2020-06-30 12:17:14 +02:00
|
|
|
package net.citizensnpcs.trait;
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
import java.util.Collection;
|
2020-07-03 09:14:55 +02:00
|
|
|
import java.util.List;
|
|
|
|
|
2020-11-17 04:39:24 +01:00
|
|
|
import org.bukkit.ChatColor;
|
2020-07-03 09:14:55 +02:00
|
|
|
import org.bukkit.Location;
|
2021-06-03 14:36:45 +02:00
|
|
|
import org.bukkit.entity.ArmorStand;
|
2020-06-30 12:17:14 +02:00
|
|
|
import org.bukkit.entity.EntityType;
|
2020-07-03 09:14:55 +02:00
|
|
|
import org.bukkit.event.player.PlayerTeleportEvent.TeleportCause;
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
import com.google.common.collect.Collections2;
|
2020-07-03 09:14:55 +02:00
|
|
|
import com.google.common.collect.Lists;
|
2020-06-30 12:17:14 +02:00
|
|
|
|
2020-07-03 09:14:55 +02:00
|
|
|
import net.citizensnpcs.Settings.Setting;
|
2020-06-30 12:17:14 +02:00
|
|
|
import net.citizensnpcs.api.CitizensAPI;
|
|
|
|
import net.citizensnpcs.api.npc.MemoryNPCDataStore;
|
|
|
|
import net.citizensnpcs.api.npc.NPC;
|
|
|
|
import net.citizensnpcs.api.npc.NPCRegistry;
|
|
|
|
import net.citizensnpcs.api.persistence.Persist;
|
|
|
|
import net.citizensnpcs.api.trait.Trait;
|
|
|
|
import net.citizensnpcs.api.trait.TraitName;
|
2020-11-17 04:39:24 +01:00
|
|
|
import net.citizensnpcs.api.util.Colorizer;
|
2021-07-04 07:07:15 +02:00
|
|
|
import net.citizensnpcs.api.util.Messaging;
|
2020-06-30 12:17:14 +02:00
|
|
|
import net.citizensnpcs.api.util.Placeholders;
|
2020-07-06 02:42:46 +02:00
|
|
|
import net.citizensnpcs.util.NMS;
|
2020-06-30 12:17:14 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Persists a hologram attached to the NPC.
|
|
|
|
*/
|
|
|
|
@TraitName("hologramtrait")
|
|
|
|
public class HologramTrait extends Trait {
|
2020-07-03 09:14:55 +02:00
|
|
|
private Location currentLoc;
|
2020-07-27 11:34:07 +02:00
|
|
|
@Persist
|
|
|
|
private HologramDirection direction = HologramDirection.BOTTOM_UP;
|
2020-07-03 09:14:55 +02:00
|
|
|
private final List<NPC> hologramNPCs = Lists.newArrayList();
|
2020-06-30 12:17:14 +02:00
|
|
|
@Persist
|
2020-07-03 09:14:55 +02:00
|
|
|
private double lineHeight = -1;
|
|
|
|
@Persist
|
|
|
|
private final List<String> lines = Lists.newArrayList();
|
2020-07-06 10:37:34 +02:00
|
|
|
private NPC nameNPC;
|
2021-02-04 03:10:48 +01:00
|
|
|
private final NPCRegistry registry = CitizensAPI.createCitizensBackedNPCRegistry(new MemoryNPCDataStore());
|
2020-06-30 12:17:14 +02:00
|
|
|
|
|
|
|
public HologramTrait() {
|
|
|
|
super("hologramtrait");
|
|
|
|
}
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
/**
|
|
|
|
* Adds a new hologram line which will displayed over an NPC's head.
|
|
|
|
*
|
|
|
|
* @param text
|
|
|
|
* The new line to add
|
|
|
|
*/
|
2020-07-03 09:14:55 +02:00
|
|
|
public void addLine(String text) {
|
|
|
|
lines.add(text);
|
2021-07-04 07:07:15 +02:00
|
|
|
onDespawn();
|
|
|
|
onSpawn();
|
2020-07-03 09:14:55 +02:00
|
|
|
}
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
/**
|
|
|
|
* Clears all hologram lines
|
|
|
|
*/
|
2020-07-06 19:07:01 +02:00
|
|
|
public void clear() {
|
2021-07-04 07:07:15 +02:00
|
|
|
onDespawn();
|
2020-07-06 19:07:01 +02:00
|
|
|
lines.clear();
|
|
|
|
}
|
|
|
|
|
2020-07-06 10:37:34 +02:00
|
|
|
private NPC createHologram(String line, double heightOffset) {
|
|
|
|
NPC hologramNPC = registry.createNPC(EntityType.ARMOR_STAND, line);
|
2020-07-17 03:52:00 +02:00
|
|
|
hologramNPC.addTrait(new ClickRedirectTrait(npc));
|
2020-09-14 11:57:58 +02:00
|
|
|
ArmorStandTrait trait = hologramNPC.getOrAddTrait(ArmorStandTrait.class);
|
2020-07-06 10:37:34 +02:00
|
|
|
trait.setVisible(false);
|
|
|
|
trait.setSmall(true);
|
|
|
|
trait.setMarker(true);
|
|
|
|
trait.setGravity(false);
|
|
|
|
trait.setHasArms(false);
|
|
|
|
trait.setHasBaseplate(false);
|
2020-07-27 11:34:07 +02:00
|
|
|
hologramNPC.spawn(currentLoc.clone().add(0,
|
|
|
|
getEntityHeight()
|
|
|
|
+ (direction == HologramDirection.BOTTOM_UP ? heightOffset : getMaxHeight() - heightOffset),
|
|
|
|
0));
|
2020-07-06 10:37:34 +02:00
|
|
|
return hologramNPC;
|
|
|
|
}
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
/**
|
|
|
|
* @return The direction that hologram lines are displayed in
|
|
|
|
*/
|
2020-10-07 13:16:41 +02:00
|
|
|
public HologramDirection getDirection() {
|
|
|
|
return direction;
|
|
|
|
}
|
|
|
|
|
2020-07-06 02:42:46 +02:00
|
|
|
private double getEntityHeight() {
|
|
|
|
return NMS.getHeight(npc.getEntity());
|
|
|
|
}
|
|
|
|
|
2020-07-03 09:14:55 +02:00
|
|
|
private double getHeight(int lineNumber) {
|
2020-07-03 16:59:18 +02:00
|
|
|
return (lineHeight == -1 ? Setting.DEFAULT_NPC_HOLOGRAM_LINE_HEIGHT.asDouble() : lineHeight) * (lineNumber + 1);
|
2020-07-03 09:14:55 +02:00
|
|
|
}
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
/**
|
|
|
|
* Note: this is implementation-specific and may be removed at a later date.
|
|
|
|
*/
|
|
|
|
public Collection<ArmorStand> getHologramEntities() {
|
|
|
|
return Collections2.transform(hologramNPCs, (n) -> (ArmorStand) n.getEntity());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return The line height between each hologram line, in blocks
|
|
|
|
*/
|
2020-10-07 13:16:41 +02:00
|
|
|
public double getLineHeight() {
|
|
|
|
return lineHeight;
|
|
|
|
}
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
/**
|
|
|
|
* @return the hologram lines, in bottom-up order
|
|
|
|
*/
|
2020-07-03 09:14:55 +02:00
|
|
|
public List<String> getLines() {
|
2021-07-19 17:02:41 +02:00
|
|
|
return lines;
|
2020-07-03 09:14:55 +02:00
|
|
|
}
|
|
|
|
|
2020-07-27 11:34:07 +02:00
|
|
|
private double getMaxHeight() {
|
|
|
|
return (lineHeight == -1 ? Setting.DEFAULT_NPC_HOLOGRAM_LINE_HEIGHT.asDouble() : lineHeight)
|
2020-08-11 16:47:45 +02:00
|
|
|
* (lines.size() + (npc.requiresNameHologram() ? 0 : 1));
|
2020-07-27 11:34:07 +02:00
|
|
|
}
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
/**
|
|
|
|
* Note: this is implementation-specific and may be removed at a later date.
|
|
|
|
*/
|
|
|
|
public ArmorStand getNameEntity() {
|
|
|
|
return nameNPC != null && nameNPC.isSpawned() ? ((ArmorStand) npc.getEntity()) : null;
|
|
|
|
}
|
|
|
|
|
2020-06-30 12:17:14 +02:00
|
|
|
@Override
|
|
|
|
public void onDespawn() {
|
2021-07-04 07:07:15 +02:00
|
|
|
if (nameNPC != null) {
|
|
|
|
nameNPC.destroy();
|
|
|
|
nameNPC = null;
|
|
|
|
}
|
|
|
|
for (NPC npc : hologramNPCs) {
|
|
|
|
npc.destroy();
|
|
|
|
}
|
|
|
|
hologramNPCs.clear();
|
2020-06-30 12:17:14 +02:00
|
|
|
}
|
|
|
|
|
2020-07-03 16:59:18 +02:00
|
|
|
@Override
|
|
|
|
public void onRemove() {
|
2021-07-04 07:07:15 +02:00
|
|
|
onDespawn();
|
2020-07-03 16:59:18 +02:00
|
|
|
}
|
|
|
|
|
2020-06-30 12:17:14 +02:00
|
|
|
@Override
|
|
|
|
public void onSpawn() {
|
2021-07-04 07:07:15 +02:00
|
|
|
currentLoc = npc.getStoredLocation();
|
|
|
|
if (npc.requiresNameHologram()
|
|
|
|
&& Boolean.parseBoolean(npc.data().<Object> get(NPC.NAMEPLATE_VISIBLE_METADATA, true).toString())) {
|
|
|
|
nameNPC = createHologram(npc.getFullName(), 0);
|
|
|
|
}
|
|
|
|
for (int i = 0; i < lines.size(); i++) {
|
|
|
|
String line = lines.get(i);
|
|
|
|
hologramNPCs.add(createHologram(Placeholders.replace(line, null, npc), getHeight(i)));
|
|
|
|
}
|
2020-07-03 09:14:55 +02:00
|
|
|
}
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
/**
|
|
|
|
* Removes the line at the specified index
|
|
|
|
*
|
|
|
|
* @param idx
|
|
|
|
*/
|
2020-07-03 09:14:55 +02:00
|
|
|
public void removeLine(int idx) {
|
|
|
|
lines.remove(idx);
|
2021-07-04 07:07:15 +02:00
|
|
|
onDespawn();
|
|
|
|
onSpawn();
|
2020-06-30 12:17:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void run() {
|
2020-07-03 09:14:55 +02:00
|
|
|
if (!npc.isSpawned()) {
|
2021-07-04 07:07:15 +02:00
|
|
|
onDespawn();
|
2020-06-30 12:17:14 +02:00
|
|
|
return;
|
|
|
|
}
|
2020-07-10 06:19:29 +02:00
|
|
|
if (npc.requiresNameHologram()) {
|
2020-07-10 06:28:39 +02:00
|
|
|
boolean visible = Boolean
|
|
|
|
.parseBoolean(npc.data().<Object> get(NPC.NAMEPLATE_VISIBLE_METADATA, true).toString());
|
2020-07-10 06:19:29 +02:00
|
|
|
if (nameNPC != null && !visible) {
|
|
|
|
nameNPC.destroy();
|
|
|
|
nameNPC = null;
|
|
|
|
} else if (nameNPC == null && visible) {
|
|
|
|
nameNPC = createHologram(npc.getFullName(), 0);
|
|
|
|
}
|
|
|
|
}
|
2020-11-03 07:36:44 +01:00
|
|
|
boolean update = currentLoc.getWorld() != npc.getStoredLocation().getWorld()
|
2021-08-15 09:25:09 +02:00
|
|
|
|| currentLoc.distance(npc.getStoredLocation()) >= 0.001;
|
2020-07-03 09:14:55 +02:00
|
|
|
if (update) {
|
|
|
|
currentLoc = npc.getStoredLocation();
|
|
|
|
}
|
2020-07-06 10:37:34 +02:00
|
|
|
if (nameNPC != null && nameNPC.isSpawned()) {
|
|
|
|
if (update) {
|
|
|
|
nameNPC.teleport(currentLoc.clone().add(0, getEntityHeight(), 0), TeleportCause.PLUGIN);
|
|
|
|
}
|
|
|
|
nameNPC.setName(npc.getFullName());
|
|
|
|
}
|
2020-07-03 09:14:55 +02:00
|
|
|
for (int i = 0; i < hologramNPCs.size(); i++) {
|
|
|
|
NPC hologramNPC = hologramNPCs.get(i);
|
2020-07-06 10:37:34 +02:00
|
|
|
if (!hologramNPC.isSpawned())
|
2020-07-03 09:14:55 +02:00
|
|
|
continue;
|
|
|
|
if (update) {
|
2020-07-06 02:42:46 +02:00
|
|
|
hologramNPC.teleport(currentLoc.clone().add(0, getEntityHeight() + getHeight(i), 0),
|
2020-07-03 09:14:55 +02:00
|
|
|
TeleportCause.PLUGIN);
|
|
|
|
}
|
2021-07-04 07:07:15 +02:00
|
|
|
if (i >= lines.size()) {
|
2021-07-16 11:23:36 +02:00
|
|
|
Messaging.severe("More hologram NPCs than lines for ID", npc.getId(), "lines", lines);
|
2021-07-04 07:07:15 +02:00
|
|
|
break;
|
|
|
|
}
|
2020-07-03 09:14:55 +02:00
|
|
|
String text = lines.get(i);
|
2020-11-17 04:39:24 +01:00
|
|
|
if (text != null && !ChatColor.stripColor(Colorizer.parseColors(text)).isEmpty()) {
|
2020-07-03 09:14:55 +02:00
|
|
|
hologramNPC.setName(Placeholders.replace(text, null, npc));
|
2020-11-17 04:36:18 +01:00
|
|
|
hologramNPC.data().set(NPC.NAMEPLATE_VISIBLE_METADATA, true);
|
2020-07-03 09:14:55 +02:00
|
|
|
} else {
|
2020-11-07 08:10:49 +01:00
|
|
|
hologramNPC.setName("");
|
2020-11-17 04:36:18 +01:00
|
|
|
hologramNPC.data().set(NPC.NAMEPLATE_VISIBLE_METADATA, false);
|
2020-07-03 09:14:55 +02:00
|
|
|
}
|
2020-06-30 12:17:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
/**
|
|
|
|
* @see #getDirection()
|
|
|
|
* @param direction
|
|
|
|
* The new direction
|
|
|
|
*/
|
2020-07-27 11:34:07 +02:00
|
|
|
public void setDirection(HologramDirection direction) {
|
|
|
|
this.direction = direction;
|
2021-07-04 07:07:15 +02:00
|
|
|
onDespawn();
|
|
|
|
onSpawn();
|
2020-07-27 11:34:07 +02:00
|
|
|
}
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
/**
|
|
|
|
* Sets the hologram line at a specific index
|
|
|
|
*
|
|
|
|
* @param idx
|
|
|
|
* The index
|
|
|
|
* @param text
|
|
|
|
* The new line
|
|
|
|
*/
|
2020-07-03 09:14:55 +02:00
|
|
|
public void setLine(int idx, String text) {
|
2020-08-11 16:57:32 +02:00
|
|
|
if (idx == lines.size()) {
|
2021-07-16 11:23:36 +02:00
|
|
|
lines.add(text);
|
2020-08-11 16:57:32 +02:00
|
|
|
} else {
|
|
|
|
lines.set(idx, text);
|
|
|
|
}
|
2021-07-04 07:07:15 +02:00
|
|
|
onDespawn();
|
|
|
|
onSpawn();
|
2020-07-03 09:14:55 +02:00
|
|
|
}
|
|
|
|
|
2021-06-03 14:36:45 +02:00
|
|
|
/**
|
|
|
|
* Sets the line height
|
|
|
|
*
|
|
|
|
* @see #getLineHeight()
|
|
|
|
* @param height
|
|
|
|
* The line height in blocks
|
|
|
|
*/
|
2020-07-03 09:14:55 +02:00
|
|
|
public void setLineHeight(double height) {
|
|
|
|
lineHeight = height;
|
2021-07-04 07:07:15 +02:00
|
|
|
onDespawn();
|
|
|
|
onSpawn();
|
2020-07-03 09:14:55 +02:00
|
|
|
}
|
2020-07-27 11:34:07 +02:00
|
|
|
|
|
|
|
public enum HologramDirection {
|
|
|
|
BOTTOM_UP,
|
|
|
|
TOP_DOWN;
|
|
|
|
}
|
2020-06-30 12:17:14 +02:00
|
|
|
}
|