diff --git a/.gitignore b/.gitignore index f9e1624..b63da45 100644 --- a/.gitignore +++ b/.gitignore @@ -1,117 +1,42 @@ -# User-specific stuff -.idea/ - -*.iml -*.ipr -*.iws - -# IntelliJ -out/ -# mpeltonen/sbt-idea plugin -.idea_modules/ - -# JIRA plugin -atlassian-ide-plugin.xml - -# Compiled class file -*.class - -# Log file -*.log - -# BlueJ files -*.ctxt - -# Package Files # -*.war -*.nar -*.ear -*.zip -*.tar.gz -*.rar - -# virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml -hs_err_pid* - -*~ - -# temporary files which can be created if a process still has a handle open of a deleted file -.fuse_hidden* - -# KDE directory preferences -.directory - -# Linux trash folder which might appear on any partition or disk -.Trash-* - -# .nfs files are created when an open file is removed but is still being accessed -.nfs* - -# General -.DS_Store -.AppleDouble -.LSOverride - -# Icon must end with two \r -Icon - -# Thumbnails -._* - -# Files that might appear in the root of a volume -.DocumentRevisions-V100 -.fseventsd -.Spotlight-V100 -.TemporaryItems -.Trashes -.VolumeIcon.icns -.com.apple.timemachine.donotpresent - -# Directories potentially created on remote AFP share -.AppleDB -.AppleDesktop -Network Trash Folder -Temporary Items -.apdisk - -# Windows thumbnail cache files -Thumbs.db -Thumbs.db:encryptable -ehthumbs.db -ehthumbs_vista.db - -# Dump file -*.stackdump - -# Folder config file -[Dd]esktop.ini - -# Recycle Bin used on file shares -$RECYCLE.BIN/ - -# Windows Installer files -*.cab -*.msi -*.msix -*.msm -*.msp - -# Windows shortcuts -*.lnk - .gradle build/ +!gradle/wrapper/gradle-wrapper.jar +!**/src/main/**/build/ +!**/src/test/**/build/ -# Ignore Gradle GUI config -gradle-app.setting +### IntelliJ IDEA ### +.idea/modules.xml +.idea/jarRepositories.xml +.idea/compiler.xml +.idea/libraries/ +*.iws +*.iml +*.ipr +out/ +!**/src/main/**/out/ +!**/src/test/**/out/ -# Cache of project -.gradletasknamecache +### Eclipse ### +.apt_generated +.classpath +.factorypath +.project +.settings +.springBeans +.sts4-cache +bin/ +!**/src/main/**/bin/ +!**/src/test/**/bin/ -**/build/ +### NetBeans ### +/nbproject/private/ +/nbbuild/ +/dist/ +/nbdist/ +/.nb-gradle/ -# Common working directory -run/ +### VS Code ### +.vscode/ -# Avoid ignoring Gradle wrapper jar file (.jar files are usually ignored) -!gradle-wrapper.jar +### Mac OS ### +.DS_Store \ No newline at end of file diff --git a/api/.gitignore b/api/.gitignore new file mode 100644 index 0000000..b63da45 --- /dev/null +++ b/api/.gitignore @@ -0,0 +1,42 @@ +.gradle +build/ +!gradle/wrapper/gradle-wrapper.jar +!**/src/main/**/build/ +!**/src/test/**/build/ + +### IntelliJ IDEA ### +.idea/modules.xml +.idea/jarRepositories.xml +.idea/compiler.xml +.idea/libraries/ +*.iws +*.iml +*.ipr +out/ +!**/src/main/**/out/ +!**/src/test/**/out/ + +### Eclipse ### +.apt_generated +.classpath +.factorypath +.project +.settings +.springBeans +.sts4-cache +bin/ +!**/src/main/**/bin/ +!**/src/test/**/bin/ + +### NetBeans ### +/nbproject/private/ +/nbbuild/ +/dist/ +/nbdist/ +/.nb-gradle/ + +### VS Code ### +.vscode/ + +### Mac OS ### +.DS_Store \ No newline at end of file diff --git a/api/build.gradle.kts b/api/build.gradle.kts new file mode 100644 index 0000000..c742b50 --- /dev/null +++ b/api/build.gradle.kts @@ -0,0 +1,4 @@ +dependencies { + compileOnly("dev.folia:folia-api:1.20.1-R0.1-SNAPSHOT") + compileOnly("me.clip:placeholderapi:2.11.5") +} \ No newline at end of file diff --git a/api/src/main/java/net/momirealms/customnameplates/api/CustomNameplatesPlugin.java b/api/src/main/java/net/momirealms/customnameplates/api/CustomNameplatesPlugin.java new file mode 100644 index 0000000..aad50cb --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/CustomNameplatesPlugin.java @@ -0,0 +1,112 @@ +package net.momirealms.customnameplates.api; + +import net.momirealms.customnameplates.api.manager.*; +import net.momirealms.customnameplates.api.scheduler.Scheduler; +import org.bukkit.configuration.file.YamlConfiguration; +import org.bukkit.plugin.java.JavaPlugin; + +public abstract class CustomNameplatesPlugin extends JavaPlugin { + + protected static CustomNameplatesPlugin instance; + protected Scheduler scheduler; + protected StorageManager storageManager; + protected VersionManager versionManager; + protected AdventureManager adventureManager; + protected RequirementManager requirementManager; + protected BossBarManager bossBarManager; + protected ImageManager imageManager; + protected PlaceholderManager placeholderManager; + protected ResourcePackManager resourcePackManager; + protected BackGroundManager backGroundManager; + protected TeamManager teamManager; + protected NameplateManager nameplateManager; + protected ActionBarManager actionBarManager; + + protected CustomNameplatesPlugin() { + instance = this; + } + + /* Get plugin instance */ + public static CustomNameplatesPlugin getInstance() { + return instance; + } + + /* Get plugin instance */ + public static CustomNameplatesPlugin get() { + return instance; + } + + /* reload the plugin */ + public abstract void reload(boolean generatePack); + + /* Get the scheduler */ + public Scheduler getScheduler() { + return scheduler; + } + + /* Get the storage manager */ + public StorageManager getStorageManager() { + return storageManager; + } + + /* Get the requirement manager */ + public RequirementManager getRequirementManager() { + return requirementManager; + } + + /* Get the image manager */ + public ImageManager getImageManager() { + return imageManager; + } + + /* Get the background manager */ + public BackGroundManager getBackGroundManager() { + return backGroundManager; + } + + /* Get the resource pack manager */ + public ResourcePackManager getResourcePackManager() { + return resourcePackManager; + } + + /* Get the adventure manager */ + public AdventureManager getAdventure() { + return adventureManager; + } + + /* Get the version manager */ + public VersionManager getVersionManager() { + return versionManager; + } + + /* Get the bossbar manager */ + public BossBarManager getBossBarManager() { + return bossBarManager; + } + + /* Get the placeholder manager */ + public PlaceholderManager getPlaceholderManager() { + return placeholderManager; + } + + /* Get the team manager */ + public TeamManager getTeamManager() { + return teamManager; + } + + /* Get the actionbar manager */ + public ActionBarManager getActionBarManager() { + return actionBarManager; + } + + /* Get the nameplate manager */ + public NameplateManager getNameplateManager() { + return nameplateManager; + } + + /* debug get config by file name */ + public abstract YamlConfiguration getConfig(String file); + + /* debug message */ + public abstract void debug(String s); +} diff --git a/src/main/java/net/momirealms/customnameplates/object/scheduler/BukkitTimerTask.java b/api/src/main/java/net/momirealms/customnameplates/api/common/Key.java similarity index 51% rename from src/main/java/net/momirealms/customnameplates/object/scheduler/BukkitTimerTask.java rename to api/src/main/java/net/momirealms/customnameplates/api/common/Key.java index efd7fe9..52829e1 100644 --- a/src/main/java/net/momirealms/customnameplates/object/scheduler/BukkitTimerTask.java +++ b/api/src/main/java/net/momirealms/customnameplates/api/common/Key.java @@ -15,25 +15,32 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.object.scheduler; +package net.momirealms.customnameplates.api.common; -import org.bukkit.scheduler.BukkitTask; +public record Key(String namespace, String value) { -public class BukkitTimerTask implements TimerTask { - - private final BukkitTask bukkitTask; - - public BukkitTimerTask(BukkitTask bukkitTask) { - this.bukkitTask = bukkitTask; + public static Key of(String namespace, String value) { + return new Key(namespace, value); } @Override - public void cancel() { - bukkitTask.cancel(); + public int hashCode() { + int result = this.namespace.hashCode(); + result = (31 * result) + this.value.hashCode(); + return result; } @Override - public boolean isCancelled() { - return bukkitTask.isCancelled(); + public boolean equals(Object obj) { + if (obj == null) { + return false; + } + if (!(obj instanceof Key key)) return false; + return this.namespace.equals(key.namespace()) && this.value.equals(key.value()); + } + + @Override + public String toString() { + return namespace + ":" + value; } } diff --git a/src/main/java/net/momirealms/customnameplates/object/placeholders/StaticText.java b/api/src/main/java/net/momirealms/customnameplates/api/common/Pair.java similarity index 76% rename from src/main/java/net/momirealms/customnameplates/object/placeholders/StaticText.java rename to api/src/main/java/net/momirealms/customnameplates/api/common/Pair.java index b9c4fac..3e5f923 100644 --- a/src/main/java/net/momirealms/customnameplates/object/placeholders/StaticText.java +++ b/api/src/main/java/net/momirealms/customnameplates/api/common/Pair.java @@ -15,14 +15,11 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.object.placeholders; +package net.momirealms.customnameplates.api.common; -public record StaticText(String text, int value, StaticState staticState) { +public record Pair(L left, R right) { - public enum StaticState { - - LEFT, - MIDDLE, - RIGHT + public static Pair of(final L left, final R right) { + return new Pair<>(left, right); } } \ No newline at end of file diff --git a/api/src/main/java/net/momirealms/customnameplates/api/common/SimpleLocation.java b/api/src/main/java/net/momirealms/customnameplates/api/common/SimpleLocation.java new file mode 100644 index 0000000..1102057 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/common/SimpleLocation.java @@ -0,0 +1,63 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.api.common; + +import org.bukkit.Location; + +import java.util.Objects; + +public record SimpleLocation(String worldName, int x, int y, int z) { + + @Override + public boolean equals(Object obj) { + if (obj == null) { + return false; + } + if (getClass() != obj.getClass()) { + return false; + } + final SimpleLocation other = (SimpleLocation) obj; + if (!Objects.equals(worldName, other.worldName())) { + return false; + } + if (Double.doubleToLongBits(this.x) != Double.doubleToLongBits(other.x)) { + return false; + } + if (Double.doubleToLongBits(this.y) != Double.doubleToLongBits(other.y)) { + return false; + } + if (Double.doubleToLongBits(this.z) != Double.doubleToLongBits(other.z)) { + return false; + } + return true; + } + + @Override + public int hashCode() { + int hash = 3; + hash = 19 * hash + (worldName != null ? worldName.hashCode() : 0); + hash = 19 * hash + (int) (Double.doubleToLongBits(this.x) ^ (Double.doubleToLongBits(this.x) >>> 32)); + hash = 19 * hash + (int) (Double.doubleToLongBits(this.y) ^ (Double.doubleToLongBits(this.y) >>> 32)); + hash = 19 * hash + (int) (Double.doubleToLongBits(this.z) ^ (Double.doubleToLongBits(this.z) >>> 32)); + return hash; + } + + public static SimpleLocation getByBukkitLocation(Location location) { + return new SimpleLocation(location.getWorld().getName(), location.getBlockX(), location.getBlockY(), location.getBlockZ()); + } +} \ No newline at end of file diff --git a/api/src/main/java/net/momirealms/customnameplates/api/common/Tuple.java b/api/src/main/java/net/momirealms/customnameplates/api/common/Tuple.java new file mode 100644 index 0000000..a1ed9e4 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/common/Tuple.java @@ -0,0 +1,59 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.api.common; + +public class Tuple { + + private L left; + private M mid; + private R right; + + public Tuple(L left, M mid, R right) { + this.left = left; + this.mid = mid; + this.right = right; + } + + public static Tuple of(final L left, final M mid, final R right) { + return new Tuple<>(left, mid, right); + } + + public L getLeft() { + return left; + } + + public void setLeft(L left) { + this.left = left; + } + + public M getMid() { + return mid; + } + + public void setMid(M mid) { + this.mid = mid; + } + + public R getRight() { + return right; + } + + public void setRight(R right) { + this.right = right; + } +} \ No newline at end of file diff --git a/api/src/main/java/net/momirealms/customnameplates/api/data/OnlineUser.java b/api/src/main/java/net/momirealms/customnameplates/api/data/OnlineUser.java new file mode 100644 index 0000000..f536e75 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/data/OnlineUser.java @@ -0,0 +1,79 @@ +package net.momirealms.customnameplates.api.data; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.mechanic.nameplate.Nameplate; +import org.bukkit.entity.Player; + +import java.util.UUID; + +public class OnlineUser { + + private final Player player; + private String nameplate; + private String bubble; + + public OnlineUser(Player player, PlayerData playerData) { + this.player = player; + this.nameplate = playerData.getNameplate(); + this.bubble = playerData.getBubble(); + } + + public PlayerData toPlayerData() { + return PlayerData.builder() + .setBubble(bubble) + .setNameplate(nameplate) + .build(); + } + + public UUID getUUID() { + return player.getUniqueId(); + } + + public Player getPlayer() { + return player; + } + + /** + * Get the original nameplate key from data + */ + public String getNameplateKey() { + return nameplate; + } + + /** + * Get the original bubble key from data + */ + public String getBubbleKey() { + return bubble; + } + + /** + * This value might be inconsistent with the key get by "getNameplateKey()" + * Because if a player doesn't have a nameplate, his nameplate would be the default one + */ + public Nameplate getNameplate() { + String temp = nameplate; + if (temp.equals("none")) { + temp = CustomNameplatesPlugin.get().getNameplateManager().getDefaultNameplate(); + } + return CustomNameplatesPlugin.get().getNameplateManager().getNameplate(temp); + } + + /** + * Set nameplate for a player + * + * @param nameplate nameplate + */ + public void setNameplate(String nameplate) { + this.nameplate = nameplate; + } + + /** + * Set bubble for a player + * + * @param bubble bubble + */ + public void setBubble(String bubble) { + this.bubble = bubble; + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/data/PlayerData.java b/api/src/main/java/net/momirealms/customnameplates/api/data/PlayerData.java new file mode 100644 index 0000000..258ad99 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/data/PlayerData.java @@ -0,0 +1,58 @@ +package net.momirealms.customnameplates.api.data; + +import com.google.gson.annotations.SerializedName; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +public class PlayerData { + + @SerializedName("nameplate") + private String nameplate; + @SerializedName("bubble") + private String bubble; + + public static Builder builder() { + return new Builder(); + } + + public static PlayerData empty() { + return new PlayerData.Builder() + .setNameplate("none") + .setBubble("none") + .build(); + } + + public String getNameplate() { + return nameplate; + } + + public String getBubble() { + return bubble; + } + + public static class Builder { + + private final PlayerData playerData; + + public Builder() { + this.playerData = new PlayerData(); + } + + @NotNull + public Builder setNameplate(@Nullable String nameplate) { + this.playerData.nameplate = nameplate; + return this; + } + + @NotNull + public Builder setBubble(@Nullable String bubble) { + this.playerData.bubble = bubble; + return this; + } + + @NotNull + public PlayerData build() { + return this.playerData; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/event/CustomNameplatesReloadEvent.java b/api/src/main/java/net/momirealms/customnameplates/api/event/CustomNameplatesReloadEvent.java new file mode 100644 index 0000000..9044641 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/event/CustomNameplatesReloadEvent.java @@ -0,0 +1,30 @@ +package net.momirealms.customnameplates.api.event; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import org.bukkit.event.Event; +import org.bukkit.event.HandlerList; +import org.jetbrains.annotations.NotNull; + +public class CustomNameplatesReloadEvent extends Event { + + private static final HandlerList handlerList = new HandlerList(); + private final CustomNameplatesPlugin plugin; + + public CustomNameplatesReloadEvent(CustomNameplatesPlugin plugin) { + this.plugin = plugin; + } + + public static HandlerList getHandlerList() { + return handlerList; + } + + @NotNull + @Override + public HandlerList getHandlers() { + return getHandlerList(); + } + + public CustomNameplatesPlugin getPlugin() { + return plugin; + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/event/NameplateDataLoadEvent.java b/api/src/main/java/net/momirealms/customnameplates/api/event/NameplateDataLoadEvent.java new file mode 100644 index 0000000..ae47291 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/event/NameplateDataLoadEvent.java @@ -0,0 +1,39 @@ +package net.momirealms.customnameplates.api.event; + +import net.momirealms.customnameplates.api.data.OnlineUser; +import org.bukkit.event.Event; +import org.bukkit.event.HandlerList; +import org.jetbrains.annotations.NotNull; + +import java.util.UUID; + +public class NameplateDataLoadEvent extends Event { + + private static final HandlerList handlerList = new HandlerList(); + private final UUID uuid; + private final OnlineUser onlineUser; + + public NameplateDataLoadEvent(UUID uuid, OnlineUser onlineUser) { + super(true); + this.uuid = uuid; + this.onlineUser = onlineUser; + } + + public UUID getUUID() { + return uuid; + } + + public OnlineUser getOnlineUser() { + return onlineUser; + } + + public static HandlerList getHandlerList() { + return handlerList; + } + + @NotNull + @Override + public HandlerList getHandlers() { + return getHandlerList(); + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerEquipBubbleEvent.java b/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerEquipBubbleEvent.java new file mode 100644 index 0000000..b92e049 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerEquipBubbleEvent.java @@ -0,0 +1,43 @@ +package net.momirealms.customnameplates.api.event; + +import org.bukkit.entity.Player; +import org.bukkit.event.Cancellable; +import org.bukkit.event.HandlerList; +import org.bukkit.event.player.PlayerEvent; +import org.jetbrains.annotations.NotNull; + +public class PlayerEquipBubbleEvent extends PlayerEvent implements Cancellable { + + private static final HandlerList handlerList = new HandlerList(); + private final String bubble; + private boolean isCancelled; + + public PlayerEquipBubbleEvent(@NotNull Player who, String bubble) { + super(who); + this.bubble = bubble; + this.isCancelled = false; + } + + @Override + public @NotNull HandlerList getHandlers() { + return handlerList; + } + + public static HandlerList getHandlerList() { + return handlerList; + } + + public String getBubble() { + return bubble; + } + + @Override + public boolean isCancelled() { + return isCancelled; + } + + @Override + public void setCancelled(boolean cancel) { + isCancelled = cancel; + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerEquipNameplateEvent.java b/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerEquipNameplateEvent.java new file mode 100644 index 0000000..522339e --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerEquipNameplateEvent.java @@ -0,0 +1,43 @@ +package net.momirealms.customnameplates.api.event; + +import org.bukkit.entity.Player; +import org.bukkit.event.Cancellable; +import org.bukkit.event.HandlerList; +import org.bukkit.event.player.PlayerEvent; +import org.jetbrains.annotations.NotNull; + +public class PlayerEquipNameplateEvent extends PlayerEvent implements Cancellable { + + private static final HandlerList handlerList = new HandlerList(); + private final String nameplate; + private boolean isCancelled; + + public PlayerEquipNameplateEvent(@NotNull Player who, String nameplate) { + super(who); + this.nameplate = nameplate; + this.isCancelled = false; + } + + @Override + public @NotNull HandlerList getHandlers() { + return handlerList; + } + + public static HandlerList getHandlerList() { + return handlerList; + } + + public String getNameplate() { + return nameplate; + } + + @Override + public boolean isCancelled() { + return isCancelled; + } + + @Override + public void setCancelled(boolean cancel) { + isCancelled = cancel; + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerUnequipBubbleEvent.java b/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerUnequipBubbleEvent.java new file mode 100644 index 0000000..4a1fe0f --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerUnequipBubbleEvent.java @@ -0,0 +1,37 @@ +package net.momirealms.customnameplates.api.event; + +import org.bukkit.entity.Player; +import org.bukkit.event.Cancellable; +import org.bukkit.event.HandlerList; +import org.bukkit.event.player.PlayerEvent; +import org.jetbrains.annotations.NotNull; + +public class PlayerUnequipBubbleEvent extends PlayerEvent implements Cancellable { + + private static final HandlerList handlerList = new HandlerList(); + private boolean isCancelled; + + public PlayerUnequipBubbleEvent(@NotNull Player who) { + super(who); + this.isCancelled = false; + } + + @Override + public @NotNull HandlerList getHandlers() { + return handlerList; + } + + public static HandlerList getHandlerList() { + return handlerList; + } + + @Override + public boolean isCancelled() { + return isCancelled; + } + + @Override + public void setCancelled(boolean cancel) { + isCancelled = cancel; + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerUnequipNameplateEvent.java b/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerUnequipNameplateEvent.java new file mode 100644 index 0000000..eb0be64 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/event/PlayerUnequipNameplateEvent.java @@ -0,0 +1,37 @@ +package net.momirealms.customnameplates.api.event; + +import org.bukkit.entity.Player; +import org.bukkit.event.Cancellable; +import org.bukkit.event.HandlerList; +import org.bukkit.event.player.PlayerEvent; +import org.jetbrains.annotations.NotNull; + +public class PlayerUnequipNameplateEvent extends PlayerEvent implements Cancellable { + + private static final HandlerList handlerList = new HandlerList(); + private boolean isCancelled; + + public PlayerUnequipNameplateEvent(@NotNull Player who) { + super(who); + this.isCancelled = false; + } + + @Override + public @NotNull HandlerList getHandlers() { + return handlerList; + } + + public static HandlerList getHandlerList() { + return handlerList; + } + + @Override + public boolean isCancelled() { + return isCancelled; + } + + @Override + public void setCancelled(boolean cancel) { + isCancelled = cancel; + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/ActionBarManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/ActionBarManager.java new file mode 100644 index 0000000..3df91be --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/ActionBarManager.java @@ -0,0 +1,17 @@ +package net.momirealms.customnameplates.api.manager; + +import org.bukkit.entity.Player; +import org.jetbrains.annotations.NotNull; + +public interface ActionBarManager { + + /** + * Get the actionbar sent by other plugins in MiniMessage format + * Return "" if no other actionbar received + * + * @param player receiver + * @return text + */ + @NotNull + String getOtherPluginActionBar(Player player); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/AdventureManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/AdventureManager.java new file mode 100644 index 0000000..6271313 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/AdventureManager.java @@ -0,0 +1,186 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.api.manager; + +import net.kyori.adventure.key.Key; +import net.kyori.adventure.sound.Sound; +import net.kyori.adventure.text.Component; +import org.bukkit.command.CommandSender; +import org.bukkit.entity.Player; + +public interface AdventureManager { + + Object getIChatComponentFromMiniMessage(String text); + + /** + * Strip all the tags from text + * + * @param text text + * @return stripped + */ + String stripTags(String text); + + /** + * Get component from text + * + * @param text text + * @return component + */ + Component getComponentFromMiniMessage(String text); + + /** + * Send a message to a command sender + * + * @param sender sender + * @param msg message + */ + void sendMessage(CommandSender sender, String msg); + + /** + * Send a message with prefix to a command sender + * + * @param sender sender + * @param msg message + */ + void sendMessageWithPrefix(CommandSender sender, String msg); + + /** + * Send a message to console + * + * @param msg message + */ + void sendConsoleMessage(String msg); + + /** + * Send a message to a player + * + * @param player player + * @param msg message + */ + void sendPlayerMessage(Player player, String msg); + + /** + * Send a title + * + * @param player player + * @param title title + * @param subtitle subtitle + * @param in in (ms) + * @param duration duration (ms) + * @param out out (ms) + */ + void sendTitle(Player player, String title, String subtitle, int in, int duration, int out); + + /** + * Send a title + * + * @param player player + * @param title title + * @param subtitle subtitle + * @param in in (ms) + * @param duration duration (ms) + * @param out out (ms) + */ + void sendTitle(Player player, Component title, Component subtitle, int in, int duration, int out); + + /** + * Send an actionbar + * + * @param player player + * @param msg msg + */ + void sendActionbar(Player player, String msg); + + /** + * Send an actionbar + * + * @param player player + * @param component component + */ + void sendActionbar(Player player, Component component); + + /** + * Play a sound to a player + * + * @param player player + * @param source sound source + * @param key sound key + * @param volume volume + * @param pitch pitch + */ + void sendSound(Player player, Sound.Source source, Key key, float volume, float pitch); + + /** + * Play a sound to a player + * + * @param player player + * @param sound sound + */ + void sendSound(Player player, Sound sound); + + /** + * Replace legacy color codes to MiniMessage format + * + * @param legacy legacy text + * @return MiniMessage format text + */ + String legacyToMiniMessage(String legacy); + + /** + * If a char is legacy color code + * + * @param c char + * @return is legacy color + */ + @SuppressWarnings("BooleanMethodIsAlwaysInverted") + boolean isColorCode(char c); + + /** + * Get legacy format text + * + * @param component component + * @return legacy format text + */ + String componentToLegacy(Component component); + + /** + * Get json by component + * + * @param component component + * @return json + */ + String componentToJson(Component component); + + /** + * Get the component in original package + * + * @param component shaded component + * @return paper component + */ + Object shadedToOriginal(Component component); + + /** + * Get MiniMessage format text from component + * + * @param component component + * @return text + */ + String getMiniMessageFormat(Component component); + + Object getIChatComponent(String json); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/BackGroundManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/BackGroundManager.java new file mode 100644 index 0000000..3ec95dc --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/BackGroundManager.java @@ -0,0 +1,40 @@ +package net.momirealms.customnameplates.api.manager; + +import net.momirealms.customnameplates.api.mechanic.background.BackGround; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.Collection; + +public interface BackGroundManager { + + /** + * Get a background's config by key + * + * @param key key + * @return background + */ + @Nullable + BackGround getBackGround(@NotNull String key); + + Collection getBackGrounds(); + + /** + * Register a background into the plugin + * This will fail if there already exists one with the same key + * + * @param key key + * @param backGround background + * @return success or not + */ + boolean registerBackGround(@NotNull String key, @NotNull BackGround backGround); + + /** + * Unregister a background by key + * This will fail if the key doesn't exist + * + * @param key key + * @return success or not + */ + boolean unregisterBackGround(@NotNull String key); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/BossBarManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/BossBarManager.java new file mode 100644 index 0000000..54abdc7 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/BossBarManager.java @@ -0,0 +1,5 @@ +package net.momirealms.customnameplates.api.manager; + +public interface BossBarManager { + +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/ImageManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/ImageManager.java new file mode 100644 index 0000000..834d9f7 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/ImageManager.java @@ -0,0 +1,40 @@ +package net.momirealms.customnameplates.api.manager; + +import net.momirealms.customnameplates.api.mechanic.character.ConfiguredChar; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.Collection; + +public interface ImageManager { + + /** + * Get an image by key + * + * @param key key + * @return image + */ + @Nullable + ConfiguredChar getImage(@NotNull String key); + + Collection getImages(); + + /** + * Register am image into the plugin + * This will fail if there already exists one with the same key + * + * @param key key + * @param configuredChar image + * @return success or not + */ + boolean registerImage(@NotNull String key, @NotNull ConfiguredChar configuredChar); + + /** + * Unregister an image by key + * This will fail if the key doesn't exist + * + * @param key key + * @return success or not + */ + boolean unregisterImage(@NotNull String key); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/NameplateManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/NameplateManager.java new file mode 100644 index 0000000..337c7ea --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/NameplateManager.java @@ -0,0 +1,158 @@ +package net.momirealms.customnameplates.api.manager; + +import net.momirealms.customnameplates.api.mechanic.nameplate.CachedNameplate; +import net.momirealms.customnameplates.api.mechanic.nameplate.Nameplate; +import net.momirealms.customnameplates.api.mechanic.nameplate.TagMode; +import net.momirealms.customnameplates.api.mechanic.tag.NameplatePlayer; +import net.momirealms.customnameplates.api.mechanic.team.TeamColor; +import org.bukkit.entity.Entity; +import org.bukkit.entity.Player; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.Collection; +import java.util.List; +import java.util.UUID; + +public interface NameplateManager { + + /** + * Get the default nameplate + */ + @NotNull + String getDefaultNameplate(); + + /** + * Put an entity's ID to map + * This map is used for quickly getting the entity instance + * Removal from the map is necessary when the entity is invalid + * Otherwise it would cause memory leak + * + * @param entityID entityID + * @param entity entity + */ + boolean putEntityIDToMap(int entityID, Entity entity); + + /** + * Remove the entity from map + * + * @param entityID entityID + * @return the removed entity + */ + @Nullable + Entity removeEntityIDFromMap(int entityID); + + /** + * Nameplates are cached in memory so they would not be frequently updated + * The update rate is decided by "refresh-frequency" in nameplate.yml + * + * @param uuid player uuid + * @param nameplate cached nameplate + */ + boolean putCachedNameplateToMap(UUID uuid, CachedNameplate nameplate); + + /** + * Remove CachedNameplate from map + * + * @param uuid player uuid + * @return removed CachedNameplate + */ + @Nullable + CachedNameplate removeCachedNameplateFromMap(UUID uuid); + + /** + * Get player by entityID from the cache + * + * @param id entityID + * @return player + */ + @Nullable + Player getPlayerByEntityID(int id); + + /** + * Get entity by entityID from the cache + * + * @param id entityID + * @return entity + */ + @Nullable + Entity getEntityByEntityID(int id); + + /** + * Update a player's cached nameplate + * The nameplate is affected by "prefix" and "suffix" option + * + * @param player player + * @return if the nameplate is updated + */ + boolean updateCachedNameplate(Player player); + + /** + * Update a player's cached nameplate + * The nameplate is affected by "prefix" and "suffix" option + * + * @param player player + * @return if the nameplate is updated + */ + boolean updateCachedNameplate(Player player, Nameplate nameplate); + + /** + * This should not be null when player's data is loaded (async process) + * + * @param player player + * @return cached nameplate + */ + @Nullable + CachedNameplate getCacheNameplate(Player player); + + /** + * Create a name tag for a player, the tag type is decided by the mode in nameplate.yml + * + * @param player player + */ + void createNameTag(Player player); + + void putNameplatePlayerToMap(NameplatePlayer player); + + NameplatePlayer removeNameplatePlayerFromMap(UUID uuid); + + String getNameplatePrefix(Player player); + + String getNameplateSuffix(Player player); + + String getFullNameTag(Player player); + + boolean registerNameplate(String key, Nameplate nameplate); + + void unEquipNameplate(Player player); + + boolean unregisterNameplate(String key); + + boolean isProxyMode(); + + TagMode getTagMode(); + + Nameplate getNameplate(String key); + + Collection getNameplates(); + + boolean containsNameplate(String key); + + List getAvailableNameplates(Player player); + + /** + * If player has permission for a certain nameplate + */ + boolean hasNameplate(Player player, String nameplate); + + /** + * Return false if nameplate doesn't exist + */ + boolean equipNameplate(Player player, String nameplate); + + TeamColor getTeamColor(Player player); + + TeamTagManager getTeamTagManager(); + + UnlimitedTagManager getUnlimitedTagManager(); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/PlaceholderManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/PlaceholderManager.java new file mode 100644 index 0000000..0463b19 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/PlaceholderManager.java @@ -0,0 +1,73 @@ +package net.momirealms.customnameplates.api.manager; + +import net.momirealms.customnameplates.api.mechanic.placeholder.*; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.List; + +public interface PlaceholderManager { + + /** + * Detect all the placeholders + * + * @param text text + * @return placeholder + */ + @NotNull + List detectPlaceholders(String text); + + /** + * Get a static text instance + * + * @param key key + * @return static text + */ + @Nullable + StaticText getStaticText(String key); + + /** + * Get a switch text instance + * + * @param key key + * @return switch text + */ + @Nullable + SwitchText getSwitchText(String key); + + /** + * Get a descent text instance + * + * @param key key + * @return descent text + */ + @Nullable + DescentText getDescentText(String key); + + /** + * Get a conditional text + * + * @param key key + * @return conditional text + */ + @Nullable + ConditionalText getConditionalText(String key); + + /** + * Get a nameplate text + * + * @param key key + * @return nameplate text + */ + @Nullable + NameplateText getNameplateText(String key); + + /** + * Get a background text + * + * @param key key + * @return background text + */ + @Nullable + BackGroundText getBackGroundText(String key); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/RequirementManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/RequirementManager.java new file mode 100644 index 0000000..6c9a77f --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/RequirementManager.java @@ -0,0 +1,104 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.api.manager; + +import net.momirealms.customnameplates.api.requirement.Condition; +import net.momirealms.customnameplates.api.requirement.Requirement; +import net.momirealms.customnameplates.api.requirement.RequirementFactory; +import org.bukkit.configuration.ConfigurationSection; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +public interface RequirementManager { + + /** + * Registers a custom requirement type with its corresponding factory. + * + * @param type The type identifier of the requirement. + * @param requirementFactory The factory responsible for creating instances of the requirement. + * @return True if registration was successful, false if the type is already registered. + */ + boolean registerRequirement(String type, RequirementFactory requirementFactory); + + /** + * Unregisters a custom requirement type. + * + * @param type The type identifier of the requirement to unregister. + * @return True if unregistration was successful, false if the type is not registered. + */ + boolean unregisterRequirement(String type); + + /** + * Retrieves an array of requirements based on a configuration section. + * + * @param section The configuration section containing requirement definitions. + * @return An array of Requirement objects based on the configuration section + */ + @Nullable Requirement[] getRequirements(ConfigurationSection section); + + /** + * Retrieves a Requirement object based on a configuration section and advanced flag. + *

+ * requirement_1: <- section + * type: xxx + * value: xxx + * + * @param section The configuration section containing requirement definitions. + * @return A Requirement object based on the configuration section, or an EmptyRequirement if the section is null or invalid. + */ + @NotNull Requirement getRequirement(ConfigurationSection section); + + /** + * Gets a requirement based on the provided type and value. + * If a valid RequirementFactory is found for the type, it is used to create the requirement. + * If no factory is found, a warning is logged, and an empty requirement instance is returned. + *

+ * world: <- type + * - world <- value + * + * @param type The type representing the requirement type. + * @param value The value associated with the requirement. + * @return A Requirement instance based on the type and value, or an EmptyRequirement if the type is invalid. + */ + @NotNull Requirement getRequirement(String type, Object value); + + /** + * Retrieves a RequirementFactory based on the specified requirement type. + * + * @param type The requirement type for which to retrieve a factory. + * @return A RequirementFactory for the specified type, or null if no factory is found. + */ + @Nullable RequirementFactory getRequirementFactory(String type); + + /** + * Checks if an array of requirements is met for a given condition. + * + * @param condition The Condition object to check against the requirements. + * @param requirements An array of Requirement instances to be evaluated. + * @return True if all requirements are met, false otherwise. Returns true if the requirements array is null. + */ + static boolean isRequirementMet(Condition condition, Requirement... requirements) { + if (requirements == null) return true; + for (Requirement requirement : requirements) { + if (!requirement.isConditionMet(condition)) { + return false; + } + } + return true; + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/ResourcePackManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/ResourcePackManager.java new file mode 100644 index 0000000..4df85fd --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/ResourcePackManager.java @@ -0,0 +1,15 @@ +package net.momirealms.customnameplates.api.manager; + +import java.io.File; + +public interface ResourcePackManager { + + /** + * Generate the resource pack + */ + void generateResourcePack(); + + void deleteDirectory(File file); + + String native2ascii(char c); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/StorageManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/StorageManager.java new file mode 100644 index 0000000..5165ca6 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/StorageManager.java @@ -0,0 +1,88 @@ +package net.momirealms.customnameplates.api.manager; + +import net.momirealms.customnameplates.api.data.OnlineUser; +import net.momirealms.customnameplates.api.data.PlayerData; +import org.jetbrains.annotations.NotNull; + +import java.util.Collection; +import java.util.Optional; +import java.util.UUID; +import java.util.concurrent.CompletableFuture; + +public interface StorageManager { + + /** + * Get online users + * + * @return online users + */ + Collection getOnlineUsers(); + + /** + * Get a player's data by uuid + * The player can be an offline one + * + * @param uuid uuid + * @return player data + */ + CompletableFuture> getPlayerData(UUID uuid); + + /** + * Save online players' data + * + * @param uuid uuid + * @return success or not + */ + CompletableFuture saveOnlinePlayerData(UUID uuid); + + /** + * Save specified data + * + * @param uuid uuid + * @param playerData playerData + * @return success or not + */ + CompletableFuture savePlayerData(UUID uuid, PlayerData playerData); + + /** + * Get an online user by uuid + * + * @param uuid uuid + * @return online user + */ + Optional getOnlineUser(UUID uuid); + + /** + * Get player data from json + * + * @param json json + * @return data + */ + @NotNull + PlayerData fromJson(String json); + + /** + * Get player data from bytes + * + * @param data data + * @return data + */ + PlayerData fromBytes(byte[] data); + + /** + * Convert player data to bytes + * + * @param playerData playerData + * @return bytes + */ + byte[] toBytes(PlayerData playerData); + + /** + * Convert player data to json + * + * @param playerData playerData + * @return json + */ + @NotNull + String toJson(@NotNull PlayerData playerData); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/TeamManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/TeamManager.java new file mode 100644 index 0000000..14e4721 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/TeamManager.java @@ -0,0 +1,47 @@ +package net.momirealms.customnameplates.api.manager; + +import net.kyori.adventure.text.Component; +import net.momirealms.customnameplates.api.mechanic.team.TeamColor; +import net.momirealms.customnameplates.api.mechanic.team.TeamTagVisibility; +import org.bukkit.entity.Player; + +public interface TeamManager { + + /** + * Create team for a player + * + * @param player player + */ + void createTeam(Player player); + + /** + * Create a team for a player on proxy + * + * @param player player + */ + void createProxyTeam(Player player); + + /** + * Remove a team for a player + * + * @param player player + */ + void removeTeam(Player player); + + /** + * Remove a team for a player on proxy + * + * @param player player + */ + void removeProxyTeam(Player player); + + void updateTeam(Player owner, Player viewer, Component prefix, Component suffix, TeamColor color, TeamTagVisibility visibility); + + /** + * Get the team player in + * + * @param player player + * @return team name + */ + String getTeamName(Player player); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/TeamTagManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/TeamTagManager.java new file mode 100644 index 0000000..ffa408d --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/TeamTagManager.java @@ -0,0 +1,29 @@ +package net.momirealms.customnameplates.api.manager; + +import net.momirealms.customnameplates.api.mechanic.tag.team.TeamPlayer; +import org.bukkit.entity.Player; +import org.jetbrains.annotations.Nullable; + +import java.util.UUID; + +public interface TeamTagManager { + + /** + * Create team tag for a player + * If failed, the return value would be null + * This happens when there already exists a team tag for a player + * + * @return team tag + */ + @Nullable + TeamPlayer createTagForPlayer(Player player, String prefix, String suffix); + + /** + * Remove a team tag from map by uuid + * + * @param uuid uuid + * @return team tag + */ + @Nullable + TeamPlayer removeTeamPlayerFromMap(UUID uuid); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/UnlimitedTagManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/UnlimitedTagManager.java new file mode 100644 index 0000000..305e03c --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/UnlimitedTagManager.java @@ -0,0 +1,56 @@ +package net.momirealms.customnameplates.api.manager; + +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.NamedEntity; +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.UnlimitedObject; +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.UnlimitedPlayer; +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.UnlimitedTagSetting; +import org.bukkit.entity.Player; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.List; +import java.util.UUID; + +public interface UnlimitedTagManager { + + /** + * Create a named entity (ArmorStand) for a player + * To apply the changes, you should add it the named player instance + * + * @param player player + * @param setting setting + * @return named entity + */ + @NotNull + NamedEntity createNamedEntity(UnlimitedPlayer player, UnlimitedTagSetting setting); + + /** + * Create unlimited tags for a player + * If failed, the return value would be null + * This happens when there already exists an UnlimitedObject for a player + * + * @param player player + * @param settings settings + * @return unlimited tag + */ + @Nullable + UnlimitedPlayer createTagForPlayer(Player player, List settings); + + /** + * Remove UnlimitedObject from map by uuid + * + * @param uuid uuid + * @return The removed unlimited object + */ + @Nullable + UnlimitedObject removeUnlimitedObjectFromMap(UUID uuid); + + /** + * Get an UnlimitedObject from map by uuid + * + * @param uuid uuid + * @return The unlimited object + */ + @Nullable + UnlimitedObject getUnlimitedObject(UUID uuid); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/VersionManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/VersionManager.java new file mode 100644 index 0000000..261ae9d --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/VersionManager.java @@ -0,0 +1,26 @@ +package net.momirealms.customnameplates.api.manager; + +import java.util.concurrent.CompletionStage; + +public interface VersionManager { + + boolean isFolia(); + + String getServerVersion(); + + CompletionStage checkUpdate(); + + boolean isVersionNewerThan1_19_R2(); + + boolean isVersionNewerThan1_20(); + + boolean isVersionNewerThan1_20_R2(); + + String getPluginVersion(); + + boolean isLatest(); + + boolean isVersionNewerThan1_19(); + + int getPackFormat(); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/manager/WidthManager.java b/api/src/main/java/net/momirealms/customnameplates/api/manager/WidthManager.java new file mode 100644 index 0000000..96b17e7 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/manager/WidthManager.java @@ -0,0 +1,5 @@ +package net.momirealms.customnameplates.api.manager; + +public interface WidthManager { + +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/background/BackGround.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/background/BackGround.java new file mode 100644 index 0000000..785c55b --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/background/BackGround.java @@ -0,0 +1,212 @@ +package net.momirealms.customnameplates.api.mechanic.background; + +import net.momirealms.customnameplates.api.mechanic.character.ConfiguredChar; +import net.momirealms.customnameplates.api.mechanic.font.OffsetFont; + +public class BackGround { + + private ConfiguredChar left, offset_1, offset_2, offset_4, offset_8, offset_16, offset_32, offset_64, offset_128, right; + private int leftMargin, rightMargin; + + private BackGround() { + } + + public BackGround( + ConfiguredChar left, + ConfiguredChar offset_1, + ConfiguredChar offset_2, + ConfiguredChar offset_4, + ConfiguredChar offset_8, + ConfiguredChar offset_16, + ConfiguredChar offset_32, + ConfiguredChar offset_64, + ConfiguredChar offset_128, + ConfiguredChar right, + int leftMargin, + int rightMargin + ) { + this.left = left; + this.offset_1 = offset_1; + this.offset_2 = offset_2; + this.offset_4 = offset_4; + this.offset_8 = offset_8; + this.offset_16 = offset_16; + this.offset_32 = offset_32; + this.offset_64 = offset_64; + this.offset_128 = offset_128; + this.right = right; + this.leftMargin = leftMargin; + this.rightMargin = rightMargin; + } + + public ConfiguredChar getLeft() { + return left; + } + + public ConfiguredChar getOffset_1() { + return offset_1; + } + + public ConfiguredChar getOffset_2() { + return offset_2; + } + + public ConfiguredChar getOffset_4() { + return offset_4; + } + + public ConfiguredChar getOffset_8() { + return offset_8; + } + + public ConfiguredChar getOffset_16() { + return offset_16; + } + + public ConfiguredChar getOffset_32() { + return offset_32; + } + + public ConfiguredChar getOffset_64() { + return offset_64; + } + + public ConfiguredChar getOffset_128() { + return offset_128; + } + + public ConfiguredChar getRight() { + return right; + } + + public static Builder builder() { + return new Builder(); + } + + public String getBackGroundImage(int n) { + String offset = OffsetFont.getShortestNegChars(n + rightMargin + 2); + n = n + leftMargin + rightMargin + 2; + StringBuilder stringBuilder = new StringBuilder(); + stringBuilder.append(left.getCharacter()); + while (n >= 128) { + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + stringBuilder.append(offset_128.getCharacter()); + n -= 128; + } + if (n - 64 >= 0) { + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + stringBuilder.append(offset_64.getCharacter()); + n -= 64; + } + if (n - 32 >= 0) { + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + stringBuilder.append(offset_32.getCharacter()); + n -= 32; + } + if (n - 16 >= 0) { + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + stringBuilder.append(offset_16.getCharacter()); + n -= 16; + } + if (n - 8 >= 0) { + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + stringBuilder.append(offset_8.getCharacter()); + n -= 8; + } + if (n - 4 >= 0) { + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + stringBuilder.append(offset_4.getCharacter()); + n -= 4; + } + if (n - 2 >= 0) { + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + stringBuilder.append(offset_2.getCharacter()); + n -= 2; + } + if (n - 1 >= 0) { + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + stringBuilder.append(offset_1.getCharacter()); + } + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + stringBuilder.append(right.getCharacter()); + stringBuilder.append(offset); + return stringBuilder.toString(); + } + + public static class Builder { + + private final BackGround backGround; + + public static Builder of() { + return new Builder(); + } + + public Builder() { + this.backGround = new BackGround(); + } + + public Builder left(ConfiguredChar configuredChar) { + backGround.left = configuredChar; + return this; + } + + public Builder right(ConfiguredChar configuredChar) { + backGround.right = configuredChar; + return this; + } + + public Builder offset_1(ConfiguredChar configuredChar) { + backGround.offset_1 = configuredChar; + return this; + } + + public Builder offset_2(ConfiguredChar configuredChar) { + backGround.offset_2 = configuredChar; + return this; + } + + public Builder offset_4(ConfiguredChar configuredChar) { + backGround.offset_4 = configuredChar; + return this; + } + + public Builder offset_8(ConfiguredChar configuredChar) { + backGround.offset_8 = configuredChar; + return this; + } + + public Builder offset_16(ConfiguredChar configuredChar) { + backGround.offset_16 = configuredChar; + return this; + } + + public Builder offset_32(ConfiguredChar configuredChar) { + backGround.offset_32 = configuredChar; + return this; + } + + public Builder offset_64(ConfiguredChar configuredChar) { + backGround.offset_64 = configuredChar; + return this; + } + + public Builder offset_128(ConfiguredChar configuredChar) { + backGround.offset_128 = configuredChar; + return this; + } + + public Builder leftMargin(int margin) { + backGround.leftMargin = margin; + return this; + } + + public Builder rightMargin(int margin) { + backGround.rightMargin = margin; + return this; + } + + public BackGround build() { + return backGround; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/character/CharacterArranger.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/character/CharacterArranger.java new file mode 100644 index 0000000..aac98ec --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/character/CharacterArranger.java @@ -0,0 +1,25 @@ +package net.momirealms.customnameplates.api.mechanic.character; + +public class CharacterArranger { + + public static char currentChar; + + public static void increase() { + currentChar = (char) (currentChar + '\u0001'); + } + + public static char getAndIncrease() { + char temp = currentChar; + increase(); + return temp; + } + + public static char increaseAndGet() { + increase(); + return currentChar; + } + + public static void reset(char c) { + currentChar = c; + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/character/ConfiguredChar.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/character/ConfiguredChar.java new file mode 100644 index 0000000..cc70116 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/character/ConfiguredChar.java @@ -0,0 +1,107 @@ +package net.momirealms.customnameplates.api.mechanic.character; + +import net.momirealms.customnameplates.api.util.LogUtils; + +public class ConfiguredChar { + + private char character; + private String pngFile; + private int height; + private int width; + private int ascent; + + private ConfiguredChar() { + + } + + public ConfiguredChar(char character, String pngFile, int height, int width, int ascent) { + this.character = character; + this.pngFile = pngFile; + this.height = height; + this.width = width; + this.ascent = ascent; + } + + public char getCharacter() { + return character; + } + + public static Builder builder() { + return new Builder(); + } + + public String getPngFile() { + return pngFile; + } + + public int getHeight() { + return height; + } + + public int getWidth() { + return width; + } + + public int getAscent() { + return ascent; + } + + public String getFile() { + return pngFile + ".png"; + } + + public static class Builder { + + private final ConfiguredChar configuredChar; + + public static Builder of() { + return new Builder(); + } + + public Builder() { + this.configuredChar = new ConfiguredChar(); + } + + public Builder character(char character) { + configuredChar.character = character; + return this; + } + + public Builder png(String png) { + configuredChar.pngFile = png; + return this; + } + + public Builder height(int height) { + configuredChar.height = height; + return this; + } + + public Builder ascent(int ascent) { + configuredChar.ascent = ascent; + if (ascent >= configuredChar.height) { + LogUtils.warn("Invalid config for " + configuredChar.pngFile); + LogUtils.warn("Ascent " + ascent + " should be no higher than Height " + configuredChar.height); + } + return this; + } + + public Builder descent(int descent) { + if (descent < 0) { + LogUtils.warn("Invalid config for " + configuredChar.pngFile); + LogUtils.warn("Descent " + descent + " should be no lower than 0"); + } + configuredChar.ascent = configuredChar.height - descent; + return this; + } + + public Builder width(int width) { + configuredChar.width = width; + return this; + } + + public ConfiguredChar build() { + return configuredChar; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/font/OffsetFont.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/font/OffsetFont.java new file mode 100644 index 0000000..4d24ab1 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/font/OffsetFont.java @@ -0,0 +1,180 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.api.mechanic.font; + +public enum OffsetFont { + + NEG_1('\uf801', -1, -3), + NEG_2('\uf802', -2, -4), + NEG_3('\uf803', -3, -5), + NEG_4('\uf804', -4, -6), + NEG_5('\uf805', -5, -7), + NEG_6('\uf806', -6, -8), + NEG_7('\uf807', -7, -9), + NEG_8('\uf808', -8, -10), + NEG_16('\uf809', -16, -18), + NEG_32('\uf80a', -32, -34), + NEG_64('\uf80b', -64, -66), + NEG_128('\uf80c', -128, -130), + POS_1('\uf811', 1, -1), + POS_2('\uf812', 2, 1), + POS_3('\uf813', 3, 2), + POS_4('\uf814', 4, 3), + POS_5('\uf815', 5, 4), + POS_6('\uf816', 6, 5), + POS_7('\uf817', 7, 6), + POS_8('\uf818', 8, 7), + POS_16('\uf819', 16, 15), + POS_32('\uf81a', 32, 31), + POS_64('\uf81b', 64, 63), + POS_128('\uf81c', 128, 127); + + private final char character; + private final int space; + private final int height; + + OffsetFont(char character, int space, int height) { + this.character = character; + this.space = space; + this.height = height; + } + + public char getCharacter() { + return this.character; + } + + public int getSpace() { + return this.space; + } + + public int getHeight() { + return this.height; + } + + public static String getOffsetChars(int offset) { + if (offset >= 0) { + return getShortestPosChars(offset); + } else { + return getShortestNegChars(-offset); + } + } + + public static String getShortestNegChars(int n) { + StringBuilder stringBuilder = new StringBuilder(); + while (n >= 128) { + stringBuilder.append(OffsetFont.NEG_128.getCharacter()); + n -= 128; + } + if (n - 64 >= 0) { + stringBuilder.append(OffsetFont.NEG_64.getCharacter()); + n -= 64; + } + if (n - 32 >= 0) { + stringBuilder.append(OffsetFont.NEG_32.getCharacter()); + n -= 32; + } + if (n - 16 >= 0) { + stringBuilder.append(OffsetFont.NEG_16.getCharacter()); + n -= 16; + } + if (n - 8 >= 0) { + stringBuilder.append(OffsetFont.NEG_8.getCharacter()); + n -= 8; + } + if (n - 7 >= 0) { + stringBuilder.append(OffsetFont.NEG_7.getCharacter()); + n -= 7; + } + if (n - 6 >= 0) { + stringBuilder.append(OffsetFont.NEG_6.getCharacter()); + n -= 6; + } + if (n - 5 >= 0) { + stringBuilder.append(OffsetFont.NEG_5.getCharacter()); + n -= 5; + } + if (n - 4 >= 0) { + stringBuilder.append(OffsetFont.NEG_4.getCharacter()); + n -= 4; + } + if (n - 3 >= 0) { + stringBuilder.append(OffsetFont.NEG_3.getCharacter()); + n -= 3; + } + if (n - 2 >= 0) { + stringBuilder.append(OffsetFont.NEG_2.getCharacter()); + n -= 2; + } + if (n - 1 >= 0) { + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + } + return stringBuilder.toString(); + } + + public static String getShortestPosChars(int n) { + StringBuilder stringBuilder = new StringBuilder(); + while (n >= 128) { + stringBuilder.append(OffsetFont.POS_128.getCharacter()); + n -= 128; + } + if (n - 64 >= 0) { + stringBuilder.append(OffsetFont.POS_64.getCharacter()); + n -= 64; + } + if (n - 32 >= 0) { + stringBuilder.append(OffsetFont.POS_32.getCharacter()); + n -= 32; + } + if (n - 16 >= 0) { + stringBuilder.append(OffsetFont.POS_16.getCharacter()); + n -= 16; + } + if (n - 8 >= 0) { + stringBuilder.append(OffsetFont.POS_8.getCharacter()); + n -= 8; + } + if (n - 7 >= 0) { + stringBuilder.append(OffsetFont.POS_7.getCharacter()); + n -= 7; + } + if (n - 6 >= 0) { + stringBuilder.append(OffsetFont.POS_6.getCharacter()); + n -= 6; + } + if (n - 5 >= 0) { + stringBuilder.append(OffsetFont.POS_5.getCharacter()); + n -= 5; + } + if (n - 4 >= 0) { + stringBuilder.append(OffsetFont.POS_4.getCharacter()); + n -= 4; + } + if (n - 3 >= 0) { + stringBuilder.append(OffsetFont.POS_3.getCharacter()); + n -= 3; + } + if (n - 2 >= 0) { + stringBuilder.append(OffsetFont.POS_2.getCharacter()); + n -= 2; + } + if (n - 1 >= 0) { + stringBuilder.append(OffsetFont.POS_1.getCharacter()); + } + return stringBuilder.toString(); + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/misc/ViewerText.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/misc/ViewerText.java new file mode 100644 index 0000000..ad7af9b --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/misc/ViewerText.java @@ -0,0 +1,111 @@ +package net.momirealms.customnameplates.api.mechanic.misc; + +import me.clip.placeholderapi.PlaceholderAPI; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import org.bukkit.entity.Entity; +import org.bukkit.entity.Player; + +import java.util.List; +import java.util.Objects; +import java.util.UUID; +import java.util.concurrent.ConcurrentHashMap; + +public class ViewerText { + + private final Player owner; + private String processedText; + private final ClaimedText[] placeholders; + private final ConcurrentHashMap valueMap; + + public ViewerText(Player owner, String rawText) { + this.processedText = rawText; + this.valueMap = new ConcurrentHashMap<>(); + this.owner = owner; + List placeholders = CustomNameplatesPlugin.get().getPlaceholderManager().detectPlaceholders(rawText); + this.placeholders = new ClaimedText[placeholders.size()]; + int i = 0; + for (String placeholder : placeholders) { + processedText = processedText.replace(placeholder, "%s"); + if (placeholder.startsWith("%viewer_")) { + this.placeholders[i] = new ClaimedText(null, "%" + placeholder.substring("%viewer_".length())); + } else { + this.placeholders[i] = new ClaimedText(owner, placeholder); + } + i++; + } + } + + public void updateForOwner() { + for (ClaimedText text : placeholders) { + text.update(); + } + } + + public boolean updateForViewer(Player viewer) { + String string; + if ("%s".equals(processedText)) { + string = placeholders[0].getValue(viewer); + } else if (placeholders.length != 0) { + Object[] values = new String[placeholders.length]; + for (int i = 0; i < placeholders.length; i++) { + values[i] = placeholders[i].getValue(viewer); + } + string = String.format(processedText, values); + } else { + string = processedText; + } + var uuid = viewer.getUniqueId(); + if (!valueMap.containsKey(uuid)) { + valueMap.put(uuid, string); + return true; + } + String previousValue = valueMap.get(uuid); + if (!previousValue.equals(string)) { + valueMap.put(uuid, string); + return true; + } + return false; + } + + public void removeViewer(Player viewer) { + valueMap.remove(viewer.getUniqueId()); + } + + public String getProcessedText() { + return processedText; + } + + public String getLatestValue(Player viewer) { + return valueMap.get(viewer.getUniqueId()); + } + + public Entity getOwner() { + return owner; + } + + public static class ClaimedText { + + private final String placeholder; + private final Player owner; + private String latestValue; + + public ClaimedText(Player owner, String placeholder) { + this.placeholder = placeholder; + this.owner = owner; + this.latestValue = null; + this.update(); + } + + public void update() { + if (owner == null) return; + this.latestValue = PlaceholderAPI.setPlaceholders(owner, placeholder); + } + + public String getValue(Player viewer) { + return Objects.requireNonNullElseGet( + latestValue, + () -> PlaceholderAPI.setPlaceholders(owner == null ? viewer : owner, placeholder) + ); + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/nameplate/CachedNameplate.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/nameplate/CachedNameplate.java new file mode 100644 index 0000000..40c1e6b --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/nameplate/CachedNameplate.java @@ -0,0 +1,70 @@ +package net.momirealms.customnameplates.api.mechanic.nameplate; + +import net.momirealms.customnameplates.api.mechanic.team.TeamColor; + +public class CachedNameplate { + + private TeamColor teamColor; + private String tagPrefix; + private String tagSuffix; + private String namePrefix; + private String nameSuffix; + private String playerName; + + public CachedNameplate() { + this.tagPrefix = ""; + this.tagSuffix = ""; + this.namePrefix = ""; + this.nameSuffix = ""; + this.playerName = ""; + this.teamColor = TeamColor.WHITE; + } + + public String getTagPrefix() { + return tagPrefix; + } + + public void setTagPrefix(String prefix) { + this.tagPrefix = prefix; + } + + public String getTagSuffix() { + return tagSuffix; + } + + public void setTagSuffix(String suffix) { + this.tagSuffix = suffix; + } + + public void setNamePrefix(String namePrefix) { + this.namePrefix = namePrefix; + } + + public void setNameSuffix(String nameSuffix) { + this.nameSuffix = nameSuffix; + } + + public String getNamePrefix() { + return namePrefix; + } + + public String getNameSuffix() { + return nameSuffix; + } + + public TeamColor getTeamColor() { + return teamColor; + } + + public void setTeamColor(TeamColor teamColor) { + this.teamColor = teamColor; + } + + public String getPlayerName() { + return playerName; + } + + public void setPlayerName(String playerName) { + this.playerName = playerName; + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/nameplate/Nameplate.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/nameplate/Nameplate.java new file mode 100644 index 0000000..bdd40e3 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/nameplate/Nameplate.java @@ -0,0 +1,166 @@ +package net.momirealms.customnameplates.api.mechanic.nameplate; + +import net.momirealms.customnameplates.api.mechanic.character.ConfiguredChar; +import net.momirealms.customnameplates.api.mechanic.font.OffsetFont; +import net.momirealms.customnameplates.api.mechanic.team.TeamColor; +import net.momirealms.customnameplates.api.util.FontUtils; + +public class Nameplate { + + private String displayName; + private TeamColor teamColor; + private String namePrefix; + private String nameSuffix; + private ConfiguredChar left; + private ConfiguredChar middle; + private ConfiguredChar right; + + private Nameplate() { + } + + public Nameplate( + String displayName, + TeamColor teamColor, + String namePrefix, + String nameSuffix, + ConfiguredChar left, + ConfiguredChar middle, + ConfiguredChar right + ) { + this.displayName = displayName; + this.teamColor = teamColor; + this.left = left; + this.middle = middle; + this.right = right; + this.namePrefix = namePrefix; + this.nameSuffix = nameSuffix; + } + + public String getDisplayName() { + return displayName; + } + + public TeamColor getTeamColor() { + return teamColor; + } + + public ConfiguredChar getLeft() { + return left; + } + + public ConfiguredChar getMiddle() { + return middle; + } + + public ConfiguredChar getRight() { + return right; + } + + public String getNamePrefix() { + if (teamColor == TeamColor.NONE) { + return ""; + } + if (teamColor == TeamColor.CUSTOM) { + return namePrefix; + } + return "<" + teamColor.name() + ">"; + } + + public String getNameSuffix() { + if (teamColor == TeamColor.NONE) { + return ""; + } + if (teamColor == TeamColor.CUSTOM) { + return nameSuffix; + } + return ""; + } + + public static Builder builder() { + return new Builder(); + } + + public String getPrefixWithFont(int textWidth) { + return FontUtils.surroundNameplateFont(getPrefix(textWidth)); + } + + public String getPrefix(int textWidth) { + StringBuilder stringBuilder = new StringBuilder(); + stringBuilder.append(OffsetFont.getShortestNegChars(textWidth % 2 == 0 ? textWidth + left.getWidth() : textWidth + left.getWidth() + 1)); + stringBuilder.append(left.getCharacter()); + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + int mid_amount = (textWidth - 1) / (middle.getWidth()); + if (mid_amount != 0) { + for (int i = 0; i < mid_amount; i++) { + stringBuilder.append(middle.getCharacter()); + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + } + stringBuilder.append(OffsetFont.getShortestNegChars(middle.getWidth() - textWidth % middle.getWidth())); // +1 + } + stringBuilder.append(middle.getCharacter()); + stringBuilder.append(OffsetFont.NEG_1.getCharacter()); + stringBuilder.append(right.getCharacter()); + stringBuilder.append(OffsetFont.getShortestNegChars(textWidth + right.getWidth() - 1)); // -1 + return stringBuilder.toString(); + } + + public String getSuffixWithFont(int textWidth) { + return FontUtils.surroundNameplateFont(getSuffix(textWidth)); + } + + public String getSuffix(int textWidth) { + return OffsetFont.getShortestNegChars(textWidth + textWidth % 2 + 1); + } + + public static class Builder { + + private final Nameplate nameplate; + + public Builder() { + this.nameplate = new Nameplate(); + } + + public static Builder of() { + return new Builder(); + } + + public Builder displayName(String display) { + nameplate.displayName = display; + return this; + } + + public Builder teamColor(TeamColor teamColor) { + nameplate.teamColor = teamColor; + return this; + } + + public Builder namePrefix(String namePrefix) { + nameplate.namePrefix = namePrefix; + return this; + } + + public Builder nameSuffix(String nameSuffix) { + nameplate.nameSuffix = nameSuffix; + return this; + } + + public Builder left(ConfiguredChar configuredChar) { + nameplate.left = configuredChar; + return this; + } + + public Builder middle(ConfiguredChar configuredChar) { + nameplate.middle = configuredChar; + return this; + } + + public Builder right(ConfiguredChar configuredChar) { + nameplate.right = configuredChar; + return this; + } + + public Nameplate build() { + return nameplate; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/nameplate/TagMode.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/nameplate/TagMode.java new file mode 100644 index 0000000..774c228 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/nameplate/TagMode.java @@ -0,0 +1,7 @@ +package net.momirealms.customnameplates.api.mechanic.nameplate; + +public enum TagMode { + TEAM, + UNLIMITED, + DISABLE +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/BackGroundText.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/BackGroundText.java new file mode 100644 index 0000000..a359e64 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/BackGroundText.java @@ -0,0 +1,65 @@ +package net.momirealms.customnameplates.api.mechanic.placeholder; + +import me.clip.placeholderapi.PlaceholderAPI; +import net.momirealms.customnameplates.api.mechanic.background.BackGround; +import net.momirealms.customnameplates.api.util.FontUtils; +import org.bukkit.OfflinePlayer; + +public class BackGroundText { + + private String text; + private BackGround backGround; + + private BackGroundText() { + } + + public BackGroundText(String text, BackGround backGround) { + this.text = text; + this.backGround = backGround; + } + + public String getText() { + return text; + } + + public BackGround getBackGround() { + return backGround; + } + + public String getValue(OfflinePlayer player) { + String parsed = PlaceholderAPI.setPlaceholders(player, text); + int parsedWidth = FontUtils.getTextWidth(parsed); + return backGround.getBackGroundImage(parsedWidth) + parsed; + } + + public static Builder builder() { + return new Builder(); + } + + public static class Builder { + + private final BackGroundText text; + + public Builder() { + this.text = new BackGroundText(); + } + + public static Builder of() { + return new Builder(); + } + + public Builder text(String value) { + text.text = value; + return this; + } + + public Builder background(BackGround backGround) { + text.backGround = backGround; + return this; + } + + public BackGroundText build() { + return text; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/CachedText.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/CachedText.java new file mode 100644 index 0000000..b52ba7e --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/CachedText.java @@ -0,0 +1,54 @@ +package net.momirealms.customnameplates.api.mechanic.placeholder; + +public class CachedText { + + private long refreshInterval; + private String text; + + private CachedText() { + } + + public CachedText(long refreshInterval, String text) { + this.refreshInterval = refreshInterval; + this.text = text; + } + + public long getRefreshInterval() { + return refreshInterval; + } + + public String getText() { + return text; + } + + public static Builder builder() { + return new Builder(); + } + + public static class Builder { + + private final CachedText text; + + public Builder() { + this.text = new CachedText(); + } + + public static Builder of() { + return new Builder(); + } + + public Builder refreshInterval(long time) { + this.text.refreshInterval = time; + return this; + } + + public Builder text(String text) { + this.text.text = text; + return this; + } + + public CachedText build() { + return text; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/ConditionalText.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/ConditionalText.java new file mode 100644 index 0000000..bf64792 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/ConditionalText.java @@ -0,0 +1,58 @@ +package net.momirealms.customnameplates.api.mechanic.placeholder; + +import me.clip.placeholderapi.PlaceholderAPI; +import net.momirealms.customnameplates.api.common.Pair; +import net.momirealms.customnameplates.api.manager.RequirementManager; +import net.momirealms.customnameplates.api.requirement.Condition; +import net.momirealms.customnameplates.api.requirement.Requirement; +import org.bukkit.OfflinePlayer; + +import java.util.List; + +public class ConditionalText { + + private List> textList; + + private ConditionalText() { + } + + public ConditionalText(List> textList) { + this.textList = textList; + } + + public static Builder builder() { + return new Builder(); + } + + public String getValue(OfflinePlayer player) { + Condition condition = new Condition(player); + for (Pair pair : textList) { + if (RequirementManager.isRequirementMet(condition, pair.right())) { + return PlaceholderAPI.setPlaceholders(player, pair.left()); + } + } + return ""; + } + + public static class Builder { + + private final ConditionalText conditionalText; + + public static Builder of() { + return new Builder(); + } + + public Builder() { + this.conditionalText = new ConditionalText(); + } + + public Builder textList(List> textList) { + conditionalText.textList = textList; + return this; + } + + public ConditionalText build() { + return conditionalText; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/DescentText.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/DescentText.java new file mode 100644 index 0000000..e3a3379 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/DescentText.java @@ -0,0 +1,68 @@ +package net.momirealms.customnameplates.api.mechanic.placeholder; + +import me.clip.placeholderapi.PlaceholderAPI; +import net.momirealms.customnameplates.api.util.FontUtils; +import org.bukkit.OfflinePlayer; + +public class DescentText { + + private int ascent; + private String text; + private boolean isUnicode; + + public DescentText(int ascent, String text, boolean isUnicode) { + this.ascent = ascent; + this.text = text; + this.isUnicode = isUnicode; + } + + private DescentText() { + + } + + public static Builder builder() { + return new Builder(); + } + + public String getValue(OfflinePlayer player) { + var parsed = PlaceholderAPI.setPlaceholders(player, text); + return isUnicode ? FontUtils.surroundAscentUnicodeFont(parsed, ascent) : FontUtils.surroundAscentFont(parsed, ascent); + } + + public static class Builder { + + private final DescentText descentText; + + public static Builder of() { + return new Builder(); + } + + public Builder() { + this.descentText = new DescentText(); + } + + public Builder ascent(int ascent) { + descentText.ascent = ascent; + return this; + } + + public Builder descent(int descent) { + descentText.ascent = 8 - descent; + return this; + } + + public Builder text(String text) { + descentText.text = text; + return this; + } + + public Builder unicode(boolean unicode) { + descentText.isUnicode = unicode; + return this; + } + + public DescentText build() { + return descentText; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/NameplateText.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/NameplateText.java new file mode 100644 index 0000000..932823c --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/NameplateText.java @@ -0,0 +1,71 @@ +package net.momirealms.customnameplates.api.mechanic.placeholder; + +import me.clip.placeholderapi.PlaceholderAPI; +import net.momirealms.customnameplates.api.mechanic.nameplate.Nameplate; +import net.momirealms.customnameplates.api.util.FontUtils; +import org.bukkit.OfflinePlayer; + +public class NameplateText { + + private String text; + private Nameplate nameplate; + + private NameplateText() { + } + + public NameplateText(String text, Nameplate nameplate) { + this.text = text; + this.nameplate = nameplate; + } + + public String getText() { + return text; + } + + public Nameplate getNameplate() { + return nameplate; + } + + public String getValue(OfflinePlayer player) { + String temp; + switch (nameplate.getTeamColor()) { + case CUSTOM -> temp = nameplate.getNamePrefix() + text + nameplate.getNameSuffix(); + case NONE -> temp = text; + default -> temp = "<" + nameplate.getTeamColor().name() + ">" + text + ""; + } + String parsed = PlaceholderAPI.setPlaceholders(player, temp); + int parsedWidth = FontUtils.getTextWidth(parsed); + return nameplate.getPrefixWithFont(parsedWidth) + parsed + nameplate.getSuffixWithFont(parsedWidth); + } + + public static Builder builder() { + return new Builder(); + } + + public static class Builder { + + private final NameplateText text; + + public Builder() { + this.text = new NameplateText(); + } + + public static Builder of() { + return new Builder(); + } + + public Builder text(String value) { + text.text = value; + return this; + } + + public Builder nameplate(Nameplate value) { + text.nameplate = value; + return this; + } + + public NameplateText build() { + return text; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/StaticText.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/StaticText.java new file mode 100644 index 0000000..a2bf8a4 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/StaticText.java @@ -0,0 +1,98 @@ +package net.momirealms.customnameplates.api.mechanic.placeholder; + +import me.clip.placeholderapi.PlaceholderAPI; +import net.momirealms.customnameplates.api.mechanic.font.OffsetFont; +import net.momirealms.customnameplates.api.util.FontUtils; +import org.bukkit.OfflinePlayer; + +public class StaticText { + + private String text; + private int value; + private StaticState staticState; + + private StaticText() { + } + + public StaticText(String text, int value, StaticState staticState) { + this.text = text; + this.value = value; + this.staticState = staticState; + } + + public String getText() { + return text; + } + + public int getValue() { + return value; + } + + public static Builder builder() { + return new Builder(); + } + + public StaticState getStaticState() { + return staticState; + } + + public String getValue(OfflinePlayer player) { + String parsed = PlaceholderAPI.setPlaceholders(player, text); + int parsedWidth = FontUtils.getTextWidth(parsed); + switch (staticState) { + case LEFT -> { + return parsed + FontUtils.surroundNameplateFont(OffsetFont.getOffsetChars(value - parsedWidth)); + } + case RIGHT -> { + return FontUtils.surroundNameplateFont(OffsetFont.getOffsetChars(value - parsedWidth)) + parsed; + } + case MIDDLE -> { + int half = (value - parsedWidth) / 2; + String left = FontUtils.surroundNameplateFont(OffsetFont.getOffsetChars(half)); + String right = FontUtils.surroundNameplateFont(OffsetFont.getOffsetChars(value - parsedWidth - half)); + return left + parsed + right; + } + default -> { + return ""; + } + } + } + + public static class Builder { + + private final StaticText text; + + public Builder() { + this.text = new StaticText(); + } + + public static Builder of() { + return new Builder(); + } + + public Builder value(int value) { + text.value = value; + return this; + } + + public Builder text(String value) { + text.text = value; + return this; + } + + public Builder state(StaticState state) { + text.staticState = state; + return this; + } + + public StaticText build() { + return text; + } + } + + public enum StaticState { + LEFT, + MIDDLE, + RIGHT + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/SwitchText.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/SwitchText.java new file mode 100644 index 0000000..4b54934 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/SwitchText.java @@ -0,0 +1,63 @@ +package net.momirealms.customnameplates.api.mechanic.placeholder; + +import me.clip.placeholderapi.PlaceholderAPI; +import org.bukkit.entity.Player; + +import java.util.HashMap; + +public class SwitchText { + + private HashMap valueMap; + private String toParse; + private String defaultValue; + + public static Builder builder() { + return new Builder(); + } + + private SwitchText() { + + } + + public SwitchText(HashMap valueMap, String toParse) { + this.valueMap = valueMap; + this.toParse = toParse; + } + + public String getValue(Player player) { + String parsed = PlaceholderAPI.setPlaceholders(player, toParse); + return valueMap.getOrDefault(parsed, defaultValue); + } + + public static class Builder { + + private final SwitchText switchText; + + public Builder() { + this.switchText = new SwitchText(); + } + + public static Builder of() { + return new Builder(); + } + + public Builder toParse(String toParse) { + this.switchText.toParse = toParse; + return this; + } + + public Builder defaultValue(String value) { + this.switchText.defaultValue = value; + return this; + } + + public Builder valueMap(HashMap valueMap) { + this.switchText.valueMap = valueMap; + return this; + } + + public SwitchText build() { + return switchText; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/VanillaHud.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/VanillaHud.java new file mode 100644 index 0000000..6a8d137 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/placeholder/VanillaHud.java @@ -0,0 +1,58 @@ +package net.momirealms.customnameplates.api.mechanic.placeholder; + +import me.clip.placeholderapi.PlaceholderAPI; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.util.FontUtils; +import net.momirealms.customnameplates.api.util.LogUtils; +import org.bukkit.entity.Player; + +public class VanillaHud { + + private final char empty; + private final char half; + private final char full; + private final String maxPapi; + private final String currentPapi; + private final boolean reverse; + + public VanillaHud(String empty, String half, String full, String maxPapi, String currentPapi, boolean reverse) { + this.empty = CustomNameplatesPlugin.get().getImageManager().getImage(empty).getCharacter(); + this.half = CustomNameplatesPlugin.get().getImageManager().getImage(half).getCharacter(); + this.full = CustomNameplatesPlugin.get().getImageManager().getImage(full).getCharacter(); + this.maxPapi = maxPapi; + this.currentPapi = currentPapi; + this.reverse = reverse; + } + + public String getValue(Player player) { + double current; + double max; + try { + current= Double.parseDouble(PlaceholderAPI.setPlaceholders(player, currentPapi)); + max = Double.parseDouble(PlaceholderAPI.setPlaceholders(player, maxPapi)); + } catch (NumberFormatException e) { + current = 1; + max = 1; + LogUtils.warn("Invalid number format when parsing: " + currentPapi + "/" + maxPapi); + } + if (current >= max) current = max; + if (current < 0) current = 0; + int point = (int) ((current / max) * 20); + int full_amount = point / 2; + int half_amount = point % 2; + int empty_amount = 10 - full_amount - half_amount; + StringBuilder builder = new StringBuilder(); + if (reverse) { + builder + .append(String.valueOf(empty).repeat(empty_amount)) + .append(String.valueOf(half).repeat(half_amount)) + .append(String.valueOf(full).repeat(full_amount)); + } else { + builder + .append(String.valueOf(full).repeat(full_amount)) + .append(String.valueOf(half).repeat(half_amount)) + .append(String.valueOf(empty).repeat(empty_amount)); + } + return FontUtils.surroundNameplateFont(builder.toString()); + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/NameplatePlayer.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/NameplatePlayer.java new file mode 100644 index 0000000..0c33ecf --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/NameplatePlayer.java @@ -0,0 +1,13 @@ +package net.momirealms.customnameplates.api.mechanic.tag; + +import net.momirealms.customnameplates.api.mechanic.nameplate.Nameplate; +import org.bukkit.entity.Player; + +public interface NameplatePlayer { + + void preview(); + + void preview(Nameplate nameplate); + + Player getOwner(); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/team/TeamPlayer.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/team/TeamPlayer.java new file mode 100644 index 0000000..b4813e7 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/team/TeamPlayer.java @@ -0,0 +1,109 @@ +package net.momirealms.customnameplates.api.mechanic.tag.team; + +import net.kyori.adventure.text.Component; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.TeamTagManager; +import net.momirealms.customnameplates.api.mechanic.misc.ViewerText; +import net.momirealms.customnameplates.api.mechanic.nameplate.Nameplate; +import net.momirealms.customnameplates.api.mechanic.tag.NameplatePlayer; +import net.momirealms.customnameplates.api.mechanic.team.TeamColor; +import net.momirealms.customnameplates.api.mechanic.team.TeamTagVisibility; +import org.bukkit.entity.Player; + +import java.util.Vector; + +public class TeamPlayer implements NameplatePlayer { + + private final TeamTagManager manager; + private final Player owner; + private final ViewerText prefix; + private final ViewerText suffix; + private final Vector nearbyPlayers; + + public TeamPlayer(TeamTagManager manager, Player owner, String prefix, String suffix) { + this.manager = manager; + this.owner = owner; + this.prefix = new ViewerText(owner, prefix); + this.suffix = new ViewerText(owner, suffix); + this.nearbyPlayers = new Vector<>(); + this.prefix.updateForOwner(); + this.suffix.updateForOwner(); + } + + public void updateForNearbyPlayers(boolean force) { + this.prefix.updateForOwner(); + this.suffix.updateForOwner(); + for (Player viewer : nearbyPlayers) { + updateForOne(viewer, force); + } + } + + public void removeNearbyPlayer(Player player) { + if (!nearbyPlayers.contains(player)) { + return; + } + nearbyPlayers.remove(player); + removeForOne(player); + prefix.removeViewer(player); + suffix.removeViewer(player); + } + + public void addNearbyPlayer(Player player) { + if (nearbyPlayers.contains(player)) { + return; + } + nearbyPlayers.add(player); + updateForOne(player, false); + } + + public void destroy() { + manager.removeTeamPlayerFromMap(owner.getUniqueId()); + for (Player viewer : nearbyPlayers) { + removeForOne(viewer); + } + nearbyPlayers.clear(); + } + + private void updateForOne(Player viewer, boolean force) { + try { + if ((prefix.updateForViewer(viewer) | suffix.updateForViewer(viewer)) || force) { + CustomNameplatesPlugin.get().getTeamManager().updateTeam( + owner, + viewer, + CustomNameplatesPlugin.get().getAdventure().getComponentFromMiniMessage(prefix.getLatestValue(viewer)), + CustomNameplatesPlugin.get().getAdventure().getComponentFromMiniMessage(suffix.getLatestValue(viewer)), + CustomNameplatesPlugin.get().getNameplateManager().getTeamColor(owner), + TeamTagVisibility.ALWAYS + ); + } + } catch (Exception e) { + e.printStackTrace(); + } + } + + private void removeForOne(Player viewer) { + CustomNameplatesPlugin.get().getTeamManager().updateTeam( + owner, + viewer, + Component.text(""), + Component.text(""), + TeamColor.WHITE, + TeamTagVisibility.ALWAYS + ); + } + + @Override + public void preview() { + + } + + @Override + public void preview(Nameplate nameplate) { + + } + + @Override + public Player getOwner() { + return owner; + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/NamedEntity.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/NamedEntity.java new file mode 100644 index 0000000..9627010 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/NamedEntity.java @@ -0,0 +1,60 @@ +package net.momirealms.customnameplates.api.mechanic.tag.unlimited; + +import net.momirealms.customnameplates.api.mechanic.misc.ViewerText; +import org.bukkit.entity.Player; +import org.bukkit.entity.Pose; + +import java.util.UUID; + +public interface NamedEntity { + + boolean canSee(Player viewer); + + void timer(); + + boolean canShow(); + + boolean isShownTo(Player viewer); + + void spawn(Player viewer, Pose pose); + + void spawn(Pose pose); + + void destroy(); + + void destroy(Player viewer); + + void teleport(double x, double y, double z, boolean onGround); + + void teleport(Player viewer, double x, double y, double z, boolean onGround); + + void setSneak(boolean sneaking, boolean respawn); + + void removePlayerFromViewers(Player player); + + void addPlayerToViewers(Player player); + + double getOffset(); + + void setOffset(double v); + + ViewerText getViewerText(); + + int getEntityId(); + + void move(short x, short y, short z, boolean onGround); + + void move(Player viewer, short x, short y, short z, boolean onGround); + + void respawn(Player viewer, Pose pose); + + void respawn(Pose pose); + + void updateText(); + + void updateText(Player viewer); + + UUID getUuid(); + + void handlePose(Pose previous, Pose pose); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/UnlimitedObject.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/UnlimitedObject.java new file mode 100644 index 0000000..f31689f --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/UnlimitedObject.java @@ -0,0 +1,39 @@ +package net.momirealms.customnameplates.api.mechanic.tag.unlimited; + +import net.momirealms.customnameplates.api.manager.UnlimitedTagManager; +import org.bukkit.entity.Entity; +import org.bukkit.entity.Player; +import org.bukkit.entity.Pose; + +import java.util.Vector; + +public abstract class UnlimitedObject { + + protected final UnlimitedTagManager manager; + protected final Entity entity; + protected final Vector nearbyPlayers; + + public UnlimitedObject(UnlimitedTagManager manager, Entity entity) { + this.manager = manager; + this.entity = entity; + this.nearbyPlayers = new Vector<>(); + } + + public Vector getNearbyPlayers() { + return nearbyPlayers; + } + + public abstract void addNearbyPlayer(Player player); + + public abstract void removeNearbyPlayer(Player player); + + public abstract void move(Player receiver, short x, short y, short z, boolean onGround); + + public abstract void teleport(Player receiver, double x, double y, double z, boolean onGround); + + public abstract void destroy(); + + public abstract void sneak(boolean sneaking, boolean flying); + + public abstract void handlePose(Pose previous, Pose pose); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/UnlimitedPlayer.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/UnlimitedPlayer.java new file mode 100644 index 0000000..dd04925 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/UnlimitedPlayer.java @@ -0,0 +1,201 @@ +package net.momirealms.customnameplates.api.mechanic.tag.unlimited; + +import net.kyori.adventure.text.Component; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.UnlimitedTagManager; +import net.momirealms.customnameplates.api.mechanic.nameplate.Nameplate; +import net.momirealms.customnameplates.api.mechanic.tag.NameplatePlayer; +import net.momirealms.customnameplates.api.mechanic.team.TeamColor; +import net.momirealms.customnameplates.api.mechanic.team.TeamTagVisibility; +import org.bukkit.entity.Player; +import org.bukkit.entity.Pose; + +import java.util.Vector; + +public class UnlimitedPlayer extends UnlimitedObject implements NameplatePlayer { + + private final Player owner; + private final Vector tags; + private double hatOffset; + + public UnlimitedPlayer(UnlimitedTagManager manager, Player player) { + super(manager, player); + this.owner = player; + this.tags = new Vector<>(); + } + + /** + * Add a tag for a player + * + * @param tag tag + */ + public void addTag(NamedEntity tag) { + if (tags.contains(tag)) { + return; + } + tags.add(tag); + for (Player all : nearbyPlayers) { + if (tag.canShow() && tag.canSee(all)) { + tag.addPlayerToViewers(all); + } + } + } + + /** + * Remove a tag for a player + * + * @param tag tag + */ + public void removeTag(NamedEntity tag) { + if (tags.remove(tag)) { + tag.destroy(); + } + } + + /** + * Get the tags that player own + * + * @return tags + */ + public Vector getTags() { + return tags; + } + + /** + * Set hat offset. This is useful for cosmetics plugins + * because hat might hide the name tags + * + * @param hatOffset hat offset + */ + public void setHatOffset(double hatOffset) { + this.hatOffset = hatOffset; + } + + @Override + public void preview() { + + } + + @Override + public void preview(Nameplate nameplate) { + + } + + /** + * Get the owner + */ + @Override + public Player getOwner() { + return owner; + } + + /** + * Get the hat offset + */ + public double getHatOffset() { + return hatOffset; + } + + /** + * Add a nearby player so he could see the tag + * This process is automatically handled by CustomNameplates + * + * @param player player + */ + @Override + public void addNearbyPlayer(Player player) { + if (nearbyPlayers.contains(player)) { + return; + } + nearbyPlayers.add(player); + addForOne(player); + for (NamedEntity tag : tags) { + if (tag.canShow() && tag.canSee(player)) { + tag.addPlayerToViewers(player); + } + } + } + + /** + * Remove a nearby player so he would no longer receive tag updates + * This process is automatically handled by CustomNameplates + * + * @param player player + */ + @Override + public void removeNearbyPlayer(Player player) { + if (!nearbyPlayers.contains(player)) { + return; + } + super.nearbyPlayers.remove(player); + removeForOne(player); + for (NamedEntity tag : tags) { + tag.removePlayerFromViewers(player); + } + } + + @Override + public void destroy() { + manager.removeUnlimitedObjectFromMap(entity.getUniqueId()); + for (Player viewer : nearbyPlayers) { + removeForOne(viewer); + } + for (NamedEntity tag : tags) { + tag.destroy(); + } + nearbyPlayers.clear(); + tags.clear(); + } + + public void move(Player receiver, short x, short y, short z, boolean onGround) { + for (NamedEntity tag : tags) { + tag.move(receiver, x, y, z, onGround); + } + } + + public void teleport(Player receiver, double x, double y, double z, boolean onGround) { + for (NamedEntity tag : tags) { + tag.teleport(receiver, x, y, z, onGround); + } + } + + public void sneak(boolean sneaking, boolean flying) { + for (NamedEntity tag : tags) { + tag.setSneak(sneaking, !flying); + } + } + + public void handlePose(Pose previous, Pose pose) { + for (NamedEntity tag : tags) { + tag.handlePose(previous, pose); + } + } + + private void addForOne(Player viewer) { + CustomNameplatesPlugin.get().getTeamManager().updateTeam( + owner, + viewer, + Component.text(""), + Component.text(""), + TeamColor.WHITE, + TeamTagVisibility.NEVER + ); + } + + private void removeForOne(Player viewer) { + CustomNameplatesPlugin.get().getTeamManager().updateTeam( + owner, + viewer, + Component.text(""), + Component.text(""), + TeamColor.WHITE, + TeamTagVisibility.ALWAYS + ); + } + + public void timer() { + for (NamedEntity tag : tags) { + tag.timer(); + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/UnlimitedTagSetting.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/UnlimitedTagSetting.java new file mode 100644 index 0000000..8905345 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/tag/unlimited/UnlimitedTagSetting.java @@ -0,0 +1,106 @@ +package net.momirealms.customnameplates.api.mechanic.tag.unlimited; + +import net.momirealms.customnameplates.api.requirement.Requirement; + +public class UnlimitedTagSetting { + + private double verticalOffset; + private String rawText; + private int checkFrequency; + private int refreshFrequency; + private Requirement[] viewerRequirements; + private Requirement[] ownerRequirements; + + private UnlimitedTagSetting() { + verticalOffset = 0; + rawText = ""; + checkFrequency = 10; + refreshFrequency = 10; + viewerRequirements = new Requirement[0]; + ownerRequirements = new Requirement[0]; + } + + public UnlimitedTagSetting(double verticalOffset, String rawText, int checkFrequency, int refreshFrequency, Requirement[] viewerRequirements, Requirement[] ownerRequirements) { + this.verticalOffset = verticalOffset; + this.rawText = rawText; + this.checkFrequency = checkFrequency; + this.refreshFrequency = refreshFrequency; + this.viewerRequirements = viewerRequirements; + this.ownerRequirements = ownerRequirements; + } + + public static Builder builder() { + return new Builder(); + } + + public double getVerticalOffset() { + return verticalOffset; + } + + public String getRawText() { + return rawText; + } + + public int getCheckFrequency() { + return checkFrequency; + } + + public int getRefreshFrequency() { + return refreshFrequency; + } + + public Requirement[] getViewerRequirements() { + return viewerRequirements; + } + + public Requirement[] getOwnerRequirements() { + return ownerRequirements; + } + + public static class Builder { + + private final UnlimitedTagSetting setting; + + public static Builder of() { + return new Builder(); + } + + public Builder() { + this.setting = new UnlimitedTagSetting(); + } + + public Builder checkFrequency(int checkFrequency) { + this.setting.checkFrequency = checkFrequency; + return this; + } + + public Builder refreshFrequency(int refreshFrequency) { + this.setting.refreshFrequency = refreshFrequency; + return this; + } + + public Builder rawText(String rawText) { + this.setting.rawText = rawText; + return this; + } + + public Builder verticalOffset(double verticalOffset) { + this.setting.verticalOffset = verticalOffset; + return this; + } + + public Builder ownerRequirements(Requirement[] ownerRequirements) { + this.setting.ownerRequirements = ownerRequirements; + return this; + } + + public Builder viewerRequirements(Requirement[] viewerRequirements) { + this.setting.viewerRequirements = viewerRequirements; + return this; + } + + public UnlimitedTagSetting build() { + return setting; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamCollisionRule.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamCollisionRule.java new file mode 100644 index 0000000..341ce87 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamCollisionRule.java @@ -0,0 +1,31 @@ +package net.momirealms.customnameplates.api.mechanic.team; + +import org.jetbrains.annotations.NotNull; + +import java.util.Arrays; + +public enum TeamCollisionRule { + + ALWAYS("always"), + NEVER("never"), + PUSH_OTHER_TEAMS("pushOtherTeams"), + PUSH_OWN_TEAM("pushOwnTeam"); + + private final String id; + + TeamCollisionRule(@NotNull String id) { + this.id = id; + } + + public String getId() { + return id; + } + + @NotNull + public static TeamCollisionRule byId(String id) { + return Arrays.stream(values()) + .filter(mode -> mode.id.equals(id)) + .findFirst() + .orElse(ALWAYS); + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamColor.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamColor.java new file mode 100644 index 0000000..fb20fdf --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamColor.java @@ -0,0 +1,29 @@ +package net.momirealms.customnameplates.api.mechanic.team; + +import java.util.Locale; + +public enum TeamColor { + + NONE, + BLACK, + DARK_BLUE, + DARK_GREEN, + DARK_AQUA, + DARK_RED, + DARK_PURPLE, + GOLD, + GRAY, + DARK_GRAY, + BLUE, + GREEN, + AQUA, + RED, + LIGHT_PURPLE, + YELLOW, + WHITE, + CUSTOM; + + public TeamColor getById(String id) throws IllegalArgumentException { + return valueOf(id.toUpperCase(Locale.ENGLISH)); + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamCreatePacket.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamCreatePacket.java new file mode 100644 index 0000000..d1cede2 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamCreatePacket.java @@ -0,0 +1,154 @@ +package net.momirealms.customnameplates.api.mechanic.team; + +import net.kyori.adventure.text.Component; + +import java.util.Collection; +import java.util.Collections; + +public class TeamCreatePacket { + + // String i + private String teamName; + // Collection j + private Collection members; + + /* + Optional k + */ + // IChatBaseComponent a + private Component teamDisplay; + // IChatBaseComponent b + private Component teamPrefix; + // IChatBaseComponent c + private Component teamSuffix; + // String d + private TeamTagVisibility tagVisibility; + // String e + private TeamCollisionRule collisionRule; + // Enum f + private TeamColor teamColor; + + private TeamCreatePacket() { + this.teamName = ""; + this.members = Collections.singleton(""); + this.teamDisplay = Component.text(""); + this.teamPrefix = Component.text(""); + this.teamSuffix = Component.text(""); + this.tagVisibility = TeamTagVisibility.ALWAYS; + this.collisionRule = TeamCollisionRule.ALWAYS; + this.teamColor = TeamColor.WHITE; + } + + public TeamCreatePacket( + String teamName, + Collection members, + Component teamDisplay, + Component teamPrefix, + Component teamSuffix, + TeamTagVisibility tagVisibility, + TeamCollisionRule collisionRule, + TeamColor teamColor + ) { + this.teamName = teamName; + this.members = members; + this.teamDisplay = teamDisplay; + this.teamPrefix = teamPrefix; + this.teamSuffix = teamSuffix; + this.tagVisibility = tagVisibility; + this.collisionRule = collisionRule; + this.teamColor = teamColor; + } + + public String getTeamName() { + return teamName; + } + + public Collection getMembers() { + return members; + } + + public Component getTeamDisplay() { + return teamDisplay; + } + + public Component getTeamPrefix() { + return teamPrefix; + } + + public Component getTeamSuffix() { + return teamSuffix; + } + + public TeamTagVisibility getTagVisibility() { + return tagVisibility; + } + + public TeamCollisionRule getCollisionRule() { + return collisionRule; + } + + public TeamColor getTeamColor() { + return teamColor; + } + + public static Builder builder() { + return new Builder(); + } + + public static class Builder { + + private final TeamCreatePacket packet; + + public Builder() { + this.packet = new TeamCreatePacket(); + } + + public static Builder of() { + return new Builder(); + } + + public Builder teamName(String name) { + packet.teamName = name; + return this; + } + + public Builder members(Collection members) { + packet.members = members; + return this; + } + + public Builder display(Component display) { + packet.teamDisplay = display; + return this; + } + + public Builder prefix(Component prefix) { + packet.teamPrefix = prefix; + return this; + } + + public Builder suffix(Component suffix) { + packet.teamSuffix = suffix; + return this; + } + + public Builder color(TeamColor color) { + packet.teamColor = color; + return this; + } + + public Builder tagVisibility(TeamTagVisibility visibility) { + packet.tagVisibility = visibility; + return this; + } + + public Builder collisionRule(TeamCollisionRule rule) { + packet.collisionRule = rule; + return this; + } + + public TeamCreatePacket build() { + return packet; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamRemovePacket.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamRemovePacket.java new file mode 100644 index 0000000..60d3038 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamRemovePacket.java @@ -0,0 +1,39 @@ +package net.momirealms.customnameplates.api.mechanic.team; + +public class TeamRemovePacket { + + private String teamName; + + private TeamRemovePacket() { + } + + public TeamRemovePacket(String teamName) { + this.teamName = teamName; + } + + public String getTeamName() { + return teamName; + } + + public static Builder builder() { + return new Builder(); + } + + public static class Builder { + + private final TeamRemovePacket packet; + + public Builder() { + this.packet = new TeamRemovePacket(); + } + + public Builder teamName(String teamName) { + packet.teamName = teamName; + return this; + } + + public TeamRemovePacket build() { + return packet; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamTagVisibility.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamTagVisibility.java new file mode 100644 index 0000000..421d983 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamTagVisibility.java @@ -0,0 +1,31 @@ +package net.momirealms.customnameplates.api.mechanic.team; + +import org.jetbrains.annotations.NotNull; + +import java.util.Arrays; + +public enum TeamTagVisibility { + + ALWAYS("always"), + HIDE_FOR_OTHER_TEAMS("hideForOtherTeams"), + HIDE_FOR_OWN_TEAM("hideForOwnTeam"), + NEVER("never"); + + private final String id; + + TeamTagVisibility(@NotNull String id) { + this.id = id; + } + + public String getId() { + return id; + } + + @NotNull + public static TeamTagVisibility byId(String id) { + return Arrays.stream(values()) + .filter(mode -> mode.id.equals(id)) + .findFirst() + .orElse(ALWAYS); + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamUpdatePacket.java b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamUpdatePacket.java new file mode 100644 index 0000000..4b53360 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/mechanic/team/TeamUpdatePacket.java @@ -0,0 +1,136 @@ +package net.momirealms.customnameplates.api.mechanic.team; + +import net.kyori.adventure.text.Component; + +public class TeamUpdatePacket { + + private String teamName; + + /* + Optional k + */ + // IChatBaseComponent a + private Component teamDisplay; + // IChatBaseComponent b + private Component teamPrefix; + // IChatBaseComponent c + private Component teamSuffix; + // String d + private TeamTagVisibility tagVisibility; + // String e + private TeamCollisionRule collisionRule; + // Enum f + private TeamColor teamColor; + + private TeamUpdatePacket() { + this.teamName = ""; + this.teamDisplay = Component.text(""); + this.teamPrefix = Component.text(""); + this.teamSuffix = Component.text(""); + this.tagVisibility = TeamTagVisibility.ALWAYS; + this.collisionRule = TeamCollisionRule.ALWAYS; + this.teamColor = TeamColor.WHITE; + } + + public TeamUpdatePacket( + String teamName, + Component teamDisplay, + Component teamPrefix, + Component teamSuffix, + TeamTagVisibility tagVisibility, + TeamCollisionRule collisionRule, + TeamColor teamColor + ) { + this.teamName = teamName; + this.teamDisplay = teamDisplay; + this.teamPrefix = teamPrefix; + this.teamSuffix = teamSuffix; + this.tagVisibility = tagVisibility; + this.collisionRule = collisionRule; + this.teamColor = teamColor; + } + + public String getTeamName() { + return teamName; + } + + public Component getTeamDisplay() { + return teamDisplay; + } + + public Component getTeamPrefix() { + return teamPrefix; + } + + public Component getTeamSuffix() { + return teamSuffix; + } + + public TeamTagVisibility getTagVisibility() { + return tagVisibility; + } + + public TeamCollisionRule getCollisionRule() { + return collisionRule; + } + + public TeamColor getTeamColor() { + return teamColor; + } + + public static Builder builder() { + return new Builder(); + } + + public static class Builder { + + private final TeamUpdatePacket packet; + + public Builder() { + this.packet = new TeamUpdatePacket(); + } + + public static Builder of() { + return new Builder(); + } + + public Builder teamName(String name) { + packet.teamName = name; + return this; + } + + public Builder display(Component display) { + packet.teamDisplay = display; + return this; + } + + public Builder prefix(Component prefix) { + packet.teamPrefix = prefix; + return this; + } + + public Builder suffix(Component suffix) { + packet.teamSuffix = suffix; + return this; + } + + public Builder color(TeamColor color) { + packet.teamColor = color; + return this; + } + + public Builder tagVisibility(TeamTagVisibility visibility) { + packet.tagVisibility = visibility; + return this; + } + + public Builder collisionRule(TeamCollisionRule rule) { + packet.collisionRule = rule; + return this; + } + + public TeamUpdatePacket build() { + return packet; + } + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/requirement/Condition.java b/api/src/main/java/net/momirealms/customnameplates/api/requirement/Condition.java new file mode 100644 index 0000000..60238bb --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/requirement/Condition.java @@ -0,0 +1,16 @@ +package net.momirealms.customnameplates.api.requirement; + +import org.bukkit.OfflinePlayer; + +public class Condition { + + private final OfflinePlayer player; + + public Condition(OfflinePlayer player) { + this.player = player; + } + + public OfflinePlayer getOfflinePlayer() { + return player; + } +} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/Requirement.java b/api/src/main/java/net/momirealms/customnameplates/api/requirement/Requirement.java similarity index 75% rename from src/main/java/net/momirealms/customnameplates/object/requirements/Requirement.java rename to api/src/main/java/net/momirealms/customnameplates/api/requirement/Requirement.java index 8ee8b79..9424121 100644 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/Requirement.java +++ b/api/src/main/java/net/momirealms/customnameplates/api/requirement/Requirement.java @@ -15,10 +15,15 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.object.requirements; - -import org.bukkit.entity.Player; +package net.momirealms.customnameplates.api.requirement; public interface Requirement { - boolean isConditionMet(Player player); + + /** + * Is condition met the requirement + * + * @param condition condition + * @return meet or not + */ + boolean isConditionMet(Condition condition); } diff --git a/api/src/main/java/net/momirealms/customnameplates/api/requirement/RequirementExpansion.java b/api/src/main/java/net/momirealms/customnameplates/api/requirement/RequirementExpansion.java new file mode 100644 index 0000000..df9ce14 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/requirement/RequirementExpansion.java @@ -0,0 +1,53 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.api.requirement; + +/** + * An abstract class representing a requirement expansion + * Requirement expansions are used to define custom requirements for various functionalities. + */ +public abstract class RequirementExpansion { + + /** + * Get the version of this requirement expansion. + * + * @return The version of the expansion. + */ + public abstract String getVersion(); + + /** + * Get the author of this requirement expansion. + * + * @return The author of the expansion. + */ + public abstract String getAuthor(); + + /** + * Get the type of requirement provided by this expansion. + * + * @return The type of requirement. + */ + public abstract String getRequirementType(); + + /** + * Get the factory for creating requirements defined by this expansion. + * + * @return The requirement factory. + */ + public abstract RequirementFactory getRequirementFactory(); +} diff --git a/src/main/java/net/momirealms/customnameplates/object/Function.java b/api/src/main/java/net/momirealms/customnameplates/api/requirement/RequirementFactory.java similarity index 64% rename from src/main/java/net/momirealms/customnameplates/object/Function.java rename to api/src/main/java/net/momirealms/customnameplates/api/requirement/RequirementFactory.java index 8b0061b..74b2840 100644 --- a/src/main/java/net/momirealms/customnameplates/object/Function.java +++ b/api/src/main/java/net/momirealms/customnameplates/api/requirement/RequirementFactory.java @@ -15,24 +15,18 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.object; +package net.momirealms.customnameplates.api.requirement; -import org.bukkit.entity.Player; +/** + * An interface for a requirement factory that builds requirements. + */ +public interface RequirementFactory { -public abstract class Function { - - public void load() { - } - - public void unload() { - } - - public void disable() { - } - - public void onJoin(Player player) { - } - - public void onQuit(Player player) { - } + /** + * Build a requirement with the given arguments. + * + * @param args The arguments used to build the requirement. + * @return The built requirement. + */ + Requirement build(Object args); } diff --git a/src/main/java/net/momirealms/customnameplates/object/scheduler/TimerTask.java b/api/src/main/java/net/momirealms/customnameplates/api/scheduler/CancellableTask.java similarity index 76% rename from src/main/java/net/momirealms/customnameplates/object/scheduler/TimerTask.java rename to api/src/main/java/net/momirealms/customnameplates/api/scheduler/CancellableTask.java index 793de07..d226bf8 100644 --- a/src/main/java/net/momirealms/customnameplates/object/scheduler/TimerTask.java +++ b/api/src/main/java/net/momirealms/customnameplates/api/scheduler/CancellableTask.java @@ -15,11 +15,19 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.object.scheduler; +package net.momirealms.customnameplates.api.scheduler; -public interface TimerTask { +public interface CancellableTask { + /** + * Cancel the task + */ void cancel(); + /** + * Get if the task is cancelled or not + * + * @return cancelled or not + */ boolean isCancelled(); } diff --git a/api/src/main/java/net/momirealms/customnameplates/api/scheduler/Scheduler.java b/api/src/main/java/net/momirealms/customnameplates/api/scheduler/Scheduler.java new file mode 100644 index 0000000..9b20769 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/scheduler/Scheduler.java @@ -0,0 +1,93 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.api.scheduler; + +import org.bukkit.Location; + +import java.util.concurrent.TimeUnit; + +public interface Scheduler { + + /** + * Runs a task synchronously on the main server thread or region thread. + * + * @param runnable The task to run. + * @param location The location associated with the task. + */ + void runTaskSync(Runnable runnable, Location location); + + /** + * Runs a task synchronously with a specified delay and period. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delayTicks The delay in ticks before the first execution. + * @param periodTicks The period between subsequent executions in ticks. + * @return A CancellableTask for managing the scheduled task. + */ + CancellableTask runTaskSyncTimer(Runnable runnable, Location location, long delayTicks, long periodTicks); + + /** + * Runs a task asynchronously with a specified delay. + * + * @param runnable The task to run. + * @param delay The delay before the task execution. + * @param timeUnit The time unit for the delay. + * @return A CancellableTask for managing the scheduled task. + */ + CancellableTask runTaskAsyncLater(Runnable runnable, long delay, TimeUnit timeUnit); + + /** + * Runs a task asynchronously. + * + * @param runnable The task to run. + */ + void runTaskAsync(Runnable runnable); + + /** + * Runs a task synchronously with a specified delay. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delay The delay before the task execution. + * @param timeUnit The time unit for the delay. + * @return A CancellableTask for managing the scheduled task. + */ + CancellableTask runTaskSyncLater(Runnable runnable, Location location, long delay, TimeUnit timeUnit); + + /** + * Runs a task synchronously with a specified delay in ticks. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delayTicks The delay in ticks before the task execution. + * @return A CancellableTask for managing the scheduled task. + */ + CancellableTask runTaskSyncLater(Runnable runnable, Location location, long delayTicks); + + /** + * Runs a task asynchronously with a specified delay and period. + * + * @param runnable The task to run. + * @param delay The delay before the first execution. + * @param period The period between subsequent executions. + * @param timeUnit The time unit for the delay and period. + * @return A CancellableTask for managing the scheduled task. + */ + CancellableTask runTaskAsyncTimer(Runnable runnable, long delay, long period, TimeUnit timeUnit); +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/util/FontUtils.java b/api/src/main/java/net/momirealms/customnameplates/api/util/FontUtils.java new file mode 100644 index 0000000..c1afdd0 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/util/FontUtils.java @@ -0,0 +1,76 @@ +package net.momirealms.customnameplates.api.util; + +public class FontUtils { + + private static String namespace; + private static String font; + + /** + * Surround the text with ascent font + * + * @param text text + * @param ascent ascent + * @return ascent font text + */ + public static String surroundAscentFont(String text, int ascent) { + return getAscentFontTag(ascent) + text + getFontTagCloser(); + } + + /** + * Surround the text with ascent unicode + * + * @param text text + * @param ascent ascent + * @return ascent font text + */ + public static String surroundAscentUnicodeFont(String text, int ascent) { + return getAscentUnicodeFontTag(ascent) + text + getFontTagCloser(); + } + + /** + * Surround the text with custom nameplates font + * + * @param text text + * @return font text + */ + public static String surroundNameplateFont(String text) { + return getMiniMessageFontTag() + text + getFontTagCloser(); + } + + private static String getAscentFontTag(int ascent) { + return ""; + } + + private static String getAscentUnicodeFontTag(int ascent) { + return ""; + } + + private static String getMiniMessageFontTag() { + return ""; + } + + private static String getFontTagCloser() { + return ""; + } + + /** + * Get a text's width + * + * @param text text + * @return width + */ + public static int getTextWidth(String text) { + return 0; + } + + /** + * Set namespace and font + * + * @param n namespace + * @param f font + */ + public static void setNameSpaceAndFont(String n, String f) { + namespace = n; + font = f; + } +} diff --git a/api/src/main/java/net/momirealms/customnameplates/api/util/LogUtils.java b/api/src/main/java/net/momirealms/customnameplates/api/util/LogUtils.java new file mode 100644 index 0000000..dde8f82 --- /dev/null +++ b/api/src/main/java/net/momirealms/customnameplates/api/util/LogUtils.java @@ -0,0 +1,77 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.api.util; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import org.jetbrains.annotations.NotNull; + +import java.util.logging.Level; + +public final class LogUtils { + + /** + * Log an informational message. + * + * @param message The message to log. + */ + public static void info(@NotNull String message) { + CustomNameplatesPlugin.getInstance().getLogger().info(message); + } + + /** + * Log a warning message. + * + * @param message The message to log. + */ + public static void warn(@NotNull String message) { + CustomNameplatesPlugin.getInstance().getLogger().warning(message); + } + + /** + * Log a severe error message. + * + * @param message The message to log. + */ + public static void severe(@NotNull String message) { + CustomNameplatesPlugin.getInstance().getLogger().severe(message); + } + + /** + * Log a warning message with a throwable exception. + * + * @param message The message to log. + * @param throwable The throwable exception to log. + */ + public static void warn(@NotNull String message, Throwable throwable) { + CustomNameplatesPlugin.getInstance().getLogger().log(Level.WARNING, message, throwable); + } + + /** + * Log a severe error message with a throwable exception. + * + * @param message The message to log. + * @param throwable The throwable exception to log. + */ + public static void severe(@NotNull String message, Throwable throwable) { + CustomNameplatesPlugin.getInstance().getLogger().log(Level.SEVERE, message, throwable); + } + + private LogUtils() { + throw new UnsupportedOperationException("This class cannot be instantiated"); + } +} diff --git a/build.gradle b/build.gradle deleted file mode 100644 index 28a7e0e..0000000 --- a/build.gradle +++ /dev/null @@ -1,81 +0,0 @@ -plugins { - id 'java' - id 'com.github.johnrengelman.shadow' version '7.1.2' -} - -group = 'net.momirealms' -version = '2.2.3.18.1' - -repositories { - maven {name = "aliyun-repo"; url = "https://maven.aliyun.com/repository/public/"} - maven {name = "sk89q-repo"; url = "https://maven.enginehub.org/repo/"} - maven {name = 'sonatype'; url = 'https://oss.sonatype.org/content/groups/public/'} - maven {name = 'sonatype-snapshot'; url = 'https://s01.oss.sonatype.org/content/repositories/snapshots/'} - maven {name = "dmulloy2-repo"; url = "https://repo.dmulloy2.net/repository/public/"} - maven {name = "clip-repo"; url = 'https://repo.extendedclip.com/content/repositories/placeholderapi/'} - maven {name = "jitpack"; url = 'https://jitpack.io'} - maven {name = "codecrafter47-repo"; url = 'https://nexus.codecrafter47.de/content/repositories/public/'} - maven {name = "opencollab-snapshot-repo"; url = 'https://repo.opencollab.dev/main/'} - maven {name = 'papermc-repo'; url = 'https://papermc.io/repo/repository/maven-public/'} - maven {name = 'William278-repo'; url = 'https://repo.william278.net/releases/'} - maven {name = 'kryptonmc-repo'; url = 'https://repo.kryptonmc.org/releases'} - mavenCentral() -} - -dependencies { - compileOnly fileTree(dir:'libs',includes:['*.jar']) - compileOnly('dev.folia:folia-api:1.20.1-R0.1-SNAPSHOT') - compileOnly ('me.clip:placeholderapi:2.11.3') - compileOnly ('com.zaxxer:HikariCP:5.0.1') - compileOnly ('commons-io:commons-io:2.11.0') - compileOnly ('dev.dejvokep:boosted-yaml:1.3.1') - compileOnly ("com.comphenix.protocol:ProtocolLib:5.1.0") - compileOnly ('net.md-5:bungeecord-api:1.19-R0.1-SNAPSHOT') - compileOnly ('com.github.LoneDev6:api-itemsadder:3.4.1-r4') - compileOnly ("com.velocitypowered:velocity-api:3.1.1") - compileOnly ('org.geysermc.geyser:api:2.2.0-SNAPSHOT') - compileOnly ("net.william278:velocitab:1.5.1") - compileOnly ('me.neznamy:tab-api:4.0.2') - compileOnly ('com.github.FrancoBM12:API-MagicCosmetics:2.2.5') - annotationProcessor ('com.velocitypowered:velocity-api:3.1.1') - - compileOnly ('org.apache.commons:commons-lang3:3.12.0') - implementation ('net.kyori:adventure-api:4.14.0') - implementation ('net.kyori:adventure-platform-bukkit:4.3.3-SNAPSHOT') - implementation ('net.kyori:adventure-text-minimessage:4.14.0') - implementation ('net.kyori:adventure-text-serializer-gson:4.14.0') - implementation ("org.bstats:bstats-bukkit:3.0.1") - implementation fileTree (dir:'libs',includes:['BiomeAPI.jar']) -} - -def targetJavaVersion = 17 -java { - def javaVersion = JavaVersion.toVersion(targetJavaVersion) - sourceCompatibility = javaVersion - targetCompatibility = javaVersion -} - -tasks.withType(JavaCompile).configureEach { - options.release = targetJavaVersion - options.encoding = "UTF-8" -} - -processResources { - def props = [version: version] - inputs.properties props - filteringCharset 'UTF-8' - filesMatching('plugin.yml') { - expand props - } -} - -shadowJar { - relocate ('net.kyori', 'net.momirealms.customnameplates.libs.net.kyori') - relocate ('org.bstats', 'net.momirealms.customnameplates.libs.org.bstats') - relocate ('net.momirealms.biomeapi', 'net.momirealms.customnameplates.libs.net.momirealms.biomeapi') -} - -tasks.register("delete", Delete).get().delete("build/libs/"+project.name+"-"+project.version+".jar") -tasks.named("build").get().dependsOn("shadowJar").finalizedBy("delete").doLast { - println("Deleting: "+ "build/libs/"+project.name+"-"+project.version+".jar") -} \ No newline at end of file diff --git a/build.gradle.kts b/build.gradle.kts new file mode 100644 index 0000000..1241825 --- /dev/null +++ b/build.gradle.kts @@ -0,0 +1,80 @@ +plugins { + id("java") + id("application") + id("maven-publish") + id("com.github.johnrengelman.shadow") version "8.1.1" +} + +allprojects { + + version = "2.3.0.0" + + apply() + apply(plugin = "java") + apply(plugin = "application") + apply(plugin = "com.github.johnrengelman.shadow") + apply(plugin = "org.gradle.maven-publish") + + application { + mainClass.set("") + } + + repositories { + maven("https://maven.aliyun.com/repository/public/") + mavenCentral() + maven("https://papermc.io/repo/repository/maven-public/") + maven("https://oss.sonatype.org/content/groups/public/") + maven("https://repo.dmulloy2.net/repository/public/") + maven("https://repo.extendedclip.com/content/repositories/placeholderapi/") + maven("https://repo.codemc.org/repository/maven-public/") + maven("https://maven.enginehub.org/repo/") + maven("https://jitpack.io/") + maven("https://mvn.lumine.io/repository/maven-public/") + maven("https://repo.rapture.pw/repository/maven-releases/") + maven("https://nexus.phoenixdevt.fr/repository/maven-public/") + maven("https://r.irepo.space/maven/") + maven("https://repo.auxilor.io/repository/maven-public/") + maven("https://betonquest.org/nexus/repository/betonquest/") + maven("https://repo.william278.net/releases/") + maven("https://s01.oss.sonatype.org/content/repositories/snapshots/") + maven("https://repo.minebench.de/") + maven("https://repo.xenondevs.xyz/releases/") + maven("https://repo.kryptonmc.org/releases") + maven("https://repo.oraxen.com/releases") + } +} + +subprojects { + tasks.processResources { + val props = mapOf("version" to version) + inputs.properties(props) + filteringCharset = "UTF-8" + filesMatching("*plugin.yml") { + expand(props) + } + } + + tasks.withType { + options.encoding = "UTF-8" + options.release.set(17) + } + + tasks.shadowJar { + destinationDirectory.set(file("$rootDir/target")) + archiveClassifier.set("") + archiveFileName.set("CustomNameplates-" + project.name + "-" + project.version + ".jar") + } + + if ("api" == project.name) { + publishing { + publications { + create("mavenJava") { + groupId = "net.momirealms" + artifactId = "CustomNameplates" + version = rootProject.version.toString() + artifact(tasks.shadowJar) + } + } + } + } +} \ No newline at end of file diff --git a/bungeecord/.gitignore b/bungeecord/.gitignore new file mode 100644 index 0000000..b63da45 --- /dev/null +++ b/bungeecord/.gitignore @@ -0,0 +1,42 @@ +.gradle +build/ +!gradle/wrapper/gradle-wrapper.jar +!**/src/main/**/build/ +!**/src/test/**/build/ + +### IntelliJ IDEA ### +.idea/modules.xml +.idea/jarRepositories.xml +.idea/compiler.xml +.idea/libraries/ +*.iws +*.iml +*.ipr +out/ +!**/src/main/**/out/ +!**/src/test/**/out/ + +### Eclipse ### +.apt_generated +.classpath +.factorypath +.project +.settings +.springBeans +.sts4-cache +bin/ +!**/src/main/**/bin/ +!**/src/test/**/bin/ + +### NetBeans ### +/nbproject/private/ +/nbbuild/ +/dist/ +/nbdist/ +/.nb-gradle/ + +### VS Code ### +.vscode/ + +### Mac OS ### +.DS_Store \ No newline at end of file diff --git a/bungeecord/build.gradle.kts b/bungeecord/build.gradle.kts new file mode 100644 index 0000000..0ce6a16 --- /dev/null +++ b/bungeecord/build.gradle.kts @@ -0,0 +1,3 @@ +dependencies { + +} \ No newline at end of file diff --git a/bungeecord/src/main/resources/plugin.yml b/bungeecord/src/main/resources/plugin.yml new file mode 100644 index 0000000..5bf5d9b --- /dev/null +++ b/bungeecord/src/main/resources/plugin.yml @@ -0,0 +1,9 @@ +name: CustomNameplates +version: '${version}' +main: net.momirealms.customnameplates.paper.Main +api-version: 1.17 +authors: [ XiaoMoMi ] +folia-supported: true +depend: + - ProtocolLib + - PlaceholderAPI \ No newline at end of file diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar index 7454180..c1962a7 100644 Binary files a/gradle/wrapper/gradle-wrapper.jar and b/gradle/wrapper/gradle-wrapper.jar differ diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties index fae0804..744c64d 100644 --- a/gradle/wrapper/gradle-wrapper.properties +++ b/gradle/wrapper/gradle-wrapper.properties @@ -1,5 +1,6 @@ distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists -distributionUrl=https\://services.gradle.org/distributions/gradle-8.1.1-bin.zip +distributionUrl=https\://services.gradle.org/distributions/gradle-8.4-bin.zip +networkTimeout=10000 zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists diff --git a/gradlew b/gradlew index 3da45c1..aeb74cb 100644 --- a/gradlew +++ b/gradlew @@ -1,7 +1,7 @@ #!/bin/sh # -# Copyright ? 2015-2021 the original authors. +# Copyright © 2015-2021 the original authors. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. @@ -32,10 +32,10 @@ # Busybox and similar reduced shells will NOT work, because this script # requires all of these POSIX shell features: # * functions; -# * expansions ?$var?, ?${var}?, ?${var:-default}?, ?${var+SET}?, -# ?${var#prefix}?, ?${var%suffix}?, and ?$( cmd )?; -# * compound commands having a testable exit status, especially ?case?; -# * various built-in commands including ?command?, ?set?, and ?ulimit?. +# * expansions «$var», «${var}», «${var:-default}», «${var+SET}», +# «${var#prefix}», «${var%suffix}», and «$( cmd )»; +# * compound commands having a testable exit status, especially «case»; +# * various built-in commands including «command», «set», and «ulimit». # # Important for patching: # @@ -55,7 +55,7 @@ # Darwin, MinGW, and NonStop. # # (3) This script is generated from the Groovy template -# https://github.com/gradle/gradle/blob/master/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt +# https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt # within the Gradle project. # # You can find Gradle at https://github.com/gradle/gradle/. @@ -80,13 +80,10 @@ do esac done -APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit - -APP_NAME="Gradle" +# This is normally unused +# shellcheck disable=SC2034 APP_BASE_NAME=${0##*/} - -# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. -DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' +APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit # Use the maximum available, or set MAX_FD != -1 to use that value. MAX_FD=maximum @@ -143,12 +140,16 @@ fi if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then case $MAX_FD in #( max*) + # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC3045 MAX_FD=$( ulimit -H -n ) || warn "Could not query maximum file descriptor limit" esac case $MAX_FD in #( '' | soft) :;; #( *) + # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC3045 ulimit -n "$MAX_FD" || warn "Could not set maximum file descriptor limit to $MAX_FD" esac @@ -193,6 +194,10 @@ if "$cygwin" || "$msys" ; then done fi + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' + # Collect all arguments for the java command; # * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of # shell script including quotes and variable substitutions, so put them in @@ -205,6 +210,12 @@ set -- \ org.gradle.wrapper.GradleWrapperMain \ "$@" +# Stop when "xargs" is not available. +if ! command -v xargs >/dev/null 2>&1 +then + die "xargs is not available" +fi + # Use "xargs" to parse quoted args. # # With -n1 it outputs one arg per line, with the quotes and backslashes removed. diff --git a/gradlew.bat b/gradlew.bat index 107acd3..93e3f59 100644 --- a/gradlew.bat +++ b/gradlew.bat @@ -14,7 +14,7 @@ @rem limitations under the License. @rem -@if "%DEBUG%" == "" @echo off +@if "%DEBUG%"=="" @echo off @rem ########################################################################## @rem @rem Gradle startup script for Windows @@ -25,7 +25,8 @@ if "%OS%"=="Windows_NT" setlocal set DIRNAME=%~dp0 -if "%DIRNAME%" == "" set DIRNAME=. +if "%DIRNAME%"=="" set DIRNAME=. +@rem This is normally unused set APP_BASE_NAME=%~n0 set APP_HOME=%DIRNAME% @@ -40,7 +41,7 @@ if defined JAVA_HOME goto findJavaFromJavaHome set JAVA_EXE=java.exe %JAVA_EXE% -version >NUL 2>&1 -if "%ERRORLEVEL%" == "0" goto execute +if %ERRORLEVEL% equ 0 goto execute echo. echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. @@ -75,13 +76,15 @@ set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar :end @rem End local scope for the variables with windows NT shell -if "%ERRORLEVEL%"=="0" goto mainEnd +if %ERRORLEVEL% equ 0 goto mainEnd :fail rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of rem the _cmd.exe /c_ return code! -if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 -exit /b 1 +set EXIT_CODE=%ERRORLEVEL% +if %EXIT_CODE% equ 0 set EXIT_CODE=1 +if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% +exit /b %EXIT_CODE% :mainEnd if "%OS%"=="Windows_NT" endlocal diff --git a/jitpack.yml b/jitpack.yml new file mode 100644 index 0000000..1e41e00 --- /dev/null +++ b/jitpack.yml @@ -0,0 +1,2 @@ +jdk: + - openjdk17 \ No newline at end of file diff --git a/libs/MagicCosmetics-2.3.7-API.jar b/libs/MagicCosmetics-2.3.7-API.jar deleted file mode 100644 index f13e6cb..0000000 Binary files a/libs/MagicCosmetics-2.3.7-API.jar and /dev/null differ diff --git a/libs/TrChat-2.0.8.jar b/libs/TrChat-2.0.8.jar deleted file mode 100644 index e21f105..0000000 Binary files a/libs/TrChat-2.0.8.jar and /dev/null differ diff --git a/libs/VentureChat-3.6.0.jar b/libs/VentureChat-3.6.0.jar deleted file mode 100644 index d275083..0000000 Binary files a/libs/VentureChat-3.6.0.jar and /dev/null differ diff --git a/libs/oraxen-api.jar b/libs/oraxen-api.jar deleted file mode 100644 index b1a3bc0..0000000 Binary files a/libs/oraxen-api.jar and /dev/null differ diff --git a/paper/.gitignore b/paper/.gitignore new file mode 100644 index 0000000..b63da45 --- /dev/null +++ b/paper/.gitignore @@ -0,0 +1,42 @@ +.gradle +build/ +!gradle/wrapper/gradle-wrapper.jar +!**/src/main/**/build/ +!**/src/test/**/build/ + +### IntelliJ IDEA ### +.idea/modules.xml +.idea/jarRepositories.xml +.idea/compiler.xml +.idea/libraries/ +*.iws +*.iml +*.ipr +out/ +!**/src/main/**/out/ +!**/src/test/**/out/ + +### Eclipse ### +.apt_generated +.classpath +.factorypath +.project +.settings +.springBeans +.sts4-cache +bin/ +!**/src/main/**/bin/ +!**/src/test/**/bin/ + +### NetBeans ### +/nbproject/private/ +/nbbuild/ +/dist/ +/nbdist/ +/.nb-gradle/ + +### VS Code ### +.vscode/ + +### Mac OS ### +.DS_Store \ No newline at end of file diff --git a/paper/build.gradle.kts b/paper/build.gradle.kts new file mode 100644 index 0000000..3c4ef7d --- /dev/null +++ b/paper/build.gradle.kts @@ -0,0 +1,62 @@ +dependencies { + // server + compileOnly("dev.folia:folia-api:1.20.1-R0.1-SNAPSHOT") + + // command + compileOnly("dev.jorel:commandapi-bukkit-core:9.3.0") + + // packet + compileOnly("com.comphenix.protocol:ProtocolLib:5.2.0-SNAPSHOT") + + // papi + compileOnly("me.clip:placeholderapi:2.11.5") + + // config + compileOnly("dev.dejvokep:boosted-yaml:1.3.1") + + // team + compileOnly("me.neznamy:tab-api:4.0.2") + compileOnly(files("libs/CMI-API-9.6.5.0.jar")) + + // Gson + compileOnly("com.google.code.gson:gson:2.10.1") + + // database + compileOnly("org.xerial:sqlite-jdbc:3.43.0.0") + compileOnly("com.h2database:h2:2.2.224") + compileOnly("org.mongodb:mongodb-driver-sync:4.11.1") + compileOnly("com.zaxxer:HikariCP:5.0.1") + compileOnly("redis.clients:jedis:5.1.0") + + // others + compileOnly("com.github.LoneDev6:api-itemsadder:3.5.0c-r5") + compileOnly("io.th0rgal:oraxen:1.165.0") + compileOnly("com.github.FrancoBM12:API-MagicCosmetics:2.2.5") + + // chat channels + compileOnly(files("libs/VentureChat-3.7.1.jar")) + compileOnly(files("libs/TrChat-2.0.11.jar")) + + // api module + implementation(project(":api")) + + // adventure + implementation("net.kyori:adventure-api:4.15.0") + implementation("net.kyori:adventure-platform-bukkit:4.3.1") + implementation("net.kyori:adventure-text-minimessage:4.15.0") + implementation("net.kyori:adventure-text-serializer-legacy:4.15.0") + + // bStats + implementation("org.bstats:bstats-bukkit:3.0.2") + + // local lib + implementation(files("libs/BiomeAPI.jar")) +} + +tasks { + shadowJar { +// relocate ("net.kyori", "net.momirealms.customnameplates.libraries") + relocate ("org.bstats", "net.momirealms.customnameplates.libraries.bstats") + relocate ("net.momirealms.biomeapi", "net.momirealms.customnameplates.libraries.biomeapi") + } +} diff --git a/libs/BiomeAPI.jar b/paper/libs/BiomeAPI.jar similarity index 67% rename from libs/BiomeAPI.jar rename to paper/libs/BiomeAPI.jar index 470bb2c..59096aa 100644 Binary files a/libs/BiomeAPI.jar and b/paper/libs/BiomeAPI.jar differ diff --git a/paper/libs/CMI-API-9.6.5.0.jar b/paper/libs/CMI-API-9.6.5.0.jar new file mode 100644 index 0000000..9653148 Binary files /dev/null and b/paper/libs/CMI-API-9.6.5.0.jar differ diff --git a/paper/libs/TrChat-2.0.11.jar b/paper/libs/TrChat-2.0.11.jar new file mode 100644 index 0000000..50bd40d Binary files /dev/null and b/paper/libs/TrChat-2.0.11.jar differ diff --git a/paper/libs/VentureChat-3.7.1.jar b/paper/libs/VentureChat-3.7.1.jar new file mode 100644 index 0000000..c646c95 Binary files /dev/null and b/paper/libs/VentureChat-3.7.1.jar differ diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/CustomNameplatesPluginImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/CustomNameplatesPluginImpl.java new file mode 100644 index 0000000..3e81972 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/CustomNameplatesPluginImpl.java @@ -0,0 +1,147 @@ +package net.momirealms.customnameplates.paper; + +import net.kyori.adventure.text.Component; +import net.kyori.adventure.text.serializer.gson.GsonComponentSerializer; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.event.CustomNameplatesReloadEvent; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.adventure.AdventureManagerImpl; +import net.momirealms.customnameplates.paper.command.CommandManager; +import net.momirealms.customnameplates.paper.helper.LibraryLoader; +import net.momirealms.customnameplates.paper.mechanic.actionbar.ActionBarManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.background.BackGroundManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.bossbar.BossBarManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.image.ImageManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.misc.CoolDownManager; +import net.momirealms.customnameplates.paper.mechanic.misc.PacketManager; +import net.momirealms.customnameplates.paper.mechanic.misc.VersionManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.nameplate.NameplateManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.pack.ResourcePackManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.placeholder.PlaceholderManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.requirement.RequirementManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.team.TeamManagerImpl; +import net.momirealms.customnameplates.paper.scheduler.SchedulerImpl; +import net.momirealms.customnameplates.paper.setting.CNConfig; +import net.momirealms.customnameplates.paper.setting.CNLocale; +import net.momirealms.customnameplates.paper.storage.StorageManagerImpl; +import net.momirealms.customnameplates.paper.util.ReflectionUtils; +import org.bukkit.configuration.file.YamlConfiguration; + +import java.io.File; +import java.util.TimeZone; + +public class CustomNameplatesPluginImpl extends CustomNameplatesPlugin { + + private CoolDownManager coolDownManager; + private PacketManager packetManager; + + @Override + public void onLoad() { + this.loadLibraries(); + ReflectionUtils.load(); + } + + @Override + public void onEnable() { + this.adventureManager = new AdventureManagerImpl(this); + this.versionManager = new VersionManagerImpl(this); + this.scheduler = new SchedulerImpl(this); + this.storageManager = new StorageManagerImpl(this); + this.requirementManager = new RequirementManagerImpl(this); + this.bossBarManager = new BossBarManagerImpl(this); + this.imageManager = new ImageManagerImpl(this); + this.placeholderManager = new PlaceholderManagerImpl(this); + this.backGroundManager = new BackGroundManagerImpl(this); + this.resourcePackManager = new ResourcePackManagerImpl(this); + this.nameplateManager = new NameplateManagerImpl(this); + this.teamManager = new TeamManagerImpl(this); + this.actionBarManager = new ActionBarManagerImpl(this); + this.coolDownManager = new CoolDownManager(this); + this.packetManager = new PacketManager(this); + new CommandManager(this).load(); + this.reload(CNConfig.generatePackOnStart); + this.versionManager.checkUpdate().thenAccept(outDated -> { + if (!outDated) this.getAdventure().sendConsoleMessage("[CustomNameplates] You are using the latest version."); + else this.getAdventure().sendConsoleMessage("[CustomNameplates] Update is available: https://polymart.org/resource/2543"); + }); + } + + @Override + public void onDisable() { + ((SchedulerImpl) this.scheduler).shutdown(); + ((ActionBarManagerImpl) actionBarManager).unload(); + ((NameplateManagerImpl) this.nameplateManager).unload(); + ((TeamManagerImpl) this.teamManager).unload(); + ((BossBarManagerImpl) this.bossBarManager).unload(); + ((ImageManagerImpl) this.imageManager).unload(); + ((BackGroundManagerImpl) this.backGroundManager).unload(); + ((PlaceholderManagerImpl) this.placeholderManager).unload(); + ((RequirementManagerImpl) this.requirementManager).unload(); + ((ResourcePackManagerImpl) this.resourcePackManager).unload(); + ((StorageManagerImpl) this.storageManager).disable(); + ((AdventureManagerImpl) this.adventureManager).close(); + } + + @Override + public void reload(boolean generatePack) { + CNConfig.load(); + CNLocale.load(); + ((SchedulerImpl) this.scheduler).reload(); + ((NameplateManagerImpl) this.nameplateManager).reload(); + ((BackGroundManagerImpl) this.backGroundManager).reload(); + ((TeamManagerImpl) this.teamManager).reload(); + ((StorageManagerImpl) this.storageManager).reload(); + ((RequirementManagerImpl) this.requirementManager).reload(); + ((BossBarManagerImpl) this.bossBarManager).reload(); + ((ActionBarManagerImpl) actionBarManager).reload(); + ((ImageManagerImpl) this.imageManager).reload(); + ((PlaceholderManagerImpl) this.placeholderManager).reload(); + ((ResourcePackManagerImpl) this.resourcePackManager).reload(); + this.resourcePackManager.generateResourcePack(); + CustomNameplatesReloadEvent event = new CustomNameplatesReloadEvent(this); + this.getServer().getPluginManager().callEvent(event); + } + + @Override + public YamlConfiguration getConfig(String file) { + File config = new File(this.getDataFolder(), file); + if (!config.exists()) this.saveResource(file, false); + return YamlConfiguration.loadConfiguration(config); + } + + @Override + public void debug(String s) { + if (CNConfig.debug) { + LogUtils.info(s); + } + } + + public CoolDownManager getCoolDownManager() { + return coolDownManager; + } + + private void loadLibraries() { + String mavenRepo = TimeZone.getDefault().getID().startsWith("Asia") ? + "https://maven.aliyun.com/repository/public/" : "https://repo.maven.apache.org/maven2/"; + LibraryLoader.loadDependencies( + "org.apache.commons:commons-pool2:2.12.0", mavenRepo, + "redis.clients:jedis:5.1.0", mavenRepo, + "dev.dejvokep:boosted-yaml:1.3.1", mavenRepo, + "com.zaxxer:HikariCP:5.0.1", mavenRepo, + "org.mariadb.jdbc:mariadb-java-client:3.3.0", mavenRepo, + "com.mysql:mysql-connector-j:8.2.0", mavenRepo, + "commons-io:commons-io:2.14.0", mavenRepo, + "com.google.code.gson:gson:2.10.1", mavenRepo, + "com.h2database:h2:2.2.224", mavenRepo, + "org.mongodb:mongodb-driver-sync:4.11.1", mavenRepo, + "org.mongodb:mongodb-driver-core:4.11.1", mavenRepo, + "org.mongodb:bson:4.11.1", mavenRepo, + "org.xerial:sqlite-jdbc:3.43.2.2", mavenRepo, + "dev.jorel:commandapi-bukkit-shade:9.3.0", mavenRepo + ); + } + + public PacketManager getPacketManager() { + return packetManager; + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/adventure/AdventureManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/adventure/AdventureManagerImpl.java new file mode 100644 index 0000000..0cb7e21 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/adventure/AdventureManagerImpl.java @@ -0,0 +1,333 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.adventure; + +import com.comphenix.protocol.PacketType; +import com.comphenix.protocol.events.PacketContainer; +import com.google.common.cache.CacheBuilder; +import com.google.common.cache.CacheLoader; +import com.google.common.cache.LoadingCache; +import net.kyori.adventure.audience.Audience; +import net.kyori.adventure.key.Key; +import net.kyori.adventure.platform.bukkit.BukkitAudiences; +import net.kyori.adventure.sound.Sound; +import net.kyori.adventure.text.Component; +import net.kyori.adventure.text.minimessage.MiniMessage; +import net.kyori.adventure.text.serializer.gson.GsonComponentSerializer; +import net.kyori.adventure.text.serializer.legacy.LegacyComponentSerializer; +import net.kyori.adventure.title.Title; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.AdventureManager; +import net.momirealms.customnameplates.paper.mechanic.misc.PacketManager; +import net.momirealms.customnameplates.paper.setting.CNConfig; +import net.momirealms.customnameplates.paper.setting.CNLocale; +import net.momirealms.customnameplates.paper.util.ReflectionUtils; +import org.bukkit.Bukkit; +import org.bukkit.command.CommandSender; +import org.bukkit.command.ConsoleCommandSender; +import org.bukkit.entity.Player; +import org.jetbrains.annotations.NotNull; + +import java.lang.reflect.InvocationTargetException; +import java.time.Duration; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.TimeUnit; + +public class AdventureManagerImpl implements AdventureManager { + + private final BukkitAudiences adventure; + private static AdventureManager instance; + private final CacheSystem cacheSystem; + + public AdventureManagerImpl(CustomNameplatesPlugin plugin) { + this.adventure = BukkitAudiences.create(plugin); + this.cacheSystem = new CacheSystem(); + instance = this; + } + + public static AdventureManager getInstance() { + return instance; + } + + public void close() { + if (adventure != null) + adventure.close(); + } + + @Override + public Object getIChatComponentFromMiniMessage(String text) { + return cacheSystem.getIChatFromCache(text); + } + + @Override + public String stripTags(String text) { + return MiniMessage.miniMessage().stripTags(text); + } + + @Override + public Component getComponentFromMiniMessage(String text) { + if (text == null) { + return Component.empty(); + } + if (CNConfig.legacyColorSupport) { + return cacheSystem.getComponentFromCache(legacyToMiniMessage(text)); + } else { + return cacheSystem.getComponentFromCache(text); + } + } + + @Override + public void sendMessage(CommandSender sender, String s) { + if (s == null) return; + if (sender instanceof Player player) sendPlayerMessage(player, s); + else if (sender instanceof ConsoleCommandSender) sendConsoleMessage(s); + } + + @Override + public void sendMessageWithPrefix(CommandSender sender, String s) { + if (s == null) return; + if (sender instanceof Player player) sendPlayerMessage(player, CNLocale.MSG_PREFIX + s); + else if (sender instanceof ConsoleCommandSender) sendConsoleMessage(CNLocale.MSG_PREFIX + s); + } + + @Override + public void sendConsoleMessage(String s) { + if (s == null) return; + Audience au = adventure.sender(Bukkit.getConsoleSender()); + au.sendMessage(getComponentFromMiniMessage(s)); + } + + @Override + public void sendPlayerMessage(Player player, String s) { + if (s == null) return; + Audience au = adventure.player(player); + au.sendMessage(getComponentFromMiniMessage(s)); + } + + @Override + public void sendTitle(Player player, String title, String subtitle, int in, int duration, int out) { + Audience au = adventure.player(player); + Title.Times times = Title.Times.times(Duration.ofMillis(in), Duration.ofMillis(duration), Duration.ofMillis(out)); + au.showTitle(Title.title(getComponentFromMiniMessage(title), getComponentFromMiniMessage(subtitle), times)); + } + + @Override + public void sendTitle(Player player, Component title, Component subtitle, int in, int duration, int out) { + Audience au = adventure.player(player); + Title.Times times = Title.Times.times(Duration.ofMillis(in), Duration.ofMillis(duration), Duration.ofMillis(out)); + au.showTitle(Title.title(title, subtitle, times)); + } + + @Override + public void sendActionbar(Player player, String text) { + PacketContainer packet = new PacketContainer(PacketType.Play.Server.SET_ACTION_BAR_TEXT); + packet.getModifier().write(0, getIChatComponent(componentToJson(getComponentFromMiniMessage(text).append(Component.score().name("np").objective("ab").build())))); + PacketManager.getInstance().send(player, packet); + } + + @Override + public void sendActionbar(Player player, Component component) { + PacketContainer packet = new PacketContainer(PacketType.Play.Server.SET_ACTION_BAR_TEXT); + packet.getModifier().write(0, getIChatComponent(componentToJson(component))); + PacketManager.getInstance().send(player, packet); + } + + @Override + public void sendSound(Player player, Sound.Source source, Key key, float volume, float pitch) { + Sound sound = Sound.sound(key, source, volume, pitch); + Audience au = adventure.player(player); + au.playSound(sound); + } + + @Override + public void sendSound(Player player, Sound sound) { + Audience au = adventure.player(player); + au.playSound(sound); + } + + @Override + public String legacyToMiniMessage(String legacy) { + StringBuilder stringBuilder = new StringBuilder(); + char[] chars = legacy.toCharArray(); + for (int i = 0; i < chars.length; i++) { + if (!isColorCode(chars[i])) { + stringBuilder.append(chars[i]); + continue; + } + if (i + 1 >= chars.length) { + stringBuilder.append(chars[i]); + continue; + } + switch (chars[i+1]) { + case '0' -> stringBuilder.append(""); + case '1' -> stringBuilder.append(""); + case '2' -> stringBuilder.append(""); + case '3' -> stringBuilder.append(""); + case '4' -> stringBuilder.append(""); + case '5' -> stringBuilder.append(""); + case '6' -> stringBuilder.append(""); + case '7' -> stringBuilder.append(""); + case '8' -> stringBuilder.append(""); + case '9' -> stringBuilder.append(""); + case 'a' -> stringBuilder.append(""); + case 'b' -> stringBuilder.append(""); + case 'c' -> stringBuilder.append(""); + case 'd' -> stringBuilder.append(""); + case 'e' -> stringBuilder.append(""); + case 'f' -> stringBuilder.append(""); + case 'r' -> stringBuilder.append(""); + case 'l' -> stringBuilder.append(""); + case 'm' -> stringBuilder.append(""); + case 'o' -> stringBuilder.append(""); + case 'n' -> stringBuilder.append(""); + case 'k' -> stringBuilder.append(""); + case 'x' -> { + if (i + 13 >= chars.length + || !isColorCode(chars[i+2]) + || !isColorCode(chars[i+4]) + || !isColorCode(chars[i+6]) + || !isColorCode(chars[i+8]) + || !isColorCode(chars[i+10]) + || !isColorCode(chars[i+12])) { + stringBuilder.append(chars[i]); + continue; + } + stringBuilder + .append("<#") + .append(chars[i+3]) + .append(chars[i+5]) + .append(chars[i+7]) + .append(chars[i+9]) + .append(chars[i+11]) + .append(chars[i+13]) + .append(">"); + i += 12; + } + default -> { + stringBuilder.append(chars[i]); + continue; + } + } + i++; + } + return stringBuilder.toString(); + } + + @Override + @SuppressWarnings("BooleanMethodIsAlwaysInverted") + public boolean isColorCode(char c) { + return c == '§' || c == '&'; + } + + @Override + public String componentToLegacy(Component component) { + return LegacyComponentSerializer.legacySection().serialize(component); + } + + @Override + public String componentToJson(Component component) { + return GsonComponentSerializer.gson().serialize(component); + } + + @Override + public Object shadedToOriginal(Component component) { + Object cp; + try { + cp = ReflectionUtils.gsonDeserializeMethod.invoke(ReflectionUtils.gsonInstance, GsonComponentSerializer.gson().serialize(component)); + } catch (InvocationTargetException | IllegalAccessException e) { + e.printStackTrace(); + return null; + } + return cp; + } + + @Override + public String getMiniMessageFormat(Component component) { + return MiniMessage.miniMessage().serialize(component); + } + + @Override + public Object getIChatComponent(String json) { + try { + return ReflectionUtils.iChatComponentMethod.invoke(null, json); + } catch (InvocationTargetException | IllegalAccessException exception) { + exception.printStackTrace(); + return ReflectionUtils.emptyComponent; + } + } + + public class CacheSystem { + + private final LoadingCache miniMessageToIChatComponentCache; + private final LoadingCache miniMessageToComponentCache; + + public CacheSystem() { + miniMessageToIChatComponentCache = CacheBuilder.newBuilder() + .maximumSize(100) + .expireAfterWrite(10, TimeUnit.MINUTES) + .build( + new CacheLoader<>() { + @NotNull + @Override + public Object load(@NotNull String text) { + return fetchIChatData(text); + } + }); + miniMessageToComponentCache = CacheBuilder.newBuilder() + .maximumSize(100) + .expireAfterWrite(10, TimeUnit.MINUTES) + .build( + new CacheLoader<>() { + @NotNull + @Override + public Component load(@NotNull String text) { + return fetchComponent(text); + } + }); + } + + @NotNull + private Object fetchIChatData(String text) { + Component component = getComponentFromMiniMessage(text); + return getIChatComponent(GsonComponentSerializer.gson().serialize(component)); + } + + @NotNull + private Component fetchComponent(String text) { + return getComponentFromMiniMessage(text); + } + + public Object getIChatFromCache(String text) { + try { + return miniMessageToIChatComponentCache.get(text); + } catch (ExecutionException e) { + e.printStackTrace(); + return ReflectionUtils.emptyComponent; + } + } + + public Component getComponentFromCache(String text) { + try { + return miniMessageToComponentCache.get(text); + } catch (ExecutionException e) { + e.printStackTrace(); + return Component.empty(); + } + } + } +} \ No newline at end of file diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/command/CommandManager.java b/paper/src/main/java/net/momirealms/customnameplates/paper/command/CommandManager.java new file mode 100644 index 0000000..f8375c9 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/command/CommandManager.java @@ -0,0 +1,99 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.command; + +import dev.jorel.commandapi.CommandAPI; +import dev.jorel.commandapi.CommandAPIBukkitConfig; +import dev.jorel.commandapi.CommandAPICommand; +import dev.jorel.commandapi.arguments.ArgumentSuggestions; +import dev.jorel.commandapi.arguments.StringArgument; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; +import net.momirealms.customnameplates.paper.adventure.AdventureManagerImpl; +import net.momirealms.customnameplates.paper.setting.CNLocale; +import org.bukkit.entity.Player; + +public class CommandManager { + + private final CustomNameplatesPluginImpl plugin; + + public CommandManager(CustomNameplatesPluginImpl plugin) { + this.plugin = plugin; + if (!CommandAPI.isLoaded()) + CommandAPI.onLoad(new CommandAPIBukkitConfig(plugin).silentLogs(true)); + } + + public void load() { + new CommandAPICommand("customnameplates") + .withAliases("nameplates", "cnameplates") + .withPermission("customnameplates.admin") + .withSubcommands( + getReloadCommand(), + getAboutCommand(), + getEquipCommand(), + getUnEquipCommand() + ) + .register(); + } + + private CommandAPICommand getEquipCommand() { + return new CommandAPICommand("equip") + .withArguments(new StringArgument("nameplate").replaceSuggestions(ArgumentSuggestions.strings(commandSenderSuggestionInfo -> plugin.getNameplateManager().getAvailableNameplates((Player) commandSenderSuggestionInfo.sender()).toArray(new String[0])))) + .executesPlayer((player, args) -> { + String nameplate = (String) args.get("nameplate"); + if (!plugin.getNameplateManager().hasNameplate(player, nameplate)) { + AdventureManagerImpl.getInstance().sendMessageWithPrefix(player, CNLocale.MSG_NAMEPLATE_NOT_AVAILABLE); + return; + } + + if (!plugin.getNameplateManager().equipNameplate(player, nameplate)) { + AdventureManagerImpl.getInstance().sendMessageWithPrefix(player, CNLocale.MSG_NAMEPLATE_NOT_EXISTS); + return; + } + + AdventureManagerImpl.getInstance().sendMessageWithPrefix(player, CNLocale.MSG_EQUIP_NAMEPLATE); + }); + } + + private CommandAPICommand getUnEquipCommand() { + return new CommandAPICommand("unequip") + .executesPlayer((player, args) -> { + plugin.getNameplateManager().unEquipNameplate(player); + AdventureManagerImpl.getInstance().sendMessageWithPrefix(player, CNLocale.MSG_UNEQUIP_NAMEPLATE); + }); + } + + private CommandAPICommand getReloadCommand() { + return new CommandAPICommand("reload") + .executes((sender, args) -> { + long time = System.currentTimeMillis(); + plugin.reload(true); + AdventureManagerImpl.getInstance().sendMessageWithPrefix(sender, CNLocale.MSG_RELOAD.replace("{time}", String.valueOf(System.currentTimeMillis()-time))); + }); + } + + private CommandAPICommand getAboutCommand() { + return new CommandAPICommand("about").executes((sender, args) -> { + AdventureManagerImpl.getInstance().sendMessage(sender, "<#3CB371>⚓ CustomNameplates - <#98FB98>" + CustomNameplatesPlugin.getInstance().getVersionManager().getPluginVersion()); + AdventureManagerImpl.getInstance().sendMessage(sender, "<#7FFFAA>A plugin that provides adjustable images for texts"); + AdventureManagerImpl.getInstance().sendMessage(sender, "<#DA70D6>\uD83E\uDDEA Author: <#FFC0CB>XiaoMoMi"); + AdventureManagerImpl.getInstance().sendMessage(sender, "<#FF7F50>\uD83D\uDD25 Contributors: <#FFA07A>TopOrigin"); + AdventureManagerImpl.getInstance().sendMessage(sender, "<#FFD700>⭐ Document <#A9A9A9>| <#FAFAD2>⛏ Github <#A9A9A9>| <#48D1CC>\uD83D\uDD14 Polymart"); + }); + } +} diff --git a/src/main/java/net/momirealms/customnameplates/helper/LibraryLoader.java b/paper/src/main/java/net/momirealms/customnameplates/paper/helper/LibraryLoader.java similarity index 53% rename from src/main/java/net/momirealms/customnameplates/helper/LibraryLoader.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/helper/LibraryLoader.java index d4cbfa4..07ac490 100644 --- a/src/main/java/net/momirealms/customnameplates/helper/LibraryLoader.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/helper/LibraryLoader.java @@ -23,11 +23,13 @@ * SOFTWARE. */ -package net.momirealms.customnameplates.helper; +package net.momirealms.customnameplates.paper.helper; import com.google.common.base.Supplier; import com.google.common.base.Suppliers; -import net.momirealms.customnameplates.CustomNameplates; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; import java.io.File; import java.io.InputStream; @@ -42,11 +44,10 @@ import java.util.StringJoiner; * Resolves {@link MavenLibrary} annotations for a class, and loads the dependency * into the classloader. */ -@NonnullByDefault public final class LibraryLoader { @SuppressWarnings("Guava") - private static final Supplier URL_INJECTOR = Suppliers.memoize(() -> URLClassLoaderAccess.create((URLClassLoader) CustomNameplates.getInstance().getClass().getClassLoader())); + private static final Supplier URL_INJECTOR = Suppliers.memoize(() -> URLClassLoaderAccess.create((URLClassLoader) CustomNameplatesPluginImpl.getInstance().getClass().getClassLoader())); /** * Resolves all {@link MavenLibrary} annotations on the given object. @@ -73,31 +74,38 @@ public final class LibraryLoader { load(new Dependency(groupId, artifactId, version, repoUrl)); } - public static void load(Dependency d) { - //Log.info(String.startFormat("Loading dependency %s:%s:%s from %s", d.getGroupId(), d.getArtifactId(), d.getVersion(), d.getRepoUrl())); - String name = d.getArtifactId() + "-" + d.getVersion(); + public static void loadDependencies(String... libs) { + if (libs == null || libs.length % 2 != 0) + return; + for (int i = 0; i < libs.length; i+=2) { + String[] split = libs[i].split(":"); + load(new Dependency( + split[0], + split[1], + split[2], + libs[i+1] + )); + } + } + public static void load(Dependency d) { +// LogUtils.info(String.format("Loading dependency %s:%s:%s", d.groupId, d.artifactId, d.version)); + String name = d.artifactId() + "-" + d.version(); File saveLocation = new File(getLibFolder(d), name + ".jar"); if (!saveLocation.exists()) { - try { - Log.info("Dependency '" + name + "' is not already in the libraries folder. Attempting to download..."); + LogUtils.info("Dependency '" + name + "' is not already in the libraries folder. Attempting to download..."); URL url = d.getUrl(); - try (InputStream is = url.openStream()) { Files.copy(is, saveLocation.toPath()); - Log.info("Dependency '" + name + "' successfully downloaded."); + LogUtils.info("Dependency '" + name + "' successfully downloaded."); } - } catch (Exception e) { e.printStackTrace(); } } - - if (!saveLocation.exists()) { + if (!saveLocation.exists()) throw new RuntimeException("Unable to download dependency: " + d); - } - try { URL_INJECTOR.get().addURL(saveLocation.toURI().toURL()); } catch (Exception e) { @@ -105,12 +113,13 @@ public final class LibraryLoader { } } + @SuppressWarnings("all") private static File getLibFolder(Dependency dependency) { - File pluginDataFolder = CustomNameplates.getInstance().getDataFolder(); + File pluginDataFolder = CustomNameplatesPlugin.getInstance().getDataFolder(); File serverDir = pluginDataFolder.getParentFile().getParentFile(); File helperDir = new File(serverDir, "libraries"); - String[] split = dependency.getGroupId().split("\\."); + String[] split = dependency.groupId().split("\\."); File jarDir; StringJoiner stringJoiner = new StringJoiner(File.separator); for (String str : split) { @@ -121,76 +130,53 @@ public final class LibraryLoader { return jarDir; } - @NonnullByDefault - public static final class Dependency { - private final String groupId; - private final String artifactId; - private final String version; - private final String repoUrl; - - public Dependency(String groupId, String artifactId, String version, String repoUrl) { - this.groupId = Objects.requireNonNull(groupId, "groupId"); - this.artifactId = Objects.requireNonNull(artifactId, "artifactId"); - this.version = Objects.requireNonNull(version, "version"); - this.repoUrl = Objects.requireNonNull(repoUrl, "repoUrl"); - } - - public String getGroupId() { - return this.groupId; - } - - public String getArtifactId() { - return this.artifactId; - } - - public String getVersion() { - return this.version; - } - - public String getRepoUrl() { - return this.repoUrl; - } - - public URL getUrl() throws MalformedURLException { - String repo = this.repoUrl; - if (!repo.endsWith("/")) { - repo += "/"; + public record Dependency(String groupId, String artifactId, String version, String repoUrl) { + public Dependency(String groupId, String artifactId, String version, String repoUrl) { + this.groupId = Objects.requireNonNull(groupId, "groupId"); + this.artifactId = Objects.requireNonNull(artifactId, "artifactId"); + this.version = Objects.requireNonNull(version, "version"); + this.repoUrl = Objects.requireNonNull(repoUrl, "repoUrl"); } - repo += "%s/%s/%s/%s-%s.jar"; - String url = String.format(repo, this.groupId.replace(".", "/"), this.artifactId, this.version, this.artifactId, this.version); - return new URL(url); - } + public URL getUrl() throws MalformedURLException { + String repo = this.repoUrl; + if (!repo.endsWith("/")) { + repo += "/"; + } + repo += "%s/%s/%s/%s-%s.jar"; - @Override - public boolean equals(Object o) { - if (o == this) return true; - if (!(o instanceof Dependency)) return false; - final Dependency other = (Dependency) o; - return this.getGroupId().equals(other.getGroupId()) && - this.getArtifactId().equals(other.getArtifactId()) && - this.getVersion().equals(other.getVersion()) && - this.getRepoUrl().equals(other.getRepoUrl()); - } + String url = String.format(repo, this.groupId.replace(".", "/"), this.artifactId, this.version, this.artifactId, this.version); + return new URL(url); + } - @Override - public int hashCode() { - final int PRIME = 59; - int result = 1; - result = result * PRIME + this.getGroupId().hashCode(); - result = result * PRIME + this.getArtifactId().hashCode(); - result = result * PRIME + this.getVersion().hashCode(); - result = result * PRIME + this.getRepoUrl().hashCode(); - return result; - } + @Override + public boolean equals(Object o) { + if (o == this) return true; + if (!(o instanceof Dependency other)) return false; + return this.groupId().equals(other.groupId()) && + this.artifactId().equals(other.artifactId()) && + this.version().equals(other.version()) && + this.repoUrl().equals(other.repoUrl()); + } - @Override - public String toString() { - return "LibraryLoader.Dependency(" + - "groupId=" + this.getGroupId() + ", " + - "artifactId=" + this.getArtifactId() + ", " + - "version=" + this.getVersion() + ", " + - "repoUrl=" + this.getRepoUrl() + ")"; + @Override + public int hashCode() { + final int PRIME = 59; + int result = 1; + result = result * PRIME + this.groupId().hashCode(); + result = result * PRIME + this.artifactId().hashCode(); + result = result * PRIME + this.version().hashCode(); + result = result * PRIME + this.repoUrl().hashCode(); + return result; + } + + @Override + public String toString() { + return "LibraryLoader.Dependency(" + + "groupId=" + this.groupId() + ", " + + "artifactId=" + this.artifactId() + ", " + + "version=" + this.version() + ", " + + "repoUrl=" + this.repoUrl() + ")"; + } } - } } diff --git a/src/main/java/net/momirealms/customnameplates/helper/MavenLibraries.java b/paper/src/main/java/net/momirealms/customnameplates/paper/helper/MavenLibraries.java similarity index 93% rename from src/main/java/net/momirealms/customnameplates/helper/MavenLibraries.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/helper/MavenLibraries.java index ca4747c..f4ce9c3 100644 --- a/src/main/java/net/momirealms/customnameplates/helper/MavenLibraries.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/helper/MavenLibraries.java @@ -23,9 +23,10 @@ * SOFTWARE. */ -package net.momirealms.customnameplates.helper; +package net.momirealms.customnameplates.paper.helper; + +import org.jetbrains.annotations.NotNull; -import javax.annotation.Nonnull; import java.lang.annotation.*; /** @@ -36,7 +37,7 @@ import java.lang.annotation.*; @Retention(RetentionPolicy.RUNTIME) public @interface MavenLibraries { - @Nonnull + @NotNull MavenLibrary[] value() default {}; } diff --git a/src/main/java/net/momirealms/customnameplates/helper/MavenLibrary.java b/paper/src/main/java/net/momirealms/customnameplates/paper/helper/MavenLibrary.java similarity index 93% rename from src/main/java/net/momirealms/customnameplates/helper/MavenLibrary.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/helper/MavenLibrary.java index 27a653a..010f319 100644 --- a/src/main/java/net/momirealms/customnameplates/helper/MavenLibrary.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/helper/MavenLibrary.java @@ -23,9 +23,10 @@ * SOFTWARE. */ -package net.momirealms.customnameplates.helper; +package net.momirealms.customnameplates.paper.helper; + +import org.jetbrains.annotations.NotNull; -import javax.annotation.Nonnull; import java.lang.annotation.*; /** @@ -42,7 +43,7 @@ public @interface MavenLibrary { * * @return the group id of the library */ - @Nonnull + @NotNull String groupId(); /** @@ -50,7 +51,7 @@ public @interface MavenLibrary { * * @return the artifact id of the library */ - @Nonnull + @NotNull String artifactId(); /** @@ -58,7 +59,7 @@ public @interface MavenLibrary { * * @return the version of the library */ - @Nonnull + @NotNull String version(); /** @@ -66,7 +67,7 @@ public @interface MavenLibrary { * * @return the repo where the library can be obtained from */ - @Nonnull + @NotNull Repository repo() default @Repository(url = "https://repo1.maven.org/maven2"); } diff --git a/src/main/java/net/momirealms/customnameplates/helper/Repository.java b/paper/src/main/java/net/momirealms/customnameplates/paper/helper/Repository.java similarity index 94% rename from src/main/java/net/momirealms/customnameplates/helper/Repository.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/helper/Repository.java index 5bb87a5..fa047ec 100644 --- a/src/main/java/net/momirealms/customnameplates/helper/Repository.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/helper/Repository.java @@ -23,9 +23,8 @@ * SOFTWARE. */ -package net.momirealms.customnameplates.helper; +package net.momirealms.customnameplates.paper.helper; -import javax.annotation.Nonnull; import java.lang.annotation.*; /** @@ -41,7 +40,6 @@ public @interface Repository { * * @return the base url of the repository */ - @Nonnull String url(); } diff --git a/src/main/java/net/momirealms/customnameplates/helper/URLClassLoaderAccess.java b/paper/src/main/java/net/momirealms/customnameplates/paper/helper/URLClassLoaderAccess.java similarity index 94% rename from src/main/java/net/momirealms/customnameplates/helper/URLClassLoaderAccess.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/helper/URLClassLoaderAccess.java index 2541880..476d11a 100644 --- a/src/main/java/net/momirealms/customnameplates/helper/URLClassLoaderAccess.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/helper/URLClassLoaderAccess.java @@ -23,9 +23,10 @@ * SOFTWARE. */ -package net.momirealms.customnameplates.helper; +package net.momirealms.customnameplates.paper.helper; + +import org.jetbrains.annotations.NotNull; -import javax.annotation.Nonnull; import java.lang.reflect.Field; import java.net.URL; import java.net.URLClassLoader; @@ -62,7 +63,7 @@ public abstract class URLClassLoaderAccess { * * @param url the URL to add */ - public abstract void addURL(@Nonnull URL url); + public abstract void addURL(@NotNull URL url); /** * Accesses using sun.misc.Unsafe, supported on Java 9+. @@ -116,7 +117,7 @@ public abstract class URLClassLoaderAccess { } @Override - public void addURL(@Nonnull URL url) { + public void addURL(@NotNull URL url) { this.unopenedURLs.add(url); this.pathURLs.add(url); } @@ -130,7 +131,7 @@ public abstract class URLClassLoaderAccess { } @Override - public void addURL(@Nonnull URL url) { + public void addURL(@NotNull URL url) { throw new UnsupportedOperationException(); } } diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/ActionBarConfig.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/ActionBarConfig.java new file mode 100644 index 0000000..9d2cc96 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/ActionBarConfig.java @@ -0,0 +1,79 @@ +package net.momirealms.customnameplates.paper.mechanic.actionbar; + +import net.momirealms.customnameplates.api.requirement.Requirement; +import net.momirealms.customnameplates.paper.mechanic.bossbar.BarColor; +import net.momirealms.customnameplates.paper.mechanic.bossbar.Overlay; +import net.momirealms.customnameplates.paper.mechanic.misc.TimeLimitText; + +public class ActionBarConfig { + + private int checkFrequency; + private Requirement[] requirements; + private TimeLimitText[] textDisplayOrder; + + private ActionBarConfig() { + checkFrequency = 1; + requirements = new Requirement[0]; + textDisplayOrder = new TimeLimitText[0]; + } + + public ActionBarConfig( + Overlay overlay, + BarColor barColor, + int checkFrequency, + Requirement[] requirements, + TimeLimitText[] textDisplayOrder + ) { + this.checkFrequency = checkFrequency; + this.requirements = requirements; + this.textDisplayOrder = textDisplayOrder; + } + + public int getCheckFrequency() { + return checkFrequency; + } + + public Requirement[] getRequirements() { + return requirements; + } + + public TimeLimitText[] getTextDisplayOrder() { + return textDisplayOrder; + } + + public static Builder builder() { + return new Builder(); + } + + public static class Builder { + + private final ActionBarConfig config; + + public Builder() { + this.config = new ActionBarConfig(); + } + + public static Builder of() { + return new Builder(); + } + + public Builder checkFrequency(int checkFrequency) { + config.checkFrequency = checkFrequency; + return this; + } + + public Builder requirement(Requirement[] requirements) { + config.requirements = requirements; + return this; + } + + public Builder displayOrder(TimeLimitText[] textDisplayOrder) { + config.textDisplayOrder = textDisplayOrder; + return this; + } + + public ActionBarConfig build() { + return config; + } + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/ActionBarManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/ActionBarManagerImpl.java new file mode 100644 index 0000000..a3b8031 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/ActionBarManagerImpl.java @@ -0,0 +1,219 @@ +package net.momirealms.customnameplates.paper.mechanic.actionbar; + +import com.comphenix.protocol.ProtocolLibrary; +import com.comphenix.protocol.events.PacketContainer; +import com.comphenix.protocol.events.PacketEvent; +import com.comphenix.protocol.wrappers.EnumWrappers; +import com.comphenix.protocol.wrappers.WrappedChatComponent; +import net.kyori.adventure.text.Component; +import net.kyori.adventure.text.TranslatableComponent; +import net.kyori.adventure.text.serializer.gson.GsonComponentSerializer; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.ActionBarManager; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.adventure.AdventureManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.actionbar.listener.ActionBarListener; +import net.momirealms.customnameplates.paper.mechanic.actionbar.listener.ChatMessageListener; +import net.momirealms.customnameplates.paper.mechanic.actionbar.listener.SystemChatListener; +import net.momirealms.customnameplates.paper.mechanic.misc.DisplayController; +import net.momirealms.customnameplates.paper.setting.CNConfig; +import net.momirealms.customnameplates.paper.util.ConfigUtils; +import org.bukkit.Bukkit; +import org.bukkit.configuration.ConfigurationSection; +import org.bukkit.configuration.file.YamlConfiguration; +import org.bukkit.entity.Player; +import org.bukkit.event.EventHandler; +import org.bukkit.event.HandlerList; +import org.bukkit.event.Listener; +import org.bukkit.event.player.PlayerJoinEvent; +import org.bukkit.event.player.PlayerQuitEvent; +import org.jetbrains.annotations.NotNull; + +import java.io.File; +import java.util.Map; +import java.util.UUID; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.TimeUnit; + +public class ActionBarManagerImpl implements ActionBarManager, Listener { + + private final ConcurrentHashMap receiverMap; + private final CustomNameplatesPlugin plugin; + private final ActionBarListener actionBarListener; + private ActionBarConfig config; + private ChatMessageListener chatMessageListener; + private SystemChatListener systemChatListener; + + public ActionBarManagerImpl(CustomNameplatesPlugin plugin) { + this.receiverMap = new ConcurrentHashMap<>(); + this.plugin = plugin; + this.actionBarListener = new ActionBarListener(this); + if (plugin.getVersionManager().isVersionNewerThan1_19()) { + this.systemChatListener = new SystemChatListener(this); + } else { + this.chatMessageListener = new ChatMessageListener(this); + } + } + + public void load() { + if (!CNConfig.actionBarModule) return; + this.loadConfigs(); + for (Player player : Bukkit.getOnlinePlayers()) { + createActionBarFor(player); + } + + Bukkit.getPluginManager().registerEvents(this, plugin); + if (actionBarListener != null) ProtocolLibrary.getProtocolManager().addPacketListener(actionBarListener); + if (systemChatListener != null) ProtocolLibrary.getProtocolManager().addPacketListener(systemChatListener); + if (chatMessageListener != null) ProtocolLibrary.getProtocolManager().addPacketListener(chatMessageListener); + } + + public void unload() { + for (ActionBarReceiver receiver : receiverMap.values()) { + receiver.cancelTask(); + receiver.destroy(); + } + receiverMap.clear(); + + HandlerList.unregisterAll(this); + if (actionBarListener != null) ProtocolLibrary.getProtocolManager().removePacketListener(actionBarListener); + if (systemChatListener != null) ProtocolLibrary.getProtocolManager().removePacketListener(systemChatListener); + if (chatMessageListener != null) ProtocolLibrary.getProtocolManager().removePacketListener(chatMessageListener); + } + + public void reload() { + unload(); + load(); + } + + private void loadConfigs() { + YamlConfiguration config = plugin.getConfig("configs" + File.separator + "actionbar.yml"); + for (Map.Entry barEntry : config.getValues(false).entrySet()) { + if (!(barEntry.getValue() instanceof ConfigurationSection section)) + return; + + this.config = ActionBarConfig.Builder.of() + .checkFrequency(section.getInt("check-frequency", 10)) + .requirement(plugin.getRequirementManager().getRequirements(section.getConfigurationSection("conditions"))) + .displayOrder(ConfigUtils.getTimeLimitTexts(section.getConfigurationSection("text-display-order"))) + .build(); + } + } + + @EventHandler + public void onPlayerJoin(PlayerJoinEvent event) { + final Player player = event.getPlayer(); + if (CNConfig.sendDelay == 0) { + createActionBarFor(player); + return; + } + this.plugin.getScheduler().runTaskAsyncLater(() -> { + createActionBarFor(player); + }, CNConfig.sendDelay * 50L, TimeUnit.MILLISECONDS); + } + + private void createActionBarFor(Player player) { + if (player == null || !player.isOnline()) { + return; + } + ActionBarReceiver receiver = new ActionBarReceiver(plugin, player, new DisplayController( + player, config.getCheckFrequency(), config.getRequirements(), config.getTextDisplayOrder() + )); + receiver.arrangeTask(); + this.putReceiverToMap(player.getUniqueId(), receiver); + } + + @EventHandler + public void onPlayerQuit(PlayerQuitEvent event) { + ActionBarReceiver receiver = receiverMap.remove(event.getPlayer().getUniqueId()); + if (receiver != null) { + receiver.cancelTask(); + } + } + + private void putReceiverToMap(UUID uuid, ActionBarReceiver actionBarReceiver) { + ActionBarReceiver previous = this.receiverMap.put(uuid, actionBarReceiver); + if (previous != null) { + LogUtils.warn("Unexpected error: Duplicated actionbar created"); + previous.cancelTask(); + } + } + + private ActionBarReceiver getReceiver(UUID uuid) { + return this.receiverMap.get(uuid); + } + + // 1.19+ + public void onReceiveSystemChatPacket(PacketEvent event) { + PacketContainer packet = event.getPacket(); + Boolean overlay = packet.getBooleans().readSafely(0); + if (overlay != null && overlay) { + ActionBarReceiver receiver = getReceiver(event.getPlayer().getUniqueId()); + if (receiver != null) { + event.setCancelled(true); + String json = packet.getStrings().readSafely(0); + if (json != null && !json.equals("")) { + Component component = GsonComponentSerializer.gson().deserialize(json); + if (component instanceof TranslatableComponent) { + // We can't get TranslatableComponent's width :( + return; + } + receiver.setOtherPluginText(AdventureManagerImpl.getInstance().getMiniMessageFormat(component), System.currentTimeMillis()); + } + } + } + } + + // lower version + public void onReceiveChatMessagePacket(PacketEvent event) { + PacketContainer packet = event.getPacket(); + EnumWrappers.ChatType type = packet.getChatTypes().readSafely(0); + if (type == EnumWrappers.ChatType.GAME_INFO) { + ActionBarReceiver receiver = getReceiver(event.getPlayer().getUniqueId()); + if (receiver != null) { + event.setCancelled(true); + WrappedChatComponent wrappedChatComponent = packet.getChatComponents().read(0); + if (wrappedChatComponent != null) { + String json = wrappedChatComponent.getJson(); + Component component = GsonComponentSerializer.gson().deserialize(json); + if (component instanceof TranslatableComponent) { + // We can't get TranslatableComponent's width :( + return; + } + receiver.setOtherPluginText(AdventureManagerImpl.getInstance().getMiniMessageFormat(component), System.currentTimeMillis()); + } + } + } + } + + public void onReceiveActionBarPacket(PacketEvent event) { + PacketContainer packet = event.getPacket(); + WrappedChatComponent wrappedChatComponent = packet.getChatComponents().read(0); + if (wrappedChatComponent != null) { + ActionBarReceiver receiver = getReceiver(event.getPlayer().getUniqueId()); + if (receiver != null) { + String strJson = wrappedChatComponent.getJson(); + // for better performance + if (strJson.contains("\"name\":\"np\",\"objective\":\"ab\"")) { + return; + } + event.setCancelled(true); + receiver.setOtherPluginText( + AdventureManagerImpl.getInstance().getMiniMessageFormat( + GsonComponentSerializer.gson().deserialize(strJson) + ), System.currentTimeMillis() + ); + } + } + } + + @NotNull + @Override + public String getOtherPluginActionBar(Player player) { + ActionBarReceiver receiver = getReceiver(player.getUniqueId()); + if (receiver != null) { + return receiver.getOtherPluginText(); + } + return ""; + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/ActionBarReceiver.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/ActionBarReceiver.java new file mode 100644 index 0000000..67d3740 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/ActionBarReceiver.java @@ -0,0 +1,102 @@ +package net.momirealms.customnameplates.paper.mechanic.actionbar; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.requirement.Condition; +import net.momirealms.customnameplates.api.scheduler.CancellableTask; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.adventure.AdventureManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.misc.DisplayController; +import org.bukkit.entity.Player; + +import java.util.concurrent.TimeUnit; + +public class ActionBarReceiver { + + private final CustomNameplatesPlugin plugin; + private final Player player; + private CancellableTask actionBarTask; + private final DisplayController controller; + private long lastUpdateTime; + private String otherPluginText; + private long expireTime; + + public ActionBarReceiver(CustomNameplatesPlugin plugin, Player player, DisplayController controller) { + this.plugin = plugin; + this.player = player; + this.controller = controller; + this.otherPluginText = ""; + } + + public void timer() { + if (System.currentTimeMillis() > expireTime) { + this.otherPluginText = ""; + } + Condition condition = new Condition(player); + switch (controller.stateCheck(condition)) { + case KEEP -> { + if (!controller.isShown()) { + return; + } + long current = System.currentTimeMillis(); + if (controller.updateText(condition) || shouldSendBeatPack(current)) { + AdventureManagerImpl.getInstance().sendActionbar(player, controller.getLatestContent()); + lastUpdateTime = current; + } + } + case UPDATE -> { + if (controller.isShown()) { + controller.initialize(); + AdventureManagerImpl.getInstance().sendActionbar(player, controller.getLatestContent()); + } else { + AdventureManagerImpl.getInstance().sendActionbar(player, ""); + } + } + } + } + + public void arrangeTask() { + if (this.actionBarTask != null && !this.actionBarTask.isCancelled()) { + LogUtils.warn("There's already an ActionBar task running"); + return; + } + this.actionBarTask = this.plugin.getScheduler().runTaskAsyncTimer(() -> { + try { + timer(); + } catch (Exception e) { + LogUtils.severe( + "Error occurred when sending ActionBars. " + + "This might not be a bug in CustomNameplates. Please report " + + "to the Plugin on the top of the following " + + "stack trace." + ); + e.printStackTrace(); + } + }, 50, 50, TimeUnit.MILLISECONDS); + } + + public void cancelTask() { + if (this.actionBarTask == null) { + LogUtils.warn("ActionBar task has been already cancelled"); + return; + } + this.actionBarTask.cancel(); + this.actionBarTask = null; + } + + public boolean shouldSendBeatPack(long current) { + return current - lastUpdateTime > 1700; + } + + public void setOtherPluginText(String text, long current) { + this.otherPluginText = text; + this.expireTime = current + 3000; + } + + public String getOtherPluginText() { + return otherPluginText; + } + + public void destroy() { + AdventureManagerImpl.getInstance().sendActionbar(player, ""); + } +} diff --git a/src/main/java/net/momirealms/customnameplates/listener/packet/ActionBarListener.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/listener/ActionBarListener.java similarity index 70% rename from src/main/java/net/momirealms/customnameplates/listener/packet/ActionBarListener.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/listener/ActionBarListener.java index 897bfcd..7518d23 100644 --- a/src/main/java/net/momirealms/customnameplates/listener/packet/ActionBarListener.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/listener/ActionBarListener.java @@ -15,21 +15,21 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.listener.packet; +package net.momirealms.customnameplates.paper.mechanic.actionbar.listener; import com.comphenix.protocol.PacketType; import com.comphenix.protocol.events.ListenerPriority; import com.comphenix.protocol.events.PacketAdapter; import com.comphenix.protocol.events.PacketEvent; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.ActionBarManager; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.mechanic.actionbar.ActionBarManagerImpl; public class ActionBarListener extends PacketAdapter { - private final ActionBarManager actionBarManager; + private final ActionBarManagerImpl actionBarManager; - public ActionBarListener(ActionBarManager actionBarManager) { - super(CustomNameplates.getInstance(), ListenerPriority.NORMAL, PacketType.Play.Server.SET_ACTION_BAR_TEXT); + public ActionBarListener(ActionBarManagerImpl actionBarManager) { + super(CustomNameplatesPlugin.getInstance(), ListenerPriority.NORMAL, PacketType.Play.Server.SET_ACTION_BAR_TEXT); this.actionBarManager = actionBarManager; } diff --git a/src/main/java/net/momirealms/customnameplates/listener/packet/ChatMessageListener.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/listener/ChatMessageListener.java similarity index 70% rename from src/main/java/net/momirealms/customnameplates/listener/packet/ChatMessageListener.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/listener/ChatMessageListener.java index e11f030..ccfbcea 100644 --- a/src/main/java/net/momirealms/customnameplates/listener/packet/ChatMessageListener.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/listener/ChatMessageListener.java @@ -15,21 +15,21 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.listener.packet; +package net.momirealms.customnameplates.paper.mechanic.actionbar.listener; import com.comphenix.protocol.PacketType; import com.comphenix.protocol.events.ListenerPriority; import com.comphenix.protocol.events.PacketAdapter; import com.comphenix.protocol.events.PacketEvent; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.ActionBarManager; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.mechanic.actionbar.ActionBarManagerImpl; public class ChatMessageListener extends PacketAdapter { - private final ActionBarManager actionBarManager; + private final ActionBarManagerImpl actionBarManager; - public ChatMessageListener(ActionBarManager actionBarManager) { - super(CustomNameplates.getInstance(), ListenerPriority.NORMAL, PacketType.Play.Server.CHAT); + public ChatMessageListener(ActionBarManagerImpl actionBarManager) { + super(CustomNameplatesPlugin.getInstance(), ListenerPriority.NORMAL, PacketType.Play.Server.CHAT); this.actionBarManager = actionBarManager; } diff --git a/src/main/java/net/momirealms/customnameplates/listener/packet/SystemChatListener.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/listener/SystemChatListener.java similarity index 70% rename from src/main/java/net/momirealms/customnameplates/listener/packet/SystemChatListener.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/listener/SystemChatListener.java index eb0d6e6..04ac7bb 100644 --- a/src/main/java/net/momirealms/customnameplates/listener/packet/SystemChatListener.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/actionbar/listener/SystemChatListener.java @@ -15,21 +15,21 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.listener.packet; +package net.momirealms.customnameplates.paper.mechanic.actionbar.listener; import com.comphenix.protocol.PacketType; import com.comphenix.protocol.events.ListenerPriority; import com.comphenix.protocol.events.PacketAdapter; import com.comphenix.protocol.events.PacketEvent; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.ActionBarManager; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.mechanic.actionbar.ActionBarManagerImpl; public class SystemChatListener extends PacketAdapter { - private final ActionBarManager actionBarManager; + private final ActionBarManagerImpl actionBarManager; - public SystemChatListener(ActionBarManager actionBarManager) { - super(CustomNameplates.getInstance(), ListenerPriority.NORMAL, PacketType.Play.Server.SYSTEM_CHAT); + public SystemChatListener(ActionBarManagerImpl actionBarManager) { + super(CustomNameplatesPlugin.getInstance(), ListenerPriority.NORMAL, PacketType.Play.Server.SYSTEM_CHAT); this.actionBarManager = actionBarManager; } diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/background/BackGroundManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/background/BackGroundManagerImpl.java new file mode 100644 index 0000000..4642f39 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/background/BackGroundManagerImpl.java @@ -0,0 +1,194 @@ +package net.momirealms.customnameplates.paper.mechanic.background; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.BackGroundManager; +import net.momirealms.customnameplates.api.mechanic.background.BackGround; +import net.momirealms.customnameplates.api.mechanic.character.CharacterArranger; +import net.momirealms.customnameplates.api.mechanic.character.ConfiguredChar; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.setting.CNConfig; +import org.bukkit.configuration.file.YamlConfiguration; +import org.jetbrains.annotations.NotNull; + +import java.io.File; +import java.util.Arrays; +import java.util.Collection; +import java.util.Comparator; +import java.util.HashMap; + +public class BackGroundManagerImpl implements BackGroundManager { + + private final HashMap backGroundMap; + private final CustomNameplatesPlugin plugin; + + public BackGroundManagerImpl(CustomNameplatesPlugin plugin) { + this.plugin = plugin; + this.backGroundMap = new HashMap<>(); + } + + public void reload() { + unload(); + load(); + } + + public void load() { + if (!CNConfig.backgroundModule) return; + this.loadConfigs(); + } + + public void unload() { + this.backGroundMap.clear(); + } + + private void loadConfigs() { + File bgFolder = new File(plugin.getDataFolder(),"contents" + File.separator + "backgrounds"); + if (!bgFolder.exists() && bgFolder.mkdirs()) { + saveDefaultBackgrounds(); + } + File[] bgConfigFiles = bgFolder.listFiles(file -> file.getName().endsWith(".yml")); + if (bgConfigFiles == null) return; + Arrays.sort(bgConfigFiles, Comparator.comparing(File::getName)); + for (File bgConfigFile : bgConfigFiles) { + String key = bgConfigFile.getName().substring(0, bgConfigFile.getName().length() - 4); + + YamlConfiguration config = YamlConfiguration.loadConfiguration(bgConfigFile); + int height = config.getInt("middle.height", 14); + int ascent = config.getInt("middle.ascent", 8); + if (config.contains("middle.descent")) ascent = height - config.getInt("middle.descent"); + + var background = BackGround.builder() + .leftMargin(config.getInt("left-margin", 1)) + .rightMargin(config.getInt("right-margin", 1)) + .left( + ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("left.image")) + .height(config.getInt("left.height")) + .ascent(config.getInt("left.ascent")) + .width(config.getInt("left.width")) + .build() + ) + .offset_1( + ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("middle.1")) + .height(height) + .ascent(ascent) + .width(1) + .build() + ) + .offset_2( + ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("middle.2")) + .height(height) + .ascent(ascent) + .width(2) + .build() + ) + .offset_4( + ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("middle.4")) + .height(height) + .ascent(ascent) + .width(4) + .build() + ) + .offset_8( + ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("middle.8")) + .height(height) + .ascent(ascent) + .width(8) + .build() + ) + .offset_16( + ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("middle.16")) + .height(height) + .ascent(ascent) + .width(16) + .build() + ) + .offset_32( + ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("middle.32")) + .height(height) + .ascent(ascent) + .width(32) + .build() + ) + .offset_64( + ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("middle.64")) + .height(height) + .ascent(ascent) + .width(64) + .build() + ) + .offset_128( + ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("middle.128")) + .height(height) + .ascent(ascent) + .width(128) + .build() + ) + .right( + ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("right.image")) + .height(config.getInt("right.height")) + .ascent(config.getInt("right.ascent")) + .width(config.getInt("right.width")) + .build() + ) + .build(); + + if (!registerBackGround(key, background)) { + LogUtils.warn("Found duplicated background: " + key); + } + } + } + + @Override + public BackGround getBackGround(@NotNull String key) { + return backGroundMap.get(key); + } + + @Override + public Collection getBackGrounds() { + return backGroundMap.values(); + } + + @Override + public boolean registerBackGround(@NotNull String key, @NotNull BackGround backGround) { + if (backGroundMap.containsKey(key)) { + return false; + } + backGroundMap.put(key, backGround); + return true; + } + + @Override + public boolean unregisterBackGround(@NotNull String key) { + return backGroundMap.remove(key) != null; + } + + private void saveDefaultBackgrounds() { + String[] bg_list = new String[]{"b0", "b1", "b2", "b4", "b8", "b16","b32","b64","b128"}; + for (String bg : bg_list) { + plugin.saveResource("contents" + File.separator + "backgrounds" + File.separator + bg + ".png", false); + } + String[] config_list = new String[]{"bedrock_1", "bedrock_2", "bedrock_3"}; + for (String config : config_list) { + plugin.saveResource("contents" + File.separator + "backgrounds" + File.separator + config + ".yml", false); + } + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BarColor.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BarColor.java new file mode 100644 index 0000000..e7df8c0 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BarColor.java @@ -0,0 +1,23 @@ +package net.momirealms.customnameplates.paper.mechanic.bossbar; + +import org.jetbrains.annotations.NotNull; + +import java.util.Locale; + +public enum BarColor { + PINK, + BLUE, + RED, + GREEN, + YELLOW, + PURPLE, + WHITE; + + public static BarColor getColor(@NotNull String name) { + try { + return BarColor.valueOf(name.toUpperCase(Locale.ENGLISH)); + } catch (IllegalArgumentException e) { + throw new RuntimeException("Color: " + name + " doesn't exist"); + } + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBar.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBar.java new file mode 100644 index 0000000..c33c93f --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBar.java @@ -0,0 +1,87 @@ +package net.momirealms.customnameplates.paper.mechanic.bossbar; + +import com.comphenix.protocol.PacketType; +import com.comphenix.protocol.events.InternalStructure; +import com.comphenix.protocol.events.PacketContainer; +import com.comphenix.protocol.wrappers.WrappedChatComponent; +import net.momirealms.customnameplates.paper.adventure.AdventureManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.misc.PacketManager; +import net.momirealms.customnameplates.paper.util.ReflectionUtils; +import org.bukkit.entity.Player; + +import java.lang.reflect.InvocationTargetException; +import java.util.UUID; + +public class BossBar { + + private final Overlay overlay; + private final BarColor barColor; + private final UUID uuid; + private final Player owner; + private String latestMiniMessage; + private boolean visible; + + public BossBar(Player owner, Overlay overlay, BarColor barColor) { + this.owner = owner; + this.overlay = overlay; + this.barColor = barColor; + this.uuid = UUID.randomUUID(); + this.visible = false; + } + + public void show() { + PacketManager.getInstance().send(owner, getCreatePacket()); + this.visible = true; + } + + public void hide() { + PacketManager.getInstance().send(owner, getRemovePacket()); + this.visible = false; + } + + public void update() { + PacketManager.getInstance().send(owner, getUpdatePacket()); + } + + public boolean isVisible() { + return visible; + } + + public void setMiniMessageText(String text) { + latestMiniMessage = text; + } + + private PacketContainer getCreatePacket() { + PacketContainer packet = new PacketContainer(PacketType.Play.Server.BOSS); + packet.getModifier().write(0, uuid); + InternalStructure internalStructure = packet.getStructures().read(1); + internalStructure.getModifier().write(0, AdventureManagerImpl.getInstance().getIChatComponentFromMiniMessage(latestMiniMessage)); + internalStructure.getFloat().write(0,1F); + internalStructure.getEnumModifier(BarColor.class, 2).write(0, barColor); + internalStructure.getEnumModifier(Overlay.class, 3).write(0, overlay); + internalStructure.getModifier().write(4, false); + internalStructure.getModifier().write(5, false); + internalStructure.getModifier().write(6, false); + return packet; + } + + private PacketContainer getUpdatePacket() { + PacketContainer packet = new PacketContainer(PacketType.Play.Server.BOSS); + packet.getModifier().write(0, uuid); + try { + Object chatComponent = AdventureManagerImpl.getInstance().getIChatComponentFromMiniMessage(latestMiniMessage); + Object updatePacket = ReflectionUtils.updateConstructor.newInstance(chatComponent); + packet.getModifier().write(1, updatePacket); + } catch (InvocationTargetException | IllegalAccessException | InstantiationException e) { + throw new RuntimeException(e); + } + return packet; + } + + private PacketContainer getRemovePacket() { + PacketContainer packet = new PacketContainer(PacketType.Play.Server.BOSS); + packet.getModifier().write(0, uuid); + packet.getModifier().write(1, ReflectionUtils.removeBossBarPacket); + return packet; + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBarConfig.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBarConfig.java new file mode 100644 index 0000000..6d47a59 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBarConfig.java @@ -0,0 +1,101 @@ +package net.momirealms.customnameplates.paper.mechanic.bossbar; + +import net.momirealms.customnameplates.api.requirement.Requirement; +import net.momirealms.customnameplates.paper.mechanic.misc.TimeLimitText; + +public class BossBarConfig { + + private Overlay overlay; + private BarColor barColor; + private int checkFrequency; + private Requirement[] requirements; + private TimeLimitText[] textDisplayOrder; + + private BossBarConfig() { + overlay = Overlay.PROGRESS; + barColor = BarColor.YELLOW; + checkFrequency = 1; + requirements = new Requirement[0]; + textDisplayOrder = new TimeLimitText[0]; + } + + public BossBarConfig( + Overlay overlay, + BarColor barColor, + int checkFrequency, + Requirement[] requirements, + TimeLimitText[] textDisplayOrder + ) { + this.overlay = overlay; + this.barColor = barColor; + this.checkFrequency = checkFrequency; + this.requirements = requirements; + this.textDisplayOrder = textDisplayOrder; + } + + public Overlay getOverlay() { + return overlay; + } + + public BarColor getBarColor() { + return barColor; + } + + public int getCheckFrequency() { + return checkFrequency; + } + + public Requirement[] getRequirements() { + return requirements; + } + + public TimeLimitText[] getTextDisplayOrder() { + return textDisplayOrder; + } + + public static Builder builder() { + return new Builder(); + } + + public static class Builder { + + private final BossBarConfig config; + + public Builder() { + this.config = new BossBarConfig(); + } + + public static Builder of() { + return new Builder(); + } + + public Builder overlay(Overlay overlay) { + config.overlay = overlay; + return this; + } + + public Builder barColor(BarColor barColor) { + config.barColor = barColor; + return this; + } + + public Builder checkFrequency(int checkFrequency) { + config.checkFrequency = checkFrequency; + return this; + } + + public Builder requirement(Requirement[] requirements) { + config.requirements = requirements; + return this; + } + + public Builder displayOrder(TimeLimitText[] textDisplayOrder) { + config.textDisplayOrder = textDisplayOrder; + return this; + } + + public BossBarConfig build() { + return config; + } + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBarManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBarManagerImpl.java new file mode 100644 index 0000000..30c2fb9 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBarManagerImpl.java @@ -0,0 +1,134 @@ +package net.momirealms.customnameplates.paper.mechanic.bossbar; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.common.Pair; +import net.momirealms.customnameplates.api.manager.BossBarManager; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.mechanic.misc.DisplayController; +import net.momirealms.customnameplates.paper.setting.CNConfig; +import net.momirealms.customnameplates.paper.util.ConfigUtils; +import org.bukkit.Bukkit; +import org.bukkit.configuration.ConfigurationSection; +import org.bukkit.configuration.file.YamlConfiguration; +import org.bukkit.entity.Player; +import org.bukkit.event.EventHandler; +import org.bukkit.event.HandlerList; +import org.bukkit.event.Listener; +import org.bukkit.event.player.PlayerJoinEvent; +import org.bukkit.event.player.PlayerQuitEvent; + +import java.io.File; +import java.util.ArrayList; +import java.util.Map; +import java.util.UUID; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.TimeUnit; + +public class BossBarManagerImpl implements BossBarManager, Listener { + + private final CustomNameplatesPlugin plugin; + private final ConcurrentHashMap receiverMap; + private BossBarConfig[] configs; + + public void load() { + if (!CNConfig.bossBarModule) return; + this.loadConfigs(); + Bukkit.getPluginManager().registerEvents(this, plugin); + for (Player player : Bukkit.getOnlinePlayers()) { + createBossBarFor(player); + } + } + + public void unload() { + HandlerList.unregisterAll(this); + for (BossBarReceiver receiver : receiverMap.values()) { + receiver.cancelTask(); + receiver.destroy(); + } + receiverMap.clear(); + } + + public void reload() { + unload(); + load(); + } + + public BossBarManagerImpl(CustomNameplatesPlugin plugin) { + this.plugin = plugin; + this.receiverMap = new ConcurrentHashMap<>(); + this.configs = new BossBarConfig[0]; + } + + private void loadConfigs() { + ArrayList configs = new ArrayList<>(); + + YamlConfiguration config = plugin.getConfig("configs" + File.separator + "bossbar.yml"); + for (Map.Entry barEntry : config.getValues(false).entrySet()) { + if (!(barEntry.getValue() instanceof ConfigurationSection section)) + return; + + var barConfig = BossBarConfig.Builder.of() + .barColor(BarColor.getColor(section.getString("color", "YELLOW"))) + .overlay(Overlay.getOverlay(section.getString("overlay", "PROGRESS"))) + .checkFrequency(section.getInt("check-frequency", 10)) + .requirement(plugin.getRequirementManager().getRequirements(section.getConfigurationSection("conditions"))) + .displayOrder(ConfigUtils.getTimeLimitTexts(section.getConfigurationSection("text-display-order"))) + .build(); + + configs.add(barConfig); + } + + this.configs = configs.toArray(new BossBarConfig[0]); + } + + @EventHandler + public void onPlayerJoin(PlayerJoinEvent event) { + final Player player = event.getPlayer(); + if (CNConfig.sendDelay == 0) { + createBossBarFor(player); + return; + } + this.plugin.getScheduler().runTaskAsyncLater(() -> { + createBossBarFor(player); + }, CNConfig.sendDelay * 50L, TimeUnit.MILLISECONDS); + } + + private void createBossBarFor(Player player) { + if (player == null || !player.isOnline()) { + return; + } + + Pair[] pairs = new Pair[configs.length]; + for (int i = 0; i < configs.length; i++) { + var config = configs[i]; + pairs[i] = Pair.of( + new DisplayController(player, config.getCheckFrequency(), config.getRequirements(), config.getTextDisplayOrder()), + new BossBar(player, config.getOverlay(), config.getBarColor()) + ); + } + + BossBarReceiver bossBarReceiver = new BossBarReceiver(plugin, player, pairs); + bossBarReceiver.arrangeTask(); + this.putReceiverToMap(player.getUniqueId(), bossBarReceiver); + } + + @EventHandler + public void onPlayerQuit(PlayerQuitEvent event) { + BossBarReceiver receiver = receiverMap.remove(event.getPlayer().getUniqueId()); + if (receiver != null) { + receiver.cancelTask(); + } + } + + private void putReceiverToMap(UUID uuid, BossBarReceiver bossBarReceiver) { + BossBarReceiver previous = this.receiverMap.put(uuid, bossBarReceiver); + if (previous != null) { + LogUtils.warn("Unexpected error: Duplicated bossbar created"); + previous.cancelTask(); + } + } + + private BossBarReceiver getReceiver(UUID uuid) { + return this.receiverMap.get(uuid); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBarReceiver.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBarReceiver.java new file mode 100644 index 0000000..24fd79b --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/BossBarReceiver.java @@ -0,0 +1,98 @@ +package net.momirealms.customnameplates.paper.mechanic.bossbar; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.common.Pair; +import net.momirealms.customnameplates.api.requirement.Condition; +import net.momirealms.customnameplates.api.scheduler.CancellableTask; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.mechanic.misc.DisplayController; +import org.bukkit.entity.Player; + +import java.util.concurrent.TimeUnit; + +public class BossBarReceiver { + + private final CustomNameplatesPlugin plugin; + private final Player player; + private final int bossBarSize; + private final Pair[] bossBars; + private CancellableTask bossBarTask; + + public BossBarReceiver(CustomNameplatesPlugin plugin, Player player, Pair[] bossBars) { + this.player = player; + this.plugin = plugin; + this.bossBars = bossBars; + this.bossBarSize = bossBars.length; + } + + public void arrangeTask() { + if (this.bossBarTask != null && !this.bossBarTask.isCancelled()) { + LogUtils.warn("There's already a BossBar task running"); + return; + } + this.bossBarTask = this.plugin.getScheduler().runTaskAsyncTimer(() -> { + try { + timer(); + } catch (Exception e) { + LogUtils.severe( + "Error occurred when sending BossBars. " + + "This might not be a bug in CustomNameplates. Please report " + + "to the Plugin on the top of the following " + + "stack trace." + ); + e.printStackTrace(); + } + }, 50, 50, TimeUnit.MILLISECONDS); + } + + public void cancelTask() { + if (this.bossBarTask == null) { + LogUtils.warn("BossBar task has been already cancelled"); + return; + } + this.bossBarTask.cancel(); + this.bossBarTask = null; + } + + public void destroy() { + for (Pair pair : bossBars) { + pair.right().hide(); + } + } + + public void timer() { + Condition condition = new Condition(this.player); + for (int i = 0; i < this.bossBarSize; i++) { + Pair pair = this.bossBars[i]; + switch (pair.left().stateCheck(condition)) { + case KEEP -> { + if (!pair.left().isShown()) { + continue; + } + if (pair.left().updateText(condition)) { + pair.right().setMiniMessageText(pair.left().getLatestContent()); + pair.right().update(); + } + } + case UPDATE -> { + var controller = pair.left(); + if (controller.isShown()) { + for (int j = i + 1; j < this.bossBarSize; j++) + if (this.bossBars[j].left().isShown()) + this.bossBars[j].right().hide(); + + controller.initialize(); + pair.right().setMiniMessageText(pair.left().getLatestContent()); + pair.right().show(); + + for (int j = i + 1; j < this.bossBarSize; j++) + if (this.bossBars[j].left().isShown()) + this.bossBars[j].right().show(); + } else { + pair.right().hide(); + } + } + } + } + } +} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/bossbar/Overlay.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/Overlay.java similarity index 64% rename from src/main/java/net/momirealms/customnameplates/object/bossbar/Overlay.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/Overlay.java index 99cf7c0..a251da6 100644 --- a/src/main/java/net/momirealms/customnameplates/object/bossbar/Overlay.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/bossbar/Overlay.java @@ -15,7 +15,11 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.object.bossbar; +package net.momirealms.customnameplates.paper.mechanic.bossbar; + +import org.jetbrains.annotations.NotNull; + +import java.util.Locale; public enum Overlay { @@ -23,5 +27,13 @@ public enum Overlay { NOTCHED_10, NOTCHED_12, NOTCHED_20, - PROGRESS + PROGRESS; + + public static Overlay getOverlay(@NotNull String name) { + try { + return Overlay.valueOf(name.toUpperCase(Locale.ENGLISH)); + } catch (IllegalArgumentException e) { + throw new RuntimeException("Overlay: " + name + " doesn't exist"); + } + } } diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/font/FontData.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/font/FontData.java new file mode 100644 index 0000000..f4c72e0 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/font/FontData.java @@ -0,0 +1,22 @@ +package net.momirealms.customnameplates.paper.mechanic.font; + +import net.momirealms.customnameplates.paper.setting.CNConfig; + +import java.util.HashMap; + +public class FontData { + + private final HashMap widthData; + + public FontData() { + this.widthData = new HashMap<>(); + } + + public void registerCharWidth(char c, short width) { + widthData.put(c, width); + } + + public short getWidth(char c) { + return widthData.getOrDefault(c, CNConfig.defaultCharWidth); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/font/WidthManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/font/WidthManagerImpl.java new file mode 100644 index 0000000..df0d8ea --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/font/WidthManagerImpl.java @@ -0,0 +1,38 @@ +package net.momirealms.customnameplates.paper.mechanic.font; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.WidthManager; + +import java.util.HashMap; + +public class WidthManagerImpl implements WidthManager { + + private final CustomNameplatesPlugin plugin; + private final HashMap fontDataMap; + + private char TAG_START = '<'; + private char TAG_END = '>'; + private char TAG_CLOSE = '/'; + private char TAG_ESCAPE = '\\'; + + public WidthManagerImpl(CustomNameplatesPlugin plugin) { + this.plugin = plugin; + this.fontDataMap = new HashMap<>(); + } + + public void reload() { + + } + + public void load() { + + } + + public void unload() { + fontDataMap.clear(); + } + + public int getTextWidth(String textWithTags) { + return 0; + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/image/ImageManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/image/ImageManagerImpl.java new file mode 100644 index 0000000..6da4e04 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/image/ImageManagerImpl.java @@ -0,0 +1,100 @@ +package net.momirealms.customnameplates.paper.mechanic.image; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.ImageManager; +import net.momirealms.customnameplates.api.mechanic.character.CharacterArranger; +import net.momirealms.customnameplates.api.mechanic.character.ConfiguredChar; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.setting.CNConfig; +import org.bukkit.configuration.file.YamlConfiguration; +import org.jetbrains.annotations.NotNull; + +import java.io.File; +import java.util.Arrays; +import java.util.Collection; +import java.util.Comparator; +import java.util.HashMap; + +public class ImageManagerImpl implements ImageManager { + + private final HashMap imageMap; + private final CustomNameplatesPlugin plugin; + + public ImageManagerImpl(CustomNameplatesPlugin plugin) { + this.plugin = plugin; + this.imageMap = new HashMap<>(); + } + + public void load() { + if (!CNConfig.imageModule) return; + loadConfigs(); + } + + public void unload() { + this.imageMap.clear(); + } + + public void reload() { + unload(); + loadConfigs(); + } + + @Override + public ConfiguredChar getImage(@NotNull String key) { + return imageMap.get(key); + } + + @Override + public Collection getImages() { + return imageMap.values(); + } + + private void loadConfigs() { + File imgFolder = new File(plugin.getDataFolder(), "contents" + File.separator + "images"); + if (!imgFolder.exists() && imgFolder.mkdirs()) { + saveDefaultImages(); + } + File[] configFiles = imgFolder.listFiles(file -> file.getName().endsWith(".yml")); + if (configFiles == null) return; + Arrays.sort(configFiles, Comparator.comparing(File::getName)); + for (File configFile : configFiles) { + + String key = configFile.getName().substring(0, configFile.getName().length() - 4); + YamlConfiguration config = YamlConfiguration.loadConfiguration(configFile); + + if (!registerImage(key, + ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("image", key)) + .height(config.getInt("height", 8)) + .width(config.getInt("width", 8)) + .ascent(config.getInt("ascent", 8)) + .build() + )) { + LogUtils.warn("Found duplicated image: " + key); + } + } + } + + @Override + public boolean registerImage(@NotNull String key, @NotNull ConfiguredChar configuredChar) { + if (imageMap.containsKey(key)) return false; + imageMap.put(key, configuredChar); + return true; + } + + @Override + public boolean unregisterImage(@NotNull String key) { + return this.imageMap.remove(key) != null; + } + + private void saveDefaultImages() { + String[] png_list = new String[]{"bell", "bubble", "clock", "coin", "compass", "weather", "stamina_0", "stamina_1", "stamina_2"}; + String[] part_list = new String[]{".png", ".yml"}; + for (String name : png_list) { + for (String part : part_list) { + plugin.saveResource("contents" + File.separator + "images" + File.separator + name + part, false); + } + } + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/CoolDownManager.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/CoolDownManager.java new file mode 100644 index 0000000..e32e513 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/CoolDownManager.java @@ -0,0 +1,107 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.mechanic.misc; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import org.bukkit.Bukkit; +import org.bukkit.event.EventHandler; +import org.bukkit.event.HandlerList; +import org.bukkit.event.Listener; +import org.bukkit.event.player.PlayerQuitEvent; + +import java.util.HashMap; +import java.util.UUID; +import java.util.concurrent.ConcurrentHashMap; + +/** + * Manages cooldowns for various actions or events. + * Keeps track of cooldown times for different keys associated with player UUIDs. + */ +public class CoolDownManager implements Listener { + + private final ConcurrentHashMap dataMap; + private final CustomNameplatesPlugin plugin; + + public CoolDownManager(CustomNameplatesPlugin plugin) { + this.dataMap = new ConcurrentHashMap<>(); + this.plugin = plugin; + } + + /** + * Checks if a player is currently in cooldown for a specific key. + * + * @param uuid The UUID of the player. + * @param key The key associated with the cooldown. + * @param time The cooldown time in milliseconds. + * @return True if the player is in cooldown, false otherwise. + */ + public boolean isCoolDown(UUID uuid, String key, long time) { + Data data = this.dataMap.computeIfAbsent(uuid, k -> new Data()); + return data.isCoolDown(key, time); + } + + public void load() { + Bukkit.getPluginManager().registerEvents(this, plugin); + } + + public void unload() { + HandlerList.unregisterAll(this); + } + + public void disable() { + unload(); + this.dataMap.clear(); + } + + /** + * Event handler for when a player quits the game. Removes their cooldown data. + * + * @param event The PlayerQuitEvent triggered when a player quits. + */ + @EventHandler + public void onQuit(PlayerQuitEvent event) { + dataMap.remove(event.getPlayer().getUniqueId()); + } + + public static class Data { + + private final HashMap coolDownMap; + + public Data() { + this.coolDownMap = new HashMap<>(); + } + + /** + * Checks if the player is in cooldown for a specific key. + * + * @param key The key associated with the cooldown. + * @param delay The cooldown delay in milliseconds. + * @return True if the player is in cooldown, false otherwise. + */ + public synchronized boolean isCoolDown(String key, long delay) { + long time = System.currentTimeMillis(); + long last = coolDownMap.getOrDefault(key, time - delay); + if (last + delay > time) { + return true; // Player is in cooldown + } else { + coolDownMap.put(key, time); + return false; // Player is not in cooldown + } + } + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/DisplayController.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/DisplayController.java new file mode 100644 index 0000000..17bb9f9 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/DisplayController.java @@ -0,0 +1,116 @@ +package net.momirealms.customnameplates.paper.mechanic.misc; + +import me.clip.placeholderapi.PlaceholderAPI; +import net.momirealms.customnameplates.api.manager.RequirementManager; +import net.momirealms.customnameplates.api.requirement.Condition; +import net.momirealms.customnameplates.api.requirement.Requirement; +import org.bukkit.entity.Player; + +public class DisplayController { + + private final Player owner; + private final int checkFrequency; + private final Requirement[] requirements; + private boolean isShown; + private int checkTimer; + private int refreshTimer; + private String latestValue; + private int timeLeft; + private int index; + private final TimeLimitText[] texts; + + public DisplayController( + Player player, + int checkFrequency, + Requirement[] requirements, + TimeLimitText[] texts + ) { + this.owner = player; + this.checkFrequency = checkFrequency; + this.checkTimer = 0; + this.refreshTimer = 0; + this.requirements = requirements; + this.texts = texts; + } + + public NextStage stateCheck(Condition condition) { + this.checkTimer++; + if (this.checkTimer % checkFrequency != 0) { + return NextStage.KEEP; + } + boolean canShow = RequirementManager.isRequirementMet(condition, requirements); + if (canShow) { + if (this.isShown) { + return NextStage.KEEP; + } else { + this.isShown = true; + return NextStage.UPDATE; + } + } else { + if (!this.isShown) { + return NextStage.KEEP; + } else { + this.isShown = false; + return NextStage.UPDATE; + } + } + } + + public boolean isShown() { + return isShown; + } + + public boolean updateText(Condition condition) { + timeLeft--; + if (timeLeft <= 0) { + do { + index++; + if (index >= texts.length) { + index = 0; + } + } while (!RequirementManager.isRequirementMet(condition, texts[index].getRequirements())); + + timeLeft = texts[index].getDuration(); + refreshTimer = 0; + return updateText(texts[index].getText()); + } + + if (texts[index].getRefreshFrequency() <= 0) { + return false; + } + + refreshTimer++; + if (refreshTimer >= texts[index].getRefreshFrequency()) { + refreshTimer = 0; + return updateText(texts[index].getText()); + } else { + return false; + } + } + + private boolean updateText(String text) { + var newText = PlaceholderAPI.setPlaceholders(owner, text); + if (newText.equals(latestValue)) { + return false; + } + latestValue = newText; + return true; + } + + public void initialize() { + index = 0; + checkTimer = 0; + refreshTimer = 0; + timeLeft = texts[0].getDuration(); + latestValue = PlaceholderAPI.setPlaceholders(owner, texts[0].getText()); + } + + public String getLatestContent() { + return latestValue; + } + + public enum NextStage { + KEEP, + UPDATE + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/MessageManager.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/MessageManager.java new file mode 100644 index 0000000..e716617 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/MessageManager.java @@ -0,0 +1,9 @@ +package net.momirealms.customnameplates.paper.mechanic.misc; + +import org.bukkit.event.Listener; + +public class MessageManager implements Listener { + + + +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/PacketManager.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/PacketManager.java new file mode 100644 index 0000000..d468b4d --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/PacketManager.java @@ -0,0 +1,32 @@ +package net.momirealms.customnameplates.paper.mechanic.misc; + +import com.comphenix.protocol.ProtocolLibrary; +import com.comphenix.protocol.ProtocolManager; +import com.comphenix.protocol.events.PacketContainer; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import org.bukkit.entity.Player; + +public class PacketManager { + + private static PacketManager instance; + private final ProtocolManager protocolManager; + private final CustomNameplatesPlugin plugin; + + public PacketManager(CustomNameplatesPlugin plugin) { + this.plugin = plugin; + this.protocolManager = ProtocolLibrary.getProtocolManager(); + instance = this; + } + + public static PacketManager getInstance() { + return instance; + } + + public void send(Player player, PacketContainer packet) { + if (!player.isOnline()) { + System.out.println(player.getName() + " not online"); + } + this.plugin.debug("Packet sent: " + packet.getType() + " to " + player.getName()); + this.protocolManager.sendServerPacket(player, packet); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/TimeLimitText.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/TimeLimitText.java new file mode 100644 index 0000000..c658112 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/TimeLimitText.java @@ -0,0 +1,82 @@ +package net.momirealms.customnameplates.paper.mechanic.misc; + +import net.momirealms.customnameplates.api.requirement.Requirement; + +public class TimeLimitText { + + private int duration; + private int refreshFrequency; + private String text; + private Requirement[] requirements; + + private TimeLimitText() { + this.duration = 100; + this.refreshFrequency = -1; + this.text = ""; + this.requirements = new Requirement[0]; + } + + public TimeLimitText(int duration, int refreshFrequency, String text, Requirement[] requirements) { + this.duration = duration; + this.text = text; + this.refreshFrequency = refreshFrequency; + this.requirements = requirements; + } + + public int getDuration() { + return duration; + } + + public String getText() { + return text; + } + + public Requirement[] getRequirements() { + return requirements; + } + + public int getRefreshFrequency() { + return refreshFrequency; + } + + public static Builder builder() { + return new Builder(); + } + + public static class Builder { + + private final TimeLimitText text; + + public Builder() { + this.text = new TimeLimitText(); + } + + public static Builder of() { + return new Builder(); + } + + public Builder duration(int duration) { + text.duration = duration; + return this; + } + + public Builder refreshFrequency(int refreshFrequency) { + text.refreshFrequency = refreshFrequency; + return this; + } + + public Builder text(String content) { + text.text = content; + return this; + } + + public Builder requirement(Requirement[] requirements) { + text.requirements = requirements; + return this; + } + + public TimeLimitText build() { + return text; + } + } +} diff --git a/src/main/java/net/momirealms/customnameplates/helper/VersionHelper.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/VersionManagerImpl.java similarity index 59% rename from src/main/java/net/momirealms/customnameplates/helper/VersionHelper.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/VersionManagerImpl.java index bb59f3d..831a401 100644 --- a/src/main/java/net/momirealms/customnameplates/helper/VersionHelper.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/misc/VersionManagerImpl.java @@ -15,120 +15,105 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.helper; +package net.momirealms.customnameplates.paper.mechanic.misc; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.ConfigManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.Bukkit; +import net.momirealms.customnameplates.api.manager.VersionManager; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; import java.io.BufferedReader; -import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import java.net.URL; import java.net.URLConnection; +import java.util.concurrent.CompletableFuture; -public class VersionHelper { +/** + * This class implements the VersionManager interface and is responsible for managing version-related information. + */ +public class VersionManagerImpl implements VersionManager { - private boolean isNewerThan1_19_R2; - private boolean isNewerThan1_20_R2; - private boolean isNewerThan1_19; - private boolean isNewerThan1_20; - private String serverVersion; - private final CustomNameplates plugin; - private final String pluginVersion; - private boolean isLatest; - private final int pack_format; + private final boolean isNewerThan1_19_R2; + private final boolean isNewerThan1_20; + private final boolean isNewerThan1_19; + private final boolean isNewerThan1_20_R2; + private final String serverVersion; + private final CustomNameplatesPluginImpl plugin; private boolean isFolia; - private boolean hasGeyser; + private final String pluginVersion; + private boolean isLatest = true; - public VersionHelper(CustomNameplates plugin) { + @SuppressWarnings("deprecation") + public VersionManagerImpl(CustomNameplatesPluginImpl plugin) { this.plugin = plugin; - this.pluginVersion = plugin.getDescription().getVersion(); - this.hasGeyser = Bukkit.getPluginManager().getPlugin("Geyser-Spigot") != null; - this.initialize(); - this.pack_format = getPack_format(serverVersion); + // Get the server version + serverVersion = plugin.getServer().getClass().getPackage().getName().split("\\.")[3]; + String[] split = serverVersion.split("_"); + int main_ver = Integer.parseInt(split[1]); + // Determine if the server version is newer than 1_19_R2 and 1_20_R1 + if (main_ver >= 20) { + isNewerThan1_20_R2 = Integer.parseInt(split[2].substring(1)) >= 2; + isNewerThan1_19_R2 = true; + isNewerThan1_20 = true; + isNewerThan1_19 = true; + } else if (main_ver == 19) { + isNewerThan1_19_R2 = Integer.parseInt(split[2].substring(1)) >= 2; + isNewerThan1_20 = false; + isNewerThan1_19 = true; + isNewerThan1_20_R2 = false; + } else { + isNewerThan1_19_R2 = false; + isNewerThan1_20 = false; + isNewerThan1_19 = false; + isNewerThan1_20_R2 = false; + } + // Check if the server is Folia try { Class.forName("io.papermc.paper.threadedregions.scheduler.AsyncScheduler"); this.isFolia = true; } catch (ClassNotFoundException ignored) { - this.isFolia = false; - } - } - - public void initialize() { - if (serverVersion == null) { - this.serverVersion = Bukkit.getServer().getClass().getPackage().getName().split("\\.")[3]; - String[] split = serverVersion.split("_"); - int main_ver = Integer.parseInt(split[1]); - if (main_ver >= 20) { - isNewerThan1_20 = true; - isNewerThan1_19_R2 = true; - isNewerThan1_19 = true; - isNewerThan1_20_R2 = Integer.parseInt(split[2].substring(1)) >= 2; - } else if (main_ver == 19) { - isNewerThan1_19_R2 = Integer.parseInt(split[2].substring(1)) >= 2; - isNewerThan1_19 = true; - } else { - isNewerThan1_19_R2 = false; - isNewerThan1_19 = false; - } } + // Get the plugin version + this.pluginVersion = plugin.getDescription().getVersion(); } + @Override public boolean isVersionNewerThan1_19_R2() { return isNewerThan1_19_R2; } - public boolean isVersionNewerThan1_19() { - return isNewerThan1_19; - } - + @Override public boolean isVersionNewerThan1_20() { return isNewerThan1_20; } + @Override public boolean isVersionNewerThan1_20_R2() { return isNewerThan1_20_R2; } - public void checkUpdate() { - plugin.getScheduler().runTaskAsync(() -> { - try { - URL url = new URL("https://api.polymart.org/v1/getResourceInfoSimple/?resource_id=2543&key=version"); - URLConnection conn = url.openConnection(); - conn.setConnectTimeout(3000); - conn.setReadTimeout(3000); - InputStream inputStream = conn.getInputStream(); - String newest = new BufferedReader(new InputStreamReader(inputStream)).readLine(); - String current = plugin.getDescription().getVersion(); - inputStream.close(); - isLatest = !compareVer(newest, current); - if (isLatest) { - AdventureUtils.consoleMessage(ConfigManager.lang.equalsIgnoreCase("chinese") ? "[CustomNameplates] 当前已是最新版本" : "[CustomNameplates] You are using the latest version."); - return; - } - if (ConfigManager.lang.equalsIgnoreCase("chinese")) { - AdventureUtils.consoleMessage("[CustomNameplates] 当前版本: " + current); - AdventureUtils.consoleMessage("[CustomNameplates] 最新版本: " + newest); - AdventureUtils.consoleMessage("[CustomNameplates] 请到 售后群https://polymart.org/resource/customnameplates.2543 获取最新版本."); - } - else { - AdventureUtils.consoleMessage("[CustomNameplates] Current version: " + current); - AdventureUtils.consoleMessage("[CustomNameplates] Latest version: " + newest); - AdventureUtils.consoleMessage("[CustomNameplates] Update is available: https://polymart.org/resource/customnameplates.2543"); - } - } catch (IOException e) { - isLatest = true; - Log.warn("Error occurred when checking update"); - } - }); + @Override + public String getPluginVersion() { + return pluginVersion; } - private int getPack_format(String version) { - switch (version) { + @Override + public boolean isLatest() { + return isLatest; + } + + @Override + public boolean isVersionNewerThan1_19() { + return isNewerThan1_19; + } + + @Override + public int getPackFormat() { + switch (serverVersion) { + case "v1_20_R3" -> { + return 22; + } case "v1_20_R2" -> { return 18; } @@ -153,6 +138,45 @@ public class VersionHelper { } } + @Override + public boolean isFolia() { + return isFolia; + } + + @Override + public String getServerVersion() { + return serverVersion; + } + + // Method to asynchronously check for plugin updates + public CompletableFuture checkUpdate() { + CompletableFuture updateFuture = new CompletableFuture<>(); + plugin.getScheduler().runTaskAsync(() -> { + try { + URL url = new URL("https://api.polymart.org/v1/getResourceInfoSimple/?resource_id=2543&key=version"); + URLConnection conn = url.openConnection(); + conn.setConnectTimeout(3000); + conn.setReadTimeout(3000); + InputStream inputStream = conn.getInputStream(); + String newest = new BufferedReader(new InputStreamReader(inputStream)).readLine(); + String current = getPluginVersion(); + inputStream.close(); + if (!compareVer(newest, current)) { + updateFuture.complete(false); + return; + } + isLatest = false; + updateFuture.complete(true); + } catch (Exception exception) { + LogUtils.warn("Error occurred when checking update."); + updateFuture.completeExceptionally(exception); + } + }); + return updateFuture; + } + + // Method to compare two version strings + // return true when update is available private boolean compareVer(String newV, String currentV) { if (newV == null || currentV == null || newV.isEmpty() || currentV.isEmpty()) { return false; @@ -173,9 +197,7 @@ public class VersionHelper { } else if (newPart.length > 1 && currentPart.length > 1) { String[] newHotfix = newPart[1].split("(?<=\\D)(?=\\d)|(?<=\\d)(?=\\D)"); String[] currentHotfix = currentPart[1].split("(?<=\\D)(?=\\d)|(?<=\\d)(?=\\D)"); - // hotfix2 & hotfix if (newHotfix.length == 2 && currentHotfix.length == 1) return true; - // hotfix3 & hotfix2 else if (newHotfix.length > 1 && currentHotfix.length > 1) { int newHotfixNum = Integer.parseInt(newHotfix[1]); int currentHotfixNum = Integer.parseInt(currentHotfix[1]); @@ -197,27 +219,6 @@ public class VersionHelper { return false; } } - // if common parts are the same, the longer is newer return newVS.length > currentVS.length; } - - public String getPluginVersion() { - return pluginVersion; - } - - public boolean isLatest() { - return isLatest || !ConfigManager.checkUpdate; - } - - public int getPack_format() { - return pack_format; - } - - public boolean isFolia() { - return isFolia; - } - - public boolean isGeyser() { - return hasGeyser; - } -} +} \ No newline at end of file diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/NameplateManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/NameplateManagerImpl.java new file mode 100644 index 0000000..e3b6804 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/NameplateManagerImpl.java @@ -0,0 +1,565 @@ +package net.momirealms.customnameplates.paper.mechanic.nameplate; + +import com.comphenix.protocol.ProtocolLibrary; +import me.clip.placeholderapi.PlaceholderAPI; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.data.OnlineUser; +import net.momirealms.customnameplates.api.event.NameplateDataLoadEvent; +import net.momirealms.customnameplates.api.manager.NameplateManager; +import net.momirealms.customnameplates.api.manager.TeamTagManager; +import net.momirealms.customnameplates.api.manager.UnlimitedTagManager; +import net.momirealms.customnameplates.api.mechanic.character.CharacterArranger; +import net.momirealms.customnameplates.api.mechanic.character.ConfiguredChar; +import net.momirealms.customnameplates.api.mechanic.nameplate.CachedNameplate; +import net.momirealms.customnameplates.api.mechanic.nameplate.Nameplate; +import net.momirealms.customnameplates.api.mechanic.nameplate.TagMode; +import net.momirealms.customnameplates.api.mechanic.tag.NameplatePlayer; +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.UnlimitedTagSetting; +import net.momirealms.customnameplates.api.mechanic.team.TeamColor; +import net.momirealms.customnameplates.api.scheduler.CancellableTask; +import net.momirealms.customnameplates.api.util.FontUtils; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.mechanic.nameplate.tag.listener.*; +import net.momirealms.customnameplates.paper.mechanic.nameplate.tag.team.TeamTagManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.nameplate.tag.unlimited.UnlimitedTagManagerImpl; +import net.momirealms.customnameplates.paper.setting.CNConfig; +import org.bukkit.Bukkit; +import org.bukkit.configuration.ConfigurationSection; +import org.bukkit.configuration.file.YamlConfiguration; +import org.bukkit.entity.Entity; +import org.bukkit.entity.Player; +import org.bukkit.event.EventHandler; +import org.bukkit.event.EventPriority; +import org.bukkit.event.HandlerList; +import org.bukkit.event.Listener; +import org.bukkit.event.entity.EntityPoseChangeEvent; +import org.bukkit.event.player.PlayerJoinEvent; +import org.bukkit.event.player.PlayerQuitEvent; +import org.bukkit.event.player.PlayerToggleSneakEvent; +import org.jetbrains.annotations.Nullable; + +import java.io.File; +import java.util.*; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.TimeUnit; + +public class NameplateManagerImpl implements NameplateManager, Listener { + + private final CustomNameplatesPlugin plugin; + /* A map for nameplate configs */ + private final HashMap nameplateMap; + /* A map for cached nameplates */ + private final ConcurrentHashMap cachedNameplateMap; + /* A map to quickly get Entity by its EntityID */ + private final ConcurrentHashMap entityID2EntityMap; + /* A map to store players that have tags */ + private final ConcurrentHashMap nameplatePlayerMap; + private CancellableTask nameplateRefreshTask; + private final TeamTagManagerImpl teamTagManager; + private final UnlimitedTagManagerImpl unlimitedTagManager; + private final EntityDestroyListener entityDestroyListener; + private final EntitySpawnListener entitySpawnListener; + private final EntityMoveListener entityMoveListener; + private final EntityLookListener entityLookListener; + private final EntityTeleportListener entityTeleportListener; + + /** + * Configs + */ + private long teamRefreshFrequency; + private String teamPrefix; + private String teamSuffix; + private boolean fixTab; + /* Is proxy server working */ + private boolean proxyMode; + /* TEAM & UNLIMITED */ + private TagMode tagMode; + private int previewDuration; + private String defaultNameplate; + private String playerName, prefix, suffix; + private long refreshFrequency; + private final List tagSettings; + + public NameplateManagerImpl(CustomNameplatesPlugin plugin) { + this.plugin = plugin; + this.nameplateMap = new HashMap<>(); + this.tagSettings = new ArrayList<>(); + + this.cachedNameplateMap = new ConcurrentHashMap<>(); + this.entityID2EntityMap = new ConcurrentHashMap<>(); + this.nameplatePlayerMap = new ConcurrentHashMap<>(); + this.teamTagManager = new TeamTagManagerImpl(this); + this.unlimitedTagManager = new UnlimitedTagManagerImpl(this); + + this.entityTeleportListener = new EntityTeleportListener(this); + this.entityDestroyListener = new EntityDestroyListener(this); + this.entitySpawnListener = new EntitySpawnListener(this); + this.entityLookListener = new EntityLookListener(this); + this.entityMoveListener = new EntityMoveListener(this); + } + + public void reload() { + unload(); + load(); + } + + public void unload() { + if (this.nameplateRefreshTask != null && !this.nameplateRefreshTask.isCancelled()) { + this.nameplateRefreshTask.cancel(); + } + this.nameplateMap.clear(); + this.tagSettings.clear(); + this.teamTagManager.unload(); + this.unlimitedTagManager.unload(); + + HandlerList.unregisterAll(this); + ProtocolLibrary.getProtocolManager().removePacketListener(entityDestroyListener); + ProtocolLibrary.getProtocolManager().removePacketListener(entitySpawnListener); + ProtocolLibrary.getProtocolManager().removePacketListener(entityLookListener); + ProtocolLibrary.getProtocolManager().removePacketListener(entityMoveListener); + ProtocolLibrary.getProtocolManager().removePacketListener(entityTeleportListener); + } + + public void load() { + if (!CNConfig.nameplateModule) return; + this.loadConfig(); + this.loadNameplates(); + + this.teamTagManager.load(teamRefreshFrequency, fixTab); + this.unlimitedTagManager.load(); + + this.nameplateRefreshTask = plugin.getScheduler().runTaskAsyncTimer(() -> { + for (OnlineUser user : plugin.getStorageManager().getOnlineUsers()) { + updateCachedNameplate(user.getPlayer(), user.getNameplate()); + } + }, refreshFrequency * 50, refreshFrequency * 50, TimeUnit.MILLISECONDS); + + for (Player online : Bukkit.getOnlinePlayers()) { + createNameTag(online); + } + + Bukkit.getPluginManager().registerEvents(this, plugin); + ProtocolLibrary.getProtocolManager().addPacketListener(entityDestroyListener); + ProtocolLibrary.getProtocolManager().addPacketListener(entitySpawnListener); + ProtocolLibrary.getProtocolManager().addPacketListener(entityLookListener); + ProtocolLibrary.getProtocolManager().addPacketListener(entityMoveListener); + ProtocolLibrary.getProtocolManager().addPacketListener(entityTeleportListener); + } + + private void loadConfig() { + YamlConfiguration config = plugin.getConfig("configs" + File.separator + "nameplate.yml"); + + tagMode = TagMode.valueOf(config.getString("mode", "TEAM").toUpperCase(Locale.ENGLISH)); + proxyMode = config.getBoolean("proxy", false); + previewDuration = config.getInt("preview-duration", 5); + defaultNameplate = config.getString("default-nameplate", "none"); + + playerName = config.getString("nameplate.player-name", "%player_name%"); + prefix = config.getString("nameplate.prefix", ""); + suffix = config.getString("nameplate.suffix", ""); + refreshFrequency = config.getInt("nameplate.refresh-frequency", 10); + + teamPrefix = config.getString("team.prefix", ""); + teamSuffix = config.getString("team.suffix", ""); + teamRefreshFrequency = config.getInt("team.refresh-frequency", 10); + fixTab = config.getBoolean("team.fix-Tab", true); + + ConfigurationSection unlimitedSection = config.getConfigurationSection("unlimited"); + if (unlimitedSection != null) { + for (Map.Entry entry : unlimitedSection.getValues(false).entrySet()) { + if (entry.getValue() instanceof ConfigurationSection innerSection) { + tagSettings.add( + UnlimitedTagSetting.builder() + .rawText(innerSection.getString("text", "")) + .refreshFrequency(innerSection.getInt("refresh-frequency", 20)) + .checkFrequency(innerSection.getInt("check-frequency", 20)) + .verticalOffset(innerSection.getDouble("vertical-offset", -1)) + .ownerRequirements(plugin.getRequirementManager().getRequirements(innerSection.getConfigurationSection("owner-conditions"))) + .viewerRequirements(plugin.getRequirementManager().getRequirements(innerSection.getConfigurationSection("viewer-conditions"))) + .build() + ); + } + } + } + } + + private void loadNameplates() { + File npFolder = new File(plugin.getDataFolder(), "contents" + File.separator + "nameplates"); + if (!npFolder.exists() && npFolder.mkdirs()) { + saveDefaultNameplates(); + } + File[] npConfigFiles = npFolder.listFiles(file -> file.getName().endsWith(".yml")); + if (npConfigFiles == null) return; + Arrays.sort(npConfigFiles, Comparator.comparing(File::getName)); + for (File npConfigFile : npConfigFiles) { + + String key = npConfigFile.getName().substring(0, npConfigFile.getName().length() - 4); + if (key.equals("none")) { + LogUtils.severe("You can't use 'none' as nameplate's key"); + continue; + } + + YamlConfiguration config = YamlConfiguration.loadConfiguration(npConfigFile); + if (!registerNameplate( + key, + Nameplate.builder() + .displayName(config.getString("display-name", key)) + .teamColor(TeamColor.valueOf(config.getString("name-color", "none").toUpperCase(Locale.ENGLISH))) + .namePrefix(config.getString("name-prefix", "")) + .nameSuffix(config.getString("name-suffix", "")) + .left(ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("left.image", key + "_left")) + .height(config.getInt("left.height", 16)) + .ascent(config.getInt("left.ascent", 12)) + .width(config.getInt("left.width", 16)) + .build()) + .right(ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("right.image", key + "_right")) + .height(config.getInt("right.height", 16)) + .ascent(config.getInt("right.ascent", 12)) + .width(config.getInt("right.width", 16)) + .build()) + .middle(ConfiguredChar.builder() + .character(CharacterArranger.getAndIncrease()) + .png(config.getString("middle.image", key + "_middle")) + .height(config.getInt("middle.height", 16)) + .ascent(config.getInt("middle.ascent", 12)) + .width(config.getInt("middle.width", 16)) + .build()) + .build()) + ) { + LogUtils.warn("Found duplicated nameplate: " + key); + } + } + } + + @EventHandler (ignoreCancelled = true, priority = EventPriority.LOW) + public void onDataLoaded(NameplateDataLoadEvent event) { + OnlineUser data = event.getOnlineUser(); + String nameplate = data.getNameplateKey(); + if (nameplate.equals("none")) { + nameplate = defaultNameplate; + } + + if (!nameplate.equals("none") && !containsNameplate(nameplate)) { + if (nameplate.equals(defaultNameplate)) { + LogUtils.severe("Default nameplate doesn't exist"); + return; + } + + LogUtils.severe("Nameplate " + nameplate + " doesn't exist. To prevent bugs, player " + event.getUUID() + " 's nameplate data is reset"); + data.setNameplate("none"); + plugin.getStorageManager().saveOnlinePlayerData(event.getUUID()); + return; + } + + Nameplate np = getNameplate(nameplate); + CachedNameplate cachedNameplate = new CachedNameplate(); + putCachedNameplateToMap(event.getUUID(), cachedNameplate); + updateCachedNameplate(cachedNameplate, data.getPlayer(), np); + } + + @EventHandler + public void onPlayerJoin(PlayerJoinEvent event) { + var player = event.getPlayer(); + this.putEntityIDToMap(player.getEntityId(), player); + if (!CNConfig.isOtherTeamPluginHooked()) + if (!proxyMode) plugin.getTeamManager().createTeam(player); + else plugin.getTeamManager().createProxyTeam(player); + this.createNameTag(player); + } + + @EventHandler + public void onPlayerQuit(PlayerQuitEvent event) { + var player = event.getPlayer(); + this.removeCachedNameplateFromMap(player.getUniqueId()); + this.removeEntityIDFromMap(player.getEntityId()); + + this.teamTagManager.handlePlayerQuit(player); + this.unlimitedTagManager.handlePlayerQuit(player); + + if (!proxyMode && !CNConfig.isOtherTeamPluginHooked()) { + plugin.getTeamManager().removeTeam(player); + } + } + + @EventHandler (ignoreCancelled = true) + public void onPlayerToggleSneak(PlayerToggleSneakEvent event) { + var player = event.getPlayer(); + unlimitedTagManager.handlePlayerSneak(player, event.isSneaking(), player.isFlying()); + } + + @EventHandler (ignoreCancelled = true) + public void onChangePose(EntityPoseChangeEvent event) { + if (event.getEntity() instanceof Player player) { + unlimitedTagManager.handlePlayerPose(player, event.getPose()); + } + } + + @Override + public boolean putEntityIDToMap(int entityID, Entity entity) { + if (this.entityID2EntityMap.containsKey(entityID)) + return false; + this.entityID2EntityMap.put(entityID, entity); + return true; + } + + @Override + public Entity removeEntityIDFromMap(int entityID) { + return this.entityID2EntityMap.remove(entityID); + } + + @Override + public boolean putCachedNameplateToMap(UUID uuid, CachedNameplate nameplate) { + if (this.cachedNameplateMap.containsKey(uuid)) { + return false; + } + this.cachedNameplateMap.put(uuid, nameplate); + return true; + } + + @Override + public CachedNameplate removeCachedNameplateFromMap(UUID uuid) { + return this.cachedNameplateMap.remove(uuid); + } + + @Override + public Player getPlayerByEntityID(int id) { + Entity entity = entityID2EntityMap.get(id); + if (entity instanceof Player player) { + return player; + } + return null; + } + + @Override + public Entity getEntityByEntityID(int id) { + return entityID2EntityMap.get(id); + } + + @Override + public boolean updateCachedNameplate(Player player) { + Optional onlineUser = plugin.getStorageManager().getOnlineUser(player.getUniqueId()); + if (onlineUser.isEmpty()) return false; + Nameplate nameplate = onlineUser.get().getNameplate(); + return updateCachedNameplate(player, nameplate); + } + + @Override + public boolean updateCachedNameplate(Player player, Nameplate nameplate) { + CachedNameplate cachedNameplate = cachedNameplateMap.get(player.getUniqueId()); + if (cachedNameplate == null) return false; + return updateCachedNameplate(cachedNameplate, player, nameplate); + } + + @Override + public CachedNameplate getCacheNameplate(Player player) { + return cachedNameplateMap.get(player.getUniqueId()); + } + + @Override + public void createNameTag(Player player) { + if (tagMode == TagMode.TEAM) { + putNameplatePlayerToMap(this.teamTagManager.createTagForPlayer(player, teamPrefix, teamSuffix)); + } else if (tagMode == TagMode.UNLIMITED) { + putNameplatePlayerToMap(this.unlimitedTagManager.createTagForPlayer(player, tagSettings)); + } + } + + @Override + public void putNameplatePlayerToMap(NameplatePlayer player) { + this.nameplatePlayerMap.put(player.getOwner().getUniqueId(), player); + } + + @Override + public NameplatePlayer removeNameplatePlayerFromMap(UUID uuid) { + return this.nameplatePlayerMap.remove(uuid); + } + + private boolean updateCachedNameplate(CachedNameplate cachedNameplate, Player player, Nameplate nameplate) { + String parsePrefix = PlaceholderAPI.setPlaceholders(player, prefix); + String parseName = PlaceholderAPI.setPlaceholders(player, playerName); + String parseSuffix = PlaceholderAPI.setPlaceholders(player, suffix); + if (nameplate != null) { + int width= FontUtils.getTextWidth( + parsePrefix + + nameplate.getNamePrefix() + + parseName + + nameplate.getNameSuffix() + + parseSuffix + ); + + cachedNameplate.setTeamColor(nameplate.getTeamColor()); + cachedNameplate.setNamePrefix(nameplate.getNamePrefix()); + cachedNameplate.setNameSuffix(nameplate.getNameSuffix()); + cachedNameplate.setTagSuffix(parseSuffix + nameplate.getSuffixWithFont(width)); + cachedNameplate.setTagPrefix(nameplate.getPrefixWithFont(width) + parsePrefix); + } else { + cachedNameplate.setTeamColor(TeamColor.NONE); + cachedNameplate.setNamePrefix(""); + cachedNameplate.setNameSuffix(""); + cachedNameplate.setTagPrefix(parsePrefix); + cachedNameplate.setTagSuffix(parseSuffix); + } + cachedNameplate.setPlayerName(parseName); + return true; + } + + @Override + public String getNameplatePrefix(Player player) { + CachedNameplate cachedNameplate = cachedNameplateMap.get(player.getUniqueId()); + if (cachedNameplate == null) return ""; + return cachedNameplate.getTagPrefix(); + } + + @Override + public String getNameplateSuffix(Player player) { + CachedNameplate cachedNameplate = cachedNameplateMap.get(player.getUniqueId()); + if (cachedNameplate == null) return ""; + return cachedNameplate.getTagSuffix(); + } + + @Override + public String getFullNameTag(Player player) { + CachedNameplate cachedNameplate = cachedNameplateMap.get(player.getUniqueId()); + if (cachedNameplate == null) { + return player.getName(); + } + + return cachedNameplate.getTagPrefix() + + cachedNameplate.getNamePrefix() + + cachedNameplate.getPlayerName() + + cachedNameplate.getNameSuffix() + + cachedNameplate.getTagSuffix(); + } + + @Override + public boolean registerNameplate(String key, Nameplate nameplate) { + if (this.nameplateMap.containsKey(key)) return false; + this.nameplateMap.put(key, nameplate); + return true; + } + + @Override + public List getAvailableNameplates(Player player) { + List nameplates = new ArrayList<>(); + for (String nameplate : nameplateMap.keySet()) { + if (hasNameplate(player, nameplate)) { + nameplates.add(nameplate); + } + } + return nameplates; + } + + @Override + public boolean equipNameplate(Player player, String nameplateKey) { + Nameplate nameplate = getNameplate(nameplateKey); + if (nameplate == null) { + return false; + } + plugin.getStorageManager().getOnlineUser(player.getUniqueId()).ifPresentOrElse(it -> { + it.setNameplate(nameplateKey); + plugin.getStorageManager().saveOnlinePlayerData(player.getUniqueId()); + }, () -> { + LogUtils.severe("Player " + player.getName() + "'s data is not loaded."); + }); + return true; + } + + @Override + public void unEquipNameplate(Player player) { + plugin.getStorageManager().getOnlineUser(player.getUniqueId()).ifPresentOrElse(it -> { + it.setNameplate("none"); + plugin.getStorageManager().saveOnlinePlayerData(player.getUniqueId()); + }, () -> { + LogUtils.severe("Player " + player.getName() + "'s data is not loaded."); + }); + } + + private void saveDefaultNameplates() { + String[] png_list = new String[]{"cat", "egg", "cheems", "wither", "xmas", "halloween", "hutao", "starsky", "trident", "rabbit"}; + String[] part_list = new String[]{"_left.png", "_middle.png", "_right.png", ".yml"}; + for (String name : png_list) { + for (String part : part_list) { + plugin.saveResource("contents" + File.separator + "nameplates" + File.separator + name + part, false); + } + } + } + + @Override + public boolean unregisterNameplate(String key) { + return this.nameplateMap.remove(key) != null; + } + + @Override + public boolean isProxyMode() { + return proxyMode; + } + + @Override + public TagMode getTagMode() { + return tagMode; + } + + @Override + @Nullable + public Nameplate getNameplate(String key) { + return nameplateMap.get(key); + } + + @Override + public Collection getNameplates() { + return nameplateMap.values(); + } + + @Override + public boolean containsNameplate(String key) { + return nameplateMap.containsKey(key); + } + + @Override + public boolean hasNameplate(Player player, String nameplate) { + return player.hasPermission("nameplates.equip." + nameplate); + } + + @Override + public TeamColor getTeamColor(Player player) { + CachedNameplate nameplate = getCacheNameplate(player); + return nameplate == null ? TeamColor.WHITE : nameplate.getTeamColor(); + } + + @Override + public String getDefaultNameplate() { + return defaultNameplate; + } + + @Override + public TeamTagManager getTeamTagManager() { + return teamTagManager; + } + + @Override + public UnlimitedTagManager getUnlimitedTagManager() { + return unlimitedTagManager; + } + + public void onEntityMove(Player receiver, int entityID, short x, short y, short z, boolean onGround) { + unlimitedTagManager.handleEntityMovePacket(receiver, entityID, x, y, z, onGround); + } + + public void onEntityDestroy(Player receiver, List list) { + teamTagManager.handleEntityDestroyPacket(receiver, list); + unlimitedTagManager.handleEntityDestroyPacket(receiver, list); + } + + public void onEntitySpawn(Player receiver, int entityID) { + teamTagManager.handleEntitySpawnPacket(receiver, entityID); + unlimitedTagManager.handleEntitySpawnPacket(receiver, entityID); + } + + public void onEntityTeleport(Player receiver, int entityID, double x, double y, double z, boolean onGround) { + unlimitedTagManager.handleEntityTeleportPacket(receiver, entityID, x, y, z, onGround); + } +} diff --git a/src/main/java/net/momirealms/customnameplates/listener/packet/EntityDestroyListener.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityDestroyListener.java similarity index 67% rename from src/main/java/net/momirealms/customnameplates/listener/packet/EntityDestroyListener.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityDestroyListener.java index 9936175..13f1bb5 100644 --- a/src/main/java/net/momirealms/customnameplates/listener/packet/EntityDestroyListener.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityDestroyListener.java @@ -15,27 +15,27 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.listener.packet; +package net.momirealms.customnameplates.paper.mechanic.nameplate.tag.listener; import com.comphenix.protocol.PacketType; import com.comphenix.protocol.events.ListenerPriority; import com.comphenix.protocol.events.PacketAdapter; import com.comphenix.protocol.events.PacketContainer; import com.comphenix.protocol.events.PacketEvent; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.carrier.AbstractPacketsHandler; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.mechanic.nameplate.NameplateManagerImpl; public class EntityDestroyListener extends PacketAdapter { - private final AbstractPacketsHandler handler; + private final NameplateManagerImpl manager; - public EntityDestroyListener(AbstractPacketsHandler handler) { - super(CustomNameplates.getInstance(), ListenerPriority.HIGHEST, PacketType.Play.Server.ENTITY_DESTROY); - this.handler = handler; + public EntityDestroyListener(NameplateManagerImpl manager) { + super(CustomNameplatesPlugin.getInstance(), ListenerPriority.HIGH, PacketType.Play.Server.ENTITY_DESTROY); + this.manager = manager; } public void onPacketSending(PacketEvent event) { PacketContainer packet = event.getPacket(); - handler.onEntityDestroy(event.getPlayer(), packet.getIntLists().read(0)); + manager.onEntityDestroy(event.getPlayer(), packet.getIntLists().read(0)); } } diff --git a/src/main/java/net/momirealms/customnameplates/listener/packet/EntityLookListener.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityLookListener.java similarity index 71% rename from src/main/java/net/momirealms/customnameplates/listener/packet/EntityLookListener.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityLookListener.java index e353ce0..c51f2f9 100644 --- a/src/main/java/net/momirealms/customnameplates/listener/packet/EntityLookListener.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityLookListener.java @@ -15,28 +15,28 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.listener.packet; +package net.momirealms.customnameplates.paper.mechanic.nameplate.tag.listener; import com.comphenix.protocol.PacketType; import com.comphenix.protocol.events.ListenerPriority; import com.comphenix.protocol.events.PacketAdapter; import com.comphenix.protocol.events.PacketContainer; import com.comphenix.protocol.events.PacketEvent; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.carrier.AbstractPacketsHandler; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.mechanic.nameplate.NameplateManagerImpl; public class EntityLookListener extends PacketAdapter { - private final AbstractPacketsHandler handler; + private final NameplateManagerImpl manager; - public EntityLookListener(AbstractPacketsHandler handler) { - super(CustomNameplates.getInstance(), ListenerPriority.HIGHEST, PacketType.Play.Server.REL_ENTITY_MOVE_LOOK); - this.handler = handler; + public EntityLookListener(NameplateManagerImpl manager) { + super(CustomNameplatesPlugin.getInstance(), ListenerPriority.HIGHEST, PacketType.Play.Server.REL_ENTITY_MOVE_LOOK); + this.manager = manager; } public void onPacketSending(PacketEvent event) { PacketContainer packet = event.getPacket(); - handler.onEntityMove(event.getPlayer(), + manager.onEntityMove(event.getPlayer(), packet.getIntegers().read(0), packet.getShorts().read(0), packet.getShorts().read(1), diff --git a/src/main/java/net/momirealms/customnameplates/listener/packet/EntityMoveListener.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityMoveListener.java similarity index 71% rename from src/main/java/net/momirealms/customnameplates/listener/packet/EntityMoveListener.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityMoveListener.java index e1801fa..86af826 100644 --- a/src/main/java/net/momirealms/customnameplates/listener/packet/EntityMoveListener.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityMoveListener.java @@ -15,28 +15,28 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.listener.packet; +package net.momirealms.customnameplates.paper.mechanic.nameplate.tag.listener; import com.comphenix.protocol.PacketType; import com.comphenix.protocol.events.ListenerPriority; import com.comphenix.protocol.events.PacketAdapter; import com.comphenix.protocol.events.PacketContainer; import com.comphenix.protocol.events.PacketEvent; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.carrier.AbstractPacketsHandler; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.mechanic.nameplate.NameplateManagerImpl; public class EntityMoveListener extends PacketAdapter { - private final AbstractPacketsHandler handler; + private final NameplateManagerImpl manager; - public EntityMoveListener(AbstractPacketsHandler handler) { - super(CustomNameplates.getInstance(), ListenerPriority.NORMAL, PacketType.Play.Server.REL_ENTITY_MOVE); - this.handler = handler; + public EntityMoveListener(NameplateManagerImpl manager) { + super(CustomNameplatesPlugin.getInstance(), ListenerPriority.HIGHEST, PacketType.Play.Server.REL_ENTITY_MOVE); + this.manager = manager; } public void onPacketSending(PacketEvent event) { PacketContainer packet = event.getPacket(); - handler.onEntityMove(event.getPlayer(), + manager.onEntityMove(event.getPlayer(), packet.getIntegers().read(0), packet.getShorts().read(0), packet.getShorts().read(1), diff --git a/src/main/java/net/momirealms/customnameplates/listener/packet/EntitySpawnListener.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntitySpawnListener.java similarity index 59% rename from src/main/java/net/momirealms/customnameplates/listener/packet/EntitySpawnListener.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntitySpawnListener.java index 890da94..8c1323e 100644 --- a/src/main/java/net/momirealms/customnameplates/listener/packet/EntitySpawnListener.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntitySpawnListener.java @@ -15,25 +15,25 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.listener.packet; +package net.momirealms.customnameplates.paper.mechanic.nameplate.tag.listener; import com.comphenix.protocol.PacketType; import com.comphenix.protocol.events.ListenerPriority; import com.comphenix.protocol.events.PacketAdapter; import com.comphenix.protocol.events.PacketEvent; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.carrier.AbstractPacketsHandler; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.mechanic.nameplate.NameplateManagerImpl; public class EntitySpawnListener extends PacketAdapter { - private final AbstractPacketsHandler handler; + private final NameplateManagerImpl manager; - public EntitySpawnListener(AbstractPacketsHandler handler) { - super(CustomNameplates.getInstance(), ListenerPriority.HIGHEST, CustomNameplates.getInstance().getVersionHelper().isVersionNewerThan1_20_R2() ? PacketType.Play.Server.SPAWN_ENTITY : PacketType.Play.Server.NAMED_ENTITY_SPAWN); - this.handler = handler; + public EntitySpawnListener(NameplateManagerImpl manager) { + super(CustomNameplatesPlugin.getInstance(), ListenerPriority.HIGH, CustomNameplatesPlugin.getInstance().getVersionManager().isVersionNewerThan1_20_R2() ? PacketType.Play.Server.SPAWN_ENTITY : PacketType.Play.Server.NAMED_ENTITY_SPAWN); + this.manager = manager; } public synchronized void onPacketSending(PacketEvent event) { - handler.onEntitySpawn(event.getPlayer(), event.getPacket().getIntegers().read(0), event); + manager.onEntitySpawn(event.getPlayer(), event.getPacket().getIntegers().read(0)); } } diff --git a/src/main/java/net/momirealms/customnameplates/listener/packet/EntityTeleportListener.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityTeleportListener.java similarity index 58% rename from src/main/java/net/momirealms/customnameplates/listener/packet/EntityTeleportListener.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityTeleportListener.java index cdc0473..59d035d 100644 --- a/src/main/java/net/momirealms/customnameplates/listener/packet/EntityTeleportListener.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/EntityTeleportListener.java @@ -15,27 +15,34 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.listener.packet; +package net.momirealms.customnameplates.paper.mechanic.nameplate.tag.listener; import com.comphenix.protocol.PacketType; import com.comphenix.protocol.events.ListenerPriority; import com.comphenix.protocol.events.PacketAdapter; import com.comphenix.protocol.events.PacketContainer; import com.comphenix.protocol.events.PacketEvent; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.carrier.AbstractPacketsHandler; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.mechanic.nameplate.NameplateManagerImpl; public class EntityTeleportListener extends PacketAdapter { - private final AbstractPacketsHandler handler; + private final NameplateManagerImpl manager; - public EntityTeleportListener(AbstractPacketsHandler handler) { - super(CustomNameplates.getInstance(), ListenerPriority.HIGHEST, PacketType.Play.Server.ENTITY_TELEPORT); - this.handler = handler; + public EntityTeleportListener(NameplateManagerImpl manager) { + super(CustomNameplatesPlugin.getInstance(), ListenerPriority.HIGHEST, PacketType.Play.Server.ENTITY_TELEPORT); + this.manager = manager; } public void onPacketSending(PacketEvent event) { PacketContainer packet = event.getPacket(); - handler.onEntityTeleport(event.getPlayer(), packet.getIntegers().read(0)); + manager.onEntityTeleport( + event.getPlayer(), + packet.getIntegers().read(0), + packet.getDoubles().read(0), + packet.getDoubles().read(1), + packet.getDoubles().read(2), + packet.getBooleans().read(0) + ); } } diff --git a/src/main/java/net/momirealms/customnameplates/listener/compatibility/MagicCosmeticsListener.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/MagicCosmeticsListener.java similarity index 54% rename from src/main/java/net/momirealms/customnameplates/listener/compatibility/MagicCosmeticsListener.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/MagicCosmeticsListener.java index 8e5f10c..52466ec 100644 --- a/src/main/java/net/momirealms/customnameplates/listener/compatibility/MagicCosmeticsListener.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/listener/MagicCosmeticsListener.java @@ -1,24 +1,22 @@ -package net.momirealms.customnameplates.listener.compatibility; +package net.momirealms.customnameplates.paper.mechanic.nameplate.tag.listener; import com.francobm.magicosmetics.api.Cosmetic; import com.francobm.magicosmetics.api.CosmeticType; -import com.francobm.magicosmetics.api.MagicAPI; import com.francobm.magicosmetics.cache.PlayerData; import com.francobm.magicosmetics.cache.cosmetics.Hat; import com.francobm.magicosmetics.events.*; -import net.momirealms.customnameplates.object.carrier.NamedEntityCarrier; -import net.momirealms.customnameplates.object.carrier.NamedEntityManager; +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.UnlimitedPlayer; +import net.momirealms.customnameplates.paper.mechanic.nameplate.tag.unlimited.UnlimitedTagManagerImpl; import org.bukkit.entity.Player; import org.bukkit.event.EventHandler; import org.bukkit.event.Listener; -import org.bukkit.event.player.PlayerChangedWorldEvent; public class MagicCosmeticsListener implements Listener { - private final NamedEntityCarrier namedEntityCarrier; + private final UnlimitedTagManagerImpl unlimitedTagManager; - public MagicCosmeticsListener(NamedEntityCarrier namedEntityCarrier) { - this.namedEntityCarrier = namedEntityCarrier; + public MagicCosmeticsListener(UnlimitedTagManagerImpl unlimitedTagManager) { + this.unlimitedTagManager = unlimitedTagManager; } @EventHandler @@ -26,9 +24,8 @@ public class MagicCosmeticsListener implements Listener { final Cosmetic cosmetic = event.getNewCosmetic(); final Player player = event.getPlayer(); if (cosmetic instanceof Hat hat) { - NamedEntityManager nem = namedEntityCarrier.getNamedEntityManager(player); - if (nem != null) { - nem.setHatOffset(hat.isHideCosmetic() ? 0 : hat.getOffSetY()); + if (unlimitedTagManager.getUnlimitedObject(player.getUniqueId()) instanceof UnlimitedPlayer unlimitedPlayer) { + unlimitedPlayer.setHatOffset(hat.isHideCosmetic() ? 0 : hat.getOffSetY()); } } } @@ -36,16 +33,15 @@ public class MagicCosmeticsListener implements Listener { @EventHandler public void onEnterBlackListWorld(PlayerChangeBlacklistEvent event) { var player = event.getPlayer(); - NamedEntityManager nem = namedEntityCarrier.getNamedEntityManager(player); - if (nem != null) { + if (unlimitedTagManager.getUnlimitedObject(player.getUniqueId()) instanceof UnlimitedPlayer unlimitedPlayer) { if (event.isInWorldBlacklist()) { - nem.setHatOffset(0); + unlimitedPlayer.setHatOffset(0); } else { PlayerData playerData = PlayerData.getPlayer(player); if (playerData != null) { final Cosmetic cosmetic = playerData.getHat(); if (cosmetic instanceof Hat hat) { - nem.setHatOffset(hat.isHideCosmetic() ? 0 : hat.getOffSetY()); + unlimitedPlayer.setHatOffset(hat.isHideCosmetic() ? 0 : hat.getOffSetY()); } } } @@ -57,9 +53,8 @@ public class MagicCosmeticsListener implements Listener { final Cosmetic cosmetic = event.getCosmetic(); final Player player = event.getPlayer(); if (cosmetic instanceof Hat hat) { - NamedEntityManager nem = namedEntityCarrier.getNamedEntityManager(player); - if (nem != null) { - nem.setHatOffset(hat.isHideCosmetic() ? 0 : hat.getOffSetY()); + if (unlimitedTagManager.getUnlimitedObject(player.getUniqueId()) instanceof UnlimitedPlayer unlimitedPlayer) { + unlimitedPlayer.setHatOffset(hat.isHideCosmetic() ? 0 : hat.getOffSetY()); } } } @@ -68,9 +63,8 @@ public class MagicCosmeticsListener implements Listener { public void onUnEquip(CosmeticUnEquipEvent event) { final Player player = event.getPlayer(); if (event.getCosmeticType() == CosmeticType.HAT) { - NamedEntityManager nem = namedEntityCarrier.getNamedEntityManager(player); - if (nem != null) { - nem.setHatOffset(0); + if (unlimitedTagManager.getUnlimitedObject(player.getUniqueId()) instanceof UnlimitedPlayer unlimitedPlayer) { + unlimitedPlayer.setHatOffset(0); } } } @@ -79,9 +73,8 @@ public class MagicCosmeticsListener implements Listener { public void onDataLoaded(PlayerDataLoadEvent event) { for (Cosmetic cosmetic : event.getEquippedCosmetics()) { if (cosmetic instanceof Hat hat) { - NamedEntityManager nem = namedEntityCarrier.getNamedEntityManager(event.getPlayerData().getOfflinePlayer().getPlayer()); - if (nem != null) { - nem.setHatOffset(hat.isHideCosmetic() ? 0 : hat.getOffSetY()); + if (unlimitedTagManager.getUnlimitedObject(event.getPlayerData().getUniqueId()) instanceof UnlimitedPlayer unlimitedPlayer) { + unlimitedPlayer.setHatOffset(hat.isHideCosmetic() ? 0 : hat.getOffSetY()); } } } diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/team/TeamTagManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/team/TeamTagManagerImpl.java new file mode 100644 index 0000000..dd9495f --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/team/TeamTagManagerImpl.java @@ -0,0 +1,128 @@ +package net.momirealms.customnameplates.paper.mechanic.nameplate.tag.team; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.NameplateManager; +import net.momirealms.customnameplates.api.manager.TeamTagManager; +import net.momirealms.customnameplates.api.mechanic.tag.team.TeamPlayer; +import net.momirealms.customnameplates.api.scheduler.CancellableTask; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.util.LocationUtils; +import org.bukkit.Bukkit; +import org.bukkit.entity.Entity; +import org.bukkit.entity.Player; +import org.jetbrains.annotations.Nullable; + +import java.util.List; +import java.util.UUID; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.TimeUnit; + +public class TeamTagManagerImpl implements TeamTagManager { + + private final NameplateManager manager; + private final ConcurrentHashMap teamPlayerMap; + private CancellableTask refreshTask; + + public TeamTagManagerImpl(NameplateManager manager) { + this.manager = manager; + this.teamPlayerMap = new ConcurrentHashMap<>(); + } + + public void load(long refreshFrequency, boolean fixTab) { + this.refreshTask = CustomNameplatesPlugin.get().getScheduler().runTaskAsyncTimer( + () -> { + try { + for (TeamPlayer teamPlayer : teamPlayerMap.values()) { + teamPlayer.updateForNearbyPlayers(false); + } + } catch (Exception e) { + LogUtils.severe( + "Error occurred when updating team tags. " + + "This might not be a bug in CustomNameplates. Please report " + + "to the Plugin on the top of the following " + + "stack trace." + ); + e.printStackTrace(); + } + }, + refreshFrequency * 50L, + refreshFrequency * 50L, + TimeUnit.MILLISECONDS + ); + if (fixTab) { + //TODO + } + } + + public void unload() { + if (this.refreshTask != null && !this.refreshTask.isCancelled()) { + this.refreshTask.cancel(); + } + for (TeamPlayer entry : teamPlayerMap.values()) { + entry.destroy(); + } + } + + @Override + @SuppressWarnings("DuplicatedCode") + public TeamPlayer createTagForPlayer(Player player, String prefix, String suffix) { + if (this.teamPlayerMap.containsKey(player.getUniqueId())) { + return null; + } + + var teamPlayer = new TeamPlayer(this, player, prefix, suffix); + this.teamPlayerMap.put( + player.getUniqueId(), + teamPlayer + ); + for (Player online : Bukkit.getOnlinePlayers()) { + if ( online == player + || !online.canSee(player) + || LocationUtils.getDistance(online, player) > 48 + || online.getWorld() != player.getWorld() + || online.isDead() + ) continue; + teamPlayer.addNearbyPlayer(online); + } + return teamPlayer; + } + + @Override + public TeamPlayer removeTeamPlayerFromMap(UUID uuid) { + return teamPlayerMap.remove(uuid); + } + + @Nullable + public TeamPlayer getTeamPlayer(UUID uuid) { + return teamPlayerMap.get(uuid); + } + + public void handleEntitySpawnPacket(Player receiver, int entityId) { + Entity spawned = manager.getEntityByEntityID(entityId); + if (spawned == null) return; + TeamPlayer teamPlayer = getTeamPlayer(spawned.getUniqueId()); + if (teamPlayer == null) return; + teamPlayer.addNearbyPlayer(receiver); + } + + public void handleEntityDestroyPacket(Player receiver, List list) { + for (int id : list) { + handleSingleEntityDestroy(receiver, id); + } + } + + public void handlePlayerQuit(Player quit) { + teamPlayerMap.remove(quit.getUniqueId()); + for (TeamPlayer teamPlayer : teamPlayerMap.values()) { + teamPlayer.removeNearbyPlayer(quit); + } + } + + private void handleSingleEntityDestroy(Player receiver, int entityID) { + Entity deSpawned = manager.getEntityByEntityID(entityID); + if (deSpawned == null) return; + TeamPlayer teamPlayer = getTeamPlayer(deSpawned.getUniqueId()); + if (teamPlayer == null) return; + teamPlayer.removeNearbyPlayer(receiver); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/unlimited/NamedEntityImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/unlimited/NamedEntityImpl.java new file mode 100644 index 0000000..0e94eb2 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/unlimited/NamedEntityImpl.java @@ -0,0 +1,393 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.mechanic.nameplate.tag.unlimited; + +import com.comphenix.protocol.PacketType; +import com.comphenix.protocol.events.PacketContainer; +import com.comphenix.protocol.wrappers.WrappedChatComponent; +import com.comphenix.protocol.wrappers.WrappedDataValue; +import com.comphenix.protocol.wrappers.WrappedDataWatcher; +import com.google.common.collect.Lists; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.RequirementManager; +import net.momirealms.customnameplates.api.mechanic.misc.ViewerText; +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.NamedEntity; +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.UnlimitedPlayer; +import net.momirealms.customnameplates.api.requirement.Condition; +import net.momirealms.customnameplates.api.requirement.Requirement; +import net.momirealms.customnameplates.paper.adventure.AdventureManagerImpl; +import net.momirealms.customnameplates.paper.mechanic.misc.PacketManager; +import org.bukkit.Location; +import org.bukkit.entity.EntityType; +import org.bukkit.entity.Player; +import org.bukkit.entity.Pose; + +import java.util.*; +import java.util.concurrent.TimeUnit; + +public class NamedEntityImpl implements NamedEntity { + + private final UUID uuid = UUID.randomUUID(); + private final UnlimitedPlayer owner; + private double yOffset; + private final int entityId; + private boolean sneaking; + private final ViewerText viewerText; + private final Requirement[] ownerRequirements; + private final Requirement[] viewerRequirements; + private final Vector viewers; + private final int refreshFrequency; + private final int checkFrequency; + private int checkTimer; + private int refreshTimer; + private boolean ownerCanShow; + + public NamedEntityImpl( + UnlimitedPlayer unlimitedPlayer, + ViewerText text, + int refreshFrequency, + int checkFrequency, + double yOffset, + Requirement[] ownerRequirements, + Requirement[] viewerRequirements + ) { + this.entityId = new Random().nextInt(Integer.MAX_VALUE); + this.owner = unlimitedPlayer; + this.yOffset = yOffset; + this.viewerText = text; + this.sneaking = unlimitedPlayer.getOwner().isSneaking(); + this.ownerRequirements = ownerRequirements; + this.viewerRequirements = viewerRequirements; + this.checkFrequency = checkFrequency; + this.refreshFrequency = refreshFrequency; + this.viewers = new Vector<>(); + this.ownerCanShow = RequirementManager.isRequirementMet(new Condition(owner.getOwner()), ownerRequirements); + } + + @Override + public boolean equals(Object o) { + if (this == o) return true; + if (o == null || getClass() != o.getClass()) return false; + NamedEntityImpl that = (NamedEntityImpl) o; + return uuid.equals(that.uuid); + } + + @Override + public int hashCode() { + return uuid.hashCode(); + } + + @Override + public boolean canShow() { + return ownerCanShow; + } + + @Override + public boolean isShownTo(Player viewer) { + return viewers.contains(viewer); + } + + @Override + public boolean canSee(Player viewer) { + Condition condition = new Condition(viewer); + return RequirementManager.isRequirementMet(condition, viewerRequirements); + } + + @Override + public void timer() { + checkTimer++; + if (checkTimer >= checkFrequency) { + checkTimer = 0; + if (!RequirementManager.isRequirementMet(new Condition(owner.getOwner()), ownerRequirements)) { + ownerCanShow = false; + for (Player all : owner.getNearbyPlayers()) { + removePlayerFromViewers(all); + } + } else { + ownerCanShow = true; + for (Player all : owner.getNearbyPlayers()) { + if (canSee(all)) { + addPlayerToViewers(all); + } else { + removePlayerFromViewers(all); + } + } + } + } + + refreshTimer++; + if (refreshTimer >= refreshFrequency) { + refreshTimer = 0; + viewerText.updateForOwner(); + updateText(); + } + } + + @Override + public void removePlayerFromViewers(Player player) { + if (!viewers.contains(player)) { + return; + } + this.viewers.remove(player); + destroy(player); + + } + + @Override + public void addPlayerToViewers(Player player) { + if (viewers.contains(player)) { + return; + } + this.viewers.add(player); + spawn(player, owner.getOwner().getPose()); + } + + @Override + public double getOffset() { + return yOffset; + } + + @Override + public void setOffset(double offset) { + if (yOffset == offset) return; + yOffset = offset; + for (Player all : viewers) { + PacketManager.getInstance().send(all, getTeleportPacket(0)); + } + } + + @Override + public ViewerText getViewerText() { + return viewerText; + } + + @Override + public void spawn(Player viewer, Pose pose) { + if (viewerText.updateForViewer(viewer)) { + for (PacketContainer packet : getSpawnPackets(viewerText.getLatestValue(viewer), pose)) { + PacketManager.getInstance().send(viewer, packet); + } + } + } + + @Override + public void spawn(Pose pose) { + for (Player all : viewers) { + spawn(all, pose); + } + } + + @Override + public void destroy() { + PacketContainer destroyPacket = new PacketContainer(PacketType.Play.Server.ENTITY_DESTROY); + destroyPacket.getIntLists().write(0, List.of(entityId)); + for (Player all : viewers) { + PacketManager.getInstance().send(all, destroyPacket); + viewerText.removeViewer(all); + } + } + + @Override + public void destroy(Player viewer) { + PacketContainer destroyPacket = new PacketContainer(PacketType.Play.Server.ENTITY_DESTROY); + destroyPacket.getIntLists().write(0, List.of(entityId)); + PacketManager.getInstance().send(viewer, destroyPacket); + viewerText.removeViewer(viewer); + } + + @Override + public void teleport(double x, double y, double z, boolean onGround) { + PacketContainer packet = getTeleportPacket(x, y, z, onGround); + for (Player all : viewers) { + PacketManager.getInstance().send(all, packet); + } + } + + @Override + public void teleport(Player viewer, double x, double y, double z, boolean onGround) { + if (viewers.contains(viewer)) { + PacketManager.getInstance().send(viewer, getTeleportPacket(x, y, z, onGround)); + } + } + + @Override + public void setSneak(boolean isSneaking, boolean onGround) { + this.sneaking = isSneaking; + if (!onGround) { + for (Player viewer : viewers) { + PacketManager.getInstance().send(viewer, getMetaPacket(viewerText.getLatestValue(viewer))); + } + } + } + + @Override + public int getEntityId() { + return entityId; + } + + @Override + public void move(short x, short y, short z, boolean onGround) { + PacketContainer packet = getMovePacket(x, y, z, onGround); + for (Player viewer : viewers) { + PacketManager.getInstance().send(viewer, packet); + } + } + + @Override + public void move(Player viewer, short x, short y, short z, boolean onGround) { + if (viewers.contains(viewer)) { + PacketContainer packet = getMovePacket(x, y, z, onGround); + PacketManager.getInstance().send(viewer, packet); + } + } + + @Override + public void respawn(Player viewer, Pose pose) { + destroy(viewer); + spawn(viewer, pose); + } + + @Override + public void respawn(Pose pose) { + for (Player viewer : viewers) { + respawn(viewer, pose); + } + } + + @Override + public void updateText() { + for (Player viewer : viewers) { + updateText(viewer); + } + } + + @Override + public void updateText(Player viewer) { + if (viewerText.updateForViewer(viewer)) { + PacketManager.getInstance().send(viewer, getMetaPacket(viewerText.getLatestValue(viewer))); + } + } + + @Override + public UUID getUuid() { + return uuid; + } + + @Override + public void handlePose(Pose previous, Pose pose) { + // Add delay to prevent the tag from appearing earlier + CustomNameplatesPlugin.get().getScheduler().runTaskAsyncLater(() -> { + for (Player viewer : viewers) { + respawn(viewer, pose); + } + }, 20, TimeUnit.MILLISECONDS); + } + + public PacketContainer getMovePacket(short x, short y, short z, boolean onGround) { + PacketContainer packet = new PacketContainer(PacketType.Play.Server.REL_ENTITY_MOVE); + packet.getIntegers().write(0, entityId); + packet.getShorts().write(0, x); + packet.getShorts().write(1, y); + packet.getShorts().write(2, z); + packet.getBooleans().write(0, onGround); + return packet; + } + + public PacketContainer getTeleportPacket(double x, double y, double z, boolean onGround) { + PacketContainer packet = new PacketContainer(PacketType.Play.Server.ENTITY_TELEPORT); + packet.getIntegers().write(0, entityId); + packet.getDoubles().write(0, x); + packet.getDoubles().write(1, y + owner.getHatOffset() + getCorrection(owner.getOwner().getPose()) + yOffset); + packet.getDoubles().write(2, z); + packet.getBooleans().write(0, onGround); + return packet; + } + + public PacketContainer getTeleportPacket(double correction) { + PacketContainer packet = new PacketContainer(PacketType.Play.Server.ENTITY_TELEPORT); + packet.getIntegers().write(0, entityId); + Location location = getEntityLocation(correction); + packet.getDoubles().write(0, location.getX()); + packet.getDoubles().write(1, location.getY()); + packet.getDoubles().write(2, location.getZ()); + return packet; + } + + protected PacketContainer getMetaPacket(String text) { + PacketContainer metaPacket = new PacketContainer(PacketType.Play.Server.ENTITY_METADATA); + metaPacket.getIntegers().write(0, entityId); + String json = AdventureManagerImpl.getInstance().componentToJson(AdventureManagerImpl.getInstance().getComponentFromMiniMessage(text)); + if (CustomNameplatesPlugin.getInstance().getVersionManager().isVersionNewerThan1_19_R2()) { + WrappedDataWatcher wrappedDataWatcher = createArmorStandDataWatcher(json); + List wrappedDataValueList = Lists.newArrayList(); + wrappedDataWatcher.getWatchableObjects().stream().filter(Objects::nonNull).forEach(entry -> wrappedDataValueList.add(new WrappedDataValue(entry.getWatcherObject().getIndex(), entry.getWatcherObject().getSerializer(), entry.getRawValue()))); + metaPacket.getDataValueCollectionModifier().write(0, wrappedDataValueList); + } else { + metaPacket.getWatchableCollectionModifier().write(0, createArmorStandDataWatcher(json).getWatchableObjects()); + } + return metaPacket; + } + + private Location getEntityLocation(double correction) { + var player = owner.getOwner(); + double x = player.getLocation().getX(); + double y = player.getLocation().getY(); + double z = player.getLocation().getZ(); + y += yOffset; + y += owner.getHatOffset(); + y += correction; + return new Location(null, x, y, z); + } + + private WrappedDataWatcher createArmorStandDataWatcher(String json) { + WrappedDataWatcher wrappedDataWatcher = new WrappedDataWatcher(); + WrappedDataWatcher.Serializer serializer1 = WrappedDataWatcher.Registry.get(Boolean.class); + WrappedDataWatcher.Serializer serializer2 = WrappedDataWatcher.Registry.get(Byte.class); + wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(2, WrappedDataWatcher.Registry.getChatComponentSerializer(true)), Optional.of(WrappedChatComponent.fromJson(json).getHandle())); + wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(3, serializer1), true); + byte flag = 0x20; + if (sneaking) flag += (byte) 0x02; + wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(0, serializer2), flag); + wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(15, serializer2), (byte) 0x01); + wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(3, serializer1), true); + return wrappedDataWatcher; + } + + private PacketContainer[] getSpawnPackets(String text, Pose pose) { + PacketContainer entityPacket = new PacketContainer(PacketType.Play.Server.SPAWN_ENTITY); + entityPacket.getModifier().write(0, entityId); + entityPacket.getModifier().write(1, uuid); + entityPacket.getEntityTypeModifier().write(0, EntityType.ARMOR_STAND); + Location location = getEntityLocation(getCorrection(pose)); + entityPacket.getDoubles().write(0, location.getX()); + entityPacket.getDoubles().write(1, location.getY()); + entityPacket.getDoubles().write(2, location.getZ()); + PacketContainer metaPacket = getMetaPacket(text); + return new PacketContainer[] {entityPacket, metaPacket}; + } + + private double getCorrection(Pose pose) { + return switch (pose) { + case STANDING -> 1.8; + case SNEAKING -> 1.5; + case SLEEPING -> 0.2; + case SWIMMING, FALL_FLYING, SPIN_ATTACK -> 0.55; + default -> 0; + }; + } +} \ No newline at end of file diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/unlimited/UnlimitedTagManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/unlimited/UnlimitedTagManagerImpl.java new file mode 100644 index 0000000..3600da8 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/nameplate/tag/unlimited/UnlimitedTagManagerImpl.java @@ -0,0 +1,199 @@ +package net.momirealms.customnameplates.paper.mechanic.nameplate.tag.unlimited; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.NameplateManager; +import net.momirealms.customnameplates.api.manager.UnlimitedTagManager; +import net.momirealms.customnameplates.api.mechanic.misc.ViewerText; +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.NamedEntity; +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.UnlimitedObject; +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.UnlimitedPlayer; +import net.momirealms.customnameplates.api.mechanic.tag.unlimited.UnlimitedTagSetting; +import net.momirealms.customnameplates.api.scheduler.CancellableTask; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.mechanic.nameplate.tag.listener.MagicCosmeticsListener; +import net.momirealms.customnameplates.paper.util.LocationUtils; +import org.bukkit.Bukkit; +import org.bukkit.entity.Entity; +import org.bukkit.entity.Player; +import org.bukkit.entity.Pose; +import org.bukkit.event.HandlerList; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.List; +import java.util.UUID; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.TimeUnit; + +public class UnlimitedTagManagerImpl implements UnlimitedTagManager { + + private final NameplateManager manager; + private final ConcurrentHashMap unlimitedEntryMap; + private CancellableTask refreshTask; + private MagicCosmeticsListener magicCosmeticsListener; + + public UnlimitedTagManagerImpl(NameplateManager nameplateManager) { + this.manager = nameplateManager; + this.unlimitedEntryMap = new ConcurrentHashMap<>(); + if (Bukkit.getPluginManager().getPlugin("MagicCosmetics") != null) { + this.magicCosmeticsListener = new MagicCosmeticsListener(this); + } + } + + public void load() { + this.refreshTask = CustomNameplatesPlugin.get().getScheduler().runTaskAsyncTimer( + () -> { + try { + for (UnlimitedObject unlimitedObject : unlimitedEntryMap.values()) { + if (unlimitedObject instanceof UnlimitedPlayer unlimitedPlayer) { + unlimitedPlayer.timer(); + } + } + } catch (Exception e) { + LogUtils.severe( + "Error occurred when updating unlimited tags. " + + "This might not be a bug in CustomNameplates. Please report " + + "to the Plugin on the top of the following " + + "stack trace." + ); + e.printStackTrace(); + } + }, + 50L, + 50L, + TimeUnit.MILLISECONDS + ); + if (this.magicCosmeticsListener != null) { + Bukkit.getPluginManager().registerEvents(magicCosmeticsListener, CustomNameplatesPlugin.get()); + } + } + + public void unload() { + if (this.refreshTask != null && !this.refreshTask.isCancelled()) { + this.refreshTask.cancel(); + } + for (UnlimitedObject entry : unlimitedEntryMap.values()) { + entry.destroy(); + } + if (this.magicCosmeticsListener != null) { + HandlerList.unregisterAll(magicCosmeticsListener); + } + } + + @NotNull + @Override + public NamedEntity createNamedEntity(UnlimitedPlayer player, UnlimitedTagSetting setting) { + return new NamedEntityImpl( + player, + new ViewerText(player.getOwner(), setting.getRawText()), + setting.getRefreshFrequency(), + setting.getCheckFrequency(), + setting.getVerticalOffset(), + setting.getOwnerRequirements(), + setting.getViewerRequirements() + ); + } + + @Override + @SuppressWarnings("DuplicatedCode") + public UnlimitedPlayer createTagForPlayer(Player player, List settings) { + if (this.unlimitedEntryMap.containsKey(player.getUniqueId())) { + return null; + } + + var unlimitedPlayer = new UnlimitedPlayer(this, player); + this.unlimitedEntryMap.put( + player.getUniqueId(), + unlimitedPlayer + ); + + for (UnlimitedTagSetting setting : settings) { + unlimitedPlayer.addTag( + createNamedEntity(unlimitedPlayer, setting) + ); + } + + for (Player online : Bukkit.getOnlinePlayers()) { + if ( online == player + || !online.canSee(player) + || LocationUtils.getDistance(online, player) > 48 + || online.getWorld() != player.getWorld() + || online.isDead() + ) continue; + unlimitedPlayer.addNearbyPlayer(online); + } + return unlimitedPlayer; + } + + @Override + public UnlimitedObject removeUnlimitedObjectFromMap(UUID uuid) { + return unlimitedEntryMap.remove(uuid); + } + + @Nullable + @Override + public UnlimitedObject getUnlimitedObject(UUID uuid) { + return unlimitedEntryMap.get(uuid); + } + + public void handleEntitySpawnPacket(Player receiver, int entityId) { + Player spawned = manager.getPlayerByEntityID(entityId); + if (spawned == null) return; + UnlimitedObject unlimitedObject = getUnlimitedObject(spawned.getUniqueId()); + if (unlimitedObject == null) return; + unlimitedObject.addNearbyPlayer(receiver); + } + + public void handlePlayerPose(Player player, Pose pose) { + UnlimitedObject unlimitedObject = getUnlimitedObject(player.getUniqueId()); + if (unlimitedObject != null) { + unlimitedObject.handlePose(player.getPose(), pose); + } + } + + public void handlePlayerQuit(Player quit) { + UnlimitedObject unlimitedObject = getUnlimitedObject(quit.getUniqueId()); + if (unlimitedObject != null) { + unlimitedObject.destroy(); + } + for (UnlimitedObject entry : unlimitedEntryMap.values()) { + entry.removeNearbyPlayer(quit); + } + } + + public void handleEntityMovePacket(Player receiver, int entityID, short x, short y, short z, boolean onGround) { + Entity mover = manager.getEntityByEntityID(entityID); + if (mover == null) return; + UnlimitedObject unlimitedObject = getUnlimitedObject(mover.getUniqueId()); + if (unlimitedObject == null) return; + unlimitedObject.move(receiver, x, y, z, onGround); + } + + public void handleEntityTeleportPacket(Player receiver, int entityID, double x, double y, double z, boolean onGround) { + Entity tp = manager.getEntityByEntityID(entityID); + if (tp == null) return; + UnlimitedObject unlimitedObject = getUnlimitedObject(tp.getUniqueId()); + if (unlimitedObject == null) return; + unlimitedObject.teleport(receiver, x, y, z, onGround); + } + + public void handleEntityDestroyPacket(Player receiver, List list) { + for (int id : list) { + handleSingleEntityDestroy(receiver, id); + } + } + + private void handleSingleEntityDestroy(Player receiver, int entityID) { + Entity deSpawned = manager.getEntityByEntityID(entityID); + if (deSpawned == null) return; + UnlimitedObject unlimitedObject = getUnlimitedObject(deSpawned.getUniqueId()); + if (unlimitedObject == null) return; + unlimitedObject.removeNearbyPlayer(receiver); + } + + public void handlePlayerSneak(Player sneaker, boolean sneaking, boolean flying) { + UnlimitedObject unlimitedObject = getUnlimitedObject(sneaker.getUniqueId()); + if (unlimitedObject == null) return; + unlimitedObject.sneak(sneaking, flying); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/pack/ResourcePackManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/pack/ResourcePackManagerImpl.java new file mode 100644 index 0000000..df14f1b --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/pack/ResourcePackManagerImpl.java @@ -0,0 +1,285 @@ +package net.momirealms.customnameplates.paper.mechanic.pack; + +import com.google.gson.JsonArray; +import com.google.gson.JsonObject; +import com.google.gson.JsonPrimitive; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.ResourcePackManager; +import net.momirealms.customnameplates.api.mechanic.background.BackGround; +import net.momirealms.customnameplates.api.mechanic.character.ConfiguredChar; +import net.momirealms.customnameplates.api.mechanic.font.OffsetFont; +import net.momirealms.customnameplates.api.mechanic.nameplate.Nameplate; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.setting.CNConfig; +import org.bukkit.Bukkit; +import org.codehaus.plexus.util.FileUtils; + +import java.io.*; +import java.nio.charset.StandardCharsets; +import java.util.ArrayList; +import java.util.List; +import java.util.Objects; + +public class ResourcePackManagerImpl implements ResourcePackManager { + + private final CustomNameplatesPlugin plugin; + + public ResourcePackManagerImpl(CustomNameplatesPlugin plugin) { + this.plugin = plugin; + } + + public void reload() { + unload(); + load(); + } + + public void load() { + + } + + public void unload() { + + } + + @Override + public void generateResourcePack() { + // delete the old one + File resourcePackFolder = new File(plugin.getDataFolder() + File.separator + "ResourcePack"); + this.deleteDirectory(resourcePackFolder); + + // create folders + File fontFolder = new File(plugin.getDataFolder(), "ResourcePack" + File.separator + "assets" + File.separator + CNConfig.namespace + File.separatorChar + "font"); + File texturesFolder = new File(plugin.getDataFolder(), "ResourcePack" + File.separator+ "assets" + File.separator + CNConfig.namespace + File.separatorChar + "textures"); + if (!fontFolder.mkdirs() || !texturesFolder.mkdirs()) { + LogUtils.severe("Failed to generate resource pack folders"); + return; + } + + // create json object + JsonObject fontJson = new JsonObject(); + JsonArray providers = new JsonArray(); + fontJson.add("providers", providers); + + // add offset characters + this.getOffsets(texturesFolder).forEach(providers::add); + // add nameplate characters + this.getNameplates(texturesFolder).forEach(providers::add); + // add bubble characters + this.getBubbles(texturesFolder).forEach(providers::add); + // add background characters + this.getBackgrounds(texturesFolder).forEach(providers::add); + // add image characters + this.getImages(texturesFolder).forEach(providers::add); + // set pack.mcmeta + this.setPackFormat(); + // save json object to file + this.saveFont(fontJson); + // copy the resource pack to hooked plugins + this.copyResourcePackToHookedPlugins(resourcePackFolder); + } + + private void saveFont(JsonObject fontJson) { + try (FileWriter fileWriter = new FileWriter( + plugin.getDataFolder() + + File.separator + "ResourcePack" + + File.separator + "assets" + + File.separator + CNConfig.namespace + + File.separator + "font" + + File.separator + CNConfig.font + ".json") + ) { + fileWriter.write(fontJson.toString().replace("\\\\", "\\")); + } catch (IOException e) { + e.printStackTrace(); + } + } + + @Override + public void deleteDirectory(File file){ + if (file.exists()) { + try { + FileUtils.deleteDirectory(file); + } catch (IOException e){ + e.printStackTrace(); + } + } + } + + @Override + public String native2ascii(char c) { + if (c > '\u007f') { + StringBuilder stringBuilder_1 = new StringBuilder("\\u"); + StringBuilder stringBuilder_2 = new StringBuilder(Integer.toHexString(c)); + stringBuilder_2.reverse(); + for (int n = 4 - stringBuilder_2.length(), i = 0; i < n; i++) stringBuilder_2.append('0'); + for (int j = 0; j < 4; j++) stringBuilder_1.append(stringBuilder_2.charAt(3 - j)); + return stringBuilder_1.toString(); + } + return Character.toString(c); + } + + @SuppressWarnings("ResultOfMethodCallIgnored") + private void saveSplit(File texturesFolder) { + try { + plugin.saveResource("space_split.png", false); + FileUtils.copyFile(new File(plugin.getDataFolder(),"space_split.png"), new File(texturesFolder, CNConfig.folderSplit.replace("\\", File.separator) + "space_split.png")); + File file = new File(plugin.getDataFolder(),"space_split.png"); + if (file.exists()) { + file.delete(); + } + } catch (IOException e){ + e.printStackTrace(); + } + } + + private void copyResourcePackToHookedPlugins(File resourcePackFolder) { + if (CNConfig.copyPackIA) { + try { + FileUtils.copyDirectory(new File(resourcePackFolder, "assets"), new File(Objects.requireNonNull(Bukkit.getPluginManager().getPlugin("ItemsAdder")).getDataFolder() + File.separator + "contents" + File.separator + "nameplates" + File.separator + "resourcepack" + File.separator + "assets") ); + } catch (IOException e) { + e.printStackTrace(); + } + } + if (CNConfig.copyPackOraxen) { + try { + FileUtils.copyDirectory(new File(resourcePackFolder, "assets"), new File(Objects.requireNonNull(Bukkit.getPluginManager().getPlugin("Oraxen")).getDataFolder() + File.separator + "pack" + File.separator + "assets")); + } catch (IOException e) { + e.printStackTrace(); + } + } + } + + private List getNameplates(File texturesFolder) { + ArrayList list = new ArrayList<>(); + if (!CNConfig.nameplateModule) return list; + for (Nameplate nameplate : plugin.getNameplateManager().getNameplates()) { + for (ConfiguredChar configuredChar : new ConfiguredChar[]{nameplate.getLeft(), nameplate.getMiddle(), nameplate.getRight()}) { + JsonObject jo = new JsonObject(); + jo.add("type", new JsonPrimitive("bitmap")); + jo.add("file", new JsonPrimitive(CNConfig.namespace + ":" + CNConfig.folderNameplate.replaceAll("\\\\", "/") + configuredChar.getFile())); + jo.add("ascent", new JsonPrimitive(configuredChar.getAscent())); + jo.add("height", new JsonPrimitive(configuredChar.getHeight())); + JsonArray ja = new JsonArray(); + ja.add(native2ascii(configuredChar.getCharacter())); + jo.add("chars", ja); + list.add(jo); + list.add(jo); + try { + FileUtils.copyFile( + new File(plugin.getDataFolder(), + "contents" + File.separator + "nameplates" + File.separator + configuredChar.getFile()), + new File(texturesFolder, + CNConfig.folderNameplate.replace("\\", File.separator) + configuredChar.getFile())); + } catch (IOException e) { + e.printStackTrace(); + } + } + } + return list; + } + + + private List getBackgrounds(File texturesFolder) { + ArrayList list = new ArrayList<>(); + if (!CNConfig.backgroundModule) return list; + for (BackGround backGround : plugin.getBackGroundManager().getBackGrounds()) { + for (ConfiguredChar configuredChar : new ConfiguredChar[]{ + backGround.getLeft(), backGround.getOffset_1(), + backGround.getOffset_2(), backGround.getOffset_4(), + backGround.getOffset_8(), backGround.getOffset_16(), + backGround.getOffset_32(), backGround.getOffset_64(), + backGround.getOffset_128(), backGround.getRight()} + ) { + JsonObject jo = new JsonObject(); + jo.add("type", new JsonPrimitive("bitmap")); + jo.add("file", new JsonPrimitive(CNConfig.namespace + ":" + CNConfig.folderNameplate.replaceAll("\\\\", "/") + configuredChar.getFile())); + jo.add("ascent", new JsonPrimitive(configuredChar.getAscent())); + jo.add("height", new JsonPrimitive(configuredChar.getHeight())); + JsonArray ja = new JsonArray(); + ja.add(native2ascii(configuredChar.getCharacter())); + jo.add("chars", ja); + list.add(jo); + try { + FileUtils.copyFile( + new File(plugin.getDataFolder(), + "contents" + File.separator + "backgrounds" + File.separator + configuredChar.getFile()), + new File(texturesFolder, + CNConfig.folderBackground.replace("\\", File.separator) + configuredChar.getFile())); + } catch (IOException e) { + e.printStackTrace(); + } + } + } + return list; + } + + private List getBubbles(File texturesFolder) { + ArrayList list = new ArrayList<>(); + if (!CNConfig.bubbleModule) return list; + return list; + } + + private List getImages(File texturesFolder) { + ArrayList list = new ArrayList<>(); + if (!CNConfig.imageModule) return list; + for (ConfiguredChar configuredChar : plugin.getImageManager().getImages()) { + JsonObject jo = new JsonObject(); + jo.add("type", new JsonPrimitive("bitmap")); + jo.add("file", new JsonPrimitive(CNConfig.namespace + ":" + CNConfig.folderNameplate.replaceAll("\\\\", "/") + configuredChar.getFile())); + jo.add("ascent", new JsonPrimitive(configuredChar.getAscent())); + jo.add("height", new JsonPrimitive(configuredChar.getHeight())); + JsonArray ja = new JsonArray(); + ja.add(native2ascii(configuredChar.getCharacter())); + jo.add("chars", ja); + list.add(jo); + list.add(jo); + try { + FileUtils.copyFile( + new File(plugin.getDataFolder(), + "contents" + File.separator + "images" + File.separator + configuredChar.getFile()), + new File(texturesFolder, + CNConfig.folderImage.replace("\\", File.separator) + configuredChar.getFile())); + } catch (IOException e) { + e.printStackTrace(); + } + } + return list; + } + + private List getOffsets(File texturesFolder) { + this.saveSplit(texturesFolder); + ArrayList list = new ArrayList<>(); + for (OffsetFont offsetFont : OffsetFont.values()) { + JsonObject jsonObject = new JsonObject(); + jsonObject.add("type", new JsonPrimitive("bitmap")); + jsonObject.add("file", new JsonPrimitive(CNConfig.namespace + ":" + CNConfig.folderSplit.replaceAll("\\\\","/") + "space_split.png")); + jsonObject.add("ascent", new JsonPrimitive(-5000)); + jsonObject.add("height", new JsonPrimitive(offsetFont.getHeight())); + final JsonArray jsonArray = new JsonArray(); + jsonArray.add(native2ascii(offsetFont.getCharacter())); + jsonObject.add("chars", jsonArray); + list.add(jsonObject); + } + return list; + } + + private void setPackFormat() { + plugin.saveResource("ResourcePack" + File.separator + "pack.mcmeta", false); + File format_file = new File(plugin.getDataFolder(), "ResourcePack" + File.separator + "pack.mcmeta"); + String line; + StringBuilder sb = new StringBuilder(); + try (BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(format_file), StandardCharsets.UTF_8))) { + while ((line = reader.readLine()) != null) { + sb.append(line).append(System.lineSeparator()); + } + } catch (IOException e) { + e.printStackTrace(); + } + try (BufferedWriter writer = new BufferedWriter( + new OutputStreamWriter(new FileOutputStream(new File(plugin.getDataFolder(), + "ResourcePack" + File.separator + "pack.mcmeta")), StandardCharsets.UTF_8))) { + writer.write(sb.toString().replace("%version%", String.valueOf(plugin.getVersionManager().getPackFormat()))); + } catch (IOException e) { + e.printStackTrace(); + } + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/placeholder/PlaceholderManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/placeholder/PlaceholderManagerImpl.java new file mode 100644 index 0000000..deb25d4 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/placeholder/PlaceholderManagerImpl.java @@ -0,0 +1,254 @@ +package net.momirealms.customnameplates.paper.mechanic.placeholder; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.common.Pair; +import net.momirealms.customnameplates.api.manager.PlaceholderManager; +import net.momirealms.customnameplates.api.mechanic.placeholder.*; +import net.momirealms.customnameplates.api.requirement.Requirement; +import net.momirealms.customnameplates.api.util.LogUtils; +import org.bukkit.configuration.ConfigurationSection; +import org.bukkit.configuration.file.YamlConfiguration; +import org.jetbrains.annotations.NotNull; + +import java.io.File; +import java.util.*; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +public class PlaceholderManagerImpl implements PlaceholderManager { + + private final Pattern placeholderPattern = Pattern.compile("%([^%]*)%"); + private final PluginPlaceholders pluginPlaceholders; + private final HashMap staticTextMap; + private final HashMap switchTextMap; + private final HashMap descentTextMap; + private final HashMap conditionalTextMap; + private final HashMap nameplateTextMap; + private final HashMap backGroundTextMap; + private final CustomNameplatesPlugin plugin; + + public PlaceholderManagerImpl(CustomNameplatesPlugin plugin) { + this.plugin = plugin; + this.pluginPlaceholders = new PluginPlaceholders(plugin, this); + this.staticTextMap = new HashMap<>(); + this.switchTextMap = new HashMap<>(); + this.descentTextMap = new HashMap<>(); + this.conditionalTextMap = new HashMap<>(); + this.nameplateTextMap = new HashMap<>(); + this.backGroundTextMap = new HashMap<>(); + } + + @NotNull + @Override + public List detectPlaceholders(String text){ + List placeholders = new ArrayList<>(); + Matcher matcher = placeholderPattern.matcher(text); + while (matcher.find()) placeholders.add(matcher.group()); + return placeholders; + } + + public void reload() { + unload(); + load(); + } + + public void load() { + this.loadConfigs(); + + if (!pluginPlaceholders.isRegistered()) + pluginPlaceholders.register(); + } + + public void unload() { + if (pluginPlaceholders.isRegistered()) + pluginPlaceholders.unregister(); + + this.staticTextMap.clear(); + this.switchTextMap.clear(); + this.descentTextMap.clear(); + this.conditionalTextMap.clear(); + } + + @Override + public StaticText getStaticText(String key) { + return staticTextMap.get(key); + } + + @Override + public SwitchText getSwitchText(String key) { + return switchTextMap.get(key); + } + + @Override + public DescentText getDescentText(String key) { + return descentTextMap.get(key); + } + + @Override + public ConditionalText getConditionalText(String key) { + return conditionalTextMap.get(key); + } + + @Override + public NameplateText getNameplateText(String key) { + return nameplateTextMap.get(key); + } + + @Override + public BackGroundText getBackGroundText(String key) { + return backGroundTextMap.get(key); + } + + public void loadConfigs() { + YamlConfiguration config = plugin.getConfig("configs" + File.separator + "custom-placeholders.yml"); + ConfigurationSection staticSection = config.getConfigurationSection("static-text"); + if (staticSection != null) { + loadStaticTexts(staticSection); + } + + ConfigurationSection switchSection = config.getConfigurationSection("switch-text"); + if (switchSection != null) { + loadSwitchTexts(switchSection); + } + + ConfigurationSection descentSection = config.getConfigurationSection("descent-text"); + if (descentSection != null) { + loadDescentTexts(descentSection, false); + } + + ConfigurationSection unicodeSection = config.getConfigurationSection("descent-unicode"); + if (unicodeSection != null) { + loadDescentTexts(unicodeSection, true); + } + + ConfigurationSection conditionalSection = config.getConfigurationSection("conditional-text"); + if (conditionalSection != null) { + loadConditionalTexts(conditionalSection); + } + + ConfigurationSection nameplateSection = config.getConfigurationSection("nameplate-text"); + if (nameplateSection != null) { + loadNameplateTexts(nameplateSection); + } + + ConfigurationSection backgroundSection = config.getConfigurationSection("background-text"); + if (backgroundSection != null) { + loadBackGroundTexts(backgroundSection); + } + } + + private void loadBackGroundTexts(ConfigurationSection section) { + for (Map.Entry entry : section.getValues(false).entrySet()) { + if (!(entry.getValue() instanceof ConfigurationSection innerSection)) + continue; + + backGroundTextMap.put(entry.getKey(), + BackGroundText.builder() + .background(Objects.requireNonNull(plugin.getBackGroundManager().getBackGround(innerSection.getString("background")))) + .text(innerSection.getString("text", "")) + .build() + ); + } + } + + private void loadNameplateTexts(ConfigurationSection section) { + for (Map.Entry entry : section.getValues(false).entrySet()) { + if (!(entry.getValue() instanceof ConfigurationSection innerSection)) + continue; + + var nameplate = plugin.getNameplateManager().getNameplate(innerSection.getString("nameplate")); + if (nameplate == null) { + LogUtils.warn("Nameplate: " + innerSection.getString("nameplate") + " doesn't exist. nameplate-text: " + entry.getKey() + " would not take effect"); + continue; + } + + nameplateTextMap.put(entry.getKey(), + NameplateText.builder() + .nameplate(nameplate) + .text(innerSection.getString("text", "")) + .build() + ); + } + } + + private void loadDescentTexts(ConfigurationSection section, boolean unicode) { + for (Map.Entry entry : section.getValues(false).entrySet()) { + if (!(entry.getValue() instanceof ConfigurationSection innerSection)) + continue; + + descentTextMap.put( + entry.getKey(), + DescentText.builder() + .descent(innerSection.getInt("descent", 0)) + .text(innerSection.getString("text", "")) + .unicode(unicode) + .build() + ); + } + } + + private void loadStaticTexts(ConfigurationSection section) { + for (Map.Entry entry : section.getValues(false).entrySet()) { + if (!(entry.getValue() instanceof ConfigurationSection innerSection)) + continue; + + staticTextMap.put( + entry.getKey(), + StaticText.builder() + .value(innerSection.getInt("value")) + .text(innerSection.getString("text")) + .state(StaticText.StaticState.valueOf(innerSection.getString("position", "middle").toUpperCase(Locale.ENGLISH))) + .build() + ); + } + } + + private void loadSwitchTexts(ConfigurationSection section) { + for (Map.Entry entry : section.getValues(false).entrySet()) { + if (!(entry.getValue() instanceof ConfigurationSection innerSection)) + continue; + + HashMap valueMap = new HashMap<>(); + ConfigurationSection valueSection = innerSection.getConfigurationSection("case"); + if (valueSection != null) { + for (String key : valueSection.getKeys(false)) { + valueMap.put(key, valueSection.getString(key)); + } + } + + switchTextMap.put( + entry.getKey(), + SwitchText.builder() + .toParse(Objects.requireNonNull(innerSection.getString("switch"))) + .defaultValue(innerSection.getString("default")) + .valueMap(valueMap) + .build() + ); + } + } + + private void loadConditionalTexts(ConfigurationSection section) { + for (Map.Entry entry : section.getValues(false).entrySet()) { + if (!(entry.getValue() instanceof ConfigurationSection innerSection)) + continue; + + ArrayList> list = new ArrayList<>(); + for (Map.Entry innerEntry : innerSection.getValues(false).entrySet()) { + if (!(innerEntry.getValue() instanceof ConfigurationSection prioritySection)) { + continue; + } + list.add(Pair.of( + prioritySection.getString("text"), + plugin.getRequirementManager().getRequirements(prioritySection.getConfigurationSection("conditions")) + )); + } + + conditionalTextMap.put( + entry.getKey(), + ConditionalText.builder() + .textList(list) + .build() + ); + } + } +} \ No newline at end of file diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/placeholder/PluginPlaceholders.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/placeholder/PluginPlaceholders.java new file mode 100644 index 0000000..a7270fa --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/placeholder/PluginPlaceholders.java @@ -0,0 +1,141 @@ +package net.momirealms.customnameplates.paper.mechanic.placeholder; + +import me.clip.placeholderapi.expansion.PlaceholderExpansion; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.PlaceholderManager; +import net.momirealms.customnameplates.api.mechanic.character.ConfiguredChar; +import net.momirealms.customnameplates.api.mechanic.font.OffsetFont; +import net.momirealms.customnameplates.api.mechanic.placeholder.*; +import net.momirealms.customnameplates.api.util.FontUtils; +import org.bukkit.OfflinePlayer; +import org.bukkit.entity.Player; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +public class PluginPlaceholders extends PlaceholderExpansion { + + private final CustomNameplatesPlugin plugin; + private final PlaceholderManager placeholderManager; + + public PluginPlaceholders(CustomNameplatesPlugin plugin, PlaceholderManager placeholderManager) { + this.plugin = plugin; + this.placeholderManager = placeholderManager; + } + + @Override + public @NotNull String getIdentifier() { + return "nameplates"; + } + + @Override + public @NotNull String getAuthor() { + return "XiaoMoMi"; + } + + @Override + public @NotNull String getVersion() { + return "2.3"; + } + + @Override + public boolean persist() { + return true; + } + + @Override + public @Nullable String onRequest(OfflinePlayer offlinePlayer, @NotNull String params) { + String[] mainSplit = params.split("_", 2); + String mainPara = mainSplit[0]; + String mainArg = mainSplit.length == 1 ? "" : mainSplit[1]; + switch (mainPara) { + case "image" -> { + ConfiguredChar configuredChar = plugin.getImageManager().getImage(mainArg); + if (configuredChar == null) return "Image not exists"; + return FontUtils.surroundNameplateFont(String.valueOf(configuredChar.getCharacter())); + } + case "image-char" -> { + ConfiguredChar configuredChar = plugin.getImageManager().getImage(mainArg); + if (configuredChar == null) return "Image not exists"; + return String.valueOf(configuredChar.getCharacter()); + } + case "offset" -> { + return FontUtils.surroundNameplateFont(OffsetFont.getOffsetChars(Integer.parseInt(mainArg))); + } + case "offset-char" -> { + return OffsetFont.getOffsetChars(Integer.parseInt(mainArg)); + } + case "checkupdate" -> { + return String.valueOf(!plugin.getVersionManager().isLatest()); + } + case "is-latest" -> { + return String.valueOf(plugin.getVersionManager().isLatest()); + } + case "static" -> { + StaticText text = placeholderManager.getStaticText(mainArg); + if (text == null) return "Static text not exists"; + return text.getValue(offlinePlayer); + } + case "unicode", "descent" -> { + DescentText descentText = placeholderManager.getDescentText(mainArg); + if (descentText == null) return "Descent text not exists"; + return descentText.getValue(offlinePlayer); + } + case "conditional" -> { + ConditionalText conditionalText = placeholderManager.getConditionalText(mainArg); + if (conditionalText == null) return "Conditional text not exists"; + return conditionalText.getValue(offlinePlayer); + } + case "nameplate" -> { + NameplateText nameplateText = placeholderManager.getNameplateText(mainArg); + if (nameplateText == null) return "Nameplate text not exists"; + return nameplateText.getValue(offlinePlayer); + } + case "background" -> { + BackGroundText backGroundText = placeholderManager.getBackGroundText(mainArg); + if (backGroundText == null) return "Background text not exists"; + return backGroundText.getValue(offlinePlayer); + } + } + + Player onlinePlayer = offlinePlayer.getPlayer(); + if (onlinePlayer == null) return null; + switch (mainPara) { + case "time" -> { + long time = onlinePlayer.getWorld().getTime(); + String ap = time >= 6000 && time < 18000 ? " PM" : " AM"; + int hours = (int) (time / 1000) ; + int minutes = (int) ((time - hours * 1000 ) * 0.06); + hours += 6; + while (hours >= 12) hours -= 12; + if (minutes < 10) return hours + ":0" + minutes + ap; + else return hours + ":" + minutes + ap; + } + case "actionbar" -> { + return plugin.getActionBarManager().getOtherPluginActionBar(onlinePlayer); + } + case "prefix" -> { + return plugin.getNameplateManager().getNameplatePrefix(onlinePlayer); + } + case "suffix" -> { + return plugin.getNameplateManager().getNameplateSuffix(onlinePlayer); + } + case "nametag" -> { + return plugin.getNameplateManager().getFullNameTag(onlinePlayer); + } + case "equipped" -> { + var optPlayer = plugin.getStorageManager().getOnlineUser(onlinePlayer.getUniqueId()); + if (optPlayer.isEmpty()) return "Data not loaded"; + switch (mainArg) { + case "nameplate" -> { + return optPlayer.get().getNameplateKey(); + } + case "bubble" -> { + return optPlayer.get().getBubbleKey(); + } + } + } + } + return null; + } + +} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiEquals.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/requirement/EmptyRequirement.java similarity index 58% rename from src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiEquals.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/requirement/EmptyRequirement.java index 257d75b..0a7ea13 100644 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiEquals.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/requirement/EmptyRequirement.java @@ -15,18 +15,20 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.object.requirements.papi; +package net.momirealms.customnameplates.paper.mechanic.requirement; -import me.clip.placeholderapi.PlaceholderAPI; -import org.bukkit.entity.Player; +import net.momirealms.customnameplates.api.requirement.Condition; +import net.momirealms.customnameplates.api.requirement.Requirement; -import java.util.Objects; +/** + * Represents an empty requirement that always returns true when checking conditions. + */ +public class EmptyRequirement implements Requirement { -public record PapiEquals(String papi, String requirement) implements PapiRequirement{ + public static EmptyRequirement instance = new EmptyRequirement(); @Override - public boolean isMet(Player player) { - String value = PlaceholderAPI.setPlaceholders(player, papi); - return Objects.equals(value, PlaceholderAPI.setPlaceholders(player, requirement)); + public boolean isConditionMet(Condition condition) { + return true; } } diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/requirement/RequirementManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/requirement/RequirementManagerImpl.java new file mode 100644 index 0000000..fd62c6c --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/requirement/RequirementManagerImpl.java @@ -0,0 +1,706 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.mechanic.requirement; + +import me.clip.placeholderapi.PlaceholderAPI; +import net.momirealms.biomeapi.BiomeAPI; +import net.momirealms.customnameplates.api.common.Pair; +import net.momirealms.customnameplates.api.manager.RequirementManager; +import net.momirealms.customnameplates.api.requirement.Requirement; +import net.momirealms.customnameplates.api.requirement.RequirementExpansion; +import net.momirealms.customnameplates.api.requirement.RequirementFactory; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; +import net.momirealms.customnameplates.paper.util.ClassUtils; +import net.momirealms.customnameplates.paper.util.ConfigUtils; +import org.bukkit.World; +import org.bukkit.configuration.ConfigurationSection; +import org.bukkit.potion.PotionEffect; +import org.bukkit.potion.PotionEffectType; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.io.File; +import java.io.IOException; +import java.lang.reflect.InvocationTargetException; +import java.util.*; + +public class RequirementManagerImpl implements RequirementManager { + + private final CustomNameplatesPluginImpl plugin; + private final String EXPANSION_FOLDER = "expansions/requirement"; + private final HashMap requirementBuilderMap; + + public RequirementManagerImpl(CustomNameplatesPluginImpl plugin) { + this.plugin = plugin; + this.requirementBuilderMap = new HashMap<>(64); + this.registerInbuiltRequirements(); + } + + public void load() { + this.loadExpansions(); + } + + public void unload() { + + } + + public void reload() { + unload(); + load(); + } + + public void disable() { + this.requirementBuilderMap.clear(); + } + + /** + * Registers a custom requirement type with its corresponding factory. + * + * @param type The type identifier of the requirement. + * @param requirementFactory The factory responsible for creating instances of the requirement. + * @return True if registration was successful, false if the type is already registered. + */ + @Override + public boolean registerRequirement(String type, RequirementFactory requirementFactory) { + if (this.requirementBuilderMap.containsKey(type)) return false; + this.requirementBuilderMap.put(type, requirementFactory); + return true; + } + + /** + * Unregisters a custom requirement type. + * + * @param type The type identifier of the requirement to unregister. + * @return True if unregistration was successful, false if the type is not registered. + */ + @Override + public boolean unregisterRequirement(String type) { + return this.requirementBuilderMap.remove(type) != null; + } + + private void registerInbuiltRequirements() { + this.registerTimeRequirement(); + this.registerYRequirement(); + this.registerContainRequirement(); + this.registerStartWithRequirement(); + this.registerEndWithRequirement(); + this.registerEqualsRequirement(); + this.registerBiomeRequirement(); + this.registerDateRequirement(); + this.registerPermissionRequirement(); + this.registerWorldRequirement(); + this.registerWeatherRequirement(); + this.registerGreaterThanRequirement(); + this.registerAndRequirement(); + this.registerOrRequirement(); + this.registerLevelRequirement(); + this.registerRandomRequirement(); + this.registerCoolDownRequirement(); + this.registerLessThanRequirement(); + this.registerNumberEqualRequirement(); + this.registerRegexRequirement(); + this.registerEnvironmentRequirement(); + this.registerPotionEffectRequirement(); + } + + /** + * Retrieves an array of requirements based on a configuration section. + * + * @param section The configuration section containing requirement definitions. + * @return An array of Requirement objects based on the configuration section + */ + @NotNull + @Override + public Requirement[] getRequirements(ConfigurationSection section) { + List requirements = new ArrayList<>(); + if (section == null) { + return requirements.toArray(new Requirement[0]); + } + for (Map.Entry entry : section.getValues(false).entrySet()) { + String typeOrName = entry.getKey(); + if (hasRequirement(typeOrName)) { + requirements.add(getRequirement(typeOrName, entry.getValue())); + } else { + requirements.add(getRequirement(section.getConfigurationSection(typeOrName))); + } + } + return requirements.toArray(new Requirement[0]); + } + + public boolean hasRequirement(String type) { + return requirementBuilderMap.containsKey(type); + } + + /** + * Retrieves a Requirement object based on a configuration section and advanced flag. + * + * @param section The configuration section containing requirement definitions. + * @return A Requirement object based on the configuration section, or an EmptyRequirement if the section is null or invalid. + */ + @NotNull + @Override + public Requirement getRequirement(ConfigurationSection section) { + if (section == null) return EmptyRequirement.instance; + String type = section.getString("type"); + if (type == null) { + LogUtils.warn("No requirement type found at " + section.getCurrentPath()); + return EmptyRequirement.instance; + } + var builder = getRequirementFactory(type); + if (builder == null) { + return EmptyRequirement.instance; + } + return builder.build(section.get("value")); + } + + /** + * Gets a requirement based on the provided key and value. + * If a valid RequirementFactory is found for the key, it is used to create the requirement. + * If no factory is found, a warning is logged, and an empty requirement instance is returned. + * + * @param type The key representing the requirement type. + * @param value The value associated with the requirement. + * @return A Requirement instance based on the key and value, or an empty requirement if not found. + */ + @Override + @NotNull + public Requirement getRequirement(String type, Object value) { + RequirementFactory factory = getRequirementFactory(type); + if (factory == null) { + LogUtils.warn("Requirement type: " + type + " doesn't exist."); + return EmptyRequirement.instance; + } + return factory.build(value); + } + + /** + * Retrieves a RequirementFactory based on the specified requirement type. + * + * @param type The requirement type for which to retrieve a factory. + * @return A RequirementFactory for the specified type, or null if no factory is found. + */ + @Override + @Nullable + public RequirementFactory getRequirementFactory(String type) { + return requirementBuilderMap.get(type); + } + + private void registerTimeRequirement() { + registerRequirement("time", (args) -> { + List> timePairs = ConfigUtils.stringListArgs(args).stream().map(it -> ConfigUtils.splitStringIntegerArgs(it, "~")).toList(); + return condition -> { + long time = Objects.requireNonNull(condition.getOfflinePlayer().getPlayer()).getWorld().getTime(); + for (Pair pair : timePairs) + if (time >= pair.left() && time <= pair.right()) + return true; + return false; + }; + }); + } + + private void registerYRequirement() { + registerRequirement("ypos", (args) -> { + List> timePairs = ConfigUtils.stringListArgs(args).stream().map(it -> ConfigUtils.splitStringIntegerArgs(it, "~")).toList(); + return condition -> { + int y = Objects.requireNonNull(condition.getOfflinePlayer().getPlayer()).getLocation().getBlockY(); + for (Pair pair : timePairs) + if (y >= pair.left() && y <= pair.right()) + return true; + return false; + }; + }); + } + + private void registerOrRequirement() { + registerRequirement("||", (args) -> { + if (args instanceof ConfigurationSection section) { + Requirement[] requirements = getRequirements(section); + return condition -> { + for (Requirement requirement : requirements) { + if (requirement.isConditionMet(condition)) { + return true; + } + } + return false; + }; + } else { + LogUtils.warn("Wrong value format found at || requirement."); + return EmptyRequirement.instance; + } + }); + } + + private void registerAndRequirement() { + registerRequirement("&&", (args) -> { + if (args instanceof ConfigurationSection section) { + Requirement[] requirements = getRequirements(section); + return condition -> { + outer: { + for (Requirement requirement : requirements) { + if (!requirement.isConditionMet(condition)) { + break outer; + } + } + return true; + } + return false; + }; + } else { + LogUtils.warn("Wrong value format found at && requirement."); + return EmptyRequirement.instance; + } + }); + } + + private void registerLevelRequirement() { + registerRequirement("level", (args) -> { + int level = (int) args; + return condition -> { + int current = Objects.requireNonNull(condition.getOfflinePlayer().getPlayer()).getLevel(); + return current >= level; + }; + }); + } + + private void registerRandomRequirement() { + registerRequirement("random", (args) -> { + double random = ConfigUtils.getDoubleValue(args); + return condition -> Math.random() < random; + }); + } + + private void registerBiomeRequirement() { + registerRequirement("biome", (args) -> { + HashSet biomes = new HashSet<>(ConfigUtils.stringListArgs(args)); + return condition -> { + String currentBiome = BiomeAPI.getBiome(Objects.requireNonNull(condition.getOfflinePlayer().getPlayer()).getLocation()); + return biomes.contains(currentBiome); + }; + }); + registerRequirement("!biome", (args) -> { + HashSet biomes = new HashSet<>(ConfigUtils.stringListArgs(args)); + return condition -> { + String currentBiome = BiomeAPI.getBiome(Objects.requireNonNull(condition.getOfflinePlayer().getPlayer()).getLocation()); + return !biomes.contains(currentBiome); + }; + }); + } + + private void registerWorldRequirement() { + registerRequirement("world", (args) -> { + HashSet worlds = new HashSet<>(ConfigUtils.stringListArgs(args)); + return condition -> worlds.contains(Objects.requireNonNull(condition.getOfflinePlayer().getPlayer()).getWorld().getName()); + }); + registerRequirement("!world", (args) -> { + HashSet worlds = new HashSet<>(ConfigUtils.stringListArgs(args)); + return condition -> !worlds.contains(Objects.requireNonNull(condition.getOfflinePlayer().getPlayer()).getWorld().getName()); + }); + } + + private void registerWeatherRequirement() { + registerRequirement("weather", (args) -> { + List weathers = ConfigUtils.stringListArgs(args); + return condition -> { + String currentWeather; + World world = Objects.requireNonNull(condition.getOfflinePlayer().getPlayer()).getWorld(); + if (world.isThundering()) currentWeather = "thunder"; + else if (world.isClearWeather()) currentWeather = "clear"; + else currentWeather = "rain"; + for (String weather : weathers) + if (weather.equalsIgnoreCase(currentWeather)) + return true; + return false; + }; + }); + } + + private void registerCoolDownRequirement() { + registerRequirement("cooldown", (args) -> { + if (args instanceof ConfigurationSection section) { + String key = section.getString("key"); + int time = section.getInt("time"); + return condition -> !plugin.getCoolDownManager().isCoolDown(condition.getOfflinePlayer().getUniqueId(), key, time); + } else { + LogUtils.warn("Wrong value format found at cooldown requirement."); + return EmptyRequirement.instance; + } + }); + } + + private void registerDateRequirement() { + registerRequirement("date", (args) -> { + HashSet dates = new HashSet<>(ConfigUtils.stringListArgs(args)); + return condition -> { + Calendar calendar = Calendar.getInstance(); + String current = (calendar.get(Calendar.MONTH) + 1) + "/" + calendar.get(Calendar.DATE); + return dates.contains(current); + }; + }); + } + + private void registerPermissionRequirement() { + registerRequirement("permission", (args) -> { + List perms = ConfigUtils.stringListArgs(args); + return condition -> { + for (String perm : perms) + if (Objects.requireNonNull(condition.getOfflinePlayer().getPlayer()).hasPermission(perm)) + return true; + return false; + }; + }); + registerRequirement("!permission", (args) -> { + List perms = ConfigUtils.stringListArgs(args); + return condition -> { + for (String perm : perms) + if (Objects.requireNonNull(condition.getOfflinePlayer().getPlayer()).hasPermission(perm)) + return false; + return true; + }; + }); + } + + @SuppressWarnings("DuplicatedCode") + private void registerGreaterThanRequirement() { + registerRequirement(">=", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return Double.parseDouble(p1) >= Double.parseDouble(p2); + }; + } else { + LogUtils.warn("Wrong value format found at >= requirement."); + return EmptyRequirement.instance; + } + }); + registerRequirement(">", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return Double.parseDouble(p1) > Double.parseDouble(p2); + }; + } else { + LogUtils.warn("Wrong value format found at > requirement."); + return EmptyRequirement.instance; + } + }); + } + + private void registerRegexRequirement() { + registerRequirement("regex", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("papi", ""); + String v2 = section.getString("regex", ""); + return condition -> PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1).matches(v2); + } else { + LogUtils.warn("Wrong value format found at regex requirement."); + return EmptyRequirement.instance; + } + }); + } + + private void registerNumberEqualRequirement() { + registerRequirement("==", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return Double.parseDouble(p1) == Double.parseDouble(p2); + }; + } else { + LogUtils.warn("Wrong value format found at !startsWith requirement."); + return EmptyRequirement.instance; + } + }); + registerRequirement("!=", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return Double.parseDouble(p1) != Double.parseDouble(p2); + }; + } else { + LogUtils.warn("Wrong value format found at !startsWith requirement."); + return EmptyRequirement.instance; + } + }); + } + + @SuppressWarnings("DuplicatedCode") + private void registerLessThanRequirement() { + registerRequirement("<", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return Double.parseDouble(p1) < Double.parseDouble(p2); + }; + } else { + LogUtils.warn("Wrong value format found at < requirement."); + return EmptyRequirement.instance; + } + }); + registerRequirement("<=", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return Double.parseDouble(p1) <= Double.parseDouble(p2); + }; + } else { + LogUtils.warn("Wrong value format found at <= requirement."); + return EmptyRequirement.instance; + } + }); + } + + private void registerStartWithRequirement() { + registerRequirement("startsWith", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return p1.startsWith(p2); + }; + } else { + LogUtils.warn("Wrong value format found at startsWith requirement."); + return EmptyRequirement.instance; + } + }); + registerRequirement("!startsWith", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return !p1.startsWith(p2); + }; + } else { + LogUtils.warn("Wrong value format found at !startsWith requirement."); + return EmptyRequirement.instance; + } + }); + } + + private void registerEndWithRequirement() { + registerRequirement("endsWith", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return p1.endsWith(p2); + }; + } else { + LogUtils.warn("Wrong value format found at endsWith requirement."); + return EmptyRequirement.instance; + } + }); + registerRequirement("!endsWith", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return !p1.endsWith(p2); + }; + } else { + LogUtils.warn("Wrong value format found at !endsWith requirement."); + return EmptyRequirement.instance; + } + }); + } + + private void registerContainRequirement() { + registerRequirement("contains", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return p1.contains(p2); + }; + } else { + LogUtils.warn("Wrong value format found at contains requirement."); + return EmptyRequirement.instance; + } + }); + registerRequirement("!contains", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return !p1.contains(p2); + }; + } else { + LogUtils.warn("Wrong value format found at !contains requirement."); + return EmptyRequirement.instance; + } + }); + } + + private void registerEqualsRequirement() { + registerRequirement("equals", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return p1.equals(p2); + }; + } else { + LogUtils.warn("Wrong value format found at equals requirement."); + return EmptyRequirement.instance; + } + }); + registerRequirement("!equals", (args) -> { + if (args instanceof ConfigurationSection section) { + String v1 = section.getString("value1", ""); + String v2 = section.getString("value2", ""); + return condition -> { + String p1 = v1.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v1) : v1; + String p2 = v2.startsWith("%") ? PlaceholderAPI.setPlaceholders(condition.getOfflinePlayer(), v2) : v2; + return !p1.equals(p2); + }; + } else { + LogUtils.warn("Wrong value format found at !equals requirement."); + return EmptyRequirement.instance; + } + }); + } + + private void registerEnvironmentRequirement() { + registerRequirement("environment", (args) -> { + List environments = ConfigUtils.stringListArgs(args); + return condition -> { + var name = condition.getOfflinePlayer().getPlayer().getWorld().getEnvironment().name().toLowerCase(Locale.ENGLISH); + return environments.contains(name); + }; + }); + registerRequirement("!environment", (args) -> { + List environments = ConfigUtils.stringListArgs(args); + return condition -> { + var name = condition.getOfflinePlayer().getPlayer().getWorld().getEnvironment().name().toLowerCase(Locale.ENGLISH); + return !environments.contains(name); + }; + }); + } + + private void registerPotionEffectRequirement() { + registerRequirement("potion-effect", (args) -> { + String potions = (String) args; + String[] split = potions.split("(<=|>=|<|>|==)", 2); + PotionEffectType type = PotionEffectType.getByName(split[0]); + if (type == null) { + LogUtils.warn("Potion effect doesn't exist: " + split[0]); + return EmptyRequirement.instance; + } + int required = Integer.parseInt(split[1]); + String operator = potions.substring(split[0].length(), potions.length() - split[1].length()); + return condition -> { + int level = -1; + PotionEffect potionEffect = Objects.requireNonNull(condition.getOfflinePlayer().getPlayer()).getPotionEffect(type); + if (potionEffect != null) { + level = potionEffect.getAmplifier(); + } + boolean result = false; + switch (operator) { + case ">=" -> { + if (level >= required) result = true; + } + case ">" -> { + if (level > required) result = true; + } + case "==" -> { + if (level == required) result = true; + } + case "!=" -> { + if (level != required) result = true; + } + case "<=" -> { + if (level <= required) result = true; + } + case "<" -> { + if (level < required) result = true; + } + } + return result; + }; + }); + } + + /** + * Loads requirement expansions from external JAR files located in the expansion folder. + * Each expansion JAR should contain classes that extends the RequirementExpansion class. + * Expansions are registered and used to create custom requirements. + * If an error occurs while loading or initializing an expansion, a warning message is logged. + */ + @SuppressWarnings("ResultOfMethodCallIgnored") + private void loadExpansions() { + File expansionFolder = new File(plugin.getDataFolder(), EXPANSION_FOLDER); + if (!expansionFolder.exists()) + expansionFolder.mkdirs(); + + List> classes = new ArrayList<>(); + File[] expansionJars = expansionFolder.listFiles(); + if (expansionJars == null) return; + for (File expansionJar : expansionJars) { + if (expansionJar.getName().endsWith(".jar")) { + try { + Class expansionClass = ClassUtils.findClass(expansionJar, RequirementExpansion.class); + classes.add(expansionClass); + } catch (IOException | ClassNotFoundException e) { + LogUtils.warn("Failed to load expansion: " + expansionJar.getName(), e); + } + } + } + try { + for (Class expansionClass : classes) { + RequirementExpansion expansion = expansionClass.getDeclaredConstructor().newInstance(); + unregisterRequirement(expansion.getRequirementType()); + registerRequirement(expansion.getRequirementType(), expansion.getRequirementFactory()); + LogUtils.info("Loaded requirement expansion: " + expansion.getRequirementType() + "[" + expansion.getVersion() + "]" + " by " + expansion.getAuthor()); + } + } catch (InvocationTargetException | InstantiationException | IllegalAccessException | NoSuchMethodException e) { + LogUtils.warn("Error occurred when creating expansion instance.", e); + } + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/Team.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/Team.java new file mode 100644 index 0000000..9edc622 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/Team.java @@ -0,0 +1,6 @@ +package net.momirealms.customnameplates.paper.mechanic.team; + +public interface Team { + + +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/TeamManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/TeamManagerImpl.java new file mode 100644 index 0000000..a0179d0 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/TeamManagerImpl.java @@ -0,0 +1,175 @@ +package net.momirealms.customnameplates.paper.mechanic.team; + +import com.comphenix.protocol.events.PacketContainer; +import com.google.common.io.ByteArrayDataInput; +import com.google.common.io.ByteArrayDataOutput; +import com.google.common.io.ByteStreams; +import net.kyori.adventure.text.Component; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.manager.TeamManager; +import net.momirealms.customnameplates.api.mechanic.team.*; +import net.momirealms.customnameplates.paper.mechanic.misc.PacketManager; +import net.momirealms.customnameplates.paper.mechanic.team.packet.TeamPacketAdaptor; +import net.momirealms.customnameplates.paper.mechanic.team.packet.TeamPacket_1_17; +import net.momirealms.customnameplates.paper.mechanic.team.provider.CMIProvider; +import net.momirealms.customnameplates.paper.mechanic.team.provider.DefaultProvider; +import net.momirealms.customnameplates.paper.mechanic.team.provider.TABProvider; +import net.momirealms.customnameplates.paper.mechanic.team.provider.TeamProvider; +import net.momirealms.customnameplates.paper.setting.CNConfig; +import org.bukkit.Bukkit; +import org.bukkit.entity.Entity; +import org.bukkit.entity.Player; +import org.bukkit.plugin.messaging.PluginMessageListener; +import org.jetbrains.annotations.NotNull; + +import java.util.Collections; +import java.util.Objects; + +public class TeamManagerImpl implements TeamManager, PluginMessageListener { + + private final CustomNameplatesPlugin plugin; + private final TeamPacketAdaptor teamPacketAdaptor; + private final TeamProvider teamProvider; + private static final String CHANNEL = "customnameplates:cnp"; + + public TeamManagerImpl(CustomNameplatesPlugin plugin) { + this.plugin = plugin; + this.teamPacketAdaptor = new TeamPacket_1_17(); + if (CNConfig.tabTeam) { + teamProvider = new TABProvider(); + } else if (CNConfig.cmiTeam) { + teamProvider = new CMIProvider(); + } else { + teamProvider = new DefaultProvider(); + } + } + + @Override + public void createTeam(Player player) { + if (CNConfig.isOtherTeamPluginHooked()) { + return; + } + PacketContainer createOwner = teamPacketAdaptor.getTeamCreatePacket( + TeamCreatePacket.builder() + .teamName(teamProvider.getTeam(player)) + .color(TeamColor.WHITE) + .prefix(Component.text("")) + .suffix(Component.text("")) + .members(Collections.singletonList(player.getName())) + .build() + ); + for (Player online : Bukkit.getOnlinePlayers()) { + PacketManager.getInstance().send(online, createOwner); + if (online == player) continue; + PacketContainer createOther = teamPacketAdaptor.getTeamCreatePacket( + TeamCreatePacket.builder() + .teamName(teamProvider.getTeam(online)) + .color(TeamColor.WHITE) + .prefix(Component.text("")) + .suffix(Component.text("")) + .members(Collections.singletonList(online.getName())) + .build() + ); + PacketManager.getInstance().send(player, createOther); + } + } + + @Override + public void createProxyTeam(Player player) { + + } + + @Override + public void removeTeam(Player player) { + if (CNConfig.isOtherTeamPluginHooked()) return; + PacketContainer packet = teamPacketAdaptor.getTeamRemovePacket( + TeamRemovePacket.builder() + .teamName(teamProvider.getTeam(player)) + .build() + ); + for (Player online : Bukkit.getOnlinePlayers()) { + if (player == online) continue; + PacketManager.getInstance().send(online, packet); + } + } + + @Override + public void removeProxyTeam(Player player) { + this.sendPluginMessage(MessageType.REMOVE, + player.getName() + ); + } + + @Override + public void updateTeam(Player owner, Player viewer, Component prefix, Component suffix, TeamColor color, TeamTagVisibility visibility) { + if (color == TeamColor.NONE || color == TeamColor.CUSTOM) + color = TeamColor.WHITE; + PacketContainer packet = teamPacketAdaptor.getTeamUpdatePacket( + TeamUpdatePacket.builder() + .teamName(teamProvider.getTeam(owner)) + .color(color) + .prefix(prefix) + .suffix(suffix) + .tagVisibility(visibility) + .build() + ); + PacketManager.getInstance().send(viewer, packet); + } + + public void reload() { + unload(); + load(); + } + + public void unload() { + Bukkit.getServer().getMessenger().unregisterIncomingPluginChannel(plugin, CHANNEL); + Bukkit.getServer().getMessenger().unregisterOutgoingPluginChannel(plugin, CHANNEL); + } + + public void load() { + Bukkit.getServer().getMessenger().registerOutgoingPluginChannel(plugin, CHANNEL); + Bukkit.getServer().getMessenger().registerIncomingPluginChannel(plugin, CHANNEL, this); + } + + @Override + public String getTeamName(Player player) { + return null; + } + + private void handleMessage(String... message) { + + } + + @Override + @SuppressWarnings("UnstableApiUsage") + public void onPluginMessageReceived(@NotNull String channel, @NotNull Player player, @NotNull byte[] message) { + if (!Objects.equals(CHANNEL, channel)) { + return; + } + ByteArrayDataInput dataInput = ByteStreams.newDataInput(message); + byte args = dataInput.readByte(); + String[] messages = new String[args]; + for (int i = 0; i < args; i++) { + messages[i] = dataInput.readUTF(); + } + handleMessage(messages); + } + + @SuppressWarnings("UnstableApiUsage") + public void sendPluginMessage(String... messages) { + ByteArrayDataOutput dataOutput = ByteStreams.newDataOutput(); + dataOutput.writeByte(messages.length); + for (String message : messages) { + dataOutput.writeUTF(message); + } + Bukkit.getOnlinePlayers().stream().findAny().ifPresent(player -> { + player.sendPluginMessage(plugin, CHANNEL, dataOutput.toByteArray()); + }); + } + + public static class MessageType { + + public static final String CREATE = "create"; + public static final String REMOVE = "remove"; + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/packet/TeamPacketAdaptor.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/packet/TeamPacketAdaptor.java new file mode 100644 index 0000000..8e38a1b --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/packet/TeamPacketAdaptor.java @@ -0,0 +1,15 @@ +package net.momirealms.customnameplates.paper.mechanic.team.packet; + +import com.comphenix.protocol.events.PacketContainer; +import net.momirealms.customnameplates.api.mechanic.team.TeamCreatePacket; +import net.momirealms.customnameplates.api.mechanic.team.TeamRemovePacket; +import net.momirealms.customnameplates.api.mechanic.team.TeamUpdatePacket; + +public interface TeamPacketAdaptor { + + PacketContainer getTeamCreatePacket(TeamCreatePacket teamCreatePacket); + + PacketContainer getTeamUpdatePacket(TeamUpdatePacket teamUpdatePacket); + + PacketContainer getTeamRemovePacket(TeamRemovePacket teamRemovePacket); +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/packet/TeamPacket_1_17.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/packet/TeamPacket_1_17.java new file mode 100644 index 0000000..6ea3e38 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/packet/TeamPacket_1_17.java @@ -0,0 +1,69 @@ +package net.momirealms.customnameplates.paper.mechanic.team.packet; + +import com.comphenix.protocol.PacketType; +import com.comphenix.protocol.events.InternalStructure; +import com.comphenix.protocol.events.PacketContainer; +import com.comphenix.protocol.utility.MinecraftReflection; +import com.comphenix.protocol.wrappers.WrappedChatComponent; +import net.kyori.adventure.text.serializer.gson.GsonComponentSerializer; +import net.momirealms.customnameplates.api.mechanic.team.TeamColor; +import net.momirealms.customnameplates.api.mechanic.team.TeamCreatePacket; +import net.momirealms.customnameplates.api.mechanic.team.TeamRemovePacket; +import net.momirealms.customnameplates.api.mechanic.team.TeamUpdatePacket; + +import java.util.Optional; + +@SuppressWarnings("DuplicatedCode") +public class TeamPacket_1_17 implements TeamPacketAdaptor { + + @Override + public PacketContainer getTeamCreatePacket(TeamCreatePacket teamCreatePacket) { + PacketContainer packet = new PacketContainer(PacketType.Play.Server.SCOREBOARD_TEAM); + // 0 = create team + packet.getModifier().write(0,0); + packet.getModifier().write(1, teamCreatePacket.getTeamName()); + packet.getModifier().write(2, teamCreatePacket.getMembers()); + Optional optionalInternalStructure = packet.getOptionalStructures().read(0); + if (optionalInternalStructure.isPresent()) { + InternalStructure is = optionalInternalStructure.get(); + // Team + is.getChatComponents().write(0, WrappedChatComponent.fromJson(GsonComponentSerializer.gson().serialize(teamCreatePacket.getTeamDisplay()))); + is.getChatComponents().write(1, WrappedChatComponent.fromJson(GsonComponentSerializer.gson().serialize(teamCreatePacket.getTeamPrefix()))); + is.getChatComponents().write(2, WrappedChatComponent.fromJson(GsonComponentSerializer.gson().serialize(teamCreatePacket.getTeamSuffix()))); + is.getModifier().write(3, teamCreatePacket.getTagVisibility().getId()); + is.getModifier().write(4, teamCreatePacket.getCollisionRule().getId()); + is.getEnumModifier(TeamColor.class, MinecraftReflection.getMinecraftClass("EnumChatFormat")).write(0, teamCreatePacket.getTeamColor()); + is.getModifier().write(6, teamCreatePacket.getMembers().size()); + } + return packet; + } + + @Override + public PacketContainer getTeamRemovePacket(TeamRemovePacket teamRemovePacket) { + PacketContainer packet = new PacketContainer(PacketType.Play.Server.SCOREBOARD_TEAM); + // 1 = remove team + packet.getModifier().write(0,1); + packet.getStrings().write(0, teamRemovePacket.getTeamName()); + return packet; + } + + @Override + public PacketContainer getTeamUpdatePacket(TeamUpdatePacket teamUpdatePacket) { + PacketContainer packet = new PacketContainer(PacketType.Play.Server.SCOREBOARD_TEAM); + // 2 = update team + packet.getModifier().write(0,2); + packet.getModifier().write(1, teamUpdatePacket.getTeamName()); + Optional optionalInternalStructure = packet.getOptionalStructures().read(0); + if (optionalInternalStructure.isPresent()) { + InternalStructure is = optionalInternalStructure.get(); + // Team + is.getChatComponents().write(0, WrappedChatComponent.fromJson(GsonComponentSerializer.gson().serialize(teamUpdatePacket.getTeamDisplay()))); + is.getChatComponents().write(1, WrappedChatComponent.fromJson(GsonComponentSerializer.gson().serialize(teamUpdatePacket.getTeamPrefix()))); + is.getChatComponents().write(2, WrappedChatComponent.fromJson(GsonComponentSerializer.gson().serialize(teamUpdatePacket.getTeamSuffix()))); + is.getModifier().write(3, teamUpdatePacket.getTagVisibility().getId()); + is.getModifier().write(4, teamUpdatePacket.getCollisionRule().getId()); + is.getEnumModifier(TeamColor.class, MinecraftReflection.getMinecraftClass("EnumChatFormat")).write(0, teamUpdatePacket.getTeamColor()); + } + return packet; + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/CMIProvider.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/CMIProvider.java new file mode 100644 index 0000000..fbadf01 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/CMIProvider.java @@ -0,0 +1,15 @@ +package net.momirealms.customnameplates.paper.mechanic.team.provider; + +import com.Zrips.CMI.CMI; +import org.bukkit.entity.Player; +import org.bukkit.scoreboard.Team; + +public class CMIProvider implements TeamProvider { + + @Override + public String getTeam(Player player) { + Team team = CMI.getInstance().getSB().getPlayerTeam(player); + if (team == null) return player.getName(); + return team.getName(); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/DefaultProvider.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/DefaultProvider.java new file mode 100644 index 0000000..1823509 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/DefaultProvider.java @@ -0,0 +1,11 @@ +package net.momirealms.customnameplates.paper.mechanic.team.provider; + +import org.bukkit.entity.Player; + +public class DefaultProvider implements TeamProvider { + + @Override + public String getTeam(Player player) { + return player.getName(); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/TABProvider.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/TABProvider.java new file mode 100644 index 0000000..9619e52 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/TABProvider.java @@ -0,0 +1,26 @@ +package net.momirealms.customnameplates.paper.mechanic.team.provider; + +import me.neznamy.tab.api.TabAPI; +import me.neznamy.tab.api.TabPlayer; +import me.neznamy.tab.api.tablist.SortingManager; +import net.momirealms.customnameplates.api.util.LogUtils; +import org.bukkit.entity.Player; + +public class TABProvider implements TeamProvider { + + private final SortingManager sortingManager; + + public TABProvider() { + this.sortingManager = TabAPI.getInstance().getSortingManager(); + if (sortingManager == null) { + LogUtils.warn("Detected that team management is disabled in TAB. Using player name as team name."); + } + } + + @Override + public String getTeam(Player player) { + TabPlayer tabPlayer = TabAPI.getInstance().getPlayer(player.getUniqueId()); + if (tabPlayer == null || sortingManager == null) return player.getName(); + return sortingManager.getOriginalTeamName(tabPlayer); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/TeamProvider.java b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/TeamProvider.java new file mode 100644 index 0000000..fe14d56 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/mechanic/team/provider/TeamProvider.java @@ -0,0 +1,8 @@ +package net.momirealms.customnameplates.paper.mechanic.team.provider; + +import org.bukkit.entity.Player; + +public interface TeamProvider { + + String getTeam(Player player); +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/BukkitSchedulerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/BukkitSchedulerImpl.java new file mode 100644 index 0000000..bce03bf --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/BukkitSchedulerImpl.java @@ -0,0 +1,100 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.scheduler; + +import net.momirealms.customnameplates.api.scheduler.CancellableTask; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; +import org.bukkit.Bukkit; +import org.bukkit.Location; +import org.bukkit.scheduler.BukkitTask; + +/** + * A scheduler implementation for synchronous tasks using Bukkit's Scheduler. + */ +public class BukkitSchedulerImpl implements SyncScheduler { + + private final CustomNameplatesPluginImpl plugin; + + public BukkitSchedulerImpl(CustomNameplatesPluginImpl plugin) { + this.plugin = plugin; + } + + /** + * Runs a synchronous task on the main server thread using Bukkit's Scheduler. + * If already on the main thread, the task is executed immediately. + * + * @param runnable The task to run. + * @param location The location associated with the task. + */ + @Override + public void runSyncTask(Runnable runnable, Location location) { + if (Bukkit.isPrimaryThread()) + runnable.run(); + else + Bukkit.getScheduler().runTask(plugin, runnable); + } + + /** + * Runs a synchronous task repeatedly with a specified delay and period using Bukkit's Scheduler. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delay The delay in ticks before the first execution. + * @param period The period between subsequent executions in ticks. + * @return A CancellableTask for managing the scheduled task. + */ + @Override + public CancellableTask runTaskSyncTimer(Runnable runnable, Location location, long delay, long period) { + return new BukkitCancellableTask(Bukkit.getScheduler().runTaskTimer(plugin, runnable, delay, period)); + } + + /** + * Runs a synchronous task with a specified delay using Bukkit's Scheduler. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delay The delay in ticks before the task execution. + * @return A CancellableTask for managing the scheduled task. + */ + @Override + public CancellableTask runTaskSyncLater(Runnable runnable, Location location, long delay) { + return new BukkitCancellableTask(Bukkit.getScheduler().runTaskLater(plugin, runnable, delay)); + } + + /** + * Represents a scheduled task using Bukkit's Scheduler that can be cancelled. + */ + public static class BukkitCancellableTask implements CancellableTask { + + private final BukkitTask bukkitTask; + + public BukkitCancellableTask(BukkitTask bukkitTask) { + this.bukkitTask = bukkitTask; + } + + @Override + public void cancel() { + this.bukkitTask.cancel(); + } + + @Override + public boolean isCancelled() { + return this.bukkitTask.isCancelled(); + } + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/FoliaSchedulerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/FoliaSchedulerImpl.java new file mode 100644 index 0000000..15946dd --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/FoliaSchedulerImpl.java @@ -0,0 +1,96 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.scheduler; + +import io.papermc.paper.threadedregions.scheduler.ScheduledTask; +import net.momirealms.customnameplates.api.scheduler.CancellableTask; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; +import org.bukkit.Bukkit; +import org.bukkit.Location; + +/** + * A scheduler implementation for "synchronous" tasks using Folia's RegionScheduler. + */ +public class FoliaSchedulerImpl implements SyncScheduler { + + private final CustomNameplatesPluginImpl plugin; + + public FoliaSchedulerImpl(CustomNameplatesPluginImpl plugin) { + this.plugin = plugin; + } + + /** + * Runs a "synchronous" task on the region thread using Folia's RegionScheduler. + * + * @param runnable The task to run. + * @param location The location associated with the task. + */ + @Override + public void runSyncTask(Runnable runnable, Location location) { + Bukkit.getRegionScheduler().execute(plugin, location, runnable); + } + + /** + * Runs a "synchronous" task repeatedly with a specified delay and period using Folia's RegionScheduler. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delay The delay in ticks before the first execution. + * @param period The period between subsequent executions in ticks. + * @return A CancellableTask for managing the scheduled task. + */ + @Override + public CancellableTask runTaskSyncTimer(Runnable runnable, Location location, long delay, long period) { + return new FoliaCancellableTask(Bukkit.getRegionScheduler().runAtFixedRate(plugin, location, (scheduledTask -> runnable.run()), delay, period)); + } + + /** + * Runs a "synchronous" task with a specified delay using Folia's RegionScheduler. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delay The delay in ticks before the task execution. + * @return A CancellableTask for managing the scheduled task. + */ + @Override + public CancellableTask runTaskSyncLater(Runnable runnable, Location location, long delay) { + return new FoliaCancellableTask(Bukkit.getRegionScheduler().runDelayed(plugin, location, (scheduledTask -> runnable.run()), delay)); + } + + /** + * Represents a scheduled task using Folia's RegionScheduler that can be cancelled. + */ + public static class FoliaCancellableTask implements CancellableTask { + + private final ScheduledTask scheduledTask; + + public FoliaCancellableTask(ScheduledTask scheduledTask) { + this.scheduledTask = scheduledTask; + } + + @Override + public void cancel() { + this.scheduledTask.cancel(); + } + + @Override + public boolean isCancelled() { + return this.scheduledTask.isCancelled(); + } + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/SchedulerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/SchedulerImpl.java new file mode 100644 index 0000000..77a849a --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/SchedulerImpl.java @@ -0,0 +1,183 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.scheduler; + +import net.momirealms.customnameplates.api.scheduler.CancellableTask; +import net.momirealms.customnameplates.api.scheduler.Scheduler; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; +import net.momirealms.customnameplates.paper.setting.CNConfig; +import org.bukkit.Location; + +import java.util.concurrent.ScheduledFuture; +import java.util.concurrent.ScheduledThreadPoolExecutor; +import java.util.concurrent.ThreadPoolExecutor; +import java.util.concurrent.TimeUnit; + +/** + * A scheduler implementation responsible for scheduling and managing tasks in a multi-threaded environment. + */ +public class SchedulerImpl implements Scheduler { + + private final SyncScheduler syncScheduler; + private final ScheduledThreadPoolExecutor schedule; + private final CustomNameplatesPluginImpl plugin; + + public SchedulerImpl(CustomNameplatesPluginImpl plugin) { + this.plugin = plugin; + this.syncScheduler = plugin.getVersionManager().isFolia() ? + new FoliaSchedulerImpl(plugin) : new BukkitSchedulerImpl(plugin); + this.schedule = new ScheduledThreadPoolExecutor(1); + this.schedule.setMaximumPoolSize(1); + this.schedule.setKeepAliveTime(30, TimeUnit.SECONDS); + this.schedule.setRejectedExecutionHandler(new ThreadPoolExecutor.DiscardOldestPolicy()); + } + + /** + * Reloads the scheduler configuration based on CustomFishingPlugin settings. + */ + public void reload() { + try { + this.schedule.setMaximumPoolSize(CNConfig.maximumPoolSize); + this.schedule.setCorePoolSize(CNConfig.corePoolSize); + this.schedule.setKeepAliveTime(CNConfig.keepAliveTime, TimeUnit.SECONDS); + } catch (IllegalArgumentException e) { + plugin.getLogger().severe("Failed to create thread pool. Please lower the corePoolSize in config.yml."); + } + } + + /** + * Shuts down the scheduler. + */ + public void shutdown() { + if (this.schedule != null && !this.schedule.isShutdown()) + this.schedule.shutdown(); + } + + /** + * Runs a task synchronously on the main server thread or region thread. + * + * @param runnable The task to run. + * @param location The location associated with the task. + */ + @Override + public void runTaskSync(Runnable runnable, Location location) { + this.syncScheduler.runSyncTask(runnable, location); + } + + /** + * Runs a task asynchronously. + * + * @param runnable The task to run. + */ + @Override + public void runTaskAsync(Runnable runnable) { + this.schedule.execute(runnable); + } + + /** + * Runs a task synchronously with a specified delay and period. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delayTicks The delay in ticks before the first execution. + * @param periodTicks The period between subsequent executions in ticks. + * @return A CancellableTask for managing the scheduled task. + */ + @Override + public CancellableTask runTaskSyncTimer(Runnable runnable, Location location, long delayTicks, long periodTicks) { + return this.syncScheduler.runTaskSyncTimer(runnable, location, delayTicks, periodTicks); + } + + /** + * Runs a task asynchronously with a specified delay. + * + * @param runnable The task to run. + * @param delay The delay before the task execution. + * @param timeUnit The time unit for the delay. + * @return A CancellableTask for managing the scheduled task. + */ + @Override + public CancellableTask runTaskAsyncLater(Runnable runnable, long delay, TimeUnit timeUnit) { + return new ScheduledTask(schedule.schedule(runnable, delay, timeUnit)); + } + + /** + * Runs a task synchronously with a specified delay. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delay The delay before the task execution. + * @param timeUnit The time unit for the delay. + * @return A CancellableTask for managing the scheduled task. + */ + @Override + public CancellableTask runTaskSyncLater(Runnable runnable, Location location, long delay, TimeUnit timeUnit) { + return new ScheduledTask(schedule.schedule(() -> { + runTaskSync(runnable, location); + }, delay, timeUnit)); + } + + /** + * Runs a task synchronously with a specified delay in ticks. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delayTicks The delay in ticks before the task execution. + * @return A CancellableTask for managing the scheduled task. + */ + @Override + public CancellableTask runTaskSyncLater(Runnable runnable, Location location, long delayTicks) { + return this.syncScheduler.runTaskSyncLater(runnable, location, delayTicks); + } + + /** + * Runs a task asynchronously with a specified delay and period. + * + * @param runnable The task to run. + * @param delay The delay before the first execution. + * @param period The period between subsequent executions. + * @param timeUnit The time unit for the delay and period. + * @return A CancellableTask for managing the scheduled task. + */ + @Override + public CancellableTask runTaskAsyncTimer(Runnable runnable, long delay, long period, TimeUnit timeUnit) { + return new ScheduledTask(schedule.scheduleAtFixedRate(runnable, delay, period, timeUnit)); + } + + /** + * Represents a thread-pool task that can be cancelled. + */ + public static class ScheduledTask implements CancellableTask { + + private final ScheduledFuture scheduledFuture; + + public ScheduledTask(ScheduledFuture scheduledFuture) { + this.scheduledFuture = scheduledFuture; + } + + @Override + public void cancel() { + this.scheduledFuture.cancel(false); + } + + @Override + public boolean isCancelled() { + return this.scheduledFuture.isCancelled(); + } + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/SyncScheduler.java b/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/SyncScheduler.java new file mode 100644 index 0000000..b6a5864 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/scheduler/SyncScheduler.java @@ -0,0 +1,53 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.scheduler; + +import net.momirealms.customnameplates.api.scheduler.CancellableTask; +import org.bukkit.Location; + +public interface SyncScheduler { + + /** + * Runs a task synchronously on the main server thread or region thread. + * + * @param runnable The task to run. + * @param location The location associated with the task. + */ + void runSyncTask(Runnable runnable, Location location); + + /** + * Runs a task synchronously with a specified delay and period. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delayTicks The delay in ticks before the first execution. + * @param periodTicks The period between subsequent executions in ticks. + * @return A CancellableTask for managing the scheduled task. + */ + CancellableTask runTaskSyncTimer(Runnable runnable, Location location, long delayTicks, long periodTicks); + + /** + * Runs a task synchronously with a specified delay in ticks. + * + * @param runnable The task to run. + * @param location The location associated with the task. + * @param delayTicks The delay in ticks before the task execution. + * @return A CancellableTask for managing the scheduled task. + */ + CancellableTask runTaskSyncLater(Runnable runnable, Location location, long delayTicks); +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/setting/CNConfig.java b/paper/src/main/java/net/momirealms/customnameplates/paper/setting/CNConfig.java new file mode 100644 index 0000000..e3f794e --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/setting/CNConfig.java @@ -0,0 +1,132 @@ +package net.momirealms.customnameplates.paper.setting; + +import dev.dejvokep.boostedyaml.YamlDocument; +import dev.dejvokep.boostedyaml.dvs.versioning.BasicVersioning; +import dev.dejvokep.boostedyaml.settings.dumper.DumperSettings; +import dev.dejvokep.boostedyaml.settings.general.GeneralSettings; +import dev.dejvokep.boostedyaml.settings.loader.LoaderSettings; +import dev.dejvokep.boostedyaml.settings.updater.UpdaterSettings; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.mechanic.character.CharacterArranger; +import net.momirealms.customnameplates.api.util.FontUtils; +import net.momirealms.customnameplates.api.util.LogUtils; +import org.bukkit.configuration.ConfigurationSection; +import org.bukkit.configuration.file.YamlConfiguration; + +import java.io.File; +import java.io.IOException; +import java.util.Objects; + +public class CNConfig { + + public static String configVersion = "22"; + public static int corePoolSize; + public static long keepAliveTime; + public static int maximumPoolSize; + public static boolean debug; + public static String language; + public static boolean updateChecker; + public static boolean metrics; + public static boolean legacyColorSupport; + public static boolean generatePackOnStart; + public static int sendDelay; + public static String namespace; + public static String font; + public static char initChar; + public static boolean copyPackIA; + public static boolean copyPackOraxen; + public static boolean trChatChannel; + public static boolean ventureChatChannel; + public static boolean nameplateModule; + public static boolean bossBarModule; + public static boolean actionBarModule; + public static boolean bubbleModule; + public static boolean backgroundModule; + public static boolean imageModule; + public static boolean tabTeam; + public static boolean cmiTeam; + public static String folderNameplate; + public static String folderImage; + public static String folderBubble; + public static String folderBackground; + public static String folderSplit; + public static boolean add_1_20_Unicodes; + public static short defaultCharWidth; + + public static void load() { + try { + YamlDocument.create( + new File(CustomNameplatesPlugin.getInstance().getDataFolder(), "config.yml"), + Objects.requireNonNull(CustomNameplatesPlugin.getInstance().getResource("config.yml")), + GeneralSettings.DEFAULT, + LoaderSettings + .builder() + .setAutoUpdate(true) + .build(), + DumperSettings.DEFAULT, + UpdaterSettings + .builder() + .setVersioning(new BasicVersioning("config-version")) + .build() + ); + loadSettings(CustomNameplatesPlugin.getInstance().getConfig("config.yml")); + } catch (IOException e) { + LogUtils.warn(e.getMessage()); + } + } + + private static void loadSettings(YamlConfiguration config) { + debug = config.getBoolean("debug", false); + + language = config.getString("lang", "english"); + updateChecker = config.getBoolean("update-checker", true); + metrics = config.getBoolean("metrics"); + + ConfigurationSection moduleSection = config.getConfigurationSection("modules"); + if (moduleSection != null) { + nameplateModule = moduleSection.getBoolean("nameplates"); + bossBarModule = moduleSection.getBoolean("bossbars"); + actionBarModule = moduleSection.getBoolean("actionbars"); + bubbleModule = moduleSection.getBoolean("bubbles"); + backgroundModule = moduleSection.getBoolean("backgrounds"); + imageModule = moduleSection.getBoolean("images"); + } + + ConfigurationSection integrationSection = config.getConfigurationSection("integrations"); + if (integrationSection != null) { + copyPackIA = integrationSection.getBoolean("resource-pack.ItemsAdder", false); + copyPackOraxen = integrationSection.getBoolean("resource-pack.Oraxen", false); + trChatChannel = integrationSection.getBoolean("chat.TrChat", false); + ventureChatChannel = integrationSection.getBoolean("chat.VentureChat", false); + tabTeam = integrationSection.getBoolean("team.TAB", false); + cmiTeam = integrationSection.getBoolean("team.CMI", false); + } + + ConfigurationSection packSection = config.getConfigurationSection("resource-pack"); + if (packSection != null) { + generatePackOnStart = !packSection.getBoolean("disable-generation-on-start", false); + namespace = packSection.getString("namespace", "nameplates"); + font = packSection.getString("font", "default"); + FontUtils.setNameSpaceAndFont(namespace, font); + + initChar = packSection.getString("initial-char", "뀁").charAt(0); + CharacterArranger.reset(initChar); + + folderNameplate = packSection.getString("image-path.nameplates","font\\nameplates\\"); + folderBubble = packSection.getString("image-path.bubbles","font\\bubbles\\"); + folderBackground = packSection.getString("image-path.backgrounds","font\\backgrounds\\"); + folderImage = packSection.getString("image-path.images","font\\images\\"); + folderSplit = packSection.getString("image-path.space-split","font\\base\\"); + + add_1_20_Unicodes = packSection.getBoolean("support-1_20-unicodes", true); + } + + corePoolSize = config.getInt("other-settings.thread-pool-settings.corePoolSize", 10); + maximumPoolSize = config.getInt("other-settings.thread-pool-settings.maximumPoolSize", 10); + keepAliveTime = config.getInt("other-settings.thread-pool-settings.keepAliveTime", 30); + } + + public static boolean isOtherTeamPluginHooked() { + return tabTeam || cmiTeam; + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/setting/CNLocale.java b/paper/src/main/java/net/momirealms/customnameplates/paper/setting/CNLocale.java new file mode 100644 index 0000000..ebd1168 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/setting/CNLocale.java @@ -0,0 +1,95 @@ +package net.momirealms.customnameplates.paper.setting; + +import dev.dejvokep.boostedyaml.YamlDocument; +import dev.dejvokep.boostedyaml.dvs.versioning.BasicVersioning; +import dev.dejvokep.boostedyaml.settings.dumper.DumperSettings; +import dev.dejvokep.boostedyaml.settings.general.GeneralSettings; +import dev.dejvokep.boostedyaml.settings.loader.LoaderSettings; +import dev.dejvokep.boostedyaml.settings.updater.UpdaterSettings; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.util.LogUtils; +import org.bukkit.configuration.ConfigurationSection; +import org.bukkit.configuration.file.YamlConfiguration; + +import java.io.File; +import java.io.IOException; +import java.util.Objects; + +public class CNLocale { + + public static String MSG_RELOAD; + public static String MSG_PREFIX; + public static String MSG_PREVIEW_COOLDOWN; + public static String MSG_PREVIEW_START; + public static String generate; + public static String noNameplate; + public static String MSG_PACK_GENERATION; + public static String MSG_EQUIP_NAMEPLATE; + public static String MSG_UNEQUIP_NAMEPLATE; + public static String MSG_FORCE_EQUIP_NAMEPLATE; + public static String np_force_unEquip; + public static String MSG_NAMEPLATE_NOT_EXISTS; + public static String MSG_NAMEPLATE_NOT_AVAILABLE; + public static String np_available; + public static String np_haveNone; + public static String bb_equip; + public static String bb_unEquip; + public static String bb_force_equip; + public static String bb_force_unEquip; + public static String bb_not_exist; + public static String bb_notAvailable; + public static String bb_available; + public static String bb_haveNone; + + public static void load() { + try { + YamlDocument.create( + new File(CustomNameplatesPlugin.getInstance().getDataFolder(), "messages/" + CNConfig.language + ".yml"), + Objects.requireNonNull(CustomNameplatesPlugin.getInstance().getResource("messages/" + CNConfig.language + ".yml")), + GeneralSettings.DEFAULT, + LoaderSettings + .builder() + .setAutoUpdate(true) + .build(), + DumperSettings.DEFAULT, + UpdaterSettings + .builder() + .setVersioning(new BasicVersioning("config-version")) + .build() + ); + } catch (IOException e) { + LogUtils.warn(e.getMessage()); + } + loadSettings(CustomNameplatesPlugin.get().getConfig("messages/" + CNConfig.language + ".yml")); + } + + public static void loadSettings(YamlConfiguration config) { + ConfigurationSection section = config.getConfigurationSection("messages"); + if (section != null) { + MSG_RELOAD = section.getString("reload"); + MSG_PREFIX = section.getString("prefix"); + MSG_PREVIEW_COOLDOWN = config.getString("messages.cooldown"); + MSG_PREVIEW_START = config.getString("messages.preview"); + generate = config.getString("messages.generate"); + MSG_PACK_GENERATION = config.getString("messages.generate-done"); + noNameplate = config.getString("messages.no-nameplate"); + MSG_EQUIP_NAMEPLATE = config.getString("messages.equip-nameplates"); + MSG_UNEQUIP_NAMEPLATE = config.getString("messages.unequip-nameplates"); + MSG_FORCE_EQUIP_NAMEPLATE = config.getString("messages.force-equip-nameplates"); + np_force_unEquip = config.getString("messages.force-unequip-nameplates"); + MSG_NAMEPLATE_NOT_EXISTS = config.getString("messages.not-exist-nameplates"); + MSG_NAMEPLATE_NOT_AVAILABLE = config.getString("messages.not-available-nameplates"); + np_available = config.getString("messages.available-nameplates"); + np_haveNone = config.getString("messages.have-no-nameplates"); + bb_equip = config.getString("messages.equip-bubbles"); + bb_unEquip = config.getString("messages.unequip-bubbles"); + bb_force_equip = config.getString("messages.force-equip-bubbles"); + bb_force_unEquip = config.getString("messages.force-unequip-bubbles"); + bb_not_exist = config.getString("messages.not-exist-bubbles"); + bb_notAvailable = config.getString("messages.not-available-bubbles"); + bb_available = config.getString("messages.available-bubbles"); + bb_haveNone = config.getString("messages.have-no-bubbles"); + } + + } +} diff --git a/src/main/java/net/momirealms/customnameplates/data/DataStorageInterface.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/DataStorageInterface.java similarity index 60% rename from src/main/java/net/momirealms/customnameplates/data/DataStorageInterface.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/storage/DataStorageInterface.java index ec0d8ba..9aed2f8 100644 --- a/src/main/java/net/momirealms/customnameplates/data/DataStorageInterface.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/DataStorageInterface.java @@ -15,14 +15,34 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.data; +package net.momirealms.customnameplates.paper.storage; +import net.momirealms.customnameplates.api.data.PlayerData; + +import java.util.Optional; import java.util.UUID; +import java.util.concurrent.CompletableFuture; public interface DataStorageInterface { + + /** + * Initialize the data resource + */ void initialize(); + + /** + * Close the data resource + */ void disable(); - PlayerData loadData(UUID uuid); - void saveData(PlayerData playerData); + + /** + * Get the storage data source type + * + * @return {@link StorageType} + */ StorageType getStorageType(); + + CompletableFuture> getPlayerData(UUID uuid); + + CompletableFuture updatePlayerData(UUID uuid, PlayerData playerData); } diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/storage/StorageManagerImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/StorageManagerImpl.java new file mode 100644 index 0000000..26c9bd0 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/StorageManagerImpl.java @@ -0,0 +1,239 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.storage; + +import com.google.gson.Gson; +import com.google.gson.GsonBuilder; +import com.google.gson.JsonSyntaxException; +import net.momirealms.customnameplates.api.data.OnlineUser; +import net.momirealms.customnameplates.api.data.PlayerData; +import net.momirealms.customnameplates.api.event.NameplateDataLoadEvent; +import net.momirealms.customnameplates.api.manager.StorageManager; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; +import net.momirealms.customnameplates.paper.storage.method.database.nosql.MongoDBImpl; +import net.momirealms.customnameplates.paper.storage.method.database.nosql.RedisManager; +import net.momirealms.customnameplates.paper.storage.method.database.sql.H2Impl; +import net.momirealms.customnameplates.paper.storage.method.database.sql.MariaDBImpl; +import net.momirealms.customnameplates.paper.storage.method.database.sql.MySQLImpl; +import net.momirealms.customnameplates.paper.storage.method.database.sql.SQLiteImpl; +import net.momirealms.customnameplates.paper.storage.method.file.JsonImpl; +import net.momirealms.customnameplates.paper.storage.method.file.YAMLImpl; +import org.bukkit.Bukkit; +import org.bukkit.configuration.file.YamlConfiguration; +import org.bukkit.event.EventHandler; +import org.bukkit.event.HandlerList; +import org.bukkit.event.Listener; +import org.bukkit.event.player.PlayerJoinEvent; +import org.bukkit.event.player.PlayerQuitEvent; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.nio.charset.StandardCharsets; +import java.util.Collection; +import java.util.Optional; +import java.util.UUID; +import java.util.concurrent.CompletableFuture; +import java.util.concurrent.ConcurrentHashMap; + +/** + * This class implements the StorageManager interface and is responsible for managing player data storage. + * It includes methods to handle player data retrieval, storage, and serialization. + */ +public class StorageManagerImpl implements Listener, StorageManager { + + private final CustomNameplatesPluginImpl plugin; + private DataStorageInterface dataSource; + private StorageType previousType; + private boolean hasRedis; + private RedisManager redisManager; + private final Gson gson; + private final ConcurrentHashMap onlineUserMap; + + public StorageManagerImpl(CustomNameplatesPluginImpl plugin) { + this.plugin = plugin; + this.gson = new GsonBuilder().create(); + this.onlineUserMap = new ConcurrentHashMap<>(64); + Bukkit.getPluginManager().registerEvents(this, plugin); + } + + @EventHandler + public void onPlayerJoin(PlayerJoinEvent event) { + var uuid = event.getPlayer().getUniqueId(); + this.getPlayerData(uuid).thenAccept(optData -> { + if (optData.isPresent()) { + var playerData = optData.get(); + var player = Bukkit.getPlayer(uuid); + if (player == null || !player.isOnline()) return; + var onlineUser = new OnlineUser(player, playerData); + this.putOnlineUserInMap(onlineUser); + NameplateDataLoadEvent syncEvent = new NameplateDataLoadEvent(uuid, onlineUser); + plugin.getServer().getPluginManager().callEvent(syncEvent); + } + }); + } + + @Override + public Collection getOnlineUsers() { + return onlineUserMap.values(); + } + + @EventHandler + public void onPlayerQuit(PlayerQuitEvent event) { + this.removeOnlineUserFromMap(event.getPlayer().getUniqueId()); + } + + public void putOnlineUserInMap(OnlineUser onlineUser) { + onlineUserMap.put(onlineUser.getUUID(), onlineUser); + } + + public void removeOnlineUserFromMap(UUID uuid) { + onlineUserMap.remove(uuid); + } + + @Override + public CompletableFuture> getPlayerData(UUID uuid) { + OnlineUser onlineUser = onlineUserMap.get(uuid); + if (onlineUser != null) { + return CompletableFuture.completedFuture(Optional.of(onlineUser.toPlayerData())); + } + return dataSource.getPlayerData(uuid); + } + + @Override + public CompletableFuture saveOnlinePlayerData(UUID uuid) { + OnlineUser onlineUser = onlineUserMap.get(uuid); + if (onlineUser == null) { + return CompletableFuture.completedFuture(false); + } + return dataSource.updatePlayerData(uuid, onlineUser.toPlayerData()); + } + + @Override + public CompletableFuture savePlayerData(UUID uuid, PlayerData playerData) { + return dataSource.updatePlayerData(uuid, playerData); + } + + @Override + public Optional getOnlineUser(UUID uuid) { + return Optional.ofNullable(onlineUserMap.get(uuid)); + } + + /** + * Reloads the storage manager configuration. + */ + public void reload() { + YamlConfiguration config = plugin.getConfig("database.yml"); + + // Check if storage type has changed and reinitialize if necessary + StorageType storageType = StorageType.valueOf(config.getString("data-storage-method", "H2")); + if (storageType != previousType) { + if (this.dataSource != null) this.dataSource.disable(); + this.previousType = storageType; + switch (storageType) { + case H2 -> this.dataSource = new H2Impl(plugin); + case JSON -> this.dataSource = new JsonImpl(plugin); + case YAML -> this.dataSource = new YAMLImpl(plugin); + case SQLite -> this.dataSource = new SQLiteImpl(plugin); + case MySQL -> this.dataSource = new MySQLImpl(plugin); + case MariaDB -> this.dataSource = new MariaDBImpl(plugin); + case MongoDB -> this.dataSource = new MongoDBImpl(plugin); + } + if (this.dataSource != null) this.dataSource.initialize(); + else LogUtils.severe("No storage type is set."); + } + + // Handle Redis configuration + if (!this.hasRedis && config.getBoolean("Redis.enable", false)) { + this.hasRedis = true; + this.redisManager = new RedisManager(plugin); + this.redisManager.initialize(); + } + + // Disable Redis if it was enabled but is now disabled + if (this.hasRedis && !config.getBoolean("Redis.enable", false) && this.redisManager != null) { + this.redisManager.disable(); + this.redisManager = null; + } + } + + /** + * Disables the storage manager and cleans up resources. + */ + public void disable() { + HandlerList.unregisterAll(this); + if (this.dataSource != null) + this.dataSource.disable(); + if (this.redisManager != null) + this.redisManager.disable(); + } + + /** + * Checks if Redis is enabled. + * + * @return True if Redis is enabled; otherwise, false. + */ + public boolean isRedisEnabled() { + return hasRedis; + } + + /** + * Gets the RedisManager instance. + * + * @return The RedisManager instance. + */ + @Nullable + public RedisManager getRedisManager() { + return redisManager; + } + + @NotNull + @Override + public PlayerData fromJson(String json) { + return gson.fromJson(json, PlayerData.class); + } + + @Override + public PlayerData fromBytes(byte[] data) { + try { + return gson.fromJson(new String(data, StandardCharsets.UTF_8), PlayerData.class); + } catch (JsonSyntaxException e) { + throw new DataSerializationException("Failed to get PlayerData from bytes", e); + } + } + + @Override + public byte[] toBytes(PlayerData playerData) { + return toJson(playerData).getBytes(StandardCharsets.UTF_8); + } + + @Override + @NotNull + public String toJson(@NotNull PlayerData playerData) { + return gson.toJson(playerData); + } + + /** + * Custom exception class for data serialization errors. + */ + public static class DataSerializationException extends RuntimeException { + protected DataSerializationException(String message, Throwable cause) { + super(message, cause); + } + } +} diff --git a/src/main/java/net/momirealms/customnameplates/data/StorageType.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/StorageType.java similarity index 83% rename from src/main/java/net/momirealms/customnameplates/data/StorageType.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/storage/StorageType.java index 70b929e..1490a1e 100644 --- a/src/main/java/net/momirealms/customnameplates/data/StorageType.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/StorageType.java @@ -15,9 +15,16 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.data; +package net.momirealms.customnameplates.paper.storage; public enum StorageType { - SQL, - YAML + + JSON, + YAML, + H2, + SQLite, + MySQL, + MariaDB, + MongoDB, + Redis } diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/AbstractStorage.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/AbstractStorage.java new file mode 100644 index 0000000..ef04144 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/AbstractStorage.java @@ -0,0 +1,43 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.storage.method; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.storage.DataStorageInterface; + +/** + * An abstract class that implements the DataStorageInterface and provides common functionality for data storage. + */ +public abstract class AbstractStorage implements DataStorageInterface { + + protected CustomNameplatesPlugin plugin; + + public AbstractStorage(CustomNameplatesPlugin plugin) { + this.plugin = plugin; + } + + @Override + public void initialize() { + // This method can be overridden in subclasses to perform initialization tasks specific to the storage type. + } + + @Override + public void disable() { + // This method can be overridden in subclasses to perform cleanup or shutdown tasks specific to the storage type. + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/nosql/MongoDBImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/nosql/MongoDBImpl.java new file mode 100644 index 0000000..7de6d2e --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/nosql/MongoDBImpl.java @@ -0,0 +1,171 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.storage.method.database.nosql; + +import com.mongodb.*; +import com.mongodb.client.MongoClient; +import com.mongodb.client.MongoClients; +import com.mongodb.client.MongoCollection; +import com.mongodb.client.MongoDatabase; +import com.mongodb.client.model.Filters; +import com.mongodb.client.model.UpdateOptions; +import com.mongodb.client.model.Updates; +import com.mongodb.client.result.UpdateResult; +import net.momirealms.customnameplates.api.data.PlayerData; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; +import net.momirealms.customnameplates.paper.storage.StorageType; +import net.momirealms.customnameplates.paper.storage.method.AbstractStorage; +import org.bson.Document; +import org.bson.UuidRepresentation; +import org.bson.conversions.Bson; +import org.bson.types.Binary; +import org.bukkit.Bukkit; +import org.bukkit.configuration.ConfigurationSection; +import org.bukkit.configuration.file.YamlConfiguration; + +import java.util.Collections; +import java.util.Optional; +import java.util.UUID; +import java.util.concurrent.CompletableFuture; + +/** + * An implementation of AbstractStorage that uses MongoDB for player data storage. + */ +public class MongoDBImpl extends AbstractStorage { + + private MongoClient mongoClient; + private MongoDatabase database; + private String collectionPrefix; + + public MongoDBImpl(CustomNameplatesPluginImpl plugin) { + super(plugin); + } + + /** + * Initialize the MongoDB connection and configuration based on the plugin's YAML configuration. + */ + @Override + public void initialize() { + YamlConfiguration config = plugin.getConfig("database.yml"); + ConfigurationSection section = config.getConfigurationSection("MongoDB"); + if (section == null) { + LogUtils.warn("Failed to load database config. It seems that your config is broken. Please regenerate a new one."); + return; + } + + collectionPrefix = section.getString("collection-prefix", "nameplates"); + var settings = MongoClientSettings.builder().uuidRepresentation(UuidRepresentation.STANDARD); + if (!section.getString("connection-uri", "").equals("")) { + settings.applyConnectionString(new ConnectionString(section.getString("connection-uri", ""))); + mongoClient = MongoClients.create(settings.build()); + return; + } + + if (section.contains("user")) { + MongoCredential credential = MongoCredential.createCredential( + section.getString("user", "root"), + section.getString("database", "minecraft"), + section.getString("password", "password").toCharArray() + ); + settings.credential(credential); + } + + settings.applyToClusterSettings(builder -> builder.hosts(Collections.singletonList(new ServerAddress( + section.getString("host", "localhost"), + section.getInt("port", 27017) + )))); + this.mongoClient = MongoClients.create(settings.build()); + this.database = mongoClient.getDatabase(section.getString("database", "minecraft")); + } + + /** + * Disable the MongoDB connection by closing the MongoClient. + */ + @Override + public void disable() { + if (this.mongoClient != null) { + this.mongoClient.close(); + } + } + + @Override + public CompletableFuture updatePlayerData(UUID uuid, PlayerData playerData) { + var future = new CompletableFuture(); + plugin.getScheduler().runTaskAsync(() -> { + MongoCollection collection = database.getCollection(getCollectionName("data")); + try { + Document query = new Document("uuid", uuid); + Bson updates = Updates.combine( + Updates.set( + "data", + new Binary(plugin.getStorageManager().toBytes(playerData)) + ) + ); + UpdateOptions options = new UpdateOptions().upsert(true); + UpdateResult result = collection.updateOne(query, updates, options); + future.complete(result.wasAcknowledged()); + } catch (MongoException e) { + future.completeExceptionally(e); + } + }); + return future; + } + + @Override + public CompletableFuture> getPlayerData(UUID uuid) { + var future = new CompletableFuture>(); + plugin.getScheduler().runTaskAsync(() -> { + MongoCollection collection = database.getCollection(getCollectionName("data")); + Document doc = collection.find(Filters.eq("uuid", uuid)).first(); + if (doc == null) { + future.complete(Optional.empty()); + } else if (Bukkit.getPlayer(uuid) != null) { + Binary binary = (Binary) doc.get("data"); + future.complete(Optional.of(plugin.getStorageManager().fromBytes(binary.getData()))); + } else { + future.complete(Optional.empty()); + } + }); + return future; + } + + /** + * Get the collection name for a specific subcategory of data. + * + * @param value The subcategory identifier. + * @return The full collection name including the prefix. + */ + public String getCollectionName(String value) { + return getCollectionPrefix() + "_" + value; + } + + /** + * Get the collection prefix used for MongoDB collections. + * + * @return The collection prefix. + */ + public String getCollectionPrefix() { + return collectionPrefix; + } + + @Override + public StorageType getStorageType() { + return StorageType.MongoDB; + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/nosql/RedisManager.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/nosql/RedisManager.java new file mode 100644 index 0000000..80cb8ee --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/nosql/RedisManager.java @@ -0,0 +1,194 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.storage.method.database.nosql; + +import net.momirealms.customnameplates.api.data.PlayerData; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; +import net.momirealms.customnameplates.paper.storage.StorageType; +import net.momirealms.customnameplates.paper.storage.method.AbstractStorage; +import org.bukkit.configuration.ConfigurationSection; +import org.bukkit.configuration.file.YamlConfiguration; +import org.jetbrains.annotations.NotNull; +import redis.clients.jedis.Jedis; +import redis.clients.jedis.JedisPool; +import redis.clients.jedis.JedisPoolConfig; +import redis.clients.jedis.exceptions.JedisException; + +import java.nio.charset.StandardCharsets; +import java.time.Duration; +import java.util.Optional; +import java.util.UUID; +import java.util.concurrent.CompletableFuture; + +/** + * A RedisManager class responsible for managing interactions with a Redis server for data storage. + */ +public class RedisManager extends AbstractStorage { + + private static RedisManager instance; + private JedisPool jedisPool; + private String password; + private int port; + private String host; + private boolean useSSL; + + public RedisManager(CustomNameplatesPluginImpl plugin) { + super(plugin); + instance = this; + } + + /** + * Get the singleton instance of the RedisManager. + * + * @return The RedisManager instance. + */ + public static RedisManager getInstance() { + return instance; + } + + /** + * Get a Jedis resource for interacting with the Redis server. + * + * @return A Jedis resource. + */ + public Jedis getJedis() { + return jedisPool.getResource(); + } + + /** + * Initialize the Redis connection and configuration based on the plugin's YAML configuration. + */ + @Override + public void initialize() { + YamlConfiguration config = plugin.getConfig("database.yml"); + ConfigurationSection section = config.getConfigurationSection("Redis"); + if (section == null) { + LogUtils.warn("Failed to load database config. It seems that your config is broken. Please regenerate a new one."); + return; + } + + JedisPoolConfig jedisPoolConfig = new JedisPoolConfig(); + jedisPoolConfig.setTestWhileIdle(true); + jedisPoolConfig.setTimeBetweenEvictionRuns(Duration.ofMillis(30000)); + jedisPoolConfig.setNumTestsPerEvictionRun(-1); + jedisPoolConfig.setMinEvictableIdleTime(Duration.ofMillis(section.getInt("MinEvictableIdleTimeMillis",1800000))); + jedisPoolConfig.setMaxTotal(section.getInt("MaxTotal",8)); + jedisPoolConfig.setMaxIdle(section.getInt("MaxIdle",8)); + jedisPoolConfig.setMinIdle(section.getInt("MinIdle",1)); + jedisPoolConfig.setMaxWait(Duration.ofMillis(section.getInt("MaxWaitMillis"))); + + password = section.getString("password", ""); + port = section.getInt("port", 6379); + host = section.getString("host", "localhost"); + useSSL = section.getBoolean("use-ssl", false); + + if (password.isBlank()) { + jedisPool = new JedisPool(jedisPoolConfig, host, port, 0, useSSL); + } else { + jedisPool = new JedisPool(jedisPoolConfig, host, port, 0, password, useSSL); + } + try (Jedis jedis = jedisPool.getResource()) { + jedis.ping(); + LogUtils.info("Redis server connected."); + } catch (JedisException e) { + LogUtils.warn("Failed to connect redis."); + } + } + + /** + * Disable the Redis connection by closing the JedisPool. + */ + @Override + public void disable() { + if (jedisPool != null && !jedisPool.isClosed()) + jedisPool.close(); + } + + /** + * Send a message to Redis on a specified channel. + * + * @param channel The Redis channel to send the message to. + * @param message The message to send. + */ + public void sendRedisMessage(@NotNull String channel, @NotNull String message) { + try (Jedis jedis = jedisPool.getResource()) { + jedis.publish(channel, message); + plugin.debug("Sent Redis message: " + message); + } + } + + @Override + public StorageType getStorageType() { + return StorageType.Redis; + } + + @Override + public CompletableFuture> getPlayerData(UUID uuid) { + var future = new CompletableFuture>(); + plugin.getScheduler().runTaskAsync(() -> { + try (Jedis jedis = jedisPool.getResource()) { + byte[] key = getRedisKey("cn_data", uuid); + byte[] data = jedis.get(key); + jedis.del(key); + if (data != null) { + future.complete(Optional.of(plugin.getStorageManager().fromBytes(data))); + plugin.debug("Redis data retrieved for " + uuid + "; normal data"); + } else { + future.complete(Optional.empty()); + plugin.debug("Redis data retrieved for " + uuid + "; empty data"); + } + } catch (Exception e) { + future.complete(Optional.empty()); + LogUtils.warn("Failed to get redis data for " + uuid, e); + } + }); + return future; + } + + @Override + public CompletableFuture updatePlayerData(UUID uuid, PlayerData playerData) { + var future = new CompletableFuture(); + plugin.getScheduler().runTaskAsync(() -> { + try (Jedis jedis = jedisPool.getResource()) { + jedis.setex( + getRedisKey("cn_data", uuid), + 300, + plugin.getStorageManager().toBytes(playerData) + ); + future.complete(true); + plugin.debug("Redis data set for " + uuid); + } catch (Exception e) { + future.complete(false); + LogUtils.warn("Failed to set redis data for player " + uuid, e); + } + }); + return future; + } + + /** + * Generate a Redis key for a specified key and UUID. + * + * @param key The key identifier. + * @param uuid The UUID to include in the key. + * @return A byte array representing the Redis key. + */ + private byte[] getRedisKey(String key, @NotNull UUID uuid) { + return (key + ":" + uuid).getBytes(StandardCharsets.UTF_8); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/AbstractHikariDatabase.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/AbstractHikariDatabase.java new file mode 100644 index 0000000..c6433b9 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/AbstractHikariDatabase.java @@ -0,0 +1,139 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.storage.method.database.sql; + +import com.zaxxer.hikari.HikariConfig; +import com.zaxxer.hikari.HikariDataSource; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.storage.StorageType; +import org.bukkit.configuration.ConfigurationSection; +import org.bukkit.configuration.file.YamlConfiguration; + +import java.sql.Connection; +import java.sql.SQLException; +import java.util.Locale; +import java.util.Map; +import java.util.Properties; + +/** + * An abstract base class for SQL databases using the HikariCP connection pool, which handles player data storage. + */ +public abstract class AbstractHikariDatabase extends AbstractSQLDatabase { + + private HikariDataSource dataSource; + private final String driverClass; + private final String sqlBrand; + + public AbstractHikariDatabase(CustomNameplatesPlugin plugin) { + super(plugin); + this.driverClass = getStorageType() == StorageType.MariaDB ? "org.mariadb.jdbc.Driver" : "com.mysql.cj.jdbc.Driver"; + this.sqlBrand = getStorageType() == StorageType.MariaDB ? "MariaDB" : "MySQL"; + try { + Class.forName(this.driverClass); + } catch (ClassNotFoundException e1) { + if (getStorageType() == StorageType.MariaDB) { + LogUtils.warn("No MariaDB driver is found"); + } else if (getStorageType() == StorageType.MySQL) { + try { + Class.forName("com.mysql.jdbc.Driver"); + LogUtils.warn("It seems that you are not using MySQL 8.0+. It's recommended to update."); + } catch (ClassNotFoundException e2) { + LogUtils.warn("No MySQL driver is found"); + } + } + } + } + + /** + * Initialize the database connection pool and create tables if they don't exist. + */ + @Override + public void initialize() { + YamlConfiguration config = plugin.getConfig("database.yml"); + ConfigurationSection section = config.getConfigurationSection(sqlBrand); + + if (section == null) { + LogUtils.warn("Failed to load database config. It seems that your config is broken. Please regenerate a new one."); + return; + } + + super.tablePrefix = section.getString("table-prefix", "customfishing"); + HikariConfig hikariConfig = new HikariConfig(); + hikariConfig.setUsername(section.getString("user", "root")); + hikariConfig.setPassword(section.getString("password", "pa55w0rd")); + hikariConfig.setJdbcUrl(String.format("jdbc:%s://%s:%s/%s%s", + sqlBrand.toLowerCase(Locale.ENGLISH), + section.getString("host", "localhost"), + section.getString("port", "3306"), + section.getString("database", "minecraft"), + section.getString("connection-parameters") + )); + hikariConfig.setDriverClassName(driverClass); + hikariConfig.setMaximumPoolSize(section.getInt("Pool-Settings.max-pool-size", 10)); + hikariConfig.setMinimumIdle(section.getInt("Pool-Settings.min-idle", 10)); + hikariConfig.setMaxLifetime(section.getLong("Pool-Settings.max-lifetime", 180000L)); + hikariConfig.setConnectionTimeout(section.getLong("Pool-Settings.time-out", 20000L)); + hikariConfig.setPoolName("CustomNameplatesHikariPool"); + try { + hikariConfig.setKeepaliveTime(section.getLong("Pool-Settings.keep-alive-time", 60000L)); + } catch (NoSuchMethodError ignored) { + } + + final Properties properties = new Properties(); + properties.putAll( + Map.of("cachePrepStmts", "true", + "prepStmtCacheSize", "250", + "prepStmtCacheSqlLimit", "2048", + "useServerPrepStmts", "true", + "useLocalSessionState", "true", + "useLocalTransactionState", "true" + )); + properties.putAll( + Map.of( + "rewriteBatchedStatements", "true", + "cacheResultSetMetadata", "true", + "cacheServerConfiguration", "true", + "elideSetAutoCommits", "true", + "maintainTimeStats", "false") + ); + hikariConfig.setDataSourceProperties(properties); + dataSource = new HikariDataSource(hikariConfig); + super.createTableIfNotExist(); + } + + /** + * Disable the database by closing the connection pool. + */ + @Override + public void disable() { + if (dataSource != null && !dataSource.isClosed()) + dataSource.close(); + } + + /** + * Get a connection to the SQL database from the connection pool. + * + * @return A database connection. + * @throws SQLException If there is an error establishing a connection. + */ + @Override + public Connection getConnection() throws SQLException { + return dataSource.getConnection(); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/AbstractSQLDatabase.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/AbstractSQLDatabase.java new file mode 100644 index 0000000..2678619 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/AbstractSQLDatabase.java @@ -0,0 +1,189 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.storage.method.database.sql; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.data.PlayerData; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.storage.method.AbstractStorage; +import org.bukkit.Bukkit; +import org.jetbrains.annotations.NotNull; + +import java.io.ByteArrayInputStream; +import java.io.IOException; +import java.nio.charset.StandardCharsets; +import java.sql.*; +import java.util.Locale; +import java.util.Objects; +import java.util.Optional; +import java.util.UUID; +import java.util.concurrent.CompletableFuture; + +/** + * An abstract base class for SQL database implementations that handle player data storage. + */ +public abstract class AbstractSQLDatabase extends AbstractStorage { + + protected String tablePrefix; + + public AbstractSQLDatabase(CustomNameplatesPlugin plugin) { + super(plugin); + } + + /** + * Get a connection to the SQL database. + * + * @return A database connection. + * @throws SQLException If there is an error establishing a connection. + */ + public abstract Connection getConnection() throws SQLException; + + /** + * Create tables for storing data if they don't exist in the database. + */ + public void createTableIfNotExist() { + try (Connection connection = getConnection()) { + final String[] databaseSchema = getSchema(getStorageType().name().toLowerCase(Locale.ENGLISH)); + try (Statement statement = connection.createStatement()) { + for (String tableCreationStatement : databaseSchema) { + statement.execute(tableCreationStatement); + } + } catch (SQLException e) { + LogUtils.warn("Failed to create tables"); + } + } catch (SQLException e) { + LogUtils.warn("Failed to get sql connection"); + } catch (IOException e) { + LogUtils.warn("Failed to get schema resource"); + } + } + + /** + * Get the SQL schema from a resource file. + * + * @param fileName The name of the schema file. + * @return An array of SQL statements to create tables. + * @throws IOException If there is an error reading the schema resource. + */ + private String[] getSchema(@NotNull String fileName) throws IOException { + return replaceSchemaPlaceholder(new String(Objects.requireNonNull(plugin.getResource("schema/" + fileName + ".sql")) + .readAllBytes(), StandardCharsets.UTF_8)).split(";"); + } + + /** + * Replace placeholder values in SQL schema with the table prefix. + * + * @param sql The SQL schema string. + * @return The SQL schema string with placeholders replaced. + */ + private String replaceSchemaPlaceholder(@NotNull String sql) { + return sql.replace("{prefix}", tablePrefix); + } + + /** + * Get the name of a database table based on a sub-table name and the table prefix. + * + * @param sub The sub-table name. + * @return The full table name. + */ + public String getTableName(String sub) { + return getTablePrefix() + "_" + sub; + } + + /** + * Get the current table prefix. + * + * @return The table prefix. + */ + public String getTablePrefix() { + return tablePrefix; + } + + @Override + public CompletableFuture updatePlayerData(UUID uuid, PlayerData playerData) { + var future = new CompletableFuture(); + plugin.getScheduler().runTaskAsync(() -> { + try ( + Connection connection = getConnection(); + PreparedStatement statement = connection.prepareStatement(String.format(SqlConstants.SQL_UPDATE_BY_UUID, getTableName("data"))) + ) { + statement.setBlob(1, new ByteArrayInputStream(plugin.getStorageManager().toBytes(playerData))); + statement.setString(2, uuid.toString()); + statement.executeUpdate(); + future.complete(true); + } catch (SQLException e) { + LogUtils.warn("Failed to update " + uuid + "'s data.", e); + future.completeExceptionally(e); + } + }); + return future; + } + + @Override + public CompletableFuture> getPlayerData(UUID uuid) { + var future = new CompletableFuture>(); + plugin.getScheduler().runTaskAsync(() -> { + try ( + Connection connection = getConnection(); + PreparedStatement statement = connection.prepareStatement(String.format(SqlConstants.SQL_SELECT_BY_UUID, getTableName("data"))) + ) { + statement.setString(1, uuid.toString()); + ResultSet rs = statement.executeQuery(); + if (rs.next()) { + Blob blob = rs.getBlob("data"); + byte[] dataByteArray = blob.getBytes(1, (int) blob.length()); + blob.free(); + future.complete(Optional.of(plugin.getStorageManager().fromBytes(dataByteArray))); + } else if (Bukkit.getPlayer(uuid) != null) { + var data = PlayerData.empty(); + this.insertPlayerData(uuid, data); + future.complete(Optional.of(data)); + } else { + future.complete(Optional.empty()); + } + } catch (SQLException e) { + LogUtils.warn("Failed to get " + uuid + "'s data.", e); + future.completeExceptionally(e); + } + }); + return future; + } + + public void insertPlayerData(UUID uuid, PlayerData playerData) { + try ( + Connection connection = getConnection(); + PreparedStatement statement = connection.prepareStatement(String.format(SqlConstants.SQL_INSERT_DATA_BY_UUID, getTableName("data"))) + ) { + statement.setString(1, uuid.toString()); + statement.setBlob(2, new ByteArrayInputStream(plugin.getStorageManager().toBytes(playerData))); + statement.execute(); + } catch (SQLException e) { + LogUtils.warn("Failed to insert " + uuid + "'s data.", e); + } + } + + /** + * Constants defining SQL statements used for database operations. + */ + public static class SqlConstants { + public static final String SQL_SELECT_BY_UUID = "SELECT * FROM `%s` WHERE `uuid` = ?"; + public static final String SQL_SELECT_ALL_UUID = "SELECT uuid FROM `%s`"; + public static final String SQL_UPDATE_BY_UUID = "UPDATE `%s` SET `data` = ? WHERE `uuid` = ?"; + public static final String SQL_INSERT_DATA_BY_UUID = "INSERT INTO `%s`(`uuid`, `data`) VALUES(?, ?)"; + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/H2Impl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/H2Impl.java new file mode 100644 index 0000000..e03c46d --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/H2Impl.java @@ -0,0 +1,73 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.storage.method.database.sql; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.storage.StorageType; +import org.bukkit.configuration.file.YamlConfiguration; +import org.h2.jdbcx.JdbcConnectionPool; + +import java.io.File; +import java.sql.Connection; +import java.sql.SQLException; + +/** + * An implementation of AbstractSQLDatabase that uses the H2 embedded database for player data storage. + */ +public class H2Impl extends AbstractSQLDatabase { + + private JdbcConnectionPool connectionPool; + + public H2Impl(CustomNameplatesPlugin plugin) { + super(plugin); + } + + /** + * Initialize the H2 database and connection pool based on the configuration. + */ + @Override + public void initialize() { + YamlConfiguration config = plugin.getConfig("database.yml"); + File databaseFile = new File(plugin.getDataFolder(), config.getString("H2.file", "data.db")); + super.tablePrefix = config.getString("H2.table-prefix", "customnameplates"); + + final String url = String.format("jdbc:h2:%s", databaseFile.getAbsolutePath()); + this.connectionPool = JdbcConnectionPool.create(url, "sa", ""); + super.createTableIfNotExist(); + } + + /** + * Disable the H2 database by disposing of the connection pool. + */ + @Override + public void disable() { + if (connectionPool != null) { + connectionPool.dispose(); + } + } + + @Override + public StorageType getStorageType() { + return StorageType.H2; + } + + @Override + public Connection getConnection() throws SQLException { + return connectionPool.getConnection(); + } +} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/ExpressionAnd.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/MariaDBImpl.java similarity index 62% rename from src/main/java/net/momirealms/customnameplates/object/requirements/papi/ExpressionAnd.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/MariaDBImpl.java index 047169b..e28e728 100644 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/ExpressionAnd.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/MariaDBImpl.java @@ -15,19 +15,19 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.object.requirements.papi; +package net.momirealms.customnameplates.paper.storage.method.database.sql; -import org.bukkit.entity.Player; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.storage.StorageType; -import java.util.List; +public class MariaDBImpl extends AbstractHikariDatabase { -public record ExpressionAnd(List requirements) implements PapiRequirement { + public MariaDBImpl(CustomNameplatesPlugin plugin) { + super(plugin); + } @Override - public boolean isMet(Player player) { - for (PapiRequirement requirement : requirements) { - if (!requirement.isMet(player)) return false; - } - return true; + public StorageType getStorageType() { + return StorageType.MariaDB; } -} \ No newline at end of file +} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/ExpressionOr.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/MySQLImpl.java similarity index 62% rename from src/main/java/net/momirealms/customnameplates/object/requirements/papi/ExpressionOr.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/MySQLImpl.java index b95e436..cadb247 100644 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/ExpressionOr.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/MySQLImpl.java @@ -15,19 +15,19 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.object.requirements.papi; +package net.momirealms.customnameplates.paper.storage.method.database.sql; -import org.bukkit.entity.Player; +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.paper.storage.StorageType; -import java.util.List; +public class MySQLImpl extends AbstractHikariDatabase { -public record ExpressionOr(List requirements) implements PapiRequirement{ + public MySQLImpl(CustomNameplatesPlugin plugin) { + super(plugin); + } @Override - public boolean isMet(Player player) { - for (PapiRequirement requirement : requirements) { - if (requirement.isMet(player)) return true; - } - return false; + public StorageType getStorageType() { + return StorageType.MySQL; } -} \ No newline at end of file +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/SQLiteImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/SQLiteImpl.java new file mode 100644 index 0000000..b79df41 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/database/sql/SQLiteImpl.java @@ -0,0 +1,176 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.storage.method.database.sql; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.data.PlayerData; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.storage.StorageType; +import org.bukkit.Bukkit; +import org.bukkit.configuration.file.YamlConfiguration; +import org.sqlite.SQLiteConfig; + +import java.io.File; +import java.io.IOException; +import java.sql.*; +import java.util.Optional; +import java.util.UUID; +import java.util.concurrent.CompletableFuture; + +/** + * An implementation of AbstractSQLDatabase that uses the SQLite database for player data storage. + */ +public class SQLiteImpl extends AbstractSQLDatabase { + + private Connection connection; + private File databaseFile; + + public SQLiteImpl(CustomNameplatesPlugin plugin) { + super(plugin); + } + + /** + * Initialize the SQLite database and connection based on the configuration. + */ + @Override + public void initialize() { + YamlConfiguration config = plugin.getConfig("database.yml"); + this.databaseFile = new File(plugin.getDataFolder(), config.getString("SQLite.file", "data") + ".db"); + super.tablePrefix = config.getString("SQLite.table-prefix", "customfishing"); + super.createTableIfNotExist(); + } + + /** + * Disable the SQLite database by closing the connection. + */ + @Override + public void disable() { + try { + if (connection != null && !connection.isClosed()) + connection.close(); + } catch (SQLException e) { + throw new RuntimeException(e); + } + } + + @Override + public CompletableFuture updatePlayerData(UUID uuid, PlayerData playerData) { + var future = new CompletableFuture(); + plugin.getScheduler().runTaskAsync(() -> { + try ( + Connection connection = getConnection(); + PreparedStatement statement = connection.prepareStatement(String.format(SqlConstants.SQL_UPDATE_BY_UUID, getTableName("data"))) + ) { + statement.setBytes(1, plugin.getStorageManager().toBytes(playerData)); + statement.setString(2, uuid.toString()); + statement.executeUpdate(); + future.complete(true); + } catch (SQLException e) { + LogUtils.warn("Failed to update " + uuid + "'s data.", e); + future.completeExceptionally(e); + } + }); + return future; + } + + @Override + public CompletableFuture> getPlayerData(UUID uuid) { + var future = new CompletableFuture>(); + plugin.getScheduler().runTaskAsync(() -> { + try ( + Connection connection = getConnection(); + PreparedStatement statement = connection.prepareStatement(String.format(SqlConstants.SQL_SELECT_BY_UUID, getTableName("data"))) + ) { + statement.setString(1, uuid.toString()); + ResultSet rs = statement.executeQuery(); + if (rs.next()) { + byte[] dataByteArray = rs.getBytes("data"); + future.complete(Optional.of(plugin.getStorageManager().fromBytes(dataByteArray))); + } else if (Bukkit.getPlayer(uuid) != null) { + var data = PlayerData.empty(); + this.insertPlayerData(uuid, data); + future.complete(Optional.of(data)); + } else { + future.complete(Optional.empty()); + } + } catch (SQLException e) { + LogUtils.warn("Failed to get " + uuid + "'s data.", e); + future.completeExceptionally(e); + } + }); + return future; + } + + @Override + public void insertPlayerData(UUID uuid, PlayerData playerData) { + try ( + Connection connection = getConnection(); + PreparedStatement statement = connection.prepareStatement(String.format(SqlConstants.SQL_INSERT_DATA_BY_UUID, getTableName("data"))) + ) { + statement.setString(1, uuid.toString()); + statement.setBytes(2, plugin.getStorageManager().toBytes(playerData)); + statement.execute(); + } catch (SQLException e) { + LogUtils.warn("Failed to insert " + uuid + "'s data.", e); + } + } + + @Override + public StorageType getStorageType() { + return StorageType.SQLite; + } + + /** + * Get a connection to the SQLite database. + * + * @return A database connection. + * @throws SQLException If there is an error establishing a connection. + */ + @Override + public Connection getConnection() throws SQLException { + if (connection == null || connection.isClosed()) { + setConnection(); + } + return connection; + } + + /** + * Set up the connection to the SQLite database. + */ + @SuppressWarnings("ResultOfMethodCallIgnored") + private void setConnection() { + try { + if (!databaseFile.exists()) databaseFile.createNewFile(); + Class.forName("org.sqlite.JDBC"); + SQLiteConfig config = new SQLiteConfig(); + config.enforceForeignKeys(true); + config.setEncoding(SQLiteConfig.Encoding.UTF8); + config.setSynchronous(SQLiteConfig.SynchronousMode.FULL); + connection = DriverManager.getConnection( + String.format("jdbc:sqlite:%s", databaseFile.getAbsolutePath()), + config.toProperties() + ); + } catch (IOException e) { + LogUtils.warn("Failed to create the SQLite database."); + } catch (SQLException e) { + LogUtils.warn("Failed to initialize SQLite database."); + } catch (ClassNotFoundException e) { + LogUtils.warn("Failed to find SQLite driver."); + } + } +} \ No newline at end of file diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/file/JsonImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/file/JsonImpl.java new file mode 100644 index 0000000..3a7fd0b --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/file/JsonImpl.java @@ -0,0 +1,128 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.storage.method.file; + +import com.google.gson.Gson; +import net.momirealms.customnameplates.api.data.PlayerData; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; +import net.momirealms.customnameplates.paper.storage.StorageType; +import net.momirealms.customnameplates.paper.storage.method.AbstractStorage; +import org.bukkit.Bukkit; + +import java.io.File; +import java.io.FileInputStream; +import java.io.FileWriter; +import java.io.IOException; +import java.nio.charset.StandardCharsets; +import java.util.Optional; +import java.util.UUID; +import java.util.concurrent.CompletableFuture; + +/** + * A data storage implementation that uses JSON files to store player data. + */ +public class JsonImpl extends AbstractStorage { + + @SuppressWarnings("ResultOfMethodCallIgnored") + public JsonImpl(CustomNameplatesPluginImpl plugin) { + super(plugin); + File folder = new File(plugin.getDataFolder(), "data"); + if (!folder.exists()) folder.mkdirs(); + } + + @Override + public StorageType getStorageType() { + return StorageType.JSON; + } + + @Override + public CompletableFuture> getPlayerData(UUID uuid) { + File file = getPlayerDataFile(uuid); + PlayerData playerData; + if (file.exists()) { + playerData = readFromJsonFile(file, PlayerData.class); + } else if (Bukkit.getPlayer(uuid) != null) { + playerData = PlayerData.empty(); + } else { + playerData = null; + } + return CompletableFuture.completedFuture(Optional.ofNullable(playerData)); + } + + /** + * Get the file associated with a player's UUID for storing JSON data. + * + * @param uuid The UUID of the player. + * @return The file for the player's data. + */ + public File getPlayerDataFile(UUID uuid) { + return new File(plugin.getDataFolder(), "data" + File.separator + uuid + ".json"); + } + + /** + * Save an object to a JSON file. + * + * @param obj The object to be saved as JSON. + * @param filepath The file path where the JSON file should be saved. + */ + public void saveToJsonFile(Object obj, File filepath) { + Gson gson = new Gson(); + try (FileWriter file = new FileWriter(filepath)) { + gson.toJson(obj, file); + } catch (IOException e) { + e.printStackTrace(); + } + } + + /** + * Read JSON content from a file and parse it into an object of the specified class. + * + * @param file The JSON file to read. + * @param classOfT The class of the object to parse the JSON into. + * @param The type of the object. + * @return The parsed object. + */ + public T readFromJsonFile(File file, Class classOfT) { + Gson gson = new Gson(); + String jsonContent = new String(readFileToByteArray(file), StandardCharsets.UTF_8); + return gson.fromJson(jsonContent, classOfT); + } + + /** + * Read the contents of a file and return them as a byte array. + * + * @param file The file to read. + * @return The byte array representing the file's content. + */ + @SuppressWarnings("ResultOfMethodCallIgnored") + public byte[] readFileToByteArray(File file) { + byte[] fileBytes = new byte[(int) file.length()]; + try (FileInputStream fis = new FileInputStream(file)) { + fis.read(fileBytes); + } catch (IOException e) { + e.printStackTrace(); + } + return fileBytes; + } + + @Override + public CompletableFuture updatePlayerData(UUID uuid, PlayerData playerData) { + this.saveToJsonFile(playerData, getPlayerDataFile(uuid)); + return CompletableFuture.completedFuture(true); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/file/YAMLImpl.java b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/file/YAMLImpl.java new file mode 100644 index 0000000..762d053 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/storage/method/file/YAMLImpl.java @@ -0,0 +1,105 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.storage.method.file; + +import net.momirealms.customnameplates.api.data.PlayerData; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl; +import net.momirealms.customnameplates.paper.storage.StorageType; +import net.momirealms.customnameplates.paper.storage.method.AbstractStorage; +import org.bukkit.Bukkit; +import org.bukkit.configuration.file.YamlConfiguration; + +import java.io.File; +import java.io.IOException; +import java.util.Optional; +import java.util.UUID; +import java.util.concurrent.CompletableFuture; + +/** + * A data storage implementation that uses YAML files to store player data, with support for legacy data. + */ +public class YAMLImpl extends AbstractStorage { + + @SuppressWarnings("ResultOfMethodCallIgnored") + public YAMLImpl(CustomNameplatesPluginImpl plugin) { + super(plugin); + File folder = new File(plugin.getDataFolder(), "data"); + if (!folder.exists()) folder.mkdirs(); + } + + @Override + public StorageType getStorageType() { + return StorageType.YAML; + } + + @Override + public CompletableFuture> getPlayerData(UUID uuid) { + File dataFile = getPlayerDataFile(uuid); + if (!dataFile.exists()) { + if (Bukkit.getPlayer(uuid) != null) { + return CompletableFuture.completedFuture(Optional.of(PlayerData.empty())); + } else { + return CompletableFuture.completedFuture(Optional.empty()); + } + } + YamlConfiguration data = readData(dataFile); + PlayerData playerData = new PlayerData.Builder() + .setBubble(data.getString("bubble", "")) + .setNameplate(data.getString("nameplate", "")) + .build(); + return CompletableFuture.completedFuture(Optional.of(playerData)); + } + + /** + * Get the file associated with a player's UUID for storing YAML data. + * + * @param uuid The UUID of the player. + * @return The file for the player's data. + */ + public File getPlayerDataFile(UUID uuid) { + return new File(plugin.getDataFolder(), "data" + File.separator + uuid + ".yml"); + } + + @SuppressWarnings("ResultOfMethodCallIgnored") + public YamlConfiguration readData(File file) { + if (!file.exists()) { + try { + file.getParentFile().mkdirs(); + file.createNewFile(); + } catch (IOException e) { + e.printStackTrace(); + LogUtils.warn("Failed to generate data files!"); + } + } + return YamlConfiguration.loadConfiguration(file); + } + + @Override + public CompletableFuture updatePlayerData(UUID uuid, PlayerData playerData) { + YamlConfiguration data = new YamlConfiguration(); + data.set("bubble", playerData.getBubble()); + data.set("nameplate", playerData.getNameplate()); + try { + data.save(getPlayerDataFile(uuid)); + } catch (IOException e) { + LogUtils.warn("Failed to save player data", e); + } + return CompletableFuture.completedFuture(true); + } +} diff --git a/src/main/java/net/momirealms/customnameplates/utils/ClassUtils.java b/paper/src/main/java/net/momirealms/customnameplates/paper/util/ClassUtils.java similarity index 98% rename from src/main/java/net/momirealms/customnameplates/utils/ClassUtils.java rename to paper/src/main/java/net/momirealms/customnameplates/paper/util/ClassUtils.java index 74a4646..ba36931 100644 --- a/src/main/java/net/momirealms/customnameplates/utils/ClassUtils.java +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/util/ClassUtils.java @@ -15,7 +15,7 @@ * along with this program. If not, see . */ -package net.momirealms.customnameplates.utils; +package net.momirealms.customnameplates.paper.util; import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/util/ConfigUtils.java b/paper/src/main/java/net/momirealms/customnameplates/paper/util/ConfigUtils.java new file mode 100644 index 0000000..de68cbd --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/util/ConfigUtils.java @@ -0,0 +1,98 @@ +package net.momirealms.customnameplates.paper.util; + +import net.momirealms.customnameplates.api.CustomNameplatesPlugin; +import net.momirealms.customnameplates.api.common.Pair; +import net.momirealms.customnameplates.api.requirement.Requirement; +import net.momirealms.customnameplates.api.util.LogUtils; +import net.momirealms.customnameplates.paper.mechanic.misc.TimeLimitText; +import org.bukkit.configuration.ConfigurationSection; + +import java.util.*; + +public class ConfigUtils { + + private ConfigUtils() {} + + /** + * Converts an object into an ArrayList of strings. + * + * @param object The input object + * @return An ArrayList of strings + */ + @SuppressWarnings("unchecked") + public static ArrayList stringListArgs(Object object) { + ArrayList list = new ArrayList<>(); + if (object instanceof String member) { + list.add(member); + } else if (object instanceof List members) { + list.addAll((Collection) members); + } else if (object instanceof String[] strings) { + list.addAll(List.of(strings)); + } + return list; + } + + /** + * Splits a string into a pair of integers using the "~" delimiter. + * + * @param value The input string + * @return A Pair of integers + */ + public static Pair splitStringIntegerArgs(String value, String regex) { + String[] split = value.split(regex); + return Pair.of(Integer.parseInt(split[0]), Integer.parseInt(split[1])); + } + + /** + * Converts an object into a double value. + * + * @param arg The input object + * @return A double value + */ + public static double getDoubleValue(Object arg) { + if (arg instanceof Double d) { + return d; + } else if (arg instanceof Integer i) { + return Double.valueOf(i); + } + return 0; + } + + public static TimeLimitText[] getTimeLimitTexts(ConfigurationSection section) { + + TreeMap map = new TreeMap<>(); + if (section == null) { + LogUtils.warn("No text display order set, this might cause bugs!"); + map.put(1, new TimeLimitText(100, -1, "", new Requirement[0])); + return getOrderedTexts(map); + } + + for (Map.Entry entry : section.getValues(false).entrySet()) { + if (!(entry.getValue() instanceof ConfigurationSection textSection)) + continue; + + var text = TimeLimitText.Builder.of() + .text(textSection.getString("text", "")) + .refreshFrequency(textSection.getInt("refresh-frequency", -1)) + .duration(textSection.getInt("duration", 200)) + .requirement(CustomNameplatesPlugin.get().getRequirementManager().getRequirements(textSection.getConfigurationSection("conditions"))) + .build(); + + int order; + try { + order = Integer.parseInt(entry.getKey()); + } catch (NumberFormatException e) { + LogUtils.warn("Invalid order format: " + entry.getKey()); + continue; + } + + map.put(order, text); + } + + return getOrderedTexts(map); + } + + private static TimeLimitText[] getOrderedTexts(TreeMap map) { + return map.values().toArray(new TimeLimitText[0]); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/util/LocationUtils.java b/paper/src/main/java/net/momirealms/customnameplates/paper/util/LocationUtils.java new file mode 100644 index 0000000..e7884c7 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/util/LocationUtils.java @@ -0,0 +1,13 @@ +package net.momirealms.customnameplates.paper.util; + +import org.bukkit.Location; +import org.bukkit.entity.Entity; + +public class LocationUtils { + + public static double getDistance(Entity e1, Entity e2) { + Location loc1 = e1.getLocation(); + Location loc2 = e2.getLocation(); + return Math.sqrt(Math.pow(loc1.getX()-loc2.getX(), 2) + Math.pow(loc1.getZ()-loc2.getZ(), 2)); + } +} diff --git a/paper/src/main/java/net/momirealms/customnameplates/paper/util/ReflectionUtils.java b/paper/src/main/java/net/momirealms/customnameplates/paper/util/ReflectionUtils.java new file mode 100644 index 0000000..a4b2695 --- /dev/null +++ b/paper/src/main/java/net/momirealms/customnameplates/paper/util/ReflectionUtils.java @@ -0,0 +1,62 @@ +/* + * Copyright (C) <2022> + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package net.momirealms.customnameplates.paper.util; + +import com.comphenix.protocol.utility.MinecraftReflection; +import net.kyori.adventure.text.Component; +import net.kyori.adventure.text.serializer.gson.GsonComponentSerializer; +import net.momirealms.customnameplates.api.util.LogUtils; + +import java.lang.reflect.Constructor; +import java.lang.reflect.Field; +import java.lang.reflect.InvocationTargetException; +import java.lang.reflect.Method; + +public class ReflectionUtils { + + private ReflectionUtils() {} + + public static Object removeBossBarPacket; + public static Constructor progressConstructor; + public static Constructor updateConstructor; + public static Method iChatComponentMethod; + public static Method gsonDeserializeMethod; + public static Object gsonInstance; + public static Object emptyComponent; + + public static void load() { + try { + Class bar = Class.forName("net.minecraft.network.protocol.game.PacketPlayOutBoss"); + Field remove = bar.getDeclaredField("f"); + remove.setAccessible(true); + removeBossBarPacket = remove.get(null); + Class packetBossClassF = Class.forName("net.minecraft.network.protocol.game.PacketPlayOutBoss$f"); + progressConstructor = packetBossClassF.getDeclaredConstructor(float.class); + progressConstructor.setAccessible(true); + Class packetBossClassE = Class.forName("net.minecraft.network.protocol.game.PacketPlayOutBoss$e"); + updateConstructor = packetBossClassE.getDeclaredConstructor(MinecraftReflection.getIChatBaseComponentClass()); + updateConstructor.setAccessible(true); + iChatComponentMethod = MinecraftReflection.getChatSerializerClass().getMethod("a", String.class); + iChatComponentMethod.setAccessible(true); + emptyComponent = iChatComponentMethod.invoke(null, "{\"text\":\"\"}"); + } catch (ClassNotFoundException | NoSuchFieldException | IllegalAccessException | NoSuchMethodException | InvocationTargetException exception) { + LogUtils.severe("Error occurred when loading reflections", exception); + exception.printStackTrace(); + } + } +} \ No newline at end of file diff --git a/src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.fsh b/paper/src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.fsh similarity index 100% rename from src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.fsh rename to paper/src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.fsh diff --git a/src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.json b/paper/src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.json similarity index 100% rename from src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.json rename to paper/src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.json diff --git a/src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.vsh b/paper/src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.vsh similarity index 100% rename from src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.vsh rename to paper/src/main/resources/ResourcePack/assets/minecraft/shaders/core/rendertype_text.vsh diff --git a/src/main/resources/ResourcePack/assets/minecraft/textures/gui/bars.png b/paper/src/main/resources/ResourcePack/assets/minecraft/textures/gui/bars.png similarity index 100% rename from src/main/resources/ResourcePack/assets/minecraft/textures/gui/bars.png rename to paper/src/main/resources/ResourcePack/assets/minecraft/textures/gui/bars.png diff --git a/src/main/resources/ResourcePack/assets/minecraft/textures/gui/sprites/boss_bar/yellow_background.png b/paper/src/main/resources/ResourcePack/assets/minecraft/textures/gui/sprites/boss_bar/yellow_background.png similarity index 100% rename from src/main/resources/ResourcePack/assets/minecraft/textures/gui/sprites/boss_bar/yellow_background.png rename to paper/src/main/resources/ResourcePack/assets/minecraft/textures/gui/sprites/boss_bar/yellow_background.png diff --git a/src/main/resources/ResourcePack/assets/minecraft/textures/gui/sprites/boss_bar/yellow_progress.png b/paper/src/main/resources/ResourcePack/assets/minecraft/textures/gui/sprites/boss_bar/yellow_progress.png similarity index 100% rename from src/main/resources/ResourcePack/assets/minecraft/textures/gui/sprites/boss_bar/yellow_progress.png rename to paper/src/main/resources/ResourcePack/assets/minecraft/textures/gui/sprites/boss_bar/yellow_progress.png diff --git a/src/main/resources/ResourcePack/pack.mcmeta b/paper/src/main/resources/ResourcePack/pack.mcmeta similarity index 100% rename from src/main/resources/ResourcePack/pack.mcmeta rename to paper/src/main/resources/ResourcePack/pack.mcmeta diff --git a/paper/src/main/resources/config.yml b/paper/src/main/resources/config.yml new file mode 100644 index 0000000..8f9675c --- /dev/null +++ b/paper/src/main/resources/config.yml @@ -0,0 +1,103 @@ +# Do not change +config-version: '22' + +# Debug mode +debug: false + +# bStats +metrics: true + +# Check update +update-checker: true + +# Language +# english/chinese/spanish/russian/turkish/french +lang: english + +# Modules +modules: + nameplates: true + backgrounds: true + bubbles: true + bossbars: true + actionbars: true + images: true + +# Integrations +integrations: + resource-pack: + # If enabled, CustomNameplates will automatically place the generated resource pack into ItemsAdder/Oraxen. + # To apply this change, run "/nameplates reload" & "/iazip" or "/oraxen reload all" + ItemsAdder: false + Oraxen: false + # Require a restart to apply this + team: + # When enabled, CustomNameplates will hook into TAB's team management + TAB: false + # When enabled, CustomNameplates will hook into CMI's team management + CMI: false + chat: + # TrChat channels + TrChat: false + # VentureChat channels + VentureChat: false + +resource-pack: + # disable resource pack generation on server start + disable-generation-on-start: false + # Namespace + namespace: "nameplates" + # Font name + font: "default" + # The initial character of the custom font + # Although Korean is used here, it does not affect the normal use of Korean in chatting, as they are not in the same font + initial-char: '뀁' + # Customize the folder where png files should be generated. + # This is useful for those who want to keep their resource pack structure in order. + image-path: + nameplates: 'font\nameplates\' + backgrounds: 'font\backgrounds\' + images: 'font\images\' + bubbles: 'font\bubbles\' + space-split: 'font\base\' + + shader: + enable: true + # Hide scoreboard numbers + hide-scoreboard-number: false + # Add Animated images support + animated-image: false + # Add ItemsAdder text effect support + ItemsAdder-text-effect: false + + transparent-bossbar: + # Make a certain bossbar transparent + color: YELLOW + # Generate the transparent bossbar for 1.20.2+ + 1_20_2+: true + # Generate the transparent bossbar for legacy versions + 1_17-1_20_1: true + + # Unicode Support for 1.20+ clients + # This would make your resource pack about 900KB bigger + support-1_20-unicodes: true + + +other-settings: + # It's recommended to use MiniMessage format. If you insist on using legacy color code "&", enable the support below. + legacy-color-code-support: false + + # Thread pool settings + thread-pool-settings: + # The size of the core Thread pool, that is, the size of the Thread pool when there is no task to execute + # Increase the size of corePoolSize when you are running a large server with many players fishing at the same time + corePoolSize: 10 + # The maximum number of threads allowed to be created in the Thread pool. The current number of threads in the Thread pool will not exceed this value + maximumPoolSize: 10 + # If a thread is idle for more than this attribute value, it will exit due to timeout + keepAliveTime: 30 + + # default width + default-character-width: 8 + # delay x ticks before actionbar/bossbar is sent to players + send-delay: 0 \ No newline at end of file diff --git a/paper/src/main/resources/configs/actionbar.yml b/paper/src/main/resources/configs/actionbar.yml new file mode 100644 index 0000000..6910ca5 --- /dev/null +++ b/paper/src/main/resources/configs/actionbar.yml @@ -0,0 +1,14 @@ +# You can only create "actionbar" section in this config file +actionbar: + check-frequency: 10 + conditions: + permission: "actionbar.show" + text-display-order: + 1: + # Measured in ticks + duration: 200 + # Text content + text: 'Welcome to my server! %player_name%' + # The frequency of refreshing the text, the lower the value is, the faster the placeholders are updated + # measured in ticks, -1 = disable refreshing + refresh-frequency: 1 \ No newline at end of file diff --git a/paper/src/main/resources/configs/bossbar.yml b/paper/src/main/resources/configs/bossbar.yml new file mode 100644 index 0000000..60f733f --- /dev/null +++ b/paper/src/main/resources/configs/bossbar.yml @@ -0,0 +1,34 @@ +# You can create as many sections as you want for more bossbars +bossbar_1: + # Color (BLUE, GREEN, PINK, PURPLE, RED, WHITE, YELLOW) + color: YELLOW + # Overlay (It can be either "progress", "notched_6", "notched_10", "notched_12" or "notched_20") + overlay: PROGRESS + # The frequency of checking conditions for a bar, measured in ticks + check-frequency: 10 + conditions: + permission: "bossbar.show" + # Bar's content would be shown as the following order + text-display-order: + 1: + # Measured in ticks + duration: 200 + # Text content + text: 'Welcome to my server! %player_name%' + # The frequency of refreshing the text, the lower the value is, the faster the placeholders are updated + # measured in ticks, -1 = disable refreshing + refresh-frequency: -1 + 2: + duration: 100 + text: 'Your location: [X: %player_x%, Y: %player_y%, Z: %player_z%]' + refresh-frequency: 1 + 3: + duration: 100 + text: 'This is a message only visible to admins' + refresh-frequency: -1 + # Optional + # When enabling conditions, make sure that players would see at least one in the order, + # otherwise it will fall into a dead cycle + conditions: + # If your player doesn't meet the conditions, it would be skipped + permission: nameplates.admin \ No newline at end of file diff --git a/src/main/resources/configs/custom-placeholders.yml b/paper/src/main/resources/configs/custom-placeholders.yml similarity index 90% rename from src/main/resources/configs/custom-placeholders.yml rename to paper/src/main/resources/configs/custom-placeholders.yml index 901b309..c24e90c 100644 --- a/src/main/resources/configs/custom-placeholders.yml +++ b/paper/src/main/resources/configs/custom-placeholders.yml @@ -163,23 +163,39 @@ descent-text: # %nameplates_unicode_{0}% descent-unicode: - normal: - text: "Just for registering" - descent: 0 hello: text: "Hello 여보세요 你好 こんにちは" descent: 5 +# %nameplates_switch_{0}% +switch-text: + season: + switch: '%customcrops_season%' + case: + 'Spring': 'Spring' + 'Summer': 'Summer' + 'Autumn': 'Autumn' + 'Winter': 'Winter' + default: 'Invalid season' + # %nameplates_vanilla_{0}% # This is useful for creating a vanilla like hud vanilla-hud: stamina_hud: reverse: true images: - empty: "%nameplates_image_stamina_0%" - half: "%nameplates_image_stamina_1%" - full: "%nameplates_image_stamina_2%" + empty: stamina_0 + half: stamina_1 + full: stamina_2 placeholder: # value/max-value can be placeholders value: '1.1' - max-value: '2' \ No newline at end of file + max-value: '2' + +# %nameplates_cached_{0}% +cached-text: + cached: + # The text to cache + text: '%super_complex_javascript%' + # How often is the value refreshed, measured in ms + refresh-interval: 10000 \ No newline at end of file diff --git a/paper/src/main/resources/configs/image-width.yml b/paper/src/main/resources/configs/image-width.yml new file mode 100644 index 0000000..06abcce --- /dev/null +++ b/paper/src/main/resources/configs/image-width.yml @@ -0,0 +1,5 @@ +minecraft:default: + '%img_xxx%': 15 + 默: 8 + +customcrops:default: diff --git a/paper/src/main/resources/configs/nameplate.yml b/paper/src/main/resources/configs/nameplate.yml new file mode 100644 index 0000000..ff9f083 --- /dev/null +++ b/paper/src/main/resources/configs/nameplate.yml @@ -0,0 +1,46 @@ +# Team / Unlimited / Disable +mode: TEAM + +# If you are using BungeeCord/Velocity, it's advised to install CustomNameplates on proxy too and enable this option. +proxy: false + +# The duration (in seconds) that the nameplate preview will last for. +preview-duration: 5 + +# Default nameplate to display if player doesn't equip any +default-nameplate: 'none' + +# This decides what %nameplates_prefix/suffix% would return +# Check xxx for examples +nameplate: + refresh-frequency: 10 + prefix: '' + player-name: '%player_name%' + suffix: '' + # Disable this if you are using a custom Tab plugin + fix-Tab: true + +# Settings for Team mode +team: + # measured in ticks + refresh-frequency: 10 + prefix: '%nameplates_prefix%' + suffix: '%nameplates_suffix%' + +# Settings for Unlimited mode +unlimited: + tag_1: + text: '%nameplates_nametag%' + vertical-offset: -1 + owner-conditions: { } + viewer-conditions: { } + tag_2: + text: "IP: %player_ip% My IP: %viewer_player_ip%" + vertical-offset: -0.7 + refresh-frequency: 10 + check-frequency: 20 + owner-conditions: { } + viewer-conditions: + condition_1: + type: permission + value: 'nameplates.admin' \ No newline at end of file diff --git a/src/main/resources/contents/backgrounds/b0.png b/paper/src/main/resources/contents/backgrounds/b0.png similarity index 100% rename from src/main/resources/contents/backgrounds/b0.png rename to paper/src/main/resources/contents/backgrounds/b0.png diff --git a/src/main/resources/contents/backgrounds/b1.png b/paper/src/main/resources/contents/backgrounds/b1.png similarity index 100% rename from src/main/resources/contents/backgrounds/b1.png rename to paper/src/main/resources/contents/backgrounds/b1.png diff --git a/src/main/resources/contents/backgrounds/b128.png b/paper/src/main/resources/contents/backgrounds/b128.png similarity index 100% rename from src/main/resources/contents/backgrounds/b128.png rename to paper/src/main/resources/contents/backgrounds/b128.png diff --git a/src/main/resources/contents/backgrounds/b16.png b/paper/src/main/resources/contents/backgrounds/b16.png similarity index 100% rename from src/main/resources/contents/backgrounds/b16.png rename to paper/src/main/resources/contents/backgrounds/b16.png diff --git a/src/main/resources/contents/backgrounds/b2.png b/paper/src/main/resources/contents/backgrounds/b2.png similarity index 100% rename from src/main/resources/contents/backgrounds/b2.png rename to paper/src/main/resources/contents/backgrounds/b2.png diff --git a/src/main/resources/contents/backgrounds/b32.png b/paper/src/main/resources/contents/backgrounds/b32.png similarity index 100% rename from src/main/resources/contents/backgrounds/b32.png rename to paper/src/main/resources/contents/backgrounds/b32.png diff --git a/src/main/resources/contents/backgrounds/b4.png b/paper/src/main/resources/contents/backgrounds/b4.png similarity index 100% rename from src/main/resources/contents/backgrounds/b4.png rename to paper/src/main/resources/contents/backgrounds/b4.png diff --git a/src/main/resources/contents/backgrounds/b64.png b/paper/src/main/resources/contents/backgrounds/b64.png similarity index 100% rename from src/main/resources/contents/backgrounds/b64.png rename to paper/src/main/resources/contents/backgrounds/b64.png diff --git a/src/main/resources/contents/backgrounds/b8.png b/paper/src/main/resources/contents/backgrounds/b8.png similarity index 100% rename from src/main/resources/contents/backgrounds/b8.png rename to paper/src/main/resources/contents/backgrounds/b8.png diff --git a/src/main/resources/contents/backgrounds/bedrock_1.yml b/paper/src/main/resources/contents/backgrounds/bedrock_1.yml similarity index 100% rename from src/main/resources/contents/backgrounds/bedrock_1.yml rename to paper/src/main/resources/contents/backgrounds/bedrock_1.yml diff --git a/src/main/resources/contents/backgrounds/bedrock_2.yml b/paper/src/main/resources/contents/backgrounds/bedrock_2.yml similarity index 100% rename from src/main/resources/contents/backgrounds/bedrock_2.yml rename to paper/src/main/resources/contents/backgrounds/bedrock_2.yml diff --git a/src/main/resources/contents/backgrounds/bedrock_3.yml b/paper/src/main/resources/contents/backgrounds/bedrock_3.yml similarity index 100% rename from src/main/resources/contents/backgrounds/bedrock_3.yml rename to paper/src/main/resources/contents/backgrounds/bedrock_3.yml diff --git a/src/main/resources/contents/bubbles/chat.yml b/paper/src/main/resources/contents/bubbles/chat.yml similarity index 100% rename from src/main/resources/contents/bubbles/chat.yml rename to paper/src/main/resources/contents/bubbles/chat.yml diff --git a/src/main/resources/contents/bubbles/chat_left.png b/paper/src/main/resources/contents/bubbles/chat_left.png similarity index 100% rename from src/main/resources/contents/bubbles/chat_left.png rename to paper/src/main/resources/contents/bubbles/chat_left.png diff --git a/src/main/resources/contents/bubbles/chat_middle.png b/paper/src/main/resources/contents/bubbles/chat_middle.png similarity index 100% rename from src/main/resources/contents/bubbles/chat_middle.png rename to paper/src/main/resources/contents/bubbles/chat_middle.png diff --git a/src/main/resources/contents/bubbles/chat_right.png b/paper/src/main/resources/contents/bubbles/chat_right.png similarity index 100% rename from src/main/resources/contents/bubbles/chat_right.png rename to paper/src/main/resources/contents/bubbles/chat_right.png diff --git a/src/main/resources/contents/bubbles/chat_tail.png b/paper/src/main/resources/contents/bubbles/chat_tail.png similarity index 100% rename from src/main/resources/contents/bubbles/chat_tail.png rename to paper/src/main/resources/contents/bubbles/chat_tail.png diff --git a/src/main/resources/contents/images/bell.png b/paper/src/main/resources/contents/images/bell.png similarity index 100% rename from src/main/resources/contents/images/bell.png rename to paper/src/main/resources/contents/images/bell.png diff --git a/src/main/resources/contents/images/bell.yml b/paper/src/main/resources/contents/images/bell.yml similarity index 100% rename from src/main/resources/contents/images/bell.yml rename to paper/src/main/resources/contents/images/bell.yml diff --git a/src/main/resources/contents/images/bubble.png b/paper/src/main/resources/contents/images/bubble.png similarity index 100% rename from src/main/resources/contents/images/bubble.png rename to paper/src/main/resources/contents/images/bubble.png diff --git a/src/main/resources/contents/images/bubble.yml b/paper/src/main/resources/contents/images/bubble.yml similarity index 100% rename from src/main/resources/contents/images/bubble.yml rename to paper/src/main/resources/contents/images/bubble.yml diff --git a/src/main/resources/contents/images/clock.png b/paper/src/main/resources/contents/images/clock.png similarity index 100% rename from src/main/resources/contents/images/clock.png rename to paper/src/main/resources/contents/images/clock.png diff --git a/src/main/resources/contents/images/clock.yml b/paper/src/main/resources/contents/images/clock.yml similarity index 100% rename from src/main/resources/contents/images/clock.yml rename to paper/src/main/resources/contents/images/clock.yml diff --git a/src/main/resources/contents/images/coin.png b/paper/src/main/resources/contents/images/coin.png similarity index 100% rename from src/main/resources/contents/images/coin.png rename to paper/src/main/resources/contents/images/coin.png diff --git a/src/main/resources/contents/images/coin.yml b/paper/src/main/resources/contents/images/coin.yml similarity index 100% rename from src/main/resources/contents/images/coin.yml rename to paper/src/main/resources/contents/images/coin.yml diff --git a/src/main/resources/contents/images/compass.png b/paper/src/main/resources/contents/images/compass.png similarity index 100% rename from src/main/resources/contents/images/compass.png rename to paper/src/main/resources/contents/images/compass.png diff --git a/src/main/resources/contents/images/compass.yml b/paper/src/main/resources/contents/images/compass.yml similarity index 100% rename from src/main/resources/contents/images/compass.yml rename to paper/src/main/resources/contents/images/compass.yml diff --git a/src/main/resources/contents/images/stamina_0.png b/paper/src/main/resources/contents/images/stamina_0.png similarity index 100% rename from src/main/resources/contents/images/stamina_0.png rename to paper/src/main/resources/contents/images/stamina_0.png diff --git a/src/main/resources/contents/images/stamina_0.yml b/paper/src/main/resources/contents/images/stamina_0.yml similarity index 100% rename from src/main/resources/contents/images/stamina_0.yml rename to paper/src/main/resources/contents/images/stamina_0.yml diff --git a/src/main/resources/contents/images/stamina_1.png b/paper/src/main/resources/contents/images/stamina_1.png similarity index 100% rename from src/main/resources/contents/images/stamina_1.png rename to paper/src/main/resources/contents/images/stamina_1.png diff --git a/src/main/resources/contents/images/stamina_1.yml b/paper/src/main/resources/contents/images/stamina_1.yml similarity index 100% rename from src/main/resources/contents/images/stamina_1.yml rename to paper/src/main/resources/contents/images/stamina_1.yml diff --git a/src/main/resources/contents/images/stamina_2.png b/paper/src/main/resources/contents/images/stamina_2.png similarity index 100% rename from src/main/resources/contents/images/stamina_2.png rename to paper/src/main/resources/contents/images/stamina_2.png diff --git a/src/main/resources/contents/images/stamina_2.yml b/paper/src/main/resources/contents/images/stamina_2.yml similarity index 100% rename from src/main/resources/contents/images/stamina_2.yml rename to paper/src/main/resources/contents/images/stamina_2.yml diff --git a/src/main/resources/contents/images/weather.png b/paper/src/main/resources/contents/images/weather.png similarity index 100% rename from src/main/resources/contents/images/weather.png rename to paper/src/main/resources/contents/images/weather.png diff --git a/src/main/resources/contents/images/weather.yml b/paper/src/main/resources/contents/images/weather.yml similarity index 100% rename from src/main/resources/contents/images/weather.yml rename to paper/src/main/resources/contents/images/weather.yml diff --git a/src/main/resources/contents/nameplates/cat.yml b/paper/src/main/resources/contents/nameplates/cat.yml similarity index 100% rename from src/main/resources/contents/nameplates/cat.yml rename to paper/src/main/resources/contents/nameplates/cat.yml diff --git a/src/main/resources/contents/nameplates/cat_left.png b/paper/src/main/resources/contents/nameplates/cat_left.png similarity index 100% rename from src/main/resources/contents/nameplates/cat_left.png rename to paper/src/main/resources/contents/nameplates/cat_left.png diff --git a/src/main/resources/contents/nameplates/cat_middle.png b/paper/src/main/resources/contents/nameplates/cat_middle.png similarity index 100% rename from src/main/resources/contents/nameplates/cat_middle.png rename to paper/src/main/resources/contents/nameplates/cat_middle.png diff --git a/src/main/resources/contents/nameplates/cat_right.png b/paper/src/main/resources/contents/nameplates/cat_right.png similarity index 100% rename from src/main/resources/contents/nameplates/cat_right.png rename to paper/src/main/resources/contents/nameplates/cat_right.png diff --git a/src/main/resources/contents/nameplates/cheems.yml b/paper/src/main/resources/contents/nameplates/cheems.yml similarity index 100% rename from src/main/resources/contents/nameplates/cheems.yml rename to paper/src/main/resources/contents/nameplates/cheems.yml diff --git a/src/main/resources/contents/nameplates/cheems_left.png b/paper/src/main/resources/contents/nameplates/cheems_left.png similarity index 100% rename from src/main/resources/contents/nameplates/cheems_left.png rename to paper/src/main/resources/contents/nameplates/cheems_left.png diff --git a/src/main/resources/contents/nameplates/cheems_middle.png b/paper/src/main/resources/contents/nameplates/cheems_middle.png similarity index 100% rename from src/main/resources/contents/nameplates/cheems_middle.png rename to paper/src/main/resources/contents/nameplates/cheems_middle.png diff --git a/src/main/resources/contents/nameplates/cheems_right.png b/paper/src/main/resources/contents/nameplates/cheems_right.png similarity index 100% rename from src/main/resources/contents/nameplates/cheems_right.png rename to paper/src/main/resources/contents/nameplates/cheems_right.png diff --git a/src/main/resources/contents/nameplates/egg.yml b/paper/src/main/resources/contents/nameplates/egg.yml similarity index 100% rename from src/main/resources/contents/nameplates/egg.yml rename to paper/src/main/resources/contents/nameplates/egg.yml diff --git a/src/main/resources/contents/nameplates/egg_left.png b/paper/src/main/resources/contents/nameplates/egg_left.png similarity index 100% rename from src/main/resources/contents/nameplates/egg_left.png rename to paper/src/main/resources/contents/nameplates/egg_left.png diff --git a/src/main/resources/contents/nameplates/egg_middle.png b/paper/src/main/resources/contents/nameplates/egg_middle.png similarity index 100% rename from src/main/resources/contents/nameplates/egg_middle.png rename to paper/src/main/resources/contents/nameplates/egg_middle.png diff --git a/src/main/resources/contents/nameplates/egg_right.png b/paper/src/main/resources/contents/nameplates/egg_right.png similarity index 100% rename from src/main/resources/contents/nameplates/egg_right.png rename to paper/src/main/resources/contents/nameplates/egg_right.png diff --git a/src/main/resources/contents/nameplates/halloween.yml b/paper/src/main/resources/contents/nameplates/halloween.yml similarity index 100% rename from src/main/resources/contents/nameplates/halloween.yml rename to paper/src/main/resources/contents/nameplates/halloween.yml diff --git a/src/main/resources/contents/nameplates/halloween_left.png b/paper/src/main/resources/contents/nameplates/halloween_left.png similarity index 100% rename from src/main/resources/contents/nameplates/halloween_left.png rename to paper/src/main/resources/contents/nameplates/halloween_left.png diff --git a/src/main/resources/contents/nameplates/halloween_middle.png b/paper/src/main/resources/contents/nameplates/halloween_middle.png similarity index 100% rename from src/main/resources/contents/nameplates/halloween_middle.png rename to paper/src/main/resources/contents/nameplates/halloween_middle.png diff --git a/src/main/resources/contents/nameplates/halloween_right.png b/paper/src/main/resources/contents/nameplates/halloween_right.png similarity index 100% rename from src/main/resources/contents/nameplates/halloween_right.png rename to paper/src/main/resources/contents/nameplates/halloween_right.png diff --git a/src/main/resources/contents/nameplates/hutao.yml b/paper/src/main/resources/contents/nameplates/hutao.yml similarity index 100% rename from src/main/resources/contents/nameplates/hutao.yml rename to paper/src/main/resources/contents/nameplates/hutao.yml diff --git a/src/main/resources/contents/nameplates/hutao_left.png b/paper/src/main/resources/contents/nameplates/hutao_left.png similarity index 100% rename from src/main/resources/contents/nameplates/hutao_left.png rename to paper/src/main/resources/contents/nameplates/hutao_left.png diff --git a/src/main/resources/contents/nameplates/hutao_middle.png b/paper/src/main/resources/contents/nameplates/hutao_middle.png similarity index 100% rename from src/main/resources/contents/nameplates/hutao_middle.png rename to paper/src/main/resources/contents/nameplates/hutao_middle.png diff --git a/src/main/resources/contents/nameplates/hutao_right.png b/paper/src/main/resources/contents/nameplates/hutao_right.png similarity index 100% rename from src/main/resources/contents/nameplates/hutao_right.png rename to paper/src/main/resources/contents/nameplates/hutao_right.png diff --git a/src/main/resources/contents/nameplates/rabbit.yml b/paper/src/main/resources/contents/nameplates/rabbit.yml similarity index 100% rename from src/main/resources/contents/nameplates/rabbit.yml rename to paper/src/main/resources/contents/nameplates/rabbit.yml diff --git a/src/main/resources/contents/nameplates/rabbit_left.png b/paper/src/main/resources/contents/nameplates/rabbit_left.png similarity index 100% rename from src/main/resources/contents/nameplates/rabbit_left.png rename to paper/src/main/resources/contents/nameplates/rabbit_left.png diff --git a/src/main/resources/contents/nameplates/rabbit_middle.png b/paper/src/main/resources/contents/nameplates/rabbit_middle.png similarity index 100% rename from src/main/resources/contents/nameplates/rabbit_middle.png rename to paper/src/main/resources/contents/nameplates/rabbit_middle.png diff --git a/src/main/resources/contents/nameplates/rabbit_right.png b/paper/src/main/resources/contents/nameplates/rabbit_right.png similarity index 100% rename from src/main/resources/contents/nameplates/rabbit_right.png rename to paper/src/main/resources/contents/nameplates/rabbit_right.png diff --git a/src/main/resources/contents/nameplates/starsky.yml b/paper/src/main/resources/contents/nameplates/starsky.yml similarity index 100% rename from src/main/resources/contents/nameplates/starsky.yml rename to paper/src/main/resources/contents/nameplates/starsky.yml diff --git a/src/main/resources/contents/nameplates/starsky_left.png b/paper/src/main/resources/contents/nameplates/starsky_left.png similarity index 100% rename from src/main/resources/contents/nameplates/starsky_left.png rename to paper/src/main/resources/contents/nameplates/starsky_left.png diff --git a/src/main/resources/contents/nameplates/starsky_middle.png b/paper/src/main/resources/contents/nameplates/starsky_middle.png similarity index 100% rename from src/main/resources/contents/nameplates/starsky_middle.png rename to paper/src/main/resources/contents/nameplates/starsky_middle.png diff --git a/src/main/resources/contents/nameplates/starsky_right.png b/paper/src/main/resources/contents/nameplates/starsky_right.png similarity index 100% rename from src/main/resources/contents/nameplates/starsky_right.png rename to paper/src/main/resources/contents/nameplates/starsky_right.png diff --git a/src/main/resources/contents/nameplates/trident.yml b/paper/src/main/resources/contents/nameplates/trident.yml similarity index 100% rename from src/main/resources/contents/nameplates/trident.yml rename to paper/src/main/resources/contents/nameplates/trident.yml diff --git a/src/main/resources/contents/nameplates/trident_left.png b/paper/src/main/resources/contents/nameplates/trident_left.png similarity index 100% rename from src/main/resources/contents/nameplates/trident_left.png rename to paper/src/main/resources/contents/nameplates/trident_left.png diff --git a/src/main/resources/contents/nameplates/trident_middle.png b/paper/src/main/resources/contents/nameplates/trident_middle.png similarity index 100% rename from src/main/resources/contents/nameplates/trident_middle.png rename to paper/src/main/resources/contents/nameplates/trident_middle.png diff --git a/src/main/resources/contents/nameplates/trident_right.png b/paper/src/main/resources/contents/nameplates/trident_right.png similarity index 100% rename from src/main/resources/contents/nameplates/trident_right.png rename to paper/src/main/resources/contents/nameplates/trident_right.png diff --git a/src/main/resources/contents/nameplates/wither.yml b/paper/src/main/resources/contents/nameplates/wither.yml similarity index 100% rename from src/main/resources/contents/nameplates/wither.yml rename to paper/src/main/resources/contents/nameplates/wither.yml diff --git a/src/main/resources/contents/nameplates/wither_left.png b/paper/src/main/resources/contents/nameplates/wither_left.png similarity index 100% rename from src/main/resources/contents/nameplates/wither_left.png rename to paper/src/main/resources/contents/nameplates/wither_left.png diff --git a/src/main/resources/contents/nameplates/wither_middle.png b/paper/src/main/resources/contents/nameplates/wither_middle.png similarity index 100% rename from src/main/resources/contents/nameplates/wither_middle.png rename to paper/src/main/resources/contents/nameplates/wither_middle.png diff --git a/src/main/resources/contents/nameplates/wither_right.png b/paper/src/main/resources/contents/nameplates/wither_right.png similarity index 100% rename from src/main/resources/contents/nameplates/wither_right.png rename to paper/src/main/resources/contents/nameplates/wither_right.png diff --git a/src/main/resources/contents/nameplates/xmas.yml b/paper/src/main/resources/contents/nameplates/xmas.yml similarity index 100% rename from src/main/resources/contents/nameplates/xmas.yml rename to paper/src/main/resources/contents/nameplates/xmas.yml diff --git a/src/main/resources/contents/nameplates/xmas_left.png b/paper/src/main/resources/contents/nameplates/xmas_left.png similarity index 100% rename from src/main/resources/contents/nameplates/xmas_left.png rename to paper/src/main/resources/contents/nameplates/xmas_left.png diff --git a/src/main/resources/contents/nameplates/xmas_middle.png b/paper/src/main/resources/contents/nameplates/xmas_middle.png similarity index 100% rename from src/main/resources/contents/nameplates/xmas_middle.png rename to paper/src/main/resources/contents/nameplates/xmas_middle.png diff --git a/src/main/resources/contents/nameplates/xmas_right.png b/paper/src/main/resources/contents/nameplates/xmas_right.png similarity index 100% rename from src/main/resources/contents/nameplates/xmas_right.png rename to paper/src/main/resources/contents/nameplates/xmas_right.png diff --git a/paper/src/main/resources/database.yml b/paper/src/main/resources/database.yml new file mode 100644 index 0000000..7b882a4 --- /dev/null +++ b/paper/src/main/resources/database.yml @@ -0,0 +1,77 @@ +# file: +# JSON +# YAML +# +# local database: +# SQLite +# H2 (preferred over SQLite) +# +# remote database: +# MySQL +# MariaDB (preferred over MySQL) +# MongoDB +# +data-storage-method: H2 + +SQLite: + file: 'sqlite' + table-prefix: nameplates + +H2: + file: 'h2' + table-prefix: nameplates + +MySQL: + host: 'localhost' + port: '3306' + user: 'root' + password: 'password' + database: 'minecraft' + connection-parameters: '?autoReconnect=true&useSSL=false&useUnicode=true&characterEncoding=UTF-8' + Pool-Settings: + max-pool-size: 10 + min-idle: 10 + max-lifetime: 180000 + keep-alive-time: 60000 + time-out: 20000 + table-prefix: nameplates + +MariaDB: + host: 'localhost' + port: '3306' + user: 'root' + password: 'password' + database: 'minecraft' + connection-parameters: '?autoReconnect=true&useSSL=false&useUnicode=true&characterEncoding=UTF-8' + Pool-Settings: + max-pool-size: 10 + min-idle: 10 + max-lifetime: 180000 + keep-alive-time: 60000 + time-out: 20000 + table-prefix: nameplates + +MongoDB: + host: 'localhost' + port: '27017' + #user: 'root' + #password: 'password' + database: 'minecraft' + # If this section is not empty, it would override the configs above + # https://www.mongodb.com/docs/manual/reference/connection-string/ + connection-uri: '' + collection-prefix: nameplates + +# Redis is optional if you are using remote database +# If you do not know how to use Redis, please do not enable it +Redis: + enable: false + host: localhost + #password: 123456 + port: 6379 + use-ssl: false + MaxTotal: 10 + MaxIdle: 10 + MinIdle: 1 + MaxWaitMillis: 30000 + MinEvictableIdleTimeMillis: 1800000 \ No newline at end of file diff --git a/src/main/resources/messages/chinese.yml b/paper/src/main/resources/messages/chinese.yml similarity index 85% rename from src/main/resources/messages/chinese.yml rename to paper/src/main/resources/messages/chinese.yml index 3742fe3..a68c7cc 100644 --- a/src/main/resources/messages/chinese.yml +++ b/paper/src/main/resources/messages/chinese.yml @@ -1,12 +1,8 @@ +config-version: "23" + messages: prefix: '[CustomNameplates] ' - no-perm: '你没有权限!' - lack-args: '参数不足!' - none-args: '非空参数' - invalid-args: '无效参数' reload: '插件已重载! 请不要忘记重载你的资源包.' - not-online: '玩家 {Player} 不在线!' - no-console: '这个指令不能由控制台执行!' cooldown: '上一个预览还没结束!' generate: '正在生成资源包...' generate-done: '资源包生成完毕...' diff --git a/src/main/resources/messages/english.yml b/paper/src/main/resources/messages/english.yml similarity index 85% rename from src/main/resources/messages/english.yml rename to paper/src/main/resources/messages/english.yml index cab94f9..be66755 100644 --- a/src/main/resources/messages/english.yml +++ b/paper/src/main/resources/messages/english.yml @@ -1,12 +1,8 @@ +config-version: "23" + messages: prefix: '[CustomNameplates] ' - no-perm: 'No Permission!' - lack-args: 'Insufficient parameters!' reload: 'Reloaded! Don''t forget to reinstall your resource pack.' - none-args: 'Non arguments' - invalid-args: 'Invalid arguments' - not-online: 'Player {Player} is not online!' - no-console: 'This command can be only executed by player!' cooldown: 'Previewing is still Ongoing!' generate: 'Resource Pack is generating..' generate-done: 'Resource Pack has been generated!' diff --git a/src/main/resources/messages/french.yml b/paper/src/main/resources/messages/french.yml similarity index 85% rename from src/main/resources/messages/french.yml rename to paper/src/main/resources/messages/french.yml index 0444843..981edc2 100644 --- a/src/main/resources/messages/french.yml +++ b/paper/src/main/resources/messages/french.yml @@ -1,12 +1,8 @@ +config-version: "23" + messages: prefix: '[CustomNameplates] ' - no-perm: 'Pas de permission !' - lack-args: 'Paramètres insuffisants !' reload: 'Rechargé ! N'oubliez pas de réinstaller votre pack de ressources.' - none-args: 'Aucun argument' - invalid-args: 'Arguments invalides' - not-online: 'Le joueur {Player} n'est pas en ligne !' - no-console: 'Cette commande ne peut être exécutée que par un joueur !' cooldown: 'La prévisualisation est toujours en cours !' generate: 'Le pack de ressources est en cours de génération...' generate-done: 'Le pack de ressources a été généré !' diff --git a/src/main/resources/messages/russian.yml b/paper/src/main/resources/messages/russian.yml similarity index 84% rename from src/main/resources/messages/russian.yml rename to paper/src/main/resources/messages/russian.yml index 29e0cf7..54ee91d 100644 --- a/src/main/resources/messages/russian.yml +++ b/paper/src/main/resources/messages/russian.yml @@ -1,12 +1,8 @@ +config-version: "23" + messages: prefix: '[CustomNameplates] ' - no-perm: 'Нет прав!' - lack-args: 'Недостаточно параметров!' reload: 'Перезагружено! Не забудьте переустановить свой ресурспак.' - none-args: 'В команде обязательны аргументы.' - invalid-args: 'Неверные аргументы' - not-online: 'Игрок {Player} не в сети!' - no-console: 'Эта команда может быть использована только игроком!' cooldown: 'Предпросмотр до сих пор активен!' generate: 'Ресурспак генерируется...' generate-done: 'Ресурспак сгенерирован!' diff --git a/src/main/resources/messages/spanish.yml b/paper/src/main/resources/messages/spanish.yml similarity index 84% rename from src/main/resources/messages/spanish.yml rename to paper/src/main/resources/messages/spanish.yml index 0ff4e30..9ff3662 100644 --- a/src/main/resources/messages/spanish.yml +++ b/paper/src/main/resources/messages/spanish.yml @@ -1,12 +1,8 @@ +config-version: "23" + messages: prefix: '[CustomNameplates] ' - no-perm: 'No tienes permisos!' - lack-args: '¡Parámetros insuficientes!' reload: 'Recargado! No olvides reinstalar el paquete de recursos.' - none-args: '¡Ningún argumento!' - invalid-args: 'Argumentos no válidos' - not-online: 'Jugador {Player} no está en línea!' - no-console: 'Este comando sólo puede ser ejecutado por el jugador.' cooldown: '¡El preestreno sigue en curso!' generate: 'El paquete de recursos está generando..' generate-done: 'Se ha generado un paquete de recursos.' diff --git a/src/main/resources/messages/turkish.yml b/paper/src/main/resources/messages/turkish.yml similarity index 85% rename from src/main/resources/messages/turkish.yml rename to paper/src/main/resources/messages/turkish.yml index 0988729..ef37d12 100644 --- a/src/main/resources/messages/turkish.yml +++ b/paper/src/main/resources/messages/turkish.yml @@ -1,13 +1,9 @@ +config-version: "23" + #Turkish language file by WinTone01 messages: prefix: '[CustomNameplates] ' - no-perm: 'İzin Yok!' - lack-args: 'Yetersiz parametreler!' reload: 'Yeniden Yüklendi! Kaynak paketinizi yeniden yüklemeyi unutmayın.' - none-args: 'Argüman bulunmuyor' - invalid-args: 'Geçersiz argümanlar' - not-online: 'Oyuncu {Player} çevrimiçi değil!' - no-console: 'Bu komut sadece oyuncular tarafından kullanılabilir!' cooldown: 'Önizleme hala devam ediyor!' generate: 'Kaynak paketi oluşturuluyor...' generate-done: 'Kaynak paketi oluşturuldu!' diff --git a/paper/src/main/resources/plugin.yml b/paper/src/main/resources/plugin.yml new file mode 100644 index 0000000..f740ea6 --- /dev/null +++ b/paper/src/main/resources/plugin.yml @@ -0,0 +1,7 @@ +name: CustomNameplates +version: '${version}' +main: net.momirealms.customnameplates.paper.CustomNameplatesPluginImpl +api-version: 1.17 +authors: [ XiaoMoMi ] +folia-supported: true +depend: [ ProtocolLib ,PlaceholderAPI ] \ No newline at end of file diff --git a/paper/src/main/resources/schema/h2.sql b/paper/src/main/resources/schema/h2.sql new file mode 100644 index 0000000..578ec71 --- /dev/null +++ b/paper/src/main/resources/schema/h2.sql @@ -0,0 +1,6 @@ +CREATE TABLE IF NOT EXISTS `{prefix}_data` +( + `uuid` char(36) NOT NULL UNIQUE, + `data` longblob NOT NULL, + PRIMARY KEY (`uuid`) +); \ No newline at end of file diff --git a/paper/src/main/resources/schema/mariadb.sql b/paper/src/main/resources/schema/mariadb.sql new file mode 100644 index 0000000..efef31d --- /dev/null +++ b/paper/src/main/resources/schema/mariadb.sql @@ -0,0 +1,10 @@ +SET DEFAULT_STORAGE_ENGINE = INNODB; + +CREATE TABLE IF NOT EXISTS `{prefix}_data` +( + `uuid` char(36) NOT NULL UNIQUE, + `data` longblob NOT NULL, + PRIMARY KEY (`uuid`) +) ENGINE = InnoDB + DEFAULT CHARSET = utf8mb4 + COLLATE = utf8mb4_unicode_ci; \ No newline at end of file diff --git a/paper/src/main/resources/schema/mysql.sql b/paper/src/main/resources/schema/mysql.sql new file mode 100644 index 0000000..578ec71 --- /dev/null +++ b/paper/src/main/resources/schema/mysql.sql @@ -0,0 +1,6 @@ +CREATE TABLE IF NOT EXISTS `{prefix}_data` +( + `uuid` char(36) NOT NULL UNIQUE, + `data` longblob NOT NULL, + PRIMARY KEY (`uuid`) +); \ No newline at end of file diff --git a/paper/src/main/resources/schema/sqlite.sql b/paper/src/main/resources/schema/sqlite.sql new file mode 100644 index 0000000..578ec71 --- /dev/null +++ b/paper/src/main/resources/schema/sqlite.sql @@ -0,0 +1,6 @@ +CREATE TABLE IF NOT EXISTS `{prefix}_data` +( + `uuid` char(36) NOT NULL UNIQUE, + `data` longblob NOT NULL, + PRIMARY KEY (`uuid`) +); \ No newline at end of file diff --git a/src/main/resources/space_split.png b/paper/src/main/resources/space_split.png similarity index 100% rename from src/main/resources/space_split.png rename to paper/src/main/resources/space_split.png diff --git a/settings.gradle b/settings.gradle deleted file mode 100644 index 6529a22..0000000 --- a/settings.gradle +++ /dev/null @@ -1 +0,0 @@ -rootProject.name = 'CustomNameplates' diff --git a/settings.gradle.kts b/settings.gradle.kts new file mode 100644 index 0000000..1c3a374 --- /dev/null +++ b/settings.gradle.kts @@ -0,0 +1,5 @@ +rootProject.name = "CustomNameplates" +include("api") +include("bungeecord") +include("velocity") +include("paper") diff --git a/src/main/java/net/momirealms/customnameplates/CustomNameplates.java b/src/main/java/net/momirealms/customnameplates/CustomNameplates.java deleted file mode 100644 index ac3ec07..0000000 --- a/src/main/java/net/momirealms/customnameplates/CustomNameplates.java +++ /dev/null @@ -1,244 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates; - -import com.comphenix.protocol.PacketType; -import com.comphenix.protocol.ProtocolLibrary; -import com.comphenix.protocol.ProtocolManager; -import com.comphenix.protocol.events.PacketContainer; -import net.kyori.adventure.platform.bukkit.BukkitAudiences; -import net.momirealms.customnameplates.api.CustomNameplatesAPI; -import net.momirealms.customnameplates.command.BubblesCommand; -import net.momirealms.customnameplates.command.NameplateCommand; -import net.momirealms.customnameplates.helper.LibraryLoader; -import net.momirealms.customnameplates.helper.VersionHelper; -import net.momirealms.customnameplates.manager.*; -import net.momirealms.customnameplates.object.scheduler.Scheduler; -import net.momirealms.customnameplates.object.scheduler.SchedulerPlatform; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bstats.bukkit.Metrics; -import org.bukkit.Bukkit; -import org.bukkit.command.PluginCommand; -import org.bukkit.plugin.java.JavaPlugin; - -import java.util.TimeZone; - -public final class CustomNameplates extends JavaPlugin { - - private static CustomNameplates plugin; - private static BukkitAudiences adventure; - private static ProtocolManager protocolManager; - private ResourceManager resourceManager; - private BossBarManager bossBarManager; - private ActionBarManager actionBarManager; - private PlaceholderManager placeholderManager; - private NameplateManager nameplateManager; - private ChatBubblesManager chatBubblesManager; - private DataManager dataManager; - private ConfigManager configManager; - private MessageManager messageManager; - private FontManager fontManager; - private VersionHelper versionHelper; - private TeamManager teamManager; - private BackgroundManager backgroundManager; - private ImageManager imageManager; - private CustomNameplatesAPI customNameplatesAPI; - private Scheduler scheduler; - - @Override - public void onLoad(){ - plugin = this; - loadLibs(); - } - - @Override - public void onEnable() { - adventure = BukkitAudiences.create(this); - protocolManager = ProtocolLibrary.getProtocolManager(); - AdventureUtils.consoleMessage("[CustomNameplates] Running on " + Bukkit.getVersion()); - this.fix(); - this.versionHelper = new VersionHelper(this); - this.scheduler = new Scheduler(this); - this.configManager = new ConfigManager(); - this.messageManager = new MessageManager(); - this.teamManager = new TeamManager(this); - this.placeholderManager = new PlaceholderManager(this); - this.actionBarManager = new ActionBarManager(this); - this.bossBarManager = new BossBarManager(this); - this.resourceManager = new ResourceManager(this); - this.dataManager = new DataManager(this); - this.nameplateManager = new NameplateManager(this); - this.backgroundManager = new BackgroundManager(this); - this.fontManager = new FontManager(this); - this.imageManager = new ImageManager(this); - this.chatBubblesManager = new ChatBubblesManager(this); - this.customNameplatesAPI = new CustomNameplatesAPI(this); - this.customNameplatesAPI.init(); - this.registerCommands(); - this.reload(false); - AdventureUtils.consoleMessage("[CustomNameplates] Plugin Enabled!"); - if (ConfigManager.enableBStats) new Metrics(this, 16649); - if (ConfigManager.checkUpdate) this.versionHelper.checkUpdate(); - if (ConfigManager.generatePackOnStart) resourceManager.generateResourcePack(); - } - - @Override - public void onDisable() { - if (actionBarManager != null) actionBarManager.unload(); - if (nameplateManager != null) nameplateManager.unload(); - if (bossBarManager != null) bossBarManager.unload(); - if (chatBubblesManager != null) chatBubblesManager.unload(); - if (placeholderManager != null) placeholderManager.unload(); - if (fontManager != null) fontManager.unload(); - if (teamManager != null) teamManager.unload(); - if (imageManager != null) imageManager.unload(); - if (backgroundManager != null) backgroundManager.unload(); - if (dataManager != null) dataManager.disable(); - if (adventure != null) adventure.close(); - } - - private void loadLibs() { - TimeZone timeZone = TimeZone.getDefault(); - String libRepo = timeZone.getID().startsWith("Asia") ? "https://maven.aliyun.com/repository/public/" : "https://repo.maven.apache.org/maven2/"; - LibraryLoader.load("commons-io","commons-io","2.13.0", libRepo); - LibraryLoader.load("org.apache.commons","commons-lang3","3.13.0", libRepo); - LibraryLoader.load("com.zaxxer","HikariCP","5.0.1", libRepo); - LibraryLoader.load("dev.dejvokep","boosted-yaml","1.3.1", libRepo); - LibraryLoader.load("org.mariadb.jdbc","mariadb-java-client","3.1.4", libRepo); - LibraryLoader.load("com.mysql","mysql-connector-j","8.2.0", libRepo); - } - - private void registerCommands() { - NameplateCommand nameplateCommand = new NameplateCommand(); - PluginCommand main = Bukkit.getPluginCommand("customnameplates"); - if (main != null) { - main.setExecutor(nameplateCommand); - main.setTabCompleter(nameplateCommand); - } - BubblesCommand bubblesCommand = new BubblesCommand(); - PluginCommand bubble = Bukkit.getPluginCommand("bubbles"); - if (bubble != null) { - bubble.setExecutor(bubblesCommand); - bubble.setTabCompleter(bubblesCommand); - } - } - - private void fix() { - //Don't delete this, a temp fix for a certain version of ProtocolLib - new PacketContainer(PacketType.Play.Server.SCOREBOARD_TEAM); - } - - public void reload(boolean generate) { - configManager.unload(); - messageManager.unload(); - bossBarManager.unload(); - actionBarManager.unload(); - placeholderManager.unload(); - nameplateManager.unload(); - teamManager.unload(); - chatBubblesManager.unload(); - imageManager.unload(); - fontManager.unload(); - backgroundManager.unload(); - dataManager.unload(); - configManager.load(); - messageManager.load(); - dataManager.load(); - // image manager must load before font manager - imageManager.load(); - fontManager.load(); - // team manager must load before nameplates manager - teamManager.load(); - nameplateManager.load(); - chatBubblesManager.load(); - backgroundManager.load(); - bossBarManager.load(); - actionBarManager.load(); - placeholderManager.load(); - - if (generate) resourceManager.generateResourcePack(); - } - - public static CustomNameplates getInstance() { - return plugin; - } - - public static BukkitAudiences getAdventure() { - return adventure; - } - - public static ProtocolManager getProtocolManager() { - return protocolManager; - } - - public ResourceManager getResourceManager() { - return this.resourceManager; - } - - public DataManager getDataManager() { - return this.dataManager; - } - - public PlaceholderManager getPlaceholderManager() { - return placeholderManager; - } - - public BossBarManager getBossBarManager() { - return bossBarManager; - } - - public ActionBarManager getActionBarManager() { - return actionBarManager; - } - - public NameplateManager getNameplateManager() { - return nameplateManager; - } - - public ChatBubblesManager getChatBubblesManager() { - return chatBubblesManager; - } - - public VersionHelper getVersionHelper() { - return versionHelper; - } - - public FontManager getFontManager() { - return fontManager; - } - - public TeamManager getTeamManager() { - return teamManager; - } - - public BackgroundManager getBackgroundManager() { - return backgroundManager; - } - - public ImageManager getImageManager() { - return imageManager; - } - - public CustomNameplatesAPI getAPI() { - return customNameplatesAPI; - } - - public SchedulerPlatform getScheduler() { - return scheduler.getInstance(); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/api/CustomNameplatesAPI.java b/src/main/java/net/momirealms/customnameplates/api/CustomNameplatesAPI.java deleted file mode 100644 index 70959d0..0000000 --- a/src/main/java/net/momirealms/customnameplates/api/CustomNameplatesAPI.java +++ /dev/null @@ -1,92 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.api; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.nameplate.NameplatesTeam; -import org.bukkit.entity.Player; - -public class CustomNameplatesAPI { - - private static CustomNameplatesAPI api; - private final CustomNameplates plugin; - - public CustomNameplatesAPI(CustomNameplates plugin) { - this.plugin = plugin; - } - - public void init() { - api = this; - } - - public static CustomNameplatesAPI getInstance() { - return api; - } - - @Deprecated - public static CustomNameplatesAPI getAPI() { - return api; - } - - public boolean doesNameplateExist(String nameplate) { - return plugin.getNameplateManager().existNameplate(nameplate); - } - - public boolean doesBubbleExist(String bubble) { - return plugin.getChatBubblesManager().existBubble(bubble); - } - - public void equipNameplate(Player player, String nameplate) { - plugin.getDataManager().equipNameplate(player, nameplate); - updateAndSave(player); - } - - private void updateAndSave(Player player) { - updateNameplateTeam(player); - plugin.getScheduler().runTaskAsync(() -> { - plugin.getDataManager().saveData(player); - }); - } - - public void equipBubble(Player player, String bubble) { - plugin.getDataManager().equipBubble(player, bubble); - plugin.getScheduler().runTaskAsync(() -> { - plugin.getDataManager().saveData(player); - }); - } - - public void unEquipNameplate(Player player) { - plugin.getDataManager().equipNameplate(player, "none"); - updateAndSave(player); - } - - public void unEquipBubble(Player player) { - plugin.getDataManager().equipBubble(player, "none"); - plugin.getScheduler().runTaskAsync(() -> { - plugin.getDataManager().saveData(player); - }); - } - - public void updateNameplateTeam(Player player) { - NameplatesTeam nameplatesTeam = plugin.getTeamManager().getNameplateTeam(player.getUniqueId()); - if (nameplatesTeam != null) { - nameplatesTeam.update(true); - plugin.getTeamManager().sendUpdateToAll(player, true); - } - } -} diff --git a/src/main/java/net/momirealms/customnameplates/api/events/BubblesEvent.java b/src/main/java/net/momirealms/customnameplates/api/events/BubblesEvent.java deleted file mode 100644 index 8f658ae..0000000 --- a/src/main/java/net/momirealms/customnameplates/api/events/BubblesEvent.java +++ /dev/null @@ -1,75 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.api.events; - -import org.bukkit.entity.Player; -import org.bukkit.event.Cancellable; -import org.bukkit.event.HandlerList; -import org.bukkit.event.player.PlayerEvent; -import org.jetbrains.annotations.NotNull; - -public class BubblesEvent extends PlayerEvent implements Cancellable { - - private boolean cancelled; - private String bubble; - private String text; - private static final HandlerList handlerList = new HandlerList(); - - public BubblesEvent(@NotNull Player who, String bubble, String text) { - super(who); - this.cancelled = false; - this.bubble = bubble; - this.text = text; - } - - @Override - public boolean isCancelled() { - return cancelled; - } - - @Override - public void setCancelled(boolean cancel) { - cancelled = cancel; - } - - public static HandlerList getHandlerList() { - return handlerList; - } - - @NotNull - @Override - public HandlerList getHandlers() { - return getHandlerList(); - } - - public String getBubble() { - return bubble; - } - - public void setBubble(String bubble) { - this.bubble = bubble; - } - - public String getText() { - return text; - } - - public void setText(String text) { - this.text = text; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/bungeecord/BungeeConfigManager.java b/src/main/java/net/momirealms/customnameplates/bungeecord/BungeeConfigManager.java deleted file mode 100644 index 7a9f1e0..0000000 --- a/src/main/java/net/momirealms/customnameplates/bungeecord/BungeeConfigManager.java +++ /dev/null @@ -1,40 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.bungeecord; - -public class BungeeConfigManager { - - private final CustomNameplatesBC plugin; - - public BungeeConfigManager(CustomNameplatesBC plugin) { - this.plugin = plugin; - } - - private boolean tab; - - public void load() { - tab = plugin.getProxy().getPluginManager().getPlugin("TAB") != null; - if (tab) { - plugin.getLogger().info("TAB hooked"); - } - } - - public boolean isTab() { - return tab; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/bungeecord/BungeeEventListener.java b/src/main/java/net/momirealms/customnameplates/bungeecord/BungeeEventListener.java deleted file mode 100644 index 35d6cb6..0000000 --- a/src/main/java/net/momirealms/customnameplates/bungeecord/BungeeEventListener.java +++ /dev/null @@ -1,71 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.bungeecord; - -import com.google.common.io.ByteArrayDataInput; -import com.google.common.io.ByteArrayDataOutput; -import com.google.common.io.ByteStreams; -import me.neznamy.tab.api.TabAPI; -import me.neznamy.tab.api.TabPlayer; -import me.neznamy.tab.api.tablist.SortingManager; -import net.md_5.bungee.api.connection.ProxiedPlayer; -import net.md_5.bungee.api.event.PluginMessageEvent; -import net.md_5.bungee.api.plugin.Listener; -import net.md_5.bungee.event.EventHandler; - -import java.util.Objects; -import java.util.Optional; - -public class BungeeEventListener implements Listener { - - private final CustomNameplatesBC plugin; - private final SortingManager sortingManager; - - public BungeeEventListener (CustomNameplatesBC plugin) { - this.plugin = plugin; - this.sortingManager = TabAPI.getInstance().getSortingManager(); - } - - @EventHandler - @SuppressWarnings("UnstableApiUsage") - public void onReceived(PluginMessageEvent event) { - String channel = event.getTag(); - if (event.isCancelled() || !Objects.equals("customnameplates:cnp", channel)) { - return; - } - ByteArrayDataInput dataInput = ByteStreams.newDataInput(event.getData()); - parseMessage(dataInput); - } - - @SuppressWarnings("UnstableApiUsage") - private void parseMessage(ByteArrayDataInput dataInput) { - String playerName = dataInput.readUTF(); - String teamName = playerName; - if (plugin.getBungeeConfig().isTab()) { - TabPlayer tabPlayer = TabAPI.getInstance().getPlayer(playerName); - if (tabPlayer != null) { - teamName = Optional.ofNullable(sortingManager.getOriginalTeamName(tabPlayer)).orElse(playerName); - } - } - ProxiedPlayer proxiedPlayer = plugin.getProxy().getPlayer(playerName); - ByteArrayDataOutput byteArrayDataOutput = ByteStreams.newDataOutput(); - byteArrayDataOutput.writeUTF(playerName); - byteArrayDataOutput.writeUTF(teamName); - proxiedPlayer.getServer().sendData("customnameplates:cnp", byteArrayDataOutput.toByteArray()); - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/bungeecord/CustomNameplatesBC.java b/src/main/java/net/momirealms/customnameplates/bungeecord/CustomNameplatesBC.java deleted file mode 100644 index 23daf86..0000000 --- a/src/main/java/net/momirealms/customnameplates/bungeecord/CustomNameplatesBC.java +++ /dev/null @@ -1,51 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.bungeecord; - -import net.md_5.bungee.api.plugin.Plugin; - -public class CustomNameplatesBC extends Plugin { - - public static CustomNameplatesBC bungeePlugin; - private BungeeEventListener bungeeEventListener; - private BungeeConfigManager bungeeConfigManager; - - @Override - public void onEnable() { - bungeePlugin = this; - this.bungeeEventListener = new BungeeEventListener(this); - this.getProxy().registerChannel("customnameplates:cnp"); - this.getProxy().getPluginManager().registerListener(this, bungeeEventListener); - this.bungeeConfigManager = new BungeeConfigManager(this); - this.bungeeConfigManager.load(); - } - - @Override - public void onDisable() { - this.getProxy().unregisterChannel("customnameplates:cnp"); - this.getProxy().getPluginManager().unregisterListener(bungeeEventListener); - } - - public BungeeConfigManager getBungeeConfig() { - return bungeeConfigManager; - } - - public static CustomNameplatesBC getPlugin() { - return bungeePlugin; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/AbstractMainCommand.java b/src/main/java/net/momirealms/customnameplates/command/AbstractMainCommand.java deleted file mode 100644 index 8fb1b5d..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/AbstractMainCommand.java +++ /dev/null @@ -1,81 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command; - -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.command.Command; -import org.bukkit.command.CommandSender; -import org.bukkit.command.TabExecutor; -import org.jetbrains.annotations.NotNull; - -import java.util.*; -import java.util.concurrent.ConcurrentHashMap; - -public abstract class AbstractMainCommand implements TabExecutor { - - protected final Map subCommandMap; - protected String name; - - public AbstractMainCommand(String name) { - this.name = name; - this.subCommandMap = new ConcurrentHashMap<>(); - } - - public boolean onCommand(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) { - List argList = Arrays.asList(args); - if (argList.size() < 1) { - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.nonArgs); - return true; - } - AbstractSubCommand subCommand = subCommandMap.get(argList.get(0)); - if (subCommand != null) { - if (!sender.hasPermission(name + "." + subCommand.getSubCommand())) { - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.noPerm); - return true; - } - return subCommand.onCommand(sender, argList.subList(1, argList.size())); - } else { - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.unavailableArgs); - return true; - } - } - - @Override - public List onTabComplete(@NotNull CommandSender sender, @NotNull Command command, @NotNull String alias, @NotNull String[] args) { - List argList = Arrays.asList(args); - if (argList.size() <= 1) { - List returnList = new ArrayList<>(subCommandMap.keySet()); - returnList.removeIf(str -> !str.startsWith(args[0]) || !sender.hasPermission(name + "." + str)); - return returnList; - } - AbstractSubCommand subCommand = subCommandMap.get(argList.get(0)); - if (subCommand != null) - return subCommand.onTabComplete(sender, argList.subList(1, argList.size())); - else - return Collections.singletonList(""); - } - - public void regSubCommand(AbstractSubCommand executor) { - subCommandMap.put(executor.getSubCommand(), executor); - } - - public Map getSubCommandMap() { - return subCommandMap; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/AbstractSubCommand.java b/src/main/java/net/momirealms/customnameplates/command/AbstractSubCommand.java deleted file mode 100644 index c9a3c03..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/AbstractSubCommand.java +++ /dev/null @@ -1,141 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.api.CustomNameplatesAPI; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.Bukkit; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.HumanEntity; -import org.bukkit.entity.Player; - -import java.util.ArrayList; -import java.util.Collections; -import java.util.List; -import java.util.Map; -import java.util.concurrent.ConcurrentHashMap; -import java.util.stream.Collectors; - -public abstract class AbstractSubCommand { - - private final String command; - private Map subCommandMap; - - public AbstractSubCommand(String command) { - this.command = command; - } - - public boolean onCommand(CommandSender sender, List args) { - if (subCommandMap == null || args.size() < 1) { - return true; - } - AbstractSubCommand subCommand = subCommandMap.get(args.get(0)); - if (subCommand == null) { - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.unavailableArgs); - } else { - subCommand.onCommand(sender, args.subList(1, args.size())); - } - return true; - } - - public List onTabComplete(CommandSender sender, List args) { - if (subCommandMap == null) - return Collections.singletonList(""); - if (args.size() <= 1) { - List returnList = new ArrayList<>(subCommandMap.keySet()); - returnList.removeIf(str -> !str.startsWith(args.get(0))); - return returnList; - } - AbstractSubCommand subCmd = subCommandMap.get(args.get(0)); - if (subCmd != null) - return subCommandMap.get(args.get(0)).onTabComplete(sender, args.subList(1, args.size())); - return Collections.singletonList(""); - } - - public String getSubCommand() { - return command; - } - - public Map getSubCommands() { - return Collections.unmodifiableMap(subCommandMap); - } - - public void regSubCommand(AbstractSubCommand command) { - if (subCommandMap == null) { - subCommandMap = new ConcurrentHashMap<>(); - } - subCommandMap.put(command.getSubCommand(), command); - } - - protected List online_players() { - return Bukkit.getOnlinePlayers().stream().map(HumanEntity::getName).collect(Collectors.toList()); - } - - protected List allNameplates() { - return new ArrayList<>(CustomNameplates.getInstance().getNameplateManager().getNameplateConfigMap().keySet()); - } - - protected List allBubbles() { - return new ArrayList<>(CustomNameplates.getInstance().getChatBubblesManager().getBubbleConfigMap().keySet()); - } - - protected boolean notExist(CommandSender commandSender, String type, String value) { - if (type.equals("nameplate")) { - if (!CustomNameplatesAPI.getInstance().doesNameplateExist(value)) { - AdventureUtils.sendMessage(commandSender, MessageManager.prefix + MessageManager.np_not_exist); - return true; - } - } else if (type.equals("bubble")) { - if (!CustomNameplatesAPI.getInstance().doesBubbleExist(value)) { - AdventureUtils.sendMessage(commandSender, MessageManager.prefix + MessageManager.bb_not_exist); - return true; - } - } - return false; - } - - protected boolean noConsoleExecute(CommandSender commandSender) { - if (!(commandSender instanceof Player)) { - AdventureUtils.consoleMessage(MessageManager.prefix + MessageManager.no_console); - return true; - } - return false; - } - - protected boolean lackArgs(CommandSender commandSender, int required, int current) { - if (required > current) { - AdventureUtils.sendMessage(commandSender, MessageManager.prefix + MessageManager.lackArgs); - return true; - } - return false; - } - - protected boolean playerNotOnline(CommandSender commandSender, String player) { - if (Bukkit.getPlayer(player) == null) { - AdventureUtils.sendMessage(commandSender, MessageManager.prefix + MessageManager.not_online.replace("{Player}", player)); - return true; - } - return false; - } - - protected List filterStartingWith(List list, String prefix) { - return list.stream().filter(s -> s.startsWith(prefix)).collect(Collectors.toList()); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/BubblesCommand.java b/src/main/java/net/momirealms/customnameplates/command/BubblesCommand.java deleted file mode 100644 index 09aab66..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/BubblesCommand.java +++ /dev/null @@ -1,36 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command; - -import net.momirealms.customnameplates.command.subcmd.*; - -public class BubblesCommand extends AbstractMainCommand { - - public BubblesCommand() { - super("bubbles"); - regDefaultSubCommands(); - } - - private void regDefaultSubCommands() { - regSubCommand(BubblesEquipCommand.INSTANCE); - regSubCommand(BubblesUnequipCommand.INSTANCE); - regSubCommand(BubblesForceEquipCommand.INSTANCE); - regSubCommand(BubblesForceUnequipCommand.INSTANCE); - regSubCommand(BubblesListCommand.INSTANCE); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/NameplateCommand.java b/src/main/java/net/momirealms/customnameplates/command/NameplateCommand.java deleted file mode 100644 index d441142..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/NameplateCommand.java +++ /dev/null @@ -1,42 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command; - -import net.momirealms.customnameplates.command.subcmd.*; - -public class NameplateCommand extends AbstractMainCommand { - - - public NameplateCommand() { - super("nameplates"); - regDefaultSubCommands(); - } - - private void regDefaultSubCommands() { - regSubCommand(ReloadCommand.INSTANCE); - regSubCommand(NameplatesEquipCommand.INSTANCE); - regSubCommand(NameplatesForceEquipCommand.INSTANCE); - regSubCommand(NameplatesUnequipCommand.INSTANCE); - regSubCommand(NameplatesForceUnequipCommand.INSTANCE); - regSubCommand(PreviewCommand.INSTANCE); - regSubCommand(ForcePreviewCommand.INSTANCE); - regSubCommand(NameplatesListCommand.INSTANCE); - regSubCommand(HelpCommand.INSTANCE); - regSubCommand(AboutCommand.INSTANCE); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/AboutCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/AboutCommand.java deleted file mode 100644 index ee8a3b5..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/AboutCommand.java +++ /dev/null @@ -1,44 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.command.CommandSender; - -import java.util.List; - -public class AboutCommand extends AbstractSubCommand { - - public static final AboutCommand INSTANCE = new AboutCommand(); - - public AboutCommand() { - super("about"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - AdventureUtils.sendMessage(sender, "<#3CB371>⚓ CustomNameplates - <#98FB98>" + CustomNameplates.getInstance().getVersionHelper().getPluginVersion()); - AdventureUtils.sendMessage(sender, "<#7FFFAA>A plugin that provides adjustable images for texts"); - AdventureUtils.sendMessage(sender, "<#DA70D6>\uD83E\uDDEA Author: <#FFC0CB>XiaoMoMi"); - AdventureUtils.sendMessage(sender, "<#FF7F50>\uD83D\uDD25 Contributors: <#FFA07A>TopOrigin"); - AdventureUtils.sendMessage(sender, "<#FFD700>⭐ Document <#A9A9A9>| <#FAFAD2>⛏ Github <#A9A9A9>| <#48D1CC>\uD83D\uDD14 Polymart"); - return true; - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesEquipCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesEquipCommand.java deleted file mode 100644 index 953240d..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesEquipCommand.java +++ /dev/null @@ -1,58 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.api.CustomNameplatesAPI; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; - -public class BubblesEquipCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new BubblesEquipCommand(); - - public BubblesEquipCommand() { - super("equip"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (noConsoleExecute(sender) || lackArgs(sender, 1, args.size()) || notExist(sender, "bubble", args.get(0))) - return true; - if (!sender.hasPermission("bubbles.equip." + args.get(0))) { - AdventureUtils.playerMessage((Player) sender, MessageManager.prefix + MessageManager.bb_notAvailable); - return true; - } - CustomNameplatesAPI.getInstance().equipBubble((Player) sender, args.get(0)); - AdventureUtils.playerMessage((Player) sender, MessageManager.prefix + MessageManager.bb_equip.replace("{Bubble}", CustomNameplates.getInstance().getChatBubblesManager().getBubbleConfig(args.get(0)).display_name())); - return true; - } - - @Override - public List onTabComplete(CommandSender sender, List args) { - if (args.size() == 1 && sender instanceof Player player) { - return filterStartingWith(CustomNameplates.getInstance().getChatBubblesManager().getAvailableBubbles(player), args.get(0)); - } - return null; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesForceEquipCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesForceEquipCommand.java deleted file mode 100644 index e023295..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesForceEquipCommand.java +++ /dev/null @@ -1,60 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.api.CustomNameplatesAPI; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.Bukkit; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; - -public class BubblesForceEquipCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new BubblesForceEquipCommand(); - - public BubblesForceEquipCommand() { - super("forceequip"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (lackArgs(sender, 2, args.size()) || playerNotOnline(sender, args.get(0)) || notExist(sender, "bubble", args.get(1))) - return true; - - Player player = Bukkit.getPlayer(args.get(0)); - CustomNameplatesAPI.getInstance().equipBubble(player, args.get(1)); - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.bb_force_equip.replace("{Bubble}", CustomNameplates.getInstance().getChatBubblesManager().getBubbleConfig(args.get(1)).display_name()).replace("{Player}", args.get(0))); - return true; - } - - @Override - public List onTabComplete(CommandSender sender, List args) { - if (args.size() == 1) { - return filterStartingWith(online_players(), args.get(0)); - } - if (args.size() == 2) { - return filterStartingWith(allBubbles(), args.get(1)); - } - return null; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesForceUnequipCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesForceUnequipCommand.java deleted file mode 100644 index 24de2b9..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesForceUnequipCommand.java +++ /dev/null @@ -1,55 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.api.CustomNameplatesAPI; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.Bukkit; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; - -public class BubblesForceUnequipCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new BubblesForceUnequipCommand(); - - public BubblesForceUnequipCommand() { - super("forceunequip"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (lackArgs(sender, 1, args.size()) || playerNotOnline(sender, args.get(0))) - return true; - Player player = Bukkit.getPlayer(args.get(0)); - CustomNameplatesAPI.getInstance().unEquipBubble(player); - AdventureUtils.sendMessage(sender,MessageManager.prefix + MessageManager.bb_force_unEquip.replace("{Player}", args.get(0))); - return true; - } - - @Override - public List onTabComplete(CommandSender sender, List args) { - if (args.size() == 1) { - return filterStartingWith(online_players(), args.get(0)); - } - return null; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesListCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesListCommand.java deleted file mode 100644 index 5d00018..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesListCommand.java +++ /dev/null @@ -1,55 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; -import java.util.StringJoiner; - -public class BubblesListCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new BubblesListCommand(); - - public BubblesListCommand() { - super("list"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (noConsoleExecute(sender)) return true; - Player player = (Player) sender; - List availableBubbles = CustomNameplates.getInstance().getChatBubblesManager().getAvailableBubbles(player); - if (availableBubbles.size() != 0) { - StringJoiner stringJoiner = new StringJoiner(", "); - for (String availableBubble : availableBubbles) { - stringJoiner.add(availableBubble); - } - AdventureUtils.playerMessage(player, MessageManager.prefix + MessageManager.bb_available.replace("{Bubbles}", stringJoiner.toString())); - } - else { - AdventureUtils.playerMessage(player, MessageManager.prefix + MessageManager.bb_haveNone); - } - return true; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesUnequipCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesUnequipCommand.java deleted file mode 100644 index d1d6e8c..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/BubblesUnequipCommand.java +++ /dev/null @@ -1,45 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.api.CustomNameplatesAPI; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; - -public class BubblesUnequipCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new BubblesUnequipCommand(); - - public BubblesUnequipCommand() { - super("unequip"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (noConsoleExecute(sender)) return true; - Player player = (Player) sender; - CustomNameplatesAPI.getInstance().unEquipBubble(player); - AdventureUtils.playerMessage(player, MessageManager.prefix + MessageManager.bb_unEquip); - return true; - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/ForcePreviewCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/ForcePreviewCommand.java deleted file mode 100644 index 2034be1..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/ForcePreviewCommand.java +++ /dev/null @@ -1,88 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.kyori.adventure.key.Key; -import net.kyori.adventure.text.Component; -import net.kyori.adventure.text.format.TextColor; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.manager.NameplateManager; -import net.momirealms.customnameplates.object.DisplayMode; -import net.momirealms.customnameplates.object.nameplate.NameplateConfig; -import net.momirealms.customnameplates.object.nameplate.NameplatesTeam; -import net.momirealms.customnameplates.utils.AdventureUtils; -import net.momirealms.customnameplates.utils.ArmorStandUtils; -import org.bukkit.Bukkit; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; - -public class ForcePreviewCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new ForcePreviewCommand(); - - public ForcePreviewCommand() { - super("forcepreview"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (lackArgs(sender, 1, args.size()) || playerNotOnline(sender, args.get(0))) return true; - Player player = Bukkit.getPlayer(args.get(0)); - assert player != null; - NameplateManager nameplateManager = CustomNameplates.getInstance().getNameplateManager(); - if (nameplateManager.isInCoolDown(player)) { - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.coolDown); - return true; - } - String nameplate = args.size() >= 2 ? args.get(1) : nameplateManager.getEquippedNameplate(player); - NameplateConfig nameplateConfig = nameplateManager.getNameplateConfig(nameplate); - if (nameplateConfig == null) { - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.np_not_exist); - return true; - } - if (nameplateManager.getMode() == DisplayMode.TEAM) { - NameplatesTeam team = CustomNameplates.getInstance().getTeamManager().getNameplateTeam(player.getUniqueId()); - if (team != null) { - Component full = team.getNameplatePrefixComponent() - .append(Component.text(player.getName()).color(TextColor.color(AdventureUtils.colorToDecimal(team.getColor()))).font(Key.key("minecraft:default")) - .append(team.getNameplateSuffixComponent())); - ArmorStandUtils.preview(full, player, (int) nameplateManager.getPreview_time()); - } - } else if (nameplateManager.getMode() == DisplayMode.ARMOR_STAND || nameplateManager.getMode() == DisplayMode.TEXT_DISPLAY) { - nameplateManager.showPlayerArmorStandTags(player, nameplate); - } else { - AdventureUtils.sendMessage(sender, MessageManager.prefix + "Nameplate is disabled."); - } - return true; - } - - @Override - public List onTabComplete(CommandSender sender, List args) { - if (args.size() == 1) { - return filterStartingWith(online_players(), args.get(0)); - } - if (args.size() == 2) { - return filterStartingWith(allNameplates(), args.get(1)); - } - return null; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/HelpCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/HelpCommand.java deleted file mode 100644 index 5dcd920..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/HelpCommand.java +++ /dev/null @@ -1,58 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.command.CommandSender; - -import java.util.List; - -public class HelpCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new HelpCommand(); - - public HelpCommand() { - super("help"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - AdventureUtils.sendMessage(sender, "<#3CB371>Command usage:"); - AdventureUtils.sendMessage(sender, " ├─<#FFFACD> "); - AdventureUtils.sendMessage(sender, " └─<#FFFACD><#E1FFFF>[Optional Augument]"); - AdventureUtils.sendMessage(sender, "<#3CB371>/nameplates"); - AdventureUtils.sendMessage(sender, " ├─help"); - AdventureUtils.sendMessage(sender, " ├─about"); - AdventureUtils.sendMessage(sender, " ├─reload <#98FB98>Reload the plugin"); - AdventureUtils.sendMessage(sender, " ├─list <#98FB98>Show a list of available nameplates"); - AdventureUtils.sendMessage(sender, " ├─equip <#FFFACD> <#98FB98>Equip a nameplate"); - AdventureUtils.sendMessage(sender, " ├─forceequip <#FFFACD> <#98FB98>Force a player to equip a nameplate"); - AdventureUtils.sendMessage(sender, " ├─unequip <#98FB98>Unequip the current nameplate"); - AdventureUtils.sendMessage(sender, " ├─forceunequip <#FFFACD> <#98FB98>Force a player to unequip his nameplate"); - AdventureUtils.sendMessage(sender, " ├─preview <#98FB98>Preview your current nameplate"); - AdventureUtils.sendMessage(sender, " └─forcepreview <#E1FFFF>[nameplate] <#98FB98>Force a player to preview the nameplate"); - AdventureUtils.sendMessage(sender, "<#3CB371>/bubbles"); - AdventureUtils.sendMessage(sender, " ├─list <#98FB98>Show a list of available bubbles"); - AdventureUtils.sendMessage(sender, " ├─equip <#FFFACD><#98FB98>Equip a bubble"); - AdventureUtils.sendMessage(sender, " ├─forceequip <#FFFACD> <#98FB98>Force a player to equip a bubble"); - AdventureUtils.sendMessage(sender, " ├─unequip <#98FB98>Unequip the current bubble"); - AdventureUtils.sendMessage(sender, " └─forceunequip <#FFFACD> <#98FB98>Force a player to unequip his bubble"); - return true; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesEquipCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesEquipCommand.java deleted file mode 100644 index ddcfc14..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesEquipCommand.java +++ /dev/null @@ -1,58 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.api.CustomNameplatesAPI; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; - -public class NameplatesEquipCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new NameplatesEquipCommand(); - - public NameplatesEquipCommand() { - super("equip"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (noConsoleExecute(sender) || lackArgs(sender, 1, args.size()) || notExist(sender, "nameplate", args.get(0))) return true; - if (!sender.hasPermission("nameplates.equip." + args.get(0))) { - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.np_notAvailable); - return true; - } - Player player = (Player) sender; - CustomNameplatesAPI.getInstance().equipNameplate(player, args.get(0)); - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.np_equip.replace("{Nameplate}", CustomNameplates.getInstance().getNameplateManager().getNameplateConfig(args.get(0)).display_name())); - return true; - } - - @Override - public List onTabComplete(CommandSender sender, List args) { - if (args.size() == 1 && sender instanceof Player player) { - return filterStartingWith(CustomNameplates.getInstance().getNameplateManager().getAvailableNameplates(player), args.get(0)); - } - return null; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesForceEquipCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesForceEquipCommand.java deleted file mode 100644 index 77bd195..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesForceEquipCommand.java +++ /dev/null @@ -1,58 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.api.CustomNameplatesAPI; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.Bukkit; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; - -public class NameplatesForceEquipCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new NameplatesForceEquipCommand(); - - public NameplatesForceEquipCommand() { - super("forceequip"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (lackArgs(sender, 2, args.size()) || playerNotOnline(sender, args.get(0)) || notExist(sender, "nameplate", args.get(1))) return true; - Player player = Bukkit.getPlayer(args.get(0)); - CustomNameplatesAPI.getInstance().equipNameplate(player, args.get(1)); - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.np_force_equip.replace("{Nameplate}", CustomNameplates.getInstance().getNameplateManager().getNameplateConfig(args.get(1)).display_name()).replace("{Player}", args.get(0))); - return true; - } - - @Override - public List onTabComplete(CommandSender sender, List args) { - if (args.size() == 1) { - return filterStartingWith(online_players(), args.get(0)); - } - if (args.size() == 2) { - return filterStartingWith(allNameplates(), args.get(1)); - } - return null; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesForceUnequipCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesForceUnequipCommand.java deleted file mode 100644 index c092cf4..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesForceUnequipCommand.java +++ /dev/null @@ -1,55 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.api.CustomNameplatesAPI; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.Bukkit; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; - -public class NameplatesForceUnequipCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new NameplatesForceUnequipCommand(); - - public NameplatesForceUnequipCommand() { - super("forceunequip"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (lackArgs(sender, 1, args.size()) || playerNotOnline(sender, args.get(0))) - return true; - Player player = Bukkit.getPlayer(args.get(0)); - CustomNameplatesAPI.getInstance().unEquipNameplate(player); - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.np_force_unEquip.replace("{Player}", args.get(0))); - return true; - } - - @Override - public List onTabComplete(CommandSender sender, List args) { - if (args.size() == 1) { - return filterStartingWith(online_players(), args.get(0)); - } - return null; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesListCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesListCommand.java deleted file mode 100644 index bce5484..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesListCommand.java +++ /dev/null @@ -1,54 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; -import java.util.StringJoiner; - -public class NameplatesListCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new NameplatesListCommand(); - - public NameplatesListCommand() { - super("list"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (noConsoleExecute(sender)) return true; - Player player = (Player) sender; - List availableNameplates = CustomNameplates.getInstance().getNameplateManager().getAvailableNameplates(player); - if (availableNameplates.size() != 0) { - StringJoiner stringJoiner = new StringJoiner(", "); - for (String availableNameplate : availableNameplates) { - stringJoiner.add(availableNameplate); - } - AdventureUtils.playerMessage(player, MessageManager.prefix + MessageManager.np_available.replace("{Nameplates}", stringJoiner.toString())); - } else { - AdventureUtils.playerMessage(player, MessageManager.prefix + MessageManager.np_haveNone); - } - return true; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesUnequipCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesUnequipCommand.java deleted file mode 100644 index 7a580fa..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/NameplatesUnequipCommand.java +++ /dev/null @@ -1,45 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.api.CustomNameplatesAPI; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; - -public class NameplatesUnequipCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new NameplatesUnequipCommand(); - - public NameplatesUnequipCommand() { - super("unequip"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (noConsoleExecute(sender)) return true; - Player player = (Player) sender; - CustomNameplatesAPI.getInstance().unEquipNameplate(player); - AdventureUtils.playerMessage(player, MessageManager.prefix + MessageManager.np_unEquip); - return true; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/PreviewCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/PreviewCommand.java deleted file mode 100644 index c977df7..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/PreviewCommand.java +++ /dev/null @@ -1,69 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.kyori.adventure.key.Key; -import net.kyori.adventure.text.Component; -import net.kyori.adventure.text.format.TextColor; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.manager.NameplateManager; -import net.momirealms.customnameplates.object.DisplayMode; -import net.momirealms.customnameplates.object.nameplate.NameplatesTeam; -import net.momirealms.customnameplates.utils.AdventureUtils; -import net.momirealms.customnameplates.utils.ArmorStandUtils; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -import java.util.List; - -public class PreviewCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new PreviewCommand(); - public PreviewCommand() { - super("preview"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - if (noConsoleExecute(sender)) return true; - Player player = (Player) sender; - NameplateManager nameplateManager = CustomNameplates.getInstance().getNameplateManager(); - if (nameplateManager.isInCoolDown(player)) { - AdventureUtils.playerMessage(player, MessageManager.prefix + MessageManager.coolDown); - return true; - } - if (nameplateManager.getMode() == DisplayMode.TEAM) { - NameplatesTeam team = CustomNameplates.getInstance().getTeamManager().getNameplateTeam(player.getUniqueId()); - if (team != null) { - Component full = team.getNameplatePrefixComponent() - .append(Component.text(player.getName()).color(TextColor.color(AdventureUtils.colorToDecimal(team.getColor()))).font(Key.key("minecraft:default")) - .append(team.getNameplateSuffixComponent())); - ArmorStandUtils.preview(full, player, (int) nameplateManager.getPreview_time()); - } - } else if (nameplateManager.getMode() == DisplayMode.ARMOR_STAND || nameplateManager.getMode() == DisplayMode.TEXT_DISPLAY) { - nameplateManager.showPlayerArmorStandTags(player); - } else { - AdventureUtils.playerMessage(player, MessageManager.prefix + "Nameplate is disabled."); - return true; - } - AdventureUtils.playerMessage(player, MessageManager.prefix + MessageManager.preview); - return true; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/command/subcmd/ReloadCommand.java b/src/main/java/net/momirealms/customnameplates/command/subcmd/ReloadCommand.java deleted file mode 100644 index 264d1ce..0000000 --- a/src/main/java/net/momirealms/customnameplates/command/subcmd/ReloadCommand.java +++ /dev/null @@ -1,43 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.command.subcmd; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.command.AbstractSubCommand; -import net.momirealms.customnameplates.manager.MessageManager; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.command.CommandSender; - -import java.util.List; - -public class ReloadCommand extends AbstractSubCommand { - - public static final AbstractSubCommand INSTANCE = new ReloadCommand(); - - private ReloadCommand() { - super("reload"); - } - - @Override - public boolean onCommand(CommandSender sender, List args) { - long time1 = System.currentTimeMillis(); - CustomNameplates.getInstance().reload(true); - AdventureUtils.sendMessage(sender, MessageManager.prefix + MessageManager.reload.replace("{time}", String.valueOf(System.currentTimeMillis() - time1))); - return true; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/data/FileStorageImpl.java b/src/main/java/net/momirealms/customnameplates/data/FileStorageImpl.java deleted file mode 100644 index 24f106b..0000000 --- a/src/main/java/net/momirealms/customnameplates/data/FileStorageImpl.java +++ /dev/null @@ -1,73 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.data; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.ConfigManager; -import net.momirealms.customnameplates.utils.ConfigUtils; -import org.bukkit.configuration.file.YamlConfiguration; - -import java.io.File; -import java.io.IOException; -import java.util.UUID; - -public class FileStorageImpl implements DataStorageInterface { - - private final CustomNameplates plugin; - - public FileStorageImpl(CustomNameplates plugin) { - this.plugin = plugin; - } - - @Override - public void initialize() { - //empty - } - - @Override - public void disable() { - //empty - } - - @Override - public PlayerData loadData(UUID uuid) { - YamlConfiguration data = ConfigUtils.readData(new File(plugin.getDataFolder(), "player_data" + File.separator + uuid + ".yml")); - String bubbles = data.getString("bubbles", ConfigManager.enableBubbles ? plugin.getChatBubblesManager().getDefaultBubble() : "none"); - String nameplate = data.getString("nameplate", ConfigManager.enableNameplates ? plugin.getNameplateManager().getDefault_nameplate() : "none"); - PlayerData playerData = new PlayerData(uuid, nameplate, bubbles); - saveData(playerData); - return playerData; - } - - @Override - public void saveData(PlayerData playerData) { - YamlConfiguration data = new YamlConfiguration(); - data.set("bubbles", playerData.getBubble()); - data.set("nameplate", playerData.getNameplate()); - try { - data.save(new File(plugin.getDataFolder(), "player_data" + File.separator + playerData.getUuid() + ".yml")); - } catch (IOException e) { - e.printStackTrace(); - } - } - - @Override - public StorageType getStorageType() { - return StorageType.YAML; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/data/MySQLStorageImpl.java b/src/main/java/net/momirealms/customnameplates/data/MySQLStorageImpl.java deleted file mode 100644 index b150c79..0000000 --- a/src/main/java/net/momirealms/customnameplates/data/MySQLStorageImpl.java +++ /dev/null @@ -1,113 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.data; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.ConfigManager; -import net.momirealms.customnameplates.utils.AdventureUtils; - -import java.sql.Connection; -import java.sql.PreparedStatement; -import java.sql.ResultSet; -import java.sql.SQLException; -import java.util.UUID; - -public class MySQLStorageImpl implements DataStorageInterface { - - private final SqlConnection sqlConnection; - private final CustomNameplates plugin; - - public MySQLStorageImpl(CustomNameplates plugin) { - this.plugin = plugin; - this.sqlConnection = new SqlConnection(this); - } - - @Override - public void initialize() { - sqlConnection.createNewHikariConfiguration(); - createTableIfNotExist(sqlConnection.getTable()); - } - - @Override - public void disable() { - sqlConnection.close(); - } - - @Override - public PlayerData loadData(UUID uuid) { - PlayerData playerData = null; - String sql = String.format(SqlConstants.SQL_SELECT_BY_UUID, sqlConnection.getTable()); - try (Connection connection = sqlConnection.getConnectionAndCheck(); PreparedStatement statement = connection.prepareStatement(sql)) { - statement.setString(1, uuid.toString()); - ResultSet rs = statement.executeQuery(); - if (rs.next()) { - String nameplate = rs.getString(2); - String bubbles = rs.getString(3); - playerData = new PlayerData(uuid, nameplate, bubbles); - } - else { - playerData = new PlayerData(uuid, - ConfigManager.enableNameplates ? plugin.getNameplateManager().getDefault_nameplate() : "none", - ConfigManager.enableBubbles ? plugin.getChatBubblesManager().getDefaultBubble() : "none"); - insertData(uuid, playerData.getNameplate(), playerData.getBubble()); - } - } catch (SQLException e) { - e.printStackTrace(); - } - return playerData; - } - - @Override - public void saveData(PlayerData playerData) { - String sql = String.format(SqlConstants.SQL_UPDATE_BY_UUID, sqlConnection.getTable()); - try (Connection connection = sqlConnection.getConnectionAndCheck(); PreparedStatement statement = connection.prepareStatement(sql)) { - statement.setString(1, playerData.getNameplate()); - statement.setString(2, playerData.getBubble()); - statement.setString(3, playerData.getUuid().toString()); - statement.executeUpdate(); - } catch (SQLException ex) { - AdventureUtils.consoleMessage("[CustomNameplates] Failed to update data for " + playerData.getUuid()); - } - } - - @Override - public StorageType getStorageType() { - return StorageType.SQL; - } - - private void createTableIfNotExist(String table) { - String sql = String.format(SqlConstants.SQL_CREATE_TABLE, table); - try (Connection connection = sqlConnection.getConnectionAndCheck(); PreparedStatement statement = connection.prepareStatement(sql)) { - statement.executeUpdate(); - } catch (SQLException ex) { - AdventureUtils.consoleMessage("[CustomNameplates] Failed to create table"); - } - } - - private void insertData(UUID uuid, String nameplate, String bubbles) { - String sql = String.format(SqlConstants.SQL_INSERT, sqlConnection.getTable()); - try (Connection connection = sqlConnection.getConnectionAndCheck(); PreparedStatement statement = connection.prepareStatement(sql)) { - statement.setString(1, uuid.toString()); - statement.setString(2, nameplate); - statement.setString(3, bubbles); - statement.executeUpdate(); - } catch (SQLException ex) { - AdventureUtils.consoleMessage("[CustomNameplates] Failed to insert data for " + uuid); - } - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/data/PlayerData.java b/src/main/java/net/momirealms/customnameplates/data/PlayerData.java deleted file mode 100644 index 5d1f156..0000000 --- a/src/main/java/net/momirealms/customnameplates/data/PlayerData.java +++ /dev/null @@ -1,54 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.data; - -import java.util.UUID; - -public class PlayerData { - - public static PlayerData EMPTY = new PlayerData(UUID.randomUUID(), "none", "none"); - private final UUID uuid; - private String nameplate; - private String bubble; - - public PlayerData(UUID uuid, String nameplate, String bubble) { - this.nameplate = nameplate; - this.bubble = bubble; - this.uuid = uuid; - } - - public String getNameplate() { - return this.nameplate; - } - - public void setNameplate(String nameplate) { - this.nameplate = nameplate; - } - - public String getBubble() { - return bubble; - } - - public void setBubble(String bubble) { - this.bubble = bubble; - } - - public UUID getUuid() { - return uuid; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/data/SqlConnection.java b/src/main/java/net/momirealms/customnameplates/data/SqlConnection.java deleted file mode 100644 index 11f1c30..0000000 --- a/src/main/java/net/momirealms/customnameplates/data/SqlConnection.java +++ /dev/null @@ -1,155 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.data; - -import com.zaxxer.hikari.HikariConfig; -import com.zaxxer.hikari.HikariDataSource; -import com.zaxxer.hikari.pool.HikariPool; -import net.momirealms.customnameplates.utils.AdventureUtils; -import net.momirealms.customnameplates.utils.ConfigUtils; -import org.bukkit.configuration.ConfigurationSection; -import org.bukkit.configuration.file.YamlConfiguration; - -import java.sql.Connection; -import java.sql.SQLException; - -public class SqlConnection { - - private boolean secondTry = false; - private boolean firstTry = false; - private HikariDataSource hikariDataSource; - private String table; - private final MySQLStorageImpl mySQLStorage; - - public SqlConnection(MySQLStorageImpl mySQLStorage) { - this.mySQLStorage = mySQLStorage; - } - - public void createNewHikariConfiguration() { - ConfigUtils.update("database.yml"); - YamlConfiguration config = ConfigUtils.getConfig("database.yml"); - String storageMode = config.getString("data-storage-method", "MySQL"); - - HikariConfig hikariConfig = new HikariConfig(); - String sql = "mysql"; - if (storageMode.equalsIgnoreCase("MariaDB")) { - try { - Class.forName("org.mariadb.jdbc.Driver"); - } catch (ClassNotFoundException e1) { - AdventureUtils.consoleMessage("[CustomNameplates] No sql driver is found."); - } - hikariConfig.setDriverClassName("org.mariadb.jdbc.Driver"); - sql = "mariadb"; - } else { - try { - Class.forName("com.mysql.jdbc.Driver"); - } catch (ClassNotFoundException e1) { - try { - Class.forName("com.mysql.cj.jdbc.Driver"); - } catch (ClassNotFoundException e2) { - AdventureUtils.consoleMessage("[CustomNameplates] No sql driver is found."); - } - } - } - table = config.getString(storageMode + ".table"); - hikariConfig.setPoolName("[CustomNameplates]"); - hikariConfig.setJdbcUrl(String.format("jdbc:%s://%s/%s", sql, config.getString(storageMode + ".host") + ":" + config.getString(storageMode + ".port"), config.getString(storageMode + ".database"))); - hikariConfig.setUsername(config.getString(storageMode + ".user")); - hikariConfig.setPassword(config.getString(storageMode + ".password")); - hikariConfig.setMaximumPoolSize(config.getInt(storageMode + ".Pool-Settings.maximum-pool-size")); - hikariConfig.setMinimumIdle(config.getInt(storageMode + ".Pool-Settings.minimum-idle")); - hikariConfig.setMaxLifetime(config.getInt(storageMode + ".Pool-Settings.maximum-lifetime")); - hikariConfig.setConnectionTimeout(3000); - hikariConfig.setIdleTimeout(hikariConfig.getMinimumIdle() < hikariConfig.getMaximumPoolSize() ? config.getInt(storageMode + ".Pool-Settings.idle-timeout") : 0); - ConfigurationSection section = config.getConfigurationSection(storageMode + ".properties"); - if (section != null) { - for (String property : section.getKeys(false)) { - hikariConfig.addDataSourceProperty(property, config.getString(storageMode + ".properties." + property)); - } - } - try { - hikariDataSource = new HikariDataSource(hikariConfig); - } catch (HikariPool.PoolInitializationException e) { - AdventureUtils.consoleMessage("[CustomNameplates] Failed to create sql connection"); - } - } - - public boolean setGlobalConnection() { - try { - createNewHikariConfiguration(); - Connection connection = getConnection(); - connection.close(); - if (secondTry) { - AdventureUtils.consoleMessage("[CustomNameplates] Successfully reconnect to SQL!"); - } else { - secondTry = true; - } - return true; - } catch (SQLException e) { - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to connect to SQL!"); - e.printStackTrace(); - close(); - return false; - } - } - - public Connection getConnection() throws SQLException { - return hikariDataSource.getConnection(); - } - - public boolean canConnect() { - if (hikariDataSource == null) { - return setGlobalConnection(); - } - if (hikariDataSource.isClosed()) { - return setGlobalConnection(); - } - return true; - } - - public void close() { - if (hikariDataSource != null && hikariDataSource.isRunning()) { - hikariDataSource.close(); - } - } - - public Connection getConnectionAndCheck() { - if (!canConnect()) { - return null; - } - try { - return getConnection(); - } catch (SQLException e) { - if (firstTry) { - firstTry = false; - close(); - return getConnectionAndCheck(); - } else { - firstTry = true; - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to connect to SQL!"); - close(); - e.printStackTrace(); - return null; - } - } - } - - public String getTable() { - return table; - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/data/SqlConstants.java b/src/main/java/net/momirealms/customnameplates/data/SqlConstants.java deleted file mode 100644 index e47301f..0000000 --- a/src/main/java/net/momirealms/customnameplates/data/SqlConstants.java +++ /dev/null @@ -1,26 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.data; - -public class SqlConstants { - - public static final String SQL_CREATE_TABLE = "CREATE TABLE IF NOT EXISTS `%s` ( `uuid` VARCHAR(36) NOT NULL , `equipped` VARCHAR(32) NOT NULL , `bubble` VARCHAR(32) NOT NULL , PRIMARY KEY (`uuid`) )"; - public static final String SQL_INSERT = "INSERT INTO `%s`(`uuid`, `equipped`, `bubble`) VALUES (?, ?, ?)"; - public static final String SQL_UPDATE_BY_UUID = "UPDATE `%s` SET `equipped` = ?, `bubble` = ? WHERE `uuid` = ?"; - public static final String SQL_SELECT_BY_UUID = "SELECT * FROM `%s` WHERE `uuid` = ?"; -} diff --git a/src/main/java/net/momirealms/customnameplates/helper/Log.java b/src/main/java/net/momirealms/customnameplates/helper/Log.java deleted file mode 100644 index 8210388..0000000 --- a/src/main/java/net/momirealms/customnameplates/helper/Log.java +++ /dev/null @@ -1,63 +0,0 @@ -/* - * This file is part of helper, licensed under the MIT License. - * - * Copyright (c) lucko (Luck) - * Copyright (c) contributors - * - * 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 net.momirealms.customnameplates.helper; - -import net.momirealms.customnameplates.CustomNameplates; -import org.bukkit.Bukkit; - -import javax.annotation.Nonnull; -import java.util.logging.Level; - -/** - * Utility for quickly accessing a logger instance without using {@link Bukkit#getLogger()} - */ -public final class Log { - - public static void info(@Nonnull String s) { - CustomNameplates.getInstance().getLogger().info(s); - } - - public static void warn(@Nonnull String s) { - CustomNameplates.getInstance().getLogger().warning(s); - } - - public static void severe(@Nonnull String s) { - CustomNameplates.getInstance().getLogger().severe(s); - } - - public static void warn(@Nonnull String s, Throwable t) { - CustomNameplates.getInstance().getLogger().log(Level.WARNING, s, t); - } - - public static void severe(@Nonnull String s, Throwable t) { - CustomNameplates.getInstance().getLogger().log(Level.SEVERE, s, t); - } - - private Log() { - throw new UnsupportedOperationException("This class cannot be instantiated"); - } - -} diff --git a/src/main/java/net/momirealms/customnameplates/helper/NonnullByDefault.java b/src/main/java/net/momirealms/customnameplates/helper/NonnullByDefault.java deleted file mode 100644 index a39e959..0000000 --- a/src/main/java/net/momirealms/customnameplates/helper/NonnullByDefault.java +++ /dev/null @@ -1,45 +0,0 @@ -/* - * This file is part of helper, licensed under the MIT License. - * - * Copyright (c) lucko (Luck) - * Copyright (c) contributors - * - * 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 net.momirealms.customnameplates.helper; - -import javax.annotation.Nonnull; -import javax.annotation.meta.TypeQualifierDefault; -import java.lang.annotation.Documented; -import java.lang.annotation.ElementType; -import java.lang.annotation.Retention; -import java.lang.annotation.RetentionPolicy; - -@Nonnull -@Documented -@TypeQualifierDefault({ - ElementType.FIELD, - ElementType.METHOD, - ElementType.PARAMETER -}) -@Retention(RetentionPolicy.RUNTIME) -public @interface NonnullByDefault { - -} diff --git a/src/main/java/net/momirealms/customnameplates/listener/AbstractChatListener.java b/src/main/java/net/momirealms/customnameplates/listener/AbstractChatListener.java deleted file mode 100644 index 2d16a01..0000000 --- a/src/main/java/net/momirealms/customnameplates/listener/AbstractChatListener.java +++ /dev/null @@ -1,30 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.listener; - -import net.momirealms.customnameplates.manager.ChatBubblesManager; -import org.bukkit.event.Listener; - -public abstract class AbstractChatListener implements Listener { - - protected ChatBubblesManager chatBubblesManager; - - public AbstractChatListener(ChatBubblesManager chatBubblesManager) { - this.chatBubblesManager = chatBubblesManager; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/listener/AsyncChatListener.java b/src/main/java/net/momirealms/customnameplates/listener/AsyncChatListener.java deleted file mode 100644 index 03905de..0000000 --- a/src/main/java/net/momirealms/customnameplates/listener/AsyncChatListener.java +++ /dev/null @@ -1,36 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.listener; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.ChatBubblesManager; -import org.bukkit.event.EventHandler; -import org.bukkit.event.player.AsyncPlayerChatEvent; - -public class AsyncChatListener extends AbstractChatListener { - - public AsyncChatListener(ChatBubblesManager chatBubblesManager) { - super(chatBubblesManager); - } - - @EventHandler - public void onChat(AsyncPlayerChatEvent event) { - if (event.isCancelled()) return; - CustomNameplates.getInstance().getScheduler().runTask(() -> chatBubblesManager.onChat(event.getPlayer(), event.getMessage()), event.getPlayer().getLocation()); - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/listener/EntityTagListener.java b/src/main/java/net/momirealms/customnameplates/listener/EntityTagListener.java deleted file mode 100644 index 25bc9d1..0000000 --- a/src/main/java/net/momirealms/customnameplates/listener/EntityTagListener.java +++ /dev/null @@ -1,54 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.listener; - -import net.momirealms.customnameplates.object.carrier.NamedEntityCarrier; -import org.bukkit.entity.Player; -import org.bukkit.event.EventHandler; -import org.bukkit.event.Listener; -import org.bukkit.event.entity.PlayerDeathEvent; -import org.bukkit.event.player.PlayerRespawnEvent; -import org.bukkit.event.player.PlayerToggleSneakEvent; -import org.bukkit.event.vehicle.VehicleExitEvent; - -public record EntityTagListener(NamedEntityCarrier namedEntityCarrier) implements Listener { - - @EventHandler - public void onSneak(PlayerToggleSneakEvent event) { - if (event.isCancelled()) return; - namedEntityCarrier.onSneak(event.getPlayer(), event.isSneaking()); - } - - @EventHandler - public void onRespawn(PlayerRespawnEvent event) { - namedEntityCarrier.onRespawn(event.getPlayer()); - } - -// @EventHandler -// public void onDeath(PlayerDeathEvent event) { -// namedEntityCarrier.onDeath(event.getPlayer()); -// } - - @EventHandler - public void onExitVehicle(VehicleExitEvent event) { - if (event.isCancelled()) return; - if (event.getExited() instanceof Player player) { - namedEntityCarrier.onSneak(player, false); - } - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/listener/JoinQuitListener.java b/src/main/java/net/momirealms/customnameplates/listener/JoinQuitListener.java deleted file mode 100644 index c24ebd9..0000000 --- a/src/main/java/net/momirealms/customnameplates/listener/JoinQuitListener.java +++ /dev/null @@ -1,37 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.listener; - -import net.momirealms.customnameplates.object.Function; -import org.bukkit.event.EventHandler; -import org.bukkit.event.Listener; -import org.bukkit.event.player.PlayerJoinEvent; -import org.bukkit.event.player.PlayerQuitEvent; - -public record JoinQuitListener(Function function) implements Listener { - - @EventHandler - public void onJoin(PlayerJoinEvent event) { - function.onJoin(event.getPlayer()); - } - - @EventHandler - public void onQuit(PlayerQuitEvent event) { - function.onQuit(event.getPlayer()); - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/listener/TrChatListener.java b/src/main/java/net/momirealms/customnameplates/listener/TrChatListener.java deleted file mode 100644 index f028c9f..0000000 --- a/src/main/java/net/momirealms/customnameplates/listener/TrChatListener.java +++ /dev/null @@ -1,40 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.listener; - -import me.arasple.mc.trchat.api.event.TrChatEvent; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.ChatBubblesManager; -import org.bukkit.event.EventHandler; - -public class TrChatListener extends AbstractChatListener { - - public TrChatListener(ChatBubblesManager chatBubblesManager) { - super(chatBubblesManager); - } - - @EventHandler - public void onTrChat(TrChatEvent event) { - if (event.isCancelled() || !event.getForward()) return; - String channelName = event.getChannel().getId(); - for (String channel : chatBubblesManager.getChannels()) { - if (channelName.equals(channel)) return; - } - CustomNameplates.getInstance().getScheduler().runTask(() -> chatBubblesManager.onChat(event.getSession().getPlayer(), event.getMessage()), event.getSession().getPlayer().getLocation()); - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/listener/VentureChatListener.java b/src/main/java/net/momirealms/customnameplates/listener/VentureChatListener.java deleted file mode 100644 index 5b50fbb..0000000 --- a/src/main/java/net/momirealms/customnameplates/listener/VentureChatListener.java +++ /dev/null @@ -1,45 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.listener; - -import mineverse.Aust1n46.chat.api.MineverseChatPlayer; -import mineverse.Aust1n46.chat.api.events.VentureChatEvent; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.ChatBubblesManager; -import org.bukkit.event.EventHandler; - -public class VentureChatListener extends AbstractChatListener { - - public VentureChatListener(ChatBubblesManager chatBubblesManager) { - super(chatBubblesManager); - } - - @EventHandler - public void onVentureChat(VentureChatEvent event) { - String channelName = event.getChannel().getName(); - for (String channel : chatBubblesManager.getChannels()) { - if (channelName.equals(channel)) return; - } - final MineverseChatPlayer chatPlayer = event.getMineverseChatPlayer(); - if (chatPlayer == null) { - return; - } - - CustomNameplates.getInstance().getScheduler().runTask(() -> chatBubblesManager.onChat(chatPlayer.getPlayer(), event.getChat().substring(1)), chatPlayer.getPlayer().getLocation()); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/ActionBarManager.java b/src/main/java/net/momirealms/customnameplates/manager/ActionBarManager.java deleted file mode 100644 index 59d8edc..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/ActionBarManager.java +++ /dev/null @@ -1,219 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import com.comphenix.protocol.events.PacketContainer; -import com.comphenix.protocol.events.PacketEvent; -import com.comphenix.protocol.wrappers.EnumWrappers; -import com.comphenix.protocol.wrappers.WrappedChatComponent; -import net.kyori.adventure.text.Component; -import net.kyori.adventure.text.ScoreComponent; -import net.kyori.adventure.text.TranslatableComponent; -import net.kyori.adventure.text.serializer.gson.GsonComponentSerializer; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.listener.JoinQuitListener; -import net.momirealms.customnameplates.listener.packet.ActionBarListener; -import net.momirealms.customnameplates.listener.packet.ChatMessageListener; -import net.momirealms.customnameplates.listener.packet.SystemChatListener; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.object.actionbar.ActionBarConfig; -import net.momirealms.customnameplates.object.actionbar.ActionBarTask; -import net.momirealms.customnameplates.object.bossbar.BossBarConfig; -import net.momirealms.customnameplates.object.bossbar.BossBarTask; -import net.momirealms.customnameplates.utils.AdventureUtils; -import net.momirealms.customnameplates.utils.ConfigUtils; -import net.momirealms.customnameplates.utils.GeyserUtils; -import org.bukkit.Bukkit; -import org.bukkit.configuration.ConfigurationSection; -import org.bukkit.configuration.file.YamlConfiguration; -import org.bukkit.entity.Player; -import org.bukkit.event.HandlerList; -import org.jetbrains.annotations.Nullable; - -import java.io.File; -import java.util.LinkedHashMap; -import java.util.UUID; -import java.util.concurrent.ConcurrentHashMap; - -public class ActionBarManager extends Function { - - private final LinkedHashMap actionBarConfigMap; - private final ConcurrentHashMap actionBarTaskMap; - private final ActionBarListener actionBarListener; - private SystemChatListener systemChatListener; - private ChatMessageListener chatMessageListener; - private final JoinQuitListener joinQuitListener; - private final CustomNameplates plugin; - - public ActionBarManager(CustomNameplates plugin) { - this.plugin = plugin; - this.actionBarConfigMap = new LinkedHashMap<>(); - this.actionBarTaskMap = new ConcurrentHashMap<>(); - this.joinQuitListener = new JoinQuitListener(this); - this.actionBarListener = new ActionBarListener(this); - if (plugin.getVersionHelper().isVersionNewerThan1_19()) { - this.systemChatListener = new SystemChatListener(this); - } else { - this.chatMessageListener = new ChatMessageListener(this); - } - } - - @Override - public void load() { - if (!ConfigManager.enableActionBar) return; - this.loadConfig(); - Bukkit.getPluginManager().registerEvents(joinQuitListener, plugin); - if (actionBarListener != null) CustomNameplates.getProtocolManager().addPacketListener(actionBarListener); - if (systemChatListener != null) CustomNameplates.getProtocolManager().addPacketListener(systemChatListener); - if (chatMessageListener != null) CustomNameplates.getProtocolManager().addPacketListener(chatMessageListener); - for (Player player : Bukkit.getOnlinePlayers()) { - onJoin(player); - } - } - - @Override - public void unload() { - for (ActionBarTask actionBarTask : actionBarTaskMap.values()) { - actionBarTask.stop(); - } - actionBarConfigMap.clear(); - if (actionBarListener != null) CustomNameplates.getProtocolManager().removePacketListener(actionBarListener); - if (systemChatListener != null) CustomNameplates.getProtocolManager().removePacketListener(systemChatListener); - if (chatMessageListener != null) CustomNameplates.getProtocolManager().removePacketListener(chatMessageListener); - HandlerList.unregisterAll(joinQuitListener); - } - - @Override - public void onJoin(Player player) { - if (ConfigManager.disableForBedrock && plugin.getVersionHelper().isGeyser() && GeyserUtils.isBedrockPlayer(player.getUniqueId())) - return; - var uuid = player.getUniqueId(); - if (ConfigManager.sendDelay <= 0) { - ActionBarTask actionBarTask = new ActionBarTask(player, actionBarConfigMap.values().toArray(new ActionBarConfig[0])); - actionBarTaskMap.put(uuid, actionBarTask); - actionBarTask.start(); - } else { - plugin.getScheduler().runTaskLater(() -> { - var onlinePlayer = Bukkit.getPlayer(uuid); - if (onlinePlayer != null) { - ActionBarTask actionBarTask = new ActionBarTask(onlinePlayer, actionBarConfigMap.values().toArray(new ActionBarConfig[0])); - actionBarTaskMap.put(uuid, actionBarTask); - actionBarTask.start(); - } - }, player.getLocation(), ConfigManager.sendDelay); - } - } - - @Override - public void onQuit(Player player) { - ActionBarTask actionBarTask = actionBarTaskMap.remove(player.getUniqueId()); - if (actionBarTask != null) actionBarTask.stop(); - } - - @Nullable - public ActionBarTask getActionBarTask(UUID uuid) { - return actionBarTaskMap.get(uuid); - } - - private void loadConfig() { - YamlConfiguration config = ConfigUtils.getConfig("configs" + File.separator + "actionbar.yml"); - for (String key : config.getKeys(false)) { - ConfigurationSection actionBarSection = config.getConfigurationSection(key); - if (actionBarSection == null) continue; - actionBarConfigMap.put(key, new ActionBarConfig( - actionBarSection.getInt("switch-interval", 15) * 20, - actionBarSection.getString("text") == null ? actionBarSection.getStringList("dynamic-text").toArray(new String[0]) : new String[]{actionBarSection.getString("text")}, - ConfigUtils.getRequirements(actionBarSection.getConfigurationSection("conditions")) - )); - } - AdventureUtils.consoleMessage("[CustomNameplates] Loaded " + actionBarConfigMap.size() + " actionbars"); - } - - public void onReceiveActionBarPacket(PacketEvent event) { - PacketContainer packet = event.getPacket(); - WrappedChatComponent wrappedChatComponent = packet.getChatComponents().read(0); - if (wrappedChatComponent != null) { - ActionBarTask actionBarTask = getActionBarTask(event.getPlayer().getUniqueId()); - if (actionBarTask != null) { - String strJson = wrappedChatComponent.getJson(); - // for better performance - if (strJson.endsWith("\"objective\":\"actionbar\"}}")) { - return; - } - - Component component = GsonComponentSerializer.gson().deserialize(strJson); - if (component instanceof ScoreComponent scoreComponent) { - if (scoreComponent.name().equals("nameplates") && scoreComponent.objective().equals("actionbar")) { - return; - } - } - event.setCancelled(true); - actionBarTask.setOtherText(AdventureUtils.getMiniMessageFormat(component), System.currentTimeMillis()); - } - } - } - - // 1.19+ - public void onReceiveSystemChatPacket(PacketEvent event) { - PacketContainer packet = event.getPacket(); - Boolean overlay = packet.getBooleans().readSafely(0); - if (overlay != null && overlay) { - ActionBarTask actionBarTask = getActionBarTask(event.getPlayer().getUniqueId()); - if (actionBarTask != null) { - event.setCancelled(true); - String json = packet.getStrings().readSafely(0); - if (json != null && !json.equals("")) { - Component component = GsonComponentSerializer.gson().deserialize(json); - if (component instanceof TranslatableComponent) { - return; - } - actionBarTask.setOtherText(AdventureUtils.getMiniMessageFormat(component), System.currentTimeMillis()); - } - } - } - } - - // lower version - public void onReceiveChatMessagePacket(PacketEvent event) { - PacketContainer packet = event.getPacket(); - EnumWrappers.ChatType type = packet.getChatTypes().readSafely(0); - if (type == EnumWrappers.ChatType.GAME_INFO) { - ActionBarTask actionBarTask = getActionBarTask(event.getPlayer().getUniqueId()); - if (actionBarTask != null) { - event.setCancelled(true); - WrappedChatComponent wrappedChatComponent = packet.getChatComponents().read(0); - if (wrappedChatComponent != null) { - String json = wrappedChatComponent.getJson(); - Component component = GsonComponentSerializer.gson().deserialize(json); - if (component instanceof TranslatableComponent) { - return; - } - actionBarTask.setOtherText(AdventureUtils.getMiniMessageFormat(component), System.currentTimeMillis()); - } - } - } - } - - public String getOtherPluginActionBarText(Player player) { - ActionBarTask actionBarTask = getActionBarTask(player.getUniqueId()); - if (actionBarTask != null) { - return actionBarTask.getOtherText(); - } - return ""; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/BackgroundManager.java b/src/main/java/net/momirealms/customnameplates/manager/BackgroundManager.java deleted file mode 100644 index e4eb247..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/BackgroundManager.java +++ /dev/null @@ -1,174 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.object.SimpleChar; -import net.momirealms.customnameplates.object.background.BackGroundConfig; -import net.momirealms.customnameplates.object.font.OffsetFont; -import net.momirealms.customnameplates.object.placeholders.BackGroundText; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.configuration.file.YamlConfiguration; - -import java.io.File; -import java.util.Arrays; -import java.util.Comparator; -import java.util.HashMap; - -public class BackgroundManager extends Function { - - private final CustomNameplates plugin; - private final HashMap backGroundConfigMap; - - public BackgroundManager(CustomNameplates plugin) { - this.plugin = plugin; - this.backGroundConfigMap = new HashMap<>(); - } - - @Override - public void load() { - if (!ConfigManager.enableBackground) return; - loadConfig(); - } - - @Override - public void unload() { - this.backGroundConfigMap.clear(); - } - - private void loadConfig() { - File bg_file = new File(plugin.getDataFolder(),"contents" + File.separator + "backgrounds"); - if (!bg_file.exists() && bg_file.mkdirs()) { - saveDefaultBackgrounds(); - } - File[] bg_config_files = bg_file.listFiles(file -> file.getName().endsWith(".yml")); - if (bg_config_files == null) return; - Arrays.sort(bg_config_files, Comparator.comparing(File::getName)); - for (File bg_config_file : bg_config_files) { - String key = bg_config_file.getName().substring(0, bg_config_file.getName().length() - 4); - YamlConfiguration config = YamlConfiguration.loadConfiguration(bg_config_file); - char oStart = ConfigManager.start_char; - char o1 = (char) (oStart + '\u0001'); char o2 = (char) (o1 + '\u0001'); - char o4 = (char) (o2 + '\u0001'); char o8 = (char) (o4 + '\u0001'); - char o16 = (char) (o8 + '\u0001'); char o32 = (char) (o16 + '\u0001'); - char o64 = (char) (o32 + '\u0001'); char o128 = (char) (o64 + '\u0001'); - char oEnd = (char) (o128 + '\u0001'); ConfigManager.start_char = (char) (oEnd + '\u0001'); - int height = config.getInt("middle.height", 14); - int ascent = config.getInt("middle.ascent", 8); - SimpleChar startChar = new SimpleChar(config.getInt("left.height"), config.getInt("left.ascent"),1 ,oStart, config.getString("left.image") + ".png"); - SimpleChar offset_1 = new SimpleChar(height, ascent,1, o1, config.getString("middle.1") + ".png"); - SimpleChar offset_2 = new SimpleChar(height, ascent,1, o2, config.getString("middle.2") + ".png"); - SimpleChar offset_4 = new SimpleChar(height, ascent,1, o4, config.getString("middle.4") + ".png"); - SimpleChar offset_8 = new SimpleChar(height, ascent,1, o8, config.getString("middle.8") + ".png"); - SimpleChar offset_16 = new SimpleChar(height, ascent,1, o16, config.getString("middle.16") + ".png"); - SimpleChar offset_32 = new SimpleChar(height, ascent,1, o32, config.getString("middle.32") + ".png"); - SimpleChar offset_64 = new SimpleChar(height, ascent,1, o64, config.getString("middle.64") + ".png"); - SimpleChar offset_128 = new SimpleChar(height, ascent,1, o128, config.getString("middle.128") + ".png"); - SimpleChar endChar = new SimpleChar(config.getInt("right.height"),config.getInt("right.ascent"),1, oEnd, config.getString("right.image") + ".png"); - backGroundConfigMap.put(key, new BackGroundConfig(startChar, offset_1, - offset_2, offset_4, - offset_8, offset_16, - offset_32, offset_64, - offset_128, endChar, - config.getInt("left-margin", 1), config.getInt("right-margin", 1))); - plugin.getFontManager().loadCustomWidth(o1, 1); - plugin.getFontManager().loadCustomWidth(o2, 2); - plugin.getFontManager().loadCustomWidth(o4, 4); - plugin.getFontManager().loadCustomWidth(o8, 8); - plugin.getFontManager().loadCustomWidth(o16, 16); - plugin.getFontManager().loadCustomWidth(o32, 32); - plugin.getFontManager().loadCustomWidth(o64, 64); - plugin.getFontManager().loadCustomWidth(o128, 128); - plugin.getFontManager().loadCustomWidth(oStart, config.getInt("left.width", 1)); - plugin.getFontManager().loadCustomWidth(oEnd, config.getInt("right.width", 1)); - } - AdventureUtils.consoleMessage("[CustomNameplates] Loaded " + backGroundConfigMap.size() + " backgrounds"); - } - - public HashMap getBackGroundConfigMap() { - return backGroundConfigMap; - } - - public String getBackGroundImage(BackGroundText backGroundText, String text) { - BackGroundConfig backGroundConfig = backGroundConfigMap.get(backGroundText.background()); - if (backGroundConfig == null) return backGroundText.background() + " NOT FOUND"; - return getBackGroundImage(backGroundConfig, text); - } - - public String getBackGroundImage(BackGroundConfig backGroundConfig, String text) { - int n = plugin.getFontManager().getTotalWidth(text); - String offset = plugin.getFontManager().getShortestNegChars(n + backGroundConfig.right_margin() + 2); - n = n + backGroundConfig.left_margin() + backGroundConfig.right_margin() + 2; - StringBuilder stringBuilder = new StringBuilder(); - stringBuilder.append(backGroundConfig.left().getChars()); - while (n >= 128) { - stringBuilder.append(OffsetFont.NEG_1.getCharacter()); - stringBuilder.append(backGroundConfig.offset_128().getChars()); - n -= 128; - } - if (n - 64 >= 0) { - stringBuilder.append(OffsetFont.NEG_1.getCharacter()); - stringBuilder.append(backGroundConfig.offset_64().getChars()); - n -= 64; - } - if (n - 32 >= 0) { - stringBuilder.append(OffsetFont.NEG_1.getCharacter()); - stringBuilder.append(backGroundConfig.offset_32().getChars()); - n -= 32; - } - if (n - 16 >= 0) { - stringBuilder.append(OffsetFont.NEG_1.getCharacter()); - stringBuilder.append(backGroundConfig.offset_16().getChars()); - n -= 16; - } - if (n - 8 >= 0) { - stringBuilder.append(OffsetFont.NEG_1.getCharacter()); - stringBuilder.append(backGroundConfig.offset_8().getChars()); - n -= 8; - } - if (n - 4 >= 0) { - stringBuilder.append(OffsetFont.NEG_1.getCharacter()); - stringBuilder.append(backGroundConfig.offset_4().getChars()); - n -= 4; - } - if (n - 2 >= 0) { - stringBuilder.append(OffsetFont.NEG_1.getCharacter()); - stringBuilder.append(backGroundConfig.offset_2().getChars()); - n -= 2; - } - if (n - 1 >= 0) { - stringBuilder.append(OffsetFont.NEG_1.getCharacter()); - stringBuilder.append(backGroundConfig.offset_1().getChars()); - } - stringBuilder.append(OffsetFont.NEG_1.getCharacter()); - stringBuilder.append(backGroundConfig.right().getChars()).append(offset); - return stringBuilder.toString(); - } - - private void saveDefaultBackgrounds() { - String[] bg_list = new String[]{"b0", "b1", "b2", "b4", "b8", "b16","b32","b64","b128"}; - for (String bg : bg_list) { - plugin.saveResource("contents" + File.separator + "backgrounds" + File.separator + bg + ".png", false); - } - String[] config_list = new String[]{"bedrock_1", "bedrock_2", "bedrock_3"}; - for (String config : config_list) { - plugin.saveResource("contents" + File.separator + "backgrounds" + File.separator + config + ".yml", false); - } - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/BossBarManager.java b/src/main/java/net/momirealms/customnameplates/manager/BossBarManager.java deleted file mode 100644 index 5774288..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/BossBarManager.java +++ /dev/null @@ -1,121 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.listener.JoinQuitListener; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.object.bossbar.BossBarConfig; -import net.momirealms.customnameplates.object.bossbar.BossBarTask; -import net.momirealms.customnameplates.object.bossbar.Overlay; -import net.momirealms.customnameplates.utils.AdventureUtils; -import net.momirealms.customnameplates.utils.ConfigUtils; -import net.momirealms.customnameplates.utils.GeyserUtils; -import org.bukkit.Bukkit; -import org.bukkit.boss.BarColor; -import org.bukkit.configuration.ConfigurationSection; -import org.bukkit.configuration.file.YamlConfiguration; -import org.bukkit.entity.Player; -import org.bukkit.event.HandlerList; - -import java.io.File; -import java.util.LinkedHashMap; -import java.util.Locale; -import java.util.UUID; -import java.util.concurrent.ConcurrentHashMap; - -public class BossBarManager extends Function { - - private final LinkedHashMap bossBars; - private final ConcurrentHashMap bossBarTaskMap; - private final JoinQuitListener joinQuitListener; - private final CustomNameplates plugin; - - public BossBarManager(CustomNameplates plugin) { - this.plugin = plugin; - this.joinQuitListener = new JoinQuitListener(this); - this.bossBarTaskMap = new ConcurrentHashMap<>(); - this.bossBars = new LinkedHashMap<>(); - } - - @Override - public void load() { - if (!ConfigManager.enableBossBar) return; - this.loadConfig(); - Bukkit.getPluginManager().registerEvents(joinQuitListener, plugin); - for (Player player : Bukkit.getOnlinePlayers()) { - onJoin(player); - } - } - - @Override - public void unload() { - for (BossBarTask bossBarTask : bossBarTaskMap.values()) { - bossBarTask.stop(); - } - bossBarTaskMap.clear(); - HandlerList.unregisterAll(joinQuitListener); - } - - public void onJoin(Player player) { - if (ConfigManager.disableForBedrock && plugin.getVersionHelper().isGeyser() && GeyserUtils.isBedrockPlayer(player.getUniqueId())) - return; - var uuid = player.getUniqueId(); - if (ConfigManager.sendDelay <= 0) { - BossBarTask bossBarTask = new BossBarTask(player, bossBars.values().toArray(new BossBarConfig[0])); - bossBarTaskMap.put(uuid, bossBarTask); - bossBarTask.start(); - } else { - plugin.getScheduler().runTaskLater(() -> { - var onlinePlayer = Bukkit.getPlayer(uuid); - if (onlinePlayer != null) { - BossBarTask bossBarTask = new BossBarTask(onlinePlayer, bossBars.values().toArray(new BossBarConfig[0])); - bossBarTaskMap.put(uuid, bossBarTask); - bossBarTask.start(); - } - }, player.getLocation(), ConfigManager.sendDelay); - } - } - - public void onQuit(Player player) { - BossBarTask bossBarTask = bossBarTaskMap.remove(player.getUniqueId()); - if (bossBarTask != null) bossBarTask.stop(); - } - - private void loadConfig() { - bossBars.clear(); - YamlConfiguration config = ConfigUtils.getConfig("configs" + File.separator + "bossbar.yml"); - for (String key : config.getKeys(false)) { - ConfigurationSection bossBarSection = config.getConfigurationSection(key); - if (bossBarSection == null) continue; - bossBars.put(key, new BossBarConfig( - bossBarSection.getString("text") == null ? bossBarSection.getStringList("dynamic-text").toArray(new String[0]) : new String[]{bossBarSection.getString("text")}, - Overlay.valueOf(bossBarSection.getString("overlay","progress").toUpperCase(Locale.ENGLISH)), - BarColor.valueOf(bossBarSection.getString("color","white").toUpperCase(Locale.ENGLISH)), - bossBarSection.getInt("switch-interval", 5) * 20, - Math.max(1, bossBarSection.getInt("refresh-rate", 1)), - ConfigUtils.getRequirements(bossBarSection.getConfigurationSection("conditions")) - )); - } - AdventureUtils.consoleMessage("[CustomNameplates] Loaded " + bossBars.size() + " bossbars"); - } - - public LinkedHashMap getBossBars() { - return bossBars; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/ChatBubblesManager.java b/src/main/java/net/momirealms/customnameplates/manager/ChatBubblesManager.java deleted file mode 100644 index 959c6c3..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/ChatBubblesManager.java +++ /dev/null @@ -1,377 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import me.clip.placeholderapi.PlaceholderAPI; -import net.kyori.adventure.text.minimessage.MiniMessage; -import net.kyori.adventure.text.serializer.gson.GsonComponentSerializer; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.api.events.BubblesEvent; -import net.momirealms.customnameplates.listener.*; -import net.momirealms.customnameplates.object.DisplayMode; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.object.SimpleChar; -import net.momirealms.customnameplates.object.bubble.BubbleConfig; -import net.momirealms.customnameplates.object.carrier.NamedEntityCarrier; -import net.momirealms.customnameplates.object.carrier.NamedEntityImpl; -import net.momirealms.customnameplates.object.carrier.NamedEntityManager; -import net.momirealms.customnameplates.object.carrier.TextDisplayMeta; -import net.momirealms.customnameplates.object.font.OffsetFont; -import net.momirealms.customnameplates.object.img.ImageParser; -import net.momirealms.customnameplates.object.img.ItemsAdderImageImpl; -import net.momirealms.customnameplates.object.img.OraxenImageImpl; -import net.momirealms.customnameplates.utils.AdventureUtils; -import net.momirealms.customnameplates.utils.ConfigUtils; -import org.bukkit.Bukkit; -import org.bukkit.GameMode; -import org.bukkit.configuration.file.YamlConfiguration; -import org.bukkit.entity.Player; -import org.bukkit.event.HandlerList; -import org.bukkit.permissions.PermissionAttachmentInfo; -import org.bukkit.plugin.PluginManager; -import org.bukkit.potion.PotionEffectType; -import org.jetbrains.annotations.Nullable; - -import java.io.File; -import java.io.IOException; -import java.util.*; -import java.util.concurrent.ConcurrentHashMap; - -public class ChatBubblesManager extends Function { - - private AbstractChatListener chatListener; - private ImageParser imageParser; - private final ConcurrentHashMap coolDownMap; - private final HashMap bubbleConfigMap; - private final CustomNameplates plugin; - private NamedEntityCarrier namedEntityCarrier; - private final JoinQuitListener joinQuitListener; - private String defaultBubble; - private String defaultStartFormat; - private String defaultEndFormat; - private String prefix; - private String suffix; - private double lineSpace; - private double yOffset; - private int stayTime; - private int coolDown; - private int maxCharLength; - private int lengthPerLine; - private String[] channels; - private TextDisplayMeta textDisplayMeta; - - public ChatBubblesManager(CustomNameplates plugin) { - this.plugin = plugin; - this.bubbleConfigMap = new HashMap<>(); - this.coolDownMap = new ConcurrentHashMap<>(); - this.joinQuitListener = new JoinQuitListener(this); - } - - @Override - public void load() { - if (!ConfigManager.enableBubbles) return; - super.load(); - this.loadConfig(); - this.loadBubbles(); - this.registerListener(); - this.registerImageParser(); - this.namedEntityCarrier.load(); - } - - @Override - public void unload() { - super.unload(); - this.imageParser = null; - this.bubbleConfigMap.clear(); - this.coolDownMap.clear(); - if (chatListener != null) HandlerList.unregisterAll(chatListener); - if (joinQuitListener != null) HandlerList.unregisterAll(joinQuitListener); - if (namedEntityCarrier != null) namedEntityCarrier.unload(); - } - - private void loadConfig() { - YamlConfiguration config = ConfigUtils.getConfig("configs" + File.separator + "bubble.yml"); - DisplayMode displayMode = DisplayMode.valueOf(config.getString("mode", "ARMOR_STAND").toUpperCase(Locale.ENGLISH)); - defaultBubble = config.getString("default-bubbles", "chat"); - prefix = config.getString("text-prefix", ""); - suffix = config.getString("text-suffix", ""); - lineSpace = config.getDouble("line-spacing"); - defaultStartFormat = config.getString("default-format.start", ""); - defaultEndFormat = config.getString("default-format.end", ""); - yOffset = config.getDouble("bottom-line-Y-offset") + (displayMode == DisplayMode.TEXT_DISPLAY ? 1.2 : 0); - stayTime = config.getInt("stay-time", 5); - coolDown = (int) (config.getDouble("cool-down", 1) * 1000); - maxCharLength = config.getInt("max-character-length", 100); - channels = config.getStringList("blacklist-channels").toArray(new String[0]); - lengthPerLine = config.getInt("characters-per-line", 30); - textDisplayMeta = ConfigUtils.getTextDisplayMeta(config.getConfigurationSection("text-display-options")); - namedEntityCarrier = new NamedEntityCarrier(plugin, displayMode, new HashMap<>()); - } - - private void registerImageParser() { - PluginManager pluginManager = Bukkit.getPluginManager(); - if (pluginManager.isPluginEnabled("Oraxen")) { - this.imageParser = new OraxenImageImpl(); - } else if (pluginManager.isPluginEnabled("ItemsAdder")) { - this.imageParser = new ItemsAdderImageImpl(); - } - } - - private void registerListener() { - PluginManager pluginManager = Bukkit.getPluginManager(); - if (ConfigManager.trChat_Hook) { - this.chatListener = new TrChatListener(this); - } else if (ConfigManager.ventureChat_Hook) { - this.chatListener = new VentureChatListener(this); - } else { - this.chatListener = new AsyncChatListener(this); - } - pluginManager.registerEvents(chatListener, plugin); - pluginManager.registerEvents(joinQuitListener, plugin); - } - - private void loadBubbles() { - File bb_file = new File(plugin.getDataFolder(), "contents" + File.separator + "bubbles"); - if (!bb_file.exists() && bb_file.mkdirs()) { - saveDefaultBubbles(); - } - File[] bb_config_files = bb_file.listFiles(file -> file.getName().endsWith(".yml")); - if (bb_config_files == null) return; - Arrays.sort(bb_config_files, Comparator.comparing(File::getName)); - for (File bb_config_file : bb_config_files) { - char left = ConfigManager.start_char; - char middle; - char right; - char tail; - ConfigManager.start_char = (char) ((tail = (char)((right = (char)((middle = (char)(ConfigManager.start_char + '\u0001')) + '\u0001')) + '\u0001')) + '\u0001'); - String key = bb_config_file.getName().substring(0, bb_config_file.getName().length() - 4); - YamlConfiguration config = YamlConfiguration.loadConfiguration(bb_config_file); - if (!config.contains("display-name")) config.set("display-name", key); - if (!config.contains("text-format.start")) config.set("text-format.start", ""); - if (!config.contains("text-format.end")) config.set("text-format.end", ""); - if (!config.contains("left.image")) config.set("left.image", key + "_left"); - if (!config.contains("left.height")) config.set("left.height", 16); - if (!config.contains("left.width")) config.set("left.width", 16); - if (!config.contains("left.ascent")) config.set("left.ascent", 12); - if (!config.contains("middle.image")) config.set("middle.image", key + "_middle"); - if (!config.contains("middle.height")) config.set("middle.height", 16); - if (!config.contains("middle.width")) config.set("middle.width", 16); - if (!config.contains("middle.ascent")) config.set("middle.ascent", 12); - if (!config.contains("right.image")) config.set("right.image", key + "_right"); - if (!config.contains("right.height")) config.set("right.height", 16); - if (!config.contains("right.width")) config.set("right.width", 16); - if (!config.contains("right.ascent")) config.set("right.ascent", 12); - if (!config.contains("tail.image")) config.set("tail.image", key + "_tail"); - if (!config.contains("tail.height")) config.set("tail.height", 16); - if (!config.contains("tail.width")) config.set("tail.width", 16); - if (!config.contains("tail.ascent")) config.set("tail.ascent", 12); - try { - config.save(bb_config_file); - } catch (IOException ignored) { - } - SimpleChar leftChar = new SimpleChar(config.getInt("left.height"), config.getInt("left.ascent"), config.getInt("left.width"), left, config.getString("left.image") + ".png"); - SimpleChar middleChar = new SimpleChar(config.getInt("middle.height"), config.getInt("middle.ascent"), config.getInt("middle.width"), middle, config.getString("middle.image") + ".png"); - SimpleChar rightChar = new SimpleChar(config.getInt("right.height"), config.getInt("right.ascent"), config.getInt("right.width"), right, config.getString("right.image") + ".png"); - SimpleChar tailChar = new SimpleChar(config.getInt("tail.height"), config.getInt("tail.ascent"), config.getInt("tail.width"), tail, config.getString("tail.image") + ".png"); - bubbleConfigMap.put(key, - new BubbleConfig( - config.getString("text-format.start"), config.getString("text-format.end"), - config.getString("display-name"), - leftChar, middleChar, - rightChar, tailChar - ) - ); - } - } - - @Override - public void onQuit(Player player) { - coolDownMap.remove(player); - } - - private boolean isCoolDown(Player player, int lines) { - long time = System.currentTimeMillis(); - if (time - (coolDownMap.getOrDefault(player, time - (long) coolDown * lines)) < (long) coolDown * lines) return true; - coolDownMap.put(player, time); - return false; - } - - public String getBubblePrefix(String text, BubbleConfig bubble) { - int totalWidth = plugin.getFontManager().getTotalWidth(text); - StringBuilder stringBuilder = new StringBuilder(); - stringBuilder.append(plugin.getFontManager().getShortestNegChars(totalWidth % 2 == 0 ? totalWidth + bubble.left().getWidth() : totalWidth + bubble.left().getWidth() + 1)) - .append(bubble.left().getChars()).append(OffsetFont.NEG_1.getCharacter()); - int mid_amount; - if (totalWidth - 1 <= bubble.tail().getWidth() ) { - mid_amount = -1; - } else { - mid_amount = (totalWidth - 1 - bubble.tail().getWidth()) / (bubble.middle().getWidth()); - } - if (mid_amount == -1) { - stringBuilder.append(bubble.tail().getChars()).append(OffsetFont.NEG_1.getCharacter()); - } else if (mid_amount == 0) { - stringBuilder.append(bubble.tail().getChars()).append(OffsetFont.NEG_1.getCharacter()); - stringBuilder.append( - plugin.getFontManager().getShortestNegChars( - bubble.middle().getWidth() - (totalWidth - 1 - bubble.tail().getWidth()) % bubble.middle().getWidth() - ) - ); - stringBuilder.append(bubble.middle().getChars()).append(OffsetFont.NEG_1.getCharacter()); - } else { - stringBuilder.append(bubble.middle().getChars()).append(OffsetFont.NEG_1.getCharacter()); - for (int i = 0; i < mid_amount; i++) { - if (i == mid_amount / 2) stringBuilder.append(bubble.tail().getChars()).append(OffsetFont.NEG_1.getCharacter()); - else stringBuilder.append(bubble.middle().getChars()).append(OffsetFont.NEG_1.getCharacter()); - } - stringBuilder.append( - plugin.getFontManager().getShortestNegChars( - bubble.middle().getWidth() - (totalWidth - 1 - bubble.tail().getWidth()) % bubble.middle().getWidth() - ) - ); - stringBuilder.append(bubble.middle().getChars()).append(OffsetFont.NEG_1.getCharacter()); - } - stringBuilder.append(bubble.right().getChars()); - stringBuilder.append(plugin.getFontManager().getShortestNegChars(totalWidth + bubble.right().getWidth())); - return stringBuilder.toString(); - } - - public void onChat(Player player, String text) { - if (player.getGameMode() == GameMode.SPECTATOR - || player.hasPotionEffect(PotionEffectType.INVISIBILITY) - || !player.hasPermission("bubbles.use") - ) return; - - String bubble = plugin.getDataManager().getEquippedBubble(player); - BubblesEvent bubblesEvent = new BubblesEvent(player, bubble, text); - Bukkit.getPluginManager().callEvent(bubblesEvent); - if (bubblesEvent.isCancelled()) { - return; - } - - bubble = bubblesEvent.getBubble(); - BubbleConfig bubbleConfig = getBubble(bubble); - String miniText = AdventureUtils.replaceLegacy(bubblesEvent.getText()); - String stripped_text = MiniMessage.miniMessage().stripTags(miniText); - if (stripped_text.length() > maxCharLength) return; - String[] split = splitString(stripped_text, lengthPerLine); - if (isCoolDown(player, split.length)) return; - for (int i = 0; i < split.length; i++) { - int finalI = i; - String finalBubble = bubble; - plugin.getScheduler().runTaskAsyncLater(() -> sendBubble(player, split[finalI], bubbleConfig, finalBubble), (long) i * (coolDown / 50) + 1); - } - } - - private void sendBubble(Player player, String text, BubbleConfig bubbleConfig, String key) { - String json; - if (bubbleConfig == null || key.equals("none")) { - text = defaultStartFormat + PlaceholderAPI.setPlaceholders(player, prefix) + text + PlaceholderAPI.setPlaceholders(player, suffix) + defaultEndFormat; - json = GsonComponentSerializer.gson().serialize(MiniMessage.miniMessage().deserialize(imageParser != null ? imageParser.parse(player, text) : text)); - } else { - String parsedPrefix = PlaceholderAPI.setPlaceholders(player, prefix); - String parsedSuffix = PlaceholderAPI.setPlaceholders(player, suffix); - String strippedPrefix = AdventureUtils.stripAllTags(parsedPrefix); - String strippedSuffix = AdventureUtils.stripAllTags(parsedSuffix); - String all = strippedPrefix + text + strippedSuffix; - String bubbleImage = getBubblePrefix(all, bubbleConfig); - String suffixImage = plugin.getFontManager().getSuffixStringWithFont(all); - String finalStr = ConfigManager.surroundWithFont(bubbleImage) + bubbleConfig.startFormat() + parsedPrefix + text + parsedSuffix + bubbleConfig.endFormat() + ConfigManager.surroundWithFont(suffixImage); - json = GsonComponentSerializer.gson().serialize(MiniMessage.miniMessage().deserialize(finalStr)); - } - - NamedEntityManager nem = namedEntityCarrier.getNamedEntityManager(player); - if (nem != null) { - double offset = yOffset; - DisplayMode nameplateMode = plugin.getNameplateManager().getMode(); - if (nameplateMode == DisplayMode.ARMOR_STAND || nameplateMode == DisplayMode.TEXT_DISPLAY) { - NamedEntityCarrier carrier = (NamedEntityCarrier) plugin.getNameplateManager().getTextCarrier(); - NamedEntityManager nameplatesNem = carrier.getNamedEntityManager(player); - if (nameplatesNem != null) { - offset += nameplatesNem.getHighestTextHeight(); - } - } - UUID uuid = UUID.randomUUID(); - nem.ascent(lineSpace); - nem.addNamedEntity(uuid, new NamedEntityImpl(nem, player, json, offset, textDisplayMeta)); - plugin.getScheduler().runTaskAsyncLater(() -> nem.removeArmorStand(uuid), stayTime * 20L); - } - } - - private String[] splitString(String str, int len) { - int size = (int) Math.ceil((double) str.length() / (double) len); - String[] result = new String[size]; - int index = 0; - for (int i = 0; i < str.length(); i += len) { - if (i + len > str.length()) { - result[index++] = str.substring(i); - } else { - result[index++] = str.substring(i, i + len); - } - } - return result; - } - - @Nullable - public BubbleConfig getBubble(String bubble) { - return bubbleConfigMap.get(bubble); - } - - public ArrayList getAvailableBubbles(Player player) { - ArrayList availableBubbles = new ArrayList<>(); - for (PermissionAttachmentInfo info : player.getEffectivePermissions()) { - String permission = info.getPermission().toLowerCase(); - if (permission.startsWith("bubbles.equip.")) { - permission = permission.substring(14); - if (bubbleConfigMap.get(permission) != null) { - availableBubbles.add(permission); - } - } - } - return availableBubbles; - } - - private void saveDefaultBubbles() { - String[] png_list = new String[]{"chat"}; - String[] part_list = new String[]{"_left.png", "_middle.png", "_right.png", "_tail.png", ".yml"}; - for (String name : png_list) { - for (String part : part_list) { - plugin.saveResource("contents" + File.separator + "bubbles" + File.separatorChar + name + part, false); - } - } - } - - public String getDefaultBubble() { - return defaultBubble; - } - - public String[] getChannels() { - return channels; - } - - public HashMap getBubbleConfigMap() { - return bubbleConfigMap; - } - - public boolean existBubble(String bubble) { - return bubbleConfigMap.containsKey(bubble); - } - - public BubbleConfig getBubbleConfig(String bubble) { - return bubbleConfigMap.get(bubble); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/ConfigManager.java b/src/main/java/net/momirealms/customnameplates/manager/ConfigManager.java deleted file mode 100644 index d689122..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/ConfigManager.java +++ /dev/null @@ -1,144 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.utils.ConfigUtils; -import org.bukkit.Bukkit; -import org.bukkit.configuration.ConfigurationSection; -import org.bukkit.configuration.file.YamlConfiguration; - -import java.io.File; - -public class ConfigManager extends Function { - - public static String namespace; - public static String font; - public static String lang; - public static String nameplates_folder_path; - public static String backgrounds_folder_path; - public static String space_split_folder_path; - public static String bubbles_folder_path; - public static String images_folder_path; - public static boolean itemsAdderHook; - public static boolean tab_hook; - public static boolean tab_BC_hook; - public static boolean velocitab_hook; - public static boolean oraxenHook; - public static boolean trChat_Hook; - public static boolean ventureChat_Hook; - public static boolean extractShader; - public static boolean extractLegacyBars; - public static boolean extractBars; - public static char start_char; - public static boolean checkUpdate; - public static boolean enableBStats; - public static boolean enableNameplates; - public static boolean enableBossBar; - public static boolean enableActionBar; - public static boolean enableBubbles; - public static boolean enableBackground; - public static boolean enableImages; - public static int default_width; - public static boolean enable1_20_Unicode; - public static boolean hideScoreboardNumber; - public static boolean iaShaderSupport; - public static boolean generatePackOnStart; - public static boolean disableForBedrock; - public static int sendDelay; - - @Override - public void load(){ - ConfigUtils.update("config.yml"); - YamlConfiguration config = ConfigUtils.getConfig("config.yml"); - lang = config.getString("lang"); - enableBStats = config.getBoolean("config.metrics", true); - checkUpdate = config.getBoolean("update-checker", true); - loadModules(config); - loadIntegrations(config); - loadResourcePack(config); - loadOtherSettings(config); - } - - private void loadIntegrations(ConfigurationSection config) { - ConfigurationSection section = config.getConfigurationSection("integrations"); - if (section != null) { - tab_hook = section.getBoolean("TAB",false) && Bukkit.getPluginManager().isPluginEnabled("TAB"); - tab_BC_hook = section.getBoolean("TAB-BC",false); - velocitab_hook = section.getBoolean("Velocitab",false); - trChat_Hook = section.getBoolean("TrChat",false) && Bukkit.getPluginManager().isPluginEnabled("TrChat"); - itemsAdderHook = section.getBoolean("ItemsAdder",false) && Bukkit.getPluginManager().isPluginEnabled("ItemsAdder"); - oraxenHook = section.getBoolean("Oraxen",false) && Bukkit.getPluginManager().isPluginEnabled("Oraxen"); - ventureChat_Hook = section.getBoolean("VentureChat",false) && Bukkit.getPluginManager().isPluginEnabled("VentureChat"); - } - } - - private void loadResourcePack(ConfigurationSection config) { - ConfigurationSection section = config.getConfigurationSection("resource-pack"); - if (section != null) { - generatePackOnStart = !section.getBoolean("disable-generation-on-start", false); - namespace = section.getString("namespace", "nameplates"); - font = section.getString("font", "default"); - start_char = section.getString("config.left-char", "뀁").charAt(0); - nameplates_folder_path = section.getString("image-path.nameplates","font\\nameplates\\"); - bubbles_folder_path = section.getString("image-path.bubbles","font\\bubbles\\"); - backgrounds_folder_path = section.getString("image-path.backgrounds","font\\backgrounds\\"); - space_split_folder_path = section.getString("image-path.space-split","font\\base\\"); - images_folder_path = section.getString("image-path.images","font\\images\\"); - extractShader = section.getBoolean("extract-shader",true); - extractLegacyBars = section.getBoolean("extract-bar-image-legacy",true); - extractBars = section.getBoolean("extract-bar-image",true); - enable1_20_Unicode = section.getBoolean("support-1_20-unicodes",false); - hideScoreboardNumber = section.getBoolean("hide-scoreboard-number",true); - iaShaderSupport = section.getBoolean("ItemsAdder-text-effect-support",true); - } - } - - private void loadModules(ConfigurationSection config) { - ConfigurationSection section = config.getConfigurationSection("modules"); - if (section != null) { - enableNameplates = section.getBoolean("nameplates"); - enableBossBar = section.getBoolean("bossbars"); - enableActionBar = section.getBoolean("actionbars"); - enableBubbles = section.getBoolean("bubbles"); - enableBackground = section.getBoolean("backgrounds"); - enableImages = section.getBoolean("images"); - } - } - - private void loadOtherSettings(ConfigurationSection config) { - ConfigurationSection section = config.getConfigurationSection("other-settings"); - if (section != null) { - default_width = section.getInt("default-character-width", 8); - disableForBedrock = section.getBoolean("disable-for-bedrock-players", false); - sendDelay = section.getInt("send-delay", 0); - } - } - - public static String getMiniMessageFontTag() { - return ""; - } - - public static String getFontTagCloser() { - return ""; - } - - public static String surroundWithFont(String text) { - return getMiniMessageFontTag() + text + getFontTagCloser(); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/DataManager.java b/src/main/java/net/momirealms/customnameplates/manager/DataManager.java deleted file mode 100644 index 2abde4d..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/DataManager.java +++ /dev/null @@ -1,167 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.data.*; -import net.momirealms.customnameplates.listener.JoinQuitListener; -import net.momirealms.customnameplates.object.DisplayMode; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.utils.ConfigUtils; -import org.bukkit.Bukkit; -import org.bukkit.configuration.file.YamlConfiguration; -import org.bukkit.entity.Player; -import org.bukkit.event.HandlerList; - -import java.util.Optional; -import java.util.UUID; -import java.util.concurrent.ConcurrentHashMap; - -public class DataManager extends Function { - - private final CustomNameplates plugin; - private DataStorageInterface dataStorageInterface; - private final JoinQuitListener joinQuitListener; - private StorageType storageType; - private final ConcurrentHashMap playerDataMap; - private final ConcurrentHashMap triedTimes; - - public DataManager(CustomNameplates plugin) { - this.plugin = plugin; - this.playerDataMap = new ConcurrentHashMap<>(); - this.joinQuitListener = new JoinQuitListener(this); - this.triedTimes = new ConcurrentHashMap<>(); - } - - @Override - public void load() { - if (loadStorageMode()) this.dataStorageInterface.initialize(); - Bukkit.getPluginManager().registerEvents(joinQuitListener, plugin); - } - - @Override - public void unload() { - YamlConfiguration config = ConfigUtils.getConfig("database.yml"); - StorageType st = config.getString("data-storage-method","YAML").equalsIgnoreCase("YAML") ? StorageType.YAML : StorageType.SQL; - if (this.dataStorageInterface != null && dataStorageInterface.getStorageType() != st) this.dataStorageInterface.disable(); - HandlerList.unregisterAll(joinQuitListener); - } - - @Override - public void disable() { - if (this.dataStorageInterface != null) { - this.dataStorageInterface.disable(); - } - } - - @Override - public void onJoin(Player player) { - plugin.getScheduler().runTaskAsync(() -> readData(player.getUniqueId())); - } - - public void readData(UUID uuid) { - Player player = Bukkit.getPlayer(uuid); - if (player == null || !player.isOnline() || !checkTriedTimes(uuid)) return; - PlayerData playerData = this.dataStorageInterface.loadData(uuid); - if (playerData == null) { - plugin.getScheduler().runTaskAsyncLater(() -> readData(uuid), 20); - } else { - playerDataMap.put(uuid, playerData); - if (!ConfigManager.enableNameplates || plugin.getNameplateManager().getMode() == DisplayMode.DISABLE) return; - plugin.getTeamManager().getTeamNameInterface().onJoin(player); - plugin.getTeamManager().createTeam(uuid); - } - } - - @Override - public void onQuit(Player player) { - PlayerData playerData = playerDataMap.remove(player.getUniqueId()); - if (playerData != null) { - plugin.getScheduler().runTaskAsync(() -> this.dataStorageInterface.saveData(playerData)); - } - plugin.getTeamManager().onQuit(player); - triedTimes.remove(player.getUniqueId()); - } - - public void saveData(Player player) { - PlayerData playerData = playerDataMap.get(player.getUniqueId()); - if (playerData != null) { - plugin.getScheduler().runTaskAsync(() -> this.dataStorageInterface.saveData(playerData)); - } - } - - public DataStorageInterface getDataStorageInterface() { - return dataStorageInterface; - } - - private boolean loadStorageMode() { - YamlConfiguration config = ConfigUtils.getConfig("database.yml"); - if (config.getString("data-storage-method","YAML").equalsIgnoreCase("YAML")) { - if (storageType != StorageType.YAML) { - this.dataStorageInterface = new FileStorageImpl(plugin); - this.storageType = StorageType.YAML; - return true; - } - } else { - if (storageType != StorageType.SQL) { - this.dataStorageInterface = new MySQLStorageImpl(plugin); - this.storageType = StorageType.SQL; - return true; - } - } - return false; - } - - public String getEquippedNameplate(Player player) { - return Optional.ofNullable(playerDataMap.get(player.getUniqueId())).orElse(PlayerData.EMPTY).getNameplate(); - } - - public String getEquippedBubble(Player player) { - return Optional.ofNullable(playerDataMap.get(player.getUniqueId())).orElse(PlayerData.EMPTY).getBubble(); - } - - public void equipNameplate(Player player, String nameplate) { - PlayerData playerData = playerDataMap.get(player.getUniqueId()); - if (playerData != null) { - playerData.setNameplate(nameplate); - } - } - - public void equipBubble(Player player, String bubble) { - PlayerData playerData = playerDataMap.get(player.getUniqueId()); - if (playerData != null) { - playerData.setBubble(bubble); - } - } - - protected boolean checkTriedTimes(UUID uuid) { - Integer previous = triedTimes.get(uuid); - if (previous == null) { - triedTimes.put(uuid, 1); - return true; - } - else if (previous > 2) { - triedTimes.remove(uuid); - return false; - } - else { - triedTimes.put(uuid, previous + 1); - return true; - } - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/FontManager.java b/src/main/java/net/momirealms/customnameplates/manager/FontManager.java deleted file mode 100644 index ee01923..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/FontManager.java +++ /dev/null @@ -1,343 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import me.clip.placeholderapi.PlaceholderAPI; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.helper.Log; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.object.SimpleChar; -import net.momirealms.customnameplates.object.font.ASCIIWidth; -import net.momirealms.customnameplates.object.font.OffsetFont; -import net.momirealms.customnameplates.utils.AdventureUtils; -import net.momirealms.customnameplates.utils.ConfigUtils; -import org.apache.commons.lang3.StringEscapeUtils; -import org.bukkit.configuration.file.YamlConfiguration; -import org.jetbrains.annotations.NotNull; -import org.jetbrains.annotations.Nullable; - -import javax.imageio.ImageIO; -import java.awt.image.BufferedImage; -import java.io.*; -import java.net.URL; -import java.net.URLConnection; -import java.util.HashMap; -import java.util.Objects; - -public class FontManager extends Function { - - private final CustomNameplates plugin; - private final HashMap customImageWidth; - private final HashMap asciiWidth; - - public FontManager(CustomNameplates plugin) { - this.plugin = plugin; - this.customImageWidth = new HashMap<>(1024); - this.asciiWidth = new HashMap<>(); - } - - @Override - public void load() { - this.saveFonts(); - this.loadASCIIWidth(); - this.loadUnicodesWidth(); - this.loadCustomWidth(); - } - - @Override - public void unload() { - customImageWidth.clear(); - asciiWidth.clear(); - } - - private void loadASCIIWidth() { - for (int i = 0; i < ASCIIWidth.values().length; i++) - asciiWidth.put(ASCIIWidth.values()[i].getCharacter(), ASCIIWidth.values()[i].getWidth()); - } - - public void loadCustomWidth(char c, int width) { - customImageWidth.put(c, width); - } - - private void loadCustomWidth() { - for (int i = 0; i < ASCIIWidth.values().length; i++) - customImageWidth.put(ASCIIWidth.values()[i].getCharacter(), ASCIIWidth.values()[i].getWidth()); - for (SimpleChar simpleChar : plugin.getImageManager().getCharacterMap().values()) - customImageWidth.put(simpleChar.getChars(), simpleChar.getWidth()); - for (OffsetFont offsetFont : OffsetFont.values()) - customImageWidth.put(offsetFont.getCharacter(), offsetFont.getSpace() - 1); - YamlConfiguration config = ConfigUtils.getConfig("configs" + File.separator + "image-width.yml"); - for (String image : config.getKeys(false)) { - String character = AdventureUtils.stripAllTags(PlaceholderAPI.setPlaceholders(null, image)); - if (character.length() == 1) customImageWidth.put(character.charAt(0), config.getInt(image, 8)); - } - } - - private void saveFonts() { - File font_file = new File(plugin.getDataFolder(), "unicodes"); - if (!font_file.exists()) { - for (int i = 0; i < 256; i++) { - saveResource("unicodes" + File.separator + "unicode_page_" + String.format("%02x", i) + ".png"); - } - } - plugin.saveResource("templates" + File.separator + "default.json", true); - plugin.saveResource("templates" + File.separator + "unicode.json", true); - } - - public String getSuffixStringWithFont(String text) { - return ConfigManager.surroundWithFont(getSuffixString(text)); - } - - public String getSuffixString(String text) { - int totalWidth = plugin.getFontManager().getTotalWidth(text); - return plugin.getFontManager().getShortestNegChars(totalWidth + totalWidth % 2 + 1); - } - - // Player name - public int getNameCharWidth(char c) { - Integer width = asciiWidth.get(c); - if (width != null) return width; - return Objects.requireNonNullElse(customImageWidth.get(c), ConfigManager.default_width); - } - - public int getTotalWidth(String text) { - int length = text.length(); - int n = 0; - for (int i = 0; i < length; i++) { - char current = text.charAt(i); - if (current != '\\' || i == length - 1 || text.charAt(i + 1) != '<') { - n += Objects.requireNonNullElse(customImageWidth.get(current), ConfigManager.default_width); - } - else { - n -= 1; - } - } - return n + length; - } - - public int getTotalPlayerNameWidth(String text) { - int length = text.length(); - int n = 0; - for (int i = 0; i < length; i++) { - n += getNameCharWidth(text.charAt(i)); - } - return n + length; - } - - public String getOffset(int offset) { - if (offset >= 0) { - return getShortestPosChars(offset); - } - else { - return getShortestNegChars(-offset); - } - } - - public String getShortestNegChars(int n) { - StringBuilder stringBuilder = new StringBuilder(); - while (n >= 128) { - stringBuilder.append(OffsetFont.NEG_128.getCharacter()); - n -= 128; - } - if (n - 64 >= 0) { - stringBuilder.append(OffsetFont.NEG_64.getCharacter()); - n -= 64; - } - if (n - 32 >= 0) { - stringBuilder.append(OffsetFont.NEG_32.getCharacter()); - n -= 32; - } - if (n - 16 >= 0) { - stringBuilder.append(OffsetFont.NEG_16.getCharacter()); - n -= 16; - } - if (n - 8 >= 0) { - stringBuilder.append(OffsetFont.NEG_8.getCharacter()); - n -= 8; - } - if (n - 7 >= 0) { - stringBuilder.append(OffsetFont.NEG_7.getCharacter()); - n -= 7; - } - if (n - 6 >= 0) { - stringBuilder.append(OffsetFont.NEG_6.getCharacter()); - n -= 6; - } - if (n - 5 >= 0) { - stringBuilder.append(OffsetFont.NEG_5.getCharacter()); - n -= 5; - } - if (n - 4 >= 0) { - stringBuilder.append(OffsetFont.NEG_4.getCharacter()); - n -= 4; - } - if (n - 3 >= 0) { - stringBuilder.append(OffsetFont.NEG_3.getCharacter()); - n -= 3; - } - if (n - 2 >= 0) { - stringBuilder.append(OffsetFont.NEG_2.getCharacter()); - n -= 2; - } - if (n - 1 >= 0) { - stringBuilder.append(OffsetFont.NEG_1.getCharacter()); - } - return stringBuilder.toString(); - } - - public String getShortestPosChars(int n) { - StringBuilder stringBuilder = new StringBuilder(); - while (n >= 128) { - stringBuilder.append(OffsetFont.POS_128.getCharacter()); - n -= 128; - } - if (n - 64 >= 0) { - stringBuilder.append(OffsetFont.POS_64.getCharacter()); - n -= 64; - } - if (n - 32 >= 0) { - stringBuilder.append(OffsetFont.POS_32.getCharacter()); - n -= 32; - } - if (n - 16 >= 0) { - stringBuilder.append(OffsetFont.POS_16.getCharacter()); - n -= 16; - } - if (n - 8 >= 0) { - stringBuilder.append(OffsetFont.POS_8.getCharacter()); - n -= 8; - } - if (n - 7 >= 0) { - stringBuilder.append(OffsetFont.POS_7.getCharacter()); - n -= 7; - } - if (n - 6 >= 0) { - stringBuilder.append(OffsetFont.POS_6.getCharacter()); - n -= 6; - } - if (n - 5 >= 0) { - stringBuilder.append(OffsetFont.POS_5.getCharacter()); - n -= 5; - } - if (n - 4 >= 0) { - stringBuilder.append(OffsetFont.POS_4.getCharacter()); - n -= 4; - } - if (n - 3 >= 0) { - stringBuilder.append(OffsetFont.POS_3.getCharacter()); - n -= 3; - } - if (n - 2 >= 0) { - stringBuilder.append(OffsetFont.POS_2.getCharacter()); - n -= 2; - } - if (n - 1 >= 0) { - stringBuilder.append(OffsetFont.POS_1.getCharacter()); - } - return stringBuilder.toString(); - } - - private void loadUnicodesWidth() { - File[] files = new File(CustomNameplates.getInstance().getDataFolder(), "unicodes").listFiles(); - if (files == null) return; - for (File file : files) { - if (!file.getName().endsWith(".png")) continue; - String unicodeStr = file.getName().substring(file.getName().length() - 6, file.getName().length() - 4); - try { - BufferedImage bufferedImage = ImageIO.read(file); - int width = bufferedImage.getWidth(); - int height = bufferedImage.getHeight(); - int single = width / 16; - int max_y = height / single; - for (int i = 0; i < 16; i++) { - for (int j = 0; j < max_y; j++) { - int x_final = i * single; - outer: - for (int x = i * single; x < (i+1) * single; x++) { - for (int y = j * single; y < (j+1) * single; y++) { - int rgb = bufferedImage.getRGB(x, y); - int alpha = (rgb >> 24) & 0xff; - if (alpha != 0) { - x_final = x; - continue outer; - } - } - } - int single_real_width = (int) (((double) (x_final - i * single) / single * 8) + 1); - String unicode = "\\u" + unicodeStr + String.format("%02x", i + j * 16); - String unicodeFinalStr = StringEscapeUtils.unescapeJava(unicode); - customImageWidth.put(unicodeFinalStr.charAt(0), single_real_width); - } - } - } - catch (IOException ignored) { - Log.warn("Error occurred when reading png files"); - } - } - } - - @Nullable - public InputStream getResource(@NotNull String filename) { - try { - URL url = plugin.getClass().getClassLoader().getResource(filename); - if (url == null) { - return null; - } - URLConnection connection = url.openConnection(); - connection.setUseCaches(false); - return connection.getInputStream(); - } catch (IOException ex) { - return null; - } - } - - public void saveResource(@NotNull String resourcePath) { - if (resourcePath.equals("")) { - throw new IllegalArgumentException("ResourcePath cannot be null or empty"); - } - - resourcePath = resourcePath.replace('\\', '/'); - InputStream in = getResource(resourcePath); - if (in == null) { - return; - } - - File outFile = new File(plugin.getDataFolder(), resourcePath); - int lastIndex = resourcePath.lastIndexOf('/'); - File outDir = new File(plugin.getDataFolder(), resourcePath.substring(0, Math.max(lastIndex, 0))); - - if (!outDir.exists()) { - outDir.mkdirs(); - } - - try { - if (!outFile.exists()) { - OutputStream out = new FileOutputStream(outFile); - byte[] buf = new byte[1024]; - int len; - while ((len = in.read(buf)) > 0) { - out.write(buf, 0, len); - } - out.close(); - in.close(); - } - } catch (IOException ex) { - Log.warn("Could not save " + outFile.getName() + " to " + outFile); - } - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/ImageManager.java b/src/main/java/net/momirealms/customnameplates/manager/ImageManager.java deleted file mode 100644 index c79447a..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/ImageManager.java +++ /dev/null @@ -1,100 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.object.SimpleChar; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.configuration.file.YamlConfiguration; -import org.jetbrains.annotations.Nullable; - -import java.io.File; -import java.io.IOException; -import java.util.Arrays; -import java.util.Comparator; -import java.util.HashMap; - -public class ImageManager extends Function { - - private final HashMap characterMap; - private final CustomNameplates plugin; - - public ImageManager(CustomNameplates plugin) { - this.plugin = plugin; - this.characterMap = new HashMap<>(); - } - - @Override - public void load() { - if (!ConfigManager.enableImages) return; - loadImages(); - } - - @Override - public void unload() { - characterMap.clear(); - } - - private void loadImages() { - File img_file = new File(plugin.getDataFolder(), "contents" + File.separator + "images"); - if (!img_file.exists() && img_file.mkdirs()) { - saveDefaultImages(); - } - File[] image_config_files = img_file.listFiles(file -> file.getName().endsWith(".yml")); - if (image_config_files == null) return; - Arrays.sort(image_config_files, Comparator.comparing(File::getName)); - for (File image_config_file : image_config_files) { - char img = ConfigManager.start_char; - ConfigManager.start_char = (char) (img + '\u0001'); - String key = image_config_file.getName().substring(0, image_config_file.getName().length() - 4); - YamlConfiguration config = YamlConfiguration.loadConfiguration(image_config_file); - if (!config.contains("height")) config.set("height", 10); - if (!config.contains("ascent")) config.set("ascent", 8); - if (!config.contains("width")) config.set("width", 10); - if (!config.contains("image")) config.set("image", key); - try { - config.save(image_config_file); - } catch (IOException e) { - e.printStackTrace(); - } - SimpleChar simpleChar = new SimpleChar(config.getInt("height"), config.getInt("ascent"), config.getInt("width"), img, config.getString("image") + ".png"); - characterMap.put(key, simpleChar); - } - AdventureUtils.consoleMessage("[CustomNameplates] Loaded " + characterMap.size() + " images"); - } - - private void saveDefaultImages() { - String[] png_list = new String[]{"bell", "bubble", "clock", "coin", "compass", "weather", "stamina_0", "stamina_1", "stamina_2"}; - String[] part_list = new String[]{".png", ".yml"}; - for (String name : png_list) { - for (String part : part_list) { - plugin.saveResource("contents" + File.separator + "images" + File.separator + name + part, false); - } - } - } - - @Nullable - public SimpleChar getImage(String key) { - return characterMap.get(key); - } - - public HashMap getCharacterMap() { - return characterMap; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/MessageManager.java b/src/main/java/net/momirealms/customnameplates/manager/MessageManager.java deleted file mode 100644 index a5d61b6..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/MessageManager.java +++ /dev/null @@ -1,91 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.utils.ConfigUtils; -import org.bukkit.configuration.file.YamlConfiguration; - -import java.io.File; - -public class MessageManager extends Function { - - public static String noPerm; - public static String prefix; - public static String lackArgs; - public static String reload; - public static String not_online; - public static String no_console; - public static String coolDown; - public static String preview; - public static String generate; - public static String noNameplate; - public static String generateDone; - public static String np_equip; - public static String np_unEquip; - public static String np_force_equip; - public static String np_force_unEquip; - public static String np_not_exist; - public static String np_notAvailable; - public static String np_available; - public static String np_haveNone; - public static String bb_equip; - public static String bb_unEquip; - public static String bb_force_equip; - public static String bb_force_unEquip; - public static String bb_not_exist; - public static String bb_notAvailable; - public static String bb_available; - public static String bb_haveNone; - public static String nonArgs; - public static String unavailableArgs; - - @Override - public void load(){ - YamlConfiguration config = ConfigUtils.getConfig("messages" + File.separator + ConfigManager.lang +".yml"); - noPerm = config.getString("messages.no-perm"); - prefix = config.getString("messages.prefix"); - lackArgs = config.getString("messages.lack-args"); - reload = config.getString("messages.reload"); - coolDown = config.getString("messages.cooldown"); - preview = config.getString("messages.preview"); - generate = config.getString("messages.generate"); - generateDone = config.getString("messages.generate-done"); - noNameplate = config.getString("messages.no-nameplate"); - not_online = config.getString("messages.not-online"); - no_console = config.getString("messages.no-console"); - nonArgs = config.getString("messages.none-args"); - unavailableArgs = config.getString("messages.invalid-args"); - np_equip = config.getString("messages.equip-nameplates"); - np_unEquip = config.getString("messages.unequip-nameplates"); - np_force_equip = config.getString("messages.force-equip-nameplates"); - np_force_unEquip = config.getString("messages.force-unequip-nameplates"); - np_not_exist = config.getString("messages.not-exist-nameplates"); - np_notAvailable = config.getString("messages.not-available-nameplates"); - np_available = config.getString("messages.available-nameplates"); - np_haveNone = config.getString("messages.have-no-nameplates"); - bb_equip = config.getString("messages.equip-bubbles"); - bb_unEquip = config.getString("messages.unequip-bubbles"); - bb_force_equip = config.getString("messages.force-equip-bubbles"); - bb_force_unEquip = config.getString("messages.force-unequip-bubbles"); - bb_not_exist = config.getString("messages.not-exist-bubbles"); - bb_notAvailable = config.getString("messages.not-available-bubbles"); - bb_available = config.getString("messages.available-bubbles"); - bb_haveNone = config.getString("messages.have-no-bubbles"); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/NameplateManager.java b/src/main/java/net/momirealms/customnameplates/manager/NameplateManager.java deleted file mode 100644 index 4a0d969..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/NameplateManager.java +++ /dev/null @@ -1,315 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.api.CustomNameplatesAPI; -import net.momirealms.customnameplates.object.ConditionalText; -import net.momirealms.customnameplates.object.DisplayMode; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.object.SimpleChar; -import net.momirealms.customnameplates.object.carrier.*; -import net.momirealms.customnameplates.object.font.OffsetFont; -import net.momirealms.customnameplates.object.nameplate.NameplateConfig; -import net.momirealms.customnameplates.object.requirements.Requirement; -import net.momirealms.customnameplates.utils.AdventureUtils; -import net.momirealms.customnameplates.utils.ConfigUtils; -import org.bukkit.ChatColor; -import org.bukkit.configuration.ConfigurationSection; -import org.bukkit.configuration.file.YamlConfiguration; -import org.bukkit.entity.Player; -import org.bukkit.permissions.PermissionAttachmentInfo; - -import java.io.File; -import java.io.IOException; -import java.util.*; - -public class NameplateManager extends Function { - - private String default_nameplate; - private String player_prefix; - private String player_suffix; - private String player_name_papi; - private long preview_time; - private DisplayMode mode; - private boolean fakeTeam; - private final HashMap nameplateConfigMap; - private final CustomNameplates plugin; - private AbstractTextCarrier textCarrier; - protected HashMap previewCoolDown = new HashMap<>(); - private boolean hidePrefix; - private boolean hideSuffix; - - public NameplateManager(CustomNameplates plugin) { - this.plugin = plugin; - this.nameplateConfigMap = new HashMap<>(); - } - - @Override - public void load() { - if (!ConfigManager.enableNameplates) return; - YamlConfiguration config = ConfigUtils.getConfig("configs" + File.separator + "nameplate.yml"); - loadConfig(config); - loadNameplates(); - loadMode(config); - } - - @Override - public void unload() { - if (this.textCarrier != null) { - this.textCarrier.unload(); - this.textCarrier = null; - } - this.nameplateConfigMap.clear(); - } - - private void loadConfig(ConfigurationSection config) { - this.default_nameplate = config.getString("default-nameplate", "none"); - this.player_name_papi = config.getString("player-name", "%player_name%"); - this.preview_time = config.getLong("preview-duration", 5); - this.fakeTeam = config.getBoolean("create-fake-team",true); - this.player_prefix = config.getString("prefix",""); - this.player_suffix = config.getString("suffix",""); - this.hidePrefix = config.getBoolean("hide-prefix-when-equipping-nameplate", false); - this.hideSuffix = config.getBoolean("hide-suffix-when-equipping-nameplate", false); - } - - private void loadNameplates() { - File np_file = new File(plugin.getDataFolder(), "contents" + File.separator + "nameplates"); - if (!np_file.exists() && np_file.mkdirs()) { - saveDefaultNameplates(); - } - File[] np_config_files = np_file.listFiles(file -> file.getName().endsWith(".yml")); - if (np_config_files == null) return; - Arrays.sort(np_config_files, Comparator.comparing(File::getName)); - for (File np_config_file : np_config_files) { - char left = ConfigManager.start_char; - char middle; - char right; - ConfigManager.start_char = (char) ((right = (char) ((middle = (char) (ConfigManager.start_char + '\u0001')) + '\u0001')) + '\u0001'); - String key = np_config_file.getName().substring(0, np_config_file.getName().length() - 4); - YamlConfiguration config = YamlConfiguration.loadConfiguration(np_config_file); - if (!config.contains("display-name")) config.set("display-name", key); - if (!config.contains("name-color")) config.set("name-color", "white"); - if (!config.contains("left.image")) config.set("left.image", key + "_left"); - if (!config.contains("left.height")) config.set("left.height", 16); - if (!config.contains("left.ascent")) config.set("left.ascent", 12); - if (!config.contains("left.width")) config.set("left.width", 16); - if (!config.contains("middle.image")) config.set("middle.image", key + "_middle"); - if (!config.contains("middle.height")) config.set("middle.height", 16); - if (!config.contains("middle.ascent")) config.set("middle.ascent", 12); - if (!config.contains("middle.width")) config.set("middle.width", 16); - if (!config.contains("right.image")) config.set("right.image", key + "_right"); - if (!config.contains("right.height")) config.set("right.height", 16); - if (!config.contains("right.ascent")) config.set("right.ascent", 12); - if (!config.contains("right.width")) config.set("right.width", 16); - try { - config.save(np_config_file); - } catch (IOException e) { - e.printStackTrace(); - } - SimpleChar leftChar = new SimpleChar(config.getInt("left.height"), config.getInt("left.ascent"), config.getInt("left.width"), left, config.getString("left.image") + ".png"); - SimpleChar middleChar = new SimpleChar(config.getInt("middle.height"), config.getInt("middle.ascent"), config.getInt("middle.width"), middle, config.getString("middle.image") + ".png"); - SimpleChar rightChar = new SimpleChar(config.getInt("right.height"), config.getInt("right.ascent"), config.getInt("right.width"), right, config.getString("right.image") + ".png"); - ChatColor color = ChatColor.valueOf(Objects.requireNonNull(config.getString("color", "WHITE")).toUpperCase(Locale.ENGLISH)); - nameplateConfigMap.put(key, new NameplateConfig(color, config.getString("display-name"), leftChar, middleChar, rightChar)); - } - AdventureUtils.consoleMessage("[CustomNameplates] Loaded " + nameplateConfigMap.size() + " nameplates"); - } - - private void saveDefaultNameplates() { - String[] png_list = new String[]{"cat", "egg", "cheems", "wither", "xmas", "halloween", "hutao", "starsky", "trident", "rabbit"}; - String[] part_list = new String[]{"_left.png", "_middle.png", "_right.png", ".yml"}; - for (String name : png_list) { - for (String part : part_list) { - plugin.saveResource("contents" + File.separator + "nameplates" + File.separator + name + part, false); - } - } - } - - private void loadMode(ConfigurationSection config) { - this.mode = DisplayMode.valueOf(config.getString("mode","Team").toUpperCase(Locale.ENGLISH)); - if (mode == DisplayMode.TEAM) { - this.textCarrier = new TeamInfoCarrier(plugin); - } else if (mode == DisplayMode.ARMOR_STAND) { - HashMap contentMap = new HashMap<>(); - ConfigurationSection armorStandSection = config.getConfigurationSection("armor_stand"); - if (armorStandSection != null) { - for (String key : armorStandSection.getKeys(false)) { - String text = armorStandSection.getString(key + ".text"); - double offset = armorStandSection.getDouble(key + ".vertical-offset"); - Requirement[] requirements = ConfigUtils.getRequirements(armorStandSection.getConfigurationSection(key + ".conditions")); - contentMap.put(new ConditionalText(requirements, text, null), offset); - } - } - this.textCarrier = new NamedEntityCarrier(plugin, mode, contentMap); - } else if (mode == DisplayMode.TEXT_DISPLAY) { - HashMap contentMap = new HashMap<>(); - ConfigurationSection textDisplaySection = config.getConfigurationSection("text_display"); - if (textDisplaySection != null) { - for (String key :textDisplaySection.getKeys(false)) { - String text = textDisplaySection.getString(key + ".text"); - double offset = textDisplaySection.getDouble(key + ".vertical-offset") + 1.2; - Requirement[] requirements = ConfigUtils.getRequirements(textDisplaySection.getConfigurationSection(key + ".conditions")); - TextDisplayMeta textDisplayMeta = ConfigUtils.getTextDisplayMeta(textDisplaySection.getConfigurationSection("options")); - contentMap.put(new ConditionalText(requirements, text, textDisplayMeta), offset); - } - } - this.textCarrier = new NamedEntityCarrier(plugin, mode, contentMap); - } else if (mode == DisplayMode.DISABLE) { - this.textCarrier = new DisableNameplate(plugin); - } - plugin.getTeamManager().setTeamPacketInterface(); - if (this.textCarrier != null) { - this.textCarrier.load(); - } - } - - public String getNameplatePrefixWithFont(String text, NameplateConfig nameplate) { - return ConfigManager.surroundWithFont(getNameplatePrefix(text, nameplate)); - } - - public String getNameplatePrefix(String text, NameplateConfig nameplate) { - int totalWidth = plugin.getFontManager().getTotalPlayerNameWidth(text); - StringBuilder stringBuilder = new StringBuilder(); - stringBuilder.append(plugin.getFontManager().getShortestNegChars(totalWidth % 2 == 0 ? totalWidth + nameplate.left().getWidth() : totalWidth + nameplate.left().getWidth() + 1)) - .append(nameplate.left().getChars()).append(OffsetFont.NEG_1.getCharacter()); - int mid_amount = (totalWidth - 1) / (nameplate.middle().getWidth()); - if (mid_amount == 0) { - stringBuilder.append(nameplate.middle().getChars()).append(OffsetFont.NEG_1.getCharacter()); - } - else { - for (int i = 0; i < mid_amount; i++) { - stringBuilder.append(nameplate.middle().getChars()).append(OffsetFont.NEG_1.getCharacter()); - } - stringBuilder.append( - plugin.getFontManager().getShortestNegChars( - nameplate.middle().getWidth() - (totalWidth - 1) % nameplate.middle().getWidth() - ) - ); - stringBuilder.append(nameplate.middle().getChars()).append(OffsetFont.NEG_1.getCharacter()); - } - stringBuilder.append(nameplate.right().getChars()); - stringBuilder.append(plugin.getFontManager().getShortestNegChars(totalWidth + nameplate.right().getWidth())); - return stringBuilder.toString(); - } - - public List getAvailableNameplates(Player player) { - List nameplates = new ArrayList<>(); - for (PermissionAttachmentInfo info : player.getEffectivePermissions()) { - String permission = info.getPermission().toLowerCase(); - if (permission.startsWith("nameplates.equip.")) { - permission = permission.substring(17); - if (nameplateConfigMap.get(permission) != null) { - nameplates.add(permission); - } - } - } - return nameplates; - } - - public boolean isInCoolDown(Player player) { - long time = System.currentTimeMillis(); - if (time - (previewCoolDown.getOrDefault(player.getUniqueId(), time - this.getPreview_time() * 1050)) < this.getPreview_time() * 1050) { - return true; - } - previewCoolDown.put(player.getUniqueId(), time); - return false; - } - - public void showPlayerArmorStandTags(Player player) { - NamedEntityCarrier namedEntityCarrier = (NamedEntityCarrier) this.getTextCarrier(); - NamedEntityManager nem = namedEntityCarrier.getNamedEntityManager(player); - nem.spawn(player); - for (int i = 1; i <= this.getPreview_time() * 20; i++) { - plugin.getScheduler().runTaskAsyncLater(()-> { - nem.teleport(player); - }, i); - } - plugin.getScheduler().runTaskAsyncLater(()-> nem.destroy(player), this.getPreview_time() * 20); - } - - public void showPlayerArmorStandTags(Player player, String nameplate) { - String current = getEquippedNameplate(player); - if (!nameplate.equals(current)) { - plugin.getDataManager().equipNameplate(player, nameplate); - CustomNameplatesAPI.getInstance().updateNameplateTeam(player); - plugin.getScheduler().runTaskAsyncLater(()-> { - plugin.getDataManager().equipNameplate(player, current); - CustomNameplatesAPI.getInstance().updateNameplateTeam(player); - },this.getPreview_time() * 20); - } - showPlayerArmorStandTags(player); - } - - public boolean existNameplate(String nameplate) { - return nameplateConfigMap.containsKey(nameplate); - } - - public AbstractTextCarrier getTextCarrier() { - return textCarrier; - } - - public HashMap getNameplateConfigMap() { - return nameplateConfigMap; - } - - public boolean isFakeTeam() { - return fakeTeam; - } - - public DisplayMode getMode() { - return mode; - } - - public String getEquippedNameplate(Player player) { - return plugin.getDataManager().getEquippedNameplate(player); - } - - public String getDefault_nameplate() { - return default_nameplate; - } - - public String getPrefix() { - return player_prefix; - } - - public String getSuffix() { - return player_suffix; - } - - public NameplateConfig getNameplateConfig(String nameplate) { - return nameplateConfigMap.get(nameplate); - } - - public String getPlayerNamePapi() { - return player_name_papi; - } - - public long getPreview_time() { - return preview_time; - } - - public boolean isPrefixHidden() { - return hidePrefix; - } - - public boolean isSuffixHidden() { - return hideSuffix; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/PlaceholderManager.java b/src/main/java/net/momirealms/customnameplates/manager/PlaceholderManager.java deleted file mode 100644 index b2ae443..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/PlaceholderManager.java +++ /dev/null @@ -1,237 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import me.clip.placeholderapi.PlaceholderAPI; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.ConditionalText; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.object.font.OffsetFont; -import net.momirealms.customnameplates.object.placeholders.*; -import net.momirealms.customnameplates.utils.ConfigUtils; -import org.bukkit.configuration.ConfigurationSection; -import org.bukkit.configuration.file.YamlConfiguration; - -import java.io.File; -import java.util.*; -import java.util.regex.Matcher; -import java.util.regex.Pattern; - -public class PlaceholderManager extends Function { - - private final NameplatePlaceholders nameplatePlaceholders; - private final Pattern placeholderPattern = Pattern.compile("%([^%]*)%"); - private final HashSet descent_fonts; - private final HashSet descent_unicode_fonts; - private final HashMap nameplateTextMap; - private final HashMap backGroundTextMap; - private final HashMap stringStaticTextMap; - private final HashMap descentTextMap; - private final HashMap descentUnicodeMap; - private final HashMap conditionalTextsMap; - private final HashMap vanillaHudMap; - private final CustomNameplates plugin; - - public PlaceholderManager(CustomNameplates plugin) { - this.plugin = plugin; - this.nameplatePlaceholders = new NameplatePlaceholders(plugin, this); - this.descent_fonts = new HashSet<>(); - this.descent_unicode_fonts = new HashSet<>(); - this.nameplateTextMap = new HashMap<>(); - this.backGroundTextMap = new HashMap<>(); - this.stringStaticTextMap = new HashMap<>(); - this.descentTextMap = new HashMap<>(); - this.conditionalTextsMap = new HashMap<>(); - this.vanillaHudMap = new HashMap<>(); - this.descentUnicodeMap = new HashMap<>(); - } - - @Override - public void load() { - this.nameplatePlaceholders.register(); - this.loadConfig(); - } - - @Override - public void unload() { - this.nameplatePlaceholders.unregister(); - this.descent_fonts.clear(); - this.nameplateTextMap.clear(); - this.backGroundTextMap.clear(); - this.stringStaticTextMap.clear(); - this.descentTextMap.clear(); - this.conditionalTextsMap.clear(); - this.vanillaHudMap.clear(); - this.descent_unicode_fonts.clear(); - this.descentUnicodeMap.clear(); - } - - private void loadConfig() { - YamlConfiguration config = ConfigUtils.getConfig("configs" + File.separator + "custom-placeholders.yml"); - - ConfigurationSection nameplateSection = config.getConfigurationSection("nameplate-text"); - if (nameplateSection != null && ConfigManager.enableNameplates) { - loadNameplateText(nameplateSection); - } - - ConfigurationSection backgroundSection = config.getConfigurationSection("background-text"); - if (backgroundSection != null && ConfigManager.enableBackground) { - loadBackgroundText(backgroundSection); - } - - ConfigurationSection staticTextSection = config.getConfigurationSection("static-text"); - if (staticTextSection != null) { - loadStaticText(staticTextSection); - } - - ConfigurationSection descentSection = config.getConfigurationSection("descent-text"); - if (descentSection != null) { - loadDescentText(descentSection); - } - - ConfigurationSection conditionalSection = config.getConfigurationSection("conditional-text"); - if (conditionalSection != null) { - loadConditionalText(conditionalSection); - } - - ConfigurationSection vanillaHudSection = config.getConfigurationSection("vanilla-hud"); - if (vanillaHudSection != null) { - loadVanillaHud(vanillaHudSection); - } - - if (plugin.getVersionHelper().isVersionNewerThan1_20() && !ConfigManager.enable1_20_Unicode) { - return; - } - - ConfigurationSection descentUnicodeSection = config.getConfigurationSection("descent-unicode"); - if (descentUnicodeSection != null) { - loadDescentUnicode(descentUnicodeSection); - } - } - - private void loadVanillaHud(ConfigurationSection section) { - for (String key : section.getKeys(false)) { - vanillaHudMap.put(key, new VanillaHud( - PlaceholderAPI.setPlaceholders(null, section.getString(key + ".images.empty", "")) + ConfigManager.surroundWithFont(String.valueOf(OffsetFont.NEG_2.getCharacter())), - PlaceholderAPI.setPlaceholders(null, section.getString(key + ".images.half", "")) + ConfigManager.surroundWithFont(String.valueOf(OffsetFont.NEG_2.getCharacter())), - PlaceholderAPI.setPlaceholders(null, section.getString(key + ".images.full", "")) + ConfigManager.surroundWithFont(String.valueOf(OffsetFont.NEG_2.getCharacter())), - section.getString(key + ".placeholder.value"), - section.getString(key + ".placeholder.max-value"), - section.getBoolean(key + ".reverse", true) - )); - } - } - - private void loadConditionalText(ConfigurationSection section) { - for (String key : section.getKeys(false)) { - ConfigurationSection innerSection = section.getConfigurationSection(key); - if (innerSection != null) { - ArrayList conditionalTexts = new ArrayList<>(); - for (String priority : innerSection.getKeys(false)) { - ConditionalText conditionalText = new ConditionalText( - ConfigUtils.getRequirements(innerSection.getConfigurationSection(priority + ".conditions")), - innerSection.getString(priority + ".text"), - null - ); - conditionalTexts.add(conditionalText); - } - conditionalTextsMap.put(key, new ConditionalTexts(conditionalTexts.toArray(new ConditionalText[0]))); - } - } - } - - private void loadDescentText(ConfigurationSection section) { - for (String key : section.getKeys(false)) { - descent_fonts.add(8 - section.getInt(key + ".descent")); - descentTextMap.put(key, new DescentText(section.getString(key + ".text"), 8 - section.getInt(key + ".descent"))); - } - } - - private void loadDescentUnicode(ConfigurationSection section) { - for (String key : section.getKeys(false)) { - descent_unicode_fonts.add(8 - section.getInt(key + ".descent")); - descentUnicodeMap.put(key, new DescentText(section.getString(key + ".text"), 8 - section.getInt(key + ".descent"))); - } - } - - private void loadStaticText(ConfigurationSection section) { - for (String key : section.getKeys(false)) { - stringStaticTextMap.put(key, new StaticText(section.getString(key + ".text"), section.getInt(key + ".value"), StaticText.StaticState.valueOf(section.getString(key + ".position", "left").toUpperCase(Locale.ENGLISH)))); - } - } - - private void loadNameplateText(ConfigurationSection section) { - for (String key : section.getKeys(false)) { - nameplateTextMap.put(key, new NameplateText(section.getString(key + ".text"), section.getString(key + ".nameplate"))); - } - } - - private void loadBackgroundText(ConfigurationSection section) { - for (String key : section.getKeys(false)) { - backGroundTextMap.put(key, new BackGroundText( - section.getString(key + ".text"), - section.getString(key + ".background"), - section.getBoolean(key + ".remove-shadow", true) - ) - ); - } - } - - public List detectPlaceholders(String text){ - List placeholders = new ArrayList<>(); - Matcher matcher = placeholderPattern.matcher(text); - while (matcher.find()) placeholders.add(matcher.group()); - return placeholders; - } - - public ConditionalTexts getConditionalTexts(String key) { - return conditionalTextsMap.get(key); - } - - public BackGroundText getBackgroundText(String key) { - return backGroundTextMap.get(key); - } - - public NameplateText getNameplateText(String key) { - return nameplateTextMap.get(key); - } - - public StaticText getStaticText(String key) { - return stringStaticTextMap.get(key); - } - - public DescentText getDescentText(String key) { - return descentTextMap.get(key); - } - - public VanillaHud getVanillaHud(String key) { - return vanillaHudMap.get(key); - } - - public HashSet getDescent_fonts() { - return descent_fonts; - } - - public HashSet getDescent_unicode_fonts() { - return descent_unicode_fonts; - } - - public DescentText getDescentUnicode(String key) { - return descentUnicodeMap.get(key); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/manager/ResourceManager.java b/src/main/java/net/momirealms/customnameplates/manager/ResourceManager.java deleted file mode 100644 index 0d3e503..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/ResourceManager.java +++ /dev/null @@ -1,418 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import com.google.gson.JsonArray; -import com.google.gson.JsonObject; -import com.google.gson.JsonPrimitive; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.SimpleChar; -import net.momirealms.customnameplates.object.background.BackGroundConfig; -import net.momirealms.customnameplates.object.bubble.BubbleConfig; -import net.momirealms.customnameplates.object.font.OffsetFont; -import net.momirealms.customnameplates.object.nameplate.NameplateConfig; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.apache.commons.io.FileUtils; -import org.bukkit.Bukkit; - -import java.io.*; -import java.nio.charset.StandardCharsets; -import java.util.ArrayList; -import java.util.List; -import java.util.Objects; - -public class ResourceManager { - - private final CustomNameplates plugin; - - public ResourceManager(CustomNameplates plugin) { - this.plugin = plugin; - } - - public void generateResourcePack() { - File resourcePack_folder = new File(plugin.getDataFolder() + File.separator + "ResourcePack"); - this.deleteDirectory(resourcePack_folder); - File font_folder = new File(plugin.getDataFolder(), "ResourcePack" + File.separator + "assets" + File.separator + ConfigManager.namespace + File.separatorChar + "font"); - File textures_folder = new File(plugin.getDataFolder(), "ResourcePack" + File.separator+ "assets" + File.separator + ConfigManager.namespace + File.separatorChar + "textures"); - if (!font_folder.mkdirs() || !textures_folder.mkdirs()) { - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to generate resource pack folders..."); - return; - } - JsonObject fontJsonObject = new JsonObject(); - JsonArray fontJsonArray = new JsonArray(); - fontJsonObject.add("providers", fontJsonArray); - this.getOffsetFontEnums().forEach(fontJsonArray::add); - this.loadNameplates(fontJsonArray, textures_folder); - this.loadBubbles(fontJsonArray, textures_folder); - this.loadBackgrounds(fontJsonArray, textures_folder); - this.loadImages(fontJsonArray, textures_folder); - this.extractDefault(); - this.saveOffsets(); - this.saveSplit(textures_folder.getPath()); - try (FileWriter fileWriter = new FileWriter( - plugin.getDataFolder() + - File.separator + "ResourcePack" + - File.separator + "assets" + - File.separator + ConfigManager.namespace + - File.separator + "font" + - File.separator + ConfigManager.font + ".json") - ) - { - fileWriter.write(fontJsonObject.toString().replace("\\\\", "\\")); - } - catch (IOException e) { - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to generate font json file."); - } - this.hookCopy(resourcePack_folder); - } - - private void loadImages(JsonArray jsonArray, File textures_file) { - if (!ConfigManager.enableImages) return; - for (SimpleChar simpleChar : plugin.getImageManager().getCharacterMap().values()) { - JsonObject jo_np = new JsonObject(); - jo_np.add("type", new JsonPrimitive("bitmap")); - jo_np.add("file", new JsonPrimitive(ConfigManager.namespace + ":" + ConfigManager.images_folder_path.replaceAll("\\\\", "/") + simpleChar.getFile())); - addCharToArray(jsonArray, simpleChar, jo_np); - try { - FileUtils.copyFile(new File(plugin.getDataFolder(), "contents" + File.separator + "images" + File.separator + simpleChar.getFile()), new File(textures_file.getPath() + File.separatorChar + ConfigManager.images_folder_path.replace("\\", File.separator) + simpleChar.getFile())); - } catch (IOException e) { - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to copy images to resource pack."); - } - } - } - - private void loadNameplates(JsonArray jsonArray, File textures_file) { - if (!ConfigManager.enableNameplates) return; - for (NameplateConfig nameplateConfig : plugin.getNameplateManager().getNameplateConfigMap().values()) { - for (SimpleChar simpleChar : new SimpleChar[]{nameplateConfig.left(), nameplateConfig.middle(), nameplateConfig.right()}) { - JsonObject jo_np = new JsonObject(); - jo_np.add("type", new JsonPrimitive("bitmap")); - jo_np.add("file", new JsonPrimitive(ConfigManager.namespace + ":" + ConfigManager.nameplates_folder_path.replaceAll("\\\\", "/") + simpleChar.getFile())); - addCharToArray(jsonArray, simpleChar, jo_np); - try { - FileUtils.copyFile(new File(plugin.getDataFolder(), "contents" + File.separator + "nameplates" + File.separator + simpleChar.getFile()), new File(textures_file.getPath() + File.separatorChar + ConfigManager.nameplates_folder_path.replace("\\", File.separator) + simpleChar.getFile())); - } catch (IOException e) { - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to copy nameplates to resource pack."); - } - } - } - } - - private void loadBubbles(JsonArray jsonArray, File textures_file) { - if (!ConfigManager.enableBubbles) return; - for (BubbleConfig bubbleConfig : plugin.getChatBubblesManager().getBubbleConfigMap().values()) { - SimpleChar[] simpleChars = new SimpleChar[]{bubbleConfig.left(), bubbleConfig.middle(), bubbleConfig.right(), bubbleConfig.tail()}; - for (SimpleChar simpleChar : simpleChars) { - JsonObject jo_bb = new JsonObject(); - jo_bb.add("type", new JsonPrimitive("bitmap")); - jo_bb.add("file", new JsonPrimitive(ConfigManager.namespace + ":" + ConfigManager.bubbles_folder_path.replaceAll("\\\\","/") + simpleChar.getFile())); - addCharToArray(jsonArray, simpleChar, jo_bb); - try { - FileUtils.copyFile(new File(plugin.getDataFolder(),"contents" + File.separator + "bubbles" + File.separator + simpleChar.getFile()), new File(textures_file.getPath() + File.separator + ConfigManager.bubbles_folder_path.replace("\\", File.separator) + simpleChar.getFile())); - } - catch (IOException e){ - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to copy bubbles to resource pack."); - } - } - } - } - - private void loadBackgrounds(JsonArray jsonArray, File textures_file) { - if (!ConfigManager.enableBackground) return; - for (BackGroundConfig backGroundConfig : plugin.getBackgroundManager().getBackGroundConfigMap().values()) { - for (SimpleChar simpleChar : new SimpleChar[]{ - backGroundConfig.left(), backGroundConfig.offset_1(), - backGroundConfig.offset_2(), backGroundConfig.offset_4(), - backGroundConfig.offset_8(), backGroundConfig.offset_16(), - backGroundConfig.offset_32(), backGroundConfig.offset_64(), - backGroundConfig.offset_128(), backGroundConfig.right()}) { - JsonObject jo_bg = new JsonObject(); - jo_bg.add("type", new JsonPrimitive("bitmap")); - jo_bg.add("file", new JsonPrimitive(ConfigManager.namespace + ":" + ConfigManager.backgrounds_folder_path.replaceAll("\\\\","/") + simpleChar.getFile())); - addCharToArray(jsonArray, simpleChar, jo_bg); - try { - FileUtils.copyFile(new File(plugin.getDataFolder(), "contents" + File.separator + "backgrounds" + File.separator + simpleChar.getFile()), new File(textures_file.getPath() + File.separatorChar + ConfigManager.backgrounds_folder_path.replace("\\", File.separator) + simpleChar.getFile())); - } catch (IOException e){ - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to copy backgrounds to resource pack."); - } - } - } - } - - private void addCharToArray(JsonArray jsonArray, SimpleChar simpleChar, JsonObject jsonObject) { - jsonObject.add("ascent", new JsonPrimitive(simpleChar.getAscent())); - jsonObject.add("height", new JsonPrimitive(simpleChar.getHeight())); - JsonArray ja_simple = new JsonArray(); - ja_simple.add(native2ascii(simpleChar.getChars())); - jsonObject.add("chars", ja_simple); - jsonArray.add(jsonObject); - } - - private void extractDefault() { - if (ConfigManager.extractShader) { - String path = "ResourcePack" + File.separator + "assets" + File.separator + "minecraft" + File.separator + "shaders" + File.separator + "core" + File.separator; - plugin.saveResource(path + "rendertype_text.fsh", true); - plugin.saveResource(path + "rendertype_text.json", true); - plugin.saveResource(path + "rendertype_text.vsh", true); - String line; - StringBuilder sb = new StringBuilder(); - try (BufferedReader reader = new BufferedReader( - new InputStreamReader( - new FileInputStream( - new File(plugin.getDataFolder(), path + "rendertype_text.vsh") - ), StandardCharsets.UTF_8 - ) - )) { - while ((line = reader.readLine()) != null) { - sb.append(line).append(System.lineSeparator()); - } - } catch (IOException e) { - e.printStackTrace(); - } - File outPut = new File(plugin.getDataFolder(), path + "rendertype_text.vsh"); - try (BufferedWriter writer = new BufferedWriter( - new OutputStreamWriter(new FileOutputStream(outPut), StandardCharsets.UTF_8))) { - writer.write(sb.toString() - .replace("{hide}", !ConfigManager.hideScoreboardNumber ? "" : - "if (Position.z == 0.0\n" + - " && gl_Position.x >= 0.94\n" + - " && gl_Position.y >= -0.35\n" + - " && vertexColor.g == 84.0/255.0\n" + - " && vertexColor.g == 84.0/255.0\n" + - " && vertexColor.r == 252.0/255.0\n" + - " && gl_VertexID <= 7\n" + - " ) {\n" + - " gl_Position = ProjMat * ModelViewMat * vec4(ScreenSize + 100.0, 0.0, ScreenSize + 100.0);\n" + - " }" - ) - .replace("{IA}", !ConfigManager.iaShaderSupport ? "" : - "if (Color.xyz == vec3(255., 255., 254.) / 255.) {\n" + - " gl_Position = ProjMat * ModelViewMat * vertex;\n" + - " vertexColor = ((.6 + .6 * cos(6. * (gl_Position.x + GameTime * 1000.) + vec4(0, 23, 21, 1))) + vec4(0., 0., 0., 1.)) * texelFetch(Sampler2, UV2 / 16, 0);\n" + - " } else if(Color.xyz == vec3(255., 255., 253.) / 255.) {\n" + - " gl_Position = ProjMat * ModelViewMat * vertex;\n" + - " vertexColor = Color * texelFetch(Sampler2, UV2 / 16, 0);\n" + - " gl_Position.y = gl_Position.y + sin(GameTime * 12000. + (gl_Position.x * 6)) / 150.;\n" + - " } else if (Color.xyz == vec3(255., 255., 252.) / 255.) {\n" + - " gl_Position = ProjMat * ModelViewMat * vertex;\n" + - " vertexColor = ((.6 + .6 * cos(6. * (gl_Position.x + GameTime * 1000.) + vec4(0, 23, 21, 1))) + vec4(0., 0., 0., 1.)) * texelFetch(Sampler2, UV2 / 16, 0);\n" + - " gl_Position.y = gl_Position.y + sin(GameTime*12000. + (gl_Position.x*6)) / 150.;\n" + - " } else if (Color.xyz == vec3(255., 255., 251.) / 255.) {\n" + - " vertexColor = Color * texelFetch(Sampler2, UV2 / 16, 0);\n" + - " float vertexId = mod(gl_VertexID, 4.0);\n" + - " if (vertex.z <= 0.) {\n" + - " if (vertexId == 3. || vertexId == 0.) vertex.y += cos(GameTime * 12000. / 4) * 0.1;\n" + - " vertex.y += max(cos(GameTime*12000. / 4) * 0.1, 0.);\n" + - " } else {\n" + - " if (vertexId == 3. || vertexId == 0.) vertex.y -= cos(GameTime * 12000. / 4) * 3;\n" + - " vertex.y -= max(cos(GameTime*12000. / 4) * 4, 0.);\n" + - " }\n" + - " gl_Position = ProjMat * ModelViewMat * vertex;\n" + - " } else if (Color.xyz == vec3(255., 254., 254.) / 255.) {\n" + - " float vertexId = mod(gl_VertexID, 4.0);\n" + - " if (vertex.z <= 0.) {\n" + - " if (vertexId == 3. || vertexId == 0.) vertex.y += cos(GameTime * 12000. / 4) * 0.1;\n" + - " vertex.y += max(cos(GameTime*12000. / 4) * 0.1, 0.);\n" + - " } else {\n" + - " if (vertexId == 3. || vertexId == 0.) vertex.y -= cos(GameTime * 12000. / 4) * 3;\n" + - " vertex.y -= max(cos(GameTime*12000. / 4) * 4, 0.);\n" + - " }\n" + - " vertexColor = ((.6 + .6 * cos(6. * (gl_Position.x + GameTime * 1000.) + vec4(0, 23, 21, 1))) + vec4(0., 0., 0., 1.)) * texelFetch(Sampler2, UV2 / 16, 0);\n" + - " gl_Position = ProjMat * ModelViewMat * vertex;\n" + - " } else ") - ); - } catch (IOException e) { - e.printStackTrace(); - } - } - if (ConfigManager.extractLegacyBars) { - String path = "ResourcePack" + File.separator + "assets" + File.separator + "minecraft" + File.separator + "textures" + File.separator + "gui" + File.separator; - plugin.saveResource(path + "bars.png", true); - } - if (ConfigManager.extractBars) { - String path = "ResourcePack" + File.separator + "assets" + File.separator + "minecraft" + File.separator + "textures" + File.separator + "gui" + File.separator + "sprites" + File.separator + "boss_bar" + File.separator; - plugin.saveResource(path + "yellow_background.png", true); - plugin.saveResource(path + "yellow_progress.png", true); - } - setPackFormat(); - } - - private void setPackFormat() { - plugin.saveResource("ResourcePack" + File.separator + "pack.mcmeta", false); - File format_file = new File(plugin.getDataFolder(), "ResourcePack" + File.separator + "pack.mcmeta"); - String line; - StringBuilder sb = new StringBuilder(); - try (BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(format_file), StandardCharsets.UTF_8))) { - while ((line = reader.readLine()) != null) { - sb.append(line).append(System.lineSeparator()); - } - } catch (IOException e) { - e.printStackTrace(); - } - try (BufferedWriter writer = new BufferedWriter( - new OutputStreamWriter(new FileOutputStream(new File(plugin.getDataFolder(), - "ResourcePack" + File.separator + "pack.mcmeta")), StandardCharsets.UTF_8))) { - writer.write(sb.toString().replace("%version%", String.valueOf(plugin.getVersionHelper().getPack_format()))); - } catch (IOException e) { - e.printStackTrace(); - } - } - - private void saveOffsets() { - for (int ascent : plugin.getPlaceholderManager().getDescent_fonts()) { - String line; - StringBuilder sb = new StringBuilder(); - try (BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(new File(plugin.getDataFolder(), "templates" + File.separator + "default.json")), StandardCharsets.UTF_8))) { - while ((line = reader.readLine()) != null) { - sb.append(line).append(System.lineSeparator()); - } - } catch (IOException e) { - e.printStackTrace(); - } - File outPut = new File(plugin.getDataFolder(), - "ResourcePack" + - File.separator + "assets" + - File.separator + ConfigManager.namespace + - File.separator + "font" + - File.separator + "ascent_" + ascent + ".json"); - try (BufferedWriter writer = new BufferedWriter( - new OutputStreamWriter(new FileOutputStream(outPut), StandardCharsets.UTF_8))) { - writer.write(sb.toString().replace("\\\\", "\\").replace("%ascent%", String.valueOf(ascent)).replace("%ASCENT%", String.valueOf(ascent+3))); - } catch (IOException e) { - e.printStackTrace(); - } - } - if (!ConfigManager.enable1_20_Unicode && plugin.getVersionHelper().isVersionNewerThan1_20()) { - AdventureUtils.consoleMessage("[CustomNameplates] For the moment decent unicode is not available on 1.20. You can enable support-1_20-unicodes in config.yml to ignore the limit."); - return; - } - for (int ascent : plugin.getPlaceholderManager().getDescent_unicode_fonts()) { - String line; - StringBuilder sb = new StringBuilder(); - try (BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(new File(plugin.getDataFolder(), "templates" + File.separator + "unicode.json")), StandardCharsets.UTF_8))) { - while ((line = reader.readLine()) != null) { - sb.append(line).append(System.lineSeparator()); - } - } catch (IOException e) { - e.printStackTrace(); - } - try (BufferedWriter writer = new BufferedWriter( - new OutputStreamWriter(new FileOutputStream(new File(plugin.getDataFolder(), - "ResourcePack" + - File.separator + "assets" + - File.separator + ConfigManager.namespace + - File.separator + "font" + - File.separator + "unicode_ascent_" + ascent + ".json")), StandardCharsets.UTF_8))) { - writer.write(sb.toString().replace("\\\\", "\\").replace("%ascent%", String.valueOf(ascent)).replace("%ASCENT%", String.valueOf(ascent+3))); - } catch (IOException e) { - e.printStackTrace(); - } - } - if (ConfigManager.enable1_20_Unicode) { - try { - FileUtils.copyDirectory(new File(plugin.getDataFolder(), "unicodes"), new File(plugin.getDataFolder(), - "ResourcePack" + - File.separator + "assets" + - File.separator + "minecraft" + - File.separator + "textures" + - File.separator + "font")); - } catch (IOException e) { - e.printStackTrace(); - } - } - } - - private void deleteDirectory(File file){ - if (file.exists()) { - try{ - FileUtils.deleteDirectory(file); - } - catch (IOException e){ - e.printStackTrace(); - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to delete the generated folder..." ); - } - } - } - - private List getOffsetFontEnums() { - ArrayList list = new ArrayList<>(); - for (OffsetFont offsetFont : OffsetFont.values()) { - list.add(this.getOffsetFontChar(offsetFont.getHeight(), offsetFont.getCharacter())); - } - return list; - } - - private JsonObject getOffsetFontChar(int height, char character) { - JsonObject jsonObject = new JsonObject(); - jsonObject.add("type", new JsonPrimitive("bitmap")); - jsonObject.add("file", new JsonPrimitive(ConfigManager.namespace + ":" + ConfigManager.space_split_folder_path.replaceAll("\\\\","/") + "space_split.png")); - jsonObject.add("ascent", new JsonPrimitive(-5000)); - jsonObject.add("height", new JsonPrimitive(height)); - final JsonArray jsonArray = new JsonArray(); - jsonArray.add(native2ascii(character)); - jsonObject.add("chars", jsonArray); - return jsonObject; - } - - private String native2ascii(char ch) { - if (ch > '\u007f') { - StringBuilder stringBuilder_1 = new StringBuilder("\\u"); - StringBuilder stringBuilder_2 = new StringBuilder(Integer.toHexString(ch)); - stringBuilder_2.reverse(); - for (int n = 4 - stringBuilder_2.length(), i = 0; i < n; i++) stringBuilder_2.append('0'); - for (int j = 0; j < 4; j++) stringBuilder_1.append(stringBuilder_2.charAt(3 - j)); - return stringBuilder_1.toString(); - } - return Character.toString(ch); - } - - @SuppressWarnings("ResultOfMethodCallIgnored") - private void saveSplit(String texture_folder_path) { - try { - plugin.saveResource("space_split.png", false); - FileUtils.copyFile(new File(plugin.getDataFolder(),"space_split.png"), new File(texture_folder_path + File.separator + ConfigManager.space_split_folder_path.replace("\\", File.separator) + "space_split.png")); - File file = new File(plugin.getDataFolder(),"space_split.png"); - if (file.exists()) { - file.delete(); - } - } catch (IOException e){ - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to copy space_split.png to resource pack..."); - } - } - - private void hookCopy(File resourcePack_folder) { - if (ConfigManager.itemsAdderHook){ - try { - FileUtils.copyDirectory(new File(resourcePack_folder, "assets"), new File(Objects.requireNonNull(Bukkit.getPluginManager().getPlugin("ItemsAdder")).getDataFolder() + File.separator + "contents" + File.separator + "nameplates" + File.separator + "resourcepack" + File.separator + "assets") ); - } - catch (IOException e){ - e.printStackTrace(); - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to copy files to ItemsAdder..."); - } - } - if (ConfigManager.oraxenHook){ - try { - FileUtils.copyDirectory(new File(resourcePack_folder, "assets"), new File(Objects.requireNonNull(Bukkit.getPluginManager().getPlugin("Oraxen")).getDataFolder() + File.separator + "pack" + File.separator + "assets")); - } - catch (IOException e){ - e.printStackTrace(); - AdventureUtils.consoleMessage("[CustomNameplates] Error! Failed to copy files to Oraxen..."); - } - } - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/manager/TeamManager.java b/src/main/java/net/momirealms/customnameplates/manager/TeamManager.java deleted file mode 100644 index 414a2b2..0000000 --- a/src/main/java/net/momirealms/customnameplates/manager/TeamManager.java +++ /dev/null @@ -1,155 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.manager; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.listener.JoinQuitListener; -import net.momirealms.customnameplates.object.DisplayMode; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.object.nameplate.NameplatesTeam; -import net.momirealms.customnameplates.object.team.TeamNameInterface; -import net.momirealms.customnameplates.object.team.TeamPacketInterface; -import net.momirealms.customnameplates.object.team.name.PlayerNameTeamImpl; -import net.momirealms.customnameplates.object.team.name.TABBungeeCordImpl; -import net.momirealms.customnameplates.object.team.name.TABImpl; -import net.momirealms.customnameplates.object.team.name.VelocitabImpl; -import net.momirealms.customnameplates.object.team.packet.TeamInfoImpl; -import net.momirealms.customnameplates.object.team.packet.TeamVisibilityImpl; -import org.bukkit.Bukkit; -import org.bukkit.entity.Player; -import org.bukkit.event.HandlerList; -import org.jetbrains.annotations.Nullable; - -import java.util.UUID; -import java.util.concurrent.ConcurrentHashMap; - -public class TeamManager extends Function { - - private final ConcurrentHashMap teams ; - private TeamPacketInterface teamPacketInterface; - private TeamNameInterface teamNameInterface; - private final JoinQuitListener joinQuitListener; - private final CustomNameplates plugin; - private final ConcurrentHashMap triedTimes; - - public TeamManager(CustomNameplates plugin) { - this.plugin = plugin; - this.teams = new ConcurrentHashMap<>(); - this.joinQuitListener = new JoinQuitListener(this); - this.triedTimes = new ConcurrentHashMap<>(); - } - - @Override - public void load() { - Bukkit.getPluginManager().registerEvents(joinQuitListener, plugin); - if (ConfigManager.tab_BC_hook) { - teamNameInterface = new TABBungeeCordImpl(); - } else if (ConfigManager.tab_hook) { - teamNameInterface = new TABImpl(); - } else if (ConfigManager.velocitab_hook) { - teamNameInterface = new VelocitabImpl(); - } else { - teamNameInterface = new PlayerNameTeamImpl(this); - } - teamNameInterface.load(); - } - - @Override - public void unload() { - HandlerList.unregisterAll(joinQuitListener); - if (teamNameInterface != null) { - teamNameInterface.unload(); - } - teamPacketInterface = null; - } - - @Override - public void onQuit(Player player) { - this.removePlayerFromTeamCache(player); - teamNameInterface.onQuit(player); - triedTimes.remove(player.getUniqueId()); - } - - public void setTeamPacketInterface() { - if (plugin.getNameplateManager().getMode() == DisplayMode.TEAM) { - teamPacketInterface = new TeamInfoImpl(this); - } else if (plugin.getNameplateManager().getMode() == DisplayMode.ARMOR_STAND || plugin.getNameplateManager().getMode() == DisplayMode.TEXT_DISPLAY) { - teamPacketInterface = new TeamVisibilityImpl(this); - } - } - - public void sendUpdateToOne(Player player) { - if (teamPacketInterface != null) teamPacketInterface.sendUpdateToOne(player); - } - - public void sendUpdateToAll(Player player, boolean force) { - if (teamPacketInterface != null) teamPacketInterface.sendUpdateToAll(player, force); - } - - public void createTeam(UUID uuid) { - Player player = Bukkit.getPlayer(uuid); - if (player == null || !player.isOnline() || !checkTriedTimes(player.getUniqueId())) return; - String teamName = getCurrentTeamName(player); - if (teamName != null) { - if (!teams.containsKey(uuid)) teams.put(uuid, new NameplatesTeam(plugin.getNameplateManager(), player, teamName)); - teamPacketInterface.sendUpdateToAll(player, true); - teamPacketInterface.sendUpdateToOne(player); - } - // wait for bc - else { - plugin.getScheduler().runTaskAsyncLater(() -> createTeam(uuid),10); - } - } - - @Nullable - public NameplatesTeam getNameplateTeam(UUID uuid) { - return teams.get(uuid); - } - - public String getCurrentTeamName(Player player) { - return teamNameInterface.getTeamName(player); - } - - public void removePlayerFromTeamCache(Player player) { - teams.remove(player.getUniqueId()); - } - - public boolean isFakeTeam() { - return plugin.getNameplateManager().isFakeTeam(); - } - - protected boolean checkTriedTimes(UUID uuid) { - Integer previous = triedTimes.get(uuid); - if (previous == null) { - triedTimes.put(uuid, 1); - return true; - } - else if (previous > 4) { - triedTimes.remove(uuid); - return false; - } - else { - triedTimes.put(uuid, previous + 1); - return true; - } - } - - public TeamNameInterface getTeamNameInterface() { - return teamNameInterface; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/ConditionalText.java b/src/main/java/net/momirealms/customnameplates/object/ConditionalText.java deleted file mode 100644 index b2c3d2d..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/ConditionalText.java +++ /dev/null @@ -1,25 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object; - -import net.momirealms.customnameplates.object.carrier.TextDisplayMeta; -import net.momirealms.customnameplates.object.requirements.Requirement; - -public record ConditionalText(Requirement[] requirements, String text, TextDisplayMeta textDisplayMeta) { - -} diff --git a/src/main/java/net/momirealms/customnameplates/object/DisplayMode.java b/src/main/java/net/momirealms/customnameplates/object/DisplayMode.java deleted file mode 100644 index 954d01b..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/DisplayMode.java +++ /dev/null @@ -1,25 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object; - -public enum DisplayMode { - TEAM, - ARMOR_STAND, - TEXT_DISPLAY, - DISABLE -} diff --git a/src/main/java/net/momirealms/customnameplates/object/DynamicText.java b/src/main/java/net/momirealms/customnameplates/object/DynamicText.java deleted file mode 100644 index 02fd4c5..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/DynamicText.java +++ /dev/null @@ -1,85 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object; - -import me.clip.placeholderapi.PlaceholderAPI; -import net.kyori.adventure.text.Component; -import net.kyori.adventure.text.serializer.gson.GsonComponentSerializer; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.entity.Player; - -import java.util.ArrayList; -import java.util.List; - -public class DynamicText { - - private final Player owner; - private String originalValue; - private String latestValue; - private String[] ownerPlaceholders; - - public DynamicText(Player owner, String rawValue) { - this.owner = owner; - analyze(rawValue); - } - - private void analyze(String value) { - List placeholdersOwner = new ArrayList<>(CustomNameplates.getInstance().getPlaceholderManager().detectPlaceholders(value)); - String origin = value; - for (String placeholder : placeholdersOwner) { - origin = origin.replace(placeholder, "%s"); - } - originalValue = origin; - ownerPlaceholders = placeholdersOwner.toArray(new String[0]); - latestValue = originalValue; - update(); - } - - public String getLatestValue() { - return latestValue; - } - - public String getLatestJson() { - return GsonComponentSerializer.gson().serialize(getLatestComponent()); - } - - public Component getLatestComponent() { - return AdventureUtils.getComponentFromMiniMessage(latestValue); - } - - public boolean update() { - if (ownerPlaceholders.length == 0) return false; - String string; - if ("%s".equals(originalValue)) { - string = PlaceholderAPI.setPlaceholders(owner, ownerPlaceholders[0]); - } - else { - Object[] values = new String[ownerPlaceholders.length]; - for (int i = 0; i < ownerPlaceholders.length; i++) { - values[i] = PlaceholderAPI.setPlaceholders(owner, ownerPlaceholders[i]); - } - string = String.format(originalValue, values); - } - if (!latestValue.equals(string)) { - latestValue = string; - return true; - } - return false; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/SimpleChar.java b/src/main/java/net/momirealms/customnameplates/object/SimpleChar.java deleted file mode 100644 index c2a9729..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/SimpleChar.java +++ /dev/null @@ -1,55 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object; - -public class SimpleChar { - - private final int height; - private final int ascent; - private final char chars; - private final String file; - private final int width; - - public SimpleChar(int height, int ascent, int width, char chars, String path) { - this.height = height; - this.ascent = ascent; - this.chars = chars; - this.file = path; - this.width = width; - } - - public int getHeight() { - return height; - } - - public int getAscent() { - return ascent; - } - - public char getChars() { - return chars; - } - - public String getFile() { - return file; - } - - public int getWidth() { - return width; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/actionbar/ActionBarConfig.java b/src/main/java/net/momirealms/customnameplates/object/actionbar/ActionBarConfig.java deleted file mode 100644 index 01dac96..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/actionbar/ActionBarConfig.java +++ /dev/null @@ -1,24 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.actionbar; - -import net.momirealms.customnameplates.object.requirements.Requirement; - -public record ActionBarConfig(int interval, String[] texts, Requirement[] conditions) { - -} diff --git a/src/main/java/net/momirealms/customnameplates/object/actionbar/ActionBarSender.java b/src/main/java/net/momirealms/customnameplates/object/actionbar/ActionBarSender.java deleted file mode 100644 index fc96c0c..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/actionbar/ActionBarSender.java +++ /dev/null @@ -1,70 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.actionbar; - -import net.kyori.adventure.text.Component; -import net.kyori.adventure.text.ScoreComponent; -import net.momirealms.customnameplates.object.DynamicText; -import net.momirealms.customnameplates.object.requirements.Requirement; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.entity.Player; - -public class ActionBarSender { - - private final Player player; - private final int switch_interval; - private int timer; - private int current_text_id; - private final DynamicText[] dynamicTexts; - private final Requirement[] requirements; - - public ActionBarSender(int switch_interval, String[] texts, Requirement[] requirements, Player player) { - this.player = player; - this.switch_interval = switch_interval; - this.requirements = requirements; - this.dynamicTexts = new DynamicText[texts.length]; - for (int i = 0; i < texts.length; i++) { - dynamicTexts[i] = new DynamicText(player, texts[i]); - } - this.current_text_id = 0; - } - - public boolean canSend() { - for (Requirement requirement : requirements) { - if (!requirement.isConditionMet(player)) { - return false; - } - } - return true; - } - - public void send() { - timer++; - if (timer >= switch_interval) { - timer = 0; - current_text_id++; - if (current_text_id >= dynamicTexts.length) { - current_text_id = 0; - } - } - dynamicTexts[current_text_id].update(); - ScoreComponent.Builder builder = Component.score().name("nameplates").objective("actionbar"); - builder.append(AdventureUtils.getComponentFromMiniMessage(dynamicTexts[current_text_id].getLatestValue())); - AdventureUtils.playerActionbar(player, builder.build()); - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/actionbar/ActionBarTask.java b/src/main/java/net/momirealms/customnameplates/object/actionbar/ActionBarTask.java deleted file mode 100644 index 9f77b8c..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/actionbar/ActionBarTask.java +++ /dev/null @@ -1,75 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.actionbar; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.scheduler.TimerTask; -import org.bukkit.entity.Player; - -public class ActionBarTask { - - private TimerTask timerTask; - private final ActionBarSender[] actionBarSenders; - private String otherText; - private long expireTime; - - public ActionBarTask(Player player, ActionBarConfig[] configs) { - this.actionBarSenders = new ActionBarSender[configs.length]; - this.otherText = ""; - for (int i = 0; i < configs.length; i++) { - ActionBarConfig actionBarConfig = configs[i]; - this.actionBarSenders[i] = new ActionBarSender( - actionBarConfig.interval(), - actionBarConfig.texts(), - actionBarConfig.conditions(), - player - ); - } - } - - public void stop() { - if (this.timerTask != null && !this.timerTask.isCancelled()) this.timerTask.cancel(); - } - - public void start() { - this.timerTask = CustomNameplates.getInstance().getScheduler().runTaskAsyncTimer(() -> { - if (System.currentTimeMillis() > getExpireTime()) { - this.otherText = ""; - } - for (ActionBarSender actionBarSender : actionBarSenders) { - if (actionBarSender.canSend()) { - actionBarSender.send(); - return; - } - } - }, 1, 1); - } - - public String getOtherText() { - return otherText; - } - - public void setOtherText(String otherText, long time) { - this.otherText = otherText; - this.expireTime = time + 4000; - } - - public long getExpireTime() { - return expireTime; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/background/BackGroundConfig.java b/src/main/java/net/momirealms/customnameplates/object/background/BackGroundConfig.java deleted file mode 100644 index ba8e568..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/background/BackGroundConfig.java +++ /dev/null @@ -1,28 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.background; - -import net.momirealms.customnameplates.object.SimpleChar; - -public record BackGroundConfig(SimpleChar left, SimpleChar offset_1, - SimpleChar offset_2, SimpleChar offset_4, - SimpleChar offset_8, SimpleChar offset_16, - SimpleChar offset_32, SimpleChar offset_64, - SimpleChar offset_128, SimpleChar right, - int left_margin, int right_margin) { -} diff --git a/src/main/java/net/momirealms/customnameplates/object/bossbar/BossBarConfig.java b/src/main/java/net/momirealms/customnameplates/object/bossbar/BossBarConfig.java deleted file mode 100644 index 13a36ee..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/bossbar/BossBarConfig.java +++ /dev/null @@ -1,25 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.bossbar; - -import net.momirealms.customnameplates.object.requirements.Requirement; -import org.bukkit.boss.BarColor; - -public record BossBarConfig(String[] texts, Overlay overlay, BarColor color, int interval, int refreshRate, Requirement[] conditions) { - -} diff --git a/src/main/java/net/momirealms/customnameplates/object/bossbar/BossBarSender.java b/src/main/java/net/momirealms/customnameplates/object/bossbar/BossBarSender.java deleted file mode 100644 index 185d418..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/bossbar/BossBarSender.java +++ /dev/null @@ -1,157 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.bossbar; - -import com.comphenix.protocol.PacketType; -import com.comphenix.protocol.events.InternalStructure; -import com.comphenix.protocol.events.PacketContainer; -import com.comphenix.protocol.utility.MinecraftReflection; -import com.comphenix.protocol.wrappers.WrappedChatComponent; -import net.kyori.adventure.text.minimessage.MiniMessage; -import net.kyori.adventure.text.serializer.gson.GsonComponentSerializer; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.DynamicText; -import net.momirealms.customnameplates.object.requirements.Requirement; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.boss.BarColor; -import org.bukkit.entity.Player; - -import java.lang.reflect.Constructor; -import java.lang.reflect.Field; -import java.lang.reflect.InvocationTargetException; -import java.lang.reflect.Method; -import java.util.UUID; - -public class BossBarSender { - - private final Player player; - private final int switch_interval; - private int timer; - private int refresh_rate; - private int current_text_id; - private final DynamicText[] dynamicTexts; - private final Requirement[] requirements; - private final UUID uuid; - private boolean force; - private boolean isShown; - private final Overlay overlay; - private final BarColor barColor; - - public BossBarSender(int switch_interval, int refreshRate, String[] texts, Requirement[] requirements, Overlay overlay, BarColor barColor,Player player) { - this.player = player; - this.switch_interval = switch_interval; - this.overlay = overlay; - this.barColor = barColor; - this.uuid = UUID.randomUUID(); - this.requirements = requirements; - this.dynamicTexts = new DynamicText[texts.length]; - for (int i = 0; i < texts.length; i++) { - dynamicTexts[i] = new DynamicText(player, texts[i]); - } - this.current_text_id = 0; - this.refresh_rate = refreshRate; - } - - public boolean isShown() { - return isShown; - } - - public boolean canSend() { - if (requirements == null) return true; - for (Requirement requirement : requirements) { - if (!requirement.isConditionMet(player)) { - return false; - } - } - return true; - } - - public void hide() { - this.sendRemovePacket(); - this.isShown = false; - } - - public void show() { - CustomNameplates.getProtocolManager().sendServerPacket(player, getCreatePacket()); - this.isShown = true; - } - - public void update() { - timer++; - if (timer % switch_interval == 0) { - current_text_id++; - if (current_text_id >= dynamicTexts.length) { - current_text_id = 0; - } - force = true; - } - if (timer % refresh_rate == 0 && dynamicTexts[current_text_id].update() || force) { - force = false; - CustomNameplates.getProtocolManager().sendServerPacket(player, getUpdatePacket()); - } - } - - private PacketContainer getCreatePacket() { - dynamicTexts[current_text_id].update(); - PacketContainer packet = new PacketContainer(PacketType.Play.Server.BOSS); - packet.getModifier().write(0, uuid); - InternalStructure internalStructure = packet.getStructures().read(1); - internalStructure.getChatComponents().write(0, WrappedChatComponent.fromJson(GsonComponentSerializer.gson().serialize(MiniMessage.miniMessage().deserialize(AdventureUtils.replaceLegacy(dynamicTexts[current_text_id].getLatestValue()))))); - internalStructure.getFloat().write(0,1F); - internalStructure.getEnumModifier(BarColor.class, 2).write(0, barColor); - internalStructure.getEnumModifier(Overlay.class, 3).write(0, overlay); - internalStructure.getModifier().write(4, false); - internalStructure.getModifier().write(5, false); - internalStructure.getModifier().write(6, false); - return packet; - } - - private PacketContainer getUpdatePacket() { - PacketContainer packet = new PacketContainer(PacketType.Play.Server.BOSS); - packet.getModifier().write(0, uuid); - try { - Method sMethod = MinecraftReflection.getChatSerializerClass().getMethod("a", String.class); - sMethod.setAccessible(true); - Object chatComponent = sMethod.invoke(null, GsonComponentSerializer.gson().serialize(MiniMessage.miniMessage().deserialize(AdventureUtils.replaceLegacy(dynamicTexts[current_text_id].getLatestValue())))); - Class packetBossClass = Class.forName("net.minecraft.network.protocol.game.PacketPlayOutBoss$e"); - Constructor packetConstructor = packetBossClass.getDeclaredConstructor(MinecraftReflection.getIChatBaseComponentClass()); - packetConstructor.setAccessible(true); - Object updatePacket = packetConstructor.newInstance(chatComponent); - packet.getModifier().write(1, updatePacket); - } catch (NoSuchMethodException | InvocationTargetException | IllegalAccessException | ClassNotFoundException | InstantiationException e) { - throw new RuntimeException(e); - } - return packet; - } - - private void sendRemovePacket() { - PacketContainer packet = new PacketContainer(PacketType.Play.Server.BOSS); - packet.getModifier().write(0, uuid); - try { - Class bar = Class.forName("net.minecraft.network.protocol.game.PacketPlayOutBoss"); - Field remove = bar.getDeclaredField("f"); - remove.setAccessible(true); - packet.getModifier().write(1, remove.get(null)); - CustomNameplates.getProtocolManager().sendServerPacket(player, packet); - } catch (ClassNotFoundException e){ - AdventureUtils.consoleMessage("[CustomNameplates] Failed to remove boss bar for " + player.getName()); - } catch (NoSuchFieldException | IllegalAccessException e) { - throw new RuntimeException(e); - } - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/bossbar/BossBarTask.java b/src/main/java/net/momirealms/customnameplates/object/bossbar/BossBarTask.java deleted file mode 100644 index ea31c3d..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/bossbar/BossBarTask.java +++ /dev/null @@ -1,72 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.bossbar; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.scheduler.TimerTask; -import org.bukkit.entity.Player; - -public class BossBarTask { - - private TimerTask timerTask; - private final BossBarSender[] bossBarSenders; - - public BossBarTask(Player player, BossBarConfig[] configs) { - this.bossBarSenders = new BossBarSender[configs.length]; - for (int i = 0; i < configs.length; i++) { - BossBarConfig bossBarConfig = configs[i]; - this.bossBarSenders[i] = new BossBarSender( - bossBarConfig.interval(), - bossBarConfig.refreshRate(), - bossBarConfig.texts(), - bossBarConfig.conditions(), - bossBarConfig.overlay(), - bossBarConfig.color(), - player - ); - } - } - - public void stop() { - if (this.timerTask != null && !this.timerTask.isCancelled()) this.timerTask.cancel(); - for (BossBarSender bossBarSender : bossBarSenders) { - bossBarSender.hide(); - } - } - - public void start() { - this.timerTask = CustomNameplates.getInstance().getScheduler().runTaskAsyncTimer(() -> { - for (BossBarSender bossBarSender : bossBarSenders) { - if (!bossBarSender.canSend()) { - if (bossBarSender.isShown()) { - bossBarSender.hide(); - return; - } - } - else { - if (bossBarSender.isShown()) { - bossBarSender.update(); - } - else { - bossBarSender.show(); - } - } - } - }, 1, 1); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/bubble/BubbleConfig.java b/src/main/java/net/momirealms/customnameplates/object/bubble/BubbleConfig.java deleted file mode 100644 index cd1d23d..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/bubble/BubbleConfig.java +++ /dev/null @@ -1,29 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.bubble; - -import net.momirealms.customnameplates.object.SimpleChar; - -public record BubbleConfig(String startFormat, - String endFormat, - String display_name, - SimpleChar left, - SimpleChar middle, - SimpleChar right, - SimpleChar tail) { -} diff --git a/src/main/java/net/momirealms/customnameplates/object/carrier/AbstractPacketsHandler.java b/src/main/java/net/momirealms/customnameplates/object/carrier/AbstractPacketsHandler.java deleted file mode 100644 index df54796..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/carrier/AbstractPacketsHandler.java +++ /dev/null @@ -1,72 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.carrier; - -import com.comphenix.protocol.events.PacketEvent; -import net.momirealms.customnameplates.object.Function; -import org.bukkit.Bukkit; -import org.bukkit.entity.Player; - -import java.util.List; -import java.util.concurrent.ConcurrentHashMap; - -public abstract class AbstractPacketsHandler extends Function { - - protected ConcurrentHashMap entityIdMap = new ConcurrentHashMap<>(); - - @Override - public void load() { - for (Player player : Bukkit.getOnlinePlayers()) { - entityIdMap.put(player.getEntityId(), player); - } - } - - @Override - public void unload() { - entityIdMap.clear(); - } - - @Override - public void onJoin(Player player) { - entityIdMap.put(player.getEntityId(), player); - } - - @Override - public void onQuit(Player player) { - entityIdMap.remove(player.getEntityId()); - } - - public void onEntityMove(Player receiver, int entityId, short x, short y, short z, boolean onGround) { - } - - public void onEntitySpawn(Player receiver, int entityId, PacketEvent event) { - } - - public void onEntityDestroy(Player receiver, List entities) { - } - - public void onEntityDestroy(Player receiver, int entity) { - } - - public void onEntityTeleport(Player player, int entityId) { - } - - public Player getPlayerFromMap(int entityID) { - return entityIdMap.get(entityID); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/carrier/AbstractTextCarrier.java b/src/main/java/net/momirealms/customnameplates/object/carrier/AbstractTextCarrier.java deleted file mode 100644 index 5db2d20..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/carrier/AbstractTextCarrier.java +++ /dev/null @@ -1,71 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.carrier; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.listener.JoinQuitListener; -import net.momirealms.customnameplates.manager.TeamManager; -import net.momirealms.customnameplates.object.DisplayMode; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.object.scheduler.TimerTask; -import org.bukkit.Bukkit; -import org.bukkit.entity.Player; -import org.bukkit.event.HandlerList; - -public abstract class AbstractTextCarrier extends Function { - - protected TimerTask refreshTask; - protected CustomNameplates plugin; - protected JoinQuitListener joinQuitListener; - protected DisplayMode displayMode; - - public AbstractTextCarrier(CustomNameplates plugin, DisplayMode displayMode) { - this.displayMode = displayMode; - this.plugin = plugin; - this.joinQuitListener = new JoinQuitListener(this); - } - - @Override - public void load() { - loadToAllPlayers(); - arrangeRefreshTask(); - Bukkit.getPluginManager().registerEvents(joinQuitListener, plugin); - } - - @Override - public void unload() { - if (refreshTask != null) refreshTask.cancel(); - HandlerList.unregisterAll(joinQuitListener); - } - - public void arrangeRefreshTask() { - //child - } - - public void loadToAllPlayers() { - TeamManager teamManager = plugin.getTeamManager(); - for (Player player : Bukkit.getOnlinePlayers()) { - teamManager.sendUpdateToAll(player, true); - teamManager.sendUpdateToOne(player); - } - } - - public DisplayMode getDisplayMode() { - return displayMode; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/carrier/DisableNameplate.java b/src/main/java/net/momirealms/customnameplates/object/carrier/DisableNameplate.java deleted file mode 100644 index 95b2fdf..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/carrier/DisableNameplate.java +++ /dev/null @@ -1,26 +0,0 @@ -package net.momirealms.customnameplates.object.carrier; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.DisplayMode; - -public class DisableNameplate extends AbstractTextCarrier { - - public DisableNameplate(CustomNameplates plugin) { - super(plugin, DisplayMode.DISABLE); - } - - @Override - public void load() { - - } - - @Override - public void unload() { - - } - - @Override - public void loadToAllPlayers() { - - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntity.java b/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntity.java deleted file mode 100644 index bb5bc68..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntity.java +++ /dev/null @@ -1,39 +0,0 @@ -package net.momirealms.customnameplates.object.carrier; - -import net.momirealms.customnameplates.object.DynamicText; -import org.bukkit.entity.Player; - -public interface NamedEntity { - - void refresh(); - - boolean canShow(); - - boolean isShown(); - - void destroy(); - - void destroy(Player viewer); - - void spawn(); - - void spawn(Player viewer); - - void teleport(); - - void teleport(Player viewer); - - void setSneak(boolean sneaking, boolean respawn); - - void respawn(Player viewer); - - double getOffset(); - - void setOffset(double v); - - DynamicText getDynamicText(); - - int getEntityId(); - - void move(Player viewer, short x, short y, short z, boolean onGround); -} diff --git a/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityCarrier.java b/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityCarrier.java deleted file mode 100644 index 324cd80..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityCarrier.java +++ /dev/null @@ -1,162 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.carrier; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.listener.EntityTagListener; -import net.momirealms.customnameplates.listener.compatibility.MagicCosmeticsListener; -import net.momirealms.customnameplates.object.ConditionalText; -import net.momirealms.customnameplates.object.DisplayMode; -import org.bukkit.Bukkit; -import org.bukkit.GameMode; -import org.bukkit.Location; -import org.bukkit.entity.Player; -import org.bukkit.event.HandlerList; -import org.jetbrains.annotations.NotNull; - -import java.util.HashMap; -import java.util.concurrent.ConcurrentHashMap; - -public class NamedEntityCarrier extends AbstractTextCarrier { - - protected final ConcurrentHashMap namedEntityManagerMap = new ConcurrentHashMap<>(); - protected EntityTagListener entityTagListener; - protected AbstractPacketsHandler handler; - protected MagicCosmeticsListener magicCosmeticsListener; - private final VehicleChecker vehicleChecker; - private final HashMap persistentText; - - public NamedEntityCarrier(CustomNameplates plugin, DisplayMode displayMode, @NotNull HashMap persistentText) { - super(plugin, displayMode); - this.vehicleChecker = new VehicleChecker(this); - this.entityTagListener = new EntityTagListener(this); - this.persistentText = persistentText; - this.handler = new NamedEntityPacketsHandler(this); - if (Bukkit.getPluginManager().getPlugin("MagicCosmetics") != null) { - this.magicCosmeticsListener = new MagicCosmeticsListener(this); - } - } - - @Override - public void load(){ - super.load(); - handler.load(); - this.vehicleChecker.load(); - Bukkit.getPluginManager().registerEvents(entityTagListener, CustomNameplates.getInstance()); - if (magicCosmeticsListener != null) Bukkit.getPluginManager().registerEvents(magicCosmeticsListener, CustomNameplates.getInstance()); - } - - @Override - public void unload(){ - super.unload(); - handler.unload(); - for (NamedEntityManager nem : namedEntityManagerMap.values()) { - nem.destroy(); - } - namedEntityManagerMap.clear(); - HandlerList.unregisterAll(entityTagListener); - this.vehicleChecker.unload(); - if (magicCosmeticsListener != null) HandlerList.unregisterAll(magicCosmeticsListener); - } - - @Override - public void onJoin(Player player) { - handler.onJoin(player); - createNamedEntityManager(player); - for (Player viewer : Bukkit.getOnlinePlayers()) { - spawnNamedEntity(viewer, player); - spawnNamedEntity(player, viewer); - } - } - - @Override - public void onQuit(Player player) { - handler.onQuit(player); - NamedEntityManager nem = namedEntityManagerMap.remove(player); - if (nem != null) { - nem.destroy(); - } - for (NamedEntityManager all : namedEntityManagerMap.values()) { - all.unregisterPlayer(player); - } - } - - @Override - public void loadToAllPlayers() { - super.loadToAllPlayers(); - for (Player all : Bukkit.getOnlinePlayers()) { - createNamedEntityManager(all); - for (Player player : Bukkit.getOnlinePlayers()) - spawnNamedEntity(player, all); - } - } - - public NamedEntityManager createNamedEntityManager(Player player) { - NamedEntityManager nem = new NamedEntityManager(this, player); - namedEntityManagerMap.put(player, nem); - return nem; - } - - public void onSneak(Player player, boolean isSneaking) { - if (player.isFlying()) - return; - NamedEntityManager nem = getNamedEntityManager(player); - if (nem != null) { - nem.setSneak(isSneaking, true); - } - } - - public void onRespawn(Player player) { - getNamedEntityManager(player).respawn(); - } - - public void onDeath(Player player) { - getNamedEntityManager(player).destroy(); - } - - @Override - public void arrangeRefreshTask() { - refreshTask = plugin.getScheduler().runTaskAsyncTimer(() -> { - for (NamedEntityManager nem : namedEntityManagerMap.values()) { - nem.refresh(false); - } - }, 1, 1); - } - - public NamedEntityManager getNamedEntityManager(Player player) { - return namedEntityManagerMap.get(player); - } - - protected void spawnNamedEntity(Player viewer, Player target) { - if (target == viewer || target.getGameMode() == GameMode.SPECTATOR || viewer.getWorld() != target.getWorld() || target.isDead()) return; - if (getDistance(target, viewer) < 48 && viewer.canSee(target)) { - NamedEntityManager nem = getNamedEntityManager(target); - nem.spawn(viewer); - } - } - - protected double getDistance(Player player1, Player player2) { - Location loc1 = player1.getLocation(); - Location loc2 = player2.getLocation(); - return Math.sqrt(Math.pow(loc1.getX()-loc2.getX(), 2) + Math.pow(loc1.getZ()-loc2.getZ(), 2)); - } - - public HashMap getPersistentText() { - return persistentText; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityImpl.java b/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityImpl.java deleted file mode 100644 index 2c5ca73..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityImpl.java +++ /dev/null @@ -1,330 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.carrier; - -import com.comphenix.protocol.PacketType; -import com.comphenix.protocol.events.PacketContainer; -import com.comphenix.protocol.wrappers.WrappedChatComponent; -import com.comphenix.protocol.wrappers.WrappedDataValue; -import com.comphenix.protocol.wrappers.WrappedDataWatcher; -import com.google.common.collect.Lists; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.DisplayMode; -import net.momirealms.customnameplates.object.DynamicText; -import net.momirealms.customnameplates.object.requirements.Requirement; -import org.bukkit.Location; -import org.bukkit.entity.Entity; -import org.bukkit.entity.EntityType; -import org.bukkit.entity.Player; -import org.bukkit.entity.Pose; -import org.bukkit.potion.PotionEffectType; -import org.jetbrains.annotations.NotNull; -import org.jetbrains.annotations.Nullable; - -import java.util.*; - -public class NamedEntityImpl implements NamedEntity { - - private final NamedEntityManager nem; - private final Player owner; - private double yOffset; - private final int entityId; - private final UUID uuid = UUID.randomUUID(); - private boolean sneaking; - private DynamicText dynamicText; - private String textJson; - private final Requirement[] requirements; - private boolean isShown; - private final TextDisplayMeta textDisplayMeta; - - //dynamic value - public NamedEntityImpl(NamedEntityManager nem, Player owner, DynamicText text, double yOffset, @NotNull Requirement[] requirements, @Nullable TextDisplayMeta textDisplayMeta) { - this.nem = nem; - this.entityId = new Random().nextInt(Integer.MAX_VALUE); - this.owner = owner; - this.yOffset = yOffset; - this.dynamicText = text; - this.sneaking = owner.isSneaking(); - this.requirements = requirements; - this.isShown = false; - this.textJson = dynamicText.getLatestJson(); - this.textDisplayMeta = textDisplayMeta; - } - - //constant value - public NamedEntityImpl(NamedEntityManager nem, Player owner, String textJson, double yOffset, @Nullable TextDisplayMeta textDisplayMeta) { - this.nem = nem; - this.entityId = new Random().nextInt(Integer.MAX_VALUE); - this.owner = owner; - this.yOffset = yOffset; - this.textJson = textJson; - this.sneaking = owner.isSneaking(); - this.requirements = null; - this.isShown = true; - this.textDisplayMeta = textDisplayMeta; - } - - @Override - public boolean canShow() { - if (owner.hasPotionEffect(PotionEffectType.INVISIBILITY)) - return false; - if (requirements == null) - return true; - for (Requirement requirement : requirements) { - if (!requirement.isConditionMet(owner)) { - return false; - } - } - return true; - } - - @Override - public void refresh() { - if (dynamicText == null) return; - textJson = dynamicText.getLatestJson(); - updateMetadata(); - } - - @Override - public double getOffset() { - return yOffset; - } - - @Override - public void setOffset(double offset) { - if (yOffset == offset) return; - yOffset = offset; - for (Player all : nem.getNearbyPlayers()) { - CustomNameplates.getProtocolManager().sendServerPacket(all, getTeleportPacket()); - } - } - - @Override - public void spawn(Player viewer) { - for (PacketContainer packet : getSpawnPackets()) { - CustomNameplates.getProtocolManager().sendServerPacket(viewer, packet); - } - } - - @Override - public void spawn() { - for (Player all : nem.getNearbyPlayers()) { - spawn(all); - } - isShown = true; - } - - @Override - public void destroy() { - PacketContainer destroyPacket = new PacketContainer(PacketType.Play.Server.ENTITY_DESTROY); - destroyPacket.getIntLists().write(0, List.of(entityId)); - for (Player all : nem.getNearbyPlayers()) { - CustomNameplates.getProtocolManager().sendServerPacket(all, destroyPacket); - } - isShown = false; - } - - @Override - public void destroy(Player viewer) { - PacketContainer destroyPacket = new PacketContainer(PacketType.Play.Server.ENTITY_DESTROY); - destroyPacket.getIntLists().write(0, List.of(entityId)); - CustomNameplates.getProtocolManager().sendServerPacket(viewer, destroyPacket); - } - - @Override - public void teleport() { - PacketContainer packet = getTeleportPacket(); - for (Player all : nem.getNearbyPlayers()) { - CustomNameplates.getProtocolManager().sendServerPacket(all, packet); - } - } - - @Override - public void teleport(Player viewer) { - if (!nem.isNearby(viewer) && viewer != owner) { - nem.spawn(viewer); - } else { - CustomNameplates.getProtocolManager().sendServerPacket(viewer, getTeleportPacket()); - } - } - - @Override - public void setSneak(boolean isSneaking, boolean respawn) { - this.sneaking = isSneaking; - if (respawn) { - for (Player viewer : nem.getNearbyPlayers()) { - respawn(viewer); - } - } else { - refresh(); - } - } - - @Override - public boolean isShown() { - return isShown; - } - - @Override - public DynamicText getDynamicText() { - return dynamicText; - } - - @Override - public int getEntityId() { - return entityId; - } - - @Override - public void move(Player viewer, short x, short y, short z, boolean onGround) { - PacketContainer packet = getMovePacket(x, y, z, onGround); - CustomNameplates.getProtocolManager().sendServerPacket(viewer, packet); - } - - @Override - public void respawn(Player viewer) { - destroy(viewer); - spawn(viewer); - } - - public PacketContainer getMovePacket(short x, short y, short z, boolean onGround) { - PacketContainer packet = new PacketContainer(PacketType.Play.Server.REL_ENTITY_MOVE); - packet.getIntegers().write(0, entityId); - packet.getShorts().write(0, x); - packet.getShorts().write(1, y); - packet.getShorts().write(2, z); - packet.getBooleans().write(0, onGround); - return packet; - } - - public PacketContainer getTeleportPacket() { - PacketContainer packet = new PacketContainer(PacketType.Play.Server.ENTITY_TELEPORT); - packet.getIntegers().write(0, entityId); - Location location = getEntityLocation(); - packet.getDoubles().write(0, location.getX()); - packet.getDoubles().write(1, location.getY()); - packet.getDoubles().write(2, location.getZ()); - return packet; - } - - public void updateMetadata() { - PacketContainer metaPacket = getMetaPacket(); - for (Player viewer : nem.getNearbyPlayers()) { - CustomNameplates.getProtocolManager().sendServerPacket(viewer, metaPacket); - } - } - - protected PacketContainer getMetaPacket() { - PacketContainer metaPacket = new PacketContainer(PacketType.Play.Server.ENTITY_METADATA); - metaPacket.getIntegers().write(0, entityId); - if (CustomNameplates.getInstance().getVersionHelper().isVersionNewerThan1_19_R2()) { - WrappedDataWatcher wrappedDataWatcher = - nem.getDisplayMode() == DisplayMode.ARMOR_STAND ? - createArmorStandDataWatcher(textJson) : createTextDisplayDataWatcher(textJson, textDisplayMeta); - List wrappedDataValueList = Lists.newArrayList(); - wrappedDataWatcher.getWatchableObjects().stream().filter(Objects::nonNull).forEach(entry -> wrappedDataValueList.add(new WrappedDataValue(entry.getWatcherObject().getIndex(), entry.getWatcherObject().getSerializer(), entry.getRawValue()))); - metaPacket.getDataValueCollectionModifier().write(0, wrappedDataValueList); - } else { - metaPacket.getWatchableCollectionModifier().write(0, createArmorStandDataWatcher(textJson).getWatchableObjects()); - } - return metaPacket; - } - - private Location getEntityLocation() { - double x = owner.getLocation().getX(); - double y = getY() + yOffset; - double z = owner.getLocation().getZ(); - if (!owner.isSleeping()) { - if (sneaking) y += 1.5; - else y += 1.8; - } - else y += 0.2; - y += nem.getHatOffset(); - return new Location(null, x, y, z); - } - - private double getY() { - Entity vehicle = owner.getVehicle(); - if (vehicle != null) { - switch (vehicle.getType()) { - case BOAT, CHEST_BOAT -> { - return vehicle.getLocation().getY() - 0.325; - } - case HORSE -> { - return vehicle.getLocation().getY() + 0.85; - } - case DONKEY -> { - return vehicle.getLocation().getY() + 0.525; - } - case PIG -> { - return vehicle.getLocation().getY() + 0.325; - } - case STRIDER -> { - return vehicle.getLocation().getY() + 1.15; - } - } - } - if (owner.getPose() == Pose.SWIMMING || owner.isGliding()) { - return owner.getLocation().getY() - 1.25; - } - return owner.getLocation().getY(); - } - - private WrappedDataWatcher createArmorStandDataWatcher(String json) { - WrappedDataWatcher wrappedDataWatcher = new WrappedDataWatcher(); - WrappedDataWatcher.Serializer serializer1 = WrappedDataWatcher.Registry.get(Boolean.class); - WrappedDataWatcher.Serializer serializer2 = WrappedDataWatcher.Registry.get(Byte.class); - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(2, WrappedDataWatcher.Registry.getChatComponentSerializer(true)), Optional.of(WrappedChatComponent.fromJson(json).getHandle())); - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(3, serializer1), true); - byte flag = 0x20; // invisibility - if (sneaking) flag += (byte) 0x02; - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(0, serializer2), flag); - // small size - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(15, serializer2), (byte) 0x01); - // visible text - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(3, serializer1), true); - return wrappedDataWatcher; - } - - private WrappedDataWatcher createTextDisplayDataWatcher(String textJson, TextDisplayMeta textDisplayMeta) { - WrappedDataWatcher wrappedDataWatcher = new WrappedDataWatcher(); - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(22, WrappedDataWatcher.Registry.getChatComponentSerializer(false)), WrappedChatComponent.fromJson(textJson)); - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(24, WrappedDataWatcher.Registry.get(Integer.class)), textDisplayMeta.backgroundColor()); - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(14, WrappedDataWatcher.Registry.get(Byte.class)), (byte) 3); - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(25, WrappedDataWatcher.Registry.get(Byte.class)), textDisplayMeta.opacity()); - int mask = 0; - if (textDisplayMeta.hasShadow()) mask += 1; - if (textDisplayMeta.isSeeThrough()) mask += 2; - if (textDisplayMeta.useDefaultBackground()) mask += 4; - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(26, WrappedDataWatcher.Registry.get(Byte.class)), (byte) mask); - return wrappedDataWatcher; - } - - private PacketContainer[] getSpawnPackets() { - PacketContainer entityPacket = new PacketContainer(PacketType.Play.Server.SPAWN_ENTITY); - entityPacket.getModifier().write(0, entityId); - entityPacket.getModifier().write(1, uuid); - entityPacket.getEntityTypeModifier().write(0, nem.getDisplayMode() == DisplayMode.ARMOR_STAND ? EntityType.ARMOR_STAND : EntityType.TEXT_DISPLAY); - Location location = getEntityLocation(); - entityPacket.getDoubles().write(0, location.getX()); - entityPacket.getDoubles().write(1, location.getY()); - entityPacket.getDoubles().write(2, location.getZ()); - PacketContainer metaPacket = getMetaPacket(); - return new PacketContainer[] {entityPacket, metaPacket}; - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityManager.java b/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityManager.java deleted file mode 100644 index 0768c41..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityManager.java +++ /dev/null @@ -1,192 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.carrier; - -import net.momirealms.customnameplates.object.ConditionalText; -import net.momirealms.customnameplates.object.DisplayMode; -import net.momirealms.customnameplates.object.DynamicText; -import org.bukkit.entity.Player; - -import java.util.*; -import java.util.concurrent.ConcurrentHashMap; - -public class NamedEntityManager { - - private final ConcurrentHashMap namedEntities = new ConcurrentHashMap<>(); - private NamedEntity[] namedEntityArray; - private final Set nearbyPlayers; - private final Player owner; - private double hatOffset; - private final NamedEntityCarrier namedEntityCarrier; - private double highestTextHeight; - - public NamedEntityManager(NamedEntityCarrier namedEntityCarrier, Player owner) { - this.owner = owner; - this.nearbyPlayers = Collections.synchronizedSet(new HashSet<>()); - this.namedEntityArray = new NamedEntityImpl[0]; - this.namedEntityCarrier = namedEntityCarrier; - this.init(); - } - - public void init() { - Set> entries = namedEntityCarrier.getPersistentText().entrySet(); - for (Map.Entry entry : entries) { - addNamedEntity(UUID.randomUUID(), new NamedEntityImpl( - this, - owner, - new DynamicText(owner, entry.getKey().text()), - entry.getValue(), - entry.getKey().requirements(), - entry.getKey().textDisplayMeta() - )); - } - } - - public void addNamedEntity(UUID uuid, NamedEntity namedEntity) { - this.namedEntities.put(uuid, namedEntity); - this.namedEntityArray = namedEntities.values().toArray(new NamedEntity[0]); - for (Player p : nearbyPlayers) namedEntity.spawn(p); - } - - public boolean isNearby(Player viewer) { - return nearbyPlayers.contains(viewer); - } - - public void spawn(Player viewer) { - nearbyPlayers.add(viewer); - for (NamedEntity fakeArmorStand : namedEntityArray) - if (fakeArmorStand.canShow()) - fakeArmorStand.spawn(viewer); - } - - public void spawn() { - for (Player player : nearbyPlayers) { - spawn(player); - } - } - - public void refresh(boolean force) { - highestTextHeight = -2; - for (NamedEntity fakeArmorStand : namedEntityArray) { - boolean canShow = fakeArmorStand.canShow(); - if (!canShow) { - if (fakeArmorStand.isShown()) { - fakeArmorStand.destroy(); - } - } else { - if (fakeArmorStand.isShown()) { - if (fakeArmorStand.getDynamicText() != null && (fakeArmorStand.getDynamicText().update() || force)) - fakeArmorStand.refresh(); - } else { - fakeArmorStand.spawn(); - } - highestTextHeight = Math.max(fakeArmorStand.getOffset(), highestTextHeight); - } - } - } - - public void destroy() { - for (NamedEntity entity : namedEntityArray) { - entity.destroy(); - } - nearbyPlayers.clear(); - } - - public void teleport(Player viewer) { - for (NamedEntity entity : namedEntityArray) { - if (entity.isShown()) - entity.teleport(viewer); - } - } - - public void move(Player viewer, short x, short y, short z, boolean onGround) { - for (NamedEntity entity : namedEntityArray) { - if (entity.isShown()) - entity.move(viewer, x, y, z, onGround); - } - } - - public void teleport() { - for (NamedEntity entity : namedEntityArray) { - if (entity.isShown()) - entity.teleport(); - } - } - - public void setSneak(boolean sneaking, boolean respawn) { - for (NamedEntity entity : namedEntityArray) { - if (entity.isShown()) - entity.setSneak(sneaking, respawn); - } - } - - public void respawn() { - for (NamedEntity entity : namedEntityArray) { - if (entity.isShown()) - for (Player viewer : nearbyPlayers) - entity.respawn(viewer); - } - } - - public void destroy(Player viewer) { - for (NamedEntity entity : namedEntityArray) { - entity.destroy(viewer); - } - unregisterPlayer(viewer); - } - - public void unregisterPlayer(Player viewer) { - nearbyPlayers.remove(viewer); - } - - public void removeArmorStand(UUID uuid) { - NamedEntity entity = namedEntities.remove(uuid); - if (entity != null) { - entity.destroy(); - namedEntityArray = namedEntities.values().toArray(new NamedEntity[0]); - } - } - - public void ascent(double lineSpace) { - for (NamedEntity entity : namedEntityArray) { - entity.setOffset(entity.getOffset() + lineSpace); - entity.teleport(); - } - } - - public double getHatOffset() { - return hatOffset; - } - - public void setHatOffset(double hatOffset) { - this.hatOffset = hatOffset; - teleport(); - } - - public DisplayMode getDisplayMode() { - return namedEntityCarrier.getDisplayMode(); - } - - public double getHighestTextHeight() { - return highestTextHeight; - } - - public Collection getNearbyPlayers() { - return nearbyPlayers; - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityPacketsHandler.java b/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityPacketsHandler.java deleted file mode 100644 index 1b3923c..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/carrier/NamedEntityPacketsHandler.java +++ /dev/null @@ -1,107 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.carrier; - -import com.comphenix.protocol.events.PacketEvent; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.listener.packet.*; -import org.bukkit.entity.Player; - -import java.util.List; - -public class NamedEntityPacketsHandler extends AbstractPacketsHandler { - - private final NamedEntityCarrier namedEntityCarrier; - private final EntityDestroyListener entityDestroyListener; - private final EntityMoveListener entityMoveListener; - private final EntitySpawnListener entitySpawnListener; - private final EntityTeleportListener entityTeleportListener; - private final EntityLookListener entityLookListener; - - public NamedEntityPacketsHandler(NamedEntityCarrier namedEntityCarrier) { - super(); - this.namedEntityCarrier = namedEntityCarrier; - this.entityDestroyListener = new EntityDestroyListener(this); - this.entityMoveListener = new EntityMoveListener(this); - this.entitySpawnListener = new EntitySpawnListener(this); - this.entityTeleportListener = new EntityTeleportListener(this); - this.entityLookListener = new EntityLookListener(this); - } - - @Override - public void load() { - super.load(); - CustomNameplates.getProtocolManager().addPacketListener(entityDestroyListener); - CustomNameplates.getProtocolManager().addPacketListener(entityMoveListener); - CustomNameplates.getProtocolManager().addPacketListener(entitySpawnListener); - CustomNameplates.getProtocolManager().addPacketListener(entityTeleportListener); - CustomNameplates.getProtocolManager().addPacketListener(entityLookListener); - } - - @Override - public void unload() { - super.unload(); - CustomNameplates.getProtocolManager().removePacketListener(entityDestroyListener); - CustomNameplates.getProtocolManager().removePacketListener(entityMoveListener); - CustomNameplates.getProtocolManager().removePacketListener(entitySpawnListener); - CustomNameplates.getProtocolManager().removePacketListener(entityTeleportListener); - CustomNameplates.getProtocolManager().removePacketListener(entityLookListener); - } - - @Override - public void onEntityMove(Player receiver, int entityId, short x, short y, short z, boolean onGround) { - Player mover = super.getPlayerFromMap(entityId); - if (mover != null) { - namedEntityCarrier.getNamedEntityManager(mover).move(receiver, x, y, z, onGround); - } - } - - @Override - public void onEntityTeleport(Player receiver, int entityId) { - Player teleporter = super.getPlayerFromMap(entityId); - if (teleporter != null) { - namedEntityCarrier.getNamedEntityManager(teleporter).teleport(receiver); - } - } - - @Override - public void onEntitySpawn(Player receiver, int entityId, PacketEvent event) { - Player spawnedPlayer = super.getPlayerFromMap(entityId); - if (spawnedPlayer != null) { - NamedEntityManager nem = namedEntityCarrier.getNamedEntityManager(spawnedPlayer); - if (nem != null) { - nem.spawn(receiver); - } - } - } - - @Override - public void onEntityDestroy(Player receiver, List entities) { - for (int entity : entities) { - onEntityDestroy(receiver, entity); - } - } - - @Override - public void onEntityDestroy(Player receiver, int entity) { - Player deSpawnedPlayer = super.getPlayerFromMap(entity); - if (deSpawnedPlayer != null) { - namedEntityCarrier.getNamedEntityManager(deSpawnedPlayer).destroy(receiver); - } - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/carrier/TeamInfoCarrier.java b/src/main/java/net/momirealms/customnameplates/object/carrier/TeamInfoCarrier.java deleted file mode 100644 index 13e9240..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/carrier/TeamInfoCarrier.java +++ /dev/null @@ -1,43 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.carrier; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.TeamManager; -import net.momirealms.customnameplates.object.DisplayMode; -import org.bukkit.Bukkit; -import org.bukkit.entity.Player; - -public class TeamInfoCarrier extends AbstractTextCarrier { - - private final TeamManager teamManager; - - public TeamInfoCarrier(CustomNameplates plugin) { - super(plugin, DisplayMode.TEAM); - this.teamManager = plugin.getTeamManager(); - } - - @Override - public void arrangeRefreshTask() { - refreshTask = plugin.getScheduler().runTaskAsyncTimer(() -> { - for (Player player : Bukkit.getOnlinePlayers()) { - teamManager.sendUpdateToAll(player, false); - } - }, 1, 1); - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/carrier/TextDisplayMeta.java b/src/main/java/net/momirealms/customnameplates/object/carrier/TextDisplayMeta.java deleted file mode 100644 index e32b1dd..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/carrier/TextDisplayMeta.java +++ /dev/null @@ -1,7 +0,0 @@ -package net.momirealms.customnameplates.object.carrier; - -public record TextDisplayMeta(boolean hasShadow, boolean isSeeThrough, boolean useDefaultBackground, - int backgroundColor, byte opacity) { - - public static TextDisplayMeta defaultValue = new TextDisplayMeta(false, false, true, 0, (byte) -1); -} diff --git a/src/main/java/net/momirealms/customnameplates/object/carrier/VehicleChecker.java b/src/main/java/net/momirealms/customnameplates/object/carrier/VehicleChecker.java deleted file mode 100644 index 2c15546..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/carrier/VehicleChecker.java +++ /dev/null @@ -1,91 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.carrier; - -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.Function; -import net.momirealms.customnameplates.object.scheduler.TimerTask; -import org.bukkit.Bukkit; -import org.bukkit.entity.Entity; -import org.bukkit.entity.Player; - -import java.util.concurrent.ConcurrentHashMap; - -public class VehicleChecker extends Function { - - private final ConcurrentHashMap playersOnVehicle; - - private final NamedEntityCarrier namedEntityCarrier; - - private TimerTask updatePosTask; - private TimerTask vehicleCheckTask; - - public VehicleChecker(NamedEntityCarrier namedEntityCarrier) { - this.namedEntityCarrier = namedEntityCarrier; - this.playersOnVehicle = new ConcurrentHashMap<>(); - } - - @Override - public void load() { - for (Player all : Bukkit.getOnlinePlayers()) { - Entity vehicle = all.getVehicle(); - if (vehicle != null) { - playersOnVehicle.put(all, vehicle); - } - } - this.updatePosTask = CustomNameplates.getInstance().getScheduler().runTaskAsyncTimer(() -> { - for (Player inVehicle : playersOnVehicle.keySet()) { - if (!inVehicle.isOnline() || namedEntityCarrier.getNamedEntityManager(inVehicle) == null) continue; - namedEntityCarrier.getNamedEntityManager(inVehicle).teleport(); - } - }, 1, 1); - this.vehicleCheckTask = CustomNameplates.getInstance().getScheduler().runTaskAsyncTimer(() -> { - for (Player player : Bukkit.getOnlinePlayers()) { - this.refresh(player); - } - },4,4); - } - - @Override - public void unload() { - this.updatePosTask.cancel(); - this.vehicleCheckTask.cancel(); - playersOnVehicle.clear(); - } - - public void onJoin(Player player) { - Entity vehicle = player.getVehicle(); - if (vehicle != null) playersOnVehicle.put(player, vehicle); - } - - public void onQuit(Player player) { - playersOnVehicle.remove(player); - } - - public void refresh(Player player) { - Entity vehicle = player.getVehicle(); - if (playersOnVehicle.containsKey(player) && vehicle == null) { - namedEntityCarrier.getNamedEntityManager(player).teleport(); - playersOnVehicle.remove(player); - } - if (!playersOnVehicle.containsKey(player) && vehicle != null) { - namedEntityCarrier.getNamedEntityManager(player).respawn(); - playersOnVehicle.put(player, vehicle); - } - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/font/ASCIIWidth.java b/src/main/java/net/momirealms/customnameplates/object/font/ASCIIWidth.java deleted file mode 100644 index 4ac3cc8..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/font/ASCIIWidth.java +++ /dev/null @@ -1,64 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.font; - -public enum ASCIIWidth { - - A('A', 5), a('a', 5), B('B', 5), b('b', 5), - C('C', 5), c('c', 5), D('D', 5), d('d', 5), - E('E', 5), e('e', 5), F('F', 5), f('f', 4), - G('G', 5), g('g', 5), H('H', 5), h('h', 5), - I('I', 3), i('i', 1), J('J', 5), j('j', 5), - K('K', 5), k('k', 4), L('L', 5), l('l', 2), - M('M', 5), m('m', 5), N('N', 5), n('n', 5), - O('O', 5), o('o', 5), P('P', 5), p('p', 5), - Q('Q', 5), q('q', 5), R('R', 5), r('r', 5), - S('S', 5), s('s', 5), T('T', 5), t('t', 3), - U('U', 5), u('u', 5), V('V', 5), v('v', 5), - W('W', 5), w('w', 5), X('X', 5), x('x', 5), - Y('Y', 5), y('y', 5), Z('Z', 5), z('z', 5), - NUM_1('1', 5), NUM_2('2', 5), NUM_3('3', 5), NUM_4('4', 5), - NUM_5('5', 5), NUM_6('6', 5), NUM_7('7', 5), NUM_8('8', 5), - NUM_9('9', 5), NUM_0('0', 5), EXCLAMATION_POINT('!', 1), AT_SYMBOL('@', 6), - NUM_SIGN('#', 5), DOLLAR_SIGN('$', 5), PERCENT('%', 5), UP_ARROW('^', 5), - AMPERSAND('&', 5), ASTERISK('*', 3), LEFT_PARENTHESIS('(', 3), - RIGHT_PARENTHESIS(')', 3), MINUS('-', 5), UNDERSCORE('_', 5), PLUS_SIGN('+', 5), - EQUALS_SIGN('=', 5), LEFT_CURL_BRACE('{', 3), RIGHT_CURL_BRACE('}', 3), - LEFT_BRACKET('[', 3), RIGHT_BRACKET(']', 3), COLON(':', 1), SEMI_COLON(';', 1), - DOUBLE_QUOTE('\"', 3), SINGLE_QUOTE('\'', 1), LEFT_ARROW('<', 4), - RIGHT_ARROW('>', 4), QUESTION_MARK('?', 5), SLASH('/', 5), - BACK_SLASH('\\', 5), LINE('|', 1), TILDE('~', 5), TICK('`', 2), - PERIOD('.', 1), COMMA(',', 1), SPACE(' ', 3), - IN_BETWEEN(' ', 3), DEFAULT('默', 8); - - private final char character; - private final int width; - - ASCIIWidth(char character, int width) { - this.character = character; - this.width = width; - } - - public char getCharacter() { - return this.character; - } - - public int getWidth() { - return this.width; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/font/OffsetFont.java b/src/main/java/net/momirealms/customnameplates/object/font/OffsetFont.java deleted file mode 100644 index 5be59bf..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/font/OffsetFont.java +++ /dev/null @@ -1,68 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.font; - -public enum OffsetFont { - - NEG_1('\uf801', -1, -3), - NEG_2('\uf802', -2, -4), - NEG_3('\uf803', -3, -5), - NEG_4('\uf804', -4, -6), - NEG_5('\uf805', -5, -7), - NEG_6('\uf806', -6, -8), - NEG_7('\uf807', -7, -9), - NEG_8('\uf808', -8, -10), - NEG_16('\uf809', -16, -18), - NEG_32('\uf80a', -32, -34), - NEG_64('\uf80b', -64, -66), - NEG_128('\uf80c', -128, -130), - POS_1('\uf811', 1, -1), - POS_2('\uf812', 2, 1), - POS_3('\uf813', 3, 2), - POS_4('\uf814', 4, 3), - POS_5('\uf815', 5, 4), - POS_6('\uf816', 6, 5), - POS_7('\uf817', 7, 6), - POS_8('\uf818', 8, 7), - POS_16('\uf819', 16, 15), - POS_32('\uf81a', 32, 31), - POS_64('\uf81b', 64, 63), - POS_128('\uf81c', 128, 127); - - private final char character; - private final int space; - private final int height; - - OffsetFont(char character, int space, int height) { - this.character = character; - this.space = space; - this.height = height; - } - - public char getCharacter() { - return this.character; - } - - public int getSpace() { - return this.space; - } - - public int getHeight() { - return this.height; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/img/ImageParser.java b/src/main/java/net/momirealms/customnameplates/object/img/ImageParser.java deleted file mode 100644 index 9fa8ff8..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/img/ImageParser.java +++ /dev/null @@ -1,25 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.img; - -import org.bukkit.entity.Player; - -public interface ImageParser { - - String parse(Player player, String text); -} diff --git a/src/main/java/net/momirealms/customnameplates/object/img/ItemsAdderImageImpl.java b/src/main/java/net/momirealms/customnameplates/object/img/ItemsAdderImageImpl.java deleted file mode 100644 index 8674ec9..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/img/ItemsAdderImageImpl.java +++ /dev/null @@ -1,29 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.img; - -import dev.lone.itemsadder.api.FontImages.FontImageWrapper; -import org.bukkit.entity.Player; - -public class ItemsAdderImageImpl implements ImageParser{ - - @Override - public String parse(Player player, String text) { - return FontImageWrapper.replaceFontImages(player, text).replace("§f","").replace("§r",""); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/img/OraxenImageImpl.java b/src/main/java/net/momirealms/customnameplates/object/img/OraxenImageImpl.java deleted file mode 100644 index 701421c..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/img/OraxenImageImpl.java +++ /dev/null @@ -1,44 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.img; - -import io.th0rgal.oraxen.OraxenPlugin; -import io.th0rgal.oraxen.font.FontManager; -import io.th0rgal.oraxen.font.Glyph; -import org.bukkit.entity.Player; - -import java.util.Map; - -public class OraxenImageImpl implements ImageParser{ - - private final FontManager fontManager; - - public OraxenImageImpl() { - this.fontManager = OraxenPlugin.get().getFontManager(); - } - - @Override - public String parse(Player player, String text) { - for (Map.Entry entry : this.fontManager.getGlyphByPlaceholderMap().entrySet()) { - if (entry.getValue().hasPermission(player)) { - text = text.replace(entry.getKey(), "" + entry.getValue().getCharacter() + ""); - } - } - return text; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/nameplate/NameplateConfig.java b/src/main/java/net/momirealms/customnameplates/object/nameplate/NameplateConfig.java deleted file mode 100644 index d06f909..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/nameplate/NameplateConfig.java +++ /dev/null @@ -1,25 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.nameplate; - -import net.momirealms.customnameplates.object.SimpleChar; -import org.bukkit.ChatColor; - -public record NameplateConfig(ChatColor color, String display_name, SimpleChar left, SimpleChar middle, SimpleChar right) { - -} diff --git a/src/main/java/net/momirealms/customnameplates/object/nameplate/NameplatesTeam.java b/src/main/java/net/momirealms/customnameplates/object/nameplate/NameplatesTeam.java deleted file mode 100644 index e3d07bb..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/nameplate/NameplatesTeam.java +++ /dev/null @@ -1,100 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.nameplate; - -import me.clip.placeholderapi.PlaceholderAPI; -import net.kyori.adventure.text.Component; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.NameplateManager; -import net.momirealms.customnameplates.object.DynamicText; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.ChatColor; -import org.bukkit.entity.Player; - -public class NameplatesTeam { - - private final NameplateManager nameplateManager; - private final Player player; - private ChatColor color; - private final DynamicText prefix; - private final DynamicText suffix; - private String nameplate_prefix; - private String nameplate_suffix; - private String nameplate; - private final String team_name; - - public NameplatesTeam(NameplateManager nameplateManager, Player player, String team_name) { - this.nameplateManager = nameplateManager; - this.team_name = team_name; - this.player = player; - this.prefix = new DynamicText(player, nameplateManager.getPrefix()); - this.suffix = new DynamicText(player, nameplateManager.getSuffix()); - update(true); - } - - public boolean update(boolean force) { - String newNameplate = nameplateManager.getEquippedNameplate(player); - NameplateConfig nameplateConfig = nameplateManager.getNameplateConfig(newNameplate); - boolean updated = prefix.update() || suffix.update() || force || !newNameplate.equals(nameplate); - if (newNameplate.equals("none") || nameplateConfig == null) { - this.color = ChatColor.WHITE; - this.nameplate = "none"; - if (updated) { - nameplate = newNameplate; - nameplate_prefix = prefix.getLatestValue(); - nameplate_suffix = suffix.getLatestValue(); - } - } else { - this.color = nameplateConfig.color(); - String name = PlaceholderAPI.setPlaceholders(player, nameplateManager.getPlayerNamePapi()); - if (updated) { - nameplate = newNameplate; - String text = (nameplateManager.isPrefixHidden() ? "" : AdventureUtils.stripAllTags(prefix.getLatestValue())) - + name + - (nameplateManager.isSuffixHidden() ? "" : AdventureUtils.stripAllTags(suffix.getLatestValue())); - this.nameplate_prefix = nameplateManager.getNameplatePrefixWithFont(text, nameplateConfig) + (nameplateManager.isPrefixHidden() ? "" : prefix.getLatestValue()); - this.nameplate_suffix = CustomNameplates.getInstance().getFontManager().getSuffixStringWithFont(text) + (nameplateManager.isPrefixHidden() ? "" : suffix.getLatestValue()); - } - } - return updated; - } - - public String getTeam_name() { - return team_name; - } - - public ChatColor getColor() { - return this.color; - } - - public String getNameplatePrefixText() { - return nameplate_prefix; - } - - public String getNameplateSuffixText() { - return nameplate_suffix; - } - - public Component getNameplatePrefixComponent() { - return AdventureUtils.getComponentFromMiniMessage(nameplate_prefix); - } - - public Component getNameplateSuffixComponent() { - return AdventureUtils.getComponentFromMiniMessage(nameplate_suffix); - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/placeholders/BackGroundText.java b/src/main/java/net/momirealms/customnameplates/object/placeholders/BackGroundText.java deleted file mode 100644 index c583a21..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/placeholders/BackGroundText.java +++ /dev/null @@ -1,22 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.placeholders; - -public record BackGroundText(String text, String background, boolean remove_shadow) { - -} diff --git a/src/main/java/net/momirealms/customnameplates/object/placeholders/ConditionalTexts.java b/src/main/java/net/momirealms/customnameplates/object/placeholders/ConditionalTexts.java deleted file mode 100644 index 86c6bc7..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/placeholders/ConditionalTexts.java +++ /dev/null @@ -1,44 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.placeholders; - -import net.momirealms.customnameplates.object.ConditionalText; -import net.momirealms.customnameplates.object.requirements.Requirement; -import org.bukkit.entity.Player; - -public class ConditionalTexts { - - private final ConditionalText[] conditionalTexts; - - public ConditionalTexts(ConditionalText[] conditionalTexts) { - this.conditionalTexts = conditionalTexts; - } - - public String getValue(Player player) { - outer: - for (ConditionalText conditionalText : conditionalTexts) { - for (Requirement requirement : conditionalText.requirements()) { - if (!requirement.isConditionMet(player)) { - continue outer; - } - } - return conditionalText.text(); - } - return ""; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/placeholders/DescentText.java b/src/main/java/net/momirealms/customnameplates/object/placeholders/DescentText.java deleted file mode 100644 index c83c109..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/placeholders/DescentText.java +++ /dev/null @@ -1,22 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.placeholders; - -public record DescentText(String text, int ascent) { - -} diff --git a/src/main/java/net/momirealms/customnameplates/object/placeholders/NameplatePlaceholders.java b/src/main/java/net/momirealms/customnameplates/object/placeholders/NameplatePlaceholders.java deleted file mode 100644 index b68e3ab..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/placeholders/NameplatePlaceholders.java +++ /dev/null @@ -1,250 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.placeholders; - -import me.clip.placeholderapi.PlaceholderAPI; -import me.clip.placeholderapi.expansion.PlaceholderExpansion; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.ConfigManager; -import net.momirealms.customnameplates.manager.FontManager; -import net.momirealms.customnameplates.manager.NameplateManager; -import net.momirealms.customnameplates.manager.PlaceholderManager; -import net.momirealms.customnameplates.object.SimpleChar; -import net.momirealms.customnameplates.object.nameplate.NameplateConfig; -import net.momirealms.customnameplates.object.nameplate.NameplatesTeam; -import net.momirealms.customnameplates.utils.AdventureUtils; -import org.bukkit.entity.Player; -import org.jetbrains.annotations.NotNull; - -public class NameplatePlaceholders extends PlaceholderExpansion { - - private final PlaceholderManager placeholderManager; - private final CustomNameplates plugin; - - public NameplatePlaceholders(CustomNameplates plugin, PlaceholderManager placeholderManager) { - this.placeholderManager = placeholderManager; - this.plugin = plugin; - } - - @Override - public @NotNull String getIdentifier() { - return "nameplates"; - } - - @Override - public @NotNull String getAuthor() { - return "XiaoMoMi"; - } - - @Override - public @NotNull String getVersion() { - return "2.2"; - } - - @Override - public boolean persist() { - return true; - } - - @Override - public String onPlaceholderRequest(Player player, String params) { - String[] mainParam = params.split("_", 2); - switch (mainParam[0]) { - case "prefix" -> { - return getPrefix(player); - } - case "suffix" -> { - return getSuffix(player); - } - case "equipped" -> { - return getEquipped(mainParam[1], player); - } - case "static" -> { - return getStatic(mainParam[1], player); - } - case "background" -> { - return getBackground(mainParam[1], player); - } - case "nameplate" -> { - return getNameplateText(mainParam[1], player); - } - case "conditional" -> { - return getConditional(mainParam[1], player); - } - case "descent" -> { - return getDescent(mainParam[1], player); - } - case "vanilla" -> { - return getVanilla(mainParam[1], player); - } - case "image" -> { - return getImage(mainParam[1]); - } - case "offset" -> { - return getOffset(mainParam[1]); - } - case "time" -> { - return getTime(player); - } - case "checkupdate" -> { - return String.valueOf(!plugin.getVersionHelper().isLatest()); - } - case "actionbar" -> { - return getOtherActionBar(player); - } - case "unicode" -> { - return getUnicodeDescent(mainParam[1], player); - } - } - return null; - } - - private String getOtherActionBar(Player player) { - return plugin.getActionBarManager().getOtherPluginActionBarText(player); - } - - private String getOffset(String s) { - return ConfigManager.surroundWithFont(plugin.getFontManager().getOffset(Integer.parseInt(s))); - } - - private String getNameplateText(String param, Player player) { - NameplateText nameplateText = placeholderManager.getNameplateText(param); - if (nameplateText == null) return param + " NOT FOUND"; - String parsed = PlaceholderAPI.setPlaceholders(player, nameplateText.text()); - NameplateManager nameplateManager = plugin.getNameplateManager(); - NameplateConfig nameplateConfig = nameplateManager.getNameplateConfig(nameplateText.nameplate()); - if (nameplateConfig == null) return nameplateText.nameplate() + " NOT FOUND"; - String text = AdventureUtils.stripAllTags(parsed); - return nameplateManager.getNameplatePrefixWithFont(text, nameplateConfig) + parsed + plugin.getFontManager().getSuffixStringWithFont(text); - } - - private String getEquipped(String param, Player player) { - if (param.equals("nameplate")) { - return plugin.getDataManager().getEquippedNameplate(player); - } else if (param.equals("bubble")) { - return plugin.getDataManager().getEquippedBubble(player); - } - return "null"; - } - - private String getTime(Player player) { - long time = player.getWorld().getTime(); - String ap = time >= 6000 && time < 18000 ? " PM" : " AM"; - int hours = (int) (time / 1000) ; - int minutes = (int) ((time - hours * 1000 ) * 0.06); - hours += 6; - while (hours >= 12) hours -= 12; - if (minutes < 10) return hours + ":0" + minutes + ap; - else return hours + ":" + minutes + ap; - } - - private String getImage(String param) { - SimpleChar simpleChar = plugin.getImageManager().getImage(param); - if (simpleChar == null) return param + " NOT FOUND"; - return ConfigManager.surroundWithFont(String.valueOf(simpleChar.getChars())); - } - - private String getPrefix(Player player) { - NameplatesTeam nameplatesTeam = plugin.getTeamManager().getNameplateTeam(player.getUniqueId()); - if (nameplatesTeam != null) return nameplatesTeam.getNameplatePrefixText(); - return ""; - } - - private String getSuffix(Player player) { - NameplatesTeam nameplatesTeam = plugin.getTeamManager().getNameplateTeam(player.getUniqueId()); - if (nameplatesTeam != null) return nameplatesTeam.getNameplateSuffixText(); - return ""; - } - - private String getStatic(String param, Player player) { - StaticText staticText = placeholderManager.getStaticText(param); - if (staticText == null) return param + " NOT FOUND"; - FontManager fontManager = plugin.getFontManager(); - String parsed = PlaceholderAPI.setPlaceholders(player, staticText.text()); - int parsedWidth = fontManager.getTotalWidth(AdventureUtils.stripAllTags(parsed)); - if (staticText.staticState() == StaticText.StaticState.LEFT) { - return parsed + ConfigManager.surroundWithFont(fontManager.getOffset(staticText.value() - parsedWidth)); - } else if (staticText.staticState() == StaticText.StaticState.RIGHT) { - return ConfigManager.surroundWithFont(fontManager.getOffset(staticText.value() - parsedWidth)) + parsed; - } else if (staticText.staticState() == StaticText.StaticState.MIDDLE) { - int half = (staticText.value() - parsedWidth) / 2; - String left = ConfigManager.surroundWithFont(fontManager.getOffset(half)); - String right = ConfigManager.surroundWithFont(fontManager.getOffset(staticText.value() - parsedWidth - half)); - return left + parsed + right; - } - return ""; - } - - private String getBackground(String param, Player player) { - BackGroundText backGroundText = placeholderManager.getBackgroundText(param); - if (backGroundText == null) return param + " NOT FOUND"; - String parsed = PlaceholderAPI.setPlaceholders(player, backGroundText.text()); - String background = plugin.getBackgroundManager().getBackGroundImage(backGroundText, AdventureUtils.stripAllTags(parsed)); - if (backGroundText.remove_shadow()) background = "<#FFFEFD>" + background + ""; - return ConfigManager.surroundWithFont(background) + parsed; - } - - private String getConditional(String param, Player player) { - ConditionalTexts conditionalTexts = placeholderManager.getConditionalTexts(param); - if (conditionalTexts == null) return param + " NOT FOUND"; - String value = conditionalTexts.getValue(player); - return PlaceholderAPI.setPlaceholders(player, value); - } - - private String getDescent(String param, Player player) { - DescentText descentText = placeholderManager.getDescentText(param); - if (descentText == null) return param + " NOT FOUND"; - String parsed = PlaceholderAPI.setPlaceholders(player, descentText.text()); - return "" + parsed + ""; - } - - private String getUnicodeDescent(String param, Player player) { - if (!ConfigManager.enable1_20_Unicode && plugin.getVersionHelper().isVersionNewerThan1_20()) { - return "Not Available on 1.20"; - } - DescentText descentText = placeholderManager.getDescentUnicode(param); - if (descentText == null) return param + " NOT FOUND"; - String parsed = PlaceholderAPI.setPlaceholders(player, descentText.text()); - return "" + parsed + ""; - } - - private String getVanilla(String param, Player player) { - VanillaHud vanillaHud = placeholderManager.getVanillaHud(param); - if (vanillaHud == null) return param + " NOT FOUND"; - double current; - double max; - try { - current= Double.parseDouble(PlaceholderAPI.setPlaceholders(player, vanillaHud.papi())); - max = Double.parseDouble(PlaceholderAPI.setPlaceholders(player, vanillaHud.max())); - } catch (NumberFormatException e) { - current = 1; - max = 1; - } - if (current >= max) current = max; - if (current < 0) current = 0; - int point = (int) ((current / max) * 20); - int full_amount = point / 2; - int half_amount = point % 2; - int empty_amount = 10 - full_amount - half_amount; - if (vanillaHud.reverse()) { - return "<#FFFEFD>" + vanillaHud.empty().repeat(empty_amount) + vanillaHud.half().repeat(half_amount) + vanillaHud.full().repeat(full_amount) + ""; - } else { - return "<#FFFEFD>" + vanillaHud.full().repeat(full_amount) + vanillaHud.half().repeat(half_amount) + vanillaHud.empty().repeat(empty_amount) + ""; - } - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/placeholders/NameplateText.java b/src/main/java/net/momirealms/customnameplates/object/placeholders/NameplateText.java deleted file mode 100644 index 3d2e243..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/placeholders/NameplateText.java +++ /dev/null @@ -1,22 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.placeholders; - -public record NameplateText(String text, String nameplate) { - -} diff --git a/src/main/java/net/momirealms/customnameplates/object/placeholders/VanillaHud.java b/src/main/java/net/momirealms/customnameplates/object/placeholders/VanillaHud.java deleted file mode 100644 index fe93097..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/placeholders/VanillaHud.java +++ /dev/null @@ -1,22 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.placeholders; - -public record VanillaHud(String empty, String half, String full, String papi, String max, boolean reverse) { - -} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/BiomeImpl.java b/src/main/java/net/momirealms/customnameplates/object/requirements/BiomeImpl.java deleted file mode 100644 index 122d1af..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/BiomeImpl.java +++ /dev/null @@ -1,31 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements; - -import net.momirealms.biomeapi.BiomeAPI; -import org.bukkit.entity.Player; - -import java.util.HashSet; - -public record BiomeImpl(HashSet biomes) implements Requirement { - - @Override - public boolean isConditionMet(Player player) { - return biomes.contains(BiomeAPI.getBiome(player.getLocation())); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/CustomPapiImpl.java b/src/main/java/net/momirealms/customnameplates/object/requirements/CustomPapiImpl.java deleted file mode 100644 index b13a0c6..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/CustomPapiImpl.java +++ /dev/null @@ -1,77 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements; - -import net.momirealms.customnameplates.object.requirements.papi.*; -import org.bukkit.configuration.MemorySection; -import org.bukkit.entity.Player; - -import java.util.ArrayList; -import java.util.List; -import java.util.Map; - -public class CustomPapiImpl implements Requirement { - - private final List papiRequirement; - - public CustomPapiImpl(Map expressions){ - papiRequirement = getRequirements(expressions); - } - - @Override - public boolean isConditionMet(Player player) { - for (PapiRequirement requirement : papiRequirement) { - if (!requirement.isMet(player)) { - return false; - } - } - return true; - } - - private List getRequirements(Map map) { - List papiRequirements = new ArrayList<>(); - map.keySet().forEach(key -> { - if (key.startsWith("&&")) { - if (map.get(key) instanceof MemorySection map2) { - papiRequirements.add(new ExpressionAnd(getRequirements(map2.getValues(false)))); - } - } else if (key.startsWith("||")) { - if (map.get(key) instanceof MemorySection map2) { - papiRequirements.add(new ExpressionOr(getRequirements(map2.getValues(false)))); - } - } else { - if (map.get(key) instanceof MemorySection map2) { - String type = map2.getString("type"); - String papi = map2.getString("papi"); - String value = map2.getString("value"); - if (value == null || papi == null || type == null) return; - switch (type){ - case "==" -> papiRequirements.add(new PapiEquals(papi, value)); - case "!=" -> papiRequirements.add(new PapiNotEquals(papi, value)); - case ">=" -> papiRequirements.add(new PapiNoLess(papi, value)); - case "<=" -> papiRequirements.add(new PapiNoLarger(papi, value)); - case "<" -> papiRequirements.add(new PapiSmaller(papi, value)); - case ">" -> papiRequirements.add(new PapiGreater(papi, value)); - case "regex" -> papiRequirements.add(new PapiRegex(papi, value)); - } - } - } - }); - return papiRequirements; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/DateImpl.java b/src/main/java/net/momirealms/customnameplates/object/requirements/DateImpl.java deleted file mode 100644 index 298a695..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/DateImpl.java +++ /dev/null @@ -1,33 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements; - -import org.bukkit.entity.Player; - -import java.util.Calendar; -import java.util.HashSet; - -public record DateImpl(HashSet dates) implements Requirement { - - @Override - public boolean isConditionMet(Player player) { - Calendar calendar = Calendar.getInstance(); - String current = (calendar.get(Calendar.MONTH) + 1) + "/" + calendar.get(Calendar.DATE); - return dates.contains(current); - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/GeyserImpl.java b/src/main/java/net/momirealms/customnameplates/object/requirements/GeyserImpl.java deleted file mode 100644 index 69a453c..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/GeyserImpl.java +++ /dev/null @@ -1,29 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements; - -import net.momirealms.customnameplates.utils.GeyserUtils; -import org.bukkit.entity.Player; - -public record GeyserImpl(boolean is) implements Requirement { - - @Override - public boolean isConditionMet(Player player) { - return GeyserUtils.isBedrockPlayer(player.getUniqueId()) == is; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/PermissionImpl.java b/src/main/java/net/momirealms/customnameplates/object/requirements/PermissionImpl.java deleted file mode 100644 index 0413746..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/PermissionImpl.java +++ /dev/null @@ -1,28 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements; - -import org.bukkit.entity.Player; - -public record PermissionImpl(String permission) implements Requirement { - - @Override - public boolean isConditionMet(Player player) { - return player.hasPermission(permission); - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/TimeImpl.java b/src/main/java/net/momirealms/customnameplates/object/requirements/TimeImpl.java deleted file mode 100644 index 3fc2f65..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/TimeImpl.java +++ /dev/null @@ -1,37 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements; - -import org.bukkit.entity.Player; - -import java.util.List; - -public record TimeImpl(List times) implements Requirement{ - - @Override - public boolean isConditionMet(Player player) { - long time = player.getWorld().getTime(); - for (String range : times) { - String[] timeMinMax = range.split("~"); - if (time > Long.parseLong(timeMinMax[0]) && time < Long.parseLong(timeMinMax[1])) { - return true; - } - } - return false; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/WeatherImpl.java b/src/main/java/net/momirealms/customnameplates/object/requirements/WeatherImpl.java deleted file mode 100644 index 94fee13..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/WeatherImpl.java +++ /dev/null @@ -1,46 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements; - -import org.bukkit.World; -import org.bukkit.entity.Player; - -import java.util.List; - -public record WeatherImpl(List weathers) implements Requirement { - - @Override - public boolean isConditionMet(Player player) { - World world = player.getWorld(); - String currentWeather; - if (world.isThundering()) { - if (world.isClearWeather()) currentWeather = "thunder"; - else currentWeather = "rainstorm"; - } else if (world.isClearWeather()) { - currentWeather = "clear"; - } else { - currentWeather = "rain"; - } - for (String weather : weathers) { - if (weather.equalsIgnoreCase(currentWeather)) { - return true; - } - } - return false; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/WorldImpl.java b/src/main/java/net/momirealms/customnameplates/object/requirements/WorldImpl.java deleted file mode 100644 index c74753a..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/WorldImpl.java +++ /dev/null @@ -1,31 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements; - -import org.bukkit.entity.Player; - -import java.util.List; - -public record WorldImpl(List worlds) implements Requirement { - - @Override - public boolean isConditionMet(Player player) { - org.bukkit.World world = player.getWorld(); - return worlds.contains(world.getName()); - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/YPosImpl.java b/src/main/java/net/momirealms/customnameplates/object/requirements/YPosImpl.java deleted file mode 100644 index 1fb8fe8..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/YPosImpl.java +++ /dev/null @@ -1,37 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements; - -import org.bukkit.entity.Player; - -import java.util.List; - -public record YPosImpl(List yPos) implements Requirement { - - @Override - public boolean isConditionMet(Player player) { - int y = (int) player.getLocation().getY(); - for (String range : yPos) { - String[] yMinMax = range.split("~"); - if (y > Integer.parseInt(yMinMax[0]) && y < Integer.parseInt(yMinMax[1])) { - return true; - } - } - return false; - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiGreater.java b/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiGreater.java deleted file mode 100644 index 129fc96..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiGreater.java +++ /dev/null @@ -1,30 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements.papi; - -import me.clip.placeholderapi.PlaceholderAPI; -import org.bukkit.entity.Player; - -public record PapiGreater(String papi, String requirement) implements PapiRequirement{ - - @Override - public boolean isMet(Player player) { - double value = Double.parseDouble(PlaceholderAPI.setPlaceholders(player, papi)); - return value > Double.parseDouble(PlaceholderAPI.setPlaceholders(player, requirement)); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiNoLarger.java b/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiNoLarger.java deleted file mode 100644 index 2a72872..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiNoLarger.java +++ /dev/null @@ -1,30 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements.papi; - -import me.clip.placeholderapi.PlaceholderAPI; -import org.bukkit.entity.Player; - -public record PapiNoLarger(String papi, String requirement) implements PapiRequirement{ - - @Override - public boolean isMet(Player player) { - double value = Double.parseDouble(PlaceholderAPI.setPlaceholders(player, papi)); - return value <= Double.parseDouble(PlaceholderAPI.setPlaceholders(player, requirement)); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiNoLess.java b/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiNoLess.java deleted file mode 100644 index d71dec5..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiNoLess.java +++ /dev/null @@ -1,30 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements.papi; - -import me.clip.placeholderapi.PlaceholderAPI; -import org.bukkit.entity.Player; - -public record PapiNoLess(String papi, String requirement) implements PapiRequirement{ - - @Override - public boolean isMet(Player player) { - double value = Double.parseDouble(PlaceholderAPI.setPlaceholders(player, papi)); - return value >= Double.parseDouble(PlaceholderAPI.setPlaceholders(player, requirement)); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiNotEquals.java b/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiNotEquals.java deleted file mode 100644 index ef80bf3..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiNotEquals.java +++ /dev/null @@ -1,32 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements.papi; - -import me.clip.placeholderapi.PlaceholderAPI; -import org.bukkit.entity.Player; - -import java.util.Objects; - -public record PapiNotEquals(String papi, String requirement) implements PapiRequirement{ - - @Override - public boolean isMet(Player player) { - String value = PlaceholderAPI.setPlaceholders(player, papi); - return !Objects.equals(value, PlaceholderAPI.setPlaceholders(player, requirement)); - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiRegex.java b/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiRegex.java deleted file mode 100644 index 5bb4bc2..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiRegex.java +++ /dev/null @@ -1,12 +0,0 @@ -package net.momirealms.customnameplates.object.requirements.papi; - -import me.clip.placeholderapi.PlaceholderAPI; -import org.bukkit.entity.Player; - -public record PapiRegex(String papi, String regex) implements PapiRequirement { - - @Override - public boolean isMet(Player player) { - return PlaceholderAPI.setPlaceholders(player, papi).matches(regex); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiRequirement.java b/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiRequirement.java deleted file mode 100644 index 3f85088..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiRequirement.java +++ /dev/null @@ -1,24 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements.papi; - -import org.bukkit.entity.Player; - -public interface PapiRequirement { - boolean isMet(Player player); -} diff --git a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiSmaller.java b/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiSmaller.java deleted file mode 100644 index 64161f1..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/requirements/papi/PapiSmaller.java +++ /dev/null @@ -1,30 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.requirements.papi; - -import me.clip.placeholderapi.PlaceholderAPI; -import org.bukkit.entity.Player; - -public record PapiSmaller(String papi, String requirement) implements PapiRequirement{ - - @Override - public boolean isMet(Player player) { - double value = Double.parseDouble(PlaceholderAPI.setPlaceholders(player, papi)); - return value < Double.parseDouble(PlaceholderAPI.setPlaceholders(player, requirement)); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/scheduler/BukkitSchedulerImpl.java b/src/main/java/net/momirealms/customnameplates/object/scheduler/BukkitSchedulerImpl.java deleted file mode 100644 index 1897c0d..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/scheduler/BukkitSchedulerImpl.java +++ /dev/null @@ -1,70 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.scheduler; - -import net.momirealms.customnameplates.CustomNameplates; -import org.bukkit.Bukkit; -import org.bukkit.Location; -import org.jetbrains.annotations.NotNull; - -import java.util.concurrent.Callable; -import java.util.concurrent.Future; - -public class BukkitSchedulerImpl implements SchedulerPlatform { - - private final CustomNameplates plugin; - - public BukkitSchedulerImpl(CustomNameplates plugin) { - this.plugin = plugin; - } - - @Override - public Future callSyncMethod(@NotNull Callable task) { - return Bukkit.getScheduler().callSyncMethod(plugin, task); - } - - @Override - public void runTask(Runnable runnable, Location location) { - Bukkit.getScheduler().runTask(plugin, runnable); - } - - @Override - public void runTaskLater(Runnable runnable, Location location, long delay) { - Bukkit.getScheduler().runTaskLater(plugin, runnable, delay); - } - - @Override - public void runTaskAsync(Runnable runnable) { - Bukkit.getScheduler().runTaskAsynchronously(plugin, runnable); - } - - @Override - public void runTaskAsyncLater(Runnable runnable, long delay) { - Bukkit.getScheduler().runTaskLaterAsynchronously(plugin, runnable, delay); - } - - @Override - public TimerTask runTaskAsyncTimer(Runnable runnable, long delay, long interval) { - return new BukkitTimerTask(Bukkit.getScheduler().runTaskTimerAsynchronously(plugin, runnable, delay, interval)); - } - - @Override - public TimerTask runTaskTimer(Runnable runnable, Location location, long delay, long interval) { - return new BukkitTimerTask(Bukkit.getScheduler().runTaskTimer(plugin, runnable, delay, interval)); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/scheduler/FoliaSchedulerImpl.java b/src/main/java/net/momirealms/customnameplates/object/scheduler/FoliaSchedulerImpl.java deleted file mode 100644 index ec56cc8..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/scheduler/FoliaSchedulerImpl.java +++ /dev/null @@ -1,71 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.scheduler; - -import io.papermc.paper.threadedregions.scheduler.ScheduledTask; -import net.momirealms.customnameplates.CustomNameplates; -import org.bukkit.Bukkit; -import org.bukkit.Location; -import org.jetbrains.annotations.NotNull; - -import java.util.concurrent.Callable; -import java.util.concurrent.Future; - -public class FoliaSchedulerImpl implements SchedulerPlatform { - - private final CustomNameplates plugin; - - public FoliaSchedulerImpl(CustomNameplates plugin) { - this.plugin = plugin; - } - - @Override - public Future callSyncMethod(@NotNull Callable task) { - return null; - } - - @Override - public void runTask(Runnable runnable, Location location) { - Bukkit.getRegionScheduler().execute(plugin, location, runnable); - } - - @Override - public void runTaskLater(Runnable runnable, Location location, long delay) { - Bukkit.getRegionScheduler().runDelayed(plugin, location, (ScheduledTask s) -> runnable.run(), delay); - } - - @Override - public void runTaskAsync(Runnable runnable) { - Bukkit.getGlobalRegionScheduler().execute(plugin, runnable); - } - - @Override - public void runTaskAsyncLater(Runnable runnable, long delay) { - Bukkit.getGlobalRegionScheduler().runDelayed(plugin, (ScheduledTask s) -> runnable.run(), delay); - } - - @Override - public TimerTask runTaskAsyncTimer(Runnable runnable, long delay, long interval) { - return new FoliaTimerTask(Bukkit.getGlobalRegionScheduler().runAtFixedRate(plugin, (ScheduledTask s) -> runnable.run(), delay, interval)); - } - - @Override - public TimerTask runTaskTimer(Runnable runnable, Location location, long delay, long interval) { - return new FoliaTimerTask(Bukkit.getRegionScheduler().runAtFixedRate(plugin, location, (ScheduledTask s) -> runnable.run(), delay, interval)); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/scheduler/FoliaTimerTask.java b/src/main/java/net/momirealms/customnameplates/object/scheduler/FoliaTimerTask.java deleted file mode 100644 index 532de4d..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/scheduler/FoliaTimerTask.java +++ /dev/null @@ -1,39 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.scheduler; - -import io.papermc.paper.threadedregions.scheduler.ScheduledTask; - -public class FoliaTimerTask implements TimerTask { - - private final ScheduledTask scheduledTask; - - public FoliaTimerTask(ScheduledTask scheduledTask) { - this.scheduledTask = scheduledTask; - } - - @Override - public void cancel() { - scheduledTask.cancel(); - } - - @Override - public boolean isCancelled() { - return scheduledTask.isCancelled(); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/scheduler/Scheduler.java b/src/main/java/net/momirealms/customnameplates/object/scheduler/Scheduler.java deleted file mode 100644 index e0ce1d8..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/scheduler/Scheduler.java +++ /dev/null @@ -1,37 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.scheduler; - -import net.momirealms.customnameplates.CustomNameplates; - -public class Scheduler { - - private final SchedulerPlatform schedulerPlatform; - - public Scheduler(CustomNameplates plugin) { - if (plugin.getVersionHelper().isFolia()) { - this.schedulerPlatform = new FoliaSchedulerImpl(plugin); - } else { - this.schedulerPlatform = new BukkitSchedulerImpl(plugin); - } - } - - public SchedulerPlatform getInstance() { - return schedulerPlatform; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/scheduler/SchedulerPlatform.java b/src/main/java/net/momirealms/customnameplates/object/scheduler/SchedulerPlatform.java deleted file mode 100644 index a6eb88b..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/scheduler/SchedulerPlatform.java +++ /dev/null @@ -1,41 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.scheduler; - -import org.bukkit.Location; -import org.jetbrains.annotations.NotNull; - -import java.util.concurrent.Callable; -import java.util.concurrent.Future; - -public interface SchedulerPlatform { - - Future callSyncMethod(@NotNull Callable task); - - void runTask(Runnable runnable, Location location); - - void runTaskLater(Runnable runnable, Location location, long delay); - - void runTaskAsync(Runnable runnable); - - void runTaskAsyncLater(Runnable runnable, long delay); - - TimerTask runTaskAsyncTimer(Runnable runnable, long delay, long interval); - - TimerTask runTaskTimer(Runnable runnable, Location location, long delay, long interval); -} diff --git a/src/main/java/net/momirealms/customnameplates/object/team/TeamNameInterface.java b/src/main/java/net/momirealms/customnameplates/object/team/TeamNameInterface.java deleted file mode 100644 index 4e201e8..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/team/TeamNameInterface.java +++ /dev/null @@ -1,29 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.team; - -import org.bukkit.entity.Player; - -public interface TeamNameInterface { - - String getTeamName(Player player); - void onJoin(Player player); - void onQuit(Player player); - void unload(); - void load(); -} diff --git a/src/main/java/net/momirealms/customnameplates/object/team/TeamPacketInterface.java b/src/main/java/net/momirealms/customnameplates/object/team/TeamPacketInterface.java deleted file mode 100644 index 4e1bc07..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/team/TeamPacketInterface.java +++ /dev/null @@ -1,26 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.team; - -import org.bukkit.entity.Player; - -public interface TeamPacketInterface { - - void sendUpdateToOne(Player player); - void sendUpdateToAll(Player player, boolean force); -} diff --git a/src/main/java/net/momirealms/customnameplates/object/team/name/PlayerNameTeamImpl.java b/src/main/java/net/momirealms/customnameplates/object/team/name/PlayerNameTeamImpl.java deleted file mode 100644 index d931fdc..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/team/name/PlayerNameTeamImpl.java +++ /dev/null @@ -1,120 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.team.name; - -import com.comphenix.protocol.PacketType; -import com.comphenix.protocol.events.InternalStructure; -import com.comphenix.protocol.events.PacketContainer; -import com.comphenix.protocol.utility.MinecraftReflection; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.TeamManager; -import net.momirealms.customnameplates.object.team.TeamNameInterface; -import org.bukkit.Bukkit; -import org.bukkit.ChatColor; -import org.bukkit.entity.Player; -import org.bukkit.scoreboard.Scoreboard; -import org.bukkit.scoreboard.Team; - -import java.util.Collections; -import java.util.Optional; - -public class PlayerNameTeamImpl implements TeamNameInterface { - - private final TeamManager teamManager; - - public PlayerNameTeamImpl(TeamManager teamManager) { - this.teamManager = teamManager; - } - - @Override - public String getTeamName(Player player) { - return player.getName(); - } - - @Override - public void onJoin(Player player) { - if (teamManager.isFakeTeam()) { - createFakeTeamToAll(player); - } else { - Scoreboard scoreboard = Bukkit.getScoreboardManager().getMainScoreboard(); - Team team = scoreboard.getTeam(player.getName()); - if (team == null) { - team = scoreboard.registerNewTeam(player.getName()); - } - team.addEntry(player.getName()); - } - } - - @Override - public void onQuit(Player player) { - if (teamManager.isFakeTeam()) { - destroyTeamToAll(player); - } else { - Scoreboard scoreboard = Bukkit.getScoreboardManager().getMainScoreboard(); - Team team = scoreboard.getTeam(player.getName()); - if (team != null) team.unregister(); - } - } - - @Override - public void unload() { - for (Player player : Bukkit.getOnlinePlayers()) { - onQuit(player); - } - } - - @Override - public void load() { - for (Player player : Bukkit.getOnlinePlayers()) { - onJoin(player); - } - } - - // Send fake team packets - public void createFakeTeamToAll(Player joinPlayer) { - PacketContainer packetToAll = getPlayerTeamCreatePacket(joinPlayer); - for (Player all : Bukkit.getOnlinePlayers()) { - CustomNameplates.getProtocolManager().sendServerPacket(joinPlayer, getPlayerTeamCreatePacket(all)); - if (joinPlayer != all) CustomNameplates.getProtocolManager().sendServerPacket(all, packetToAll); - } - } - - // Get team create packet - private PacketContainer getPlayerTeamCreatePacket(Player joinPlayer) { - PacketContainer packetToAll = new PacketContainer(PacketType.Play.Server.SCOREBOARD_TEAM); - packetToAll.getIntegers().write(0,0); - packetToAll.getStrings().write(0, joinPlayer.getName()); - packetToAll.getModifier().write(2, Collections.singletonList(joinPlayer.getName())); - Optional optionalInternalStructure = packetToAll.getOptionalStructures().read(0); - if (optionalInternalStructure.isPresent()) { - InternalStructure internalStructure = optionalInternalStructure.get(); - internalStructure.getEnumModifier(ChatColor.class, MinecraftReflection.getMinecraftClass("EnumChatFormat")).write(0,ChatColor.WHITE); - } - return packetToAll; - } - - // Get team destroy packet - public void destroyTeamToAll(Player quitPlayer) { - PacketContainer packetToAll = new PacketContainer(PacketType.Play.Server.SCOREBOARD_TEAM); - packetToAll.getIntegers().write(0,1); - packetToAll.getStrings().write(0, quitPlayer.getName()); - for (Player all : Bukkit.getOnlinePlayers()) { - CustomNameplates.getProtocolManager().sendServerPacket(all, packetToAll); - } - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/team/name/TABBungeeCordImpl.java b/src/main/java/net/momirealms/customnameplates/object/team/name/TABBungeeCordImpl.java deleted file mode 100644 index 8d9bd2b..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/team/name/TABBungeeCordImpl.java +++ /dev/null @@ -1,108 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.team.name; - -import com.google.common.io.ByteArrayDataInput; -import com.google.common.io.ByteArrayDataOutput; -import com.google.common.io.ByteStreams; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.team.TeamNameInterface; -import org.bukkit.Bukkit; -import org.bukkit.entity.Player; -import org.bukkit.plugin.messaging.PluginMessageListener; -import org.jetbrains.annotations.NotNull; - -import java.util.Objects; -import java.util.concurrent.ConcurrentHashMap; - -public class TABBungeeCordImpl implements TeamNameInterface { - - private final ConcurrentHashMap teamNameMap; - private final ProxyDataListener proxyDataListener; - - public TABBungeeCordImpl() { - this.teamNameMap = new ConcurrentHashMap<>(); - this.proxyDataListener = new ProxyDataListener(this); - } - - @Override - public void load() { - Bukkit.getServer().getMessenger().registerOutgoingPluginChannel(CustomNameplates.getInstance(), "customnameplates:cnp"); - Bukkit.getServer().getMessenger().registerIncomingPluginChannel(CustomNameplates.getInstance(), "customnameplates:cnp", proxyDataListener); - } - - @Override - public void unload() { - this.teamNameMap.clear(); - Bukkit.getServer().getMessenger().unregisterIncomingPluginChannel(CustomNameplates.getInstance(), "customnameplates:cnp"); - Bukkit.getServer().getMessenger().unregisterOutgoingPluginChannel(CustomNameplates.getInstance(), "customnameplates:cnp"); - } - - @Override - public String getTeamName(Player player) { - String teamName = teamNameMap.get(player.getName()); - if (teamName == null) { - sendRequest(player); - return null; - } else { - return teamName; - } - } - - @Override - public void onJoin(Player player) { - sendRequest(player); - } - - @Override - public void onQuit(Player player) { - teamNameMap.remove(player.getName()); - } - - @SuppressWarnings("UnstableApiUsage") - private void sendRequest(Player player) { - ByteArrayDataOutput dataOutput = ByteStreams.newDataOutput(); - dataOutput.writeUTF(player.getName()); - player.sendPluginMessage(CustomNameplates.getInstance(), "customnameplates:cnp", dataOutput.toByteArray()); - } - - public void addPlayerToCache(String playerName, String teamName) { - teamNameMap.put(playerName, teamName); - } - - public static class ProxyDataListener implements PluginMessageListener { - - private final TABBungeeCordImpl TABBungeeCordImpl; - - public ProxyDataListener(TABBungeeCordImpl TABBungeeCordImpl) { - this.TABBungeeCordImpl = TABBungeeCordImpl; - } - - @Override - @SuppressWarnings("UnstableApiUsage") - public void onPluginMessageReceived(@NotNull String channel, @NotNull Player player, @NotNull byte[] message) { - if (!Objects.equals("customnameplates:cnp", channel)) { - return; - } - ByteArrayDataInput dataInput = ByteStreams.newDataInput(message); - String playerName = dataInput.readUTF(); - String teamName = dataInput.readUTF(); - TABBungeeCordImpl.addPlayerToCache(playerName, teamName); - } - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/team/name/TABImpl.java b/src/main/java/net/momirealms/customnameplates/object/team/name/TABImpl.java deleted file mode 100644 index 72ae7c3..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/team/name/TABImpl.java +++ /dev/null @@ -1,60 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.team.name; - -import me.neznamy.tab.api.TabAPI; -import me.neznamy.tab.api.TabPlayer; -import me.neznamy.tab.api.tablist.SortingManager; -import net.momirealms.customnameplates.object.team.TeamNameInterface; -import org.bukkit.entity.Player; - -public class TABImpl implements TeamNameInterface { - - private final SortingManager sortingManager; - - public TABImpl() { - sortingManager = TabAPI.getInstance().getSortingManager(); - } - - @Override - public String getTeamName(Player player) { - TabPlayer tabPlayer = TabAPI.getInstance().getPlayer(player.getUniqueId()); - if (tabPlayer == null) return player.getName(); - return sortingManager.getOriginalTeamName(tabPlayer); - } - - @Override - public void onJoin(Player player) { - - } - - @Override - public void onQuit(Player player) { - - } - - @Override - public void unload() { - - } - - @Override - public void load() { - - } -} \ No newline at end of file diff --git a/src/main/java/net/momirealms/customnameplates/object/team/name/VelocitabImpl.java b/src/main/java/net/momirealms/customnameplates/object/team/name/VelocitabImpl.java deleted file mode 100644 index 634ba6d..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/team/name/VelocitabImpl.java +++ /dev/null @@ -1,108 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.team.name; - -import com.google.common.io.ByteArrayDataInput; -import com.google.common.io.ByteArrayDataOutput; -import com.google.common.io.ByteStreams; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.object.team.TeamNameInterface; -import org.bukkit.Bukkit; -import org.bukkit.entity.Player; -import org.bukkit.plugin.messaging.PluginMessageListener; -import org.jetbrains.annotations.NotNull; - -import java.util.Objects; -import java.util.concurrent.ConcurrentHashMap; - -public class VelocitabImpl implements TeamNameInterface { - - private final ConcurrentHashMap teamNameMap; - private final ProxyDataListener proxyDataListener; - - public VelocitabImpl() { - this.teamNameMap = new ConcurrentHashMap<>(); - this.proxyDataListener = new ProxyDataListener(this); - } - - @Override - public void load() { - Bukkit.getServer().getMessenger().registerOutgoingPluginChannel(CustomNameplates.getInstance(), "customnameplates:cnp"); - Bukkit.getServer().getMessenger().registerIncomingPluginChannel(CustomNameplates.getInstance(), "customnameplates:cnp", proxyDataListener); - } - - @Override - public void unload() { - this.teamNameMap.clear(); - Bukkit.getServer().getMessenger().unregisterIncomingPluginChannel(CustomNameplates.getInstance(), "customnameplates:cnp"); - Bukkit.getServer().getMessenger().unregisterOutgoingPluginChannel(CustomNameplates.getInstance(), "customnameplates:cnp"); - } - - @Override - public String getTeamName(Player player) { - String teamName = teamNameMap.get(player.getName()); - if (teamName == null) { - sendRequest(player); - return null; - } else { - return teamName; - } - } - - @Override - public void onJoin(Player player) { - sendRequest(player); - } - - @Override - public void onQuit(Player player) { - teamNameMap.remove(player.getName()); - } - - @SuppressWarnings("UnstableApiUsage") - private void sendRequest(Player player) { - ByteArrayDataOutput dataOutput = ByteStreams.newDataOutput(); - dataOutput.writeUTF(player.getName()); - player.sendPluginMessage(CustomNameplates.getInstance(), "customnameplates:cnp", dataOutput.toByteArray()); - } - - public void addPlayerToCache(String playerName, String teamName) { - teamNameMap.put(playerName, teamName); - } - - public static class ProxyDataListener implements PluginMessageListener { - - private final VelocitabImpl TABBungeeCordImpl; - - public ProxyDataListener(VelocitabImpl TABBungeeCordImpl) { - this.TABBungeeCordImpl = TABBungeeCordImpl; - } - - @Override - @SuppressWarnings("UnstableApiUsage") - public void onPluginMessageReceived(@NotNull String channel, @NotNull Player player, @NotNull byte[] message) { - if (!Objects.equals("customnameplates:cnp", channel)) { - return; - } - ByteArrayDataInput dataInput = ByteStreams.newDataInput(message); - String playerName = dataInput.readUTF(); - String teamName = dataInput.readUTF(); - TABBungeeCordImpl.addPlayerToCache(playerName, teamName); - } - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/team/packet/TeamInfoImpl.java b/src/main/java/net/momirealms/customnameplates/object/team/packet/TeamInfoImpl.java deleted file mode 100644 index d3734f0..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/team/packet/TeamInfoImpl.java +++ /dev/null @@ -1,85 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.team.packet; - -import com.comphenix.protocol.PacketType; -import com.comphenix.protocol.events.InternalStructure; -import com.comphenix.protocol.events.PacketContainer; -import com.comphenix.protocol.utility.MinecraftReflection; -import com.comphenix.protocol.wrappers.WrappedChatComponent; -import net.kyori.adventure.text.serializer.gson.GsonComponentSerializer; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.TeamManager; -import net.momirealms.customnameplates.object.nameplate.NameplatesTeam; -import net.momirealms.customnameplates.object.team.TeamPacketInterface; -import org.bukkit.Bukkit; -import org.bukkit.ChatColor; -import org.bukkit.entity.Player; - -import java.util.Optional; - -public class TeamInfoImpl implements TeamPacketInterface { - - private final TeamManager teamManager; - - public TeamInfoImpl(TeamManager teamManager) { - this.teamManager = teamManager; - } - - // this method would only be used when a player join the server - @Override - public void sendUpdateToOne(Player player) { - for (Player otherPlayer : Bukkit.getOnlinePlayers()) { - if (player == otherPlayer) continue; - PacketContainer packet = new PacketContainer(PacketType.Play.Server.SCOREBOARD_TEAM); - NameplatesTeam nameplatesTeam = teamManager.getNameplateTeam(otherPlayer.getUniqueId()); - if (nameplatesTeam == null) continue; - packet.getStrings().write(0, nameplatesTeam.getTeam_name()); - //update - packet.getIntegers().write(0,2); - sendPacket(player, packet, nameplatesTeam); - } - } - - // this method would be used when joining the server, refresh task, equip new nameplate - @Override - public void sendUpdateToAll(Player player, boolean force) { - NameplatesTeam nameplatesTeam = teamManager.getNameplateTeam(player.getUniqueId()); - if (nameplatesTeam != null && nameplatesTeam.update(force)) { - for (Player otherPlayer : Bukkit.getOnlinePlayers()) { - PacketContainer packet = new PacketContainer(PacketType.Play.Server.SCOREBOARD_TEAM); - packet.getStrings().write(0, nameplatesTeam.getTeam_name()); - //update - packet.getIntegers().write(0,2); - sendPacket(otherPlayer, packet, nameplatesTeam); - } - } - } - - private void sendPacket(Player player, PacketContainer packet, NameplatesTeam nameplatesTeam) { - Optional optionalInternalStructure = packet.getOptionalStructures().read(0); - if (optionalInternalStructure.isPresent()) { - InternalStructure internalStructure = optionalInternalStructure.get(); - internalStructure.getStrings().write(0, "always"); - internalStructure.getChatComponents().write(1, WrappedChatComponent.fromJson(GsonComponentSerializer.gson().serialize(nameplatesTeam.getNameplatePrefixComponent()))); - internalStructure.getChatComponents().write(2, WrappedChatComponent.fromJson(GsonComponentSerializer.gson().serialize(nameplatesTeam.getNameplateSuffixComponent()))); - internalStructure.getEnumModifier(ChatColor.class, MinecraftReflection.getMinecraftClass("EnumChatFormat")).write(0, nameplatesTeam.getColor()); - CustomNameplates.getProtocolManager().sendServerPacket(player, packet); - } - } -} diff --git a/src/main/java/net/momirealms/customnameplates/object/team/packet/TeamVisibilityImpl.java b/src/main/java/net/momirealms/customnameplates/object/team/packet/TeamVisibilityImpl.java deleted file mode 100644 index 37a4aeb..0000000 --- a/src/main/java/net/momirealms/customnameplates/object/team/packet/TeamVisibilityImpl.java +++ /dev/null @@ -1,78 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.object.team.packet; - -import com.comphenix.protocol.PacketType; -import com.comphenix.protocol.events.InternalStructure; -import com.comphenix.protocol.events.PacketContainer; -import com.comphenix.protocol.utility.MinecraftReflection; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.manager.TeamManager; -import net.momirealms.customnameplates.object.nameplate.NameplatesTeam; -import net.momirealms.customnameplates.object.team.TeamPacketInterface; -import org.bukkit.Bukkit; -import org.bukkit.ChatColor; -import org.bukkit.entity.Player; - -import java.util.Optional; - -public class TeamVisibilityImpl implements TeamPacketInterface { - - private final TeamManager teamManager; - - public TeamVisibilityImpl(TeamManager teamManager) { - this.teamManager = teamManager; - } - - // this method would only be used when a player join the server - @Override - public void sendUpdateToOne(Player player) { - for (Player otherPlayer : Bukkit.getOnlinePlayers()) { - if (player == otherPlayer) continue; - NameplatesTeam nameplatesTeam = teamManager.getNameplateTeam(otherPlayer.getUniqueId()); - if (nameplatesTeam != null) { - PacketContainer packet = new PacketContainer(PacketType.Play.Server.SCOREBOARD_TEAM); - packet.getIntegers().write(0,2); - packet.getStrings().write(0, nameplatesTeam.getTeam_name()); - Optional optionalInternalStructure = packet.getOptionalStructures().read(0); - optionalInternalStructure.ifPresent(internalStructure -> sendPackets(player, packet, internalStructure)); - } - } - } - - // this method would only be used when a player join the server - @Override - public void sendUpdateToAll(Player player, boolean force) { - NameplatesTeam nameplatesTeam = teamManager.getNameplateTeam(player.getUniqueId()); - if (nameplatesTeam != null) { - for (Player otherPlayer : Bukkit.getOnlinePlayers()) { - PacketContainer packet = new PacketContainer(PacketType.Play.Server.SCOREBOARD_TEAM); - packet.getStrings().write(0, nameplatesTeam.getTeam_name()); - packet.getIntegers().write(0,2); - Optional optionalInternalStructure = packet.getOptionalStructures().read(0); - optionalInternalStructure.ifPresent(internalStructure -> sendPackets(otherPlayer, packet, internalStructure)); - } - } - } - - private void sendPackets(Player player, PacketContainer packet, InternalStructure internalStructure) { - internalStructure.getStrings().write(0, "never"); - internalStructure.getEnumModifier(ChatColor.class, MinecraftReflection.getMinecraftClass("EnumChatFormat")).write(0, ChatColor.WHITE); - CustomNameplates.getProtocolManager().sendServerPacket(player, packet); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/utils/AdventureUtils.java b/src/main/java/net/momirealms/customnameplates/utils/AdventureUtils.java deleted file mode 100644 index ea204c7..0000000 --- a/src/main/java/net/momirealms/customnameplates/utils/AdventureUtils.java +++ /dev/null @@ -1,157 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.utils; - -import net.kyori.adventure.audience.Audience; -import net.kyori.adventure.text.Component; -import net.kyori.adventure.text.minimessage.MiniMessage; -import net.momirealms.customnameplates.CustomNameplates; -import org.bukkit.Bukkit; -import org.bukkit.ChatColor; -import org.bukkit.command.CommandSender; -import org.bukkit.entity.Player; - -public class AdventureUtils { - - public static Component getComponentFromMiniMessage(String text) { - return MiniMessage.miniMessage().deserialize(replaceLegacy(text)); - } - - public static void sendMessage(CommandSender sender, String s) { - if (s == null) return; - if (sender instanceof Player player) playerMessage(player, s); - else consoleMessage(s); - } - - public static void consoleMessage(String s) { - if (s == null) return; - Audience au = CustomNameplates.getAdventure().sender(Bukkit.getConsoleSender()); - au.sendMessage(getComponentFromMiniMessage(s)); - } - - public static void playerMessage(Player player, String s) { - if (s == null) return; - Audience au = CustomNameplates.getAdventure().player(player); - au.sendMessage(getComponentFromMiniMessage(s)); - } - - public static void playerActionbar(Player player, Component component) { - Audience au = CustomNameplates.getAdventure().player(player); - au.sendActionBar(component); - } - - public static String replaceLegacy(String legacy) { - StringBuilder stringBuilder = new StringBuilder(); - char[] chars = legacy.toCharArray(); - for (int i = 0; i < chars.length; i++) { - if (isColorCode(chars[i])) { - if (i + 1 < chars.length) { - switch (chars[i+1]) { - case '0' -> stringBuilder.append(""); - case '1' -> stringBuilder.append(""); - case '2' -> stringBuilder.append(""); - case '3' -> stringBuilder.append(""); - case '4' -> stringBuilder.append(""); - case '5' -> stringBuilder.append(""); - case '6' -> stringBuilder.append(""); - case '7' -> stringBuilder.append(""); - case '8' -> stringBuilder.append(""); - case '9' -> stringBuilder.append(""); - case 'a' -> stringBuilder.append(""); - case 'b' -> stringBuilder.append(""); - case 'c' -> stringBuilder.append(""); - case 'd' -> stringBuilder.append(""); - case 'e' -> stringBuilder.append(""); - case 'f' -> stringBuilder.append(""); - case 'r' -> stringBuilder.append(""); - case 'l' -> stringBuilder.append(""); - case 'm' -> stringBuilder.append(""); - case 'o' -> stringBuilder.append(""); - case 'n' -> stringBuilder.append(""); - case 'k' -> stringBuilder.append(""); - case 'x' -> { - if (i + 13 >= chars.length - || !isColorCode(chars[i+2]) - || !isColorCode(chars[i+4]) - || !isColorCode(chars[i+6]) - || !isColorCode(chars[i+8]) - || !isColorCode(chars[i+10]) - || !isColorCode(chars[i+12])) { - stringBuilder.append(chars[i]); - continue; - } - stringBuilder - .append("<#") - .append(chars[i+3]) - .append(chars[i+5]) - .append(chars[i+7]) - .append(chars[i+9]) - .append(chars[i+11]) - .append(chars[i+13]) - .append(">"); - i += 12; - } - default -> { - stringBuilder.append(chars[i]); - continue; - } - } - i++; - } else { - stringBuilder.append(chars[i]); - } - } else { - stringBuilder.append(chars[i]); - } - } - return stringBuilder.toString(); - } - - private static boolean isColorCode(char c) { - return c == '§' || c == '&'; - } - - public static int colorToDecimal(ChatColor color){ - switch (String.valueOf(color.getChar())){ - case "0" -> {return 0;} - case "c" -> {return 16733525;} - case "6" -> {return 16755200;} - case "4" -> {return 11141120;} - case "e" -> {return 16777045;} - case "2" -> {return 43520;} - case "a" -> {return 5635925;} - case "b" -> {return 5636095;} - case "3" -> {return 43690;} - case "1" -> {return 170;} - case "9" -> {return 5592575;} - case "d" -> {return 16733695;} - case "5" -> {return 11141290;} - case "8" -> {return 5592405;} - case "7" -> {return 11184810;} - default -> {return 16777215;} - } - } - - public static String stripAllTags(String text) { - return MiniMessage.miniMessage().stripTags(replaceLegacy(text)); - } - - public static String getMiniMessageFormat(Component component) { - return MiniMessage.miniMessage().serialize(component); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/utils/ArmorStandUtils.java b/src/main/java/net/momirealms/customnameplates/utils/ArmorStandUtils.java deleted file mode 100644 index 863d0e4..0000000 --- a/src/main/java/net/momirealms/customnameplates/utils/ArmorStandUtils.java +++ /dev/null @@ -1,101 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.utils; - -import com.comphenix.protocol.PacketType; -import com.comphenix.protocol.ProtocolLibrary; -import com.comphenix.protocol.events.PacketContainer; -import com.comphenix.protocol.wrappers.WrappedChatComponent; -import com.comphenix.protocol.wrappers.WrappedDataValue; -import com.comphenix.protocol.wrappers.WrappedDataWatcher; -import com.google.common.collect.Lists; -import net.kyori.adventure.text.Component; -import net.kyori.adventure.text.serializer.gson.GsonComponentSerializer; -import net.momirealms.customnameplates.CustomNameplates; -import org.bukkit.Location; -import org.bukkit.entity.EntityType; -import org.bukkit.entity.Player; - -import java.util.*; - -public class ArmorStandUtils { - - public static void preview(Component component, Player player, int duration) { - int id = new Random().nextInt(Integer.MAX_VALUE); - sendSpawnPacket(player, id); - sendMetaPacket(player, id, component); - for (int i = 1; i < duration * 20 - 1; i++){ - CustomNameplates.getInstance().getScheduler().runTaskAsyncLater(()-> sendTeleportPacket(player, id), i); - } - CustomNameplates.getInstance().getScheduler().runTaskAsyncLater(()-> sendDestroyPacket(player, id), duration * 20L); - } - - public static void sendSpawnPacket(Player player, int id) { - PacketContainer spawnPacket = new PacketContainer(PacketType.Play.Server.SPAWN_ENTITY); - spawnPacket.getModifier().write(0, id); - spawnPacket.getModifier().write(1, UUID.randomUUID()); - spawnPacket.getEntityTypeModifier().write(0, EntityType.ARMOR_STAND); - Location location = player.getLocation(); - spawnPacket.getDoubles().write(0, location.getX()); - spawnPacket.getDoubles().write(1, location.getY()+0.8); - spawnPacket.getDoubles().write(2, location.getZ()); - ProtocolLibrary.getProtocolManager().sendServerPacket(player, spawnPacket); - } - - public static void sendMetaPacket(Player player, int id, Component component) { - PacketContainer metaPacket = new PacketContainer(PacketType.Play.Server.ENTITY_METADATA); - WrappedDataWatcher wrappedDataWatcher = new WrappedDataWatcher(); - WrappedDataWatcher.Serializer serializer1 = WrappedDataWatcher.Registry.get(Boolean.class); - WrappedDataWatcher.Serializer serializer2 = WrappedDataWatcher.Registry.get(Byte.class); - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(2, WrappedDataWatcher.Registry.getChatComponentSerializer(true)), Optional.of(WrappedChatComponent.fromJson(GsonComponentSerializer.gson().serialize(component)).getHandle())); - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(3, serializer1), true); - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(5, serializer1), true); - byte mask1 = 0x20; - byte mask2 = 0x01; - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(0, serializer2), mask1); - wrappedDataWatcher.setObject(new WrappedDataWatcher.WrappedDataWatcherObject(15, serializer2), mask2); - metaPacket.getModifier().write(0, id); - if (CustomNameplates.getInstance().getVersionHelper().isVersionNewerThan1_19_R2()) { - List wrappedDataValueList = Lists.newArrayList(); - wrappedDataWatcher.getWatchableObjects().stream().filter(Objects::nonNull).forEach(entry -> { - final WrappedDataWatcher.WrappedDataWatcherObject dataWatcherObject = entry.getWatcherObject(); - wrappedDataValueList.add(new WrappedDataValue(dataWatcherObject.getIndex(), dataWatcherObject.getSerializer(), entry.getRawValue())); - }); - metaPacket.getDataValueCollectionModifier().write(0, wrappedDataValueList); - } else { - metaPacket.getWatchableCollectionModifier().write(0, wrappedDataWatcher.getWatchableObjects()); - } - ProtocolLibrary.getProtocolManager().sendServerPacket(player, metaPacket); - } - - public static void sendTeleportPacket(Player player, int id){ - PacketContainer packet = new PacketContainer(PacketType.Play.Server.ENTITY_TELEPORT); - packet.getIntegers().write(0, id); - Location location = player.getLocation(); - packet.getDoubles().write(0, location.getX()); - packet.getDoubles().write(1, location.getY()+0.8); - packet.getDoubles().write(2, location.getZ()); - ProtocolLibrary.getProtocolManager().sendServerPacket(player, packet); - } - - public static void sendDestroyPacket(Player player, int id) { - PacketContainer packet = new PacketContainer(PacketType.Play.Server.ENTITY_DESTROY); - packet.getIntLists().write(0, List.of(id)); - ProtocolLibrary.getProtocolManager().sendServerPacket(player, packet); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/utils/ConfigUtils.java b/src/main/java/net/momirealms/customnameplates/utils/ConfigUtils.java deleted file mode 100644 index dde72a3..0000000 --- a/src/main/java/net/momirealms/customnameplates/utils/ConfigUtils.java +++ /dev/null @@ -1,117 +0,0 @@ -/* - * Copyright (C) <2022> - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package net.momirealms.customnameplates.utils; - -import dev.dejvokep.boostedyaml.YamlDocument; -import dev.dejvokep.boostedyaml.dvs.versioning.BasicVersioning; -import dev.dejvokep.boostedyaml.settings.dumper.DumperSettings; -import dev.dejvokep.boostedyaml.settings.general.GeneralSettings; -import dev.dejvokep.boostedyaml.settings.loader.LoaderSettings; -import dev.dejvokep.boostedyaml.settings.updater.UpdaterSettings; -import net.momirealms.customnameplates.CustomNameplates; -import net.momirealms.customnameplates.helper.Log; -import net.momirealms.customnameplates.object.carrier.TextDisplayMeta; -import net.momirealms.customnameplates.object.requirements.*; -import org.bukkit.configuration.ConfigurationSection; -import org.bukkit.configuration.file.YamlConfiguration; - -import java.io.File; -import java.io.IOException; -import java.util.ArrayList; -import java.util.HashSet; -import java.util.List; -import java.util.Objects; - -public class ConfigUtils { - - public static void update(String file_name){ - try { - YamlDocument.create( - new File(CustomNameplates.getInstance().getDataFolder(), file_name), - Objects.requireNonNull(CustomNameplates.getInstance().getResource(file_name)), - GeneralSettings.DEFAULT, - LoaderSettings.builder().setAutoUpdate(true).build(), - DumperSettings.DEFAULT, - UpdaterSettings.builder().setVersioning(new BasicVersioning("config-version") - ).build()); - } catch (IOException e){ - Log.warn(e.getMessage()); - } - } - - public static YamlConfiguration getConfig(String config_name) { - File file = new File(CustomNameplates.getInstance().getDataFolder(), config_name); - if (!file.exists()) CustomNameplates.getInstance().saveResource(config_name, false); - return YamlConfiguration.loadConfiguration(file); - } - - public static YamlConfiguration readData(File file) { - if (!file.exists()) { - try { - file.getParentFile().mkdirs(); - if (!file.createNewFile()) { - AdventureUtils.consoleMessage("[CustomNameplates] Failed to generate data files!"); - } - } catch (IOException e) { - e.printStackTrace(); - AdventureUtils.consoleMessage("[CustomNameplates] Failed to generate data files!"); - } - } - return YamlConfiguration.loadConfiguration(file); - } - - public static Requirement[] getRequirements(ConfigurationSection section) { - List requirements = new ArrayList<>(); - if (section != null) { - for (String type : section.getKeys(false)) { - switch (type) { - case "biome" -> requirements.add(new BiomeImpl(new HashSet<>(section.getStringList(type)))); - case "weather" -> requirements.add(new WeatherImpl(section.getStringList(type))); - case "ypos" -> requirements.add(new YPosImpl(section.getStringList(type))); - case "world" -> requirements.add(new WorldImpl(section.getStringList(type))); - case "permission" -> requirements.add(new PermissionImpl(section.getString(type))); - case "time" -> requirements.add(new TimeImpl(section.getStringList(type))); - case "date" -> requirements.add(new DateImpl(new HashSet<>(section.getStringList(type)))); - case "geyser" -> requirements.add(new GeyserImpl(section.getBoolean(type))); - case "papi-condition" -> requirements.add(new CustomPapiImpl(Objects.requireNonNull(section.getConfigurationSection(type)).getValues(false))); - } - } - } - return requirements.toArray(new Requirement[0]); - } - - public static TextDisplayMeta getTextDisplayMeta(ConfigurationSection section) { - if (section == null) return TextDisplayMeta.defaultValue; - return new TextDisplayMeta( - section.getBoolean("has-shadow", false), - section.getBoolean("is-see-through", false), - section.getBoolean("use-default-background-color", false), - ConfigUtils.rgbToDecimal(section.getString("background-color", "0,0,0,128")), - (byte) section.getInt("text-opacity") - ); - } - - public static int rgbToDecimal(String rgba) { - String[] split = rgba.split(","); - int r = Integer.parseInt(split[0]); - int g = Integer.parseInt(split[1]); - int b = Integer.parseInt(split[2]); - int a = Integer.parseInt(split[3]); - return (a << 24) | (r << 16) | (g << 8) | b; - } -} diff --git a/src/main/java/net/momirealms/customnameplates/utils/GeyserUtils.java b/src/main/java/net/momirealms/customnameplates/utils/GeyserUtils.java deleted file mode 100644 index 39ee7f9..0000000 --- a/src/main/java/net/momirealms/customnameplates/utils/GeyserUtils.java +++ /dev/null @@ -1,12 +0,0 @@ -package net.momirealms.customnameplates.utils; - -import org.geysermc.api.Geyser; - -import java.util.UUID; - -public class GeyserUtils { - - public static boolean isBedrockPlayer(UUID uuid) { - return Geyser.api().isBedrockPlayer(uuid); - } -} diff --git a/src/main/java/net/momirealms/customnameplates/velocity/CustomNameplatesVC.java b/src/main/java/net/momirealms/customnameplates/velocity/CustomNameplatesVC.java deleted file mode 100644 index 2d26a0c..0000000 --- a/src/main/java/net/momirealms/customnameplates/velocity/CustomNameplatesVC.java +++ /dev/null @@ -1,87 +0,0 @@ -package net.momirealms.customnameplates.velocity; - -import com.google.common.io.ByteArrayDataInput; -import com.google.common.io.ByteArrayDataOutput; -import com.google.common.io.ByteStreams; -import com.google.inject.Inject; -import com.velocitypowered.api.event.Subscribe; -import com.velocitypowered.api.event.connection.PluginMessageEvent; -import com.velocitypowered.api.event.proxy.ProxyInitializeEvent; -import com.velocitypowered.api.event.proxy.ProxyShutdownEvent; -import com.velocitypowered.api.plugin.Dependency; -import com.velocitypowered.api.plugin.Plugin; -import com.velocitypowered.api.plugin.PluginContainer; -import com.velocitypowered.api.proxy.Player; -import com.velocitypowered.api.proxy.ProxyServer; -import com.velocitypowered.api.proxy.messages.MinecraftChannelIdentifier; -import net.william278.velocitab.Velocitab; -import net.william278.velocitab.api.VelocitabAPI; -import org.slf4j.Logger; - -import java.time.Duration; -import java.util.Optional; - -@Plugin(id = "customnameplates", name = "CustomNameplates", version = "2.2", authors = {"XiaoMoMi"}, - dependencies = { - @Dependency(id = "velocitab") - } -) -public class CustomNameplatesVC { - - private static CustomNameplatesVC instance; - private final ProxyServer server; - private final Logger logger; - private Velocitab tab; - - @Inject - public CustomNameplatesVC(ProxyServer server, Logger logger) { - instance = this; - this.server = server; - this.logger = logger; - } - - public static CustomNameplatesVC getInstance() { - return instance; - } - - @Subscribe - public void onProxyInitialization(ProxyInitializeEvent event) { - server.getChannelRegistrar().register(MinecraftChannelIdentifier.from("customnameplates:cnp")); - Optional optContainer = server.getPluginManager().getPlugin("velocitab"); - optContainer.ifPresent(pluginContainer -> tab = (Velocitab) pluginContainer.getInstance().get()); - } - - @Subscribe - public void onProxyShutdown(ProxyShutdownEvent event) { - server.getChannelRegistrar().unregister(MinecraftChannelIdentifier.from("customnameplates:cnp")); - } - - public ProxyServer getServer() { - return server; - } - - public Logger getLogger() { - return logger; - } - - @Subscribe - @SuppressWarnings("UnstableApiUsage") - public void onPluginMessage(PluginMessageEvent event) { - if (!event.getIdentifier().getId().equals("customnameplates:cnp")) { - return; - } - ByteArrayDataInput dataInput = ByteStreams.newDataInput(event.getData()); - String playerName = dataInput.readUTF(); - Optional optPlayer = server.getPlayer(playerName); - if (optPlayer.isEmpty()) return; - var player = VelocitabAPI.getInstance().getUser(optPlayer.get()); - player.ifPresent(presentPlayer -> presentPlayer.getTeamName(tab).thenAccept(team -> { - server.getScheduler().buildTask(this, () -> { - ByteArrayDataOutput byteArrayDataOutput = ByteStreams.newDataOutput(); - byteArrayDataOutput.writeUTF(playerName); - byteArrayDataOutput.writeUTF(team); - optPlayer.get().getCurrentServer().ifPresent(it -> it.sendPluginMessage(MinecraftChannelIdentifier.from("customnameplates:cnp"), byteArrayDataOutput.toByteArray())); - }).delay(Duration.ofSeconds(1)).schedule(); - })); - } -} diff --git a/src/main/resources/bungee.yml b/src/main/resources/bungee.yml deleted file mode 100644 index 1444876..0000000 --- a/src/main/resources/bungee.yml +++ /dev/null @@ -1,5 +0,0 @@ -name: CustomNameplates -main: net.momirealms.customnameplates.bungeecord.CustomNameplatesBC -version: '2.2.1.2' -author: XiaoMoMi -softDepends: [TAB, BungeeTabListPlus] \ No newline at end of file diff --git a/src/main/resources/config.yml b/src/main/resources/config.yml deleted file mode 100644 index 5956ab5..0000000 --- a/src/main/resources/config.yml +++ /dev/null @@ -1,101 +0,0 @@ -# Do not change -config-version: '22' - -# bStats -metrics: true - -# check update -# 检查更新 -update-checker: true - -# Language -# english/chinese/spanish/russian/turkish/french -lang: english - -# Modules -# 模块 -modules: - nameplates: true - backgrounds: true - bubbles: true - bossbars: true - actionbars: true - images: true - -# Should the plugin hook into other plugins -integrations: - # When enabled, the plugin will automatically place the resource pack - # folder into the resource pack generated by ItemsAdder/Oraxen. - # 启用后,插件会自动将生成的资源包发送到IA或者oraxen插件文件夹内 - ItemsAdder: false - Oraxen: false - # When enabled, the plugin will hook into TAB's team management - # require a restart to apply this - # 启用后,插件会使用TAB插件的队伍管理系统 - TAB: false - # When enabled, plugin will receive team data from TAB on BungeeCord - # You need to install CustomNameplates on BungeeCord too, otherwise you might be kicked from the server if you enabled "create-fake-team" in nameplate.yml - # 启用后,插件会从BungeeCord端接收TAB的队伍信息 - TAB-BC: false - # Receive Team info from Velocitab - # 启用后,插件会从Velocity端接收Velocitab的队伍信息 - Velocitab: false - # TrChat channels - # TrChat频道 - TrChat: false - # VentureChat channels - # VentureChat频道 - VentureChat: false - -resource-pack: - # disable resource pack generation on server start - # 关闭开服时候重新生成资源包 - disable-generation-on-start: false - # extract the shader - # 启用shader - extract-shader: true - # extract the transparent bossbar for 1.20.2+ - # 解压新版本bossbar图片 - extract-bar-image: true - # extract the transparent bars.png for legacy versions - # 解压旧版本bossbar图片 - extract-bar-image-legacy: true - # Your namespace - # 命名空间 - namespace: "nameplates" - # Font Name - # 字体名 - font: "default" - # Customize the folder where png files should be generated. - # This is useful for those who want to keep their resource pack structure in order. - # 自定义文件路径 - image-path: - nameplates: 'font\nameplates\' - backgrounds: 'font\backgrounds\' - images: 'font\images\' - bubbles: 'font\bubbles\' - space-split: 'font\base\' - # The initial character of a font - # 字体的初始字符 - initial-char: '뀁' - # Unicode Support for 1.20+ clients - # This would make your resource pack about 900KB bigger - # 1.20 Unicode 字符支持 - support-1_20-unicodes: true - # Hide scoreboard numbers - # 合并隐藏计分板数字shader - hide-scoreboard-number: true - # Added ItemsAdder text effect shader support - # 合并IA text effect shader - ItemsAdder-text-effect-support: true - -other-settings: - # default width - # 默认文字宽度 - default-character-width: 8 - # disable bossbar/actionbar for bedrock players - # 为基岩玩家禁用bossbar和actionbar - disable-for-bedrock-players: false - # delay x ticks before actionbar/bossbar is sent to players - # 延迟xticks发送bossbar或actionbar - send-delay: 0 \ No newline at end of file diff --git a/src/main/resources/configs/actionbar.yml b/src/main/resources/configs/actionbar.yml deleted file mode 100644 index 86d8e20..0000000 --- a/src/main/resources/configs/actionbar.yml +++ /dev/null @@ -1,10 +0,0 @@ -actionbar_1: - text: '%nameplates_conditional_stamina%%nameplates_static_money_hud%%nameplates_conditional_income_actionbar%' - conditions: - papi-condition: - condition_2: - type: '==' - papi: '%player_gamemode%' - value: 'SURVIVAL' -actionbar_2: - text: '%nameplates_actionbar%' \ No newline at end of file diff --git a/src/main/resources/configs/bossbar.yml b/src/main/resources/configs/bossbar.yml deleted file mode 100644 index 2f3c401..0000000 --- a/src/main/resources/configs/bossbar.yml +++ /dev/null @@ -1,35 +0,0 @@ -basic_info_hud: - color: YELLOW - overlay: PROGRESS - dynamic-text: - - '%nameplates_background_weather% %nameplates_background_position%' - - '%nameplates_background_time% %nameplates_background_hello%' - #- '%nameplates_conditional_region%' - # switch the line of text every 15 seconds - switch-interval: 15 - # Refresh the text every 1 tick - refresh-rate: 1 - conditions: - permission: bossbar.show - -update_info: - color: YELLOW - overlay: PROGRESS - text: '%nameplates_background_update%' - switch-interval: 15 - refresh-rate: 20 - conditions: - permission: customnameplates.admin - papi-condition: - condition_1: - type: '==' - papi: '%nameplates_checkupdate%' - value: 'true' - -halloween: - color: YELLOW - overlay: PROGRESS - text: '%nameplates_nameplate_halloween%' - conditions: - date: - - 10/31 \ No newline at end of file diff --git a/src/main/resources/configs/bubble.yml b/src/main/resources/configs/bubble.yml deleted file mode 100644 index 4971532..0000000 --- a/src/main/resources/configs/bubble.yml +++ /dev/null @@ -1,57 +0,0 @@ -# Armor_Stand / Text_Display (1.19.4+) -mode: Armor_Stand - -# Text display options -text-display-options: - # RGBA - background-color: 0,0,0,0 - has-shadow: false - is-see-through: false - use-default-background-color: false - text-opacity: -1 - -# blacklist channels -# 黑名单聊天频道 -blacklist-channels: - - Private - - Staff - -# Player's default chatting bubbles -# 玩家默认聊天气泡 -default-bubbles: 'chat' - -# Text startFormat when no bubbles equipped -# 无气泡时的默认聊天格式 -default-format: - start: '' - end: '' - -# Add additional prefix and suffix to the text -# 文字的前缀与后缀 -text-prefix: '' -text-suffix: '' - -# Space between two bubbles -# 两行气泡之间的间距 -line-spacing: 0.4 - -# This decides where the bottom line is -# 第一行气泡的起始位置 -bottom-line-Y-offset: 0.4 - -# Seconds -# This decides how long will the chat bubble remain -# 气泡的停留时间 -stay-time: 8 - -# Plugin would send another line of bubble if the text's length exceeds a certain value -# 换行所需的字符数 -characters-per-line: 30 - -# Plugin would ignore player's chat message when its length is over a certain value -# 单次最大文字长度 -max-character-length: 100 - -# Cool down (seconds) -# 发送冷却 -cool-down: 1 \ No newline at end of file diff --git a/src/main/resources/configs/image-width.yml b/src/main/resources/configs/image-width.yml deleted file mode 100644 index 3105a1f..0000000 --- a/src/main/resources/configs/image-width.yml +++ /dev/null @@ -1,2212 +0,0 @@ -# Single unicode character -'默': 8 -# Placeholder -'%img_coin%': 10 -# northern latin -¡: 1 -‰: 7 -­: 3 -·: 1 -₴: 7 -≠: 5 -¿: 5 -×: 5 -Ø: 5 -Þ: 5 -һ: 5 -ð: 5 -ø: 5 -þ: 5 -Α: 5 -Β: 5 -Γ: 5 -Δ: 5 -Ε: 5 -Ζ: 5 -Η: 5 -Θ: 5 -Ι: 3 -Κ: 5 -Λ: 5 -Μ: 5 -Ν: 5 -Ξ: 5 -Ο: 5 -Π: 5 -Ρ: 5 -Σ: 5 -Τ: 5 -Υ: 5 -Φ: 5 -Χ: 5 -Ψ: 5 -Ω: 5 -α: 5 -β: 5 -γ: 5 -δ: 5 -ε: 5 -ζ: 5 -η: 5 -θ: 5 -ι: 2 -κ: 4 -λ: 5 -μ: 5 -ν: 5 -ξ: 5 -ο: 5 -π: 5 -ρ: 5 -ς: 5 -σ: 6 -τ: 5 -υ: 5 -φ: 5 -χ: 5 -ψ: 5 -ω: 7 -Ђ: 7 -Ѕ: 5 -І: 3 -Ј: 5 -Љ: 7 -Њ: 6 -Ћ: 6 -А: 5 -Б: 5 -В: 5 -Г: 5 -Д: 6 -Е: 5 -Ж: 7 -З: 5 -И: 5 -К: 5 -Л: 5 -М: 5 -Н: 5 -О: 5 -П: 5 -Р: 5 -С: 5 -Т: 5 -У: 5 -Ф: 7 -Х: 5 -Ц: 6 -Ч: 5 -Ш: 7 -Щ: 8 -Ъ: 6 -Ы: 7 -Ь: 5 -Э: 5 -Ю: 7 -Я: 5 -а: 5 -б: 5 -в: 5 -г: 4 -д: 6 -е: 5 -ж: 5 -з: 5 -и: 5 -к: 4 -л: 5 -м: 5 -н: 5 -о: 5 -п: 5 -р: 5 -с: 5 -т: 5 -у: 5 -ф: 5 -х: 5 -ц: 5 -ч: 5 -ш: 5 -щ: 6 -ъ: 6 -ы: 6 -ь: 5 -э: 5 -ю: 7 -я: 5 -є: 5 -ѕ: 5 -і: 1 -ј: 5 -љ: 7 -њ: 6 -–: 6 -—: 8 -‘: 2 -’: 2 -“: 4 -”: 4 -„: 4 -…: 7 -⁊: 4 -←: 7 -↑: 5 -→: 7 -↓: 5 -⇄: 7 -+: 5 -Ə: 5 -ə: 5 -ɛ: 5 -ɪ: 3 -Ү: 5 -ү: 5 -Ө: 5 -ө: 5 -ʻ: 1 -ˌ: 1 -;: 1 -ĸ: 4 -ẞ: 5 -ß: 5 -₽: 6 -€: 6 -Ѣ: 6 -ѣ: 6 -Ѵ: 6 -ѵ: 6 -Ӏ: 3 -Ѳ: 5 -ѳ: 5 -⁰: 4 -¹: 3 -³: 4 -⁴: 4 -⁵: 4 -⁶: 4 -⁷: 4 -⁸: 4 -⁹: 4 -⁺: 3 -⁻: 3 -⁼: 3 -⁽: 2 -⁾: 2 -ⁱ: 1 -™: 8 -ʔ: 5 -ʕ: 5 -⧈: 7 -⚔: 7 -☠: 7 -Қ: 6 -қ: 5 -Ғ: 6 -ғ: 5 -Ұ: 5 -ұ: 5 -Ә: 5 -ә: 5 -Җ: 8 -җ: 6 -Ң: 6 -ң: 6 -Һ: 5 -א: 5 -ב: 5 -ג: 4 -ד: 5 -ה: 5 -ו: 1 -ז: 3 -ח: 5 -ט: 5 -י: 1 -כ: 5 -ל: 5 -מ: 5 -ם: 5 -נ: 3 -ן: 1 -ס: 5 -ע: 5 -פ: 5 -ף: 5 -צ: 5 -ץ: 5 -ק: 5 -ר: 5 -¢: 5 -¤: 7 -¥: 5 -©: 7 -®: 7 -µ: 5 -¶: 6 -¼: 7 -½: 7 -¾: 7 -·: 1 -‐: 3 -‚: 2 -†: 5 -‡: 5 -•: 2 -‱: 8 -′: 2 -″: 4 -‴: 6 -‵: 2 -‶: 4 -‷: 6 -‹: 3 -›: 3 -※: 7 -‼: 3 -‽: 5 -⁂: 7 -⁈: 7 -⁉: 7 -⁋: 6 -⁎: 3 -⁏: 1 -⁑: 3 -⁒: 3 -⁗: 8 -℗: 7 -−: 5 -∓: 5 -∞: 7 -☀: 8 -☁: 8 -☈: 7 -Є: 5 -☲: 5 -☵: 5 -☽: 8 -♀: 3 -♂: 5 -⚥: 5 -♠: 5 -♣: 5 -♥: 5 -♦: 5 -♩: 3 -♪: 5 -♫: 7 -♬: 7 -♭: 3 -♮: 3 -♯: 5 -⚀: 7 -⚁: 7 -⚂: 7 -⚃: 7 -⚄: 7 -⚅: 7 -ʬ: 5 -⚡: 5 -⛏: 8 -✔: 6 -❄: 7 -❌: 6 -❤: 7 -⭐: 7 -⸘: 5 -⸮: 5 -⸵: 1 -⸸: 5 -⹁: 1 -⹋: 5 -⥝: 5 -ᘔ: 5 -Ɛ: 5 -߈: 5 -ϛ: 5 -ㄥ: 5 -Ɐ: 5 -ᗺ: 5 -Ɔ: 5 -ᗡ: 5 -Ǝ: 5 -Ⅎ: 5 -⅁: 5 -Ʞ: 5 -Ꞁ: 5 -Ԁ: 5 -Ꝺ: 5 -ᴚ: 5 -⟘: 5 -∩: 5 -Ʌ: 5 -⅄: 5 -ɐ: 5 -ɔ: 5 -ǝ: 5 -ɟ: 4 -ᵷ: 5 -ɥ: 5 -ᴉ: 1 -ɾ: 5 -ʞ: 4 -ꞁ: 2 -ɯ: 5 -ɹ: 5 -ʇ: 3 -ʌ: 5 -ʍ: 5 -ʎ: 5 -Ա: 5 -Բ: 5 -Գ: 6 -Դ: 6 -Զ: 5 -Է: 5 -Թ: 6 -Ժ: 6 -Ի: 5 -Լ: 4 -Խ: 6 -Ծ: 5 -Կ: 5 -Հ: 5 -Ձ: 5 -Ղ: 6 -Ճ: 5 -Մ: 6 -Յ: 5 -Ն: 5 -Շ: 5 -Ո: 5 -Չ: 5 -Ջ: 5 -Ռ: 6 -Ս: 5 -Վ: 5 -Տ: 5 -Ր: 5 -Ց: 5 -Ւ: 4 -Փ: 5 -Ք: 5 -Օ: 5 -Ֆ: 5 -ՙ: 2 -ա: 5 -բ: 5 -գ: 6 -դ: 6 -ե: 5 -զ: 6 -է: 4 -ը: 5 -թ: 5 -ժ: 6 -ի: 5 -լ: 2 -խ: 6 -ծ: 5 -կ: 5 -հ: 5 -ձ: 5 -ղ: 6 -ճ: 5 -մ: 6 -յ: 5 -ն: 6 -շ: 5 -ո: 5 -չ: 5 -պ: 5 -ջ: 5 -ռ: 5 -ս: 5 -վ: 6 -տ: 7 -ր: 5 -ց: 5 -ւ: 4 -փ: 7 -ք: 6 -օ: 5 -ֆ: 5 -և: 6 -ש: 5 -ת: 5 -Ը: 5 -՚: 2 -՛: 2 -՜: 3 -՝: 2 -՞: 3 -՟: 5 -ՠ: 5 -ֈ: 6 -֏: 5 -¯: 5 -ſ: 3 -Ʒ: 5 -ʒ: 5 -Ƿ: 5 -ƿ: 5 -Ȝ: 5 -ȝ: 5 -Ȥ: 5 -ȥ: 5 -˙: 1 -Ꝛ: 5 -ꝛ: 4 -‑: 3 -⅋: 5 -⏏: 5 -⏩: 6 -⏪: 6 -⏭: 7 -⏮: 7 -⏯: 6 -⏴: 3 -⏵: 3 -⏶: 5 -⏷: 5 -⏸: 3 -⏹: 5 -⏺: 5 -⏻: 7 -⏼: 7 -⏽: 1 -⭘: 7 -▲: 5 -▶: 6 -▼: 5 -◀: 6 -●: 4 -◦: 5 -◘: 3 -⚓: 7 -⛨: 7 -IJ: 5 -ij: 4 -lj: 7 -Ꜩ: 7 -ꜩ: 6 -ꜹ: 8 -ꜻ: 8 -ff: 7 -fi: 5 -fl: 6 -ffi: 8 -ſt: 5 -�: 7 -Ե: 5 -Պ: 5 -ᚠ: 4 -ᚢ: 5 -ᚣ: 5 -ᚤ: 5 -ᚥ: 5 -ᚦ: 4 -ᚧ: 4 -ᚨ: 3 -ᚩ: 4 -ᚪ: 4 -ᚫ: 4 -ᚬ: 5 -ᚭ: 3 -ᚮ: 3 -ᚯ: 5 -ᚰ: 5 -ᚱ: 4 -ᚲ: 3 -ᚳ: 4 -ᚴ: 4 -ᚶ: 5 -ᚷ: 5 -ᚸ: 5 -ᚹ: 4 -ᚺ: 5 -ᚻ: 5 -ᚼ: 5 -ᚽ: 3 -ᚾ: 5 -ᚿ: 3 -ᛀ: 5 -ᛁ: 1 -ᛂ: 3 -ᛃ: 6 -ᛄ: 5 -ᛅ: 5 -ᛆ: 3 -ᛇ: 5 -ᛈ: 4 -ᛉ: 5 -ᛊ: 3 -ᛋ: 5 -ᛌ: 1 -ᛍ: 3 -ᛎ: 5 -ᛏ: 5 -ᛐ: 3 -ᛑ: 4 -ᛒ: 4 -ᛓ: 3 -ᛔ: 4 -ᛕ: 4 -ᛖ: 5 -ᛗ: 5 -ᛘ: 5 -ᛙ: 3 -ᛚ: 3 -ᛛ: 4 -ᛜ: 5 -ᛝ: 5 -ᛞ: 5 -ᛟ: 6 -ᛠ: 7 -ᛡ: 5 -ᛢ: 7 -ᛣ: 5 -ᛤ: 5 -ᛥ: 5 -ᛦ: 5 -ᛧ: 1 -ᛨ: 5 -ᛩ: 4 -ᛪ: 7 -᛫: 2 -᛬: 1 -᛭: 5 -ᛮ: 5 -ᛯ: 5 -ᛰ: 5 -ᛱ: 5 -ᛲ: 5 -ᛳ: 7 -ᛴ: 4 -ᛵ: 2 -ᛶ: 5 -ᛷ: 5 -ᛸ: 5 -☺: 5 -☻: 7 -¦: 1 -☹: 5 -ך: 5 -׳: 2 -״: 4 -װ: 3 -ױ: 3 -ײ: 3 -־: 5 -׃: 1 -׆: 3 -´: 2 -¨: 3 -ᴀ: 5 -ʙ: 5 -ᴄ: 5 -ᴅ: 5 -ᴇ: 5 -ꜰ: 5 -ɢ: 5 -ʜ: 5 -ᴊ: 5 -ᴋ: 5 -ʟ: 5 -ᴍ: 5 -ɴ: 5 -ᴏ: 5 -ᴘ: 5 -ꞯ: 5 -ʀ: 5 -ꜱ: 5 -ᴛ: 5 -ᴜ: 5 -ᴠ: 5 -ᴡ: 5 -ʏ: 5 -ᴢ: 5 -§: 5 -ɱ: 5 -ɳ: 6 -ɲ: 6 -ʈ: 3 -ɖ: 6 -ɡ: 5 -ʡ: 5 -ɕ: 5 -ʑ: 5 -ɸ: 5 -ʝ: 5 -ʢ: 5 -ɻ: 6 -ʁ: 5 -ɦ: 5 -ʋ: 5 -ɰ: 5 -ɬ: 4 -ɮ: 5 -ʘ: 5 -ǀ: 1 -ǃ: 1 -ǂ: 5 -ǁ: 3 -ɓ: 5 -ɗ: 6 -ᶑ: 6 -ʄ: 6 -ɠ: 6 -ʛ: 6 -ɧ: 5 -ɫ: 5 -ɨ: 3 -ʉ: 7 -ʊ: 5 -ɘ: 5 -ɵ: 5 -ɤ: 5 -ɜ: 5 -ɞ: 5 -ɑ: 5 -ɒ: 5 -ɚ: 7 -ɝ: 7 -Ɓ: 7 -Ɖ: 6 -Ƒ: 6 -Ʃ: 5 -Ʋ: 5 -Ⴀ: 5 -Ⴁ: 7 -Ⴂ: 7 -Ⴃ: 5 -Ⴄ: 5 -Ⴅ: 7 -Ⴆ: 7 -Ⴇ: 7 -Ⴈ: 5 -Ⴉ: 5 -Ⴊ: 7 -Ⴋ: 6 -Ⴌ: 5 -Ⴍ: 7 -Ⴎ: 7 -Ⴏ: 6 -Ⴐ: 7 -Ⴑ: 5 -Ⴒ: 5 -Ⴓ: 7 -Ⴔ: 5 -Ⴕ: 5 -Ⴖ: 5 -Ⴗ: 5 -Ⴘ: 5 -Ⴙ: 5 -Ⴚ: 5 -Ⴛ: 7 -Ⴜ: 5 -Ⴝ: 5 -Ⴞ: 5 -Ⴟ: 5 -Ⴠ: 7 -Ⴡ: 5 -Ⴢ: 5 -Ⴣ: 5 -Ⴤ: 5 -Ⴥ: 7 -Ⴧ: 5 -Ⴭ: 5 -ა: 5 -ბ: 5 -გ: 5 -დ: 5 -ე: 5 -ვ: 5 -ზ: 5 -თ: 5 -ი: 5 -კ: 5 -ლ: 7 -მ: 5 -ნ: 5 -ო: 5 -პ: 5 -ჟ: 5 -რ: 5 -ს: 5 -ტ: 5 -უ: 5 -ფ: 5 -ქ: 5 -ღ: 5 -ყ: 5 -შ: 5 -ჩ: 5 -ც: 5 -ძ: 5 -წ: 5 -ჭ: 5 -ხ: 5 -ჯ: 4 -ჰ: 5 -ჱ: 6 -ჲ: 5 -ჳ: 5 -ჴ: 5 -ჵ: 5 -ჶ: 5 -ჷ: 5 -ჸ: 5 -ჹ: 5 -ჺ: 5 -჻: 3 -ჼ: 3 -ჽ: 5 -ჾ: 5 -ჿ: 5 -תּ: 5 -שׂ: 5 -פֿ: 5 -פּ: 5 -כּ: 4 -ײַ: 3 -יִ: 1 -וֹ: 2 -וּ: 3 -בֿ: 5 -בּ: 5 -ꜧ: 5 -Ꜧ: 5 -ɺ: 5 -ⱱ: 7 -ʠ: 6 -ʗ: 5 -ʖ: 5 -ɭ: 2 -ɷ: 7 -ɿ: 5 -ʅ: 5 -ʆ: 6 -ʓ: 5 -ʚ: 5 -₪: 7 -₾: 6 -֊: 5 -ⴀ: 5 -ⴁ: 6 -ⴂ: 6 -ⴃ: 5 -ⴄ: 5 -ⴅ: 5 -ⴆ: 6 -ⴡ: 5 -ⴇ: 5 -ⴈ: 5 -ⴉ: 5 -ⴊ: 5 -ⴋ: 5 -ⴌ: 5 -ⴢ: 5 -ⴍ: 5 -ⴎ: 6 -ⴏ: 6 -ⴐ: 5 -ⴑ: 5 -ⴒ: 5 -ⴣ: 5 -ⴓ: 5 -ⴔ: 5 -ⴕ: 5 -ⴖ: 5 -ⴗ: 5 -ⴘ: 5 -ⴙ: 5 -ⴚ: 5 -ⴛ: 5 -ⴜ: 5 -ⴝ: 6 -ⴞ: 5 -ⴤ: 5 -ⴟ: 5 -ⴠ: 5 -ⴥ: 5 -⅛: 8 -⅜: 8 -⅝: 8 -⅞: 8 -⅓: 8 -⅔: 8 -✉: 7 -☂: 7 -☔: 7 -☄: 7 -⛄: 6 -☃: 7 -⌛: 7 -⌚: 7 -⚐: 7 -✎: 7 -❣: 5 -♤: 5 -♧: 5 -♡: 5 -♢: 5 -⛈: 8 -☰: 5 -☱: 5 -☳: 5 -☴: 5 -☶: 5 -☷: 5 -↔: 8 -⇒: 8 -⇏: 8 -⇔: 8 -⇵: 6 -∀: 7 -∃: 5 -∄: 5 -∉: 5 -∋: 5 -∌: 5 -⊂: 5 -⊃: 5 -⊄: 5 -⊅: 5 -∧: 5 -∨: 5 -⊻: 5 -⊼: 5 -⊽: 5 -∥: 3 -≢: 6 -⋆: 3 -∑: 5 -⊤: 5 -⊥: 5 -⊢: 5 -⊨: 5 -≔: 7 -∁: 4 -∴: 5 -∵: 5 -∛: 7 -∜: 7 -∂: 5 -⋃: 5 -⊆: 5 -⊇: 5 -□: 5 -△: 5 -▷: 6 -▽: 5 -◁: 6 -◆: 5 -◇: 5 -○: 5 -◎: 7 -☆: 7 -★: 7 -✘: 6 -₀: 4 -₁: 3 -₂: 4 -₃: 4 -₄: 4 -₅: 4 -₆: 4 -₇: 4 -₈: 4 -₉: 4 -₊: 3 -₋: 3 -₌: 3 -₍: 2 -₎: 2 -∫: 5 -∮: 5 -∝: 6 -⌀: 7 -⌂: 7 -⌘: 7 -〒: 5 -ɼ: 5 -Ƅ: 6 -ƅ: 6 -ẟ: 5 -Ƚ: 6 -ƚ: 3 -ƛ: 5 -Ƞ: 5 -ƞ: 5 -Ɵ: 5 -Ƨ: 5 -ƨ: 5 -ƪ: 5 -Ƹ: 5 -ƹ: 5 -ƻ: 5 -Ƽ: 5 -ƽ: 5 -ƾ: 5 -ȡ: 7 -ȴ: 3 -ȵ: 7 -ȶ: 4 -Ⱥ: 5 -ⱥ: 6 -Ȼ: 5 -ȼ: 5 -Ɇ: 5 -ɇ: 6 -Ⱦ: 5 -ⱦ: 5 -Ɂ: 5 -ɂ: 5 -Ƀ: 6 -Ʉ: 7 -Ɉ: 6 -ɉ: 6 -Ɋ: 6 -ɋ: 6 -Ɍ: 6 -ɍ: 6 -Ɏ: 7 -ɏ: 7 -ẜ: 4 -ẝ: 4 -Ỽ: 5 -ỽ: 4 -Ỿ: 5 -ỿ: 6 -₦: 7 -₩: 7 -₫: 6 -₭: 6 -₮: 5 -₰: 5 -₱: 7 -₲: 5 -₳: 7 -₵: 5 -₶: 5 -₷: 7 -₸: 5 -₹: 5 -₺: 6 -₻: 7 -₼: 5 -₿: 5 -Ꞩ: 6 -ꞩ: 5 -⯪: 7 -⯫: 7 -Ɑ: 5 -✂: 7 -⏳: 7 -⚑: 7 -₠: 5 -₡: 5 -₢: 5 -₣: 6 -₤: 5 -₥: 5 -⅐: 8 -⅑: 8 -⅕: 8 -⅖: 8 -⅗: 8 -⅙: 7 -⅚: 7 -⅟: 6 -↉: 8 -⚗: 7 -#ascent start -Á: 5 -Â: 5 -Ã: 5 -Ä: 5 -Å: 5 -Æ: 9 -Ç: 5 -È: 5 -É: 5 -Ê: 5 -Ë: 5 -Ì: 3 -Í: 3 -Î: 3 -Ï: 3 -Ð: 6 -Ñ: 5 -Ò: 5 -Ó: 5 -Ô: 5 -Õ: 5 -Ö: 5 -Ù: 5 -Ú: 5 -Û: 5 -Ü: 5 -Ý: 5 -à: 5 -á: 5 -â: 5 -ã: 5 -ä: 5 -å: 5 -æ: 9 -ç: 5 -ì: 2 -í: 2 -î: 3 -ï: 3 -ñ: 5 -ò: 5 -ó: 5 -ô: 5 -õ: 5 -ö: 5 -ù: 5 -ú: 5 -û: 5 -ü: 5 -ý: 5 -ÿ: 5 -Ā: 5 -ā: 5 -Ă: 5 -ă: 5 -Ą: 5 -ą: 5 -Ć: 5 -ć: 5 -Ĉ: 5 -ĉ: 5 -Ċ: 5 -ċ: 5 -Č: 5 -č: 5 -Ď: 5 -ď: 7 -Đ: 6 -đ: 6 -Ē: 5 -ē: 5 -Ĕ: 5 -ĕ: 5 -Ė: 5 -ė: 5 -Ę: 5 -ę: 5 -Ě: 5 -ě: 5 -Ĝ: 5 -ĝ: 5 -Ḡ: 5 -ḡ: 5 -Ğ: 5 -ğ: 5 -Ġ: 5 -ġ: 5 -Ģ: 5 -ģ: 5 -Ĥ: 5 -ĥ: 5 -Ħ: 7 -ħ: 6 -Ĩ: 4 -ĩ: 4 -Ī: 3 -ī: 3 -Ĭ: 4 -ĭ: 4 -Į: 3 -į: 2 -İ: 3 -ı: 1 -Ĵ: 5 -ĵ: 5 -Ķ: 5 -ķ: 4 -Ĺ: 5 -ĺ: 2 -Ļ: 5 -ļ: 2 -Ľ: 5 -ľ: 3 -Ŀ: 5 -ŀ: 3 -Ł: 6 -ł: 4 -Ń: 5 -ń: 5 -Ņ: 5 -ņ: 5 -Ň: 5 -ň: 5 -Ŋ: 5 -ŋ: 5 -Ō: 5 -ō: 5 -Ŏ: 5 -ŏ: 5 -Ő: 5 -ő: 5 -Œ: 9 -œ: 9 -Ŕ: 5 -ŕ: 5 -Ŗ: 5 -ŗ: 5 -Ř: 5 -ř: 5 -Ś: 5 -ś: 5 -Ŝ: 5 -ŝ: 5 -Ş: 5 -ş: 5 -Š: 5 -š: 5 -Ţ: 5 -ţ: 3 -Ť: 5 -ť: 4 -Ŧ: 5 -ŧ: 3 -Ũ: 5 -ũ: 5 -Ū: 5 -ū: 5 -Ŭ: 5 -ŭ: 5 -Ů: 5 -ů: 5 -Ű: 5 -ű: 5 -Ų: 5 -ų: 5 -Ŵ: 5 -ŵ: 5 -Ŷ: 5 -ŷ: 5 -Ÿ: 5 -Ź: 5 -ź: 5 -Ż: 5 -ż: 5 -Ž: 5 -ž: 5 -Ǽ: 9 -ǽ: 9 -Ǿ: 5 -ǿ: 5 -Ș: 5 -ș: 5 -Ț: 5 -ț: 3 -Ά: 7 -Έ: 7 -Ή: 7 -Ί: 5 -Ό: 7 -Ύ: 7 -Ώ: 7 -ΐ: 5 -Ϊ: 3 -Ϋ: 5 -ά: 5 -έ: 5 -ή: 5 -ί: 2 -ΰ: 5 -ϊ: 3 -ϋ: 5 -ό: 5 -ύ: 5 -ώ: 7 -Ѐ: 5 -Ё: 5 -Ѓ: 5 -Ї: 3 -Ќ: 5 -Ѝ: 5 -Ў: 5 -Й: 5 -й: 5 -ѐ: 5 -ё: 5 -ђ: 6 -ѓ: 4 -ї: 3 -ћ: 6 -ќ: 4 -ѝ: 5 -ў: 5 -џ: 5 -Ґ: 5 -ґ: 5 -Ḃ: 5 -ḃ: 5 -Ḋ: 5 -ḋ: 5 -Ḟ: 5 -ḟ: 4 -Ḣ: 5 -ḣ: 5 -Ḱ: 5 -ḱ: 4 -Ṁ: 5 -ṁ: 5 -Ṗ: 5 -ṗ: 5 -Ṡ: 5 -ṡ: 5 -Ṫ: 5 -ṫ: 3 -Ẁ: 5 -ẁ: 5 -Ẃ: 5 -ẃ: 5 -Ẅ: 5 -ẅ: 5 -Ỳ: 5 -ỳ: 5 -è: 5 -é: 5 -ê: 5 -ë: 5 -ʼn: 7 -ǧ: 5 -ǫ: 5 -Џ: 5 -ḍ: 5 -ḥ: 5 -ṛ: 5 -ṭ: 3 -Ẓ: 5 -Ị: 3 -ị: 1 -Ọ: 5 -ọ: 5 -Ụ: 5 -ụ: 5 -№: 9 -ȇ: 5 -Ɣ: 5 -ɣ: 5 -ʃ: 5 -⁇: 9 -DZ: 9 -Dz: 9 -dz: 9 -DŽ: 9 -Dž: 9 -dž: 9 -LJ: 9 -Lj: 9 -NJ: 9 -Nj: 9 -nj: 9 -ℹ: 9 -ᵫ: 9 -Ꜳ: 9 -ꜳ: 9 -Ꜵ: 9 -ꜵ: 9 -Ꜷ: 9 -ꜷ: 9 -Ꜹ: 9 -Ꜻ: 9 -Ꜽ: 9 -ꜽ: 8 -Ꝏ: 9 -ꝏ: 9 -Ꝡ: 5 -ꝡ: 5 -ffl: 9 -st: 9 -ᚡ: 4 -ᚵ: 4 -Ơ: 7 -ơ: 7 -Ư: 7 -ư: 7 -Ắ: 5 -ắ: 5 -Ấ: 5 -ấ: 5 -Ế: 5 -ế: 5 -ố: 5 -Ớ: 7 -ớ: 7 -Ứ: 7 -ứ: 7 -Ằ: 5 -ằ: 5 -Ầ: 5 -ầ: 5 -Ề: 5 -ề: 5 -ồ: 5 -Ờ: 7 -ờ: 7 -Ừ: 7 -ừ: 7 -Ả: 5 -ả: 5 -Ẳ: 5 -ẳ: 5 -Ẩ: 5 -ẩ: 5 -Ẻ: 5 -ẻ: 5 -ổ: 5 -Ở: 7 -Ể: 5 -ể: 5 -Ỉ: 3 -ỉ: 3 -Ỏ: 5 -ỏ: 5 -Ổ: 5 -ở: 7 -Ủ: 5 -ủ: 5 -Ử: 7 -ử: 7 -Ỷ: 5 -ỷ: 5 -Ạ: 5 -ạ: 5 -Ặ: 5 -ặ: 5 -Ậ: 5 -ậ: 5 -Ẹ: 5 -ẹ: 5 -Ệ: 5 -ệ: 5 -Ộ: 5 -ộ: 5 -Ợ: 7 -ợ: 7 -Ự: 7 -ự: 7 -Ỵ: 5 -ỵ: 5 -Ố: 5 -ƕ: 8 -Ẫ: 5 -ẫ: 5 -Ỗ: 5 -ỗ: 5 -ữ: 7 -☞: 9 -☜: 9 -☮: 9 -Ẵ: 5 -ẵ: 5 -Ẽ: 5 -ẽ: 5 -Ễ: 5 -ễ: 5 -Ồ: 5 -Ỡ: 7 -ỡ: 7 -Ữ: 7 -Ỹ: 5 -ỹ: 5 -Ҙ: 5 -ҙ: 5 -Ҡ: 6 -ҡ: 5 -Ҫ: 5 -ҫ: 5 -Ƕ: 8 -⚠: 9 -⓪: 9 -①: 9 -②: 9 -③: 9 -④: 9 -⑤: 9 -⑥: 9 -⑦: 9 -⑧: 9 -⑨: 9 -⑩: 9 -⑪: 9 -⑫: 9 -⑬: 9 -⑭: 9 -⑮: 9 -⑯: 9 -⑰: 9 -⑱: 9 -⑲: 9 -⑳: 9 -Ⓐ: 9 -Ⓑ: 9 -Ⓒ: 9 -Ⓓ: 9 -Ⓔ: 9 -Ⓕ: 9 -Ⓖ: 9 -Ⓗ: 9 -Ⓘ: 9 -Ⓙ: 9 -Ⓚ: 9 -Ⓛ: 9 -Ⓜ: 9 -Ⓝ: 9 -Ⓞ: 9 -Ⓟ: 9 -Ⓠ: 9 -Ⓡ: 9 -Ⓢ: 9 -Ⓣ: 9 -Ⓤ: 9 -Ⓥ: 9 -Ⓦ: 9 -Ⓧ: 9 -Ⓨ: 9 -Ⓩ: 9 -ⓐ: 9 -ⓑ: 9 -ⓒ: 9 -ⓓ: 9 -ⓔ: 9 -ⓕ: 9 -ⓖ: 9 -ⓗ: 9 -ⓘ: 9 -ⓙ: 9 -ⓚ: 9 -ⓛ: 9 -ⓜ: 9 -ⓝ: 9 -ⓞ: 9 -ⓟ: 9 -ⓠ: 9 -ⓡ: 9 -ⓢ: 9 -ⓣ: 9 -ⓤ: 9 -ⓥ: 9 -ⓦ: 9 -ⓧ: 9 -ⓨ: 9 -ⓩ: 9 -̧: 2 -ʂ: 5 -ʐ: 6 -ɶ: 9 -Ǎ: 5 -ǎ: 5 -Ǟ: 5 -ǟ: 5 -Ǻ: 5 -ǻ: 5 -Ȃ: 5 -ȃ: 5 -Ȧ: 5 -ȧ: 5 -Ǡ: 5 -ǡ: 5 -Ḁ: 5 -ḁ: 5 -Ȁ: 5 -ȁ: 5 -Ḇ: 5 -ḇ: 5 -Ḅ: 5 -ḅ: 5 -ᵬ: 7 -Ḉ: 5 -ḉ: 5 -Ḑ: 5 -ḑ: 5 -Ḓ: 5 -ḓ: 5 -Ḏ: 5 -ḏ: 5 -Ḍ: 5 -ᵭ: 7 -Ḕ: 5 -ḕ: 5 -Ḗ: 5 -ḗ: 5 -Ḙ: 5 -ḙ: 5 -Ḝ: 5 -ḝ: 5 -Ȩ: 5 -ȩ: 5 -Ḛ: 5 -ḛ: 5 -Ȅ: 5 -ȅ: 5 -Ȇ: 5 -ᵮ: 5 -Ǵ: 5 -ǵ: 5 -Ǧ: 5 -Ḧ: 5 -ḧ: 5 -Ḩ: 5 -ḩ: 5 -Ḫ: 5 -ḫ: 5 -Ȟ: 5 -ȟ: 5 -Ḥ: 5 -ẖ: 5 -Ḯ: 3 -ḯ: 3 -Ȋ: 4 -ȋ: 4 -Ǐ: 3 -ǐ: 3 -Ȉ: 5 -ȉ: 5 -Ḭ: 4 -ḭ: 4 -ǰ: 5 -ȷ: 5 -Ǩ: 5 -ǩ: 4 -Ḳ: 5 -ḳ: 4 -Ḵ: 5 -ḵ: 4 -Ḻ: 5 -ḻ: 3 -Ḽ: 5 -ḽ: 3 -Ḷ: 5 -ḷ: 2 -Ḹ: 5 -ḹ: 3 -Ɫ: 7 -Ḿ: 5 -ḿ: 5 -Ṃ: 5 -ṃ: 5 -ᵯ: 7 -Ṅ: 5 -ṅ: 5 -Ṇ: 5 -ṇ: 5 -Ṋ: 5 -ṋ: 5 -Ǹ: 5 -ǹ: 5 -Ṉ: 5 -ṉ: 5 -ᵰ: 7 -Ǭ: 5 -ǭ: 5 -Ȭ: 5 -ȭ: 5 -Ṍ: 5 -ṍ: 5 -Ṏ: 5 -ṏ: 5 -Ṑ: 5 -ṑ: 5 -Ṓ: 5 -ṓ: 5 -Ȏ: 5 -ȏ: 5 -Ȫ: 5 -ȫ: 5 -Ǒ: 5 -ǒ: 5 -Ȯ: 5 -ȯ: 5 -Ȱ: 5 -ȱ: 5 -Ȍ: 5 -ȍ: 5 -Ǫ: 5 -Ṕ: 5 -ṕ: 5 -ᵱ: 7 -Ȓ: 5 -ȓ: 5 -Ṙ: 5 -ṙ: 5 -Ṝ: 5 -ṝ: 5 -Ṟ: 5 -ṟ: 5 -Ȑ: 5 -ȑ: 5 -Ṛ: 5 -ᵳ: 7 -ᵲ: 7 -Ṥ: 5 -ṥ: 5 -Ṧ: 5 -ṧ: 5 -Ṣ: 5 -ṣ: 5 -Ṩ: 5 -ṩ: 5 -ᵴ: 7 -Ṱ: 5 -ṱ: 3 -Ṯ: 5 -ṯ: 3 -Ṭ: 5 -ẗ: 3 -ᵵ: 5 -Ṳ: 5 -ṳ: 5 -Ṷ: 5 -ṷ: 5 -Ṹ: 5 -ṹ: 5 -Ṻ: 5 -ṻ: 5 -Ǔ: 5 -ǔ: 5 -Ǖ: 5 -ǖ: 5 -Ǘ: 5 -ǘ: 5 -Ǚ: 5 -ǚ: 5 -Ǜ: 5 -ǜ: 5 -Ṵ: 5 -ṵ: 5 -Ȕ: 5 -ȕ: 5 -Ȗ: 5 -Ṿ: 5 -ṿ: 5 -Ṽ: 5 -ṽ: 5 -Ẇ: 5 -ẇ: 5 -Ẉ: 5 -ẉ: 5 -ẘ: 5 -Ẍ: 5 -ẍ: 5 -Ẋ: 5 -ẋ: 5 -Ȳ: 5 -ȳ: 5 -Ẏ: 5 -ẏ: 5 -ẙ: 5 -Ẕ: 5 -ẕ: 5 -Ẑ: 5 -ẑ: 5 -ẓ: 5 -ᵶ: 5 -Ǯ: 5 -ǯ: 5 -ẛ: 3 -Ꜿ: 5 -ꜿ: 5 -Ǣ: 9 -ǣ: 9 -ᵺ: 8 -ỻ: 6 -ᴂ: 9 -ᴔ: 9 -ꭣ: 9 -ȸ: 9 -ʣ: 9 -ʥ: 9 -ʤ: 9 -ʩ: 9 -ʪ: 7 -ʫ: 7 -ȹ: 9 -ʨ: 8 -ʦ: 8 -ʧ: 7 -ꭐ: 5 -ꭑ: 5 -₧: 9 -Ỻ: 7 -אַ: 5 -אָ: 5 -ƀ: 6 -Ƃ: 5 -ƃ: 5 -Ƈ: 6 -ƈ: 6 -Ɗ: 7 -Ƌ: 5 -ƌ: 5 -Ɠ: 6 -Ǥ: 6 -ǥ: 6 -Ɨ: 3 -Ɩ: 3 -ɩ: 2 -Ƙ: 5 -ƙ: 4 -Ɲ: 6 -Ƥ: 7 -ƥ: 5 -ɽ: 5 -Ʀ: 5 -Ƭ: 6 -ƭ: 3 -ƫ: 3 -Ʈ: 5 -ȗ: 5 -Ʊ: 5 -Ɯ: 5 -Ƴ: 6 -ƴ: 7 -Ƶ: 5 -ƶ: 5 -Ƣ: 7 -ƣ: 7 -Ȣ: 5 -ȣ: 5 -ʭ: 5 -ʮ: 6 -ʯ: 7 -ﬔ: 9 -ﬕ: 9 -ﬗ: 9 -ﬖ: 9 -ﬓ: 9 -Ӑ: 5 -ӑ: 5 -Ӓ: 5 -ӓ: 5 -Ӷ: 5 -ӷ: 4 -Ҕ: 5 -ҕ: 4 -Ӗ: 5 -ӗ: 5 -Ҽ: 7 -ҽ: 7 -Ҿ: 7 -ҿ: 7 -Ӛ: 5 -ӛ: 5 -Ӝ: 7 -ӝ: 5 -Ӂ: 7 -ӂ: 5 -Ӟ: 5 -ӟ: 5 -Ӣ: 5 -ӣ: 5 -Ӥ: 5 -ӥ: 5 -Ӧ: 5 -ӧ: 5 -Ӫ: 5 -ӫ: 5 -Ӱ: 5 -ӱ: 5 -Ӯ: 5 -ӯ: 5 -Ӳ: 5 -ӳ: 5 -Ӵ: 5 -ӵ: 5 -Ӹ: 7 -ӹ: 6 -Ӭ: 5 -ӭ: 5 -Ѷ: 6 -ѷ: 6 -Ӕ: 9 -Ӻ: 6 -Ԃ: 8 -Ꚃ: 5 -Ꚁ: 6 -Ꚉ: 6 -Ԫ: 9 -Ԭ: 6 -Ꚅ: 9 -Ԅ: 7 -Ԑ: 5 -Ӡ: 5 -Ԇ: 6 -Ҋ: 6 -Ӄ: 5 -Ҟ: 6 -Ҝ: 5 -Ԟ: 5 -Ԛ: 5 -Ӆ: 6 -Ԯ: 6 -Ԓ: 5 -Ԡ: 8 -Ԉ: 8 -Ԕ: 8 -Ӎ: 6 -Ӊ: 6 -Ԩ: 6 -Ӈ: 5 -Ҥ: 6 -Ԣ: 8 -Ԋ: 8 -Ҩ: 6 -Ԥ: 6 -Ҧ: 8 -Ҏ: 5 -Ԗ: 8 -Ԍ: 5 -Ꚑ: 5 -Ҭ: 5 -Ꚋ: 6 -Ꚍ: 5 -Ԏ: 6 -Ҳ: 6 -Ӽ: 5 -Ӿ: 5 -Ԧ: 6 -Ꚕ: 5 -Ҵ: 7 -Ꚏ: 6 -Ҷ: 6 -Ӌ: 5 -Ҹ: 5 -Ꚓ: 6 -Ꚗ: 8 -Ꚇ: 5 -Ҍ: 6 -Ԙ: 9 -Ԝ: 5 -ӕ: 9 -ӻ: 5 -ԃ: 8 -ꚃ: 5 -ꚁ: 6 -ꚉ: 6 -ԫ: 9 -ԭ: 6 -ꚅ: 8 -ԅ: 7 -ԑ: 5 -ӡ: 5 -ԇ: 6 -ҋ: 6 -ӄ: 4 -ҟ: 5 -ҝ: 5 -ԟ: 5 -ԛ: 5 -ӆ: 6 -ԯ: 6 -ԓ: 5 -ԡ: 8 -ԉ: 8 -ԕ: 8 -ӎ: 6 -ӊ: 6 -ԩ: 6 -ӈ: 5 -ҥ: 6 -ԣ: 8 -ԋ: 8 -ҩ: 6 -ԥ: 6 -ҧ: 8 -ҏ: 5 -ԗ: 8 -ԍ: 5 -ꚑ: 5 -ҭ: 5 -ꚋ: 6 -ꚍ: 5 -ԏ: 6 -ҳ: 6 -ӽ: 5 -ӿ: 5 -ԧ: 6 -ꚕ: 5 -ҵ: 6 -ꚏ: 5 -ҷ: 6 -ӌ: 5 -ҹ: 5 -ꚓ: 6 -ꚗ: 6 -ꚇ: 5 -ҍ: 6 -ԙ: 9 -ԝ: 5 -Ἀ: 8 -ἀ: 5 -Ἁ: 8 -ἁ: 5 -Ἂ: 9 -ἂ: 5 -Ἃ: 9 -ἃ: 5 -Ἄ: 9 -ἄ: 5 -Ἅ: 9 -ἅ: 6 -Ἆ: 8 -ἆ: 5 -Ἇ: 8 -ἇ: 5 -Ὰ: 8 -ὰ: 5 -Ᾰ: 5 -ᾰ: 5 -Ᾱ: 5 -ᾱ: 5 -Ά: 8 -ά: 5 -ᾈ: 8 -ᾀ: 5 -ᾉ: 8 -ᾁ: 5 -ᾊ: 9 -ᾂ: 5 -ᾋ: 9 -ᾃ: 5 -ᾌ: 9 -ᾄ: 5 -ᾍ: 9 -ᾅ: 5 -ᾎ: 8 -ᾆ: 5 -ᾏ: 8 -ᾇ: 5 -ᾼ: 5 -ᾴ: 5 -ᾶ: 5 -ᾷ: 5 -ᾲ: 5 -ᾳ: 5 -Ἐ: 8 -ἐ: 5 -Ἑ: 8 -ἑ: 5 -Ἒ: 9 -ἒ: 5 -Ἓ: 9 -ἓ: 5 -Ἔ: 9 -ἔ: 5 -Ἕ: 9 -ἕ: 5 -Ὲ: 8 -Έ: 8 -ὲ: 5 -έ: 5 -Ἠ: 8 -ἠ: 5 -Ὴ: 8 -ὴ: 5 -Ἡ: 8 -ἡ: 5 -Ἢ: 9 -ἢ: 5 -Ἣ: 9 -ἣ: 5 -Ἤ: 9 -ἤ: 5 -Ἥ: 9 -ἥ: 5 -Ἦ: 8 -ἦ: 5 -Ἧ: 8 -ἧ: 5 -ᾘ: 8 -ᾐ: 5 -ᾙ: 8 -ᾑ: 5 -ᾚ: 9 -ᾒ: 5 -ᾛ: 9 -ᾓ: 5 -ᾜ: 9 -ᾔ: 5 -ᾝ: 9 -ᾕ: 5 -ᾞ: 8 -ᾖ: 5 -ᾟ: 8 -ᾗ: 5 -Ή: 8 -ή: 5 -ῌ: 5 -ῃ: 5 -ῂ: 5 -ῄ: 5 -ῆ: 5 -ῇ: 5 -Ὶ: 6 -ὶ: 3 -Ί: 6 -ί: 2 -Ἰ: 6 -ἰ: 3 -Ἱ: 9 -ἱ: 2 -Ἲ: 8 -ἲ: 5 -Ἳ: 7 -ἳ: 5 -Ἴ: 8 -ἴ: 5 -Ἵ: 7 -ἵ: 5 -Ἶ: 6 -ἶ: 4 -Ἷ: 6 -ἷ: 4 -Ῐ: 4 -ῐ: 4 -Ῑ: 3 -ῑ: 3 -ῒ: 3 -ΐ: 3 -ῖ: 4 -ῗ: 4 -Ὸ: 8 -ὸ: 5 -Ό: 8 -ό: 5 -Ὀ: 8 -ὀ: 5 -Ὁ: 8 -ὁ: 5 -Ὂ: 9 -ὂ: 5 -Ὃ: 9 -ὃ: 5 -Ὄ: 9 -ὄ: 5 -Ὅ: 9 -ὅ: 5 -Ῥ: 8 -ῤ: 5 -ῥ: 5 -Ὺ: 8 -ὺ: 5 -Ύ: 8 -ύ: 5 -Ὑ: 8 -ὑ: 5 -Ὓ: 9 -ὓ: 5 -Ὕ: 9 -ὕ: 5 -Ὗ: 8 -ὗ: 5 -Ῠ: 5 -ῠ: 5 -Ῡ: 5 -ῡ: 5 -ϓ: 9 -ϔ: 6 -ῢ: 5 -ΰ: 5 -ῧ: 5 -ὐ: 5 -ὒ: 5 -ὔ: 5 -ῦ: 5 -ὖ: 5 -Ὼ: 8 -ὼ: 7 -Ώ: 8 -ώ: 7 -Ὠ: 8 -ὠ: 7 -Ὡ: 8 -ὡ: 7 -Ὢ: 9 -ὢ: 7 -Ὣ: 9 -ὣ: 7 -Ὤ: 9 -ὤ: 7 -Ὥ: 9 -ὥ: 7 -Ὦ: 8 -ὦ: 7 -Ὧ: 9 -ὧ: 7 -ᾨ: 8 -ᾠ: 7 -ᾩ: 8 -ᾡ: 7 -ᾪ: 9 -ᾢ: 7 -ᾫ: 9 -ᾣ: 7 -ᾬ: 9 -ᾤ: 7 -ᾭ: 9 -ᾥ: 7 -ᾮ: 8 -ᾦ: 7 -ᾯ: 9 -ᾧ: 7 -ῼ: 5 -ῳ: 7 -ῲ: 7 -ῴ: 7 -ῶ: 7 -ῷ: 7 -☯: 9 -☐: 9 -☑: 9 -☒: 9 -ƍ: 5 -ƺ: 5 -Ȿ: 5 -ȿ: 5 -Ɀ: 5 -ɀ: 5 -ᶀ: 5 -Ꞔ: 5 -ꞔ: 5 -ᶁ: 6 -ᶂ: 4 -ᶃ: 6 -ꞕ: 6 -ᶄ: 5 -ᶅ: 2 -ᶆ: 6 -ᶇ: 6 -ᶈ: 5 -ᶉ: 5 -ᶊ: 5 -ᶋ: 5 -ᶌ: 5 -ᶍ: 6 -Ᶎ: 6 -ᶎ: 6 -ᶏ: 6 -ᶐ: 6 -ᶒ: 6 -ᶓ: 6 -ᶔ: 5 -ᶕ: 6 -ᶖ: 2 -ᶗ: 6 -ᶘ: 5 -ᶙ: 6 -ᶚ: 5 -ẚ: 5 -⅒: 9 -⅘: 9 -₨: 9 -₯: 9 \ No newline at end of file diff --git a/src/main/resources/configs/nameplate.yml b/src/main/resources/configs/nameplate.yml deleted file mode 100644 index 5352db2..0000000 --- a/src/main/resources/configs/nameplate.yml +++ /dev/null @@ -1,64 +0,0 @@ -# mode: Team / Armor_Stand / Text_Display(1.19.4+) / Disable -# 模式 队伍/盔甲架/文字/禁用 -mode: Team - -# Disable this if you meet "Team xxx already exists" when you are running a BungeeCord server -# You can also install BungeeTabListPlus or TAB on BungeeCord side to fix -# 当你使用BungeeCord服务端的时候请禁用此选项或在bc上安装BungeeTabListPlus或TAB -create-fake-team: true - -# This decides what %nameplates_prefix/suffix% would return -# 玩家名变量,这会影响铭牌的长度判断 -player-name: '%player_name%' - -# The duration (in seconds) that the nameplate preview will last for. -# 预览持续时间 -preview-duration: 5 - -# The default nameplate. -# 默认铭牌, none代表无 -# "none" represents no default nameplate. -default-nameplate: 'none' - -# Prefix & Suffix -# Rejoin the server to apply the changes -# 前缀和后缀,重新进入服务器以应用更改 -prefix: '' -suffix: '' - -# Should prefix & suffix be hidden when equipping nameplate -# 前后缀是否在佩戴铭牌的时候隐藏 -hide-prefix-when-equipping-nameplate: false -hide-suffix-when-equipping-nameplate: false - -armor_stand: - text_1: - text: '%nameplates_prefix%%nameplates_descent_name%%nameplates_suffix%' - vertical-offset: -1 - text_2: - text: '%vault_prefix%' - vertical-offset: -0.7 - conditions: - papi-condition: - condition_1: - type: '!=' - papi: '%vault_prefix%' - value: '' - -# Using text display is not recommended because it lacks the smooth movement. -# This limitation is due to the way the client handles the packet, and it cannot be resolved from the server side. -text_display: - text_1: - text: '%nameplates_prefix%%nameplates_descent_name%%nameplates_suffix%' - vertical-offset: -1 - options: - is-see-through: false - text_2: - text: '%vault_prefix%' - vertical-offset: -0.7 - conditions: - papi-condition: - condition_1: - type: '!=' - papi: '%vault_prefix%' - value: '' \ No newline at end of file diff --git a/src/main/resources/database.yml b/src/main/resources/database.yml deleted file mode 100644 index 41b073a..0000000 --- a/src/main/resources/database.yml +++ /dev/null @@ -1,42 +0,0 @@ -config-version: '15' - -# YAML/MySQL/MariaDB -# 存储模式 -data-storage-method: YAML - -# MySQL settings -MySQL: - host: localhost - port: 3306 - user: root - password: password - database: minecraft - table: nameplates_data - properties: - cachePrepStmts: 'true' - prepStmtCacheSize: '250' - prepStmtCacheSqlLimit: '2048' - useSSL: 'false' - verifyServerCertificate: 'false' - Pool-Settings: - maximum-pool-size: 10 - minimum-idle: 10 - maximum-lifetime: 180000 - idle-timeout: 60000 - -MariaDB: - host: localhost - port: 3306 - user: root - password: password - database: minecraft - table: nameplates_data - properties: - cachePrepStmts: 'true' - prepStmtCacheSize: '250' - prepStmtCacheSqlLimit: '2048' - Pool-Settings: - maximum-pool-size: 10 - minimum-idle: 10 - maximum-lifetime: 180000 - idle-timeout: 60000 \ No newline at end of file diff --git a/src/main/resources/plugin.yml b/src/main/resources/plugin.yml deleted file mode 100644 index fd76ad2..0000000 --- a/src/main/resources/plugin.yml +++ /dev/null @@ -1,82 +0,0 @@ -name: CustomNameplates -version: '${version}' -main: net.momirealms.customnameplates.CustomNameplates -api-version: 1.17 -authors: [ XiaoMoMi ] -folia-supported: true -depend: - - ProtocolLib - - PlaceholderAPI -softdepend: - - ItemsAdder - - TAB - - Oraxen - - TrChat - - MagicCosmetics -commands: - customnameplates: - usage: /customnameplates - aliases: - - nameplates - chatbubbles: - usage: /chatbubbles - aliases: - - bubbles - -permissions: - nameplates.*: - description: Gives access to all nameplates commands - children: - nameplates.reload: true - nameplates.forceequip: true - nameplates.forceunequip: true - nameplates.help: true - nameplates.unequip: true - nameplates.preview: true - nameplates.list: true - nameplates.about: true - - nameplates.reload: - default: op - nameplates.forceequip: - default: op - nameplates.forceunequip: - default: op - nameplates.help: - default: op - nameplates.about: - default: op - nameplates.unequip: - default: true - nameplates.preview: - default: true - nameplates.list: - default: true - nameplates.equip: - default: true - - bubbles.*: - description: Gives access to all bubbles commands - children: - bubbles.unequip: true - bubbles.equip: true - bubbles.help: true - bubbles.forceequip: true - bubbles.forceunequip: true - bubbles.list: true - bubbles.use: true - - bubbles.help: - default: op - bubbles.equip: - default: true - bubbles.unequip: - default: true - bubbles.list: - default: true - bubbles.forceequip: - default: op - bubbles.forceunequip: - default: op - bubbles.use: - default: true \ No newline at end of file diff --git a/src/main/resources/templates/default.json b/src/main/resources/templates/default.json deleted file mode 100644 index a6f291d..0000000 --- a/src/main/resources/templates/default.json +++ /dev/null @@ -1,191 +0,0 @@ -{ - "providers": [ - { - "type": "space", - "advances": { - " ": 4, - "\u200c": 0 - } - }, - { - "type": "bitmap", - "file": "minecraft:font/nonlatin_european.png", - "ascent": %ascent%, - "chars": [ - "\u00a1\u2030\u00ad\u00b7\u20b4\u2260\u00bf\u00d7\u00d8\u00de\u04bb\u00f0\u00f8\u00fe\u0391\u0392", - "\u0393\u0394\u0395\u0396\u0397\u0398\u0399\u039a\u039b\u039c\u039d\u039e\u039f\u03a0\u03a1\u03a3", - "\u03a4\u03a5\u03a6\u03a7\u03a8\u03a9\u03b1\u03b2\u03b3\u03b4\u03b5\u03b6\u03b7\u03b8\u03b9\u03ba", - "\u03bb\u03bc\u03bd\u03be\u03bf\u03c0\u03c1\u03c2\u03c3\u03c4\u03c5\u03c6\u03c7\u03c8\u03c9\u0402", - "\u0405\u0406\u0408\u0409\u040a\u040b\u0410\u0411\u0412\u0413\u0414\u0415\u0416\u0417\u0418\u041a", - "\u041b\u041c\u041d\u041e\u041f\u0420\u0421\u0422\u0423\u0424\u0425\u0426\u0427\u0428\u0429\u042a", - "\u042b\u042c\u042d\u042e\u042f\u0430\u0431\u0432\u0433\u0434\u0435\u0436\u0437\u0438\u043a\u043b", - "\u043c\u043d\u043e\u043f\u0440\u0441\u0442\u0443\u0444\u0445\u0446\u0447\u0448\u0449\u044a\u044b", - "\u044c\u044d\u044e\u044f\u0454\u0455\u0456\u0458\u0459\u045a\u2013\u2014\u2018\u2019\u201c\u201d", - "\u201e\u2026\u204a\u2190\u2191\u2192\u2193\u21c4\uff0b\u018f\u0259\u025b\u026a\u04ae\u04af\u04e8", - "\u04e9\u02bb\u02cc\u037e\u0138\u1e9e\u00df\u20bd\u20ac\u0462\u0463\u0474\u0475\u04c0\u0472\u0473", - "\u2070\u00b9\u00b3\u2074\u2075\u2076\u2077\u2078\u2079\u207a\u207b\u207c\u207d\u207e\u2071\u2122", - "\u0294\u0295\u29c8\u2694\u2620\u049a\u049b\u0492\u0493\u04b0\u04b1\u04d8\u04d9\u0496\u0497\u04a2", - "\u04a3\u04ba\u05d0\u05d1\u05d2\u05d3\u05d4\u05d5\u05d6\u05d7\u05d8\u05d9\u05db\u05dc\u05de\u05dd", - "\u05e0\u05df\u05e1\u05e2\u05e4\u05e3\u05e6\u05e5\u05e7\u05e8\u00a2\u00a4\u00a5\u00a9\u00ae\u00b5", - "\u00b6\u00bc\u00bd\u00be\u0387\u2010\u201a\u2020\u2021\u2022\u2031\u2032\u2033\u2034\u2035\u2036", - "\u2037\u2039\u203a\u203b\u203c\u203d\u2042\u2048\u2049\u204b\u204e\u204f\u2051\u2052\u2057\u2117", - "\u2212\u2213\u221e\u2600\u2601\u2608\u0404\u2632\u2635\u263d\u2640\u2642\u26a5\u2660\u2663\u2665", - "\u2666\u2669\u266a\u266b\u266c\u266d\u266e\u266f\u2680\u2681\u2682\u2683\u2684\u2685\u02ac\u26a1", - "\u26cf\u2714\u2744\u274c\u2764\u2b50\u2e18\u2e2e\u2e35\u2e38\u2e41\u2e4b\u295d\u1614\u0190\u07c8", - "\u03db\u3125\u2c6f\u15fa\u0186\u15e1\u018e\u2132\u2141\ua7b0\ua780\u0500\ua779\u1d1a\u27d8\u2229", - "\u0245\u2144\u0250\u0254\u01dd\u025f\u1d77\u0265\u1d09\u027e\u029e\ua781\u026f\u0279\u0287\u028c", - "\u028d\u028e\u0531\u0532\u0533\u0534\u0536\u0537\u0539\u053a\u053b\u053c\u053d\u053e\u053f\u0540", - "\u0541\u0542\u0543\u0544\u0545\u0546\u0547\u0548\u0549\u054b\u054c\u054d\u054e\u054f\u0550\u0551", - "\u0552\u0553\u0554\u0555\u0556\u0559\u0561\u0562\u0563\u0564\u0565\u0566\u0567\u0568\u0569\u056a", - "\u056b\u056c\u056d\u056e\u056f\u0570\u0571\u0572\u0573\u0574\u0575\u0576\u0577\u0578\u0579\u057a", - "\u057b\u057c\u057d\u057e\u057f\u0580\u0581\u0582\u0583\u0584\u0585\u0586\u0587\u05e9\u05ea\u0538", - "\u055a\u055b\u055c\u055d\u055e\u055f\u0560\u0588\u058f\u00af\u017f\u01b7\u0292\u01f7\u01bf\u021c", - "\u021d\u0224\u0225\u02d9\ua75a\ua75b\u2011\u214b\u23cf\u23e9\u23ea\u23ed\u23ee\u23ef\u23f4\u23f5", - "\u23f6\u23f7\u23f8\u23f9\u23fa\u23fb\u23fc\u23fd\u2b58\u25b2\u25b6\u25bc\u25c0\u25cf\u25e6\u25d8", - "\u2693\u26e8\u0132\u0133\u01c9\ua728\ua729\ua739\ua73b\ufb00\ufb01\ufb02\ufb03\ufb05\ufffd\u0535", - "\u054a\u16a0\u16a2\u16a3\u16a4\u16a5\u16a6\u16a7\u16a8\u16a9\u16aa\u16ab\u16ac\u16ad\u16ae\u16af", - "\u16b0\u16b1\u16b2\u16b3\u16b4\u16b6\u16b7\u16b8\u16b9\u16ba\u16bb\u16bc\u16bd\u16be\u16bf\u16c0", - "\u16c1\u16c2\u16c3\u16c4\u16c5\u16c6\u16c7\u16c8\u16c9\u16ca\u16cb\u16cc\u16cd\u16ce\u16cf\u16d0", - "\u16d1\u16d2\u16d3\u16d4\u16d5\u16d6\u16d7\u16d8\u16d9\u16da\u16db\u16dc\u16dd\u16de\u16df\u16e0", - "\u16e1\u16e2\u16e3\u16e4\u16e5\u16e6\u16e7\u16e8\u16e9\u16ea\u16eb\u16ec\u16ed\u16ee\u16ef\u16f0", - "\u16f1\u16f2\u16f3\u16f4\u16f5\u16f6\u16f7\u16f8\u263a\u263b\u00a6\u2639\u05da\u05f3\u05f4\u05f0", - "\u05f1\u05f2\u05be\u05c3\u05c6\u00b4\u00a8\u1d00\u0299\u1d04\u1d05\u1d07\ua730\u0262\u029c\u1d0a", - "\u1d0b\u029f\u1d0d\u0274\u1d0f\u1d18\ua7af\u0280\ua731\u1d1b\u1d1c\u1d20\u1d21\u028f\u1d22\u00a7", - "\u0271\u0273\u0272\u0288\u0256\u0261\u02a1\u0255\u0291\u0278\u029d\u02a2\u027b\u0281\u0266\u028b", - "\u0270\u026c\u026e\u0298\u01c0\u01c3\u01c2\u01c1\u0253\u0257\u1d91\u0284\u0260\u029b\u0267\u026b", - "\u0268\u0289\u028a\u0258\u0275\u0264\u025c\u025e\u0251\u0252\u025a\u025d\u0181\u0189\u0191\u01a9", - "\u01b2\u10a0\u10a1\u10a2\u10a3\u10a4\u10a5\u10a6\u10a7\u10a8\u10a9\u10aa\u10ab\u10ac\u10ad\u10ae", - "\u10af\u10b0\u10b1\u10b2\u10b3\u10b4\u10b5\u10b6\u10b7\u10b8\u10b9\u10ba\u10bb\u10bc\u10bd\u10be", - "\u10bf\u10c0\u10c1\u10c2\u10c3\u10c4\u10c5\u10c7\u10cd\u10d0\u10d1\u10d2\u10d3\u10d4\u10d5\u10d6", - "\u10d7\u10d8\u10d9\u10da\u10db\u10dc\u10dd\u10de\u10df\u10e0\u10e1\u10e2\u10e3\u10e4\u10e5\u10e6", - "\u10e7\u10e8\u10e9\u10ea\u10eb\u10ec\u10ed\u10ee\u10ef\u10f0\u10f1\u10f2\u10f3\u10f4\u10f5\u10f6", - "\u10f7\u10f8\u10f9\u10fa\u10fb\u10fc\u10fd\u10fe\u10ff\ufb4a\ufb2b\ufb4e\ufb44\ufb3b\ufb1f\ufb1d", - "\ufb4b\ufb35\ufb4c\ufb31\ua727\ua726\u027a\u2c71\u02a0\u0297\u0296\u026d\u0277\u027f\u0285\u0286", - "\u0293\u029a\u20aa\u20be\u058a\u2d00\u2d01\u2d02\u2d03\u2d04\u2d05\u2d06\u2d21\u2d07\u2d08\u2d09", - "\u2d0a\u2d0b\u2d0c\u2d22\u2d0d\u2d0e\u2d0f\u2d10\u2d11\u2d12\u2d23\u2d13\u2d14\u2d15\u2d16\u2d17", - "\u2d18\u2d19\u2d1a\u2d1b\u2d1c\u2d1d\u2d1e\u2d24\u2d1f\u2d20\u2d25\u215b\u215c\u215d\u215e\u2153", - "\u2154\u2709\u2602\u2614\u2604\u26c4\u2603\u231b\u231a\u2690\u270e\u2763\u2664\u2667\u2661\u2662", - "\u26c8\u2630\u2631\u2633\u2634\u2636\u2637\u2194\u21d2\u21cf\u21d4\u21f5\u2200\u2203\u2204\u2209", - "\u220b\u220c\u2282\u2283\u2284\u2285\u2227\u2228\u22bb\u22bc\u22bd\u2225\u2262\u22c6\u2211\u22a4", - "\u22a5\u22a2\u22a8\u2254\u2201\u2234\u2235\u221b\u221c\u2202\u22c3\u2286\u2287\u25a1\u25b3\u25b7", - "\u25bd\u25c1\u25c6\u25c7\u25cb\u25ce\u2606\u2605\u2718\u2080\u2081\u2082\u2083\u2084\u2085\u2086", - "\u2087\u2088\u2089\u208a\u208b\u208c\u208d\u208e\u222b\u222e\u221d\u2300\u2302\u2318\u3012\u027c", - "\u0184\u0185\u1e9f\u023d\u019a\u019b\u0220\u019e\u019f\u01a7\u01a8\u01aa\u01b8\u01b9\u01bb\u01bc", - "\u01bd\u01be\u0221\u0234\u0235\u0236\u023a\u2c65\u023b\u023c\u0246\u0247\u023e\u2c66\u0241\u0242", - "\u0243\u0244\u0248\u0249\u024a\u024b\u024c\u024d\u024e\u024f\u1e9c\u1e9d\u1efc\u1efd\u1efe\u1eff", - "\ua7a8\ua7a9\ud800\udf30\ud800\udf31\ud800\udf32\ud800\udf33\ud800\udf34\ud800\udf35\ud800\udf36\ud800\udf37\ud800\udf38\ud800\udf39\ud800\udf3a\ud800\udf3b\ud800\udf3c\ud800\udf3d", - "\ud800\udf3e\ud800\udf3f\ud800\udf40\ud800\udf41\ud800\udf42\ud800\udf43\ud800\udf44\ud800\udf45\ud800\udf46\ud800\udf47\ud800\udf48\ud800\udf49\ud800\udf4a\ud83c\udf27\ud83d\udd25\ud83c\udf0a", - "\u2150\u2151\u2155\u2156\u2157\u2159\u215a\u215f\u2189\ud83d\udde1\ud83c\udff9\ud83e\ude93\ud83d\udd31\ud83c\udfa3\ud83e\uddea\u2697", - "\u2bea\u2beb\u2c6d\ud83d\udee1\u2702\ud83c\udf56\ud83e\udea3\ud83d\udd14\u23f3\u2691\u20a0\u20a1\u20a2\u20a3\u20a4\u20a5", - "\u20a6\u20a9\u20ab\u20ad\u20ae\u20b0\u20b1\u20b2\u20b3\u20b5\u20b6\u20b7\u20b8\u20b9\u20ba\u20bb", - "\u20bc\u20bf\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/accented.png", - "height": 12, - "ascent": %ASCENT%, - "chars": [ - "\u00c0\u00c1\u00c2\u00c3\u00c4\u00c5\u00c6\u00c7\u00c8\u00c9\u00ca\u00cb\u00cc\u00cd\u00ce\u00cf", - "\u00d0\u00d1\u00d2\u00d3\u00d4\u00d5\u00d6\u00d9\u00da\u00db\u00dc\u00dd\u00e0\u00e1\u00e2\u00e3", - "\u00e4\u00e5\u00e6\u00e7\u00ec\u00ed\u00ee\u00ef\u00f1\u00f2\u00f3\u00f4\u00f5\u00f6\u00f9\u00fa", - "\u00fb\u00fc\u00fd\u00ff\u0100\u0101\u0102\u0103\u0104\u0105\u0106\u0107\u0108\u0109\u010a\u010b", - "\u010c\u010d\u010e\u010f\u0110\u0111\u0112\u0113\u0114\u0115\u0116\u0117\u0118\u0119\u011a\u011b", - "\u011c\u011d\u1e20\u1e21\u011e\u011f\u0120\u0121\u0122\u0123\u0124\u0125\u0126\u0127\u0128\u0129", - "\u012a\u012b\u012c\u012d\u012e\u012f\u0130\u0131\u0134\u0135\u0136\u0137\u0139\u013a\u013b\u013c", - "\u013d\u013e\u013f\u0140\u0141\u0142\u0143\u0144\u0145\u0146\u0147\u0148\u014a\u014b\u014c\u014d", - "\u014e\u014f\u0150\u0151\u0152\u0153\u0154\u0155\u0156\u0157\u0158\u0159\u015a\u015b\u015c\u015d", - "\u015e\u015f\u0160\u0161\u0162\u0163\u0164\u0165\u0166\u0167\u0168\u0169\u016a\u016b\u016c\u016d", - "\u016e\u016f\u0170\u0171\u0172\u0173\u0174\u0175\u0176\u0177\u0178\u0179\u017a\u017b\u017c\u017d", - "\u017e\u01fc\u01fd\u01fe\u01ff\u0218\u0219\u021a\u021b\u0386\u0388\u0389\u038a\u038c\u038e\u038f", - "\u0390\u03aa\u03ab\u03ac\u03ad\u03ae\u03af\u03b0\u03ca\u03cb\u03cc\u03cd\u03ce\u0400\u0401\u0403", - "\u0407\u040c\u040d\u040e\u0419\u0439\u0450\u0451\u0452\u0453\u0457\u045b\u045c\u045d\u045e\u045f", - "\u0490\u0491\u1e02\u1e03\u1e0a\u1e0b\u1e1e\u1e1f\u1e22\u1e23\u1e30\u1e31\u1e40\u1e41\u1e56\u1e57", - "\u1e60\u1e61\u1e6a\u1e6b\u1e80\u1e81\u1e82\u1e83\u1e84\u1e85\u1ef2\u1ef3\u00e8\u00e9\u00ea\u00eb", - "\u0149\u01e7\u01eb\u040f\u1e0d\u1e25\u1e5b\u1e6d\u1e92\u1eca\u1ecb\u1ecc\u1ecd\u1ee4\u1ee5\u2116", - "\u0207\u0194\u0263\u0283\u2047\u01f1\u01f2\u01f3\u01c4\u01c5\u01c6\u01c7\u01c8\u01ca\u01cb\u01cc", - "\u2139\u1d6b\ua732\ua733\ua734\ua735\ua736\ua737\ua738\ua73a\ua73c\ua73d\ua74e\ua74f\ua760\ua761", - "\ufb04\ufb06\u16a1\u16b5\u01a0\u01a1\u01af\u01b0\u1eae\u1eaf\u1ea4\u1ea5\u1ebe\u1ebf\u1ed1\u1eda", - "\u1edb\u1ee8\u1ee9\u1eb0\u1eb1\u1ea6\u1ea7\u1ec0\u1ec1\u1ed3\u1edc\u1edd\u1eea\u1eeb\u1ea2\u1ea3", - "\u1eb2\u1eb3\u1ea8\u1ea9\u1eba\u1ebb\u1ed5\u1ede\u1ec2\u1ec3\u1ec8\u1ec9\u1ece\u1ecf\u1ed4\u1edf", - "\u1ee6\u1ee7\u1eec\u1eed\u1ef6\u1ef7\u1ea0\u1ea1\u1eb6\u1eb7\u1eac\u1ead\u1eb8\u1eb9\u1ec6\u1ec7", - "\u1ed8\u1ed9\u1ee2\u1ee3\u1ef0\u1ef1\u1ef4\u1ef5\u1ed0\u0195\u1eaa\u1eab\u1ed6\u1ed7\u1eef\u261e", - "\u261c\u262e\u1eb4\u1eb5\u1ebc\u1ebd\u1ec4\u1ec5\u1ed2\u1ee0\u1ee1\u1eee\u1ef8\u1ef9\u0498\u0499", - "\u04a0\u04a1\u04aa\u04ab\u01f6\u26a0\u24ea\u2460\u2461\u2462\u2463\u2464\u2465\u2466\u2467\u2468", - "\u2469\u246a\u246b\u246c\u246d\u246e\u246f\u2470\u2471\u2472\u2473\u24b6\u24b7\u24b8\u24b9\u24ba", - "\u24bb\u24bc\u24bd\u24be\u24bf\u24c0\u24c1\u24c2\u24c3\u24c4\u24c5\u24c6\u24c7\u24c8\u24c9\u24ca", - "\u24cb\u24cc\u24cd\u24ce\u24cf\u24d0\u24d1\u24d2\u24d3\u24d4\u24d5\u24d6\u24d7\u24d8\u24d9\u24da", - "\u24db\u24dc\u24dd\u24de\u24df\u24e0\u24e1\u24e2\u24e3\u24e4\u24e5\u24e6\u24e7\u24e8\u24e9\u0327", - "\u0282\u0290\u0276\u01cd\u01ce\u01de\u01df\u01fa\u01fb\u0202\u0203\u0226\u0227\u01e0\u01e1\u1e00", - "\u1e01\u0200\u0201\u1e06\u1e07\u1e04\u1e05\u1d6c\u1e08\u1e09\u1e10\u1e11\u1e12\u1e13\u1e0e\u1e0f", - "\u1e0c\u1d6d\u1e14\u1e15\u1e16\u1e17\u1e18\u1e19\u1e1c\u1e1d\u0228\u0229\u1e1a\u1e1b\u0204\u0205", - "\u0206\u1d6e\u01f4\u01f5\u01e6\u1e26\u1e27\u1e28\u1e29\u1e2a\u1e2b\u021e\u021f\u1e24\u1e96\u1e2e", - "\u1e2f\u020a\u020b\u01cf\u01d0\u0208\u0209\u1e2c\u1e2d\u01f0\u0237\u01e8\u01e9\u1e32\u1e33\u1e34", - "\u1e35\u1e3a\u1e3b\u1e3c\u1e3d\u1e36\u1e37\u1e38\u1e39\u2c62\u1e3e\u1e3f\u1e42\u1e43\u1d6f\u1e44", - "\u1e45\u1e46\u1e47\u1e4a\u1e4b\u01f8\u01f9\u1e48\u1e49\u1d70\u01ec\u01ed\u022c\u022d\u1e4c\u1e4d", - "\u1e4e\u1e4f\u1e50\u1e51\u1e52\u1e53\u020e\u020f\u022a\u022b\u01d1\u01d2\u022e\u022f\u0230\u0231", - "\u020c\u020d\u01ea\u1e54\u1e55\u1d71\u0212\u0213\u1e58\u1e59\u1e5c\u1e5d\u1e5e\u1e5f\u0210\u0211", - "\u1e5a\u1d73\u1d72\u1e64\u1e65\u1e66\u1e67\u1e62\u1e63\u1e68\u1e69\u1d74\u1e70\u1e71\u1e6e\u1e6f", - "\u1e6c\u1e97\u1d75\u1e72\u1e73\u1e76\u1e77\u1e78\u1e79\u1e7a\u1e7b\u01d3\u01d4\u01d5\u01d6\u01d7", - "\u01d8\u01d9\u01da\u01db\u01dc\u1e74\u1e75\u0214\u0215\u0216\u1e7e\u1e7f\u1e7c\u1e7d\u1e86\u1e87", - "\u1e88\u1e89\u1e98\u1e8c\u1e8d\u1e8a\u1e8b\u0232\u0233\u1e8e\u1e8f\u1e99\u1e94\u1e95\u1e90\u1e91", - "\u1e93\u1d76\u01ee\u01ef\u1e9b\ua73e\ua73f\u01e2\u01e3\u1d7a\u1efb\u1d02\u1d14\uab63\u0238\u02a3", - "\u02a5\u02a4\u02a9\u02aa\u02ab\u0239\u02a8\u02a6\u02a7\uab50\uab51\u20a7\u1efa\ufb2e\ufb2f\u0180", - "\u0182\u0183\u0187\u0188\u018a\u018b\u018c\u0193\u01e4\u01e5\u0197\u0196\u0269\u0198\u0199\u019d", - "\u01a4\u01a5\u027d\u01a6\u01ac\u01ad\u01ab\u01ae\u0217\u01b1\u019c\u01b3\u01b4\u01b5\u01b6\u01a2", - "\u01a3\u0222\u0223\u02ad\u02ae\u02af\ufb14\ufb15\ufb17\ufb16\ufb13\u04d0\u04d1\u04d2\u04d3\u04f6", - "\u04f7\u0494\u0495\u04d6\u04d7\u04bc\u04bd\u04be\u04bf\u04da\u04db\u04dc\u04dd\u04c1\u04c2\u04de", - "\u04df\u04e2\u04e3\u04e4\u04e5\u04e6\u04e7\u04ea\u04eb\u04f0\u04f1\u04ee\u04ef\u04f2\u04f3\u04f4", - "\u04f5\u04f8\u04f9\u04ec\u04ed\u0476\u0477\u04d4\u04fa\u0502\ua682\ua680\ua688\u052a\u052c\ua684", - "\u0504\u0510\u04e0\u0506\u048a\u04c3\u049e\u049c\u051e\u051a\u04c5\u052e\u0512\u0520\u0508\u0514", - "\u04cd\u04c9\u0528\u04c7\u04a4\u0522\u050a\u04a8\u0524\u04a6\u048e\u0516\u050c\ua690\u04ac\ua68a", - "\ua68c\u050e\u04b2\u04fc\u04fe\u0526\ua694\u04b4\ua68e\u04b6\u04cb\u04b8\ua692\ua696\ua686\u048c", - "\u0518\u051c\u04d5\u04fb\u0503\ua683\ua681\ua689\u052b\u052d\ua685\u0505\u0511\u04e1\u0507\u048b", - "\u04c4\u049f\u049d\u051f\u051b\u04c6\u052f\u0513\u0521\u0509\u0515\u04ce\u04ca\u0529\u04c8\u04a5", - "\u0523\u050b\u04a9\u0525\u04a7\u048f\u0517\u050d\ua691\u04ad\ua68b\ua68d\u050f\u04b3\u04fd\u04ff", - "\u0527\ua695\u04b5\ua68f\u04b7\u04cc\u04b9\ua693\ua697\ua687\u048d\u0519\u051d\u1f08\u1f00\u1f09", - "\u1f01\u1f0a\u1f02\u1f0b\u1f03\u1f0c\u1f04\u1f0d\u1f05\u1f0e\u1f06\u1f0f\u1f07\u1fba\u1f70\u1fb8", - "\u1fb0\u1fb9\u1fb1\u1fbb\u1f71\u1f88\u1f80\u1f89\u1f81\u1f8a\u1f82\u1f8b\u1f83\u1f8c\u1f84\u1f8d", - "\u1f85\u1f8e\u1f86\u1f8f\u1f87\u1fbc\u1fb4\u1fb6\u1fb7\u1fb2\u1fb3\u1f18\u1f10\u1f19\u1f11\u1f1a", - "\u1f12\u1f1b\u1f13\u1f1c\u1f14\u1f1d\u1f15\u1fc8\u1fc9\u1f72\u1f73\u1f28\u1f20\u1fca\u1f74\u1f29", - "\u1f21\u1f2a\u1f22\u1f2b\u1f23\u1f2c\u1f24\u1f2d\u1f25\u1f2e\u1f26\u1f2f\u1f27\u1f98\u1f90\u1f99", - "\u1f91\u1f9a\u1f92\u1f9b\u1f93\u1f9c\u1f94\u1f9d\u1f95\u1f9e\u1f96\u1f9f\u1f97\u1fcb\u1f75\u1fcc", - "\u1fc3\u1fc2\u1fc4\u1fc6\u1fc7\u1fda\u1f76\u1fdb\u1f77\u1f38\u1f30\u1f39\u1f31\u1f3a\u1f32\u1f3b", - "\u1f33\u1f3c\u1f34\u1f3d\u1f35\u1f3e\u1f36\u1f3f\u1f37\u1fd8\u1fd0\u1fd9\u1fd1\u1fd2\u1fd3\u1fd6", - "\u1fd7\u1ff8\u1f78\u1ff9\u1f79\u1f48\u1f40\u1f49\u1f41\u1f4a\u1f42\u1f4b\u1f43\u1f4c\u1f44\u1f4d", - "\u1f45\u1fec\u1fe4\u1fe5\u1fea\u1f7a\u1feb\u1f7b\u1f59\u1f51\u1f5b\u1f53\u1f5d\u1f55\u1f5f\u1f57", - "\u1fe8\u1fe0\u1fe9\u1fe1\u03d3\u03d4\u1fe2\u1fe3\u1fe7\u1f50\u1f52\u1f54\u1fe6\u1f56\u1ffa\u1f7c", - "\u1ffb\u1f7d\u1f68\u1f60\u1f69\u1f61\u1f6a\u1f62\u1f6b\u1f63\u1f6c\u1f64\u1f6d\u1f65\u1f6e\u1f66", - "\u1f6f\u1f67\u1fa8\u1fa0\u1fa9\u1fa1\u1faa\u1fa2\u1fab\u1fa3\u1fac\u1fa4\u1fad\u1fa5\u1fae\u1fa6", - "\u1faf\u1fa7\u1ffc\u1ff3\u1ff2\u1ff4\u1ff6\u1ff7\u262f\u2610\u2611\u2612\u018d\u01ba\u2c7e\u023f", - "\u2c7f\u0240\u1d80\ua7c4\ua794\u1d81\u1d82\u1d83\ua795\u1d84\u1d85\u1d86\u1d87\u1d88\u1d89\u1d8a", - "\u1d8b\u1d8c\u1d8d\ua7c6\u1d8e\u1d8f\u1d90\u1d92\u1d93\u1d94\u1d95\u1d96\u1d97\u1d98\u1d99\u1d9a", - "\u1e9a\u2152\u2158\u20a8\u20af\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/ascii.png", - "ascent": %ascent%, - "chars": [ - "\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000", - "\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000", - "\u0020\u0021\u0022\u0023\u0024\u0025\u0026\u0027\u0028\u0029\u002a\u002b\u002c\u002d\u002e\u002f", - "\u0030\u0031\u0032\u0033\u0034\u0035\u0036\u0037\u0038\u0039\u003a\u003b\u003c\u003d\u003e\u003f", - "\u0040\u0041\u0042\u0043\u0044\u0045\u0046\u0047\u0048\u0049\u004a\u004b\u004c\u004d\u004e\u004f", - "\u0050\u0051\u0052\u0053\u0054\u0055\u0056\u0057\u0058\u0059\u005a\u005b\u005c\u005d\u005e\u005f", - "\u0060\u0061\u0062\u0063\u0064\u0065\u0066\u0067\u0068\u0069\u006a\u006b\u006c\u006d\u006e\u006f", - "\u0070\u0071\u0072\u0073\u0074\u0075\u0076\u0077\u0078\u0079\u007a\u007b\u007c\u007d\u007e\u0000", - "\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000", - "\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u00a3\u0000\u0000\u0192", - "\u0000\u0000\u0000\u0000\u0000\u0000\u00aa\u00ba\u0000\u0000\u00ac\u0000\u0000\u0000\u00ab\u00bb", - "\u2591\u2592\u2593\u2502\u2524\u2561\u2562\u2556\u2555\u2563\u2551\u2557\u255d\u255c\u255b\u2510", - "\u2514\u2534\u252c\u251c\u2500\u253c\u255e\u255f\u255a\u2554\u2569\u2566\u2560\u2550\u256c\u2567", - "\u2568\u2564\u2565\u2559\u2558\u2552\u2553\u256b\u256a\u2518\u250c\u2588\u2584\u258c\u2590\u2580", - "\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u2205\u2208\u0000", - "\u2261\u00b1\u2265\u2264\u2320\u2321\u00f7\u2248\u00b0\u2219\u0000\u221a\u207f\u00b2\u25a0\u0000" - ] - } - ] -} diff --git a/src/main/resources/templates/unicode.json b/src/main/resources/templates/unicode.json deleted file mode 100644 index 2e72eb9..0000000 --- a/src/main/resources/templates/unicode.json +++ /dev/null @@ -1,5297 +0,0 @@ -{ - "providers": [ - { - "type": "space", - "advances": { - " ": 4, - "\u200c": 0 - } - }, - { - "type": "bitmap", - "file": "minecraft:font/nonlatin_european.png", - "ascent": %ascent%, - "chars": [ - "\u00a1\u2030\u00ad\u00b7\u20b4\u2260\u00bf\u00d7\u00d8\u00de\u04bb\u00f0\u00f8\u00fe\u0391\u0392", - "\u0393\u0394\u0395\u0396\u0397\u0398\u0399\u039a\u039b\u039c\u039d\u039e\u039f\u03a0\u03a1\u03a3", - "\u03a4\u03a5\u03a6\u03a7\u03a8\u03a9\u03b1\u03b2\u03b3\u03b4\u03b5\u03b6\u03b7\u03b8\u03b9\u03ba", - "\u03bb\u03bc\u03bd\u03be\u03bf\u03c0\u03c1\u03c2\u03c3\u03c4\u03c5\u03c6\u03c7\u03c8\u03c9\u0402", - "\u0405\u0406\u0408\u0409\u040a\u040b\u0410\u0411\u0412\u0413\u0414\u0415\u0416\u0417\u0418\u041a", - "\u041b\u041c\u041d\u041e\u041f\u0420\u0421\u0422\u0423\u0424\u0425\u0426\u0427\u0428\u0429\u042a", - "\u042b\u042c\u042d\u042e\u042f\u0430\u0431\u0432\u0433\u0434\u0435\u0436\u0437\u0438\u043a\u043b", - "\u043c\u043d\u043e\u043f\u0440\u0441\u0442\u0443\u0444\u0445\u0446\u0447\u0448\u0449\u044a\u044b", - "\u044c\u044d\u044e\u044f\u0454\u0455\u0456\u0458\u0459\u045a\u2013\u2014\u2018\u2019\u201c\u201d", - "\u201e\u2026\u204a\u2190\u2191\u2192\u2193\u21c4\uff0b\u018f\u0259\u025b\u026a\u04ae\u04af\u04e8", - "\u04e9\u02bb\u02cc\u037e\u0138\u1e9e\u00df\u20bd\u20ac\u0462\u0463\u0474\u0475\u04c0\u0472\u0473", - "\u2070\u00b9\u00b3\u2074\u2075\u2076\u2077\u2078\u2079\u207a\u207b\u207c\u207d\u207e\u2071\u2122", - "\u0294\u0295\u29c8\u2694\u2620\u049a\u049b\u0492\u0493\u04b0\u04b1\u04d8\u04d9\u0496\u0497\u04a2", - "\u04a3\u04ba\u05d0\u05d1\u05d2\u05d3\u05d4\u05d5\u05d6\u05d7\u05d8\u05d9\u05db\u05dc\u05de\u05dd", - "\u05e0\u05df\u05e1\u05e2\u05e4\u05e3\u05e6\u05e5\u05e7\u05e8\u00a2\u00a4\u00a5\u00a9\u00ae\u00b5", - "\u00b6\u00bc\u00bd\u00be\u0387\u2010\u201a\u2020\u2021\u2022\u2031\u2032\u2033\u2034\u2035\u2036", - "\u2037\u2039\u203a\u203b\u203c\u203d\u2042\u2048\u2049\u204b\u204e\u204f\u2051\u2052\u2057\u2117", - "\u2212\u2213\u221e\u2600\u2601\u2608\u0404\u2632\u2635\u263d\u2640\u2642\u26a5\u2660\u2663\u2665", - "\u2666\u2669\u266a\u266b\u266c\u266d\u266e\u266f\u2680\u2681\u2682\u2683\u2684\u2685\u02ac\u26a1", - "\u26cf\u2714\u2744\u274c\u2764\u2b50\u2e18\u2e2e\u2e35\u2e38\u2e41\u2e4b\u295d\u1614\u0190\u07c8", - "\u03db\u3125\u2c6f\u15fa\u0186\u15e1\u018e\u2132\u2141\ua7b0\ua780\u0500\ua779\u1d1a\u27d8\u2229", - "\u0245\u2144\u0250\u0254\u01dd\u025f\u1d77\u0265\u1d09\u027e\u029e\ua781\u026f\u0279\u0287\u028c", - "\u028d\u028e\u0531\u0532\u0533\u0534\u0536\u0537\u0539\u053a\u053b\u053c\u053d\u053e\u053f\u0540", - "\u0541\u0542\u0543\u0544\u0545\u0546\u0547\u0548\u0549\u054b\u054c\u054d\u054e\u054f\u0550\u0551", - "\u0552\u0553\u0554\u0555\u0556\u0559\u0561\u0562\u0563\u0564\u0565\u0566\u0567\u0568\u0569\u056a", - "\u056b\u056c\u056d\u056e\u056f\u0570\u0571\u0572\u0573\u0574\u0575\u0576\u0577\u0578\u0579\u057a", - "\u057b\u057c\u057d\u057e\u057f\u0580\u0581\u0582\u0583\u0584\u0585\u0586\u0587\u05e9\u05ea\u0538", - "\u055a\u055b\u055c\u055d\u055e\u055f\u0560\u0588\u058f\u00af\u017f\u01b7\u0292\u01f7\u01bf\u021c", - "\u021d\u0224\u0225\u02d9\ua75a\ua75b\u2011\u214b\u23cf\u23e9\u23ea\u23ed\u23ee\u23ef\u23f4\u23f5", - "\u23f6\u23f7\u23f8\u23f9\u23fa\u23fb\u23fc\u23fd\u2b58\u25b2\u25b6\u25bc\u25c0\u25cf\u25e6\u25d8", - "\u2693\u26e8\u0132\u0133\u01c9\ua728\ua729\ua739\ua73b\ufb00\ufb01\ufb02\ufb03\ufb05\ufffd\u0535", - "\u054a\u16a0\u16a2\u16a3\u16a4\u16a5\u16a6\u16a7\u16a8\u16a9\u16aa\u16ab\u16ac\u16ad\u16ae\u16af", - "\u16b0\u16b1\u16b2\u16b3\u16b4\u16b6\u16b7\u16b8\u16b9\u16ba\u16bb\u16bc\u16bd\u16be\u16bf\u16c0", - "\u16c1\u16c2\u16c3\u16c4\u16c5\u16c6\u16c7\u16c8\u16c9\u16ca\u16cb\u16cc\u16cd\u16ce\u16cf\u16d0", - "\u16d1\u16d2\u16d3\u16d4\u16d5\u16d6\u16d7\u16d8\u16d9\u16da\u16db\u16dc\u16dd\u16de\u16df\u16e0", - "\u16e1\u16e2\u16e3\u16e4\u16e5\u16e6\u16e7\u16e8\u16e9\u16ea\u16eb\u16ec\u16ed\u16ee\u16ef\u16f0", - "\u16f1\u16f2\u16f3\u16f4\u16f5\u16f6\u16f7\u16f8\u263a\u263b\u00a6\u2639\u05da\u05f3\u05f4\u05f0", - "\u05f1\u05f2\u05be\u05c3\u05c6\u00b4\u00a8\u1d00\u0299\u1d04\u1d05\u1d07\ua730\u0262\u029c\u1d0a", - "\u1d0b\u029f\u1d0d\u0274\u1d0f\u1d18\ua7af\u0280\ua731\u1d1b\u1d1c\u1d20\u1d21\u028f\u1d22\u00a7", - "\u0271\u0273\u0272\u0288\u0256\u0261\u02a1\u0255\u0291\u0278\u029d\u02a2\u027b\u0281\u0266\u028b", - "\u0270\u026c\u026e\u0298\u01c0\u01c3\u01c2\u01c1\u0253\u0257\u1d91\u0284\u0260\u029b\u0267\u026b", - "\u0268\u0289\u028a\u0258\u0275\u0264\u025c\u025e\u0251\u0252\u025a\u025d\u0181\u0189\u0191\u01a9", - "\u01b2\u10a0\u10a1\u10a2\u10a3\u10a4\u10a5\u10a6\u10a7\u10a8\u10a9\u10aa\u10ab\u10ac\u10ad\u10ae", - "\u10af\u10b0\u10b1\u10b2\u10b3\u10b4\u10b5\u10b6\u10b7\u10b8\u10b9\u10ba\u10bb\u10bc\u10bd\u10be", - "\u10bf\u10c0\u10c1\u10c2\u10c3\u10c4\u10c5\u10c7\u10cd\u10d0\u10d1\u10d2\u10d3\u10d4\u10d5\u10d6", - "\u10d7\u10d8\u10d9\u10da\u10db\u10dc\u10dd\u10de\u10df\u10e0\u10e1\u10e2\u10e3\u10e4\u10e5\u10e6", - "\u10e7\u10e8\u10e9\u10ea\u10eb\u10ec\u10ed\u10ee\u10ef\u10f0\u10f1\u10f2\u10f3\u10f4\u10f5\u10f6", - "\u10f7\u10f8\u10f9\u10fa\u10fb\u10fc\u10fd\u10fe\u10ff\ufb4a\ufb2b\ufb4e\ufb44\ufb3b\ufb1f\ufb1d", - "\ufb4b\ufb35\ufb4c\ufb31\ua727\ua726\u027a\u2c71\u02a0\u0297\u0296\u026d\u0277\u027f\u0285\u0286", - "\u0293\u029a\u20aa\u20be\u058a\u2d00\u2d01\u2d02\u2d03\u2d04\u2d05\u2d06\u2d21\u2d07\u2d08\u2d09", - "\u2d0a\u2d0b\u2d0c\u2d22\u2d0d\u2d0e\u2d0f\u2d10\u2d11\u2d12\u2d23\u2d13\u2d14\u2d15\u2d16\u2d17", - "\u2d18\u2d19\u2d1a\u2d1b\u2d1c\u2d1d\u2d1e\u2d24\u2d1f\u2d20\u2d25\u215b\u215c\u215d\u215e\u2153", - "\u2154\u2709\u2602\u2614\u2604\u26c4\u2603\u231b\u231a\u2690\u270e\u2763\u2664\u2667\u2661\u2662", - "\u26c8\u2630\u2631\u2633\u2634\u2636\u2637\u2194\u21d2\u21cf\u21d4\u21f5\u2200\u2203\u2204\u2209", - "\u220b\u220c\u2282\u2283\u2284\u2285\u2227\u2228\u22bb\u22bc\u22bd\u2225\u2262\u22c6\u2211\u22a4", - "\u22a5\u22a2\u22a8\u2254\u2201\u2234\u2235\u221b\u221c\u2202\u22c3\u2286\u2287\u25a1\u25b3\u25b7", - "\u25bd\u25c1\u25c6\u25c7\u25cb\u25ce\u2606\u2605\u2718\u2080\u2081\u2082\u2083\u2084\u2085\u2086", - "\u2087\u2088\u2089\u208a\u208b\u208c\u208d\u208e\u222b\u222e\u221d\u2300\u2302\u2318\u3012\u027c", - "\u0184\u0185\u1e9f\u023d\u019a\u019b\u0220\u019e\u019f\u01a7\u01a8\u01aa\u01b8\u01b9\u01bb\u01bc", - "\u01bd\u01be\u0221\u0234\u0235\u0236\u023a\u2c65\u023b\u023c\u0246\u0247\u023e\u2c66\u0241\u0242", - "\u0243\u0244\u0248\u0249\u024a\u024b\u024c\u024d\u024e\u024f\u1e9c\u1e9d\u1efc\u1efd\u1efe\u1eff", - "\ua7a8\ua7a9\ud800\udf30\ud800\udf31\ud800\udf32\ud800\udf33\ud800\udf34\ud800\udf35\ud800\udf36\ud800\udf37\ud800\udf38\ud800\udf39\ud800\udf3a\ud800\udf3b\ud800\udf3c\ud800\udf3d", - "\ud800\udf3e\ud800\udf3f\ud800\udf40\ud800\udf41\ud800\udf42\ud800\udf43\ud800\udf44\ud800\udf45\ud800\udf46\ud800\udf47\ud800\udf48\ud800\udf49\ud800\udf4a\ud83c\udf27\ud83d\udd25\ud83c\udf0a", - "\u2150\u2151\u2155\u2156\u2157\u2159\u215a\u215f\u2189\ud83d\udde1\ud83c\udff9\ud83e\ude93\ud83d\udd31\ud83c\udfa3\ud83e\uddea\u2697", - "\u2bea\u2beb\u2c6d\ud83d\udee1\u2702\ud83c\udf56\ud83e\udea3\ud83d\udd14\u23f3\u2691\u20a0\u20a1\u20a2\u20a3\u20a4\u20a5", - "\u20a6\u20a9\u20ab\u20ad\u20ae\u20b0\u20b1\u20b2\u20b3\u20b5\u20b6\u20b7\u20b8\u20b9\u20ba\u20bb", - "\u20bc\u20bf\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/accented.png", - "height": 12, - "ascent": %ASCENT%, - "chars": [ - "\u00c0\u00c1\u00c2\u00c3\u00c4\u00c5\u00c6\u00c7\u00c8\u00c9\u00ca\u00cb\u00cc\u00cd\u00ce\u00cf", - "\u00d0\u00d1\u00d2\u00d3\u00d4\u00d5\u00d6\u00d9\u00da\u00db\u00dc\u00dd\u00e0\u00e1\u00e2\u00e3", - "\u00e4\u00e5\u00e6\u00e7\u00ec\u00ed\u00ee\u00ef\u00f1\u00f2\u00f3\u00f4\u00f5\u00f6\u00f9\u00fa", - "\u00fb\u00fc\u00fd\u00ff\u0100\u0101\u0102\u0103\u0104\u0105\u0106\u0107\u0108\u0109\u010a\u010b", - "\u010c\u010d\u010e\u010f\u0110\u0111\u0112\u0113\u0114\u0115\u0116\u0117\u0118\u0119\u011a\u011b", - "\u011c\u011d\u1e20\u1e21\u011e\u011f\u0120\u0121\u0122\u0123\u0124\u0125\u0126\u0127\u0128\u0129", - "\u012a\u012b\u012c\u012d\u012e\u012f\u0130\u0131\u0134\u0135\u0136\u0137\u0139\u013a\u013b\u013c", - "\u013d\u013e\u013f\u0140\u0141\u0142\u0143\u0144\u0145\u0146\u0147\u0148\u014a\u014b\u014c\u014d", - "\u014e\u014f\u0150\u0151\u0152\u0153\u0154\u0155\u0156\u0157\u0158\u0159\u015a\u015b\u015c\u015d", - "\u015e\u015f\u0160\u0161\u0162\u0163\u0164\u0165\u0166\u0167\u0168\u0169\u016a\u016b\u016c\u016d", - "\u016e\u016f\u0170\u0171\u0172\u0173\u0174\u0175\u0176\u0177\u0178\u0179\u017a\u017b\u017c\u017d", - "\u017e\u01fc\u01fd\u01fe\u01ff\u0218\u0219\u021a\u021b\u0386\u0388\u0389\u038a\u038c\u038e\u038f", - "\u0390\u03aa\u03ab\u03ac\u03ad\u03ae\u03af\u03b0\u03ca\u03cb\u03cc\u03cd\u03ce\u0400\u0401\u0403", - "\u0407\u040c\u040d\u040e\u0419\u0439\u0450\u0451\u0452\u0453\u0457\u045b\u045c\u045d\u045e\u045f", - "\u0490\u0491\u1e02\u1e03\u1e0a\u1e0b\u1e1e\u1e1f\u1e22\u1e23\u1e30\u1e31\u1e40\u1e41\u1e56\u1e57", - "\u1e60\u1e61\u1e6a\u1e6b\u1e80\u1e81\u1e82\u1e83\u1e84\u1e85\u1ef2\u1ef3\u00e8\u00e9\u00ea\u00eb", - "\u0149\u01e7\u01eb\u040f\u1e0d\u1e25\u1e5b\u1e6d\u1e92\u1eca\u1ecb\u1ecc\u1ecd\u1ee4\u1ee5\u2116", - "\u0207\u0194\u0263\u0283\u2047\u01f1\u01f2\u01f3\u01c4\u01c5\u01c6\u01c7\u01c8\u01ca\u01cb\u01cc", - "\u2139\u1d6b\ua732\ua733\ua734\ua735\ua736\ua737\ua738\ua73a\ua73c\ua73d\ua74e\ua74f\ua760\ua761", - "\ufb04\ufb06\u16a1\u16b5\u01a0\u01a1\u01af\u01b0\u1eae\u1eaf\u1ea4\u1ea5\u1ebe\u1ebf\u1ed1\u1eda", - "\u1edb\u1ee8\u1ee9\u1eb0\u1eb1\u1ea6\u1ea7\u1ec0\u1ec1\u1ed3\u1edc\u1edd\u1eea\u1eeb\u1ea2\u1ea3", - "\u1eb2\u1eb3\u1ea8\u1ea9\u1eba\u1ebb\u1ed5\u1ede\u1ec2\u1ec3\u1ec8\u1ec9\u1ece\u1ecf\u1ed4\u1edf", - "\u1ee6\u1ee7\u1eec\u1eed\u1ef6\u1ef7\u1ea0\u1ea1\u1eb6\u1eb7\u1eac\u1ead\u1eb8\u1eb9\u1ec6\u1ec7", - "\u1ed8\u1ed9\u1ee2\u1ee3\u1ef0\u1ef1\u1ef4\u1ef5\u1ed0\u0195\u1eaa\u1eab\u1ed6\u1ed7\u1eef\u261e", - "\u261c\u262e\u1eb4\u1eb5\u1ebc\u1ebd\u1ec4\u1ec5\u1ed2\u1ee0\u1ee1\u1eee\u1ef8\u1ef9\u0498\u0499", - "\u04a0\u04a1\u04aa\u04ab\u01f6\u26a0\u24ea\u2460\u2461\u2462\u2463\u2464\u2465\u2466\u2467\u2468", - "\u2469\u246a\u246b\u246c\u246d\u246e\u246f\u2470\u2471\u2472\u2473\u24b6\u24b7\u24b8\u24b9\u24ba", - "\u24bb\u24bc\u24bd\u24be\u24bf\u24c0\u24c1\u24c2\u24c3\u24c4\u24c5\u24c6\u24c7\u24c8\u24c9\u24ca", - "\u24cb\u24cc\u24cd\u24ce\u24cf\u24d0\u24d1\u24d2\u24d3\u24d4\u24d5\u24d6\u24d7\u24d8\u24d9\u24da", - "\u24db\u24dc\u24dd\u24de\u24df\u24e0\u24e1\u24e2\u24e3\u24e4\u24e5\u24e6\u24e7\u24e8\u24e9\u0327", - "\u0282\u0290\u0276\u01cd\u01ce\u01de\u01df\u01fa\u01fb\u0202\u0203\u0226\u0227\u01e0\u01e1\u1e00", - "\u1e01\u0200\u0201\u1e06\u1e07\u1e04\u1e05\u1d6c\u1e08\u1e09\u1e10\u1e11\u1e12\u1e13\u1e0e\u1e0f", - "\u1e0c\u1d6d\u1e14\u1e15\u1e16\u1e17\u1e18\u1e19\u1e1c\u1e1d\u0228\u0229\u1e1a\u1e1b\u0204\u0205", - "\u0206\u1d6e\u01f4\u01f5\u01e6\u1e26\u1e27\u1e28\u1e29\u1e2a\u1e2b\u021e\u021f\u1e24\u1e96\u1e2e", - "\u1e2f\u020a\u020b\u01cf\u01d0\u0208\u0209\u1e2c\u1e2d\u01f0\u0237\u01e8\u01e9\u1e32\u1e33\u1e34", - "\u1e35\u1e3a\u1e3b\u1e3c\u1e3d\u1e36\u1e37\u1e38\u1e39\u2c62\u1e3e\u1e3f\u1e42\u1e43\u1d6f\u1e44", - "\u1e45\u1e46\u1e47\u1e4a\u1e4b\u01f8\u01f9\u1e48\u1e49\u1d70\u01ec\u01ed\u022c\u022d\u1e4c\u1e4d", - "\u1e4e\u1e4f\u1e50\u1e51\u1e52\u1e53\u020e\u020f\u022a\u022b\u01d1\u01d2\u022e\u022f\u0230\u0231", - "\u020c\u020d\u01ea\u1e54\u1e55\u1d71\u0212\u0213\u1e58\u1e59\u1e5c\u1e5d\u1e5e\u1e5f\u0210\u0211", - "\u1e5a\u1d73\u1d72\u1e64\u1e65\u1e66\u1e67\u1e62\u1e63\u1e68\u1e69\u1d74\u1e70\u1e71\u1e6e\u1e6f", - "\u1e6c\u1e97\u1d75\u1e72\u1e73\u1e76\u1e77\u1e78\u1e79\u1e7a\u1e7b\u01d3\u01d4\u01d5\u01d6\u01d7", - "\u01d8\u01d9\u01da\u01db\u01dc\u1e74\u1e75\u0214\u0215\u0216\u1e7e\u1e7f\u1e7c\u1e7d\u1e86\u1e87", - "\u1e88\u1e89\u1e98\u1e8c\u1e8d\u1e8a\u1e8b\u0232\u0233\u1e8e\u1e8f\u1e99\u1e94\u1e95\u1e90\u1e91", - "\u1e93\u1d76\u01ee\u01ef\u1e9b\ua73e\ua73f\u01e2\u01e3\u1d7a\u1efb\u1d02\u1d14\uab63\u0238\u02a3", - "\u02a5\u02a4\u02a9\u02aa\u02ab\u0239\u02a8\u02a6\u02a7\uab50\uab51\u20a7\u1efa\ufb2e\ufb2f\u0180", - "\u0182\u0183\u0187\u0188\u018a\u018b\u018c\u0193\u01e4\u01e5\u0197\u0196\u0269\u0198\u0199\u019d", - "\u01a4\u01a5\u027d\u01a6\u01ac\u01ad\u01ab\u01ae\u0217\u01b1\u019c\u01b3\u01b4\u01b5\u01b6\u01a2", - "\u01a3\u0222\u0223\u02ad\u02ae\u02af\ufb14\ufb15\ufb17\ufb16\ufb13\u04d0\u04d1\u04d2\u04d3\u04f6", - "\u04f7\u0494\u0495\u04d6\u04d7\u04bc\u04bd\u04be\u04bf\u04da\u04db\u04dc\u04dd\u04c1\u04c2\u04de", - "\u04df\u04e2\u04e3\u04e4\u04e5\u04e6\u04e7\u04ea\u04eb\u04f0\u04f1\u04ee\u04ef\u04f2\u04f3\u04f4", - "\u04f5\u04f8\u04f9\u04ec\u04ed\u0476\u0477\u04d4\u04fa\u0502\ua682\ua680\ua688\u052a\u052c\ua684", - "\u0504\u0510\u04e0\u0506\u048a\u04c3\u049e\u049c\u051e\u051a\u04c5\u052e\u0512\u0520\u0508\u0514", - "\u04cd\u04c9\u0528\u04c7\u04a4\u0522\u050a\u04a8\u0524\u04a6\u048e\u0516\u050c\ua690\u04ac\ua68a", - "\ua68c\u050e\u04b2\u04fc\u04fe\u0526\ua694\u04b4\ua68e\u04b6\u04cb\u04b8\ua692\ua696\ua686\u048c", - "\u0518\u051c\u04d5\u04fb\u0503\ua683\ua681\ua689\u052b\u052d\ua685\u0505\u0511\u04e1\u0507\u048b", - "\u04c4\u049f\u049d\u051f\u051b\u04c6\u052f\u0513\u0521\u0509\u0515\u04ce\u04ca\u0529\u04c8\u04a5", - "\u0523\u050b\u04a9\u0525\u04a7\u048f\u0517\u050d\ua691\u04ad\ua68b\ua68d\u050f\u04b3\u04fd\u04ff", - "\u0527\ua695\u04b5\ua68f\u04b7\u04cc\u04b9\ua693\ua697\ua687\u048d\u0519\u051d\u1f08\u1f00\u1f09", - "\u1f01\u1f0a\u1f02\u1f0b\u1f03\u1f0c\u1f04\u1f0d\u1f05\u1f0e\u1f06\u1f0f\u1f07\u1fba\u1f70\u1fb8", - "\u1fb0\u1fb9\u1fb1\u1fbb\u1f71\u1f88\u1f80\u1f89\u1f81\u1f8a\u1f82\u1f8b\u1f83\u1f8c\u1f84\u1f8d", - "\u1f85\u1f8e\u1f86\u1f8f\u1f87\u1fbc\u1fb4\u1fb6\u1fb7\u1fb2\u1fb3\u1f18\u1f10\u1f19\u1f11\u1f1a", - "\u1f12\u1f1b\u1f13\u1f1c\u1f14\u1f1d\u1f15\u1fc8\u1fc9\u1f72\u1f73\u1f28\u1f20\u1fca\u1f74\u1f29", - "\u1f21\u1f2a\u1f22\u1f2b\u1f23\u1f2c\u1f24\u1f2d\u1f25\u1f2e\u1f26\u1f2f\u1f27\u1f98\u1f90\u1f99", - "\u1f91\u1f9a\u1f92\u1f9b\u1f93\u1f9c\u1f94\u1f9d\u1f95\u1f9e\u1f96\u1f9f\u1f97\u1fcb\u1f75\u1fcc", - "\u1fc3\u1fc2\u1fc4\u1fc6\u1fc7\u1fda\u1f76\u1fdb\u1f77\u1f38\u1f30\u1f39\u1f31\u1f3a\u1f32\u1f3b", - "\u1f33\u1f3c\u1f34\u1f3d\u1f35\u1f3e\u1f36\u1f3f\u1f37\u1fd8\u1fd0\u1fd9\u1fd1\u1fd2\u1fd3\u1fd6", - "\u1fd7\u1ff8\u1f78\u1ff9\u1f79\u1f48\u1f40\u1f49\u1f41\u1f4a\u1f42\u1f4b\u1f43\u1f4c\u1f44\u1f4d", - "\u1f45\u1fec\u1fe4\u1fe5\u1fea\u1f7a\u1feb\u1f7b\u1f59\u1f51\u1f5b\u1f53\u1f5d\u1f55\u1f5f\u1f57", - "\u1fe8\u1fe0\u1fe9\u1fe1\u03d3\u03d4\u1fe2\u1fe3\u1fe7\u1f50\u1f52\u1f54\u1fe6\u1f56\u1ffa\u1f7c", - "\u1ffb\u1f7d\u1f68\u1f60\u1f69\u1f61\u1f6a\u1f62\u1f6b\u1f63\u1f6c\u1f64\u1f6d\u1f65\u1f6e\u1f66", - "\u1f6f\u1f67\u1fa8\u1fa0\u1fa9\u1fa1\u1faa\u1fa2\u1fab\u1fa3\u1fac\u1fa4\u1fad\u1fa5\u1fae\u1fa6", - "\u1faf\u1fa7\u1ffc\u1ff3\u1ff2\u1ff4\u1ff6\u1ff7\u262f\u2610\u2611\u2612\u018d\u01ba\u2c7e\u023f", - "\u2c7f\u0240\u1d80\ua7c4\ua794\u1d81\u1d82\u1d83\ua795\u1d84\u1d85\u1d86\u1d87\u1d88\u1d89\u1d8a", - "\u1d8b\u1d8c\u1d8d\ua7c6\u1d8e\u1d8f\u1d90\u1d92\u1d93\u1d94\u1d95\u1d96\u1d97\u1d98\u1d99\u1d9a", - "\u1e9a\u2152\u2158\u20a8\u20af\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/ascii.png", - "ascent": %ascent%, - "chars": [ - "\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000", - "\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000", - "\u0020\u0021\u0022\u0023\u0024\u0025\u0026\u0027\u0028\u0029\u002a\u002b\u002c\u002d\u002e\u002f", - "\u0030\u0031\u0032\u0033\u0034\u0035\u0036\u0037\u0038\u0039\u003a\u003b\u003c\u003d\u003e\u003f", - "\u0040\u0041\u0042\u0043\u0044\u0045\u0046\u0047\u0048\u0049\u004a\u004b\u004c\u004d\u004e\u004f", - "\u0050\u0051\u0052\u0053\u0054\u0055\u0056\u0057\u0058\u0059\u005a\u005b\u005c\u005d\u005e\u005f", - "\u0060\u0061\u0062\u0063\u0064\u0065\u0066\u0067\u0068\u0069\u006a\u006b\u006c\u006d\u006e\u006f", - "\u0070\u0071\u0072\u0073\u0074\u0075\u0076\u0077\u0078\u0079\u007a\u007b\u007c\u007d\u007e\u0000", - "\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000", - "\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u00a3\u0000\u0000\u0192", - "\u0000\u0000\u0000\u0000\u0000\u0000\u00aa\u00ba\u0000\u0000\u00ac\u0000\u0000\u0000\u00ab\u00bb", - "\u2591\u2592\u2593\u2502\u2524\u2561\u2562\u2556\u2555\u2563\u2551\u2557\u255d\u255c\u255b\u2510", - "\u2514\u2534\u252c\u251c\u2500\u253c\u255e\u255f\u255a\u2554\u2569\u2566\u2560\u2550\u256c\u2567", - "\u2568\u2564\u2565\u2559\u2558\u2552\u2553\u256b\u256a\u2518\u250c\u2588\u2584\u258c\u2590\u2580", - "\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u0000\u2205\u2208\u0000", - "\u2261\u00b1\u2265\u2264\u2320\u2321\u00f7\u2248\u00b0\u2219\u0000\u221a\u207f\u00b2\u25a0\u0000" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_00.png", - "ascent": %ascent%, - "chars": [ - "\u0000\u0001\u0002\u0003\u0004\u0005\u0006\u0007\u0008\u0009\u000a\u000b\u000c\u000d\u000e\u000f", - "\u0010\u0011\u0012\u0013\u0014\u0015\u0016\u0017\u0018\u0019\u001a\u001b\u001c\u001d\u001e\u001f", - "\u0020\u0021\u0022\u0023\u0024\u0025\u0026\u0027\u0028\u0029\u002a\u002b\u002c\u002d\u002e\u002f", - "\u0030\u0031\u0032\u0033\u0034\u0035\u0036\u0037\u0038\u0039\u003a\u003b\u003c\u003d\u003e\u003f", - "\u0040\u0041\u0042\u0043\u0044\u0045\u0046\u0047\u0048\u0049\u004a\u004b\u004c\u004d\u004e\u004f", - "\u0050\u0051\u0052\u0053\u0054\u0055\u0056\u0057\u0058\u0059\u005a\u005b\u005c\u005d\u005e\u005f", - "\u0060\u0061\u0062\u0063\u0064\u0065\u0066\u0067\u0068\u0069\u006a\u006b\u006c\u006d\u006e\u006f", - "\u0070\u0071\u0072\u0073\u0074\u0075\u0076\u0077\u0078\u0079\u007a\u007b\u007c\u007d\u007e\u007f", - "\u0080\u0081\u0082\u0083\u0084\u0085\u0086\u0087\u0088\u0089\u008a\u008b\u008c\u008d\u008e\u008f", - "\u0090\u0091\u0092\u0093\u0094\u0095\u0096\u0097\u0098\u0099\u009a\u009b\u009c\u009d\u009e\u009f", - "\u00a0\u00a1\u00a2\u00a3\u00a4\u00a5\u00a6\u00a7\u00a8\u00a9\u00aa\u00ab\u00ac\u00ad\u00ae\u00af", - "\u00b0\u00b1\u00b2\u00b3\u00b4\u00b5\u00b6\u00b7\u00b8\u00b9\u00ba\u00bb\u00bc\u00bd\u00be\u00bf", - "\u00c0\u00c1\u00c2\u00c3\u00c4\u00c5\u00c6\u00c7\u00c8\u00c9\u00ca\u00cb\u00cc\u00cd\u00ce\u00cf", - "\u00d0\u00d1\u00d2\u00d3\u00d4\u00d5\u00d6\u00d7\u00d8\u00d9\u00da\u00db\u00dc\u00dd\u00de\u00df", - "\u00e0\u00e1\u00e2\u00e3\u00e4\u00e5\u00e6\u00e7\u00e8\u00e9\u00ea\u00eb\u00ec\u00ed\u00ee\u00ef", - "\u00f0\u00f1\u00f2\u00f3\u00f4\u00f5\u00f6\u00f7\u00f8\u00f9\u00fa\u00fb\u00fc\u00fd\u00fe\u00ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_01.png", - "ascent": %ascent%, - "chars": [ - "\u0100\u0101\u0102\u0103\u0104\u0105\u0106\u0107\u0108\u0109\u010a\u010b\u010c\u010d\u010e\u010f", - "\u0110\u0111\u0112\u0113\u0114\u0115\u0116\u0117\u0118\u0119\u011a\u011b\u011c\u011d\u011e\u011f", - "\u0120\u0121\u0122\u0123\u0124\u0125\u0126\u0127\u0128\u0129\u012a\u012b\u012c\u012d\u012e\u012f", - "\u0130\u0131\u0132\u0133\u0134\u0135\u0136\u0137\u0138\u0139\u013a\u013b\u013c\u013d\u013e\u013f", - "\u0140\u0141\u0142\u0143\u0144\u0145\u0146\u0147\u0148\u0149\u014a\u014b\u014c\u014d\u014e\u014f", - "\u0150\u0151\u0152\u0153\u0154\u0155\u0156\u0157\u0158\u0159\u015a\u015b\u015c\u015d\u015e\u015f", - "\u0160\u0161\u0162\u0163\u0164\u0165\u0166\u0167\u0168\u0169\u016a\u016b\u016c\u016d\u016e\u016f", - "\u0170\u0171\u0172\u0173\u0174\u0175\u0176\u0177\u0178\u0179\u017a\u017b\u017c\u017d\u017e\u017f", - "\u0180\u0181\u0182\u0183\u0184\u0185\u0186\u0187\u0188\u0189\u018a\u018b\u018c\u018d\u018e\u018f", - "\u0190\u0191\u0192\u0193\u0194\u0195\u0196\u0197\u0198\u0199\u019a\u019b\u019c\u019d\u019e\u019f", - "\u01a0\u01a1\u01a2\u01a3\u01a4\u01a5\u01a6\u01a7\u01a8\u01a9\u01aa\u01ab\u01ac\u01ad\u01ae\u01af", - "\u01b0\u01b1\u01b2\u01b3\u01b4\u01b5\u01b6\u01b7\u01b8\u01b9\u01ba\u01bb\u01bc\u01bd\u01be\u01bf", - "\u01c0\u01c1\u01c2\u01c3\u01c4\u01c5\u01c6\u01c7\u01c8\u01c9\u01ca\u01cb\u01cc\u01cd\u01ce\u01cf", - "\u01d0\u01d1\u01d2\u01d3\u01d4\u01d5\u01d6\u01d7\u01d8\u01d9\u01da\u01db\u01dc\u01dd\u01de\u01df", - "\u01e0\u01e1\u01e2\u01e3\u01e4\u01e5\u01e6\u01e7\u01e8\u01e9\u01ea\u01eb\u01ec\u01ed\u01ee\u01ef", - "\u01f0\u01f1\u01f2\u01f3\u01f4\u01f5\u01f6\u01f7\u01f8\u01f9\u01fa\u01fb\u01fc\u01fd\u01fe\u01ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_02.png", - "ascent": %ascent%, - "chars": [ - "\u0200\u0201\u0202\u0203\u0204\u0205\u0206\u0207\u0208\u0209\u020a\u020b\u020c\u020d\u020e\u020f", - "\u0210\u0211\u0212\u0213\u0214\u0215\u0216\u0217\u0218\u0219\u021a\u021b\u021c\u021d\u021e\u021f", - "\u0220\u0221\u0222\u0223\u0224\u0225\u0226\u0227\u0228\u0229\u022a\u022b\u022c\u022d\u022e\u022f", - "\u0230\u0231\u0232\u0233\u0234\u0235\u0236\u0237\u0238\u0239\u023a\u023b\u023c\u023d\u023e\u023f", - "\u0240\u0241\u0242\u0243\u0244\u0245\u0246\u0247\u0248\u0249\u024a\u024b\u024c\u024d\u024e\u024f", - "\u0250\u0251\u0252\u0253\u0254\u0255\u0256\u0257\u0258\u0259\u025a\u025b\u025c\u025d\u025e\u025f", - "\u0260\u0261\u0262\u0263\u0264\u0265\u0266\u0267\u0268\u0269\u026a\u026b\u026c\u026d\u026e\u026f", - "\u0270\u0271\u0272\u0273\u0274\u0275\u0276\u0277\u0278\u0279\u027a\u027b\u027c\u027d\u027e\u027f", - "\u0280\u0281\u0282\u0283\u0284\u0285\u0286\u0287\u0288\u0289\u028a\u028b\u028c\u028d\u028e\u028f", - "\u0290\u0291\u0292\u0293\u0294\u0295\u0296\u0297\u0298\u0299\u029a\u029b\u029c\u029d\u029e\u029f", - "\u02a0\u02a1\u02a2\u02a3\u02a4\u02a5\u02a6\u02a7\u02a8\u02a9\u02aa\u02ab\u02ac\u02ad\u02ae\u02af", - "\u02b0\u02b1\u02b2\u02b3\u02b4\u02b5\u02b6\u02b7\u02b8\u02b9\u02ba\u02bb\u02bc\u02bd\u02be\u02bf", - "\u02c0\u02c1\u02c2\u02c3\u02c4\u02c5\u02c6\u02c7\u02c8\u02c9\u02ca\u02cb\u02cc\u02cd\u02ce\u02cf", - "\u02d0\u02d1\u02d2\u02d3\u02d4\u02d5\u02d6\u02d7\u02d8\u02d9\u02da\u02db\u02dc\u02dd\u02de\u02df", - "\u02e0\u02e1\u02e2\u02e3\u02e4\u02e5\u02e6\u02e7\u02e8\u02e9\u02ea\u02eb\u02ec\u02ed\u02ee\u02ef", - "\u02f0\u02f1\u02f2\u02f3\u02f4\u02f5\u02f6\u02f7\u02f8\u02f9\u02fa\u02fb\u02fc\u02fd\u02fe\u02ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_03.png", - "ascent": %ascent%, - "chars": [ - "\u0300\u0301\u0302\u0303\u0304\u0305\u0306\u0307\u0308\u0309\u030a\u030b\u030c\u030d\u030e\u030f", - "\u0310\u0311\u0312\u0313\u0314\u0315\u0316\u0317\u0318\u0319\u031a\u031b\u031c\u031d\u031e\u031f", - "\u0320\u0321\u0322\u0323\u0324\u0325\u0326\u0327\u0328\u0329\u032a\u032b\u032c\u032d\u032e\u032f", - "\u0330\u0331\u0332\u0333\u0334\u0335\u0336\u0337\u0338\u0339\u033a\u033b\u033c\u033d\u033e\u033f", - "\u0340\u0341\u0342\u0343\u0344\u0345\u0346\u0347\u0348\u0349\u034a\u034b\u034c\u034d\u034e\u034f", - "\u0350\u0351\u0352\u0353\u0354\u0355\u0356\u0357\u0358\u0359\u035a\u035b\u035c\u035d\u035e\u035f", - "\u0360\u0361\u0362\u0363\u0364\u0365\u0366\u0367\u0368\u0369\u036a\u036b\u036c\u036d\u036e\u036f", - "\u0370\u0371\u0372\u0373\u0374\u0375\u0376\u0377\u0378\u0379\u037a\u037b\u037c\u037d\u037e\u037f", - "\u0380\u0381\u0382\u0383\u0384\u0385\u0386\u0387\u0388\u0389\u038a\u038b\u038c\u038d\u038e\u038f", - "\u0390\u0391\u0392\u0393\u0394\u0395\u0396\u0397\u0398\u0399\u039a\u039b\u039c\u039d\u039e\u039f", - "\u03a0\u03a1\u03a2\u03a3\u03a4\u03a5\u03a6\u03a7\u03a8\u03a9\u03aa\u03ab\u03ac\u03ad\u03ae\u03af", - "\u03b0\u03b1\u03b2\u03b3\u03b4\u03b5\u03b6\u03b7\u03b8\u03b9\u03ba\u03bb\u03bc\u03bd\u03be\u03bf", - "\u03c0\u03c1\u03c2\u03c3\u03c4\u03c5\u03c6\u03c7\u03c8\u03c9\u03ca\u03cb\u03cc\u03cd\u03ce\u03cf", - "\u03d0\u03d1\u03d2\u03d3\u03d4\u03d5\u03d6\u03d7\u03d8\u03d9\u03da\u03db\u03dc\u03dd\u03de\u03df", - "\u03e0\u03e1\u03e2\u03e3\u03e4\u03e5\u03e6\u03e7\u03e8\u03e9\u03ea\u03eb\u03ec\u03ed\u03ee\u03ef", - "\u03f0\u03f1\u03f2\u03f3\u03f4\u03f5\u03f6\u03f7\u03f8\u03f9\u03fa\u03fb\u03fc\u03fd\u03fe\u03ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_04.png", - "ascent": %ascent%, - "chars": [ - "\u0400\u0401\u0402\u0403\u0404\u0405\u0406\u0407\u0408\u0409\u040a\u040b\u040c\u040d\u040e\u040f", - "\u0410\u0411\u0412\u0413\u0414\u0415\u0416\u0417\u0418\u0419\u041a\u041b\u041c\u041d\u041e\u041f", - "\u0420\u0421\u0422\u0423\u0424\u0425\u0426\u0427\u0428\u0429\u042a\u042b\u042c\u042d\u042e\u042f", - "\u0430\u0431\u0432\u0433\u0434\u0435\u0436\u0437\u0438\u0439\u043a\u043b\u043c\u043d\u043e\u043f", - "\u0440\u0441\u0442\u0443\u0444\u0445\u0446\u0447\u0448\u0449\u044a\u044b\u044c\u044d\u044e\u044f", - "\u0450\u0451\u0452\u0453\u0454\u0455\u0456\u0457\u0458\u0459\u045a\u045b\u045c\u045d\u045e\u045f", - "\u0460\u0461\u0462\u0463\u0464\u0465\u0466\u0467\u0468\u0469\u046a\u046b\u046c\u046d\u046e\u046f", - "\u0470\u0471\u0472\u0473\u0474\u0475\u0476\u0477\u0478\u0479\u047a\u047b\u047c\u047d\u047e\u047f", - "\u0480\u0481\u0482\u0483\u0484\u0485\u0486\u0487\u0488\u0489\u048a\u048b\u048c\u048d\u048e\u048f", - "\u0490\u0491\u0492\u0493\u0494\u0495\u0496\u0497\u0498\u0499\u049a\u049b\u049c\u049d\u049e\u049f", - "\u04a0\u04a1\u04a2\u04a3\u04a4\u04a5\u04a6\u04a7\u04a8\u04a9\u04aa\u04ab\u04ac\u04ad\u04ae\u04af", - "\u04b0\u04b1\u04b2\u04b3\u04b4\u04b5\u04b6\u04b7\u04b8\u04b9\u04ba\u04bb\u04bc\u04bd\u04be\u04bf", - "\u04c0\u04c1\u04c2\u04c3\u04c4\u04c5\u04c6\u04c7\u04c8\u04c9\u04ca\u04cb\u04cc\u04cd\u04ce\u04cf", - "\u04d0\u04d1\u04d2\u04d3\u04d4\u04d5\u04d6\u04d7\u04d8\u04d9\u04da\u04db\u04dc\u04dd\u04de\u04df", - "\u04e0\u04e1\u04e2\u04e3\u04e4\u04e5\u04e6\u04e7\u04e8\u04e9\u04ea\u04eb\u04ec\u04ed\u04ee\u04ef", - "\u04f0\u04f1\u04f2\u04f3\u04f4\u04f5\u04f6\u04f7\u04f8\u04f9\u04fa\u04fb\u04fc\u04fd\u04fe\u04ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_05.png", - "ascent": %ascent%, - "chars": [ - "\u0500\u0501\u0502\u0503\u0504\u0505\u0506\u0507\u0508\u0509\u050a\u050b\u050c\u050d\u050e\u050f", - "\u0510\u0511\u0512\u0513\u0514\u0515\u0516\u0517\u0518\u0519\u051a\u051b\u051c\u051d\u051e\u051f", - "\u0520\u0521\u0522\u0523\u0524\u0525\u0526\u0527\u0528\u0529\u052a\u052b\u052c\u052d\u052e\u052f", - "\u0530\u0531\u0532\u0533\u0534\u0535\u0536\u0537\u0538\u0539\u053a\u053b\u053c\u053d\u053e\u053f", - "\u0540\u0541\u0542\u0543\u0544\u0545\u0546\u0547\u0548\u0549\u054a\u054b\u054c\u054d\u054e\u054f", - "\u0550\u0551\u0552\u0553\u0554\u0555\u0556\u0557\u0558\u0559\u055a\u055b\u055c\u055d\u055e\u055f", - "\u0560\u0561\u0562\u0563\u0564\u0565\u0566\u0567\u0568\u0569\u056a\u056b\u056c\u056d\u056e\u056f", - "\u0570\u0571\u0572\u0573\u0574\u0575\u0576\u0577\u0578\u0579\u057a\u057b\u057c\u057d\u057e\u057f", - "\u0580\u0581\u0582\u0583\u0584\u0585\u0586\u0587\u0588\u0589\u058a\u058b\u058c\u058d\u058e\u058f", - "\u0590\u0591\u0592\u0593\u0594\u0595\u0596\u0597\u0598\u0599\u059a\u059b\u059c\u059d\u059e\u059f", - "\u05a0\u05a1\u05a2\u05a3\u05a4\u05a5\u05a6\u05a7\u05a8\u05a9\u05aa\u05ab\u05ac\u05ad\u05ae\u05af", - "\u05b0\u05b1\u05b2\u05b3\u05b4\u05b5\u05b6\u05b7\u05b8\u05b9\u05ba\u05bb\u05bc\u05bd\u05be\u05bf", - "\u05c0\u05c1\u05c2\u05c3\u05c4\u05c5\u05c6\u05c7\u05c8\u05c9\u05ca\u05cb\u05cc\u05cd\u05ce\u05cf", - "\u05d0\u05d1\u05d2\u05d3\u05d4\u05d5\u05d6\u05d7\u05d8\u05d9\u05da\u05db\u05dc\u05dd\u05de\u05df", - "\u05e0\u05e1\u05e2\u05e3\u05e4\u05e5\u05e6\u05e7\u05e8\u05e9\u05ea\u05eb\u05ec\u05ed\u05ee\u05ef", - "\u05f0\u05f1\u05f2\u05f3\u05f4\u05f5\u05f6\u05f7\u05f8\u05f9\u05fa\u05fb\u05fc\u05fd\u05fe\u05ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_06.png", - "ascent": %ascent%, - "chars": [ - "\u0600\u0601\u0602\u0603\u0604\u0605\u0606\u0607\u0608\u0609\u060a\u060b\u060c\u060d\u060e\u060f", - "\u0610\u0611\u0612\u0613\u0614\u0615\u0616\u0617\u0618\u0619\u061a\u061b\u061c\u061d\u061e\u061f", - "\u0620\u0621\u0622\u0623\u0624\u0625\u0626\u0627\u0628\u0629\u062a\u062b\u062c\u062d\u062e\u062f", - "\u0630\u0631\u0632\u0633\u0634\u0635\u0636\u0637\u0638\u0639\u063a\u063b\u063c\u063d\u063e\u063f", - "\u0640\u0641\u0642\u0643\u0644\u0645\u0646\u0647\u0648\u0649\u064a\u064b\u064c\u064d\u064e\u064f", - "\u0650\u0651\u0652\u0653\u0654\u0655\u0656\u0657\u0658\u0659\u065a\u065b\u065c\u065d\u065e\u065f", - "\u0660\u0661\u0662\u0663\u0664\u0665\u0666\u0667\u0668\u0669\u066a\u066b\u066c\u066d\u066e\u066f", - "\u0670\u0671\u0672\u0673\u0674\u0675\u0676\u0677\u0678\u0679\u067a\u067b\u067c\u067d\u067e\u067f", - "\u0680\u0681\u0682\u0683\u0684\u0685\u0686\u0687\u0688\u0689\u068a\u068b\u068c\u068d\u068e\u068f", - "\u0690\u0691\u0692\u0693\u0694\u0695\u0696\u0697\u0698\u0699\u069a\u069b\u069c\u069d\u069e\u069f", - "\u06a0\u06a1\u06a2\u06a3\u06a4\u06a5\u06a6\u06a7\u06a8\u06a9\u06aa\u06ab\u06ac\u06ad\u06ae\u06af", - "\u06b0\u06b1\u06b2\u06b3\u06b4\u06b5\u06b6\u06b7\u06b8\u06b9\u06ba\u06bb\u06bc\u06bd\u06be\u06bf", - "\u06c0\u06c1\u06c2\u06c3\u06c4\u06c5\u06c6\u06c7\u06c8\u06c9\u06ca\u06cb\u06cc\u06cd\u06ce\u06cf", - "\u06d0\u06d1\u06d2\u06d3\u06d4\u06d5\u06d6\u06d7\u06d8\u06d9\u06da\u06db\u06dc\u06dd\u06de\u06df", - "\u06e0\u06e1\u06e2\u06e3\u06e4\u06e5\u06e6\u06e7\u06e8\u06e9\u06ea\u06eb\u06ec\u06ed\u06ee\u06ef", - "\u06f0\u06f1\u06f2\u06f3\u06f4\u06f5\u06f6\u06f7\u06f8\u06f9\u06fa\u06fb\u06fc\u06fd\u06fe\u06ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_07.png", - "ascent": %ascent%, - "chars": [ - "\u0700\u0701\u0702\u0703\u0704\u0705\u0706\u0707\u0708\u0709\u070a\u070b\u070c\u070d\u070e\u070f", - "\u0710\u0711\u0712\u0713\u0714\u0715\u0716\u0717\u0718\u0719\u071a\u071b\u071c\u071d\u071e\u071f", - "\u0720\u0721\u0722\u0723\u0724\u0725\u0726\u0727\u0728\u0729\u072a\u072b\u072c\u072d\u072e\u072f", - "\u0730\u0731\u0732\u0733\u0734\u0735\u0736\u0737\u0738\u0739\u073a\u073b\u073c\u073d\u073e\u073f", - "\u0740\u0741\u0742\u0743\u0744\u0745\u0746\u0747\u0748\u0749\u074a\u074b\u074c\u074d\u074e\u074f", - "\u0750\u0751\u0752\u0753\u0754\u0755\u0756\u0757\u0758\u0759\u075a\u075b\u075c\u075d\u075e\u075f", - "\u0760\u0761\u0762\u0763\u0764\u0765\u0766\u0767\u0768\u0769\u076a\u076b\u076c\u076d\u076e\u076f", - "\u0770\u0771\u0772\u0773\u0774\u0775\u0776\u0777\u0778\u0779\u077a\u077b\u077c\u077d\u077e\u077f", - "\u0780\u0781\u0782\u0783\u0784\u0785\u0786\u0787\u0788\u0789\u078a\u078b\u078c\u078d\u078e\u078f", - "\u0790\u0791\u0792\u0793\u0794\u0795\u0796\u0797\u0798\u0799\u079a\u079b\u079c\u079d\u079e\u079f", - "\u07a0\u07a1\u07a2\u07a3\u07a4\u07a5\u07a6\u07a7\u07a8\u07a9\u07aa\u07ab\u07ac\u07ad\u07ae\u07af", - "\u07b0\u07b1\u07b2\u07b3\u07b4\u07b5\u07b6\u07b7\u07b8\u07b9\u07ba\u07bb\u07bc\u07bd\u07be\u07bf", - "\u07c0\u07c1\u07c2\u07c3\u07c4\u07c5\u07c6\u07c7\u07c8\u07c9\u07ca\u07cb\u07cc\u07cd\u07ce\u07cf", - "\u07d0\u07d1\u07d2\u07d3\u07d4\u07d5\u07d6\u07d7\u07d8\u07d9\u07da\u07db\u07dc\u07dd\u07de\u07df", - "\u07e0\u07e1\u07e2\u07e3\u07e4\u07e5\u07e6\u07e7\u07e8\u07e9\u07ea\u07eb\u07ec\u07ed\u07ee\u07ef", - "\u07f0\u07f1\u07f2\u07f3\u07f4\u07f5\u07f6\u07f7\u07f8\u07f9\u07fa\u07fb\u07fc\u07fd\u07fe\u07ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_09.png", - "ascent": %ascent%, - "chars": [ - "\u0900\u0901\u0902\u0903\u0904\u0905\u0906\u0907\u0908\u0909\u090a\u090b\u090c\u090d\u090e\u090f", - "\u0910\u0911\u0912\u0913\u0914\u0915\u0916\u0917\u0918\u0919\u091a\u091b\u091c\u091d\u091e\u091f", - "\u0920\u0921\u0922\u0923\u0924\u0925\u0926\u0927\u0928\u0929\u092a\u092b\u092c\u092d\u092e\u092f", - "\u0930\u0931\u0932\u0933\u0934\u0935\u0936\u0937\u0938\u0939\u093a\u093b\u093c\u093d\u093e\u093f", - "\u0940\u0941\u0942\u0943\u0944\u0945\u0946\u0947\u0948\u0949\u094a\u094b\u094c\u094d\u094e\u094f", - "\u0950\u0951\u0952\u0953\u0954\u0955\u0956\u0957\u0958\u0959\u095a\u095b\u095c\u095d\u095e\u095f", - "\u0960\u0961\u0962\u0963\u0964\u0965\u0966\u0967\u0968\u0969\u096a\u096b\u096c\u096d\u096e\u096f", - "\u0970\u0971\u0972\u0973\u0974\u0975\u0976\u0977\u0978\u0979\u097a\u097b\u097c\u097d\u097e\u097f", - "\u0980\u0981\u0982\u0983\u0984\u0985\u0986\u0987\u0988\u0989\u098a\u098b\u098c\u098d\u098e\u098f", - "\u0990\u0991\u0992\u0993\u0994\u0995\u0996\u0997\u0998\u0999\u099a\u099b\u099c\u099d\u099e\u099f", - "\u09a0\u09a1\u09a2\u09a3\u09a4\u09a5\u09a6\u09a7\u09a8\u09a9\u09aa\u09ab\u09ac\u09ad\u09ae\u09af", - "\u09b0\u09b1\u09b2\u09b3\u09b4\u09b5\u09b6\u09b7\u09b8\u09b9\u09ba\u09bb\u09bc\u09bd\u09be\u09bf", - "\u09c0\u09c1\u09c2\u09c3\u09c4\u09c5\u09c6\u09c7\u09c8\u09c9\u09ca\u09cb\u09cc\u09cd\u09ce\u09cf", - "\u09d0\u09d1\u09d2\u09d3\u09d4\u09d5\u09d6\u09d7\u09d8\u09d9\u09da\u09db\u09dc\u09dd\u09de\u09df", - "\u09e0\u09e1\u09e2\u09e3\u09e4\u09e5\u09e6\u09e7\u09e8\u09e9\u09ea\u09eb\u09ec\u09ed\u09ee\u09ef", - "\u09f0\u09f1\u09f2\u09f3\u09f4\u09f5\u09f6\u09f7\u09f8\u09f9\u09fa\u09fb\u09fc\u09fd\u09fe\u09ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_0a.png", - "ascent": %ascent%, - "chars": [ - "\u0a00\u0a01\u0a02\u0a03\u0a04\u0a05\u0a06\u0a07\u0a08\u0a09\u0a0a\u0a0b\u0a0c\u0a0d\u0a0e\u0a0f", - "\u0a10\u0a11\u0a12\u0a13\u0a14\u0a15\u0a16\u0a17\u0a18\u0a19\u0a1a\u0a1b\u0a1c\u0a1d\u0a1e\u0a1f", - "\u0a20\u0a21\u0a22\u0a23\u0a24\u0a25\u0a26\u0a27\u0a28\u0a29\u0a2a\u0a2b\u0a2c\u0a2d\u0a2e\u0a2f", - "\u0a30\u0a31\u0a32\u0a33\u0a34\u0a35\u0a36\u0a37\u0a38\u0a39\u0a3a\u0a3b\u0a3c\u0a3d\u0a3e\u0a3f", - "\u0a40\u0a41\u0a42\u0a43\u0a44\u0a45\u0a46\u0a47\u0a48\u0a49\u0a4a\u0a4b\u0a4c\u0a4d\u0a4e\u0a4f", - "\u0a50\u0a51\u0a52\u0a53\u0a54\u0a55\u0a56\u0a57\u0a58\u0a59\u0a5a\u0a5b\u0a5c\u0a5d\u0a5e\u0a5f", - "\u0a60\u0a61\u0a62\u0a63\u0a64\u0a65\u0a66\u0a67\u0a68\u0a69\u0a6a\u0a6b\u0a6c\u0a6d\u0a6e\u0a6f", - "\u0a70\u0a71\u0a72\u0a73\u0a74\u0a75\u0a76\u0a77\u0a78\u0a79\u0a7a\u0a7b\u0a7c\u0a7d\u0a7e\u0a7f", - "\u0a80\u0a81\u0a82\u0a83\u0a84\u0a85\u0a86\u0a87\u0a88\u0a89\u0a8a\u0a8b\u0a8c\u0a8d\u0a8e\u0a8f", - "\u0a90\u0a91\u0a92\u0a93\u0a94\u0a95\u0a96\u0a97\u0a98\u0a99\u0a9a\u0a9b\u0a9c\u0a9d\u0a9e\u0a9f", - "\u0aa0\u0aa1\u0aa2\u0aa3\u0aa4\u0aa5\u0aa6\u0aa7\u0aa8\u0aa9\u0aaa\u0aab\u0aac\u0aad\u0aae\u0aaf", - "\u0ab0\u0ab1\u0ab2\u0ab3\u0ab4\u0ab5\u0ab6\u0ab7\u0ab8\u0ab9\u0aba\u0abb\u0abc\u0abd\u0abe\u0abf", - "\u0ac0\u0ac1\u0ac2\u0ac3\u0ac4\u0ac5\u0ac6\u0ac7\u0ac8\u0ac9\u0aca\u0acb\u0acc\u0acd\u0ace\u0acf", - "\u0ad0\u0ad1\u0ad2\u0ad3\u0ad4\u0ad5\u0ad6\u0ad7\u0ad8\u0ad9\u0ada\u0adb\u0adc\u0add\u0ade\u0adf", - "\u0ae0\u0ae1\u0ae2\u0ae3\u0ae4\u0ae5\u0ae6\u0ae7\u0ae8\u0ae9\u0aea\u0aeb\u0aec\u0aed\u0aee\u0aef", - "\u0af0\u0af1\u0af2\u0af3\u0af4\u0af5\u0af6\u0af7\u0af8\u0af9\u0afa\u0afb\u0afc\u0afd\u0afe\u0aff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_0b.png", - "ascent": %ascent%, - "chars": [ - "\u0b00\u0b01\u0b02\u0b03\u0b04\u0b05\u0b06\u0b07\u0b08\u0b09\u0b0a\u0b0b\u0b0c\u0b0d\u0b0e\u0b0f", - "\u0b10\u0b11\u0b12\u0b13\u0b14\u0b15\u0b16\u0b17\u0b18\u0b19\u0b1a\u0b1b\u0b1c\u0b1d\u0b1e\u0b1f", - "\u0b20\u0b21\u0b22\u0b23\u0b24\u0b25\u0b26\u0b27\u0b28\u0b29\u0b2a\u0b2b\u0b2c\u0b2d\u0b2e\u0b2f", - "\u0b30\u0b31\u0b32\u0b33\u0b34\u0b35\u0b36\u0b37\u0b38\u0b39\u0b3a\u0b3b\u0b3c\u0b3d\u0b3e\u0b3f", - "\u0b40\u0b41\u0b42\u0b43\u0b44\u0b45\u0b46\u0b47\u0b48\u0b49\u0b4a\u0b4b\u0b4c\u0b4d\u0b4e\u0b4f", - "\u0b50\u0b51\u0b52\u0b53\u0b54\u0b55\u0b56\u0b57\u0b58\u0b59\u0b5a\u0b5b\u0b5c\u0b5d\u0b5e\u0b5f", - "\u0b60\u0b61\u0b62\u0b63\u0b64\u0b65\u0b66\u0b67\u0b68\u0b69\u0b6a\u0b6b\u0b6c\u0b6d\u0b6e\u0b6f", - "\u0b70\u0b71\u0b72\u0b73\u0b74\u0b75\u0b76\u0b77\u0b78\u0b79\u0b7a\u0b7b\u0b7c\u0b7d\u0b7e\u0b7f", - "\u0b80\u0b81\u0b82\u0b83\u0b84\u0b85\u0b86\u0b87\u0b88\u0b89\u0b8a\u0b8b\u0b8c\u0b8d\u0b8e\u0b8f", - "\u0b90\u0b91\u0b92\u0b93\u0b94\u0b95\u0b96\u0b97\u0b98\u0b99\u0b9a\u0b9b\u0b9c\u0b9d\u0b9e\u0b9f", - "\u0ba0\u0ba1\u0ba2\u0ba3\u0ba4\u0ba5\u0ba6\u0ba7\u0ba8\u0ba9\u0baa\u0bab\u0bac\u0bad\u0bae\u0baf", - "\u0bb0\u0bb1\u0bb2\u0bb3\u0bb4\u0bb5\u0bb6\u0bb7\u0bb8\u0bb9\u0bba\u0bbb\u0bbc\u0bbd\u0bbe\u0bbf", - "\u0bc0\u0bc1\u0bc2\u0bc3\u0bc4\u0bc5\u0bc6\u0bc7\u0bc8\u0bc9\u0bca\u0bcb\u0bcc\u0bcd\u0bce\u0bcf", - "\u0bd0\u0bd1\u0bd2\u0bd3\u0bd4\u0bd5\u0bd6\u0bd7\u0bd8\u0bd9\u0bda\u0bdb\u0bdc\u0bdd\u0bde\u0bdf", - "\u0be0\u0be1\u0be2\u0be3\u0be4\u0be5\u0be6\u0be7\u0be8\u0be9\u0bea\u0beb\u0bec\u0bed\u0bee\u0bef", - "\u0bf0\u0bf1\u0bf2\u0bf3\u0bf4\u0bf5\u0bf6\u0bf7\u0bf8\u0bf9\u0bfa\u0bfb\u0bfc\u0bfd\u0bfe\u0bff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_0c.png", - "ascent": %ascent%, - "chars": [ - "\u0c00\u0c01\u0c02\u0c03\u0c04\u0c05\u0c06\u0c07\u0c08\u0c09\u0c0a\u0c0b\u0c0c\u0c0d\u0c0e\u0c0f", - "\u0c10\u0c11\u0c12\u0c13\u0c14\u0c15\u0c16\u0c17\u0c18\u0c19\u0c1a\u0c1b\u0c1c\u0c1d\u0c1e\u0c1f", - "\u0c20\u0c21\u0c22\u0c23\u0c24\u0c25\u0c26\u0c27\u0c28\u0c29\u0c2a\u0c2b\u0c2c\u0c2d\u0c2e\u0c2f", - "\u0c30\u0c31\u0c32\u0c33\u0c34\u0c35\u0c36\u0c37\u0c38\u0c39\u0c3a\u0c3b\u0c3c\u0c3d\u0c3e\u0c3f", - "\u0c40\u0c41\u0c42\u0c43\u0c44\u0c45\u0c46\u0c47\u0c48\u0c49\u0c4a\u0c4b\u0c4c\u0c4d\u0c4e\u0c4f", - "\u0c50\u0c51\u0c52\u0c53\u0c54\u0c55\u0c56\u0c57\u0c58\u0c59\u0c5a\u0c5b\u0c5c\u0c5d\u0c5e\u0c5f", - "\u0c60\u0c61\u0c62\u0c63\u0c64\u0c65\u0c66\u0c67\u0c68\u0c69\u0c6a\u0c6b\u0c6c\u0c6d\u0c6e\u0c6f", - "\u0c70\u0c71\u0c72\u0c73\u0c74\u0c75\u0c76\u0c77\u0c78\u0c79\u0c7a\u0c7b\u0c7c\u0c7d\u0c7e\u0c7f", - "\u0c80\u0c81\u0c82\u0c83\u0c84\u0c85\u0c86\u0c87\u0c88\u0c89\u0c8a\u0c8b\u0c8c\u0c8d\u0c8e\u0c8f", - "\u0c90\u0c91\u0c92\u0c93\u0c94\u0c95\u0c96\u0c97\u0c98\u0c99\u0c9a\u0c9b\u0c9c\u0c9d\u0c9e\u0c9f", - "\u0ca0\u0ca1\u0ca2\u0ca3\u0ca4\u0ca5\u0ca6\u0ca7\u0ca8\u0ca9\u0caa\u0cab\u0cac\u0cad\u0cae\u0caf", - "\u0cb0\u0cb1\u0cb2\u0cb3\u0cb4\u0cb5\u0cb6\u0cb7\u0cb8\u0cb9\u0cba\u0cbb\u0cbc\u0cbd\u0cbe\u0cbf", - "\u0cc0\u0cc1\u0cc2\u0cc3\u0cc4\u0cc5\u0cc6\u0cc7\u0cc8\u0cc9\u0cca\u0ccb\u0ccc\u0ccd\u0cce\u0ccf", - "\u0cd0\u0cd1\u0cd2\u0cd3\u0cd4\u0cd5\u0cd6\u0cd7\u0cd8\u0cd9\u0cda\u0cdb\u0cdc\u0cdd\u0cde\u0cdf", - "\u0ce0\u0ce1\u0ce2\u0ce3\u0ce4\u0ce5\u0ce6\u0ce7\u0ce8\u0ce9\u0cea\u0ceb\u0cec\u0ced\u0cee\u0cef", - "\u0cf0\u0cf1\u0cf2\u0cf3\u0cf4\u0cf5\u0cf6\u0cf7\u0cf8\u0cf9\u0cfa\u0cfb\u0cfc\u0cfd\u0cfe\u0cff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_0d.png", - "ascent": %ascent%, - "chars": [ - "\u0d00\u0d01\u0d02\u0d03\u0d04\u0d05\u0d06\u0d07\u0d08\u0d09\u0d0a\u0d0b\u0d0c\u0d0d\u0d0e\u0d0f", - "\u0d10\u0d11\u0d12\u0d13\u0d14\u0d15\u0d16\u0d17\u0d18\u0d19\u0d1a\u0d1b\u0d1c\u0d1d\u0d1e\u0d1f", - "\u0d20\u0d21\u0d22\u0d23\u0d24\u0d25\u0d26\u0d27\u0d28\u0d29\u0d2a\u0d2b\u0d2c\u0d2d\u0d2e\u0d2f", - "\u0d30\u0d31\u0d32\u0d33\u0d34\u0d35\u0d36\u0d37\u0d38\u0d39\u0d3a\u0d3b\u0d3c\u0d3d\u0d3e\u0d3f", - "\u0d40\u0d41\u0d42\u0d43\u0d44\u0d45\u0d46\u0d47\u0d48\u0d49\u0d4a\u0d4b\u0d4c\u0d4d\u0d4e\u0d4f", - "\u0d50\u0d51\u0d52\u0d53\u0d54\u0d55\u0d56\u0d57\u0d58\u0d59\u0d5a\u0d5b\u0d5c\u0d5d\u0d5e\u0d5f", - "\u0d60\u0d61\u0d62\u0d63\u0d64\u0d65\u0d66\u0d67\u0d68\u0d69\u0d6a\u0d6b\u0d6c\u0d6d\u0d6e\u0d6f", - "\u0d70\u0d71\u0d72\u0d73\u0d74\u0d75\u0d76\u0d77\u0d78\u0d79\u0d7a\u0d7b\u0d7c\u0d7d\u0d7e\u0d7f", - "\u0d80\u0d81\u0d82\u0d83\u0d84\u0d85\u0d86\u0d87\u0d88\u0d89\u0d8a\u0d8b\u0d8c\u0d8d\u0d8e\u0d8f", - "\u0d90\u0d91\u0d92\u0d93\u0d94\u0d95\u0d96\u0d97\u0d98\u0d99\u0d9a\u0d9b\u0d9c\u0d9d\u0d9e\u0d9f", - "\u0da0\u0da1\u0da2\u0da3\u0da4\u0da5\u0da6\u0da7\u0da8\u0da9\u0daa\u0dab\u0dac\u0dad\u0dae\u0daf", - "\u0db0\u0db1\u0db2\u0db3\u0db4\u0db5\u0db6\u0db7\u0db8\u0db9\u0dba\u0dbb\u0dbc\u0dbd\u0dbe\u0dbf", - "\u0dc0\u0dc1\u0dc2\u0dc3\u0dc4\u0dc5\u0dc6\u0dc7\u0dc8\u0dc9\u0dca\u0dcb\u0dcc\u0dcd\u0dce\u0dcf", - "\u0dd0\u0dd1\u0dd2\u0dd3\u0dd4\u0dd5\u0dd6\u0dd7\u0dd8\u0dd9\u0dda\u0ddb\u0ddc\u0ddd\u0dde\u0ddf", - "\u0de0\u0de1\u0de2\u0de3\u0de4\u0de5\u0de6\u0de7\u0de8\u0de9\u0dea\u0deb\u0dec\u0ded\u0dee\u0def", - "\u0df0\u0df1\u0df2\u0df3\u0df4\u0df5\u0df6\u0df7\u0df8\u0df9\u0dfa\u0dfb\u0dfc\u0dfd\u0dfe\u0dff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_0e.png", - "ascent": %ascent%, - "chars": [ - "\u0e00\u0e01\u0e02\u0e03\u0e04\u0e05\u0e06\u0e07\u0e08\u0e09\u0e0a\u0e0b\u0e0c\u0e0d\u0e0e\u0e0f", - "\u0e10\u0e11\u0e12\u0e13\u0e14\u0e15\u0e16\u0e17\u0e18\u0e19\u0e1a\u0e1b\u0e1c\u0e1d\u0e1e\u0e1f", - "\u0e20\u0e21\u0e22\u0e23\u0e24\u0e25\u0e26\u0e27\u0e28\u0e29\u0e2a\u0e2b\u0e2c\u0e2d\u0e2e\u0e2f", - "\u0e30\u0e31\u0e32\u0e33\u0e34\u0e35\u0e36\u0e37\u0e38\u0e39\u0e3a\u0e3b\u0e3c\u0e3d\u0e3e\u0e3f", - "\u0e40\u0e41\u0e42\u0e43\u0e44\u0e45\u0e46\u0e47\u0e48\u0e49\u0e4a\u0e4b\u0e4c\u0e4d\u0e4e\u0e4f", - "\u0e50\u0e51\u0e52\u0e53\u0e54\u0e55\u0e56\u0e57\u0e58\u0e59\u0e5a\u0e5b\u0e5c\u0e5d\u0e5e\u0e5f", - "\u0e60\u0e61\u0e62\u0e63\u0e64\u0e65\u0e66\u0e67\u0e68\u0e69\u0e6a\u0e6b\u0e6c\u0e6d\u0e6e\u0e6f", - "\u0e70\u0e71\u0e72\u0e73\u0e74\u0e75\u0e76\u0e77\u0e78\u0e79\u0e7a\u0e7b\u0e7c\u0e7d\u0e7e\u0e7f", - "\u0e80\u0e81\u0e82\u0e83\u0e84\u0e85\u0e86\u0e87\u0e88\u0e89\u0e8a\u0e8b\u0e8c\u0e8d\u0e8e\u0e8f", - "\u0e90\u0e91\u0e92\u0e93\u0e94\u0e95\u0e96\u0e97\u0e98\u0e99\u0e9a\u0e9b\u0e9c\u0e9d\u0e9e\u0e9f", - "\u0ea0\u0ea1\u0ea2\u0ea3\u0ea4\u0ea5\u0ea6\u0ea7\u0ea8\u0ea9\u0eaa\u0eab\u0eac\u0ead\u0eae\u0eaf", - "\u0eb0\u0eb1\u0eb2\u0eb3\u0eb4\u0eb5\u0eb6\u0eb7\u0eb8\u0eb9\u0eba\u0ebb\u0ebc\u0ebd\u0ebe\u0ebf", - "\u0ec0\u0ec1\u0ec2\u0ec3\u0ec4\u0ec5\u0ec6\u0ec7\u0ec8\u0ec9\u0eca\u0ecb\u0ecc\u0ecd\u0ece\u0ecf", - "\u0ed0\u0ed1\u0ed2\u0ed3\u0ed4\u0ed5\u0ed6\u0ed7\u0ed8\u0ed9\u0eda\u0edb\u0edc\u0edd\u0ede\u0edf", - "\u0ee0\u0ee1\u0ee2\u0ee3\u0ee4\u0ee5\u0ee6\u0ee7\u0ee8\u0ee9\u0eea\u0eeb\u0eec\u0eed\u0eee\u0eef", - "\u0ef0\u0ef1\u0ef2\u0ef3\u0ef4\u0ef5\u0ef6\u0ef7\u0ef8\u0ef9\u0efa\u0efb\u0efc\u0efd\u0efe\u0eff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_0f.png", - "ascent": %ascent%, - "chars": [ - "\u0f00\u0f01\u0f02\u0f03\u0f04\u0f05\u0f06\u0f07\u0f08\u0f09\u0f0a\u0f0b\u0f0c\u0f0d\u0f0e\u0f0f", - "\u0f10\u0f11\u0f12\u0f13\u0f14\u0f15\u0f16\u0f17\u0f18\u0f19\u0f1a\u0f1b\u0f1c\u0f1d\u0f1e\u0f1f", - "\u0f20\u0f21\u0f22\u0f23\u0f24\u0f25\u0f26\u0f27\u0f28\u0f29\u0f2a\u0f2b\u0f2c\u0f2d\u0f2e\u0f2f", - "\u0f30\u0f31\u0f32\u0f33\u0f34\u0f35\u0f36\u0f37\u0f38\u0f39\u0f3a\u0f3b\u0f3c\u0f3d\u0f3e\u0f3f", - "\u0f40\u0f41\u0f42\u0f43\u0f44\u0f45\u0f46\u0f47\u0f48\u0f49\u0f4a\u0f4b\u0f4c\u0f4d\u0f4e\u0f4f", - "\u0f50\u0f51\u0f52\u0f53\u0f54\u0f55\u0f56\u0f57\u0f58\u0f59\u0f5a\u0f5b\u0f5c\u0f5d\u0f5e\u0f5f", - "\u0f60\u0f61\u0f62\u0f63\u0f64\u0f65\u0f66\u0f67\u0f68\u0f69\u0f6a\u0f6b\u0f6c\u0f6d\u0f6e\u0f6f", - "\u0f70\u0f71\u0f72\u0f73\u0f74\u0f75\u0f76\u0f77\u0f78\u0f79\u0f7a\u0f7b\u0f7c\u0f7d\u0f7e\u0f7f", - "\u0f80\u0f81\u0f82\u0f83\u0f84\u0f85\u0f86\u0f87\u0f88\u0f89\u0f8a\u0f8b\u0f8c\u0f8d\u0f8e\u0f8f", - "\u0f90\u0f91\u0f92\u0f93\u0f94\u0f95\u0f96\u0f97\u0f98\u0f99\u0f9a\u0f9b\u0f9c\u0f9d\u0f9e\u0f9f", - "\u0fa0\u0fa1\u0fa2\u0fa3\u0fa4\u0fa5\u0fa6\u0fa7\u0fa8\u0fa9\u0faa\u0fab\u0fac\u0fad\u0fae\u0faf", - "\u0fb0\u0fb1\u0fb2\u0fb3\u0fb4\u0fb5\u0fb6\u0fb7\u0fb8\u0fb9\u0fba\u0fbb\u0fbc\u0fbd\u0fbe\u0fbf", - "\u0fc0\u0fc1\u0fc2\u0fc3\u0fc4\u0fc5\u0fc6\u0fc7\u0fc8\u0fc9\u0fca\u0fcb\u0fcc\u0fcd\u0fce\u0fcf", - "\u0fd0\u0fd1\u0fd2\u0fd3\u0fd4\u0fd5\u0fd6\u0fd7\u0fd8\u0fd9\u0fda\u0fdb\u0fdc\u0fdd\u0fde\u0fdf", - "\u0fe0\u0fe1\u0fe2\u0fe3\u0fe4\u0fe5\u0fe6\u0fe7\u0fe8\u0fe9\u0fea\u0feb\u0fec\u0fed\u0fee\u0fef", - "\u0ff0\u0ff1\u0ff2\u0ff3\u0ff4\u0ff5\u0ff6\u0ff7\u0ff8\u0ff9\u0ffa\u0ffb\u0ffc\u0ffd\u0ffe\u0fff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_10.png", - "ascent": %ascent%, - "chars": [ - "\u1000\u1001\u1002\u1003\u1004\u1005\u1006\u1007\u1008\u1009\u100a\u100b\u100c\u100d\u100e\u100f", - "\u1010\u1011\u1012\u1013\u1014\u1015\u1016\u1017\u1018\u1019\u101a\u101b\u101c\u101d\u101e\u101f", - "\u1020\u1021\u1022\u1023\u1024\u1025\u1026\u1027\u1028\u1029\u102a\u102b\u102c\u102d\u102e\u102f", - "\u1030\u1031\u1032\u1033\u1034\u1035\u1036\u1037\u1038\u1039\u103a\u103b\u103c\u103d\u103e\u103f", - "\u1040\u1041\u1042\u1043\u1044\u1045\u1046\u1047\u1048\u1049\u104a\u104b\u104c\u104d\u104e\u104f", - "\u1050\u1051\u1052\u1053\u1054\u1055\u1056\u1057\u1058\u1059\u105a\u105b\u105c\u105d\u105e\u105f", - "\u1060\u1061\u1062\u1063\u1064\u1065\u1066\u1067\u1068\u1069\u106a\u106b\u106c\u106d\u106e\u106f", - "\u1070\u1071\u1072\u1073\u1074\u1075\u1076\u1077\u1078\u1079\u107a\u107b\u107c\u107d\u107e\u107f", - "\u1080\u1081\u1082\u1083\u1084\u1085\u1086\u1087\u1088\u1089\u108a\u108b\u108c\u108d\u108e\u108f", - "\u1090\u1091\u1092\u1093\u1094\u1095\u1096\u1097\u1098\u1099\u109a\u109b\u109c\u109d\u109e\u109f", - "\u10a0\u10a1\u10a2\u10a3\u10a4\u10a5\u10a6\u10a7\u10a8\u10a9\u10aa\u10ab\u10ac\u10ad\u10ae\u10af", - "\u10b0\u10b1\u10b2\u10b3\u10b4\u10b5\u10b6\u10b7\u10b8\u10b9\u10ba\u10bb\u10bc\u10bd\u10be\u10bf", - "\u10c0\u10c1\u10c2\u10c3\u10c4\u10c5\u10c6\u10c7\u10c8\u10c9\u10ca\u10cb\u10cc\u10cd\u10ce\u10cf", - "\u10d0\u10d1\u10d2\u10d3\u10d4\u10d5\u10d6\u10d7\u10d8\u10d9\u10da\u10db\u10dc\u10dd\u10de\u10df", - "\u10e0\u10e1\u10e2\u10e3\u10e4\u10e5\u10e6\u10e7\u10e8\u10e9\u10ea\u10eb\u10ec\u10ed\u10ee\u10ef", - "\u10f0\u10f1\u10f2\u10f3\u10f4\u10f5\u10f6\u10f7\u10f8\u10f9\u10fa\u10fb\u10fc\u10fd\u10fe\u10ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_11.png", - "ascent": %ascent%, - "chars": [ - "\u1100\u1101\u1102\u1103\u1104\u1105\u1106\u1107\u1108\u1109\u110a\u110b\u110c\u110d\u110e\u110f", - "\u1110\u1111\u1112\u1113\u1114\u1115\u1116\u1117\u1118\u1119\u111a\u111b\u111c\u111d\u111e\u111f", - "\u1120\u1121\u1122\u1123\u1124\u1125\u1126\u1127\u1128\u1129\u112a\u112b\u112c\u112d\u112e\u112f", - "\u1130\u1131\u1132\u1133\u1134\u1135\u1136\u1137\u1138\u1139\u113a\u113b\u113c\u113d\u113e\u113f", - "\u1140\u1141\u1142\u1143\u1144\u1145\u1146\u1147\u1148\u1149\u114a\u114b\u114c\u114d\u114e\u114f", - "\u1150\u1151\u1152\u1153\u1154\u1155\u1156\u1157\u1158\u1159\u115a\u115b\u115c\u115d\u115e\u115f", - "\u1160\u1161\u1162\u1163\u1164\u1165\u1166\u1167\u1168\u1169\u116a\u116b\u116c\u116d\u116e\u116f", - "\u1170\u1171\u1172\u1173\u1174\u1175\u1176\u1177\u1178\u1179\u117a\u117b\u117c\u117d\u117e\u117f", - "\u1180\u1181\u1182\u1183\u1184\u1185\u1186\u1187\u1188\u1189\u118a\u118b\u118c\u118d\u118e\u118f", - "\u1190\u1191\u1192\u1193\u1194\u1195\u1196\u1197\u1198\u1199\u119a\u119b\u119c\u119d\u119e\u119f", - "\u11a0\u11a1\u11a2\u11a3\u11a4\u11a5\u11a6\u11a7\u11a8\u11a9\u11aa\u11ab\u11ac\u11ad\u11ae\u11af", - "\u11b0\u11b1\u11b2\u11b3\u11b4\u11b5\u11b6\u11b7\u11b8\u11b9\u11ba\u11bb\u11bc\u11bd\u11be\u11bf", - "\u11c0\u11c1\u11c2\u11c3\u11c4\u11c5\u11c6\u11c7\u11c8\u11c9\u11ca\u11cb\u11cc\u11cd\u11ce\u11cf", - "\u11d0\u11d1\u11d2\u11d3\u11d4\u11d5\u11d6\u11d7\u11d8\u11d9\u11da\u11db\u11dc\u11dd\u11de\u11df", - "\u11e0\u11e1\u11e2\u11e3\u11e4\u11e5\u11e6\u11e7\u11e8\u11e9\u11ea\u11eb\u11ec\u11ed\u11ee\u11ef", - "\u11f0\u11f1\u11f2\u11f3\u11f4\u11f5\u11f6\u11f7\u11f8\u11f9\u11fa\u11fb\u11fc\u11fd\u11fe\u11ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_12.png", - "ascent": %ascent%, - "chars": [ - "\u1200\u1201\u1202\u1203\u1204\u1205\u1206\u1207\u1208\u1209\u120a\u120b\u120c\u120d\u120e\u120f", - "\u1210\u1211\u1212\u1213\u1214\u1215\u1216\u1217\u1218\u1219\u121a\u121b\u121c\u121d\u121e\u121f", - "\u1220\u1221\u1222\u1223\u1224\u1225\u1226\u1227\u1228\u1229\u122a\u122b\u122c\u122d\u122e\u122f", - "\u1230\u1231\u1232\u1233\u1234\u1235\u1236\u1237\u1238\u1239\u123a\u123b\u123c\u123d\u123e\u123f", - "\u1240\u1241\u1242\u1243\u1244\u1245\u1246\u1247\u1248\u1249\u124a\u124b\u124c\u124d\u124e\u124f", - "\u1250\u1251\u1252\u1253\u1254\u1255\u1256\u1257\u1258\u1259\u125a\u125b\u125c\u125d\u125e\u125f", - "\u1260\u1261\u1262\u1263\u1264\u1265\u1266\u1267\u1268\u1269\u126a\u126b\u126c\u126d\u126e\u126f", - "\u1270\u1271\u1272\u1273\u1274\u1275\u1276\u1277\u1278\u1279\u127a\u127b\u127c\u127d\u127e\u127f", - "\u1280\u1281\u1282\u1283\u1284\u1285\u1286\u1287\u1288\u1289\u128a\u128b\u128c\u128d\u128e\u128f", - "\u1290\u1291\u1292\u1293\u1294\u1295\u1296\u1297\u1298\u1299\u129a\u129b\u129c\u129d\u129e\u129f", - "\u12a0\u12a1\u12a2\u12a3\u12a4\u12a5\u12a6\u12a7\u12a8\u12a9\u12aa\u12ab\u12ac\u12ad\u12ae\u12af", - "\u12b0\u12b1\u12b2\u12b3\u12b4\u12b5\u12b6\u12b7\u12b8\u12b9\u12ba\u12bb\u12bc\u12bd\u12be\u12bf", - "\u12c0\u12c1\u12c2\u12c3\u12c4\u12c5\u12c6\u12c7\u12c8\u12c9\u12ca\u12cb\u12cc\u12cd\u12ce\u12cf", - "\u12d0\u12d1\u12d2\u12d3\u12d4\u12d5\u12d6\u12d7\u12d8\u12d9\u12da\u12db\u12dc\u12dd\u12de\u12df", - "\u12e0\u12e1\u12e2\u12e3\u12e4\u12e5\u12e6\u12e7\u12e8\u12e9\u12ea\u12eb\u12ec\u12ed\u12ee\u12ef", - "\u12f0\u12f1\u12f2\u12f3\u12f4\u12f5\u12f6\u12f7\u12f8\u12f9\u12fa\u12fb\u12fc\u12fd\u12fe\u12ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_13.png", - "ascent": %ascent%, - "chars": [ - "\u1300\u1301\u1302\u1303\u1304\u1305\u1306\u1307\u1308\u1309\u130a\u130b\u130c\u130d\u130e\u130f", - "\u1310\u1311\u1312\u1313\u1314\u1315\u1316\u1317\u1318\u1319\u131a\u131b\u131c\u131d\u131e\u131f", - "\u1320\u1321\u1322\u1323\u1324\u1325\u1326\u1327\u1328\u1329\u132a\u132b\u132c\u132d\u132e\u132f", - "\u1330\u1331\u1332\u1333\u1334\u1335\u1336\u1337\u1338\u1339\u133a\u133b\u133c\u133d\u133e\u133f", - "\u1340\u1341\u1342\u1343\u1344\u1345\u1346\u1347\u1348\u1349\u134a\u134b\u134c\u134d\u134e\u134f", - "\u1350\u1351\u1352\u1353\u1354\u1355\u1356\u1357\u1358\u1359\u135a\u135b\u135c\u135d\u135e\u135f", - "\u1360\u1361\u1362\u1363\u1364\u1365\u1366\u1367\u1368\u1369\u136a\u136b\u136c\u136d\u136e\u136f", - "\u1370\u1371\u1372\u1373\u1374\u1375\u1376\u1377\u1378\u1379\u137a\u137b\u137c\u137d\u137e\u137f", - "\u1380\u1381\u1382\u1383\u1384\u1385\u1386\u1387\u1388\u1389\u138a\u138b\u138c\u138d\u138e\u138f", - "\u1390\u1391\u1392\u1393\u1394\u1395\u1396\u1397\u1398\u1399\u139a\u139b\u139c\u139d\u139e\u139f", - "\u13a0\u13a1\u13a2\u13a3\u13a4\u13a5\u13a6\u13a7\u13a8\u13a9\u13aa\u13ab\u13ac\u13ad\u13ae\u13af", - "\u13b0\u13b1\u13b2\u13b3\u13b4\u13b5\u13b6\u13b7\u13b8\u13b9\u13ba\u13bb\u13bc\u13bd\u13be\u13bf", - "\u13c0\u13c1\u13c2\u13c3\u13c4\u13c5\u13c6\u13c7\u13c8\u13c9\u13ca\u13cb\u13cc\u13cd\u13ce\u13cf", - "\u13d0\u13d1\u13d2\u13d3\u13d4\u13d5\u13d6\u13d7\u13d8\u13d9\u13da\u13db\u13dc\u13dd\u13de\u13df", - "\u13e0\u13e1\u13e2\u13e3\u13e4\u13e5\u13e6\u13e7\u13e8\u13e9\u13ea\u13eb\u13ec\u13ed\u13ee\u13ef", - "\u13f0\u13f1\u13f2\u13f3\u13f4\u13f5\u13f6\u13f7\u13f8\u13f9\u13fa\u13fb\u13fc\u13fd\u13fe\u13ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_14.png", - "ascent": %ascent%, - "chars": [ - "\u1400\u1401\u1402\u1403\u1404\u1405\u1406\u1407\u1408\u1409\u140a\u140b\u140c\u140d\u140e\u140f", - "\u1410\u1411\u1412\u1413\u1414\u1415\u1416\u1417\u1418\u1419\u141a\u141b\u141c\u141d\u141e\u141f", - "\u1420\u1421\u1422\u1423\u1424\u1425\u1426\u1427\u1428\u1429\u142a\u142b\u142c\u142d\u142e\u142f", - "\u1430\u1431\u1432\u1433\u1434\u1435\u1436\u1437\u1438\u1439\u143a\u143b\u143c\u143d\u143e\u143f", - "\u1440\u1441\u1442\u1443\u1444\u1445\u1446\u1447\u1448\u1449\u144a\u144b\u144c\u144d\u144e\u144f", - "\u1450\u1451\u1452\u1453\u1454\u1455\u1456\u1457\u1458\u1459\u145a\u145b\u145c\u145d\u145e\u145f", - "\u1460\u1461\u1462\u1463\u1464\u1465\u1466\u1467\u1468\u1469\u146a\u146b\u146c\u146d\u146e\u146f", - "\u1470\u1471\u1472\u1473\u1474\u1475\u1476\u1477\u1478\u1479\u147a\u147b\u147c\u147d\u147e\u147f", - "\u1480\u1481\u1482\u1483\u1484\u1485\u1486\u1487\u1488\u1489\u148a\u148b\u148c\u148d\u148e\u148f", - "\u1490\u1491\u1492\u1493\u1494\u1495\u1496\u1497\u1498\u1499\u149a\u149b\u149c\u149d\u149e\u149f", - "\u14a0\u14a1\u14a2\u14a3\u14a4\u14a5\u14a6\u14a7\u14a8\u14a9\u14aa\u14ab\u14ac\u14ad\u14ae\u14af", - "\u14b0\u14b1\u14b2\u14b3\u14b4\u14b5\u14b6\u14b7\u14b8\u14b9\u14ba\u14bb\u14bc\u14bd\u14be\u14bf", - "\u14c0\u14c1\u14c2\u14c3\u14c4\u14c5\u14c6\u14c7\u14c8\u14c9\u14ca\u14cb\u14cc\u14cd\u14ce\u14cf", - "\u14d0\u14d1\u14d2\u14d3\u14d4\u14d5\u14d6\u14d7\u14d8\u14d9\u14da\u14db\u14dc\u14dd\u14de\u14df", - "\u14e0\u14e1\u14e2\u14e3\u14e4\u14e5\u14e6\u14e7\u14e8\u14e9\u14ea\u14eb\u14ec\u14ed\u14ee\u14ef", - "\u14f0\u14f1\u14f2\u14f3\u14f4\u14f5\u14f6\u14f7\u14f8\u14f9\u14fa\u14fb\u14fc\u14fd\u14fe\u14ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_15.png", - "ascent": %ascent%, - "chars": [ - "\u1500\u1501\u1502\u1503\u1504\u1505\u1506\u1507\u1508\u1509\u150a\u150b\u150c\u150d\u150e\u150f", - "\u1510\u1511\u1512\u1513\u1514\u1515\u1516\u1517\u1518\u1519\u151a\u151b\u151c\u151d\u151e\u151f", - "\u1520\u1521\u1522\u1523\u1524\u1525\u1526\u1527\u1528\u1529\u152a\u152b\u152c\u152d\u152e\u152f", - "\u1530\u1531\u1532\u1533\u1534\u1535\u1536\u1537\u1538\u1539\u153a\u153b\u153c\u153d\u153e\u153f", - "\u1540\u1541\u1542\u1543\u1544\u1545\u1546\u1547\u1548\u1549\u154a\u154b\u154c\u154d\u154e\u154f", - "\u1550\u1551\u1552\u1553\u1554\u1555\u1556\u1557\u1558\u1559\u155a\u155b\u155c\u155d\u155e\u155f", - "\u1560\u1561\u1562\u1563\u1564\u1565\u1566\u1567\u1568\u1569\u156a\u156b\u156c\u156d\u156e\u156f", - "\u1570\u1571\u1572\u1573\u1574\u1575\u1576\u1577\u1578\u1579\u157a\u157b\u157c\u157d\u157e\u157f", - "\u1580\u1581\u1582\u1583\u1584\u1585\u1586\u1587\u1588\u1589\u158a\u158b\u158c\u158d\u158e\u158f", - "\u1590\u1591\u1592\u1593\u1594\u1595\u1596\u1597\u1598\u1599\u159a\u159b\u159c\u159d\u159e\u159f", - "\u15a0\u15a1\u15a2\u15a3\u15a4\u15a5\u15a6\u15a7\u15a8\u15a9\u15aa\u15ab\u15ac\u15ad\u15ae\u15af", - "\u15b0\u15b1\u15b2\u15b3\u15b4\u15b5\u15b6\u15b7\u15b8\u15b9\u15ba\u15bb\u15bc\u15bd\u15be\u15bf", - "\u15c0\u15c1\u15c2\u15c3\u15c4\u15c5\u15c6\u15c7\u15c8\u15c9\u15ca\u15cb\u15cc\u15cd\u15ce\u15cf", - "\u15d0\u15d1\u15d2\u15d3\u15d4\u15d5\u15d6\u15d7\u15d8\u15d9\u15da\u15db\u15dc\u15dd\u15de\u15df", - "\u15e0\u15e1\u15e2\u15e3\u15e4\u15e5\u15e6\u15e7\u15e8\u15e9\u15ea\u15eb\u15ec\u15ed\u15ee\u15ef", - "\u15f0\u15f1\u15f2\u15f3\u15f4\u15f5\u15f6\u15f7\u15f8\u15f9\u15fa\u15fb\u15fc\u15fd\u15fe\u15ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_16.png", - "ascent": %ascent%, - "chars": [ - "\u1600\u1601\u1602\u1603\u1604\u1605\u1606\u1607\u1608\u1609\u160a\u160b\u160c\u160d\u160e\u160f", - "\u1610\u1611\u1612\u1613\u1614\u1615\u1616\u1617\u1618\u1619\u161a\u161b\u161c\u161d\u161e\u161f", - "\u1620\u1621\u1622\u1623\u1624\u1625\u1626\u1627\u1628\u1629\u162a\u162b\u162c\u162d\u162e\u162f", - "\u1630\u1631\u1632\u1633\u1634\u1635\u1636\u1637\u1638\u1639\u163a\u163b\u163c\u163d\u163e\u163f", - "\u1640\u1641\u1642\u1643\u1644\u1645\u1646\u1647\u1648\u1649\u164a\u164b\u164c\u164d\u164e\u164f", - "\u1650\u1651\u1652\u1653\u1654\u1655\u1656\u1657\u1658\u1659\u165a\u165b\u165c\u165d\u165e\u165f", - "\u1660\u1661\u1662\u1663\u1664\u1665\u1666\u1667\u1668\u1669\u166a\u166b\u166c\u166d\u166e\u166f", - "\u1670\u1671\u1672\u1673\u1674\u1675\u1676\u1677\u1678\u1679\u167a\u167b\u167c\u167d\u167e\u167f", - "\u1680\u1681\u1682\u1683\u1684\u1685\u1686\u1687\u1688\u1689\u168a\u168b\u168c\u168d\u168e\u168f", - "\u1690\u1691\u1692\u1693\u1694\u1695\u1696\u1697\u1698\u1699\u169a\u169b\u169c\u169d\u169e\u169f", - "\u16a0\u16a1\u16a2\u16a3\u16a4\u16a5\u16a6\u16a7\u16a8\u16a9\u16aa\u16ab\u16ac\u16ad\u16ae\u16af", - "\u16b0\u16b1\u16b2\u16b3\u16b4\u16b5\u16b6\u16b7\u16b8\u16b9\u16ba\u16bb\u16bc\u16bd\u16be\u16bf", - "\u16c0\u16c1\u16c2\u16c3\u16c4\u16c5\u16c6\u16c7\u16c8\u16c9\u16ca\u16cb\u16cc\u16cd\u16ce\u16cf", - "\u16d0\u16d1\u16d2\u16d3\u16d4\u16d5\u16d6\u16d7\u16d8\u16d9\u16da\u16db\u16dc\u16dd\u16de\u16df", - "\u16e0\u16e1\u16e2\u16e3\u16e4\u16e5\u16e6\u16e7\u16e8\u16e9\u16ea\u16eb\u16ec\u16ed\u16ee\u16ef", - "\u16f0\u16f1\u16f2\u16f3\u16f4\u16f5\u16f6\u16f7\u16f8\u16f9\u16fa\u16fb\u16fc\u16fd\u16fe\u16ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_17.png", - "ascent": %ascent%, - "chars": [ - "\u1700\u1701\u1702\u1703\u1704\u1705\u1706\u1707\u1708\u1709\u170a\u170b\u170c\u170d\u170e\u170f", - "\u1710\u1711\u1712\u1713\u1714\u1715\u1716\u1717\u1718\u1719\u171a\u171b\u171c\u171d\u171e\u171f", - "\u1720\u1721\u1722\u1723\u1724\u1725\u1726\u1727\u1728\u1729\u172a\u172b\u172c\u172d\u172e\u172f", - "\u1730\u1731\u1732\u1733\u1734\u1735\u1736\u1737\u1738\u1739\u173a\u173b\u173c\u173d\u173e\u173f", - "\u1740\u1741\u1742\u1743\u1744\u1745\u1746\u1747\u1748\u1749\u174a\u174b\u174c\u174d\u174e\u174f", - "\u1750\u1751\u1752\u1753\u1754\u1755\u1756\u1757\u1758\u1759\u175a\u175b\u175c\u175d\u175e\u175f", - "\u1760\u1761\u1762\u1763\u1764\u1765\u1766\u1767\u1768\u1769\u176a\u176b\u176c\u176d\u176e\u176f", - "\u1770\u1771\u1772\u1773\u1774\u1775\u1776\u1777\u1778\u1779\u177a\u177b\u177c\u177d\u177e\u177f", - "\u1780\u1781\u1782\u1783\u1784\u1785\u1786\u1787\u1788\u1789\u178a\u178b\u178c\u178d\u178e\u178f", - "\u1790\u1791\u1792\u1793\u1794\u1795\u1796\u1797\u1798\u1799\u179a\u179b\u179c\u179d\u179e\u179f", - "\u17a0\u17a1\u17a2\u17a3\u17a4\u17a5\u17a6\u17a7\u17a8\u17a9\u17aa\u17ab\u17ac\u17ad\u17ae\u17af", - "\u17b0\u17b1\u17b2\u17b3\u17b4\u17b5\u17b6\u17b7\u17b8\u17b9\u17ba\u17bb\u17bc\u17bd\u17be\u17bf", - "\u17c0\u17c1\u17c2\u17c3\u17c4\u17c5\u17c6\u17c7\u17c8\u17c9\u17ca\u17cb\u17cc\u17cd\u17ce\u17cf", - "\u17d0\u17d1\u17d2\u17d3\u17d4\u17d5\u17d6\u17d7\u17d8\u17d9\u17da\u17db\u17dc\u17dd\u17de\u17df", - "\u17e0\u17e1\u17e2\u17e3\u17e4\u17e5\u17e6\u17e7\u17e8\u17e9\u17ea\u17eb\u17ec\u17ed\u17ee\u17ef", - "\u17f0\u17f1\u17f2\u17f3\u17f4\u17f5\u17f6\u17f7\u17f8\u17f9\u17fa\u17fb\u17fc\u17fd\u17fe\u17ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_18.png", - "ascent": %ascent%, - "chars": [ - "\u1800\u1801\u1802\u1803\u1804\u1805\u1806\u1807\u1808\u1809\u180a\u180b\u180c\u180d\u180e\u180f", - "\u1810\u1811\u1812\u1813\u1814\u1815\u1816\u1817\u1818\u1819\u181a\u181b\u181c\u181d\u181e\u181f", - "\u1820\u1821\u1822\u1823\u1824\u1825\u1826\u1827\u1828\u1829\u182a\u182b\u182c\u182d\u182e\u182f", - "\u1830\u1831\u1832\u1833\u1834\u1835\u1836\u1837\u1838\u1839\u183a\u183b\u183c\u183d\u183e\u183f", - "\u1840\u1841\u1842\u1843\u1844\u1845\u1846\u1847\u1848\u1849\u184a\u184b\u184c\u184d\u184e\u184f", - "\u1850\u1851\u1852\u1853\u1854\u1855\u1856\u1857\u1858\u1859\u185a\u185b\u185c\u185d\u185e\u185f", - "\u1860\u1861\u1862\u1863\u1864\u1865\u1866\u1867\u1868\u1869\u186a\u186b\u186c\u186d\u186e\u186f", - "\u1870\u1871\u1872\u1873\u1874\u1875\u1876\u1877\u1878\u1879\u187a\u187b\u187c\u187d\u187e\u187f", - "\u1880\u1881\u1882\u1883\u1884\u1885\u1886\u1887\u1888\u1889\u188a\u188b\u188c\u188d\u188e\u188f", - "\u1890\u1891\u1892\u1893\u1894\u1895\u1896\u1897\u1898\u1899\u189a\u189b\u189c\u189d\u189e\u189f", - "\u18a0\u18a1\u18a2\u18a3\u18a4\u18a5\u18a6\u18a7\u18a8\u18a9\u18aa\u18ab\u18ac\u18ad\u18ae\u18af", - "\u18b0\u18b1\u18b2\u18b3\u18b4\u18b5\u18b6\u18b7\u18b8\u18b9\u18ba\u18bb\u18bc\u18bd\u18be\u18bf", - "\u18c0\u18c1\u18c2\u18c3\u18c4\u18c5\u18c6\u18c7\u18c8\u18c9\u18ca\u18cb\u18cc\u18cd\u18ce\u18cf", - "\u18d0\u18d1\u18d2\u18d3\u18d4\u18d5\u18d6\u18d7\u18d8\u18d9\u18da\u18db\u18dc\u18dd\u18de\u18df", - "\u18e0\u18e1\u18e2\u18e3\u18e4\u18e5\u18e6\u18e7\u18e8\u18e9\u18ea\u18eb\u18ec\u18ed\u18ee\u18ef", - "\u18f0\u18f1\u18f2\u18f3\u18f4\u18f5\u18f6\u18f7\u18f8\u18f9\u18fa\u18fb\u18fc\u18fd\u18fe\u18ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_19.png", - "ascent": %ascent%, - "chars": [ - "\u1900\u1901\u1902\u1903\u1904\u1905\u1906\u1907\u1908\u1909\u190a\u190b\u190c\u190d\u190e\u190f", - "\u1910\u1911\u1912\u1913\u1914\u1915\u1916\u1917\u1918\u1919\u191a\u191b\u191c\u191d\u191e\u191f", - "\u1920\u1921\u1922\u1923\u1924\u1925\u1926\u1927\u1928\u1929\u192a\u192b\u192c\u192d\u192e\u192f", - "\u1930\u1931\u1932\u1933\u1934\u1935\u1936\u1937\u1938\u1939\u193a\u193b\u193c\u193d\u193e\u193f", - "\u1940\u1941\u1942\u1943\u1944\u1945\u1946\u1947\u1948\u1949\u194a\u194b\u194c\u194d\u194e\u194f", - "\u1950\u1951\u1952\u1953\u1954\u1955\u1956\u1957\u1958\u1959\u195a\u195b\u195c\u195d\u195e\u195f", - "\u1960\u1961\u1962\u1963\u1964\u1965\u1966\u1967\u1968\u1969\u196a\u196b\u196c\u196d\u196e\u196f", - "\u1970\u1971\u1972\u1973\u1974\u1975\u1976\u1977\u1978\u1979\u197a\u197b\u197c\u197d\u197e\u197f", - "\u1980\u1981\u1982\u1983\u1984\u1985\u1986\u1987\u1988\u1989\u198a\u198b\u198c\u198d\u198e\u198f", - "\u1990\u1991\u1992\u1993\u1994\u1995\u1996\u1997\u1998\u1999\u199a\u199b\u199c\u199d\u199e\u199f", - "\u19a0\u19a1\u19a2\u19a3\u19a4\u19a5\u19a6\u19a7\u19a8\u19a9\u19aa\u19ab\u19ac\u19ad\u19ae\u19af", - "\u19b0\u19b1\u19b2\u19b3\u19b4\u19b5\u19b6\u19b7\u19b8\u19b9\u19ba\u19bb\u19bc\u19bd\u19be\u19bf", - "\u19c0\u19c1\u19c2\u19c3\u19c4\u19c5\u19c6\u19c7\u19c8\u19c9\u19ca\u19cb\u19cc\u19cd\u19ce\u19cf", - "\u19d0\u19d1\u19d2\u19d3\u19d4\u19d5\u19d6\u19d7\u19d8\u19d9\u19da\u19db\u19dc\u19dd\u19de\u19df", - "\u19e0\u19e1\u19e2\u19e3\u19e4\u19e5\u19e6\u19e7\u19e8\u19e9\u19ea\u19eb\u19ec\u19ed\u19ee\u19ef", - "\u19f0\u19f1\u19f2\u19f3\u19f4\u19f5\u19f6\u19f7\u19f8\u19f9\u19fa\u19fb\u19fc\u19fd\u19fe\u19ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_1a.png", - "ascent": %ascent%, - "chars": [ - "\u1a00\u1a01\u1a02\u1a03\u1a04\u1a05\u1a06\u1a07\u1a08\u1a09\u1a0a\u1a0b\u1a0c\u1a0d\u1a0e\u1a0f", - "\u1a10\u1a11\u1a12\u1a13\u1a14\u1a15\u1a16\u1a17\u1a18\u1a19\u1a1a\u1a1b\u1a1c\u1a1d\u1a1e\u1a1f", - "\u1a20\u1a21\u1a22\u1a23\u1a24\u1a25\u1a26\u1a27\u1a28\u1a29\u1a2a\u1a2b\u1a2c\u1a2d\u1a2e\u1a2f", - "\u1a30\u1a31\u1a32\u1a33\u1a34\u1a35\u1a36\u1a37\u1a38\u1a39\u1a3a\u1a3b\u1a3c\u1a3d\u1a3e\u1a3f", - "\u1a40\u1a41\u1a42\u1a43\u1a44\u1a45\u1a46\u1a47\u1a48\u1a49\u1a4a\u1a4b\u1a4c\u1a4d\u1a4e\u1a4f", - "\u1a50\u1a51\u1a52\u1a53\u1a54\u1a55\u1a56\u1a57\u1a58\u1a59\u1a5a\u1a5b\u1a5c\u1a5d\u1a5e\u1a5f", - "\u1a60\u1a61\u1a62\u1a63\u1a64\u1a65\u1a66\u1a67\u1a68\u1a69\u1a6a\u1a6b\u1a6c\u1a6d\u1a6e\u1a6f", - "\u1a70\u1a71\u1a72\u1a73\u1a74\u1a75\u1a76\u1a77\u1a78\u1a79\u1a7a\u1a7b\u1a7c\u1a7d\u1a7e\u1a7f", - "\u1a80\u1a81\u1a82\u1a83\u1a84\u1a85\u1a86\u1a87\u1a88\u1a89\u1a8a\u1a8b\u1a8c\u1a8d\u1a8e\u1a8f", - "\u1a90\u1a91\u1a92\u1a93\u1a94\u1a95\u1a96\u1a97\u1a98\u1a99\u1a9a\u1a9b\u1a9c\u1a9d\u1a9e\u1a9f", - "\u1aa0\u1aa1\u1aa2\u1aa3\u1aa4\u1aa5\u1aa6\u1aa7\u1aa8\u1aa9\u1aaa\u1aab\u1aac\u1aad\u1aae\u1aaf", - "\u1ab0\u1ab1\u1ab2\u1ab3\u1ab4\u1ab5\u1ab6\u1ab7\u1ab8\u1ab9\u1aba\u1abb\u1abc\u1abd\u1abe\u1abf", - "\u1ac0\u1ac1\u1ac2\u1ac3\u1ac4\u1ac5\u1ac6\u1ac7\u1ac8\u1ac9\u1aca\u1acb\u1acc\u1acd\u1ace\u1acf", - "\u1ad0\u1ad1\u1ad2\u1ad3\u1ad4\u1ad5\u1ad6\u1ad7\u1ad8\u1ad9\u1ada\u1adb\u1adc\u1add\u1ade\u1adf", - "\u1ae0\u1ae1\u1ae2\u1ae3\u1ae4\u1ae5\u1ae6\u1ae7\u1ae8\u1ae9\u1aea\u1aeb\u1aec\u1aed\u1aee\u1aef", - "\u1af0\u1af1\u1af2\u1af3\u1af4\u1af5\u1af6\u1af7\u1af8\u1af9\u1afa\u1afb\u1afc\u1afd\u1afe\u1aff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_1b.png", - "ascent": %ascent%, - "chars": [ - "\u1b00\u1b01\u1b02\u1b03\u1b04\u1b05\u1b06\u1b07\u1b08\u1b09\u1b0a\u1b0b\u1b0c\u1b0d\u1b0e\u1b0f", - "\u1b10\u1b11\u1b12\u1b13\u1b14\u1b15\u1b16\u1b17\u1b18\u1b19\u1b1a\u1b1b\u1b1c\u1b1d\u1b1e\u1b1f", - "\u1b20\u1b21\u1b22\u1b23\u1b24\u1b25\u1b26\u1b27\u1b28\u1b29\u1b2a\u1b2b\u1b2c\u1b2d\u1b2e\u1b2f", - "\u1b30\u1b31\u1b32\u1b33\u1b34\u1b35\u1b36\u1b37\u1b38\u1b39\u1b3a\u1b3b\u1b3c\u1b3d\u1b3e\u1b3f", - "\u1b40\u1b41\u1b42\u1b43\u1b44\u1b45\u1b46\u1b47\u1b48\u1b49\u1b4a\u1b4b\u1b4c\u1b4d\u1b4e\u1b4f", - "\u1b50\u1b51\u1b52\u1b53\u1b54\u1b55\u1b56\u1b57\u1b58\u1b59\u1b5a\u1b5b\u1b5c\u1b5d\u1b5e\u1b5f", - "\u1b60\u1b61\u1b62\u1b63\u1b64\u1b65\u1b66\u1b67\u1b68\u1b69\u1b6a\u1b6b\u1b6c\u1b6d\u1b6e\u1b6f", - "\u1b70\u1b71\u1b72\u1b73\u1b74\u1b75\u1b76\u1b77\u1b78\u1b79\u1b7a\u1b7b\u1b7c\u1b7d\u1b7e\u1b7f", - "\u1b80\u1b81\u1b82\u1b83\u1b84\u1b85\u1b86\u1b87\u1b88\u1b89\u1b8a\u1b8b\u1b8c\u1b8d\u1b8e\u1b8f", - "\u1b90\u1b91\u1b92\u1b93\u1b94\u1b95\u1b96\u1b97\u1b98\u1b99\u1b9a\u1b9b\u1b9c\u1b9d\u1b9e\u1b9f", - "\u1ba0\u1ba1\u1ba2\u1ba3\u1ba4\u1ba5\u1ba6\u1ba7\u1ba8\u1ba9\u1baa\u1bab\u1bac\u1bad\u1bae\u1baf", - "\u1bb0\u1bb1\u1bb2\u1bb3\u1bb4\u1bb5\u1bb6\u1bb7\u1bb8\u1bb9\u1bba\u1bbb\u1bbc\u1bbd\u1bbe\u1bbf", - "\u1bc0\u1bc1\u1bc2\u1bc3\u1bc4\u1bc5\u1bc6\u1bc7\u1bc8\u1bc9\u1bca\u1bcb\u1bcc\u1bcd\u1bce\u1bcf", - "\u1bd0\u1bd1\u1bd2\u1bd3\u1bd4\u1bd5\u1bd6\u1bd7\u1bd8\u1bd9\u1bda\u1bdb\u1bdc\u1bdd\u1bde\u1bdf", - "\u1be0\u1be1\u1be2\u1be3\u1be4\u1be5\u1be6\u1be7\u1be8\u1be9\u1bea\u1beb\u1bec\u1bed\u1bee\u1bef", - "\u1bf0\u1bf1\u1bf2\u1bf3\u1bf4\u1bf5\u1bf6\u1bf7\u1bf8\u1bf9\u1bfa\u1bfb\u1bfc\u1bfd\u1bfe\u1bff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_1c.png", - "ascent": %ascent%, - "chars": [ - "\u1c00\u1c01\u1c02\u1c03\u1c04\u1c05\u1c06\u1c07\u1c08\u1c09\u1c0a\u1c0b\u1c0c\u1c0d\u1c0e\u1c0f", - "\u1c10\u1c11\u1c12\u1c13\u1c14\u1c15\u1c16\u1c17\u1c18\u1c19\u1c1a\u1c1b\u1c1c\u1c1d\u1c1e\u1c1f", - "\u1c20\u1c21\u1c22\u1c23\u1c24\u1c25\u1c26\u1c27\u1c28\u1c29\u1c2a\u1c2b\u1c2c\u1c2d\u1c2e\u1c2f", - "\u1c30\u1c31\u1c32\u1c33\u1c34\u1c35\u1c36\u1c37\u1c38\u1c39\u1c3a\u1c3b\u1c3c\u1c3d\u1c3e\u1c3f", - "\u1c40\u1c41\u1c42\u1c43\u1c44\u1c45\u1c46\u1c47\u1c48\u1c49\u1c4a\u1c4b\u1c4c\u1c4d\u1c4e\u1c4f", - "\u1c50\u1c51\u1c52\u1c53\u1c54\u1c55\u1c56\u1c57\u1c58\u1c59\u1c5a\u1c5b\u1c5c\u1c5d\u1c5e\u1c5f", - "\u1c60\u1c61\u1c62\u1c63\u1c64\u1c65\u1c66\u1c67\u1c68\u1c69\u1c6a\u1c6b\u1c6c\u1c6d\u1c6e\u1c6f", - "\u1c70\u1c71\u1c72\u1c73\u1c74\u1c75\u1c76\u1c77\u1c78\u1c79\u1c7a\u1c7b\u1c7c\u1c7d\u1c7e\u1c7f", - "\u1c80\u1c81\u1c82\u1c83\u1c84\u1c85\u1c86\u1c87\u1c88\u1c89\u1c8a\u1c8b\u1c8c\u1c8d\u1c8e\u1c8f", - "\u1c90\u1c91\u1c92\u1c93\u1c94\u1c95\u1c96\u1c97\u1c98\u1c99\u1c9a\u1c9b\u1c9c\u1c9d\u1c9e\u1c9f", - "\u1ca0\u1ca1\u1ca2\u1ca3\u1ca4\u1ca5\u1ca6\u1ca7\u1ca8\u1ca9\u1caa\u1cab\u1cac\u1cad\u1cae\u1caf", - "\u1cb0\u1cb1\u1cb2\u1cb3\u1cb4\u1cb5\u1cb6\u1cb7\u1cb8\u1cb9\u1cba\u1cbb\u1cbc\u1cbd\u1cbe\u1cbf", - "\u1cc0\u1cc1\u1cc2\u1cc3\u1cc4\u1cc5\u1cc6\u1cc7\u1cc8\u1cc9\u1cca\u1ccb\u1ccc\u1ccd\u1cce\u1ccf", - "\u1cd0\u1cd1\u1cd2\u1cd3\u1cd4\u1cd5\u1cd6\u1cd7\u1cd8\u1cd9\u1cda\u1cdb\u1cdc\u1cdd\u1cde\u1cdf", - "\u1ce0\u1ce1\u1ce2\u1ce3\u1ce4\u1ce5\u1ce6\u1ce7\u1ce8\u1ce9\u1cea\u1ceb\u1cec\u1ced\u1cee\u1cef", - "\u1cf0\u1cf1\u1cf2\u1cf3\u1cf4\u1cf5\u1cf6\u1cf7\u1cf8\u1cf9\u1cfa\u1cfb\u1cfc\u1cfd\u1cfe\u1cff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_1d.png", - "ascent": %ascent%, - "chars": [ - "\u1d00\u1d01\u1d02\u1d03\u1d04\u1d05\u1d06\u1d07\u1d08\u1d09\u1d0a\u1d0b\u1d0c\u1d0d\u1d0e\u1d0f", - "\u1d10\u1d11\u1d12\u1d13\u1d14\u1d15\u1d16\u1d17\u1d18\u1d19\u1d1a\u1d1b\u1d1c\u1d1d\u1d1e\u1d1f", - "\u1d20\u1d21\u1d22\u1d23\u1d24\u1d25\u1d26\u1d27\u1d28\u1d29\u1d2a\u1d2b\u1d2c\u1d2d\u1d2e\u1d2f", - "\u1d30\u1d31\u1d32\u1d33\u1d34\u1d35\u1d36\u1d37\u1d38\u1d39\u1d3a\u1d3b\u1d3c\u1d3d\u1d3e\u1d3f", - "\u1d40\u1d41\u1d42\u1d43\u1d44\u1d45\u1d46\u1d47\u1d48\u1d49\u1d4a\u1d4b\u1d4c\u1d4d\u1d4e\u1d4f", - "\u1d50\u1d51\u1d52\u1d53\u1d54\u1d55\u1d56\u1d57\u1d58\u1d59\u1d5a\u1d5b\u1d5c\u1d5d\u1d5e\u1d5f", - "\u1d60\u1d61\u1d62\u1d63\u1d64\u1d65\u1d66\u1d67\u1d68\u1d69\u1d6a\u1d6b\u1d6c\u1d6d\u1d6e\u1d6f", - "\u1d70\u1d71\u1d72\u1d73\u1d74\u1d75\u1d76\u1d77\u1d78\u1d79\u1d7a\u1d7b\u1d7c\u1d7d\u1d7e\u1d7f", - "\u1d80\u1d81\u1d82\u1d83\u1d84\u1d85\u1d86\u1d87\u1d88\u1d89\u1d8a\u1d8b\u1d8c\u1d8d\u1d8e\u1d8f", - "\u1d90\u1d91\u1d92\u1d93\u1d94\u1d95\u1d96\u1d97\u1d98\u1d99\u1d9a\u1d9b\u1d9c\u1d9d\u1d9e\u1d9f", - "\u1da0\u1da1\u1da2\u1da3\u1da4\u1da5\u1da6\u1da7\u1da8\u1da9\u1daa\u1dab\u1dac\u1dad\u1dae\u1daf", - "\u1db0\u1db1\u1db2\u1db3\u1db4\u1db5\u1db6\u1db7\u1db8\u1db9\u1dba\u1dbb\u1dbc\u1dbd\u1dbe\u1dbf", - "\u1dc0\u1dc1\u1dc2\u1dc3\u1dc4\u1dc5\u1dc6\u1dc7\u1dc8\u1dc9\u1dca\u1dcb\u1dcc\u1dcd\u1dce\u1dcf", - "\u1dd0\u1dd1\u1dd2\u1dd3\u1dd4\u1dd5\u1dd6\u1dd7\u1dd8\u1dd9\u1dda\u1ddb\u1ddc\u1ddd\u1dde\u1ddf", - "\u1de0\u1de1\u1de2\u1de3\u1de4\u1de5\u1de6\u1de7\u1de8\u1de9\u1dea\u1deb\u1dec\u1ded\u1dee\u1def", - "\u1df0\u1df1\u1df2\u1df3\u1df4\u1df5\u1df6\u1df7\u1df8\u1df9\u1dfa\u1dfb\u1dfc\u1dfd\u1dfe\u1dff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_1e.png", - "ascent": %ascent%, - "chars": [ - "\u1e00\u1e01\u1e02\u1e03\u1e04\u1e05\u1e06\u1e07\u1e08\u1e09\u1e0a\u1e0b\u1e0c\u1e0d\u1e0e\u1e0f", - "\u1e10\u1e11\u1e12\u1e13\u1e14\u1e15\u1e16\u1e17\u1e18\u1e19\u1e1a\u1e1b\u1e1c\u1e1d\u1e1e\u1e1f", - "\u1e20\u1e21\u1e22\u1e23\u1e24\u1e25\u1e26\u1e27\u1e28\u1e29\u1e2a\u1e2b\u1e2c\u1e2d\u1e2e\u1e2f", - "\u1e30\u1e31\u1e32\u1e33\u1e34\u1e35\u1e36\u1e37\u1e38\u1e39\u1e3a\u1e3b\u1e3c\u1e3d\u1e3e\u1e3f", - "\u1e40\u1e41\u1e42\u1e43\u1e44\u1e45\u1e46\u1e47\u1e48\u1e49\u1e4a\u1e4b\u1e4c\u1e4d\u1e4e\u1e4f", - "\u1e50\u1e51\u1e52\u1e53\u1e54\u1e55\u1e56\u1e57\u1e58\u1e59\u1e5a\u1e5b\u1e5c\u1e5d\u1e5e\u1e5f", - "\u1e60\u1e61\u1e62\u1e63\u1e64\u1e65\u1e66\u1e67\u1e68\u1e69\u1e6a\u1e6b\u1e6c\u1e6d\u1e6e\u1e6f", - "\u1e70\u1e71\u1e72\u1e73\u1e74\u1e75\u1e76\u1e77\u1e78\u1e79\u1e7a\u1e7b\u1e7c\u1e7d\u1e7e\u1e7f", - "\u1e80\u1e81\u1e82\u1e83\u1e84\u1e85\u1e86\u1e87\u1e88\u1e89\u1e8a\u1e8b\u1e8c\u1e8d\u1e8e\u1e8f", - "\u1e90\u1e91\u1e92\u1e93\u1e94\u1e95\u1e96\u1e97\u1e98\u1e99\u1e9a\u1e9b\u1e9c\u1e9d\u1e9e\u1e9f", - "\u1ea0\u1ea1\u1ea2\u1ea3\u1ea4\u1ea5\u1ea6\u1ea7\u1ea8\u1ea9\u1eaa\u1eab\u1eac\u1ead\u1eae\u1eaf", - "\u1eb0\u1eb1\u1eb2\u1eb3\u1eb4\u1eb5\u1eb6\u1eb7\u1eb8\u1eb9\u1eba\u1ebb\u1ebc\u1ebd\u1ebe\u1ebf", - "\u1ec0\u1ec1\u1ec2\u1ec3\u1ec4\u1ec5\u1ec6\u1ec7\u1ec8\u1ec9\u1eca\u1ecb\u1ecc\u1ecd\u1ece\u1ecf", - "\u1ed0\u1ed1\u1ed2\u1ed3\u1ed4\u1ed5\u1ed6\u1ed7\u1ed8\u1ed9\u1eda\u1edb\u1edc\u1edd\u1ede\u1edf", - "\u1ee0\u1ee1\u1ee2\u1ee3\u1ee4\u1ee5\u1ee6\u1ee7\u1ee8\u1ee9\u1eea\u1eeb\u1eec\u1eed\u1eee\u1eef", - "\u1ef0\u1ef1\u1ef2\u1ef3\u1ef4\u1ef5\u1ef6\u1ef7\u1ef8\u1ef9\u1efa\u1efb\u1efc\u1efd\u1efe\u1eff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_1f.png", - "ascent": %ascent%, - "chars": [ - "\u1f00\u1f01\u1f02\u1f03\u1f04\u1f05\u1f06\u1f07\u1f08\u1f09\u1f0a\u1f0b\u1f0c\u1f0d\u1f0e\u1f0f", - "\u1f10\u1f11\u1f12\u1f13\u1f14\u1f15\u1f16\u1f17\u1f18\u1f19\u1f1a\u1f1b\u1f1c\u1f1d\u1f1e\u1f1f", - "\u1f20\u1f21\u1f22\u1f23\u1f24\u1f25\u1f26\u1f27\u1f28\u1f29\u1f2a\u1f2b\u1f2c\u1f2d\u1f2e\u1f2f", - "\u1f30\u1f31\u1f32\u1f33\u1f34\u1f35\u1f36\u1f37\u1f38\u1f39\u1f3a\u1f3b\u1f3c\u1f3d\u1f3e\u1f3f", - "\u1f40\u1f41\u1f42\u1f43\u1f44\u1f45\u1f46\u1f47\u1f48\u1f49\u1f4a\u1f4b\u1f4c\u1f4d\u1f4e\u1f4f", - "\u1f50\u1f51\u1f52\u1f53\u1f54\u1f55\u1f56\u1f57\u1f58\u1f59\u1f5a\u1f5b\u1f5c\u1f5d\u1f5e\u1f5f", - "\u1f60\u1f61\u1f62\u1f63\u1f64\u1f65\u1f66\u1f67\u1f68\u1f69\u1f6a\u1f6b\u1f6c\u1f6d\u1f6e\u1f6f", - "\u1f70\u1f71\u1f72\u1f73\u1f74\u1f75\u1f76\u1f77\u1f78\u1f79\u1f7a\u1f7b\u1f7c\u1f7d\u1f7e\u1f7f", - "\u1f80\u1f81\u1f82\u1f83\u1f84\u1f85\u1f86\u1f87\u1f88\u1f89\u1f8a\u1f8b\u1f8c\u1f8d\u1f8e\u1f8f", - "\u1f90\u1f91\u1f92\u1f93\u1f94\u1f95\u1f96\u1f97\u1f98\u1f99\u1f9a\u1f9b\u1f9c\u1f9d\u1f9e\u1f9f", - "\u1fa0\u1fa1\u1fa2\u1fa3\u1fa4\u1fa5\u1fa6\u1fa7\u1fa8\u1fa9\u1faa\u1fab\u1fac\u1fad\u1fae\u1faf", - "\u1fb0\u1fb1\u1fb2\u1fb3\u1fb4\u1fb5\u1fb6\u1fb7\u1fb8\u1fb9\u1fba\u1fbb\u1fbc\u1fbd\u1fbe\u1fbf", - "\u1fc0\u1fc1\u1fc2\u1fc3\u1fc4\u1fc5\u1fc6\u1fc7\u1fc8\u1fc9\u1fca\u1fcb\u1fcc\u1fcd\u1fce\u1fcf", - "\u1fd0\u1fd1\u1fd2\u1fd3\u1fd4\u1fd5\u1fd6\u1fd7\u1fd8\u1fd9\u1fda\u1fdb\u1fdc\u1fdd\u1fde\u1fdf", - "\u1fe0\u1fe1\u1fe2\u1fe3\u1fe4\u1fe5\u1fe6\u1fe7\u1fe8\u1fe9\u1fea\u1feb\u1fec\u1fed\u1fee\u1fef", - "\u1ff0\u1ff1\u1ff2\u1ff3\u1ff4\u1ff5\u1ff6\u1ff7\u1ff8\u1ff9\u1ffa\u1ffb\u1ffc\u1ffd\u1ffe\u1fff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_20.png", - "ascent": %ascent%, - "chars": [ - "\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u200b\u200c\u200d\u200e\u200f", - "\u2010\u2011\u2012\u2013\u2014\u2015\u2016\u2017\u2018\u2019\u201a\u201b\u201c\u201d\u201e\u201f", - "\u2020\u2021\u2022\u2023\u2024\u2025\u2026\u2027\u2028\u2029\u202a\u202b\u202c\u202d\u202e\u202f", - "\u2030\u2031\u2032\u2033\u2034\u2035\u2036\u2037\u2038\u2039\u203a\u203b\u203c\u203d\u203e\u203f", - "\u2040\u2041\u2042\u2043\u2044\u2045\u2046\u2047\u2048\u2049\u204a\u204b\u204c\u204d\u204e\u204f", - "\u2050\u2051\u2052\u2053\u2054\u2055\u2056\u2057\u2058\u2059\u205a\u205b\u205c\u205d\u205e\u205f", - "\u2060\u2061\u2062\u2063\u2064\u2065\u2066\u2067\u2068\u2069\u206a\u206b\u206c\u206d\u206e\u206f", - "\u2070\u2071\u2072\u2073\u2074\u2075\u2076\u2077\u2078\u2079\u207a\u207b\u207c\u207d\u207e\u207f", - "\u2080\u2081\u2082\u2083\u2084\u2085\u2086\u2087\u2088\u2089\u208a\u208b\u208c\u208d\u208e\u208f", - "\u2090\u2091\u2092\u2093\u2094\u2095\u2096\u2097\u2098\u2099\u209a\u209b\u209c\u209d\u209e\u209f", - "\u20a0\u20a1\u20a2\u20a3\u20a4\u20a5\u20a6\u20a7\u20a8\u20a9\u20aa\u20ab\u20ac\u20ad\u20ae\u20af", - "\u20b0\u20b1\u20b2\u20b3\u20b4\u20b5\u20b6\u20b7\u20b8\u20b9\u20ba\u20bb\u20bc\u20bd\u20be\u20bf", - "\u20c0\u20c1\u20c2\u20c3\u20c4\u20c5\u20c6\u20c7\u20c8\u20c9\u20ca\u20cb\u20cc\u20cd\u20ce\u20cf", - "\u20d0\u20d1\u20d2\u20d3\u20d4\u20d5\u20d6\u20d7\u20d8\u20d9\u20da\u20db\u20dc\u20dd\u20de\u20df", - "\u20e0\u20e1\u20e2\u20e3\u20e4\u20e5\u20e6\u20e7\u20e8\u20e9\u20ea\u20eb\u20ec\u20ed\u20ee\u20ef", - "\u20f0\u20f1\u20f2\u20f3\u20f4\u20f5\u20f6\u20f7\u20f8\u20f9\u20fa\u20fb\u20fc\u20fd\u20fe\u20ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_21.png", - "ascent": %ascent%, - "chars": [ - "\u2100\u2101\u2102\u2103\u2104\u2105\u2106\u2107\u2108\u2109\u210a\u210b\u210c\u210d\u210e\u210f", - "\u2110\u2111\u2112\u2113\u2114\u2115\u2116\u2117\u2118\u2119\u211a\u211b\u211c\u211d\u211e\u211f", - "\u2120\u2121\u2122\u2123\u2124\u2125\u2126\u2127\u2128\u2129\u212a\u212b\u212c\u212d\u212e\u212f", - "\u2130\u2131\u2132\u2133\u2134\u2135\u2136\u2137\u2138\u2139\u213a\u213b\u213c\u213d\u213e\u213f", - "\u2140\u2141\u2142\u2143\u2144\u2145\u2146\u2147\u2148\u2149\u214a\u214b\u214c\u214d\u214e\u214f", - "\u2150\u2151\u2152\u2153\u2154\u2155\u2156\u2157\u2158\u2159\u215a\u215b\u215c\u215d\u215e\u215f", - "\u2160\u2161\u2162\u2163\u2164\u2165\u2166\u2167\u2168\u2169\u216a\u216b\u216c\u216d\u216e\u216f", - "\u2170\u2171\u2172\u2173\u2174\u2175\u2176\u2177\u2178\u2179\u217a\u217b\u217c\u217d\u217e\u217f", - "\u2180\u2181\u2182\u2183\u2184\u2185\u2186\u2187\u2188\u2189\u218a\u218b\u218c\u218d\u218e\u218f", - "\u2190\u2191\u2192\u2193\u2194\u2195\u2196\u2197\u2198\u2199\u219a\u219b\u219c\u219d\u219e\u219f", - "\u21a0\u21a1\u21a2\u21a3\u21a4\u21a5\u21a6\u21a7\u21a8\u21a9\u21aa\u21ab\u21ac\u21ad\u21ae\u21af", - "\u21b0\u21b1\u21b2\u21b3\u21b4\u21b5\u21b6\u21b7\u21b8\u21b9\u21ba\u21bb\u21bc\u21bd\u21be\u21bf", - "\u21c0\u21c1\u21c2\u21c3\u21c4\u21c5\u21c6\u21c7\u21c8\u21c9\u21ca\u21cb\u21cc\u21cd\u21ce\u21cf", - "\u21d0\u21d1\u21d2\u21d3\u21d4\u21d5\u21d6\u21d7\u21d8\u21d9\u21da\u21db\u21dc\u21dd\u21de\u21df", - "\u21e0\u21e1\u21e2\u21e3\u21e4\u21e5\u21e6\u21e7\u21e8\u21e9\u21ea\u21eb\u21ec\u21ed\u21ee\u21ef", - "\u21f0\u21f1\u21f2\u21f3\u21f4\u21f5\u21f6\u21f7\u21f8\u21f9\u21fa\u21fb\u21fc\u21fd\u21fe\u21ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_22.png", - "ascent": %ascent%, - "chars": [ - "\u2200\u2201\u2202\u2203\u2204\u2205\u2206\u2207\u2208\u2209\u220a\u220b\u220c\u220d\u220e\u220f", - "\u2210\u2211\u2212\u2213\u2214\u2215\u2216\u2217\u2218\u2219\u221a\u221b\u221c\u221d\u221e\u221f", - "\u2220\u2221\u2222\u2223\u2224\u2225\u2226\u2227\u2228\u2229\u222a\u222b\u222c\u222d\u222e\u222f", - "\u2230\u2231\u2232\u2233\u2234\u2235\u2236\u2237\u2238\u2239\u223a\u223b\u223c\u223d\u223e\u223f", - "\u2240\u2241\u2242\u2243\u2244\u2245\u2246\u2247\u2248\u2249\u224a\u224b\u224c\u224d\u224e\u224f", - "\u2250\u2251\u2252\u2253\u2254\u2255\u2256\u2257\u2258\u2259\u225a\u225b\u225c\u225d\u225e\u225f", - "\u2260\u2261\u2262\u2263\u2264\u2265\u2266\u2267\u2268\u2269\u226a\u226b\u226c\u226d\u226e\u226f", - "\u2270\u2271\u2272\u2273\u2274\u2275\u2276\u2277\u2278\u2279\u227a\u227b\u227c\u227d\u227e\u227f", - "\u2280\u2281\u2282\u2283\u2284\u2285\u2286\u2287\u2288\u2289\u228a\u228b\u228c\u228d\u228e\u228f", - "\u2290\u2291\u2292\u2293\u2294\u2295\u2296\u2297\u2298\u2299\u229a\u229b\u229c\u229d\u229e\u229f", - "\u22a0\u22a1\u22a2\u22a3\u22a4\u22a5\u22a6\u22a7\u22a8\u22a9\u22aa\u22ab\u22ac\u22ad\u22ae\u22af", - "\u22b0\u22b1\u22b2\u22b3\u22b4\u22b5\u22b6\u22b7\u22b8\u22b9\u22ba\u22bb\u22bc\u22bd\u22be\u22bf", - "\u22c0\u22c1\u22c2\u22c3\u22c4\u22c5\u22c6\u22c7\u22c8\u22c9\u22ca\u22cb\u22cc\u22cd\u22ce\u22cf", - "\u22d0\u22d1\u22d2\u22d3\u22d4\u22d5\u22d6\u22d7\u22d8\u22d9\u22da\u22db\u22dc\u22dd\u22de\u22df", - "\u22e0\u22e1\u22e2\u22e3\u22e4\u22e5\u22e6\u22e7\u22e8\u22e9\u22ea\u22eb\u22ec\u22ed\u22ee\u22ef", - "\u22f0\u22f1\u22f2\u22f3\u22f4\u22f5\u22f6\u22f7\u22f8\u22f9\u22fa\u22fb\u22fc\u22fd\u22fe\u22ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_23.png", - "ascent": %ascent%, - "chars": [ - "\u2300\u2301\u2302\u2303\u2304\u2305\u2306\u2307\u2308\u2309\u230a\u230b\u230c\u230d\u230e\u230f", - "\u2310\u2311\u2312\u2313\u2314\u2315\u2316\u2317\u2318\u2319\u231a\u231b\u231c\u231d\u231e\u231f", - "\u2320\u2321\u2322\u2323\u2324\u2325\u2326\u2327\u2328\u2329\u232a\u232b\u232c\u232d\u232e\u232f", - "\u2330\u2331\u2332\u2333\u2334\u2335\u2336\u2337\u2338\u2339\u233a\u233b\u233c\u233d\u233e\u233f", - "\u2340\u2341\u2342\u2343\u2344\u2345\u2346\u2347\u2348\u2349\u234a\u234b\u234c\u234d\u234e\u234f", - "\u2350\u2351\u2352\u2353\u2354\u2355\u2356\u2357\u2358\u2359\u235a\u235b\u235c\u235d\u235e\u235f", - "\u2360\u2361\u2362\u2363\u2364\u2365\u2366\u2367\u2368\u2369\u236a\u236b\u236c\u236d\u236e\u236f", - "\u2370\u2371\u2372\u2373\u2374\u2375\u2376\u2377\u2378\u2379\u237a\u237b\u237c\u237d\u237e\u237f", - "\u2380\u2381\u2382\u2383\u2384\u2385\u2386\u2387\u2388\u2389\u238a\u238b\u238c\u238d\u238e\u238f", - "\u2390\u2391\u2392\u2393\u2394\u2395\u2396\u2397\u2398\u2399\u239a\u239b\u239c\u239d\u239e\u239f", - "\u23a0\u23a1\u23a2\u23a3\u23a4\u23a5\u23a6\u23a7\u23a8\u23a9\u23aa\u23ab\u23ac\u23ad\u23ae\u23af", - "\u23b0\u23b1\u23b2\u23b3\u23b4\u23b5\u23b6\u23b7\u23b8\u23b9\u23ba\u23bb\u23bc\u23bd\u23be\u23bf", - "\u23c0\u23c1\u23c2\u23c3\u23c4\u23c5\u23c6\u23c7\u23c8\u23c9\u23ca\u23cb\u23cc\u23cd\u23ce\u23cf", - "\u23d0\u23d1\u23d2\u23d3\u23d4\u23d5\u23d6\u23d7\u23d8\u23d9\u23da\u23db\u23dc\u23dd\u23de\u23df", - "\u23e0\u23e1\u23e2\u23e3\u23e4\u23e5\u23e6\u23e7\u23e8\u23e9\u23ea\u23eb\u23ec\u23ed\u23ee\u23ef", - "\u23f0\u23f1\u23f2\u23f3\u23f4\u23f5\u23f6\u23f7\u23f8\u23f9\u23fa\u23fb\u23fc\u23fd\u23fe\u23ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_24.png", - "ascent": %ascent%, - "chars": [ - "\u2400\u2401\u2402\u2403\u2404\u2405\u2406\u2407\u2408\u2409\u240a\u240b\u240c\u240d\u240e\u240f", - "\u2410\u2411\u2412\u2413\u2414\u2415\u2416\u2417\u2418\u2419\u241a\u241b\u241c\u241d\u241e\u241f", - "\u2420\u2421\u2422\u2423\u2424\u2425\u2426\u2427\u2428\u2429\u242a\u242b\u242c\u242d\u242e\u242f", - "\u2430\u2431\u2432\u2433\u2434\u2435\u2436\u2437\u2438\u2439\u243a\u243b\u243c\u243d\u243e\u243f", - "\u2440\u2441\u2442\u2443\u2444\u2445\u2446\u2447\u2448\u2449\u244a\u244b\u244c\u244d\u244e\u244f", - "\u2450\u2451\u2452\u2453\u2454\u2455\u2456\u2457\u2458\u2459\u245a\u245b\u245c\u245d\u245e\u245f", - "\u2460\u2461\u2462\u2463\u2464\u2465\u2466\u2467\u2468\u2469\u246a\u246b\u246c\u246d\u246e\u246f", - "\u2470\u2471\u2472\u2473\u2474\u2475\u2476\u2477\u2478\u2479\u247a\u247b\u247c\u247d\u247e\u247f", - "\u2480\u2481\u2482\u2483\u2484\u2485\u2486\u2487\u2488\u2489\u248a\u248b\u248c\u248d\u248e\u248f", - "\u2490\u2491\u2492\u2493\u2494\u2495\u2496\u2497\u2498\u2499\u249a\u249b\u249c\u249d\u249e\u249f", - "\u24a0\u24a1\u24a2\u24a3\u24a4\u24a5\u24a6\u24a7\u24a8\u24a9\u24aa\u24ab\u24ac\u24ad\u24ae\u24af", - "\u24b0\u24b1\u24b2\u24b3\u24b4\u24b5\u24b6\u24b7\u24b8\u24b9\u24ba\u24bb\u24bc\u24bd\u24be\u24bf", - "\u24c0\u24c1\u24c2\u24c3\u24c4\u24c5\u24c6\u24c7\u24c8\u24c9\u24ca\u24cb\u24cc\u24cd\u24ce\u24cf", - "\u24d0\u24d1\u24d2\u24d3\u24d4\u24d5\u24d6\u24d7\u24d8\u24d9\u24da\u24db\u24dc\u24dd\u24de\u24df", - "\u24e0\u24e1\u24e2\u24e3\u24e4\u24e5\u24e6\u24e7\u24e8\u24e9\u24ea\u24eb\u24ec\u24ed\u24ee\u24ef", - "\u24f0\u24f1\u24f2\u24f3\u24f4\u24f5\u24f6\u24f7\u24f8\u24f9\u24fa\u24fb\u24fc\u24fd\u24fe\u24ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_25.png", - "ascent": %ascent%, - "chars": [ - "\u2500\u2501\u2502\u2503\u2504\u2505\u2506\u2507\u2508\u2509\u250a\u250b\u250c\u250d\u250e\u250f", - "\u2510\u2511\u2512\u2513\u2514\u2515\u2516\u2517\u2518\u2519\u251a\u251b\u251c\u251d\u251e\u251f", - "\u2520\u2521\u2522\u2523\u2524\u2525\u2526\u2527\u2528\u2529\u252a\u252b\u252c\u252d\u252e\u252f", - "\u2530\u2531\u2532\u2533\u2534\u2535\u2536\u2537\u2538\u2539\u253a\u253b\u253c\u253d\u253e\u253f", - "\u2540\u2541\u2542\u2543\u2544\u2545\u2546\u2547\u2548\u2549\u254a\u254b\u254c\u254d\u254e\u254f", - "\u2550\u2551\u2552\u2553\u2554\u2555\u2556\u2557\u2558\u2559\u255a\u255b\u255c\u255d\u255e\u255f", - "\u2560\u2561\u2562\u2563\u2564\u2565\u2566\u2567\u2568\u2569\u256a\u256b\u256c\u256d\u256e\u256f", - "\u2570\u2571\u2572\u2573\u2574\u2575\u2576\u2577\u2578\u2579\u257a\u257b\u257c\u257d\u257e\u257f", - "\u2580\u2581\u2582\u2583\u2584\u2585\u2586\u2587\u2588\u2589\u258a\u258b\u258c\u258d\u258e\u258f", - "\u2590\u2591\u2592\u2593\u2594\u2595\u2596\u2597\u2598\u2599\u259a\u259b\u259c\u259d\u259e\u259f", - "\u25a0\u25a1\u25a2\u25a3\u25a4\u25a5\u25a6\u25a7\u25a8\u25a9\u25aa\u25ab\u25ac\u25ad\u25ae\u25af", - "\u25b0\u25b1\u25b2\u25b3\u25b4\u25b5\u25b6\u25b7\u25b8\u25b9\u25ba\u25bb\u25bc\u25bd\u25be\u25bf", - "\u25c0\u25c1\u25c2\u25c3\u25c4\u25c5\u25c6\u25c7\u25c8\u25c9\u25ca\u25cb\u25cc\u25cd\u25ce\u25cf", - "\u25d0\u25d1\u25d2\u25d3\u25d4\u25d5\u25d6\u25d7\u25d8\u25d9\u25da\u25db\u25dc\u25dd\u25de\u25df", - "\u25e0\u25e1\u25e2\u25e3\u25e4\u25e5\u25e6\u25e7\u25e8\u25e9\u25ea\u25eb\u25ec\u25ed\u25ee\u25ef", - "\u25f0\u25f1\u25f2\u25f3\u25f4\u25f5\u25f6\u25f7\u25f8\u25f9\u25fa\u25fb\u25fc\u25fd\u25fe\u25ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_26.png", - "ascent": %ascent%, - "chars": [ - "\u2600\u2601\u2602\u2603\u2604\u2605\u2606\u2607\u2608\u2609\u260a\u260b\u260c\u260d\u260e\u260f", - "\u2610\u2611\u2612\u2613\u2614\u2615\u2616\u2617\u2618\u2619\u261a\u261b\u261c\u261d\u261e\u261f", - "\u2620\u2621\u2622\u2623\u2624\u2625\u2626\u2627\u2628\u2629\u262a\u262b\u262c\u262d\u262e\u262f", - "\u2630\u2631\u2632\u2633\u2634\u2635\u2636\u2637\u2638\u2639\u263a\u263b\u263c\u263d\u263e\u263f", - "\u2640\u2641\u2642\u2643\u2644\u2645\u2646\u2647\u2648\u2649\u264a\u264b\u264c\u264d\u264e\u264f", - "\u2650\u2651\u2652\u2653\u2654\u2655\u2656\u2657\u2658\u2659\u265a\u265b\u265c\u265d\u265e\u265f", - "\u2660\u2661\u2662\u2663\u2664\u2665\u2666\u2667\u2668\u2669\u266a\u266b\u266c\u266d\u266e\u266f", - "\u2670\u2671\u2672\u2673\u2674\u2675\u2676\u2677\u2678\u2679\u267a\u267b\u267c\u267d\u267e\u267f", - "\u2680\u2681\u2682\u2683\u2684\u2685\u2686\u2687\u2688\u2689\u268a\u268b\u268c\u268d\u268e\u268f", - "\u2690\u2691\u2692\u2693\u2694\u2695\u2696\u2697\u2698\u2699\u269a\u269b\u269c\u269d\u269e\u269f", - "\u26a0\u26a1\u26a2\u26a3\u26a4\u26a5\u26a6\u26a7\u26a8\u26a9\u26aa\u26ab\u26ac\u26ad\u26ae\u26af", - "\u26b0\u26b1\u26b2\u26b3\u26b4\u26b5\u26b6\u26b7\u26b8\u26b9\u26ba\u26bb\u26bc\u26bd\u26be\u26bf", - "\u26c0\u26c1\u26c2\u26c3\u26c4\u26c5\u26c6\u26c7\u26c8\u26c9\u26ca\u26cb\u26cc\u26cd\u26ce\u26cf", - "\u26d0\u26d1\u26d2\u26d3\u26d4\u26d5\u26d6\u26d7\u26d8\u26d9\u26da\u26db\u26dc\u26dd\u26de\u26df", - "\u26e0\u26e1\u26e2\u26e3\u26e4\u26e5\u26e6\u26e7\u26e8\u26e9\u26ea\u26eb\u26ec\u26ed\u26ee\u26ef", - "\u26f0\u26f1\u26f2\u26f3\u26f4\u26f5\u26f6\u26f7\u26f8\u26f9\u26fa\u26fb\u26fc\u26fd\u26fe\u26ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_27.png", - "ascent": %ascent%, - "chars": [ - "\u2700\u2701\u2702\u2703\u2704\u2705\u2706\u2707\u2708\u2709\u270a\u270b\u270c\u270d\u270e\u270f", - "\u2710\u2711\u2712\u2713\u2714\u2715\u2716\u2717\u2718\u2719\u271a\u271b\u271c\u271d\u271e\u271f", - "\u2720\u2721\u2722\u2723\u2724\u2725\u2726\u2727\u2728\u2729\u272a\u272b\u272c\u272d\u272e\u272f", - "\u2730\u2731\u2732\u2733\u2734\u2735\u2736\u2737\u2738\u2739\u273a\u273b\u273c\u273d\u273e\u273f", - "\u2740\u2741\u2742\u2743\u2744\u2745\u2746\u2747\u2748\u2749\u274a\u274b\u274c\u274d\u274e\u274f", - "\u2750\u2751\u2752\u2753\u2754\u2755\u2756\u2757\u2758\u2759\u275a\u275b\u275c\u275d\u275e\u275f", - "\u2760\u2761\u2762\u2763\u2764\u2765\u2766\u2767\u2768\u2769\u276a\u276b\u276c\u276d\u276e\u276f", - "\u2770\u2771\u2772\u2773\u2774\u2775\u2776\u2777\u2778\u2779\u277a\u277b\u277c\u277d\u277e\u277f", - "\u2780\u2781\u2782\u2783\u2784\u2785\u2786\u2787\u2788\u2789\u278a\u278b\u278c\u278d\u278e\u278f", - "\u2790\u2791\u2792\u2793\u2794\u2795\u2796\u2797\u2798\u2799\u279a\u279b\u279c\u279d\u279e\u279f", - "\u27a0\u27a1\u27a2\u27a3\u27a4\u27a5\u27a6\u27a7\u27a8\u27a9\u27aa\u27ab\u27ac\u27ad\u27ae\u27af", - "\u27b0\u27b1\u27b2\u27b3\u27b4\u27b5\u27b6\u27b7\u27b8\u27b9\u27ba\u27bb\u27bc\u27bd\u27be\u27bf", - "\u27c0\u27c1\u27c2\u27c3\u27c4\u27c5\u27c6\u27c7\u27c8\u27c9\u27ca\u27cb\u27cc\u27cd\u27ce\u27cf", - "\u27d0\u27d1\u27d2\u27d3\u27d4\u27d5\u27d6\u27d7\u27d8\u27d9\u27da\u27db\u27dc\u27dd\u27de\u27df", - "\u27e0\u27e1\u27e2\u27e3\u27e4\u27e5\u27e6\u27e7\u27e8\u27e9\u27ea\u27eb\u27ec\u27ed\u27ee\u27ef", - "\u27f0\u27f1\u27f2\u27f3\u27f4\u27f5\u27f6\u27f7\u27f8\u27f9\u27fa\u27fb\u27fc\u27fd\u27fe\u27ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_28.png", - "ascent": %ascent%, - "chars": [ - "\u2800\u2801\u2802\u2803\u2804\u2805\u2806\u2807\u2808\u2809\u280a\u280b\u280c\u280d\u280e\u280f", - "\u2810\u2811\u2812\u2813\u2814\u2815\u2816\u2817\u2818\u2819\u281a\u281b\u281c\u281d\u281e\u281f", - "\u2820\u2821\u2822\u2823\u2824\u2825\u2826\u2827\u2828\u2829\u282a\u282b\u282c\u282d\u282e\u282f", - "\u2830\u2831\u2832\u2833\u2834\u2835\u2836\u2837\u2838\u2839\u283a\u283b\u283c\u283d\u283e\u283f", - "\u2840\u2841\u2842\u2843\u2844\u2845\u2846\u2847\u2848\u2849\u284a\u284b\u284c\u284d\u284e\u284f", - "\u2850\u2851\u2852\u2853\u2854\u2855\u2856\u2857\u2858\u2859\u285a\u285b\u285c\u285d\u285e\u285f", - "\u2860\u2861\u2862\u2863\u2864\u2865\u2866\u2867\u2868\u2869\u286a\u286b\u286c\u286d\u286e\u286f", - "\u2870\u2871\u2872\u2873\u2874\u2875\u2876\u2877\u2878\u2879\u287a\u287b\u287c\u287d\u287e\u287f", - "\u2880\u2881\u2882\u2883\u2884\u2885\u2886\u2887\u2888\u2889\u288a\u288b\u288c\u288d\u288e\u288f", - "\u2890\u2891\u2892\u2893\u2894\u2895\u2896\u2897\u2898\u2899\u289a\u289b\u289c\u289d\u289e\u289f", - "\u28a0\u28a1\u28a2\u28a3\u28a4\u28a5\u28a6\u28a7\u28a8\u28a9\u28aa\u28ab\u28ac\u28ad\u28ae\u28af", - "\u28b0\u28b1\u28b2\u28b3\u28b4\u28b5\u28b6\u28b7\u28b8\u28b9\u28ba\u28bb\u28bc\u28bd\u28be\u28bf", - "\u28c0\u28c1\u28c2\u28c3\u28c4\u28c5\u28c6\u28c7\u28c8\u28c9\u28ca\u28cb\u28cc\u28cd\u28ce\u28cf", - "\u28d0\u28d1\u28d2\u28d3\u28d4\u28d5\u28d6\u28d7\u28d8\u28d9\u28da\u28db\u28dc\u28dd\u28de\u28df", - "\u28e0\u28e1\u28e2\u28e3\u28e4\u28e5\u28e6\u28e7\u28e8\u28e9\u28ea\u28eb\u28ec\u28ed\u28ee\u28ef", - "\u28f0\u28f1\u28f2\u28f3\u28f4\u28f5\u28f6\u28f7\u28f8\u28f9\u28fa\u28fb\u28fc\u28fd\u28fe\u28ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_29.png", - "ascent": %ascent%, - "chars": [ - "\u2900\u2901\u2902\u2903\u2904\u2905\u2906\u2907\u2908\u2909\u290a\u290b\u290c\u290d\u290e\u290f", - "\u2910\u2911\u2912\u2913\u2914\u2915\u2916\u2917\u2918\u2919\u291a\u291b\u291c\u291d\u291e\u291f", - "\u2920\u2921\u2922\u2923\u2924\u2925\u2926\u2927\u2928\u2929\u292a\u292b\u292c\u292d\u292e\u292f", - "\u2930\u2931\u2932\u2933\u2934\u2935\u2936\u2937\u2938\u2939\u293a\u293b\u293c\u293d\u293e\u293f", - "\u2940\u2941\u2942\u2943\u2944\u2945\u2946\u2947\u2948\u2949\u294a\u294b\u294c\u294d\u294e\u294f", - "\u2950\u2951\u2952\u2953\u2954\u2955\u2956\u2957\u2958\u2959\u295a\u295b\u295c\u295d\u295e\u295f", - "\u2960\u2961\u2962\u2963\u2964\u2965\u2966\u2967\u2968\u2969\u296a\u296b\u296c\u296d\u296e\u296f", - "\u2970\u2971\u2972\u2973\u2974\u2975\u2976\u2977\u2978\u2979\u297a\u297b\u297c\u297d\u297e\u297f", - "\u2980\u2981\u2982\u2983\u2984\u2985\u2986\u2987\u2988\u2989\u298a\u298b\u298c\u298d\u298e\u298f", - "\u2990\u2991\u2992\u2993\u2994\u2995\u2996\u2997\u2998\u2999\u299a\u299b\u299c\u299d\u299e\u299f", - "\u29a0\u29a1\u29a2\u29a3\u29a4\u29a5\u29a6\u29a7\u29a8\u29a9\u29aa\u29ab\u29ac\u29ad\u29ae\u29af", - "\u29b0\u29b1\u29b2\u29b3\u29b4\u29b5\u29b6\u29b7\u29b8\u29b9\u29ba\u29bb\u29bc\u29bd\u29be\u29bf", - "\u29c0\u29c1\u29c2\u29c3\u29c4\u29c5\u29c6\u29c7\u29c8\u29c9\u29ca\u29cb\u29cc\u29cd\u29ce\u29cf", - "\u29d0\u29d1\u29d2\u29d3\u29d4\u29d5\u29d6\u29d7\u29d8\u29d9\u29da\u29db\u29dc\u29dd\u29de\u29df", - "\u29e0\u29e1\u29e2\u29e3\u29e4\u29e5\u29e6\u29e7\u29e8\u29e9\u29ea\u29eb\u29ec\u29ed\u29ee\u29ef", - "\u29f0\u29f1\u29f2\u29f3\u29f4\u29f5\u29f6\u29f7\u29f8\u29f9\u29fa\u29fb\u29fc\u29fd\u29fe\u29ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_2a.png", - "ascent": %ascent%, - "chars": [ - "\u2a00\u2a01\u2a02\u2a03\u2a04\u2a05\u2a06\u2a07\u2a08\u2a09\u2a0a\u2a0b\u2a0c\u2a0d\u2a0e\u2a0f", - "\u2a10\u2a11\u2a12\u2a13\u2a14\u2a15\u2a16\u2a17\u2a18\u2a19\u2a1a\u2a1b\u2a1c\u2a1d\u2a1e\u2a1f", - "\u2a20\u2a21\u2a22\u2a23\u2a24\u2a25\u2a26\u2a27\u2a28\u2a29\u2a2a\u2a2b\u2a2c\u2a2d\u2a2e\u2a2f", - "\u2a30\u2a31\u2a32\u2a33\u2a34\u2a35\u2a36\u2a37\u2a38\u2a39\u2a3a\u2a3b\u2a3c\u2a3d\u2a3e\u2a3f", - "\u2a40\u2a41\u2a42\u2a43\u2a44\u2a45\u2a46\u2a47\u2a48\u2a49\u2a4a\u2a4b\u2a4c\u2a4d\u2a4e\u2a4f", - "\u2a50\u2a51\u2a52\u2a53\u2a54\u2a55\u2a56\u2a57\u2a58\u2a59\u2a5a\u2a5b\u2a5c\u2a5d\u2a5e\u2a5f", - "\u2a60\u2a61\u2a62\u2a63\u2a64\u2a65\u2a66\u2a67\u2a68\u2a69\u2a6a\u2a6b\u2a6c\u2a6d\u2a6e\u2a6f", - "\u2a70\u2a71\u2a72\u2a73\u2a74\u2a75\u2a76\u2a77\u2a78\u2a79\u2a7a\u2a7b\u2a7c\u2a7d\u2a7e\u2a7f", - "\u2a80\u2a81\u2a82\u2a83\u2a84\u2a85\u2a86\u2a87\u2a88\u2a89\u2a8a\u2a8b\u2a8c\u2a8d\u2a8e\u2a8f", - "\u2a90\u2a91\u2a92\u2a93\u2a94\u2a95\u2a96\u2a97\u2a98\u2a99\u2a9a\u2a9b\u2a9c\u2a9d\u2a9e\u2a9f", - "\u2aa0\u2aa1\u2aa2\u2aa3\u2aa4\u2aa5\u2aa6\u2aa7\u2aa8\u2aa9\u2aaa\u2aab\u2aac\u2aad\u2aae\u2aaf", - "\u2ab0\u2ab1\u2ab2\u2ab3\u2ab4\u2ab5\u2ab6\u2ab7\u2ab8\u2ab9\u2aba\u2abb\u2abc\u2abd\u2abe\u2abf", - "\u2ac0\u2ac1\u2ac2\u2ac3\u2ac4\u2ac5\u2ac6\u2ac7\u2ac8\u2ac9\u2aca\u2acb\u2acc\u2acd\u2ace\u2acf", - "\u2ad0\u2ad1\u2ad2\u2ad3\u2ad4\u2ad5\u2ad6\u2ad7\u2ad8\u2ad9\u2ada\u2adb\u2adc\u2add\u2ade\u2adf", - "\u2ae0\u2ae1\u2ae2\u2ae3\u2ae4\u2ae5\u2ae6\u2ae7\u2ae8\u2ae9\u2aea\u2aeb\u2aec\u2aed\u2aee\u2aef", - "\u2af0\u2af1\u2af2\u2af3\u2af4\u2af5\u2af6\u2af7\u2af8\u2af9\u2afa\u2afb\u2afc\u2afd\u2afe\u2aff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_2b.png", - "ascent": %ascent%, - "chars": [ - "\u2b00\u2b01\u2b02\u2b03\u2b04\u2b05\u2b06\u2b07\u2b08\u2b09\u2b0a\u2b0b\u2b0c\u2b0d\u2b0e\u2b0f", - "\u2b10\u2b11\u2b12\u2b13\u2b14\u2b15\u2b16\u2b17\u2b18\u2b19\u2b1a\u2b1b\u2b1c\u2b1d\u2b1e\u2b1f", - "\u2b20\u2b21\u2b22\u2b23\u2b24\u2b25\u2b26\u2b27\u2b28\u2b29\u2b2a\u2b2b\u2b2c\u2b2d\u2b2e\u2b2f", - "\u2b30\u2b31\u2b32\u2b33\u2b34\u2b35\u2b36\u2b37\u2b38\u2b39\u2b3a\u2b3b\u2b3c\u2b3d\u2b3e\u2b3f", - "\u2b40\u2b41\u2b42\u2b43\u2b44\u2b45\u2b46\u2b47\u2b48\u2b49\u2b4a\u2b4b\u2b4c\u2b4d\u2b4e\u2b4f", - "\u2b50\u2b51\u2b52\u2b53\u2b54\u2b55\u2b56\u2b57\u2b58\u2b59\u2b5a\u2b5b\u2b5c\u2b5d\u2b5e\u2b5f", - "\u2b60\u2b61\u2b62\u2b63\u2b64\u2b65\u2b66\u2b67\u2b68\u2b69\u2b6a\u2b6b\u2b6c\u2b6d\u2b6e\u2b6f", - "\u2b70\u2b71\u2b72\u2b73\u2b74\u2b75\u2b76\u2b77\u2b78\u2b79\u2b7a\u2b7b\u2b7c\u2b7d\u2b7e\u2b7f", - "\u2b80\u2b81\u2b82\u2b83\u2b84\u2b85\u2b86\u2b87\u2b88\u2b89\u2b8a\u2b8b\u2b8c\u2b8d\u2b8e\u2b8f", - "\u2b90\u2b91\u2b92\u2b93\u2b94\u2b95\u2b96\u2b97\u2b98\u2b99\u2b9a\u2b9b\u2b9c\u2b9d\u2b9e\u2b9f", - "\u2ba0\u2ba1\u2ba2\u2ba3\u2ba4\u2ba5\u2ba6\u2ba7\u2ba8\u2ba9\u2baa\u2bab\u2bac\u2bad\u2bae\u2baf", - "\u2bb0\u2bb1\u2bb2\u2bb3\u2bb4\u2bb5\u2bb6\u2bb7\u2bb8\u2bb9\u2bba\u2bbb\u2bbc\u2bbd\u2bbe\u2bbf", - "\u2bc0\u2bc1\u2bc2\u2bc3\u2bc4\u2bc5\u2bc6\u2bc7\u2bc8\u2bc9\u2bca\u2bcb\u2bcc\u2bcd\u2bce\u2bcf", - "\u2bd0\u2bd1\u2bd2\u2bd3\u2bd4\u2bd5\u2bd6\u2bd7\u2bd8\u2bd9\u2bda\u2bdb\u2bdc\u2bdd\u2bde\u2bdf", - "\u2be0\u2be1\u2be2\u2be3\u2be4\u2be5\u2be6\u2be7\u2be8\u2be9\u2bea\u2beb\u2bec\u2bed\u2bee\u2bef", - "\u2bf0\u2bf1\u2bf2\u2bf3\u2bf4\u2bf5\u2bf6\u2bf7\u2bf8\u2bf9\u2bfa\u2bfb\u2bfc\u2bfd\u2bfe\u2bff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_2c.png", - "ascent": %ascent%, - "chars": [ - "\u2c00\u2c01\u2c02\u2c03\u2c04\u2c05\u2c06\u2c07\u2c08\u2c09\u2c0a\u2c0b\u2c0c\u2c0d\u2c0e\u2c0f", - "\u2c10\u2c11\u2c12\u2c13\u2c14\u2c15\u2c16\u2c17\u2c18\u2c19\u2c1a\u2c1b\u2c1c\u2c1d\u2c1e\u2c1f", - "\u2c20\u2c21\u2c22\u2c23\u2c24\u2c25\u2c26\u2c27\u2c28\u2c29\u2c2a\u2c2b\u2c2c\u2c2d\u2c2e\u2c2f", - "\u2c30\u2c31\u2c32\u2c33\u2c34\u2c35\u2c36\u2c37\u2c38\u2c39\u2c3a\u2c3b\u2c3c\u2c3d\u2c3e\u2c3f", - "\u2c40\u2c41\u2c42\u2c43\u2c44\u2c45\u2c46\u2c47\u2c48\u2c49\u2c4a\u2c4b\u2c4c\u2c4d\u2c4e\u2c4f", - "\u2c50\u2c51\u2c52\u2c53\u2c54\u2c55\u2c56\u2c57\u2c58\u2c59\u2c5a\u2c5b\u2c5c\u2c5d\u2c5e\u2c5f", - "\u2c60\u2c61\u2c62\u2c63\u2c64\u2c65\u2c66\u2c67\u2c68\u2c69\u2c6a\u2c6b\u2c6c\u2c6d\u2c6e\u2c6f", - "\u2c70\u2c71\u2c72\u2c73\u2c74\u2c75\u2c76\u2c77\u2c78\u2c79\u2c7a\u2c7b\u2c7c\u2c7d\u2c7e\u2c7f", - "\u2c80\u2c81\u2c82\u2c83\u2c84\u2c85\u2c86\u2c87\u2c88\u2c89\u2c8a\u2c8b\u2c8c\u2c8d\u2c8e\u2c8f", - "\u2c90\u2c91\u2c92\u2c93\u2c94\u2c95\u2c96\u2c97\u2c98\u2c99\u2c9a\u2c9b\u2c9c\u2c9d\u2c9e\u2c9f", - "\u2ca0\u2ca1\u2ca2\u2ca3\u2ca4\u2ca5\u2ca6\u2ca7\u2ca8\u2ca9\u2caa\u2cab\u2cac\u2cad\u2cae\u2caf", - "\u2cb0\u2cb1\u2cb2\u2cb3\u2cb4\u2cb5\u2cb6\u2cb7\u2cb8\u2cb9\u2cba\u2cbb\u2cbc\u2cbd\u2cbe\u2cbf", - "\u2cc0\u2cc1\u2cc2\u2cc3\u2cc4\u2cc5\u2cc6\u2cc7\u2cc8\u2cc9\u2cca\u2ccb\u2ccc\u2ccd\u2cce\u2ccf", - "\u2cd0\u2cd1\u2cd2\u2cd3\u2cd4\u2cd5\u2cd6\u2cd7\u2cd8\u2cd9\u2cda\u2cdb\u2cdc\u2cdd\u2cde\u2cdf", - "\u2ce0\u2ce1\u2ce2\u2ce3\u2ce4\u2ce5\u2ce6\u2ce7\u2ce8\u2ce9\u2cea\u2ceb\u2cec\u2ced\u2cee\u2cef", - "\u2cf0\u2cf1\u2cf2\u2cf3\u2cf4\u2cf5\u2cf6\u2cf7\u2cf8\u2cf9\u2cfa\u2cfb\u2cfc\u2cfd\u2cfe\u2cff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_2d.png", - "ascent": %ascent%, - "chars": [ - "\u2d00\u2d01\u2d02\u2d03\u2d04\u2d05\u2d06\u2d07\u2d08\u2d09\u2d0a\u2d0b\u2d0c\u2d0d\u2d0e\u2d0f", - "\u2d10\u2d11\u2d12\u2d13\u2d14\u2d15\u2d16\u2d17\u2d18\u2d19\u2d1a\u2d1b\u2d1c\u2d1d\u2d1e\u2d1f", - "\u2d20\u2d21\u2d22\u2d23\u2d24\u2d25\u2d26\u2d27\u2d28\u2d29\u2d2a\u2d2b\u2d2c\u2d2d\u2d2e\u2d2f", - "\u2d30\u2d31\u2d32\u2d33\u2d34\u2d35\u2d36\u2d37\u2d38\u2d39\u2d3a\u2d3b\u2d3c\u2d3d\u2d3e\u2d3f", - "\u2d40\u2d41\u2d42\u2d43\u2d44\u2d45\u2d46\u2d47\u2d48\u2d49\u2d4a\u2d4b\u2d4c\u2d4d\u2d4e\u2d4f", - "\u2d50\u2d51\u2d52\u2d53\u2d54\u2d55\u2d56\u2d57\u2d58\u2d59\u2d5a\u2d5b\u2d5c\u2d5d\u2d5e\u2d5f", - "\u2d60\u2d61\u2d62\u2d63\u2d64\u2d65\u2d66\u2d67\u2d68\u2d69\u2d6a\u2d6b\u2d6c\u2d6d\u2d6e\u2d6f", - "\u2d70\u2d71\u2d72\u2d73\u2d74\u2d75\u2d76\u2d77\u2d78\u2d79\u2d7a\u2d7b\u2d7c\u2d7d\u2d7e\u2d7f", - "\u2d80\u2d81\u2d82\u2d83\u2d84\u2d85\u2d86\u2d87\u2d88\u2d89\u2d8a\u2d8b\u2d8c\u2d8d\u2d8e\u2d8f", - "\u2d90\u2d91\u2d92\u2d93\u2d94\u2d95\u2d96\u2d97\u2d98\u2d99\u2d9a\u2d9b\u2d9c\u2d9d\u2d9e\u2d9f", - "\u2da0\u2da1\u2da2\u2da3\u2da4\u2da5\u2da6\u2da7\u2da8\u2da9\u2daa\u2dab\u2dac\u2dad\u2dae\u2daf", - "\u2db0\u2db1\u2db2\u2db3\u2db4\u2db5\u2db6\u2db7\u2db8\u2db9\u2dba\u2dbb\u2dbc\u2dbd\u2dbe\u2dbf", - "\u2dc0\u2dc1\u2dc2\u2dc3\u2dc4\u2dc5\u2dc6\u2dc7\u2dc8\u2dc9\u2dca\u2dcb\u2dcc\u2dcd\u2dce\u2dcf", - "\u2dd0\u2dd1\u2dd2\u2dd3\u2dd4\u2dd5\u2dd6\u2dd7\u2dd8\u2dd9\u2dda\u2ddb\u2ddc\u2ddd\u2dde\u2ddf", - "\u2de0\u2de1\u2de2\u2de3\u2de4\u2de5\u2de6\u2de7\u2de8\u2de9\u2dea\u2deb\u2dec\u2ded\u2dee\u2def", - "\u2df0\u2df1\u2df2\u2df3\u2df4\u2df5\u2df6\u2df7\u2df8\u2df9\u2dfa\u2dfb\u2dfc\u2dfd\u2dfe\u2dff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_2e.png", - "ascent": %ascent%, - "chars": [ - "\u2e00\u2e01\u2e02\u2e03\u2e04\u2e05\u2e06\u2e07\u2e08\u2e09\u2e0a\u2e0b\u2e0c\u2e0d\u2e0e\u2e0f", - "\u2e10\u2e11\u2e12\u2e13\u2e14\u2e15\u2e16\u2e17\u2e18\u2e19\u2e1a\u2e1b\u2e1c\u2e1d\u2e1e\u2e1f", - "\u2e20\u2e21\u2e22\u2e23\u2e24\u2e25\u2e26\u2e27\u2e28\u2e29\u2e2a\u2e2b\u2e2c\u2e2d\u2e2e\u2e2f", - "\u2e30\u2e31\u2e32\u2e33\u2e34\u2e35\u2e36\u2e37\u2e38\u2e39\u2e3a\u2e3b\u2e3c\u2e3d\u2e3e\u2e3f", - "\u2e40\u2e41\u2e42\u2e43\u2e44\u2e45\u2e46\u2e47\u2e48\u2e49\u2e4a\u2e4b\u2e4c\u2e4d\u2e4e\u2e4f", - "\u2e50\u2e51\u2e52\u2e53\u2e54\u2e55\u2e56\u2e57\u2e58\u2e59\u2e5a\u2e5b\u2e5c\u2e5d\u2e5e\u2e5f", - "\u2e60\u2e61\u2e62\u2e63\u2e64\u2e65\u2e66\u2e67\u2e68\u2e69\u2e6a\u2e6b\u2e6c\u2e6d\u2e6e\u2e6f", - "\u2e70\u2e71\u2e72\u2e73\u2e74\u2e75\u2e76\u2e77\u2e78\u2e79\u2e7a\u2e7b\u2e7c\u2e7d\u2e7e\u2e7f", - "\u2e80\u2e81\u2e82\u2e83\u2e84\u2e85\u2e86\u2e87\u2e88\u2e89\u2e8a\u2e8b\u2e8c\u2e8d\u2e8e\u2e8f", - "\u2e90\u2e91\u2e92\u2e93\u2e94\u2e95\u2e96\u2e97\u2e98\u2e99\u2e9a\u2e9b\u2e9c\u2e9d\u2e9e\u2e9f", - "\u2ea0\u2ea1\u2ea2\u2ea3\u2ea4\u2ea5\u2ea6\u2ea7\u2ea8\u2ea9\u2eaa\u2eab\u2eac\u2ead\u2eae\u2eaf", - "\u2eb0\u2eb1\u2eb2\u2eb3\u2eb4\u2eb5\u2eb6\u2eb7\u2eb8\u2eb9\u2eba\u2ebb\u2ebc\u2ebd\u2ebe\u2ebf", - "\u2ec0\u2ec1\u2ec2\u2ec3\u2ec4\u2ec5\u2ec6\u2ec7\u2ec8\u2ec9\u2eca\u2ecb\u2ecc\u2ecd\u2ece\u2ecf", - "\u2ed0\u2ed1\u2ed2\u2ed3\u2ed4\u2ed5\u2ed6\u2ed7\u2ed8\u2ed9\u2eda\u2edb\u2edc\u2edd\u2ede\u2edf", - "\u2ee0\u2ee1\u2ee2\u2ee3\u2ee4\u2ee5\u2ee6\u2ee7\u2ee8\u2ee9\u2eea\u2eeb\u2eec\u2eed\u2eee\u2eef", - "\u2ef0\u2ef1\u2ef2\u2ef3\u2ef4\u2ef5\u2ef6\u2ef7\u2ef8\u2ef9\u2efa\u2efb\u2efc\u2efd\u2efe\u2eff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_2f.png", - "ascent": %ascent%, - "chars": [ - "\u2f00\u2f01\u2f02\u2f03\u2f04\u2f05\u2f06\u2f07\u2f08\u2f09\u2f0a\u2f0b\u2f0c\u2f0d\u2f0e\u2f0f", - "\u2f10\u2f11\u2f12\u2f13\u2f14\u2f15\u2f16\u2f17\u2f18\u2f19\u2f1a\u2f1b\u2f1c\u2f1d\u2f1e\u2f1f", - "\u2f20\u2f21\u2f22\u2f23\u2f24\u2f25\u2f26\u2f27\u2f28\u2f29\u2f2a\u2f2b\u2f2c\u2f2d\u2f2e\u2f2f", - "\u2f30\u2f31\u2f32\u2f33\u2f34\u2f35\u2f36\u2f37\u2f38\u2f39\u2f3a\u2f3b\u2f3c\u2f3d\u2f3e\u2f3f", - "\u2f40\u2f41\u2f42\u2f43\u2f44\u2f45\u2f46\u2f47\u2f48\u2f49\u2f4a\u2f4b\u2f4c\u2f4d\u2f4e\u2f4f", - "\u2f50\u2f51\u2f52\u2f53\u2f54\u2f55\u2f56\u2f57\u2f58\u2f59\u2f5a\u2f5b\u2f5c\u2f5d\u2f5e\u2f5f", - "\u2f60\u2f61\u2f62\u2f63\u2f64\u2f65\u2f66\u2f67\u2f68\u2f69\u2f6a\u2f6b\u2f6c\u2f6d\u2f6e\u2f6f", - "\u2f70\u2f71\u2f72\u2f73\u2f74\u2f75\u2f76\u2f77\u2f78\u2f79\u2f7a\u2f7b\u2f7c\u2f7d\u2f7e\u2f7f", - "\u2f80\u2f81\u2f82\u2f83\u2f84\u2f85\u2f86\u2f87\u2f88\u2f89\u2f8a\u2f8b\u2f8c\u2f8d\u2f8e\u2f8f", - "\u2f90\u2f91\u2f92\u2f93\u2f94\u2f95\u2f96\u2f97\u2f98\u2f99\u2f9a\u2f9b\u2f9c\u2f9d\u2f9e\u2f9f", - "\u2fa0\u2fa1\u2fa2\u2fa3\u2fa4\u2fa5\u2fa6\u2fa7\u2fa8\u2fa9\u2faa\u2fab\u2fac\u2fad\u2fae\u2faf", - "\u2fb0\u2fb1\u2fb2\u2fb3\u2fb4\u2fb5\u2fb6\u2fb7\u2fb8\u2fb9\u2fba\u2fbb\u2fbc\u2fbd\u2fbe\u2fbf", - "\u2fc0\u2fc1\u2fc2\u2fc3\u2fc4\u2fc5\u2fc6\u2fc7\u2fc8\u2fc9\u2fca\u2fcb\u2fcc\u2fcd\u2fce\u2fcf", - "\u2fd0\u2fd1\u2fd2\u2fd3\u2fd4\u2fd5\u2fd6\u2fd7\u2fd8\u2fd9\u2fda\u2fdb\u2fdc\u2fdd\u2fde\u2fdf", - "\u2fe0\u2fe1\u2fe2\u2fe3\u2fe4\u2fe5\u2fe6\u2fe7\u2fe8\u2fe9\u2fea\u2feb\u2fec\u2fed\u2fee\u2fef", - "\u2ff0\u2ff1\u2ff2\u2ff3\u2ff4\u2ff5\u2ff6\u2ff7\u2ff8\u2ff9\u2ffa\u2ffb\u2ffc\u2ffd\u2ffe\u2fff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_30.png", - "ascent": %ascent%, - "chars": [ - "\u3000\u3001\u3002\u3003\u3004\u3005\u3006\u3007\u3008\u3009\u300a\u300b\u300c\u300d\u300e\u300f", - "\u3010\u3011\u3012\u3013\u3014\u3015\u3016\u3017\u3018\u3019\u301a\u301b\u301c\u301d\u301e\u301f", - "\u3020\u3021\u3022\u3023\u3024\u3025\u3026\u3027\u3028\u3029\u302a\u302b\u302c\u302d\u302e\u302f", - "\u3030\u3031\u3032\u3033\u3034\u3035\u3036\u3037\u3038\u3039\u303a\u303b\u303c\u303d\u303e\u303f", - "\u3040\u3041\u3042\u3043\u3044\u3045\u3046\u3047\u3048\u3049\u304a\u304b\u304c\u304d\u304e\u304f", - "\u3050\u3051\u3052\u3053\u3054\u3055\u3056\u3057\u3058\u3059\u305a\u305b\u305c\u305d\u305e\u305f", - "\u3060\u3061\u3062\u3063\u3064\u3065\u3066\u3067\u3068\u3069\u306a\u306b\u306c\u306d\u306e\u306f", - "\u3070\u3071\u3072\u3073\u3074\u3075\u3076\u3077\u3078\u3079\u307a\u307b\u307c\u307d\u307e\u307f", - "\u3080\u3081\u3082\u3083\u3084\u3085\u3086\u3087\u3088\u3089\u308a\u308b\u308c\u308d\u308e\u308f", - "\u3090\u3091\u3092\u3093\u3094\u3095\u3096\u3097\u3098\u3099\u309a\u309b\u309c\u309d\u309e\u309f", - "\u30a0\u30a1\u30a2\u30a3\u30a4\u30a5\u30a6\u30a7\u30a8\u30a9\u30aa\u30ab\u30ac\u30ad\u30ae\u30af", - "\u30b0\u30b1\u30b2\u30b3\u30b4\u30b5\u30b6\u30b7\u30b8\u30b9\u30ba\u30bb\u30bc\u30bd\u30be\u30bf", - "\u30c0\u30c1\u30c2\u30c3\u30c4\u30c5\u30c6\u30c7\u30c8\u30c9\u30ca\u30cb\u30cc\u30cd\u30ce\u30cf", - "\u30d0\u30d1\u30d2\u30d3\u30d4\u30d5\u30d6\u30d7\u30d8\u30d9\u30da\u30db\u30dc\u30dd\u30de\u30df", - "\u30e0\u30e1\u30e2\u30e3\u30e4\u30e5\u30e6\u30e7\u30e8\u30e9\u30ea\u30eb\u30ec\u30ed\u30ee\u30ef", - "\u30f0\u30f1\u30f2\u30f3\u30f4\u30f5\u30f6\u30f7\u30f8\u30f9\u30fa\u30fb\u30fc\u30fd\u30fe\u30ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_31.png", - "ascent": %ascent%, - "chars": [ - "\u3100\u3101\u3102\u3103\u3104\u3105\u3106\u3107\u3108\u3109\u310a\u310b\u310c\u310d\u310e\u310f", - "\u3110\u3111\u3112\u3113\u3114\u3115\u3116\u3117\u3118\u3119\u311a\u311b\u311c\u311d\u311e\u311f", - "\u3120\u3121\u3122\u3123\u3124\u3125\u3126\u3127\u3128\u3129\u312a\u312b\u312c\u312d\u312e\u312f", - "\u3130\u3131\u3132\u3133\u3134\u3135\u3136\u3137\u3138\u3139\u313a\u313b\u313c\u313d\u313e\u313f", - "\u3140\u3141\u3142\u3143\u3144\u3145\u3146\u3147\u3148\u3149\u314a\u314b\u314c\u314d\u314e\u314f", - "\u3150\u3151\u3152\u3153\u3154\u3155\u3156\u3157\u3158\u3159\u315a\u315b\u315c\u315d\u315e\u315f", - "\u3160\u3161\u3162\u3163\u3164\u3165\u3166\u3167\u3168\u3169\u316a\u316b\u316c\u316d\u316e\u316f", - "\u3170\u3171\u3172\u3173\u3174\u3175\u3176\u3177\u3178\u3179\u317a\u317b\u317c\u317d\u317e\u317f", - "\u3180\u3181\u3182\u3183\u3184\u3185\u3186\u3187\u3188\u3189\u318a\u318b\u318c\u318d\u318e\u318f", - "\u3190\u3191\u3192\u3193\u3194\u3195\u3196\u3197\u3198\u3199\u319a\u319b\u319c\u319d\u319e\u319f", - "\u31a0\u31a1\u31a2\u31a3\u31a4\u31a5\u31a6\u31a7\u31a8\u31a9\u31aa\u31ab\u31ac\u31ad\u31ae\u31af", - "\u31b0\u31b1\u31b2\u31b3\u31b4\u31b5\u31b6\u31b7\u31b8\u31b9\u31ba\u31bb\u31bc\u31bd\u31be\u31bf", - "\u31c0\u31c1\u31c2\u31c3\u31c4\u31c5\u31c6\u31c7\u31c8\u31c9\u31ca\u31cb\u31cc\u31cd\u31ce\u31cf", - "\u31d0\u31d1\u31d2\u31d3\u31d4\u31d5\u31d6\u31d7\u31d8\u31d9\u31da\u31db\u31dc\u31dd\u31de\u31df", - "\u31e0\u31e1\u31e2\u31e3\u31e4\u31e5\u31e6\u31e7\u31e8\u31e9\u31ea\u31eb\u31ec\u31ed\u31ee\u31ef", - "\u31f0\u31f1\u31f2\u31f3\u31f4\u31f5\u31f6\u31f7\u31f8\u31f9\u31fa\u31fb\u31fc\u31fd\u31fe\u31ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_32.png", - "ascent": %ascent%, - "chars": [ - "\u3200\u3201\u3202\u3203\u3204\u3205\u3206\u3207\u3208\u3209\u320a\u320b\u320c\u320d\u320e\u320f", - "\u3210\u3211\u3212\u3213\u3214\u3215\u3216\u3217\u3218\u3219\u321a\u321b\u321c\u321d\u321e\u321f", - "\u3220\u3221\u3222\u3223\u3224\u3225\u3226\u3227\u3228\u3229\u322a\u322b\u322c\u322d\u322e\u322f", - "\u3230\u3231\u3232\u3233\u3234\u3235\u3236\u3237\u3238\u3239\u323a\u323b\u323c\u323d\u323e\u323f", - "\u3240\u3241\u3242\u3243\u3244\u3245\u3246\u3247\u3248\u3249\u324a\u324b\u324c\u324d\u324e\u324f", - "\u3250\u3251\u3252\u3253\u3254\u3255\u3256\u3257\u3258\u3259\u325a\u325b\u325c\u325d\u325e\u325f", - "\u3260\u3261\u3262\u3263\u3264\u3265\u3266\u3267\u3268\u3269\u326a\u326b\u326c\u326d\u326e\u326f", - "\u3270\u3271\u3272\u3273\u3274\u3275\u3276\u3277\u3278\u3279\u327a\u327b\u327c\u327d\u327e\u327f", - "\u3280\u3281\u3282\u3283\u3284\u3285\u3286\u3287\u3288\u3289\u328a\u328b\u328c\u328d\u328e\u328f", - "\u3290\u3291\u3292\u3293\u3294\u3295\u3296\u3297\u3298\u3299\u329a\u329b\u329c\u329d\u329e\u329f", - "\u32a0\u32a1\u32a2\u32a3\u32a4\u32a5\u32a6\u32a7\u32a8\u32a9\u32aa\u32ab\u32ac\u32ad\u32ae\u32af", - "\u32b0\u32b1\u32b2\u32b3\u32b4\u32b5\u32b6\u32b7\u32b8\u32b9\u32ba\u32bb\u32bc\u32bd\u32be\u32bf", - "\u32c0\u32c1\u32c2\u32c3\u32c4\u32c5\u32c6\u32c7\u32c8\u32c9\u32ca\u32cb\u32cc\u32cd\u32ce\u32cf", - "\u32d0\u32d1\u32d2\u32d3\u32d4\u32d5\u32d6\u32d7\u32d8\u32d9\u32da\u32db\u32dc\u32dd\u32de\u32df", - "\u32e0\u32e1\u32e2\u32e3\u32e4\u32e5\u32e6\u32e7\u32e8\u32e9\u32ea\u32eb\u32ec\u32ed\u32ee\u32ef", - "\u32f0\u32f1\u32f2\u32f3\u32f4\u32f5\u32f6\u32f7\u32f8\u32f9\u32fa\u32fb\u32fc\u32fd\u32fe\u32ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_33.png", - "ascent": %ascent%, - "chars": [ - "\u3300\u3301\u3302\u3303\u3304\u3305\u3306\u3307\u3308\u3309\u330a\u330b\u330c\u330d\u330e\u330f", - "\u3310\u3311\u3312\u3313\u3314\u3315\u3316\u3317\u3318\u3319\u331a\u331b\u331c\u331d\u331e\u331f", - "\u3320\u3321\u3322\u3323\u3324\u3325\u3326\u3327\u3328\u3329\u332a\u332b\u332c\u332d\u332e\u332f", - "\u3330\u3331\u3332\u3333\u3334\u3335\u3336\u3337\u3338\u3339\u333a\u333b\u333c\u333d\u333e\u333f", - "\u3340\u3341\u3342\u3343\u3344\u3345\u3346\u3347\u3348\u3349\u334a\u334b\u334c\u334d\u334e\u334f", - "\u3350\u3351\u3352\u3353\u3354\u3355\u3356\u3357\u3358\u3359\u335a\u335b\u335c\u335d\u335e\u335f", - "\u3360\u3361\u3362\u3363\u3364\u3365\u3366\u3367\u3368\u3369\u336a\u336b\u336c\u336d\u336e\u336f", - "\u3370\u3371\u3372\u3373\u3374\u3375\u3376\u3377\u3378\u3379\u337a\u337b\u337c\u337d\u337e\u337f", - "\u3380\u3381\u3382\u3383\u3384\u3385\u3386\u3387\u3388\u3389\u338a\u338b\u338c\u338d\u338e\u338f", - "\u3390\u3391\u3392\u3393\u3394\u3395\u3396\u3397\u3398\u3399\u339a\u339b\u339c\u339d\u339e\u339f", - "\u33a0\u33a1\u33a2\u33a3\u33a4\u33a5\u33a6\u33a7\u33a8\u33a9\u33aa\u33ab\u33ac\u33ad\u33ae\u33af", - "\u33b0\u33b1\u33b2\u33b3\u33b4\u33b5\u33b6\u33b7\u33b8\u33b9\u33ba\u33bb\u33bc\u33bd\u33be\u33bf", - "\u33c0\u33c1\u33c2\u33c3\u33c4\u33c5\u33c6\u33c7\u33c8\u33c9\u33ca\u33cb\u33cc\u33cd\u33ce\u33cf", - "\u33d0\u33d1\u33d2\u33d3\u33d4\u33d5\u33d6\u33d7\u33d8\u33d9\u33da\u33db\u33dc\u33dd\u33de\u33df", - "\u33e0\u33e1\u33e2\u33e3\u33e4\u33e5\u33e6\u33e7\u33e8\u33e9\u33ea\u33eb\u33ec\u33ed\u33ee\u33ef", - "\u33f0\u33f1\u33f2\u33f3\u33f4\u33f5\u33f6\u33f7\u33f8\u33f9\u33fa\u33fb\u33fc\u33fd\u33fe\u33ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_34.png", - "ascent": %ascent%, - "chars": [ - "\u3400\u3401\u3402\u3403\u3404\u3405\u3406\u3407\u3408\u3409\u340a\u340b\u340c\u340d\u340e\u340f", - "\u3410\u3411\u3412\u3413\u3414\u3415\u3416\u3417\u3418\u3419\u341a\u341b\u341c\u341d\u341e\u341f", - "\u3420\u3421\u3422\u3423\u3424\u3425\u3426\u3427\u3428\u3429\u342a\u342b\u342c\u342d\u342e\u342f", - "\u3430\u3431\u3432\u3433\u3434\u3435\u3436\u3437\u3438\u3439\u343a\u343b\u343c\u343d\u343e\u343f", - "\u3440\u3441\u3442\u3443\u3444\u3445\u3446\u3447\u3448\u3449\u344a\u344b\u344c\u344d\u344e\u344f", - "\u3450\u3451\u3452\u3453\u3454\u3455\u3456\u3457\u3458\u3459\u345a\u345b\u345c\u345d\u345e\u345f", - "\u3460\u3461\u3462\u3463\u3464\u3465\u3466\u3467\u3468\u3469\u346a\u346b\u346c\u346d\u346e\u346f", - "\u3470\u3471\u3472\u3473\u3474\u3475\u3476\u3477\u3478\u3479\u347a\u347b\u347c\u347d\u347e\u347f", - "\u3480\u3481\u3482\u3483\u3484\u3485\u3486\u3487\u3488\u3489\u348a\u348b\u348c\u348d\u348e\u348f", - "\u3490\u3491\u3492\u3493\u3494\u3495\u3496\u3497\u3498\u3499\u349a\u349b\u349c\u349d\u349e\u349f", - "\u34a0\u34a1\u34a2\u34a3\u34a4\u34a5\u34a6\u34a7\u34a8\u34a9\u34aa\u34ab\u34ac\u34ad\u34ae\u34af", - "\u34b0\u34b1\u34b2\u34b3\u34b4\u34b5\u34b6\u34b7\u34b8\u34b9\u34ba\u34bb\u34bc\u34bd\u34be\u34bf", - "\u34c0\u34c1\u34c2\u34c3\u34c4\u34c5\u34c6\u34c7\u34c8\u34c9\u34ca\u34cb\u34cc\u34cd\u34ce\u34cf", - "\u34d0\u34d1\u34d2\u34d3\u34d4\u34d5\u34d6\u34d7\u34d8\u34d9\u34da\u34db\u34dc\u34dd\u34de\u34df", - "\u34e0\u34e1\u34e2\u34e3\u34e4\u34e5\u34e6\u34e7\u34e8\u34e9\u34ea\u34eb\u34ec\u34ed\u34ee\u34ef", - "\u34f0\u34f1\u34f2\u34f3\u34f4\u34f5\u34f6\u34f7\u34f8\u34f9\u34fa\u34fb\u34fc\u34fd\u34fe\u34ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_35.png", - "ascent": %ascent%, - "chars": [ - "\u3500\u3501\u3502\u3503\u3504\u3505\u3506\u3507\u3508\u3509\u350a\u350b\u350c\u350d\u350e\u350f", - "\u3510\u3511\u3512\u3513\u3514\u3515\u3516\u3517\u3518\u3519\u351a\u351b\u351c\u351d\u351e\u351f", - "\u3520\u3521\u3522\u3523\u3524\u3525\u3526\u3527\u3528\u3529\u352a\u352b\u352c\u352d\u352e\u352f", - "\u3530\u3531\u3532\u3533\u3534\u3535\u3536\u3537\u3538\u3539\u353a\u353b\u353c\u353d\u353e\u353f", - "\u3540\u3541\u3542\u3543\u3544\u3545\u3546\u3547\u3548\u3549\u354a\u354b\u354c\u354d\u354e\u354f", - "\u3550\u3551\u3552\u3553\u3554\u3555\u3556\u3557\u3558\u3559\u355a\u355b\u355c\u355d\u355e\u355f", - "\u3560\u3561\u3562\u3563\u3564\u3565\u3566\u3567\u3568\u3569\u356a\u356b\u356c\u356d\u356e\u356f", - "\u3570\u3571\u3572\u3573\u3574\u3575\u3576\u3577\u3578\u3579\u357a\u357b\u357c\u357d\u357e\u357f", - "\u3580\u3581\u3582\u3583\u3584\u3585\u3586\u3587\u3588\u3589\u358a\u358b\u358c\u358d\u358e\u358f", - "\u3590\u3591\u3592\u3593\u3594\u3595\u3596\u3597\u3598\u3599\u359a\u359b\u359c\u359d\u359e\u359f", - "\u35a0\u35a1\u35a2\u35a3\u35a4\u35a5\u35a6\u35a7\u35a8\u35a9\u35aa\u35ab\u35ac\u35ad\u35ae\u35af", - "\u35b0\u35b1\u35b2\u35b3\u35b4\u35b5\u35b6\u35b7\u35b8\u35b9\u35ba\u35bb\u35bc\u35bd\u35be\u35bf", - "\u35c0\u35c1\u35c2\u35c3\u35c4\u35c5\u35c6\u35c7\u35c8\u35c9\u35ca\u35cb\u35cc\u35cd\u35ce\u35cf", - "\u35d0\u35d1\u35d2\u35d3\u35d4\u35d5\u35d6\u35d7\u35d8\u35d9\u35da\u35db\u35dc\u35dd\u35de\u35df", - "\u35e0\u35e1\u35e2\u35e3\u35e4\u35e5\u35e6\u35e7\u35e8\u35e9\u35ea\u35eb\u35ec\u35ed\u35ee\u35ef", - "\u35f0\u35f1\u35f2\u35f3\u35f4\u35f5\u35f6\u35f7\u35f8\u35f9\u35fa\u35fb\u35fc\u35fd\u35fe\u35ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_36.png", - "ascent": %ascent%, - "chars": [ - "\u3600\u3601\u3602\u3603\u3604\u3605\u3606\u3607\u3608\u3609\u360a\u360b\u360c\u360d\u360e\u360f", - "\u3610\u3611\u3612\u3613\u3614\u3615\u3616\u3617\u3618\u3619\u361a\u361b\u361c\u361d\u361e\u361f", - "\u3620\u3621\u3622\u3623\u3624\u3625\u3626\u3627\u3628\u3629\u362a\u362b\u362c\u362d\u362e\u362f", - "\u3630\u3631\u3632\u3633\u3634\u3635\u3636\u3637\u3638\u3639\u363a\u363b\u363c\u363d\u363e\u363f", - "\u3640\u3641\u3642\u3643\u3644\u3645\u3646\u3647\u3648\u3649\u364a\u364b\u364c\u364d\u364e\u364f", - "\u3650\u3651\u3652\u3653\u3654\u3655\u3656\u3657\u3658\u3659\u365a\u365b\u365c\u365d\u365e\u365f", - "\u3660\u3661\u3662\u3663\u3664\u3665\u3666\u3667\u3668\u3669\u366a\u366b\u366c\u366d\u366e\u366f", - "\u3670\u3671\u3672\u3673\u3674\u3675\u3676\u3677\u3678\u3679\u367a\u367b\u367c\u367d\u367e\u367f", - "\u3680\u3681\u3682\u3683\u3684\u3685\u3686\u3687\u3688\u3689\u368a\u368b\u368c\u368d\u368e\u368f", - "\u3690\u3691\u3692\u3693\u3694\u3695\u3696\u3697\u3698\u3699\u369a\u369b\u369c\u369d\u369e\u369f", - "\u36a0\u36a1\u36a2\u36a3\u36a4\u36a5\u36a6\u36a7\u36a8\u36a9\u36aa\u36ab\u36ac\u36ad\u36ae\u36af", - "\u36b0\u36b1\u36b2\u36b3\u36b4\u36b5\u36b6\u36b7\u36b8\u36b9\u36ba\u36bb\u36bc\u36bd\u36be\u36bf", - "\u36c0\u36c1\u36c2\u36c3\u36c4\u36c5\u36c6\u36c7\u36c8\u36c9\u36ca\u36cb\u36cc\u36cd\u36ce\u36cf", - "\u36d0\u36d1\u36d2\u36d3\u36d4\u36d5\u36d6\u36d7\u36d8\u36d9\u36da\u36db\u36dc\u36dd\u36de\u36df", - "\u36e0\u36e1\u36e2\u36e3\u36e4\u36e5\u36e6\u36e7\u36e8\u36e9\u36ea\u36eb\u36ec\u36ed\u36ee\u36ef", - "\u36f0\u36f1\u36f2\u36f3\u36f4\u36f5\u36f6\u36f7\u36f8\u36f9\u36fa\u36fb\u36fc\u36fd\u36fe\u36ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_37.png", - "ascent": %ascent%, - "chars": [ - "\u3700\u3701\u3702\u3703\u3704\u3705\u3706\u3707\u3708\u3709\u370a\u370b\u370c\u370d\u370e\u370f", - "\u3710\u3711\u3712\u3713\u3714\u3715\u3716\u3717\u3718\u3719\u371a\u371b\u371c\u371d\u371e\u371f", - "\u3720\u3721\u3722\u3723\u3724\u3725\u3726\u3727\u3728\u3729\u372a\u372b\u372c\u372d\u372e\u372f", - "\u3730\u3731\u3732\u3733\u3734\u3735\u3736\u3737\u3738\u3739\u373a\u373b\u373c\u373d\u373e\u373f", - "\u3740\u3741\u3742\u3743\u3744\u3745\u3746\u3747\u3748\u3749\u374a\u374b\u374c\u374d\u374e\u374f", - "\u3750\u3751\u3752\u3753\u3754\u3755\u3756\u3757\u3758\u3759\u375a\u375b\u375c\u375d\u375e\u375f", - "\u3760\u3761\u3762\u3763\u3764\u3765\u3766\u3767\u3768\u3769\u376a\u376b\u376c\u376d\u376e\u376f", - "\u3770\u3771\u3772\u3773\u3774\u3775\u3776\u3777\u3778\u3779\u377a\u377b\u377c\u377d\u377e\u377f", - "\u3780\u3781\u3782\u3783\u3784\u3785\u3786\u3787\u3788\u3789\u378a\u378b\u378c\u378d\u378e\u378f", - "\u3790\u3791\u3792\u3793\u3794\u3795\u3796\u3797\u3798\u3799\u379a\u379b\u379c\u379d\u379e\u379f", - "\u37a0\u37a1\u37a2\u37a3\u37a4\u37a5\u37a6\u37a7\u37a8\u37a9\u37aa\u37ab\u37ac\u37ad\u37ae\u37af", - "\u37b0\u37b1\u37b2\u37b3\u37b4\u37b5\u37b6\u37b7\u37b8\u37b9\u37ba\u37bb\u37bc\u37bd\u37be\u37bf", - "\u37c0\u37c1\u37c2\u37c3\u37c4\u37c5\u37c6\u37c7\u37c8\u37c9\u37ca\u37cb\u37cc\u37cd\u37ce\u37cf", - "\u37d0\u37d1\u37d2\u37d3\u37d4\u37d5\u37d6\u37d7\u37d8\u37d9\u37da\u37db\u37dc\u37dd\u37de\u37df", - "\u37e0\u37e1\u37e2\u37e3\u37e4\u37e5\u37e6\u37e7\u37e8\u37e9\u37ea\u37eb\u37ec\u37ed\u37ee\u37ef", - "\u37f0\u37f1\u37f2\u37f3\u37f4\u37f5\u37f6\u37f7\u37f8\u37f9\u37fa\u37fb\u37fc\u37fd\u37fe\u37ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_38.png", - "ascent": %ascent%, - "chars": [ - "\u3800\u3801\u3802\u3803\u3804\u3805\u3806\u3807\u3808\u3809\u380a\u380b\u380c\u380d\u380e\u380f", - "\u3810\u3811\u3812\u3813\u3814\u3815\u3816\u3817\u3818\u3819\u381a\u381b\u381c\u381d\u381e\u381f", - "\u3820\u3821\u3822\u3823\u3824\u3825\u3826\u3827\u3828\u3829\u382a\u382b\u382c\u382d\u382e\u382f", - "\u3830\u3831\u3832\u3833\u3834\u3835\u3836\u3837\u3838\u3839\u383a\u383b\u383c\u383d\u383e\u383f", - "\u3840\u3841\u3842\u3843\u3844\u3845\u3846\u3847\u3848\u3849\u384a\u384b\u384c\u384d\u384e\u384f", - "\u3850\u3851\u3852\u3853\u3854\u3855\u3856\u3857\u3858\u3859\u385a\u385b\u385c\u385d\u385e\u385f", - "\u3860\u3861\u3862\u3863\u3864\u3865\u3866\u3867\u3868\u3869\u386a\u386b\u386c\u386d\u386e\u386f", - "\u3870\u3871\u3872\u3873\u3874\u3875\u3876\u3877\u3878\u3879\u387a\u387b\u387c\u387d\u387e\u387f", - "\u3880\u3881\u3882\u3883\u3884\u3885\u3886\u3887\u3888\u3889\u388a\u388b\u388c\u388d\u388e\u388f", - "\u3890\u3891\u3892\u3893\u3894\u3895\u3896\u3897\u3898\u3899\u389a\u389b\u389c\u389d\u389e\u389f", - "\u38a0\u38a1\u38a2\u38a3\u38a4\u38a5\u38a6\u38a7\u38a8\u38a9\u38aa\u38ab\u38ac\u38ad\u38ae\u38af", - "\u38b0\u38b1\u38b2\u38b3\u38b4\u38b5\u38b6\u38b7\u38b8\u38b9\u38ba\u38bb\u38bc\u38bd\u38be\u38bf", - "\u38c0\u38c1\u38c2\u38c3\u38c4\u38c5\u38c6\u38c7\u38c8\u38c9\u38ca\u38cb\u38cc\u38cd\u38ce\u38cf", - "\u38d0\u38d1\u38d2\u38d3\u38d4\u38d5\u38d6\u38d7\u38d8\u38d9\u38da\u38db\u38dc\u38dd\u38de\u38df", - "\u38e0\u38e1\u38e2\u38e3\u38e4\u38e5\u38e6\u38e7\u38e8\u38e9\u38ea\u38eb\u38ec\u38ed\u38ee\u38ef", - "\u38f0\u38f1\u38f2\u38f3\u38f4\u38f5\u38f6\u38f7\u38f8\u38f9\u38fa\u38fb\u38fc\u38fd\u38fe\u38ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_39.png", - "ascent": %ascent%, - "chars": [ - "\u3900\u3901\u3902\u3903\u3904\u3905\u3906\u3907\u3908\u3909\u390a\u390b\u390c\u390d\u390e\u390f", - "\u3910\u3911\u3912\u3913\u3914\u3915\u3916\u3917\u3918\u3919\u391a\u391b\u391c\u391d\u391e\u391f", - "\u3920\u3921\u3922\u3923\u3924\u3925\u3926\u3927\u3928\u3929\u392a\u392b\u392c\u392d\u392e\u392f", - "\u3930\u3931\u3932\u3933\u3934\u3935\u3936\u3937\u3938\u3939\u393a\u393b\u393c\u393d\u393e\u393f", - "\u3940\u3941\u3942\u3943\u3944\u3945\u3946\u3947\u3948\u3949\u394a\u394b\u394c\u394d\u394e\u394f", - "\u3950\u3951\u3952\u3953\u3954\u3955\u3956\u3957\u3958\u3959\u395a\u395b\u395c\u395d\u395e\u395f", - "\u3960\u3961\u3962\u3963\u3964\u3965\u3966\u3967\u3968\u3969\u396a\u396b\u396c\u396d\u396e\u396f", - "\u3970\u3971\u3972\u3973\u3974\u3975\u3976\u3977\u3978\u3979\u397a\u397b\u397c\u397d\u397e\u397f", - "\u3980\u3981\u3982\u3983\u3984\u3985\u3986\u3987\u3988\u3989\u398a\u398b\u398c\u398d\u398e\u398f", - "\u3990\u3991\u3992\u3993\u3994\u3995\u3996\u3997\u3998\u3999\u399a\u399b\u399c\u399d\u399e\u399f", - "\u39a0\u39a1\u39a2\u39a3\u39a4\u39a5\u39a6\u39a7\u39a8\u39a9\u39aa\u39ab\u39ac\u39ad\u39ae\u39af", - "\u39b0\u39b1\u39b2\u39b3\u39b4\u39b5\u39b6\u39b7\u39b8\u39b9\u39ba\u39bb\u39bc\u39bd\u39be\u39bf", - "\u39c0\u39c1\u39c2\u39c3\u39c4\u39c5\u39c6\u39c7\u39c8\u39c9\u39ca\u39cb\u39cc\u39cd\u39ce\u39cf", - "\u39d0\u39d1\u39d2\u39d3\u39d4\u39d5\u39d6\u39d7\u39d8\u39d9\u39da\u39db\u39dc\u39dd\u39de\u39df", - "\u39e0\u39e1\u39e2\u39e3\u39e4\u39e5\u39e6\u39e7\u39e8\u39e9\u39ea\u39eb\u39ec\u39ed\u39ee\u39ef", - "\u39f0\u39f1\u39f2\u39f3\u39f4\u39f5\u39f6\u39f7\u39f8\u39f9\u39fa\u39fb\u39fc\u39fd\u39fe\u39ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_3a.png", - "ascent": %ascent%, - "chars": [ - "\u3a00\u3a01\u3a02\u3a03\u3a04\u3a05\u3a06\u3a07\u3a08\u3a09\u3a0a\u3a0b\u3a0c\u3a0d\u3a0e\u3a0f", - "\u3a10\u3a11\u3a12\u3a13\u3a14\u3a15\u3a16\u3a17\u3a18\u3a19\u3a1a\u3a1b\u3a1c\u3a1d\u3a1e\u3a1f", - "\u3a20\u3a21\u3a22\u3a23\u3a24\u3a25\u3a26\u3a27\u3a28\u3a29\u3a2a\u3a2b\u3a2c\u3a2d\u3a2e\u3a2f", - "\u3a30\u3a31\u3a32\u3a33\u3a34\u3a35\u3a36\u3a37\u3a38\u3a39\u3a3a\u3a3b\u3a3c\u3a3d\u3a3e\u3a3f", - "\u3a40\u3a41\u3a42\u3a43\u3a44\u3a45\u3a46\u3a47\u3a48\u3a49\u3a4a\u3a4b\u3a4c\u3a4d\u3a4e\u3a4f", - "\u3a50\u3a51\u3a52\u3a53\u3a54\u3a55\u3a56\u3a57\u3a58\u3a59\u3a5a\u3a5b\u3a5c\u3a5d\u3a5e\u3a5f", - "\u3a60\u3a61\u3a62\u3a63\u3a64\u3a65\u3a66\u3a67\u3a68\u3a69\u3a6a\u3a6b\u3a6c\u3a6d\u3a6e\u3a6f", - "\u3a70\u3a71\u3a72\u3a73\u3a74\u3a75\u3a76\u3a77\u3a78\u3a79\u3a7a\u3a7b\u3a7c\u3a7d\u3a7e\u3a7f", - "\u3a80\u3a81\u3a82\u3a83\u3a84\u3a85\u3a86\u3a87\u3a88\u3a89\u3a8a\u3a8b\u3a8c\u3a8d\u3a8e\u3a8f", - "\u3a90\u3a91\u3a92\u3a93\u3a94\u3a95\u3a96\u3a97\u3a98\u3a99\u3a9a\u3a9b\u3a9c\u3a9d\u3a9e\u3a9f", - "\u3aa0\u3aa1\u3aa2\u3aa3\u3aa4\u3aa5\u3aa6\u3aa7\u3aa8\u3aa9\u3aaa\u3aab\u3aac\u3aad\u3aae\u3aaf", - "\u3ab0\u3ab1\u3ab2\u3ab3\u3ab4\u3ab5\u3ab6\u3ab7\u3ab8\u3ab9\u3aba\u3abb\u3abc\u3abd\u3abe\u3abf", - "\u3ac0\u3ac1\u3ac2\u3ac3\u3ac4\u3ac5\u3ac6\u3ac7\u3ac8\u3ac9\u3aca\u3acb\u3acc\u3acd\u3ace\u3acf", - "\u3ad0\u3ad1\u3ad2\u3ad3\u3ad4\u3ad5\u3ad6\u3ad7\u3ad8\u3ad9\u3ada\u3adb\u3adc\u3add\u3ade\u3adf", - "\u3ae0\u3ae1\u3ae2\u3ae3\u3ae4\u3ae5\u3ae6\u3ae7\u3ae8\u3ae9\u3aea\u3aeb\u3aec\u3aed\u3aee\u3aef", - "\u3af0\u3af1\u3af2\u3af3\u3af4\u3af5\u3af6\u3af7\u3af8\u3af9\u3afa\u3afb\u3afc\u3afd\u3afe\u3aff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_3b.png", - "ascent": %ascent%, - "chars": [ - "\u3b00\u3b01\u3b02\u3b03\u3b04\u3b05\u3b06\u3b07\u3b08\u3b09\u3b0a\u3b0b\u3b0c\u3b0d\u3b0e\u3b0f", - "\u3b10\u3b11\u3b12\u3b13\u3b14\u3b15\u3b16\u3b17\u3b18\u3b19\u3b1a\u3b1b\u3b1c\u3b1d\u3b1e\u3b1f", - "\u3b20\u3b21\u3b22\u3b23\u3b24\u3b25\u3b26\u3b27\u3b28\u3b29\u3b2a\u3b2b\u3b2c\u3b2d\u3b2e\u3b2f", - "\u3b30\u3b31\u3b32\u3b33\u3b34\u3b35\u3b36\u3b37\u3b38\u3b39\u3b3a\u3b3b\u3b3c\u3b3d\u3b3e\u3b3f", - "\u3b40\u3b41\u3b42\u3b43\u3b44\u3b45\u3b46\u3b47\u3b48\u3b49\u3b4a\u3b4b\u3b4c\u3b4d\u3b4e\u3b4f", - "\u3b50\u3b51\u3b52\u3b53\u3b54\u3b55\u3b56\u3b57\u3b58\u3b59\u3b5a\u3b5b\u3b5c\u3b5d\u3b5e\u3b5f", - "\u3b60\u3b61\u3b62\u3b63\u3b64\u3b65\u3b66\u3b67\u3b68\u3b69\u3b6a\u3b6b\u3b6c\u3b6d\u3b6e\u3b6f", - "\u3b70\u3b71\u3b72\u3b73\u3b74\u3b75\u3b76\u3b77\u3b78\u3b79\u3b7a\u3b7b\u3b7c\u3b7d\u3b7e\u3b7f", - "\u3b80\u3b81\u3b82\u3b83\u3b84\u3b85\u3b86\u3b87\u3b88\u3b89\u3b8a\u3b8b\u3b8c\u3b8d\u3b8e\u3b8f", - "\u3b90\u3b91\u3b92\u3b93\u3b94\u3b95\u3b96\u3b97\u3b98\u3b99\u3b9a\u3b9b\u3b9c\u3b9d\u3b9e\u3b9f", - "\u3ba0\u3ba1\u3ba2\u3ba3\u3ba4\u3ba5\u3ba6\u3ba7\u3ba8\u3ba9\u3baa\u3bab\u3bac\u3bad\u3bae\u3baf", - "\u3bb0\u3bb1\u3bb2\u3bb3\u3bb4\u3bb5\u3bb6\u3bb7\u3bb8\u3bb9\u3bba\u3bbb\u3bbc\u3bbd\u3bbe\u3bbf", - "\u3bc0\u3bc1\u3bc2\u3bc3\u3bc4\u3bc5\u3bc6\u3bc7\u3bc8\u3bc9\u3bca\u3bcb\u3bcc\u3bcd\u3bce\u3bcf", - "\u3bd0\u3bd1\u3bd2\u3bd3\u3bd4\u3bd5\u3bd6\u3bd7\u3bd8\u3bd9\u3bda\u3bdb\u3bdc\u3bdd\u3bde\u3bdf", - "\u3be0\u3be1\u3be2\u3be3\u3be4\u3be5\u3be6\u3be7\u3be8\u3be9\u3bea\u3beb\u3bec\u3bed\u3bee\u3bef", - "\u3bf0\u3bf1\u3bf2\u3bf3\u3bf4\u3bf5\u3bf6\u3bf7\u3bf8\u3bf9\u3bfa\u3bfb\u3bfc\u3bfd\u3bfe\u3bff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_3c.png", - "ascent": %ascent%, - "chars": [ - "\u3c00\u3c01\u3c02\u3c03\u3c04\u3c05\u3c06\u3c07\u3c08\u3c09\u3c0a\u3c0b\u3c0c\u3c0d\u3c0e\u3c0f", - "\u3c10\u3c11\u3c12\u3c13\u3c14\u3c15\u3c16\u3c17\u3c18\u3c19\u3c1a\u3c1b\u3c1c\u3c1d\u3c1e\u3c1f", - "\u3c20\u3c21\u3c22\u3c23\u3c24\u3c25\u3c26\u3c27\u3c28\u3c29\u3c2a\u3c2b\u3c2c\u3c2d\u3c2e\u3c2f", - "\u3c30\u3c31\u3c32\u3c33\u3c34\u3c35\u3c36\u3c37\u3c38\u3c39\u3c3a\u3c3b\u3c3c\u3c3d\u3c3e\u3c3f", - "\u3c40\u3c41\u3c42\u3c43\u3c44\u3c45\u3c46\u3c47\u3c48\u3c49\u3c4a\u3c4b\u3c4c\u3c4d\u3c4e\u3c4f", - "\u3c50\u3c51\u3c52\u3c53\u3c54\u3c55\u3c56\u3c57\u3c58\u3c59\u3c5a\u3c5b\u3c5c\u3c5d\u3c5e\u3c5f", - "\u3c60\u3c61\u3c62\u3c63\u3c64\u3c65\u3c66\u3c67\u3c68\u3c69\u3c6a\u3c6b\u3c6c\u3c6d\u3c6e\u3c6f", - "\u3c70\u3c71\u3c72\u3c73\u3c74\u3c75\u3c76\u3c77\u3c78\u3c79\u3c7a\u3c7b\u3c7c\u3c7d\u3c7e\u3c7f", - "\u3c80\u3c81\u3c82\u3c83\u3c84\u3c85\u3c86\u3c87\u3c88\u3c89\u3c8a\u3c8b\u3c8c\u3c8d\u3c8e\u3c8f", - "\u3c90\u3c91\u3c92\u3c93\u3c94\u3c95\u3c96\u3c97\u3c98\u3c99\u3c9a\u3c9b\u3c9c\u3c9d\u3c9e\u3c9f", - "\u3ca0\u3ca1\u3ca2\u3ca3\u3ca4\u3ca5\u3ca6\u3ca7\u3ca8\u3ca9\u3caa\u3cab\u3cac\u3cad\u3cae\u3caf", - "\u3cb0\u3cb1\u3cb2\u3cb3\u3cb4\u3cb5\u3cb6\u3cb7\u3cb8\u3cb9\u3cba\u3cbb\u3cbc\u3cbd\u3cbe\u3cbf", - "\u3cc0\u3cc1\u3cc2\u3cc3\u3cc4\u3cc5\u3cc6\u3cc7\u3cc8\u3cc9\u3cca\u3ccb\u3ccc\u3ccd\u3cce\u3ccf", - "\u3cd0\u3cd1\u3cd2\u3cd3\u3cd4\u3cd5\u3cd6\u3cd7\u3cd8\u3cd9\u3cda\u3cdb\u3cdc\u3cdd\u3cde\u3cdf", - "\u3ce0\u3ce1\u3ce2\u3ce3\u3ce4\u3ce5\u3ce6\u3ce7\u3ce8\u3ce9\u3cea\u3ceb\u3cec\u3ced\u3cee\u3cef", - "\u3cf0\u3cf1\u3cf2\u3cf3\u3cf4\u3cf5\u3cf6\u3cf7\u3cf8\u3cf9\u3cfa\u3cfb\u3cfc\u3cfd\u3cfe\u3cff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_3d.png", - "ascent": %ascent%, - "chars": [ - "\u3d00\u3d01\u3d02\u3d03\u3d04\u3d05\u3d06\u3d07\u3d08\u3d09\u3d0a\u3d0b\u3d0c\u3d0d\u3d0e\u3d0f", - "\u3d10\u3d11\u3d12\u3d13\u3d14\u3d15\u3d16\u3d17\u3d18\u3d19\u3d1a\u3d1b\u3d1c\u3d1d\u3d1e\u3d1f", - "\u3d20\u3d21\u3d22\u3d23\u3d24\u3d25\u3d26\u3d27\u3d28\u3d29\u3d2a\u3d2b\u3d2c\u3d2d\u3d2e\u3d2f", - "\u3d30\u3d31\u3d32\u3d33\u3d34\u3d35\u3d36\u3d37\u3d38\u3d39\u3d3a\u3d3b\u3d3c\u3d3d\u3d3e\u3d3f", - "\u3d40\u3d41\u3d42\u3d43\u3d44\u3d45\u3d46\u3d47\u3d48\u3d49\u3d4a\u3d4b\u3d4c\u3d4d\u3d4e\u3d4f", - "\u3d50\u3d51\u3d52\u3d53\u3d54\u3d55\u3d56\u3d57\u3d58\u3d59\u3d5a\u3d5b\u3d5c\u3d5d\u3d5e\u3d5f", - "\u3d60\u3d61\u3d62\u3d63\u3d64\u3d65\u3d66\u3d67\u3d68\u3d69\u3d6a\u3d6b\u3d6c\u3d6d\u3d6e\u3d6f", - "\u3d70\u3d71\u3d72\u3d73\u3d74\u3d75\u3d76\u3d77\u3d78\u3d79\u3d7a\u3d7b\u3d7c\u3d7d\u3d7e\u3d7f", - "\u3d80\u3d81\u3d82\u3d83\u3d84\u3d85\u3d86\u3d87\u3d88\u3d89\u3d8a\u3d8b\u3d8c\u3d8d\u3d8e\u3d8f", - "\u3d90\u3d91\u3d92\u3d93\u3d94\u3d95\u3d96\u3d97\u3d98\u3d99\u3d9a\u3d9b\u3d9c\u3d9d\u3d9e\u3d9f", - "\u3da0\u3da1\u3da2\u3da3\u3da4\u3da5\u3da6\u3da7\u3da8\u3da9\u3daa\u3dab\u3dac\u3dad\u3dae\u3daf", - "\u3db0\u3db1\u3db2\u3db3\u3db4\u3db5\u3db6\u3db7\u3db8\u3db9\u3dba\u3dbb\u3dbc\u3dbd\u3dbe\u3dbf", - "\u3dc0\u3dc1\u3dc2\u3dc3\u3dc4\u3dc5\u3dc6\u3dc7\u3dc8\u3dc9\u3dca\u3dcb\u3dcc\u3dcd\u3dce\u3dcf", - "\u3dd0\u3dd1\u3dd2\u3dd3\u3dd4\u3dd5\u3dd6\u3dd7\u3dd8\u3dd9\u3dda\u3ddb\u3ddc\u3ddd\u3dde\u3ddf", - "\u3de0\u3de1\u3de2\u3de3\u3de4\u3de5\u3de6\u3de7\u3de8\u3de9\u3dea\u3deb\u3dec\u3ded\u3dee\u3def", - "\u3df0\u3df1\u3df2\u3df3\u3df4\u3df5\u3df6\u3df7\u3df8\u3df9\u3dfa\u3dfb\u3dfc\u3dfd\u3dfe\u3dff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_3e.png", - "ascent": %ascent%, - "chars": [ - "\u3e00\u3e01\u3e02\u3e03\u3e04\u3e05\u3e06\u3e07\u3e08\u3e09\u3e0a\u3e0b\u3e0c\u3e0d\u3e0e\u3e0f", - "\u3e10\u3e11\u3e12\u3e13\u3e14\u3e15\u3e16\u3e17\u3e18\u3e19\u3e1a\u3e1b\u3e1c\u3e1d\u3e1e\u3e1f", - "\u3e20\u3e21\u3e22\u3e23\u3e24\u3e25\u3e26\u3e27\u3e28\u3e29\u3e2a\u3e2b\u3e2c\u3e2d\u3e2e\u3e2f", - "\u3e30\u3e31\u3e32\u3e33\u3e34\u3e35\u3e36\u3e37\u3e38\u3e39\u3e3a\u3e3b\u3e3c\u3e3d\u3e3e\u3e3f", - "\u3e40\u3e41\u3e42\u3e43\u3e44\u3e45\u3e46\u3e47\u3e48\u3e49\u3e4a\u3e4b\u3e4c\u3e4d\u3e4e\u3e4f", - "\u3e50\u3e51\u3e52\u3e53\u3e54\u3e55\u3e56\u3e57\u3e58\u3e59\u3e5a\u3e5b\u3e5c\u3e5d\u3e5e\u3e5f", - "\u3e60\u3e61\u3e62\u3e63\u3e64\u3e65\u3e66\u3e67\u3e68\u3e69\u3e6a\u3e6b\u3e6c\u3e6d\u3e6e\u3e6f", - "\u3e70\u3e71\u3e72\u3e73\u3e74\u3e75\u3e76\u3e77\u3e78\u3e79\u3e7a\u3e7b\u3e7c\u3e7d\u3e7e\u3e7f", - "\u3e80\u3e81\u3e82\u3e83\u3e84\u3e85\u3e86\u3e87\u3e88\u3e89\u3e8a\u3e8b\u3e8c\u3e8d\u3e8e\u3e8f", - "\u3e90\u3e91\u3e92\u3e93\u3e94\u3e95\u3e96\u3e97\u3e98\u3e99\u3e9a\u3e9b\u3e9c\u3e9d\u3e9e\u3e9f", - "\u3ea0\u3ea1\u3ea2\u3ea3\u3ea4\u3ea5\u3ea6\u3ea7\u3ea8\u3ea9\u3eaa\u3eab\u3eac\u3ead\u3eae\u3eaf", - "\u3eb0\u3eb1\u3eb2\u3eb3\u3eb4\u3eb5\u3eb6\u3eb7\u3eb8\u3eb9\u3eba\u3ebb\u3ebc\u3ebd\u3ebe\u3ebf", - "\u3ec0\u3ec1\u3ec2\u3ec3\u3ec4\u3ec5\u3ec6\u3ec7\u3ec8\u3ec9\u3eca\u3ecb\u3ecc\u3ecd\u3ece\u3ecf", - "\u3ed0\u3ed1\u3ed2\u3ed3\u3ed4\u3ed5\u3ed6\u3ed7\u3ed8\u3ed9\u3eda\u3edb\u3edc\u3edd\u3ede\u3edf", - "\u3ee0\u3ee1\u3ee2\u3ee3\u3ee4\u3ee5\u3ee6\u3ee7\u3ee8\u3ee9\u3eea\u3eeb\u3eec\u3eed\u3eee\u3eef", - "\u3ef0\u3ef1\u3ef2\u3ef3\u3ef4\u3ef5\u3ef6\u3ef7\u3ef8\u3ef9\u3efa\u3efb\u3efc\u3efd\u3efe\u3eff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_3f.png", - "ascent": %ascent%, - "chars": [ - "\u3f00\u3f01\u3f02\u3f03\u3f04\u3f05\u3f06\u3f07\u3f08\u3f09\u3f0a\u3f0b\u3f0c\u3f0d\u3f0e\u3f0f", - "\u3f10\u3f11\u3f12\u3f13\u3f14\u3f15\u3f16\u3f17\u3f18\u3f19\u3f1a\u3f1b\u3f1c\u3f1d\u3f1e\u3f1f", - "\u3f20\u3f21\u3f22\u3f23\u3f24\u3f25\u3f26\u3f27\u3f28\u3f29\u3f2a\u3f2b\u3f2c\u3f2d\u3f2e\u3f2f", - "\u3f30\u3f31\u3f32\u3f33\u3f34\u3f35\u3f36\u3f37\u3f38\u3f39\u3f3a\u3f3b\u3f3c\u3f3d\u3f3e\u3f3f", - "\u3f40\u3f41\u3f42\u3f43\u3f44\u3f45\u3f46\u3f47\u3f48\u3f49\u3f4a\u3f4b\u3f4c\u3f4d\u3f4e\u3f4f", - "\u3f50\u3f51\u3f52\u3f53\u3f54\u3f55\u3f56\u3f57\u3f58\u3f59\u3f5a\u3f5b\u3f5c\u3f5d\u3f5e\u3f5f", - "\u3f60\u3f61\u3f62\u3f63\u3f64\u3f65\u3f66\u3f67\u3f68\u3f69\u3f6a\u3f6b\u3f6c\u3f6d\u3f6e\u3f6f", - "\u3f70\u3f71\u3f72\u3f73\u3f74\u3f75\u3f76\u3f77\u3f78\u3f79\u3f7a\u3f7b\u3f7c\u3f7d\u3f7e\u3f7f", - "\u3f80\u3f81\u3f82\u3f83\u3f84\u3f85\u3f86\u3f87\u3f88\u3f89\u3f8a\u3f8b\u3f8c\u3f8d\u3f8e\u3f8f", - "\u3f90\u3f91\u3f92\u3f93\u3f94\u3f95\u3f96\u3f97\u3f98\u3f99\u3f9a\u3f9b\u3f9c\u3f9d\u3f9e\u3f9f", - "\u3fa0\u3fa1\u3fa2\u3fa3\u3fa4\u3fa5\u3fa6\u3fa7\u3fa8\u3fa9\u3faa\u3fab\u3fac\u3fad\u3fae\u3faf", - "\u3fb0\u3fb1\u3fb2\u3fb3\u3fb4\u3fb5\u3fb6\u3fb7\u3fb8\u3fb9\u3fba\u3fbb\u3fbc\u3fbd\u3fbe\u3fbf", - "\u3fc0\u3fc1\u3fc2\u3fc3\u3fc4\u3fc5\u3fc6\u3fc7\u3fc8\u3fc9\u3fca\u3fcb\u3fcc\u3fcd\u3fce\u3fcf", - "\u3fd0\u3fd1\u3fd2\u3fd3\u3fd4\u3fd5\u3fd6\u3fd7\u3fd8\u3fd9\u3fda\u3fdb\u3fdc\u3fdd\u3fde\u3fdf", - "\u3fe0\u3fe1\u3fe2\u3fe3\u3fe4\u3fe5\u3fe6\u3fe7\u3fe8\u3fe9\u3fea\u3feb\u3fec\u3fed\u3fee\u3fef", - "\u3ff0\u3ff1\u3ff2\u3ff3\u3ff4\u3ff5\u3ff6\u3ff7\u3ff8\u3ff9\u3ffa\u3ffb\u3ffc\u3ffd\u3ffe\u3fff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_40.png", - "ascent": %ascent%, - "chars": [ - "\u4000\u4001\u4002\u4003\u4004\u4005\u4006\u4007\u4008\u4009\u400a\u400b\u400c\u400d\u400e\u400f", - "\u4010\u4011\u4012\u4013\u4014\u4015\u4016\u4017\u4018\u4019\u401a\u401b\u401c\u401d\u401e\u401f", - "\u4020\u4021\u4022\u4023\u4024\u4025\u4026\u4027\u4028\u4029\u402a\u402b\u402c\u402d\u402e\u402f", - "\u4030\u4031\u4032\u4033\u4034\u4035\u4036\u4037\u4038\u4039\u403a\u403b\u403c\u403d\u403e\u403f", - "\u4040\u4041\u4042\u4043\u4044\u4045\u4046\u4047\u4048\u4049\u404a\u404b\u404c\u404d\u404e\u404f", - "\u4050\u4051\u4052\u4053\u4054\u4055\u4056\u4057\u4058\u4059\u405a\u405b\u405c\u405d\u405e\u405f", - "\u4060\u4061\u4062\u4063\u4064\u4065\u4066\u4067\u4068\u4069\u406a\u406b\u406c\u406d\u406e\u406f", - "\u4070\u4071\u4072\u4073\u4074\u4075\u4076\u4077\u4078\u4079\u407a\u407b\u407c\u407d\u407e\u407f", - "\u4080\u4081\u4082\u4083\u4084\u4085\u4086\u4087\u4088\u4089\u408a\u408b\u408c\u408d\u408e\u408f", - "\u4090\u4091\u4092\u4093\u4094\u4095\u4096\u4097\u4098\u4099\u409a\u409b\u409c\u409d\u409e\u409f", - "\u40a0\u40a1\u40a2\u40a3\u40a4\u40a5\u40a6\u40a7\u40a8\u40a9\u40aa\u40ab\u40ac\u40ad\u40ae\u40af", - "\u40b0\u40b1\u40b2\u40b3\u40b4\u40b5\u40b6\u40b7\u40b8\u40b9\u40ba\u40bb\u40bc\u40bd\u40be\u40bf", - "\u40c0\u40c1\u40c2\u40c3\u40c4\u40c5\u40c6\u40c7\u40c8\u40c9\u40ca\u40cb\u40cc\u40cd\u40ce\u40cf", - "\u40d0\u40d1\u40d2\u40d3\u40d4\u40d5\u40d6\u40d7\u40d8\u40d9\u40da\u40db\u40dc\u40dd\u40de\u40df", - "\u40e0\u40e1\u40e2\u40e3\u40e4\u40e5\u40e6\u40e7\u40e8\u40e9\u40ea\u40eb\u40ec\u40ed\u40ee\u40ef", - "\u40f0\u40f1\u40f2\u40f3\u40f4\u40f5\u40f6\u40f7\u40f8\u40f9\u40fa\u40fb\u40fc\u40fd\u40fe\u40ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_41.png", - "ascent": %ascent%, - "chars": [ - "\u4100\u4101\u4102\u4103\u4104\u4105\u4106\u4107\u4108\u4109\u410a\u410b\u410c\u410d\u410e\u410f", - "\u4110\u4111\u4112\u4113\u4114\u4115\u4116\u4117\u4118\u4119\u411a\u411b\u411c\u411d\u411e\u411f", - "\u4120\u4121\u4122\u4123\u4124\u4125\u4126\u4127\u4128\u4129\u412a\u412b\u412c\u412d\u412e\u412f", - "\u4130\u4131\u4132\u4133\u4134\u4135\u4136\u4137\u4138\u4139\u413a\u413b\u413c\u413d\u413e\u413f", - "\u4140\u4141\u4142\u4143\u4144\u4145\u4146\u4147\u4148\u4149\u414a\u414b\u414c\u414d\u414e\u414f", - "\u4150\u4151\u4152\u4153\u4154\u4155\u4156\u4157\u4158\u4159\u415a\u415b\u415c\u415d\u415e\u415f", - "\u4160\u4161\u4162\u4163\u4164\u4165\u4166\u4167\u4168\u4169\u416a\u416b\u416c\u416d\u416e\u416f", - "\u4170\u4171\u4172\u4173\u4174\u4175\u4176\u4177\u4178\u4179\u417a\u417b\u417c\u417d\u417e\u417f", - "\u4180\u4181\u4182\u4183\u4184\u4185\u4186\u4187\u4188\u4189\u418a\u418b\u418c\u418d\u418e\u418f", - "\u4190\u4191\u4192\u4193\u4194\u4195\u4196\u4197\u4198\u4199\u419a\u419b\u419c\u419d\u419e\u419f", - "\u41a0\u41a1\u41a2\u41a3\u41a4\u41a5\u41a6\u41a7\u41a8\u41a9\u41aa\u41ab\u41ac\u41ad\u41ae\u41af", - "\u41b0\u41b1\u41b2\u41b3\u41b4\u41b5\u41b6\u41b7\u41b8\u41b9\u41ba\u41bb\u41bc\u41bd\u41be\u41bf", - "\u41c0\u41c1\u41c2\u41c3\u41c4\u41c5\u41c6\u41c7\u41c8\u41c9\u41ca\u41cb\u41cc\u41cd\u41ce\u41cf", - "\u41d0\u41d1\u41d2\u41d3\u41d4\u41d5\u41d6\u41d7\u41d8\u41d9\u41da\u41db\u41dc\u41dd\u41de\u41df", - "\u41e0\u41e1\u41e2\u41e3\u41e4\u41e5\u41e6\u41e7\u41e8\u41e9\u41ea\u41eb\u41ec\u41ed\u41ee\u41ef", - "\u41f0\u41f1\u41f2\u41f3\u41f4\u41f5\u41f6\u41f7\u41f8\u41f9\u41fa\u41fb\u41fc\u41fd\u41fe\u41ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_42.png", - "ascent": %ascent%, - "chars": [ - "\u4200\u4201\u4202\u4203\u4204\u4205\u4206\u4207\u4208\u4209\u420a\u420b\u420c\u420d\u420e\u420f", - "\u4210\u4211\u4212\u4213\u4214\u4215\u4216\u4217\u4218\u4219\u421a\u421b\u421c\u421d\u421e\u421f", - "\u4220\u4221\u4222\u4223\u4224\u4225\u4226\u4227\u4228\u4229\u422a\u422b\u422c\u422d\u422e\u422f", - "\u4230\u4231\u4232\u4233\u4234\u4235\u4236\u4237\u4238\u4239\u423a\u423b\u423c\u423d\u423e\u423f", - "\u4240\u4241\u4242\u4243\u4244\u4245\u4246\u4247\u4248\u4249\u424a\u424b\u424c\u424d\u424e\u424f", - "\u4250\u4251\u4252\u4253\u4254\u4255\u4256\u4257\u4258\u4259\u425a\u425b\u425c\u425d\u425e\u425f", - "\u4260\u4261\u4262\u4263\u4264\u4265\u4266\u4267\u4268\u4269\u426a\u426b\u426c\u426d\u426e\u426f", - "\u4270\u4271\u4272\u4273\u4274\u4275\u4276\u4277\u4278\u4279\u427a\u427b\u427c\u427d\u427e\u427f", - "\u4280\u4281\u4282\u4283\u4284\u4285\u4286\u4287\u4288\u4289\u428a\u428b\u428c\u428d\u428e\u428f", - "\u4290\u4291\u4292\u4293\u4294\u4295\u4296\u4297\u4298\u4299\u429a\u429b\u429c\u429d\u429e\u429f", - "\u42a0\u42a1\u42a2\u42a3\u42a4\u42a5\u42a6\u42a7\u42a8\u42a9\u42aa\u42ab\u42ac\u42ad\u42ae\u42af", - "\u42b0\u42b1\u42b2\u42b3\u42b4\u42b5\u42b6\u42b7\u42b8\u42b9\u42ba\u42bb\u42bc\u42bd\u42be\u42bf", - "\u42c0\u42c1\u42c2\u42c3\u42c4\u42c5\u42c6\u42c7\u42c8\u42c9\u42ca\u42cb\u42cc\u42cd\u42ce\u42cf", - "\u42d0\u42d1\u42d2\u42d3\u42d4\u42d5\u42d6\u42d7\u42d8\u42d9\u42da\u42db\u42dc\u42dd\u42de\u42df", - "\u42e0\u42e1\u42e2\u42e3\u42e4\u42e5\u42e6\u42e7\u42e8\u42e9\u42ea\u42eb\u42ec\u42ed\u42ee\u42ef", - "\u42f0\u42f1\u42f2\u42f3\u42f4\u42f5\u42f6\u42f7\u42f8\u42f9\u42fa\u42fb\u42fc\u42fd\u42fe\u42ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_43.png", - "ascent": %ascent%, - "chars": [ - "\u4300\u4301\u4302\u4303\u4304\u4305\u4306\u4307\u4308\u4309\u430a\u430b\u430c\u430d\u430e\u430f", - "\u4310\u4311\u4312\u4313\u4314\u4315\u4316\u4317\u4318\u4319\u431a\u431b\u431c\u431d\u431e\u431f", - "\u4320\u4321\u4322\u4323\u4324\u4325\u4326\u4327\u4328\u4329\u432a\u432b\u432c\u432d\u432e\u432f", - "\u4330\u4331\u4332\u4333\u4334\u4335\u4336\u4337\u4338\u4339\u433a\u433b\u433c\u433d\u433e\u433f", - "\u4340\u4341\u4342\u4343\u4344\u4345\u4346\u4347\u4348\u4349\u434a\u434b\u434c\u434d\u434e\u434f", - "\u4350\u4351\u4352\u4353\u4354\u4355\u4356\u4357\u4358\u4359\u435a\u435b\u435c\u435d\u435e\u435f", - "\u4360\u4361\u4362\u4363\u4364\u4365\u4366\u4367\u4368\u4369\u436a\u436b\u436c\u436d\u436e\u436f", - "\u4370\u4371\u4372\u4373\u4374\u4375\u4376\u4377\u4378\u4379\u437a\u437b\u437c\u437d\u437e\u437f", - "\u4380\u4381\u4382\u4383\u4384\u4385\u4386\u4387\u4388\u4389\u438a\u438b\u438c\u438d\u438e\u438f", - "\u4390\u4391\u4392\u4393\u4394\u4395\u4396\u4397\u4398\u4399\u439a\u439b\u439c\u439d\u439e\u439f", - "\u43a0\u43a1\u43a2\u43a3\u43a4\u43a5\u43a6\u43a7\u43a8\u43a9\u43aa\u43ab\u43ac\u43ad\u43ae\u43af", - "\u43b0\u43b1\u43b2\u43b3\u43b4\u43b5\u43b6\u43b7\u43b8\u43b9\u43ba\u43bb\u43bc\u43bd\u43be\u43bf", - "\u43c0\u43c1\u43c2\u43c3\u43c4\u43c5\u43c6\u43c7\u43c8\u43c9\u43ca\u43cb\u43cc\u43cd\u43ce\u43cf", - "\u43d0\u43d1\u43d2\u43d3\u43d4\u43d5\u43d6\u43d7\u43d8\u43d9\u43da\u43db\u43dc\u43dd\u43de\u43df", - "\u43e0\u43e1\u43e2\u43e3\u43e4\u43e5\u43e6\u43e7\u43e8\u43e9\u43ea\u43eb\u43ec\u43ed\u43ee\u43ef", - "\u43f0\u43f1\u43f2\u43f3\u43f4\u43f5\u43f6\u43f7\u43f8\u43f9\u43fa\u43fb\u43fc\u43fd\u43fe\u43ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_44.png", - "ascent": %ascent%, - "chars": [ - "\u4400\u4401\u4402\u4403\u4404\u4405\u4406\u4407\u4408\u4409\u440a\u440b\u440c\u440d\u440e\u440f", - "\u4410\u4411\u4412\u4413\u4414\u4415\u4416\u4417\u4418\u4419\u441a\u441b\u441c\u441d\u441e\u441f", - "\u4420\u4421\u4422\u4423\u4424\u4425\u4426\u4427\u4428\u4429\u442a\u442b\u442c\u442d\u442e\u442f", - "\u4430\u4431\u4432\u4433\u4434\u4435\u4436\u4437\u4438\u4439\u443a\u443b\u443c\u443d\u443e\u443f", - "\u4440\u4441\u4442\u4443\u4444\u4445\u4446\u4447\u4448\u4449\u444a\u444b\u444c\u444d\u444e\u444f", - "\u4450\u4451\u4452\u4453\u4454\u4455\u4456\u4457\u4458\u4459\u445a\u445b\u445c\u445d\u445e\u445f", - "\u4460\u4461\u4462\u4463\u4464\u4465\u4466\u4467\u4468\u4469\u446a\u446b\u446c\u446d\u446e\u446f", - "\u4470\u4471\u4472\u4473\u4474\u4475\u4476\u4477\u4478\u4479\u447a\u447b\u447c\u447d\u447e\u447f", - "\u4480\u4481\u4482\u4483\u4484\u4485\u4486\u4487\u4488\u4489\u448a\u448b\u448c\u448d\u448e\u448f", - "\u4490\u4491\u4492\u4493\u4494\u4495\u4496\u4497\u4498\u4499\u449a\u449b\u449c\u449d\u449e\u449f", - "\u44a0\u44a1\u44a2\u44a3\u44a4\u44a5\u44a6\u44a7\u44a8\u44a9\u44aa\u44ab\u44ac\u44ad\u44ae\u44af", - "\u44b0\u44b1\u44b2\u44b3\u44b4\u44b5\u44b6\u44b7\u44b8\u44b9\u44ba\u44bb\u44bc\u44bd\u44be\u44bf", - "\u44c0\u44c1\u44c2\u44c3\u44c4\u44c5\u44c6\u44c7\u44c8\u44c9\u44ca\u44cb\u44cc\u44cd\u44ce\u44cf", - "\u44d0\u44d1\u44d2\u44d3\u44d4\u44d5\u44d6\u44d7\u44d8\u44d9\u44da\u44db\u44dc\u44dd\u44de\u44df", - "\u44e0\u44e1\u44e2\u44e3\u44e4\u44e5\u44e6\u44e7\u44e8\u44e9\u44ea\u44eb\u44ec\u44ed\u44ee\u44ef", - "\u44f0\u44f1\u44f2\u44f3\u44f4\u44f5\u44f6\u44f7\u44f8\u44f9\u44fa\u44fb\u44fc\u44fd\u44fe\u44ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_45.png", - "ascent": %ascent%, - "chars": [ - "\u4500\u4501\u4502\u4503\u4504\u4505\u4506\u4507\u4508\u4509\u450a\u450b\u450c\u450d\u450e\u450f", - "\u4510\u4511\u4512\u4513\u4514\u4515\u4516\u4517\u4518\u4519\u451a\u451b\u451c\u451d\u451e\u451f", - "\u4520\u4521\u4522\u4523\u4524\u4525\u4526\u4527\u4528\u4529\u452a\u452b\u452c\u452d\u452e\u452f", - "\u4530\u4531\u4532\u4533\u4534\u4535\u4536\u4537\u4538\u4539\u453a\u453b\u453c\u453d\u453e\u453f", - "\u4540\u4541\u4542\u4543\u4544\u4545\u4546\u4547\u4548\u4549\u454a\u454b\u454c\u454d\u454e\u454f", - "\u4550\u4551\u4552\u4553\u4554\u4555\u4556\u4557\u4558\u4559\u455a\u455b\u455c\u455d\u455e\u455f", - "\u4560\u4561\u4562\u4563\u4564\u4565\u4566\u4567\u4568\u4569\u456a\u456b\u456c\u456d\u456e\u456f", - "\u4570\u4571\u4572\u4573\u4574\u4575\u4576\u4577\u4578\u4579\u457a\u457b\u457c\u457d\u457e\u457f", - "\u4580\u4581\u4582\u4583\u4584\u4585\u4586\u4587\u4588\u4589\u458a\u458b\u458c\u458d\u458e\u458f", - "\u4590\u4591\u4592\u4593\u4594\u4595\u4596\u4597\u4598\u4599\u459a\u459b\u459c\u459d\u459e\u459f", - "\u45a0\u45a1\u45a2\u45a3\u45a4\u45a5\u45a6\u45a7\u45a8\u45a9\u45aa\u45ab\u45ac\u45ad\u45ae\u45af", - "\u45b0\u45b1\u45b2\u45b3\u45b4\u45b5\u45b6\u45b7\u45b8\u45b9\u45ba\u45bb\u45bc\u45bd\u45be\u45bf", - "\u45c0\u45c1\u45c2\u45c3\u45c4\u45c5\u45c6\u45c7\u45c8\u45c9\u45ca\u45cb\u45cc\u45cd\u45ce\u45cf", - "\u45d0\u45d1\u45d2\u45d3\u45d4\u45d5\u45d6\u45d7\u45d8\u45d9\u45da\u45db\u45dc\u45dd\u45de\u45df", - "\u45e0\u45e1\u45e2\u45e3\u45e4\u45e5\u45e6\u45e7\u45e8\u45e9\u45ea\u45eb\u45ec\u45ed\u45ee\u45ef", - "\u45f0\u45f1\u45f2\u45f3\u45f4\u45f5\u45f6\u45f7\u45f8\u45f9\u45fa\u45fb\u45fc\u45fd\u45fe\u45ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_46.png", - "ascent": %ascent%, - "chars": [ - "\u4600\u4601\u4602\u4603\u4604\u4605\u4606\u4607\u4608\u4609\u460a\u460b\u460c\u460d\u460e\u460f", - "\u4610\u4611\u4612\u4613\u4614\u4615\u4616\u4617\u4618\u4619\u461a\u461b\u461c\u461d\u461e\u461f", - "\u4620\u4621\u4622\u4623\u4624\u4625\u4626\u4627\u4628\u4629\u462a\u462b\u462c\u462d\u462e\u462f", - "\u4630\u4631\u4632\u4633\u4634\u4635\u4636\u4637\u4638\u4639\u463a\u463b\u463c\u463d\u463e\u463f", - "\u4640\u4641\u4642\u4643\u4644\u4645\u4646\u4647\u4648\u4649\u464a\u464b\u464c\u464d\u464e\u464f", - "\u4650\u4651\u4652\u4653\u4654\u4655\u4656\u4657\u4658\u4659\u465a\u465b\u465c\u465d\u465e\u465f", - "\u4660\u4661\u4662\u4663\u4664\u4665\u4666\u4667\u4668\u4669\u466a\u466b\u466c\u466d\u466e\u466f", - "\u4670\u4671\u4672\u4673\u4674\u4675\u4676\u4677\u4678\u4679\u467a\u467b\u467c\u467d\u467e\u467f", - "\u4680\u4681\u4682\u4683\u4684\u4685\u4686\u4687\u4688\u4689\u468a\u468b\u468c\u468d\u468e\u468f", - "\u4690\u4691\u4692\u4693\u4694\u4695\u4696\u4697\u4698\u4699\u469a\u469b\u469c\u469d\u469e\u469f", - "\u46a0\u46a1\u46a2\u46a3\u46a4\u46a5\u46a6\u46a7\u46a8\u46a9\u46aa\u46ab\u46ac\u46ad\u46ae\u46af", - "\u46b0\u46b1\u46b2\u46b3\u46b4\u46b5\u46b6\u46b7\u46b8\u46b9\u46ba\u46bb\u46bc\u46bd\u46be\u46bf", - "\u46c0\u46c1\u46c2\u46c3\u46c4\u46c5\u46c6\u46c7\u46c8\u46c9\u46ca\u46cb\u46cc\u46cd\u46ce\u46cf", - "\u46d0\u46d1\u46d2\u46d3\u46d4\u46d5\u46d6\u46d7\u46d8\u46d9\u46da\u46db\u46dc\u46dd\u46de\u46df", - "\u46e0\u46e1\u46e2\u46e3\u46e4\u46e5\u46e6\u46e7\u46e8\u46e9\u46ea\u46eb\u46ec\u46ed\u46ee\u46ef", - "\u46f0\u46f1\u46f2\u46f3\u46f4\u46f5\u46f6\u46f7\u46f8\u46f9\u46fa\u46fb\u46fc\u46fd\u46fe\u46ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_47.png", - "ascent": %ascent%, - "chars": [ - "\u4700\u4701\u4702\u4703\u4704\u4705\u4706\u4707\u4708\u4709\u470a\u470b\u470c\u470d\u470e\u470f", - "\u4710\u4711\u4712\u4713\u4714\u4715\u4716\u4717\u4718\u4719\u471a\u471b\u471c\u471d\u471e\u471f", - "\u4720\u4721\u4722\u4723\u4724\u4725\u4726\u4727\u4728\u4729\u472a\u472b\u472c\u472d\u472e\u472f", - "\u4730\u4731\u4732\u4733\u4734\u4735\u4736\u4737\u4738\u4739\u473a\u473b\u473c\u473d\u473e\u473f", - "\u4740\u4741\u4742\u4743\u4744\u4745\u4746\u4747\u4748\u4749\u474a\u474b\u474c\u474d\u474e\u474f", - "\u4750\u4751\u4752\u4753\u4754\u4755\u4756\u4757\u4758\u4759\u475a\u475b\u475c\u475d\u475e\u475f", - "\u4760\u4761\u4762\u4763\u4764\u4765\u4766\u4767\u4768\u4769\u476a\u476b\u476c\u476d\u476e\u476f", - "\u4770\u4771\u4772\u4773\u4774\u4775\u4776\u4777\u4778\u4779\u477a\u477b\u477c\u477d\u477e\u477f", - "\u4780\u4781\u4782\u4783\u4784\u4785\u4786\u4787\u4788\u4789\u478a\u478b\u478c\u478d\u478e\u478f", - "\u4790\u4791\u4792\u4793\u4794\u4795\u4796\u4797\u4798\u4799\u479a\u479b\u479c\u479d\u479e\u479f", - "\u47a0\u47a1\u47a2\u47a3\u47a4\u47a5\u47a6\u47a7\u47a8\u47a9\u47aa\u47ab\u47ac\u47ad\u47ae\u47af", - "\u47b0\u47b1\u47b2\u47b3\u47b4\u47b5\u47b6\u47b7\u47b8\u47b9\u47ba\u47bb\u47bc\u47bd\u47be\u47bf", - "\u47c0\u47c1\u47c2\u47c3\u47c4\u47c5\u47c6\u47c7\u47c8\u47c9\u47ca\u47cb\u47cc\u47cd\u47ce\u47cf", - "\u47d0\u47d1\u47d2\u47d3\u47d4\u47d5\u47d6\u47d7\u47d8\u47d9\u47da\u47db\u47dc\u47dd\u47de\u47df", - "\u47e0\u47e1\u47e2\u47e3\u47e4\u47e5\u47e6\u47e7\u47e8\u47e9\u47ea\u47eb\u47ec\u47ed\u47ee\u47ef", - "\u47f0\u47f1\u47f2\u47f3\u47f4\u47f5\u47f6\u47f7\u47f8\u47f9\u47fa\u47fb\u47fc\u47fd\u47fe\u47ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_48.png", - "ascent": %ascent%, - "chars": [ - "\u4800\u4801\u4802\u4803\u4804\u4805\u4806\u4807\u4808\u4809\u480a\u480b\u480c\u480d\u480e\u480f", - "\u4810\u4811\u4812\u4813\u4814\u4815\u4816\u4817\u4818\u4819\u481a\u481b\u481c\u481d\u481e\u481f", - "\u4820\u4821\u4822\u4823\u4824\u4825\u4826\u4827\u4828\u4829\u482a\u482b\u482c\u482d\u482e\u482f", - "\u4830\u4831\u4832\u4833\u4834\u4835\u4836\u4837\u4838\u4839\u483a\u483b\u483c\u483d\u483e\u483f", - "\u4840\u4841\u4842\u4843\u4844\u4845\u4846\u4847\u4848\u4849\u484a\u484b\u484c\u484d\u484e\u484f", - "\u4850\u4851\u4852\u4853\u4854\u4855\u4856\u4857\u4858\u4859\u485a\u485b\u485c\u485d\u485e\u485f", - "\u4860\u4861\u4862\u4863\u4864\u4865\u4866\u4867\u4868\u4869\u486a\u486b\u486c\u486d\u486e\u486f", - "\u4870\u4871\u4872\u4873\u4874\u4875\u4876\u4877\u4878\u4879\u487a\u487b\u487c\u487d\u487e\u487f", - "\u4880\u4881\u4882\u4883\u4884\u4885\u4886\u4887\u4888\u4889\u488a\u488b\u488c\u488d\u488e\u488f", - "\u4890\u4891\u4892\u4893\u4894\u4895\u4896\u4897\u4898\u4899\u489a\u489b\u489c\u489d\u489e\u489f", - "\u48a0\u48a1\u48a2\u48a3\u48a4\u48a5\u48a6\u48a7\u48a8\u48a9\u48aa\u48ab\u48ac\u48ad\u48ae\u48af", - "\u48b0\u48b1\u48b2\u48b3\u48b4\u48b5\u48b6\u48b7\u48b8\u48b9\u48ba\u48bb\u48bc\u48bd\u48be\u48bf", - "\u48c0\u48c1\u48c2\u48c3\u48c4\u48c5\u48c6\u48c7\u48c8\u48c9\u48ca\u48cb\u48cc\u48cd\u48ce\u48cf", - "\u48d0\u48d1\u48d2\u48d3\u48d4\u48d5\u48d6\u48d7\u48d8\u48d9\u48da\u48db\u48dc\u48dd\u48de\u48df", - "\u48e0\u48e1\u48e2\u48e3\u48e4\u48e5\u48e6\u48e7\u48e8\u48e9\u48ea\u48eb\u48ec\u48ed\u48ee\u48ef", - "\u48f0\u48f1\u48f2\u48f3\u48f4\u48f5\u48f6\u48f7\u48f8\u48f9\u48fa\u48fb\u48fc\u48fd\u48fe\u48ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_49.png", - "ascent": %ascent%, - "chars": [ - "\u4900\u4901\u4902\u4903\u4904\u4905\u4906\u4907\u4908\u4909\u490a\u490b\u490c\u490d\u490e\u490f", - "\u4910\u4911\u4912\u4913\u4914\u4915\u4916\u4917\u4918\u4919\u491a\u491b\u491c\u491d\u491e\u491f", - "\u4920\u4921\u4922\u4923\u4924\u4925\u4926\u4927\u4928\u4929\u492a\u492b\u492c\u492d\u492e\u492f", - "\u4930\u4931\u4932\u4933\u4934\u4935\u4936\u4937\u4938\u4939\u493a\u493b\u493c\u493d\u493e\u493f", - "\u4940\u4941\u4942\u4943\u4944\u4945\u4946\u4947\u4948\u4949\u494a\u494b\u494c\u494d\u494e\u494f", - "\u4950\u4951\u4952\u4953\u4954\u4955\u4956\u4957\u4958\u4959\u495a\u495b\u495c\u495d\u495e\u495f", - "\u4960\u4961\u4962\u4963\u4964\u4965\u4966\u4967\u4968\u4969\u496a\u496b\u496c\u496d\u496e\u496f", - "\u4970\u4971\u4972\u4973\u4974\u4975\u4976\u4977\u4978\u4979\u497a\u497b\u497c\u497d\u497e\u497f", - "\u4980\u4981\u4982\u4983\u4984\u4985\u4986\u4987\u4988\u4989\u498a\u498b\u498c\u498d\u498e\u498f", - "\u4990\u4991\u4992\u4993\u4994\u4995\u4996\u4997\u4998\u4999\u499a\u499b\u499c\u499d\u499e\u499f", - "\u49a0\u49a1\u49a2\u49a3\u49a4\u49a5\u49a6\u49a7\u49a8\u49a9\u49aa\u49ab\u49ac\u49ad\u49ae\u49af", - "\u49b0\u49b1\u49b2\u49b3\u49b4\u49b5\u49b6\u49b7\u49b8\u49b9\u49ba\u49bb\u49bc\u49bd\u49be\u49bf", - "\u49c0\u49c1\u49c2\u49c3\u49c4\u49c5\u49c6\u49c7\u49c8\u49c9\u49ca\u49cb\u49cc\u49cd\u49ce\u49cf", - "\u49d0\u49d1\u49d2\u49d3\u49d4\u49d5\u49d6\u49d7\u49d8\u49d9\u49da\u49db\u49dc\u49dd\u49de\u49df", - "\u49e0\u49e1\u49e2\u49e3\u49e4\u49e5\u49e6\u49e7\u49e8\u49e9\u49ea\u49eb\u49ec\u49ed\u49ee\u49ef", - "\u49f0\u49f1\u49f2\u49f3\u49f4\u49f5\u49f6\u49f7\u49f8\u49f9\u49fa\u49fb\u49fc\u49fd\u49fe\u49ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_4a.png", - "ascent": %ascent%, - "chars": [ - "\u4a00\u4a01\u4a02\u4a03\u4a04\u4a05\u4a06\u4a07\u4a08\u4a09\u4a0a\u4a0b\u4a0c\u4a0d\u4a0e\u4a0f", - "\u4a10\u4a11\u4a12\u4a13\u4a14\u4a15\u4a16\u4a17\u4a18\u4a19\u4a1a\u4a1b\u4a1c\u4a1d\u4a1e\u4a1f", - "\u4a20\u4a21\u4a22\u4a23\u4a24\u4a25\u4a26\u4a27\u4a28\u4a29\u4a2a\u4a2b\u4a2c\u4a2d\u4a2e\u4a2f", - "\u4a30\u4a31\u4a32\u4a33\u4a34\u4a35\u4a36\u4a37\u4a38\u4a39\u4a3a\u4a3b\u4a3c\u4a3d\u4a3e\u4a3f", - "\u4a40\u4a41\u4a42\u4a43\u4a44\u4a45\u4a46\u4a47\u4a48\u4a49\u4a4a\u4a4b\u4a4c\u4a4d\u4a4e\u4a4f", - "\u4a50\u4a51\u4a52\u4a53\u4a54\u4a55\u4a56\u4a57\u4a58\u4a59\u4a5a\u4a5b\u4a5c\u4a5d\u4a5e\u4a5f", - "\u4a60\u4a61\u4a62\u4a63\u4a64\u4a65\u4a66\u4a67\u4a68\u4a69\u4a6a\u4a6b\u4a6c\u4a6d\u4a6e\u4a6f", - "\u4a70\u4a71\u4a72\u4a73\u4a74\u4a75\u4a76\u4a77\u4a78\u4a79\u4a7a\u4a7b\u4a7c\u4a7d\u4a7e\u4a7f", - "\u4a80\u4a81\u4a82\u4a83\u4a84\u4a85\u4a86\u4a87\u4a88\u4a89\u4a8a\u4a8b\u4a8c\u4a8d\u4a8e\u4a8f", - "\u4a90\u4a91\u4a92\u4a93\u4a94\u4a95\u4a96\u4a97\u4a98\u4a99\u4a9a\u4a9b\u4a9c\u4a9d\u4a9e\u4a9f", - "\u4aa0\u4aa1\u4aa2\u4aa3\u4aa4\u4aa5\u4aa6\u4aa7\u4aa8\u4aa9\u4aaa\u4aab\u4aac\u4aad\u4aae\u4aaf", - "\u4ab0\u4ab1\u4ab2\u4ab3\u4ab4\u4ab5\u4ab6\u4ab7\u4ab8\u4ab9\u4aba\u4abb\u4abc\u4abd\u4abe\u4abf", - "\u4ac0\u4ac1\u4ac2\u4ac3\u4ac4\u4ac5\u4ac6\u4ac7\u4ac8\u4ac9\u4aca\u4acb\u4acc\u4acd\u4ace\u4acf", - "\u4ad0\u4ad1\u4ad2\u4ad3\u4ad4\u4ad5\u4ad6\u4ad7\u4ad8\u4ad9\u4ada\u4adb\u4adc\u4add\u4ade\u4adf", - "\u4ae0\u4ae1\u4ae2\u4ae3\u4ae4\u4ae5\u4ae6\u4ae7\u4ae8\u4ae9\u4aea\u4aeb\u4aec\u4aed\u4aee\u4aef", - "\u4af0\u4af1\u4af2\u4af3\u4af4\u4af5\u4af6\u4af7\u4af8\u4af9\u4afa\u4afb\u4afc\u4afd\u4afe\u4aff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_4b.png", - "ascent": %ascent%, - "chars": [ - "\u4b00\u4b01\u4b02\u4b03\u4b04\u4b05\u4b06\u4b07\u4b08\u4b09\u4b0a\u4b0b\u4b0c\u4b0d\u4b0e\u4b0f", - "\u4b10\u4b11\u4b12\u4b13\u4b14\u4b15\u4b16\u4b17\u4b18\u4b19\u4b1a\u4b1b\u4b1c\u4b1d\u4b1e\u4b1f", - "\u4b20\u4b21\u4b22\u4b23\u4b24\u4b25\u4b26\u4b27\u4b28\u4b29\u4b2a\u4b2b\u4b2c\u4b2d\u4b2e\u4b2f", - "\u4b30\u4b31\u4b32\u4b33\u4b34\u4b35\u4b36\u4b37\u4b38\u4b39\u4b3a\u4b3b\u4b3c\u4b3d\u4b3e\u4b3f", - "\u4b40\u4b41\u4b42\u4b43\u4b44\u4b45\u4b46\u4b47\u4b48\u4b49\u4b4a\u4b4b\u4b4c\u4b4d\u4b4e\u4b4f", - "\u4b50\u4b51\u4b52\u4b53\u4b54\u4b55\u4b56\u4b57\u4b58\u4b59\u4b5a\u4b5b\u4b5c\u4b5d\u4b5e\u4b5f", - "\u4b60\u4b61\u4b62\u4b63\u4b64\u4b65\u4b66\u4b67\u4b68\u4b69\u4b6a\u4b6b\u4b6c\u4b6d\u4b6e\u4b6f", - "\u4b70\u4b71\u4b72\u4b73\u4b74\u4b75\u4b76\u4b77\u4b78\u4b79\u4b7a\u4b7b\u4b7c\u4b7d\u4b7e\u4b7f", - "\u4b80\u4b81\u4b82\u4b83\u4b84\u4b85\u4b86\u4b87\u4b88\u4b89\u4b8a\u4b8b\u4b8c\u4b8d\u4b8e\u4b8f", - "\u4b90\u4b91\u4b92\u4b93\u4b94\u4b95\u4b96\u4b97\u4b98\u4b99\u4b9a\u4b9b\u4b9c\u4b9d\u4b9e\u4b9f", - "\u4ba0\u4ba1\u4ba2\u4ba3\u4ba4\u4ba5\u4ba6\u4ba7\u4ba8\u4ba9\u4baa\u4bab\u4bac\u4bad\u4bae\u4baf", - "\u4bb0\u4bb1\u4bb2\u4bb3\u4bb4\u4bb5\u4bb6\u4bb7\u4bb8\u4bb9\u4bba\u4bbb\u4bbc\u4bbd\u4bbe\u4bbf", - "\u4bc0\u4bc1\u4bc2\u4bc3\u4bc4\u4bc5\u4bc6\u4bc7\u4bc8\u4bc9\u4bca\u4bcb\u4bcc\u4bcd\u4bce\u4bcf", - "\u4bd0\u4bd1\u4bd2\u4bd3\u4bd4\u4bd5\u4bd6\u4bd7\u4bd8\u4bd9\u4bda\u4bdb\u4bdc\u4bdd\u4bde\u4bdf", - "\u4be0\u4be1\u4be2\u4be3\u4be4\u4be5\u4be6\u4be7\u4be8\u4be9\u4bea\u4beb\u4bec\u4bed\u4bee\u4bef", - "\u4bf0\u4bf1\u4bf2\u4bf3\u4bf4\u4bf5\u4bf6\u4bf7\u4bf8\u4bf9\u4bfa\u4bfb\u4bfc\u4bfd\u4bfe\u4bff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_4c.png", - "ascent": %ascent%, - "chars": [ - "\u4c00\u4c01\u4c02\u4c03\u4c04\u4c05\u4c06\u4c07\u4c08\u4c09\u4c0a\u4c0b\u4c0c\u4c0d\u4c0e\u4c0f", - "\u4c10\u4c11\u4c12\u4c13\u4c14\u4c15\u4c16\u4c17\u4c18\u4c19\u4c1a\u4c1b\u4c1c\u4c1d\u4c1e\u4c1f", - "\u4c20\u4c21\u4c22\u4c23\u4c24\u4c25\u4c26\u4c27\u4c28\u4c29\u4c2a\u4c2b\u4c2c\u4c2d\u4c2e\u4c2f", - "\u4c30\u4c31\u4c32\u4c33\u4c34\u4c35\u4c36\u4c37\u4c38\u4c39\u4c3a\u4c3b\u4c3c\u4c3d\u4c3e\u4c3f", - "\u4c40\u4c41\u4c42\u4c43\u4c44\u4c45\u4c46\u4c47\u4c48\u4c49\u4c4a\u4c4b\u4c4c\u4c4d\u4c4e\u4c4f", - "\u4c50\u4c51\u4c52\u4c53\u4c54\u4c55\u4c56\u4c57\u4c58\u4c59\u4c5a\u4c5b\u4c5c\u4c5d\u4c5e\u4c5f", - "\u4c60\u4c61\u4c62\u4c63\u4c64\u4c65\u4c66\u4c67\u4c68\u4c69\u4c6a\u4c6b\u4c6c\u4c6d\u4c6e\u4c6f", - "\u4c70\u4c71\u4c72\u4c73\u4c74\u4c75\u4c76\u4c77\u4c78\u4c79\u4c7a\u4c7b\u4c7c\u4c7d\u4c7e\u4c7f", - "\u4c80\u4c81\u4c82\u4c83\u4c84\u4c85\u4c86\u4c87\u4c88\u4c89\u4c8a\u4c8b\u4c8c\u4c8d\u4c8e\u4c8f", - "\u4c90\u4c91\u4c92\u4c93\u4c94\u4c95\u4c96\u4c97\u4c98\u4c99\u4c9a\u4c9b\u4c9c\u4c9d\u4c9e\u4c9f", - "\u4ca0\u4ca1\u4ca2\u4ca3\u4ca4\u4ca5\u4ca6\u4ca7\u4ca8\u4ca9\u4caa\u4cab\u4cac\u4cad\u4cae\u4caf", - "\u4cb0\u4cb1\u4cb2\u4cb3\u4cb4\u4cb5\u4cb6\u4cb7\u4cb8\u4cb9\u4cba\u4cbb\u4cbc\u4cbd\u4cbe\u4cbf", - "\u4cc0\u4cc1\u4cc2\u4cc3\u4cc4\u4cc5\u4cc6\u4cc7\u4cc8\u4cc9\u4cca\u4ccb\u4ccc\u4ccd\u4cce\u4ccf", - "\u4cd0\u4cd1\u4cd2\u4cd3\u4cd4\u4cd5\u4cd6\u4cd7\u4cd8\u4cd9\u4cda\u4cdb\u4cdc\u4cdd\u4cde\u4cdf", - "\u4ce0\u4ce1\u4ce2\u4ce3\u4ce4\u4ce5\u4ce6\u4ce7\u4ce8\u4ce9\u4cea\u4ceb\u4cec\u4ced\u4cee\u4cef", - "\u4cf0\u4cf1\u4cf2\u4cf3\u4cf4\u4cf5\u4cf6\u4cf7\u4cf8\u4cf9\u4cfa\u4cfb\u4cfc\u4cfd\u4cfe\u4cff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_4d.png", - "ascent": %ascent%, - "chars": [ - "\u4d00\u4d01\u4d02\u4d03\u4d04\u4d05\u4d06\u4d07\u4d08\u4d09\u4d0a\u4d0b\u4d0c\u4d0d\u4d0e\u4d0f", - "\u4d10\u4d11\u4d12\u4d13\u4d14\u4d15\u4d16\u4d17\u4d18\u4d19\u4d1a\u4d1b\u4d1c\u4d1d\u4d1e\u4d1f", - "\u4d20\u4d21\u4d22\u4d23\u4d24\u4d25\u4d26\u4d27\u4d28\u4d29\u4d2a\u4d2b\u4d2c\u4d2d\u4d2e\u4d2f", - "\u4d30\u4d31\u4d32\u4d33\u4d34\u4d35\u4d36\u4d37\u4d38\u4d39\u4d3a\u4d3b\u4d3c\u4d3d\u4d3e\u4d3f", - "\u4d40\u4d41\u4d42\u4d43\u4d44\u4d45\u4d46\u4d47\u4d48\u4d49\u4d4a\u4d4b\u4d4c\u4d4d\u4d4e\u4d4f", - "\u4d50\u4d51\u4d52\u4d53\u4d54\u4d55\u4d56\u4d57\u4d58\u4d59\u4d5a\u4d5b\u4d5c\u4d5d\u4d5e\u4d5f", - "\u4d60\u4d61\u4d62\u4d63\u4d64\u4d65\u4d66\u4d67\u4d68\u4d69\u4d6a\u4d6b\u4d6c\u4d6d\u4d6e\u4d6f", - "\u4d70\u4d71\u4d72\u4d73\u4d74\u4d75\u4d76\u4d77\u4d78\u4d79\u4d7a\u4d7b\u4d7c\u4d7d\u4d7e\u4d7f", - "\u4d80\u4d81\u4d82\u4d83\u4d84\u4d85\u4d86\u4d87\u4d88\u4d89\u4d8a\u4d8b\u4d8c\u4d8d\u4d8e\u4d8f", - "\u4d90\u4d91\u4d92\u4d93\u4d94\u4d95\u4d96\u4d97\u4d98\u4d99\u4d9a\u4d9b\u4d9c\u4d9d\u4d9e\u4d9f", - "\u4da0\u4da1\u4da2\u4da3\u4da4\u4da5\u4da6\u4da7\u4da8\u4da9\u4daa\u4dab\u4dac\u4dad\u4dae\u4daf", - "\u4db0\u4db1\u4db2\u4db3\u4db4\u4db5\u4db6\u4db7\u4db8\u4db9\u4dba\u4dbb\u4dbc\u4dbd\u4dbe\u4dbf", - "\u4dc0\u4dc1\u4dc2\u4dc3\u4dc4\u4dc5\u4dc6\u4dc7\u4dc8\u4dc9\u4dca\u4dcb\u4dcc\u4dcd\u4dce\u4dcf", - "\u4dd0\u4dd1\u4dd2\u4dd3\u4dd4\u4dd5\u4dd6\u4dd7\u4dd8\u4dd9\u4dda\u4ddb\u4ddc\u4ddd\u4dde\u4ddf", - "\u4de0\u4de1\u4de2\u4de3\u4de4\u4de5\u4de6\u4de7\u4de8\u4de9\u4dea\u4deb\u4dec\u4ded\u4dee\u4def", - "\u4df0\u4df1\u4df2\u4df3\u4df4\u4df5\u4df6\u4df7\u4df8\u4df9\u4dfa\u4dfb\u4dfc\u4dfd\u4dfe\u4dff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_4e.png", - "ascent": %ascent%, - "chars": [ - "\u4e00\u4e01\u4e02\u4e03\u4e04\u4e05\u4e06\u4e07\u4e08\u4e09\u4e0a\u4e0b\u4e0c\u4e0d\u4e0e\u4e0f", - "\u4e10\u4e11\u4e12\u4e13\u4e14\u4e15\u4e16\u4e17\u4e18\u4e19\u4e1a\u4e1b\u4e1c\u4e1d\u4e1e\u4e1f", - "\u4e20\u4e21\u4e22\u4e23\u4e24\u4e25\u4e26\u4e27\u4e28\u4e29\u4e2a\u4e2b\u4e2c\u4e2d\u4e2e\u4e2f", - "\u4e30\u4e31\u4e32\u4e33\u4e34\u4e35\u4e36\u4e37\u4e38\u4e39\u4e3a\u4e3b\u4e3c\u4e3d\u4e3e\u4e3f", - "\u4e40\u4e41\u4e42\u4e43\u4e44\u4e45\u4e46\u4e47\u4e48\u4e49\u4e4a\u4e4b\u4e4c\u4e4d\u4e4e\u4e4f", - "\u4e50\u4e51\u4e52\u4e53\u4e54\u4e55\u4e56\u4e57\u4e58\u4e59\u4e5a\u4e5b\u4e5c\u4e5d\u4e5e\u4e5f", - "\u4e60\u4e61\u4e62\u4e63\u4e64\u4e65\u4e66\u4e67\u4e68\u4e69\u4e6a\u4e6b\u4e6c\u4e6d\u4e6e\u4e6f", - "\u4e70\u4e71\u4e72\u4e73\u4e74\u4e75\u4e76\u4e77\u4e78\u4e79\u4e7a\u4e7b\u4e7c\u4e7d\u4e7e\u4e7f", - "\u4e80\u4e81\u4e82\u4e83\u4e84\u4e85\u4e86\u4e87\u4e88\u4e89\u4e8a\u4e8b\u4e8c\u4e8d\u4e8e\u4e8f", - "\u4e90\u4e91\u4e92\u4e93\u4e94\u4e95\u4e96\u4e97\u4e98\u4e99\u4e9a\u4e9b\u4e9c\u4e9d\u4e9e\u4e9f", - "\u4ea0\u4ea1\u4ea2\u4ea3\u4ea4\u4ea5\u4ea6\u4ea7\u4ea8\u4ea9\u4eaa\u4eab\u4eac\u4ead\u4eae\u4eaf", - "\u4eb0\u4eb1\u4eb2\u4eb3\u4eb4\u4eb5\u4eb6\u4eb7\u4eb8\u4eb9\u4eba\u4ebb\u4ebc\u4ebd\u4ebe\u4ebf", - "\u4ec0\u4ec1\u4ec2\u4ec3\u4ec4\u4ec5\u4ec6\u4ec7\u4ec8\u4ec9\u4eca\u4ecb\u4ecc\u4ecd\u4ece\u4ecf", - "\u4ed0\u4ed1\u4ed2\u4ed3\u4ed4\u4ed5\u4ed6\u4ed7\u4ed8\u4ed9\u4eda\u4edb\u4edc\u4edd\u4ede\u4edf", - "\u4ee0\u4ee1\u4ee2\u4ee3\u4ee4\u4ee5\u4ee6\u4ee7\u4ee8\u4ee9\u4eea\u4eeb\u4eec\u4eed\u4eee\u4eef", - "\u4ef0\u4ef1\u4ef2\u4ef3\u4ef4\u4ef5\u4ef6\u4ef7\u4ef8\u4ef9\u4efa\u4efb\u4efc\u4efd\u4efe\u4eff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_4f.png", - "ascent": %ascent%, - "chars": [ - "\u4f00\u4f01\u4f02\u4f03\u4f04\u4f05\u4f06\u4f07\u4f08\u4f09\u4f0a\u4f0b\u4f0c\u4f0d\u4f0e\u4f0f", - "\u4f10\u4f11\u4f12\u4f13\u4f14\u4f15\u4f16\u4f17\u4f18\u4f19\u4f1a\u4f1b\u4f1c\u4f1d\u4f1e\u4f1f", - "\u4f20\u4f21\u4f22\u4f23\u4f24\u4f25\u4f26\u4f27\u4f28\u4f29\u4f2a\u4f2b\u4f2c\u4f2d\u4f2e\u4f2f", - "\u4f30\u4f31\u4f32\u4f33\u4f34\u4f35\u4f36\u4f37\u4f38\u4f39\u4f3a\u4f3b\u4f3c\u4f3d\u4f3e\u4f3f", - "\u4f40\u4f41\u4f42\u4f43\u4f44\u4f45\u4f46\u4f47\u4f48\u4f49\u4f4a\u4f4b\u4f4c\u4f4d\u4f4e\u4f4f", - "\u4f50\u4f51\u4f52\u4f53\u4f54\u4f55\u4f56\u4f57\u4f58\u4f59\u4f5a\u4f5b\u4f5c\u4f5d\u4f5e\u4f5f", - "\u4f60\u4f61\u4f62\u4f63\u4f64\u4f65\u4f66\u4f67\u4f68\u4f69\u4f6a\u4f6b\u4f6c\u4f6d\u4f6e\u4f6f", - "\u4f70\u4f71\u4f72\u4f73\u4f74\u4f75\u4f76\u4f77\u4f78\u4f79\u4f7a\u4f7b\u4f7c\u4f7d\u4f7e\u4f7f", - "\u4f80\u4f81\u4f82\u4f83\u4f84\u4f85\u4f86\u4f87\u4f88\u4f89\u4f8a\u4f8b\u4f8c\u4f8d\u4f8e\u4f8f", - "\u4f90\u4f91\u4f92\u4f93\u4f94\u4f95\u4f96\u4f97\u4f98\u4f99\u4f9a\u4f9b\u4f9c\u4f9d\u4f9e\u4f9f", - "\u4fa0\u4fa1\u4fa2\u4fa3\u4fa4\u4fa5\u4fa6\u4fa7\u4fa8\u4fa9\u4faa\u4fab\u4fac\u4fad\u4fae\u4faf", - "\u4fb0\u4fb1\u4fb2\u4fb3\u4fb4\u4fb5\u4fb6\u4fb7\u4fb8\u4fb9\u4fba\u4fbb\u4fbc\u4fbd\u4fbe\u4fbf", - "\u4fc0\u4fc1\u4fc2\u4fc3\u4fc4\u4fc5\u4fc6\u4fc7\u4fc8\u4fc9\u4fca\u4fcb\u4fcc\u4fcd\u4fce\u4fcf", - "\u4fd0\u4fd1\u4fd2\u4fd3\u4fd4\u4fd5\u4fd6\u4fd7\u4fd8\u4fd9\u4fda\u4fdb\u4fdc\u4fdd\u4fde\u4fdf", - "\u4fe0\u4fe1\u4fe2\u4fe3\u4fe4\u4fe5\u4fe6\u4fe7\u4fe8\u4fe9\u4fea\u4feb\u4fec\u4fed\u4fee\u4fef", - "\u4ff0\u4ff1\u4ff2\u4ff3\u4ff4\u4ff5\u4ff6\u4ff7\u4ff8\u4ff9\u4ffa\u4ffb\u4ffc\u4ffd\u4ffe\u4fff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_50.png", - "ascent": %ascent%, - "chars": [ - "\u5000\u5001\u5002\u5003\u5004\u5005\u5006\u5007\u5008\u5009\u500a\u500b\u500c\u500d\u500e\u500f", - "\u5010\u5011\u5012\u5013\u5014\u5015\u5016\u5017\u5018\u5019\u501a\u501b\u501c\u501d\u501e\u501f", - "\u5020\u5021\u5022\u5023\u5024\u5025\u5026\u5027\u5028\u5029\u502a\u502b\u502c\u502d\u502e\u502f", - "\u5030\u5031\u5032\u5033\u5034\u5035\u5036\u5037\u5038\u5039\u503a\u503b\u503c\u503d\u503e\u503f", - "\u5040\u5041\u5042\u5043\u5044\u5045\u5046\u5047\u5048\u5049\u504a\u504b\u504c\u504d\u504e\u504f", - "\u5050\u5051\u5052\u5053\u5054\u5055\u5056\u5057\u5058\u5059\u505a\u505b\u505c\u505d\u505e\u505f", - "\u5060\u5061\u5062\u5063\u5064\u5065\u5066\u5067\u5068\u5069\u506a\u506b\u506c\u506d\u506e\u506f", - "\u5070\u5071\u5072\u5073\u5074\u5075\u5076\u5077\u5078\u5079\u507a\u507b\u507c\u507d\u507e\u507f", - "\u5080\u5081\u5082\u5083\u5084\u5085\u5086\u5087\u5088\u5089\u508a\u508b\u508c\u508d\u508e\u508f", - "\u5090\u5091\u5092\u5093\u5094\u5095\u5096\u5097\u5098\u5099\u509a\u509b\u509c\u509d\u509e\u509f", - "\u50a0\u50a1\u50a2\u50a3\u50a4\u50a5\u50a6\u50a7\u50a8\u50a9\u50aa\u50ab\u50ac\u50ad\u50ae\u50af", - "\u50b0\u50b1\u50b2\u50b3\u50b4\u50b5\u50b6\u50b7\u50b8\u50b9\u50ba\u50bb\u50bc\u50bd\u50be\u50bf", - "\u50c0\u50c1\u50c2\u50c3\u50c4\u50c5\u50c6\u50c7\u50c8\u50c9\u50ca\u50cb\u50cc\u50cd\u50ce\u50cf", - "\u50d0\u50d1\u50d2\u50d3\u50d4\u50d5\u50d6\u50d7\u50d8\u50d9\u50da\u50db\u50dc\u50dd\u50de\u50df", - "\u50e0\u50e1\u50e2\u50e3\u50e4\u50e5\u50e6\u50e7\u50e8\u50e9\u50ea\u50eb\u50ec\u50ed\u50ee\u50ef", - "\u50f0\u50f1\u50f2\u50f3\u50f4\u50f5\u50f6\u50f7\u50f8\u50f9\u50fa\u50fb\u50fc\u50fd\u50fe\u50ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_51.png", - "ascent": %ascent%, - "chars": [ - "\u5100\u5101\u5102\u5103\u5104\u5105\u5106\u5107\u5108\u5109\u510a\u510b\u510c\u510d\u510e\u510f", - "\u5110\u5111\u5112\u5113\u5114\u5115\u5116\u5117\u5118\u5119\u511a\u511b\u511c\u511d\u511e\u511f", - "\u5120\u5121\u5122\u5123\u5124\u5125\u5126\u5127\u5128\u5129\u512a\u512b\u512c\u512d\u512e\u512f", - "\u5130\u5131\u5132\u5133\u5134\u5135\u5136\u5137\u5138\u5139\u513a\u513b\u513c\u513d\u513e\u513f", - "\u5140\u5141\u5142\u5143\u5144\u5145\u5146\u5147\u5148\u5149\u514a\u514b\u514c\u514d\u514e\u514f", - "\u5150\u5151\u5152\u5153\u5154\u5155\u5156\u5157\u5158\u5159\u515a\u515b\u515c\u515d\u515e\u515f", - "\u5160\u5161\u5162\u5163\u5164\u5165\u5166\u5167\u5168\u5169\u516a\u516b\u516c\u516d\u516e\u516f", - "\u5170\u5171\u5172\u5173\u5174\u5175\u5176\u5177\u5178\u5179\u517a\u517b\u517c\u517d\u517e\u517f", - "\u5180\u5181\u5182\u5183\u5184\u5185\u5186\u5187\u5188\u5189\u518a\u518b\u518c\u518d\u518e\u518f", - "\u5190\u5191\u5192\u5193\u5194\u5195\u5196\u5197\u5198\u5199\u519a\u519b\u519c\u519d\u519e\u519f", - "\u51a0\u51a1\u51a2\u51a3\u51a4\u51a5\u51a6\u51a7\u51a8\u51a9\u51aa\u51ab\u51ac\u51ad\u51ae\u51af", - "\u51b0\u51b1\u51b2\u51b3\u51b4\u51b5\u51b6\u51b7\u51b8\u51b9\u51ba\u51bb\u51bc\u51bd\u51be\u51bf", - "\u51c0\u51c1\u51c2\u51c3\u51c4\u51c5\u51c6\u51c7\u51c8\u51c9\u51ca\u51cb\u51cc\u51cd\u51ce\u51cf", - "\u51d0\u51d1\u51d2\u51d3\u51d4\u51d5\u51d6\u51d7\u51d8\u51d9\u51da\u51db\u51dc\u51dd\u51de\u51df", - "\u51e0\u51e1\u51e2\u51e3\u51e4\u51e5\u51e6\u51e7\u51e8\u51e9\u51ea\u51eb\u51ec\u51ed\u51ee\u51ef", - "\u51f0\u51f1\u51f2\u51f3\u51f4\u51f5\u51f6\u51f7\u51f8\u51f9\u51fa\u51fb\u51fc\u51fd\u51fe\u51ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_52.png", - "ascent": %ascent%, - "chars": [ - "\u5200\u5201\u5202\u5203\u5204\u5205\u5206\u5207\u5208\u5209\u520a\u520b\u520c\u520d\u520e\u520f", - "\u5210\u5211\u5212\u5213\u5214\u5215\u5216\u5217\u5218\u5219\u521a\u521b\u521c\u521d\u521e\u521f", - "\u5220\u5221\u5222\u5223\u5224\u5225\u5226\u5227\u5228\u5229\u522a\u522b\u522c\u522d\u522e\u522f", - "\u5230\u5231\u5232\u5233\u5234\u5235\u5236\u5237\u5238\u5239\u523a\u523b\u523c\u523d\u523e\u523f", - "\u5240\u5241\u5242\u5243\u5244\u5245\u5246\u5247\u5248\u5249\u524a\u524b\u524c\u524d\u524e\u524f", - "\u5250\u5251\u5252\u5253\u5254\u5255\u5256\u5257\u5258\u5259\u525a\u525b\u525c\u525d\u525e\u525f", - "\u5260\u5261\u5262\u5263\u5264\u5265\u5266\u5267\u5268\u5269\u526a\u526b\u526c\u526d\u526e\u526f", - "\u5270\u5271\u5272\u5273\u5274\u5275\u5276\u5277\u5278\u5279\u527a\u527b\u527c\u527d\u527e\u527f", - "\u5280\u5281\u5282\u5283\u5284\u5285\u5286\u5287\u5288\u5289\u528a\u528b\u528c\u528d\u528e\u528f", - "\u5290\u5291\u5292\u5293\u5294\u5295\u5296\u5297\u5298\u5299\u529a\u529b\u529c\u529d\u529e\u529f", - "\u52a0\u52a1\u52a2\u52a3\u52a4\u52a5\u52a6\u52a7\u52a8\u52a9\u52aa\u52ab\u52ac\u52ad\u52ae\u52af", - "\u52b0\u52b1\u52b2\u52b3\u52b4\u52b5\u52b6\u52b7\u52b8\u52b9\u52ba\u52bb\u52bc\u52bd\u52be\u52bf", - "\u52c0\u52c1\u52c2\u52c3\u52c4\u52c5\u52c6\u52c7\u52c8\u52c9\u52ca\u52cb\u52cc\u52cd\u52ce\u52cf", - "\u52d0\u52d1\u52d2\u52d3\u52d4\u52d5\u52d6\u52d7\u52d8\u52d9\u52da\u52db\u52dc\u52dd\u52de\u52df", - "\u52e0\u52e1\u52e2\u52e3\u52e4\u52e5\u52e6\u52e7\u52e8\u52e9\u52ea\u52eb\u52ec\u52ed\u52ee\u52ef", - "\u52f0\u52f1\u52f2\u52f3\u52f4\u52f5\u52f6\u52f7\u52f8\u52f9\u52fa\u52fb\u52fc\u52fd\u52fe\u52ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_53.png", - "ascent": %ascent%, - "chars": [ - "\u5300\u5301\u5302\u5303\u5304\u5305\u5306\u5307\u5308\u5309\u530a\u530b\u530c\u530d\u530e\u530f", - "\u5310\u5311\u5312\u5313\u5314\u5315\u5316\u5317\u5318\u5319\u531a\u531b\u531c\u531d\u531e\u531f", - "\u5320\u5321\u5322\u5323\u5324\u5325\u5326\u5327\u5328\u5329\u532a\u532b\u532c\u532d\u532e\u532f", - "\u5330\u5331\u5332\u5333\u5334\u5335\u5336\u5337\u5338\u5339\u533a\u533b\u533c\u533d\u533e\u533f", - "\u5340\u5341\u5342\u5343\u5344\u5345\u5346\u5347\u5348\u5349\u534a\u534b\u534c\u534d\u534e\u534f", - "\u5350\u5351\u5352\u5353\u5354\u5355\u5356\u5357\u5358\u5359\u535a\u535b\u535c\u535d\u535e\u535f", - "\u5360\u5361\u5362\u5363\u5364\u5365\u5366\u5367\u5368\u5369\u536a\u536b\u536c\u536d\u536e\u536f", - "\u5370\u5371\u5372\u5373\u5374\u5375\u5376\u5377\u5378\u5379\u537a\u537b\u537c\u537d\u537e\u537f", - "\u5380\u5381\u5382\u5383\u5384\u5385\u5386\u5387\u5388\u5389\u538a\u538b\u538c\u538d\u538e\u538f", - "\u5390\u5391\u5392\u5393\u5394\u5395\u5396\u5397\u5398\u5399\u539a\u539b\u539c\u539d\u539e\u539f", - "\u53a0\u53a1\u53a2\u53a3\u53a4\u53a5\u53a6\u53a7\u53a8\u53a9\u53aa\u53ab\u53ac\u53ad\u53ae\u53af", - "\u53b0\u53b1\u53b2\u53b3\u53b4\u53b5\u53b6\u53b7\u53b8\u53b9\u53ba\u53bb\u53bc\u53bd\u53be\u53bf", - "\u53c0\u53c1\u53c2\u53c3\u53c4\u53c5\u53c6\u53c7\u53c8\u53c9\u53ca\u53cb\u53cc\u53cd\u53ce\u53cf", - "\u53d0\u53d1\u53d2\u53d3\u53d4\u53d5\u53d6\u53d7\u53d8\u53d9\u53da\u53db\u53dc\u53dd\u53de\u53df", - "\u53e0\u53e1\u53e2\u53e3\u53e4\u53e5\u53e6\u53e7\u53e8\u53e9\u53ea\u53eb\u53ec\u53ed\u53ee\u53ef", - "\u53f0\u53f1\u53f2\u53f3\u53f4\u53f5\u53f6\u53f7\u53f8\u53f9\u53fa\u53fb\u53fc\u53fd\u53fe\u53ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_54.png", - "ascent": %ascent%, - "chars": [ - "\u5400\u5401\u5402\u5403\u5404\u5405\u5406\u5407\u5408\u5409\u540a\u540b\u540c\u540d\u540e\u540f", - "\u5410\u5411\u5412\u5413\u5414\u5415\u5416\u5417\u5418\u5419\u541a\u541b\u541c\u541d\u541e\u541f", - "\u5420\u5421\u5422\u5423\u5424\u5425\u5426\u5427\u5428\u5429\u542a\u542b\u542c\u542d\u542e\u542f", - "\u5430\u5431\u5432\u5433\u5434\u5435\u5436\u5437\u5438\u5439\u543a\u543b\u543c\u543d\u543e\u543f", - "\u5440\u5441\u5442\u5443\u5444\u5445\u5446\u5447\u5448\u5449\u544a\u544b\u544c\u544d\u544e\u544f", - "\u5450\u5451\u5452\u5453\u5454\u5455\u5456\u5457\u5458\u5459\u545a\u545b\u545c\u545d\u545e\u545f", - "\u5460\u5461\u5462\u5463\u5464\u5465\u5466\u5467\u5468\u5469\u546a\u546b\u546c\u546d\u546e\u546f", - "\u5470\u5471\u5472\u5473\u5474\u5475\u5476\u5477\u5478\u5479\u547a\u547b\u547c\u547d\u547e\u547f", - "\u5480\u5481\u5482\u5483\u5484\u5485\u5486\u5487\u5488\u5489\u548a\u548b\u548c\u548d\u548e\u548f", - "\u5490\u5491\u5492\u5493\u5494\u5495\u5496\u5497\u5498\u5499\u549a\u549b\u549c\u549d\u549e\u549f", - "\u54a0\u54a1\u54a2\u54a3\u54a4\u54a5\u54a6\u54a7\u54a8\u54a9\u54aa\u54ab\u54ac\u54ad\u54ae\u54af", - "\u54b0\u54b1\u54b2\u54b3\u54b4\u54b5\u54b6\u54b7\u54b8\u54b9\u54ba\u54bb\u54bc\u54bd\u54be\u54bf", - "\u54c0\u54c1\u54c2\u54c3\u54c4\u54c5\u54c6\u54c7\u54c8\u54c9\u54ca\u54cb\u54cc\u54cd\u54ce\u54cf", - "\u54d0\u54d1\u54d2\u54d3\u54d4\u54d5\u54d6\u54d7\u54d8\u54d9\u54da\u54db\u54dc\u54dd\u54de\u54df", - "\u54e0\u54e1\u54e2\u54e3\u54e4\u54e5\u54e6\u54e7\u54e8\u54e9\u54ea\u54eb\u54ec\u54ed\u54ee\u54ef", - "\u54f0\u54f1\u54f2\u54f3\u54f4\u54f5\u54f6\u54f7\u54f8\u54f9\u54fa\u54fb\u54fc\u54fd\u54fe\u54ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_55.png", - "ascent": %ascent%, - "chars": [ - "\u5500\u5501\u5502\u5503\u5504\u5505\u5506\u5507\u5508\u5509\u550a\u550b\u550c\u550d\u550e\u550f", - "\u5510\u5511\u5512\u5513\u5514\u5515\u5516\u5517\u5518\u5519\u551a\u551b\u551c\u551d\u551e\u551f", - "\u5520\u5521\u5522\u5523\u5524\u5525\u5526\u5527\u5528\u5529\u552a\u552b\u552c\u552d\u552e\u552f", - "\u5530\u5531\u5532\u5533\u5534\u5535\u5536\u5537\u5538\u5539\u553a\u553b\u553c\u553d\u553e\u553f", - "\u5540\u5541\u5542\u5543\u5544\u5545\u5546\u5547\u5548\u5549\u554a\u554b\u554c\u554d\u554e\u554f", - "\u5550\u5551\u5552\u5553\u5554\u5555\u5556\u5557\u5558\u5559\u555a\u555b\u555c\u555d\u555e\u555f", - "\u5560\u5561\u5562\u5563\u5564\u5565\u5566\u5567\u5568\u5569\u556a\u556b\u556c\u556d\u556e\u556f", - "\u5570\u5571\u5572\u5573\u5574\u5575\u5576\u5577\u5578\u5579\u557a\u557b\u557c\u557d\u557e\u557f", - "\u5580\u5581\u5582\u5583\u5584\u5585\u5586\u5587\u5588\u5589\u558a\u558b\u558c\u558d\u558e\u558f", - "\u5590\u5591\u5592\u5593\u5594\u5595\u5596\u5597\u5598\u5599\u559a\u559b\u559c\u559d\u559e\u559f", - "\u55a0\u55a1\u55a2\u55a3\u55a4\u55a5\u55a6\u55a7\u55a8\u55a9\u55aa\u55ab\u55ac\u55ad\u55ae\u55af", - "\u55b0\u55b1\u55b2\u55b3\u55b4\u55b5\u55b6\u55b7\u55b8\u55b9\u55ba\u55bb\u55bc\u55bd\u55be\u55bf", - "\u55c0\u55c1\u55c2\u55c3\u55c4\u55c5\u55c6\u55c7\u55c8\u55c9\u55ca\u55cb\u55cc\u55cd\u55ce\u55cf", - "\u55d0\u55d1\u55d2\u55d3\u55d4\u55d5\u55d6\u55d7\u55d8\u55d9\u55da\u55db\u55dc\u55dd\u55de\u55df", - "\u55e0\u55e1\u55e2\u55e3\u55e4\u55e5\u55e6\u55e7\u55e8\u55e9\u55ea\u55eb\u55ec\u55ed\u55ee\u55ef", - "\u55f0\u55f1\u55f2\u55f3\u55f4\u55f5\u55f6\u55f7\u55f8\u55f9\u55fa\u55fb\u55fc\u55fd\u55fe\u55ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_56.png", - "ascent": %ascent%, - "chars": [ - "\u5600\u5601\u5602\u5603\u5604\u5605\u5606\u5607\u5608\u5609\u560a\u560b\u560c\u560d\u560e\u560f", - "\u5610\u5611\u5612\u5613\u5614\u5615\u5616\u5617\u5618\u5619\u561a\u561b\u561c\u561d\u561e\u561f", - "\u5620\u5621\u5622\u5623\u5624\u5625\u5626\u5627\u5628\u5629\u562a\u562b\u562c\u562d\u562e\u562f", - "\u5630\u5631\u5632\u5633\u5634\u5635\u5636\u5637\u5638\u5639\u563a\u563b\u563c\u563d\u563e\u563f", - "\u5640\u5641\u5642\u5643\u5644\u5645\u5646\u5647\u5648\u5649\u564a\u564b\u564c\u564d\u564e\u564f", - "\u5650\u5651\u5652\u5653\u5654\u5655\u5656\u5657\u5658\u5659\u565a\u565b\u565c\u565d\u565e\u565f", - "\u5660\u5661\u5662\u5663\u5664\u5665\u5666\u5667\u5668\u5669\u566a\u566b\u566c\u566d\u566e\u566f", - "\u5670\u5671\u5672\u5673\u5674\u5675\u5676\u5677\u5678\u5679\u567a\u567b\u567c\u567d\u567e\u567f", - "\u5680\u5681\u5682\u5683\u5684\u5685\u5686\u5687\u5688\u5689\u568a\u568b\u568c\u568d\u568e\u568f", - "\u5690\u5691\u5692\u5693\u5694\u5695\u5696\u5697\u5698\u5699\u569a\u569b\u569c\u569d\u569e\u569f", - "\u56a0\u56a1\u56a2\u56a3\u56a4\u56a5\u56a6\u56a7\u56a8\u56a9\u56aa\u56ab\u56ac\u56ad\u56ae\u56af", - "\u56b0\u56b1\u56b2\u56b3\u56b4\u56b5\u56b6\u56b7\u56b8\u56b9\u56ba\u56bb\u56bc\u56bd\u56be\u56bf", - "\u56c0\u56c1\u56c2\u56c3\u56c4\u56c5\u56c6\u56c7\u56c8\u56c9\u56ca\u56cb\u56cc\u56cd\u56ce\u56cf", - "\u56d0\u56d1\u56d2\u56d3\u56d4\u56d5\u56d6\u56d7\u56d8\u56d9\u56da\u56db\u56dc\u56dd\u56de\u56df", - "\u56e0\u56e1\u56e2\u56e3\u56e4\u56e5\u56e6\u56e7\u56e8\u56e9\u56ea\u56eb\u56ec\u56ed\u56ee\u56ef", - "\u56f0\u56f1\u56f2\u56f3\u56f4\u56f5\u56f6\u56f7\u56f8\u56f9\u56fa\u56fb\u56fc\u56fd\u56fe\u56ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_57.png", - "ascent": %ascent%, - "chars": [ - "\u5700\u5701\u5702\u5703\u5704\u5705\u5706\u5707\u5708\u5709\u570a\u570b\u570c\u570d\u570e\u570f", - "\u5710\u5711\u5712\u5713\u5714\u5715\u5716\u5717\u5718\u5719\u571a\u571b\u571c\u571d\u571e\u571f", - "\u5720\u5721\u5722\u5723\u5724\u5725\u5726\u5727\u5728\u5729\u572a\u572b\u572c\u572d\u572e\u572f", - "\u5730\u5731\u5732\u5733\u5734\u5735\u5736\u5737\u5738\u5739\u573a\u573b\u573c\u573d\u573e\u573f", - "\u5740\u5741\u5742\u5743\u5744\u5745\u5746\u5747\u5748\u5749\u574a\u574b\u574c\u574d\u574e\u574f", - "\u5750\u5751\u5752\u5753\u5754\u5755\u5756\u5757\u5758\u5759\u575a\u575b\u575c\u575d\u575e\u575f", - "\u5760\u5761\u5762\u5763\u5764\u5765\u5766\u5767\u5768\u5769\u576a\u576b\u576c\u576d\u576e\u576f", - "\u5770\u5771\u5772\u5773\u5774\u5775\u5776\u5777\u5778\u5779\u577a\u577b\u577c\u577d\u577e\u577f", - "\u5780\u5781\u5782\u5783\u5784\u5785\u5786\u5787\u5788\u5789\u578a\u578b\u578c\u578d\u578e\u578f", - "\u5790\u5791\u5792\u5793\u5794\u5795\u5796\u5797\u5798\u5799\u579a\u579b\u579c\u579d\u579e\u579f", - "\u57a0\u57a1\u57a2\u57a3\u57a4\u57a5\u57a6\u57a7\u57a8\u57a9\u57aa\u57ab\u57ac\u57ad\u57ae\u57af", - "\u57b0\u57b1\u57b2\u57b3\u57b4\u57b5\u57b6\u57b7\u57b8\u57b9\u57ba\u57bb\u57bc\u57bd\u57be\u57bf", - "\u57c0\u57c1\u57c2\u57c3\u57c4\u57c5\u57c6\u57c7\u57c8\u57c9\u57ca\u57cb\u57cc\u57cd\u57ce\u57cf", - "\u57d0\u57d1\u57d2\u57d3\u57d4\u57d5\u57d6\u57d7\u57d8\u57d9\u57da\u57db\u57dc\u57dd\u57de\u57df", - "\u57e0\u57e1\u57e2\u57e3\u57e4\u57e5\u57e6\u57e7\u57e8\u57e9\u57ea\u57eb\u57ec\u57ed\u57ee\u57ef", - "\u57f0\u57f1\u57f2\u57f3\u57f4\u57f5\u57f6\u57f7\u57f8\u57f9\u57fa\u57fb\u57fc\u57fd\u57fe\u57ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_58.png", - "ascent": %ascent%, - "chars": [ - "\u5800\u5801\u5802\u5803\u5804\u5805\u5806\u5807\u5808\u5809\u580a\u580b\u580c\u580d\u580e\u580f", - "\u5810\u5811\u5812\u5813\u5814\u5815\u5816\u5817\u5818\u5819\u581a\u581b\u581c\u581d\u581e\u581f", - "\u5820\u5821\u5822\u5823\u5824\u5825\u5826\u5827\u5828\u5829\u582a\u582b\u582c\u582d\u582e\u582f", - "\u5830\u5831\u5832\u5833\u5834\u5835\u5836\u5837\u5838\u5839\u583a\u583b\u583c\u583d\u583e\u583f", - "\u5840\u5841\u5842\u5843\u5844\u5845\u5846\u5847\u5848\u5849\u584a\u584b\u584c\u584d\u584e\u584f", - "\u5850\u5851\u5852\u5853\u5854\u5855\u5856\u5857\u5858\u5859\u585a\u585b\u585c\u585d\u585e\u585f", - "\u5860\u5861\u5862\u5863\u5864\u5865\u5866\u5867\u5868\u5869\u586a\u586b\u586c\u586d\u586e\u586f", - "\u5870\u5871\u5872\u5873\u5874\u5875\u5876\u5877\u5878\u5879\u587a\u587b\u587c\u587d\u587e\u587f", - "\u5880\u5881\u5882\u5883\u5884\u5885\u5886\u5887\u5888\u5889\u588a\u588b\u588c\u588d\u588e\u588f", - "\u5890\u5891\u5892\u5893\u5894\u5895\u5896\u5897\u5898\u5899\u589a\u589b\u589c\u589d\u589e\u589f", - "\u58a0\u58a1\u58a2\u58a3\u58a4\u58a5\u58a6\u58a7\u58a8\u58a9\u58aa\u58ab\u58ac\u58ad\u58ae\u58af", - "\u58b0\u58b1\u58b2\u58b3\u58b4\u58b5\u58b6\u58b7\u58b8\u58b9\u58ba\u58bb\u58bc\u58bd\u58be\u58bf", - "\u58c0\u58c1\u58c2\u58c3\u58c4\u58c5\u58c6\u58c7\u58c8\u58c9\u58ca\u58cb\u58cc\u58cd\u58ce\u58cf", - "\u58d0\u58d1\u58d2\u58d3\u58d4\u58d5\u58d6\u58d7\u58d8\u58d9\u58da\u58db\u58dc\u58dd\u58de\u58df", - "\u58e0\u58e1\u58e2\u58e3\u58e4\u58e5\u58e6\u58e7\u58e8\u58e9\u58ea\u58eb\u58ec\u58ed\u58ee\u58ef", - "\u58f0\u58f1\u58f2\u58f3\u58f4\u58f5\u58f6\u58f7\u58f8\u58f9\u58fa\u58fb\u58fc\u58fd\u58fe\u58ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_59.png", - "ascent": %ascent%, - "chars": [ - "\u5900\u5901\u5902\u5903\u5904\u5905\u5906\u5907\u5908\u5909\u590a\u590b\u590c\u590d\u590e\u590f", - "\u5910\u5911\u5912\u5913\u5914\u5915\u5916\u5917\u5918\u5919\u591a\u591b\u591c\u591d\u591e\u591f", - "\u5920\u5921\u5922\u5923\u5924\u5925\u5926\u5927\u5928\u5929\u592a\u592b\u592c\u592d\u592e\u592f", - "\u5930\u5931\u5932\u5933\u5934\u5935\u5936\u5937\u5938\u5939\u593a\u593b\u593c\u593d\u593e\u593f", - "\u5940\u5941\u5942\u5943\u5944\u5945\u5946\u5947\u5948\u5949\u594a\u594b\u594c\u594d\u594e\u594f", - "\u5950\u5951\u5952\u5953\u5954\u5955\u5956\u5957\u5958\u5959\u595a\u595b\u595c\u595d\u595e\u595f", - "\u5960\u5961\u5962\u5963\u5964\u5965\u5966\u5967\u5968\u5969\u596a\u596b\u596c\u596d\u596e\u596f", - "\u5970\u5971\u5972\u5973\u5974\u5975\u5976\u5977\u5978\u5979\u597a\u597b\u597c\u597d\u597e\u597f", - "\u5980\u5981\u5982\u5983\u5984\u5985\u5986\u5987\u5988\u5989\u598a\u598b\u598c\u598d\u598e\u598f", - "\u5990\u5991\u5992\u5993\u5994\u5995\u5996\u5997\u5998\u5999\u599a\u599b\u599c\u599d\u599e\u599f", - "\u59a0\u59a1\u59a2\u59a3\u59a4\u59a5\u59a6\u59a7\u59a8\u59a9\u59aa\u59ab\u59ac\u59ad\u59ae\u59af", - "\u59b0\u59b1\u59b2\u59b3\u59b4\u59b5\u59b6\u59b7\u59b8\u59b9\u59ba\u59bb\u59bc\u59bd\u59be\u59bf", - "\u59c0\u59c1\u59c2\u59c3\u59c4\u59c5\u59c6\u59c7\u59c8\u59c9\u59ca\u59cb\u59cc\u59cd\u59ce\u59cf", - "\u59d0\u59d1\u59d2\u59d3\u59d4\u59d5\u59d6\u59d7\u59d8\u59d9\u59da\u59db\u59dc\u59dd\u59de\u59df", - "\u59e0\u59e1\u59e2\u59e3\u59e4\u59e5\u59e6\u59e7\u59e8\u59e9\u59ea\u59eb\u59ec\u59ed\u59ee\u59ef", - "\u59f0\u59f1\u59f2\u59f3\u59f4\u59f5\u59f6\u59f7\u59f8\u59f9\u59fa\u59fb\u59fc\u59fd\u59fe\u59ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_5a.png", - "ascent": %ascent%, - "chars": [ - "\u5a00\u5a01\u5a02\u5a03\u5a04\u5a05\u5a06\u5a07\u5a08\u5a09\u5a0a\u5a0b\u5a0c\u5a0d\u5a0e\u5a0f", - "\u5a10\u5a11\u5a12\u5a13\u5a14\u5a15\u5a16\u5a17\u5a18\u5a19\u5a1a\u5a1b\u5a1c\u5a1d\u5a1e\u5a1f", - "\u5a20\u5a21\u5a22\u5a23\u5a24\u5a25\u5a26\u5a27\u5a28\u5a29\u5a2a\u5a2b\u5a2c\u5a2d\u5a2e\u5a2f", - "\u5a30\u5a31\u5a32\u5a33\u5a34\u5a35\u5a36\u5a37\u5a38\u5a39\u5a3a\u5a3b\u5a3c\u5a3d\u5a3e\u5a3f", - "\u5a40\u5a41\u5a42\u5a43\u5a44\u5a45\u5a46\u5a47\u5a48\u5a49\u5a4a\u5a4b\u5a4c\u5a4d\u5a4e\u5a4f", - "\u5a50\u5a51\u5a52\u5a53\u5a54\u5a55\u5a56\u5a57\u5a58\u5a59\u5a5a\u5a5b\u5a5c\u5a5d\u5a5e\u5a5f", - "\u5a60\u5a61\u5a62\u5a63\u5a64\u5a65\u5a66\u5a67\u5a68\u5a69\u5a6a\u5a6b\u5a6c\u5a6d\u5a6e\u5a6f", - "\u5a70\u5a71\u5a72\u5a73\u5a74\u5a75\u5a76\u5a77\u5a78\u5a79\u5a7a\u5a7b\u5a7c\u5a7d\u5a7e\u5a7f", - "\u5a80\u5a81\u5a82\u5a83\u5a84\u5a85\u5a86\u5a87\u5a88\u5a89\u5a8a\u5a8b\u5a8c\u5a8d\u5a8e\u5a8f", - "\u5a90\u5a91\u5a92\u5a93\u5a94\u5a95\u5a96\u5a97\u5a98\u5a99\u5a9a\u5a9b\u5a9c\u5a9d\u5a9e\u5a9f", - "\u5aa0\u5aa1\u5aa2\u5aa3\u5aa4\u5aa5\u5aa6\u5aa7\u5aa8\u5aa9\u5aaa\u5aab\u5aac\u5aad\u5aae\u5aaf", - "\u5ab0\u5ab1\u5ab2\u5ab3\u5ab4\u5ab5\u5ab6\u5ab7\u5ab8\u5ab9\u5aba\u5abb\u5abc\u5abd\u5abe\u5abf", - "\u5ac0\u5ac1\u5ac2\u5ac3\u5ac4\u5ac5\u5ac6\u5ac7\u5ac8\u5ac9\u5aca\u5acb\u5acc\u5acd\u5ace\u5acf", - "\u5ad0\u5ad1\u5ad2\u5ad3\u5ad4\u5ad5\u5ad6\u5ad7\u5ad8\u5ad9\u5ada\u5adb\u5adc\u5add\u5ade\u5adf", - "\u5ae0\u5ae1\u5ae2\u5ae3\u5ae4\u5ae5\u5ae6\u5ae7\u5ae8\u5ae9\u5aea\u5aeb\u5aec\u5aed\u5aee\u5aef", - "\u5af0\u5af1\u5af2\u5af3\u5af4\u5af5\u5af6\u5af7\u5af8\u5af9\u5afa\u5afb\u5afc\u5afd\u5afe\u5aff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_5b.png", - "ascent": %ascent%, - "chars": [ - "\u5b00\u5b01\u5b02\u5b03\u5b04\u5b05\u5b06\u5b07\u5b08\u5b09\u5b0a\u5b0b\u5b0c\u5b0d\u5b0e\u5b0f", - "\u5b10\u5b11\u5b12\u5b13\u5b14\u5b15\u5b16\u5b17\u5b18\u5b19\u5b1a\u5b1b\u5b1c\u5b1d\u5b1e\u5b1f", - "\u5b20\u5b21\u5b22\u5b23\u5b24\u5b25\u5b26\u5b27\u5b28\u5b29\u5b2a\u5b2b\u5b2c\u5b2d\u5b2e\u5b2f", - "\u5b30\u5b31\u5b32\u5b33\u5b34\u5b35\u5b36\u5b37\u5b38\u5b39\u5b3a\u5b3b\u5b3c\u5b3d\u5b3e\u5b3f", - "\u5b40\u5b41\u5b42\u5b43\u5b44\u5b45\u5b46\u5b47\u5b48\u5b49\u5b4a\u5b4b\u5b4c\u5b4d\u5b4e\u5b4f", - "\u5b50\u5b51\u5b52\u5b53\u5b54\u5b55\u5b56\u5b57\u5b58\u5b59\u5b5a\u5b5b\u5b5c\u5b5d\u5b5e\u5b5f", - "\u5b60\u5b61\u5b62\u5b63\u5b64\u5b65\u5b66\u5b67\u5b68\u5b69\u5b6a\u5b6b\u5b6c\u5b6d\u5b6e\u5b6f", - "\u5b70\u5b71\u5b72\u5b73\u5b74\u5b75\u5b76\u5b77\u5b78\u5b79\u5b7a\u5b7b\u5b7c\u5b7d\u5b7e\u5b7f", - "\u5b80\u5b81\u5b82\u5b83\u5b84\u5b85\u5b86\u5b87\u5b88\u5b89\u5b8a\u5b8b\u5b8c\u5b8d\u5b8e\u5b8f", - "\u5b90\u5b91\u5b92\u5b93\u5b94\u5b95\u5b96\u5b97\u5b98\u5b99\u5b9a\u5b9b\u5b9c\u5b9d\u5b9e\u5b9f", - "\u5ba0\u5ba1\u5ba2\u5ba3\u5ba4\u5ba5\u5ba6\u5ba7\u5ba8\u5ba9\u5baa\u5bab\u5bac\u5bad\u5bae\u5baf", - "\u5bb0\u5bb1\u5bb2\u5bb3\u5bb4\u5bb5\u5bb6\u5bb7\u5bb8\u5bb9\u5bba\u5bbb\u5bbc\u5bbd\u5bbe\u5bbf", - "\u5bc0\u5bc1\u5bc2\u5bc3\u5bc4\u5bc5\u5bc6\u5bc7\u5bc8\u5bc9\u5bca\u5bcb\u5bcc\u5bcd\u5bce\u5bcf", - "\u5bd0\u5bd1\u5bd2\u5bd3\u5bd4\u5bd5\u5bd6\u5bd7\u5bd8\u5bd9\u5bda\u5bdb\u5bdc\u5bdd\u5bde\u5bdf", - "\u5be0\u5be1\u5be2\u5be3\u5be4\u5be5\u5be6\u5be7\u5be8\u5be9\u5bea\u5beb\u5bec\u5bed\u5bee\u5bef", - "\u5bf0\u5bf1\u5bf2\u5bf3\u5bf4\u5bf5\u5bf6\u5bf7\u5bf8\u5bf9\u5bfa\u5bfb\u5bfc\u5bfd\u5bfe\u5bff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_5c.png", - "ascent": %ascent%, - "chars": [ - "\u5c00\u5c01\u5c02\u5c03\u5c04\u5c05\u5c06\u5c07\u5c08\u5c09\u5c0a\u5c0b\u5c0c\u5c0d\u5c0e\u5c0f", - "\u5c10\u5c11\u5c12\u5c13\u5c14\u5c15\u5c16\u5c17\u5c18\u5c19\u5c1a\u5c1b\u5c1c\u5c1d\u5c1e\u5c1f", - "\u5c20\u5c21\u5c22\u5c23\u5c24\u5c25\u5c26\u5c27\u5c28\u5c29\u5c2a\u5c2b\u5c2c\u5c2d\u5c2e\u5c2f", - "\u5c30\u5c31\u5c32\u5c33\u5c34\u5c35\u5c36\u5c37\u5c38\u5c39\u5c3a\u5c3b\u5c3c\u5c3d\u5c3e\u5c3f", - "\u5c40\u5c41\u5c42\u5c43\u5c44\u5c45\u5c46\u5c47\u5c48\u5c49\u5c4a\u5c4b\u5c4c\u5c4d\u5c4e\u5c4f", - "\u5c50\u5c51\u5c52\u5c53\u5c54\u5c55\u5c56\u5c57\u5c58\u5c59\u5c5a\u5c5b\u5c5c\u5c5d\u5c5e\u5c5f", - "\u5c60\u5c61\u5c62\u5c63\u5c64\u5c65\u5c66\u5c67\u5c68\u5c69\u5c6a\u5c6b\u5c6c\u5c6d\u5c6e\u5c6f", - "\u5c70\u5c71\u5c72\u5c73\u5c74\u5c75\u5c76\u5c77\u5c78\u5c79\u5c7a\u5c7b\u5c7c\u5c7d\u5c7e\u5c7f", - "\u5c80\u5c81\u5c82\u5c83\u5c84\u5c85\u5c86\u5c87\u5c88\u5c89\u5c8a\u5c8b\u5c8c\u5c8d\u5c8e\u5c8f", - "\u5c90\u5c91\u5c92\u5c93\u5c94\u5c95\u5c96\u5c97\u5c98\u5c99\u5c9a\u5c9b\u5c9c\u5c9d\u5c9e\u5c9f", - "\u5ca0\u5ca1\u5ca2\u5ca3\u5ca4\u5ca5\u5ca6\u5ca7\u5ca8\u5ca9\u5caa\u5cab\u5cac\u5cad\u5cae\u5caf", - "\u5cb0\u5cb1\u5cb2\u5cb3\u5cb4\u5cb5\u5cb6\u5cb7\u5cb8\u5cb9\u5cba\u5cbb\u5cbc\u5cbd\u5cbe\u5cbf", - "\u5cc0\u5cc1\u5cc2\u5cc3\u5cc4\u5cc5\u5cc6\u5cc7\u5cc8\u5cc9\u5cca\u5ccb\u5ccc\u5ccd\u5cce\u5ccf", - "\u5cd0\u5cd1\u5cd2\u5cd3\u5cd4\u5cd5\u5cd6\u5cd7\u5cd8\u5cd9\u5cda\u5cdb\u5cdc\u5cdd\u5cde\u5cdf", - "\u5ce0\u5ce1\u5ce2\u5ce3\u5ce4\u5ce5\u5ce6\u5ce7\u5ce8\u5ce9\u5cea\u5ceb\u5cec\u5ced\u5cee\u5cef", - "\u5cf0\u5cf1\u5cf2\u5cf3\u5cf4\u5cf5\u5cf6\u5cf7\u5cf8\u5cf9\u5cfa\u5cfb\u5cfc\u5cfd\u5cfe\u5cff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_5d.png", - "ascent": %ascent%, - "chars": [ - "\u5d00\u5d01\u5d02\u5d03\u5d04\u5d05\u5d06\u5d07\u5d08\u5d09\u5d0a\u5d0b\u5d0c\u5d0d\u5d0e\u5d0f", - "\u5d10\u5d11\u5d12\u5d13\u5d14\u5d15\u5d16\u5d17\u5d18\u5d19\u5d1a\u5d1b\u5d1c\u5d1d\u5d1e\u5d1f", - "\u5d20\u5d21\u5d22\u5d23\u5d24\u5d25\u5d26\u5d27\u5d28\u5d29\u5d2a\u5d2b\u5d2c\u5d2d\u5d2e\u5d2f", - "\u5d30\u5d31\u5d32\u5d33\u5d34\u5d35\u5d36\u5d37\u5d38\u5d39\u5d3a\u5d3b\u5d3c\u5d3d\u5d3e\u5d3f", - "\u5d40\u5d41\u5d42\u5d43\u5d44\u5d45\u5d46\u5d47\u5d48\u5d49\u5d4a\u5d4b\u5d4c\u5d4d\u5d4e\u5d4f", - "\u5d50\u5d51\u5d52\u5d53\u5d54\u5d55\u5d56\u5d57\u5d58\u5d59\u5d5a\u5d5b\u5d5c\u5d5d\u5d5e\u5d5f", - "\u5d60\u5d61\u5d62\u5d63\u5d64\u5d65\u5d66\u5d67\u5d68\u5d69\u5d6a\u5d6b\u5d6c\u5d6d\u5d6e\u5d6f", - "\u5d70\u5d71\u5d72\u5d73\u5d74\u5d75\u5d76\u5d77\u5d78\u5d79\u5d7a\u5d7b\u5d7c\u5d7d\u5d7e\u5d7f", - "\u5d80\u5d81\u5d82\u5d83\u5d84\u5d85\u5d86\u5d87\u5d88\u5d89\u5d8a\u5d8b\u5d8c\u5d8d\u5d8e\u5d8f", - "\u5d90\u5d91\u5d92\u5d93\u5d94\u5d95\u5d96\u5d97\u5d98\u5d99\u5d9a\u5d9b\u5d9c\u5d9d\u5d9e\u5d9f", - "\u5da0\u5da1\u5da2\u5da3\u5da4\u5da5\u5da6\u5da7\u5da8\u5da9\u5daa\u5dab\u5dac\u5dad\u5dae\u5daf", - "\u5db0\u5db1\u5db2\u5db3\u5db4\u5db5\u5db6\u5db7\u5db8\u5db9\u5dba\u5dbb\u5dbc\u5dbd\u5dbe\u5dbf", - "\u5dc0\u5dc1\u5dc2\u5dc3\u5dc4\u5dc5\u5dc6\u5dc7\u5dc8\u5dc9\u5dca\u5dcb\u5dcc\u5dcd\u5dce\u5dcf", - "\u5dd0\u5dd1\u5dd2\u5dd3\u5dd4\u5dd5\u5dd6\u5dd7\u5dd8\u5dd9\u5dda\u5ddb\u5ddc\u5ddd\u5dde\u5ddf", - "\u5de0\u5de1\u5de2\u5de3\u5de4\u5de5\u5de6\u5de7\u5de8\u5de9\u5dea\u5deb\u5dec\u5ded\u5dee\u5def", - "\u5df0\u5df1\u5df2\u5df3\u5df4\u5df5\u5df6\u5df7\u5df8\u5df9\u5dfa\u5dfb\u5dfc\u5dfd\u5dfe\u5dff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_5e.png", - "ascent": %ascent%, - "chars": [ - "\u5e00\u5e01\u5e02\u5e03\u5e04\u5e05\u5e06\u5e07\u5e08\u5e09\u5e0a\u5e0b\u5e0c\u5e0d\u5e0e\u5e0f", - "\u5e10\u5e11\u5e12\u5e13\u5e14\u5e15\u5e16\u5e17\u5e18\u5e19\u5e1a\u5e1b\u5e1c\u5e1d\u5e1e\u5e1f", - "\u5e20\u5e21\u5e22\u5e23\u5e24\u5e25\u5e26\u5e27\u5e28\u5e29\u5e2a\u5e2b\u5e2c\u5e2d\u5e2e\u5e2f", - "\u5e30\u5e31\u5e32\u5e33\u5e34\u5e35\u5e36\u5e37\u5e38\u5e39\u5e3a\u5e3b\u5e3c\u5e3d\u5e3e\u5e3f", - "\u5e40\u5e41\u5e42\u5e43\u5e44\u5e45\u5e46\u5e47\u5e48\u5e49\u5e4a\u5e4b\u5e4c\u5e4d\u5e4e\u5e4f", - "\u5e50\u5e51\u5e52\u5e53\u5e54\u5e55\u5e56\u5e57\u5e58\u5e59\u5e5a\u5e5b\u5e5c\u5e5d\u5e5e\u5e5f", - "\u5e60\u5e61\u5e62\u5e63\u5e64\u5e65\u5e66\u5e67\u5e68\u5e69\u5e6a\u5e6b\u5e6c\u5e6d\u5e6e\u5e6f", - "\u5e70\u5e71\u5e72\u5e73\u5e74\u5e75\u5e76\u5e77\u5e78\u5e79\u5e7a\u5e7b\u5e7c\u5e7d\u5e7e\u5e7f", - "\u5e80\u5e81\u5e82\u5e83\u5e84\u5e85\u5e86\u5e87\u5e88\u5e89\u5e8a\u5e8b\u5e8c\u5e8d\u5e8e\u5e8f", - "\u5e90\u5e91\u5e92\u5e93\u5e94\u5e95\u5e96\u5e97\u5e98\u5e99\u5e9a\u5e9b\u5e9c\u5e9d\u5e9e\u5e9f", - "\u5ea0\u5ea1\u5ea2\u5ea3\u5ea4\u5ea5\u5ea6\u5ea7\u5ea8\u5ea9\u5eaa\u5eab\u5eac\u5ead\u5eae\u5eaf", - "\u5eb0\u5eb1\u5eb2\u5eb3\u5eb4\u5eb5\u5eb6\u5eb7\u5eb8\u5eb9\u5eba\u5ebb\u5ebc\u5ebd\u5ebe\u5ebf", - "\u5ec0\u5ec1\u5ec2\u5ec3\u5ec4\u5ec5\u5ec6\u5ec7\u5ec8\u5ec9\u5eca\u5ecb\u5ecc\u5ecd\u5ece\u5ecf", - "\u5ed0\u5ed1\u5ed2\u5ed3\u5ed4\u5ed5\u5ed6\u5ed7\u5ed8\u5ed9\u5eda\u5edb\u5edc\u5edd\u5ede\u5edf", - "\u5ee0\u5ee1\u5ee2\u5ee3\u5ee4\u5ee5\u5ee6\u5ee7\u5ee8\u5ee9\u5eea\u5eeb\u5eec\u5eed\u5eee\u5eef", - "\u5ef0\u5ef1\u5ef2\u5ef3\u5ef4\u5ef5\u5ef6\u5ef7\u5ef8\u5ef9\u5efa\u5efb\u5efc\u5efd\u5efe\u5eff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_5f.png", - "ascent": %ascent%, - "chars": [ - "\u5f00\u5f01\u5f02\u5f03\u5f04\u5f05\u5f06\u5f07\u5f08\u5f09\u5f0a\u5f0b\u5f0c\u5f0d\u5f0e\u5f0f", - "\u5f10\u5f11\u5f12\u5f13\u5f14\u5f15\u5f16\u5f17\u5f18\u5f19\u5f1a\u5f1b\u5f1c\u5f1d\u5f1e\u5f1f", - "\u5f20\u5f21\u5f22\u5f23\u5f24\u5f25\u5f26\u5f27\u5f28\u5f29\u5f2a\u5f2b\u5f2c\u5f2d\u5f2e\u5f2f", - "\u5f30\u5f31\u5f32\u5f33\u5f34\u5f35\u5f36\u5f37\u5f38\u5f39\u5f3a\u5f3b\u5f3c\u5f3d\u5f3e\u5f3f", - "\u5f40\u5f41\u5f42\u5f43\u5f44\u5f45\u5f46\u5f47\u5f48\u5f49\u5f4a\u5f4b\u5f4c\u5f4d\u5f4e\u5f4f", - "\u5f50\u5f51\u5f52\u5f53\u5f54\u5f55\u5f56\u5f57\u5f58\u5f59\u5f5a\u5f5b\u5f5c\u5f5d\u5f5e\u5f5f", - "\u5f60\u5f61\u5f62\u5f63\u5f64\u5f65\u5f66\u5f67\u5f68\u5f69\u5f6a\u5f6b\u5f6c\u5f6d\u5f6e\u5f6f", - "\u5f70\u5f71\u5f72\u5f73\u5f74\u5f75\u5f76\u5f77\u5f78\u5f79\u5f7a\u5f7b\u5f7c\u5f7d\u5f7e\u5f7f", - "\u5f80\u5f81\u5f82\u5f83\u5f84\u5f85\u5f86\u5f87\u5f88\u5f89\u5f8a\u5f8b\u5f8c\u5f8d\u5f8e\u5f8f", - "\u5f90\u5f91\u5f92\u5f93\u5f94\u5f95\u5f96\u5f97\u5f98\u5f99\u5f9a\u5f9b\u5f9c\u5f9d\u5f9e\u5f9f", - "\u5fa0\u5fa1\u5fa2\u5fa3\u5fa4\u5fa5\u5fa6\u5fa7\u5fa8\u5fa9\u5faa\u5fab\u5fac\u5fad\u5fae\u5faf", - "\u5fb0\u5fb1\u5fb2\u5fb3\u5fb4\u5fb5\u5fb6\u5fb7\u5fb8\u5fb9\u5fba\u5fbb\u5fbc\u5fbd\u5fbe\u5fbf", - "\u5fc0\u5fc1\u5fc2\u5fc3\u5fc4\u5fc5\u5fc6\u5fc7\u5fc8\u5fc9\u5fca\u5fcb\u5fcc\u5fcd\u5fce\u5fcf", - "\u5fd0\u5fd1\u5fd2\u5fd3\u5fd4\u5fd5\u5fd6\u5fd7\u5fd8\u5fd9\u5fda\u5fdb\u5fdc\u5fdd\u5fde\u5fdf", - "\u5fe0\u5fe1\u5fe2\u5fe3\u5fe4\u5fe5\u5fe6\u5fe7\u5fe8\u5fe9\u5fea\u5feb\u5fec\u5fed\u5fee\u5fef", - "\u5ff0\u5ff1\u5ff2\u5ff3\u5ff4\u5ff5\u5ff6\u5ff7\u5ff8\u5ff9\u5ffa\u5ffb\u5ffc\u5ffd\u5ffe\u5fff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_60.png", - "ascent": %ascent%, - "chars": [ - "\u6000\u6001\u6002\u6003\u6004\u6005\u6006\u6007\u6008\u6009\u600a\u600b\u600c\u600d\u600e\u600f", - "\u6010\u6011\u6012\u6013\u6014\u6015\u6016\u6017\u6018\u6019\u601a\u601b\u601c\u601d\u601e\u601f", - "\u6020\u6021\u6022\u6023\u6024\u6025\u6026\u6027\u6028\u6029\u602a\u602b\u602c\u602d\u602e\u602f", - "\u6030\u6031\u6032\u6033\u6034\u6035\u6036\u6037\u6038\u6039\u603a\u603b\u603c\u603d\u603e\u603f", - "\u6040\u6041\u6042\u6043\u6044\u6045\u6046\u6047\u6048\u6049\u604a\u604b\u604c\u604d\u604e\u604f", - "\u6050\u6051\u6052\u6053\u6054\u6055\u6056\u6057\u6058\u6059\u605a\u605b\u605c\u605d\u605e\u605f", - "\u6060\u6061\u6062\u6063\u6064\u6065\u6066\u6067\u6068\u6069\u606a\u606b\u606c\u606d\u606e\u606f", - "\u6070\u6071\u6072\u6073\u6074\u6075\u6076\u6077\u6078\u6079\u607a\u607b\u607c\u607d\u607e\u607f", - "\u6080\u6081\u6082\u6083\u6084\u6085\u6086\u6087\u6088\u6089\u608a\u608b\u608c\u608d\u608e\u608f", - "\u6090\u6091\u6092\u6093\u6094\u6095\u6096\u6097\u6098\u6099\u609a\u609b\u609c\u609d\u609e\u609f", - "\u60a0\u60a1\u60a2\u60a3\u60a4\u60a5\u60a6\u60a7\u60a8\u60a9\u60aa\u60ab\u60ac\u60ad\u60ae\u60af", - "\u60b0\u60b1\u60b2\u60b3\u60b4\u60b5\u60b6\u60b7\u60b8\u60b9\u60ba\u60bb\u60bc\u60bd\u60be\u60bf", - "\u60c0\u60c1\u60c2\u60c3\u60c4\u60c5\u60c6\u60c7\u60c8\u60c9\u60ca\u60cb\u60cc\u60cd\u60ce\u60cf", - "\u60d0\u60d1\u60d2\u60d3\u60d4\u60d5\u60d6\u60d7\u60d8\u60d9\u60da\u60db\u60dc\u60dd\u60de\u60df", - "\u60e0\u60e1\u60e2\u60e3\u60e4\u60e5\u60e6\u60e7\u60e8\u60e9\u60ea\u60eb\u60ec\u60ed\u60ee\u60ef", - "\u60f0\u60f1\u60f2\u60f3\u60f4\u60f5\u60f6\u60f7\u60f8\u60f9\u60fa\u60fb\u60fc\u60fd\u60fe\u60ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_61.png", - "ascent": %ascent%, - "chars": [ - "\u6100\u6101\u6102\u6103\u6104\u6105\u6106\u6107\u6108\u6109\u610a\u610b\u610c\u610d\u610e\u610f", - "\u6110\u6111\u6112\u6113\u6114\u6115\u6116\u6117\u6118\u6119\u611a\u611b\u611c\u611d\u611e\u611f", - "\u6120\u6121\u6122\u6123\u6124\u6125\u6126\u6127\u6128\u6129\u612a\u612b\u612c\u612d\u612e\u612f", - "\u6130\u6131\u6132\u6133\u6134\u6135\u6136\u6137\u6138\u6139\u613a\u613b\u613c\u613d\u613e\u613f", - "\u6140\u6141\u6142\u6143\u6144\u6145\u6146\u6147\u6148\u6149\u614a\u614b\u614c\u614d\u614e\u614f", - "\u6150\u6151\u6152\u6153\u6154\u6155\u6156\u6157\u6158\u6159\u615a\u615b\u615c\u615d\u615e\u615f", - "\u6160\u6161\u6162\u6163\u6164\u6165\u6166\u6167\u6168\u6169\u616a\u616b\u616c\u616d\u616e\u616f", - "\u6170\u6171\u6172\u6173\u6174\u6175\u6176\u6177\u6178\u6179\u617a\u617b\u617c\u617d\u617e\u617f", - "\u6180\u6181\u6182\u6183\u6184\u6185\u6186\u6187\u6188\u6189\u618a\u618b\u618c\u618d\u618e\u618f", - "\u6190\u6191\u6192\u6193\u6194\u6195\u6196\u6197\u6198\u6199\u619a\u619b\u619c\u619d\u619e\u619f", - "\u61a0\u61a1\u61a2\u61a3\u61a4\u61a5\u61a6\u61a7\u61a8\u61a9\u61aa\u61ab\u61ac\u61ad\u61ae\u61af", - "\u61b0\u61b1\u61b2\u61b3\u61b4\u61b5\u61b6\u61b7\u61b8\u61b9\u61ba\u61bb\u61bc\u61bd\u61be\u61bf", - "\u61c0\u61c1\u61c2\u61c3\u61c4\u61c5\u61c6\u61c7\u61c8\u61c9\u61ca\u61cb\u61cc\u61cd\u61ce\u61cf", - "\u61d0\u61d1\u61d2\u61d3\u61d4\u61d5\u61d6\u61d7\u61d8\u61d9\u61da\u61db\u61dc\u61dd\u61de\u61df", - "\u61e0\u61e1\u61e2\u61e3\u61e4\u61e5\u61e6\u61e7\u61e8\u61e9\u61ea\u61eb\u61ec\u61ed\u61ee\u61ef", - "\u61f0\u61f1\u61f2\u61f3\u61f4\u61f5\u61f6\u61f7\u61f8\u61f9\u61fa\u61fb\u61fc\u61fd\u61fe\u61ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_62.png", - "ascent": %ascent%, - "chars": [ - "\u6200\u6201\u6202\u6203\u6204\u6205\u6206\u6207\u6208\u6209\u620a\u620b\u620c\u620d\u620e\u620f", - "\u6210\u6211\u6212\u6213\u6214\u6215\u6216\u6217\u6218\u6219\u621a\u621b\u621c\u621d\u621e\u621f", - "\u6220\u6221\u6222\u6223\u6224\u6225\u6226\u6227\u6228\u6229\u622a\u622b\u622c\u622d\u622e\u622f", - "\u6230\u6231\u6232\u6233\u6234\u6235\u6236\u6237\u6238\u6239\u623a\u623b\u623c\u623d\u623e\u623f", - "\u6240\u6241\u6242\u6243\u6244\u6245\u6246\u6247\u6248\u6249\u624a\u624b\u624c\u624d\u624e\u624f", - "\u6250\u6251\u6252\u6253\u6254\u6255\u6256\u6257\u6258\u6259\u625a\u625b\u625c\u625d\u625e\u625f", - "\u6260\u6261\u6262\u6263\u6264\u6265\u6266\u6267\u6268\u6269\u626a\u626b\u626c\u626d\u626e\u626f", - "\u6270\u6271\u6272\u6273\u6274\u6275\u6276\u6277\u6278\u6279\u627a\u627b\u627c\u627d\u627e\u627f", - "\u6280\u6281\u6282\u6283\u6284\u6285\u6286\u6287\u6288\u6289\u628a\u628b\u628c\u628d\u628e\u628f", - "\u6290\u6291\u6292\u6293\u6294\u6295\u6296\u6297\u6298\u6299\u629a\u629b\u629c\u629d\u629e\u629f", - "\u62a0\u62a1\u62a2\u62a3\u62a4\u62a5\u62a6\u62a7\u62a8\u62a9\u62aa\u62ab\u62ac\u62ad\u62ae\u62af", - "\u62b0\u62b1\u62b2\u62b3\u62b4\u62b5\u62b6\u62b7\u62b8\u62b9\u62ba\u62bb\u62bc\u62bd\u62be\u62bf", - "\u62c0\u62c1\u62c2\u62c3\u62c4\u62c5\u62c6\u62c7\u62c8\u62c9\u62ca\u62cb\u62cc\u62cd\u62ce\u62cf", - "\u62d0\u62d1\u62d2\u62d3\u62d4\u62d5\u62d6\u62d7\u62d8\u62d9\u62da\u62db\u62dc\u62dd\u62de\u62df", - "\u62e0\u62e1\u62e2\u62e3\u62e4\u62e5\u62e6\u62e7\u62e8\u62e9\u62ea\u62eb\u62ec\u62ed\u62ee\u62ef", - "\u62f0\u62f1\u62f2\u62f3\u62f4\u62f5\u62f6\u62f7\u62f8\u62f9\u62fa\u62fb\u62fc\u62fd\u62fe\u62ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_63.png", - "ascent": %ascent%, - "chars": [ - "\u6300\u6301\u6302\u6303\u6304\u6305\u6306\u6307\u6308\u6309\u630a\u630b\u630c\u630d\u630e\u630f", - "\u6310\u6311\u6312\u6313\u6314\u6315\u6316\u6317\u6318\u6319\u631a\u631b\u631c\u631d\u631e\u631f", - "\u6320\u6321\u6322\u6323\u6324\u6325\u6326\u6327\u6328\u6329\u632a\u632b\u632c\u632d\u632e\u632f", - "\u6330\u6331\u6332\u6333\u6334\u6335\u6336\u6337\u6338\u6339\u633a\u633b\u633c\u633d\u633e\u633f", - "\u6340\u6341\u6342\u6343\u6344\u6345\u6346\u6347\u6348\u6349\u634a\u634b\u634c\u634d\u634e\u634f", - "\u6350\u6351\u6352\u6353\u6354\u6355\u6356\u6357\u6358\u6359\u635a\u635b\u635c\u635d\u635e\u635f", - "\u6360\u6361\u6362\u6363\u6364\u6365\u6366\u6367\u6368\u6369\u636a\u636b\u636c\u636d\u636e\u636f", - "\u6370\u6371\u6372\u6373\u6374\u6375\u6376\u6377\u6378\u6379\u637a\u637b\u637c\u637d\u637e\u637f", - "\u6380\u6381\u6382\u6383\u6384\u6385\u6386\u6387\u6388\u6389\u638a\u638b\u638c\u638d\u638e\u638f", - "\u6390\u6391\u6392\u6393\u6394\u6395\u6396\u6397\u6398\u6399\u639a\u639b\u639c\u639d\u639e\u639f", - "\u63a0\u63a1\u63a2\u63a3\u63a4\u63a5\u63a6\u63a7\u63a8\u63a9\u63aa\u63ab\u63ac\u63ad\u63ae\u63af", - "\u63b0\u63b1\u63b2\u63b3\u63b4\u63b5\u63b6\u63b7\u63b8\u63b9\u63ba\u63bb\u63bc\u63bd\u63be\u63bf", - "\u63c0\u63c1\u63c2\u63c3\u63c4\u63c5\u63c6\u63c7\u63c8\u63c9\u63ca\u63cb\u63cc\u63cd\u63ce\u63cf", - "\u63d0\u63d1\u63d2\u63d3\u63d4\u63d5\u63d6\u63d7\u63d8\u63d9\u63da\u63db\u63dc\u63dd\u63de\u63df", - "\u63e0\u63e1\u63e2\u63e3\u63e4\u63e5\u63e6\u63e7\u63e8\u63e9\u63ea\u63eb\u63ec\u63ed\u63ee\u63ef", - "\u63f0\u63f1\u63f2\u63f3\u63f4\u63f5\u63f6\u63f7\u63f8\u63f9\u63fa\u63fb\u63fc\u63fd\u63fe\u63ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_64.png", - "ascent": %ascent%, - "chars": [ - "\u6400\u6401\u6402\u6403\u6404\u6405\u6406\u6407\u6408\u6409\u640a\u640b\u640c\u640d\u640e\u640f", - "\u6410\u6411\u6412\u6413\u6414\u6415\u6416\u6417\u6418\u6419\u641a\u641b\u641c\u641d\u641e\u641f", - "\u6420\u6421\u6422\u6423\u6424\u6425\u6426\u6427\u6428\u6429\u642a\u642b\u642c\u642d\u642e\u642f", - "\u6430\u6431\u6432\u6433\u6434\u6435\u6436\u6437\u6438\u6439\u643a\u643b\u643c\u643d\u643e\u643f", - "\u6440\u6441\u6442\u6443\u6444\u6445\u6446\u6447\u6448\u6449\u644a\u644b\u644c\u644d\u644e\u644f", - "\u6450\u6451\u6452\u6453\u6454\u6455\u6456\u6457\u6458\u6459\u645a\u645b\u645c\u645d\u645e\u645f", - "\u6460\u6461\u6462\u6463\u6464\u6465\u6466\u6467\u6468\u6469\u646a\u646b\u646c\u646d\u646e\u646f", - "\u6470\u6471\u6472\u6473\u6474\u6475\u6476\u6477\u6478\u6479\u647a\u647b\u647c\u647d\u647e\u647f", - "\u6480\u6481\u6482\u6483\u6484\u6485\u6486\u6487\u6488\u6489\u648a\u648b\u648c\u648d\u648e\u648f", - "\u6490\u6491\u6492\u6493\u6494\u6495\u6496\u6497\u6498\u6499\u649a\u649b\u649c\u649d\u649e\u649f", - "\u64a0\u64a1\u64a2\u64a3\u64a4\u64a5\u64a6\u64a7\u64a8\u64a9\u64aa\u64ab\u64ac\u64ad\u64ae\u64af", - "\u64b0\u64b1\u64b2\u64b3\u64b4\u64b5\u64b6\u64b7\u64b8\u64b9\u64ba\u64bb\u64bc\u64bd\u64be\u64bf", - "\u64c0\u64c1\u64c2\u64c3\u64c4\u64c5\u64c6\u64c7\u64c8\u64c9\u64ca\u64cb\u64cc\u64cd\u64ce\u64cf", - "\u64d0\u64d1\u64d2\u64d3\u64d4\u64d5\u64d6\u64d7\u64d8\u64d9\u64da\u64db\u64dc\u64dd\u64de\u64df", - "\u64e0\u64e1\u64e2\u64e3\u64e4\u64e5\u64e6\u64e7\u64e8\u64e9\u64ea\u64eb\u64ec\u64ed\u64ee\u64ef", - "\u64f0\u64f1\u64f2\u64f3\u64f4\u64f5\u64f6\u64f7\u64f8\u64f9\u64fa\u64fb\u64fc\u64fd\u64fe\u64ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_65.png", - "ascent": %ascent%, - "chars": [ - "\u6500\u6501\u6502\u6503\u6504\u6505\u6506\u6507\u6508\u6509\u650a\u650b\u650c\u650d\u650e\u650f", - "\u6510\u6511\u6512\u6513\u6514\u6515\u6516\u6517\u6518\u6519\u651a\u651b\u651c\u651d\u651e\u651f", - "\u6520\u6521\u6522\u6523\u6524\u6525\u6526\u6527\u6528\u6529\u652a\u652b\u652c\u652d\u652e\u652f", - "\u6530\u6531\u6532\u6533\u6534\u6535\u6536\u6537\u6538\u6539\u653a\u653b\u653c\u653d\u653e\u653f", - "\u6540\u6541\u6542\u6543\u6544\u6545\u6546\u6547\u6548\u6549\u654a\u654b\u654c\u654d\u654e\u654f", - "\u6550\u6551\u6552\u6553\u6554\u6555\u6556\u6557\u6558\u6559\u655a\u655b\u655c\u655d\u655e\u655f", - "\u6560\u6561\u6562\u6563\u6564\u6565\u6566\u6567\u6568\u6569\u656a\u656b\u656c\u656d\u656e\u656f", - "\u6570\u6571\u6572\u6573\u6574\u6575\u6576\u6577\u6578\u6579\u657a\u657b\u657c\u657d\u657e\u657f", - "\u6580\u6581\u6582\u6583\u6584\u6585\u6586\u6587\u6588\u6589\u658a\u658b\u658c\u658d\u658e\u658f", - "\u6590\u6591\u6592\u6593\u6594\u6595\u6596\u6597\u6598\u6599\u659a\u659b\u659c\u659d\u659e\u659f", - "\u65a0\u65a1\u65a2\u65a3\u65a4\u65a5\u65a6\u65a7\u65a8\u65a9\u65aa\u65ab\u65ac\u65ad\u65ae\u65af", - "\u65b0\u65b1\u65b2\u65b3\u65b4\u65b5\u65b6\u65b7\u65b8\u65b9\u65ba\u65bb\u65bc\u65bd\u65be\u65bf", - "\u65c0\u65c1\u65c2\u65c3\u65c4\u65c5\u65c6\u65c7\u65c8\u65c9\u65ca\u65cb\u65cc\u65cd\u65ce\u65cf", - "\u65d0\u65d1\u65d2\u65d3\u65d4\u65d5\u65d6\u65d7\u65d8\u65d9\u65da\u65db\u65dc\u65dd\u65de\u65df", - "\u65e0\u65e1\u65e2\u65e3\u65e4\u65e5\u65e6\u65e7\u65e8\u65e9\u65ea\u65eb\u65ec\u65ed\u65ee\u65ef", - "\u65f0\u65f1\u65f2\u65f3\u65f4\u65f5\u65f6\u65f7\u65f8\u65f9\u65fa\u65fb\u65fc\u65fd\u65fe\u65ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_66.png", - "ascent": %ascent%, - "chars": [ - "\u6600\u6601\u6602\u6603\u6604\u6605\u6606\u6607\u6608\u6609\u660a\u660b\u660c\u660d\u660e\u660f", - "\u6610\u6611\u6612\u6613\u6614\u6615\u6616\u6617\u6618\u6619\u661a\u661b\u661c\u661d\u661e\u661f", - "\u6620\u6621\u6622\u6623\u6624\u6625\u6626\u6627\u6628\u6629\u662a\u662b\u662c\u662d\u662e\u662f", - "\u6630\u6631\u6632\u6633\u6634\u6635\u6636\u6637\u6638\u6639\u663a\u663b\u663c\u663d\u663e\u663f", - "\u6640\u6641\u6642\u6643\u6644\u6645\u6646\u6647\u6648\u6649\u664a\u664b\u664c\u664d\u664e\u664f", - "\u6650\u6651\u6652\u6653\u6654\u6655\u6656\u6657\u6658\u6659\u665a\u665b\u665c\u665d\u665e\u665f", - "\u6660\u6661\u6662\u6663\u6664\u6665\u6666\u6667\u6668\u6669\u666a\u666b\u666c\u666d\u666e\u666f", - "\u6670\u6671\u6672\u6673\u6674\u6675\u6676\u6677\u6678\u6679\u667a\u667b\u667c\u667d\u667e\u667f", - "\u6680\u6681\u6682\u6683\u6684\u6685\u6686\u6687\u6688\u6689\u668a\u668b\u668c\u668d\u668e\u668f", - "\u6690\u6691\u6692\u6693\u6694\u6695\u6696\u6697\u6698\u6699\u669a\u669b\u669c\u669d\u669e\u669f", - "\u66a0\u66a1\u66a2\u66a3\u66a4\u66a5\u66a6\u66a7\u66a8\u66a9\u66aa\u66ab\u66ac\u66ad\u66ae\u66af", - "\u66b0\u66b1\u66b2\u66b3\u66b4\u66b5\u66b6\u66b7\u66b8\u66b9\u66ba\u66bb\u66bc\u66bd\u66be\u66bf", - "\u66c0\u66c1\u66c2\u66c3\u66c4\u66c5\u66c6\u66c7\u66c8\u66c9\u66ca\u66cb\u66cc\u66cd\u66ce\u66cf", - "\u66d0\u66d1\u66d2\u66d3\u66d4\u66d5\u66d6\u66d7\u66d8\u66d9\u66da\u66db\u66dc\u66dd\u66de\u66df", - "\u66e0\u66e1\u66e2\u66e3\u66e4\u66e5\u66e6\u66e7\u66e8\u66e9\u66ea\u66eb\u66ec\u66ed\u66ee\u66ef", - "\u66f0\u66f1\u66f2\u66f3\u66f4\u66f5\u66f6\u66f7\u66f8\u66f9\u66fa\u66fb\u66fc\u66fd\u66fe\u66ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_67.png", - "ascent": %ascent%, - "chars": [ - "\u6700\u6701\u6702\u6703\u6704\u6705\u6706\u6707\u6708\u6709\u670a\u670b\u670c\u670d\u670e\u670f", - "\u6710\u6711\u6712\u6713\u6714\u6715\u6716\u6717\u6718\u6719\u671a\u671b\u671c\u671d\u671e\u671f", - "\u6720\u6721\u6722\u6723\u6724\u6725\u6726\u6727\u6728\u6729\u672a\u672b\u672c\u672d\u672e\u672f", - "\u6730\u6731\u6732\u6733\u6734\u6735\u6736\u6737\u6738\u6739\u673a\u673b\u673c\u673d\u673e\u673f", - "\u6740\u6741\u6742\u6743\u6744\u6745\u6746\u6747\u6748\u6749\u674a\u674b\u674c\u674d\u674e\u674f", - "\u6750\u6751\u6752\u6753\u6754\u6755\u6756\u6757\u6758\u6759\u675a\u675b\u675c\u675d\u675e\u675f", - "\u6760\u6761\u6762\u6763\u6764\u6765\u6766\u6767\u6768\u6769\u676a\u676b\u676c\u676d\u676e\u676f", - "\u6770\u6771\u6772\u6773\u6774\u6775\u6776\u6777\u6778\u6779\u677a\u677b\u677c\u677d\u677e\u677f", - "\u6780\u6781\u6782\u6783\u6784\u6785\u6786\u6787\u6788\u6789\u678a\u678b\u678c\u678d\u678e\u678f", - "\u6790\u6791\u6792\u6793\u6794\u6795\u6796\u6797\u6798\u6799\u679a\u679b\u679c\u679d\u679e\u679f", - "\u67a0\u67a1\u67a2\u67a3\u67a4\u67a5\u67a6\u67a7\u67a8\u67a9\u67aa\u67ab\u67ac\u67ad\u67ae\u67af", - "\u67b0\u67b1\u67b2\u67b3\u67b4\u67b5\u67b6\u67b7\u67b8\u67b9\u67ba\u67bb\u67bc\u67bd\u67be\u67bf", - "\u67c0\u67c1\u67c2\u67c3\u67c4\u67c5\u67c6\u67c7\u67c8\u67c9\u67ca\u67cb\u67cc\u67cd\u67ce\u67cf", - "\u67d0\u67d1\u67d2\u67d3\u67d4\u67d5\u67d6\u67d7\u67d8\u67d9\u67da\u67db\u67dc\u67dd\u67de\u67df", - "\u67e0\u67e1\u67e2\u67e3\u67e4\u67e5\u67e6\u67e7\u67e8\u67e9\u67ea\u67eb\u67ec\u67ed\u67ee\u67ef", - "\u67f0\u67f1\u67f2\u67f3\u67f4\u67f5\u67f6\u67f7\u67f8\u67f9\u67fa\u67fb\u67fc\u67fd\u67fe\u67ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_68.png", - "ascent": %ascent%, - "chars": [ - "\u6800\u6801\u6802\u6803\u6804\u6805\u6806\u6807\u6808\u6809\u680a\u680b\u680c\u680d\u680e\u680f", - "\u6810\u6811\u6812\u6813\u6814\u6815\u6816\u6817\u6818\u6819\u681a\u681b\u681c\u681d\u681e\u681f", - "\u6820\u6821\u6822\u6823\u6824\u6825\u6826\u6827\u6828\u6829\u682a\u682b\u682c\u682d\u682e\u682f", - "\u6830\u6831\u6832\u6833\u6834\u6835\u6836\u6837\u6838\u6839\u683a\u683b\u683c\u683d\u683e\u683f", - "\u6840\u6841\u6842\u6843\u6844\u6845\u6846\u6847\u6848\u6849\u684a\u684b\u684c\u684d\u684e\u684f", - "\u6850\u6851\u6852\u6853\u6854\u6855\u6856\u6857\u6858\u6859\u685a\u685b\u685c\u685d\u685e\u685f", - "\u6860\u6861\u6862\u6863\u6864\u6865\u6866\u6867\u6868\u6869\u686a\u686b\u686c\u686d\u686e\u686f", - "\u6870\u6871\u6872\u6873\u6874\u6875\u6876\u6877\u6878\u6879\u687a\u687b\u687c\u687d\u687e\u687f", - "\u6880\u6881\u6882\u6883\u6884\u6885\u6886\u6887\u6888\u6889\u688a\u688b\u688c\u688d\u688e\u688f", - "\u6890\u6891\u6892\u6893\u6894\u6895\u6896\u6897\u6898\u6899\u689a\u689b\u689c\u689d\u689e\u689f", - "\u68a0\u68a1\u68a2\u68a3\u68a4\u68a5\u68a6\u68a7\u68a8\u68a9\u68aa\u68ab\u68ac\u68ad\u68ae\u68af", - "\u68b0\u68b1\u68b2\u68b3\u68b4\u68b5\u68b6\u68b7\u68b8\u68b9\u68ba\u68bb\u68bc\u68bd\u68be\u68bf", - "\u68c0\u68c1\u68c2\u68c3\u68c4\u68c5\u68c6\u68c7\u68c8\u68c9\u68ca\u68cb\u68cc\u68cd\u68ce\u68cf", - "\u68d0\u68d1\u68d2\u68d3\u68d4\u68d5\u68d6\u68d7\u68d8\u68d9\u68da\u68db\u68dc\u68dd\u68de\u68df", - "\u68e0\u68e1\u68e2\u68e3\u68e4\u68e5\u68e6\u68e7\u68e8\u68e9\u68ea\u68eb\u68ec\u68ed\u68ee\u68ef", - "\u68f0\u68f1\u68f2\u68f3\u68f4\u68f5\u68f6\u68f7\u68f8\u68f9\u68fa\u68fb\u68fc\u68fd\u68fe\u68ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_69.png", - "ascent": %ascent%, - "chars": [ - "\u6900\u6901\u6902\u6903\u6904\u6905\u6906\u6907\u6908\u6909\u690a\u690b\u690c\u690d\u690e\u690f", - "\u6910\u6911\u6912\u6913\u6914\u6915\u6916\u6917\u6918\u6919\u691a\u691b\u691c\u691d\u691e\u691f", - "\u6920\u6921\u6922\u6923\u6924\u6925\u6926\u6927\u6928\u6929\u692a\u692b\u692c\u692d\u692e\u692f", - "\u6930\u6931\u6932\u6933\u6934\u6935\u6936\u6937\u6938\u6939\u693a\u693b\u693c\u693d\u693e\u693f", - "\u6940\u6941\u6942\u6943\u6944\u6945\u6946\u6947\u6948\u6949\u694a\u694b\u694c\u694d\u694e\u694f", - "\u6950\u6951\u6952\u6953\u6954\u6955\u6956\u6957\u6958\u6959\u695a\u695b\u695c\u695d\u695e\u695f", - "\u6960\u6961\u6962\u6963\u6964\u6965\u6966\u6967\u6968\u6969\u696a\u696b\u696c\u696d\u696e\u696f", - "\u6970\u6971\u6972\u6973\u6974\u6975\u6976\u6977\u6978\u6979\u697a\u697b\u697c\u697d\u697e\u697f", - "\u6980\u6981\u6982\u6983\u6984\u6985\u6986\u6987\u6988\u6989\u698a\u698b\u698c\u698d\u698e\u698f", - "\u6990\u6991\u6992\u6993\u6994\u6995\u6996\u6997\u6998\u6999\u699a\u699b\u699c\u699d\u699e\u699f", - "\u69a0\u69a1\u69a2\u69a3\u69a4\u69a5\u69a6\u69a7\u69a8\u69a9\u69aa\u69ab\u69ac\u69ad\u69ae\u69af", - "\u69b0\u69b1\u69b2\u69b3\u69b4\u69b5\u69b6\u69b7\u69b8\u69b9\u69ba\u69bb\u69bc\u69bd\u69be\u69bf", - "\u69c0\u69c1\u69c2\u69c3\u69c4\u69c5\u69c6\u69c7\u69c8\u69c9\u69ca\u69cb\u69cc\u69cd\u69ce\u69cf", - "\u69d0\u69d1\u69d2\u69d3\u69d4\u69d5\u69d6\u69d7\u69d8\u69d9\u69da\u69db\u69dc\u69dd\u69de\u69df", - "\u69e0\u69e1\u69e2\u69e3\u69e4\u69e5\u69e6\u69e7\u69e8\u69e9\u69ea\u69eb\u69ec\u69ed\u69ee\u69ef", - "\u69f0\u69f1\u69f2\u69f3\u69f4\u69f5\u69f6\u69f7\u69f8\u69f9\u69fa\u69fb\u69fc\u69fd\u69fe\u69ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_6a.png", - "ascent": %ascent%, - "chars": [ - "\u6a00\u6a01\u6a02\u6a03\u6a04\u6a05\u6a06\u6a07\u6a08\u6a09\u6a0a\u6a0b\u6a0c\u6a0d\u6a0e\u6a0f", - "\u6a10\u6a11\u6a12\u6a13\u6a14\u6a15\u6a16\u6a17\u6a18\u6a19\u6a1a\u6a1b\u6a1c\u6a1d\u6a1e\u6a1f", - "\u6a20\u6a21\u6a22\u6a23\u6a24\u6a25\u6a26\u6a27\u6a28\u6a29\u6a2a\u6a2b\u6a2c\u6a2d\u6a2e\u6a2f", - "\u6a30\u6a31\u6a32\u6a33\u6a34\u6a35\u6a36\u6a37\u6a38\u6a39\u6a3a\u6a3b\u6a3c\u6a3d\u6a3e\u6a3f", - "\u6a40\u6a41\u6a42\u6a43\u6a44\u6a45\u6a46\u6a47\u6a48\u6a49\u6a4a\u6a4b\u6a4c\u6a4d\u6a4e\u6a4f", - "\u6a50\u6a51\u6a52\u6a53\u6a54\u6a55\u6a56\u6a57\u6a58\u6a59\u6a5a\u6a5b\u6a5c\u6a5d\u6a5e\u6a5f", - "\u6a60\u6a61\u6a62\u6a63\u6a64\u6a65\u6a66\u6a67\u6a68\u6a69\u6a6a\u6a6b\u6a6c\u6a6d\u6a6e\u6a6f", - "\u6a70\u6a71\u6a72\u6a73\u6a74\u6a75\u6a76\u6a77\u6a78\u6a79\u6a7a\u6a7b\u6a7c\u6a7d\u6a7e\u6a7f", - "\u6a80\u6a81\u6a82\u6a83\u6a84\u6a85\u6a86\u6a87\u6a88\u6a89\u6a8a\u6a8b\u6a8c\u6a8d\u6a8e\u6a8f", - "\u6a90\u6a91\u6a92\u6a93\u6a94\u6a95\u6a96\u6a97\u6a98\u6a99\u6a9a\u6a9b\u6a9c\u6a9d\u6a9e\u6a9f", - "\u6aa0\u6aa1\u6aa2\u6aa3\u6aa4\u6aa5\u6aa6\u6aa7\u6aa8\u6aa9\u6aaa\u6aab\u6aac\u6aad\u6aae\u6aaf", - "\u6ab0\u6ab1\u6ab2\u6ab3\u6ab4\u6ab5\u6ab6\u6ab7\u6ab8\u6ab9\u6aba\u6abb\u6abc\u6abd\u6abe\u6abf", - "\u6ac0\u6ac1\u6ac2\u6ac3\u6ac4\u6ac5\u6ac6\u6ac7\u6ac8\u6ac9\u6aca\u6acb\u6acc\u6acd\u6ace\u6acf", - "\u6ad0\u6ad1\u6ad2\u6ad3\u6ad4\u6ad5\u6ad6\u6ad7\u6ad8\u6ad9\u6ada\u6adb\u6adc\u6add\u6ade\u6adf", - "\u6ae0\u6ae1\u6ae2\u6ae3\u6ae4\u6ae5\u6ae6\u6ae7\u6ae8\u6ae9\u6aea\u6aeb\u6aec\u6aed\u6aee\u6aef", - "\u6af0\u6af1\u6af2\u6af3\u6af4\u6af5\u6af6\u6af7\u6af8\u6af9\u6afa\u6afb\u6afc\u6afd\u6afe\u6aff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_6b.png", - "ascent": %ascent%, - "chars": [ - "\u6b00\u6b01\u6b02\u6b03\u6b04\u6b05\u6b06\u6b07\u6b08\u6b09\u6b0a\u6b0b\u6b0c\u6b0d\u6b0e\u6b0f", - "\u6b10\u6b11\u6b12\u6b13\u6b14\u6b15\u6b16\u6b17\u6b18\u6b19\u6b1a\u6b1b\u6b1c\u6b1d\u6b1e\u6b1f", - "\u6b20\u6b21\u6b22\u6b23\u6b24\u6b25\u6b26\u6b27\u6b28\u6b29\u6b2a\u6b2b\u6b2c\u6b2d\u6b2e\u6b2f", - "\u6b30\u6b31\u6b32\u6b33\u6b34\u6b35\u6b36\u6b37\u6b38\u6b39\u6b3a\u6b3b\u6b3c\u6b3d\u6b3e\u6b3f", - "\u6b40\u6b41\u6b42\u6b43\u6b44\u6b45\u6b46\u6b47\u6b48\u6b49\u6b4a\u6b4b\u6b4c\u6b4d\u6b4e\u6b4f", - "\u6b50\u6b51\u6b52\u6b53\u6b54\u6b55\u6b56\u6b57\u6b58\u6b59\u6b5a\u6b5b\u6b5c\u6b5d\u6b5e\u6b5f", - "\u6b60\u6b61\u6b62\u6b63\u6b64\u6b65\u6b66\u6b67\u6b68\u6b69\u6b6a\u6b6b\u6b6c\u6b6d\u6b6e\u6b6f", - "\u6b70\u6b71\u6b72\u6b73\u6b74\u6b75\u6b76\u6b77\u6b78\u6b79\u6b7a\u6b7b\u6b7c\u6b7d\u6b7e\u6b7f", - "\u6b80\u6b81\u6b82\u6b83\u6b84\u6b85\u6b86\u6b87\u6b88\u6b89\u6b8a\u6b8b\u6b8c\u6b8d\u6b8e\u6b8f", - "\u6b90\u6b91\u6b92\u6b93\u6b94\u6b95\u6b96\u6b97\u6b98\u6b99\u6b9a\u6b9b\u6b9c\u6b9d\u6b9e\u6b9f", - "\u6ba0\u6ba1\u6ba2\u6ba3\u6ba4\u6ba5\u6ba6\u6ba7\u6ba8\u6ba9\u6baa\u6bab\u6bac\u6bad\u6bae\u6baf", - "\u6bb0\u6bb1\u6bb2\u6bb3\u6bb4\u6bb5\u6bb6\u6bb7\u6bb8\u6bb9\u6bba\u6bbb\u6bbc\u6bbd\u6bbe\u6bbf", - "\u6bc0\u6bc1\u6bc2\u6bc3\u6bc4\u6bc5\u6bc6\u6bc7\u6bc8\u6bc9\u6bca\u6bcb\u6bcc\u6bcd\u6bce\u6bcf", - "\u6bd0\u6bd1\u6bd2\u6bd3\u6bd4\u6bd5\u6bd6\u6bd7\u6bd8\u6bd9\u6bda\u6bdb\u6bdc\u6bdd\u6bde\u6bdf", - "\u6be0\u6be1\u6be2\u6be3\u6be4\u6be5\u6be6\u6be7\u6be8\u6be9\u6bea\u6beb\u6bec\u6bed\u6bee\u6bef", - "\u6bf0\u6bf1\u6bf2\u6bf3\u6bf4\u6bf5\u6bf6\u6bf7\u6bf8\u6bf9\u6bfa\u6bfb\u6bfc\u6bfd\u6bfe\u6bff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_6c.png", - "ascent": %ascent%, - "chars": [ - "\u6c00\u6c01\u6c02\u6c03\u6c04\u6c05\u6c06\u6c07\u6c08\u6c09\u6c0a\u6c0b\u6c0c\u6c0d\u6c0e\u6c0f", - "\u6c10\u6c11\u6c12\u6c13\u6c14\u6c15\u6c16\u6c17\u6c18\u6c19\u6c1a\u6c1b\u6c1c\u6c1d\u6c1e\u6c1f", - "\u6c20\u6c21\u6c22\u6c23\u6c24\u6c25\u6c26\u6c27\u6c28\u6c29\u6c2a\u6c2b\u6c2c\u6c2d\u6c2e\u6c2f", - "\u6c30\u6c31\u6c32\u6c33\u6c34\u6c35\u6c36\u6c37\u6c38\u6c39\u6c3a\u6c3b\u6c3c\u6c3d\u6c3e\u6c3f", - "\u6c40\u6c41\u6c42\u6c43\u6c44\u6c45\u6c46\u6c47\u6c48\u6c49\u6c4a\u6c4b\u6c4c\u6c4d\u6c4e\u6c4f", - "\u6c50\u6c51\u6c52\u6c53\u6c54\u6c55\u6c56\u6c57\u6c58\u6c59\u6c5a\u6c5b\u6c5c\u6c5d\u6c5e\u6c5f", - "\u6c60\u6c61\u6c62\u6c63\u6c64\u6c65\u6c66\u6c67\u6c68\u6c69\u6c6a\u6c6b\u6c6c\u6c6d\u6c6e\u6c6f", - "\u6c70\u6c71\u6c72\u6c73\u6c74\u6c75\u6c76\u6c77\u6c78\u6c79\u6c7a\u6c7b\u6c7c\u6c7d\u6c7e\u6c7f", - "\u6c80\u6c81\u6c82\u6c83\u6c84\u6c85\u6c86\u6c87\u6c88\u6c89\u6c8a\u6c8b\u6c8c\u6c8d\u6c8e\u6c8f", - "\u6c90\u6c91\u6c92\u6c93\u6c94\u6c95\u6c96\u6c97\u6c98\u6c99\u6c9a\u6c9b\u6c9c\u6c9d\u6c9e\u6c9f", - "\u6ca0\u6ca1\u6ca2\u6ca3\u6ca4\u6ca5\u6ca6\u6ca7\u6ca8\u6ca9\u6caa\u6cab\u6cac\u6cad\u6cae\u6caf", - "\u6cb0\u6cb1\u6cb2\u6cb3\u6cb4\u6cb5\u6cb6\u6cb7\u6cb8\u6cb9\u6cba\u6cbb\u6cbc\u6cbd\u6cbe\u6cbf", - "\u6cc0\u6cc1\u6cc2\u6cc3\u6cc4\u6cc5\u6cc6\u6cc7\u6cc8\u6cc9\u6cca\u6ccb\u6ccc\u6ccd\u6cce\u6ccf", - "\u6cd0\u6cd1\u6cd2\u6cd3\u6cd4\u6cd5\u6cd6\u6cd7\u6cd8\u6cd9\u6cda\u6cdb\u6cdc\u6cdd\u6cde\u6cdf", - "\u6ce0\u6ce1\u6ce2\u6ce3\u6ce4\u6ce5\u6ce6\u6ce7\u6ce8\u6ce9\u6cea\u6ceb\u6cec\u6ced\u6cee\u6cef", - "\u6cf0\u6cf1\u6cf2\u6cf3\u6cf4\u6cf5\u6cf6\u6cf7\u6cf8\u6cf9\u6cfa\u6cfb\u6cfc\u6cfd\u6cfe\u6cff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_6d.png", - "ascent": %ascent%, - "chars": [ - "\u6d00\u6d01\u6d02\u6d03\u6d04\u6d05\u6d06\u6d07\u6d08\u6d09\u6d0a\u6d0b\u6d0c\u6d0d\u6d0e\u6d0f", - "\u6d10\u6d11\u6d12\u6d13\u6d14\u6d15\u6d16\u6d17\u6d18\u6d19\u6d1a\u6d1b\u6d1c\u6d1d\u6d1e\u6d1f", - "\u6d20\u6d21\u6d22\u6d23\u6d24\u6d25\u6d26\u6d27\u6d28\u6d29\u6d2a\u6d2b\u6d2c\u6d2d\u6d2e\u6d2f", - "\u6d30\u6d31\u6d32\u6d33\u6d34\u6d35\u6d36\u6d37\u6d38\u6d39\u6d3a\u6d3b\u6d3c\u6d3d\u6d3e\u6d3f", - "\u6d40\u6d41\u6d42\u6d43\u6d44\u6d45\u6d46\u6d47\u6d48\u6d49\u6d4a\u6d4b\u6d4c\u6d4d\u6d4e\u6d4f", - "\u6d50\u6d51\u6d52\u6d53\u6d54\u6d55\u6d56\u6d57\u6d58\u6d59\u6d5a\u6d5b\u6d5c\u6d5d\u6d5e\u6d5f", - "\u6d60\u6d61\u6d62\u6d63\u6d64\u6d65\u6d66\u6d67\u6d68\u6d69\u6d6a\u6d6b\u6d6c\u6d6d\u6d6e\u6d6f", - "\u6d70\u6d71\u6d72\u6d73\u6d74\u6d75\u6d76\u6d77\u6d78\u6d79\u6d7a\u6d7b\u6d7c\u6d7d\u6d7e\u6d7f", - "\u6d80\u6d81\u6d82\u6d83\u6d84\u6d85\u6d86\u6d87\u6d88\u6d89\u6d8a\u6d8b\u6d8c\u6d8d\u6d8e\u6d8f", - "\u6d90\u6d91\u6d92\u6d93\u6d94\u6d95\u6d96\u6d97\u6d98\u6d99\u6d9a\u6d9b\u6d9c\u6d9d\u6d9e\u6d9f", - "\u6da0\u6da1\u6da2\u6da3\u6da4\u6da5\u6da6\u6da7\u6da8\u6da9\u6daa\u6dab\u6dac\u6dad\u6dae\u6daf", - "\u6db0\u6db1\u6db2\u6db3\u6db4\u6db5\u6db6\u6db7\u6db8\u6db9\u6dba\u6dbb\u6dbc\u6dbd\u6dbe\u6dbf", - "\u6dc0\u6dc1\u6dc2\u6dc3\u6dc4\u6dc5\u6dc6\u6dc7\u6dc8\u6dc9\u6dca\u6dcb\u6dcc\u6dcd\u6dce\u6dcf", - "\u6dd0\u6dd1\u6dd2\u6dd3\u6dd4\u6dd5\u6dd6\u6dd7\u6dd8\u6dd9\u6dda\u6ddb\u6ddc\u6ddd\u6dde\u6ddf", - "\u6de0\u6de1\u6de2\u6de3\u6de4\u6de5\u6de6\u6de7\u6de8\u6de9\u6dea\u6deb\u6dec\u6ded\u6dee\u6def", - "\u6df0\u6df1\u6df2\u6df3\u6df4\u6df5\u6df6\u6df7\u6df8\u6df9\u6dfa\u6dfb\u6dfc\u6dfd\u6dfe\u6dff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_6e.png", - "ascent": %ascent%, - "chars": [ - "\u6e00\u6e01\u6e02\u6e03\u6e04\u6e05\u6e06\u6e07\u6e08\u6e09\u6e0a\u6e0b\u6e0c\u6e0d\u6e0e\u6e0f", - "\u6e10\u6e11\u6e12\u6e13\u6e14\u6e15\u6e16\u6e17\u6e18\u6e19\u6e1a\u6e1b\u6e1c\u6e1d\u6e1e\u6e1f", - "\u6e20\u6e21\u6e22\u6e23\u6e24\u6e25\u6e26\u6e27\u6e28\u6e29\u6e2a\u6e2b\u6e2c\u6e2d\u6e2e\u6e2f", - "\u6e30\u6e31\u6e32\u6e33\u6e34\u6e35\u6e36\u6e37\u6e38\u6e39\u6e3a\u6e3b\u6e3c\u6e3d\u6e3e\u6e3f", - "\u6e40\u6e41\u6e42\u6e43\u6e44\u6e45\u6e46\u6e47\u6e48\u6e49\u6e4a\u6e4b\u6e4c\u6e4d\u6e4e\u6e4f", - "\u6e50\u6e51\u6e52\u6e53\u6e54\u6e55\u6e56\u6e57\u6e58\u6e59\u6e5a\u6e5b\u6e5c\u6e5d\u6e5e\u6e5f", - "\u6e60\u6e61\u6e62\u6e63\u6e64\u6e65\u6e66\u6e67\u6e68\u6e69\u6e6a\u6e6b\u6e6c\u6e6d\u6e6e\u6e6f", - "\u6e70\u6e71\u6e72\u6e73\u6e74\u6e75\u6e76\u6e77\u6e78\u6e79\u6e7a\u6e7b\u6e7c\u6e7d\u6e7e\u6e7f", - "\u6e80\u6e81\u6e82\u6e83\u6e84\u6e85\u6e86\u6e87\u6e88\u6e89\u6e8a\u6e8b\u6e8c\u6e8d\u6e8e\u6e8f", - "\u6e90\u6e91\u6e92\u6e93\u6e94\u6e95\u6e96\u6e97\u6e98\u6e99\u6e9a\u6e9b\u6e9c\u6e9d\u6e9e\u6e9f", - "\u6ea0\u6ea1\u6ea2\u6ea3\u6ea4\u6ea5\u6ea6\u6ea7\u6ea8\u6ea9\u6eaa\u6eab\u6eac\u6ead\u6eae\u6eaf", - "\u6eb0\u6eb1\u6eb2\u6eb3\u6eb4\u6eb5\u6eb6\u6eb7\u6eb8\u6eb9\u6eba\u6ebb\u6ebc\u6ebd\u6ebe\u6ebf", - "\u6ec0\u6ec1\u6ec2\u6ec3\u6ec4\u6ec5\u6ec6\u6ec7\u6ec8\u6ec9\u6eca\u6ecb\u6ecc\u6ecd\u6ece\u6ecf", - "\u6ed0\u6ed1\u6ed2\u6ed3\u6ed4\u6ed5\u6ed6\u6ed7\u6ed8\u6ed9\u6eda\u6edb\u6edc\u6edd\u6ede\u6edf", - "\u6ee0\u6ee1\u6ee2\u6ee3\u6ee4\u6ee5\u6ee6\u6ee7\u6ee8\u6ee9\u6eea\u6eeb\u6eec\u6eed\u6eee\u6eef", - "\u6ef0\u6ef1\u6ef2\u6ef3\u6ef4\u6ef5\u6ef6\u6ef7\u6ef8\u6ef9\u6efa\u6efb\u6efc\u6efd\u6efe\u6eff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_6f.png", - "ascent": %ascent%, - "chars": [ - "\u6f00\u6f01\u6f02\u6f03\u6f04\u6f05\u6f06\u6f07\u6f08\u6f09\u6f0a\u6f0b\u6f0c\u6f0d\u6f0e\u6f0f", - "\u6f10\u6f11\u6f12\u6f13\u6f14\u6f15\u6f16\u6f17\u6f18\u6f19\u6f1a\u6f1b\u6f1c\u6f1d\u6f1e\u6f1f", - "\u6f20\u6f21\u6f22\u6f23\u6f24\u6f25\u6f26\u6f27\u6f28\u6f29\u6f2a\u6f2b\u6f2c\u6f2d\u6f2e\u6f2f", - "\u6f30\u6f31\u6f32\u6f33\u6f34\u6f35\u6f36\u6f37\u6f38\u6f39\u6f3a\u6f3b\u6f3c\u6f3d\u6f3e\u6f3f", - "\u6f40\u6f41\u6f42\u6f43\u6f44\u6f45\u6f46\u6f47\u6f48\u6f49\u6f4a\u6f4b\u6f4c\u6f4d\u6f4e\u6f4f", - "\u6f50\u6f51\u6f52\u6f53\u6f54\u6f55\u6f56\u6f57\u6f58\u6f59\u6f5a\u6f5b\u6f5c\u6f5d\u6f5e\u6f5f", - "\u6f60\u6f61\u6f62\u6f63\u6f64\u6f65\u6f66\u6f67\u6f68\u6f69\u6f6a\u6f6b\u6f6c\u6f6d\u6f6e\u6f6f", - "\u6f70\u6f71\u6f72\u6f73\u6f74\u6f75\u6f76\u6f77\u6f78\u6f79\u6f7a\u6f7b\u6f7c\u6f7d\u6f7e\u6f7f", - "\u6f80\u6f81\u6f82\u6f83\u6f84\u6f85\u6f86\u6f87\u6f88\u6f89\u6f8a\u6f8b\u6f8c\u6f8d\u6f8e\u6f8f", - "\u6f90\u6f91\u6f92\u6f93\u6f94\u6f95\u6f96\u6f97\u6f98\u6f99\u6f9a\u6f9b\u6f9c\u6f9d\u6f9e\u6f9f", - "\u6fa0\u6fa1\u6fa2\u6fa3\u6fa4\u6fa5\u6fa6\u6fa7\u6fa8\u6fa9\u6faa\u6fab\u6fac\u6fad\u6fae\u6faf", - "\u6fb0\u6fb1\u6fb2\u6fb3\u6fb4\u6fb5\u6fb6\u6fb7\u6fb8\u6fb9\u6fba\u6fbb\u6fbc\u6fbd\u6fbe\u6fbf", - "\u6fc0\u6fc1\u6fc2\u6fc3\u6fc4\u6fc5\u6fc6\u6fc7\u6fc8\u6fc9\u6fca\u6fcb\u6fcc\u6fcd\u6fce\u6fcf", - "\u6fd0\u6fd1\u6fd2\u6fd3\u6fd4\u6fd5\u6fd6\u6fd7\u6fd8\u6fd9\u6fda\u6fdb\u6fdc\u6fdd\u6fde\u6fdf", - "\u6fe0\u6fe1\u6fe2\u6fe3\u6fe4\u6fe5\u6fe6\u6fe7\u6fe8\u6fe9\u6fea\u6feb\u6fec\u6fed\u6fee\u6fef", - "\u6ff0\u6ff1\u6ff2\u6ff3\u6ff4\u6ff5\u6ff6\u6ff7\u6ff8\u6ff9\u6ffa\u6ffb\u6ffc\u6ffd\u6ffe\u6fff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_70.png", - "ascent": %ascent%, - "chars": [ - "\u7000\u7001\u7002\u7003\u7004\u7005\u7006\u7007\u7008\u7009\u700a\u700b\u700c\u700d\u700e\u700f", - "\u7010\u7011\u7012\u7013\u7014\u7015\u7016\u7017\u7018\u7019\u701a\u701b\u701c\u701d\u701e\u701f", - "\u7020\u7021\u7022\u7023\u7024\u7025\u7026\u7027\u7028\u7029\u702a\u702b\u702c\u702d\u702e\u702f", - "\u7030\u7031\u7032\u7033\u7034\u7035\u7036\u7037\u7038\u7039\u703a\u703b\u703c\u703d\u703e\u703f", - "\u7040\u7041\u7042\u7043\u7044\u7045\u7046\u7047\u7048\u7049\u704a\u704b\u704c\u704d\u704e\u704f", - "\u7050\u7051\u7052\u7053\u7054\u7055\u7056\u7057\u7058\u7059\u705a\u705b\u705c\u705d\u705e\u705f", - "\u7060\u7061\u7062\u7063\u7064\u7065\u7066\u7067\u7068\u7069\u706a\u706b\u706c\u706d\u706e\u706f", - "\u7070\u7071\u7072\u7073\u7074\u7075\u7076\u7077\u7078\u7079\u707a\u707b\u707c\u707d\u707e\u707f", - "\u7080\u7081\u7082\u7083\u7084\u7085\u7086\u7087\u7088\u7089\u708a\u708b\u708c\u708d\u708e\u708f", - "\u7090\u7091\u7092\u7093\u7094\u7095\u7096\u7097\u7098\u7099\u709a\u709b\u709c\u709d\u709e\u709f", - "\u70a0\u70a1\u70a2\u70a3\u70a4\u70a5\u70a6\u70a7\u70a8\u70a9\u70aa\u70ab\u70ac\u70ad\u70ae\u70af", - "\u70b0\u70b1\u70b2\u70b3\u70b4\u70b5\u70b6\u70b7\u70b8\u70b9\u70ba\u70bb\u70bc\u70bd\u70be\u70bf", - "\u70c0\u70c1\u70c2\u70c3\u70c4\u70c5\u70c6\u70c7\u70c8\u70c9\u70ca\u70cb\u70cc\u70cd\u70ce\u70cf", - "\u70d0\u70d1\u70d2\u70d3\u70d4\u70d5\u70d6\u70d7\u70d8\u70d9\u70da\u70db\u70dc\u70dd\u70de\u70df", - "\u70e0\u70e1\u70e2\u70e3\u70e4\u70e5\u70e6\u70e7\u70e8\u70e9\u70ea\u70eb\u70ec\u70ed\u70ee\u70ef", - "\u70f0\u70f1\u70f2\u70f3\u70f4\u70f5\u70f6\u70f7\u70f8\u70f9\u70fa\u70fb\u70fc\u70fd\u70fe\u70ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_71.png", - "ascent": %ascent%, - "chars": [ - "\u7100\u7101\u7102\u7103\u7104\u7105\u7106\u7107\u7108\u7109\u710a\u710b\u710c\u710d\u710e\u710f", - "\u7110\u7111\u7112\u7113\u7114\u7115\u7116\u7117\u7118\u7119\u711a\u711b\u711c\u711d\u711e\u711f", - "\u7120\u7121\u7122\u7123\u7124\u7125\u7126\u7127\u7128\u7129\u712a\u712b\u712c\u712d\u712e\u712f", - "\u7130\u7131\u7132\u7133\u7134\u7135\u7136\u7137\u7138\u7139\u713a\u713b\u713c\u713d\u713e\u713f", - "\u7140\u7141\u7142\u7143\u7144\u7145\u7146\u7147\u7148\u7149\u714a\u714b\u714c\u714d\u714e\u714f", - "\u7150\u7151\u7152\u7153\u7154\u7155\u7156\u7157\u7158\u7159\u715a\u715b\u715c\u715d\u715e\u715f", - "\u7160\u7161\u7162\u7163\u7164\u7165\u7166\u7167\u7168\u7169\u716a\u716b\u716c\u716d\u716e\u716f", - "\u7170\u7171\u7172\u7173\u7174\u7175\u7176\u7177\u7178\u7179\u717a\u717b\u717c\u717d\u717e\u717f", - "\u7180\u7181\u7182\u7183\u7184\u7185\u7186\u7187\u7188\u7189\u718a\u718b\u718c\u718d\u718e\u718f", - "\u7190\u7191\u7192\u7193\u7194\u7195\u7196\u7197\u7198\u7199\u719a\u719b\u719c\u719d\u719e\u719f", - "\u71a0\u71a1\u71a2\u71a3\u71a4\u71a5\u71a6\u71a7\u71a8\u71a9\u71aa\u71ab\u71ac\u71ad\u71ae\u71af", - "\u71b0\u71b1\u71b2\u71b3\u71b4\u71b5\u71b6\u71b7\u71b8\u71b9\u71ba\u71bb\u71bc\u71bd\u71be\u71bf", - "\u71c0\u71c1\u71c2\u71c3\u71c4\u71c5\u71c6\u71c7\u71c8\u71c9\u71ca\u71cb\u71cc\u71cd\u71ce\u71cf", - "\u71d0\u71d1\u71d2\u71d3\u71d4\u71d5\u71d6\u71d7\u71d8\u71d9\u71da\u71db\u71dc\u71dd\u71de\u71df", - "\u71e0\u71e1\u71e2\u71e3\u71e4\u71e5\u71e6\u71e7\u71e8\u71e9\u71ea\u71eb\u71ec\u71ed\u71ee\u71ef", - "\u71f0\u71f1\u71f2\u71f3\u71f4\u71f5\u71f6\u71f7\u71f8\u71f9\u71fa\u71fb\u71fc\u71fd\u71fe\u71ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_72.png", - "ascent": %ascent%, - "chars": [ - "\u7200\u7201\u7202\u7203\u7204\u7205\u7206\u7207\u7208\u7209\u720a\u720b\u720c\u720d\u720e\u720f", - "\u7210\u7211\u7212\u7213\u7214\u7215\u7216\u7217\u7218\u7219\u721a\u721b\u721c\u721d\u721e\u721f", - "\u7220\u7221\u7222\u7223\u7224\u7225\u7226\u7227\u7228\u7229\u722a\u722b\u722c\u722d\u722e\u722f", - "\u7230\u7231\u7232\u7233\u7234\u7235\u7236\u7237\u7238\u7239\u723a\u723b\u723c\u723d\u723e\u723f", - "\u7240\u7241\u7242\u7243\u7244\u7245\u7246\u7247\u7248\u7249\u724a\u724b\u724c\u724d\u724e\u724f", - "\u7250\u7251\u7252\u7253\u7254\u7255\u7256\u7257\u7258\u7259\u725a\u725b\u725c\u725d\u725e\u725f", - "\u7260\u7261\u7262\u7263\u7264\u7265\u7266\u7267\u7268\u7269\u726a\u726b\u726c\u726d\u726e\u726f", - "\u7270\u7271\u7272\u7273\u7274\u7275\u7276\u7277\u7278\u7279\u727a\u727b\u727c\u727d\u727e\u727f", - "\u7280\u7281\u7282\u7283\u7284\u7285\u7286\u7287\u7288\u7289\u728a\u728b\u728c\u728d\u728e\u728f", - "\u7290\u7291\u7292\u7293\u7294\u7295\u7296\u7297\u7298\u7299\u729a\u729b\u729c\u729d\u729e\u729f", - "\u72a0\u72a1\u72a2\u72a3\u72a4\u72a5\u72a6\u72a7\u72a8\u72a9\u72aa\u72ab\u72ac\u72ad\u72ae\u72af", - "\u72b0\u72b1\u72b2\u72b3\u72b4\u72b5\u72b6\u72b7\u72b8\u72b9\u72ba\u72bb\u72bc\u72bd\u72be\u72bf", - "\u72c0\u72c1\u72c2\u72c3\u72c4\u72c5\u72c6\u72c7\u72c8\u72c9\u72ca\u72cb\u72cc\u72cd\u72ce\u72cf", - "\u72d0\u72d1\u72d2\u72d3\u72d4\u72d5\u72d6\u72d7\u72d8\u72d9\u72da\u72db\u72dc\u72dd\u72de\u72df", - "\u72e0\u72e1\u72e2\u72e3\u72e4\u72e5\u72e6\u72e7\u72e8\u72e9\u72ea\u72eb\u72ec\u72ed\u72ee\u72ef", - "\u72f0\u72f1\u72f2\u72f3\u72f4\u72f5\u72f6\u72f7\u72f8\u72f9\u72fa\u72fb\u72fc\u72fd\u72fe\u72ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_73.png", - "ascent": %ascent%, - "chars": [ - "\u7300\u7301\u7302\u7303\u7304\u7305\u7306\u7307\u7308\u7309\u730a\u730b\u730c\u730d\u730e\u730f", - "\u7310\u7311\u7312\u7313\u7314\u7315\u7316\u7317\u7318\u7319\u731a\u731b\u731c\u731d\u731e\u731f", - "\u7320\u7321\u7322\u7323\u7324\u7325\u7326\u7327\u7328\u7329\u732a\u732b\u732c\u732d\u732e\u732f", - "\u7330\u7331\u7332\u7333\u7334\u7335\u7336\u7337\u7338\u7339\u733a\u733b\u733c\u733d\u733e\u733f", - "\u7340\u7341\u7342\u7343\u7344\u7345\u7346\u7347\u7348\u7349\u734a\u734b\u734c\u734d\u734e\u734f", - "\u7350\u7351\u7352\u7353\u7354\u7355\u7356\u7357\u7358\u7359\u735a\u735b\u735c\u735d\u735e\u735f", - "\u7360\u7361\u7362\u7363\u7364\u7365\u7366\u7367\u7368\u7369\u736a\u736b\u736c\u736d\u736e\u736f", - "\u7370\u7371\u7372\u7373\u7374\u7375\u7376\u7377\u7378\u7379\u737a\u737b\u737c\u737d\u737e\u737f", - "\u7380\u7381\u7382\u7383\u7384\u7385\u7386\u7387\u7388\u7389\u738a\u738b\u738c\u738d\u738e\u738f", - "\u7390\u7391\u7392\u7393\u7394\u7395\u7396\u7397\u7398\u7399\u739a\u739b\u739c\u739d\u739e\u739f", - "\u73a0\u73a1\u73a2\u73a3\u73a4\u73a5\u73a6\u73a7\u73a8\u73a9\u73aa\u73ab\u73ac\u73ad\u73ae\u73af", - "\u73b0\u73b1\u73b2\u73b3\u73b4\u73b5\u73b6\u73b7\u73b8\u73b9\u73ba\u73bb\u73bc\u73bd\u73be\u73bf", - "\u73c0\u73c1\u73c2\u73c3\u73c4\u73c5\u73c6\u73c7\u73c8\u73c9\u73ca\u73cb\u73cc\u73cd\u73ce\u73cf", - "\u73d0\u73d1\u73d2\u73d3\u73d4\u73d5\u73d6\u73d7\u73d8\u73d9\u73da\u73db\u73dc\u73dd\u73de\u73df", - "\u73e0\u73e1\u73e2\u73e3\u73e4\u73e5\u73e6\u73e7\u73e8\u73e9\u73ea\u73eb\u73ec\u73ed\u73ee\u73ef", - "\u73f0\u73f1\u73f2\u73f3\u73f4\u73f5\u73f6\u73f7\u73f8\u73f9\u73fa\u73fb\u73fc\u73fd\u73fe\u73ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_74.png", - "ascent": %ascent%, - "chars": [ - "\u7400\u7401\u7402\u7403\u7404\u7405\u7406\u7407\u7408\u7409\u740a\u740b\u740c\u740d\u740e\u740f", - "\u7410\u7411\u7412\u7413\u7414\u7415\u7416\u7417\u7418\u7419\u741a\u741b\u741c\u741d\u741e\u741f", - "\u7420\u7421\u7422\u7423\u7424\u7425\u7426\u7427\u7428\u7429\u742a\u742b\u742c\u742d\u742e\u742f", - "\u7430\u7431\u7432\u7433\u7434\u7435\u7436\u7437\u7438\u7439\u743a\u743b\u743c\u743d\u743e\u743f", - "\u7440\u7441\u7442\u7443\u7444\u7445\u7446\u7447\u7448\u7449\u744a\u744b\u744c\u744d\u744e\u744f", - "\u7450\u7451\u7452\u7453\u7454\u7455\u7456\u7457\u7458\u7459\u745a\u745b\u745c\u745d\u745e\u745f", - "\u7460\u7461\u7462\u7463\u7464\u7465\u7466\u7467\u7468\u7469\u746a\u746b\u746c\u746d\u746e\u746f", - "\u7470\u7471\u7472\u7473\u7474\u7475\u7476\u7477\u7478\u7479\u747a\u747b\u747c\u747d\u747e\u747f", - "\u7480\u7481\u7482\u7483\u7484\u7485\u7486\u7487\u7488\u7489\u748a\u748b\u748c\u748d\u748e\u748f", - "\u7490\u7491\u7492\u7493\u7494\u7495\u7496\u7497\u7498\u7499\u749a\u749b\u749c\u749d\u749e\u749f", - "\u74a0\u74a1\u74a2\u74a3\u74a4\u74a5\u74a6\u74a7\u74a8\u74a9\u74aa\u74ab\u74ac\u74ad\u74ae\u74af", - "\u74b0\u74b1\u74b2\u74b3\u74b4\u74b5\u74b6\u74b7\u74b8\u74b9\u74ba\u74bb\u74bc\u74bd\u74be\u74bf", - "\u74c0\u74c1\u74c2\u74c3\u74c4\u74c5\u74c6\u74c7\u74c8\u74c9\u74ca\u74cb\u74cc\u74cd\u74ce\u74cf", - "\u74d0\u74d1\u74d2\u74d3\u74d4\u74d5\u74d6\u74d7\u74d8\u74d9\u74da\u74db\u74dc\u74dd\u74de\u74df", - "\u74e0\u74e1\u74e2\u74e3\u74e4\u74e5\u74e6\u74e7\u74e8\u74e9\u74ea\u74eb\u74ec\u74ed\u74ee\u74ef", - "\u74f0\u74f1\u74f2\u74f3\u74f4\u74f5\u74f6\u74f7\u74f8\u74f9\u74fa\u74fb\u74fc\u74fd\u74fe\u74ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_75.png", - "ascent": %ascent%, - "chars": [ - "\u7500\u7501\u7502\u7503\u7504\u7505\u7506\u7507\u7508\u7509\u750a\u750b\u750c\u750d\u750e\u750f", - "\u7510\u7511\u7512\u7513\u7514\u7515\u7516\u7517\u7518\u7519\u751a\u751b\u751c\u751d\u751e\u751f", - "\u7520\u7521\u7522\u7523\u7524\u7525\u7526\u7527\u7528\u7529\u752a\u752b\u752c\u752d\u752e\u752f", - "\u7530\u7531\u7532\u7533\u7534\u7535\u7536\u7537\u7538\u7539\u753a\u753b\u753c\u753d\u753e\u753f", - "\u7540\u7541\u7542\u7543\u7544\u7545\u7546\u7547\u7548\u7549\u754a\u754b\u754c\u754d\u754e\u754f", - "\u7550\u7551\u7552\u7553\u7554\u7555\u7556\u7557\u7558\u7559\u755a\u755b\u755c\u755d\u755e\u755f", - "\u7560\u7561\u7562\u7563\u7564\u7565\u7566\u7567\u7568\u7569\u756a\u756b\u756c\u756d\u756e\u756f", - "\u7570\u7571\u7572\u7573\u7574\u7575\u7576\u7577\u7578\u7579\u757a\u757b\u757c\u757d\u757e\u757f", - "\u7580\u7581\u7582\u7583\u7584\u7585\u7586\u7587\u7588\u7589\u758a\u758b\u758c\u758d\u758e\u758f", - "\u7590\u7591\u7592\u7593\u7594\u7595\u7596\u7597\u7598\u7599\u759a\u759b\u759c\u759d\u759e\u759f", - "\u75a0\u75a1\u75a2\u75a3\u75a4\u75a5\u75a6\u75a7\u75a8\u75a9\u75aa\u75ab\u75ac\u75ad\u75ae\u75af", - "\u75b0\u75b1\u75b2\u75b3\u75b4\u75b5\u75b6\u75b7\u75b8\u75b9\u75ba\u75bb\u75bc\u75bd\u75be\u75bf", - "\u75c0\u75c1\u75c2\u75c3\u75c4\u75c5\u75c6\u75c7\u75c8\u75c9\u75ca\u75cb\u75cc\u75cd\u75ce\u75cf", - "\u75d0\u75d1\u75d2\u75d3\u75d4\u75d5\u75d6\u75d7\u75d8\u75d9\u75da\u75db\u75dc\u75dd\u75de\u75df", - "\u75e0\u75e1\u75e2\u75e3\u75e4\u75e5\u75e6\u75e7\u75e8\u75e9\u75ea\u75eb\u75ec\u75ed\u75ee\u75ef", - "\u75f0\u75f1\u75f2\u75f3\u75f4\u75f5\u75f6\u75f7\u75f8\u75f9\u75fa\u75fb\u75fc\u75fd\u75fe\u75ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_76.png", - "ascent": %ascent%, - "chars": [ - "\u7600\u7601\u7602\u7603\u7604\u7605\u7606\u7607\u7608\u7609\u760a\u760b\u760c\u760d\u760e\u760f", - "\u7610\u7611\u7612\u7613\u7614\u7615\u7616\u7617\u7618\u7619\u761a\u761b\u761c\u761d\u761e\u761f", - "\u7620\u7621\u7622\u7623\u7624\u7625\u7626\u7627\u7628\u7629\u762a\u762b\u762c\u762d\u762e\u762f", - "\u7630\u7631\u7632\u7633\u7634\u7635\u7636\u7637\u7638\u7639\u763a\u763b\u763c\u763d\u763e\u763f", - "\u7640\u7641\u7642\u7643\u7644\u7645\u7646\u7647\u7648\u7649\u764a\u764b\u764c\u764d\u764e\u764f", - "\u7650\u7651\u7652\u7653\u7654\u7655\u7656\u7657\u7658\u7659\u765a\u765b\u765c\u765d\u765e\u765f", - "\u7660\u7661\u7662\u7663\u7664\u7665\u7666\u7667\u7668\u7669\u766a\u766b\u766c\u766d\u766e\u766f", - "\u7670\u7671\u7672\u7673\u7674\u7675\u7676\u7677\u7678\u7679\u767a\u767b\u767c\u767d\u767e\u767f", - "\u7680\u7681\u7682\u7683\u7684\u7685\u7686\u7687\u7688\u7689\u768a\u768b\u768c\u768d\u768e\u768f", - "\u7690\u7691\u7692\u7693\u7694\u7695\u7696\u7697\u7698\u7699\u769a\u769b\u769c\u769d\u769e\u769f", - "\u76a0\u76a1\u76a2\u76a3\u76a4\u76a5\u76a6\u76a7\u76a8\u76a9\u76aa\u76ab\u76ac\u76ad\u76ae\u76af", - "\u76b0\u76b1\u76b2\u76b3\u76b4\u76b5\u76b6\u76b7\u76b8\u76b9\u76ba\u76bb\u76bc\u76bd\u76be\u76bf", - "\u76c0\u76c1\u76c2\u76c3\u76c4\u76c5\u76c6\u76c7\u76c8\u76c9\u76ca\u76cb\u76cc\u76cd\u76ce\u76cf", - "\u76d0\u76d1\u76d2\u76d3\u76d4\u76d5\u76d6\u76d7\u76d8\u76d9\u76da\u76db\u76dc\u76dd\u76de\u76df", - "\u76e0\u76e1\u76e2\u76e3\u76e4\u76e5\u76e6\u76e7\u76e8\u76e9\u76ea\u76eb\u76ec\u76ed\u76ee\u76ef", - "\u76f0\u76f1\u76f2\u76f3\u76f4\u76f5\u76f6\u76f7\u76f8\u76f9\u76fa\u76fb\u76fc\u76fd\u76fe\u76ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_77.png", - "ascent": %ascent%, - "chars": [ - "\u7700\u7701\u7702\u7703\u7704\u7705\u7706\u7707\u7708\u7709\u770a\u770b\u770c\u770d\u770e\u770f", - "\u7710\u7711\u7712\u7713\u7714\u7715\u7716\u7717\u7718\u7719\u771a\u771b\u771c\u771d\u771e\u771f", - "\u7720\u7721\u7722\u7723\u7724\u7725\u7726\u7727\u7728\u7729\u772a\u772b\u772c\u772d\u772e\u772f", - "\u7730\u7731\u7732\u7733\u7734\u7735\u7736\u7737\u7738\u7739\u773a\u773b\u773c\u773d\u773e\u773f", - "\u7740\u7741\u7742\u7743\u7744\u7745\u7746\u7747\u7748\u7749\u774a\u774b\u774c\u774d\u774e\u774f", - "\u7750\u7751\u7752\u7753\u7754\u7755\u7756\u7757\u7758\u7759\u775a\u775b\u775c\u775d\u775e\u775f", - "\u7760\u7761\u7762\u7763\u7764\u7765\u7766\u7767\u7768\u7769\u776a\u776b\u776c\u776d\u776e\u776f", - "\u7770\u7771\u7772\u7773\u7774\u7775\u7776\u7777\u7778\u7779\u777a\u777b\u777c\u777d\u777e\u777f", - "\u7780\u7781\u7782\u7783\u7784\u7785\u7786\u7787\u7788\u7789\u778a\u778b\u778c\u778d\u778e\u778f", - "\u7790\u7791\u7792\u7793\u7794\u7795\u7796\u7797\u7798\u7799\u779a\u779b\u779c\u779d\u779e\u779f", - "\u77a0\u77a1\u77a2\u77a3\u77a4\u77a5\u77a6\u77a7\u77a8\u77a9\u77aa\u77ab\u77ac\u77ad\u77ae\u77af", - "\u77b0\u77b1\u77b2\u77b3\u77b4\u77b5\u77b6\u77b7\u77b8\u77b9\u77ba\u77bb\u77bc\u77bd\u77be\u77bf", - "\u77c0\u77c1\u77c2\u77c3\u77c4\u77c5\u77c6\u77c7\u77c8\u77c9\u77ca\u77cb\u77cc\u77cd\u77ce\u77cf", - "\u77d0\u77d1\u77d2\u77d3\u77d4\u77d5\u77d6\u77d7\u77d8\u77d9\u77da\u77db\u77dc\u77dd\u77de\u77df", - "\u77e0\u77e1\u77e2\u77e3\u77e4\u77e5\u77e6\u77e7\u77e8\u77e9\u77ea\u77eb\u77ec\u77ed\u77ee\u77ef", - "\u77f0\u77f1\u77f2\u77f3\u77f4\u77f5\u77f6\u77f7\u77f8\u77f9\u77fa\u77fb\u77fc\u77fd\u77fe\u77ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_78.png", - "ascent": %ascent%, - "chars": [ - "\u7800\u7801\u7802\u7803\u7804\u7805\u7806\u7807\u7808\u7809\u780a\u780b\u780c\u780d\u780e\u780f", - "\u7810\u7811\u7812\u7813\u7814\u7815\u7816\u7817\u7818\u7819\u781a\u781b\u781c\u781d\u781e\u781f", - "\u7820\u7821\u7822\u7823\u7824\u7825\u7826\u7827\u7828\u7829\u782a\u782b\u782c\u782d\u782e\u782f", - "\u7830\u7831\u7832\u7833\u7834\u7835\u7836\u7837\u7838\u7839\u783a\u783b\u783c\u783d\u783e\u783f", - "\u7840\u7841\u7842\u7843\u7844\u7845\u7846\u7847\u7848\u7849\u784a\u784b\u784c\u784d\u784e\u784f", - "\u7850\u7851\u7852\u7853\u7854\u7855\u7856\u7857\u7858\u7859\u785a\u785b\u785c\u785d\u785e\u785f", - "\u7860\u7861\u7862\u7863\u7864\u7865\u7866\u7867\u7868\u7869\u786a\u786b\u786c\u786d\u786e\u786f", - "\u7870\u7871\u7872\u7873\u7874\u7875\u7876\u7877\u7878\u7879\u787a\u787b\u787c\u787d\u787e\u787f", - "\u7880\u7881\u7882\u7883\u7884\u7885\u7886\u7887\u7888\u7889\u788a\u788b\u788c\u788d\u788e\u788f", - "\u7890\u7891\u7892\u7893\u7894\u7895\u7896\u7897\u7898\u7899\u789a\u789b\u789c\u789d\u789e\u789f", - "\u78a0\u78a1\u78a2\u78a3\u78a4\u78a5\u78a6\u78a7\u78a8\u78a9\u78aa\u78ab\u78ac\u78ad\u78ae\u78af", - "\u78b0\u78b1\u78b2\u78b3\u78b4\u78b5\u78b6\u78b7\u78b8\u78b9\u78ba\u78bb\u78bc\u78bd\u78be\u78bf", - "\u78c0\u78c1\u78c2\u78c3\u78c4\u78c5\u78c6\u78c7\u78c8\u78c9\u78ca\u78cb\u78cc\u78cd\u78ce\u78cf", - "\u78d0\u78d1\u78d2\u78d3\u78d4\u78d5\u78d6\u78d7\u78d8\u78d9\u78da\u78db\u78dc\u78dd\u78de\u78df", - "\u78e0\u78e1\u78e2\u78e3\u78e4\u78e5\u78e6\u78e7\u78e8\u78e9\u78ea\u78eb\u78ec\u78ed\u78ee\u78ef", - "\u78f0\u78f1\u78f2\u78f3\u78f4\u78f5\u78f6\u78f7\u78f8\u78f9\u78fa\u78fb\u78fc\u78fd\u78fe\u78ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_79.png", - "ascent": %ascent%, - "chars": [ - "\u7900\u7901\u7902\u7903\u7904\u7905\u7906\u7907\u7908\u7909\u790a\u790b\u790c\u790d\u790e\u790f", - "\u7910\u7911\u7912\u7913\u7914\u7915\u7916\u7917\u7918\u7919\u791a\u791b\u791c\u791d\u791e\u791f", - "\u7920\u7921\u7922\u7923\u7924\u7925\u7926\u7927\u7928\u7929\u792a\u792b\u792c\u792d\u792e\u792f", - "\u7930\u7931\u7932\u7933\u7934\u7935\u7936\u7937\u7938\u7939\u793a\u793b\u793c\u793d\u793e\u793f", - "\u7940\u7941\u7942\u7943\u7944\u7945\u7946\u7947\u7948\u7949\u794a\u794b\u794c\u794d\u794e\u794f", - "\u7950\u7951\u7952\u7953\u7954\u7955\u7956\u7957\u7958\u7959\u795a\u795b\u795c\u795d\u795e\u795f", - "\u7960\u7961\u7962\u7963\u7964\u7965\u7966\u7967\u7968\u7969\u796a\u796b\u796c\u796d\u796e\u796f", - "\u7970\u7971\u7972\u7973\u7974\u7975\u7976\u7977\u7978\u7979\u797a\u797b\u797c\u797d\u797e\u797f", - "\u7980\u7981\u7982\u7983\u7984\u7985\u7986\u7987\u7988\u7989\u798a\u798b\u798c\u798d\u798e\u798f", - "\u7990\u7991\u7992\u7993\u7994\u7995\u7996\u7997\u7998\u7999\u799a\u799b\u799c\u799d\u799e\u799f", - "\u79a0\u79a1\u79a2\u79a3\u79a4\u79a5\u79a6\u79a7\u79a8\u79a9\u79aa\u79ab\u79ac\u79ad\u79ae\u79af", - "\u79b0\u79b1\u79b2\u79b3\u79b4\u79b5\u79b6\u79b7\u79b8\u79b9\u79ba\u79bb\u79bc\u79bd\u79be\u79bf", - "\u79c0\u79c1\u79c2\u79c3\u79c4\u79c5\u79c6\u79c7\u79c8\u79c9\u79ca\u79cb\u79cc\u79cd\u79ce\u79cf", - "\u79d0\u79d1\u79d2\u79d3\u79d4\u79d5\u79d6\u79d7\u79d8\u79d9\u79da\u79db\u79dc\u79dd\u79de\u79df", - "\u79e0\u79e1\u79e2\u79e3\u79e4\u79e5\u79e6\u79e7\u79e8\u79e9\u79ea\u79eb\u79ec\u79ed\u79ee\u79ef", - "\u79f0\u79f1\u79f2\u79f3\u79f4\u79f5\u79f6\u79f7\u79f8\u79f9\u79fa\u79fb\u79fc\u79fd\u79fe\u79ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_7a.png", - "ascent": %ascent%, - "chars": [ - "\u7a00\u7a01\u7a02\u7a03\u7a04\u7a05\u7a06\u7a07\u7a08\u7a09\u7a0a\u7a0b\u7a0c\u7a0d\u7a0e\u7a0f", - "\u7a10\u7a11\u7a12\u7a13\u7a14\u7a15\u7a16\u7a17\u7a18\u7a19\u7a1a\u7a1b\u7a1c\u7a1d\u7a1e\u7a1f", - "\u7a20\u7a21\u7a22\u7a23\u7a24\u7a25\u7a26\u7a27\u7a28\u7a29\u7a2a\u7a2b\u7a2c\u7a2d\u7a2e\u7a2f", - "\u7a30\u7a31\u7a32\u7a33\u7a34\u7a35\u7a36\u7a37\u7a38\u7a39\u7a3a\u7a3b\u7a3c\u7a3d\u7a3e\u7a3f", - "\u7a40\u7a41\u7a42\u7a43\u7a44\u7a45\u7a46\u7a47\u7a48\u7a49\u7a4a\u7a4b\u7a4c\u7a4d\u7a4e\u7a4f", - "\u7a50\u7a51\u7a52\u7a53\u7a54\u7a55\u7a56\u7a57\u7a58\u7a59\u7a5a\u7a5b\u7a5c\u7a5d\u7a5e\u7a5f", - "\u7a60\u7a61\u7a62\u7a63\u7a64\u7a65\u7a66\u7a67\u7a68\u7a69\u7a6a\u7a6b\u7a6c\u7a6d\u7a6e\u7a6f", - "\u7a70\u7a71\u7a72\u7a73\u7a74\u7a75\u7a76\u7a77\u7a78\u7a79\u7a7a\u7a7b\u7a7c\u7a7d\u7a7e\u7a7f", - "\u7a80\u7a81\u7a82\u7a83\u7a84\u7a85\u7a86\u7a87\u7a88\u7a89\u7a8a\u7a8b\u7a8c\u7a8d\u7a8e\u7a8f", - "\u7a90\u7a91\u7a92\u7a93\u7a94\u7a95\u7a96\u7a97\u7a98\u7a99\u7a9a\u7a9b\u7a9c\u7a9d\u7a9e\u7a9f", - "\u7aa0\u7aa1\u7aa2\u7aa3\u7aa4\u7aa5\u7aa6\u7aa7\u7aa8\u7aa9\u7aaa\u7aab\u7aac\u7aad\u7aae\u7aaf", - "\u7ab0\u7ab1\u7ab2\u7ab3\u7ab4\u7ab5\u7ab6\u7ab7\u7ab8\u7ab9\u7aba\u7abb\u7abc\u7abd\u7abe\u7abf", - "\u7ac0\u7ac1\u7ac2\u7ac3\u7ac4\u7ac5\u7ac6\u7ac7\u7ac8\u7ac9\u7aca\u7acb\u7acc\u7acd\u7ace\u7acf", - "\u7ad0\u7ad1\u7ad2\u7ad3\u7ad4\u7ad5\u7ad6\u7ad7\u7ad8\u7ad9\u7ada\u7adb\u7adc\u7add\u7ade\u7adf", - "\u7ae0\u7ae1\u7ae2\u7ae3\u7ae4\u7ae5\u7ae6\u7ae7\u7ae8\u7ae9\u7aea\u7aeb\u7aec\u7aed\u7aee\u7aef", - "\u7af0\u7af1\u7af2\u7af3\u7af4\u7af5\u7af6\u7af7\u7af8\u7af9\u7afa\u7afb\u7afc\u7afd\u7afe\u7aff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_7b.png", - "ascent": %ascent%, - "chars": [ - "\u7b00\u7b01\u7b02\u7b03\u7b04\u7b05\u7b06\u7b07\u7b08\u7b09\u7b0a\u7b0b\u7b0c\u7b0d\u7b0e\u7b0f", - "\u7b10\u7b11\u7b12\u7b13\u7b14\u7b15\u7b16\u7b17\u7b18\u7b19\u7b1a\u7b1b\u7b1c\u7b1d\u7b1e\u7b1f", - "\u7b20\u7b21\u7b22\u7b23\u7b24\u7b25\u7b26\u7b27\u7b28\u7b29\u7b2a\u7b2b\u7b2c\u7b2d\u7b2e\u7b2f", - "\u7b30\u7b31\u7b32\u7b33\u7b34\u7b35\u7b36\u7b37\u7b38\u7b39\u7b3a\u7b3b\u7b3c\u7b3d\u7b3e\u7b3f", - "\u7b40\u7b41\u7b42\u7b43\u7b44\u7b45\u7b46\u7b47\u7b48\u7b49\u7b4a\u7b4b\u7b4c\u7b4d\u7b4e\u7b4f", - "\u7b50\u7b51\u7b52\u7b53\u7b54\u7b55\u7b56\u7b57\u7b58\u7b59\u7b5a\u7b5b\u7b5c\u7b5d\u7b5e\u7b5f", - "\u7b60\u7b61\u7b62\u7b63\u7b64\u7b65\u7b66\u7b67\u7b68\u7b69\u7b6a\u7b6b\u7b6c\u7b6d\u7b6e\u7b6f", - "\u7b70\u7b71\u7b72\u7b73\u7b74\u7b75\u7b76\u7b77\u7b78\u7b79\u7b7a\u7b7b\u7b7c\u7b7d\u7b7e\u7b7f", - "\u7b80\u7b81\u7b82\u7b83\u7b84\u7b85\u7b86\u7b87\u7b88\u7b89\u7b8a\u7b8b\u7b8c\u7b8d\u7b8e\u7b8f", - "\u7b90\u7b91\u7b92\u7b93\u7b94\u7b95\u7b96\u7b97\u7b98\u7b99\u7b9a\u7b9b\u7b9c\u7b9d\u7b9e\u7b9f", - "\u7ba0\u7ba1\u7ba2\u7ba3\u7ba4\u7ba5\u7ba6\u7ba7\u7ba8\u7ba9\u7baa\u7bab\u7bac\u7bad\u7bae\u7baf", - "\u7bb0\u7bb1\u7bb2\u7bb3\u7bb4\u7bb5\u7bb6\u7bb7\u7bb8\u7bb9\u7bba\u7bbb\u7bbc\u7bbd\u7bbe\u7bbf", - "\u7bc0\u7bc1\u7bc2\u7bc3\u7bc4\u7bc5\u7bc6\u7bc7\u7bc8\u7bc9\u7bca\u7bcb\u7bcc\u7bcd\u7bce\u7bcf", - "\u7bd0\u7bd1\u7bd2\u7bd3\u7bd4\u7bd5\u7bd6\u7bd7\u7bd8\u7bd9\u7bda\u7bdb\u7bdc\u7bdd\u7bde\u7bdf", - "\u7be0\u7be1\u7be2\u7be3\u7be4\u7be5\u7be6\u7be7\u7be8\u7be9\u7bea\u7beb\u7bec\u7bed\u7bee\u7bef", - "\u7bf0\u7bf1\u7bf2\u7bf3\u7bf4\u7bf5\u7bf6\u7bf7\u7bf8\u7bf9\u7bfa\u7bfb\u7bfc\u7bfd\u7bfe\u7bff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_7c.png", - "ascent": %ascent%, - "chars": [ - "\u7c00\u7c01\u7c02\u7c03\u7c04\u7c05\u7c06\u7c07\u7c08\u7c09\u7c0a\u7c0b\u7c0c\u7c0d\u7c0e\u7c0f", - "\u7c10\u7c11\u7c12\u7c13\u7c14\u7c15\u7c16\u7c17\u7c18\u7c19\u7c1a\u7c1b\u7c1c\u7c1d\u7c1e\u7c1f", - "\u7c20\u7c21\u7c22\u7c23\u7c24\u7c25\u7c26\u7c27\u7c28\u7c29\u7c2a\u7c2b\u7c2c\u7c2d\u7c2e\u7c2f", - "\u7c30\u7c31\u7c32\u7c33\u7c34\u7c35\u7c36\u7c37\u7c38\u7c39\u7c3a\u7c3b\u7c3c\u7c3d\u7c3e\u7c3f", - "\u7c40\u7c41\u7c42\u7c43\u7c44\u7c45\u7c46\u7c47\u7c48\u7c49\u7c4a\u7c4b\u7c4c\u7c4d\u7c4e\u7c4f", - "\u7c50\u7c51\u7c52\u7c53\u7c54\u7c55\u7c56\u7c57\u7c58\u7c59\u7c5a\u7c5b\u7c5c\u7c5d\u7c5e\u7c5f", - "\u7c60\u7c61\u7c62\u7c63\u7c64\u7c65\u7c66\u7c67\u7c68\u7c69\u7c6a\u7c6b\u7c6c\u7c6d\u7c6e\u7c6f", - "\u7c70\u7c71\u7c72\u7c73\u7c74\u7c75\u7c76\u7c77\u7c78\u7c79\u7c7a\u7c7b\u7c7c\u7c7d\u7c7e\u7c7f", - "\u7c80\u7c81\u7c82\u7c83\u7c84\u7c85\u7c86\u7c87\u7c88\u7c89\u7c8a\u7c8b\u7c8c\u7c8d\u7c8e\u7c8f", - "\u7c90\u7c91\u7c92\u7c93\u7c94\u7c95\u7c96\u7c97\u7c98\u7c99\u7c9a\u7c9b\u7c9c\u7c9d\u7c9e\u7c9f", - "\u7ca0\u7ca1\u7ca2\u7ca3\u7ca4\u7ca5\u7ca6\u7ca7\u7ca8\u7ca9\u7caa\u7cab\u7cac\u7cad\u7cae\u7caf", - "\u7cb0\u7cb1\u7cb2\u7cb3\u7cb4\u7cb5\u7cb6\u7cb7\u7cb8\u7cb9\u7cba\u7cbb\u7cbc\u7cbd\u7cbe\u7cbf", - "\u7cc0\u7cc1\u7cc2\u7cc3\u7cc4\u7cc5\u7cc6\u7cc7\u7cc8\u7cc9\u7cca\u7ccb\u7ccc\u7ccd\u7cce\u7ccf", - "\u7cd0\u7cd1\u7cd2\u7cd3\u7cd4\u7cd5\u7cd6\u7cd7\u7cd8\u7cd9\u7cda\u7cdb\u7cdc\u7cdd\u7cde\u7cdf", - "\u7ce0\u7ce1\u7ce2\u7ce3\u7ce4\u7ce5\u7ce6\u7ce7\u7ce8\u7ce9\u7cea\u7ceb\u7cec\u7ced\u7cee\u7cef", - "\u7cf0\u7cf1\u7cf2\u7cf3\u7cf4\u7cf5\u7cf6\u7cf7\u7cf8\u7cf9\u7cfa\u7cfb\u7cfc\u7cfd\u7cfe\u7cff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_7d.png", - "ascent": %ascent%, - "chars": [ - "\u7d00\u7d01\u7d02\u7d03\u7d04\u7d05\u7d06\u7d07\u7d08\u7d09\u7d0a\u7d0b\u7d0c\u7d0d\u7d0e\u7d0f", - "\u7d10\u7d11\u7d12\u7d13\u7d14\u7d15\u7d16\u7d17\u7d18\u7d19\u7d1a\u7d1b\u7d1c\u7d1d\u7d1e\u7d1f", - "\u7d20\u7d21\u7d22\u7d23\u7d24\u7d25\u7d26\u7d27\u7d28\u7d29\u7d2a\u7d2b\u7d2c\u7d2d\u7d2e\u7d2f", - "\u7d30\u7d31\u7d32\u7d33\u7d34\u7d35\u7d36\u7d37\u7d38\u7d39\u7d3a\u7d3b\u7d3c\u7d3d\u7d3e\u7d3f", - "\u7d40\u7d41\u7d42\u7d43\u7d44\u7d45\u7d46\u7d47\u7d48\u7d49\u7d4a\u7d4b\u7d4c\u7d4d\u7d4e\u7d4f", - "\u7d50\u7d51\u7d52\u7d53\u7d54\u7d55\u7d56\u7d57\u7d58\u7d59\u7d5a\u7d5b\u7d5c\u7d5d\u7d5e\u7d5f", - "\u7d60\u7d61\u7d62\u7d63\u7d64\u7d65\u7d66\u7d67\u7d68\u7d69\u7d6a\u7d6b\u7d6c\u7d6d\u7d6e\u7d6f", - "\u7d70\u7d71\u7d72\u7d73\u7d74\u7d75\u7d76\u7d77\u7d78\u7d79\u7d7a\u7d7b\u7d7c\u7d7d\u7d7e\u7d7f", - "\u7d80\u7d81\u7d82\u7d83\u7d84\u7d85\u7d86\u7d87\u7d88\u7d89\u7d8a\u7d8b\u7d8c\u7d8d\u7d8e\u7d8f", - "\u7d90\u7d91\u7d92\u7d93\u7d94\u7d95\u7d96\u7d97\u7d98\u7d99\u7d9a\u7d9b\u7d9c\u7d9d\u7d9e\u7d9f", - "\u7da0\u7da1\u7da2\u7da3\u7da4\u7da5\u7da6\u7da7\u7da8\u7da9\u7daa\u7dab\u7dac\u7dad\u7dae\u7daf", - "\u7db0\u7db1\u7db2\u7db3\u7db4\u7db5\u7db6\u7db7\u7db8\u7db9\u7dba\u7dbb\u7dbc\u7dbd\u7dbe\u7dbf", - "\u7dc0\u7dc1\u7dc2\u7dc3\u7dc4\u7dc5\u7dc6\u7dc7\u7dc8\u7dc9\u7dca\u7dcb\u7dcc\u7dcd\u7dce\u7dcf", - "\u7dd0\u7dd1\u7dd2\u7dd3\u7dd4\u7dd5\u7dd6\u7dd7\u7dd8\u7dd9\u7dda\u7ddb\u7ddc\u7ddd\u7dde\u7ddf", - "\u7de0\u7de1\u7de2\u7de3\u7de4\u7de5\u7de6\u7de7\u7de8\u7de9\u7dea\u7deb\u7dec\u7ded\u7dee\u7def", - "\u7df0\u7df1\u7df2\u7df3\u7df4\u7df5\u7df6\u7df7\u7df8\u7df9\u7dfa\u7dfb\u7dfc\u7dfd\u7dfe\u7dff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_7e.png", - "ascent": %ascent%, - "chars": [ - "\u7e00\u7e01\u7e02\u7e03\u7e04\u7e05\u7e06\u7e07\u7e08\u7e09\u7e0a\u7e0b\u7e0c\u7e0d\u7e0e\u7e0f", - "\u7e10\u7e11\u7e12\u7e13\u7e14\u7e15\u7e16\u7e17\u7e18\u7e19\u7e1a\u7e1b\u7e1c\u7e1d\u7e1e\u7e1f", - "\u7e20\u7e21\u7e22\u7e23\u7e24\u7e25\u7e26\u7e27\u7e28\u7e29\u7e2a\u7e2b\u7e2c\u7e2d\u7e2e\u7e2f", - "\u7e30\u7e31\u7e32\u7e33\u7e34\u7e35\u7e36\u7e37\u7e38\u7e39\u7e3a\u7e3b\u7e3c\u7e3d\u7e3e\u7e3f", - "\u7e40\u7e41\u7e42\u7e43\u7e44\u7e45\u7e46\u7e47\u7e48\u7e49\u7e4a\u7e4b\u7e4c\u7e4d\u7e4e\u7e4f", - "\u7e50\u7e51\u7e52\u7e53\u7e54\u7e55\u7e56\u7e57\u7e58\u7e59\u7e5a\u7e5b\u7e5c\u7e5d\u7e5e\u7e5f", - "\u7e60\u7e61\u7e62\u7e63\u7e64\u7e65\u7e66\u7e67\u7e68\u7e69\u7e6a\u7e6b\u7e6c\u7e6d\u7e6e\u7e6f", - "\u7e70\u7e71\u7e72\u7e73\u7e74\u7e75\u7e76\u7e77\u7e78\u7e79\u7e7a\u7e7b\u7e7c\u7e7d\u7e7e\u7e7f", - "\u7e80\u7e81\u7e82\u7e83\u7e84\u7e85\u7e86\u7e87\u7e88\u7e89\u7e8a\u7e8b\u7e8c\u7e8d\u7e8e\u7e8f", - "\u7e90\u7e91\u7e92\u7e93\u7e94\u7e95\u7e96\u7e97\u7e98\u7e99\u7e9a\u7e9b\u7e9c\u7e9d\u7e9e\u7e9f", - "\u7ea0\u7ea1\u7ea2\u7ea3\u7ea4\u7ea5\u7ea6\u7ea7\u7ea8\u7ea9\u7eaa\u7eab\u7eac\u7ead\u7eae\u7eaf", - "\u7eb0\u7eb1\u7eb2\u7eb3\u7eb4\u7eb5\u7eb6\u7eb7\u7eb8\u7eb9\u7eba\u7ebb\u7ebc\u7ebd\u7ebe\u7ebf", - "\u7ec0\u7ec1\u7ec2\u7ec3\u7ec4\u7ec5\u7ec6\u7ec7\u7ec8\u7ec9\u7eca\u7ecb\u7ecc\u7ecd\u7ece\u7ecf", - "\u7ed0\u7ed1\u7ed2\u7ed3\u7ed4\u7ed5\u7ed6\u7ed7\u7ed8\u7ed9\u7eda\u7edb\u7edc\u7edd\u7ede\u7edf", - "\u7ee0\u7ee1\u7ee2\u7ee3\u7ee4\u7ee5\u7ee6\u7ee7\u7ee8\u7ee9\u7eea\u7eeb\u7eec\u7eed\u7eee\u7eef", - "\u7ef0\u7ef1\u7ef2\u7ef3\u7ef4\u7ef5\u7ef6\u7ef7\u7ef8\u7ef9\u7efa\u7efb\u7efc\u7efd\u7efe\u7eff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_7f.png", - "ascent": %ascent%, - "chars": [ - "\u7f00\u7f01\u7f02\u7f03\u7f04\u7f05\u7f06\u7f07\u7f08\u7f09\u7f0a\u7f0b\u7f0c\u7f0d\u7f0e\u7f0f", - "\u7f10\u7f11\u7f12\u7f13\u7f14\u7f15\u7f16\u7f17\u7f18\u7f19\u7f1a\u7f1b\u7f1c\u7f1d\u7f1e\u7f1f", - "\u7f20\u7f21\u7f22\u7f23\u7f24\u7f25\u7f26\u7f27\u7f28\u7f29\u7f2a\u7f2b\u7f2c\u7f2d\u7f2e\u7f2f", - "\u7f30\u7f31\u7f32\u7f33\u7f34\u7f35\u7f36\u7f37\u7f38\u7f39\u7f3a\u7f3b\u7f3c\u7f3d\u7f3e\u7f3f", - "\u7f40\u7f41\u7f42\u7f43\u7f44\u7f45\u7f46\u7f47\u7f48\u7f49\u7f4a\u7f4b\u7f4c\u7f4d\u7f4e\u7f4f", - "\u7f50\u7f51\u7f52\u7f53\u7f54\u7f55\u7f56\u7f57\u7f58\u7f59\u7f5a\u7f5b\u7f5c\u7f5d\u7f5e\u7f5f", - "\u7f60\u7f61\u7f62\u7f63\u7f64\u7f65\u7f66\u7f67\u7f68\u7f69\u7f6a\u7f6b\u7f6c\u7f6d\u7f6e\u7f6f", - "\u7f70\u7f71\u7f72\u7f73\u7f74\u7f75\u7f76\u7f77\u7f78\u7f79\u7f7a\u7f7b\u7f7c\u7f7d\u7f7e\u7f7f", - "\u7f80\u7f81\u7f82\u7f83\u7f84\u7f85\u7f86\u7f87\u7f88\u7f89\u7f8a\u7f8b\u7f8c\u7f8d\u7f8e\u7f8f", - "\u7f90\u7f91\u7f92\u7f93\u7f94\u7f95\u7f96\u7f97\u7f98\u7f99\u7f9a\u7f9b\u7f9c\u7f9d\u7f9e\u7f9f", - "\u7fa0\u7fa1\u7fa2\u7fa3\u7fa4\u7fa5\u7fa6\u7fa7\u7fa8\u7fa9\u7faa\u7fab\u7fac\u7fad\u7fae\u7faf", - "\u7fb0\u7fb1\u7fb2\u7fb3\u7fb4\u7fb5\u7fb6\u7fb7\u7fb8\u7fb9\u7fba\u7fbb\u7fbc\u7fbd\u7fbe\u7fbf", - "\u7fc0\u7fc1\u7fc2\u7fc3\u7fc4\u7fc5\u7fc6\u7fc7\u7fc8\u7fc9\u7fca\u7fcb\u7fcc\u7fcd\u7fce\u7fcf", - "\u7fd0\u7fd1\u7fd2\u7fd3\u7fd4\u7fd5\u7fd6\u7fd7\u7fd8\u7fd9\u7fda\u7fdb\u7fdc\u7fdd\u7fde\u7fdf", - "\u7fe0\u7fe1\u7fe2\u7fe3\u7fe4\u7fe5\u7fe6\u7fe7\u7fe8\u7fe9\u7fea\u7feb\u7fec\u7fed\u7fee\u7fef", - "\u7ff0\u7ff1\u7ff2\u7ff3\u7ff4\u7ff5\u7ff6\u7ff7\u7ff8\u7ff9\u7ffa\u7ffb\u7ffc\u7ffd\u7ffe\u7fff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_80.png", - "ascent": %ascent%, - "chars": [ - "\u8000\u8001\u8002\u8003\u8004\u8005\u8006\u8007\u8008\u8009\u800a\u800b\u800c\u800d\u800e\u800f", - "\u8010\u8011\u8012\u8013\u8014\u8015\u8016\u8017\u8018\u8019\u801a\u801b\u801c\u801d\u801e\u801f", - "\u8020\u8021\u8022\u8023\u8024\u8025\u8026\u8027\u8028\u8029\u802a\u802b\u802c\u802d\u802e\u802f", - "\u8030\u8031\u8032\u8033\u8034\u8035\u8036\u8037\u8038\u8039\u803a\u803b\u803c\u803d\u803e\u803f", - "\u8040\u8041\u8042\u8043\u8044\u8045\u8046\u8047\u8048\u8049\u804a\u804b\u804c\u804d\u804e\u804f", - "\u8050\u8051\u8052\u8053\u8054\u8055\u8056\u8057\u8058\u8059\u805a\u805b\u805c\u805d\u805e\u805f", - "\u8060\u8061\u8062\u8063\u8064\u8065\u8066\u8067\u8068\u8069\u806a\u806b\u806c\u806d\u806e\u806f", - "\u8070\u8071\u8072\u8073\u8074\u8075\u8076\u8077\u8078\u8079\u807a\u807b\u807c\u807d\u807e\u807f", - "\u8080\u8081\u8082\u8083\u8084\u8085\u8086\u8087\u8088\u8089\u808a\u808b\u808c\u808d\u808e\u808f", - "\u8090\u8091\u8092\u8093\u8094\u8095\u8096\u8097\u8098\u8099\u809a\u809b\u809c\u809d\u809e\u809f", - "\u80a0\u80a1\u80a2\u80a3\u80a4\u80a5\u80a6\u80a7\u80a8\u80a9\u80aa\u80ab\u80ac\u80ad\u80ae\u80af", - "\u80b0\u80b1\u80b2\u80b3\u80b4\u80b5\u80b6\u80b7\u80b8\u80b9\u80ba\u80bb\u80bc\u80bd\u80be\u80bf", - "\u80c0\u80c1\u80c2\u80c3\u80c4\u80c5\u80c6\u80c7\u80c8\u80c9\u80ca\u80cb\u80cc\u80cd\u80ce\u80cf", - "\u80d0\u80d1\u80d2\u80d3\u80d4\u80d5\u80d6\u80d7\u80d8\u80d9\u80da\u80db\u80dc\u80dd\u80de\u80df", - "\u80e0\u80e1\u80e2\u80e3\u80e4\u80e5\u80e6\u80e7\u80e8\u80e9\u80ea\u80eb\u80ec\u80ed\u80ee\u80ef", - "\u80f0\u80f1\u80f2\u80f3\u80f4\u80f5\u80f6\u80f7\u80f8\u80f9\u80fa\u80fb\u80fc\u80fd\u80fe\u80ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_81.png", - "ascent": %ascent%, - "chars": [ - "\u8100\u8101\u8102\u8103\u8104\u8105\u8106\u8107\u8108\u8109\u810a\u810b\u810c\u810d\u810e\u810f", - "\u8110\u8111\u8112\u8113\u8114\u8115\u8116\u8117\u8118\u8119\u811a\u811b\u811c\u811d\u811e\u811f", - "\u8120\u8121\u8122\u8123\u8124\u8125\u8126\u8127\u8128\u8129\u812a\u812b\u812c\u812d\u812e\u812f", - "\u8130\u8131\u8132\u8133\u8134\u8135\u8136\u8137\u8138\u8139\u813a\u813b\u813c\u813d\u813e\u813f", - "\u8140\u8141\u8142\u8143\u8144\u8145\u8146\u8147\u8148\u8149\u814a\u814b\u814c\u814d\u814e\u814f", - "\u8150\u8151\u8152\u8153\u8154\u8155\u8156\u8157\u8158\u8159\u815a\u815b\u815c\u815d\u815e\u815f", - "\u8160\u8161\u8162\u8163\u8164\u8165\u8166\u8167\u8168\u8169\u816a\u816b\u816c\u816d\u816e\u816f", - "\u8170\u8171\u8172\u8173\u8174\u8175\u8176\u8177\u8178\u8179\u817a\u817b\u817c\u817d\u817e\u817f", - "\u8180\u8181\u8182\u8183\u8184\u8185\u8186\u8187\u8188\u8189\u818a\u818b\u818c\u818d\u818e\u818f", - "\u8190\u8191\u8192\u8193\u8194\u8195\u8196\u8197\u8198\u8199\u819a\u819b\u819c\u819d\u819e\u819f", - "\u81a0\u81a1\u81a2\u81a3\u81a4\u81a5\u81a6\u81a7\u81a8\u81a9\u81aa\u81ab\u81ac\u81ad\u81ae\u81af", - "\u81b0\u81b1\u81b2\u81b3\u81b4\u81b5\u81b6\u81b7\u81b8\u81b9\u81ba\u81bb\u81bc\u81bd\u81be\u81bf", - "\u81c0\u81c1\u81c2\u81c3\u81c4\u81c5\u81c6\u81c7\u81c8\u81c9\u81ca\u81cb\u81cc\u81cd\u81ce\u81cf", - "\u81d0\u81d1\u81d2\u81d3\u81d4\u81d5\u81d6\u81d7\u81d8\u81d9\u81da\u81db\u81dc\u81dd\u81de\u81df", - "\u81e0\u81e1\u81e2\u81e3\u81e4\u81e5\u81e6\u81e7\u81e8\u81e9\u81ea\u81eb\u81ec\u81ed\u81ee\u81ef", - "\u81f0\u81f1\u81f2\u81f3\u81f4\u81f5\u81f6\u81f7\u81f8\u81f9\u81fa\u81fb\u81fc\u81fd\u81fe\u81ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_82.png", - "ascent": %ascent%, - "chars": [ - "\u8200\u8201\u8202\u8203\u8204\u8205\u8206\u8207\u8208\u8209\u820a\u820b\u820c\u820d\u820e\u820f", - "\u8210\u8211\u8212\u8213\u8214\u8215\u8216\u8217\u8218\u8219\u821a\u821b\u821c\u821d\u821e\u821f", - "\u8220\u8221\u8222\u8223\u8224\u8225\u8226\u8227\u8228\u8229\u822a\u822b\u822c\u822d\u822e\u822f", - "\u8230\u8231\u8232\u8233\u8234\u8235\u8236\u8237\u8238\u8239\u823a\u823b\u823c\u823d\u823e\u823f", - "\u8240\u8241\u8242\u8243\u8244\u8245\u8246\u8247\u8248\u8249\u824a\u824b\u824c\u824d\u824e\u824f", - "\u8250\u8251\u8252\u8253\u8254\u8255\u8256\u8257\u8258\u8259\u825a\u825b\u825c\u825d\u825e\u825f", - "\u8260\u8261\u8262\u8263\u8264\u8265\u8266\u8267\u8268\u8269\u826a\u826b\u826c\u826d\u826e\u826f", - "\u8270\u8271\u8272\u8273\u8274\u8275\u8276\u8277\u8278\u8279\u827a\u827b\u827c\u827d\u827e\u827f", - "\u8280\u8281\u8282\u8283\u8284\u8285\u8286\u8287\u8288\u8289\u828a\u828b\u828c\u828d\u828e\u828f", - "\u8290\u8291\u8292\u8293\u8294\u8295\u8296\u8297\u8298\u8299\u829a\u829b\u829c\u829d\u829e\u829f", - "\u82a0\u82a1\u82a2\u82a3\u82a4\u82a5\u82a6\u82a7\u82a8\u82a9\u82aa\u82ab\u82ac\u82ad\u82ae\u82af", - "\u82b0\u82b1\u82b2\u82b3\u82b4\u82b5\u82b6\u82b7\u82b8\u82b9\u82ba\u82bb\u82bc\u82bd\u82be\u82bf", - "\u82c0\u82c1\u82c2\u82c3\u82c4\u82c5\u82c6\u82c7\u82c8\u82c9\u82ca\u82cb\u82cc\u82cd\u82ce\u82cf", - "\u82d0\u82d1\u82d2\u82d3\u82d4\u82d5\u82d6\u82d7\u82d8\u82d9\u82da\u82db\u82dc\u82dd\u82de\u82df", - "\u82e0\u82e1\u82e2\u82e3\u82e4\u82e5\u82e6\u82e7\u82e8\u82e9\u82ea\u82eb\u82ec\u82ed\u82ee\u82ef", - "\u82f0\u82f1\u82f2\u82f3\u82f4\u82f5\u82f6\u82f7\u82f8\u82f9\u82fa\u82fb\u82fc\u82fd\u82fe\u82ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_83.png", - "ascent": %ascent%, - "chars": [ - "\u8300\u8301\u8302\u8303\u8304\u8305\u8306\u8307\u8308\u8309\u830a\u830b\u830c\u830d\u830e\u830f", - "\u8310\u8311\u8312\u8313\u8314\u8315\u8316\u8317\u8318\u8319\u831a\u831b\u831c\u831d\u831e\u831f", - "\u8320\u8321\u8322\u8323\u8324\u8325\u8326\u8327\u8328\u8329\u832a\u832b\u832c\u832d\u832e\u832f", - "\u8330\u8331\u8332\u8333\u8334\u8335\u8336\u8337\u8338\u8339\u833a\u833b\u833c\u833d\u833e\u833f", - "\u8340\u8341\u8342\u8343\u8344\u8345\u8346\u8347\u8348\u8349\u834a\u834b\u834c\u834d\u834e\u834f", - "\u8350\u8351\u8352\u8353\u8354\u8355\u8356\u8357\u8358\u8359\u835a\u835b\u835c\u835d\u835e\u835f", - "\u8360\u8361\u8362\u8363\u8364\u8365\u8366\u8367\u8368\u8369\u836a\u836b\u836c\u836d\u836e\u836f", - "\u8370\u8371\u8372\u8373\u8374\u8375\u8376\u8377\u8378\u8379\u837a\u837b\u837c\u837d\u837e\u837f", - "\u8380\u8381\u8382\u8383\u8384\u8385\u8386\u8387\u8388\u8389\u838a\u838b\u838c\u838d\u838e\u838f", - "\u8390\u8391\u8392\u8393\u8394\u8395\u8396\u8397\u8398\u8399\u839a\u839b\u839c\u839d\u839e\u839f", - "\u83a0\u83a1\u83a2\u83a3\u83a4\u83a5\u83a6\u83a7\u83a8\u83a9\u83aa\u83ab\u83ac\u83ad\u83ae\u83af", - "\u83b0\u83b1\u83b2\u83b3\u83b4\u83b5\u83b6\u83b7\u83b8\u83b9\u83ba\u83bb\u83bc\u83bd\u83be\u83bf", - "\u83c0\u83c1\u83c2\u83c3\u83c4\u83c5\u83c6\u83c7\u83c8\u83c9\u83ca\u83cb\u83cc\u83cd\u83ce\u83cf", - "\u83d0\u83d1\u83d2\u83d3\u83d4\u83d5\u83d6\u83d7\u83d8\u83d9\u83da\u83db\u83dc\u83dd\u83de\u83df", - "\u83e0\u83e1\u83e2\u83e3\u83e4\u83e5\u83e6\u83e7\u83e8\u83e9\u83ea\u83eb\u83ec\u83ed\u83ee\u83ef", - "\u83f0\u83f1\u83f2\u83f3\u83f4\u83f5\u83f6\u83f7\u83f8\u83f9\u83fa\u83fb\u83fc\u83fd\u83fe\u83ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_84.png", - "ascent": %ascent%, - "chars": [ - "\u8400\u8401\u8402\u8403\u8404\u8405\u8406\u8407\u8408\u8409\u840a\u840b\u840c\u840d\u840e\u840f", - "\u8410\u8411\u8412\u8413\u8414\u8415\u8416\u8417\u8418\u8419\u841a\u841b\u841c\u841d\u841e\u841f", - "\u8420\u8421\u8422\u8423\u8424\u8425\u8426\u8427\u8428\u8429\u842a\u842b\u842c\u842d\u842e\u842f", - "\u8430\u8431\u8432\u8433\u8434\u8435\u8436\u8437\u8438\u8439\u843a\u843b\u843c\u843d\u843e\u843f", - "\u8440\u8441\u8442\u8443\u8444\u8445\u8446\u8447\u8448\u8449\u844a\u844b\u844c\u844d\u844e\u844f", - "\u8450\u8451\u8452\u8453\u8454\u8455\u8456\u8457\u8458\u8459\u845a\u845b\u845c\u845d\u845e\u845f", - "\u8460\u8461\u8462\u8463\u8464\u8465\u8466\u8467\u8468\u8469\u846a\u846b\u846c\u846d\u846e\u846f", - "\u8470\u8471\u8472\u8473\u8474\u8475\u8476\u8477\u8478\u8479\u847a\u847b\u847c\u847d\u847e\u847f", - "\u8480\u8481\u8482\u8483\u8484\u8485\u8486\u8487\u8488\u8489\u848a\u848b\u848c\u848d\u848e\u848f", - "\u8490\u8491\u8492\u8493\u8494\u8495\u8496\u8497\u8498\u8499\u849a\u849b\u849c\u849d\u849e\u849f", - "\u84a0\u84a1\u84a2\u84a3\u84a4\u84a5\u84a6\u84a7\u84a8\u84a9\u84aa\u84ab\u84ac\u84ad\u84ae\u84af", - "\u84b0\u84b1\u84b2\u84b3\u84b4\u84b5\u84b6\u84b7\u84b8\u84b9\u84ba\u84bb\u84bc\u84bd\u84be\u84bf", - "\u84c0\u84c1\u84c2\u84c3\u84c4\u84c5\u84c6\u84c7\u84c8\u84c9\u84ca\u84cb\u84cc\u84cd\u84ce\u84cf", - "\u84d0\u84d1\u84d2\u84d3\u84d4\u84d5\u84d6\u84d7\u84d8\u84d9\u84da\u84db\u84dc\u84dd\u84de\u84df", - "\u84e0\u84e1\u84e2\u84e3\u84e4\u84e5\u84e6\u84e7\u84e8\u84e9\u84ea\u84eb\u84ec\u84ed\u84ee\u84ef", - "\u84f0\u84f1\u84f2\u84f3\u84f4\u84f5\u84f6\u84f7\u84f8\u84f9\u84fa\u84fb\u84fc\u84fd\u84fe\u84ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_85.png", - "ascent": %ascent%, - "chars": [ - "\u8500\u8501\u8502\u8503\u8504\u8505\u8506\u8507\u8508\u8509\u850a\u850b\u850c\u850d\u850e\u850f", - "\u8510\u8511\u8512\u8513\u8514\u8515\u8516\u8517\u8518\u8519\u851a\u851b\u851c\u851d\u851e\u851f", - "\u8520\u8521\u8522\u8523\u8524\u8525\u8526\u8527\u8528\u8529\u852a\u852b\u852c\u852d\u852e\u852f", - "\u8530\u8531\u8532\u8533\u8534\u8535\u8536\u8537\u8538\u8539\u853a\u853b\u853c\u853d\u853e\u853f", - "\u8540\u8541\u8542\u8543\u8544\u8545\u8546\u8547\u8548\u8549\u854a\u854b\u854c\u854d\u854e\u854f", - "\u8550\u8551\u8552\u8553\u8554\u8555\u8556\u8557\u8558\u8559\u855a\u855b\u855c\u855d\u855e\u855f", - "\u8560\u8561\u8562\u8563\u8564\u8565\u8566\u8567\u8568\u8569\u856a\u856b\u856c\u856d\u856e\u856f", - "\u8570\u8571\u8572\u8573\u8574\u8575\u8576\u8577\u8578\u8579\u857a\u857b\u857c\u857d\u857e\u857f", - "\u8580\u8581\u8582\u8583\u8584\u8585\u8586\u8587\u8588\u8589\u858a\u858b\u858c\u858d\u858e\u858f", - "\u8590\u8591\u8592\u8593\u8594\u8595\u8596\u8597\u8598\u8599\u859a\u859b\u859c\u859d\u859e\u859f", - "\u85a0\u85a1\u85a2\u85a3\u85a4\u85a5\u85a6\u85a7\u85a8\u85a9\u85aa\u85ab\u85ac\u85ad\u85ae\u85af", - "\u85b0\u85b1\u85b2\u85b3\u85b4\u85b5\u85b6\u85b7\u85b8\u85b9\u85ba\u85bb\u85bc\u85bd\u85be\u85bf", - "\u85c0\u85c1\u85c2\u85c3\u85c4\u85c5\u85c6\u85c7\u85c8\u85c9\u85ca\u85cb\u85cc\u85cd\u85ce\u85cf", - "\u85d0\u85d1\u85d2\u85d3\u85d4\u85d5\u85d6\u85d7\u85d8\u85d9\u85da\u85db\u85dc\u85dd\u85de\u85df", - "\u85e0\u85e1\u85e2\u85e3\u85e4\u85e5\u85e6\u85e7\u85e8\u85e9\u85ea\u85eb\u85ec\u85ed\u85ee\u85ef", - "\u85f0\u85f1\u85f2\u85f3\u85f4\u85f5\u85f6\u85f7\u85f8\u85f9\u85fa\u85fb\u85fc\u85fd\u85fe\u85ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_86.png", - "ascent": %ascent%, - "chars": [ - "\u8600\u8601\u8602\u8603\u8604\u8605\u8606\u8607\u8608\u8609\u860a\u860b\u860c\u860d\u860e\u860f", - "\u8610\u8611\u8612\u8613\u8614\u8615\u8616\u8617\u8618\u8619\u861a\u861b\u861c\u861d\u861e\u861f", - "\u8620\u8621\u8622\u8623\u8624\u8625\u8626\u8627\u8628\u8629\u862a\u862b\u862c\u862d\u862e\u862f", - "\u8630\u8631\u8632\u8633\u8634\u8635\u8636\u8637\u8638\u8639\u863a\u863b\u863c\u863d\u863e\u863f", - "\u8640\u8641\u8642\u8643\u8644\u8645\u8646\u8647\u8648\u8649\u864a\u864b\u864c\u864d\u864e\u864f", - "\u8650\u8651\u8652\u8653\u8654\u8655\u8656\u8657\u8658\u8659\u865a\u865b\u865c\u865d\u865e\u865f", - "\u8660\u8661\u8662\u8663\u8664\u8665\u8666\u8667\u8668\u8669\u866a\u866b\u866c\u866d\u866e\u866f", - "\u8670\u8671\u8672\u8673\u8674\u8675\u8676\u8677\u8678\u8679\u867a\u867b\u867c\u867d\u867e\u867f", - "\u8680\u8681\u8682\u8683\u8684\u8685\u8686\u8687\u8688\u8689\u868a\u868b\u868c\u868d\u868e\u868f", - "\u8690\u8691\u8692\u8693\u8694\u8695\u8696\u8697\u8698\u8699\u869a\u869b\u869c\u869d\u869e\u869f", - "\u86a0\u86a1\u86a2\u86a3\u86a4\u86a5\u86a6\u86a7\u86a8\u86a9\u86aa\u86ab\u86ac\u86ad\u86ae\u86af", - "\u86b0\u86b1\u86b2\u86b3\u86b4\u86b5\u86b6\u86b7\u86b8\u86b9\u86ba\u86bb\u86bc\u86bd\u86be\u86bf", - "\u86c0\u86c1\u86c2\u86c3\u86c4\u86c5\u86c6\u86c7\u86c8\u86c9\u86ca\u86cb\u86cc\u86cd\u86ce\u86cf", - "\u86d0\u86d1\u86d2\u86d3\u86d4\u86d5\u86d6\u86d7\u86d8\u86d9\u86da\u86db\u86dc\u86dd\u86de\u86df", - "\u86e0\u86e1\u86e2\u86e3\u86e4\u86e5\u86e6\u86e7\u86e8\u86e9\u86ea\u86eb\u86ec\u86ed\u86ee\u86ef", - "\u86f0\u86f1\u86f2\u86f3\u86f4\u86f5\u86f6\u86f7\u86f8\u86f9\u86fa\u86fb\u86fc\u86fd\u86fe\u86ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_87.png", - "ascent": %ascent%, - "chars": [ - "\u8700\u8701\u8702\u8703\u8704\u8705\u8706\u8707\u8708\u8709\u870a\u870b\u870c\u870d\u870e\u870f", - "\u8710\u8711\u8712\u8713\u8714\u8715\u8716\u8717\u8718\u8719\u871a\u871b\u871c\u871d\u871e\u871f", - "\u8720\u8721\u8722\u8723\u8724\u8725\u8726\u8727\u8728\u8729\u872a\u872b\u872c\u872d\u872e\u872f", - "\u8730\u8731\u8732\u8733\u8734\u8735\u8736\u8737\u8738\u8739\u873a\u873b\u873c\u873d\u873e\u873f", - "\u8740\u8741\u8742\u8743\u8744\u8745\u8746\u8747\u8748\u8749\u874a\u874b\u874c\u874d\u874e\u874f", - "\u8750\u8751\u8752\u8753\u8754\u8755\u8756\u8757\u8758\u8759\u875a\u875b\u875c\u875d\u875e\u875f", - "\u8760\u8761\u8762\u8763\u8764\u8765\u8766\u8767\u8768\u8769\u876a\u876b\u876c\u876d\u876e\u876f", - "\u8770\u8771\u8772\u8773\u8774\u8775\u8776\u8777\u8778\u8779\u877a\u877b\u877c\u877d\u877e\u877f", - "\u8780\u8781\u8782\u8783\u8784\u8785\u8786\u8787\u8788\u8789\u878a\u878b\u878c\u878d\u878e\u878f", - "\u8790\u8791\u8792\u8793\u8794\u8795\u8796\u8797\u8798\u8799\u879a\u879b\u879c\u879d\u879e\u879f", - "\u87a0\u87a1\u87a2\u87a3\u87a4\u87a5\u87a6\u87a7\u87a8\u87a9\u87aa\u87ab\u87ac\u87ad\u87ae\u87af", - "\u87b0\u87b1\u87b2\u87b3\u87b4\u87b5\u87b6\u87b7\u87b8\u87b9\u87ba\u87bb\u87bc\u87bd\u87be\u87bf", - "\u87c0\u87c1\u87c2\u87c3\u87c4\u87c5\u87c6\u87c7\u87c8\u87c9\u87ca\u87cb\u87cc\u87cd\u87ce\u87cf", - "\u87d0\u87d1\u87d2\u87d3\u87d4\u87d5\u87d6\u87d7\u87d8\u87d9\u87da\u87db\u87dc\u87dd\u87de\u87df", - "\u87e0\u87e1\u87e2\u87e3\u87e4\u87e5\u87e6\u87e7\u87e8\u87e9\u87ea\u87eb\u87ec\u87ed\u87ee\u87ef", - "\u87f0\u87f1\u87f2\u87f3\u87f4\u87f5\u87f6\u87f7\u87f8\u87f9\u87fa\u87fb\u87fc\u87fd\u87fe\u87ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_88.png", - "ascent": %ascent%, - "chars": [ - "\u8800\u8801\u8802\u8803\u8804\u8805\u8806\u8807\u8808\u8809\u880a\u880b\u880c\u880d\u880e\u880f", - "\u8810\u8811\u8812\u8813\u8814\u8815\u8816\u8817\u8818\u8819\u881a\u881b\u881c\u881d\u881e\u881f", - "\u8820\u8821\u8822\u8823\u8824\u8825\u8826\u8827\u8828\u8829\u882a\u882b\u882c\u882d\u882e\u882f", - "\u8830\u8831\u8832\u8833\u8834\u8835\u8836\u8837\u8838\u8839\u883a\u883b\u883c\u883d\u883e\u883f", - "\u8840\u8841\u8842\u8843\u8844\u8845\u8846\u8847\u8848\u8849\u884a\u884b\u884c\u884d\u884e\u884f", - "\u8850\u8851\u8852\u8853\u8854\u8855\u8856\u8857\u8858\u8859\u885a\u885b\u885c\u885d\u885e\u885f", - "\u8860\u8861\u8862\u8863\u8864\u8865\u8866\u8867\u8868\u8869\u886a\u886b\u886c\u886d\u886e\u886f", - "\u8870\u8871\u8872\u8873\u8874\u8875\u8876\u8877\u8878\u8879\u887a\u887b\u887c\u887d\u887e\u887f", - "\u8880\u8881\u8882\u8883\u8884\u8885\u8886\u8887\u8888\u8889\u888a\u888b\u888c\u888d\u888e\u888f", - "\u8890\u8891\u8892\u8893\u8894\u8895\u8896\u8897\u8898\u8899\u889a\u889b\u889c\u889d\u889e\u889f", - "\u88a0\u88a1\u88a2\u88a3\u88a4\u88a5\u88a6\u88a7\u88a8\u88a9\u88aa\u88ab\u88ac\u88ad\u88ae\u88af", - "\u88b0\u88b1\u88b2\u88b3\u88b4\u88b5\u88b6\u88b7\u88b8\u88b9\u88ba\u88bb\u88bc\u88bd\u88be\u88bf", - "\u88c0\u88c1\u88c2\u88c3\u88c4\u88c5\u88c6\u88c7\u88c8\u88c9\u88ca\u88cb\u88cc\u88cd\u88ce\u88cf", - "\u88d0\u88d1\u88d2\u88d3\u88d4\u88d5\u88d6\u88d7\u88d8\u88d9\u88da\u88db\u88dc\u88dd\u88de\u88df", - "\u88e0\u88e1\u88e2\u88e3\u88e4\u88e5\u88e6\u88e7\u88e8\u88e9\u88ea\u88eb\u88ec\u88ed\u88ee\u88ef", - "\u88f0\u88f1\u88f2\u88f3\u88f4\u88f5\u88f6\u88f7\u88f8\u88f9\u88fa\u88fb\u88fc\u88fd\u88fe\u88ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_89.png", - "ascent": %ascent%, - "chars": [ - "\u8900\u8901\u8902\u8903\u8904\u8905\u8906\u8907\u8908\u8909\u890a\u890b\u890c\u890d\u890e\u890f", - "\u8910\u8911\u8912\u8913\u8914\u8915\u8916\u8917\u8918\u8919\u891a\u891b\u891c\u891d\u891e\u891f", - "\u8920\u8921\u8922\u8923\u8924\u8925\u8926\u8927\u8928\u8929\u892a\u892b\u892c\u892d\u892e\u892f", - "\u8930\u8931\u8932\u8933\u8934\u8935\u8936\u8937\u8938\u8939\u893a\u893b\u893c\u893d\u893e\u893f", - "\u8940\u8941\u8942\u8943\u8944\u8945\u8946\u8947\u8948\u8949\u894a\u894b\u894c\u894d\u894e\u894f", - "\u8950\u8951\u8952\u8953\u8954\u8955\u8956\u8957\u8958\u8959\u895a\u895b\u895c\u895d\u895e\u895f", - "\u8960\u8961\u8962\u8963\u8964\u8965\u8966\u8967\u8968\u8969\u896a\u896b\u896c\u896d\u896e\u896f", - "\u8970\u8971\u8972\u8973\u8974\u8975\u8976\u8977\u8978\u8979\u897a\u897b\u897c\u897d\u897e\u897f", - "\u8980\u8981\u8982\u8983\u8984\u8985\u8986\u8987\u8988\u8989\u898a\u898b\u898c\u898d\u898e\u898f", - "\u8990\u8991\u8992\u8993\u8994\u8995\u8996\u8997\u8998\u8999\u899a\u899b\u899c\u899d\u899e\u899f", - "\u89a0\u89a1\u89a2\u89a3\u89a4\u89a5\u89a6\u89a7\u89a8\u89a9\u89aa\u89ab\u89ac\u89ad\u89ae\u89af", - "\u89b0\u89b1\u89b2\u89b3\u89b4\u89b5\u89b6\u89b7\u89b8\u89b9\u89ba\u89bb\u89bc\u89bd\u89be\u89bf", - "\u89c0\u89c1\u89c2\u89c3\u89c4\u89c5\u89c6\u89c7\u89c8\u89c9\u89ca\u89cb\u89cc\u89cd\u89ce\u89cf", - "\u89d0\u89d1\u89d2\u89d3\u89d4\u89d5\u89d6\u89d7\u89d8\u89d9\u89da\u89db\u89dc\u89dd\u89de\u89df", - "\u89e0\u89e1\u89e2\u89e3\u89e4\u89e5\u89e6\u89e7\u89e8\u89e9\u89ea\u89eb\u89ec\u89ed\u89ee\u89ef", - "\u89f0\u89f1\u89f2\u89f3\u89f4\u89f5\u89f6\u89f7\u89f8\u89f9\u89fa\u89fb\u89fc\u89fd\u89fe\u89ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_8a.png", - "ascent": %ascent%, - "chars": [ - "\u8a00\u8a01\u8a02\u8a03\u8a04\u8a05\u8a06\u8a07\u8a08\u8a09\u8a0a\u8a0b\u8a0c\u8a0d\u8a0e\u8a0f", - "\u8a10\u8a11\u8a12\u8a13\u8a14\u8a15\u8a16\u8a17\u8a18\u8a19\u8a1a\u8a1b\u8a1c\u8a1d\u8a1e\u8a1f", - "\u8a20\u8a21\u8a22\u8a23\u8a24\u8a25\u8a26\u8a27\u8a28\u8a29\u8a2a\u8a2b\u8a2c\u8a2d\u8a2e\u8a2f", - "\u8a30\u8a31\u8a32\u8a33\u8a34\u8a35\u8a36\u8a37\u8a38\u8a39\u8a3a\u8a3b\u8a3c\u8a3d\u8a3e\u8a3f", - "\u8a40\u8a41\u8a42\u8a43\u8a44\u8a45\u8a46\u8a47\u8a48\u8a49\u8a4a\u8a4b\u8a4c\u8a4d\u8a4e\u8a4f", - "\u8a50\u8a51\u8a52\u8a53\u8a54\u8a55\u8a56\u8a57\u8a58\u8a59\u8a5a\u8a5b\u8a5c\u8a5d\u8a5e\u8a5f", - "\u8a60\u8a61\u8a62\u8a63\u8a64\u8a65\u8a66\u8a67\u8a68\u8a69\u8a6a\u8a6b\u8a6c\u8a6d\u8a6e\u8a6f", - "\u8a70\u8a71\u8a72\u8a73\u8a74\u8a75\u8a76\u8a77\u8a78\u8a79\u8a7a\u8a7b\u8a7c\u8a7d\u8a7e\u8a7f", - "\u8a80\u8a81\u8a82\u8a83\u8a84\u8a85\u8a86\u8a87\u8a88\u8a89\u8a8a\u8a8b\u8a8c\u8a8d\u8a8e\u8a8f", - "\u8a90\u8a91\u8a92\u8a93\u8a94\u8a95\u8a96\u8a97\u8a98\u8a99\u8a9a\u8a9b\u8a9c\u8a9d\u8a9e\u8a9f", - "\u8aa0\u8aa1\u8aa2\u8aa3\u8aa4\u8aa5\u8aa6\u8aa7\u8aa8\u8aa9\u8aaa\u8aab\u8aac\u8aad\u8aae\u8aaf", - "\u8ab0\u8ab1\u8ab2\u8ab3\u8ab4\u8ab5\u8ab6\u8ab7\u8ab8\u8ab9\u8aba\u8abb\u8abc\u8abd\u8abe\u8abf", - "\u8ac0\u8ac1\u8ac2\u8ac3\u8ac4\u8ac5\u8ac6\u8ac7\u8ac8\u8ac9\u8aca\u8acb\u8acc\u8acd\u8ace\u8acf", - "\u8ad0\u8ad1\u8ad2\u8ad3\u8ad4\u8ad5\u8ad6\u8ad7\u8ad8\u8ad9\u8ada\u8adb\u8adc\u8add\u8ade\u8adf", - "\u8ae0\u8ae1\u8ae2\u8ae3\u8ae4\u8ae5\u8ae6\u8ae7\u8ae8\u8ae9\u8aea\u8aeb\u8aec\u8aed\u8aee\u8aef", - "\u8af0\u8af1\u8af2\u8af3\u8af4\u8af5\u8af6\u8af7\u8af8\u8af9\u8afa\u8afb\u8afc\u8afd\u8afe\u8aff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_8b.png", - "ascent": %ascent%, - "chars": [ - "\u8b00\u8b01\u8b02\u8b03\u8b04\u8b05\u8b06\u8b07\u8b08\u8b09\u8b0a\u8b0b\u8b0c\u8b0d\u8b0e\u8b0f", - "\u8b10\u8b11\u8b12\u8b13\u8b14\u8b15\u8b16\u8b17\u8b18\u8b19\u8b1a\u8b1b\u8b1c\u8b1d\u8b1e\u8b1f", - "\u8b20\u8b21\u8b22\u8b23\u8b24\u8b25\u8b26\u8b27\u8b28\u8b29\u8b2a\u8b2b\u8b2c\u8b2d\u8b2e\u8b2f", - "\u8b30\u8b31\u8b32\u8b33\u8b34\u8b35\u8b36\u8b37\u8b38\u8b39\u8b3a\u8b3b\u8b3c\u8b3d\u8b3e\u8b3f", - "\u8b40\u8b41\u8b42\u8b43\u8b44\u8b45\u8b46\u8b47\u8b48\u8b49\u8b4a\u8b4b\u8b4c\u8b4d\u8b4e\u8b4f", - "\u8b50\u8b51\u8b52\u8b53\u8b54\u8b55\u8b56\u8b57\u8b58\u8b59\u8b5a\u8b5b\u8b5c\u8b5d\u8b5e\u8b5f", - "\u8b60\u8b61\u8b62\u8b63\u8b64\u8b65\u8b66\u8b67\u8b68\u8b69\u8b6a\u8b6b\u8b6c\u8b6d\u8b6e\u8b6f", - "\u8b70\u8b71\u8b72\u8b73\u8b74\u8b75\u8b76\u8b77\u8b78\u8b79\u8b7a\u8b7b\u8b7c\u8b7d\u8b7e\u8b7f", - "\u8b80\u8b81\u8b82\u8b83\u8b84\u8b85\u8b86\u8b87\u8b88\u8b89\u8b8a\u8b8b\u8b8c\u8b8d\u8b8e\u8b8f", - "\u8b90\u8b91\u8b92\u8b93\u8b94\u8b95\u8b96\u8b97\u8b98\u8b99\u8b9a\u8b9b\u8b9c\u8b9d\u8b9e\u8b9f", - "\u8ba0\u8ba1\u8ba2\u8ba3\u8ba4\u8ba5\u8ba6\u8ba7\u8ba8\u8ba9\u8baa\u8bab\u8bac\u8bad\u8bae\u8baf", - "\u8bb0\u8bb1\u8bb2\u8bb3\u8bb4\u8bb5\u8bb6\u8bb7\u8bb8\u8bb9\u8bba\u8bbb\u8bbc\u8bbd\u8bbe\u8bbf", - "\u8bc0\u8bc1\u8bc2\u8bc3\u8bc4\u8bc5\u8bc6\u8bc7\u8bc8\u8bc9\u8bca\u8bcb\u8bcc\u8bcd\u8bce\u8bcf", - "\u8bd0\u8bd1\u8bd2\u8bd3\u8bd4\u8bd5\u8bd6\u8bd7\u8bd8\u8bd9\u8bda\u8bdb\u8bdc\u8bdd\u8bde\u8bdf", - "\u8be0\u8be1\u8be2\u8be3\u8be4\u8be5\u8be6\u8be7\u8be8\u8be9\u8bea\u8beb\u8bec\u8bed\u8bee\u8bef", - "\u8bf0\u8bf1\u8bf2\u8bf3\u8bf4\u8bf5\u8bf6\u8bf7\u8bf8\u8bf9\u8bfa\u8bfb\u8bfc\u8bfd\u8bfe\u8bff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_8c.png", - "ascent": %ascent%, - "chars": [ - "\u8c00\u8c01\u8c02\u8c03\u8c04\u8c05\u8c06\u8c07\u8c08\u8c09\u8c0a\u8c0b\u8c0c\u8c0d\u8c0e\u8c0f", - "\u8c10\u8c11\u8c12\u8c13\u8c14\u8c15\u8c16\u8c17\u8c18\u8c19\u8c1a\u8c1b\u8c1c\u8c1d\u8c1e\u8c1f", - "\u8c20\u8c21\u8c22\u8c23\u8c24\u8c25\u8c26\u8c27\u8c28\u8c29\u8c2a\u8c2b\u8c2c\u8c2d\u8c2e\u8c2f", - "\u8c30\u8c31\u8c32\u8c33\u8c34\u8c35\u8c36\u8c37\u8c38\u8c39\u8c3a\u8c3b\u8c3c\u8c3d\u8c3e\u8c3f", - "\u8c40\u8c41\u8c42\u8c43\u8c44\u8c45\u8c46\u8c47\u8c48\u8c49\u8c4a\u8c4b\u8c4c\u8c4d\u8c4e\u8c4f", - "\u8c50\u8c51\u8c52\u8c53\u8c54\u8c55\u8c56\u8c57\u8c58\u8c59\u8c5a\u8c5b\u8c5c\u8c5d\u8c5e\u8c5f", - "\u8c60\u8c61\u8c62\u8c63\u8c64\u8c65\u8c66\u8c67\u8c68\u8c69\u8c6a\u8c6b\u8c6c\u8c6d\u8c6e\u8c6f", - "\u8c70\u8c71\u8c72\u8c73\u8c74\u8c75\u8c76\u8c77\u8c78\u8c79\u8c7a\u8c7b\u8c7c\u8c7d\u8c7e\u8c7f", - "\u8c80\u8c81\u8c82\u8c83\u8c84\u8c85\u8c86\u8c87\u8c88\u8c89\u8c8a\u8c8b\u8c8c\u8c8d\u8c8e\u8c8f", - "\u8c90\u8c91\u8c92\u8c93\u8c94\u8c95\u8c96\u8c97\u8c98\u8c99\u8c9a\u8c9b\u8c9c\u8c9d\u8c9e\u8c9f", - "\u8ca0\u8ca1\u8ca2\u8ca3\u8ca4\u8ca5\u8ca6\u8ca7\u8ca8\u8ca9\u8caa\u8cab\u8cac\u8cad\u8cae\u8caf", - "\u8cb0\u8cb1\u8cb2\u8cb3\u8cb4\u8cb5\u8cb6\u8cb7\u8cb8\u8cb9\u8cba\u8cbb\u8cbc\u8cbd\u8cbe\u8cbf", - "\u8cc0\u8cc1\u8cc2\u8cc3\u8cc4\u8cc5\u8cc6\u8cc7\u8cc8\u8cc9\u8cca\u8ccb\u8ccc\u8ccd\u8cce\u8ccf", - "\u8cd0\u8cd1\u8cd2\u8cd3\u8cd4\u8cd5\u8cd6\u8cd7\u8cd8\u8cd9\u8cda\u8cdb\u8cdc\u8cdd\u8cde\u8cdf", - "\u8ce0\u8ce1\u8ce2\u8ce3\u8ce4\u8ce5\u8ce6\u8ce7\u8ce8\u8ce9\u8cea\u8ceb\u8cec\u8ced\u8cee\u8cef", - "\u8cf0\u8cf1\u8cf2\u8cf3\u8cf4\u8cf5\u8cf6\u8cf7\u8cf8\u8cf9\u8cfa\u8cfb\u8cfc\u8cfd\u8cfe\u8cff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_8d.png", - "ascent": %ascent%, - "chars": [ - "\u8d00\u8d01\u8d02\u8d03\u8d04\u8d05\u8d06\u8d07\u8d08\u8d09\u8d0a\u8d0b\u8d0c\u8d0d\u8d0e\u8d0f", - "\u8d10\u8d11\u8d12\u8d13\u8d14\u8d15\u8d16\u8d17\u8d18\u8d19\u8d1a\u8d1b\u8d1c\u8d1d\u8d1e\u8d1f", - "\u8d20\u8d21\u8d22\u8d23\u8d24\u8d25\u8d26\u8d27\u8d28\u8d29\u8d2a\u8d2b\u8d2c\u8d2d\u8d2e\u8d2f", - "\u8d30\u8d31\u8d32\u8d33\u8d34\u8d35\u8d36\u8d37\u8d38\u8d39\u8d3a\u8d3b\u8d3c\u8d3d\u8d3e\u8d3f", - "\u8d40\u8d41\u8d42\u8d43\u8d44\u8d45\u8d46\u8d47\u8d48\u8d49\u8d4a\u8d4b\u8d4c\u8d4d\u8d4e\u8d4f", - "\u8d50\u8d51\u8d52\u8d53\u8d54\u8d55\u8d56\u8d57\u8d58\u8d59\u8d5a\u8d5b\u8d5c\u8d5d\u8d5e\u8d5f", - "\u8d60\u8d61\u8d62\u8d63\u8d64\u8d65\u8d66\u8d67\u8d68\u8d69\u8d6a\u8d6b\u8d6c\u8d6d\u8d6e\u8d6f", - "\u8d70\u8d71\u8d72\u8d73\u8d74\u8d75\u8d76\u8d77\u8d78\u8d79\u8d7a\u8d7b\u8d7c\u8d7d\u8d7e\u8d7f", - "\u8d80\u8d81\u8d82\u8d83\u8d84\u8d85\u8d86\u8d87\u8d88\u8d89\u8d8a\u8d8b\u8d8c\u8d8d\u8d8e\u8d8f", - "\u8d90\u8d91\u8d92\u8d93\u8d94\u8d95\u8d96\u8d97\u8d98\u8d99\u8d9a\u8d9b\u8d9c\u8d9d\u8d9e\u8d9f", - "\u8da0\u8da1\u8da2\u8da3\u8da4\u8da5\u8da6\u8da7\u8da8\u8da9\u8daa\u8dab\u8dac\u8dad\u8dae\u8daf", - "\u8db0\u8db1\u8db2\u8db3\u8db4\u8db5\u8db6\u8db7\u8db8\u8db9\u8dba\u8dbb\u8dbc\u8dbd\u8dbe\u8dbf", - "\u8dc0\u8dc1\u8dc2\u8dc3\u8dc4\u8dc5\u8dc6\u8dc7\u8dc8\u8dc9\u8dca\u8dcb\u8dcc\u8dcd\u8dce\u8dcf", - "\u8dd0\u8dd1\u8dd2\u8dd3\u8dd4\u8dd5\u8dd6\u8dd7\u8dd8\u8dd9\u8dda\u8ddb\u8ddc\u8ddd\u8dde\u8ddf", - "\u8de0\u8de1\u8de2\u8de3\u8de4\u8de5\u8de6\u8de7\u8de8\u8de9\u8dea\u8deb\u8dec\u8ded\u8dee\u8def", - "\u8df0\u8df1\u8df2\u8df3\u8df4\u8df5\u8df6\u8df7\u8df8\u8df9\u8dfa\u8dfb\u8dfc\u8dfd\u8dfe\u8dff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_8e.png", - "ascent": %ascent%, - "chars": [ - "\u8e00\u8e01\u8e02\u8e03\u8e04\u8e05\u8e06\u8e07\u8e08\u8e09\u8e0a\u8e0b\u8e0c\u8e0d\u8e0e\u8e0f", - "\u8e10\u8e11\u8e12\u8e13\u8e14\u8e15\u8e16\u8e17\u8e18\u8e19\u8e1a\u8e1b\u8e1c\u8e1d\u8e1e\u8e1f", - "\u8e20\u8e21\u8e22\u8e23\u8e24\u8e25\u8e26\u8e27\u8e28\u8e29\u8e2a\u8e2b\u8e2c\u8e2d\u8e2e\u8e2f", - "\u8e30\u8e31\u8e32\u8e33\u8e34\u8e35\u8e36\u8e37\u8e38\u8e39\u8e3a\u8e3b\u8e3c\u8e3d\u8e3e\u8e3f", - "\u8e40\u8e41\u8e42\u8e43\u8e44\u8e45\u8e46\u8e47\u8e48\u8e49\u8e4a\u8e4b\u8e4c\u8e4d\u8e4e\u8e4f", - "\u8e50\u8e51\u8e52\u8e53\u8e54\u8e55\u8e56\u8e57\u8e58\u8e59\u8e5a\u8e5b\u8e5c\u8e5d\u8e5e\u8e5f", - "\u8e60\u8e61\u8e62\u8e63\u8e64\u8e65\u8e66\u8e67\u8e68\u8e69\u8e6a\u8e6b\u8e6c\u8e6d\u8e6e\u8e6f", - "\u8e70\u8e71\u8e72\u8e73\u8e74\u8e75\u8e76\u8e77\u8e78\u8e79\u8e7a\u8e7b\u8e7c\u8e7d\u8e7e\u8e7f", - "\u8e80\u8e81\u8e82\u8e83\u8e84\u8e85\u8e86\u8e87\u8e88\u8e89\u8e8a\u8e8b\u8e8c\u8e8d\u8e8e\u8e8f", - "\u8e90\u8e91\u8e92\u8e93\u8e94\u8e95\u8e96\u8e97\u8e98\u8e99\u8e9a\u8e9b\u8e9c\u8e9d\u8e9e\u8e9f", - "\u8ea0\u8ea1\u8ea2\u8ea3\u8ea4\u8ea5\u8ea6\u8ea7\u8ea8\u8ea9\u8eaa\u8eab\u8eac\u8ead\u8eae\u8eaf", - "\u8eb0\u8eb1\u8eb2\u8eb3\u8eb4\u8eb5\u8eb6\u8eb7\u8eb8\u8eb9\u8eba\u8ebb\u8ebc\u8ebd\u8ebe\u8ebf", - "\u8ec0\u8ec1\u8ec2\u8ec3\u8ec4\u8ec5\u8ec6\u8ec7\u8ec8\u8ec9\u8eca\u8ecb\u8ecc\u8ecd\u8ece\u8ecf", - "\u8ed0\u8ed1\u8ed2\u8ed3\u8ed4\u8ed5\u8ed6\u8ed7\u8ed8\u8ed9\u8eda\u8edb\u8edc\u8edd\u8ede\u8edf", - "\u8ee0\u8ee1\u8ee2\u8ee3\u8ee4\u8ee5\u8ee6\u8ee7\u8ee8\u8ee9\u8eea\u8eeb\u8eec\u8eed\u8eee\u8eef", - "\u8ef0\u8ef1\u8ef2\u8ef3\u8ef4\u8ef5\u8ef6\u8ef7\u8ef8\u8ef9\u8efa\u8efb\u8efc\u8efd\u8efe\u8eff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_8f.png", - "ascent": %ascent%, - "chars": [ - "\u8f00\u8f01\u8f02\u8f03\u8f04\u8f05\u8f06\u8f07\u8f08\u8f09\u8f0a\u8f0b\u8f0c\u8f0d\u8f0e\u8f0f", - "\u8f10\u8f11\u8f12\u8f13\u8f14\u8f15\u8f16\u8f17\u8f18\u8f19\u8f1a\u8f1b\u8f1c\u8f1d\u8f1e\u8f1f", - "\u8f20\u8f21\u8f22\u8f23\u8f24\u8f25\u8f26\u8f27\u8f28\u8f29\u8f2a\u8f2b\u8f2c\u8f2d\u8f2e\u8f2f", - "\u8f30\u8f31\u8f32\u8f33\u8f34\u8f35\u8f36\u8f37\u8f38\u8f39\u8f3a\u8f3b\u8f3c\u8f3d\u8f3e\u8f3f", - "\u8f40\u8f41\u8f42\u8f43\u8f44\u8f45\u8f46\u8f47\u8f48\u8f49\u8f4a\u8f4b\u8f4c\u8f4d\u8f4e\u8f4f", - "\u8f50\u8f51\u8f52\u8f53\u8f54\u8f55\u8f56\u8f57\u8f58\u8f59\u8f5a\u8f5b\u8f5c\u8f5d\u8f5e\u8f5f", - "\u8f60\u8f61\u8f62\u8f63\u8f64\u8f65\u8f66\u8f67\u8f68\u8f69\u8f6a\u8f6b\u8f6c\u8f6d\u8f6e\u8f6f", - "\u8f70\u8f71\u8f72\u8f73\u8f74\u8f75\u8f76\u8f77\u8f78\u8f79\u8f7a\u8f7b\u8f7c\u8f7d\u8f7e\u8f7f", - "\u8f80\u8f81\u8f82\u8f83\u8f84\u8f85\u8f86\u8f87\u8f88\u8f89\u8f8a\u8f8b\u8f8c\u8f8d\u8f8e\u8f8f", - "\u8f90\u8f91\u8f92\u8f93\u8f94\u8f95\u8f96\u8f97\u8f98\u8f99\u8f9a\u8f9b\u8f9c\u8f9d\u8f9e\u8f9f", - "\u8fa0\u8fa1\u8fa2\u8fa3\u8fa4\u8fa5\u8fa6\u8fa7\u8fa8\u8fa9\u8faa\u8fab\u8fac\u8fad\u8fae\u8faf", - "\u8fb0\u8fb1\u8fb2\u8fb3\u8fb4\u8fb5\u8fb6\u8fb7\u8fb8\u8fb9\u8fba\u8fbb\u8fbc\u8fbd\u8fbe\u8fbf", - "\u8fc0\u8fc1\u8fc2\u8fc3\u8fc4\u8fc5\u8fc6\u8fc7\u8fc8\u8fc9\u8fca\u8fcb\u8fcc\u8fcd\u8fce\u8fcf", - "\u8fd0\u8fd1\u8fd2\u8fd3\u8fd4\u8fd5\u8fd6\u8fd7\u8fd8\u8fd9\u8fda\u8fdb\u8fdc\u8fdd\u8fde\u8fdf", - "\u8fe0\u8fe1\u8fe2\u8fe3\u8fe4\u8fe5\u8fe6\u8fe7\u8fe8\u8fe9\u8fea\u8feb\u8fec\u8fed\u8fee\u8fef", - "\u8ff0\u8ff1\u8ff2\u8ff3\u8ff4\u8ff5\u8ff6\u8ff7\u8ff8\u8ff9\u8ffa\u8ffb\u8ffc\u8ffd\u8ffe\u8fff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_90.png", - "ascent": %ascent%, - "chars": [ - "\u9000\u9001\u9002\u9003\u9004\u9005\u9006\u9007\u9008\u9009\u900a\u900b\u900c\u900d\u900e\u900f", - "\u9010\u9011\u9012\u9013\u9014\u9015\u9016\u9017\u9018\u9019\u901a\u901b\u901c\u901d\u901e\u901f", - "\u9020\u9021\u9022\u9023\u9024\u9025\u9026\u9027\u9028\u9029\u902a\u902b\u902c\u902d\u902e\u902f", - "\u9030\u9031\u9032\u9033\u9034\u9035\u9036\u9037\u9038\u9039\u903a\u903b\u903c\u903d\u903e\u903f", - "\u9040\u9041\u9042\u9043\u9044\u9045\u9046\u9047\u9048\u9049\u904a\u904b\u904c\u904d\u904e\u904f", - "\u9050\u9051\u9052\u9053\u9054\u9055\u9056\u9057\u9058\u9059\u905a\u905b\u905c\u905d\u905e\u905f", - "\u9060\u9061\u9062\u9063\u9064\u9065\u9066\u9067\u9068\u9069\u906a\u906b\u906c\u906d\u906e\u906f", - "\u9070\u9071\u9072\u9073\u9074\u9075\u9076\u9077\u9078\u9079\u907a\u907b\u907c\u907d\u907e\u907f", - "\u9080\u9081\u9082\u9083\u9084\u9085\u9086\u9087\u9088\u9089\u908a\u908b\u908c\u908d\u908e\u908f", - "\u9090\u9091\u9092\u9093\u9094\u9095\u9096\u9097\u9098\u9099\u909a\u909b\u909c\u909d\u909e\u909f", - "\u90a0\u90a1\u90a2\u90a3\u90a4\u90a5\u90a6\u90a7\u90a8\u90a9\u90aa\u90ab\u90ac\u90ad\u90ae\u90af", - "\u90b0\u90b1\u90b2\u90b3\u90b4\u90b5\u90b6\u90b7\u90b8\u90b9\u90ba\u90bb\u90bc\u90bd\u90be\u90bf", - "\u90c0\u90c1\u90c2\u90c3\u90c4\u90c5\u90c6\u90c7\u90c8\u90c9\u90ca\u90cb\u90cc\u90cd\u90ce\u90cf", - "\u90d0\u90d1\u90d2\u90d3\u90d4\u90d5\u90d6\u90d7\u90d8\u90d9\u90da\u90db\u90dc\u90dd\u90de\u90df", - "\u90e0\u90e1\u90e2\u90e3\u90e4\u90e5\u90e6\u90e7\u90e8\u90e9\u90ea\u90eb\u90ec\u90ed\u90ee\u90ef", - "\u90f0\u90f1\u90f2\u90f3\u90f4\u90f5\u90f6\u90f7\u90f8\u90f9\u90fa\u90fb\u90fc\u90fd\u90fe\u90ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_91.png", - "ascent": %ascent%, - "chars": [ - "\u9100\u9101\u9102\u9103\u9104\u9105\u9106\u9107\u9108\u9109\u910a\u910b\u910c\u910d\u910e\u910f", - "\u9110\u9111\u9112\u9113\u9114\u9115\u9116\u9117\u9118\u9119\u911a\u911b\u911c\u911d\u911e\u911f", - "\u9120\u9121\u9122\u9123\u9124\u9125\u9126\u9127\u9128\u9129\u912a\u912b\u912c\u912d\u912e\u912f", - "\u9130\u9131\u9132\u9133\u9134\u9135\u9136\u9137\u9138\u9139\u913a\u913b\u913c\u913d\u913e\u913f", - "\u9140\u9141\u9142\u9143\u9144\u9145\u9146\u9147\u9148\u9149\u914a\u914b\u914c\u914d\u914e\u914f", - "\u9150\u9151\u9152\u9153\u9154\u9155\u9156\u9157\u9158\u9159\u915a\u915b\u915c\u915d\u915e\u915f", - "\u9160\u9161\u9162\u9163\u9164\u9165\u9166\u9167\u9168\u9169\u916a\u916b\u916c\u916d\u916e\u916f", - "\u9170\u9171\u9172\u9173\u9174\u9175\u9176\u9177\u9178\u9179\u917a\u917b\u917c\u917d\u917e\u917f", - "\u9180\u9181\u9182\u9183\u9184\u9185\u9186\u9187\u9188\u9189\u918a\u918b\u918c\u918d\u918e\u918f", - "\u9190\u9191\u9192\u9193\u9194\u9195\u9196\u9197\u9198\u9199\u919a\u919b\u919c\u919d\u919e\u919f", - "\u91a0\u91a1\u91a2\u91a3\u91a4\u91a5\u91a6\u91a7\u91a8\u91a9\u91aa\u91ab\u91ac\u91ad\u91ae\u91af", - "\u91b0\u91b1\u91b2\u91b3\u91b4\u91b5\u91b6\u91b7\u91b8\u91b9\u91ba\u91bb\u91bc\u91bd\u91be\u91bf", - "\u91c0\u91c1\u91c2\u91c3\u91c4\u91c5\u91c6\u91c7\u91c8\u91c9\u91ca\u91cb\u91cc\u91cd\u91ce\u91cf", - "\u91d0\u91d1\u91d2\u91d3\u91d4\u91d5\u91d6\u91d7\u91d8\u91d9\u91da\u91db\u91dc\u91dd\u91de\u91df", - "\u91e0\u91e1\u91e2\u91e3\u91e4\u91e5\u91e6\u91e7\u91e8\u91e9\u91ea\u91eb\u91ec\u91ed\u91ee\u91ef", - "\u91f0\u91f1\u91f2\u91f3\u91f4\u91f5\u91f6\u91f7\u91f8\u91f9\u91fa\u91fb\u91fc\u91fd\u91fe\u91ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_92.png", - "ascent": %ascent%, - "chars": [ - "\u9200\u9201\u9202\u9203\u9204\u9205\u9206\u9207\u9208\u9209\u920a\u920b\u920c\u920d\u920e\u920f", - "\u9210\u9211\u9212\u9213\u9214\u9215\u9216\u9217\u9218\u9219\u921a\u921b\u921c\u921d\u921e\u921f", - "\u9220\u9221\u9222\u9223\u9224\u9225\u9226\u9227\u9228\u9229\u922a\u922b\u922c\u922d\u922e\u922f", - "\u9230\u9231\u9232\u9233\u9234\u9235\u9236\u9237\u9238\u9239\u923a\u923b\u923c\u923d\u923e\u923f", - "\u9240\u9241\u9242\u9243\u9244\u9245\u9246\u9247\u9248\u9249\u924a\u924b\u924c\u924d\u924e\u924f", - "\u9250\u9251\u9252\u9253\u9254\u9255\u9256\u9257\u9258\u9259\u925a\u925b\u925c\u925d\u925e\u925f", - "\u9260\u9261\u9262\u9263\u9264\u9265\u9266\u9267\u9268\u9269\u926a\u926b\u926c\u926d\u926e\u926f", - "\u9270\u9271\u9272\u9273\u9274\u9275\u9276\u9277\u9278\u9279\u927a\u927b\u927c\u927d\u927e\u927f", - "\u9280\u9281\u9282\u9283\u9284\u9285\u9286\u9287\u9288\u9289\u928a\u928b\u928c\u928d\u928e\u928f", - "\u9290\u9291\u9292\u9293\u9294\u9295\u9296\u9297\u9298\u9299\u929a\u929b\u929c\u929d\u929e\u929f", - "\u92a0\u92a1\u92a2\u92a3\u92a4\u92a5\u92a6\u92a7\u92a8\u92a9\u92aa\u92ab\u92ac\u92ad\u92ae\u92af", - "\u92b0\u92b1\u92b2\u92b3\u92b4\u92b5\u92b6\u92b7\u92b8\u92b9\u92ba\u92bb\u92bc\u92bd\u92be\u92bf", - "\u92c0\u92c1\u92c2\u92c3\u92c4\u92c5\u92c6\u92c7\u92c8\u92c9\u92ca\u92cb\u92cc\u92cd\u92ce\u92cf", - "\u92d0\u92d1\u92d2\u92d3\u92d4\u92d5\u92d6\u92d7\u92d8\u92d9\u92da\u92db\u92dc\u92dd\u92de\u92df", - "\u92e0\u92e1\u92e2\u92e3\u92e4\u92e5\u92e6\u92e7\u92e8\u92e9\u92ea\u92eb\u92ec\u92ed\u92ee\u92ef", - "\u92f0\u92f1\u92f2\u92f3\u92f4\u92f5\u92f6\u92f7\u92f8\u92f9\u92fa\u92fb\u92fc\u92fd\u92fe\u92ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_93.png", - "ascent": %ascent%, - "chars": [ - "\u9300\u9301\u9302\u9303\u9304\u9305\u9306\u9307\u9308\u9309\u930a\u930b\u930c\u930d\u930e\u930f", - "\u9310\u9311\u9312\u9313\u9314\u9315\u9316\u9317\u9318\u9319\u931a\u931b\u931c\u931d\u931e\u931f", - "\u9320\u9321\u9322\u9323\u9324\u9325\u9326\u9327\u9328\u9329\u932a\u932b\u932c\u932d\u932e\u932f", - "\u9330\u9331\u9332\u9333\u9334\u9335\u9336\u9337\u9338\u9339\u933a\u933b\u933c\u933d\u933e\u933f", - "\u9340\u9341\u9342\u9343\u9344\u9345\u9346\u9347\u9348\u9349\u934a\u934b\u934c\u934d\u934e\u934f", - "\u9350\u9351\u9352\u9353\u9354\u9355\u9356\u9357\u9358\u9359\u935a\u935b\u935c\u935d\u935e\u935f", - "\u9360\u9361\u9362\u9363\u9364\u9365\u9366\u9367\u9368\u9369\u936a\u936b\u936c\u936d\u936e\u936f", - "\u9370\u9371\u9372\u9373\u9374\u9375\u9376\u9377\u9378\u9379\u937a\u937b\u937c\u937d\u937e\u937f", - "\u9380\u9381\u9382\u9383\u9384\u9385\u9386\u9387\u9388\u9389\u938a\u938b\u938c\u938d\u938e\u938f", - "\u9390\u9391\u9392\u9393\u9394\u9395\u9396\u9397\u9398\u9399\u939a\u939b\u939c\u939d\u939e\u939f", - "\u93a0\u93a1\u93a2\u93a3\u93a4\u93a5\u93a6\u93a7\u93a8\u93a9\u93aa\u93ab\u93ac\u93ad\u93ae\u93af", - "\u93b0\u93b1\u93b2\u93b3\u93b4\u93b5\u93b6\u93b7\u93b8\u93b9\u93ba\u93bb\u93bc\u93bd\u93be\u93bf", - "\u93c0\u93c1\u93c2\u93c3\u93c4\u93c5\u93c6\u93c7\u93c8\u93c9\u93ca\u93cb\u93cc\u93cd\u93ce\u93cf", - "\u93d0\u93d1\u93d2\u93d3\u93d4\u93d5\u93d6\u93d7\u93d8\u93d9\u93da\u93db\u93dc\u93dd\u93de\u93df", - "\u93e0\u93e1\u93e2\u93e3\u93e4\u93e5\u93e6\u93e7\u93e8\u93e9\u93ea\u93eb\u93ec\u93ed\u93ee\u93ef", - "\u93f0\u93f1\u93f2\u93f3\u93f4\u93f5\u93f6\u93f7\u93f8\u93f9\u93fa\u93fb\u93fc\u93fd\u93fe\u93ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_94.png", - "ascent": %ascent%, - "chars": [ - "\u9400\u9401\u9402\u9403\u9404\u9405\u9406\u9407\u9408\u9409\u940a\u940b\u940c\u940d\u940e\u940f", - "\u9410\u9411\u9412\u9413\u9414\u9415\u9416\u9417\u9418\u9419\u941a\u941b\u941c\u941d\u941e\u941f", - "\u9420\u9421\u9422\u9423\u9424\u9425\u9426\u9427\u9428\u9429\u942a\u942b\u942c\u942d\u942e\u942f", - "\u9430\u9431\u9432\u9433\u9434\u9435\u9436\u9437\u9438\u9439\u943a\u943b\u943c\u943d\u943e\u943f", - "\u9440\u9441\u9442\u9443\u9444\u9445\u9446\u9447\u9448\u9449\u944a\u944b\u944c\u944d\u944e\u944f", - "\u9450\u9451\u9452\u9453\u9454\u9455\u9456\u9457\u9458\u9459\u945a\u945b\u945c\u945d\u945e\u945f", - "\u9460\u9461\u9462\u9463\u9464\u9465\u9466\u9467\u9468\u9469\u946a\u946b\u946c\u946d\u946e\u946f", - "\u9470\u9471\u9472\u9473\u9474\u9475\u9476\u9477\u9478\u9479\u947a\u947b\u947c\u947d\u947e\u947f", - "\u9480\u9481\u9482\u9483\u9484\u9485\u9486\u9487\u9488\u9489\u948a\u948b\u948c\u948d\u948e\u948f", - "\u9490\u9491\u9492\u9493\u9494\u9495\u9496\u9497\u9498\u9499\u949a\u949b\u949c\u949d\u949e\u949f", - "\u94a0\u94a1\u94a2\u94a3\u94a4\u94a5\u94a6\u94a7\u94a8\u94a9\u94aa\u94ab\u94ac\u94ad\u94ae\u94af", - "\u94b0\u94b1\u94b2\u94b3\u94b4\u94b5\u94b6\u94b7\u94b8\u94b9\u94ba\u94bb\u94bc\u94bd\u94be\u94bf", - "\u94c0\u94c1\u94c2\u94c3\u94c4\u94c5\u94c6\u94c7\u94c8\u94c9\u94ca\u94cb\u94cc\u94cd\u94ce\u94cf", - "\u94d0\u94d1\u94d2\u94d3\u94d4\u94d5\u94d6\u94d7\u94d8\u94d9\u94da\u94db\u94dc\u94dd\u94de\u94df", - "\u94e0\u94e1\u94e2\u94e3\u94e4\u94e5\u94e6\u94e7\u94e8\u94e9\u94ea\u94eb\u94ec\u94ed\u94ee\u94ef", - "\u94f0\u94f1\u94f2\u94f3\u94f4\u94f5\u94f6\u94f7\u94f8\u94f9\u94fa\u94fb\u94fc\u94fd\u94fe\u94ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_95.png", - "ascent": %ascent%, - "chars": [ - "\u9500\u9501\u9502\u9503\u9504\u9505\u9506\u9507\u9508\u9509\u950a\u950b\u950c\u950d\u950e\u950f", - "\u9510\u9511\u9512\u9513\u9514\u9515\u9516\u9517\u9518\u9519\u951a\u951b\u951c\u951d\u951e\u951f", - "\u9520\u9521\u9522\u9523\u9524\u9525\u9526\u9527\u9528\u9529\u952a\u952b\u952c\u952d\u952e\u952f", - "\u9530\u9531\u9532\u9533\u9534\u9535\u9536\u9537\u9538\u9539\u953a\u953b\u953c\u953d\u953e\u953f", - "\u9540\u9541\u9542\u9543\u9544\u9545\u9546\u9547\u9548\u9549\u954a\u954b\u954c\u954d\u954e\u954f", - "\u9550\u9551\u9552\u9553\u9554\u9555\u9556\u9557\u9558\u9559\u955a\u955b\u955c\u955d\u955e\u955f", - "\u9560\u9561\u9562\u9563\u9564\u9565\u9566\u9567\u9568\u9569\u956a\u956b\u956c\u956d\u956e\u956f", - "\u9570\u9571\u9572\u9573\u9574\u9575\u9576\u9577\u9578\u9579\u957a\u957b\u957c\u957d\u957e\u957f", - "\u9580\u9581\u9582\u9583\u9584\u9585\u9586\u9587\u9588\u9589\u958a\u958b\u958c\u958d\u958e\u958f", - "\u9590\u9591\u9592\u9593\u9594\u9595\u9596\u9597\u9598\u9599\u959a\u959b\u959c\u959d\u959e\u959f", - "\u95a0\u95a1\u95a2\u95a3\u95a4\u95a5\u95a6\u95a7\u95a8\u95a9\u95aa\u95ab\u95ac\u95ad\u95ae\u95af", - "\u95b0\u95b1\u95b2\u95b3\u95b4\u95b5\u95b6\u95b7\u95b8\u95b9\u95ba\u95bb\u95bc\u95bd\u95be\u95bf", - "\u95c0\u95c1\u95c2\u95c3\u95c4\u95c5\u95c6\u95c7\u95c8\u95c9\u95ca\u95cb\u95cc\u95cd\u95ce\u95cf", - "\u95d0\u95d1\u95d2\u95d3\u95d4\u95d5\u95d6\u95d7\u95d8\u95d9\u95da\u95db\u95dc\u95dd\u95de\u95df", - "\u95e0\u95e1\u95e2\u95e3\u95e4\u95e5\u95e6\u95e7\u95e8\u95e9\u95ea\u95eb\u95ec\u95ed\u95ee\u95ef", - "\u95f0\u95f1\u95f2\u95f3\u95f4\u95f5\u95f6\u95f7\u95f8\u95f9\u95fa\u95fb\u95fc\u95fd\u95fe\u95ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_96.png", - "ascent": %ascent%, - "chars": [ - "\u9600\u9601\u9602\u9603\u9604\u9605\u9606\u9607\u9608\u9609\u960a\u960b\u960c\u960d\u960e\u960f", - "\u9610\u9611\u9612\u9613\u9614\u9615\u9616\u9617\u9618\u9619\u961a\u961b\u961c\u961d\u961e\u961f", - "\u9620\u9621\u9622\u9623\u9624\u9625\u9626\u9627\u9628\u9629\u962a\u962b\u962c\u962d\u962e\u962f", - "\u9630\u9631\u9632\u9633\u9634\u9635\u9636\u9637\u9638\u9639\u963a\u963b\u963c\u963d\u963e\u963f", - "\u9640\u9641\u9642\u9643\u9644\u9645\u9646\u9647\u9648\u9649\u964a\u964b\u964c\u964d\u964e\u964f", - "\u9650\u9651\u9652\u9653\u9654\u9655\u9656\u9657\u9658\u9659\u965a\u965b\u965c\u965d\u965e\u965f", - "\u9660\u9661\u9662\u9663\u9664\u9665\u9666\u9667\u9668\u9669\u966a\u966b\u966c\u966d\u966e\u966f", - "\u9670\u9671\u9672\u9673\u9674\u9675\u9676\u9677\u9678\u9679\u967a\u967b\u967c\u967d\u967e\u967f", - "\u9680\u9681\u9682\u9683\u9684\u9685\u9686\u9687\u9688\u9689\u968a\u968b\u968c\u968d\u968e\u968f", - "\u9690\u9691\u9692\u9693\u9694\u9695\u9696\u9697\u9698\u9699\u969a\u969b\u969c\u969d\u969e\u969f", - "\u96a0\u96a1\u96a2\u96a3\u96a4\u96a5\u96a6\u96a7\u96a8\u96a9\u96aa\u96ab\u96ac\u96ad\u96ae\u96af", - "\u96b0\u96b1\u96b2\u96b3\u96b4\u96b5\u96b6\u96b7\u96b8\u96b9\u96ba\u96bb\u96bc\u96bd\u96be\u96bf", - "\u96c0\u96c1\u96c2\u96c3\u96c4\u96c5\u96c6\u96c7\u96c8\u96c9\u96ca\u96cb\u96cc\u96cd\u96ce\u96cf", - "\u96d0\u96d1\u96d2\u96d3\u96d4\u96d5\u96d6\u96d7\u96d8\u96d9\u96da\u96db\u96dc\u96dd\u96de\u96df", - "\u96e0\u96e1\u96e2\u96e3\u96e4\u96e5\u96e6\u96e7\u96e8\u96e9\u96ea\u96eb\u96ec\u96ed\u96ee\u96ef", - "\u96f0\u96f1\u96f2\u96f3\u96f4\u96f5\u96f6\u96f7\u96f8\u96f9\u96fa\u96fb\u96fc\u96fd\u96fe\u96ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_97.png", - "ascent": %ascent%, - "chars": [ - "\u9700\u9701\u9702\u9703\u9704\u9705\u9706\u9707\u9708\u9709\u970a\u970b\u970c\u970d\u970e\u970f", - "\u9710\u9711\u9712\u9713\u9714\u9715\u9716\u9717\u9718\u9719\u971a\u971b\u971c\u971d\u971e\u971f", - "\u9720\u9721\u9722\u9723\u9724\u9725\u9726\u9727\u9728\u9729\u972a\u972b\u972c\u972d\u972e\u972f", - "\u9730\u9731\u9732\u9733\u9734\u9735\u9736\u9737\u9738\u9739\u973a\u973b\u973c\u973d\u973e\u973f", - "\u9740\u9741\u9742\u9743\u9744\u9745\u9746\u9747\u9748\u9749\u974a\u974b\u974c\u974d\u974e\u974f", - "\u9750\u9751\u9752\u9753\u9754\u9755\u9756\u9757\u9758\u9759\u975a\u975b\u975c\u975d\u975e\u975f", - "\u9760\u9761\u9762\u9763\u9764\u9765\u9766\u9767\u9768\u9769\u976a\u976b\u976c\u976d\u976e\u976f", - "\u9770\u9771\u9772\u9773\u9774\u9775\u9776\u9777\u9778\u9779\u977a\u977b\u977c\u977d\u977e\u977f", - "\u9780\u9781\u9782\u9783\u9784\u9785\u9786\u9787\u9788\u9789\u978a\u978b\u978c\u978d\u978e\u978f", - "\u9790\u9791\u9792\u9793\u9794\u9795\u9796\u9797\u9798\u9799\u979a\u979b\u979c\u979d\u979e\u979f", - "\u97a0\u97a1\u97a2\u97a3\u97a4\u97a5\u97a6\u97a7\u97a8\u97a9\u97aa\u97ab\u97ac\u97ad\u97ae\u97af", - "\u97b0\u97b1\u97b2\u97b3\u97b4\u97b5\u97b6\u97b7\u97b8\u97b9\u97ba\u97bb\u97bc\u97bd\u97be\u97bf", - "\u97c0\u97c1\u97c2\u97c3\u97c4\u97c5\u97c6\u97c7\u97c8\u97c9\u97ca\u97cb\u97cc\u97cd\u97ce\u97cf", - "\u97d0\u97d1\u97d2\u97d3\u97d4\u97d5\u97d6\u97d7\u97d8\u97d9\u97da\u97db\u97dc\u97dd\u97de\u97df", - "\u97e0\u97e1\u97e2\u97e3\u97e4\u97e5\u97e6\u97e7\u97e8\u97e9\u97ea\u97eb\u97ec\u97ed\u97ee\u97ef", - "\u97f0\u97f1\u97f2\u97f3\u97f4\u97f5\u97f6\u97f7\u97f8\u97f9\u97fa\u97fb\u97fc\u97fd\u97fe\u97ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_98.png", - "ascent": %ascent%, - "chars": [ - "\u9800\u9801\u9802\u9803\u9804\u9805\u9806\u9807\u9808\u9809\u980a\u980b\u980c\u980d\u980e\u980f", - "\u9810\u9811\u9812\u9813\u9814\u9815\u9816\u9817\u9818\u9819\u981a\u981b\u981c\u981d\u981e\u981f", - "\u9820\u9821\u9822\u9823\u9824\u9825\u9826\u9827\u9828\u9829\u982a\u982b\u982c\u982d\u982e\u982f", - "\u9830\u9831\u9832\u9833\u9834\u9835\u9836\u9837\u9838\u9839\u983a\u983b\u983c\u983d\u983e\u983f", - "\u9840\u9841\u9842\u9843\u9844\u9845\u9846\u9847\u9848\u9849\u984a\u984b\u984c\u984d\u984e\u984f", - "\u9850\u9851\u9852\u9853\u9854\u9855\u9856\u9857\u9858\u9859\u985a\u985b\u985c\u985d\u985e\u985f", - "\u9860\u9861\u9862\u9863\u9864\u9865\u9866\u9867\u9868\u9869\u986a\u986b\u986c\u986d\u986e\u986f", - "\u9870\u9871\u9872\u9873\u9874\u9875\u9876\u9877\u9878\u9879\u987a\u987b\u987c\u987d\u987e\u987f", - "\u9880\u9881\u9882\u9883\u9884\u9885\u9886\u9887\u9888\u9889\u988a\u988b\u988c\u988d\u988e\u988f", - "\u9890\u9891\u9892\u9893\u9894\u9895\u9896\u9897\u9898\u9899\u989a\u989b\u989c\u989d\u989e\u989f", - "\u98a0\u98a1\u98a2\u98a3\u98a4\u98a5\u98a6\u98a7\u98a8\u98a9\u98aa\u98ab\u98ac\u98ad\u98ae\u98af", - "\u98b0\u98b1\u98b2\u98b3\u98b4\u98b5\u98b6\u98b7\u98b8\u98b9\u98ba\u98bb\u98bc\u98bd\u98be\u98bf", - "\u98c0\u98c1\u98c2\u98c3\u98c4\u98c5\u98c6\u98c7\u98c8\u98c9\u98ca\u98cb\u98cc\u98cd\u98ce\u98cf", - "\u98d0\u98d1\u98d2\u98d3\u98d4\u98d5\u98d6\u98d7\u98d8\u98d9\u98da\u98db\u98dc\u98dd\u98de\u98df", - "\u98e0\u98e1\u98e2\u98e3\u98e4\u98e5\u98e6\u98e7\u98e8\u98e9\u98ea\u98eb\u98ec\u98ed\u98ee\u98ef", - "\u98f0\u98f1\u98f2\u98f3\u98f4\u98f5\u98f6\u98f7\u98f8\u98f9\u98fa\u98fb\u98fc\u98fd\u98fe\u98ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_99.png", - "ascent": %ascent%, - "chars": [ - "\u9900\u9901\u9902\u9903\u9904\u9905\u9906\u9907\u9908\u9909\u990a\u990b\u990c\u990d\u990e\u990f", - "\u9910\u9911\u9912\u9913\u9914\u9915\u9916\u9917\u9918\u9919\u991a\u991b\u991c\u991d\u991e\u991f", - "\u9920\u9921\u9922\u9923\u9924\u9925\u9926\u9927\u9928\u9929\u992a\u992b\u992c\u992d\u992e\u992f", - "\u9930\u9931\u9932\u9933\u9934\u9935\u9936\u9937\u9938\u9939\u993a\u993b\u993c\u993d\u993e\u993f", - "\u9940\u9941\u9942\u9943\u9944\u9945\u9946\u9947\u9948\u9949\u994a\u994b\u994c\u994d\u994e\u994f", - "\u9950\u9951\u9952\u9953\u9954\u9955\u9956\u9957\u9958\u9959\u995a\u995b\u995c\u995d\u995e\u995f", - "\u9960\u9961\u9962\u9963\u9964\u9965\u9966\u9967\u9968\u9969\u996a\u996b\u996c\u996d\u996e\u996f", - "\u9970\u9971\u9972\u9973\u9974\u9975\u9976\u9977\u9978\u9979\u997a\u997b\u997c\u997d\u997e\u997f", - "\u9980\u9981\u9982\u9983\u9984\u9985\u9986\u9987\u9988\u9989\u998a\u998b\u998c\u998d\u998e\u998f", - "\u9990\u9991\u9992\u9993\u9994\u9995\u9996\u9997\u9998\u9999\u999a\u999b\u999c\u999d\u999e\u999f", - "\u99a0\u99a1\u99a2\u99a3\u99a4\u99a5\u99a6\u99a7\u99a8\u99a9\u99aa\u99ab\u99ac\u99ad\u99ae\u99af", - "\u99b0\u99b1\u99b2\u99b3\u99b4\u99b5\u99b6\u99b7\u99b8\u99b9\u99ba\u99bb\u99bc\u99bd\u99be\u99bf", - "\u99c0\u99c1\u99c2\u99c3\u99c4\u99c5\u99c6\u99c7\u99c8\u99c9\u99ca\u99cb\u99cc\u99cd\u99ce\u99cf", - "\u99d0\u99d1\u99d2\u99d3\u99d4\u99d5\u99d6\u99d7\u99d8\u99d9\u99da\u99db\u99dc\u99dd\u99de\u99df", - "\u99e0\u99e1\u99e2\u99e3\u99e4\u99e5\u99e6\u99e7\u99e8\u99e9\u99ea\u99eb\u99ec\u99ed\u99ee\u99ef", - "\u99f0\u99f1\u99f2\u99f3\u99f4\u99f5\u99f6\u99f7\u99f8\u99f9\u99fa\u99fb\u99fc\u99fd\u99fe\u99ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_9a.png", - "ascent": %ascent%, - "chars": [ - "\u9a00\u9a01\u9a02\u9a03\u9a04\u9a05\u9a06\u9a07\u9a08\u9a09\u9a0a\u9a0b\u9a0c\u9a0d\u9a0e\u9a0f", - "\u9a10\u9a11\u9a12\u9a13\u9a14\u9a15\u9a16\u9a17\u9a18\u9a19\u9a1a\u9a1b\u9a1c\u9a1d\u9a1e\u9a1f", - "\u9a20\u9a21\u9a22\u9a23\u9a24\u9a25\u9a26\u9a27\u9a28\u9a29\u9a2a\u9a2b\u9a2c\u9a2d\u9a2e\u9a2f", - "\u9a30\u9a31\u9a32\u9a33\u9a34\u9a35\u9a36\u9a37\u9a38\u9a39\u9a3a\u9a3b\u9a3c\u9a3d\u9a3e\u9a3f", - "\u9a40\u9a41\u9a42\u9a43\u9a44\u9a45\u9a46\u9a47\u9a48\u9a49\u9a4a\u9a4b\u9a4c\u9a4d\u9a4e\u9a4f", - "\u9a50\u9a51\u9a52\u9a53\u9a54\u9a55\u9a56\u9a57\u9a58\u9a59\u9a5a\u9a5b\u9a5c\u9a5d\u9a5e\u9a5f", - "\u9a60\u9a61\u9a62\u9a63\u9a64\u9a65\u9a66\u9a67\u9a68\u9a69\u9a6a\u9a6b\u9a6c\u9a6d\u9a6e\u9a6f", - "\u9a70\u9a71\u9a72\u9a73\u9a74\u9a75\u9a76\u9a77\u9a78\u9a79\u9a7a\u9a7b\u9a7c\u9a7d\u9a7e\u9a7f", - "\u9a80\u9a81\u9a82\u9a83\u9a84\u9a85\u9a86\u9a87\u9a88\u9a89\u9a8a\u9a8b\u9a8c\u9a8d\u9a8e\u9a8f", - "\u9a90\u9a91\u9a92\u9a93\u9a94\u9a95\u9a96\u9a97\u9a98\u9a99\u9a9a\u9a9b\u9a9c\u9a9d\u9a9e\u9a9f", - "\u9aa0\u9aa1\u9aa2\u9aa3\u9aa4\u9aa5\u9aa6\u9aa7\u9aa8\u9aa9\u9aaa\u9aab\u9aac\u9aad\u9aae\u9aaf", - "\u9ab0\u9ab1\u9ab2\u9ab3\u9ab4\u9ab5\u9ab6\u9ab7\u9ab8\u9ab9\u9aba\u9abb\u9abc\u9abd\u9abe\u9abf", - "\u9ac0\u9ac1\u9ac2\u9ac3\u9ac4\u9ac5\u9ac6\u9ac7\u9ac8\u9ac9\u9aca\u9acb\u9acc\u9acd\u9ace\u9acf", - "\u9ad0\u9ad1\u9ad2\u9ad3\u9ad4\u9ad5\u9ad6\u9ad7\u9ad8\u9ad9\u9ada\u9adb\u9adc\u9add\u9ade\u9adf", - "\u9ae0\u9ae1\u9ae2\u9ae3\u9ae4\u9ae5\u9ae6\u9ae7\u9ae8\u9ae9\u9aea\u9aeb\u9aec\u9aed\u9aee\u9aef", - "\u9af0\u9af1\u9af2\u9af3\u9af4\u9af5\u9af6\u9af7\u9af8\u9af9\u9afa\u9afb\u9afc\u9afd\u9afe\u9aff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_9b.png", - "ascent": %ascent%, - "chars": [ - "\u9b00\u9b01\u9b02\u9b03\u9b04\u9b05\u9b06\u9b07\u9b08\u9b09\u9b0a\u9b0b\u9b0c\u9b0d\u9b0e\u9b0f", - "\u9b10\u9b11\u9b12\u9b13\u9b14\u9b15\u9b16\u9b17\u9b18\u9b19\u9b1a\u9b1b\u9b1c\u9b1d\u9b1e\u9b1f", - "\u9b20\u9b21\u9b22\u9b23\u9b24\u9b25\u9b26\u9b27\u9b28\u9b29\u9b2a\u9b2b\u9b2c\u9b2d\u9b2e\u9b2f", - "\u9b30\u9b31\u9b32\u9b33\u9b34\u9b35\u9b36\u9b37\u9b38\u9b39\u9b3a\u9b3b\u9b3c\u9b3d\u9b3e\u9b3f", - "\u9b40\u9b41\u9b42\u9b43\u9b44\u9b45\u9b46\u9b47\u9b48\u9b49\u9b4a\u9b4b\u9b4c\u9b4d\u9b4e\u9b4f", - "\u9b50\u9b51\u9b52\u9b53\u9b54\u9b55\u9b56\u9b57\u9b58\u9b59\u9b5a\u9b5b\u9b5c\u9b5d\u9b5e\u9b5f", - "\u9b60\u9b61\u9b62\u9b63\u9b64\u9b65\u9b66\u9b67\u9b68\u9b69\u9b6a\u9b6b\u9b6c\u9b6d\u9b6e\u9b6f", - "\u9b70\u9b71\u9b72\u9b73\u9b74\u9b75\u9b76\u9b77\u9b78\u9b79\u9b7a\u9b7b\u9b7c\u9b7d\u9b7e\u9b7f", - "\u9b80\u9b81\u9b82\u9b83\u9b84\u9b85\u9b86\u9b87\u9b88\u9b89\u9b8a\u9b8b\u9b8c\u9b8d\u9b8e\u9b8f", - "\u9b90\u9b91\u9b92\u9b93\u9b94\u9b95\u9b96\u9b97\u9b98\u9b99\u9b9a\u9b9b\u9b9c\u9b9d\u9b9e\u9b9f", - "\u9ba0\u9ba1\u9ba2\u9ba3\u9ba4\u9ba5\u9ba6\u9ba7\u9ba8\u9ba9\u9baa\u9bab\u9bac\u9bad\u9bae\u9baf", - "\u9bb0\u9bb1\u9bb2\u9bb3\u9bb4\u9bb5\u9bb6\u9bb7\u9bb8\u9bb9\u9bba\u9bbb\u9bbc\u9bbd\u9bbe\u9bbf", - "\u9bc0\u9bc1\u9bc2\u9bc3\u9bc4\u9bc5\u9bc6\u9bc7\u9bc8\u9bc9\u9bca\u9bcb\u9bcc\u9bcd\u9bce\u9bcf", - "\u9bd0\u9bd1\u9bd2\u9bd3\u9bd4\u9bd5\u9bd6\u9bd7\u9bd8\u9bd9\u9bda\u9bdb\u9bdc\u9bdd\u9bde\u9bdf", - "\u9be0\u9be1\u9be2\u9be3\u9be4\u9be5\u9be6\u9be7\u9be8\u9be9\u9bea\u9beb\u9bec\u9bed\u9bee\u9bef", - "\u9bf0\u9bf1\u9bf2\u9bf3\u9bf4\u9bf5\u9bf6\u9bf7\u9bf8\u9bf9\u9bfa\u9bfb\u9bfc\u9bfd\u9bfe\u9bff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_9c.png", - "ascent": %ascent%, - "chars": [ - "\u9c00\u9c01\u9c02\u9c03\u9c04\u9c05\u9c06\u9c07\u9c08\u9c09\u9c0a\u9c0b\u9c0c\u9c0d\u9c0e\u9c0f", - "\u9c10\u9c11\u9c12\u9c13\u9c14\u9c15\u9c16\u9c17\u9c18\u9c19\u9c1a\u9c1b\u9c1c\u9c1d\u9c1e\u9c1f", - "\u9c20\u9c21\u9c22\u9c23\u9c24\u9c25\u9c26\u9c27\u9c28\u9c29\u9c2a\u9c2b\u9c2c\u9c2d\u9c2e\u9c2f", - "\u9c30\u9c31\u9c32\u9c33\u9c34\u9c35\u9c36\u9c37\u9c38\u9c39\u9c3a\u9c3b\u9c3c\u9c3d\u9c3e\u9c3f", - "\u9c40\u9c41\u9c42\u9c43\u9c44\u9c45\u9c46\u9c47\u9c48\u9c49\u9c4a\u9c4b\u9c4c\u9c4d\u9c4e\u9c4f", - "\u9c50\u9c51\u9c52\u9c53\u9c54\u9c55\u9c56\u9c57\u9c58\u9c59\u9c5a\u9c5b\u9c5c\u9c5d\u9c5e\u9c5f", - "\u9c60\u9c61\u9c62\u9c63\u9c64\u9c65\u9c66\u9c67\u9c68\u9c69\u9c6a\u9c6b\u9c6c\u9c6d\u9c6e\u9c6f", - "\u9c70\u9c71\u9c72\u9c73\u9c74\u9c75\u9c76\u9c77\u9c78\u9c79\u9c7a\u9c7b\u9c7c\u9c7d\u9c7e\u9c7f", - "\u9c80\u9c81\u9c82\u9c83\u9c84\u9c85\u9c86\u9c87\u9c88\u9c89\u9c8a\u9c8b\u9c8c\u9c8d\u9c8e\u9c8f", - "\u9c90\u9c91\u9c92\u9c93\u9c94\u9c95\u9c96\u9c97\u9c98\u9c99\u9c9a\u9c9b\u9c9c\u9c9d\u9c9e\u9c9f", - "\u9ca0\u9ca1\u9ca2\u9ca3\u9ca4\u9ca5\u9ca6\u9ca7\u9ca8\u9ca9\u9caa\u9cab\u9cac\u9cad\u9cae\u9caf", - "\u9cb0\u9cb1\u9cb2\u9cb3\u9cb4\u9cb5\u9cb6\u9cb7\u9cb8\u9cb9\u9cba\u9cbb\u9cbc\u9cbd\u9cbe\u9cbf", - "\u9cc0\u9cc1\u9cc2\u9cc3\u9cc4\u9cc5\u9cc6\u9cc7\u9cc8\u9cc9\u9cca\u9ccb\u9ccc\u9ccd\u9cce\u9ccf", - "\u9cd0\u9cd1\u9cd2\u9cd3\u9cd4\u9cd5\u9cd6\u9cd7\u9cd8\u9cd9\u9cda\u9cdb\u9cdc\u9cdd\u9cde\u9cdf", - "\u9ce0\u9ce1\u9ce2\u9ce3\u9ce4\u9ce5\u9ce6\u9ce7\u9ce8\u9ce9\u9cea\u9ceb\u9cec\u9ced\u9cee\u9cef", - "\u9cf0\u9cf1\u9cf2\u9cf3\u9cf4\u9cf5\u9cf6\u9cf7\u9cf8\u9cf9\u9cfa\u9cfb\u9cfc\u9cfd\u9cfe\u9cff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_9d.png", - "ascent": %ascent%, - "chars": [ - "\u9d00\u9d01\u9d02\u9d03\u9d04\u9d05\u9d06\u9d07\u9d08\u9d09\u9d0a\u9d0b\u9d0c\u9d0d\u9d0e\u9d0f", - "\u9d10\u9d11\u9d12\u9d13\u9d14\u9d15\u9d16\u9d17\u9d18\u9d19\u9d1a\u9d1b\u9d1c\u9d1d\u9d1e\u9d1f", - "\u9d20\u9d21\u9d22\u9d23\u9d24\u9d25\u9d26\u9d27\u9d28\u9d29\u9d2a\u9d2b\u9d2c\u9d2d\u9d2e\u9d2f", - "\u9d30\u9d31\u9d32\u9d33\u9d34\u9d35\u9d36\u9d37\u9d38\u9d39\u9d3a\u9d3b\u9d3c\u9d3d\u9d3e\u9d3f", - "\u9d40\u9d41\u9d42\u9d43\u9d44\u9d45\u9d46\u9d47\u9d48\u9d49\u9d4a\u9d4b\u9d4c\u9d4d\u9d4e\u9d4f", - "\u9d50\u9d51\u9d52\u9d53\u9d54\u9d55\u9d56\u9d57\u9d58\u9d59\u9d5a\u9d5b\u9d5c\u9d5d\u9d5e\u9d5f", - "\u9d60\u9d61\u9d62\u9d63\u9d64\u9d65\u9d66\u9d67\u9d68\u9d69\u9d6a\u9d6b\u9d6c\u9d6d\u9d6e\u9d6f", - "\u9d70\u9d71\u9d72\u9d73\u9d74\u9d75\u9d76\u9d77\u9d78\u9d79\u9d7a\u9d7b\u9d7c\u9d7d\u9d7e\u9d7f", - "\u9d80\u9d81\u9d82\u9d83\u9d84\u9d85\u9d86\u9d87\u9d88\u9d89\u9d8a\u9d8b\u9d8c\u9d8d\u9d8e\u9d8f", - "\u9d90\u9d91\u9d92\u9d93\u9d94\u9d95\u9d96\u9d97\u9d98\u9d99\u9d9a\u9d9b\u9d9c\u9d9d\u9d9e\u9d9f", - "\u9da0\u9da1\u9da2\u9da3\u9da4\u9da5\u9da6\u9da7\u9da8\u9da9\u9daa\u9dab\u9dac\u9dad\u9dae\u9daf", - "\u9db0\u9db1\u9db2\u9db3\u9db4\u9db5\u9db6\u9db7\u9db8\u9db9\u9dba\u9dbb\u9dbc\u9dbd\u9dbe\u9dbf", - "\u9dc0\u9dc1\u9dc2\u9dc3\u9dc4\u9dc5\u9dc6\u9dc7\u9dc8\u9dc9\u9dca\u9dcb\u9dcc\u9dcd\u9dce\u9dcf", - "\u9dd0\u9dd1\u9dd2\u9dd3\u9dd4\u9dd5\u9dd6\u9dd7\u9dd8\u9dd9\u9dda\u9ddb\u9ddc\u9ddd\u9dde\u9ddf", - "\u9de0\u9de1\u9de2\u9de3\u9de4\u9de5\u9de6\u9de7\u9de8\u9de9\u9dea\u9deb\u9dec\u9ded\u9dee\u9def", - "\u9df0\u9df1\u9df2\u9df3\u9df4\u9df5\u9df6\u9df7\u9df8\u9df9\u9dfa\u9dfb\u9dfc\u9dfd\u9dfe\u9dff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_9e.png", - "ascent": %ascent%, - "chars": [ - "\u9e00\u9e01\u9e02\u9e03\u9e04\u9e05\u9e06\u9e07\u9e08\u9e09\u9e0a\u9e0b\u9e0c\u9e0d\u9e0e\u9e0f", - "\u9e10\u9e11\u9e12\u9e13\u9e14\u9e15\u9e16\u9e17\u9e18\u9e19\u9e1a\u9e1b\u9e1c\u9e1d\u9e1e\u9e1f", - "\u9e20\u9e21\u9e22\u9e23\u9e24\u9e25\u9e26\u9e27\u9e28\u9e29\u9e2a\u9e2b\u9e2c\u9e2d\u9e2e\u9e2f", - "\u9e30\u9e31\u9e32\u9e33\u9e34\u9e35\u9e36\u9e37\u9e38\u9e39\u9e3a\u9e3b\u9e3c\u9e3d\u9e3e\u9e3f", - "\u9e40\u9e41\u9e42\u9e43\u9e44\u9e45\u9e46\u9e47\u9e48\u9e49\u9e4a\u9e4b\u9e4c\u9e4d\u9e4e\u9e4f", - "\u9e50\u9e51\u9e52\u9e53\u9e54\u9e55\u9e56\u9e57\u9e58\u9e59\u9e5a\u9e5b\u9e5c\u9e5d\u9e5e\u9e5f", - "\u9e60\u9e61\u9e62\u9e63\u9e64\u9e65\u9e66\u9e67\u9e68\u9e69\u9e6a\u9e6b\u9e6c\u9e6d\u9e6e\u9e6f", - "\u9e70\u9e71\u9e72\u9e73\u9e74\u9e75\u9e76\u9e77\u9e78\u9e79\u9e7a\u9e7b\u9e7c\u9e7d\u9e7e\u9e7f", - "\u9e80\u9e81\u9e82\u9e83\u9e84\u9e85\u9e86\u9e87\u9e88\u9e89\u9e8a\u9e8b\u9e8c\u9e8d\u9e8e\u9e8f", - "\u9e90\u9e91\u9e92\u9e93\u9e94\u9e95\u9e96\u9e97\u9e98\u9e99\u9e9a\u9e9b\u9e9c\u9e9d\u9e9e\u9e9f", - "\u9ea0\u9ea1\u9ea2\u9ea3\u9ea4\u9ea5\u9ea6\u9ea7\u9ea8\u9ea9\u9eaa\u9eab\u9eac\u9ead\u9eae\u9eaf", - "\u9eb0\u9eb1\u9eb2\u9eb3\u9eb4\u9eb5\u9eb6\u9eb7\u9eb8\u9eb9\u9eba\u9ebb\u9ebc\u9ebd\u9ebe\u9ebf", - "\u9ec0\u9ec1\u9ec2\u9ec3\u9ec4\u9ec5\u9ec6\u9ec7\u9ec8\u9ec9\u9eca\u9ecb\u9ecc\u9ecd\u9ece\u9ecf", - "\u9ed0\u9ed1\u9ed2\u9ed3\u9ed4\u9ed5\u9ed6\u9ed7\u9ed8\u9ed9\u9eda\u9edb\u9edc\u9edd\u9ede\u9edf", - "\u9ee0\u9ee1\u9ee2\u9ee3\u9ee4\u9ee5\u9ee6\u9ee7\u9ee8\u9ee9\u9eea\u9eeb\u9eec\u9eed\u9eee\u9eef", - "\u9ef0\u9ef1\u9ef2\u9ef3\u9ef4\u9ef5\u9ef6\u9ef7\u9ef8\u9ef9\u9efa\u9efb\u9efc\u9efd\u9efe\u9eff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_9f.png", - "ascent": %ascent%, - "chars": [ - "\u9f00\u9f01\u9f02\u9f03\u9f04\u9f05\u9f06\u9f07\u9f08\u9f09\u9f0a\u9f0b\u9f0c\u9f0d\u9f0e\u9f0f", - "\u9f10\u9f11\u9f12\u9f13\u9f14\u9f15\u9f16\u9f17\u9f18\u9f19\u9f1a\u9f1b\u9f1c\u9f1d\u9f1e\u9f1f", - "\u9f20\u9f21\u9f22\u9f23\u9f24\u9f25\u9f26\u9f27\u9f28\u9f29\u9f2a\u9f2b\u9f2c\u9f2d\u9f2e\u9f2f", - "\u9f30\u9f31\u9f32\u9f33\u9f34\u9f35\u9f36\u9f37\u9f38\u9f39\u9f3a\u9f3b\u9f3c\u9f3d\u9f3e\u9f3f", - "\u9f40\u9f41\u9f42\u9f43\u9f44\u9f45\u9f46\u9f47\u9f48\u9f49\u9f4a\u9f4b\u9f4c\u9f4d\u9f4e\u9f4f", - "\u9f50\u9f51\u9f52\u9f53\u9f54\u9f55\u9f56\u9f57\u9f58\u9f59\u9f5a\u9f5b\u9f5c\u9f5d\u9f5e\u9f5f", - "\u9f60\u9f61\u9f62\u9f63\u9f64\u9f65\u9f66\u9f67\u9f68\u9f69\u9f6a\u9f6b\u9f6c\u9f6d\u9f6e\u9f6f", - "\u9f70\u9f71\u9f72\u9f73\u9f74\u9f75\u9f76\u9f77\u9f78\u9f79\u9f7a\u9f7b\u9f7c\u9f7d\u9f7e\u9f7f", - "\u9f80\u9f81\u9f82\u9f83\u9f84\u9f85\u9f86\u9f87\u9f88\u9f89\u9f8a\u9f8b\u9f8c\u9f8d\u9f8e\u9f8f", - "\u9f90\u9f91\u9f92\u9f93\u9f94\u9f95\u9f96\u9f97\u9f98\u9f99\u9f9a\u9f9b\u9f9c\u9f9d\u9f9e\u9f9f", - "\u9fa0\u9fa1\u9fa2\u9fa3\u9fa4\u9fa5\u9fa6\u9fa7\u9fa8\u9fa9\u9faa\u9fab\u9fac\u9fad\u9fae\u9faf", - "\u9fb0\u9fb1\u9fb2\u9fb3\u9fb4\u9fb5\u9fb6\u9fb7\u9fb8\u9fb9\u9fba\u9fbb\u9fbc\u9fbd\u9fbe\u9fbf", - "\u9fc0\u9fc1\u9fc2\u9fc3\u9fc4\u9fc5\u9fc6\u9fc7\u9fc8\u9fc9\u9fca\u9fcb\u9fcc\u9fcd\u9fce\u9fcf", - "\u9fd0\u9fd1\u9fd2\u9fd3\u9fd4\u9fd5\u9fd6\u9fd7\u9fd8\u9fd9\u9fda\u9fdb\u9fdc\u9fdd\u9fde\u9fdf", - "\u9fe0\u9fe1\u9fe2\u9fe3\u9fe4\u9fe5\u9fe6\u9fe7\u9fe8\u9fe9\u9fea\u9feb\u9fec\u9fed\u9fee\u9fef", - "\u9ff0\u9ff1\u9ff2\u9ff3\u9ff4\u9ff5\u9ff6\u9ff7\u9ff8\u9ff9\u9ffa\u9ffb\u9ffc\u9ffd\u9ffe\u9fff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_a0.png", - "ascent": %ascent%, - "chars": [ - "\ua000\ua001\ua002\ua003\ua004\ua005\ua006\ua007\ua008\ua009\ua00a\ua00b\ua00c\ua00d\ua00e\ua00f", - "\ua010\ua011\ua012\ua013\ua014\ua015\ua016\ua017\ua018\ua019\ua01a\ua01b\ua01c\ua01d\ua01e\ua01f", - "\ua020\ua021\ua022\ua023\ua024\ua025\ua026\ua027\ua028\ua029\ua02a\ua02b\ua02c\ua02d\ua02e\ua02f", - "\ua030\ua031\ua032\ua033\ua034\ua035\ua036\ua037\ua038\ua039\ua03a\ua03b\ua03c\ua03d\ua03e\ua03f", - "\ua040\ua041\ua042\ua043\ua044\ua045\ua046\ua047\ua048\ua049\ua04a\ua04b\ua04c\ua04d\ua04e\ua04f", - "\ua050\ua051\ua052\ua053\ua054\ua055\ua056\ua057\ua058\ua059\ua05a\ua05b\ua05c\ua05d\ua05e\ua05f", - "\ua060\ua061\ua062\ua063\ua064\ua065\ua066\ua067\ua068\ua069\ua06a\ua06b\ua06c\ua06d\ua06e\ua06f", - "\ua070\ua071\ua072\ua073\ua074\ua075\ua076\ua077\ua078\ua079\ua07a\ua07b\ua07c\ua07d\ua07e\ua07f", - "\ua080\ua081\ua082\ua083\ua084\ua085\ua086\ua087\ua088\ua089\ua08a\ua08b\ua08c\ua08d\ua08e\ua08f", - "\ua090\ua091\ua092\ua093\ua094\ua095\ua096\ua097\ua098\ua099\ua09a\ua09b\ua09c\ua09d\ua09e\ua09f", - "\ua0a0\ua0a1\ua0a2\ua0a3\ua0a4\ua0a5\ua0a6\ua0a7\ua0a8\ua0a9\ua0aa\ua0ab\ua0ac\ua0ad\ua0ae\ua0af", - "\ua0b0\ua0b1\ua0b2\ua0b3\ua0b4\ua0b5\ua0b6\ua0b7\ua0b8\ua0b9\ua0ba\ua0bb\ua0bc\ua0bd\ua0be\ua0bf", - "\ua0c0\ua0c1\ua0c2\ua0c3\ua0c4\ua0c5\ua0c6\ua0c7\ua0c8\ua0c9\ua0ca\ua0cb\ua0cc\ua0cd\ua0ce\ua0cf", - "\ua0d0\ua0d1\ua0d2\ua0d3\ua0d4\ua0d5\ua0d6\ua0d7\ua0d8\ua0d9\ua0da\ua0db\ua0dc\ua0dd\ua0de\ua0df", - "\ua0e0\ua0e1\ua0e2\ua0e3\ua0e4\ua0e5\ua0e6\ua0e7\ua0e8\ua0e9\ua0ea\ua0eb\ua0ec\ua0ed\ua0ee\ua0ef", - "\ua0f0\ua0f1\ua0f2\ua0f3\ua0f4\ua0f5\ua0f6\ua0f7\ua0f8\ua0f9\ua0fa\ua0fb\ua0fc\ua0fd\ua0fe\ua0ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_a1.png", - "ascent": %ascent%, - "chars": [ - "\ua100\ua101\ua102\ua103\ua104\ua105\ua106\ua107\ua108\ua109\ua10a\ua10b\ua10c\ua10d\ua10e\ua10f", - "\ua110\ua111\ua112\ua113\ua114\ua115\ua116\ua117\ua118\ua119\ua11a\ua11b\ua11c\ua11d\ua11e\ua11f", - "\ua120\ua121\ua122\ua123\ua124\ua125\ua126\ua127\ua128\ua129\ua12a\ua12b\ua12c\ua12d\ua12e\ua12f", - "\ua130\ua131\ua132\ua133\ua134\ua135\ua136\ua137\ua138\ua139\ua13a\ua13b\ua13c\ua13d\ua13e\ua13f", - "\ua140\ua141\ua142\ua143\ua144\ua145\ua146\ua147\ua148\ua149\ua14a\ua14b\ua14c\ua14d\ua14e\ua14f", - "\ua150\ua151\ua152\ua153\ua154\ua155\ua156\ua157\ua158\ua159\ua15a\ua15b\ua15c\ua15d\ua15e\ua15f", - "\ua160\ua161\ua162\ua163\ua164\ua165\ua166\ua167\ua168\ua169\ua16a\ua16b\ua16c\ua16d\ua16e\ua16f", - "\ua170\ua171\ua172\ua173\ua174\ua175\ua176\ua177\ua178\ua179\ua17a\ua17b\ua17c\ua17d\ua17e\ua17f", - "\ua180\ua181\ua182\ua183\ua184\ua185\ua186\ua187\ua188\ua189\ua18a\ua18b\ua18c\ua18d\ua18e\ua18f", - "\ua190\ua191\ua192\ua193\ua194\ua195\ua196\ua197\ua198\ua199\ua19a\ua19b\ua19c\ua19d\ua19e\ua19f", - "\ua1a0\ua1a1\ua1a2\ua1a3\ua1a4\ua1a5\ua1a6\ua1a7\ua1a8\ua1a9\ua1aa\ua1ab\ua1ac\ua1ad\ua1ae\ua1af", - "\ua1b0\ua1b1\ua1b2\ua1b3\ua1b4\ua1b5\ua1b6\ua1b7\ua1b8\ua1b9\ua1ba\ua1bb\ua1bc\ua1bd\ua1be\ua1bf", - "\ua1c0\ua1c1\ua1c2\ua1c3\ua1c4\ua1c5\ua1c6\ua1c7\ua1c8\ua1c9\ua1ca\ua1cb\ua1cc\ua1cd\ua1ce\ua1cf", - "\ua1d0\ua1d1\ua1d2\ua1d3\ua1d4\ua1d5\ua1d6\ua1d7\ua1d8\ua1d9\ua1da\ua1db\ua1dc\ua1dd\ua1de\ua1df", - "\ua1e0\ua1e1\ua1e2\ua1e3\ua1e4\ua1e5\ua1e6\ua1e7\ua1e8\ua1e9\ua1ea\ua1eb\ua1ec\ua1ed\ua1ee\ua1ef", - "\ua1f0\ua1f1\ua1f2\ua1f3\ua1f4\ua1f5\ua1f6\ua1f7\ua1f8\ua1f9\ua1fa\ua1fb\ua1fc\ua1fd\ua1fe\ua1ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_a2.png", - "ascent": %ascent%, - "chars": [ - "\ua200\ua201\ua202\ua203\ua204\ua205\ua206\ua207\ua208\ua209\ua20a\ua20b\ua20c\ua20d\ua20e\ua20f", - "\ua210\ua211\ua212\ua213\ua214\ua215\ua216\ua217\ua218\ua219\ua21a\ua21b\ua21c\ua21d\ua21e\ua21f", - "\ua220\ua221\ua222\ua223\ua224\ua225\ua226\ua227\ua228\ua229\ua22a\ua22b\ua22c\ua22d\ua22e\ua22f", - "\ua230\ua231\ua232\ua233\ua234\ua235\ua236\ua237\ua238\ua239\ua23a\ua23b\ua23c\ua23d\ua23e\ua23f", - "\ua240\ua241\ua242\ua243\ua244\ua245\ua246\ua247\ua248\ua249\ua24a\ua24b\ua24c\ua24d\ua24e\ua24f", - "\ua250\ua251\ua252\ua253\ua254\ua255\ua256\ua257\ua258\ua259\ua25a\ua25b\ua25c\ua25d\ua25e\ua25f", - "\ua260\ua261\ua262\ua263\ua264\ua265\ua266\ua267\ua268\ua269\ua26a\ua26b\ua26c\ua26d\ua26e\ua26f", - "\ua270\ua271\ua272\ua273\ua274\ua275\ua276\ua277\ua278\ua279\ua27a\ua27b\ua27c\ua27d\ua27e\ua27f", - "\ua280\ua281\ua282\ua283\ua284\ua285\ua286\ua287\ua288\ua289\ua28a\ua28b\ua28c\ua28d\ua28e\ua28f", - "\ua290\ua291\ua292\ua293\ua294\ua295\ua296\ua297\ua298\ua299\ua29a\ua29b\ua29c\ua29d\ua29e\ua29f", - "\ua2a0\ua2a1\ua2a2\ua2a3\ua2a4\ua2a5\ua2a6\ua2a7\ua2a8\ua2a9\ua2aa\ua2ab\ua2ac\ua2ad\ua2ae\ua2af", - "\ua2b0\ua2b1\ua2b2\ua2b3\ua2b4\ua2b5\ua2b6\ua2b7\ua2b8\ua2b9\ua2ba\ua2bb\ua2bc\ua2bd\ua2be\ua2bf", - "\ua2c0\ua2c1\ua2c2\ua2c3\ua2c4\ua2c5\ua2c6\ua2c7\ua2c8\ua2c9\ua2ca\ua2cb\ua2cc\ua2cd\ua2ce\ua2cf", - "\ua2d0\ua2d1\ua2d2\ua2d3\ua2d4\ua2d5\ua2d6\ua2d7\ua2d8\ua2d9\ua2da\ua2db\ua2dc\ua2dd\ua2de\ua2df", - "\ua2e0\ua2e1\ua2e2\ua2e3\ua2e4\ua2e5\ua2e6\ua2e7\ua2e8\ua2e9\ua2ea\ua2eb\ua2ec\ua2ed\ua2ee\ua2ef", - "\ua2f0\ua2f1\ua2f2\ua2f3\ua2f4\ua2f5\ua2f6\ua2f7\ua2f8\ua2f9\ua2fa\ua2fb\ua2fc\ua2fd\ua2fe\ua2ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_a3.png", - "ascent": %ascent%, - "chars": [ - "\ua300\ua301\ua302\ua303\ua304\ua305\ua306\ua307\ua308\ua309\ua30a\ua30b\ua30c\ua30d\ua30e\ua30f", - "\ua310\ua311\ua312\ua313\ua314\ua315\ua316\ua317\ua318\ua319\ua31a\ua31b\ua31c\ua31d\ua31e\ua31f", - "\ua320\ua321\ua322\ua323\ua324\ua325\ua326\ua327\ua328\ua329\ua32a\ua32b\ua32c\ua32d\ua32e\ua32f", - "\ua330\ua331\ua332\ua333\ua334\ua335\ua336\ua337\ua338\ua339\ua33a\ua33b\ua33c\ua33d\ua33e\ua33f", - "\ua340\ua341\ua342\ua343\ua344\ua345\ua346\ua347\ua348\ua349\ua34a\ua34b\ua34c\ua34d\ua34e\ua34f", - "\ua350\ua351\ua352\ua353\ua354\ua355\ua356\ua357\ua358\ua359\ua35a\ua35b\ua35c\ua35d\ua35e\ua35f", - "\ua360\ua361\ua362\ua363\ua364\ua365\ua366\ua367\ua368\ua369\ua36a\ua36b\ua36c\ua36d\ua36e\ua36f", - "\ua370\ua371\ua372\ua373\ua374\ua375\ua376\ua377\ua378\ua379\ua37a\ua37b\ua37c\ua37d\ua37e\ua37f", - "\ua380\ua381\ua382\ua383\ua384\ua385\ua386\ua387\ua388\ua389\ua38a\ua38b\ua38c\ua38d\ua38e\ua38f", - "\ua390\ua391\ua392\ua393\ua394\ua395\ua396\ua397\ua398\ua399\ua39a\ua39b\ua39c\ua39d\ua39e\ua39f", - "\ua3a0\ua3a1\ua3a2\ua3a3\ua3a4\ua3a5\ua3a6\ua3a7\ua3a8\ua3a9\ua3aa\ua3ab\ua3ac\ua3ad\ua3ae\ua3af", - "\ua3b0\ua3b1\ua3b2\ua3b3\ua3b4\ua3b5\ua3b6\ua3b7\ua3b8\ua3b9\ua3ba\ua3bb\ua3bc\ua3bd\ua3be\ua3bf", - "\ua3c0\ua3c1\ua3c2\ua3c3\ua3c4\ua3c5\ua3c6\ua3c7\ua3c8\ua3c9\ua3ca\ua3cb\ua3cc\ua3cd\ua3ce\ua3cf", - "\ua3d0\ua3d1\ua3d2\ua3d3\ua3d4\ua3d5\ua3d6\ua3d7\ua3d8\ua3d9\ua3da\ua3db\ua3dc\ua3dd\ua3de\ua3df", - "\ua3e0\ua3e1\ua3e2\ua3e3\ua3e4\ua3e5\ua3e6\ua3e7\ua3e8\ua3e9\ua3ea\ua3eb\ua3ec\ua3ed\ua3ee\ua3ef", - "\ua3f0\ua3f1\ua3f2\ua3f3\ua3f4\ua3f5\ua3f6\ua3f7\ua3f8\ua3f9\ua3fa\ua3fb\ua3fc\ua3fd\ua3fe\ua3ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_a4.png", - "ascent": %ascent%, - "chars": [ - "\ua400\ua401\ua402\ua403\ua404\ua405\ua406\ua407\ua408\ua409\ua40a\ua40b\ua40c\ua40d\ua40e\ua40f", - "\ua410\ua411\ua412\ua413\ua414\ua415\ua416\ua417\ua418\ua419\ua41a\ua41b\ua41c\ua41d\ua41e\ua41f", - "\ua420\ua421\ua422\ua423\ua424\ua425\ua426\ua427\ua428\ua429\ua42a\ua42b\ua42c\ua42d\ua42e\ua42f", - "\ua430\ua431\ua432\ua433\ua434\ua435\ua436\ua437\ua438\ua439\ua43a\ua43b\ua43c\ua43d\ua43e\ua43f", - "\ua440\ua441\ua442\ua443\ua444\ua445\ua446\ua447\ua448\ua449\ua44a\ua44b\ua44c\ua44d\ua44e\ua44f", - "\ua450\ua451\ua452\ua453\ua454\ua455\ua456\ua457\ua458\ua459\ua45a\ua45b\ua45c\ua45d\ua45e\ua45f", - "\ua460\ua461\ua462\ua463\ua464\ua465\ua466\ua467\ua468\ua469\ua46a\ua46b\ua46c\ua46d\ua46e\ua46f", - "\ua470\ua471\ua472\ua473\ua474\ua475\ua476\ua477\ua478\ua479\ua47a\ua47b\ua47c\ua47d\ua47e\ua47f", - "\ua480\ua481\ua482\ua483\ua484\ua485\ua486\ua487\ua488\ua489\ua48a\ua48b\ua48c\ua48d\ua48e\ua48f", - "\ua490\ua491\ua492\ua493\ua494\ua495\ua496\ua497\ua498\ua499\ua49a\ua49b\ua49c\ua49d\ua49e\ua49f", - "\ua4a0\ua4a1\ua4a2\ua4a3\ua4a4\ua4a5\ua4a6\ua4a7\ua4a8\ua4a9\ua4aa\ua4ab\ua4ac\ua4ad\ua4ae\ua4af", - "\ua4b0\ua4b1\ua4b2\ua4b3\ua4b4\ua4b5\ua4b6\ua4b7\ua4b8\ua4b9\ua4ba\ua4bb\ua4bc\ua4bd\ua4be\ua4bf", - "\ua4c0\ua4c1\ua4c2\ua4c3\ua4c4\ua4c5\ua4c6\ua4c7\ua4c8\ua4c9\ua4ca\ua4cb\ua4cc\ua4cd\ua4ce\ua4cf", - "\ua4d0\ua4d1\ua4d2\ua4d3\ua4d4\ua4d5\ua4d6\ua4d7\ua4d8\ua4d9\ua4da\ua4db\ua4dc\ua4dd\ua4de\ua4df", - "\ua4e0\ua4e1\ua4e2\ua4e3\ua4e4\ua4e5\ua4e6\ua4e7\ua4e8\ua4e9\ua4ea\ua4eb\ua4ec\ua4ed\ua4ee\ua4ef", - "\ua4f0\ua4f1\ua4f2\ua4f3\ua4f4\ua4f5\ua4f6\ua4f7\ua4f8\ua4f9\ua4fa\ua4fb\ua4fc\ua4fd\ua4fe\ua4ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_a5.png", - "ascent": %ascent%, - "chars": [ - "\ua500\ua501\ua502\ua503\ua504\ua505\ua506\ua507\ua508\ua509\ua50a\ua50b\ua50c\ua50d\ua50e\ua50f", - "\ua510\ua511\ua512\ua513\ua514\ua515\ua516\ua517\ua518\ua519\ua51a\ua51b\ua51c\ua51d\ua51e\ua51f", - "\ua520\ua521\ua522\ua523\ua524\ua525\ua526\ua527\ua528\ua529\ua52a\ua52b\ua52c\ua52d\ua52e\ua52f", - "\ua530\ua531\ua532\ua533\ua534\ua535\ua536\ua537\ua538\ua539\ua53a\ua53b\ua53c\ua53d\ua53e\ua53f", - "\ua540\ua541\ua542\ua543\ua544\ua545\ua546\ua547\ua548\ua549\ua54a\ua54b\ua54c\ua54d\ua54e\ua54f", - "\ua550\ua551\ua552\ua553\ua554\ua555\ua556\ua557\ua558\ua559\ua55a\ua55b\ua55c\ua55d\ua55e\ua55f", - "\ua560\ua561\ua562\ua563\ua564\ua565\ua566\ua567\ua568\ua569\ua56a\ua56b\ua56c\ua56d\ua56e\ua56f", - "\ua570\ua571\ua572\ua573\ua574\ua575\ua576\ua577\ua578\ua579\ua57a\ua57b\ua57c\ua57d\ua57e\ua57f", - "\ua580\ua581\ua582\ua583\ua584\ua585\ua586\ua587\ua588\ua589\ua58a\ua58b\ua58c\ua58d\ua58e\ua58f", - "\ua590\ua591\ua592\ua593\ua594\ua595\ua596\ua597\ua598\ua599\ua59a\ua59b\ua59c\ua59d\ua59e\ua59f", - "\ua5a0\ua5a1\ua5a2\ua5a3\ua5a4\ua5a5\ua5a6\ua5a7\ua5a8\ua5a9\ua5aa\ua5ab\ua5ac\ua5ad\ua5ae\ua5af", - "\ua5b0\ua5b1\ua5b2\ua5b3\ua5b4\ua5b5\ua5b6\ua5b7\ua5b8\ua5b9\ua5ba\ua5bb\ua5bc\ua5bd\ua5be\ua5bf", - "\ua5c0\ua5c1\ua5c2\ua5c3\ua5c4\ua5c5\ua5c6\ua5c7\ua5c8\ua5c9\ua5ca\ua5cb\ua5cc\ua5cd\ua5ce\ua5cf", - "\ua5d0\ua5d1\ua5d2\ua5d3\ua5d4\ua5d5\ua5d6\ua5d7\ua5d8\ua5d9\ua5da\ua5db\ua5dc\ua5dd\ua5de\ua5df", - "\ua5e0\ua5e1\ua5e2\ua5e3\ua5e4\ua5e5\ua5e6\ua5e7\ua5e8\ua5e9\ua5ea\ua5eb\ua5ec\ua5ed\ua5ee\ua5ef", - "\ua5f0\ua5f1\ua5f2\ua5f3\ua5f4\ua5f5\ua5f6\ua5f7\ua5f8\ua5f9\ua5fa\ua5fb\ua5fc\ua5fd\ua5fe\ua5ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_a6.png", - "ascent": %ascent%, - "chars": [ - "\ua600\ua601\ua602\ua603\ua604\ua605\ua606\ua607\ua608\ua609\ua60a\ua60b\ua60c\ua60d\ua60e\ua60f", - "\ua610\ua611\ua612\ua613\ua614\ua615\ua616\ua617\ua618\ua619\ua61a\ua61b\ua61c\ua61d\ua61e\ua61f", - "\ua620\ua621\ua622\ua623\ua624\ua625\ua626\ua627\ua628\ua629\ua62a\ua62b\ua62c\ua62d\ua62e\ua62f", - "\ua630\ua631\ua632\ua633\ua634\ua635\ua636\ua637\ua638\ua639\ua63a\ua63b\ua63c\ua63d\ua63e\ua63f", - "\ua640\ua641\ua642\ua643\ua644\ua645\ua646\ua647\ua648\ua649\ua64a\ua64b\ua64c\ua64d\ua64e\ua64f", - "\ua650\ua651\ua652\ua653\ua654\ua655\ua656\ua657\ua658\ua659\ua65a\ua65b\ua65c\ua65d\ua65e\ua65f", - "\ua660\ua661\ua662\ua663\ua664\ua665\ua666\ua667\ua668\ua669\ua66a\ua66b\ua66c\ua66d\ua66e\ua66f", - "\ua670\ua671\ua672\ua673\ua674\ua675\ua676\ua677\ua678\ua679\ua67a\ua67b\ua67c\ua67d\ua67e\ua67f", - "\ua680\ua681\ua682\ua683\ua684\ua685\ua686\ua687\ua688\ua689\ua68a\ua68b\ua68c\ua68d\ua68e\ua68f", - "\ua690\ua691\ua692\ua693\ua694\ua695\ua696\ua697\ua698\ua699\ua69a\ua69b\ua69c\ua69d\ua69e\ua69f", - "\ua6a0\ua6a1\ua6a2\ua6a3\ua6a4\ua6a5\ua6a6\ua6a7\ua6a8\ua6a9\ua6aa\ua6ab\ua6ac\ua6ad\ua6ae\ua6af", - "\ua6b0\ua6b1\ua6b2\ua6b3\ua6b4\ua6b5\ua6b6\ua6b7\ua6b8\ua6b9\ua6ba\ua6bb\ua6bc\ua6bd\ua6be\ua6bf", - "\ua6c0\ua6c1\ua6c2\ua6c3\ua6c4\ua6c5\ua6c6\ua6c7\ua6c8\ua6c9\ua6ca\ua6cb\ua6cc\ua6cd\ua6ce\ua6cf", - "\ua6d0\ua6d1\ua6d2\ua6d3\ua6d4\ua6d5\ua6d6\ua6d7\ua6d8\ua6d9\ua6da\ua6db\ua6dc\ua6dd\ua6de\ua6df", - "\ua6e0\ua6e1\ua6e2\ua6e3\ua6e4\ua6e5\ua6e6\ua6e7\ua6e8\ua6e9\ua6ea\ua6eb\ua6ec\ua6ed\ua6ee\ua6ef", - "\ua6f0\ua6f1\ua6f2\ua6f3\ua6f4\ua6f5\ua6f6\ua6f7\ua6f8\ua6f9\ua6fa\ua6fb\ua6fc\ua6fd\ua6fe\ua6ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_a7.png", - "ascent": %ascent%, - "chars": [ - "\ua700\ua701\ua702\ua703\ua704\ua705\ua706\ua707\ua708\ua709\ua70a\ua70b\ua70c\ua70d\ua70e\ua70f", - "\ua710\ua711\ua712\ua713\ua714\ua715\ua716\ua717\ua718\ua719\ua71a\ua71b\ua71c\ua71d\ua71e\ua71f", - "\ua720\ua721\ua722\ua723\ua724\ua725\ua726\ua727\ua728\ua729\ua72a\ua72b\ua72c\ua72d\ua72e\ua72f", - "\ua730\ua731\ua732\ua733\ua734\ua735\ua736\ua737\ua738\ua739\ua73a\ua73b\ua73c\ua73d\ua73e\ua73f", - "\ua740\ua741\ua742\ua743\ua744\ua745\ua746\ua747\ua748\ua749\ua74a\ua74b\ua74c\ua74d\ua74e\ua74f", - "\ua750\ua751\ua752\ua753\ua754\ua755\ua756\ua757\ua758\ua759\ua75a\ua75b\ua75c\ua75d\ua75e\ua75f", - "\ua760\ua761\ua762\ua763\ua764\ua765\ua766\ua767\ua768\ua769\ua76a\ua76b\ua76c\ua76d\ua76e\ua76f", - "\ua770\ua771\ua772\ua773\ua774\ua775\ua776\ua777\ua778\ua779\ua77a\ua77b\ua77c\ua77d\ua77e\ua77f", - "\ua780\ua781\ua782\ua783\ua784\ua785\ua786\ua787\ua788\ua789\ua78a\ua78b\ua78c\ua78d\ua78e\ua78f", - "\ua790\ua791\ua792\ua793\ua794\ua795\ua796\ua797\ua798\ua799\ua79a\ua79b\ua79c\ua79d\ua79e\ua79f", - "\ua7a0\ua7a1\ua7a2\ua7a3\ua7a4\ua7a5\ua7a6\ua7a7\ua7a8\ua7a9\ua7aa\ua7ab\ua7ac\ua7ad\ua7ae\ua7af", - "\ua7b0\ua7b1\ua7b2\ua7b3\ua7b4\ua7b5\ua7b6\ua7b7\ua7b8\ua7b9\ua7ba\ua7bb\ua7bc\ua7bd\ua7be\ua7bf", - "\ua7c0\ua7c1\ua7c2\ua7c3\ua7c4\ua7c5\ua7c6\ua7c7\ua7c8\ua7c9\ua7ca\ua7cb\ua7cc\ua7cd\ua7ce\ua7cf", - "\ua7d0\ua7d1\ua7d2\ua7d3\ua7d4\ua7d5\ua7d6\ua7d7\ua7d8\ua7d9\ua7da\ua7db\ua7dc\ua7dd\ua7de\ua7df", - "\ua7e0\ua7e1\ua7e2\ua7e3\ua7e4\ua7e5\ua7e6\ua7e7\ua7e8\ua7e9\ua7ea\ua7eb\ua7ec\ua7ed\ua7ee\ua7ef", - "\ua7f0\ua7f1\ua7f2\ua7f3\ua7f4\ua7f5\ua7f6\ua7f7\ua7f8\ua7f9\ua7fa\ua7fb\ua7fc\ua7fd\ua7fe\ua7ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_a8.png", - "ascent": %ascent%, - "chars": [ - "\ua800\ua801\ua802\ua803\ua804\ua805\ua806\ua807\ua808\ua809\ua80a\ua80b\ua80c\ua80d\ua80e\ua80f", - "\ua810\ua811\ua812\ua813\ua814\ua815\ua816\ua817\ua818\ua819\ua81a\ua81b\ua81c\ua81d\ua81e\ua81f", - "\ua820\ua821\ua822\ua823\ua824\ua825\ua826\ua827\ua828\ua829\ua82a\ua82b\ua82c\ua82d\ua82e\ua82f", - "\ua830\ua831\ua832\ua833\ua834\ua835\ua836\ua837\ua838\ua839\ua83a\ua83b\ua83c\ua83d\ua83e\ua83f", - "\ua840\ua841\ua842\ua843\ua844\ua845\ua846\ua847\ua848\ua849\ua84a\ua84b\ua84c\ua84d\ua84e\ua84f", - "\ua850\ua851\ua852\ua853\ua854\ua855\ua856\ua857\ua858\ua859\ua85a\ua85b\ua85c\ua85d\ua85e\ua85f", - "\ua860\ua861\ua862\ua863\ua864\ua865\ua866\ua867\ua868\ua869\ua86a\ua86b\ua86c\ua86d\ua86e\ua86f", - "\ua870\ua871\ua872\ua873\ua874\ua875\ua876\ua877\ua878\ua879\ua87a\ua87b\ua87c\ua87d\ua87e\ua87f", - "\ua880\ua881\ua882\ua883\ua884\ua885\ua886\ua887\ua888\ua889\ua88a\ua88b\ua88c\ua88d\ua88e\ua88f", - "\ua890\ua891\ua892\ua893\ua894\ua895\ua896\ua897\ua898\ua899\ua89a\ua89b\ua89c\ua89d\ua89e\ua89f", - "\ua8a0\ua8a1\ua8a2\ua8a3\ua8a4\ua8a5\ua8a6\ua8a7\ua8a8\ua8a9\ua8aa\ua8ab\ua8ac\ua8ad\ua8ae\ua8af", - "\ua8b0\ua8b1\ua8b2\ua8b3\ua8b4\ua8b5\ua8b6\ua8b7\ua8b8\ua8b9\ua8ba\ua8bb\ua8bc\ua8bd\ua8be\ua8bf", - "\ua8c0\ua8c1\ua8c2\ua8c3\ua8c4\ua8c5\ua8c6\ua8c7\ua8c8\ua8c9\ua8ca\ua8cb\ua8cc\ua8cd\ua8ce\ua8cf", - "\ua8d0\ua8d1\ua8d2\ua8d3\ua8d4\ua8d5\ua8d6\ua8d7\ua8d8\ua8d9\ua8da\ua8db\ua8dc\ua8dd\ua8de\ua8df", - "\ua8e0\ua8e1\ua8e2\ua8e3\ua8e4\ua8e5\ua8e6\ua8e7\ua8e8\ua8e9\ua8ea\ua8eb\ua8ec\ua8ed\ua8ee\ua8ef", - "\ua8f0\ua8f1\ua8f2\ua8f3\ua8f4\ua8f5\ua8f6\ua8f7\ua8f8\ua8f9\ua8fa\ua8fb\ua8fc\ua8fd\ua8fe\ua8ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_a9.png", - "ascent": %ascent%, - "chars": [ - "\ua900\ua901\ua902\ua903\ua904\ua905\ua906\ua907\ua908\ua909\ua90a\ua90b\ua90c\ua90d\ua90e\ua90f", - "\ua910\ua911\ua912\ua913\ua914\ua915\ua916\ua917\ua918\ua919\ua91a\ua91b\ua91c\ua91d\ua91e\ua91f", - "\ua920\ua921\ua922\ua923\ua924\ua925\ua926\ua927\ua928\ua929\ua92a\ua92b\ua92c\ua92d\ua92e\ua92f", - "\ua930\ua931\ua932\ua933\ua934\ua935\ua936\ua937\ua938\ua939\ua93a\ua93b\ua93c\ua93d\ua93e\ua93f", - "\ua940\ua941\ua942\ua943\ua944\ua945\ua946\ua947\ua948\ua949\ua94a\ua94b\ua94c\ua94d\ua94e\ua94f", - "\ua950\ua951\ua952\ua953\ua954\ua955\ua956\ua957\ua958\ua959\ua95a\ua95b\ua95c\ua95d\ua95e\ua95f", - "\ua960\ua961\ua962\ua963\ua964\ua965\ua966\ua967\ua968\ua969\ua96a\ua96b\ua96c\ua96d\ua96e\ua96f", - "\ua970\ua971\ua972\ua973\ua974\ua975\ua976\ua977\ua978\ua979\ua97a\ua97b\ua97c\ua97d\ua97e\ua97f", - "\ua980\ua981\ua982\ua983\ua984\ua985\ua986\ua987\ua988\ua989\ua98a\ua98b\ua98c\ua98d\ua98e\ua98f", - "\ua990\ua991\ua992\ua993\ua994\ua995\ua996\ua997\ua998\ua999\ua99a\ua99b\ua99c\ua99d\ua99e\ua99f", - "\ua9a0\ua9a1\ua9a2\ua9a3\ua9a4\ua9a5\ua9a6\ua9a7\ua9a8\ua9a9\ua9aa\ua9ab\ua9ac\ua9ad\ua9ae\ua9af", - "\ua9b0\ua9b1\ua9b2\ua9b3\ua9b4\ua9b5\ua9b6\ua9b7\ua9b8\ua9b9\ua9ba\ua9bb\ua9bc\ua9bd\ua9be\ua9bf", - "\ua9c0\ua9c1\ua9c2\ua9c3\ua9c4\ua9c5\ua9c6\ua9c7\ua9c8\ua9c9\ua9ca\ua9cb\ua9cc\ua9cd\ua9ce\ua9cf", - "\ua9d0\ua9d1\ua9d2\ua9d3\ua9d4\ua9d5\ua9d6\ua9d7\ua9d8\ua9d9\ua9da\ua9db\ua9dc\ua9dd\ua9de\ua9df", - "\ua9e0\ua9e1\ua9e2\ua9e3\ua9e4\ua9e5\ua9e6\ua9e7\ua9e8\ua9e9\ua9ea\ua9eb\ua9ec\ua9ed\ua9ee\ua9ef", - "\ua9f0\ua9f1\ua9f2\ua9f3\ua9f4\ua9f5\ua9f6\ua9f7\ua9f8\ua9f9\ua9fa\ua9fb\ua9fc\ua9fd\ua9fe\ua9ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_aa.png", - "ascent": %ascent%, - "chars": [ - "\uaa00\uaa01\uaa02\uaa03\uaa04\uaa05\uaa06\uaa07\uaa08\uaa09\uaa0a\uaa0b\uaa0c\uaa0d\uaa0e\uaa0f", - "\uaa10\uaa11\uaa12\uaa13\uaa14\uaa15\uaa16\uaa17\uaa18\uaa19\uaa1a\uaa1b\uaa1c\uaa1d\uaa1e\uaa1f", - "\uaa20\uaa21\uaa22\uaa23\uaa24\uaa25\uaa26\uaa27\uaa28\uaa29\uaa2a\uaa2b\uaa2c\uaa2d\uaa2e\uaa2f", - "\uaa30\uaa31\uaa32\uaa33\uaa34\uaa35\uaa36\uaa37\uaa38\uaa39\uaa3a\uaa3b\uaa3c\uaa3d\uaa3e\uaa3f", - "\uaa40\uaa41\uaa42\uaa43\uaa44\uaa45\uaa46\uaa47\uaa48\uaa49\uaa4a\uaa4b\uaa4c\uaa4d\uaa4e\uaa4f", - "\uaa50\uaa51\uaa52\uaa53\uaa54\uaa55\uaa56\uaa57\uaa58\uaa59\uaa5a\uaa5b\uaa5c\uaa5d\uaa5e\uaa5f", - "\uaa60\uaa61\uaa62\uaa63\uaa64\uaa65\uaa66\uaa67\uaa68\uaa69\uaa6a\uaa6b\uaa6c\uaa6d\uaa6e\uaa6f", - "\uaa70\uaa71\uaa72\uaa73\uaa74\uaa75\uaa76\uaa77\uaa78\uaa79\uaa7a\uaa7b\uaa7c\uaa7d\uaa7e\uaa7f", - "\uaa80\uaa81\uaa82\uaa83\uaa84\uaa85\uaa86\uaa87\uaa88\uaa89\uaa8a\uaa8b\uaa8c\uaa8d\uaa8e\uaa8f", - "\uaa90\uaa91\uaa92\uaa93\uaa94\uaa95\uaa96\uaa97\uaa98\uaa99\uaa9a\uaa9b\uaa9c\uaa9d\uaa9e\uaa9f", - "\uaaa0\uaaa1\uaaa2\uaaa3\uaaa4\uaaa5\uaaa6\uaaa7\uaaa8\uaaa9\uaaaa\uaaab\uaaac\uaaad\uaaae\uaaaf", - "\uaab0\uaab1\uaab2\uaab3\uaab4\uaab5\uaab6\uaab7\uaab8\uaab9\uaaba\uaabb\uaabc\uaabd\uaabe\uaabf", - "\uaac0\uaac1\uaac2\uaac3\uaac4\uaac5\uaac6\uaac7\uaac8\uaac9\uaaca\uaacb\uaacc\uaacd\uaace\uaacf", - "\uaad0\uaad1\uaad2\uaad3\uaad4\uaad5\uaad6\uaad7\uaad8\uaad9\uaada\uaadb\uaadc\uaadd\uaade\uaadf", - "\uaae0\uaae1\uaae2\uaae3\uaae4\uaae5\uaae6\uaae7\uaae8\uaae9\uaaea\uaaeb\uaaec\uaaed\uaaee\uaaef", - "\uaaf0\uaaf1\uaaf2\uaaf3\uaaf4\uaaf5\uaaf6\uaaf7\uaaf8\uaaf9\uaafa\uaafb\uaafc\uaafd\uaafe\uaaff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_ab.png", - "ascent": %ascent%, - "chars": [ - "\uab00\uab01\uab02\uab03\uab04\uab05\uab06\uab07\uab08\uab09\uab0a\uab0b\uab0c\uab0d\uab0e\uab0f", - "\uab10\uab11\uab12\uab13\uab14\uab15\uab16\uab17\uab18\uab19\uab1a\uab1b\uab1c\uab1d\uab1e\uab1f", - "\uab20\uab21\uab22\uab23\uab24\uab25\uab26\uab27\uab28\uab29\uab2a\uab2b\uab2c\uab2d\uab2e\uab2f", - "\uab30\uab31\uab32\uab33\uab34\uab35\uab36\uab37\uab38\uab39\uab3a\uab3b\uab3c\uab3d\uab3e\uab3f", - "\uab40\uab41\uab42\uab43\uab44\uab45\uab46\uab47\uab48\uab49\uab4a\uab4b\uab4c\uab4d\uab4e\uab4f", - "\uab50\uab51\uab52\uab53\uab54\uab55\uab56\uab57\uab58\uab59\uab5a\uab5b\uab5c\uab5d\uab5e\uab5f", - "\uab60\uab61\uab62\uab63\uab64\uab65\uab66\uab67\uab68\uab69\uab6a\uab6b\uab6c\uab6d\uab6e\uab6f", - "\uab70\uab71\uab72\uab73\uab74\uab75\uab76\uab77\uab78\uab79\uab7a\uab7b\uab7c\uab7d\uab7e\uab7f", - "\uab80\uab81\uab82\uab83\uab84\uab85\uab86\uab87\uab88\uab89\uab8a\uab8b\uab8c\uab8d\uab8e\uab8f", - "\uab90\uab91\uab92\uab93\uab94\uab95\uab96\uab97\uab98\uab99\uab9a\uab9b\uab9c\uab9d\uab9e\uab9f", - "\uaba0\uaba1\uaba2\uaba3\uaba4\uaba5\uaba6\uaba7\uaba8\uaba9\uabaa\uabab\uabac\uabad\uabae\uabaf", - "\uabb0\uabb1\uabb2\uabb3\uabb4\uabb5\uabb6\uabb7\uabb8\uabb9\uabba\uabbb\uabbc\uabbd\uabbe\uabbf", - "\uabc0\uabc1\uabc2\uabc3\uabc4\uabc5\uabc6\uabc7\uabc8\uabc9\uabca\uabcb\uabcc\uabcd\uabce\uabcf", - "\uabd0\uabd1\uabd2\uabd3\uabd4\uabd5\uabd6\uabd7\uabd8\uabd9\uabda\uabdb\uabdc\uabdd\uabde\uabdf", - "\uabe0\uabe1\uabe2\uabe3\uabe4\uabe5\uabe6\uabe7\uabe8\uabe9\uabea\uabeb\uabec\uabed\uabee\uabef", - "\uabf0\uabf1\uabf2\uabf3\uabf4\uabf5\uabf6\uabf7\uabf8\uabf9\uabfa\uabfb\uabfc\uabfd\uabfe\uabff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_ac.png", - "ascent": %ascent%, - "chars": [ - "\uac00\uac01\uac02\uac03\uac04\uac05\uac06\uac07\uac08\uac09\uac0a\uac0b\uac0c\uac0d\uac0e\uac0f", - "\uac10\uac11\uac12\uac13\uac14\uac15\uac16\uac17\uac18\uac19\uac1a\uac1b\uac1c\uac1d\uac1e\uac1f", - "\uac20\uac21\uac22\uac23\uac24\uac25\uac26\uac27\uac28\uac29\uac2a\uac2b\uac2c\uac2d\uac2e\uac2f", - "\uac30\uac31\uac32\uac33\uac34\uac35\uac36\uac37\uac38\uac39\uac3a\uac3b\uac3c\uac3d\uac3e\uac3f", - "\uac40\uac41\uac42\uac43\uac44\uac45\uac46\uac47\uac48\uac49\uac4a\uac4b\uac4c\uac4d\uac4e\uac4f", - "\uac50\uac51\uac52\uac53\uac54\uac55\uac56\uac57\uac58\uac59\uac5a\uac5b\uac5c\uac5d\uac5e\uac5f", - "\uac60\uac61\uac62\uac63\uac64\uac65\uac66\uac67\uac68\uac69\uac6a\uac6b\uac6c\uac6d\uac6e\uac6f", - "\uac70\uac71\uac72\uac73\uac74\uac75\uac76\uac77\uac78\uac79\uac7a\uac7b\uac7c\uac7d\uac7e\uac7f", - "\uac80\uac81\uac82\uac83\uac84\uac85\uac86\uac87\uac88\uac89\uac8a\uac8b\uac8c\uac8d\uac8e\uac8f", - "\uac90\uac91\uac92\uac93\uac94\uac95\uac96\uac97\uac98\uac99\uac9a\uac9b\uac9c\uac9d\uac9e\uac9f", - "\uaca0\uaca1\uaca2\uaca3\uaca4\uaca5\uaca6\uaca7\uaca8\uaca9\uacaa\uacab\uacac\uacad\uacae\uacaf", - "\uacb0\uacb1\uacb2\uacb3\uacb4\uacb5\uacb6\uacb7\uacb8\uacb9\uacba\uacbb\uacbc\uacbd\uacbe\uacbf", - "\uacc0\uacc1\uacc2\uacc3\uacc4\uacc5\uacc6\uacc7\uacc8\uacc9\uacca\uaccb\uaccc\uaccd\uacce\uaccf", - "\uacd0\uacd1\uacd2\uacd3\uacd4\uacd5\uacd6\uacd7\uacd8\uacd9\uacda\uacdb\uacdc\uacdd\uacde\uacdf", - "\uace0\uace1\uace2\uace3\uace4\uace5\uace6\uace7\uace8\uace9\uacea\uaceb\uacec\uaced\uacee\uacef", - "\uacf0\uacf1\uacf2\uacf3\uacf4\uacf5\uacf6\uacf7\uacf8\uacf9\uacfa\uacfb\uacfc\uacfd\uacfe\uacff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_ad.png", - "ascent": %ascent%, - "chars": [ - "\uad00\uad01\uad02\uad03\uad04\uad05\uad06\uad07\uad08\uad09\uad0a\uad0b\uad0c\uad0d\uad0e\uad0f", - "\uad10\uad11\uad12\uad13\uad14\uad15\uad16\uad17\uad18\uad19\uad1a\uad1b\uad1c\uad1d\uad1e\uad1f", - "\uad20\uad21\uad22\uad23\uad24\uad25\uad26\uad27\uad28\uad29\uad2a\uad2b\uad2c\uad2d\uad2e\uad2f", - "\uad30\uad31\uad32\uad33\uad34\uad35\uad36\uad37\uad38\uad39\uad3a\uad3b\uad3c\uad3d\uad3e\uad3f", - "\uad40\uad41\uad42\uad43\uad44\uad45\uad46\uad47\uad48\uad49\uad4a\uad4b\uad4c\uad4d\uad4e\uad4f", - "\uad50\uad51\uad52\uad53\uad54\uad55\uad56\uad57\uad58\uad59\uad5a\uad5b\uad5c\uad5d\uad5e\uad5f", - "\uad60\uad61\uad62\uad63\uad64\uad65\uad66\uad67\uad68\uad69\uad6a\uad6b\uad6c\uad6d\uad6e\uad6f", - "\uad70\uad71\uad72\uad73\uad74\uad75\uad76\uad77\uad78\uad79\uad7a\uad7b\uad7c\uad7d\uad7e\uad7f", - "\uad80\uad81\uad82\uad83\uad84\uad85\uad86\uad87\uad88\uad89\uad8a\uad8b\uad8c\uad8d\uad8e\uad8f", - "\uad90\uad91\uad92\uad93\uad94\uad95\uad96\uad97\uad98\uad99\uad9a\uad9b\uad9c\uad9d\uad9e\uad9f", - "\uada0\uada1\uada2\uada3\uada4\uada5\uada6\uada7\uada8\uada9\uadaa\uadab\uadac\uadad\uadae\uadaf", - "\uadb0\uadb1\uadb2\uadb3\uadb4\uadb5\uadb6\uadb7\uadb8\uadb9\uadba\uadbb\uadbc\uadbd\uadbe\uadbf", - "\uadc0\uadc1\uadc2\uadc3\uadc4\uadc5\uadc6\uadc7\uadc8\uadc9\uadca\uadcb\uadcc\uadcd\uadce\uadcf", - "\uadd0\uadd1\uadd2\uadd3\uadd4\uadd5\uadd6\uadd7\uadd8\uadd9\uadda\uaddb\uaddc\uaddd\uadde\uaddf", - "\uade0\uade1\uade2\uade3\uade4\uade5\uade6\uade7\uade8\uade9\uadea\uadeb\uadec\uaded\uadee\uadef", - "\uadf0\uadf1\uadf2\uadf3\uadf4\uadf5\uadf6\uadf7\uadf8\uadf9\uadfa\uadfb\uadfc\uadfd\uadfe\uadff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_ae.png", - "ascent": %ascent%, - "chars": [ - "\uae00\uae01\uae02\uae03\uae04\uae05\uae06\uae07\uae08\uae09\uae0a\uae0b\uae0c\uae0d\uae0e\uae0f", - "\uae10\uae11\uae12\uae13\uae14\uae15\uae16\uae17\uae18\uae19\uae1a\uae1b\uae1c\uae1d\uae1e\uae1f", - "\uae20\uae21\uae22\uae23\uae24\uae25\uae26\uae27\uae28\uae29\uae2a\uae2b\uae2c\uae2d\uae2e\uae2f", - "\uae30\uae31\uae32\uae33\uae34\uae35\uae36\uae37\uae38\uae39\uae3a\uae3b\uae3c\uae3d\uae3e\uae3f", - "\uae40\uae41\uae42\uae43\uae44\uae45\uae46\uae47\uae48\uae49\uae4a\uae4b\uae4c\uae4d\uae4e\uae4f", - "\uae50\uae51\uae52\uae53\uae54\uae55\uae56\uae57\uae58\uae59\uae5a\uae5b\uae5c\uae5d\uae5e\uae5f", - "\uae60\uae61\uae62\uae63\uae64\uae65\uae66\uae67\uae68\uae69\uae6a\uae6b\uae6c\uae6d\uae6e\uae6f", - "\uae70\uae71\uae72\uae73\uae74\uae75\uae76\uae77\uae78\uae79\uae7a\uae7b\uae7c\uae7d\uae7e\uae7f", - "\uae80\uae81\uae82\uae83\uae84\uae85\uae86\uae87\uae88\uae89\uae8a\uae8b\uae8c\uae8d\uae8e\uae8f", - "\uae90\uae91\uae92\uae93\uae94\uae95\uae96\uae97\uae98\uae99\uae9a\uae9b\uae9c\uae9d\uae9e\uae9f", - "\uaea0\uaea1\uaea2\uaea3\uaea4\uaea5\uaea6\uaea7\uaea8\uaea9\uaeaa\uaeab\uaeac\uaead\uaeae\uaeaf", - "\uaeb0\uaeb1\uaeb2\uaeb3\uaeb4\uaeb5\uaeb6\uaeb7\uaeb8\uaeb9\uaeba\uaebb\uaebc\uaebd\uaebe\uaebf", - "\uaec0\uaec1\uaec2\uaec3\uaec4\uaec5\uaec6\uaec7\uaec8\uaec9\uaeca\uaecb\uaecc\uaecd\uaece\uaecf", - "\uaed0\uaed1\uaed2\uaed3\uaed4\uaed5\uaed6\uaed7\uaed8\uaed9\uaeda\uaedb\uaedc\uaedd\uaede\uaedf", - "\uaee0\uaee1\uaee2\uaee3\uaee4\uaee5\uaee6\uaee7\uaee8\uaee9\uaeea\uaeeb\uaeec\uaeed\uaeee\uaeef", - "\uaef0\uaef1\uaef2\uaef3\uaef4\uaef5\uaef6\uaef7\uaef8\uaef9\uaefa\uaefb\uaefc\uaefd\uaefe\uaeff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_af.png", - "ascent": %ascent%, - "chars": [ - "\uaf00\uaf01\uaf02\uaf03\uaf04\uaf05\uaf06\uaf07\uaf08\uaf09\uaf0a\uaf0b\uaf0c\uaf0d\uaf0e\uaf0f", - "\uaf10\uaf11\uaf12\uaf13\uaf14\uaf15\uaf16\uaf17\uaf18\uaf19\uaf1a\uaf1b\uaf1c\uaf1d\uaf1e\uaf1f", - "\uaf20\uaf21\uaf22\uaf23\uaf24\uaf25\uaf26\uaf27\uaf28\uaf29\uaf2a\uaf2b\uaf2c\uaf2d\uaf2e\uaf2f", - "\uaf30\uaf31\uaf32\uaf33\uaf34\uaf35\uaf36\uaf37\uaf38\uaf39\uaf3a\uaf3b\uaf3c\uaf3d\uaf3e\uaf3f", - "\uaf40\uaf41\uaf42\uaf43\uaf44\uaf45\uaf46\uaf47\uaf48\uaf49\uaf4a\uaf4b\uaf4c\uaf4d\uaf4e\uaf4f", - "\uaf50\uaf51\uaf52\uaf53\uaf54\uaf55\uaf56\uaf57\uaf58\uaf59\uaf5a\uaf5b\uaf5c\uaf5d\uaf5e\uaf5f", - "\uaf60\uaf61\uaf62\uaf63\uaf64\uaf65\uaf66\uaf67\uaf68\uaf69\uaf6a\uaf6b\uaf6c\uaf6d\uaf6e\uaf6f", - "\uaf70\uaf71\uaf72\uaf73\uaf74\uaf75\uaf76\uaf77\uaf78\uaf79\uaf7a\uaf7b\uaf7c\uaf7d\uaf7e\uaf7f", - "\uaf80\uaf81\uaf82\uaf83\uaf84\uaf85\uaf86\uaf87\uaf88\uaf89\uaf8a\uaf8b\uaf8c\uaf8d\uaf8e\uaf8f", - "\uaf90\uaf91\uaf92\uaf93\uaf94\uaf95\uaf96\uaf97\uaf98\uaf99\uaf9a\uaf9b\uaf9c\uaf9d\uaf9e\uaf9f", - "\uafa0\uafa1\uafa2\uafa3\uafa4\uafa5\uafa6\uafa7\uafa8\uafa9\uafaa\uafab\uafac\uafad\uafae\uafaf", - "\uafb0\uafb1\uafb2\uafb3\uafb4\uafb5\uafb6\uafb7\uafb8\uafb9\uafba\uafbb\uafbc\uafbd\uafbe\uafbf", - "\uafc0\uafc1\uafc2\uafc3\uafc4\uafc5\uafc6\uafc7\uafc8\uafc9\uafca\uafcb\uafcc\uafcd\uafce\uafcf", - "\uafd0\uafd1\uafd2\uafd3\uafd4\uafd5\uafd6\uafd7\uafd8\uafd9\uafda\uafdb\uafdc\uafdd\uafde\uafdf", - "\uafe0\uafe1\uafe2\uafe3\uafe4\uafe5\uafe6\uafe7\uafe8\uafe9\uafea\uafeb\uafec\uafed\uafee\uafef", - "\uaff0\uaff1\uaff2\uaff3\uaff4\uaff5\uaff6\uaff7\uaff8\uaff9\uaffa\uaffb\uaffc\uaffd\uaffe\uafff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_b0.png", - "ascent": %ascent%, - "chars": [ - "\ub000\ub001\ub002\ub003\ub004\ub005\ub006\ub007\ub008\ub009\ub00a\ub00b\ub00c\ub00d\ub00e\ub00f", - "\ub010\ub011\ub012\ub013\ub014\ub015\ub016\ub017\ub018\ub019\ub01a\ub01b\ub01c\ub01d\ub01e\ub01f", - "\ub020\ub021\ub022\ub023\ub024\ub025\ub026\ub027\ub028\ub029\ub02a\ub02b\ub02c\ub02d\ub02e\ub02f", - "\ub030\ub031\ub032\ub033\ub034\ub035\ub036\ub037\ub038\ub039\ub03a\ub03b\ub03c\ub03d\ub03e\ub03f", - "\ub040\ub041\ub042\ub043\ub044\ub045\ub046\ub047\ub048\ub049\ub04a\ub04b\ub04c\ub04d\ub04e\ub04f", - "\ub050\ub051\ub052\ub053\ub054\ub055\ub056\ub057\ub058\ub059\ub05a\ub05b\ub05c\ub05d\ub05e\ub05f", - "\ub060\ub061\ub062\ub063\ub064\ub065\ub066\ub067\ub068\ub069\ub06a\ub06b\ub06c\ub06d\ub06e\ub06f", - "\ub070\ub071\ub072\ub073\ub074\ub075\ub076\ub077\ub078\ub079\ub07a\ub07b\ub07c\ub07d\ub07e\ub07f", - "\ub080\ub081\ub082\ub083\ub084\ub085\ub086\ub087\ub088\ub089\ub08a\ub08b\ub08c\ub08d\ub08e\ub08f", - "\ub090\ub091\ub092\ub093\ub094\ub095\ub096\ub097\ub098\ub099\ub09a\ub09b\ub09c\ub09d\ub09e\ub09f", - "\ub0a0\ub0a1\ub0a2\ub0a3\ub0a4\ub0a5\ub0a6\ub0a7\ub0a8\ub0a9\ub0aa\ub0ab\ub0ac\ub0ad\ub0ae\ub0af", - "\ub0b0\ub0b1\ub0b2\ub0b3\ub0b4\ub0b5\ub0b6\ub0b7\ub0b8\ub0b9\ub0ba\ub0bb\ub0bc\ub0bd\ub0be\ub0bf", - "\ub0c0\ub0c1\ub0c2\ub0c3\ub0c4\ub0c5\ub0c6\ub0c7\ub0c8\ub0c9\ub0ca\ub0cb\ub0cc\ub0cd\ub0ce\ub0cf", - "\ub0d0\ub0d1\ub0d2\ub0d3\ub0d4\ub0d5\ub0d6\ub0d7\ub0d8\ub0d9\ub0da\ub0db\ub0dc\ub0dd\ub0de\ub0df", - "\ub0e0\ub0e1\ub0e2\ub0e3\ub0e4\ub0e5\ub0e6\ub0e7\ub0e8\ub0e9\ub0ea\ub0eb\ub0ec\ub0ed\ub0ee\ub0ef", - "\ub0f0\ub0f1\ub0f2\ub0f3\ub0f4\ub0f5\ub0f6\ub0f7\ub0f8\ub0f9\ub0fa\ub0fb\ub0fc\ub0fd\ub0fe\ub0ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_b1.png", - "ascent": %ascent%, - "chars": [ - "\ub100\ub101\ub102\ub103\ub104\ub105\ub106\ub107\ub108\ub109\ub10a\ub10b\ub10c\ub10d\ub10e\ub10f", - "\ub110\ub111\ub112\ub113\ub114\ub115\ub116\ub117\ub118\ub119\ub11a\ub11b\ub11c\ub11d\ub11e\ub11f", - "\ub120\ub121\ub122\ub123\ub124\ub125\ub126\ub127\ub128\ub129\ub12a\ub12b\ub12c\ub12d\ub12e\ub12f", - "\ub130\ub131\ub132\ub133\ub134\ub135\ub136\ub137\ub138\ub139\ub13a\ub13b\ub13c\ub13d\ub13e\ub13f", - "\ub140\ub141\ub142\ub143\ub144\ub145\ub146\ub147\ub148\ub149\ub14a\ub14b\ub14c\ub14d\ub14e\ub14f", - "\ub150\ub151\ub152\ub153\ub154\ub155\ub156\ub157\ub158\ub159\ub15a\ub15b\ub15c\ub15d\ub15e\ub15f", - "\ub160\ub161\ub162\ub163\ub164\ub165\ub166\ub167\ub168\ub169\ub16a\ub16b\ub16c\ub16d\ub16e\ub16f", - "\ub170\ub171\ub172\ub173\ub174\ub175\ub176\ub177\ub178\ub179\ub17a\ub17b\ub17c\ub17d\ub17e\ub17f", - "\ub180\ub181\ub182\ub183\ub184\ub185\ub186\ub187\ub188\ub189\ub18a\ub18b\ub18c\ub18d\ub18e\ub18f", - "\ub190\ub191\ub192\ub193\ub194\ub195\ub196\ub197\ub198\ub199\ub19a\ub19b\ub19c\ub19d\ub19e\ub19f", - "\ub1a0\ub1a1\ub1a2\ub1a3\ub1a4\ub1a5\ub1a6\ub1a7\ub1a8\ub1a9\ub1aa\ub1ab\ub1ac\ub1ad\ub1ae\ub1af", - "\ub1b0\ub1b1\ub1b2\ub1b3\ub1b4\ub1b5\ub1b6\ub1b7\ub1b8\ub1b9\ub1ba\ub1bb\ub1bc\ub1bd\ub1be\ub1bf", - "\ub1c0\ub1c1\ub1c2\ub1c3\ub1c4\ub1c5\ub1c6\ub1c7\ub1c8\ub1c9\ub1ca\ub1cb\ub1cc\ub1cd\ub1ce\ub1cf", - "\ub1d0\ub1d1\ub1d2\ub1d3\ub1d4\ub1d5\ub1d6\ub1d7\ub1d8\ub1d9\ub1da\ub1db\ub1dc\ub1dd\ub1de\ub1df", - "\ub1e0\ub1e1\ub1e2\ub1e3\ub1e4\ub1e5\ub1e6\ub1e7\ub1e8\ub1e9\ub1ea\ub1eb\ub1ec\ub1ed\ub1ee\ub1ef", - "\ub1f0\ub1f1\ub1f2\ub1f3\ub1f4\ub1f5\ub1f6\ub1f7\ub1f8\ub1f9\ub1fa\ub1fb\ub1fc\ub1fd\ub1fe\ub1ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_b2.png", - "ascent": %ascent%, - "chars": [ - "\ub200\ub201\ub202\ub203\ub204\ub205\ub206\ub207\ub208\ub209\ub20a\ub20b\ub20c\ub20d\ub20e\ub20f", - "\ub210\ub211\ub212\ub213\ub214\ub215\ub216\ub217\ub218\ub219\ub21a\ub21b\ub21c\ub21d\ub21e\ub21f", - "\ub220\ub221\ub222\ub223\ub224\ub225\ub226\ub227\ub228\ub229\ub22a\ub22b\ub22c\ub22d\ub22e\ub22f", - "\ub230\ub231\ub232\ub233\ub234\ub235\ub236\ub237\ub238\ub239\ub23a\ub23b\ub23c\ub23d\ub23e\ub23f", - "\ub240\ub241\ub242\ub243\ub244\ub245\ub246\ub247\ub248\ub249\ub24a\ub24b\ub24c\ub24d\ub24e\ub24f", - "\ub250\ub251\ub252\ub253\ub254\ub255\ub256\ub257\ub258\ub259\ub25a\ub25b\ub25c\ub25d\ub25e\ub25f", - "\ub260\ub261\ub262\ub263\ub264\ub265\ub266\ub267\ub268\ub269\ub26a\ub26b\ub26c\ub26d\ub26e\ub26f", - "\ub270\ub271\ub272\ub273\ub274\ub275\ub276\ub277\ub278\ub279\ub27a\ub27b\ub27c\ub27d\ub27e\ub27f", - "\ub280\ub281\ub282\ub283\ub284\ub285\ub286\ub287\ub288\ub289\ub28a\ub28b\ub28c\ub28d\ub28e\ub28f", - "\ub290\ub291\ub292\ub293\ub294\ub295\ub296\ub297\ub298\ub299\ub29a\ub29b\ub29c\ub29d\ub29e\ub29f", - "\ub2a0\ub2a1\ub2a2\ub2a3\ub2a4\ub2a5\ub2a6\ub2a7\ub2a8\ub2a9\ub2aa\ub2ab\ub2ac\ub2ad\ub2ae\ub2af", - "\ub2b0\ub2b1\ub2b2\ub2b3\ub2b4\ub2b5\ub2b6\ub2b7\ub2b8\ub2b9\ub2ba\ub2bb\ub2bc\ub2bd\ub2be\ub2bf", - "\ub2c0\ub2c1\ub2c2\ub2c3\ub2c4\ub2c5\ub2c6\ub2c7\ub2c8\ub2c9\ub2ca\ub2cb\ub2cc\ub2cd\ub2ce\ub2cf", - "\ub2d0\ub2d1\ub2d2\ub2d3\ub2d4\ub2d5\ub2d6\ub2d7\ub2d8\ub2d9\ub2da\ub2db\ub2dc\ub2dd\ub2de\ub2df", - "\ub2e0\ub2e1\ub2e2\ub2e3\ub2e4\ub2e5\ub2e6\ub2e7\ub2e8\ub2e9\ub2ea\ub2eb\ub2ec\ub2ed\ub2ee\ub2ef", - "\ub2f0\ub2f1\ub2f2\ub2f3\ub2f4\ub2f5\ub2f6\ub2f7\ub2f8\ub2f9\ub2fa\ub2fb\ub2fc\ub2fd\ub2fe\ub2ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_b3.png", - "ascent": %ascent%, - "chars": [ - "\ub300\ub301\ub302\ub303\ub304\ub305\ub306\ub307\ub308\ub309\ub30a\ub30b\ub30c\ub30d\ub30e\ub30f", - "\ub310\ub311\ub312\ub313\ub314\ub315\ub316\ub317\ub318\ub319\ub31a\ub31b\ub31c\ub31d\ub31e\ub31f", - "\ub320\ub321\ub322\ub323\ub324\ub325\ub326\ub327\ub328\ub329\ub32a\ub32b\ub32c\ub32d\ub32e\ub32f", - "\ub330\ub331\ub332\ub333\ub334\ub335\ub336\ub337\ub338\ub339\ub33a\ub33b\ub33c\ub33d\ub33e\ub33f", - "\ub340\ub341\ub342\ub343\ub344\ub345\ub346\ub347\ub348\ub349\ub34a\ub34b\ub34c\ub34d\ub34e\ub34f", - "\ub350\ub351\ub352\ub353\ub354\ub355\ub356\ub357\ub358\ub359\ub35a\ub35b\ub35c\ub35d\ub35e\ub35f", - "\ub360\ub361\ub362\ub363\ub364\ub365\ub366\ub367\ub368\ub369\ub36a\ub36b\ub36c\ub36d\ub36e\ub36f", - "\ub370\ub371\ub372\ub373\ub374\ub375\ub376\ub377\ub378\ub379\ub37a\ub37b\ub37c\ub37d\ub37e\ub37f", - "\ub380\ub381\ub382\ub383\ub384\ub385\ub386\ub387\ub388\ub389\ub38a\ub38b\ub38c\ub38d\ub38e\ub38f", - "\ub390\ub391\ub392\ub393\ub394\ub395\ub396\ub397\ub398\ub399\ub39a\ub39b\ub39c\ub39d\ub39e\ub39f", - "\ub3a0\ub3a1\ub3a2\ub3a3\ub3a4\ub3a5\ub3a6\ub3a7\ub3a8\ub3a9\ub3aa\ub3ab\ub3ac\ub3ad\ub3ae\ub3af", - "\ub3b0\ub3b1\ub3b2\ub3b3\ub3b4\ub3b5\ub3b6\ub3b7\ub3b8\ub3b9\ub3ba\ub3bb\ub3bc\ub3bd\ub3be\ub3bf", - "\ub3c0\ub3c1\ub3c2\ub3c3\ub3c4\ub3c5\ub3c6\ub3c7\ub3c8\ub3c9\ub3ca\ub3cb\ub3cc\ub3cd\ub3ce\ub3cf", - "\ub3d0\ub3d1\ub3d2\ub3d3\ub3d4\ub3d5\ub3d6\ub3d7\ub3d8\ub3d9\ub3da\ub3db\ub3dc\ub3dd\ub3de\ub3df", - "\ub3e0\ub3e1\ub3e2\ub3e3\ub3e4\ub3e5\ub3e6\ub3e7\ub3e8\ub3e9\ub3ea\ub3eb\ub3ec\ub3ed\ub3ee\ub3ef", - "\ub3f0\ub3f1\ub3f2\ub3f3\ub3f4\ub3f5\ub3f6\ub3f7\ub3f8\ub3f9\ub3fa\ub3fb\ub3fc\ub3fd\ub3fe\ub3ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_b4.png", - "ascent": %ascent%, - "chars": [ - "\ub400\ub401\ub402\ub403\ub404\ub405\ub406\ub407\ub408\ub409\ub40a\ub40b\ub40c\ub40d\ub40e\ub40f", - "\ub410\ub411\ub412\ub413\ub414\ub415\ub416\ub417\ub418\ub419\ub41a\ub41b\ub41c\ub41d\ub41e\ub41f", - "\ub420\ub421\ub422\ub423\ub424\ub425\ub426\ub427\ub428\ub429\ub42a\ub42b\ub42c\ub42d\ub42e\ub42f", - "\ub430\ub431\ub432\ub433\ub434\ub435\ub436\ub437\ub438\ub439\ub43a\ub43b\ub43c\ub43d\ub43e\ub43f", - "\ub440\ub441\ub442\ub443\ub444\ub445\ub446\ub447\ub448\ub449\ub44a\ub44b\ub44c\ub44d\ub44e\ub44f", - "\ub450\ub451\ub452\ub453\ub454\ub455\ub456\ub457\ub458\ub459\ub45a\ub45b\ub45c\ub45d\ub45e\ub45f", - "\ub460\ub461\ub462\ub463\ub464\ub465\ub466\ub467\ub468\ub469\ub46a\ub46b\ub46c\ub46d\ub46e\ub46f", - "\ub470\ub471\ub472\ub473\ub474\ub475\ub476\ub477\ub478\ub479\ub47a\ub47b\ub47c\ub47d\ub47e\ub47f", - "\ub480\ub481\ub482\ub483\ub484\ub485\ub486\ub487\ub488\ub489\ub48a\ub48b\ub48c\ub48d\ub48e\ub48f", - "\ub490\ub491\ub492\ub493\ub494\ub495\ub496\ub497\ub498\ub499\ub49a\ub49b\ub49c\ub49d\ub49e\ub49f", - "\ub4a0\ub4a1\ub4a2\ub4a3\ub4a4\ub4a5\ub4a6\ub4a7\ub4a8\ub4a9\ub4aa\ub4ab\ub4ac\ub4ad\ub4ae\ub4af", - "\ub4b0\ub4b1\ub4b2\ub4b3\ub4b4\ub4b5\ub4b6\ub4b7\ub4b8\ub4b9\ub4ba\ub4bb\ub4bc\ub4bd\ub4be\ub4bf", - "\ub4c0\ub4c1\ub4c2\ub4c3\ub4c4\ub4c5\ub4c6\ub4c7\ub4c8\ub4c9\ub4ca\ub4cb\ub4cc\ub4cd\ub4ce\ub4cf", - "\ub4d0\ub4d1\ub4d2\ub4d3\ub4d4\ub4d5\ub4d6\ub4d7\ub4d8\ub4d9\ub4da\ub4db\ub4dc\ub4dd\ub4de\ub4df", - "\ub4e0\ub4e1\ub4e2\ub4e3\ub4e4\ub4e5\ub4e6\ub4e7\ub4e8\ub4e9\ub4ea\ub4eb\ub4ec\ub4ed\ub4ee\ub4ef", - "\ub4f0\ub4f1\ub4f2\ub4f3\ub4f4\ub4f5\ub4f6\ub4f7\ub4f8\ub4f9\ub4fa\ub4fb\ub4fc\ub4fd\ub4fe\ub4ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_b5.png", - "ascent": %ascent%, - "chars": [ - "\ub500\ub501\ub502\ub503\ub504\ub505\ub506\ub507\ub508\ub509\ub50a\ub50b\ub50c\ub50d\ub50e\ub50f", - "\ub510\ub511\ub512\ub513\ub514\ub515\ub516\ub517\ub518\ub519\ub51a\ub51b\ub51c\ub51d\ub51e\ub51f", - "\ub520\ub521\ub522\ub523\ub524\ub525\ub526\ub527\ub528\ub529\ub52a\ub52b\ub52c\ub52d\ub52e\ub52f", - "\ub530\ub531\ub532\ub533\ub534\ub535\ub536\ub537\ub538\ub539\ub53a\ub53b\ub53c\ub53d\ub53e\ub53f", - "\ub540\ub541\ub542\ub543\ub544\ub545\ub546\ub547\ub548\ub549\ub54a\ub54b\ub54c\ub54d\ub54e\ub54f", - "\ub550\ub551\ub552\ub553\ub554\ub555\ub556\ub557\ub558\ub559\ub55a\ub55b\ub55c\ub55d\ub55e\ub55f", - "\ub560\ub561\ub562\ub563\ub564\ub565\ub566\ub567\ub568\ub569\ub56a\ub56b\ub56c\ub56d\ub56e\ub56f", - "\ub570\ub571\ub572\ub573\ub574\ub575\ub576\ub577\ub578\ub579\ub57a\ub57b\ub57c\ub57d\ub57e\ub57f", - "\ub580\ub581\ub582\ub583\ub584\ub585\ub586\ub587\ub588\ub589\ub58a\ub58b\ub58c\ub58d\ub58e\ub58f", - "\ub590\ub591\ub592\ub593\ub594\ub595\ub596\ub597\ub598\ub599\ub59a\ub59b\ub59c\ub59d\ub59e\ub59f", - "\ub5a0\ub5a1\ub5a2\ub5a3\ub5a4\ub5a5\ub5a6\ub5a7\ub5a8\ub5a9\ub5aa\ub5ab\ub5ac\ub5ad\ub5ae\ub5af", - "\ub5b0\ub5b1\ub5b2\ub5b3\ub5b4\ub5b5\ub5b6\ub5b7\ub5b8\ub5b9\ub5ba\ub5bb\ub5bc\ub5bd\ub5be\ub5bf", - "\ub5c0\ub5c1\ub5c2\ub5c3\ub5c4\ub5c5\ub5c6\ub5c7\ub5c8\ub5c9\ub5ca\ub5cb\ub5cc\ub5cd\ub5ce\ub5cf", - "\ub5d0\ub5d1\ub5d2\ub5d3\ub5d4\ub5d5\ub5d6\ub5d7\ub5d8\ub5d9\ub5da\ub5db\ub5dc\ub5dd\ub5de\ub5df", - "\ub5e0\ub5e1\ub5e2\ub5e3\ub5e4\ub5e5\ub5e6\ub5e7\ub5e8\ub5e9\ub5ea\ub5eb\ub5ec\ub5ed\ub5ee\ub5ef", - "\ub5f0\ub5f1\ub5f2\ub5f3\ub5f4\ub5f5\ub5f6\ub5f7\ub5f8\ub5f9\ub5fa\ub5fb\ub5fc\ub5fd\ub5fe\ub5ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_b6.png", - "ascent": %ascent%, - "chars": [ - "\ub600\ub601\ub602\ub603\ub604\ub605\ub606\ub607\ub608\ub609\ub60a\ub60b\ub60c\ub60d\ub60e\ub60f", - "\ub610\ub611\ub612\ub613\ub614\ub615\ub616\ub617\ub618\ub619\ub61a\ub61b\ub61c\ub61d\ub61e\ub61f", - "\ub620\ub621\ub622\ub623\ub624\ub625\ub626\ub627\ub628\ub629\ub62a\ub62b\ub62c\ub62d\ub62e\ub62f", - "\ub630\ub631\ub632\ub633\ub634\ub635\ub636\ub637\ub638\ub639\ub63a\ub63b\ub63c\ub63d\ub63e\ub63f", - "\ub640\ub641\ub642\ub643\ub644\ub645\ub646\ub647\ub648\ub649\ub64a\ub64b\ub64c\ub64d\ub64e\ub64f", - "\ub650\ub651\ub652\ub653\ub654\ub655\ub656\ub657\ub658\ub659\ub65a\ub65b\ub65c\ub65d\ub65e\ub65f", - "\ub660\ub661\ub662\ub663\ub664\ub665\ub666\ub667\ub668\ub669\ub66a\ub66b\ub66c\ub66d\ub66e\ub66f", - "\ub670\ub671\ub672\ub673\ub674\ub675\ub676\ub677\ub678\ub679\ub67a\ub67b\ub67c\ub67d\ub67e\ub67f", - "\ub680\ub681\ub682\ub683\ub684\ub685\ub686\ub687\ub688\ub689\ub68a\ub68b\ub68c\ub68d\ub68e\ub68f", - "\ub690\ub691\ub692\ub693\ub694\ub695\ub696\ub697\ub698\ub699\ub69a\ub69b\ub69c\ub69d\ub69e\ub69f", - "\ub6a0\ub6a1\ub6a2\ub6a3\ub6a4\ub6a5\ub6a6\ub6a7\ub6a8\ub6a9\ub6aa\ub6ab\ub6ac\ub6ad\ub6ae\ub6af", - "\ub6b0\ub6b1\ub6b2\ub6b3\ub6b4\ub6b5\ub6b6\ub6b7\ub6b8\ub6b9\ub6ba\ub6bb\ub6bc\ub6bd\ub6be\ub6bf", - "\ub6c0\ub6c1\ub6c2\ub6c3\ub6c4\ub6c5\ub6c6\ub6c7\ub6c8\ub6c9\ub6ca\ub6cb\ub6cc\ub6cd\ub6ce\ub6cf", - "\ub6d0\ub6d1\ub6d2\ub6d3\ub6d4\ub6d5\ub6d6\ub6d7\ub6d8\ub6d9\ub6da\ub6db\ub6dc\ub6dd\ub6de\ub6df", - "\ub6e0\ub6e1\ub6e2\ub6e3\ub6e4\ub6e5\ub6e6\ub6e7\ub6e8\ub6e9\ub6ea\ub6eb\ub6ec\ub6ed\ub6ee\ub6ef", - "\ub6f0\ub6f1\ub6f2\ub6f3\ub6f4\ub6f5\ub6f6\ub6f7\ub6f8\ub6f9\ub6fa\ub6fb\ub6fc\ub6fd\ub6fe\ub6ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_b7.png", - "ascent": %ascent%, - "chars": [ - "\ub700\ub701\ub702\ub703\ub704\ub705\ub706\ub707\ub708\ub709\ub70a\ub70b\ub70c\ub70d\ub70e\ub70f", - "\ub710\ub711\ub712\ub713\ub714\ub715\ub716\ub717\ub718\ub719\ub71a\ub71b\ub71c\ub71d\ub71e\ub71f", - "\ub720\ub721\ub722\ub723\ub724\ub725\ub726\ub727\ub728\ub729\ub72a\ub72b\ub72c\ub72d\ub72e\ub72f", - "\ub730\ub731\ub732\ub733\ub734\ub735\ub736\ub737\ub738\ub739\ub73a\ub73b\ub73c\ub73d\ub73e\ub73f", - "\ub740\ub741\ub742\ub743\ub744\ub745\ub746\ub747\ub748\ub749\ub74a\ub74b\ub74c\ub74d\ub74e\ub74f", - "\ub750\ub751\ub752\ub753\ub754\ub755\ub756\ub757\ub758\ub759\ub75a\ub75b\ub75c\ub75d\ub75e\ub75f", - "\ub760\ub761\ub762\ub763\ub764\ub765\ub766\ub767\ub768\ub769\ub76a\ub76b\ub76c\ub76d\ub76e\ub76f", - "\ub770\ub771\ub772\ub773\ub774\ub775\ub776\ub777\ub778\ub779\ub77a\ub77b\ub77c\ub77d\ub77e\ub77f", - "\ub780\ub781\ub782\ub783\ub784\ub785\ub786\ub787\ub788\ub789\ub78a\ub78b\ub78c\ub78d\ub78e\ub78f", - "\ub790\ub791\ub792\ub793\ub794\ub795\ub796\ub797\ub798\ub799\ub79a\ub79b\ub79c\ub79d\ub79e\ub79f", - "\ub7a0\ub7a1\ub7a2\ub7a3\ub7a4\ub7a5\ub7a6\ub7a7\ub7a8\ub7a9\ub7aa\ub7ab\ub7ac\ub7ad\ub7ae\ub7af", - "\ub7b0\ub7b1\ub7b2\ub7b3\ub7b4\ub7b5\ub7b6\ub7b7\ub7b8\ub7b9\ub7ba\ub7bb\ub7bc\ub7bd\ub7be\ub7bf", - "\ub7c0\ub7c1\ub7c2\ub7c3\ub7c4\ub7c5\ub7c6\ub7c7\ub7c8\ub7c9\ub7ca\ub7cb\ub7cc\ub7cd\ub7ce\ub7cf", - "\ub7d0\ub7d1\ub7d2\ub7d3\ub7d4\ub7d5\ub7d6\ub7d7\ub7d8\ub7d9\ub7da\ub7db\ub7dc\ub7dd\ub7de\ub7df", - "\ub7e0\ub7e1\ub7e2\ub7e3\ub7e4\ub7e5\ub7e6\ub7e7\ub7e8\ub7e9\ub7ea\ub7eb\ub7ec\ub7ed\ub7ee\ub7ef", - "\ub7f0\ub7f1\ub7f2\ub7f3\ub7f4\ub7f5\ub7f6\ub7f7\ub7f8\ub7f9\ub7fa\ub7fb\ub7fc\ub7fd\ub7fe\ub7ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_b8.png", - "ascent": %ascent%, - "chars": [ - "\ub800\ub801\ub802\ub803\ub804\ub805\ub806\ub807\ub808\ub809\ub80a\ub80b\ub80c\ub80d\ub80e\ub80f", - "\ub810\ub811\ub812\ub813\ub814\ub815\ub816\ub817\ub818\ub819\ub81a\ub81b\ub81c\ub81d\ub81e\ub81f", - "\ub820\ub821\ub822\ub823\ub824\ub825\ub826\ub827\ub828\ub829\ub82a\ub82b\ub82c\ub82d\ub82e\ub82f", - "\ub830\ub831\ub832\ub833\ub834\ub835\ub836\ub837\ub838\ub839\ub83a\ub83b\ub83c\ub83d\ub83e\ub83f", - "\ub840\ub841\ub842\ub843\ub844\ub845\ub846\ub847\ub848\ub849\ub84a\ub84b\ub84c\ub84d\ub84e\ub84f", - "\ub850\ub851\ub852\ub853\ub854\ub855\ub856\ub857\ub858\ub859\ub85a\ub85b\ub85c\ub85d\ub85e\ub85f", - "\ub860\ub861\ub862\ub863\ub864\ub865\ub866\ub867\ub868\ub869\ub86a\ub86b\ub86c\ub86d\ub86e\ub86f", - "\ub870\ub871\ub872\ub873\ub874\ub875\ub876\ub877\ub878\ub879\ub87a\ub87b\ub87c\ub87d\ub87e\ub87f", - "\ub880\ub881\ub882\ub883\ub884\ub885\ub886\ub887\ub888\ub889\ub88a\ub88b\ub88c\ub88d\ub88e\ub88f", - "\ub890\ub891\ub892\ub893\ub894\ub895\ub896\ub897\ub898\ub899\ub89a\ub89b\ub89c\ub89d\ub89e\ub89f", - "\ub8a0\ub8a1\ub8a2\ub8a3\ub8a4\ub8a5\ub8a6\ub8a7\ub8a8\ub8a9\ub8aa\ub8ab\ub8ac\ub8ad\ub8ae\ub8af", - "\ub8b0\ub8b1\ub8b2\ub8b3\ub8b4\ub8b5\ub8b6\ub8b7\ub8b8\ub8b9\ub8ba\ub8bb\ub8bc\ub8bd\ub8be\ub8bf", - "\ub8c0\ub8c1\ub8c2\ub8c3\ub8c4\ub8c5\ub8c6\ub8c7\ub8c8\ub8c9\ub8ca\ub8cb\ub8cc\ub8cd\ub8ce\ub8cf", - "\ub8d0\ub8d1\ub8d2\ub8d3\ub8d4\ub8d5\ub8d6\ub8d7\ub8d8\ub8d9\ub8da\ub8db\ub8dc\ub8dd\ub8de\ub8df", - "\ub8e0\ub8e1\ub8e2\ub8e3\ub8e4\ub8e5\ub8e6\ub8e7\ub8e8\ub8e9\ub8ea\ub8eb\ub8ec\ub8ed\ub8ee\ub8ef", - "\ub8f0\ub8f1\ub8f2\ub8f3\ub8f4\ub8f5\ub8f6\ub8f7\ub8f8\ub8f9\ub8fa\ub8fb\ub8fc\ub8fd\ub8fe\ub8ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_b9.png", - "ascent": %ascent%, - "chars": [ - "\ub900\ub901\ub902\ub903\ub904\ub905\ub906\ub907\ub908\ub909\ub90a\ub90b\ub90c\ub90d\ub90e\ub90f", - "\ub910\ub911\ub912\ub913\ub914\ub915\ub916\ub917\ub918\ub919\ub91a\ub91b\ub91c\ub91d\ub91e\ub91f", - "\ub920\ub921\ub922\ub923\ub924\ub925\ub926\ub927\ub928\ub929\ub92a\ub92b\ub92c\ub92d\ub92e\ub92f", - "\ub930\ub931\ub932\ub933\ub934\ub935\ub936\ub937\ub938\ub939\ub93a\ub93b\ub93c\ub93d\ub93e\ub93f", - "\ub940\ub941\ub942\ub943\ub944\ub945\ub946\ub947\ub948\ub949\ub94a\ub94b\ub94c\ub94d\ub94e\ub94f", - "\ub950\ub951\ub952\ub953\ub954\ub955\ub956\ub957\ub958\ub959\ub95a\ub95b\ub95c\ub95d\ub95e\ub95f", - "\ub960\ub961\ub962\ub963\ub964\ub965\ub966\ub967\ub968\ub969\ub96a\ub96b\ub96c\ub96d\ub96e\ub96f", - "\ub970\ub971\ub972\ub973\ub974\ub975\ub976\ub977\ub978\ub979\ub97a\ub97b\ub97c\ub97d\ub97e\ub97f", - "\ub980\ub981\ub982\ub983\ub984\ub985\ub986\ub987\ub988\ub989\ub98a\ub98b\ub98c\ub98d\ub98e\ub98f", - "\ub990\ub991\ub992\ub993\ub994\ub995\ub996\ub997\ub998\ub999\ub99a\ub99b\ub99c\ub99d\ub99e\ub99f", - "\ub9a0\ub9a1\ub9a2\ub9a3\ub9a4\ub9a5\ub9a6\ub9a7\ub9a8\ub9a9\ub9aa\ub9ab\ub9ac\ub9ad\ub9ae\ub9af", - "\ub9b0\ub9b1\ub9b2\ub9b3\ub9b4\ub9b5\ub9b6\ub9b7\ub9b8\ub9b9\ub9ba\ub9bb\ub9bc\ub9bd\ub9be\ub9bf", - "\ub9c0\ub9c1\ub9c2\ub9c3\ub9c4\ub9c5\ub9c6\ub9c7\ub9c8\ub9c9\ub9ca\ub9cb\ub9cc\ub9cd\ub9ce\ub9cf", - "\ub9d0\ub9d1\ub9d2\ub9d3\ub9d4\ub9d5\ub9d6\ub9d7\ub9d8\ub9d9\ub9da\ub9db\ub9dc\ub9dd\ub9de\ub9df", - "\ub9e0\ub9e1\ub9e2\ub9e3\ub9e4\ub9e5\ub9e6\ub9e7\ub9e8\ub9e9\ub9ea\ub9eb\ub9ec\ub9ed\ub9ee\ub9ef", - "\ub9f0\ub9f1\ub9f2\ub9f3\ub9f4\ub9f5\ub9f6\ub9f7\ub9f8\ub9f9\ub9fa\ub9fb\ub9fc\ub9fd\ub9fe\ub9ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_ba.png", - "ascent": %ascent%, - "chars": [ - "\uba00\uba01\uba02\uba03\uba04\uba05\uba06\uba07\uba08\uba09\uba0a\uba0b\uba0c\uba0d\uba0e\uba0f", - "\uba10\uba11\uba12\uba13\uba14\uba15\uba16\uba17\uba18\uba19\uba1a\uba1b\uba1c\uba1d\uba1e\uba1f", - "\uba20\uba21\uba22\uba23\uba24\uba25\uba26\uba27\uba28\uba29\uba2a\uba2b\uba2c\uba2d\uba2e\uba2f", - "\uba30\uba31\uba32\uba33\uba34\uba35\uba36\uba37\uba38\uba39\uba3a\uba3b\uba3c\uba3d\uba3e\uba3f", - "\uba40\uba41\uba42\uba43\uba44\uba45\uba46\uba47\uba48\uba49\uba4a\uba4b\uba4c\uba4d\uba4e\uba4f", - "\uba50\uba51\uba52\uba53\uba54\uba55\uba56\uba57\uba58\uba59\uba5a\uba5b\uba5c\uba5d\uba5e\uba5f", - "\uba60\uba61\uba62\uba63\uba64\uba65\uba66\uba67\uba68\uba69\uba6a\uba6b\uba6c\uba6d\uba6e\uba6f", - "\uba70\uba71\uba72\uba73\uba74\uba75\uba76\uba77\uba78\uba79\uba7a\uba7b\uba7c\uba7d\uba7e\uba7f", - "\uba80\uba81\uba82\uba83\uba84\uba85\uba86\uba87\uba88\uba89\uba8a\uba8b\uba8c\uba8d\uba8e\uba8f", - "\uba90\uba91\uba92\uba93\uba94\uba95\uba96\uba97\uba98\uba99\uba9a\uba9b\uba9c\uba9d\uba9e\uba9f", - "\ubaa0\ubaa1\ubaa2\ubaa3\ubaa4\ubaa5\ubaa6\ubaa7\ubaa8\ubaa9\ubaaa\ubaab\ubaac\ubaad\ubaae\ubaaf", - "\ubab0\ubab1\ubab2\ubab3\ubab4\ubab5\ubab6\ubab7\ubab8\ubab9\ubaba\ubabb\ubabc\ubabd\ubabe\ubabf", - "\ubac0\ubac1\ubac2\ubac3\ubac4\ubac5\ubac6\ubac7\ubac8\ubac9\ubaca\ubacb\ubacc\ubacd\ubace\ubacf", - "\ubad0\ubad1\ubad2\ubad3\ubad4\ubad5\ubad6\ubad7\ubad8\ubad9\ubada\ubadb\ubadc\ubadd\ubade\ubadf", - "\ubae0\ubae1\ubae2\ubae3\ubae4\ubae5\ubae6\ubae7\ubae8\ubae9\ubaea\ubaeb\ubaec\ubaed\ubaee\ubaef", - "\ubaf0\ubaf1\ubaf2\ubaf3\ubaf4\ubaf5\ubaf6\ubaf7\ubaf8\ubaf9\ubafa\ubafb\ubafc\ubafd\ubafe\ubaff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_bb.png", - "ascent": %ascent%, - "chars": [ - "\ubb00\ubb01\ubb02\ubb03\ubb04\ubb05\ubb06\ubb07\ubb08\ubb09\ubb0a\ubb0b\ubb0c\ubb0d\ubb0e\ubb0f", - "\ubb10\ubb11\ubb12\ubb13\ubb14\ubb15\ubb16\ubb17\ubb18\ubb19\ubb1a\ubb1b\ubb1c\ubb1d\ubb1e\ubb1f", - "\ubb20\ubb21\ubb22\ubb23\ubb24\ubb25\ubb26\ubb27\ubb28\ubb29\ubb2a\ubb2b\ubb2c\ubb2d\ubb2e\ubb2f", - "\ubb30\ubb31\ubb32\ubb33\ubb34\ubb35\ubb36\ubb37\ubb38\ubb39\ubb3a\ubb3b\ubb3c\ubb3d\ubb3e\ubb3f", - "\ubb40\ubb41\ubb42\ubb43\ubb44\ubb45\ubb46\ubb47\ubb48\ubb49\ubb4a\ubb4b\ubb4c\ubb4d\ubb4e\ubb4f", - "\ubb50\ubb51\ubb52\ubb53\ubb54\ubb55\ubb56\ubb57\ubb58\ubb59\ubb5a\ubb5b\ubb5c\ubb5d\ubb5e\ubb5f", - "\ubb60\ubb61\ubb62\ubb63\ubb64\ubb65\ubb66\ubb67\ubb68\ubb69\ubb6a\ubb6b\ubb6c\ubb6d\ubb6e\ubb6f", - "\ubb70\ubb71\ubb72\ubb73\ubb74\ubb75\ubb76\ubb77\ubb78\ubb79\ubb7a\ubb7b\ubb7c\ubb7d\ubb7e\ubb7f", - "\ubb80\ubb81\ubb82\ubb83\ubb84\ubb85\ubb86\ubb87\ubb88\ubb89\ubb8a\ubb8b\ubb8c\ubb8d\ubb8e\ubb8f", - "\ubb90\ubb91\ubb92\ubb93\ubb94\ubb95\ubb96\ubb97\ubb98\ubb99\ubb9a\ubb9b\ubb9c\ubb9d\ubb9e\ubb9f", - "\ubba0\ubba1\ubba2\ubba3\ubba4\ubba5\ubba6\ubba7\ubba8\ubba9\ubbaa\ubbab\ubbac\ubbad\ubbae\ubbaf", - "\ubbb0\ubbb1\ubbb2\ubbb3\ubbb4\ubbb5\ubbb6\ubbb7\ubbb8\ubbb9\ubbba\ubbbb\ubbbc\ubbbd\ubbbe\ubbbf", - "\ubbc0\ubbc1\ubbc2\ubbc3\ubbc4\ubbc5\ubbc6\ubbc7\ubbc8\ubbc9\ubbca\ubbcb\ubbcc\ubbcd\ubbce\ubbcf", - "\ubbd0\ubbd1\ubbd2\ubbd3\ubbd4\ubbd5\ubbd6\ubbd7\ubbd8\ubbd9\ubbda\ubbdb\ubbdc\ubbdd\ubbde\ubbdf", - "\ubbe0\ubbe1\ubbe2\ubbe3\ubbe4\ubbe5\ubbe6\ubbe7\ubbe8\ubbe9\ubbea\ubbeb\ubbec\ubbed\ubbee\ubbef", - "\ubbf0\ubbf1\ubbf2\ubbf3\ubbf4\ubbf5\ubbf6\ubbf7\ubbf8\ubbf9\ubbfa\ubbfb\ubbfc\ubbfd\ubbfe\ubbff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_bc.png", - "ascent": %ascent%, - "chars": [ - "\ubc00\ubc01\ubc02\ubc03\ubc04\ubc05\ubc06\ubc07\ubc08\ubc09\ubc0a\ubc0b\ubc0c\ubc0d\ubc0e\ubc0f", - "\ubc10\ubc11\ubc12\ubc13\ubc14\ubc15\ubc16\ubc17\ubc18\ubc19\ubc1a\ubc1b\ubc1c\ubc1d\ubc1e\ubc1f", - "\ubc20\ubc21\ubc22\ubc23\ubc24\ubc25\ubc26\ubc27\ubc28\ubc29\ubc2a\ubc2b\ubc2c\ubc2d\ubc2e\ubc2f", - "\ubc30\ubc31\ubc32\ubc33\ubc34\ubc35\ubc36\ubc37\ubc38\ubc39\ubc3a\ubc3b\ubc3c\ubc3d\ubc3e\ubc3f", - "\ubc40\ubc41\ubc42\ubc43\ubc44\ubc45\ubc46\ubc47\ubc48\ubc49\ubc4a\ubc4b\ubc4c\ubc4d\ubc4e\ubc4f", - "\ubc50\ubc51\ubc52\ubc53\ubc54\ubc55\ubc56\ubc57\ubc58\ubc59\ubc5a\ubc5b\ubc5c\ubc5d\ubc5e\ubc5f", - "\ubc60\ubc61\ubc62\ubc63\ubc64\ubc65\ubc66\ubc67\ubc68\ubc69\ubc6a\ubc6b\ubc6c\ubc6d\ubc6e\ubc6f", - "\ubc70\ubc71\ubc72\ubc73\ubc74\ubc75\ubc76\ubc77\ubc78\ubc79\ubc7a\ubc7b\ubc7c\ubc7d\ubc7e\ubc7f", - "\ubc80\ubc81\ubc82\ubc83\ubc84\ubc85\ubc86\ubc87\ubc88\ubc89\ubc8a\ubc8b\ubc8c\ubc8d\ubc8e\ubc8f", - "\ubc90\ubc91\ubc92\ubc93\ubc94\ubc95\ubc96\ubc97\ubc98\ubc99\ubc9a\ubc9b\ubc9c\ubc9d\ubc9e\ubc9f", - "\ubca0\ubca1\ubca2\ubca3\ubca4\ubca5\ubca6\ubca7\ubca8\ubca9\ubcaa\ubcab\ubcac\ubcad\ubcae\ubcaf", - "\ubcb0\ubcb1\ubcb2\ubcb3\ubcb4\ubcb5\ubcb6\ubcb7\ubcb8\ubcb9\ubcba\ubcbb\ubcbc\ubcbd\ubcbe\ubcbf", - "\ubcc0\ubcc1\ubcc2\ubcc3\ubcc4\ubcc5\ubcc6\ubcc7\ubcc8\ubcc9\ubcca\ubccb\ubccc\ubccd\ubcce\ubccf", - "\ubcd0\ubcd1\ubcd2\ubcd3\ubcd4\ubcd5\ubcd6\ubcd7\ubcd8\ubcd9\ubcda\ubcdb\ubcdc\ubcdd\ubcde\ubcdf", - "\ubce0\ubce1\ubce2\ubce3\ubce4\ubce5\ubce6\ubce7\ubce8\ubce9\ubcea\ubceb\ubcec\ubced\ubcee\ubcef", - "\ubcf0\ubcf1\ubcf2\ubcf3\ubcf4\ubcf5\ubcf6\ubcf7\ubcf8\ubcf9\ubcfa\ubcfb\ubcfc\ubcfd\ubcfe\ubcff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_bd.png", - "ascent": %ascent%, - "chars": [ - "\ubd00\ubd01\ubd02\ubd03\ubd04\ubd05\ubd06\ubd07\ubd08\ubd09\ubd0a\ubd0b\ubd0c\ubd0d\ubd0e\ubd0f", - "\ubd10\ubd11\ubd12\ubd13\ubd14\ubd15\ubd16\ubd17\ubd18\ubd19\ubd1a\ubd1b\ubd1c\ubd1d\ubd1e\ubd1f", - "\ubd20\ubd21\ubd22\ubd23\ubd24\ubd25\ubd26\ubd27\ubd28\ubd29\ubd2a\ubd2b\ubd2c\ubd2d\ubd2e\ubd2f", - "\ubd30\ubd31\ubd32\ubd33\ubd34\ubd35\ubd36\ubd37\ubd38\ubd39\ubd3a\ubd3b\ubd3c\ubd3d\ubd3e\ubd3f", - "\ubd40\ubd41\ubd42\ubd43\ubd44\ubd45\ubd46\ubd47\ubd48\ubd49\ubd4a\ubd4b\ubd4c\ubd4d\ubd4e\ubd4f", - "\ubd50\ubd51\ubd52\ubd53\ubd54\ubd55\ubd56\ubd57\ubd58\ubd59\ubd5a\ubd5b\ubd5c\ubd5d\ubd5e\ubd5f", - "\ubd60\ubd61\ubd62\ubd63\ubd64\ubd65\ubd66\ubd67\ubd68\ubd69\ubd6a\ubd6b\ubd6c\ubd6d\ubd6e\ubd6f", - "\ubd70\ubd71\ubd72\ubd73\ubd74\ubd75\ubd76\ubd77\ubd78\ubd79\ubd7a\ubd7b\ubd7c\ubd7d\ubd7e\ubd7f", - "\ubd80\ubd81\ubd82\ubd83\ubd84\ubd85\ubd86\ubd87\ubd88\ubd89\ubd8a\ubd8b\ubd8c\ubd8d\ubd8e\ubd8f", - "\ubd90\ubd91\ubd92\ubd93\ubd94\ubd95\ubd96\ubd97\ubd98\ubd99\ubd9a\ubd9b\ubd9c\ubd9d\ubd9e\ubd9f", - "\ubda0\ubda1\ubda2\ubda3\ubda4\ubda5\ubda6\ubda7\ubda8\ubda9\ubdaa\ubdab\ubdac\ubdad\ubdae\ubdaf", - "\ubdb0\ubdb1\ubdb2\ubdb3\ubdb4\ubdb5\ubdb6\ubdb7\ubdb8\ubdb9\ubdba\ubdbb\ubdbc\ubdbd\ubdbe\ubdbf", - "\ubdc0\ubdc1\ubdc2\ubdc3\ubdc4\ubdc5\ubdc6\ubdc7\ubdc8\ubdc9\ubdca\ubdcb\ubdcc\ubdcd\ubdce\ubdcf", - "\ubdd0\ubdd1\ubdd2\ubdd3\ubdd4\ubdd5\ubdd6\ubdd7\ubdd8\ubdd9\ubdda\ubddb\ubddc\ubddd\ubdde\ubddf", - "\ubde0\ubde1\ubde2\ubde3\ubde4\ubde5\ubde6\ubde7\ubde8\ubde9\ubdea\ubdeb\ubdec\ubded\ubdee\ubdef", - "\ubdf0\ubdf1\ubdf2\ubdf3\ubdf4\ubdf5\ubdf6\ubdf7\ubdf8\ubdf9\ubdfa\ubdfb\ubdfc\ubdfd\ubdfe\ubdff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_be.png", - "ascent": %ascent%, - "chars": [ - "\ube00\ube01\ube02\ube03\ube04\ube05\ube06\ube07\ube08\ube09\ube0a\ube0b\ube0c\ube0d\ube0e\ube0f", - "\ube10\ube11\ube12\ube13\ube14\ube15\ube16\ube17\ube18\ube19\ube1a\ube1b\ube1c\ube1d\ube1e\ube1f", - "\ube20\ube21\ube22\ube23\ube24\ube25\ube26\ube27\ube28\ube29\ube2a\ube2b\ube2c\ube2d\ube2e\ube2f", - "\ube30\ube31\ube32\ube33\ube34\ube35\ube36\ube37\ube38\ube39\ube3a\ube3b\ube3c\ube3d\ube3e\ube3f", - "\ube40\ube41\ube42\ube43\ube44\ube45\ube46\ube47\ube48\ube49\ube4a\ube4b\ube4c\ube4d\ube4e\ube4f", - "\ube50\ube51\ube52\ube53\ube54\ube55\ube56\ube57\ube58\ube59\ube5a\ube5b\ube5c\ube5d\ube5e\ube5f", - "\ube60\ube61\ube62\ube63\ube64\ube65\ube66\ube67\ube68\ube69\ube6a\ube6b\ube6c\ube6d\ube6e\ube6f", - "\ube70\ube71\ube72\ube73\ube74\ube75\ube76\ube77\ube78\ube79\ube7a\ube7b\ube7c\ube7d\ube7e\ube7f", - "\ube80\ube81\ube82\ube83\ube84\ube85\ube86\ube87\ube88\ube89\ube8a\ube8b\ube8c\ube8d\ube8e\ube8f", - "\ube90\ube91\ube92\ube93\ube94\ube95\ube96\ube97\ube98\ube99\ube9a\ube9b\ube9c\ube9d\ube9e\ube9f", - "\ubea0\ubea1\ubea2\ubea3\ubea4\ubea5\ubea6\ubea7\ubea8\ubea9\ubeaa\ubeab\ubeac\ubead\ubeae\ubeaf", - "\ubeb0\ubeb1\ubeb2\ubeb3\ubeb4\ubeb5\ubeb6\ubeb7\ubeb8\ubeb9\ubeba\ubebb\ubebc\ubebd\ubebe\ubebf", - "\ubec0\ubec1\ubec2\ubec3\ubec4\ubec5\ubec6\ubec7\ubec8\ubec9\ubeca\ubecb\ubecc\ubecd\ubece\ubecf", - "\ubed0\ubed1\ubed2\ubed3\ubed4\ubed5\ubed6\ubed7\ubed8\ubed9\ubeda\ubedb\ubedc\ubedd\ubede\ubedf", - "\ubee0\ubee1\ubee2\ubee3\ubee4\ubee5\ubee6\ubee7\ubee8\ubee9\ubeea\ubeeb\ubeec\ubeed\ubeee\ubeef", - "\ubef0\ubef1\ubef2\ubef3\ubef4\ubef5\ubef6\ubef7\ubef8\ubef9\ubefa\ubefb\ubefc\ubefd\ubefe\ubeff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_bf.png", - "ascent": %ascent%, - "chars": [ - "\ubf00\ubf01\ubf02\ubf03\ubf04\ubf05\ubf06\ubf07\ubf08\ubf09\ubf0a\ubf0b\ubf0c\ubf0d\ubf0e\ubf0f", - "\ubf10\ubf11\ubf12\ubf13\ubf14\ubf15\ubf16\ubf17\ubf18\ubf19\ubf1a\ubf1b\ubf1c\ubf1d\ubf1e\ubf1f", - "\ubf20\ubf21\ubf22\ubf23\ubf24\ubf25\ubf26\ubf27\ubf28\ubf29\ubf2a\ubf2b\ubf2c\ubf2d\ubf2e\ubf2f", - "\ubf30\ubf31\ubf32\ubf33\ubf34\ubf35\ubf36\ubf37\ubf38\ubf39\ubf3a\ubf3b\ubf3c\ubf3d\ubf3e\ubf3f", - "\ubf40\ubf41\ubf42\ubf43\ubf44\ubf45\ubf46\ubf47\ubf48\ubf49\ubf4a\ubf4b\ubf4c\ubf4d\ubf4e\ubf4f", - "\ubf50\ubf51\ubf52\ubf53\ubf54\ubf55\ubf56\ubf57\ubf58\ubf59\ubf5a\ubf5b\ubf5c\ubf5d\ubf5e\ubf5f", - "\ubf60\ubf61\ubf62\ubf63\ubf64\ubf65\ubf66\ubf67\ubf68\ubf69\ubf6a\ubf6b\ubf6c\ubf6d\ubf6e\ubf6f", - "\ubf70\ubf71\ubf72\ubf73\ubf74\ubf75\ubf76\ubf77\ubf78\ubf79\ubf7a\ubf7b\ubf7c\ubf7d\ubf7e\ubf7f", - "\ubf80\ubf81\ubf82\ubf83\ubf84\ubf85\ubf86\ubf87\ubf88\ubf89\ubf8a\ubf8b\ubf8c\ubf8d\ubf8e\ubf8f", - "\ubf90\ubf91\ubf92\ubf93\ubf94\ubf95\ubf96\ubf97\ubf98\ubf99\ubf9a\ubf9b\ubf9c\ubf9d\ubf9e\ubf9f", - "\ubfa0\ubfa1\ubfa2\ubfa3\ubfa4\ubfa5\ubfa6\ubfa7\ubfa8\ubfa9\ubfaa\ubfab\ubfac\ubfad\ubfae\ubfaf", - "\ubfb0\ubfb1\ubfb2\ubfb3\ubfb4\ubfb5\ubfb6\ubfb7\ubfb8\ubfb9\ubfba\ubfbb\ubfbc\ubfbd\ubfbe\ubfbf", - "\ubfc0\ubfc1\ubfc2\ubfc3\ubfc4\ubfc5\ubfc6\ubfc7\ubfc8\ubfc9\ubfca\ubfcb\ubfcc\ubfcd\ubfce\ubfcf", - "\ubfd0\ubfd1\ubfd2\ubfd3\ubfd4\ubfd5\ubfd6\ubfd7\ubfd8\ubfd9\ubfda\ubfdb\ubfdc\ubfdd\ubfde\ubfdf", - "\ubfe0\ubfe1\ubfe2\ubfe3\ubfe4\ubfe5\ubfe6\ubfe7\ubfe8\ubfe9\ubfea\ubfeb\ubfec\ubfed\ubfee\ubfef", - "\ubff0\ubff1\ubff2\ubff3\ubff4\ubff5\ubff6\ubff7\ubff8\ubff9\ubffa\ubffb\ubffc\ubffd\ubffe\ubfff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_c0.png", - "ascent": %ascent%, - "chars": [ - "\uc000\uc001\uc002\uc003\uc004\uc005\uc006\uc007\uc008\uc009\uc00a\uc00b\uc00c\uc00d\uc00e\uc00f", - "\uc010\uc011\uc012\uc013\uc014\uc015\uc016\uc017\uc018\uc019\uc01a\uc01b\uc01c\uc01d\uc01e\uc01f", - "\uc020\uc021\uc022\uc023\uc024\uc025\uc026\uc027\uc028\uc029\uc02a\uc02b\uc02c\uc02d\uc02e\uc02f", - "\uc030\uc031\uc032\uc033\uc034\uc035\uc036\uc037\uc038\uc039\uc03a\uc03b\uc03c\uc03d\uc03e\uc03f", - "\uc040\uc041\uc042\uc043\uc044\uc045\uc046\uc047\uc048\uc049\uc04a\uc04b\uc04c\uc04d\uc04e\uc04f", - "\uc050\uc051\uc052\uc053\uc054\uc055\uc056\uc057\uc058\uc059\uc05a\uc05b\uc05c\uc05d\uc05e\uc05f", - "\uc060\uc061\uc062\uc063\uc064\uc065\uc066\uc067\uc068\uc069\uc06a\uc06b\uc06c\uc06d\uc06e\uc06f", - "\uc070\uc071\uc072\uc073\uc074\uc075\uc076\uc077\uc078\uc079\uc07a\uc07b\uc07c\uc07d\uc07e\uc07f", - "\uc080\uc081\uc082\uc083\uc084\uc085\uc086\uc087\uc088\uc089\uc08a\uc08b\uc08c\uc08d\uc08e\uc08f", - "\uc090\uc091\uc092\uc093\uc094\uc095\uc096\uc097\uc098\uc099\uc09a\uc09b\uc09c\uc09d\uc09e\uc09f", - "\uc0a0\uc0a1\uc0a2\uc0a3\uc0a4\uc0a5\uc0a6\uc0a7\uc0a8\uc0a9\uc0aa\uc0ab\uc0ac\uc0ad\uc0ae\uc0af", - "\uc0b0\uc0b1\uc0b2\uc0b3\uc0b4\uc0b5\uc0b6\uc0b7\uc0b8\uc0b9\uc0ba\uc0bb\uc0bc\uc0bd\uc0be\uc0bf", - "\uc0c0\uc0c1\uc0c2\uc0c3\uc0c4\uc0c5\uc0c6\uc0c7\uc0c8\uc0c9\uc0ca\uc0cb\uc0cc\uc0cd\uc0ce\uc0cf", - "\uc0d0\uc0d1\uc0d2\uc0d3\uc0d4\uc0d5\uc0d6\uc0d7\uc0d8\uc0d9\uc0da\uc0db\uc0dc\uc0dd\uc0de\uc0df", - "\uc0e0\uc0e1\uc0e2\uc0e3\uc0e4\uc0e5\uc0e6\uc0e7\uc0e8\uc0e9\uc0ea\uc0eb\uc0ec\uc0ed\uc0ee\uc0ef", - "\uc0f0\uc0f1\uc0f2\uc0f3\uc0f4\uc0f5\uc0f6\uc0f7\uc0f8\uc0f9\uc0fa\uc0fb\uc0fc\uc0fd\uc0fe\uc0ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_c1.png", - "ascent": %ascent%, - "chars": [ - "\uc100\uc101\uc102\uc103\uc104\uc105\uc106\uc107\uc108\uc109\uc10a\uc10b\uc10c\uc10d\uc10e\uc10f", - "\uc110\uc111\uc112\uc113\uc114\uc115\uc116\uc117\uc118\uc119\uc11a\uc11b\uc11c\uc11d\uc11e\uc11f", - "\uc120\uc121\uc122\uc123\uc124\uc125\uc126\uc127\uc128\uc129\uc12a\uc12b\uc12c\uc12d\uc12e\uc12f", - "\uc130\uc131\uc132\uc133\uc134\uc135\uc136\uc137\uc138\uc139\uc13a\uc13b\uc13c\uc13d\uc13e\uc13f", - "\uc140\uc141\uc142\uc143\uc144\uc145\uc146\uc147\uc148\uc149\uc14a\uc14b\uc14c\uc14d\uc14e\uc14f", - "\uc150\uc151\uc152\uc153\uc154\uc155\uc156\uc157\uc158\uc159\uc15a\uc15b\uc15c\uc15d\uc15e\uc15f", - "\uc160\uc161\uc162\uc163\uc164\uc165\uc166\uc167\uc168\uc169\uc16a\uc16b\uc16c\uc16d\uc16e\uc16f", - "\uc170\uc171\uc172\uc173\uc174\uc175\uc176\uc177\uc178\uc179\uc17a\uc17b\uc17c\uc17d\uc17e\uc17f", - "\uc180\uc181\uc182\uc183\uc184\uc185\uc186\uc187\uc188\uc189\uc18a\uc18b\uc18c\uc18d\uc18e\uc18f", - "\uc190\uc191\uc192\uc193\uc194\uc195\uc196\uc197\uc198\uc199\uc19a\uc19b\uc19c\uc19d\uc19e\uc19f", - "\uc1a0\uc1a1\uc1a2\uc1a3\uc1a4\uc1a5\uc1a6\uc1a7\uc1a8\uc1a9\uc1aa\uc1ab\uc1ac\uc1ad\uc1ae\uc1af", - "\uc1b0\uc1b1\uc1b2\uc1b3\uc1b4\uc1b5\uc1b6\uc1b7\uc1b8\uc1b9\uc1ba\uc1bb\uc1bc\uc1bd\uc1be\uc1bf", - "\uc1c0\uc1c1\uc1c2\uc1c3\uc1c4\uc1c5\uc1c6\uc1c7\uc1c8\uc1c9\uc1ca\uc1cb\uc1cc\uc1cd\uc1ce\uc1cf", - "\uc1d0\uc1d1\uc1d2\uc1d3\uc1d4\uc1d5\uc1d6\uc1d7\uc1d8\uc1d9\uc1da\uc1db\uc1dc\uc1dd\uc1de\uc1df", - "\uc1e0\uc1e1\uc1e2\uc1e3\uc1e4\uc1e5\uc1e6\uc1e7\uc1e8\uc1e9\uc1ea\uc1eb\uc1ec\uc1ed\uc1ee\uc1ef", - "\uc1f0\uc1f1\uc1f2\uc1f3\uc1f4\uc1f5\uc1f6\uc1f7\uc1f8\uc1f9\uc1fa\uc1fb\uc1fc\uc1fd\uc1fe\uc1ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_c2.png", - "ascent": %ascent%, - "chars": [ - "\uc200\uc201\uc202\uc203\uc204\uc205\uc206\uc207\uc208\uc209\uc20a\uc20b\uc20c\uc20d\uc20e\uc20f", - "\uc210\uc211\uc212\uc213\uc214\uc215\uc216\uc217\uc218\uc219\uc21a\uc21b\uc21c\uc21d\uc21e\uc21f", - "\uc220\uc221\uc222\uc223\uc224\uc225\uc226\uc227\uc228\uc229\uc22a\uc22b\uc22c\uc22d\uc22e\uc22f", - "\uc230\uc231\uc232\uc233\uc234\uc235\uc236\uc237\uc238\uc239\uc23a\uc23b\uc23c\uc23d\uc23e\uc23f", - "\uc240\uc241\uc242\uc243\uc244\uc245\uc246\uc247\uc248\uc249\uc24a\uc24b\uc24c\uc24d\uc24e\uc24f", - "\uc250\uc251\uc252\uc253\uc254\uc255\uc256\uc257\uc258\uc259\uc25a\uc25b\uc25c\uc25d\uc25e\uc25f", - "\uc260\uc261\uc262\uc263\uc264\uc265\uc266\uc267\uc268\uc269\uc26a\uc26b\uc26c\uc26d\uc26e\uc26f", - "\uc270\uc271\uc272\uc273\uc274\uc275\uc276\uc277\uc278\uc279\uc27a\uc27b\uc27c\uc27d\uc27e\uc27f", - "\uc280\uc281\uc282\uc283\uc284\uc285\uc286\uc287\uc288\uc289\uc28a\uc28b\uc28c\uc28d\uc28e\uc28f", - "\uc290\uc291\uc292\uc293\uc294\uc295\uc296\uc297\uc298\uc299\uc29a\uc29b\uc29c\uc29d\uc29e\uc29f", - "\uc2a0\uc2a1\uc2a2\uc2a3\uc2a4\uc2a5\uc2a6\uc2a7\uc2a8\uc2a9\uc2aa\uc2ab\uc2ac\uc2ad\uc2ae\uc2af", - "\uc2b0\uc2b1\uc2b2\uc2b3\uc2b4\uc2b5\uc2b6\uc2b7\uc2b8\uc2b9\uc2ba\uc2bb\uc2bc\uc2bd\uc2be\uc2bf", - "\uc2c0\uc2c1\uc2c2\uc2c3\uc2c4\uc2c5\uc2c6\uc2c7\uc2c8\uc2c9\uc2ca\uc2cb\uc2cc\uc2cd\uc2ce\uc2cf", - "\uc2d0\uc2d1\uc2d2\uc2d3\uc2d4\uc2d5\uc2d6\uc2d7\uc2d8\uc2d9\uc2da\uc2db\uc2dc\uc2dd\uc2de\uc2df", - "\uc2e0\uc2e1\uc2e2\uc2e3\uc2e4\uc2e5\uc2e6\uc2e7\uc2e8\uc2e9\uc2ea\uc2eb\uc2ec\uc2ed\uc2ee\uc2ef", - "\uc2f0\uc2f1\uc2f2\uc2f3\uc2f4\uc2f5\uc2f6\uc2f7\uc2f8\uc2f9\uc2fa\uc2fb\uc2fc\uc2fd\uc2fe\uc2ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_c3.png", - "ascent": %ascent%, - "chars": [ - "\uc300\uc301\uc302\uc303\uc304\uc305\uc306\uc307\uc308\uc309\uc30a\uc30b\uc30c\uc30d\uc30e\uc30f", - "\uc310\uc311\uc312\uc313\uc314\uc315\uc316\uc317\uc318\uc319\uc31a\uc31b\uc31c\uc31d\uc31e\uc31f", - "\uc320\uc321\uc322\uc323\uc324\uc325\uc326\uc327\uc328\uc329\uc32a\uc32b\uc32c\uc32d\uc32e\uc32f", - "\uc330\uc331\uc332\uc333\uc334\uc335\uc336\uc337\uc338\uc339\uc33a\uc33b\uc33c\uc33d\uc33e\uc33f", - "\uc340\uc341\uc342\uc343\uc344\uc345\uc346\uc347\uc348\uc349\uc34a\uc34b\uc34c\uc34d\uc34e\uc34f", - "\uc350\uc351\uc352\uc353\uc354\uc355\uc356\uc357\uc358\uc359\uc35a\uc35b\uc35c\uc35d\uc35e\uc35f", - "\uc360\uc361\uc362\uc363\uc364\uc365\uc366\uc367\uc368\uc369\uc36a\uc36b\uc36c\uc36d\uc36e\uc36f", - "\uc370\uc371\uc372\uc373\uc374\uc375\uc376\uc377\uc378\uc379\uc37a\uc37b\uc37c\uc37d\uc37e\uc37f", - "\uc380\uc381\uc382\uc383\uc384\uc385\uc386\uc387\uc388\uc389\uc38a\uc38b\uc38c\uc38d\uc38e\uc38f", - "\uc390\uc391\uc392\uc393\uc394\uc395\uc396\uc397\uc398\uc399\uc39a\uc39b\uc39c\uc39d\uc39e\uc39f", - "\uc3a0\uc3a1\uc3a2\uc3a3\uc3a4\uc3a5\uc3a6\uc3a7\uc3a8\uc3a9\uc3aa\uc3ab\uc3ac\uc3ad\uc3ae\uc3af", - "\uc3b0\uc3b1\uc3b2\uc3b3\uc3b4\uc3b5\uc3b6\uc3b7\uc3b8\uc3b9\uc3ba\uc3bb\uc3bc\uc3bd\uc3be\uc3bf", - "\uc3c0\uc3c1\uc3c2\uc3c3\uc3c4\uc3c5\uc3c6\uc3c7\uc3c8\uc3c9\uc3ca\uc3cb\uc3cc\uc3cd\uc3ce\uc3cf", - "\uc3d0\uc3d1\uc3d2\uc3d3\uc3d4\uc3d5\uc3d6\uc3d7\uc3d8\uc3d9\uc3da\uc3db\uc3dc\uc3dd\uc3de\uc3df", - "\uc3e0\uc3e1\uc3e2\uc3e3\uc3e4\uc3e5\uc3e6\uc3e7\uc3e8\uc3e9\uc3ea\uc3eb\uc3ec\uc3ed\uc3ee\uc3ef", - "\uc3f0\uc3f1\uc3f2\uc3f3\uc3f4\uc3f5\uc3f6\uc3f7\uc3f8\uc3f9\uc3fa\uc3fb\uc3fc\uc3fd\uc3fe\uc3ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_c4.png", - "ascent": %ascent%, - "chars": [ - "\uc400\uc401\uc402\uc403\uc404\uc405\uc406\uc407\uc408\uc409\uc40a\uc40b\uc40c\uc40d\uc40e\uc40f", - "\uc410\uc411\uc412\uc413\uc414\uc415\uc416\uc417\uc418\uc419\uc41a\uc41b\uc41c\uc41d\uc41e\uc41f", - "\uc420\uc421\uc422\uc423\uc424\uc425\uc426\uc427\uc428\uc429\uc42a\uc42b\uc42c\uc42d\uc42e\uc42f", - "\uc430\uc431\uc432\uc433\uc434\uc435\uc436\uc437\uc438\uc439\uc43a\uc43b\uc43c\uc43d\uc43e\uc43f", - "\uc440\uc441\uc442\uc443\uc444\uc445\uc446\uc447\uc448\uc449\uc44a\uc44b\uc44c\uc44d\uc44e\uc44f", - "\uc450\uc451\uc452\uc453\uc454\uc455\uc456\uc457\uc458\uc459\uc45a\uc45b\uc45c\uc45d\uc45e\uc45f", - "\uc460\uc461\uc462\uc463\uc464\uc465\uc466\uc467\uc468\uc469\uc46a\uc46b\uc46c\uc46d\uc46e\uc46f", - "\uc470\uc471\uc472\uc473\uc474\uc475\uc476\uc477\uc478\uc479\uc47a\uc47b\uc47c\uc47d\uc47e\uc47f", - "\uc480\uc481\uc482\uc483\uc484\uc485\uc486\uc487\uc488\uc489\uc48a\uc48b\uc48c\uc48d\uc48e\uc48f", - "\uc490\uc491\uc492\uc493\uc494\uc495\uc496\uc497\uc498\uc499\uc49a\uc49b\uc49c\uc49d\uc49e\uc49f", - "\uc4a0\uc4a1\uc4a2\uc4a3\uc4a4\uc4a5\uc4a6\uc4a7\uc4a8\uc4a9\uc4aa\uc4ab\uc4ac\uc4ad\uc4ae\uc4af", - "\uc4b0\uc4b1\uc4b2\uc4b3\uc4b4\uc4b5\uc4b6\uc4b7\uc4b8\uc4b9\uc4ba\uc4bb\uc4bc\uc4bd\uc4be\uc4bf", - "\uc4c0\uc4c1\uc4c2\uc4c3\uc4c4\uc4c5\uc4c6\uc4c7\uc4c8\uc4c9\uc4ca\uc4cb\uc4cc\uc4cd\uc4ce\uc4cf", - "\uc4d0\uc4d1\uc4d2\uc4d3\uc4d4\uc4d5\uc4d6\uc4d7\uc4d8\uc4d9\uc4da\uc4db\uc4dc\uc4dd\uc4de\uc4df", - "\uc4e0\uc4e1\uc4e2\uc4e3\uc4e4\uc4e5\uc4e6\uc4e7\uc4e8\uc4e9\uc4ea\uc4eb\uc4ec\uc4ed\uc4ee\uc4ef", - "\uc4f0\uc4f1\uc4f2\uc4f3\uc4f4\uc4f5\uc4f6\uc4f7\uc4f8\uc4f9\uc4fa\uc4fb\uc4fc\uc4fd\uc4fe\uc4ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_c5.png", - "ascent": %ascent%, - "chars": [ - "\uc500\uc501\uc502\uc503\uc504\uc505\uc506\uc507\uc508\uc509\uc50a\uc50b\uc50c\uc50d\uc50e\uc50f", - "\uc510\uc511\uc512\uc513\uc514\uc515\uc516\uc517\uc518\uc519\uc51a\uc51b\uc51c\uc51d\uc51e\uc51f", - "\uc520\uc521\uc522\uc523\uc524\uc525\uc526\uc527\uc528\uc529\uc52a\uc52b\uc52c\uc52d\uc52e\uc52f", - "\uc530\uc531\uc532\uc533\uc534\uc535\uc536\uc537\uc538\uc539\uc53a\uc53b\uc53c\uc53d\uc53e\uc53f", - "\uc540\uc541\uc542\uc543\uc544\uc545\uc546\uc547\uc548\uc549\uc54a\uc54b\uc54c\uc54d\uc54e\uc54f", - "\uc550\uc551\uc552\uc553\uc554\uc555\uc556\uc557\uc558\uc559\uc55a\uc55b\uc55c\uc55d\uc55e\uc55f", - "\uc560\uc561\uc562\uc563\uc564\uc565\uc566\uc567\uc568\uc569\uc56a\uc56b\uc56c\uc56d\uc56e\uc56f", - "\uc570\uc571\uc572\uc573\uc574\uc575\uc576\uc577\uc578\uc579\uc57a\uc57b\uc57c\uc57d\uc57e\uc57f", - "\uc580\uc581\uc582\uc583\uc584\uc585\uc586\uc587\uc588\uc589\uc58a\uc58b\uc58c\uc58d\uc58e\uc58f", - "\uc590\uc591\uc592\uc593\uc594\uc595\uc596\uc597\uc598\uc599\uc59a\uc59b\uc59c\uc59d\uc59e\uc59f", - "\uc5a0\uc5a1\uc5a2\uc5a3\uc5a4\uc5a5\uc5a6\uc5a7\uc5a8\uc5a9\uc5aa\uc5ab\uc5ac\uc5ad\uc5ae\uc5af", - "\uc5b0\uc5b1\uc5b2\uc5b3\uc5b4\uc5b5\uc5b6\uc5b7\uc5b8\uc5b9\uc5ba\uc5bb\uc5bc\uc5bd\uc5be\uc5bf", - "\uc5c0\uc5c1\uc5c2\uc5c3\uc5c4\uc5c5\uc5c6\uc5c7\uc5c8\uc5c9\uc5ca\uc5cb\uc5cc\uc5cd\uc5ce\uc5cf", - "\uc5d0\uc5d1\uc5d2\uc5d3\uc5d4\uc5d5\uc5d6\uc5d7\uc5d8\uc5d9\uc5da\uc5db\uc5dc\uc5dd\uc5de\uc5df", - "\uc5e0\uc5e1\uc5e2\uc5e3\uc5e4\uc5e5\uc5e6\uc5e7\uc5e8\uc5e9\uc5ea\uc5eb\uc5ec\uc5ed\uc5ee\uc5ef", - "\uc5f0\uc5f1\uc5f2\uc5f3\uc5f4\uc5f5\uc5f6\uc5f7\uc5f8\uc5f9\uc5fa\uc5fb\uc5fc\uc5fd\uc5fe\uc5ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_c6.png", - "ascent": %ascent%, - "chars": [ - "\uc600\uc601\uc602\uc603\uc604\uc605\uc606\uc607\uc608\uc609\uc60a\uc60b\uc60c\uc60d\uc60e\uc60f", - "\uc610\uc611\uc612\uc613\uc614\uc615\uc616\uc617\uc618\uc619\uc61a\uc61b\uc61c\uc61d\uc61e\uc61f", - "\uc620\uc621\uc622\uc623\uc624\uc625\uc626\uc627\uc628\uc629\uc62a\uc62b\uc62c\uc62d\uc62e\uc62f", - "\uc630\uc631\uc632\uc633\uc634\uc635\uc636\uc637\uc638\uc639\uc63a\uc63b\uc63c\uc63d\uc63e\uc63f", - "\uc640\uc641\uc642\uc643\uc644\uc645\uc646\uc647\uc648\uc649\uc64a\uc64b\uc64c\uc64d\uc64e\uc64f", - "\uc650\uc651\uc652\uc653\uc654\uc655\uc656\uc657\uc658\uc659\uc65a\uc65b\uc65c\uc65d\uc65e\uc65f", - "\uc660\uc661\uc662\uc663\uc664\uc665\uc666\uc667\uc668\uc669\uc66a\uc66b\uc66c\uc66d\uc66e\uc66f", - "\uc670\uc671\uc672\uc673\uc674\uc675\uc676\uc677\uc678\uc679\uc67a\uc67b\uc67c\uc67d\uc67e\uc67f", - "\uc680\uc681\uc682\uc683\uc684\uc685\uc686\uc687\uc688\uc689\uc68a\uc68b\uc68c\uc68d\uc68e\uc68f", - "\uc690\uc691\uc692\uc693\uc694\uc695\uc696\uc697\uc698\uc699\uc69a\uc69b\uc69c\uc69d\uc69e\uc69f", - "\uc6a0\uc6a1\uc6a2\uc6a3\uc6a4\uc6a5\uc6a6\uc6a7\uc6a8\uc6a9\uc6aa\uc6ab\uc6ac\uc6ad\uc6ae\uc6af", - "\uc6b0\uc6b1\uc6b2\uc6b3\uc6b4\uc6b5\uc6b6\uc6b7\uc6b8\uc6b9\uc6ba\uc6bb\uc6bc\uc6bd\uc6be\uc6bf", - "\uc6c0\uc6c1\uc6c2\uc6c3\uc6c4\uc6c5\uc6c6\uc6c7\uc6c8\uc6c9\uc6ca\uc6cb\uc6cc\uc6cd\uc6ce\uc6cf", - "\uc6d0\uc6d1\uc6d2\uc6d3\uc6d4\uc6d5\uc6d6\uc6d7\uc6d8\uc6d9\uc6da\uc6db\uc6dc\uc6dd\uc6de\uc6df", - "\uc6e0\uc6e1\uc6e2\uc6e3\uc6e4\uc6e5\uc6e6\uc6e7\uc6e8\uc6e9\uc6ea\uc6eb\uc6ec\uc6ed\uc6ee\uc6ef", - "\uc6f0\uc6f1\uc6f2\uc6f3\uc6f4\uc6f5\uc6f6\uc6f7\uc6f8\uc6f9\uc6fa\uc6fb\uc6fc\uc6fd\uc6fe\uc6ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_c7.png", - "ascent": %ascent%, - "chars": [ - "\uc700\uc701\uc702\uc703\uc704\uc705\uc706\uc707\uc708\uc709\uc70a\uc70b\uc70c\uc70d\uc70e\uc70f", - "\uc710\uc711\uc712\uc713\uc714\uc715\uc716\uc717\uc718\uc719\uc71a\uc71b\uc71c\uc71d\uc71e\uc71f", - "\uc720\uc721\uc722\uc723\uc724\uc725\uc726\uc727\uc728\uc729\uc72a\uc72b\uc72c\uc72d\uc72e\uc72f", - "\uc730\uc731\uc732\uc733\uc734\uc735\uc736\uc737\uc738\uc739\uc73a\uc73b\uc73c\uc73d\uc73e\uc73f", - "\uc740\uc741\uc742\uc743\uc744\uc745\uc746\uc747\uc748\uc749\uc74a\uc74b\uc74c\uc74d\uc74e\uc74f", - "\uc750\uc751\uc752\uc753\uc754\uc755\uc756\uc757\uc758\uc759\uc75a\uc75b\uc75c\uc75d\uc75e\uc75f", - "\uc760\uc761\uc762\uc763\uc764\uc765\uc766\uc767\uc768\uc769\uc76a\uc76b\uc76c\uc76d\uc76e\uc76f", - "\uc770\uc771\uc772\uc773\uc774\uc775\uc776\uc777\uc778\uc779\uc77a\uc77b\uc77c\uc77d\uc77e\uc77f", - "\uc780\uc781\uc782\uc783\uc784\uc785\uc786\uc787\uc788\uc789\uc78a\uc78b\uc78c\uc78d\uc78e\uc78f", - "\uc790\uc791\uc792\uc793\uc794\uc795\uc796\uc797\uc798\uc799\uc79a\uc79b\uc79c\uc79d\uc79e\uc79f", - "\uc7a0\uc7a1\uc7a2\uc7a3\uc7a4\uc7a5\uc7a6\uc7a7\uc7a8\uc7a9\uc7aa\uc7ab\uc7ac\uc7ad\uc7ae\uc7af", - "\uc7b0\uc7b1\uc7b2\uc7b3\uc7b4\uc7b5\uc7b6\uc7b7\uc7b8\uc7b9\uc7ba\uc7bb\uc7bc\uc7bd\uc7be\uc7bf", - "\uc7c0\uc7c1\uc7c2\uc7c3\uc7c4\uc7c5\uc7c6\uc7c7\uc7c8\uc7c9\uc7ca\uc7cb\uc7cc\uc7cd\uc7ce\uc7cf", - "\uc7d0\uc7d1\uc7d2\uc7d3\uc7d4\uc7d5\uc7d6\uc7d7\uc7d8\uc7d9\uc7da\uc7db\uc7dc\uc7dd\uc7de\uc7df", - "\uc7e0\uc7e1\uc7e2\uc7e3\uc7e4\uc7e5\uc7e6\uc7e7\uc7e8\uc7e9\uc7ea\uc7eb\uc7ec\uc7ed\uc7ee\uc7ef", - "\uc7f0\uc7f1\uc7f2\uc7f3\uc7f4\uc7f5\uc7f6\uc7f7\uc7f8\uc7f9\uc7fa\uc7fb\uc7fc\uc7fd\uc7fe\uc7ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_c8.png", - "ascent": %ascent%, - "chars": [ - "\uc800\uc801\uc802\uc803\uc804\uc805\uc806\uc807\uc808\uc809\uc80a\uc80b\uc80c\uc80d\uc80e\uc80f", - "\uc810\uc811\uc812\uc813\uc814\uc815\uc816\uc817\uc818\uc819\uc81a\uc81b\uc81c\uc81d\uc81e\uc81f", - "\uc820\uc821\uc822\uc823\uc824\uc825\uc826\uc827\uc828\uc829\uc82a\uc82b\uc82c\uc82d\uc82e\uc82f", - "\uc830\uc831\uc832\uc833\uc834\uc835\uc836\uc837\uc838\uc839\uc83a\uc83b\uc83c\uc83d\uc83e\uc83f", - "\uc840\uc841\uc842\uc843\uc844\uc845\uc846\uc847\uc848\uc849\uc84a\uc84b\uc84c\uc84d\uc84e\uc84f", - "\uc850\uc851\uc852\uc853\uc854\uc855\uc856\uc857\uc858\uc859\uc85a\uc85b\uc85c\uc85d\uc85e\uc85f", - "\uc860\uc861\uc862\uc863\uc864\uc865\uc866\uc867\uc868\uc869\uc86a\uc86b\uc86c\uc86d\uc86e\uc86f", - "\uc870\uc871\uc872\uc873\uc874\uc875\uc876\uc877\uc878\uc879\uc87a\uc87b\uc87c\uc87d\uc87e\uc87f", - "\uc880\uc881\uc882\uc883\uc884\uc885\uc886\uc887\uc888\uc889\uc88a\uc88b\uc88c\uc88d\uc88e\uc88f", - "\uc890\uc891\uc892\uc893\uc894\uc895\uc896\uc897\uc898\uc899\uc89a\uc89b\uc89c\uc89d\uc89e\uc89f", - "\uc8a0\uc8a1\uc8a2\uc8a3\uc8a4\uc8a5\uc8a6\uc8a7\uc8a8\uc8a9\uc8aa\uc8ab\uc8ac\uc8ad\uc8ae\uc8af", - "\uc8b0\uc8b1\uc8b2\uc8b3\uc8b4\uc8b5\uc8b6\uc8b7\uc8b8\uc8b9\uc8ba\uc8bb\uc8bc\uc8bd\uc8be\uc8bf", - "\uc8c0\uc8c1\uc8c2\uc8c3\uc8c4\uc8c5\uc8c6\uc8c7\uc8c8\uc8c9\uc8ca\uc8cb\uc8cc\uc8cd\uc8ce\uc8cf", - "\uc8d0\uc8d1\uc8d2\uc8d3\uc8d4\uc8d5\uc8d6\uc8d7\uc8d8\uc8d9\uc8da\uc8db\uc8dc\uc8dd\uc8de\uc8df", - "\uc8e0\uc8e1\uc8e2\uc8e3\uc8e4\uc8e5\uc8e6\uc8e7\uc8e8\uc8e9\uc8ea\uc8eb\uc8ec\uc8ed\uc8ee\uc8ef", - "\uc8f0\uc8f1\uc8f2\uc8f3\uc8f4\uc8f5\uc8f6\uc8f7\uc8f8\uc8f9\uc8fa\uc8fb\uc8fc\uc8fd\uc8fe\uc8ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_c9.png", - "ascent": %ascent%, - "chars": [ - "\uc900\uc901\uc902\uc903\uc904\uc905\uc906\uc907\uc908\uc909\uc90a\uc90b\uc90c\uc90d\uc90e\uc90f", - "\uc910\uc911\uc912\uc913\uc914\uc915\uc916\uc917\uc918\uc919\uc91a\uc91b\uc91c\uc91d\uc91e\uc91f", - "\uc920\uc921\uc922\uc923\uc924\uc925\uc926\uc927\uc928\uc929\uc92a\uc92b\uc92c\uc92d\uc92e\uc92f", - "\uc930\uc931\uc932\uc933\uc934\uc935\uc936\uc937\uc938\uc939\uc93a\uc93b\uc93c\uc93d\uc93e\uc93f", - "\uc940\uc941\uc942\uc943\uc944\uc945\uc946\uc947\uc948\uc949\uc94a\uc94b\uc94c\uc94d\uc94e\uc94f", - "\uc950\uc951\uc952\uc953\uc954\uc955\uc956\uc957\uc958\uc959\uc95a\uc95b\uc95c\uc95d\uc95e\uc95f", - "\uc960\uc961\uc962\uc963\uc964\uc965\uc966\uc967\uc968\uc969\uc96a\uc96b\uc96c\uc96d\uc96e\uc96f", - "\uc970\uc971\uc972\uc973\uc974\uc975\uc976\uc977\uc978\uc979\uc97a\uc97b\uc97c\uc97d\uc97e\uc97f", - "\uc980\uc981\uc982\uc983\uc984\uc985\uc986\uc987\uc988\uc989\uc98a\uc98b\uc98c\uc98d\uc98e\uc98f", - "\uc990\uc991\uc992\uc993\uc994\uc995\uc996\uc997\uc998\uc999\uc99a\uc99b\uc99c\uc99d\uc99e\uc99f", - "\uc9a0\uc9a1\uc9a2\uc9a3\uc9a4\uc9a5\uc9a6\uc9a7\uc9a8\uc9a9\uc9aa\uc9ab\uc9ac\uc9ad\uc9ae\uc9af", - "\uc9b0\uc9b1\uc9b2\uc9b3\uc9b4\uc9b5\uc9b6\uc9b7\uc9b8\uc9b9\uc9ba\uc9bb\uc9bc\uc9bd\uc9be\uc9bf", - "\uc9c0\uc9c1\uc9c2\uc9c3\uc9c4\uc9c5\uc9c6\uc9c7\uc9c8\uc9c9\uc9ca\uc9cb\uc9cc\uc9cd\uc9ce\uc9cf", - "\uc9d0\uc9d1\uc9d2\uc9d3\uc9d4\uc9d5\uc9d6\uc9d7\uc9d8\uc9d9\uc9da\uc9db\uc9dc\uc9dd\uc9de\uc9df", - "\uc9e0\uc9e1\uc9e2\uc9e3\uc9e4\uc9e5\uc9e6\uc9e7\uc9e8\uc9e9\uc9ea\uc9eb\uc9ec\uc9ed\uc9ee\uc9ef", - "\uc9f0\uc9f1\uc9f2\uc9f3\uc9f4\uc9f5\uc9f6\uc9f7\uc9f8\uc9f9\uc9fa\uc9fb\uc9fc\uc9fd\uc9fe\uc9ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_ca.png", - "ascent": %ascent%, - "chars": [ - "\uca00\uca01\uca02\uca03\uca04\uca05\uca06\uca07\uca08\uca09\uca0a\uca0b\uca0c\uca0d\uca0e\uca0f", - "\uca10\uca11\uca12\uca13\uca14\uca15\uca16\uca17\uca18\uca19\uca1a\uca1b\uca1c\uca1d\uca1e\uca1f", - "\uca20\uca21\uca22\uca23\uca24\uca25\uca26\uca27\uca28\uca29\uca2a\uca2b\uca2c\uca2d\uca2e\uca2f", - "\uca30\uca31\uca32\uca33\uca34\uca35\uca36\uca37\uca38\uca39\uca3a\uca3b\uca3c\uca3d\uca3e\uca3f", - "\uca40\uca41\uca42\uca43\uca44\uca45\uca46\uca47\uca48\uca49\uca4a\uca4b\uca4c\uca4d\uca4e\uca4f", - "\uca50\uca51\uca52\uca53\uca54\uca55\uca56\uca57\uca58\uca59\uca5a\uca5b\uca5c\uca5d\uca5e\uca5f", - "\uca60\uca61\uca62\uca63\uca64\uca65\uca66\uca67\uca68\uca69\uca6a\uca6b\uca6c\uca6d\uca6e\uca6f", - "\uca70\uca71\uca72\uca73\uca74\uca75\uca76\uca77\uca78\uca79\uca7a\uca7b\uca7c\uca7d\uca7e\uca7f", - "\uca80\uca81\uca82\uca83\uca84\uca85\uca86\uca87\uca88\uca89\uca8a\uca8b\uca8c\uca8d\uca8e\uca8f", - "\uca90\uca91\uca92\uca93\uca94\uca95\uca96\uca97\uca98\uca99\uca9a\uca9b\uca9c\uca9d\uca9e\uca9f", - "\ucaa0\ucaa1\ucaa2\ucaa3\ucaa4\ucaa5\ucaa6\ucaa7\ucaa8\ucaa9\ucaaa\ucaab\ucaac\ucaad\ucaae\ucaaf", - "\ucab0\ucab1\ucab2\ucab3\ucab4\ucab5\ucab6\ucab7\ucab8\ucab9\ucaba\ucabb\ucabc\ucabd\ucabe\ucabf", - "\ucac0\ucac1\ucac2\ucac3\ucac4\ucac5\ucac6\ucac7\ucac8\ucac9\ucaca\ucacb\ucacc\ucacd\ucace\ucacf", - "\ucad0\ucad1\ucad2\ucad3\ucad4\ucad5\ucad6\ucad7\ucad8\ucad9\ucada\ucadb\ucadc\ucadd\ucade\ucadf", - "\ucae0\ucae1\ucae2\ucae3\ucae4\ucae5\ucae6\ucae7\ucae8\ucae9\ucaea\ucaeb\ucaec\ucaed\ucaee\ucaef", - "\ucaf0\ucaf1\ucaf2\ucaf3\ucaf4\ucaf5\ucaf6\ucaf7\ucaf8\ucaf9\ucafa\ucafb\ucafc\ucafd\ucafe\ucaff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_cb.png", - "ascent": %ascent%, - "chars": [ - "\ucb00\ucb01\ucb02\ucb03\ucb04\ucb05\ucb06\ucb07\ucb08\ucb09\ucb0a\ucb0b\ucb0c\ucb0d\ucb0e\ucb0f", - "\ucb10\ucb11\ucb12\ucb13\ucb14\ucb15\ucb16\ucb17\ucb18\ucb19\ucb1a\ucb1b\ucb1c\ucb1d\ucb1e\ucb1f", - "\ucb20\ucb21\ucb22\ucb23\ucb24\ucb25\ucb26\ucb27\ucb28\ucb29\ucb2a\ucb2b\ucb2c\ucb2d\ucb2e\ucb2f", - "\ucb30\ucb31\ucb32\ucb33\ucb34\ucb35\ucb36\ucb37\ucb38\ucb39\ucb3a\ucb3b\ucb3c\ucb3d\ucb3e\ucb3f", - "\ucb40\ucb41\ucb42\ucb43\ucb44\ucb45\ucb46\ucb47\ucb48\ucb49\ucb4a\ucb4b\ucb4c\ucb4d\ucb4e\ucb4f", - "\ucb50\ucb51\ucb52\ucb53\ucb54\ucb55\ucb56\ucb57\ucb58\ucb59\ucb5a\ucb5b\ucb5c\ucb5d\ucb5e\ucb5f", - "\ucb60\ucb61\ucb62\ucb63\ucb64\ucb65\ucb66\ucb67\ucb68\ucb69\ucb6a\ucb6b\ucb6c\ucb6d\ucb6e\ucb6f", - "\ucb70\ucb71\ucb72\ucb73\ucb74\ucb75\ucb76\ucb77\ucb78\ucb79\ucb7a\ucb7b\ucb7c\ucb7d\ucb7e\ucb7f", - "\ucb80\ucb81\ucb82\ucb83\ucb84\ucb85\ucb86\ucb87\ucb88\ucb89\ucb8a\ucb8b\ucb8c\ucb8d\ucb8e\ucb8f", - "\ucb90\ucb91\ucb92\ucb93\ucb94\ucb95\ucb96\ucb97\ucb98\ucb99\ucb9a\ucb9b\ucb9c\ucb9d\ucb9e\ucb9f", - "\ucba0\ucba1\ucba2\ucba3\ucba4\ucba5\ucba6\ucba7\ucba8\ucba9\ucbaa\ucbab\ucbac\ucbad\ucbae\ucbaf", - "\ucbb0\ucbb1\ucbb2\ucbb3\ucbb4\ucbb5\ucbb6\ucbb7\ucbb8\ucbb9\ucbba\ucbbb\ucbbc\ucbbd\ucbbe\ucbbf", - "\ucbc0\ucbc1\ucbc2\ucbc3\ucbc4\ucbc5\ucbc6\ucbc7\ucbc8\ucbc9\ucbca\ucbcb\ucbcc\ucbcd\ucbce\ucbcf", - "\ucbd0\ucbd1\ucbd2\ucbd3\ucbd4\ucbd5\ucbd6\ucbd7\ucbd8\ucbd9\ucbda\ucbdb\ucbdc\ucbdd\ucbde\ucbdf", - "\ucbe0\ucbe1\ucbe2\ucbe3\ucbe4\ucbe5\ucbe6\ucbe7\ucbe8\ucbe9\ucbea\ucbeb\ucbec\ucbed\ucbee\ucbef", - "\ucbf0\ucbf1\ucbf2\ucbf3\ucbf4\ucbf5\ucbf6\ucbf7\ucbf8\ucbf9\ucbfa\ucbfb\ucbfc\ucbfd\ucbfe\ucbff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_cc.png", - "ascent": %ascent%, - "chars": [ - "\ucc00\ucc01\ucc02\ucc03\ucc04\ucc05\ucc06\ucc07\ucc08\ucc09\ucc0a\ucc0b\ucc0c\ucc0d\ucc0e\ucc0f", - "\ucc10\ucc11\ucc12\ucc13\ucc14\ucc15\ucc16\ucc17\ucc18\ucc19\ucc1a\ucc1b\ucc1c\ucc1d\ucc1e\ucc1f", - "\ucc20\ucc21\ucc22\ucc23\ucc24\ucc25\ucc26\ucc27\ucc28\ucc29\ucc2a\ucc2b\ucc2c\ucc2d\ucc2e\ucc2f", - "\ucc30\ucc31\ucc32\ucc33\ucc34\ucc35\ucc36\ucc37\ucc38\ucc39\ucc3a\ucc3b\ucc3c\ucc3d\ucc3e\ucc3f", - "\ucc40\ucc41\ucc42\ucc43\ucc44\ucc45\ucc46\ucc47\ucc48\ucc49\ucc4a\ucc4b\ucc4c\ucc4d\ucc4e\ucc4f", - "\ucc50\ucc51\ucc52\ucc53\ucc54\ucc55\ucc56\ucc57\ucc58\ucc59\ucc5a\ucc5b\ucc5c\ucc5d\ucc5e\ucc5f", - "\ucc60\ucc61\ucc62\ucc63\ucc64\ucc65\ucc66\ucc67\ucc68\ucc69\ucc6a\ucc6b\ucc6c\ucc6d\ucc6e\ucc6f", - "\ucc70\ucc71\ucc72\ucc73\ucc74\ucc75\ucc76\ucc77\ucc78\ucc79\ucc7a\ucc7b\ucc7c\ucc7d\ucc7e\ucc7f", - "\ucc80\ucc81\ucc82\ucc83\ucc84\ucc85\ucc86\ucc87\ucc88\ucc89\ucc8a\ucc8b\ucc8c\ucc8d\ucc8e\ucc8f", - "\ucc90\ucc91\ucc92\ucc93\ucc94\ucc95\ucc96\ucc97\ucc98\ucc99\ucc9a\ucc9b\ucc9c\ucc9d\ucc9e\ucc9f", - "\ucca0\ucca1\ucca2\ucca3\ucca4\ucca5\ucca6\ucca7\ucca8\ucca9\uccaa\uccab\uccac\uccad\uccae\uccaf", - "\uccb0\uccb1\uccb2\uccb3\uccb4\uccb5\uccb6\uccb7\uccb8\uccb9\uccba\uccbb\uccbc\uccbd\uccbe\uccbf", - "\uccc0\uccc1\uccc2\uccc3\uccc4\uccc5\uccc6\uccc7\uccc8\uccc9\uccca\ucccb\ucccc\ucccd\uccce\ucccf", - "\uccd0\uccd1\uccd2\uccd3\uccd4\uccd5\uccd6\uccd7\uccd8\uccd9\uccda\uccdb\uccdc\uccdd\uccde\uccdf", - "\ucce0\ucce1\ucce2\ucce3\ucce4\ucce5\ucce6\ucce7\ucce8\ucce9\uccea\ucceb\uccec\ucced\uccee\uccef", - "\uccf0\uccf1\uccf2\uccf3\uccf4\uccf5\uccf6\uccf7\uccf8\uccf9\uccfa\uccfb\uccfc\uccfd\uccfe\uccff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_cd.png", - "ascent": %ascent%, - "chars": [ - "\ucd00\ucd01\ucd02\ucd03\ucd04\ucd05\ucd06\ucd07\ucd08\ucd09\ucd0a\ucd0b\ucd0c\ucd0d\ucd0e\ucd0f", - "\ucd10\ucd11\ucd12\ucd13\ucd14\ucd15\ucd16\ucd17\ucd18\ucd19\ucd1a\ucd1b\ucd1c\ucd1d\ucd1e\ucd1f", - "\ucd20\ucd21\ucd22\ucd23\ucd24\ucd25\ucd26\ucd27\ucd28\ucd29\ucd2a\ucd2b\ucd2c\ucd2d\ucd2e\ucd2f", - "\ucd30\ucd31\ucd32\ucd33\ucd34\ucd35\ucd36\ucd37\ucd38\ucd39\ucd3a\ucd3b\ucd3c\ucd3d\ucd3e\ucd3f", - "\ucd40\ucd41\ucd42\ucd43\ucd44\ucd45\ucd46\ucd47\ucd48\ucd49\ucd4a\ucd4b\ucd4c\ucd4d\ucd4e\ucd4f", - "\ucd50\ucd51\ucd52\ucd53\ucd54\ucd55\ucd56\ucd57\ucd58\ucd59\ucd5a\ucd5b\ucd5c\ucd5d\ucd5e\ucd5f", - "\ucd60\ucd61\ucd62\ucd63\ucd64\ucd65\ucd66\ucd67\ucd68\ucd69\ucd6a\ucd6b\ucd6c\ucd6d\ucd6e\ucd6f", - "\ucd70\ucd71\ucd72\ucd73\ucd74\ucd75\ucd76\ucd77\ucd78\ucd79\ucd7a\ucd7b\ucd7c\ucd7d\ucd7e\ucd7f", - "\ucd80\ucd81\ucd82\ucd83\ucd84\ucd85\ucd86\ucd87\ucd88\ucd89\ucd8a\ucd8b\ucd8c\ucd8d\ucd8e\ucd8f", - "\ucd90\ucd91\ucd92\ucd93\ucd94\ucd95\ucd96\ucd97\ucd98\ucd99\ucd9a\ucd9b\ucd9c\ucd9d\ucd9e\ucd9f", - "\ucda0\ucda1\ucda2\ucda3\ucda4\ucda5\ucda6\ucda7\ucda8\ucda9\ucdaa\ucdab\ucdac\ucdad\ucdae\ucdaf", - "\ucdb0\ucdb1\ucdb2\ucdb3\ucdb4\ucdb5\ucdb6\ucdb7\ucdb8\ucdb9\ucdba\ucdbb\ucdbc\ucdbd\ucdbe\ucdbf", - "\ucdc0\ucdc1\ucdc2\ucdc3\ucdc4\ucdc5\ucdc6\ucdc7\ucdc8\ucdc9\ucdca\ucdcb\ucdcc\ucdcd\ucdce\ucdcf", - "\ucdd0\ucdd1\ucdd2\ucdd3\ucdd4\ucdd5\ucdd6\ucdd7\ucdd8\ucdd9\ucdda\ucddb\ucddc\ucddd\ucdde\ucddf", - "\ucde0\ucde1\ucde2\ucde3\ucde4\ucde5\ucde6\ucde7\ucde8\ucde9\ucdea\ucdeb\ucdec\ucded\ucdee\ucdef", - "\ucdf0\ucdf1\ucdf2\ucdf3\ucdf4\ucdf5\ucdf6\ucdf7\ucdf8\ucdf9\ucdfa\ucdfb\ucdfc\ucdfd\ucdfe\ucdff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_ce.png", - "ascent": %ascent%, - "chars": [ - "\uce00\uce01\uce02\uce03\uce04\uce05\uce06\uce07\uce08\uce09\uce0a\uce0b\uce0c\uce0d\uce0e\uce0f", - "\uce10\uce11\uce12\uce13\uce14\uce15\uce16\uce17\uce18\uce19\uce1a\uce1b\uce1c\uce1d\uce1e\uce1f", - "\uce20\uce21\uce22\uce23\uce24\uce25\uce26\uce27\uce28\uce29\uce2a\uce2b\uce2c\uce2d\uce2e\uce2f", - "\uce30\uce31\uce32\uce33\uce34\uce35\uce36\uce37\uce38\uce39\uce3a\uce3b\uce3c\uce3d\uce3e\uce3f", - "\uce40\uce41\uce42\uce43\uce44\uce45\uce46\uce47\uce48\uce49\uce4a\uce4b\uce4c\uce4d\uce4e\uce4f", - "\uce50\uce51\uce52\uce53\uce54\uce55\uce56\uce57\uce58\uce59\uce5a\uce5b\uce5c\uce5d\uce5e\uce5f", - "\uce60\uce61\uce62\uce63\uce64\uce65\uce66\uce67\uce68\uce69\uce6a\uce6b\uce6c\uce6d\uce6e\uce6f", - "\uce70\uce71\uce72\uce73\uce74\uce75\uce76\uce77\uce78\uce79\uce7a\uce7b\uce7c\uce7d\uce7e\uce7f", - "\uce80\uce81\uce82\uce83\uce84\uce85\uce86\uce87\uce88\uce89\uce8a\uce8b\uce8c\uce8d\uce8e\uce8f", - "\uce90\uce91\uce92\uce93\uce94\uce95\uce96\uce97\uce98\uce99\uce9a\uce9b\uce9c\uce9d\uce9e\uce9f", - "\ucea0\ucea1\ucea2\ucea3\ucea4\ucea5\ucea6\ucea7\ucea8\ucea9\uceaa\uceab\uceac\ucead\uceae\uceaf", - "\uceb0\uceb1\uceb2\uceb3\uceb4\uceb5\uceb6\uceb7\uceb8\uceb9\uceba\ucebb\ucebc\ucebd\ucebe\ucebf", - "\ucec0\ucec1\ucec2\ucec3\ucec4\ucec5\ucec6\ucec7\ucec8\ucec9\uceca\ucecb\ucecc\ucecd\ucece\ucecf", - "\uced0\uced1\uced2\uced3\uced4\uced5\uced6\uced7\uced8\uced9\uceda\ucedb\ucedc\ucedd\ucede\ucedf", - "\ucee0\ucee1\ucee2\ucee3\ucee4\ucee5\ucee6\ucee7\ucee8\ucee9\uceea\uceeb\uceec\uceed\uceee\uceef", - "\ucef0\ucef1\ucef2\ucef3\ucef4\ucef5\ucef6\ucef7\ucef8\ucef9\ucefa\ucefb\ucefc\ucefd\ucefe\uceff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_cf.png", - "ascent": %ascent%, - "chars": [ - "\ucf00\ucf01\ucf02\ucf03\ucf04\ucf05\ucf06\ucf07\ucf08\ucf09\ucf0a\ucf0b\ucf0c\ucf0d\ucf0e\ucf0f", - "\ucf10\ucf11\ucf12\ucf13\ucf14\ucf15\ucf16\ucf17\ucf18\ucf19\ucf1a\ucf1b\ucf1c\ucf1d\ucf1e\ucf1f", - "\ucf20\ucf21\ucf22\ucf23\ucf24\ucf25\ucf26\ucf27\ucf28\ucf29\ucf2a\ucf2b\ucf2c\ucf2d\ucf2e\ucf2f", - "\ucf30\ucf31\ucf32\ucf33\ucf34\ucf35\ucf36\ucf37\ucf38\ucf39\ucf3a\ucf3b\ucf3c\ucf3d\ucf3e\ucf3f", - "\ucf40\ucf41\ucf42\ucf43\ucf44\ucf45\ucf46\ucf47\ucf48\ucf49\ucf4a\ucf4b\ucf4c\ucf4d\ucf4e\ucf4f", - "\ucf50\ucf51\ucf52\ucf53\ucf54\ucf55\ucf56\ucf57\ucf58\ucf59\ucf5a\ucf5b\ucf5c\ucf5d\ucf5e\ucf5f", - "\ucf60\ucf61\ucf62\ucf63\ucf64\ucf65\ucf66\ucf67\ucf68\ucf69\ucf6a\ucf6b\ucf6c\ucf6d\ucf6e\ucf6f", - "\ucf70\ucf71\ucf72\ucf73\ucf74\ucf75\ucf76\ucf77\ucf78\ucf79\ucf7a\ucf7b\ucf7c\ucf7d\ucf7e\ucf7f", - "\ucf80\ucf81\ucf82\ucf83\ucf84\ucf85\ucf86\ucf87\ucf88\ucf89\ucf8a\ucf8b\ucf8c\ucf8d\ucf8e\ucf8f", - "\ucf90\ucf91\ucf92\ucf93\ucf94\ucf95\ucf96\ucf97\ucf98\ucf99\ucf9a\ucf9b\ucf9c\ucf9d\ucf9e\ucf9f", - "\ucfa0\ucfa1\ucfa2\ucfa3\ucfa4\ucfa5\ucfa6\ucfa7\ucfa8\ucfa9\ucfaa\ucfab\ucfac\ucfad\ucfae\ucfaf", - "\ucfb0\ucfb1\ucfb2\ucfb3\ucfb4\ucfb5\ucfb6\ucfb7\ucfb8\ucfb9\ucfba\ucfbb\ucfbc\ucfbd\ucfbe\ucfbf", - "\ucfc0\ucfc1\ucfc2\ucfc3\ucfc4\ucfc5\ucfc6\ucfc7\ucfc8\ucfc9\ucfca\ucfcb\ucfcc\ucfcd\ucfce\ucfcf", - "\ucfd0\ucfd1\ucfd2\ucfd3\ucfd4\ucfd5\ucfd6\ucfd7\ucfd8\ucfd9\ucfda\ucfdb\ucfdc\ucfdd\ucfde\ucfdf", - "\ucfe0\ucfe1\ucfe2\ucfe3\ucfe4\ucfe5\ucfe6\ucfe7\ucfe8\ucfe9\ucfea\ucfeb\ucfec\ucfed\ucfee\ucfef", - "\ucff0\ucff1\ucff2\ucff3\ucff4\ucff5\ucff6\ucff7\ucff8\ucff9\ucffa\ucffb\ucffc\ucffd\ucffe\ucfff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_d0.png", - "ascent": %ascent%, - "chars": [ - "\ud000\ud001\ud002\ud003\ud004\ud005\ud006\ud007\ud008\ud009\ud00a\ud00b\ud00c\ud00d\ud00e\ud00f", - "\ud010\ud011\ud012\ud013\ud014\ud015\ud016\ud017\ud018\ud019\ud01a\ud01b\ud01c\ud01d\ud01e\ud01f", - "\ud020\ud021\ud022\ud023\ud024\ud025\ud026\ud027\ud028\ud029\ud02a\ud02b\ud02c\ud02d\ud02e\ud02f", - "\ud030\ud031\ud032\ud033\ud034\ud035\ud036\ud037\ud038\ud039\ud03a\ud03b\ud03c\ud03d\ud03e\ud03f", - "\ud040\ud041\ud042\ud043\ud044\ud045\ud046\ud047\ud048\ud049\ud04a\ud04b\ud04c\ud04d\ud04e\ud04f", - "\ud050\ud051\ud052\ud053\ud054\ud055\ud056\ud057\ud058\ud059\ud05a\ud05b\ud05c\ud05d\ud05e\ud05f", - "\ud060\ud061\ud062\ud063\ud064\ud065\ud066\ud067\ud068\ud069\ud06a\ud06b\ud06c\ud06d\ud06e\ud06f", - "\ud070\ud071\ud072\ud073\ud074\ud075\ud076\ud077\ud078\ud079\ud07a\ud07b\ud07c\ud07d\ud07e\ud07f", - "\ud080\ud081\ud082\ud083\ud084\ud085\ud086\ud087\ud088\ud089\ud08a\ud08b\ud08c\ud08d\ud08e\ud08f", - "\ud090\ud091\ud092\ud093\ud094\ud095\ud096\ud097\ud098\ud099\ud09a\ud09b\ud09c\ud09d\ud09e\ud09f", - "\ud0a0\ud0a1\ud0a2\ud0a3\ud0a4\ud0a5\ud0a6\ud0a7\ud0a8\ud0a9\ud0aa\ud0ab\ud0ac\ud0ad\ud0ae\ud0af", - "\ud0b0\ud0b1\ud0b2\ud0b3\ud0b4\ud0b5\ud0b6\ud0b7\ud0b8\ud0b9\ud0ba\ud0bb\ud0bc\ud0bd\ud0be\ud0bf", - "\ud0c0\ud0c1\ud0c2\ud0c3\ud0c4\ud0c5\ud0c6\ud0c7\ud0c8\ud0c9\ud0ca\ud0cb\ud0cc\ud0cd\ud0ce\ud0cf", - "\ud0d0\ud0d1\ud0d2\ud0d3\ud0d4\ud0d5\ud0d6\ud0d7\ud0d8\ud0d9\ud0da\ud0db\ud0dc\ud0dd\ud0de\ud0df", - "\ud0e0\ud0e1\ud0e2\ud0e3\ud0e4\ud0e5\ud0e6\ud0e7\ud0e8\ud0e9\ud0ea\ud0eb\ud0ec\ud0ed\ud0ee\ud0ef", - "\ud0f0\ud0f1\ud0f2\ud0f3\ud0f4\ud0f5\ud0f6\ud0f7\ud0f8\ud0f9\ud0fa\ud0fb\ud0fc\ud0fd\ud0fe\ud0ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_d1.png", - "ascent": %ascent%, - "chars": [ - "\ud100\ud101\ud102\ud103\ud104\ud105\ud106\ud107\ud108\ud109\ud10a\ud10b\ud10c\ud10d\ud10e\ud10f", - "\ud110\ud111\ud112\ud113\ud114\ud115\ud116\ud117\ud118\ud119\ud11a\ud11b\ud11c\ud11d\ud11e\ud11f", - "\ud120\ud121\ud122\ud123\ud124\ud125\ud126\ud127\ud128\ud129\ud12a\ud12b\ud12c\ud12d\ud12e\ud12f", - "\ud130\ud131\ud132\ud133\ud134\ud135\ud136\ud137\ud138\ud139\ud13a\ud13b\ud13c\ud13d\ud13e\ud13f", - "\ud140\ud141\ud142\ud143\ud144\ud145\ud146\ud147\ud148\ud149\ud14a\ud14b\ud14c\ud14d\ud14e\ud14f", - "\ud150\ud151\ud152\ud153\ud154\ud155\ud156\ud157\ud158\ud159\ud15a\ud15b\ud15c\ud15d\ud15e\ud15f", - "\ud160\ud161\ud162\ud163\ud164\ud165\ud166\ud167\ud168\ud169\ud16a\ud16b\ud16c\ud16d\ud16e\ud16f", - "\ud170\ud171\ud172\ud173\ud174\ud175\ud176\ud177\ud178\ud179\ud17a\ud17b\ud17c\ud17d\ud17e\ud17f", - "\ud180\ud181\ud182\ud183\ud184\ud185\ud186\ud187\ud188\ud189\ud18a\ud18b\ud18c\ud18d\ud18e\ud18f", - "\ud190\ud191\ud192\ud193\ud194\ud195\ud196\ud197\ud198\ud199\ud19a\ud19b\ud19c\ud19d\ud19e\ud19f", - "\ud1a0\ud1a1\ud1a2\ud1a3\ud1a4\ud1a5\ud1a6\ud1a7\ud1a8\ud1a9\ud1aa\ud1ab\ud1ac\ud1ad\ud1ae\ud1af", - "\ud1b0\ud1b1\ud1b2\ud1b3\ud1b4\ud1b5\ud1b6\ud1b7\ud1b8\ud1b9\ud1ba\ud1bb\ud1bc\ud1bd\ud1be\ud1bf", - "\ud1c0\ud1c1\ud1c2\ud1c3\ud1c4\ud1c5\ud1c6\ud1c7\ud1c8\ud1c9\ud1ca\ud1cb\ud1cc\ud1cd\ud1ce\ud1cf", - "\ud1d0\ud1d1\ud1d2\ud1d3\ud1d4\ud1d5\ud1d6\ud1d7\ud1d8\ud1d9\ud1da\ud1db\ud1dc\ud1dd\ud1de\ud1df", - "\ud1e0\ud1e1\ud1e2\ud1e3\ud1e4\ud1e5\ud1e6\ud1e7\ud1e8\ud1e9\ud1ea\ud1eb\ud1ec\ud1ed\ud1ee\ud1ef", - "\ud1f0\ud1f1\ud1f2\ud1f3\ud1f4\ud1f5\ud1f6\ud1f7\ud1f8\ud1f9\ud1fa\ud1fb\ud1fc\ud1fd\ud1fe\ud1ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_d2.png", - "ascent": %ascent%, - "chars": [ - "\ud200\ud201\ud202\ud203\ud204\ud205\ud206\ud207\ud208\ud209\ud20a\ud20b\ud20c\ud20d\ud20e\ud20f", - "\ud210\ud211\ud212\ud213\ud214\ud215\ud216\ud217\ud218\ud219\ud21a\ud21b\ud21c\ud21d\ud21e\ud21f", - "\ud220\ud221\ud222\ud223\ud224\ud225\ud226\ud227\ud228\ud229\ud22a\ud22b\ud22c\ud22d\ud22e\ud22f", - "\ud230\ud231\ud232\ud233\ud234\ud235\ud236\ud237\ud238\ud239\ud23a\ud23b\ud23c\ud23d\ud23e\ud23f", - "\ud240\ud241\ud242\ud243\ud244\ud245\ud246\ud247\ud248\ud249\ud24a\ud24b\ud24c\ud24d\ud24e\ud24f", - "\ud250\ud251\ud252\ud253\ud254\ud255\ud256\ud257\ud258\ud259\ud25a\ud25b\ud25c\ud25d\ud25e\ud25f", - "\ud260\ud261\ud262\ud263\ud264\ud265\ud266\ud267\ud268\ud269\ud26a\ud26b\ud26c\ud26d\ud26e\ud26f", - "\ud270\ud271\ud272\ud273\ud274\ud275\ud276\ud277\ud278\ud279\ud27a\ud27b\ud27c\ud27d\ud27e\ud27f", - "\ud280\ud281\ud282\ud283\ud284\ud285\ud286\ud287\ud288\ud289\ud28a\ud28b\ud28c\ud28d\ud28e\ud28f", - "\ud290\ud291\ud292\ud293\ud294\ud295\ud296\ud297\ud298\ud299\ud29a\ud29b\ud29c\ud29d\ud29e\ud29f", - "\ud2a0\ud2a1\ud2a2\ud2a3\ud2a4\ud2a5\ud2a6\ud2a7\ud2a8\ud2a9\ud2aa\ud2ab\ud2ac\ud2ad\ud2ae\ud2af", - "\ud2b0\ud2b1\ud2b2\ud2b3\ud2b4\ud2b5\ud2b6\ud2b7\ud2b8\ud2b9\ud2ba\ud2bb\ud2bc\ud2bd\ud2be\ud2bf", - "\ud2c0\ud2c1\ud2c2\ud2c3\ud2c4\ud2c5\ud2c6\ud2c7\ud2c8\ud2c9\ud2ca\ud2cb\ud2cc\ud2cd\ud2ce\ud2cf", - "\ud2d0\ud2d1\ud2d2\ud2d3\ud2d4\ud2d5\ud2d6\ud2d7\ud2d8\ud2d9\ud2da\ud2db\ud2dc\ud2dd\ud2de\ud2df", - "\ud2e0\ud2e1\ud2e2\ud2e3\ud2e4\ud2e5\ud2e6\ud2e7\ud2e8\ud2e9\ud2ea\ud2eb\ud2ec\ud2ed\ud2ee\ud2ef", - "\ud2f0\ud2f1\ud2f2\ud2f3\ud2f4\ud2f5\ud2f6\ud2f7\ud2f8\ud2f9\ud2fa\ud2fb\ud2fc\ud2fd\ud2fe\ud2ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_d3.png", - "ascent": %ascent%, - "chars": [ - "\ud300\ud301\ud302\ud303\ud304\ud305\ud306\ud307\ud308\ud309\ud30a\ud30b\ud30c\ud30d\ud30e\ud30f", - "\ud310\ud311\ud312\ud313\ud314\ud315\ud316\ud317\ud318\ud319\ud31a\ud31b\ud31c\ud31d\ud31e\ud31f", - "\ud320\ud321\ud322\ud323\ud324\ud325\ud326\ud327\ud328\ud329\ud32a\ud32b\ud32c\ud32d\ud32e\ud32f", - "\ud330\ud331\ud332\ud333\ud334\ud335\ud336\ud337\ud338\ud339\ud33a\ud33b\ud33c\ud33d\ud33e\ud33f", - "\ud340\ud341\ud342\ud343\ud344\ud345\ud346\ud347\ud348\ud349\ud34a\ud34b\ud34c\ud34d\ud34e\ud34f", - "\ud350\ud351\ud352\ud353\ud354\ud355\ud356\ud357\ud358\ud359\ud35a\ud35b\ud35c\ud35d\ud35e\ud35f", - "\ud360\ud361\ud362\ud363\ud364\ud365\ud366\ud367\ud368\ud369\ud36a\ud36b\ud36c\ud36d\ud36e\ud36f", - "\ud370\ud371\ud372\ud373\ud374\ud375\ud376\ud377\ud378\ud379\ud37a\ud37b\ud37c\ud37d\ud37e\ud37f", - "\ud380\ud381\ud382\ud383\ud384\ud385\ud386\ud387\ud388\ud389\ud38a\ud38b\ud38c\ud38d\ud38e\ud38f", - "\ud390\ud391\ud392\ud393\ud394\ud395\ud396\ud397\ud398\ud399\ud39a\ud39b\ud39c\ud39d\ud39e\ud39f", - "\ud3a0\ud3a1\ud3a2\ud3a3\ud3a4\ud3a5\ud3a6\ud3a7\ud3a8\ud3a9\ud3aa\ud3ab\ud3ac\ud3ad\ud3ae\ud3af", - "\ud3b0\ud3b1\ud3b2\ud3b3\ud3b4\ud3b5\ud3b6\ud3b7\ud3b8\ud3b9\ud3ba\ud3bb\ud3bc\ud3bd\ud3be\ud3bf", - "\ud3c0\ud3c1\ud3c2\ud3c3\ud3c4\ud3c5\ud3c6\ud3c7\ud3c8\ud3c9\ud3ca\ud3cb\ud3cc\ud3cd\ud3ce\ud3cf", - "\ud3d0\ud3d1\ud3d2\ud3d3\ud3d4\ud3d5\ud3d6\ud3d7\ud3d8\ud3d9\ud3da\ud3db\ud3dc\ud3dd\ud3de\ud3df", - "\ud3e0\ud3e1\ud3e2\ud3e3\ud3e4\ud3e5\ud3e6\ud3e7\ud3e8\ud3e9\ud3ea\ud3eb\ud3ec\ud3ed\ud3ee\ud3ef", - "\ud3f0\ud3f1\ud3f2\ud3f3\ud3f4\ud3f5\ud3f6\ud3f7\ud3f8\ud3f9\ud3fa\ud3fb\ud3fc\ud3fd\ud3fe\ud3ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_d4.png", - "ascent": %ascent%, - "chars": [ - "\ud400\ud401\ud402\ud403\ud404\ud405\ud406\ud407\ud408\ud409\ud40a\ud40b\ud40c\ud40d\ud40e\ud40f", - "\ud410\ud411\ud412\ud413\ud414\ud415\ud416\ud417\ud418\ud419\ud41a\ud41b\ud41c\ud41d\ud41e\ud41f", - "\ud420\ud421\ud422\ud423\ud424\ud425\ud426\ud427\ud428\ud429\ud42a\ud42b\ud42c\ud42d\ud42e\ud42f", - "\ud430\ud431\ud432\ud433\ud434\ud435\ud436\ud437\ud438\ud439\ud43a\ud43b\ud43c\ud43d\ud43e\ud43f", - "\ud440\ud441\ud442\ud443\ud444\ud445\ud446\ud447\ud448\ud449\ud44a\ud44b\ud44c\ud44d\ud44e\ud44f", - "\ud450\ud451\ud452\ud453\ud454\ud455\ud456\ud457\ud458\ud459\ud45a\ud45b\ud45c\ud45d\ud45e\ud45f", - "\ud460\ud461\ud462\ud463\ud464\ud465\ud466\ud467\ud468\ud469\ud46a\ud46b\ud46c\ud46d\ud46e\ud46f", - "\ud470\ud471\ud472\ud473\ud474\ud475\ud476\ud477\ud478\ud479\ud47a\ud47b\ud47c\ud47d\ud47e\ud47f", - "\ud480\ud481\ud482\ud483\ud484\ud485\ud486\ud487\ud488\ud489\ud48a\ud48b\ud48c\ud48d\ud48e\ud48f", - "\ud490\ud491\ud492\ud493\ud494\ud495\ud496\ud497\ud498\ud499\ud49a\ud49b\ud49c\ud49d\ud49e\ud49f", - "\ud4a0\ud4a1\ud4a2\ud4a3\ud4a4\ud4a5\ud4a6\ud4a7\ud4a8\ud4a9\ud4aa\ud4ab\ud4ac\ud4ad\ud4ae\ud4af", - "\ud4b0\ud4b1\ud4b2\ud4b3\ud4b4\ud4b5\ud4b6\ud4b7\ud4b8\ud4b9\ud4ba\ud4bb\ud4bc\ud4bd\ud4be\ud4bf", - "\ud4c0\ud4c1\ud4c2\ud4c3\ud4c4\ud4c5\ud4c6\ud4c7\ud4c8\ud4c9\ud4ca\ud4cb\ud4cc\ud4cd\ud4ce\ud4cf", - "\ud4d0\ud4d1\ud4d2\ud4d3\ud4d4\ud4d5\ud4d6\ud4d7\ud4d8\ud4d9\ud4da\ud4db\ud4dc\ud4dd\ud4de\ud4df", - "\ud4e0\ud4e1\ud4e2\ud4e3\ud4e4\ud4e5\ud4e6\ud4e7\ud4e8\ud4e9\ud4ea\ud4eb\ud4ec\ud4ed\ud4ee\ud4ef", - "\ud4f0\ud4f1\ud4f2\ud4f3\ud4f4\ud4f5\ud4f6\ud4f7\ud4f8\ud4f9\ud4fa\ud4fb\ud4fc\ud4fd\ud4fe\ud4ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_d5.png", - "ascent": %ascent%, - "chars": [ - "\ud500\ud501\ud502\ud503\ud504\ud505\ud506\ud507\ud508\ud509\ud50a\ud50b\ud50c\ud50d\ud50e\ud50f", - "\ud510\ud511\ud512\ud513\ud514\ud515\ud516\ud517\ud518\ud519\ud51a\ud51b\ud51c\ud51d\ud51e\ud51f", - "\ud520\ud521\ud522\ud523\ud524\ud525\ud526\ud527\ud528\ud529\ud52a\ud52b\ud52c\ud52d\ud52e\ud52f", - "\ud530\ud531\ud532\ud533\ud534\ud535\ud536\ud537\ud538\ud539\ud53a\ud53b\ud53c\ud53d\ud53e\ud53f", - "\ud540\ud541\ud542\ud543\ud544\ud545\ud546\ud547\ud548\ud549\ud54a\ud54b\ud54c\ud54d\ud54e\ud54f", - "\ud550\ud551\ud552\ud553\ud554\ud555\ud556\ud557\ud558\ud559\ud55a\ud55b\ud55c\ud55d\ud55e\ud55f", - "\ud560\ud561\ud562\ud563\ud564\ud565\ud566\ud567\ud568\ud569\ud56a\ud56b\ud56c\ud56d\ud56e\ud56f", - "\ud570\ud571\ud572\ud573\ud574\ud575\ud576\ud577\ud578\ud579\ud57a\ud57b\ud57c\ud57d\ud57e\ud57f", - "\ud580\ud581\ud582\ud583\ud584\ud585\ud586\ud587\ud588\ud589\ud58a\ud58b\ud58c\ud58d\ud58e\ud58f", - "\ud590\ud591\ud592\ud593\ud594\ud595\ud596\ud597\ud598\ud599\ud59a\ud59b\ud59c\ud59d\ud59e\ud59f", - "\ud5a0\ud5a1\ud5a2\ud5a3\ud5a4\ud5a5\ud5a6\ud5a7\ud5a8\ud5a9\ud5aa\ud5ab\ud5ac\ud5ad\ud5ae\ud5af", - "\ud5b0\ud5b1\ud5b2\ud5b3\ud5b4\ud5b5\ud5b6\ud5b7\ud5b8\ud5b9\ud5ba\ud5bb\ud5bc\ud5bd\ud5be\ud5bf", - "\ud5c0\ud5c1\ud5c2\ud5c3\ud5c4\ud5c5\ud5c6\ud5c7\ud5c8\ud5c9\ud5ca\ud5cb\ud5cc\ud5cd\ud5ce\ud5cf", - "\ud5d0\ud5d1\ud5d2\ud5d3\ud5d4\ud5d5\ud5d6\ud5d7\ud5d8\ud5d9\ud5da\ud5db\ud5dc\ud5dd\ud5de\ud5df", - "\ud5e0\ud5e1\ud5e2\ud5e3\ud5e4\ud5e5\ud5e6\ud5e7\ud5e8\ud5e9\ud5ea\ud5eb\ud5ec\ud5ed\ud5ee\ud5ef", - "\ud5f0\ud5f1\ud5f2\ud5f3\ud5f4\ud5f5\ud5f6\ud5f7\ud5f8\ud5f9\ud5fa\ud5fb\ud5fc\ud5fd\ud5fe\ud5ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_d6.png", - "ascent": %ascent%, - "chars": [ - "\ud600\ud601\ud602\ud603\ud604\ud605\ud606\ud607\ud608\ud609\ud60a\ud60b\ud60c\ud60d\ud60e\ud60f", - "\ud610\ud611\ud612\ud613\ud614\ud615\ud616\ud617\ud618\ud619\ud61a\ud61b\ud61c\ud61d\ud61e\ud61f", - "\ud620\ud621\ud622\ud623\ud624\ud625\ud626\ud627\ud628\ud629\ud62a\ud62b\ud62c\ud62d\ud62e\ud62f", - "\ud630\ud631\ud632\ud633\ud634\ud635\ud636\ud637\ud638\ud639\ud63a\ud63b\ud63c\ud63d\ud63e\ud63f", - "\ud640\ud641\ud642\ud643\ud644\ud645\ud646\ud647\ud648\ud649\ud64a\ud64b\ud64c\ud64d\ud64e\ud64f", - "\ud650\ud651\ud652\ud653\ud654\ud655\ud656\ud657\ud658\ud659\ud65a\ud65b\ud65c\ud65d\ud65e\ud65f", - "\ud660\ud661\ud662\ud663\ud664\ud665\ud666\ud667\ud668\ud669\ud66a\ud66b\ud66c\ud66d\ud66e\ud66f", - "\ud670\ud671\ud672\ud673\ud674\ud675\ud676\ud677\ud678\ud679\ud67a\ud67b\ud67c\ud67d\ud67e\ud67f", - "\ud680\ud681\ud682\ud683\ud684\ud685\ud686\ud687\ud688\ud689\ud68a\ud68b\ud68c\ud68d\ud68e\ud68f", - "\ud690\ud691\ud692\ud693\ud694\ud695\ud696\ud697\ud698\ud699\ud69a\ud69b\ud69c\ud69d\ud69e\ud69f", - "\ud6a0\ud6a1\ud6a2\ud6a3\ud6a4\ud6a5\ud6a6\ud6a7\ud6a8\ud6a9\ud6aa\ud6ab\ud6ac\ud6ad\ud6ae\ud6af", - "\ud6b0\ud6b1\ud6b2\ud6b3\ud6b4\ud6b5\ud6b6\ud6b7\ud6b8\ud6b9\ud6ba\ud6bb\ud6bc\ud6bd\ud6be\ud6bf", - "\ud6c0\ud6c1\ud6c2\ud6c3\ud6c4\ud6c5\ud6c6\ud6c7\ud6c8\ud6c9\ud6ca\ud6cb\ud6cc\ud6cd\ud6ce\ud6cf", - "\ud6d0\ud6d1\ud6d2\ud6d3\ud6d4\ud6d5\ud6d6\ud6d7\ud6d8\ud6d9\ud6da\ud6db\ud6dc\ud6dd\ud6de\ud6df", - "\ud6e0\ud6e1\ud6e2\ud6e3\ud6e4\ud6e5\ud6e6\ud6e7\ud6e8\ud6e9\ud6ea\ud6eb\ud6ec\ud6ed\ud6ee\ud6ef", - "\ud6f0\ud6f1\ud6f2\ud6f3\ud6f4\ud6f5\ud6f6\ud6f7\ud6f8\ud6f9\ud6fa\ud6fb\ud6fc\ud6fd\ud6fe\ud6ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_d7.png", - "ascent": %ascent%, - "chars": [ - "\ud700\ud701\ud702\ud703\ud704\ud705\ud706\ud707\ud708\ud709\ud70a\ud70b\ud70c\ud70d\ud70e\ud70f", - "\ud710\ud711\ud712\ud713\ud714\ud715\ud716\ud717\ud718\ud719\ud71a\ud71b\ud71c\ud71d\ud71e\ud71f", - "\ud720\ud721\ud722\ud723\ud724\ud725\ud726\ud727\ud728\ud729\ud72a\ud72b\ud72c\ud72d\ud72e\ud72f", - "\ud730\ud731\ud732\ud733\ud734\ud735\ud736\ud737\ud738\ud739\ud73a\ud73b\ud73c\ud73d\ud73e\ud73f", - "\ud740\ud741\ud742\ud743\ud744\ud745\ud746\ud747\ud748\ud749\ud74a\ud74b\ud74c\ud74d\ud74e\ud74f", - "\ud750\ud751\ud752\ud753\ud754\ud755\ud756\ud757\ud758\ud759\ud75a\ud75b\ud75c\ud75d\ud75e\ud75f", - "\ud760\ud761\ud762\ud763\ud764\ud765\ud766\ud767\ud768\ud769\ud76a\ud76b\ud76c\ud76d\ud76e\ud76f", - "\ud770\ud771\ud772\ud773\ud774\ud775\ud776\ud777\ud778\ud779\ud77a\ud77b\ud77c\ud77d\ud77e\ud77f", - "\ud780\ud781\ud782\ud783\ud784\ud785\ud786\ud787\ud788\ud789\ud78a\ud78b\ud78c\ud78d\ud78e\ud78f", - "\ud790\ud791\ud792\ud793\ud794\ud795\ud796\ud797\ud798\ud799\ud79a\ud79b\ud79c\ud79d\ud79e\ud79f", - "\ud7a0\ud7a1\ud7a2\ud7a3\ud7a4\ud7a5\ud7a6\ud7a7\ud7a8\ud7a9\ud7aa\ud7ab\ud7ac\ud7ad\ud7ae\ud7af", - "\ud7b0\ud7b1\ud7b2\ud7b3\ud7b4\ud7b5\ud7b6\ud7b7\ud7b8\ud7b9\ud7ba\ud7bb\ud7bc\ud7bd\ud7be\ud7bf", - "\ud7c0\ud7c1\ud7c2\ud7c3\ud7c4\ud7c5\ud7c6\ud7c7\ud7c8\ud7c9\ud7ca\ud7cb\ud7cc\ud7cd\ud7ce\ud7cf", - "\ud7d0\ud7d1\ud7d2\ud7d3\ud7d4\ud7d5\ud7d6\ud7d7\ud7d8\ud7d9\ud7da\ud7db\ud7dc\ud7dd\ud7de\ud7df", - "\ud7e0\ud7e1\ud7e2\ud7e3\ud7e4\ud7e5\ud7e6\ud7e7\ud7e8\ud7e9\ud7ea\ud7eb\ud7ec\ud7ed\ud7ee\ud7ef", - "\ud7f0\ud7f1\ud7f2\ud7f3\ud7f4\ud7f5\ud7f6\ud7f7\ud7f8\ud7f9\ud7fa\ud7fb\ud7fc\ud7fd\ud7fe\ud7ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_f9.png", - "ascent": %ascent%, - "chars": [ - "\uf900\uf901\uf902\uf903\uf904\uf905\uf906\uf907\uf908\uf909\uf90a\uf90b\uf90c\uf90d\uf90e\uf90f", - "\uf910\uf911\uf912\uf913\uf914\uf915\uf916\uf917\uf918\uf919\uf91a\uf91b\uf91c\uf91d\uf91e\uf91f", - "\uf920\uf921\uf922\uf923\uf924\uf925\uf926\uf927\uf928\uf929\uf92a\uf92b\uf92c\uf92d\uf92e\uf92f", - "\uf930\uf931\uf932\uf933\uf934\uf935\uf936\uf937\uf938\uf939\uf93a\uf93b\uf93c\uf93d\uf93e\uf93f", - "\uf940\uf941\uf942\uf943\uf944\uf945\uf946\uf947\uf948\uf949\uf94a\uf94b\uf94c\uf94d\uf94e\uf94f", - "\uf950\uf951\uf952\uf953\uf954\uf955\uf956\uf957\uf958\uf959\uf95a\uf95b\uf95c\uf95d\uf95e\uf95f", - "\uf960\uf961\uf962\uf963\uf964\uf965\uf966\uf967\uf968\uf969\uf96a\uf96b\uf96c\uf96d\uf96e\uf96f", - "\uf970\uf971\uf972\uf973\uf974\uf975\uf976\uf977\uf978\uf979\uf97a\uf97b\uf97c\uf97d\uf97e\uf97f", - "\uf980\uf981\uf982\uf983\uf984\uf985\uf986\uf987\uf988\uf989\uf98a\uf98b\uf98c\uf98d\uf98e\uf98f", - "\uf990\uf991\uf992\uf993\uf994\uf995\uf996\uf997\uf998\uf999\uf99a\uf99b\uf99c\uf99d\uf99e\uf99f", - "\uf9a0\uf9a1\uf9a2\uf9a3\uf9a4\uf9a5\uf9a6\uf9a7\uf9a8\uf9a9\uf9aa\uf9ab\uf9ac\uf9ad\uf9ae\uf9af", - "\uf9b0\uf9b1\uf9b2\uf9b3\uf9b4\uf9b5\uf9b6\uf9b7\uf9b8\uf9b9\uf9ba\uf9bb\uf9bc\uf9bd\uf9be\uf9bf", - "\uf9c0\uf9c1\uf9c2\uf9c3\uf9c4\uf9c5\uf9c6\uf9c7\uf9c8\uf9c9\uf9ca\uf9cb\uf9cc\uf9cd\uf9ce\uf9cf", - "\uf9d0\uf9d1\uf9d2\uf9d3\uf9d4\uf9d5\uf9d6\uf9d7\uf9d8\uf9d9\uf9da\uf9db\uf9dc\uf9dd\uf9de\uf9df", - "\uf9e0\uf9e1\uf9e2\uf9e3\uf9e4\uf9e5\uf9e6\uf9e7\uf9e8\uf9e9\uf9ea\uf9eb\uf9ec\uf9ed\uf9ee\uf9ef", - "\uf9f0\uf9f1\uf9f2\uf9f3\uf9f4\uf9f5\uf9f6\uf9f7\uf9f8\uf9f9\uf9fa\uf9fb\uf9fc\uf9fd\uf9fe\uf9ff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_fa.png", - "ascent": %ascent%, - "chars": [ - "\ufa00\ufa01\ufa02\ufa03\ufa04\ufa05\ufa06\ufa07\ufa08\ufa09\ufa0a\ufa0b\ufa0c\ufa0d\ufa0e\ufa0f", - "\ufa10\ufa11\ufa12\ufa13\ufa14\ufa15\ufa16\ufa17\ufa18\ufa19\ufa1a\ufa1b\ufa1c\ufa1d\ufa1e\ufa1f", - "\ufa20\ufa21\ufa22\ufa23\ufa24\ufa25\ufa26\ufa27\ufa28\ufa29\ufa2a\ufa2b\ufa2c\ufa2d\ufa2e\ufa2f", - "\ufa30\ufa31\ufa32\ufa33\ufa34\ufa35\ufa36\ufa37\ufa38\ufa39\ufa3a\ufa3b\ufa3c\ufa3d\ufa3e\ufa3f", - "\ufa40\ufa41\ufa42\ufa43\ufa44\ufa45\ufa46\ufa47\ufa48\ufa49\ufa4a\ufa4b\ufa4c\ufa4d\ufa4e\ufa4f", - "\ufa50\ufa51\ufa52\ufa53\ufa54\ufa55\ufa56\ufa57\ufa58\ufa59\ufa5a\ufa5b\ufa5c\ufa5d\ufa5e\ufa5f", - "\ufa60\ufa61\ufa62\ufa63\ufa64\ufa65\ufa66\ufa67\ufa68\ufa69\ufa6a\ufa6b\ufa6c\ufa6d\ufa6e\ufa6f", - "\ufa70\ufa71\ufa72\ufa73\ufa74\ufa75\ufa76\ufa77\ufa78\ufa79\ufa7a\ufa7b\ufa7c\ufa7d\ufa7e\ufa7f", - "\ufa80\ufa81\ufa82\ufa83\ufa84\ufa85\ufa86\ufa87\ufa88\ufa89\ufa8a\ufa8b\ufa8c\ufa8d\ufa8e\ufa8f", - "\ufa90\ufa91\ufa92\ufa93\ufa94\ufa95\ufa96\ufa97\ufa98\ufa99\ufa9a\ufa9b\ufa9c\ufa9d\ufa9e\ufa9f", - "\ufaa0\ufaa1\ufaa2\ufaa3\ufaa4\ufaa5\ufaa6\ufaa7\ufaa8\ufaa9\ufaaa\ufaab\ufaac\ufaad\ufaae\ufaaf", - "\ufab0\ufab1\ufab2\ufab3\ufab4\ufab5\ufab6\ufab7\ufab8\ufab9\ufaba\ufabb\ufabc\ufabd\ufabe\ufabf", - "\ufac0\ufac1\ufac2\ufac3\ufac4\ufac5\ufac6\ufac7\ufac8\ufac9\ufaca\ufacb\ufacc\ufacd\uface\ufacf", - "\ufad0\ufad1\ufad2\ufad3\ufad4\ufad5\ufad6\ufad7\ufad8\ufad9\ufada\ufadb\ufadc\ufadd\ufade\ufadf", - "\ufae0\ufae1\ufae2\ufae3\ufae4\ufae5\ufae6\ufae7\ufae8\ufae9\ufaea\ufaeb\ufaec\ufaed\ufaee\ufaef", - "\ufaf0\ufaf1\ufaf2\ufaf3\ufaf4\ufaf5\ufaf6\ufaf7\ufaf8\ufaf9\ufafa\ufafb\ufafc\ufafd\ufafe\ufaff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_fb.png", - "ascent": %ascent%, - "chars": [ - "\ufb00\ufb01\ufb02\ufb03\ufb04\ufb05\ufb06\ufb07\ufb08\ufb09\ufb0a\ufb0b\ufb0c\ufb0d\ufb0e\ufb0f", - "\ufb10\ufb11\ufb12\ufb13\ufb14\ufb15\ufb16\ufb17\ufb18\ufb19\ufb1a\ufb1b\ufb1c\ufb1d\ufb1e\ufb1f", - "\ufb20\ufb21\ufb22\ufb23\ufb24\ufb25\ufb26\ufb27\ufb28\ufb29\ufb2a\ufb2b\ufb2c\ufb2d\ufb2e\ufb2f", - "\ufb30\ufb31\ufb32\ufb33\ufb34\ufb35\ufb36\ufb37\ufb38\ufb39\ufb3a\ufb3b\ufb3c\ufb3d\ufb3e\ufb3f", - "\ufb40\ufb41\ufb42\ufb43\ufb44\ufb45\ufb46\ufb47\ufb48\ufb49\ufb4a\ufb4b\ufb4c\ufb4d\ufb4e\ufb4f", - "\ufb50\ufb51\ufb52\ufb53\ufb54\ufb55\ufb56\ufb57\ufb58\ufb59\ufb5a\ufb5b\ufb5c\ufb5d\ufb5e\ufb5f", - "\ufb60\ufb61\ufb62\ufb63\ufb64\ufb65\ufb66\ufb67\ufb68\ufb69\ufb6a\ufb6b\ufb6c\ufb6d\ufb6e\ufb6f", - "\ufb70\ufb71\ufb72\ufb73\ufb74\ufb75\ufb76\ufb77\ufb78\ufb79\ufb7a\ufb7b\ufb7c\ufb7d\ufb7e\ufb7f", - "\ufb80\ufb81\ufb82\ufb83\ufb84\ufb85\ufb86\ufb87\ufb88\ufb89\ufb8a\ufb8b\ufb8c\ufb8d\ufb8e\ufb8f", - "\ufb90\ufb91\ufb92\ufb93\ufb94\ufb95\ufb96\ufb97\ufb98\ufb99\ufb9a\ufb9b\ufb9c\ufb9d\ufb9e\ufb9f", - "\ufba0\ufba1\ufba2\ufba3\ufba4\ufba5\ufba6\ufba7\ufba8\ufba9\ufbaa\ufbab\ufbac\ufbad\ufbae\ufbaf", - "\ufbb0\ufbb1\ufbb2\ufbb3\ufbb4\ufbb5\ufbb6\ufbb7\ufbb8\ufbb9\ufbba\ufbbb\ufbbc\ufbbd\ufbbe\ufbbf", - "\ufbc0\ufbc1\ufbc2\ufbc3\ufbc4\ufbc5\ufbc6\ufbc7\ufbc8\ufbc9\ufbca\ufbcb\ufbcc\ufbcd\ufbce\ufbcf", - "\ufbd0\ufbd1\ufbd2\ufbd3\ufbd4\ufbd5\ufbd6\ufbd7\ufbd8\ufbd9\ufbda\ufbdb\ufbdc\ufbdd\ufbde\ufbdf", - "\ufbe0\ufbe1\ufbe2\ufbe3\ufbe4\ufbe5\ufbe6\ufbe7\ufbe8\ufbe9\ufbea\ufbeb\ufbec\ufbed\ufbee\ufbef", - "\ufbf0\ufbf1\ufbf2\ufbf3\ufbf4\ufbf5\ufbf6\ufbf7\ufbf8\ufbf9\ufbfa\ufbfb\ufbfc\ufbfd\ufbfe\ufbff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_fc.png", - "ascent": %ascent%, - "chars": [ - "\ufc00\ufc01\ufc02\ufc03\ufc04\ufc05\ufc06\ufc07\ufc08\ufc09\ufc0a\ufc0b\ufc0c\ufc0d\ufc0e\ufc0f", - "\ufc10\ufc11\ufc12\ufc13\ufc14\ufc15\ufc16\ufc17\ufc18\ufc19\ufc1a\ufc1b\ufc1c\ufc1d\ufc1e\ufc1f", - "\ufc20\ufc21\ufc22\ufc23\ufc24\ufc25\ufc26\ufc27\ufc28\ufc29\ufc2a\ufc2b\ufc2c\ufc2d\ufc2e\ufc2f", - "\ufc30\ufc31\ufc32\ufc33\ufc34\ufc35\ufc36\ufc37\ufc38\ufc39\ufc3a\ufc3b\ufc3c\ufc3d\ufc3e\ufc3f", - "\ufc40\ufc41\ufc42\ufc43\ufc44\ufc45\ufc46\ufc47\ufc48\ufc49\ufc4a\ufc4b\ufc4c\ufc4d\ufc4e\ufc4f", - "\ufc50\ufc51\ufc52\ufc53\ufc54\ufc55\ufc56\ufc57\ufc58\ufc59\ufc5a\ufc5b\ufc5c\ufc5d\ufc5e\ufc5f", - "\ufc60\ufc61\ufc62\ufc63\ufc64\ufc65\ufc66\ufc67\ufc68\ufc69\ufc6a\ufc6b\ufc6c\ufc6d\ufc6e\ufc6f", - "\ufc70\ufc71\ufc72\ufc73\ufc74\ufc75\ufc76\ufc77\ufc78\ufc79\ufc7a\ufc7b\ufc7c\ufc7d\ufc7e\ufc7f", - "\ufc80\ufc81\ufc82\ufc83\ufc84\ufc85\ufc86\ufc87\ufc88\ufc89\ufc8a\ufc8b\ufc8c\ufc8d\ufc8e\ufc8f", - "\ufc90\ufc91\ufc92\ufc93\ufc94\ufc95\ufc96\ufc97\ufc98\ufc99\ufc9a\ufc9b\ufc9c\ufc9d\ufc9e\ufc9f", - "\ufca0\ufca1\ufca2\ufca3\ufca4\ufca5\ufca6\ufca7\ufca8\ufca9\ufcaa\ufcab\ufcac\ufcad\ufcae\ufcaf", - "\ufcb0\ufcb1\ufcb2\ufcb3\ufcb4\ufcb5\ufcb6\ufcb7\ufcb8\ufcb9\ufcba\ufcbb\ufcbc\ufcbd\ufcbe\ufcbf", - "\ufcc0\ufcc1\ufcc2\ufcc3\ufcc4\ufcc5\ufcc6\ufcc7\ufcc8\ufcc9\ufcca\ufccb\ufccc\ufccd\ufcce\ufccf", - "\ufcd0\ufcd1\ufcd2\ufcd3\ufcd4\ufcd5\ufcd6\ufcd7\ufcd8\ufcd9\ufcda\ufcdb\ufcdc\ufcdd\ufcde\ufcdf", - "\ufce0\ufce1\ufce2\ufce3\ufce4\ufce5\ufce6\ufce7\ufce8\ufce9\ufcea\ufceb\ufcec\ufced\ufcee\ufcef", - "\ufcf0\ufcf1\ufcf2\ufcf3\ufcf4\ufcf5\ufcf6\ufcf7\ufcf8\ufcf9\ufcfa\ufcfb\ufcfc\ufcfd\ufcfe\ufcff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_fd.png", - "ascent": %ascent%, - "chars": [ - "\ufd00\ufd01\ufd02\ufd03\ufd04\ufd05\ufd06\ufd07\ufd08\ufd09\ufd0a\ufd0b\ufd0c\ufd0d\ufd0e\ufd0f", - "\ufd10\ufd11\ufd12\ufd13\ufd14\ufd15\ufd16\ufd17\ufd18\ufd19\ufd1a\ufd1b\ufd1c\ufd1d\ufd1e\ufd1f", - "\ufd20\ufd21\ufd22\ufd23\ufd24\ufd25\ufd26\ufd27\ufd28\ufd29\ufd2a\ufd2b\ufd2c\ufd2d\ufd2e\ufd2f", - "\ufd30\ufd31\ufd32\ufd33\ufd34\ufd35\ufd36\ufd37\ufd38\ufd39\ufd3a\ufd3b\ufd3c\ufd3d\ufd3e\ufd3f", - "\ufd40\ufd41\ufd42\ufd43\ufd44\ufd45\ufd46\ufd47\ufd48\ufd49\ufd4a\ufd4b\ufd4c\ufd4d\ufd4e\ufd4f", - "\ufd50\ufd51\ufd52\ufd53\ufd54\ufd55\ufd56\ufd57\ufd58\ufd59\ufd5a\ufd5b\ufd5c\ufd5d\ufd5e\ufd5f", - "\ufd60\ufd61\ufd62\ufd63\ufd64\ufd65\ufd66\ufd67\ufd68\ufd69\ufd6a\ufd6b\ufd6c\ufd6d\ufd6e\ufd6f", - "\ufd70\ufd71\ufd72\ufd73\ufd74\ufd75\ufd76\ufd77\ufd78\ufd79\ufd7a\ufd7b\ufd7c\ufd7d\ufd7e\ufd7f", - "\ufd80\ufd81\ufd82\ufd83\ufd84\ufd85\ufd86\ufd87\ufd88\ufd89\ufd8a\ufd8b\ufd8c\ufd8d\ufd8e\ufd8f", - "\ufd90\ufd91\ufd92\ufd93\ufd94\ufd95\ufd96\ufd97\ufd98\ufd99\ufd9a\ufd9b\ufd9c\ufd9d\ufd9e\ufd9f", - "\ufda0\ufda1\ufda2\ufda3\ufda4\ufda5\ufda6\ufda7\ufda8\ufda9\ufdaa\ufdab\ufdac\ufdad\ufdae\ufdaf", - "\ufdb0\ufdb1\ufdb2\ufdb3\ufdb4\ufdb5\ufdb6\ufdb7\ufdb8\ufdb9\ufdba\ufdbb\ufdbc\ufdbd\ufdbe\ufdbf", - "\ufdc0\ufdc1\ufdc2\ufdc3\ufdc4\ufdc5\ufdc6\ufdc7\ufdc8\ufdc9\ufdca\ufdcb\ufdcc\ufdcd\ufdce\ufdcf", - "\ufdd0\ufdd1\ufdd2\ufdd3\ufdd4\ufdd5\ufdd6\ufdd7\ufdd8\ufdd9\ufdda\ufddb\ufddc\ufddd\ufdde\ufddf", - "\ufde0\ufde1\ufde2\ufde3\ufde4\ufde5\ufde6\ufde7\ufde8\ufde9\ufdea\ufdeb\ufdec\ufded\ufdee\ufdef", - "\ufdf0\ufdf1\ufdf2\ufdf3\ufdf4\ufdf5\ufdf6\ufdf7\ufdf8\ufdf9\ufdfa\ufdfb\ufdfc\ufdfd\ufdfe\ufdff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_fe.png", - "ascent": %ascent%, - "chars": [ - "\ufe00\ufe01\ufe02\ufe03\ufe04\ufe05\ufe06\ufe07\ufe08\ufe09\ufe0a\ufe0b\ufe0c\ufe0d\ufe0e\ufe0f", - "\ufe10\ufe11\ufe12\ufe13\ufe14\ufe15\ufe16\ufe17\ufe18\ufe19\ufe1a\ufe1b\ufe1c\ufe1d\ufe1e\ufe1f", - "\ufe20\ufe21\ufe22\ufe23\ufe24\ufe25\ufe26\ufe27\ufe28\ufe29\ufe2a\ufe2b\ufe2c\ufe2d\ufe2e\ufe2f", - "\ufe30\ufe31\ufe32\ufe33\ufe34\ufe35\ufe36\ufe37\ufe38\ufe39\ufe3a\ufe3b\ufe3c\ufe3d\ufe3e\ufe3f", - "\ufe40\ufe41\ufe42\ufe43\ufe44\ufe45\ufe46\ufe47\ufe48\ufe49\ufe4a\ufe4b\ufe4c\ufe4d\ufe4e\ufe4f", - "\ufe50\ufe51\ufe52\ufe53\ufe54\ufe55\ufe56\ufe57\ufe58\ufe59\ufe5a\ufe5b\ufe5c\ufe5d\ufe5e\ufe5f", - "\ufe60\ufe61\ufe62\ufe63\ufe64\ufe65\ufe66\ufe67\ufe68\ufe69\ufe6a\ufe6b\ufe6c\ufe6d\ufe6e\ufe6f", - "\ufe70\ufe71\ufe72\ufe73\ufe74\ufe75\ufe76\ufe77\ufe78\ufe79\ufe7a\ufe7b\ufe7c\ufe7d\ufe7e\ufe7f", - "\ufe80\ufe81\ufe82\ufe83\ufe84\ufe85\ufe86\ufe87\ufe88\ufe89\ufe8a\ufe8b\ufe8c\ufe8d\ufe8e\ufe8f", - "\ufe90\ufe91\ufe92\ufe93\ufe94\ufe95\ufe96\ufe97\ufe98\ufe99\ufe9a\ufe9b\ufe9c\ufe9d\ufe9e\ufe9f", - "\ufea0\ufea1\ufea2\ufea3\ufea4\ufea5\ufea6\ufea7\ufea8\ufea9\ufeaa\ufeab\ufeac\ufead\ufeae\ufeaf", - "\ufeb0\ufeb1\ufeb2\ufeb3\ufeb4\ufeb5\ufeb6\ufeb7\ufeb8\ufeb9\ufeba\ufebb\ufebc\ufebd\ufebe\ufebf", - "\ufec0\ufec1\ufec2\ufec3\ufec4\ufec5\ufec6\ufec7\ufec8\ufec9\ufeca\ufecb\ufecc\ufecd\ufece\ufecf", - "\ufed0\ufed1\ufed2\ufed3\ufed4\ufed5\ufed6\ufed7\ufed8\ufed9\ufeda\ufedb\ufedc\ufedd\ufede\ufedf", - "\ufee0\ufee1\ufee2\ufee3\ufee4\ufee5\ufee6\ufee7\ufee8\ufee9\ufeea\ufeeb\ufeec\ufeed\ufeee\ufeef", - "\ufef0\ufef1\ufef2\ufef3\ufef4\ufef5\ufef6\ufef7\ufef8\ufef9\ufefa\ufefb\ufefc\ufefd\ufefe\ufeff" - ] - }, - { - "type": "bitmap", - "file": "minecraft:font/unicode_page_ff.png", - "ascent": %ascent%, - "chars": [ - "\uff00\uff01\uff02\uff03\uff04\uff05\uff06\uff07\uff08\uff09\uff0a\uff0b\uff0c\uff0d\uff0e\uff0f", - "\uff10\uff11\uff12\uff13\uff14\uff15\uff16\uff17\uff18\uff19\uff1a\uff1b\uff1c\uff1d\uff1e\uff1f", - "\uff20\uff21\uff22\uff23\uff24\uff25\uff26\uff27\uff28\uff29\uff2a\uff2b\uff2c\uff2d\uff2e\uff2f", - "\uff30\uff31\uff32\uff33\uff34\uff35\uff36\uff37\uff38\uff39\uff3a\uff3b\uff3c\uff3d\uff3e\uff3f", - "\uff40\uff41\uff42\uff43\uff44\uff45\uff46\uff47\uff48\uff49\uff4a\uff4b\uff4c\uff4d\uff4e\uff4f", - "\uff50\uff51\uff52\uff53\uff54\uff55\uff56\uff57\uff58\uff59\uff5a\uff5b\uff5c\uff5d\uff5e\uff5f", - "\uff60\uff61\uff62\uff63\uff64\uff65\uff66\uff67\uff68\uff69\uff6a\uff6b\uff6c\uff6d\uff6e\uff6f", - "\uff70\uff71\uff72\uff73\uff74\uff75\uff76\uff77\uff78\uff79\uff7a\uff7b\uff7c\uff7d\uff7e\uff7f", - "\uff80\uff81\uff82\uff83\uff84\uff85\uff86\uff87\uff88\uff89\uff8a\uff8b\uff8c\uff8d\uff8e\uff8f", - "\uff90\uff91\uff92\uff93\uff94\uff95\uff96\uff97\uff98\uff99\uff9a\uff9b\uff9c\uff9d\uff9e\uff9f", - "\uffa0\uffa1\uffa2\uffa3\uffa4\uffa5\uffa6\uffa7\uffa8\uffa9\uffaa\uffab\uffac\uffad\uffae\uffaf", - "\uffb0\uffb1\uffb2\uffb3\uffb4\uffb5\uffb6\uffb7\uffb8\uffb9\uffba\uffbb\uffbc\uffbd\uffbe\uffbf", - "\uffc0\uffc1\uffc2\uffc3\uffc4\uffc5\uffc6\uffc7\uffc8\uffc9\uffca\uffcb\uffcc\uffcd\uffce\uffcf", - "\uffd0\uffd1\uffd2\uffd3\uffd4\uffd5\uffd6\uffd7\uffd8\uffd9\uffda\uffdb\uffdc\uffdd\uffde\uffdf", - "\uffe0\uffe1\uffe2\uffe3\uffe4\uffe5\uffe6\uffe7\uffe8\uffe9\uffea\uffeb\uffec\uffed\uffee\uffef", - "\ufff0\ufff1\ufff2\ufff3\ufff4\ufff5\ufff6\ufff7\ufff8\ufff9\ufffa\ufffb\ufffc\ufffd\ufffe\uffff" - ] - } - ] -} \ No newline at end of file diff --git a/src/main/resources/unicodes/unicode_page_00.png b/src/main/resources/unicodes/unicode_page_00.png deleted file mode 100644 index 0ddd184..0000000 Binary files a/src/main/resources/unicodes/unicode_page_00.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_01.png b/src/main/resources/unicodes/unicode_page_01.png deleted file mode 100644 index a1d7b01..0000000 Binary files a/src/main/resources/unicodes/unicode_page_01.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_02.png b/src/main/resources/unicodes/unicode_page_02.png deleted file mode 100644 index b437ceb..0000000 Binary files a/src/main/resources/unicodes/unicode_page_02.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_03.png b/src/main/resources/unicodes/unicode_page_03.png deleted file mode 100644 index 59fbbab..0000000 Binary files a/src/main/resources/unicodes/unicode_page_03.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_04.png b/src/main/resources/unicodes/unicode_page_04.png deleted file mode 100644 index 75a4523..0000000 Binary files a/src/main/resources/unicodes/unicode_page_04.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_05.png b/src/main/resources/unicodes/unicode_page_05.png deleted file mode 100644 index 32b1c88..0000000 Binary files a/src/main/resources/unicodes/unicode_page_05.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_06.png b/src/main/resources/unicodes/unicode_page_06.png deleted file mode 100644 index 825e877..0000000 Binary files a/src/main/resources/unicodes/unicode_page_06.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_07.png b/src/main/resources/unicodes/unicode_page_07.png deleted file mode 100644 index 865b092..0000000 Binary files a/src/main/resources/unicodes/unicode_page_07.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_09.png b/src/main/resources/unicodes/unicode_page_09.png deleted file mode 100644 index d6407e3..0000000 Binary files a/src/main/resources/unicodes/unicode_page_09.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_0a.png b/src/main/resources/unicodes/unicode_page_0a.png deleted file mode 100644 index a858b78..0000000 Binary files a/src/main/resources/unicodes/unicode_page_0a.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_0b.png b/src/main/resources/unicodes/unicode_page_0b.png deleted file mode 100644 index 3dd0aca..0000000 Binary files a/src/main/resources/unicodes/unicode_page_0b.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_0c.png b/src/main/resources/unicodes/unicode_page_0c.png deleted file mode 100644 index 2f75930..0000000 Binary files a/src/main/resources/unicodes/unicode_page_0c.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_0d.png b/src/main/resources/unicodes/unicode_page_0d.png deleted file mode 100644 index 8eff12c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_0d.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_0e.png b/src/main/resources/unicodes/unicode_page_0e.png deleted file mode 100644 index 03b8f53..0000000 Binary files a/src/main/resources/unicodes/unicode_page_0e.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_0f.png b/src/main/resources/unicodes/unicode_page_0f.png deleted file mode 100644 index b89ee65..0000000 Binary files a/src/main/resources/unicodes/unicode_page_0f.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_10.png b/src/main/resources/unicodes/unicode_page_10.png deleted file mode 100644 index b3eb746..0000000 Binary files a/src/main/resources/unicodes/unicode_page_10.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_11.png b/src/main/resources/unicodes/unicode_page_11.png deleted file mode 100644 index 85e714c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_11.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_12.png b/src/main/resources/unicodes/unicode_page_12.png deleted file mode 100644 index a6580cd..0000000 Binary files a/src/main/resources/unicodes/unicode_page_12.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_13.png b/src/main/resources/unicodes/unicode_page_13.png deleted file mode 100644 index a847ba7..0000000 Binary files a/src/main/resources/unicodes/unicode_page_13.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_14.png b/src/main/resources/unicodes/unicode_page_14.png deleted file mode 100644 index bec8830..0000000 Binary files a/src/main/resources/unicodes/unicode_page_14.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_15.png b/src/main/resources/unicodes/unicode_page_15.png deleted file mode 100644 index fb372b2..0000000 Binary files a/src/main/resources/unicodes/unicode_page_15.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_16.png b/src/main/resources/unicodes/unicode_page_16.png deleted file mode 100644 index ee6efed..0000000 Binary files a/src/main/resources/unicodes/unicode_page_16.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_17.png b/src/main/resources/unicodes/unicode_page_17.png deleted file mode 100644 index 7c06ef1..0000000 Binary files a/src/main/resources/unicodes/unicode_page_17.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_18.png b/src/main/resources/unicodes/unicode_page_18.png deleted file mode 100644 index 1032942..0000000 Binary files a/src/main/resources/unicodes/unicode_page_18.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_19.png b/src/main/resources/unicodes/unicode_page_19.png deleted file mode 100644 index 28dce5d..0000000 Binary files a/src/main/resources/unicodes/unicode_page_19.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_1a.png b/src/main/resources/unicodes/unicode_page_1a.png deleted file mode 100644 index edf8116..0000000 Binary files a/src/main/resources/unicodes/unicode_page_1a.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_1b.png b/src/main/resources/unicodes/unicode_page_1b.png deleted file mode 100644 index 3ddb4c6..0000000 Binary files a/src/main/resources/unicodes/unicode_page_1b.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_1c.png b/src/main/resources/unicodes/unicode_page_1c.png deleted file mode 100644 index 8884fd2..0000000 Binary files a/src/main/resources/unicodes/unicode_page_1c.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_1d.png b/src/main/resources/unicodes/unicode_page_1d.png deleted file mode 100644 index ca0b077..0000000 Binary files a/src/main/resources/unicodes/unicode_page_1d.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_1e.png b/src/main/resources/unicodes/unicode_page_1e.png deleted file mode 100644 index 9206f32..0000000 Binary files a/src/main/resources/unicodes/unicode_page_1e.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_1f.png b/src/main/resources/unicodes/unicode_page_1f.png deleted file mode 100644 index 7e28632..0000000 Binary files a/src/main/resources/unicodes/unicode_page_1f.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_20.png b/src/main/resources/unicodes/unicode_page_20.png deleted file mode 100644 index 986dbbf..0000000 Binary files a/src/main/resources/unicodes/unicode_page_20.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_21.png b/src/main/resources/unicodes/unicode_page_21.png deleted file mode 100644 index b2c5ed4..0000000 Binary files a/src/main/resources/unicodes/unicode_page_21.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_22.png b/src/main/resources/unicodes/unicode_page_22.png deleted file mode 100644 index be83259..0000000 Binary files a/src/main/resources/unicodes/unicode_page_22.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_23.png b/src/main/resources/unicodes/unicode_page_23.png deleted file mode 100644 index 09a9de8..0000000 Binary files a/src/main/resources/unicodes/unicode_page_23.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_24.png b/src/main/resources/unicodes/unicode_page_24.png deleted file mode 100644 index 17da640..0000000 Binary files a/src/main/resources/unicodes/unicode_page_24.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_25.png b/src/main/resources/unicodes/unicode_page_25.png deleted file mode 100644 index 6ac17b6..0000000 Binary files a/src/main/resources/unicodes/unicode_page_25.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_26.png b/src/main/resources/unicodes/unicode_page_26.png deleted file mode 100644 index 615dfa6..0000000 Binary files a/src/main/resources/unicodes/unicode_page_26.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_27.png b/src/main/resources/unicodes/unicode_page_27.png deleted file mode 100644 index bbe69f7..0000000 Binary files a/src/main/resources/unicodes/unicode_page_27.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_28.png b/src/main/resources/unicodes/unicode_page_28.png deleted file mode 100644 index 7d818ff..0000000 Binary files a/src/main/resources/unicodes/unicode_page_28.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_29.png b/src/main/resources/unicodes/unicode_page_29.png deleted file mode 100644 index 0947b65..0000000 Binary files a/src/main/resources/unicodes/unicode_page_29.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_2a.png b/src/main/resources/unicodes/unicode_page_2a.png deleted file mode 100644 index 75319b5..0000000 Binary files a/src/main/resources/unicodes/unicode_page_2a.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_2b.png b/src/main/resources/unicodes/unicode_page_2b.png deleted file mode 100644 index 93390c2..0000000 Binary files a/src/main/resources/unicodes/unicode_page_2b.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_2c.png b/src/main/resources/unicodes/unicode_page_2c.png deleted file mode 100644 index daea036..0000000 Binary files a/src/main/resources/unicodes/unicode_page_2c.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_2d.png b/src/main/resources/unicodes/unicode_page_2d.png deleted file mode 100644 index 5753d02..0000000 Binary files a/src/main/resources/unicodes/unicode_page_2d.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_2e.png b/src/main/resources/unicodes/unicode_page_2e.png deleted file mode 100644 index 1c0a222..0000000 Binary files a/src/main/resources/unicodes/unicode_page_2e.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_2f.png b/src/main/resources/unicodes/unicode_page_2f.png deleted file mode 100644 index 054bc40..0000000 Binary files a/src/main/resources/unicodes/unicode_page_2f.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_30.png b/src/main/resources/unicodes/unicode_page_30.png deleted file mode 100644 index ab73978..0000000 Binary files a/src/main/resources/unicodes/unicode_page_30.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_31.png b/src/main/resources/unicodes/unicode_page_31.png deleted file mode 100644 index ae293b9..0000000 Binary files a/src/main/resources/unicodes/unicode_page_31.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_32.png b/src/main/resources/unicodes/unicode_page_32.png deleted file mode 100644 index 229ab5c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_32.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_33.png b/src/main/resources/unicodes/unicode_page_33.png deleted file mode 100644 index b2ecd11..0000000 Binary files a/src/main/resources/unicodes/unicode_page_33.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_34.png b/src/main/resources/unicodes/unicode_page_34.png deleted file mode 100644 index 6309118..0000000 Binary files a/src/main/resources/unicodes/unicode_page_34.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_35.png b/src/main/resources/unicodes/unicode_page_35.png deleted file mode 100644 index a740bdc..0000000 Binary files a/src/main/resources/unicodes/unicode_page_35.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_36.png b/src/main/resources/unicodes/unicode_page_36.png deleted file mode 100644 index 5d449c3..0000000 Binary files a/src/main/resources/unicodes/unicode_page_36.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_37.png b/src/main/resources/unicodes/unicode_page_37.png deleted file mode 100644 index c4707c6..0000000 Binary files a/src/main/resources/unicodes/unicode_page_37.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_38.png b/src/main/resources/unicodes/unicode_page_38.png deleted file mode 100644 index a2d05c8..0000000 Binary files a/src/main/resources/unicodes/unicode_page_38.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_39.png b/src/main/resources/unicodes/unicode_page_39.png deleted file mode 100644 index 7e02109..0000000 Binary files a/src/main/resources/unicodes/unicode_page_39.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_3a.png b/src/main/resources/unicodes/unicode_page_3a.png deleted file mode 100644 index 548fa07..0000000 Binary files a/src/main/resources/unicodes/unicode_page_3a.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_3b.png b/src/main/resources/unicodes/unicode_page_3b.png deleted file mode 100644 index 9da10b1..0000000 Binary files a/src/main/resources/unicodes/unicode_page_3b.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_3c.png b/src/main/resources/unicodes/unicode_page_3c.png deleted file mode 100644 index c4bbe7d..0000000 Binary files a/src/main/resources/unicodes/unicode_page_3c.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_3d.png b/src/main/resources/unicodes/unicode_page_3d.png deleted file mode 100644 index 2df1f96..0000000 Binary files a/src/main/resources/unicodes/unicode_page_3d.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_3e.png b/src/main/resources/unicodes/unicode_page_3e.png deleted file mode 100644 index e0c7178..0000000 Binary files a/src/main/resources/unicodes/unicode_page_3e.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_3f.png b/src/main/resources/unicodes/unicode_page_3f.png deleted file mode 100644 index 81f03b7..0000000 Binary files a/src/main/resources/unicodes/unicode_page_3f.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_40.png b/src/main/resources/unicodes/unicode_page_40.png deleted file mode 100644 index c2910f4..0000000 Binary files a/src/main/resources/unicodes/unicode_page_40.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_41.png b/src/main/resources/unicodes/unicode_page_41.png deleted file mode 100644 index 1e6edfc..0000000 Binary files a/src/main/resources/unicodes/unicode_page_41.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_42.png b/src/main/resources/unicodes/unicode_page_42.png deleted file mode 100644 index feff714..0000000 Binary files a/src/main/resources/unicodes/unicode_page_42.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_43.png b/src/main/resources/unicodes/unicode_page_43.png deleted file mode 100644 index eda2899..0000000 Binary files a/src/main/resources/unicodes/unicode_page_43.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_44.png b/src/main/resources/unicodes/unicode_page_44.png deleted file mode 100644 index 1373dc8..0000000 Binary files a/src/main/resources/unicodes/unicode_page_44.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_45.png b/src/main/resources/unicodes/unicode_page_45.png deleted file mode 100644 index a8d4db3..0000000 Binary files a/src/main/resources/unicodes/unicode_page_45.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_46.png b/src/main/resources/unicodes/unicode_page_46.png deleted file mode 100644 index 0da9635..0000000 Binary files a/src/main/resources/unicodes/unicode_page_46.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_47.png b/src/main/resources/unicodes/unicode_page_47.png deleted file mode 100644 index 68eedf4..0000000 Binary files a/src/main/resources/unicodes/unicode_page_47.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_48.png b/src/main/resources/unicodes/unicode_page_48.png deleted file mode 100644 index 30e2e83..0000000 Binary files a/src/main/resources/unicodes/unicode_page_48.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_49.png b/src/main/resources/unicodes/unicode_page_49.png deleted file mode 100644 index 046646a..0000000 Binary files a/src/main/resources/unicodes/unicode_page_49.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_4a.png b/src/main/resources/unicodes/unicode_page_4a.png deleted file mode 100644 index f1ffb0e..0000000 Binary files a/src/main/resources/unicodes/unicode_page_4a.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_4b.png b/src/main/resources/unicodes/unicode_page_4b.png deleted file mode 100644 index eaab3e4..0000000 Binary files a/src/main/resources/unicodes/unicode_page_4b.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_4c.png b/src/main/resources/unicodes/unicode_page_4c.png deleted file mode 100644 index 4e4b7a2..0000000 Binary files a/src/main/resources/unicodes/unicode_page_4c.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_4d.png b/src/main/resources/unicodes/unicode_page_4d.png deleted file mode 100644 index d460d70..0000000 Binary files a/src/main/resources/unicodes/unicode_page_4d.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_4e.png b/src/main/resources/unicodes/unicode_page_4e.png deleted file mode 100644 index be8546c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_4e.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_4f.png b/src/main/resources/unicodes/unicode_page_4f.png deleted file mode 100644 index cdf58f3..0000000 Binary files a/src/main/resources/unicodes/unicode_page_4f.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_50.png b/src/main/resources/unicodes/unicode_page_50.png deleted file mode 100644 index a31aff5..0000000 Binary files a/src/main/resources/unicodes/unicode_page_50.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_51.png b/src/main/resources/unicodes/unicode_page_51.png deleted file mode 100644 index 72cb5ab..0000000 Binary files a/src/main/resources/unicodes/unicode_page_51.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_52.png b/src/main/resources/unicodes/unicode_page_52.png deleted file mode 100644 index 81bbd78..0000000 Binary files a/src/main/resources/unicodes/unicode_page_52.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_53.png b/src/main/resources/unicodes/unicode_page_53.png deleted file mode 100644 index 238c830..0000000 Binary files a/src/main/resources/unicodes/unicode_page_53.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_54.png b/src/main/resources/unicodes/unicode_page_54.png deleted file mode 100644 index 3f616e3..0000000 Binary files a/src/main/resources/unicodes/unicode_page_54.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_55.png b/src/main/resources/unicodes/unicode_page_55.png deleted file mode 100644 index 698ce0d..0000000 Binary files a/src/main/resources/unicodes/unicode_page_55.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_56.png b/src/main/resources/unicodes/unicode_page_56.png deleted file mode 100644 index f0dad79..0000000 Binary files a/src/main/resources/unicodes/unicode_page_56.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_57.png b/src/main/resources/unicodes/unicode_page_57.png deleted file mode 100644 index 715e1ef..0000000 Binary files a/src/main/resources/unicodes/unicode_page_57.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_58.png b/src/main/resources/unicodes/unicode_page_58.png deleted file mode 100644 index beabcc1..0000000 Binary files a/src/main/resources/unicodes/unicode_page_58.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_59.png b/src/main/resources/unicodes/unicode_page_59.png deleted file mode 100644 index f623511..0000000 Binary files a/src/main/resources/unicodes/unicode_page_59.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_5a.png b/src/main/resources/unicodes/unicode_page_5a.png deleted file mode 100644 index 1e83420..0000000 Binary files a/src/main/resources/unicodes/unicode_page_5a.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_5b.png b/src/main/resources/unicodes/unicode_page_5b.png deleted file mode 100644 index 852f074..0000000 Binary files a/src/main/resources/unicodes/unicode_page_5b.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_5c.png b/src/main/resources/unicodes/unicode_page_5c.png deleted file mode 100644 index 412d289..0000000 Binary files a/src/main/resources/unicodes/unicode_page_5c.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_5d.png b/src/main/resources/unicodes/unicode_page_5d.png deleted file mode 100644 index 2c4714c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_5d.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_5e.png b/src/main/resources/unicodes/unicode_page_5e.png deleted file mode 100644 index 4eb5267..0000000 Binary files a/src/main/resources/unicodes/unicode_page_5e.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_5f.png b/src/main/resources/unicodes/unicode_page_5f.png deleted file mode 100644 index 151f8d4..0000000 Binary files a/src/main/resources/unicodes/unicode_page_5f.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_60.png b/src/main/resources/unicodes/unicode_page_60.png deleted file mode 100644 index 333efc0..0000000 Binary files a/src/main/resources/unicodes/unicode_page_60.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_61.png b/src/main/resources/unicodes/unicode_page_61.png deleted file mode 100644 index 9c18392..0000000 Binary files a/src/main/resources/unicodes/unicode_page_61.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_62.png b/src/main/resources/unicodes/unicode_page_62.png deleted file mode 100644 index 9555a9d..0000000 Binary files a/src/main/resources/unicodes/unicode_page_62.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_63.png b/src/main/resources/unicodes/unicode_page_63.png deleted file mode 100644 index 4ab408a..0000000 Binary files a/src/main/resources/unicodes/unicode_page_63.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_64.png b/src/main/resources/unicodes/unicode_page_64.png deleted file mode 100644 index ec8e90b..0000000 Binary files a/src/main/resources/unicodes/unicode_page_64.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_65.png b/src/main/resources/unicodes/unicode_page_65.png deleted file mode 100644 index d9da268..0000000 Binary files a/src/main/resources/unicodes/unicode_page_65.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_66.png b/src/main/resources/unicodes/unicode_page_66.png deleted file mode 100644 index bf39744..0000000 Binary files a/src/main/resources/unicodes/unicode_page_66.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_67.png b/src/main/resources/unicodes/unicode_page_67.png deleted file mode 100644 index c5d51c1..0000000 Binary files a/src/main/resources/unicodes/unicode_page_67.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_68.png b/src/main/resources/unicodes/unicode_page_68.png deleted file mode 100644 index 321e8ee..0000000 Binary files a/src/main/resources/unicodes/unicode_page_68.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_69.png b/src/main/resources/unicodes/unicode_page_69.png deleted file mode 100644 index 1a969ac..0000000 Binary files a/src/main/resources/unicodes/unicode_page_69.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_6a.png b/src/main/resources/unicodes/unicode_page_6a.png deleted file mode 100644 index b67475b..0000000 Binary files a/src/main/resources/unicodes/unicode_page_6a.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_6b.png b/src/main/resources/unicodes/unicode_page_6b.png deleted file mode 100644 index 82e083a..0000000 Binary files a/src/main/resources/unicodes/unicode_page_6b.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_6c.png b/src/main/resources/unicodes/unicode_page_6c.png deleted file mode 100644 index d2b50af..0000000 Binary files a/src/main/resources/unicodes/unicode_page_6c.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_6d.png b/src/main/resources/unicodes/unicode_page_6d.png deleted file mode 100644 index 2e1dfd1..0000000 Binary files a/src/main/resources/unicodes/unicode_page_6d.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_6e.png b/src/main/resources/unicodes/unicode_page_6e.png deleted file mode 100644 index 495a4c1..0000000 Binary files a/src/main/resources/unicodes/unicode_page_6e.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_6f.png b/src/main/resources/unicodes/unicode_page_6f.png deleted file mode 100644 index 80e4993..0000000 Binary files a/src/main/resources/unicodes/unicode_page_6f.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_70.png b/src/main/resources/unicodes/unicode_page_70.png deleted file mode 100644 index 15694bb..0000000 Binary files a/src/main/resources/unicodes/unicode_page_70.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_71.png b/src/main/resources/unicodes/unicode_page_71.png deleted file mode 100644 index ff04b6a..0000000 Binary files a/src/main/resources/unicodes/unicode_page_71.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_72.png b/src/main/resources/unicodes/unicode_page_72.png deleted file mode 100644 index 67c6461..0000000 Binary files a/src/main/resources/unicodes/unicode_page_72.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_73.png b/src/main/resources/unicodes/unicode_page_73.png deleted file mode 100644 index 4bafafa..0000000 Binary files a/src/main/resources/unicodes/unicode_page_73.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_74.png b/src/main/resources/unicodes/unicode_page_74.png deleted file mode 100644 index ec0f591..0000000 Binary files a/src/main/resources/unicodes/unicode_page_74.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_75.png b/src/main/resources/unicodes/unicode_page_75.png deleted file mode 100644 index 00e2c71..0000000 Binary files a/src/main/resources/unicodes/unicode_page_75.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_76.png b/src/main/resources/unicodes/unicode_page_76.png deleted file mode 100644 index e76fbee..0000000 Binary files a/src/main/resources/unicodes/unicode_page_76.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_77.png b/src/main/resources/unicodes/unicode_page_77.png deleted file mode 100644 index 08a22ff..0000000 Binary files a/src/main/resources/unicodes/unicode_page_77.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_78.png b/src/main/resources/unicodes/unicode_page_78.png deleted file mode 100644 index d6e3418..0000000 Binary files a/src/main/resources/unicodes/unicode_page_78.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_79.png b/src/main/resources/unicodes/unicode_page_79.png deleted file mode 100644 index b11efa8..0000000 Binary files a/src/main/resources/unicodes/unicode_page_79.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_7a.png b/src/main/resources/unicodes/unicode_page_7a.png deleted file mode 100644 index f3ff655..0000000 Binary files a/src/main/resources/unicodes/unicode_page_7a.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_7b.png b/src/main/resources/unicodes/unicode_page_7b.png deleted file mode 100644 index 6b3038c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_7b.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_7c.png b/src/main/resources/unicodes/unicode_page_7c.png deleted file mode 100644 index 3ffe7a7..0000000 Binary files a/src/main/resources/unicodes/unicode_page_7c.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_7d.png b/src/main/resources/unicodes/unicode_page_7d.png deleted file mode 100644 index 700ab75..0000000 Binary files a/src/main/resources/unicodes/unicode_page_7d.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_7e.png b/src/main/resources/unicodes/unicode_page_7e.png deleted file mode 100644 index 34c8c54..0000000 Binary files a/src/main/resources/unicodes/unicode_page_7e.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_7f.png b/src/main/resources/unicodes/unicode_page_7f.png deleted file mode 100644 index ad3b070..0000000 Binary files a/src/main/resources/unicodes/unicode_page_7f.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_80.png b/src/main/resources/unicodes/unicode_page_80.png deleted file mode 100644 index f9a954c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_80.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_81.png b/src/main/resources/unicodes/unicode_page_81.png deleted file mode 100644 index 97d5667..0000000 Binary files a/src/main/resources/unicodes/unicode_page_81.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_82.png b/src/main/resources/unicodes/unicode_page_82.png deleted file mode 100644 index 5acf8aa..0000000 Binary files a/src/main/resources/unicodes/unicode_page_82.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_83.png b/src/main/resources/unicodes/unicode_page_83.png deleted file mode 100644 index b1a29e4..0000000 Binary files a/src/main/resources/unicodes/unicode_page_83.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_84.png b/src/main/resources/unicodes/unicode_page_84.png deleted file mode 100644 index 9dd7e51..0000000 Binary files a/src/main/resources/unicodes/unicode_page_84.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_85.png b/src/main/resources/unicodes/unicode_page_85.png deleted file mode 100644 index 6f72b8b..0000000 Binary files a/src/main/resources/unicodes/unicode_page_85.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_86.png b/src/main/resources/unicodes/unicode_page_86.png deleted file mode 100644 index 91d82f3..0000000 Binary files a/src/main/resources/unicodes/unicode_page_86.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_87.png b/src/main/resources/unicodes/unicode_page_87.png deleted file mode 100644 index 002bbc6..0000000 Binary files a/src/main/resources/unicodes/unicode_page_87.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_88.png b/src/main/resources/unicodes/unicode_page_88.png deleted file mode 100644 index 2388727..0000000 Binary files a/src/main/resources/unicodes/unicode_page_88.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_89.png b/src/main/resources/unicodes/unicode_page_89.png deleted file mode 100644 index e35974a..0000000 Binary files a/src/main/resources/unicodes/unicode_page_89.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_8a.png b/src/main/resources/unicodes/unicode_page_8a.png deleted file mode 100644 index 008a185..0000000 Binary files a/src/main/resources/unicodes/unicode_page_8a.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_8b.png b/src/main/resources/unicodes/unicode_page_8b.png deleted file mode 100644 index 360f9c6..0000000 Binary files a/src/main/resources/unicodes/unicode_page_8b.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_8c.png b/src/main/resources/unicodes/unicode_page_8c.png deleted file mode 100644 index 12b78b5..0000000 Binary files a/src/main/resources/unicodes/unicode_page_8c.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_8d.png b/src/main/resources/unicodes/unicode_page_8d.png deleted file mode 100644 index 10de16c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_8d.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_8e.png b/src/main/resources/unicodes/unicode_page_8e.png deleted file mode 100644 index 8e4f17c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_8e.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_8f.png b/src/main/resources/unicodes/unicode_page_8f.png deleted file mode 100644 index 6e4c941..0000000 Binary files a/src/main/resources/unicodes/unicode_page_8f.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_90.png b/src/main/resources/unicodes/unicode_page_90.png deleted file mode 100644 index a95436c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_90.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_91.png b/src/main/resources/unicodes/unicode_page_91.png deleted file mode 100644 index d7bad07..0000000 Binary files a/src/main/resources/unicodes/unicode_page_91.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_92.png b/src/main/resources/unicodes/unicode_page_92.png deleted file mode 100644 index 3676d3f..0000000 Binary files a/src/main/resources/unicodes/unicode_page_92.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_93.png b/src/main/resources/unicodes/unicode_page_93.png deleted file mode 100644 index d59c7d2..0000000 Binary files a/src/main/resources/unicodes/unicode_page_93.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_94.png b/src/main/resources/unicodes/unicode_page_94.png deleted file mode 100644 index 8f38140..0000000 Binary files a/src/main/resources/unicodes/unicode_page_94.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_95.png b/src/main/resources/unicodes/unicode_page_95.png deleted file mode 100644 index 13cf8c7..0000000 Binary files a/src/main/resources/unicodes/unicode_page_95.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_96.png b/src/main/resources/unicodes/unicode_page_96.png deleted file mode 100644 index 6a7838c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_96.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_97.png b/src/main/resources/unicodes/unicode_page_97.png deleted file mode 100644 index 081c1af..0000000 Binary files a/src/main/resources/unicodes/unicode_page_97.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_98.png b/src/main/resources/unicodes/unicode_page_98.png deleted file mode 100644 index 2bdcbf9..0000000 Binary files a/src/main/resources/unicodes/unicode_page_98.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_99.png b/src/main/resources/unicodes/unicode_page_99.png deleted file mode 100644 index 5789f19..0000000 Binary files a/src/main/resources/unicodes/unicode_page_99.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_9a.png b/src/main/resources/unicodes/unicode_page_9a.png deleted file mode 100644 index 8aacdc6..0000000 Binary files a/src/main/resources/unicodes/unicode_page_9a.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_9b.png b/src/main/resources/unicodes/unicode_page_9b.png deleted file mode 100644 index 34a54bd..0000000 Binary files a/src/main/resources/unicodes/unicode_page_9b.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_9c.png b/src/main/resources/unicodes/unicode_page_9c.png deleted file mode 100644 index 57e4a71..0000000 Binary files a/src/main/resources/unicodes/unicode_page_9c.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_9d.png b/src/main/resources/unicodes/unicode_page_9d.png deleted file mode 100644 index 0a079be..0000000 Binary files a/src/main/resources/unicodes/unicode_page_9d.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_9e.png b/src/main/resources/unicodes/unicode_page_9e.png deleted file mode 100644 index 80ce2b3..0000000 Binary files a/src/main/resources/unicodes/unicode_page_9e.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_9f.png b/src/main/resources/unicodes/unicode_page_9f.png deleted file mode 100644 index a6100ae..0000000 Binary files a/src/main/resources/unicodes/unicode_page_9f.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_a0.png b/src/main/resources/unicodes/unicode_page_a0.png deleted file mode 100644 index 70c2b20..0000000 Binary files a/src/main/resources/unicodes/unicode_page_a0.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_a1.png b/src/main/resources/unicodes/unicode_page_a1.png deleted file mode 100644 index b650759..0000000 Binary files a/src/main/resources/unicodes/unicode_page_a1.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_a2.png b/src/main/resources/unicodes/unicode_page_a2.png deleted file mode 100644 index 44b5a8d..0000000 Binary files a/src/main/resources/unicodes/unicode_page_a2.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_a3.png b/src/main/resources/unicodes/unicode_page_a3.png deleted file mode 100644 index a6987c7..0000000 Binary files a/src/main/resources/unicodes/unicode_page_a3.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_a4.png b/src/main/resources/unicodes/unicode_page_a4.png deleted file mode 100644 index 4dce2f2..0000000 Binary files a/src/main/resources/unicodes/unicode_page_a4.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_a5.png b/src/main/resources/unicodes/unicode_page_a5.png deleted file mode 100644 index cec113c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_a5.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_a6.png b/src/main/resources/unicodes/unicode_page_a6.png deleted file mode 100644 index d28ad13..0000000 Binary files a/src/main/resources/unicodes/unicode_page_a6.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_a7.png b/src/main/resources/unicodes/unicode_page_a7.png deleted file mode 100644 index b41be02..0000000 Binary files a/src/main/resources/unicodes/unicode_page_a7.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_a8.png b/src/main/resources/unicodes/unicode_page_a8.png deleted file mode 100644 index 8a21e03..0000000 Binary files a/src/main/resources/unicodes/unicode_page_a8.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_a9.png b/src/main/resources/unicodes/unicode_page_a9.png deleted file mode 100644 index 0970c14..0000000 Binary files a/src/main/resources/unicodes/unicode_page_a9.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_aa.png b/src/main/resources/unicodes/unicode_page_aa.png deleted file mode 100644 index 8192712..0000000 Binary files a/src/main/resources/unicodes/unicode_page_aa.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_ab.png b/src/main/resources/unicodes/unicode_page_ab.png deleted file mode 100644 index b771a71..0000000 Binary files a/src/main/resources/unicodes/unicode_page_ab.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_ac.png b/src/main/resources/unicodes/unicode_page_ac.png deleted file mode 100644 index 2818b95..0000000 Binary files a/src/main/resources/unicodes/unicode_page_ac.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_ad.png b/src/main/resources/unicodes/unicode_page_ad.png deleted file mode 100644 index 9d53d25..0000000 Binary files a/src/main/resources/unicodes/unicode_page_ad.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_ae.png b/src/main/resources/unicodes/unicode_page_ae.png deleted file mode 100644 index b10a7be..0000000 Binary files a/src/main/resources/unicodes/unicode_page_ae.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_af.png b/src/main/resources/unicodes/unicode_page_af.png deleted file mode 100644 index 538e377..0000000 Binary files a/src/main/resources/unicodes/unicode_page_af.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_b0.png b/src/main/resources/unicodes/unicode_page_b0.png deleted file mode 100644 index 0bac850..0000000 Binary files a/src/main/resources/unicodes/unicode_page_b0.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_b1.png b/src/main/resources/unicodes/unicode_page_b1.png deleted file mode 100644 index 0f0f56e..0000000 Binary files a/src/main/resources/unicodes/unicode_page_b1.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_b2.png b/src/main/resources/unicodes/unicode_page_b2.png deleted file mode 100644 index c064012..0000000 Binary files a/src/main/resources/unicodes/unicode_page_b2.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_b3.png b/src/main/resources/unicodes/unicode_page_b3.png deleted file mode 100644 index daa46be..0000000 Binary files a/src/main/resources/unicodes/unicode_page_b3.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_b4.png b/src/main/resources/unicodes/unicode_page_b4.png deleted file mode 100644 index 9cd89ff..0000000 Binary files a/src/main/resources/unicodes/unicode_page_b4.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_b5.png b/src/main/resources/unicodes/unicode_page_b5.png deleted file mode 100644 index d50990d..0000000 Binary files a/src/main/resources/unicodes/unicode_page_b5.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_b6.png b/src/main/resources/unicodes/unicode_page_b6.png deleted file mode 100644 index 7cbdbe5..0000000 Binary files a/src/main/resources/unicodes/unicode_page_b6.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_b7.png b/src/main/resources/unicodes/unicode_page_b7.png deleted file mode 100644 index bb82532..0000000 Binary files a/src/main/resources/unicodes/unicode_page_b7.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_b8.png b/src/main/resources/unicodes/unicode_page_b8.png deleted file mode 100644 index d5d32dd..0000000 Binary files a/src/main/resources/unicodes/unicode_page_b8.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_b9.png b/src/main/resources/unicodes/unicode_page_b9.png deleted file mode 100644 index acda6e0..0000000 Binary files a/src/main/resources/unicodes/unicode_page_b9.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_ba.png b/src/main/resources/unicodes/unicode_page_ba.png deleted file mode 100644 index 47d0a75..0000000 Binary files a/src/main/resources/unicodes/unicode_page_ba.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_bb.png b/src/main/resources/unicodes/unicode_page_bb.png deleted file mode 100644 index 39a4b8f..0000000 Binary files a/src/main/resources/unicodes/unicode_page_bb.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_bc.png b/src/main/resources/unicodes/unicode_page_bc.png deleted file mode 100644 index f942978..0000000 Binary files a/src/main/resources/unicodes/unicode_page_bc.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_bd.png b/src/main/resources/unicodes/unicode_page_bd.png deleted file mode 100644 index 783a79b..0000000 Binary files a/src/main/resources/unicodes/unicode_page_bd.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_be.png b/src/main/resources/unicodes/unicode_page_be.png deleted file mode 100644 index 2df8f23..0000000 Binary files a/src/main/resources/unicodes/unicode_page_be.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_bf.png b/src/main/resources/unicodes/unicode_page_bf.png deleted file mode 100644 index da9f3c6..0000000 Binary files a/src/main/resources/unicodes/unicode_page_bf.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_c0.png b/src/main/resources/unicodes/unicode_page_c0.png deleted file mode 100644 index 21ee4be..0000000 Binary files a/src/main/resources/unicodes/unicode_page_c0.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_c1.png b/src/main/resources/unicodes/unicode_page_c1.png deleted file mode 100644 index c7cc199..0000000 Binary files a/src/main/resources/unicodes/unicode_page_c1.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_c2.png b/src/main/resources/unicodes/unicode_page_c2.png deleted file mode 100644 index 5e8dfdb..0000000 Binary files a/src/main/resources/unicodes/unicode_page_c2.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_c3.png b/src/main/resources/unicodes/unicode_page_c3.png deleted file mode 100644 index 949ca7c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_c3.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_c4.png b/src/main/resources/unicodes/unicode_page_c4.png deleted file mode 100644 index 85580db..0000000 Binary files a/src/main/resources/unicodes/unicode_page_c4.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_c5.png b/src/main/resources/unicodes/unicode_page_c5.png deleted file mode 100644 index 656e69f..0000000 Binary files a/src/main/resources/unicodes/unicode_page_c5.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_c6.png b/src/main/resources/unicodes/unicode_page_c6.png deleted file mode 100644 index e9e9e77..0000000 Binary files a/src/main/resources/unicodes/unicode_page_c6.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_c7.png b/src/main/resources/unicodes/unicode_page_c7.png deleted file mode 100644 index 7b3971e..0000000 Binary files a/src/main/resources/unicodes/unicode_page_c7.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_c8.png b/src/main/resources/unicodes/unicode_page_c8.png deleted file mode 100644 index 5b02f56..0000000 Binary files a/src/main/resources/unicodes/unicode_page_c8.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_c9.png b/src/main/resources/unicodes/unicode_page_c9.png deleted file mode 100644 index f162dc4..0000000 Binary files a/src/main/resources/unicodes/unicode_page_c9.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_ca.png b/src/main/resources/unicodes/unicode_page_ca.png deleted file mode 100644 index bb52b26..0000000 Binary files a/src/main/resources/unicodes/unicode_page_ca.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_cb.png b/src/main/resources/unicodes/unicode_page_cb.png deleted file mode 100644 index bc38e37..0000000 Binary files a/src/main/resources/unicodes/unicode_page_cb.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_cc.png b/src/main/resources/unicodes/unicode_page_cc.png deleted file mode 100644 index 4c82ee7..0000000 Binary files a/src/main/resources/unicodes/unicode_page_cc.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_cd.png b/src/main/resources/unicodes/unicode_page_cd.png deleted file mode 100644 index 806205f..0000000 Binary files a/src/main/resources/unicodes/unicode_page_cd.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_ce.png b/src/main/resources/unicodes/unicode_page_ce.png deleted file mode 100644 index ef0a4a7..0000000 Binary files a/src/main/resources/unicodes/unicode_page_ce.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_cf.png b/src/main/resources/unicodes/unicode_page_cf.png deleted file mode 100644 index 2249985..0000000 Binary files a/src/main/resources/unicodes/unicode_page_cf.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_d0.png b/src/main/resources/unicodes/unicode_page_d0.png deleted file mode 100644 index 4438d1d..0000000 Binary files a/src/main/resources/unicodes/unicode_page_d0.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_d1.png b/src/main/resources/unicodes/unicode_page_d1.png deleted file mode 100644 index cb6afe8..0000000 Binary files a/src/main/resources/unicodes/unicode_page_d1.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_d2.png b/src/main/resources/unicodes/unicode_page_d2.png deleted file mode 100644 index 62ae825..0000000 Binary files a/src/main/resources/unicodes/unicode_page_d2.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_d3.png b/src/main/resources/unicodes/unicode_page_d3.png deleted file mode 100644 index a3d5989..0000000 Binary files a/src/main/resources/unicodes/unicode_page_d3.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_d4.png b/src/main/resources/unicodes/unicode_page_d4.png deleted file mode 100644 index 893b356..0000000 Binary files a/src/main/resources/unicodes/unicode_page_d4.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_d5.png b/src/main/resources/unicodes/unicode_page_d5.png deleted file mode 100644 index 4f7774d..0000000 Binary files a/src/main/resources/unicodes/unicode_page_d5.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_d6.png b/src/main/resources/unicodes/unicode_page_d6.png deleted file mode 100644 index f72847f..0000000 Binary files a/src/main/resources/unicodes/unicode_page_d6.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_d7.png b/src/main/resources/unicodes/unicode_page_d7.png deleted file mode 100644 index 845a04b..0000000 Binary files a/src/main/resources/unicodes/unicode_page_d7.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_f9.png b/src/main/resources/unicodes/unicode_page_f9.png deleted file mode 100644 index 02c37fd..0000000 Binary files a/src/main/resources/unicodes/unicode_page_f9.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_fa.png b/src/main/resources/unicodes/unicode_page_fa.png deleted file mode 100644 index fda4560..0000000 Binary files a/src/main/resources/unicodes/unicode_page_fa.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_fb.png b/src/main/resources/unicodes/unicode_page_fb.png deleted file mode 100644 index c900e2a..0000000 Binary files a/src/main/resources/unicodes/unicode_page_fb.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_fc.png b/src/main/resources/unicodes/unicode_page_fc.png deleted file mode 100644 index 49d460c..0000000 Binary files a/src/main/resources/unicodes/unicode_page_fc.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_fd.png b/src/main/resources/unicodes/unicode_page_fd.png deleted file mode 100644 index ea8e3e0..0000000 Binary files a/src/main/resources/unicodes/unicode_page_fd.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_fe.png b/src/main/resources/unicodes/unicode_page_fe.png deleted file mode 100644 index cacee8a..0000000 Binary files a/src/main/resources/unicodes/unicode_page_fe.png and /dev/null differ diff --git a/src/main/resources/unicodes/unicode_page_ff.png b/src/main/resources/unicodes/unicode_page_ff.png deleted file mode 100644 index c14110d..0000000 Binary files a/src/main/resources/unicodes/unicode_page_ff.png and /dev/null differ diff --git a/velocity/.gitignore b/velocity/.gitignore new file mode 100644 index 0000000..b63da45 --- /dev/null +++ b/velocity/.gitignore @@ -0,0 +1,42 @@ +.gradle +build/ +!gradle/wrapper/gradle-wrapper.jar +!**/src/main/**/build/ +!**/src/test/**/build/ + +### IntelliJ IDEA ### +.idea/modules.xml +.idea/jarRepositories.xml +.idea/compiler.xml +.idea/libraries/ +*.iws +*.iml +*.ipr +out/ +!**/src/main/**/out/ +!**/src/test/**/out/ + +### Eclipse ### +.apt_generated +.classpath +.factorypath +.project +.settings +.springBeans +.sts4-cache +bin/ +!**/src/main/**/bin/ +!**/src/test/**/bin/ + +### NetBeans ### +/nbproject/private/ +/nbbuild/ +/dist/ +/nbdist/ +/.nb-gradle/ + +### VS Code ### +.vscode/ + +### Mac OS ### +.DS_Store \ No newline at end of file diff --git a/velocity/build.gradle.kts b/velocity/build.gradle.kts new file mode 100644 index 0000000..571b417 --- /dev/null +++ b/velocity/build.gradle.kts @@ -0,0 +1,2 @@ +dependencies { +} \ No newline at end of file diff --git a/velocity/src/main/resources/plugin.yml b/velocity/src/main/resources/plugin.yml new file mode 100644 index 0000000..5bf5d9b --- /dev/null +++ b/velocity/src/main/resources/plugin.yml @@ -0,0 +1,9 @@ +name: CustomNameplates +version: '${version}' +main: net.momirealms.customnameplates.paper.Main +api-version: 1.17 +authors: [ XiaoMoMi ] +folia-supported: true +depend: + - ProtocolLib + - PlaceholderAPI \ No newline at end of file