Dateien nach "src/main/java/pb/ajneb97/lib/fastboard" hochladen
This commit is contained in:
161
src/main/java/pb/ajneb97/lib/fastboard/FastBoard.java
Normal file
161
src/main/java/pb/ajneb97/lib/fastboard/FastBoard.java
Normal file
@@ -0,0 +1,161 @@
|
||||
/*
|
||||
* This file is part of FastBoard, licensed under the MIT License.
|
||||
*
|
||||
* Copyright (c) 2019-2023 MrMicky
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in all
|
||||
* copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
* SOFTWARE.
|
||||
*/
|
||||
package pb.ajneb97.lib.fastboard;
|
||||
|
||||
import org.bukkit.ChatColor;
|
||||
import org.bukkit.entity.Player;
|
||||
|
||||
import java.lang.invoke.MethodHandle;
|
||||
import java.lang.invoke.MethodHandles;
|
||||
import java.lang.reflect.Array;
|
||||
import java.util.Objects;
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
public class FastBoard extends FastBoardBase<String> {
|
||||
|
||||
private static final MethodHandle MESSAGE_FROM_STRING;
|
||||
private static final Object EMPTY_MESSAGE;
|
||||
|
||||
static {
|
||||
try {
|
||||
MethodHandles.Lookup lookup = MethodHandles.lookup();
|
||||
Class<?> craftChatMessageClass = FastReflection.obcClass("util.CraftChatMessage");
|
||||
MESSAGE_FROM_STRING = lookup.unreflect(craftChatMessageClass.getMethod("fromString", String.class));
|
||||
EMPTY_MESSAGE = Array.get(MESSAGE_FROM_STRING.invoke(""), 0);
|
||||
} catch (Throwable t) {
|
||||
throw new ExceptionInInitializerError(t);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
public FastBoard(Player player) {
|
||||
super(player);
|
||||
}
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
@Override
|
||||
public void updateTitle(String title) {
|
||||
Objects.requireNonNull(title, "title");
|
||||
|
||||
if (!VersionType.V1_13.isHigherOrEqual() && title.length() > 32) {
|
||||
throw new IllegalArgumentException("Title is longer than 32 chars");
|
||||
}
|
||||
|
||||
super.updateTitle(title);
|
||||
}
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
@Override
|
||||
public void updateLines(String... lines) {
|
||||
Objects.requireNonNull(lines, "lines");
|
||||
|
||||
if (!VersionType.V1_13.isHigherOrEqual()) {
|
||||
int lineCount = 0;
|
||||
for (String s : lines) {
|
||||
if (s != null && s.length() > 30) {
|
||||
throw new IllegalArgumentException("Line " + lineCount + " is longer than 30 chars");
|
||||
}
|
||||
lineCount++;
|
||||
}
|
||||
}
|
||||
|
||||
super.updateLines(lines);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void sendLineChange(int score) throws Throwable {
|
||||
int maxLength = hasLinesMaxLength() ? 16 : 1024;
|
||||
String line = getLineByScore(score);
|
||||
String prefix;
|
||||
String suffix = "";
|
||||
|
||||
if (line == null || line.isEmpty()) {
|
||||
prefix = COLOR_CODES[score] + ChatColor.RESET;
|
||||
} else if (line.length() <= maxLength) {
|
||||
prefix = line;
|
||||
} else {
|
||||
// Prevent splitting color codes
|
||||
int index = line.charAt(maxLength - 1) == ChatColor.COLOR_CHAR
|
||||
? (maxLength - 1) : maxLength;
|
||||
prefix = line.substring(0, index);
|
||||
String suffixTmp = line.substring(index);
|
||||
ChatColor chatColor = null;
|
||||
|
||||
if (suffixTmp.length() >= 2 && suffixTmp.charAt(0) == ChatColor.COLOR_CHAR) {
|
||||
chatColor = ChatColor.getByChar(suffixTmp.charAt(1));
|
||||
}
|
||||
|
||||
String color = ChatColor.getLastColors(prefix);
|
||||
boolean addColor = chatColor == null || chatColor.isFormat();
|
||||
|
||||
suffix = (addColor ? (color.isEmpty() ? ChatColor.RESET.toString() : color) : "") + suffixTmp;
|
||||
}
|
||||
|
||||
if (prefix.length() > maxLength || suffix.length() > maxLength) {
|
||||
// Something went wrong, just cut to prevent client crash/kick
|
||||
prefix = prefix.substring(0, Math.min(maxLength, prefix.length()));
|
||||
suffix = suffix.substring(0, Math.min(maxLength, suffix.length()));
|
||||
}
|
||||
|
||||
sendTeamPacket(score, TeamMode.UPDATE, prefix, suffix);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Object toMinecraftComponent(String line) throws Throwable {
|
||||
if (line == null || line.isEmpty()) {
|
||||
return EMPTY_MESSAGE;
|
||||
}
|
||||
|
||||
return Array.get(MESSAGE_FROM_STRING.invoke(line), 0);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected String serializeLine(String value) {
|
||||
return value;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected String emptyLine() {
|
||||
return "";
|
||||
}
|
||||
|
||||
/**
|
||||
* Return if the player has a prefix/suffix characters limit.
|
||||
* By default, it returns true only in 1.12 or lower.
|
||||
* This method can be overridden to fix compatibility with some versions support plugin.
|
||||
*
|
||||
* @return max length
|
||||
*/
|
||||
protected boolean hasLinesMaxLength() {
|
||||
return !VersionType.V1_13.isHigherOrEqual();
|
||||
}
|
||||
}
|
832
src/main/java/pb/ajneb97/lib/fastboard/FastBoardBase.java
Normal file
832
src/main/java/pb/ajneb97/lib/fastboard/FastBoardBase.java
Normal file
@@ -0,0 +1,832 @@
|
||||
/*
|
||||
* This file is part of FastBoard, licensed under the MIT License.
|
||||
*
|
||||
* Copyright (c) 2019-2023 MrMicky
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in all
|
||||
* copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
* SOFTWARE.
|
||||
*/
|
||||
package pb.ajneb97.lib.fastboard;
|
||||
|
||||
import org.bukkit.ChatColor;
|
||||
import org.bukkit.entity.Player;
|
||||
|
||||
import java.lang.invoke.MethodHandle;
|
||||
import java.lang.invoke.MethodHandles;
|
||||
import java.lang.invoke.MethodType;
|
||||
import java.lang.reflect.Field;
|
||||
import java.lang.reflect.Method;
|
||||
import java.lang.reflect.Modifier;
|
||||
import java.util.*;
|
||||
import java.util.concurrent.ThreadLocalRandom;
|
||||
import java.util.stream.Stream;
|
||||
|
||||
/**
|
||||
* Lightweight packet-based scoreboard API for Bukkit plugins.
|
||||
* It can be safely used asynchronously as everything is at packet level.
|
||||
* <p>
|
||||
* The project is on <a href="https://github.com/MrMicky-FR/FastBoard">GitHub</a>.
|
||||
*
|
||||
* @author MrMicky
|
||||
* @version 2.1.5
|
||||
*/
|
||||
public abstract class FastBoardBase<T> {
|
||||
|
||||
private static final Map<Class<?>, Field[]> PACKETS = new HashMap<>(8);
|
||||
protected static final String[] COLOR_CODES = Arrays.stream(ChatColor.values())
|
||||
.map(Object::toString)
|
||||
.toArray(String[]::new);
|
||||
private static final VersionType VERSION_TYPE;
|
||||
// Packets and components
|
||||
private static final Class<?> CHAT_COMPONENT_CLASS;
|
||||
private static final Class<?> CHAT_FORMAT_ENUM;
|
||||
private static final Object RESET_FORMATTING;
|
||||
private static final MethodHandle PLAYER_CONNECTION;
|
||||
private static final MethodHandle SEND_PACKET;
|
||||
private static final MethodHandle PLAYER_GET_HANDLE;
|
||||
private static final MethodHandle FIXED_NUMBER_FORMAT;
|
||||
// Scoreboard packets
|
||||
private static final FastReflection.PacketConstructor PACKET_SB_OBJ;
|
||||
private static final FastReflection.PacketConstructor PACKET_SB_DISPLAY_OBJ;
|
||||
private static final FastReflection.PacketConstructor PACKET_SB_TEAM;
|
||||
private static final FastReflection.PacketConstructor PACKET_SB_SERIALIZABLE_TEAM;
|
||||
private static final MethodHandle PACKET_SB_SET_SCORE;
|
||||
private static final MethodHandle PACKET_SB_RESET_SCORE;
|
||||
private static final boolean SCORE_OPTIONAL_COMPONENTS;
|
||||
// Scoreboard enums
|
||||
private static final Class<?> DISPLAY_SLOT_TYPE;
|
||||
private static final Class<?> ENUM_SB_HEALTH_DISPLAY;
|
||||
private static final Class<?> ENUM_SB_ACTION;
|
||||
private static final Class<?> ENUM_VISIBILITY;
|
||||
private static final Class<?> ENUM_COLLISION_RULE;
|
||||
private static final Object BLANK_NUMBER_FORMAT;
|
||||
private static final Object SIDEBAR_DISPLAY_SLOT;
|
||||
private static final Object ENUM_SB_HEALTH_DISPLAY_INTEGER;
|
||||
private static final Object ENUM_SB_ACTION_CHANGE;
|
||||
private static final Object ENUM_SB_ACTION_REMOVE;
|
||||
private static final Object ENUM_VISIBILITY_ALWAYS;
|
||||
private static final Object ENUM_COLLISION_RULE_ALWAYS;
|
||||
|
||||
static {
|
||||
try {
|
||||
MethodHandles.Lookup lookup = MethodHandles.lookup();
|
||||
|
||||
if (FastReflection.isRepackaged()) {
|
||||
VERSION_TYPE = VersionType.V1_17;
|
||||
} else if (FastReflection.nmsOptionalClass(null, "ScoreboardServer$Action").isPresent()
|
||||
|| FastReflection.nmsOptionalClass(null, "ServerScoreboard$Method").isPresent()) {
|
||||
VERSION_TYPE = VersionType.V1_13;
|
||||
} else if (FastReflection.nmsOptionalClass(null, "IScoreboardCriteria$EnumScoreboardHealthDisplay").isPresent()
|
||||
|| FastReflection.nmsOptionalClass(null, "ObjectiveCriteria$RenderType").isPresent()) {
|
||||
VERSION_TYPE = VersionType.V1_8;
|
||||
} else {
|
||||
VERSION_TYPE = VersionType.V1_7;
|
||||
}
|
||||
|
||||
String gameProtocolPackage = "network.protocol.game";
|
||||
Class<?> craftPlayerClass = FastReflection.obcClass("entity.CraftPlayer");
|
||||
Class<?> entityPlayerClass = FastReflection.nmsClass("server.level", "EntityPlayer", "ServerPlayer");
|
||||
Class<?> playerConnectionClass = FastReflection.nmsClass("server.network", "PlayerConnection", "ServerGamePacketListenerImpl");
|
||||
Class<?> packetClass = FastReflection.nmsClass("network.protocol", "Packet");
|
||||
Class<?> packetSbObjClass = FastReflection.nmsClass(gameProtocolPackage, "PacketPlayOutScoreboardObjective", "ClientboundSetObjectivePacket");
|
||||
Class<?> packetSbDisplayObjClass = FastReflection.nmsClass(gameProtocolPackage, "PacketPlayOutScoreboardDisplayObjective", "ClientboundSetDisplayObjectivePacket");
|
||||
Class<?> packetSbScoreClass = FastReflection.nmsClass(gameProtocolPackage, "PacketPlayOutScoreboardScore", "ClientboundSetScorePacket");
|
||||
Class<?> packetSbTeamClass = FastReflection.nmsClass(gameProtocolPackage, "PacketPlayOutScoreboardTeam", "ClientboundSetPlayerTeamPacket");
|
||||
Class<?> sbTeamClass = VersionType.V1_17.isHigherOrEqual()
|
||||
? FastReflection.innerClass(packetSbTeamClass, innerClass -> !innerClass.isEnum()) : null;
|
||||
Field playerConnectionField = Arrays.stream(entityPlayerClass.getFields())
|
||||
.filter(field -> field.getType().isAssignableFrom(playerConnectionClass))
|
||||
.findFirst().orElseThrow(NoSuchFieldException::new);
|
||||
Method sendPacketMethod = Stream.concat(
|
||||
Arrays.stream(playerConnectionClass.getSuperclass().getMethods()),
|
||||
Arrays.stream(playerConnectionClass.getMethods())
|
||||
)
|
||||
.filter(m -> m.getParameterCount() == 1 && m.getParameterTypes()[0] == packetClass)
|
||||
.findFirst().orElseThrow(NoSuchMethodException::new);
|
||||
Optional<Class<?>> displaySlotEnum = FastReflection.nmsOptionalClass("world.scores", "DisplaySlot");
|
||||
CHAT_COMPONENT_CLASS = FastReflection.nmsClass("network.chat", "IChatBaseComponent","Component");
|
||||
CHAT_FORMAT_ENUM = FastReflection.nmsClass(null, "EnumChatFormat", "ChatFormatting");
|
||||
DISPLAY_SLOT_TYPE = displaySlotEnum.orElse(int.class);
|
||||
RESET_FORMATTING = FastReflection.enumValueOf(CHAT_FORMAT_ENUM, "RESET", 21);
|
||||
SIDEBAR_DISPLAY_SLOT = displaySlotEnum.isPresent() ? FastReflection.enumValueOf(DISPLAY_SLOT_TYPE, "SIDEBAR", 1) : 1;
|
||||
PLAYER_GET_HANDLE = lookup.findVirtual(craftPlayerClass, "getHandle", MethodType.methodType(entityPlayerClass));
|
||||
PLAYER_CONNECTION = lookup.unreflectGetter(playerConnectionField);
|
||||
SEND_PACKET = lookup.unreflect(sendPacketMethod);
|
||||
PACKET_SB_OBJ = FastReflection.findPacketConstructor(packetSbObjClass, lookup);
|
||||
PACKET_SB_DISPLAY_OBJ = FastReflection.findPacketConstructor(packetSbDisplayObjClass, lookup);
|
||||
|
||||
Optional<Class<?>> numberFormat = FastReflection.nmsOptionalClass("network.chat.numbers", "NumberFormat");
|
||||
MethodHandle packetSbSetScore;
|
||||
MethodHandle packetSbResetScore = null;
|
||||
MethodHandle fixedFormatConstructor = null;
|
||||
Object blankNumberFormat = null;
|
||||
boolean scoreOptionalComponents = false;
|
||||
|
||||
if (numberFormat.isPresent()) { // 1.20.3
|
||||
Class<?> blankFormatClass = FastReflection.nmsClass("network.chat.numbers", "BlankFormat");
|
||||
Class<?> fixedFormatClass = FastReflection.nmsClass("network.chat.numbers", "FixedFormat");
|
||||
Class<?> resetScoreClass = FastReflection.nmsClass(gameProtocolPackage, "ClientboundResetScorePacket");
|
||||
MethodType scoreType = MethodType.methodType(void.class, String.class, String.class, int.class, CHAT_COMPONENT_CLASS, numberFormat.get());
|
||||
MethodType scoreTypeOptional = MethodType.methodType(void.class, String.class, String.class, int.class, Optional.class, Optional.class);
|
||||
MethodType removeScoreType = MethodType.methodType(void.class, String.class, String.class);
|
||||
MethodType fixedFormatType = MethodType.methodType(void.class, CHAT_COMPONENT_CLASS);
|
||||
Optional<Field> blankField = Arrays.stream(blankFormatClass.getFields()).filter(f -> f.getType() == blankFormatClass).findAny();
|
||||
// Fields are of type Optional in 1.20.5+
|
||||
Optional<MethodHandle> optionalScorePacket = FastReflection.optionalConstructor(packetSbScoreClass, lookup, scoreTypeOptional);
|
||||
fixedFormatConstructor = lookup.findConstructor(fixedFormatClass, fixedFormatType);
|
||||
packetSbSetScore = optionalScorePacket.isPresent() ? optionalScorePacket.get()
|
||||
: lookup.findConstructor(packetSbScoreClass, scoreType);
|
||||
scoreOptionalComponents = optionalScorePacket.isPresent();
|
||||
packetSbResetScore = lookup.findConstructor(resetScoreClass, removeScoreType);
|
||||
blankNumberFormat = blankField.isPresent() ? blankField.get().get(null) : null;
|
||||
} else if (VersionType.V1_17.isHigherOrEqual()) {
|
||||
Class<?> enumSbAction = FastReflection.nmsClass("server", "ScoreboardServer$Action", "ServerScoreboard$Method");
|
||||
MethodType scoreType = MethodType.methodType(void.class, enumSbAction, String.class, String.class, int.class);
|
||||
packetSbSetScore = lookup.findConstructor(packetSbScoreClass, scoreType);
|
||||
} else {
|
||||
packetSbSetScore = lookup.findConstructor(packetSbScoreClass, MethodType.methodType(void.class));
|
||||
}
|
||||
|
||||
PACKET_SB_SET_SCORE = packetSbSetScore;
|
||||
PACKET_SB_RESET_SCORE = packetSbResetScore;
|
||||
PACKET_SB_TEAM = FastReflection.findPacketConstructor(packetSbTeamClass, lookup);
|
||||
PACKET_SB_SERIALIZABLE_TEAM = sbTeamClass != null ? FastReflection.findPacketConstructor(sbTeamClass, lookup) : null;
|
||||
FIXED_NUMBER_FORMAT = fixedFormatConstructor;
|
||||
BLANK_NUMBER_FORMAT = blankNumberFormat;
|
||||
SCORE_OPTIONAL_COMPONENTS = scoreOptionalComponents;
|
||||
|
||||
if (VersionType.V1_17.isHigherOrEqual()) {
|
||||
ENUM_VISIBILITY = FastReflection.nmsClass("world.scores", "ScoreboardTeamBase$EnumNameTagVisibility", "Team$Visibility");
|
||||
ENUM_COLLISION_RULE = FastReflection.nmsClass("world.scores", "ScoreboardTeamBase$EnumTeamPush", "Team$CollisionRule");
|
||||
ENUM_VISIBILITY_ALWAYS = FastReflection.enumValueOf(ENUM_VISIBILITY, "ALWAYS", 0);
|
||||
ENUM_COLLISION_RULE_ALWAYS = FastReflection.enumValueOf(ENUM_COLLISION_RULE, "ALWAYS", 0);
|
||||
} else {
|
||||
ENUM_VISIBILITY = null;
|
||||
ENUM_COLLISION_RULE = null;
|
||||
ENUM_VISIBILITY_ALWAYS = null;
|
||||
ENUM_COLLISION_RULE_ALWAYS = null;
|
||||
}
|
||||
|
||||
for (Class<?> clazz : Arrays.asList(packetSbObjClass, packetSbDisplayObjClass, packetSbScoreClass, packetSbTeamClass, sbTeamClass)) {
|
||||
if (clazz == null) {
|
||||
continue;
|
||||
}
|
||||
Field[] fields = Arrays.stream(clazz.getDeclaredFields())
|
||||
.filter(field -> !Modifier.isStatic(field.getModifiers()))
|
||||
.toArray(Field[]::new);
|
||||
for (Field field : fields) {
|
||||
field.setAccessible(true);
|
||||
}
|
||||
PACKETS.put(clazz, fields);
|
||||
}
|
||||
|
||||
if (VersionType.V1_8.isHigherOrEqual()) {
|
||||
String enumSbActionClass = VersionType.V1_13.isHigherOrEqual()
|
||||
? "ScoreboardServer$Action"
|
||||
: "PacketPlayOutScoreboardScore$EnumScoreboardAction";
|
||||
ENUM_SB_HEALTH_DISPLAY = FastReflection.nmsClass("world.scores.criteria", "IScoreboardCriteria$EnumScoreboardHealthDisplay", "ObjectiveCriteria$RenderType");
|
||||
ENUM_SB_ACTION = FastReflection.nmsOptionalClass("server", enumSbActionClass, "ServerScoreboard$Method").orElse(null);
|
||||
ENUM_SB_HEALTH_DISPLAY_INTEGER = FastReflection.enumValueOf(ENUM_SB_HEALTH_DISPLAY, "INTEGER", 0);
|
||||
ENUM_SB_ACTION_CHANGE = ENUM_SB_ACTION != null ? FastReflection.enumValueOf(ENUM_SB_ACTION, "CHANGE", 0) : null;
|
||||
ENUM_SB_ACTION_REMOVE = ENUM_SB_ACTION != null ? FastReflection.enumValueOf(ENUM_SB_ACTION, "REMOVE", 1) : null;
|
||||
} else {
|
||||
ENUM_SB_HEALTH_DISPLAY = null;
|
||||
ENUM_SB_ACTION = null;
|
||||
ENUM_SB_HEALTH_DISPLAY_INTEGER = null;
|
||||
ENUM_SB_ACTION_CHANGE = null;
|
||||
ENUM_SB_ACTION_REMOVE = null;
|
||||
}
|
||||
} catch (Throwable t) {
|
||||
throw new ExceptionInInitializerError(t);
|
||||
}
|
||||
}
|
||||
|
||||
private final Player player;
|
||||
private final String id;
|
||||
|
||||
private final List<T> lines = new ArrayList<>();
|
||||
private final List<T> scores = new ArrayList<>();
|
||||
private T title = emptyLine();
|
||||
|
||||
private boolean deleted = false;
|
||||
|
||||
/**
|
||||
* Creates a new FastBoard.
|
||||
*
|
||||
* @param player the owner of the scoreboard
|
||||
*/
|
||||
protected FastBoardBase(Player player) {
|
||||
this.player = Objects.requireNonNull(player, "player");
|
||||
this.id = "fb-" + Integer.toHexString(ThreadLocalRandom.current().nextInt());
|
||||
|
||||
try {
|
||||
sendObjectivePacket(ObjectiveMode.CREATE);
|
||||
sendDisplayObjectivePacket();
|
||||
} catch (Throwable t) {
|
||||
throw new RuntimeException("Unable to create scoreboard", t);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the scoreboard title.
|
||||
*
|
||||
* @return the scoreboard title
|
||||
*/
|
||||
public T getTitle() {
|
||||
return this.title;
|
||||
}
|
||||
|
||||
/**
|
||||
* Update the scoreboard title.
|
||||
*
|
||||
* @param title the new scoreboard title
|
||||
* @throws IllegalArgumentException if the title is longer than 32 chars on 1.12 or lower
|
||||
* @throws IllegalStateException if {@link #delete()} was call before
|
||||
*/
|
||||
public void updateTitle(T title) {
|
||||
if (this.title.equals(Objects.requireNonNull(title, "title"))) {
|
||||
return;
|
||||
}
|
||||
|
||||
this.title = title;
|
||||
|
||||
try {
|
||||
sendObjectivePacket(ObjectiveMode.UPDATE);
|
||||
} catch (Throwable t) {
|
||||
throw new RuntimeException("Unable to update scoreboard title", t);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the scoreboard lines.
|
||||
*
|
||||
* @return the scoreboard lines
|
||||
*/
|
||||
public List<T> getLines() {
|
||||
return new ArrayList<>(this.lines);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the specified scoreboard line.
|
||||
*
|
||||
* @param line the line number
|
||||
* @return the line
|
||||
* @throws IndexOutOfBoundsException if the line is higher than {@code size}
|
||||
*/
|
||||
public T getLine(int line) {
|
||||
checkLineNumber(line, true, false);
|
||||
|
||||
return this.lines.get(line);
|
||||
}
|
||||
|
||||
/**
|
||||
* Get how a specific line's score is displayed. On 1.20.2 or below, the value returned isn't used.
|
||||
*
|
||||
* @param line the line number
|
||||
* @return the text of how the line is displayed
|
||||
* @throws IndexOutOfBoundsException if the line is higher than {@code size}
|
||||
*/
|
||||
public Optional<T> getScore(int line) {
|
||||
checkLineNumber(line, true, false);
|
||||
|
||||
return Optional.ofNullable(this.scores.get(line));
|
||||
}
|
||||
|
||||
/**
|
||||
* Update a single scoreboard line.
|
||||
*
|
||||
* @param line the line number
|
||||
* @param text the new line text
|
||||
* @throws IndexOutOfBoundsException if the line is higher than {@link #size() size() + 1}
|
||||
*/
|
||||
public synchronized void updateLine(int line, T text) {
|
||||
updateLine(line, text, null);
|
||||
}
|
||||
|
||||
/**
|
||||
* Update a single scoreboard line including how its score is displayed.
|
||||
* The score will only be displayed on 1.20.3 and higher.
|
||||
*
|
||||
* @param line the line number
|
||||
* @param text the new line text
|
||||
* @param scoreText the new line's score, if null will not change current value
|
||||
* @throws IndexOutOfBoundsException if the line is higher than {@link #size() size() + 1}
|
||||
*/
|
||||
public synchronized void updateLine(int line, T text, T scoreText) {
|
||||
checkLineNumber(line, false, false);
|
||||
|
||||
try {
|
||||
if (line < size()) {
|
||||
this.lines.set(line, text);
|
||||
this.scores.set(line, scoreText);
|
||||
|
||||
sendLineChange(getScoreByLine(line));
|
||||
|
||||
if (customScoresSupported()) {
|
||||
sendScorePacket(getScoreByLine(line), ScoreboardAction.CHANGE);
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
List<T> newLines = new ArrayList<>(this.lines);
|
||||
List<T> newScores = new ArrayList<>(this.scores);
|
||||
|
||||
if (line > size()) {
|
||||
for (int i = size(); i < line; i++) {
|
||||
newLines.add(emptyLine());
|
||||
newScores.add(null);
|
||||
}
|
||||
}
|
||||
|
||||
newLines.add(text);
|
||||
newScores.add(scoreText);
|
||||
|
||||
updateLines(newLines, newScores);
|
||||
} catch (Throwable t) {
|
||||
throw new RuntimeException("Unable to update scoreboard lines", t);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove a scoreboard line.
|
||||
*
|
||||
* @param line the line number
|
||||
*/
|
||||
public synchronized void removeLine(int line) {
|
||||
checkLineNumber(line, false, false);
|
||||
|
||||
if (line >= size()) {
|
||||
return;
|
||||
}
|
||||
|
||||
List<T> newLines = new ArrayList<>(this.lines);
|
||||
List<T> newScores = new ArrayList<>(this.scores);
|
||||
newLines.remove(line);
|
||||
newScores.remove(line);
|
||||
updateLines(newLines, newScores);
|
||||
}
|
||||
|
||||
/**
|
||||
* Update all the scoreboard lines.
|
||||
*
|
||||
* @param lines the new lines
|
||||
* @throws IllegalArgumentException if one line is longer than 30 chars on 1.12 or lower
|
||||
* @throws IllegalStateException if {@link #delete()} was call before
|
||||
*/
|
||||
public void updateLines(T... lines) {
|
||||
updateLines(Arrays.asList(lines));
|
||||
}
|
||||
|
||||
/**
|
||||
* Update the lines of the scoreboard
|
||||
*
|
||||
* @param lines the new scoreboard lines
|
||||
* @throws IllegalArgumentException if one line is longer than 30 chars on 1.12 or lower
|
||||
* @throws IllegalStateException if {@link #delete()} was call before
|
||||
*/
|
||||
public synchronized void updateLines(Collection<T> lines) {
|
||||
updateLines(lines, null);
|
||||
}
|
||||
|
||||
/**
|
||||
* Update the lines and how their score is displayed on the scoreboard.
|
||||
* The scores will only be displayed for servers on 1.20.3 and higher.
|
||||
*
|
||||
* @param lines the new scoreboard lines
|
||||
* @param scores the set for how each line's score should be, if null will fall back to default (blank)
|
||||
* @throws IllegalArgumentException if one line is longer than 30 chars on 1.12 or lower
|
||||
* @throws IllegalArgumentException if lines and scores are not the same size
|
||||
* @throws IllegalStateException if {@link #delete()} was call before
|
||||
*/
|
||||
public synchronized void updateLines(Collection<T> lines, Collection<T> scores) {
|
||||
Objects.requireNonNull(lines, "lines");
|
||||
checkLineNumber(lines.size(), false, true);
|
||||
|
||||
if (scores != null && scores.size() != lines.size()) {
|
||||
throw new IllegalArgumentException("The size of the scores must match the size of the board");
|
||||
}
|
||||
|
||||
List<T> oldLines = new ArrayList<>(this.lines);
|
||||
this.lines.clear();
|
||||
this.lines.addAll(lines);
|
||||
|
||||
List<T> oldScores = new ArrayList<>(this.scores);
|
||||
this.scores.clear();
|
||||
this.scores.addAll(scores != null ? scores : Collections.nCopies(lines.size(), null));
|
||||
|
||||
int linesSize = this.lines.size();
|
||||
|
||||
try {
|
||||
if (oldLines.size() != linesSize) {
|
||||
List<T> oldLinesCopy = new ArrayList<>(oldLines);
|
||||
|
||||
if (oldLines.size() > linesSize) {
|
||||
for (int i = oldLinesCopy.size(); i > linesSize; i--) {
|
||||
sendTeamPacket(i - 1, TeamMode.REMOVE);
|
||||
sendScorePacket(i - 1, ScoreboardAction.REMOVE);
|
||||
oldLines.remove(0);
|
||||
}
|
||||
} else {
|
||||
for (int i = oldLinesCopy.size(); i < linesSize; i++) {
|
||||
sendScorePacket(i, ScoreboardAction.CHANGE);
|
||||
sendTeamPacket(i, TeamMode.CREATE, null, null);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for (int i = 0; i < linesSize; i++) {
|
||||
if (!Objects.equals(getLineByScore(oldLines, i), getLineByScore(i))) {
|
||||
sendLineChange(i);
|
||||
}
|
||||
if (!Objects.equals(getLineByScore(oldScores, i), getLineByScore(this.scores, i))) {
|
||||
sendScorePacket(i, ScoreboardAction.CHANGE);
|
||||
}
|
||||
}
|
||||
} catch (Throwable t) {
|
||||
throw new RuntimeException("Unable to update scoreboard lines", t);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Update how a specified line's score is displayed on the scoreboard. A null value will reset the displayed
|
||||
* text back to default. The scores will only be displayed for servers on 1.20.3 and higher.
|
||||
*
|
||||
* @param line the line number
|
||||
* @param text the text to be displayed as the score. if null, no score will be displayed
|
||||
* @throws IllegalArgumentException if the line number is not in range
|
||||
* @throws IllegalStateException if {@link #delete()} was call before
|
||||
*/
|
||||
public synchronized void updateScore(int line, T text) {
|
||||
checkLineNumber(line, true, false);
|
||||
|
||||
this.scores.set(line, text);
|
||||
|
||||
try {
|
||||
if (customScoresSupported()) {
|
||||
sendScorePacket(getScoreByLine(line), ScoreboardAction.CHANGE);
|
||||
}
|
||||
} catch (Throwable e) {
|
||||
throw new RuntimeException("Unable to update line score", e);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Reset a line's score back to default (blank). The score will only be displayed for servers on 1.20.3 and higher.
|
||||
*
|
||||
* @param line the line number
|
||||
* @throws IllegalArgumentException if the line number is not in range
|
||||
* @throws IllegalStateException if {@link #delete()} was call before
|
||||
*/
|
||||
public synchronized void removeScore(int line) {
|
||||
updateScore(line, null);
|
||||
}
|
||||
|
||||
/**
|
||||
* Update how all lines' scores are displayed. A value of null will reset the displayed text back to default.
|
||||
* The scores will only be displayed for servers on 1.20.3 and higher.
|
||||
*
|
||||
* @param texts the set of texts to be displayed as the scores
|
||||
* @throws IllegalArgumentException if the size of the texts does not match the current size of the board
|
||||
* @throws IllegalStateException if {@link #delete()} was call before
|
||||
*/
|
||||
public synchronized void updateScores(T... texts) {
|
||||
updateScores(Arrays.asList(texts));
|
||||
}
|
||||
|
||||
/**
|
||||
* Update how all lines' scores are displayed. A null value will reset the displayed
|
||||
* text back to default (blank). Only available on 1.20.3+ servers.
|
||||
*
|
||||
* @param texts the set of texts to be displayed as the scores
|
||||
* @throws IllegalArgumentException if the size of the texts does not match the current size of the board
|
||||
* @throws IllegalStateException if {@link #delete()} was call before
|
||||
*/
|
||||
public synchronized void updateScores(Collection<T> texts) {
|
||||
Objects.requireNonNull(texts, "texts");
|
||||
|
||||
if (this.scores.size() != this.lines.size()) {
|
||||
throw new IllegalArgumentException("The size of the scores must match the size of the board");
|
||||
}
|
||||
|
||||
List<T> newScores = new ArrayList<>(texts);
|
||||
for (int i = 0; i < this.scores.size(); i++) {
|
||||
if (Objects.equals(this.scores.get(i), newScores.get(i))) {
|
||||
continue;
|
||||
}
|
||||
|
||||
this.scores.set(i, newScores.get(i));
|
||||
|
||||
try {
|
||||
if (customScoresSupported()) {
|
||||
sendScorePacket(getScoreByLine(i), ScoreboardAction.CHANGE);
|
||||
}
|
||||
} catch (Throwable e) {
|
||||
throw new RuntimeException("Unable to update scores", e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the player who has the scoreboard.
|
||||
*
|
||||
* @return current player for this FastBoard
|
||||
*/
|
||||
public Player getPlayer() {
|
||||
return this.player;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the scoreboard id.
|
||||
*
|
||||
* @return the id
|
||||
*/
|
||||
public String getId() {
|
||||
return this.id;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get if the scoreboard is deleted.
|
||||
*
|
||||
* @return true if the scoreboard is deleted
|
||||
*/
|
||||
public boolean isDeleted() {
|
||||
return this.deleted;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get if the server supports custom scoreboard scores (1.20.3+ servers only).
|
||||
*
|
||||
* @return true if the server supports custom scores
|
||||
*/
|
||||
public boolean customScoresSupported() {
|
||||
return BLANK_NUMBER_FORMAT != null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the scoreboard size (the number of lines).
|
||||
*
|
||||
* @return the size
|
||||
*/
|
||||
public int size() {
|
||||
return this.lines.size();
|
||||
}
|
||||
|
||||
/**
|
||||
* Delete this FastBoard, and will remove the scoreboard for the associated player if he is online.
|
||||
* After this, all uses of {@link #updateLines} and {@link #updateTitle} will throw an {@link IllegalStateException}
|
||||
*
|
||||
* @throws IllegalStateException if this was already call before
|
||||
*/
|
||||
public void delete() {
|
||||
try {
|
||||
for (int i = 0; i < this.lines.size(); i++) {
|
||||
sendTeamPacket(i, TeamMode.REMOVE);
|
||||
}
|
||||
|
||||
sendObjectivePacket(ObjectiveMode.REMOVE);
|
||||
} catch (Throwable t) {
|
||||
throw new RuntimeException("Unable to delete scoreboard", t);
|
||||
}
|
||||
|
||||
this.deleted = true;
|
||||
}
|
||||
|
||||
protected abstract void sendLineChange(int score) throws Throwable;
|
||||
|
||||
protected abstract Object toMinecraftComponent(T value) throws Throwable;
|
||||
|
||||
protected abstract String serializeLine(T value);
|
||||
|
||||
protected abstract T emptyLine();
|
||||
|
||||
private void checkLineNumber(int line, boolean checkInRange, boolean checkMax) {
|
||||
if (line < 0) {
|
||||
throw new IllegalArgumentException("Line number must be positive");
|
||||
}
|
||||
|
||||
if (checkInRange && line >= this.lines.size()) {
|
||||
throw new IllegalArgumentException("Line number must be under " + this.lines.size());
|
||||
}
|
||||
|
||||
if (checkMax && line >= COLOR_CODES.length - 1) {
|
||||
throw new IllegalArgumentException("Line number is too high: " + line);
|
||||
}
|
||||
}
|
||||
|
||||
protected int getScoreByLine(int line) {
|
||||
return this.lines.size() - line - 1;
|
||||
}
|
||||
|
||||
protected T getLineByScore(int score) {
|
||||
return getLineByScore(this.lines, score);
|
||||
}
|
||||
|
||||
protected T getLineByScore(List<T> lines, int score) {
|
||||
return score < lines.size() ? lines.get(lines.size() - score - 1) : null;
|
||||
}
|
||||
|
||||
protected void sendObjectivePacket(ObjectiveMode mode) throws Throwable {
|
||||
Object packet = PACKET_SB_OBJ.invoke();
|
||||
|
||||
setField(packet, String.class, this.id);
|
||||
setField(packet, int.class, mode.ordinal());
|
||||
|
||||
if (mode != ObjectiveMode.REMOVE) {
|
||||
setComponentField(packet, this.title, 1);
|
||||
setField(packet, Optional.class, Optional.empty()); // Number format for 1.20.5+, previously nullable
|
||||
|
||||
if (VersionType.V1_8.isHigherOrEqual()) {
|
||||
setField(packet, ENUM_SB_HEALTH_DISPLAY, ENUM_SB_HEALTH_DISPLAY_INTEGER);
|
||||
}
|
||||
} else if (VERSION_TYPE == VersionType.V1_7) {
|
||||
setField(packet, String.class, "", 1);
|
||||
}
|
||||
|
||||
sendPacket(packet);
|
||||
}
|
||||
|
||||
protected void sendDisplayObjectivePacket() throws Throwable {
|
||||
Object packet = PACKET_SB_DISPLAY_OBJ.invoke();
|
||||
|
||||
setField(packet, DISPLAY_SLOT_TYPE, SIDEBAR_DISPLAY_SLOT); // Position
|
||||
setField(packet, String.class, this.id); // Score Name
|
||||
|
||||
sendPacket(packet);
|
||||
}
|
||||
|
||||
protected void sendScorePacket(int score, ScoreboardAction action) throws Throwable {
|
||||
if (VersionType.V1_17.isHigherOrEqual()) {
|
||||
sendModernScorePacket(score, action);
|
||||
return;
|
||||
}
|
||||
|
||||
Object packet = PACKET_SB_SET_SCORE.invoke();
|
||||
|
||||
setField(packet, String.class, COLOR_CODES[score], 0); // Player Name
|
||||
|
||||
if (VersionType.V1_8.isHigherOrEqual()) {
|
||||
Object enumAction = action == ScoreboardAction.REMOVE
|
||||
? ENUM_SB_ACTION_REMOVE : ENUM_SB_ACTION_CHANGE;
|
||||
setField(packet, ENUM_SB_ACTION, enumAction);
|
||||
} else {
|
||||
setField(packet, int.class, action.ordinal(), 1); // Action
|
||||
}
|
||||
|
||||
if (action == ScoreboardAction.CHANGE) {
|
||||
setField(packet, String.class, this.id, 1); // Objective Name
|
||||
setField(packet, int.class, score); // Score
|
||||
}
|
||||
|
||||
sendPacket(packet);
|
||||
}
|
||||
|
||||
private void sendModernScorePacket(int score, ScoreboardAction action) throws Throwable {
|
||||
String objName = COLOR_CODES[score];
|
||||
Object enumAction = action == ScoreboardAction.REMOVE
|
||||
? ENUM_SB_ACTION_REMOVE : ENUM_SB_ACTION_CHANGE;
|
||||
|
||||
if (PACKET_SB_RESET_SCORE == null) { // Pre 1.20.3
|
||||
sendPacket(PACKET_SB_SET_SCORE.invoke(enumAction, this.id, objName, score));
|
||||
return;
|
||||
}
|
||||
|
||||
if (action == ScoreboardAction.REMOVE) {
|
||||
sendPacket(PACKET_SB_RESET_SCORE.invoke(objName, this.id));
|
||||
return;
|
||||
}
|
||||
|
||||
T scoreFormat = getLineByScore(this.scores, score);
|
||||
Object format = scoreFormat != null
|
||||
? FIXED_NUMBER_FORMAT.invoke(toMinecraftComponent(scoreFormat))
|
||||
: BLANK_NUMBER_FORMAT;
|
||||
Object scorePacket = SCORE_OPTIONAL_COMPONENTS
|
||||
? PACKET_SB_SET_SCORE.invoke(objName, this.id, score, Optional.empty(), Optional.of(format))
|
||||
: PACKET_SB_SET_SCORE.invoke(objName, this.id, score, null, format);
|
||||
|
||||
sendPacket(scorePacket);
|
||||
}
|
||||
|
||||
protected void sendTeamPacket(int score, TeamMode mode) throws Throwable {
|
||||
sendTeamPacket(score, mode, null, null);
|
||||
}
|
||||
|
||||
protected void sendTeamPacket(int score, TeamMode mode, T prefix, T suffix)
|
||||
throws Throwable {
|
||||
if (mode == TeamMode.ADD_PLAYERS || mode == TeamMode.REMOVE_PLAYERS) {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
|
||||
Object packet = PACKET_SB_TEAM.invoke();
|
||||
|
||||
setField(packet, String.class, this.id + ':' + score); // Team name
|
||||
setField(packet, int.class, mode.ordinal(), VERSION_TYPE == VersionType.V1_8 ? 1 : 0); // Update mode
|
||||
|
||||
if (mode == TeamMode.REMOVE) {
|
||||
sendPacket(packet);
|
||||
return;
|
||||
}
|
||||
|
||||
if (VersionType.V1_17.isHigherOrEqual()) {
|
||||
Object team = PACKET_SB_SERIALIZABLE_TEAM.invoke();
|
||||
// Since the packet is initialized with null values, we need to change more things.
|
||||
setComponentField(team, null, 0); // Display name
|
||||
setField(team, CHAT_FORMAT_ENUM, RESET_FORMATTING); // Color
|
||||
setComponentField(team, prefix, 1); // Prefix
|
||||
setComponentField(team, suffix, 2); // Suffix
|
||||
setField(team, String.class, "always", 0); // Visibility before 1.21.5
|
||||
setField(team, String.class, "always", 1); // Collisions before 1.21.5
|
||||
setField(team, ENUM_VISIBILITY, ENUM_VISIBILITY_ALWAYS, 0); // 1.21.5+
|
||||
setField(team, ENUM_COLLISION_RULE, ENUM_COLLISION_RULE_ALWAYS, 0); // 1.21.5+
|
||||
setField(packet, Optional.class, Optional.of(team));
|
||||
} else {
|
||||
setComponentField(packet, prefix, 2); // Prefix
|
||||
setComponentField(packet, suffix, 3); // Suffix
|
||||
setField(packet, String.class, "always", 4); // Visibility for 1.8+
|
||||
setField(packet, String.class, "always", 5); // Collisions for 1.9+
|
||||
}
|
||||
|
||||
if (mode == TeamMode.CREATE) {
|
||||
setField(packet, Collection.class, Collections.singletonList(COLOR_CODES[score])); // Players in the team
|
||||
}
|
||||
|
||||
sendPacket(packet);
|
||||
}
|
||||
|
||||
private void sendPacket(Object packet) throws Throwable {
|
||||
if (this.deleted) {
|
||||
throw new IllegalStateException("This FastBoard is deleted");
|
||||
}
|
||||
|
||||
if (this.player.isOnline()) {
|
||||
Object entityPlayer = PLAYER_GET_HANDLE.invoke(this.player);
|
||||
Object playerConnection = PLAYER_CONNECTION.invoke(entityPlayer);
|
||||
SEND_PACKET.invoke(playerConnection, packet);
|
||||
}
|
||||
}
|
||||
|
||||
private void setField(Object object, Class<?> fieldType, Object value)
|
||||
throws ReflectiveOperationException {
|
||||
setField(object, fieldType, value, 0);
|
||||
}
|
||||
|
||||
private void setField(Object packet, Class<?> fieldType, Object value, int count)
|
||||
throws ReflectiveOperationException {
|
||||
int i = 0;
|
||||
for (Field field : PACKETS.get(packet.getClass())) {
|
||||
if (field.getType() == fieldType && count == i++) {
|
||||
field.set(packet, value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void setComponentField(Object packet, T value, int count) throws Throwable {
|
||||
if (!VersionType.V1_13.isHigherOrEqual()) {
|
||||
String line = value != null ? serializeLine(value) : "";
|
||||
setField(packet, String.class, line, count);
|
||||
return;
|
||||
}
|
||||
|
||||
int i = 0;
|
||||
for (Field field : PACKETS.get(packet.getClass())) {
|
||||
if ((field.getType() == String.class || field.getType() == CHAT_COMPONENT_CLASS) && count == i++) {
|
||||
field.set(packet, toMinecraftComponent(value));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public enum ObjectiveMode {
|
||||
CREATE, REMOVE, UPDATE
|
||||
}
|
||||
|
||||
public enum TeamMode {
|
||||
CREATE, REMOVE, UPDATE, ADD_PLAYERS, REMOVE_PLAYERS
|
||||
}
|
||||
|
||||
public enum ScoreboardAction {
|
||||
CHANGE, REMOVE
|
||||
}
|
||||
|
||||
enum VersionType {
|
||||
V1_7, V1_8, V1_13, V1_17;
|
||||
|
||||
public boolean isHigherOrEqual() {
|
||||
return VERSION_TYPE.ordinal() >= ordinal();
|
||||
}
|
||||
}
|
||||
}
|
167
src/main/java/pb/ajneb97/lib/fastboard/FastReflection.java
Normal file
167
src/main/java/pb/ajneb97/lib/fastboard/FastReflection.java
Normal file
@@ -0,0 +1,167 @@
|
||||
/*
|
||||
* This file is part of FastBoard, licensed under the MIT License.
|
||||
*
|
||||
* Copyright (c) 2019-2023 MrMicky
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in all
|
||||
* copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
* SOFTWARE.
|
||||
*/
|
||||
package pb.ajneb97.lib.fastboard;
|
||||
|
||||
import org.bukkit.Bukkit;
|
||||
|
||||
import java.lang.invoke.MethodHandle;
|
||||
import java.lang.invoke.MethodHandles;
|
||||
import java.lang.invoke.MethodType;
|
||||
import java.lang.reflect.Field;
|
||||
import java.util.Optional;
|
||||
import java.util.function.Predicate;
|
||||
|
||||
/**
|
||||
* Small reflection utility class to use CraftBukkit and NMS.
|
||||
*
|
||||
* @author MrMicky
|
||||
*/
|
||||
public final class FastReflection {
|
||||
|
||||
private static final String NM_PACKAGE = "net.minecraft";
|
||||
private static final String OBC_PACKAGE = Bukkit.getServer().getClass().getPackage().getName();
|
||||
private static final String NMS_PACKAGE = OBC_PACKAGE.replace("org.bukkit.craftbukkit", NM_PACKAGE + ".server");
|
||||
|
||||
private static final MethodType VOID_METHOD_TYPE = MethodType.methodType(void.class);
|
||||
private static final boolean NMS_REPACKAGED = optionalClass(NM_PACKAGE + ".network.protocol.Packet").isPresent();
|
||||
private static final boolean MOJANG_MAPPINGS = optionalClass(NM_PACKAGE + ".network.chat.Component").isPresent();
|
||||
|
||||
private static volatile Object theUnsafe;
|
||||
|
||||
private FastReflection() {
|
||||
throw new UnsupportedOperationException();
|
||||
}
|
||||
|
||||
public static boolean isRepackaged() {
|
||||
return NMS_REPACKAGED;
|
||||
}
|
||||
|
||||
public static String nmsClassName(String post1_17package, String className) {
|
||||
if (NMS_REPACKAGED) {
|
||||
String classPackage = post1_17package == null ? NM_PACKAGE : NM_PACKAGE + '.' + post1_17package;
|
||||
|
||||
return classPackage + '.' + className;
|
||||
}
|
||||
|
||||
return NMS_PACKAGE + '.' + className;
|
||||
}
|
||||
|
||||
public static Class<?> nmsClass(String post1_17package, String className) throws ClassNotFoundException {
|
||||
return Class.forName(nmsClassName(post1_17package, className));
|
||||
}
|
||||
|
||||
public static Class<?> nmsClass(String post1_17package, String spigotClass, String mojangClass) throws ClassNotFoundException {
|
||||
return nmsClass(post1_17package, MOJANG_MAPPINGS ? mojangClass : spigotClass);
|
||||
}
|
||||
|
||||
public static Optional<Class<?>> nmsOptionalClass(String post1_17package, String className) {
|
||||
return optionalClass(nmsClassName(post1_17package, className));
|
||||
}
|
||||
|
||||
public static Optional<Class<?>> nmsOptionalClass(String post1_17package, String spigotClass, String mojangClass) {
|
||||
return optionalClass(nmsClassName(post1_17package, MOJANG_MAPPINGS ? mojangClass : spigotClass));
|
||||
}
|
||||
|
||||
public static String obcClassName(String className) {
|
||||
return OBC_PACKAGE + '.' + className;
|
||||
}
|
||||
|
||||
public static Class<?> obcClass(String className) throws ClassNotFoundException {
|
||||
return Class.forName(obcClassName(className));
|
||||
}
|
||||
|
||||
public static Optional<Class<?>> obcOptionalClass(String className) {
|
||||
return optionalClass(obcClassName(className));
|
||||
}
|
||||
|
||||
public static Optional<Class<?>> optionalClass(String className) {
|
||||
try {
|
||||
return Optional.of(Class.forName(className));
|
||||
} catch (ClassNotFoundException e) {
|
||||
return Optional.empty();
|
||||
}
|
||||
}
|
||||
|
||||
public static Object enumValueOf(Class<?> enumClass, String enumName) {
|
||||
return Enum.valueOf(enumClass.asSubclass(Enum.class), enumName);
|
||||
}
|
||||
|
||||
public static Object enumValueOf(Class<?> enumClass, String enumName, int fallbackOrdinal) {
|
||||
try {
|
||||
return enumValueOf(enumClass, enumName);
|
||||
} catch (IllegalArgumentException e) {
|
||||
Object[] constants = enumClass.getEnumConstants();
|
||||
if (constants.length > fallbackOrdinal) {
|
||||
return constants[fallbackOrdinal];
|
||||
}
|
||||
throw e;
|
||||
}
|
||||
}
|
||||
|
||||
static Class<?> innerClass(Class<?> parentClass, Predicate<Class<?>> classPredicate) throws ClassNotFoundException {
|
||||
for (Class<?> innerClass : parentClass.getDeclaredClasses()) {
|
||||
if (classPredicate.test(innerClass)) {
|
||||
return innerClass;
|
||||
}
|
||||
}
|
||||
throw new ClassNotFoundException("No class in " + parentClass.getCanonicalName() + " matches the predicate.");
|
||||
}
|
||||
|
||||
static Optional<MethodHandle> optionalConstructor(Class<?> declaringClass, MethodHandles.Lookup lookup, MethodType type) throws IllegalAccessException {
|
||||
try {
|
||||
return Optional.of(lookup.findConstructor(declaringClass, type));
|
||||
} catch (NoSuchMethodException e) {
|
||||
return Optional.empty();
|
||||
}
|
||||
}
|
||||
|
||||
public static PacketConstructor findPacketConstructor(Class<?> packetClass, MethodHandles.Lookup lookup) throws Exception {
|
||||
try {
|
||||
MethodHandle constructor = lookup.findConstructor(packetClass, VOID_METHOD_TYPE);
|
||||
return constructor::invoke;
|
||||
} catch (NoSuchMethodException | IllegalAccessException e) {
|
||||
// try below with Unsafe
|
||||
}
|
||||
|
||||
if (theUnsafe == null) {
|
||||
synchronized (FastReflection.class) {
|
||||
if (theUnsafe == null) {
|
||||
Class<?> unsafeClass = Class.forName("sun.misc.Unsafe");
|
||||
Field theUnsafeField = unsafeClass.getDeclaredField("theUnsafe");
|
||||
theUnsafeField.setAccessible(true);
|
||||
theUnsafe = theUnsafeField.get(null);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
MethodType allocateMethodType = MethodType.methodType(Object.class, Class.class);
|
||||
MethodHandle allocateMethod = lookup.findVirtual(theUnsafe.getClass(), "allocateInstance", allocateMethodType);
|
||||
return () -> allocateMethod.invoke(theUnsafe, packetClass);
|
||||
}
|
||||
|
||||
@FunctionalInterface
|
||||
interface PacketConstructor {
|
||||
Object invoke() throws Throwable;
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user