1
1
package io .papermc .paper .adventure ;
2
2
3
- import com .google .gson .JsonElement ;
4
- import com .google .gson .JsonParser ;
5
- import com .mojang .brigadier .exceptions .CommandSyntaxException ;
6
3
import com .mojang .datafixers .util .Either ;
7
- import com .mojang .datafixers .util .Pair ;
8
4
import com .mojang .serialization .Codec ;
9
5
import com .mojang .serialization .DataResult ;
10
- import com .mojang .serialization .DynamicOps ;
11
- import com .mojang .serialization .JsonOps ;
12
6
import com .mojang .serialization .MapCodec ;
13
- import com .mojang .serialization .codecs .RecordCodecBuilder ;
14
7
import java .util .Collections ;
15
8
import java .util .List ;
16
9
import java .util .Map ;
17
10
import java .util .Optional ;
18
- import java .util .UUID ;
19
11
import java .util .function .Consumer ;
20
12
import java .util .function .Function ;
21
13
import java .util .function .Predicate ;
14
+ import com .mojang .serialization .codecs .RecordCodecBuilder ;
22
15
import net .kyori .adventure .key .Key ;
23
16
import net .kyori .adventure .text .BlockNBTComponent ;
24
17
import net .kyori .adventure .text .Component ;
40
33
import net .kyori .adventure .text .format .Style ;
41
34
import net .kyori .adventure .text .format .TextColor ;
42
35
import net .kyori .adventure .text .format .TextDecoration ;
43
- import net .kyori .adventure .text .serializer .plain .PlainTextComponentSerializer ;
44
36
import net .minecraft .commands .arguments .selector .SelectorPattern ;
45
37
import net .minecraft .core .UUIDUtil ;
46
38
import net .minecraft .core .registries .BuiltInRegistries ;
47
- import net .minecraft .nbt .CompoundTag ;
48
- import net .minecraft .nbt .NbtOps ;
49
- import net .minecraft .nbt .Tag ;
50
- import net .minecraft .nbt .TagParser ;
51
39
import net .minecraft .network .RegistryFriendlyByteBuf ;
52
40
import net .minecraft .network .chat .ComponentSerialization ;
53
41
import net .minecraft .network .chat .contents .KeybindContents ;
54
42
import net .minecraft .network .chat .contents .ScoreContents ;
55
43
import net .minecraft .network .chat .contents .TranslatableContents ;
56
44
import net .minecraft .network .codec .ByteBufCodecs ;
57
45
import net .minecraft .network .codec .StreamCodec ;
58
- import net .minecraft .resources .RegistryOps ;
59
46
import net .minecraft .util .ExtraCodecs ;
60
47
import net .minecraft .util .StringRepresentable ;
61
48
import net .minecraft .world .item .Item ;
76
63
@ DefaultQualifier (NonNull .class )
77
64
public final class AdventureCodecs {
78
65
79
- public static final Codec <Component > COMPONENT_CODEC = recursive ("adventure Component" , AdventureCodecs ::createCodec );
66
+ public static final Codec <Component > COMPONENT_CODEC = recursive ("adventure Component" , AdventureCodecs ::createCodec );
80
67
public static final StreamCodec <RegistryFriendlyByteBuf , Component > STREAM_COMPONENT_CODEC = ByteBufCodecs .fromCodecWithRegistriesTrusted (COMPONENT_CODEC );
81
68
82
69
static final Codec <ShadowColor > SHADOW_COLOR_CODEC = ExtraCodecs .ARGB_COLOR_CODEC .xmap (ShadowColor ::shadowColor , ShadowColor ::value );
@@ -101,102 +88,92 @@ public final class AdventureCodecs {
101
88
return Key .parseable (s ) ? DataResult .success (Key .key (s )) : DataResult .error (() -> "Cannot convert " + s + " to adventure Key" );
102
89
}, Key ::asString );
103
90
104
- static final Codec <ClickEvent .Action > CLICK_EVENT_ACTION_CODEC = Codec .STRING .comapFlatMap (s -> {
105
- final ClickEvent .@ Nullable Action value = ClickEvent .Action .NAMES .value (s );
106
- return value != null ? DataResult .success (value ) : DataResult .error (() -> "Cannot convert " + s + " to adventure ClickEvent$Action" );
107
- }, ClickEvent .Action .NAMES ::keyOrThrow );
108
- static final Codec <ClickEvent > CLICK_EVENT_CODEC = RecordCodecBuilder .create ((instance ) -> {
109
- return instance .group (
110
- CLICK_EVENT_ACTION_CODEC .fieldOf ("action" ).forGetter (ClickEvent ::action ),
111
- Codec .STRING .fieldOf ("value" ).forGetter (ClickEvent ::value )
112
- ).apply (instance , ClickEvent ::clickEvent );
113
- });
114
-
115
- static Codec <HoverEvent .ShowEntity > showEntityCodec (final Codec <Component > componentCodec ) {
116
- return RecordCodecBuilder .create ((instance ) -> {
117
- return instance .group (
118
- KEY_CODEC .fieldOf ("type" ).forGetter (HoverEvent .ShowEntity ::type ),
119
- UUIDUtil .LENIENT_CODEC .fieldOf ("id" ).forGetter (HoverEvent .ShowEntity ::id ),
120
- componentCodec .lenientOptionalFieldOf ("name" ).forGetter (he -> Optional .ofNullable (he .name ()))
121
- ).apply (instance , (key , uuid , component ) -> {
122
- return HoverEvent .ShowEntity .showEntity (key , uuid , component .orElse (null ));
123
- });
124
- });
125
- }
126
-
127
- static Codec <HoverEvent .ShowItem > showItemCodec (final Codec <Component > componentCodec ) {
128
- return net .minecraft .network .chat .HoverEvent .ShowItem .CODEC .xmap (internal -> {
129
- @ Subst ("key" ) final String typeKey = internal .item ().getItemHolder ().unwrapKey ().orElseThrow ().location ().toString ();
130
- return HoverEvent .ShowItem .showItem (Key .key (typeKey ), internal .item ().getCount (), PaperAdventure .asAdventure (internal .item ().getComponentsPatch ()));
131
- }, adventure -> {
132
- final Item itemType = BuiltInRegistries .ITEM .getValue (PaperAdventure .asVanilla (adventure .item ()));
133
- final Map <Key , DataComponentValue > dataComponentsMap = adventure .dataComponents ();
134
- final ItemStack stack = new ItemStack (BuiltInRegistries .ITEM .wrapAsHolder (itemType ), adventure .count (), PaperAdventure .asVanilla (dataComponentsMap ));
135
- return new net .minecraft .network .chat .HoverEvent .ShowItem (stack );
136
- }).codec ();
137
- }
138
-
139
- static final HoverEventType <HoverEvent .ShowEntity > SHOW_ENTITY_HOVER_EVENT_TYPE = new HoverEventType <>(AdventureCodecs ::showEntityCodec , HoverEvent .Action .SHOW_ENTITY , "show_entity" , AdventureCodecs ::legacyDeserializeEntity );
140
- static final HoverEventType <HoverEvent .ShowItem > SHOW_ITEM_HOVER_EVENT_TYPE = new HoverEventType <>(AdventureCodecs ::showItemCodec , HoverEvent .Action .SHOW_ITEM , "show_item" , AdventureCodecs ::legacyDeserializeItem );
141
- static final HoverEventType <Component > SHOW_TEXT_HOVER_EVENT_TYPE = new HoverEventType <>(identity (), HoverEvent .Action .SHOW_TEXT , "show_text" , (component , registryOps , codec ) -> DataResult .success (component ));
142
- static final Codec <HoverEventType <?>> HOVER_EVENT_TYPE_CODEC = StringRepresentable .fromValues (() -> new HoverEventType <?>[]{ SHOW_ENTITY_HOVER_EVENT_TYPE , SHOW_ITEM_HOVER_EVENT_TYPE , SHOW_TEXT_HOVER_EVENT_TYPE });
143
-
144
- static DataResult <HoverEvent .ShowEntity > legacyDeserializeEntity (final Component component , final @ Nullable RegistryOps <?> ops , final Codec <Component > componentCodec ) {
145
- try {
146
- final CompoundTag tag = TagParser .parseCompoundFully (PlainTextComponentSerializer .plainText ().serialize (component ));
147
- final DynamicOps <JsonElement > dynamicOps = ops != null ? ops .withParent (JsonOps .INSTANCE ) : JsonOps .INSTANCE ;
148
- final DataResult <Component > entityNameResult = componentCodec .parse (dynamicOps , JsonParser .parseString (tag .getStringOr ("name" , "" )));
149
- @ Subst ("key" ) final String keyString = tag .getStringOr ("type" , "" );
150
- final UUID entityUUID = UUID .fromString (tag .getStringOr ("id" , "" ));
151
- return entityNameResult .map (name -> HoverEvent .ShowEntity .showEntity (Key .key (keyString ), entityUUID , name ));
152
- } catch (final Exception ex ) {
153
- return DataResult .error (() -> "Failed to parse tooltip: " + ex .getMessage ());
91
+ /*
92
+ * Click
93
+ */
94
+ static final MapCodec <ClickEvent > OPEN_URL_CODEC = mapCodec ((instance ) -> instance .group (
95
+ Codec .STRING .fieldOf ("url" ).forGetter (a -> !a .value ().contains ("://" ) ? "https://" + a .value () : a .value ())
96
+ ).apply (instance , ClickEvent ::openUrl ));
97
+ static final MapCodec <ClickEvent > OPEN_FILE_CODEC = mapCodec ((instance ) -> instance .group (
98
+ Codec .STRING .fieldOf ("path" ).forGetter (ClickEvent ::value )
99
+ ).apply (instance , ClickEvent ::openFile ));
100
+ static final MapCodec <ClickEvent > RUN_COMMAND_CODEC = mapCodec ((instance ) -> instance .group (
101
+ Codec .STRING .fieldOf ("command" ).forGetter (ClickEvent ::value )
102
+ ).apply (instance , ClickEvent ::runCommand ));
103
+ static final MapCodec <ClickEvent > SUGGEST_COMMAND_CODEC = mapCodec ((instance ) -> instance .group (
104
+ Codec .STRING .fieldOf ("command" ).forGetter (ClickEvent ::value )
105
+ ).apply (instance , ClickEvent ::suggestCommand ));
106
+ static final MapCodec <ClickEvent > CHANGE_PAGE_CODEC = mapCodec ((instance ) -> instance .group (
107
+ ExtraCodecs .POSITIVE_INT .fieldOf ("page" ).forGetter (a -> Integer .parseInt (a .value ()))
108
+ ).apply (instance , ClickEvent ::changePage ));
109
+ static final MapCodec <ClickEvent > COPY_TO_CLIPBOARD_CODEC = mapCodec ((instance ) -> instance .group (
110
+ Codec .STRING .fieldOf ("value" ).forGetter (ClickEvent ::value )
111
+ ).apply (instance , ClickEvent ::copyToClipboard ));
112
+
113
+ static final ClickEventType OPEN_URL_CLICK_EVENT_TYPE = new ClickEventType (OPEN_URL_CODEC , "open_url" );
114
+ static final ClickEventType OPEN_FILE_CLICK_EVENT_TYPE = new ClickEventType (OPEN_FILE_CODEC , "open_file" );
115
+ static final ClickEventType RUN_COMMAND_CLICK_EVENT_TYPE = new ClickEventType (RUN_COMMAND_CODEC , "run_command" );
116
+ static final ClickEventType SUGGEST_COMMAND_CLICK_EVENT_TYPE = new ClickEventType (SUGGEST_COMMAND_CODEC , "suggest_command" );
117
+ static final ClickEventType CHANGE_PAGE_CLICK_EVENT_TYPE = new ClickEventType (CHANGE_PAGE_CODEC , "change_page" );
118
+ static final ClickEventType COPY_TO_CLIPBOARD_CLICK_EVENT_TYPE = new ClickEventType (COPY_TO_CLIPBOARD_CODEC , "copy_to_clipboard" );
119
+ static final Codec <ClickEventType > CLICK_EVENT_TYPE_CODEC = StringRepresentable .fromValues (() -> new ClickEventType []{OPEN_URL_CLICK_EVENT_TYPE , OPEN_FILE_CLICK_EVENT_TYPE , RUN_COMMAND_CLICK_EVENT_TYPE , SUGGEST_COMMAND_CLICK_EVENT_TYPE , CHANGE_PAGE_CLICK_EVENT_TYPE , COPY_TO_CLIPBOARD_CLICK_EVENT_TYPE });
120
+
121
+ record ClickEventType (MapCodec <ClickEvent > codec , String id ) implements StringRepresentable {
122
+ @ Override
123
+ public String getSerializedName () {
124
+ return this .id ;
154
125
}
155
126
}
156
127
157
- static DataResult <HoverEvent .ShowItem > legacyDeserializeItem (final Component component , final @ Nullable RegistryOps <?> ops , final Codec <Component > componentCodec ) {
158
- try {
159
- final CompoundTag tag = TagParser .parseCompoundFully (PlainTextComponentSerializer .plainText ().serialize (component ));
160
- final DynamicOps <Tag > dynamicOps = ops != null ? ops .withParent (NbtOps .INSTANCE ) : NbtOps .INSTANCE ;
161
- final DataResult <ItemStack > stackResult = ItemStack .CODEC .parse (dynamicOps , tag );
162
- return stackResult .map (stack -> {
163
- @ Subst ("key:value" ) final String location = stack .getItemHolder ().unwrapKey ().orElseThrow ().location ().toString ();
164
- return HoverEvent .ShowItem .showItem (Key .key (location ), stack .getCount (), PaperAdventure .asAdventure (stack .getComponentsPatch ()));
165
- });
166
- } catch (final CommandSyntaxException ex ) {
167
- return DataResult .error (() -> "Failed to parse item tag: " + ex .getMessage ());
128
+ private static final Function <ClickEvent , ClickEventType > GET_CLICK_EVENT_TYPE = he -> {
129
+ if (he .action () == ClickEvent .Action .OPEN_URL ) {
130
+ return OPEN_URL_CLICK_EVENT_TYPE ;
131
+ } else if (he .action () == ClickEvent .Action .OPEN_FILE ) {
132
+ return OPEN_FILE_CLICK_EVENT_TYPE ;
133
+ } else if (he .action () == ClickEvent .Action .RUN_COMMAND ) {
134
+ return RUN_COMMAND_CLICK_EVENT_TYPE ;
135
+ } else if (he .action () == ClickEvent .Action .SUGGEST_COMMAND ) {
136
+ return SUGGEST_COMMAND_CLICK_EVENT_TYPE ;
137
+ } else if (he .action () == ClickEvent .Action .CHANGE_PAGE ) {
138
+ return CHANGE_PAGE_CLICK_EVENT_TYPE ;
139
+ } else if (he .action () == ClickEvent .Action .COPY_TO_CLIPBOARD ) {
140
+ return COPY_TO_CLIPBOARD_CLICK_EVENT_TYPE ;
141
+ } else {
142
+ throw new IllegalStateException ();
168
143
}
169
- }
144
+ };
170
145
171
- @ FunctionalInterface
172
- interface LegacyDeserializer <T > {
173
- DataResult <T > apply (Component component , @ Nullable RegistryOps <?> ops , Codec <Component > componentCodec );
174
- }
146
+ static final Codec <ClickEvent > CLICK_EVENT_CODEC = CLICK_EVENT_TYPE_CODEC .dispatch ("action" , GET_CLICK_EVENT_TYPE , ClickEventType ::codec );
147
+
148
+ /*
149
+ * HOVER
150
+ */
151
+ static final MapCodec <HoverEvent <Component >> SHOW_TEXT_CODEC = mapCodec ((instance ) -> instance .group (
152
+ COMPONENT_CODEC .fieldOf ("value" ).forGetter (HoverEvent ::value )
153
+ ).apply (instance , HoverEvent ::showText ));
154
+
155
+ static final MapCodec <HoverEvent <HoverEvent .ShowEntity >> SHOW_ENTITY_CODEC = mapCodec ((instance ) -> instance .group (
156
+ KEY_CODEC .fieldOf ("id" ).forGetter (a -> a .value ().type ()),
157
+ UUIDUtil .LENIENT_CODEC .fieldOf ("uuid" ).forGetter (a -> a .value ().id ()),
158
+ COMPONENT_CODEC .lenientOptionalFieldOf ("name" ).forGetter (a -> Optional .ofNullable (a .value ().name ()))
159
+ ).apply (instance , (key , uuid , component ) -> HoverEvent .showEntity (key , uuid , component .orElse (null ))));
160
+
161
+ static final MapCodec <HoverEvent <HoverEvent .ShowItem >> SHOW_ITEM_CODEC = net .minecraft .network .chat .HoverEvent .ShowItem .CODEC .xmap (internal -> {
162
+ @ Subst ("key" ) final String typeKey = internal .item ().getItemHolder ().unwrapKey ().orElseThrow ().location ().toString ();
163
+ return HoverEvent .showItem (Key .key (typeKey ), internal .item ().getCount (), PaperAdventure .asAdventure (internal .item ().getComponentsPatch ()));
164
+ }, adventure -> {
165
+ final Item itemType = BuiltInRegistries .ITEM .getValue (PaperAdventure .asVanilla (adventure .value ().item ()));
166
+ final Map <Key , DataComponentValue > dataComponentsMap = adventure .value ().dataComponents ();
167
+ final ItemStack stack = new ItemStack (BuiltInRegistries .ITEM .wrapAsHolder (itemType ), adventure .value ().count (), PaperAdventure .asVanilla (dataComponentsMap ));
168
+ return new net .minecraft .network .chat .HoverEvent .ShowItem (stack );
169
+ });
175
170
176
- record HoverEventType <V >(Function <Codec <Component >, MapCodec <HoverEvent <V >>> codec , String id , Function <Codec <Component >, MapCodec <HoverEvent <V >>> legacyCodec ) implements StringRepresentable {
177
- HoverEventType (final Function <Codec <Component >, Codec <V >> contentCodec , final HoverEvent .Action <V > action , final String id , final LegacyDeserializer <V > legacyDeserializer ) {
178
- this (cc -> contentCodec .apply (cc ).xmap (v -> HoverEvent .hoverEvent (action , v ), HoverEvent ::value ).fieldOf ("contents" ),
179
- id ,
180
- codec -> (new Codec <HoverEvent <V >>() {
181
- public <D > DataResult <Pair <HoverEvent <V >, D >> decode (final DynamicOps <D > dynamicOps , final D object ) {
182
- return codec .decode (dynamicOps , object ).flatMap (pair -> {
183
- final DataResult <V > dataResult ;
184
- if (dynamicOps instanceof final RegistryOps <D > registryOps ) {
185
- dataResult = legacyDeserializer .apply (pair .getFirst (), registryOps , codec );
186
- } else {
187
- dataResult = legacyDeserializer .apply (pair .getFirst (), null , codec );
188
- }
189
-
190
- return dataResult .map (value -> Pair .of (HoverEvent .hoverEvent (action , value ), pair .getSecond ()));
191
- });
192
- }
193
-
194
- public <D > DataResult <D > encode (final HoverEvent <V > hoverEvent , final DynamicOps <D > dynamicOps , final D object ) {
195
- return DataResult .error (() -> "Can't encode in legacy format" );
196
- }
197
- }).fieldOf ("value" )
198
- );
199
- }
171
+ static final HoverEventType <HoverEvent .ShowEntity > SHOW_ENTITY_HOVER_EVENT_TYPE = new HoverEventType <>(SHOW_ENTITY_CODEC , "show_entity" );
172
+ static final HoverEventType <HoverEvent .ShowItem > SHOW_ITEM_HOVER_EVENT_TYPE = new HoverEventType <>(SHOW_ITEM_CODEC , "show_item" );
173
+ static final HoverEventType <Component > SHOW_TEXT_HOVER_EVENT_TYPE = new HoverEventType <>(SHOW_TEXT_CODEC , "show_text" );
174
+ static final Codec <HoverEventType <?>> HOVER_EVENT_TYPE_CODEC = StringRepresentable .fromValues (() -> new HoverEventType <?>[]{SHOW_ENTITY_HOVER_EVENT_TYPE , SHOW_ITEM_HOVER_EVENT_TYPE , SHOW_TEXT_HOVER_EVENT_TYPE });
175
+
176
+ record HoverEventType <V >(MapCodec <HoverEvent <V >> codec , String id ) implements StringRepresentable {
200
177
@ Override
201
178
public String getSerializedName () {
202
179
return this .id ;
@@ -214,11 +191,12 @@ public String getSerializedName() {
214
191
throw new IllegalStateException ();
215
192
}
216
193
};
217
- static final Codec <HoverEvent <?>> HOVER_EVENT_CODEC = Codec .withAlternative (
218
- HOVER_EVENT_TYPE_CODEC .<HoverEvent <?>>dispatchMap ("action" , GET_HOVER_EVENT_TYPE , het -> het .codec .apply (COMPONENT_CODEC )).codec (),
219
- HOVER_EVENT_TYPE_CODEC .<HoverEvent <?>>dispatchMap ("action" , GET_HOVER_EVENT_TYPE , het -> het .legacyCodec .apply (COMPONENT_CODEC )).codec ()
220
- );
221
194
195
+ static final Codec <HoverEvent <?>> HOVER_EVENT_CODEC = HOVER_EVENT_TYPE_CODEC .dispatch ("action" , GET_HOVER_EVENT_TYPE , HoverEventType ::codec );
196
+
197
+ /*
198
+ * Style
199
+ */
222
200
public static final MapCodec <Style > STYLE_MAP_CODEC = mapCodec ((instance ) -> {
223
201
return instance .group (
224
202
TEXT_COLOR_CODEC .optionalFieldOf ("color" ).forGetter (nullableGetter (Style ::color )),
@@ -228,8 +206,8 @@ public String getSerializedName() {
228
206
Codec .BOOL .optionalFieldOf ("underlined" ).forGetter (decorationGetter (TextDecoration .UNDERLINED )),
229
207
Codec .BOOL .optionalFieldOf ("strikethrough" ).forGetter (decorationGetter (TextDecoration .STRIKETHROUGH )),
230
208
Codec .BOOL .optionalFieldOf ("obfuscated" ).forGetter (decorationGetter (TextDecoration .OBFUSCATED )),
231
- CLICK_EVENT_CODEC .optionalFieldOf ("clickEvent " ).forGetter (nullableGetter (Style ::clickEvent )),
232
- HOVER_EVENT_CODEC .optionalFieldOf ("hoverEvent " ).forGetter (nullableGetter (Style ::hoverEvent )),
209
+ CLICK_EVENT_CODEC .optionalFieldOf ("click_event " ).forGetter (nullableGetter (Style ::clickEvent )),
210
+ HOVER_EVENT_CODEC .optionalFieldOf ("hover_event " ).forGetter (nullableGetter (Style ::hoverEvent )),
233
211
Codec .STRING .optionalFieldOf ("insertion" ).forGetter (nullableGetter (Style ::insertion )),
234
212
KEY_CODEC .optionalFieldOf ("font" ).forGetter (nullableGetter (Style ::font ))
235
213
).apply (instance , (textColor , shadowColor , bold , italic , underlined , strikethrough , obfuscated , clickEvent , hoverEvent , insertion , font ) -> {
@@ -248,6 +226,10 @@ public String getSerializedName() {
248
226
});
249
227
});
250
228
});
229
+
230
+ /*
231
+ * Misc
232
+ */
251
233
static Consumer <Boolean > styleBooleanConsumer (final Style .Builder builder , final TextDecoration decoration ) {
252
234
return b -> builder .decoration (decoration , b );
253
235
}
0 commit comments