Citizens2/v1_14_R1/src/main/java/net/citizensnpcs/nms/v1_14_R1/util/PlayerlistTracker.java

200 lines
7.9 KiB
Java
Raw Normal View History

2019-04-24 15:01:22 +02:00
package net.citizensnpcs.nms.v1_14_R1.util;
2019-04-25 09:43:44 +02:00
import java.lang.invoke.MethodHandle;
import java.util.Map;
import java.util.Set;
2023-02-22 17:06:13 +01:00
import org.bukkit.Bukkit;
import org.bukkit.entity.EntityType;
import org.bukkit.entity.Player;
2023-02-22 17:06:13 +01:00
import com.google.common.collect.ForwardingMap;
import com.google.common.collect.ForwardingSet;
import net.citizensnpcs.Settings.Setting;
import net.citizensnpcs.api.CitizensAPI;
import net.citizensnpcs.api.event.NPCLinkToPlayerEvent;
2023-04-16 19:47:32 +02:00
import net.citizensnpcs.api.event.NPCSeenByPlayerEvent;
2022-12-18 17:36:53 +01:00
import net.citizensnpcs.api.npc.NPC;
2019-04-24 15:01:22 +02:00
import net.citizensnpcs.nms.v1_14_R1.entity.EntityHumanNPC;
2022-12-18 17:36:53 +01:00
import net.citizensnpcs.npc.ai.NPCHolder;
2019-04-24 15:01:22 +02:00
import net.citizensnpcs.util.NMS;
import net.minecraft.server.v1_14_R1.Entity;
import net.minecraft.server.v1_14_R1.EntityPlayer;
import net.minecraft.server.v1_14_R1.EntityTrackerEntry;
import net.minecraft.server.v1_14_R1.PacketPlayOutAnimation;
2019-04-24 15:01:22 +02:00
import net.minecraft.server.v1_14_R1.PlayerChunkMap;
import net.minecraft.server.v1_14_R1.PlayerChunkMap.EntityTracker;
public class PlayerlistTracker extends PlayerChunkMap.EntityTracker {
private final Entity tracker;
private Map<EntityPlayer, Boolean> trackingMap;
2019-04-24 15:01:22 +02:00
public PlayerlistTracker(PlayerChunkMap map, Entity entity, int i, int j, boolean flag) {
map.super(entity, i, j, flag);
this.tracker = entity;
if (TRACKING_MAP_SETTER != null) {
try {
Map<EntityPlayer, Boolean> delegate = (Map<EntityPlayer, Boolean>) TRACKING_MAP_GETTER.invoke(this);
trackingMap = delegate;
TRACKING_MAP_SETTER.invoke(this, new ForwardingMap<EntityPlayer, Boolean>() {
@Override
protected Map<EntityPlayer, Boolean> delegate() {
return delegate;
}
@Override
public Boolean put(EntityPlayer player, Boolean value) {
Boolean res = super.put(player, value);
if (res == null) {
updateLastPlayer(player);
}
return res;
}
});
} catch (Throwable e) {
e.printStackTrace();
}
} else {
try {
Set<EntityPlayer> delegate = super.trackedPlayers;
TRACKING_SET_SETTER.invoke(this, new ForwardingSet<EntityPlayer>() {
@Override
public boolean add(EntityPlayer player) {
boolean res = super.add(player);
if (res) {
updateLastPlayer(player);
}
return res;
}
@Override
protected Set<EntityPlayer> delegate() {
return delegate;
}
});
} catch (Throwable e) {
e.printStackTrace();
}
}
2019-04-24 15:01:22 +02:00
}
public PlayerlistTracker(PlayerChunkMap map, EntityTracker entry) {
2020-05-12 17:12:39 +02:00
this(map, getTracker(entry), getTrackingDistance(entry), getD(entry), getE(entry));
2019-04-24 15:01:22 +02:00
}
private boolean isTracked(EntityPlayer player) {
return trackingMap != null ? trackingMap.containsKey(player) : trackedPlayers.contains(player);
}
public void updateLastPlayer(EntityPlayer lastUpdatedPlayer) {
if (lastUpdatedPlayer != null) {
Bukkit.getScheduler().scheduleSyncDelayedTask(CitizensAPI.getPlugin(),
() -> Bukkit.getPluginManager().callEvent(new NPCLinkToPlayerEvent(((NPCHolder) tracker).getNPC(),
lastUpdatedPlayer.getBukkitEntity())));
}
if (tracker.dead || lastUpdatedPlayer == null || tracker.getBukkitEntity().getType() != EntityType.PLAYER)
return;
final EntityPlayer entityplayer = lastUpdatedPlayer;
NMS.sendTabListAdd(entityplayer.getBukkitEntity(), (Player) tracker.getBukkitEntity());
2023-08-06 12:57:42 +02:00
NPC npc = ((NPCHolder) tracker).getNPC();
if (npc.data().get(NPC.Metadata.RESET_YAW_ON_SPAWN, Setting.RESET_YAW_ON_SPAWN.asBoolean())) {
Bukkit.getScheduler().scheduleSyncDelayedTask(CitizensAPI.getPlugin(),
() -> NMSImpl.sendPacket(entityplayer.getBukkitEntity(), new PacketPlayOutAnimation(tracker, 0)),
1);
}
if (!Setting.DISABLE_TABLIST.asBoolean())
return;
Bukkit.getScheduler().scheduleSyncDelayedTask(CitizensAPI.getPlugin(),
() -> NMS.sendTabListRemove(entityplayer.getBukkitEntity(), (Player) tracker.getBukkitEntity()),
Setting.TABLIST_REMOVE_PACKET_DELAY.asTicks());
2019-04-24 15:01:22 +02:00
}
@Override
public void updatePlayer(final EntityPlayer entityplayer) {
if (entityplayer instanceof EntityHumanNPC)
return;
if (!isTracked(entityplayer) && tracker instanceof NPCHolder) {
2023-03-26 20:52:56 +02:00
NPC npc = ((NPCHolder) tracker).getNPC();
2023-04-16 19:47:32 +02:00
NPCSeenByPlayerEvent event = new NPCSeenByPlayerEvent(npc, entityplayer.getBukkitEntity());
Bukkit.getPluginManager().callEvent(event);
if (event.isCancelled())
return;
2023-03-26 20:52:56 +02:00
Integer trackingRange = npc.data().<Integer> get(NPC.Metadata.TRACKING_RANGE);
if (TRACKING_RANGE_SETTER != null && trackingRange != null
&& npc.data().get("last-tracking-range", -1) != trackingRange.intValue()) {
try {
TRACKING_RANGE_SETTER.invoke(this, trackingRange);
npc.data().set("last-tracking-range", trackingRange);
} catch (Throwable e) {
e.printStackTrace();
}
}
2019-04-24 15:01:22 +02:00
}
2023-03-26 20:52:56 +02:00
super.updatePlayer(entityplayer);
2019-04-24 15:01:22 +02:00
}
private static int getD(EntityTracker entry) {
try {
2019-04-25 09:43:44 +02:00
return (int) D.invoke(TRACKER_ENTRY.invoke(entry));
} catch (Throwable e) {
2019-04-24 15:01:22 +02:00
e.printStackTrace();
}
return 0;
}
private static boolean getE(EntityTracker entry) {
try {
2019-04-25 09:43:44 +02:00
return (boolean) E.invoke(TRACKER_ENTRY.invoke(entry));
} catch (Throwable e) {
2019-04-24 15:01:22 +02:00
e.printStackTrace();
}
return false;
}
private static Entity getTracker(EntityTracker entry) {
try {
2019-04-25 09:43:44 +02:00
return (Entity) TRACKER.invoke(entry);
} catch (Throwable e) {
2019-04-24 15:01:22 +02:00
e.printStackTrace();
}
return null;
}
2020-05-12 17:12:39 +02:00
private static int getTrackingDistance(EntityTracker entry) {
2019-04-24 15:01:22 +02:00
try {
2022-12-18 17:36:53 +01:00
return (Integer) TRACKING_RANGE.invoke(entry);
2019-04-25 09:43:44 +02:00
} catch (Throwable e) {
2019-04-24 15:01:22 +02:00
e.printStackTrace();
}
2020-05-12 17:12:39 +02:00
return 0;
2019-04-24 15:01:22 +02:00
}
2019-04-25 09:43:44 +02:00
private static final MethodHandle D = NMS.getGetter(EntityTrackerEntry.class, "d");
private static final MethodHandle E = NMS.getGetter(EntityTrackerEntry.class, "e");
private static final MethodHandle TRACKER = NMS.getGetter(EntityTracker.class, "tracker");
private static final MethodHandle TRACKER_ENTRY = NMS.getGetter(EntityTracker.class, "trackerEntry");
private static MethodHandle TRACKING_MAP_GETTER;
private static MethodHandle TRACKING_MAP_SETTER;
2022-12-18 17:36:53 +01:00
private static final MethodHandle TRACKING_RANGE = NMS.getGetter(EntityTracker.class, "trackingDistance");
2023-03-26 20:52:56 +02:00
private static final MethodHandle TRACKING_RANGE_SETTER = NMS.getFirstFinalSetter(EntityTracker.class, int.class);
private static final MethodHandle TRACKING_SET_SETTER = NMS.getFirstFinalSetter(EntityTracker.class, Set.class);
static {
try {
// Old paper versions override the tracked player set to be a map
if (EntityTracker.class.getField("trackedPlayerMap") != null) {
TRACKING_MAP_SETTER = NMS.getFirstSetter(EntityTracker.class, Map.class);
TRACKING_MAP_GETTER = NMS.getFirstGetter(EntityTracker.class, Map.class);
}
} catch (Exception e) {
}
}
2019-04-24 15:01:22 +02:00
}