mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-17 06:48:23 +01:00
Improve legacy format serialization more
This should now complete legacy serialization to avoid ever changing the output content. This removes the concept of "Default Color" from the method as that entire concept was flawed and broke the intent of chat components. Going to actually PR this patch to Spigot soon. This now puts us back at a point where any data saved pre Spigot breaking things will still save back the exact same way as before, but new component -> legacy will now be fixed to not insert undesirable default colors (such as black) into the legacy string, and instead use the proper reset code. This means you can now safety get the text from a book and put it in chat or an entity display name without worry about black color codes or other undesired color codes leaking into the new context where that color doesn't make sense.
This commit is contained in:
parent
7ae7528cfb
commit
4c4b8f4d3e
1 changed files with 52 additions and 36 deletions
|
@ -1,6 +1,6 @@
|
||||||
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
||||||
From: wea_ondara <wea_ondara@alpenblock.net>
|
From: Aikar <aikar@aikar.co>
|
||||||
Date: Sat, 2 Nov 2019 22:25:40 +0100
|
Date: Sun, 31 May 2020 21:55:52 -0400
|
||||||
Subject: [PATCH] Fix serialization of colors from components
|
Subject: [PATCH] Fix serialization of colors from components
|
||||||
|
|
||||||
This patch fixes the serialization of display names, item lores and
|
This patch fixes the serialization of display names, item lores and
|
||||||
|
@ -13,6 +13,9 @@ saving an ItemStack in a Yaml config).
|
||||||
|
|
||||||
Spigot has now made the issue worse and expanded the scope to more places.
|
Spigot has now made the issue worse and expanded the scope to more places.
|
||||||
|
|
||||||
|
Original work by (but impl pretty much fully changed):
|
||||||
|
Co-Authored-By: wea_ondara <wea_ondara@alpenblock.net>
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/util/CraftChatMessage.java b/src/main/java/org/bukkit/craftbukkit/util/CraftChatMessage.java
|
diff --git a/src/main/java/org/bukkit/craftbukkit/util/CraftChatMessage.java b/src/main/java/org/bukkit/craftbukkit/util/CraftChatMessage.java
|
||||||
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/util/CraftChatMessage.java
|
--- a/src/main/java/org/bukkit/craftbukkit/util/CraftChatMessage.java
|
||||||
|
@ -26,28 +29,36 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
||||||
} else if (format.isFormat()) {
|
} else if (format.isFormat()) {
|
||||||
switch (format) {
|
switch (format) {
|
||||||
case BOLD:
|
case BOLD:
|
||||||
|
@@ -0,0 +0,0 @@ public final class CraftChatMessage {
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
if (keepNewlines) {
|
||||||
|
- currentChatComponent.addSibling(new ChatComponentText("\n"));
|
||||||
|
+ // Paper start - retain formatting for new lines
|
||||||
|
+ ChatComponentText ichatbasecomponent = new ChatComponentText("\n");
|
||||||
|
+ ichatbasecomponent.setChatModifier(modifier.clone());
|
||||||
|
+ currentChatComponent.addSibling(ichatbasecomponent);
|
||||||
|
+ // Paper end
|
||||||
|
} else {
|
||||||
|
currentChatComponent = null;
|
||||||
|
}
|
||||||
@@ -0,0 +0,0 @@ public final class CraftChatMessage {
|
@@ -0,0 +0,0 @@ public final class CraftChatMessage {
|
||||||
if (component == null) return "";
|
if (component == null) return "";
|
||||||
StringBuilder out = new StringBuilder();
|
StringBuilder out = new StringBuilder();
|
||||||
|
|
||||||
+ // Paper start - fix deletion of color codes at the beginning of result string
|
+ // Paper start - fix deletion of color codes at the beginning of result string
|
||||||
+ boolean first = true;
|
+ boolean hadColor = false;
|
||||||
for (IChatBaseComponent c : (Iterable<IChatBaseComponent>) component) {
|
for (IChatBaseComponent c : (Iterable<IChatBaseComponent>) component) {
|
||||||
ChatModifier modi = c.getChatModifier();
|
ChatModifier modi = c.getChatModifier();
|
||||||
- out.append(modi.getColor() == null ? defaultColor : modi.getColor());
|
- out.append(modi.getColor() == null ? defaultColor : modi.getColor());
|
||||||
+ EnumChatFormat color = modi.getColor();
|
+ EnumChatFormat color = modi.getColor();
|
||||||
+ if (first) {
|
+ if (!c.getText().isEmpty() || color != null) {
|
||||||
+ if (color != null) {
|
+ if (color != null) {
|
||||||
+ out.append(color);
|
+ out.append(color);
|
||||||
+ }
|
+ hadColor = true;
|
||||||
+ if (!c.getText().isEmpty() || color != null) {
|
+ } else if (hadColor) {
|
||||||
+ first = false;
|
+ out.append(ChatColor.RESET);
|
||||||
+ }
|
+ hadColor = false;
|
||||||
+ } else if (!c.getText().isEmpty() || color != null) {
|
|
||||||
+ if (color != null) {
|
|
||||||
+ out.append(color);
|
|
||||||
+ } else if (defaultColor != null) {
|
|
||||||
+ out.append(defaultColor);
|
|
||||||
+ }
|
+ }
|
||||||
+ }
|
+ }
|
||||||
+ // Paper end
|
+ // Paper end
|
||||||
|
@ -71,9 +82,7 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
||||||
@@ -0,0 +0,0 @@
|
@@ -0,0 +0,0 @@
|
||||||
+package org.bukkit.craftbukkit;
|
+package org.bukkit.craftbukkit;
|
||||||
+
|
+
|
||||||
+import net.minecraft.server.EnumChatFormat;
|
|
||||||
+import net.minecraft.server.IChatBaseComponent;
|
+import net.minecraft.server.IChatBaseComponent;
|
||||||
+import org.bukkit.ChatColor;
|
|
||||||
+import org.bukkit.craftbukkit.util.CraftChatMessage;
|
+import org.bukkit.craftbukkit.util.CraftChatMessage;
|
||||||
+import org.bukkit.support.AbstractTestingBase;
|
+import org.bukkit.support.AbstractTestingBase;
|
||||||
+import org.junit.Test;
|
+import org.junit.Test;
|
||||||
|
@ -83,21 +92,25 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
||||||
+public class CraftChatMessageTest extends AbstractTestingBase {
|
+public class CraftChatMessageTest extends AbstractTestingBase {
|
||||||
+ @Test
|
+ @Test
|
||||||
+ public void testSimpleStrings() {
|
+ public void testSimpleStrings() {
|
||||||
+ testString("&fFoo", EnumChatFormat.WHITE);
|
+ testString("§fFoo");
|
||||||
+ testString("Foo", EnumChatFormat.WHITE);
|
+ testString("Foo");
|
||||||
+ testString("Foo&bBar", EnumChatFormat.WHITE);
|
+ testString("Foo§bBar");
|
||||||
+ testString("Foo&bBar", EnumChatFormat.AQUA);
|
+ testString("Foo§bBar");
|
||||||
+ testString("&fFoo&bBar", EnumChatFormat.WHITE);
|
+ testString("§fFoo§bBar");
|
||||||
+ testString("&rFoo", EnumChatFormat.WHITE);
|
+ testString("§fFoo§bBar§rBaz");
|
||||||
|
+ testString("§rFoo");
|
||||||
|
+ testString("Test§0\n§0\n§5Test");
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ @Test
|
+ @Test
|
||||||
+ public void testComponents() {
|
+ public void testComponents() {
|
||||||
+ testComponent("Foo&bBar&fBaz", EnumChatFormat.WHITE, create("Foo", "&bBar", "Baz"));
|
+ testComponent("Foo§bBar§rBaz", create("Foo", "§bBar", "Baz"));
|
||||||
+ testComponent("&fFoo&bBar&fBaz", EnumChatFormat.WHITE, create("", "&fFoo", "&bBar", "Baz"));
|
+ testComponent("§fFoo§bBar§rBaz", create("", "§fFoo", "§bBar", "Baz"));
|
||||||
+ testComponent("Foo&bBar&fBaz", EnumChatFormat.WHITE, create("", "Foo", "&bBar", "Baz"));
|
+ testComponent("§fFoo§bBar§rBaz", create("", "§fFoo", "§bBar", "", "Baz"));
|
||||||
+ testComponent("&fFoo&bBar&fBaz", EnumChatFormat.WHITE, create("&fFoo", "&bBar", "Baz"));
|
+ testComponent("§fFoo§bBar§rBaz", create("§fFoo", "§bBar", "Baz"));
|
||||||
+ testComponent("F&foo&bBar&fBaz", EnumChatFormat.WHITE, create("F&foo", "&bBar", "Baz"));
|
+ testComponent("Foo§bBar§rBaz", create("", "Foo", "§bBar", "Baz"));
|
||||||
|
+ testComponent("§fFoo§bBar§rBaz", create("§fFoo", "§bBar", "Baz"));
|
||||||
|
+ testComponent("F§foo§bBar§rBaz", create("F§foo", "§bBar", "Baz"));
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ private IChatBaseComponent create(String txt, String ...rest) {
|
+ private IChatBaseComponent create(String txt, String ...rest) {
|
||||||
|
@ -110,18 +123,21 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ private IChatBaseComponent toComp(String txt) {
|
+ private IChatBaseComponent toComp(String txt) {
|
||||||
+ return CraftChatMessage.fromString(ChatColor.translateAlternateColorCodes('&', txt))[0];
|
+ return CraftChatMessage.fromString(txt, true)[0];
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ private void testString(String expected, EnumChatFormat defColor) {
|
+ private void testString(String expected) {
|
||||||
+ expected = ChatColor.translateAlternateColorCodes('&', expected);
|
+ IChatBaseComponent cmp = toComp(expected);
|
||||||
+ IChatBaseComponent cmp = CraftChatMessage.fromStringOrNull(expected);
|
+ String actual = CraftChatMessage.fromComponent(cmp);
|
||||||
+ testComponent(expected, defColor, cmp);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private void testComponent(String expected, EnumChatFormat defColor, IChatBaseComponent components) {
|
|
||||||
+ expected = ChatColor.translateAlternateColorCodes('&', expected);
|
|
||||||
+ String actual = CraftChatMessage.fromComponent(components, defColor);
|
|
||||||
+ assertEquals(expected, actual);
|
+ assertEquals(expected, actual);
|
||||||
+ }
|
+ }
|
||||||
|
+
|
||||||
|
+ private void testComponent(String expected, IChatBaseComponent components) {
|
||||||
|
+ String actual = CraftChatMessage.fromComponent(components);
|
||||||
|
+ assertEquals(expected, actual);
|
||||||
|
+
|
||||||
|
+ IChatBaseComponent expectedCmp = toComp(expected);
|
||||||
|
+ String actualExpectedCmp = CraftChatMessage.fromComponent(expectedCmp);
|
||||||
|
+ assertEquals(expected, actualExpectedCmp);
|
||||||
|
+ }
|
||||||
+}
|
+}
|
||||||
|
|
Loading…
Reference in a new issue