mirror of
https://github.com/IntellectualSites/PlotSquared.git
synced 2025-04-06 19:46:23 +02:00
339 lines
14 KiB
Java
339 lines
14 KiB
Java
/*
|
|
* _____ _ _ _____ _
|
|
* | __ \| | | | / ____| | |
|
|
* | |__) | | ___ | |_| (___ __ _ _ _ __ _ _ __ ___ __| |
|
|
* | ___/| |/ _ \| __|\___ \ / _` | | | |/ _` | '__/ _ \/ _` |
|
|
* | | | | (_) | |_ ____) | (_| | |_| | (_| | | | __/ (_| |
|
|
* |_| |_|\___/ \__|_____/ \__, |\__,_|\__,_|_| \___|\__,_|
|
|
* | |
|
|
* |_|
|
|
* PlotSquared plot management system for Minecraft
|
|
* Copyright (C) 2020 IntellectualSites
|
|
*
|
|
* 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
|
|
* (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 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
package com.plotsquared.bukkit.chat;
|
|
|
|
import com.google.common.base.Preconditions;
|
|
import com.google.common.collect.ImmutableMap;
|
|
import com.google.gson.stream.JsonWriter;
|
|
import org.bukkit.configuration.serialization.ConfigurationSerializable;
|
|
import org.bukkit.configuration.serialization.ConfigurationSerialization;
|
|
|
|
import java.io.IOException;
|
|
import java.util.HashMap;
|
|
import java.util.Map;
|
|
|
|
/**
|
|
* Represents a textual component of a message part.
|
|
* This can be used to not only represent string literals in a JSON message,
|
|
* but also to represent localized strings and other text values.
|
|
* <p>Different instances of this class can be created with static constructor methods.</p>
|
|
*/
|
|
public abstract class TextualComponent implements Cloneable {
|
|
|
|
static {
|
|
ConfigurationSerialization.registerClass(TextualComponent.ArbitraryTextTypeComponent.class);
|
|
ConfigurationSerialization.registerClass(TextualComponent.ComplexTextTypeComponent.class);
|
|
}
|
|
|
|
static TextualComponent deserialize(Map<String, Object> map) {
|
|
if (map.containsKey("key") && map.size() == 2 && map.containsKey("value")) {
|
|
// Arbitrary text component
|
|
return ArbitraryTextTypeComponent.deserialize(map);
|
|
} else if (map.size() >= 2 && map.containsKey("key") && !map
|
|
.containsKey("value") /* It contains keys that START WITH value */) {
|
|
// Complex JSON object
|
|
return ComplexTextTypeComponent.deserialize(map);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
static boolean isTextKey(String key) {
|
|
return key.equals("translate") || key.equals("text") || key.equals("score") || key
|
|
.equals("selector");
|
|
}
|
|
|
|
static boolean isTranslatableText(TextualComponent component) {
|
|
return component instanceof ComplexTextTypeComponent && component.getKey()
|
|
.equals("translate");
|
|
}
|
|
|
|
/**
|
|
* Create a textual component representing a string literal.
|
|
*
|
|
* <p>This is the default type of textual component when a single string
|
|
* literal is given to a method.
|
|
*
|
|
* @param textValue The text which will be represented.
|
|
* @return The text component representing the specified literal text.
|
|
*/
|
|
public static TextualComponent rawText(String textValue) {
|
|
return new ArbitraryTextTypeComponent("text", textValue);
|
|
}
|
|
|
|
/**
|
|
* Create a textual component representing a localized string.
|
|
* The client will see this text component as their localized version of the specified string <em>key</em>, which can be overridden by a
|
|
* resource pack.
|
|
* <p>
|
|
* If the specified translation key is not present on the client resource pack, the translation key will be displayed as a string literal to
|
|
* the client.
|
|
* </p>
|
|
*
|
|
* @param translateKey The string key which maps to localized text.
|
|
* @return The text component representing the specified localized text.
|
|
*/
|
|
public static TextualComponent localizedText(String translateKey) {
|
|
return new ArbitraryTextTypeComponent("translate", translateKey);
|
|
}
|
|
|
|
private static void throwUnsupportedSnapshot() {
|
|
throw new UnsupportedOperationException(
|
|
"This feature is only supported in snapshot releases.");
|
|
}
|
|
|
|
/**
|
|
* Create a textual component representing a scoreboard value.
|
|
* The client will see their own score for the specified objective as the text represented by this component.
|
|
* <p>
|
|
* <b>This method is currently guaranteed to throw an {@code UnsupportedOperationException} as it is only supported on snapshot clients.</b>
|
|
* </p>
|
|
*
|
|
* @param scoreboardObjective The name of the objective for which to display the score.
|
|
* @return The text component representing the specified scoreboard score (for the viewing player), or {@code null} if an error occurs during
|
|
* JSON serialization.
|
|
*/
|
|
public static TextualComponent objectiveScore(String scoreboardObjective) {
|
|
return objectiveScore("*", scoreboardObjective);
|
|
}
|
|
|
|
/**
|
|
* Create a textual component representing a scoreboard value.
|
|
* The client will see the score of the specified player for the specified objective as the text represented by this component.
|
|
*
|
|
* <p><b>This method is currently guaranteed to throw an {@code UnsupportedOperationException}
|
|
* as it is only supported on snapshot clients.</b>
|
|
*
|
|
* @param playerName The name of the player whos score will be shown. If
|
|
* this string represents the single-character sequence
|
|
* "*", the viewing player's score will be displayed.
|
|
* Standard minecraft selectors (@a, @p, etc)
|
|
* are <em>not</em> supported.
|
|
* @param scoreboardObjective The name of the objective for
|
|
* which to display the score.
|
|
* @return The text component representing the specified scoreboard score
|
|
* for the specified player, or {@code null} if an error occurs during JSON serialization.
|
|
*/
|
|
public static TextualComponent objectiveScore(String playerName, String scoreboardObjective) {
|
|
throwUnsupportedSnapshot(); // Remove this line when the feature is released to non-snapshot versions, in addition to updating ALL THE
|
|
// OVERLOADS documentation accordingly
|
|
|
|
return new ComplexTextTypeComponent("score",
|
|
ImmutableMap.<String, String>builder().put("name", playerName)
|
|
.put("objective", scoreboardObjective).build());
|
|
}
|
|
|
|
/**
|
|
* Create a textual component representing a player name, retrievable by using a standard minecraft selector.
|
|
* The client will see the players or entities captured by the specified selector as the text represented by this component.
|
|
* <p>
|
|
* <b>This method is currently guaranteed to throw an {@code UnsupportedOperationException} as it is only supported on snapshot clients.</b>
|
|
* </p>
|
|
*
|
|
* @param selector The minecraft player or entity selector which will capture the entities whose string representations will be displayed in
|
|
* the place of this text component.
|
|
* @return The text component representing the name of the entities captured by the selector.
|
|
*/
|
|
public static TextualComponent selector(String selector) {
|
|
throwUnsupportedSnapshot(); // Remove this line when the feature is released to non-snapshot versions, in addition to updating ALL THE
|
|
// OVERLOADS documentation accordingly
|
|
|
|
return new ArbitraryTextTypeComponent("selector", selector);
|
|
}
|
|
|
|
@Override public String toString() {
|
|
return getReadableString();
|
|
}
|
|
|
|
/**
|
|
* @return The JSON key used to represent text components of this type.
|
|
*/
|
|
public abstract String getKey();
|
|
|
|
/**
|
|
* @return A readable String
|
|
*/
|
|
public abstract String getReadableString();
|
|
|
|
/**
|
|
* Clones a textual component instance.
|
|
* The returned object should not reference this textual component instance, but should maintain the same key and value.
|
|
*/
|
|
@Override public abstract TextualComponent clone() throws CloneNotSupportedException;
|
|
|
|
/**
|
|
* Writes the text data represented by this textual component to the specified JSON writer object.
|
|
* A new object within the writer is not started.
|
|
*
|
|
* @param writer The object to which to write the JSON data.
|
|
* @throws IOException If an error occurs while writing to the stream.
|
|
*/
|
|
public abstract void writeJson(JsonWriter writer) throws IOException;
|
|
|
|
/**
|
|
* Internal class used to represent all types of text components.
|
|
* Exception validating done is on keys and values.
|
|
*/
|
|
private static final class ArbitraryTextTypeComponent extends TextualComponent
|
|
implements ConfigurationSerializable {
|
|
|
|
private String key;
|
|
private String value;
|
|
|
|
public ArbitraryTextTypeComponent(String key, String value) {
|
|
setKey(key);
|
|
setValue(value);
|
|
}
|
|
|
|
public static ArbitraryTextTypeComponent deserialize(Map<String, Object> map) {
|
|
return new ArbitraryTextTypeComponent(map.get("key").toString(),
|
|
map.get("value").toString());
|
|
}
|
|
|
|
@Override public String getKey() {
|
|
return key;
|
|
}
|
|
|
|
public void setKey(String key) {
|
|
Preconditions
|
|
.checkArgument(key != null && !key.isEmpty(), "The key must be specified.");
|
|
this.key = key;
|
|
}
|
|
|
|
public String getValue() {
|
|
return value;
|
|
}
|
|
|
|
public void setValue(String value) {
|
|
Preconditions.checkArgument(value != null, "The value must be specified.");
|
|
this.value = value;
|
|
}
|
|
|
|
@Override public TextualComponent clone() throws CloneNotSupportedException {
|
|
// Since this is a private and final class, we can just reinstantiate this class instead of casting super.clone
|
|
return new ArbitraryTextTypeComponent(getKey(), getValue());
|
|
}
|
|
|
|
@Override public void writeJson(JsonWriter writer) throws IOException {
|
|
writer.name(getKey()).value(getValue());
|
|
}
|
|
|
|
@Override @SuppressWarnings("serial") public Map<String, Object> serialize() {
|
|
return new HashMap<String, Object>() {
|
|
{
|
|
put("key", getKey());
|
|
put("value", getValue());
|
|
}
|
|
};
|
|
}
|
|
|
|
@Override public String getReadableString() {
|
|
return getValue();
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Internal class used to represent a text component with a nested JSON
|
|
* value.
|
|
*
|
|
* <p>Exception validating done is on keys and values.
|
|
*/
|
|
private static final class ComplexTextTypeComponent extends TextualComponent
|
|
implements ConfigurationSerializable {
|
|
|
|
private String key;
|
|
private Map<String, String> value;
|
|
|
|
public ComplexTextTypeComponent(String key, Map<String, String> values) {
|
|
setKey(key);
|
|
setValue(values);
|
|
}
|
|
|
|
public static ComplexTextTypeComponent deserialize(Map<String, Object> map) {
|
|
String key = null;
|
|
Map<String, String> value = new HashMap<>();
|
|
for (Map.Entry<String, Object> valEntry : map.entrySet()) {
|
|
if (valEntry.getKey().equals("key")) {
|
|
key = (String) valEntry.getValue();
|
|
} else if (valEntry.getKey().startsWith("value.")) {
|
|
value.put(valEntry.getKey().substring(6) /* Strips out the value prefix */,
|
|
valEntry.getValue().toString());
|
|
}
|
|
}
|
|
return new ComplexTextTypeComponent(key, value);
|
|
}
|
|
|
|
@Override public String getKey() {
|
|
return key;
|
|
}
|
|
|
|
public void setKey(String key) {
|
|
Preconditions
|
|
.checkArgument(key != null && !key.isEmpty(), "The key must be specified.");
|
|
this.key = key;
|
|
}
|
|
|
|
public Map<String, String> getValue() {
|
|
return value;
|
|
}
|
|
|
|
public void setValue(Map<String, String> value) {
|
|
Preconditions.checkArgument(value != null, "The value must be specified.");
|
|
this.value = value;
|
|
}
|
|
|
|
@Override public TextualComponent clone() {
|
|
// Since this is a private and final class, we can just reinstantiate this class instead of casting super.clone
|
|
return new ComplexTextTypeComponent(getKey(), getValue());
|
|
}
|
|
|
|
@Override public void writeJson(JsonWriter writer) throws IOException {
|
|
writer.name(getKey());
|
|
writer.beginObject();
|
|
for (Map.Entry<String, String> jsonPair : value.entrySet()) {
|
|
writer.name(jsonPair.getKey()).value(jsonPair.getValue());
|
|
}
|
|
writer.endObject();
|
|
}
|
|
|
|
@Override @SuppressWarnings("serial") public Map<String, Object> serialize() {
|
|
return new java.util.HashMap<String, Object>() {
|
|
{
|
|
put("key", getKey());
|
|
for (Map.Entry<String, String> valEntry : getValue().entrySet()) {
|
|
put("value." + valEntry.getKey(), valEntry.getValue());
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
@Override public String getReadableString() {
|
|
return getKey();
|
|
}
|
|
}
|
|
}
|