geforkt von Mirrors/FastAsyncWorldEdit
Replace the message system
Dieser Commit ist enthalten in:
Ursprung
d56cd96282
Commit
5b1573a24e
@ -142,9 +142,11 @@ subprojects {
|
|||||||
include(dependency('com.sk89q:jchronic:0.2.4a'))
|
include(dependency('com.sk89q:jchronic:0.2.4a'))
|
||||||
include(dependency('com.thoughtworks.paranamer:paranamer:2.6'))
|
include(dependency('com.thoughtworks.paranamer:paranamer:2.6'))
|
||||||
include(dependency('com.sk89q.lib:jlibnoise:1.0.0'))
|
include(dependency('com.sk89q.lib:jlibnoise:1.0.0'))
|
||||||
relocate('net.kyori.text', 'com.sk89q.worldedit.util.formatting.text') {
|
|
||||||
include(dependency('net.kyori:text-api:2.0.0'))
|
include(dependency('net.kyori:text-api:2.0.0'))
|
||||||
}
|
include(dependency("net.kyori:text-serializer-gson:2.0.0"))
|
||||||
|
include(dependency("net.kyori:text-serializer-legacy:2.0.0"))
|
||||||
|
|
||||||
|
relocate('net.kyori.text', 'com.sk89q.worldedit.util.formatting.text')
|
||||||
}
|
}
|
||||||
exclude 'GradleStart**'
|
exclude 'GradleStart**'
|
||||||
exclude '.cache'
|
exclude '.cache'
|
||||||
|
@ -43,6 +43,7 @@ shadowJar {
|
|||||||
include(dependency(':worldedit-core'))
|
include(dependency(':worldedit-core'))
|
||||||
include(dependency('org.slf4j:slf4j-api'))
|
include(dependency('org.slf4j:slf4j-api'))
|
||||||
include(dependency("org.slf4j:slf4j-jdk14"))
|
include(dependency("org.slf4j:slf4j-jdk14"))
|
||||||
|
include(dependency("net.kyori:text-adapter-bukkit:1.0.3"))
|
||||||
relocate ("org.bstats", "com.sk89q.worldedit.bukkit.bstats") {
|
relocate ("org.bstats", "com.sk89q.worldedit.bukkit.bstats") {
|
||||||
include(dependency("org.bstats:bstats-bukkit:1.4"))
|
include(dependency("org.bstats:bstats-bukkit:1.4"))
|
||||||
}
|
}
|
||||||
|
@ -16,6 +16,8 @@ dependencies {
|
|||||||
compile 'com.googlecode.json-simple:json-simple:1.1.1'
|
compile 'com.googlecode.json-simple:json-simple:1.1.1'
|
||||||
compile 'org.slf4j:slf4j-api:1.7.26'
|
compile 'org.slf4j:slf4j-api:1.7.26'
|
||||||
compile 'net.kyori:text-api:2.0.0'
|
compile 'net.kyori:text-api:2.0.0'
|
||||||
|
compile 'net.kyori:text-serializer-gson:2.0.0'
|
||||||
|
compile 'net.kyori:text-serializer-legacy:2.0.0'
|
||||||
//compile 'net.sf.trove4j:trove4j:3.0.3'
|
//compile 'net.sf.trove4j:trove4j:3.0.3'
|
||||||
testCompile 'org.mockito:mockito-core:1.9.0-rc1'
|
testCompile 'org.mockito:mockito-core:1.9.0-rc1'
|
||||||
}
|
}
|
||||||
|
@ -55,15 +55,14 @@ import com.sk89q.worldedit.regions.selector.SphereRegionSelector;
|
|||||||
import com.sk89q.worldedit.session.ClipboardHolder;
|
import com.sk89q.worldedit.session.ClipboardHolder;
|
||||||
import com.sk89q.worldedit.util.Countable;
|
import com.sk89q.worldedit.util.Countable;
|
||||||
import com.sk89q.worldedit.util.Location;
|
import com.sk89q.worldedit.util.Location;
|
||||||
import com.sk89q.worldedit.util.formatting.ColorCodeBuilder;
|
|
||||||
import com.sk89q.worldedit.util.formatting.Style;
|
|
||||||
import com.sk89q.worldedit.util.formatting.StyledFragment;
|
|
||||||
import com.sk89q.worldedit.util.formatting.component.CommandListBox;
|
import com.sk89q.worldedit.util.formatting.component.CommandListBox;
|
||||||
|
import com.sk89q.worldedit.util.formatting.component.Subtle;
|
||||||
import com.sk89q.worldedit.world.World;
|
import com.sk89q.worldedit.world.World;
|
||||||
import com.sk89q.worldedit.world.block.BaseBlock;
|
import com.sk89q.worldedit.world.block.BaseBlock;
|
||||||
import com.sk89q.worldedit.world.block.BlockState;
|
import com.sk89q.worldedit.world.block.BlockState;
|
||||||
import com.sk89q.worldedit.world.item.ItemTypes;
|
import com.sk89q.worldedit.world.item.ItemTypes;
|
||||||
import com.sk89q.worldedit.world.storage.ChunkStore;
|
import com.sk89q.worldedit.world.storage.ChunkStore;
|
||||||
|
import net.kyori.text.Component;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@ -754,9 +753,8 @@ public class SelectionCommands {
|
|||||||
limit.ifPresent(integer -> player.print(integer + " points maximum."));
|
limit.ifPresent(integer -> player.print(integer + " points maximum."));
|
||||||
} else {
|
} else {
|
||||||
CommandListBox box = new CommandListBox("Selection modes");
|
CommandListBox box = new CommandListBox("Selection modes");
|
||||||
StyledFragment contents = box.getContents();
|
Component contents = box.getContents();
|
||||||
StyledFragment tip = contents.createFragment(Style.RED);
|
contents.append(new Subtle("Select one of the modes below:").append(Component.newline()));
|
||||||
tip.append("Select one of the modes below:").newLine();
|
|
||||||
|
|
||||||
box.appendCommand("cuboid", "Select two corners of a cuboid");
|
box.appendCommand("cuboid", "Select two corners of a cuboid");
|
||||||
box.appendCommand("extend", "Fast cuboid selection mode");
|
box.appendCommand("extend", "Fast cuboid selection mode");
|
||||||
@ -766,7 +764,7 @@ public class SelectionCommands {
|
|||||||
box.appendCommand("cyl", "Select a cylinder");
|
box.appendCommand("cyl", "Select a cylinder");
|
||||||
box.appendCommand("convex", "Select a convex polyhedral");
|
box.appendCommand("convex", "Select a convex polyhedral");
|
||||||
|
|
||||||
player.printRaw(ColorCodeBuilder.asColorCodes(box));
|
player.print(box);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -58,15 +58,16 @@ import com.sk89q.worldedit.util.command.CommandMapping;
|
|||||||
import com.sk89q.worldedit.util.command.Dispatcher;
|
import com.sk89q.worldedit.util.command.Dispatcher;
|
||||||
import com.sk89q.worldedit.util.command.PrimaryAliasComparator;
|
import com.sk89q.worldedit.util.command.PrimaryAliasComparator;
|
||||||
import com.sk89q.worldedit.util.command.binding.Text;
|
import com.sk89q.worldedit.util.command.binding.Text;
|
||||||
import com.sk89q.worldedit.util.formatting.ColorCodeBuilder;
|
|
||||||
import com.sk89q.worldedit.util.formatting.Style;
|
|
||||||
import com.sk89q.worldedit.util.formatting.StyledFragment;
|
|
||||||
import com.sk89q.worldedit.util.formatting.component.Code;
|
import com.sk89q.worldedit.util.formatting.component.Code;
|
||||||
import com.sk89q.worldedit.util.formatting.component.CommandListBox;
|
import com.sk89q.worldedit.util.formatting.component.CommandListBox;
|
||||||
import com.sk89q.worldedit.util.formatting.component.CommandUsageBox;
|
import com.sk89q.worldedit.util.formatting.component.CommandUsageBox;
|
||||||
|
import com.sk89q.worldedit.util.formatting.component.Error;
|
||||||
import com.sk89q.worldedit.world.World;
|
import com.sk89q.worldedit.world.World;
|
||||||
import com.sk89q.worldedit.world.block.BaseBlock;
|
import com.sk89q.worldedit.world.block.BaseBlock;
|
||||||
import com.sk89q.worldedit.world.block.BlockTypes;
|
import com.sk89q.worldedit.world.block.BlockTypes;
|
||||||
|
import net.kyori.text.Component;
|
||||||
|
import net.kyori.text.TextComponent;
|
||||||
|
import net.kyori.text.format.TextColor;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@ -669,17 +670,17 @@ public class UtilityCommands {
|
|||||||
|
|
||||||
// Box
|
// Box
|
||||||
CommandListBox box = new CommandListBox(String.format("Help: page %d/%d ", page + 1, pageTotal));
|
CommandListBox box = new CommandListBox(String.format("Help: page %d/%d ", page + 1, pageTotal));
|
||||||
StyledFragment contents = box.getContents();
|
Component contents = box.getContents();
|
||||||
StyledFragment tip = contents.createFragment(Style.GRAY);
|
Component tip = contents.append(TextComponent.of("", TextColor.GRAY));
|
||||||
|
|
||||||
if (offset >= aliases.size()) {
|
if (offset >= aliases.size()) {
|
||||||
tip.createFragment(Style.RED).append(String.format("There is no page %d (total number of pages is %d).", page + 1, pageTotal)).newLine();
|
tip.append(new Error(String.format("There is no page %d (total number of pages is %d).", page + 1, pageTotal))).append(Component.newline());
|
||||||
} else {
|
} else {
|
||||||
List<CommandMapping> list = aliases.subList(offset, Math.min(offset + perPage, aliases.size()));
|
List<CommandMapping> list = aliases.subList(offset, Math.min(offset + perPage, aliases.size()));
|
||||||
|
|
||||||
tip.append("Type ");
|
tip.append(TextComponent.of("Type "));
|
||||||
tip.append(new Code().append("//help ").append("<command> [<page>]"));
|
tip.append(new Code("//help ").append(TextComponent.of("<command> [<page>]")));
|
||||||
tip.append(" for more information.").newLine();
|
tip.append(TextComponent.of(" for more information.")).append(Component.newline());
|
||||||
|
|
||||||
// Add each command
|
// Add each command
|
||||||
for (CommandMapping mapping : list) {
|
for (CommandMapping mapping : list) {
|
||||||
@ -696,10 +697,10 @@ public class UtilityCommands {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
actor.printRaw(ColorCodeBuilder.asColorCodes(box));
|
actor.print(box);
|
||||||
} else {
|
} else {
|
||||||
CommandUsageBox box = new CommandUsageBox(callable, Joiner.on(" ").join(visited));
|
CommandUsageBox box = new CommandUsageBox(callable, Joiner.on(" ").join(visited));
|
||||||
actor.printRaw(ColorCodeBuilder.asColorCodes(box));
|
actor.print(box);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -77,7 +77,6 @@ import com.sk89q.worldedit.util.command.parametric.ExceptionConverter;
|
|||||||
import com.sk89q.worldedit.util.command.parametric.LegacyCommandsHandler;
|
import com.sk89q.worldedit.util.command.parametric.LegacyCommandsHandler;
|
||||||
import com.sk89q.worldedit.util.command.parametric.ParametricBuilder;
|
import com.sk89q.worldedit.util.command.parametric.ParametricBuilder;
|
||||||
import com.sk89q.worldedit.util.eventbus.Subscribe;
|
import com.sk89q.worldedit.util.eventbus.Subscribe;
|
||||||
import com.sk89q.worldedit.util.formatting.ColorCodeBuilder;
|
|
||||||
import com.sk89q.worldedit.util.formatting.component.CommandUsageBox;
|
import com.sk89q.worldedit.util.formatting.component.CommandUsageBox;
|
||||||
import com.sk89q.worldedit.util.logging.DynamicStreamHandler;
|
import com.sk89q.worldedit.util.logging.DynamicStreamHandler;
|
||||||
import com.sk89q.worldedit.util.logging.LogFormat;
|
import com.sk89q.worldedit.util.logging.LogFormat;
|
||||||
@ -300,7 +299,7 @@ public final class CommandManager {
|
|||||||
actor.printError("You are not permitted to do that. Are you in the right mode?");
|
actor.printError("You are not permitted to do that. Are you in the right mode?");
|
||||||
} catch (InvalidUsageException e) {
|
} catch (InvalidUsageException e) {
|
||||||
if (e.isFullHelpSuggested()) {
|
if (e.isFullHelpSuggested()) {
|
||||||
actor.printRaw(ColorCodeBuilder.asColorCodes(new CommandUsageBox(e.getCommand(), e.getCommandUsed("/", ""), locals)));
|
actor.print(new CommandUsageBox(e.getCommand(), e.getCommandUsed("/", ""), locals));
|
||||||
String message = e.getMessage();
|
String message = e.getMessage();
|
||||||
if (message != null) {
|
if (message != null) {
|
||||||
actor.printError(message);
|
actor.printError(message);
|
||||||
|
@ -1,274 +0,0 @@
|
|||||||
/*
|
|
||||||
* WorldEdit, a Minecraft world manipulation toolkit
|
|
||||||
* Copyright (C) sk89q <http://www.sk89q.com>
|
|
||||||
* Copyright (C) WorldEdit team and contributors
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU Lesser General Public License as published by the
|
|
||||||
* Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) 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 Lesser General Public License
|
|
||||||
* for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package com.sk89q.worldedit.util.formatting;
|
|
||||||
|
|
||||||
import com.google.common.base.Joiner;
|
|
||||||
|
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.LinkedList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
public class ColorCodeBuilder {
|
|
||||||
|
|
||||||
private static final ColorCodeBuilder instance = new ColorCodeBuilder();
|
|
||||||
private static final Joiner newLineJoiner = Joiner.on("\n");
|
|
||||||
public static final int GUARANTEED_NO_WRAP_CHAT_PAGE_WIDTH = 47;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convert a message into color-coded text.
|
|
||||||
*
|
|
||||||
* @param message the message
|
|
||||||
* @return a list of lines
|
|
||||||
*/
|
|
||||||
public String[] build(StyledFragment message) {
|
|
||||||
StringBuilder builder = new StringBuilder();
|
|
||||||
buildFragment(builder, message, message.getStyle(), new StyleSet());
|
|
||||||
return builder.toString().split("\r?\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Build a fragment.
|
|
||||||
*
|
|
||||||
* @param builder the string builder
|
|
||||||
* @param message the message
|
|
||||||
* @param parentStyle the parent style
|
|
||||||
* @param lastStyle the last style
|
|
||||||
* @return the last style used
|
|
||||||
*/
|
|
||||||
private StyleSet buildFragment(StringBuilder builder, StyledFragment message, StyleSet parentStyle, StyleSet lastStyle) {
|
|
||||||
for (Fragment node : message.getChildren()) {
|
|
||||||
if (node instanceof StyledFragment) {
|
|
||||||
StyledFragment fragment = (StyledFragment) node;
|
|
||||||
lastStyle = buildFragment(
|
|
||||||
builder, fragment,
|
|
||||||
parentStyle.extend(message.getStyle()), lastStyle);
|
|
||||||
} else {
|
|
||||||
StyleSet style = parentStyle.extend(message.getStyle());
|
|
||||||
builder.append(getAdditive(style, lastStyle));
|
|
||||||
builder.append(node);
|
|
||||||
lastStyle = style;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return lastStyle;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the formatting codes.
|
|
||||||
*
|
|
||||||
* @param style the style
|
|
||||||
* @return the color codes
|
|
||||||
*/
|
|
||||||
public static String getFormattingCode(StyleSet style) {
|
|
||||||
StringBuilder builder = new StringBuilder();
|
|
||||||
if (style.isBold()) {
|
|
||||||
builder.append(Style.BOLD);
|
|
||||||
}
|
|
||||||
if (style.isItalic()) {
|
|
||||||
builder.append(Style.ITALIC);
|
|
||||||
}
|
|
||||||
if (style.isUnderline()) {
|
|
||||||
builder.append(Style.UNDERLINE);
|
|
||||||
}
|
|
||||||
if (style.isStrikethrough()) {
|
|
||||||
builder.append(Style.STRIKETHROUGH);
|
|
||||||
}
|
|
||||||
if (style.isObfuscated()) {
|
|
||||||
builder.append(Style.OBFUSCATED);
|
|
||||||
}
|
|
||||||
return builder.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the formatting and color codes.
|
|
||||||
*
|
|
||||||
* @param style the style
|
|
||||||
* @return the color codes
|
|
||||||
*/
|
|
||||||
public static String getCode(StyleSet style) {
|
|
||||||
StringBuilder builder = new StringBuilder();
|
|
||||||
builder.append(getFormattingCode(style));
|
|
||||||
if (style.getColor() != null) {
|
|
||||||
builder.append(style.getColor());
|
|
||||||
}
|
|
||||||
return builder.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the additional color codes needed to set the given style when the current
|
|
||||||
* style is the other given one.
|
|
||||||
*
|
|
||||||
* @param resetTo the style to reset to
|
|
||||||
* @param resetFrom the style to reset from
|
|
||||||
* @return the color codes
|
|
||||||
*/
|
|
||||||
public static String getAdditive(StyleSet resetTo, StyleSet resetFrom) {
|
|
||||||
if (!resetFrom.hasFormatting() && resetTo.hasFormatting()) {
|
|
||||||
StringBuilder builder = new StringBuilder();
|
|
||||||
builder.append(getFormattingCode(resetTo));
|
|
||||||
if (resetFrom.getColor() != resetTo.getColor()) {
|
|
||||||
builder.append(resetTo.getColor());
|
|
||||||
}
|
|
||||||
return builder.toString();
|
|
||||||
} else if (!resetFrom.hasEqualFormatting(resetTo) ||
|
|
||||||
(resetFrom.getColor() != null && resetTo.getColor() == null)) {
|
|
||||||
// Have to set reset code and add back all the formatting codes
|
|
||||||
return Style.RESET + getCode(resetTo);
|
|
||||||
} else {
|
|
||||||
if (resetFrom.getColor() != resetTo.getColor()) {
|
|
||||||
return String.valueOf(resetTo.getColor());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Word wrap the given text and maintain color codes throughout lines.
|
|
||||||
*
|
|
||||||
* <p>This is borrowed from Bukkit.</p>
|
|
||||||
*
|
|
||||||
* @param rawString the raw string
|
|
||||||
* @param lineLength the maximum line length
|
|
||||||
* @return a list of lines
|
|
||||||
*/
|
|
||||||
private String[] wordWrap(String rawString, int lineLength) {
|
|
||||||
// A null string is a single line
|
|
||||||
if (rawString == null) {
|
|
||||||
return new String[] {""};
|
|
||||||
}
|
|
||||||
|
|
||||||
// A string shorter than the lineWidth is a single line
|
|
||||||
if (rawString.length() <= lineLength && !rawString.contains("\n")) {
|
|
||||||
return new String[] {rawString};
|
|
||||||
}
|
|
||||||
|
|
||||||
char[] rawChars = (rawString + ' ').toCharArray(); // add a trailing space to trigger pagination
|
|
||||||
StringBuilder word = new StringBuilder();
|
|
||||||
StringBuilder line = new StringBuilder();
|
|
||||||
List<String> lines = new LinkedList<>();
|
|
||||||
int lineColorChars = 0;
|
|
||||||
|
|
||||||
for (int i = 0; i < rawChars.length; i++) {
|
|
||||||
char c = rawChars[i];
|
|
||||||
|
|
||||||
// skip chat color modifiers
|
|
||||||
if (c == Style.COLOR_CHAR) {
|
|
||||||
word.append(Style.getByChar(rawChars[i + 1]));
|
|
||||||
lineColorChars += 2;
|
|
||||||
i++; // Eat the next character as we have already processed it
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (c == ' ' || c == '\n') {
|
|
||||||
if (line.length() == 0 && word.length() > lineLength) { // special case: extremely long word begins a line
|
|
||||||
String wordStr = word.toString();
|
|
||||||
String transformed;
|
|
||||||
if ((transformed = transform(wordStr)) != null) {
|
|
||||||
line.append(transformed);
|
|
||||||
} else {
|
|
||||||
lines.addAll(Arrays.asList(word.toString().split("(?<=\\G.{" + lineLength + "})")));
|
|
||||||
}
|
|
||||||
} else if (line.length() + word.length() - lineColorChars == lineLength) { // Line exactly the correct length...newline
|
|
||||||
line.append(' ');
|
|
||||||
line.append(word);
|
|
||||||
lines.add(line.toString());
|
|
||||||
line = new StringBuilder();
|
|
||||||
lineColorChars = 0;
|
|
||||||
} else if (line.length() + 1 + word.length() - lineColorChars > lineLength) { // Line too long...break the line
|
|
||||||
String wordStr = word.toString();
|
|
||||||
String transformed;
|
|
||||||
if (word.length() > lineLength && (transformed = transform(wordStr)) != null) {
|
|
||||||
if (line.length() + 1 + transformed.length() - lineColorChars > lineLength) {
|
|
||||||
lines.add(line.toString());
|
|
||||||
line = new StringBuilder(transformed);
|
|
||||||
lineColorChars = 0;
|
|
||||||
} else {
|
|
||||||
if (line.length() > 0) {
|
|
||||||
line.append(' ');
|
|
||||||
}
|
|
||||||
line.append(transformed);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
for (String partialWord : wordStr.split("(?<=\\G.{" + lineLength + "})")) {
|
|
||||||
lines.add(line.toString());
|
|
||||||
line = new StringBuilder(partialWord);
|
|
||||||
}
|
|
||||||
lineColorChars = 0;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (line.length() > 0) {
|
|
||||||
line.append(' ');
|
|
||||||
}
|
|
||||||
line.append(word);
|
|
||||||
}
|
|
||||||
word = new StringBuilder();
|
|
||||||
|
|
||||||
if (c == '\n') { // Newline forces the line to flush
|
|
||||||
lines.add(line.toString());
|
|
||||||
line = new StringBuilder();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
word.append(c);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if(line.length() > 0) { // Only add the last line if there is anything to add
|
|
||||||
lines.add(line.toString());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Iterate over the wrapped lines, applying the last color from one line to the beginning of the next
|
|
||||||
if (lines.get(0).isEmpty() || lines.get(0).charAt(0) != Style.COLOR_CHAR) {
|
|
||||||
lines.set(0, Style.WHITE + lines.get(0));
|
|
||||||
}
|
|
||||||
for (int i = 1; i < lines.size(); i++) {
|
|
||||||
final String pLine = lines.get(i-1);
|
|
||||||
final String subLine = lines.get(i);
|
|
||||||
|
|
||||||
char color = pLine.charAt(pLine.lastIndexOf(Style.COLOR_CHAR) + 1);
|
|
||||||
if (subLine.isEmpty() || subLine.charAt(0) != Style.COLOR_CHAR) {
|
|
||||||
lines.set(i, Style.getByChar(color) + subLine);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return lines.toArray(new String[lines.size()]);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Callback for transforming a word, such as a URL.
|
|
||||||
*
|
|
||||||
* @param word the word
|
|
||||||
* @return the transformed value, or null to do nothing
|
|
||||||
*/
|
|
||||||
protected String transform(String word) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Convert the given styled fragment into color codes.
|
|
||||||
*
|
|
||||||
* @param fragment the fragment
|
|
||||||
* @return color codes
|
|
||||||
*/
|
|
||||||
public static String asColorCodes(StyledFragment fragment) {
|
|
||||||
return newLineJoiner.join(instance.build(fragment));
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -1,121 +0,0 @@
|
|||||||
/*
|
|
||||||
* WorldEdit, a Minecraft world manipulation toolkit
|
|
||||||
* Copyright (C) sk89q <http://www.sk89q.com>
|
|
||||||
* Copyright (C) WorldEdit team and contributors
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU Lesser General Public License as published by the
|
|
||||||
* Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) 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 Lesser General Public License
|
|
||||||
* for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package com.sk89q.worldedit.util.formatting;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A fragment of text.
|
|
||||||
*/
|
|
||||||
public class Fragment {
|
|
||||||
|
|
||||||
private final StringBuilder builder = new StringBuilder();
|
|
||||||
private final List<Fragment> children = new ArrayList<>();
|
|
||||||
private Fragment lastText;
|
|
||||||
|
|
||||||
Fragment() {
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<Fragment> getChildren() {
|
|
||||||
return children;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected Fragment lastText() {
|
|
||||||
Fragment text;
|
|
||||||
if (!children.isEmpty()) {
|
|
||||||
text = children.get(children.size() - 1);
|
|
||||||
if (text == lastText) {
|
|
||||||
return text;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
text = new Fragment();
|
|
||||||
this.lastText = text;
|
|
||||||
children.add(text);
|
|
||||||
return text;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment append(Fragment fragment) {
|
|
||||||
children.add(fragment);
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment append(String str) {
|
|
||||||
builder.append(Style.stripColor(str));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment append(Object obj) {
|
|
||||||
append(String.valueOf(obj));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment append(StringBuffer sb) {
|
|
||||||
append(String.valueOf(sb));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment append(CharSequence s) {
|
|
||||||
append(String.valueOf(s));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment append(boolean b) {
|
|
||||||
append(String.valueOf(b));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment append(char c) {
|
|
||||||
append(String.valueOf(c));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment append(int i) {
|
|
||||||
append(String.valueOf(i));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment append(long lng) {
|
|
||||||
append(String.valueOf(lng));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment append(float f) {
|
|
||||||
append(String.valueOf(f));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment append(double d) {
|
|
||||||
append(String.valueOf(d));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Fragment newLine() {
|
|
||||||
append("\n");
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String toString() {
|
|
||||||
return builder.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -1,274 +0,0 @@
|
|||||||
/*
|
|
||||||
* WorldEdit, a Minecraft world manipulation toolkit
|
|
||||||
* Copyright (C) sk89q <http://www.sk89q.com>
|
|
||||||
* Copyright (C) WorldEdit team and contributors
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU Lesser General Public License as published by the
|
|
||||||
* Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) 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 Lesser General Public License
|
|
||||||
* for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package com.sk89q.worldedit.util.formatting;
|
|
||||||
|
|
||||||
import static com.google.common.base.Preconditions.checkArgument;
|
|
||||||
import static com.google.common.base.Preconditions.checkNotNull;
|
|
||||||
|
|
||||||
import com.google.common.collect.Maps;
|
|
||||||
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.regex.Pattern;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* All supported color values for chat.
|
|
||||||
*/
|
|
||||||
public enum Style {
|
|
||||||
/**
|
|
||||||
* Represents black
|
|
||||||
*/
|
|
||||||
BLACK('0', 0x00),
|
|
||||||
/**
|
|
||||||
* Represents dark blue
|
|
||||||
*/
|
|
||||||
BLUE_DARK('1', 0x1),
|
|
||||||
/**
|
|
||||||
* Represents dark green
|
|
||||||
*/
|
|
||||||
GREEN_DARK('2', 0x2),
|
|
||||||
/**
|
|
||||||
* Represents dark blue (aqua)
|
|
||||||
*/
|
|
||||||
CYAN_DARK('3', 0x3),
|
|
||||||
/**
|
|
||||||
* Represents dark red
|
|
||||||
*/
|
|
||||||
RED_DARK('4', 0x4),
|
|
||||||
/**
|
|
||||||
* Represents dark purple
|
|
||||||
*/
|
|
||||||
PURPLE_DARK('5', 0x5),
|
|
||||||
/**
|
|
||||||
* Represents gold
|
|
||||||
*/
|
|
||||||
YELLOW_DARK('6', 0x6),
|
|
||||||
/**
|
|
||||||
* Represents gray
|
|
||||||
*/
|
|
||||||
GRAY('7', 0x7),
|
|
||||||
/**
|
|
||||||
* Represents dark gray
|
|
||||||
*/
|
|
||||||
GRAY_DARK('8', 0x8),
|
|
||||||
/**
|
|
||||||
* Represents blue
|
|
||||||
*/
|
|
||||||
BLUE('9', 0x9),
|
|
||||||
/**
|
|
||||||
* Represents green
|
|
||||||
*/
|
|
||||||
GREEN('a', 0xA),
|
|
||||||
/**
|
|
||||||
* Represents aqua
|
|
||||||
*/
|
|
||||||
CYAN('b', 0xB),
|
|
||||||
/**
|
|
||||||
* Represents red
|
|
||||||
*/
|
|
||||||
RED('c', 0xC),
|
|
||||||
/**
|
|
||||||
* Represents light purple
|
|
||||||
*/
|
|
||||||
PURPLE('d', 0xD),
|
|
||||||
/**
|
|
||||||
* Represents yellow
|
|
||||||
*/
|
|
||||||
YELLOW('e', 0xE),
|
|
||||||
/**
|
|
||||||
* Represents white
|
|
||||||
*/
|
|
||||||
WHITE('f', 0xF),
|
|
||||||
/**
|
|
||||||
* Represents magical characters that change around randomly
|
|
||||||
*/
|
|
||||||
OBFUSCATED('k', 0x10, true),
|
|
||||||
/**
|
|
||||||
* Makes the text bold.
|
|
||||||
*/
|
|
||||||
BOLD('l', 0x11, true),
|
|
||||||
/**
|
|
||||||
* Makes a line appear through the text.
|
|
||||||
*/
|
|
||||||
STRIKETHROUGH('m', 0x12, true),
|
|
||||||
/**
|
|
||||||
* Makes the text appear underlined.
|
|
||||||
*/
|
|
||||||
UNDERLINE('n', 0x13, true),
|
|
||||||
/**
|
|
||||||
* Makes the text italic.
|
|
||||||
*/
|
|
||||||
ITALIC('o', 0x14, true),
|
|
||||||
/**
|
|
||||||
* Resets all previous chat colors or formats.
|
|
||||||
*/
|
|
||||||
RESET('r', 0x15);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The special character which prefixes all chat color codes. Use this if you need to dynamically
|
|
||||||
* convert color codes from your custom format.
|
|
||||||
*/
|
|
||||||
public static final char COLOR_CHAR = '\u00A7';
|
|
||||||
private static final Pattern STRIP_COLOR_PATTERN = Pattern.compile("(?i)" + COLOR_CHAR + "[0-9A-FK-OR]");
|
|
||||||
|
|
||||||
private final int intCode;
|
|
||||||
private final char code;
|
|
||||||
private final boolean isFormat;
|
|
||||||
private final String toString;
|
|
||||||
private final static Map<Integer, Style> BY_ID = Maps.newHashMap();
|
|
||||||
private final static Map<Character, Style> BY_CHAR = Maps.newHashMap();
|
|
||||||
|
|
||||||
Style(char code, int intCode) {
|
|
||||||
this(code, intCode, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
Style(char code, int intCode, boolean isFormat) {
|
|
||||||
this.code = code;
|
|
||||||
this.intCode = intCode;
|
|
||||||
this.isFormat = isFormat;
|
|
||||||
this.toString = new String(new char[] {COLOR_CHAR, code});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the char value associated with this color
|
|
||||||
*
|
|
||||||
* @return A char value of this color code
|
|
||||||
*/
|
|
||||||
public char getChar() {
|
|
||||||
return code;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String toString() {
|
|
||||||
return toString;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if this code is a format code as opposed to a color code.
|
|
||||||
*
|
|
||||||
* @return the if the code is a formatting code
|
|
||||||
*/
|
|
||||||
public boolean isFormat() {
|
|
||||||
return isFormat;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if this code is a color code as opposed to a format code.
|
|
||||||
*
|
|
||||||
* @return the if the code is a color
|
|
||||||
*/
|
|
||||||
public boolean isColor() {
|
|
||||||
return !isFormat && this != RESET;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the color represented by the specified color code
|
|
||||||
*
|
|
||||||
* @param code Code to check
|
|
||||||
* @return Associative Style with the given code, or null if it doesn't exist
|
|
||||||
*/
|
|
||||||
public static Style getByChar(char code) {
|
|
||||||
return BY_CHAR.get(code);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the color represented by the specified color code
|
|
||||||
*
|
|
||||||
* @param code Code to check
|
|
||||||
* @return Associative Style with the given code, or null if it doesn't exist
|
|
||||||
*/
|
|
||||||
public static Style getByChar(String code) {
|
|
||||||
checkNotNull(code);
|
|
||||||
checkArgument(!code.isEmpty(), "Code must have at least one character");
|
|
||||||
|
|
||||||
return BY_CHAR.get(code.charAt(0));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Strips the given message of all color codes
|
|
||||||
*
|
|
||||||
* @param input String to strip of color
|
|
||||||
* @return A copy of the input string, without any coloring
|
|
||||||
*/
|
|
||||||
public static String stripColor(final String input) {
|
|
||||||
if (input == null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return STRIP_COLOR_PATTERN.matcher(input).replaceAll("");
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Translates a string using an alternate color code character into a string that uses the internal
|
|
||||||
* ChatColor.COLOR_CODE color code character. The alternate color code character will only be replaced
|
|
||||||
* if it is immediately followed by 0-9, A-F, a-f, K-O, k-o, R or r.
|
|
||||||
*
|
|
||||||
* @param altColorChar The alternate color code character to replace. Ex: &
|
|
||||||
* @param textToTranslate Text containing the alternate color code character.
|
|
||||||
* @return Text containing the ChatColor.COLOR_CODE color code character.
|
|
||||||
*/
|
|
||||||
public static String translateAlternateColorCodes(char altColorChar, String textToTranslate) {
|
|
||||||
char[] b = textToTranslate.toCharArray();
|
|
||||||
for (int i = 0; i < b.length - 1; i++) {
|
|
||||||
if (b[i] == altColorChar && "0123456789AaBbCcDdEeFfKkLlMmNnOoRr".indexOf(b[i+1]) > -1) {
|
|
||||||
b[i] = Style.COLOR_CHAR;
|
|
||||||
b[i+1] = Character.toLowerCase(b[i+1]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return new String(b);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Gets the ChatColors used at the end of the given input string.
|
|
||||||
*
|
|
||||||
* @param input Input string to retrieve the colors from.
|
|
||||||
* @return Any remaining ChatColors to pass onto the next line.
|
|
||||||
*/
|
|
||||||
public static String getLastColors(String input) {
|
|
||||||
String result = "";
|
|
||||||
int length = input.length();
|
|
||||||
|
|
||||||
// Search backwards from the end as it is faster
|
|
||||||
for (int index = length - 1; index > -1; index--) {
|
|
||||||
char section = input.charAt(index);
|
|
||||||
if (section == COLOR_CHAR && index < length - 1) {
|
|
||||||
char c = input.charAt(index + 1);
|
|
||||||
Style color = getByChar(c);
|
|
||||||
|
|
||||||
if (color != null) {
|
|
||||||
result = color + result;
|
|
||||||
|
|
||||||
// Once we find a color or reset we can stop searching
|
|
||||||
if (color.isColor() || color == RESET) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
static {
|
|
||||||
for (Style color : values()) {
|
|
||||||
BY_ID.put(color.intCode, color);
|
|
||||||
BY_CHAR.put(color.code, color);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,320 +0,0 @@
|
|||||||
/*
|
|
||||||
* WorldEdit, a Minecraft world manipulation toolkit
|
|
||||||
* Copyright (C) sk89q <http://www.sk89q.com>
|
|
||||||
* Copyright (C) WorldEdit team and contributors
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU Lesser General Public License as published by the
|
|
||||||
* Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) 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 Lesser General Public License
|
|
||||||
* for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package com.sk89q.worldedit.util.formatting;
|
|
||||||
|
|
||||||
import java.util.Objects;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Represents set of styles, such as color, bold, etc.
|
|
||||||
*/
|
|
||||||
public class StyleSet {
|
|
||||||
|
|
||||||
private Boolean bold;
|
|
||||||
private Boolean italic;
|
|
||||||
private Boolean underline;
|
|
||||||
private Boolean strikethrough;
|
|
||||||
private Boolean obfuscated;
|
|
||||||
private String insertion;
|
|
||||||
private Style color;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new style set with no properties set.
|
|
||||||
*/
|
|
||||||
public StyleSet() {
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new style set with the given styles.
|
|
||||||
*
|
|
||||||
* <p>{@link Style#RESET} will be ignored if provided.</p>
|
|
||||||
*
|
|
||||||
* @param styles a list of styles
|
|
||||||
*/
|
|
||||||
public StyleSet(Style... styles) {
|
|
||||||
for (Style style : styles) {
|
|
||||||
if (style.isColor()) {
|
|
||||||
color = style;
|
|
||||||
} else if (style == Style.BOLD) {
|
|
||||||
bold = true;
|
|
||||||
} else if (style == Style.ITALIC) {
|
|
||||||
italic = true;
|
|
||||||
} else if (style == Style.UNDERLINE) {
|
|
||||||
underline = true;
|
|
||||||
} else if (style == Style.STRIKETHROUGH) {
|
|
||||||
strikethrough = true;
|
|
||||||
} else if (style == Style.OBFUSCATED) {
|
|
||||||
obfuscated = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether this style set is bold.
|
|
||||||
*
|
|
||||||
* @return true, false, or null if unset
|
|
||||||
*/
|
|
||||||
public Boolean getBold() {
|
|
||||||
return bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether the text is bold.
|
|
||||||
*
|
|
||||||
* @return true if bold
|
|
||||||
*/
|
|
||||||
public boolean isBold() {
|
|
||||||
return getBold() != null && getBold();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set whether the text is bold.
|
|
||||||
*
|
|
||||||
* @param bold true, false, or null to unset
|
|
||||||
*/
|
|
||||||
public void setBold(Boolean bold) {
|
|
||||||
this.bold = bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether this style set is italicized.
|
|
||||||
*
|
|
||||||
* @return true, false, or null if unset
|
|
||||||
*/
|
|
||||||
public Boolean getItalic() {
|
|
||||||
return italic;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether the text is italicized.
|
|
||||||
*
|
|
||||||
* @return true if italicized
|
|
||||||
*/
|
|
||||||
public boolean isItalic() {
|
|
||||||
return getItalic() != null && getItalic();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set whether the text is italicized.
|
|
||||||
*
|
|
||||||
* @param italic false, or null to unset
|
|
||||||
*/
|
|
||||||
public void setItalic(Boolean italic) {
|
|
||||||
this.italic = italic;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether this style set is underlined.
|
|
||||||
*
|
|
||||||
* @return true, false, or null if unset
|
|
||||||
*/
|
|
||||||
public Boolean getUnderline() {
|
|
||||||
return underline;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether the text is underlined.
|
|
||||||
*
|
|
||||||
* @return true if underlined
|
|
||||||
*/
|
|
||||||
public boolean isUnderline() {
|
|
||||||
return getUnderline() != null && getUnderline();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set whether the text is underline.
|
|
||||||
*
|
|
||||||
* @param underline false, or null to unset
|
|
||||||
*/
|
|
||||||
public void setUnderline(Boolean underline) {
|
|
||||||
this.underline = underline;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether this style set is stricken through.
|
|
||||||
*
|
|
||||||
* @return true, false, or null if unset
|
|
||||||
*/
|
|
||||||
public Boolean getStrikethrough() {
|
|
||||||
return strikethrough;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether the text is stricken through.
|
|
||||||
*
|
|
||||||
* @return true if there is strikethrough applied
|
|
||||||
*/
|
|
||||||
public boolean isStrikethrough() {
|
|
||||||
return getStrikethrough() != null && getStrikethrough();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set whether the text is stricken through.
|
|
||||||
*
|
|
||||||
* @param strikethrough false, or null to unset
|
|
||||||
*/
|
|
||||||
public void setStrikethrough(Boolean strikethrough) {
|
|
||||||
this.strikethrough = strikethrough;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether this style set is obfuscated.
|
|
||||||
*
|
|
||||||
* @return true, false, or null if unset
|
|
||||||
*/
|
|
||||||
public Boolean getObfuscated() {
|
|
||||||
return obfuscated;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether this style set is obfuscated.
|
|
||||||
*
|
|
||||||
* @return true if there is obfuscation applied
|
|
||||||
*/
|
|
||||||
public boolean isObfuscated() {
|
|
||||||
return getObfuscated() != null && getObfuscated();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set whether the text is obfuscated.
|
|
||||||
*
|
|
||||||
* @param obfuscated false, or null to unset
|
|
||||||
*/
|
|
||||||
public void setObfuscated(Boolean obfuscated) {
|
|
||||||
this.obfuscated = obfuscated;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get this style set's insertion, if present.
|
|
||||||
*
|
|
||||||
* @return the insertion, or null if unset
|
|
||||||
*/
|
|
||||||
public String getInsertion() {
|
|
||||||
return insertion;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get whether this style set has an insertion.
|
|
||||||
*
|
|
||||||
* @return true if there is an insertion
|
|
||||||
*/
|
|
||||||
public boolean hasInsertion() {
|
|
||||||
return insertion != null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the style set's insertion.
|
|
||||||
*
|
|
||||||
* @param insertion the insertion, or null to unset
|
|
||||||
*/
|
|
||||||
public void setInsertion(String insertion) {
|
|
||||||
this.insertion = insertion;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the color of the text.
|
|
||||||
*
|
|
||||||
* @return true, false, or null if unset
|
|
||||||
*/
|
|
||||||
public Style getColor() {
|
|
||||||
return color;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the color of the text.
|
|
||||||
*
|
|
||||||
* @param color the color
|
|
||||||
*/
|
|
||||||
public void setColor(Style color) {
|
|
||||||
this.color = color;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return whether text formatting (bold, italics, underline, strikethrough) is set.
|
|
||||||
*
|
|
||||||
* @return true if formatting is set
|
|
||||||
*/
|
|
||||||
public boolean hasFormatting() {
|
|
||||||
return getBold() != null || getItalic() != null
|
|
||||||
|| getUnderline() != null || getStrikethrough() != null
|
|
||||||
|| getObfuscated() != null || getInsertion() != null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Return where the text formatting of the given style set is different from
|
|
||||||
* that assigned to this one.
|
|
||||||
*
|
|
||||||
* @param other the other style set
|
|
||||||
* @return true if there is a difference
|
|
||||||
*/
|
|
||||||
public boolean hasEqualFormatting(StyleSet other) {
|
|
||||||
return getBold() == other.getBold() && getItalic() == other.getItalic()
|
|
||||||
&& getUnderline() == other.getUnderline() &&
|
|
||||||
getStrikethrough() == other.getStrikethrough() &&
|
|
||||||
getObfuscated() == other.getObfuscated() &&
|
|
||||||
Objects.equals(getInsertion(), other.getInsertion());
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new instance with styles inherited from this one but with new styles
|
|
||||||
* from the given style set.
|
|
||||||
*
|
|
||||||
* @param style the style set
|
|
||||||
* @return a new style set instance
|
|
||||||
*/
|
|
||||||
public StyleSet extend(StyleSet style) {
|
|
||||||
StyleSet newStyle = clone();
|
|
||||||
if (style.getBold() != null) {
|
|
||||||
newStyle.setBold(style.getBold());
|
|
||||||
}
|
|
||||||
if (style.getItalic() != null) {
|
|
||||||
newStyle.setItalic(style.getItalic());
|
|
||||||
}
|
|
||||||
if (style.getUnderline() != null) {
|
|
||||||
newStyle.setUnderline(style.getUnderline());
|
|
||||||
}
|
|
||||||
if (style.getStrikethrough() != null) {
|
|
||||||
newStyle.setStrikethrough(style.getStrikethrough());
|
|
||||||
}
|
|
||||||
if (style.getObfuscated() != null) {
|
|
||||||
newStyle.setObfuscated(style.getObfuscated());
|
|
||||||
}
|
|
||||||
if (style.getInsertion() != null) {
|
|
||||||
newStyle.setInsertion(style.getInsertion());
|
|
||||||
}
|
|
||||||
if (style.getColor() != null) {
|
|
||||||
newStyle.setColor(style.getColor());
|
|
||||||
}
|
|
||||||
return newStyle;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyleSet clone() {
|
|
||||||
StyleSet style = new StyleSet();
|
|
||||||
style.setBold(getBold());
|
|
||||||
style.setItalic(getItalic());
|
|
||||||
style.setUnderline(getUnderline());
|
|
||||||
style.setStrikethrough(getStrikethrough());
|
|
||||||
style.setObfuscated(getObfuscated());
|
|
||||||
style.setInsertion(getInsertion());
|
|
||||||
style.setColor(getColor());
|
|
||||||
return style;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -1,121 +0,0 @@
|
|||||||
/*
|
|
||||||
* WorldEdit, a Minecraft world manipulation toolkit
|
|
||||||
* Copyright (C) sk89q <http://www.sk89q.com>
|
|
||||||
* Copyright (C) WorldEdit team and contributors
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU Lesser General Public License as published by the
|
|
||||||
* Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) 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 Lesser General Public License
|
|
||||||
* for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package com.sk89q.worldedit.util.formatting;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A fragment of text that can be styled.
|
|
||||||
*/
|
|
||||||
public class StyledFragment extends Fragment {
|
|
||||||
|
|
||||||
private StyleSet style;
|
|
||||||
|
|
||||||
public StyledFragment() {
|
|
||||||
style = new StyleSet();
|
|
||||||
}
|
|
||||||
|
|
||||||
public StyledFragment(StyleSet style) {
|
|
||||||
this.style = style;
|
|
||||||
}
|
|
||||||
|
|
||||||
public StyledFragment(Style... styles) {
|
|
||||||
this.style = new StyleSet(styles);
|
|
||||||
}
|
|
||||||
|
|
||||||
public StyleSet getStyle() {
|
|
||||||
return style;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setStyles(StyleSet style) {
|
|
||||||
this.style = style;
|
|
||||||
}
|
|
||||||
|
|
||||||
public StyledFragment createFragment(Style... styles) {
|
|
||||||
StyledFragment fragment = new StyledFragment(styles);
|
|
||||||
append(fragment);
|
|
||||||
return fragment;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyledFragment append(String str) {
|
|
||||||
lastText().append(str);
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyledFragment append(Object obj) {
|
|
||||||
append(String.valueOf(obj));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyledFragment append(StringBuffer sb) {
|
|
||||||
append(String.valueOf(sb));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyledFragment append(CharSequence s) {
|
|
||||||
append(String.valueOf(s));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyledFragment append(boolean b) {
|
|
||||||
append(String.valueOf(b));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyledFragment append(char c) {
|
|
||||||
append(String.valueOf(c));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyledFragment append(int i) {
|
|
||||||
append(String.valueOf(i));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyledFragment append(long lng) {
|
|
||||||
append(String.valueOf(lng));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyledFragment append(float f) {
|
|
||||||
append(String.valueOf(f));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyledFragment append(double d) {
|
|
||||||
append(String.valueOf(d));
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public StyledFragment newLine() {
|
|
||||||
append("\n");
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -19,19 +19,19 @@
|
|||||||
|
|
||||||
package com.sk89q.worldedit.util.formatting.component;
|
package com.sk89q.worldedit.util.formatting.component;
|
||||||
|
|
||||||
import com.sk89q.worldedit.util.formatting.Style;
|
import net.kyori.text.TextComponent;
|
||||||
import com.sk89q.worldedit.util.formatting.StyledFragment;
|
import net.kyori.text.format.TextColor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents a fragment representing a command that is to be typed.
|
* Represents a fragment representing a command that is to be typed.
|
||||||
*/
|
*/
|
||||||
public class Code extends StyledFragment {
|
public class Code extends TextComponent {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new instance.
|
* Create a new instance.
|
||||||
*/
|
*/
|
||||||
public Code() {
|
public Code(String message) {
|
||||||
super(Style.CYAN);
|
super(builder(message).color(TextColor.AQUA));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -19,7 +19,9 @@
|
|||||||
|
|
||||||
package com.sk89q.worldedit.util.formatting.component;
|
package com.sk89q.worldedit.util.formatting.component;
|
||||||
|
|
||||||
import com.sk89q.worldedit.util.formatting.Style;
|
import net.kyori.text.Component;
|
||||||
|
import net.kyori.text.TextComponent;
|
||||||
|
import net.kyori.text.format.TextColor;
|
||||||
|
|
||||||
public class CommandListBox extends MessageBox {
|
public class CommandListBox extends MessageBox {
|
||||||
|
|
||||||
@ -36,10 +38,10 @@ public class CommandListBox extends MessageBox {
|
|||||||
|
|
||||||
public CommandListBox appendCommand(String alias, String description) {
|
public CommandListBox appendCommand(String alias, String description) {
|
||||||
if (!first) {
|
if (!first) {
|
||||||
getContents().newLine();
|
getContents().append(Component.newline());
|
||||||
}
|
}
|
||||||
getContents().createFragment(Style.YELLOW_DARK).append(alias).append(": ");
|
getContents().append(TextComponent.of(alias, TextColor.GOLD).append(TextComponent.of(": ")));
|
||||||
getContents().append(description);
|
getContents().append(TextComponent.of(description));
|
||||||
first = false;
|
first = false;
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
@ -28,7 +28,8 @@ import com.sk89q.worldedit.util.command.CommandMapping;
|
|||||||
import com.sk89q.worldedit.util.command.Description;
|
import com.sk89q.worldedit.util.command.Description;
|
||||||
import com.sk89q.worldedit.util.command.Dispatcher;
|
import com.sk89q.worldedit.util.command.Dispatcher;
|
||||||
import com.sk89q.worldedit.util.command.PrimaryAliasComparator;
|
import com.sk89q.worldedit.util.command.PrimaryAliasComparator;
|
||||||
import com.sk89q.worldedit.util.formatting.StyledFragment;
|
import net.kyori.text.Component;
|
||||||
|
import net.kyori.text.TextComponent;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@ -38,7 +39,7 @@ import javax.annotation.Nullable;
|
|||||||
/**
|
/**
|
||||||
* A box to describe usage of a command.
|
* A box to describe usage of a command.
|
||||||
*/
|
*/
|
||||||
public class CommandUsageBox extends StyledFragment {
|
public class CommandUsageBox extends TextComponent {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new usage box.
|
* Create a new usage box.
|
||||||
@ -58,6 +59,7 @@ public class CommandUsageBox extends StyledFragment {
|
|||||||
* @param locals list of locals to use
|
* @param locals list of locals to use
|
||||||
*/
|
*/
|
||||||
public CommandUsageBox(CommandCallable command, String commandString, @Nullable CommandLocals locals) {
|
public CommandUsageBox(CommandCallable command, String commandString, @Nullable CommandLocals locals) {
|
||||||
|
super(builder());
|
||||||
checkNotNull(command);
|
checkNotNull(command);
|
||||||
checkNotNull(commandString);
|
checkNotNull(commandString);
|
||||||
if (command instanceof Dispatcher) {
|
if (command instanceof Dispatcher) {
|
||||||
@ -85,23 +87,23 @@ public class CommandUsageBox extends StyledFragment {
|
|||||||
|
|
||||||
private void attachCommandUsage(Description description, String commandString) {
|
private void attachCommandUsage(Description description, String commandString) {
|
||||||
MessageBox box = new MessageBox("Help for " + commandString);
|
MessageBox box = new MessageBox("Help for " + commandString);
|
||||||
StyledFragment contents = box.getContents();
|
Component contents = box.getContents();
|
||||||
|
|
||||||
if (description.getUsage() != null) {
|
if (description.getUsage() != null) {
|
||||||
contents.append(new Label().append("Usage: "));
|
contents.append(new Label("Usage: "));
|
||||||
contents.append(description.getUsage());
|
contents.append(TextComponent.of(description.getUsage()));
|
||||||
} else {
|
} else {
|
||||||
contents.append(new Subtle().append("Usage information is not available."));
|
contents.append(new Subtle("Usage information is not available."));
|
||||||
}
|
}
|
||||||
|
|
||||||
contents.newLine();
|
contents.append(Component.newline());
|
||||||
|
|
||||||
if (description.getHelp() != null) {
|
if (description.getHelp() != null) {
|
||||||
contents.append(description.getHelp());
|
contents.append(TextComponent.of(description.getHelp()));
|
||||||
} else if (description.getDescription() != null) {
|
} else if (description.getDescription() != null) {
|
||||||
contents.append(description.getDescription());
|
contents.append(TextComponent.of(description.getDescription()));
|
||||||
} else {
|
} else {
|
||||||
contents.append(new Subtle().append("No further help is available."));
|
contents.append(new Subtle("No further help is available."));
|
||||||
}
|
}
|
||||||
|
|
||||||
append(box);
|
append(box);
|
||||||
|
@ -19,19 +19,19 @@
|
|||||||
|
|
||||||
package com.sk89q.worldedit.util.formatting.component;
|
package com.sk89q.worldedit.util.formatting.component;
|
||||||
|
|
||||||
import com.sk89q.worldedit.util.formatting.Style;
|
import net.kyori.text.TextComponent;
|
||||||
import com.sk89q.worldedit.util.formatting.StyledFragment;
|
import net.kyori.text.format.TextColor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents a fragment representing an error.
|
* Represents a fragment representing an error.
|
||||||
*/
|
*/
|
||||||
public class Error extends StyledFragment {
|
public class Error extends TextComponent {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new instance.
|
* Create a new instance.
|
||||||
*/
|
*/
|
||||||
public Error() {
|
public Error(String message) {
|
||||||
super(Style.RED);
|
super(builder(message).color(TextColor.RED));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -19,19 +19,19 @@
|
|||||||
|
|
||||||
package com.sk89q.worldedit.util.formatting.component;
|
package com.sk89q.worldedit.util.formatting.component;
|
||||||
|
|
||||||
import com.sk89q.worldedit.util.formatting.Style;
|
import net.kyori.text.TextComponent;
|
||||||
import com.sk89q.worldedit.util.formatting.StyledFragment;
|
import net.kyori.text.format.TextColor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents a fragment representing a label.
|
* Represents a fragment representing a label.
|
||||||
*/
|
*/
|
||||||
public class Label extends StyledFragment {
|
public class Label extends TextComponent {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new instance.
|
* Create a new instance.
|
||||||
*/
|
*/
|
||||||
public Label() {
|
public Label(String message) {
|
||||||
super(Style.YELLOW);
|
super(builder(message).color(TextColor.YELLOW));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -21,36 +21,39 @@ package com.sk89q.worldedit.util.formatting.component;
|
|||||||
|
|
||||||
import static com.google.common.base.Preconditions.checkNotNull;
|
import static com.google.common.base.Preconditions.checkNotNull;
|
||||||
|
|
||||||
import com.sk89q.worldedit.util.formatting.ColorCodeBuilder;
|
import net.kyori.text.Component;
|
||||||
import com.sk89q.worldedit.util.formatting.Style;
|
import net.kyori.text.TextComponent;
|
||||||
import com.sk89q.worldedit.util.formatting.StyledFragment;
|
import net.kyori.text.format.TextColor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Makes for a box with a border above and below.
|
* Makes for a box with a border above and below.
|
||||||
*/
|
*/
|
||||||
public class MessageBox extends StyledFragment {
|
public class MessageBox extends TextComponent {
|
||||||
|
|
||||||
private final StyledFragment contents = new StyledFragment();
|
public static final int GUARANTEED_NO_WRAP_CHAT_PAGE_WIDTH = 47;
|
||||||
|
|
||||||
|
private final Component contents = Component.empty();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new box.
|
* Create a new box.
|
||||||
*/
|
*/
|
||||||
public MessageBox(String title) {
|
public MessageBox(String title) {
|
||||||
|
super(builder());
|
||||||
checkNotNull(title);
|
checkNotNull(title);
|
||||||
|
|
||||||
int leftOver = ColorCodeBuilder.GUARANTEED_NO_WRAP_CHAT_PAGE_WIDTH - title.length() - 2;
|
int leftOver = GUARANTEED_NO_WRAP_CHAT_PAGE_WIDTH - title.length() - 2;
|
||||||
int leftSide = (int) Math.floor(leftOver * 1.0/3);
|
int leftSide = (int) Math.floor(leftOver * 1.0/3);
|
||||||
int rightSide = (int) Math.floor(leftOver * 2.0/3);
|
int rightSide = (int) Math.floor(leftOver * 2.0/3);
|
||||||
if (leftSide > 0) {
|
if (leftSide > 0) {
|
||||||
createFragment(Style.YELLOW).append(createBorder(leftSide));
|
append(TextComponent.of(createBorder(leftSide), TextColor.YELLOW));
|
||||||
}
|
}
|
||||||
append(" ");
|
append(Component.space());
|
||||||
append(title);
|
append(TextComponent.of(title));
|
||||||
append(" ");
|
append(Component.space());
|
||||||
if (rightSide > 0) {
|
if (rightSide > 0) {
|
||||||
createFragment(Style.YELLOW).append(createBorder(rightSide));
|
append(TextComponent.of(createBorder(rightSide), TextColor.YELLOW));
|
||||||
}
|
}
|
||||||
newLine();
|
append(Component.newline());
|
||||||
append(contents);
|
append(contents);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,7 +70,7 @@ public class MessageBox extends StyledFragment {
|
|||||||
*
|
*
|
||||||
* @return the contents
|
* @return the contents
|
||||||
*/
|
*/
|
||||||
public StyledFragment getContents() {
|
public Component getContents() {
|
||||||
return contents;
|
return contents;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,19 +19,19 @@
|
|||||||
|
|
||||||
package com.sk89q.worldedit.util.formatting.component;
|
package com.sk89q.worldedit.util.formatting.component;
|
||||||
|
|
||||||
import com.sk89q.worldedit.util.formatting.Style;
|
import net.kyori.text.TextComponent;
|
||||||
import com.sk89q.worldedit.util.formatting.StyledFragment;
|
import net.kyori.text.format.TextColor;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents a subtle part of the message.
|
* Represents a subtle part of the message.
|
||||||
*/
|
*/
|
||||||
public class Subtle extends StyledFragment {
|
public class Subtle extends TextComponent {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new instance.
|
* Create a new instance.
|
||||||
*/
|
*/
|
||||||
public Subtle() {
|
public Subtle(String message) {
|
||||||
super(Style.GRAY);
|
super(builder(message).color(TextColor.GRAY));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -91,6 +91,7 @@ shadowJar {
|
|||||||
include(dependency(':worldedit-core'))
|
include(dependency(':worldedit-core'))
|
||||||
include(dependency('org.slf4j:slf4j-api'))
|
include(dependency('org.slf4j:slf4j-api'))
|
||||||
include(dependency("org.apache.logging.log4j:log4j-slf4j-impl"))
|
include(dependency("org.apache.logging.log4j:log4j-slf4j-impl"))
|
||||||
|
include(dependency("net.kyori:text-serializer-gson:2.0.0"))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -43,6 +43,7 @@ shadowJar {
|
|||||||
dependencies {
|
dependencies {
|
||||||
include(dependency(':worldedit-core'))
|
include(dependency(':worldedit-core'))
|
||||||
include(dependency('org.bstats:bstats-sponge:1.4'))
|
include(dependency('org.bstats:bstats-sponge:1.4'))
|
||||||
|
include(dependency("net.kyori:text-adapter-spongeapi:1.0.3"))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Laden…
x
In neuem Issue referenzieren
Einen Benutzer sperren