2017-12-30 08:36:36 +01:00
|
|
|
package net.citizensnpcs.trait;
|
|
|
|
|
|
|
|
import java.util.Collections;
|
|
|
|
import java.util.Comparator;
|
|
|
|
import java.util.List;
|
2019-04-26 15:14:15 +02:00
|
|
|
import java.util.Random;
|
2017-12-30 08:36:36 +01:00
|
|
|
|
|
|
|
import org.bukkit.GameMode;
|
|
|
|
import org.bukkit.Location;
|
|
|
|
import org.bukkit.entity.Entity;
|
|
|
|
import org.bukkit.entity.EntityType;
|
|
|
|
import org.bukkit.entity.LivingEntity;
|
|
|
|
import org.bukkit.entity.Player;
|
2018-06-07 11:01:38 +02:00
|
|
|
import org.bukkit.metadata.MetadataValue;
|
2017-12-30 08:36:36 +01:00
|
|
|
import org.bukkit.potion.PotionEffectType;
|
|
|
|
|
|
|
|
import net.citizensnpcs.Settings.Setting;
|
|
|
|
import net.citizensnpcs.api.CitizensAPI;
|
|
|
|
import net.citizensnpcs.api.command.CommandConfigurable;
|
|
|
|
import net.citizensnpcs.api.command.CommandContext;
|
2018-03-09 13:07:08 +01:00
|
|
|
import net.citizensnpcs.api.persistence.Persist;
|
2017-12-30 08:36:36 +01:00
|
|
|
import net.citizensnpcs.api.trait.Trait;
|
|
|
|
import net.citizensnpcs.api.trait.TraitName;
|
|
|
|
import net.citizensnpcs.api.util.DataKey;
|
2019-04-25 11:11:02 +02:00
|
|
|
import net.citizensnpcs.util.NMS;
|
2017-12-30 08:36:36 +01:00
|
|
|
import net.citizensnpcs.util.Util;
|
|
|
|
|
2019-05-16 13:58:29 +02:00
|
|
|
/**
|
|
|
|
* Persists the /npc lookclose metadata
|
|
|
|
*
|
|
|
|
*/
|
2017-12-30 08:36:36 +01:00
|
|
|
@TraitName("lookclose")
|
|
|
|
public class LookClose extends Trait implements Toggleable, CommandConfigurable {
|
2018-03-09 13:07:08 +01:00
|
|
|
@Persist("enabled")
|
2017-12-30 08:36:36 +01:00
|
|
|
private boolean enabled = Setting.DEFAULT_LOOK_CLOSE.asBoolean();
|
2019-04-26 15:14:15 +02:00
|
|
|
@Persist
|
|
|
|
private boolean enableRandomLook = Setting.DEFAULT_RANDOM_LOOK_CLOSE.asBoolean();
|
2017-12-30 08:36:36 +01:00
|
|
|
private Player lookingAt;
|
2019-04-26 15:14:15 +02:00
|
|
|
@Persist
|
|
|
|
private int randomLookDelay = Setting.DEFAULT_RANDOM_LOOK_DELAY.asInt();
|
|
|
|
@Persist
|
|
|
|
private float[] randomPitchRange = { -10, 0 };
|
|
|
|
@Persist
|
|
|
|
private float[] randomYawRange = { 0, 360 };
|
2017-12-30 08:36:36 +01:00
|
|
|
private double range = Setting.DEFAULT_LOOK_CLOSE_RANGE.asDouble();
|
2018-03-09 13:07:08 +01:00
|
|
|
@Persist("realisticlooking")
|
2017-12-30 08:36:36 +01:00
|
|
|
private boolean realisticLooking = Setting.DEFAULT_REALISTIC_LOOKING.asBoolean();
|
2019-04-26 15:14:15 +02:00
|
|
|
private int t;
|
2017-12-30 08:36:36 +01:00
|
|
|
|
|
|
|
public LookClose() {
|
|
|
|
super("lookclose");
|
|
|
|
}
|
|
|
|
|
2019-05-16 13:58:29 +02:00
|
|
|
/**
|
2020-02-25 06:43:19 +01:00
|
|
|
* Returns whether the target can be seen. Will use realistic line of sight if
|
|
|
|
* {@link #setRealisticLooking(boolean)} is true.
|
2019-05-16 13:58:29 +02:00
|
|
|
*/
|
|
|
|
public boolean canSeeTarget() {
|
2017-12-30 08:36:36 +01:00
|
|
|
return realisticLooking && npc.getEntity() instanceof LivingEntity
|
2019-04-25 11:11:02 +02:00
|
|
|
? ((LivingEntity) npc.getEntity()).hasLineOfSight(lookingAt)
|
2019-05-16 13:58:29 +02:00
|
|
|
: lookingAt != null && lookingAt.isValid();
|
2017-12-30 08:36:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void configure(CommandContext args) {
|
2018-03-09 13:07:08 +01:00
|
|
|
range = args.getFlagDouble("range", args.getFlagDouble("r", range));
|
2017-12-30 08:36:36 +01:00
|
|
|
realisticLooking = args.hasFlag('r');
|
|
|
|
}
|
|
|
|
|
2019-05-16 13:58:29 +02:00
|
|
|
/**
|
|
|
|
* Finds a new look-close target
|
|
|
|
*/
|
|
|
|
public void findNewTarget() {
|
2017-12-30 08:36:36 +01:00
|
|
|
List<Entity> nearby = npc.getEntity().getNearbyEntities(range, range, range);
|
|
|
|
Collections.sort(nearby, new Comparator<Entity>() {
|
|
|
|
@Override
|
|
|
|
public int compare(Entity o1, Entity o2) {
|
|
|
|
Location l1 = o1.getLocation(CACHE_LOCATION);
|
|
|
|
Location l2 = o2.getLocation(CACHE_LOCATION2);
|
|
|
|
if (!NPC_LOCATION.getWorld().equals(l1.getWorld()) || !NPC_LOCATION.getWorld().equals(l2.getWorld())) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return Double.compare(l1.distanceSquared(NPC_LOCATION), l2.distanceSquared(NPC_LOCATION));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
for (Entity entity : nearby) {
|
|
|
|
if (entity.getType() != EntityType.PLAYER || ((Player) entity).getGameMode() == GameMode.SPECTATOR
|
|
|
|
|| ((Player) entity).hasPotionEffect(PotionEffectType.INVISIBILITY)
|
2018-06-07 11:01:38 +02:00
|
|
|
|| isPluginVanished((Player) entity)
|
2017-12-30 08:36:36 +01:00
|
|
|
|| entity.getLocation(CACHE_LOCATION).getWorld() != NPC_LOCATION.getWorld()
|
|
|
|
|| CitizensAPI.getNPCRegistry().getNPC(entity) != null)
|
|
|
|
continue;
|
|
|
|
lookingAt = (Player) entity;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean hasInvalidTarget() {
|
|
|
|
if (lookingAt == null)
|
|
|
|
return true;
|
|
|
|
if (!lookingAt.isOnline() || lookingAt.getWorld() != npc.getEntity().getWorld()
|
|
|
|
|| lookingAt.getLocation(PLAYER_LOCATION).distanceSquared(NPC_LOCATION) > range) {
|
|
|
|
lookingAt = null;
|
|
|
|
}
|
|
|
|
return lookingAt == null;
|
|
|
|
}
|
|
|
|
|
2018-06-07 11:01:38 +02:00
|
|
|
private boolean isPluginVanished(Player player) {
|
|
|
|
for (MetadataValue meta : player.getMetadata("vanished")) {
|
2020-02-25 06:43:19 +01:00
|
|
|
if (meta.asBoolean()) {
|
2018-06-07 11:01:38 +02:00
|
|
|
return true;
|
2020-02-25 06:43:19 +01:00
|
|
|
}
|
2018-06-07 11:01:38 +02:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-12-30 08:36:36 +01:00
|
|
|
@Override
|
2019-04-26 15:14:15 +02:00
|
|
|
public void load(DataKey key) {
|
|
|
|
range = key.getDouble("range");
|
2017-12-30 08:36:36 +01:00
|
|
|
}
|
|
|
|
|
2019-05-16 13:58:29 +02:00
|
|
|
/**
|
|
|
|
* Enables/disables the trait
|
|
|
|
*/
|
2017-12-30 08:36:36 +01:00
|
|
|
public void lookClose(boolean lookClose) {
|
|
|
|
enabled = lookClose;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void onDespawn() {
|
|
|
|
lookingAt = null;
|
|
|
|
}
|
|
|
|
|
2019-04-26 15:14:15 +02:00
|
|
|
private void randomLook() {
|
|
|
|
Random rand = new Random();
|
|
|
|
float pitch = rand.doubles(randomPitchRange[0], randomPitchRange[1]).iterator().next().floatValue(),
|
|
|
|
yaw = rand.doubles(randomYawRange[0], randomYawRange[1]).iterator().next().floatValue();
|
|
|
|
Util.assumePose(npc.getEntity(), yaw, pitch);
|
|
|
|
}
|
|
|
|
|
2017-12-30 08:36:36 +01:00
|
|
|
@Override
|
|
|
|
public void run() {
|
2020-02-25 06:43:19 +01:00
|
|
|
if (!enabled || !npc.isSpawned()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (npc.getNavigator().isNavigating() && Setting.DISABLE_LOOKCLOSE_WHILE_NAVIGATING.asBoolean()) {
|
2017-12-30 08:36:36 +01:00
|
|
|
return;
|
2020-02-25 06:43:19 +01:00
|
|
|
}
|
2019-09-23 15:12:21 +02:00
|
|
|
// TODO: remove in a later version, defaults weren't saving properly
|
2019-09-23 15:30:12 +02:00
|
|
|
if (randomPitchRange == null || randomPitchRange.length != 2) {
|
2019-09-23 15:12:21 +02:00
|
|
|
randomPitchRange = new float[] { -10, 0 };
|
|
|
|
}
|
2019-09-23 15:30:12 +02:00
|
|
|
if (randomYawRange == null || randomYawRange.length != 2) {
|
2019-09-23 15:12:21 +02:00
|
|
|
randomYawRange = new float[] { 0, 360 };
|
|
|
|
}
|
2017-12-30 08:36:36 +01:00
|
|
|
npc.getEntity().getLocation(NPC_LOCATION);
|
|
|
|
if (hasInvalidTarget()) {
|
|
|
|
findNewTarget();
|
|
|
|
}
|
2020-02-25 06:43:19 +01:00
|
|
|
if (npc.getNavigator().isNavigating()) {
|
|
|
|
npc.getNavigator().setPaused(lookingAt != null);
|
|
|
|
} else if (lookingAt == null && enableRandomLook && t <= 0) {
|
2019-04-26 15:14:15 +02:00
|
|
|
randomLook();
|
|
|
|
t = randomLookDelay;
|
|
|
|
}
|
|
|
|
t--;
|
2017-12-30 08:36:36 +01:00
|
|
|
if (lookingAt != null && canSeeTarget()) {
|
|
|
|
Util.faceEntity(npc.getEntity(), lookingAt);
|
2019-04-25 11:11:02 +02:00
|
|
|
if (npc.getEntity().getType().name().toLowerCase().contains("shulker")) {
|
|
|
|
NMS.setPeekShulker(npc.getEntity(), 100 - (int) Math
|
|
|
|
.floor(npc.getStoredLocation().distanceSquared(lookingAt.getLocation(PLAYER_LOCATION))));
|
|
|
|
}
|
2017-12-30 08:36:36 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void save(DataKey key) {
|
|
|
|
key.setDouble("range", range);
|
|
|
|
}
|
|
|
|
|
2019-05-16 13:58:29 +02:00
|
|
|
/**
|
2020-02-25 06:43:19 +01:00
|
|
|
* Enables random looking - will look at a random {@link Location} every so
|
|
|
|
* often if enabled.
|
2019-05-16 13:58:29 +02:00
|
|
|
*/
|
2019-04-26 15:14:15 +02:00
|
|
|
public void setRandomLook(boolean enableRandomLook) {
|
|
|
|
this.enableRandomLook = enableRandomLook;
|
|
|
|
}
|
|
|
|
|
2019-05-16 13:58:29 +02:00
|
|
|
/**
|
|
|
|
* Sets the delay between random looking in ticks
|
|
|
|
*/
|
2019-04-26 15:14:15 +02:00
|
|
|
public void setRandomLookDelay(int delay) {
|
|
|
|
this.randomLookDelay = delay;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setRandomLookPitchRange(float min, float max) {
|
|
|
|
this.randomPitchRange = new float[] { min, max };
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setRandomLookYawRange(float min, float max) {
|
|
|
|
this.randomYawRange = new float[] { min, max };
|
|
|
|
}
|
|
|
|
|
2019-05-16 13:58:29 +02:00
|
|
|
/**
|
|
|
|
* Sets the maximum range in blocks to look at other Entities
|
|
|
|
*/
|
2017-12-30 08:36:36 +01:00
|
|
|
public void setRange(int range) {
|
|
|
|
this.range = range;
|
|
|
|
}
|
|
|
|
|
2019-05-16 13:58:29 +02:00
|
|
|
/**
|
2020-02-25 06:43:19 +01:00
|
|
|
* Enables/disables realistic looking (using line of sight checks). More
|
|
|
|
* computationally expensive.
|
2019-05-16 13:58:29 +02:00
|
|
|
*/
|
2017-12-30 08:36:36 +01:00
|
|
|
public void setRealisticLooking(boolean realistic) {
|
|
|
|
this.realisticLooking = realistic;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean toggle() {
|
|
|
|
enabled = !enabled;
|
|
|
|
return enabled;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public String toString() {
|
|
|
|
return "LookClose{" + enabled + "}";
|
|
|
|
}
|
|
|
|
|
|
|
|
private static final Location CACHE_LOCATION = new Location(null, 0, 0, 0);
|
|
|
|
private static final Location CACHE_LOCATION2 = new Location(null, 0, 0, 0);
|
|
|
|
private static final Location NPC_LOCATION = new Location(null, 0, 0, 0);
|
|
|
|
private static final Location PLAYER_LOCATION = new Location(null, 0, 0, 0);
|
2012-02-03 10:20:48 +01:00
|
|
|
}
|