Mirror von
https://github.com/GeyserMC/Geyser.git
synchronisiert 2024-11-20 06:50:09 +01:00
Fix uppercase item attribute modifier names (#3780)
* Check for hide attributes flag, and "Name" -> "AttributeName" * Operation tag is not required? * Only process each modifier once * Ignore `minecraft:` namespace if present * No `Operation` is implicitly ADD, fix knockback_resistance check
Dieser Commit ist enthalten in:
Ursprung
178fb2136f
Commit
ba4e37075d
@ -32,12 +32,14 @@ import com.github.steveice10.opennbt.tag.builtin.*;
|
|||||||
import net.kyori.adventure.text.Component;
|
import net.kyori.adventure.text.Component;
|
||||||
import net.kyori.adventure.text.format.NamedTextColor;
|
import net.kyori.adventure.text.format.NamedTextColor;
|
||||||
import org.checkerframework.checker.nullness.qual.NonNull;
|
import org.checkerframework.checker.nullness.qual.NonNull;
|
||||||
|
import org.checkerframework.checker.nullness.qual.Nullable;
|
||||||
import org.cloudburstmc.nbt.NbtList;
|
import org.cloudburstmc.nbt.NbtList;
|
||||||
import org.cloudburstmc.nbt.NbtMap;
|
import org.cloudburstmc.nbt.NbtMap;
|
||||||
import org.cloudburstmc.nbt.NbtMapBuilder;
|
import org.cloudburstmc.nbt.NbtMapBuilder;
|
||||||
import org.cloudburstmc.nbt.NbtType;
|
import org.cloudburstmc.nbt.NbtType;
|
||||||
import org.cloudburstmc.protocol.bedrock.data.defintions.ItemDefinition;
|
import org.cloudburstmc.protocol.bedrock.data.defintions.ItemDefinition;
|
||||||
import org.cloudburstmc.protocol.bedrock.data.inventory.ItemData;
|
import org.cloudburstmc.protocol.bedrock.data.inventory.ItemData;
|
||||||
|
import org.geysermc.geyser.GeyserImpl;
|
||||||
import org.geysermc.geyser.inventory.GeyserItemStack;
|
import org.geysermc.geyser.inventory.GeyserItemStack;
|
||||||
import org.geysermc.geyser.item.type.Item;
|
import org.geysermc.geyser.item.type.Item;
|
||||||
import org.geysermc.geyser.registry.BlockRegistries;
|
import org.geysermc.geyser.registry.BlockRegistries;
|
||||||
@ -54,6 +56,14 @@ import java.text.DecimalFormat;
|
|||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
|
||||||
public final class ItemTranslator {
|
public final class ItemTranslator {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The order of these slots is their display order on Java Edition clients
|
||||||
|
*/
|
||||||
|
private static final String[] ALL_SLOTS = new String[]{"mainhand", "offhand", "feet", "legs", "chest", "head"};
|
||||||
|
private static final DecimalFormat ATTRIBUTE_FORMAT = new DecimalFormat("0.#####");
|
||||||
|
private static final byte HIDE_ATTRIBUTES_FLAG = 1 << 1;
|
||||||
|
|
||||||
private ItemTranslator() {
|
private ItemTranslator() {
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -118,7 +128,11 @@ public final class ItemTranslator {
|
|||||||
nbt = translateDisplayProperties(session, nbt, bedrockItem);
|
nbt = translateDisplayProperties(session, nbt, bedrockItem);
|
||||||
|
|
||||||
if (nbt != null) {
|
if (nbt != null) {
|
||||||
addAttributes(nbt, javaItem, session.locale());
|
Tag hideFlags = nbt.get("HideFlags");
|
||||||
|
if (hideFlags == null || !hasFlagPresent(hideFlags, HIDE_ATTRIBUTES_FLAG)) {
|
||||||
|
// only add if the hide attribute modifiers flag is not present
|
||||||
|
addAttributeLore(nbt, session.locale());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (session.isAdvancedTooltips()) {
|
if (session.isAdvancedTooltips()) {
|
||||||
@ -149,79 +163,107 @@ public final class ItemTranslator {
|
|||||||
return builder;
|
return builder;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static CompoundTag addAttributes(CompoundTag nbt, Item item, String language) {
|
/**
|
||||||
ListTag modifiers = nbt.get("AttributeModifiers");
|
* Bedrock Edition does not see attribute modifiers like Java Edition does,
|
||||||
if (modifiers == null) return nbt;
|
* so we add them as lore instead.
|
||||||
CompoundTag newNbt = nbt;
|
*
|
||||||
if (newNbt == null) {
|
* @param nbt the NBT of the ItemStack
|
||||||
newNbt = new CompoundTag("nbt");
|
* @param language the locale of the player
|
||||||
CompoundTag display = new CompoundTag("display");
|
*/
|
||||||
display.put(new ListTag("Lore"));
|
private static void addAttributeLore(CompoundTag nbt, String language) {
|
||||||
newNbt.put(display);
|
ListTag attributeModifiers = nbt.get("AttributeModifiers");
|
||||||
|
if (attributeModifiers == null) {
|
||||||
|
return; // nothing to convert to lore
|
||||||
}
|
}
|
||||||
CompoundTag compoundTag = newNbt.get("display");
|
|
||||||
if (compoundTag == null) {
|
|
||||||
compoundTag = new CompoundTag("display");
|
|
||||||
}
|
|
||||||
ListTag listTag = compoundTag.get("Lore");
|
|
||||||
|
|
||||||
if (listTag == null) {
|
CompoundTag displayTag = nbt.get("display");
|
||||||
listTag = new ListTag("Lore");
|
if (displayTag == null) {
|
||||||
|
displayTag = new CompoundTag("display");
|
||||||
}
|
}
|
||||||
String[] allSlots = new String[]{"mainhand", "offhand", "feet", "legs", "chest", "head"};
|
ListTag lore = displayTag.get("Lore");
|
||||||
DecimalFormat decimalFormat = new DecimalFormat("0.#####");
|
if (lore == null) {
|
||||||
Map<String, List<Tag>> slotsToModifiers = new HashMap<>();
|
lore = new ListTag("Lore");
|
||||||
for (String slot : allSlots) {
|
|
||||||
slotsToModifiers.put(slot, new ArrayList<>());
|
|
||||||
}
|
}
|
||||||
for (Tag modifier : modifiers) {
|
|
||||||
Map<String, Tag> modifierValue = (Map) modifier.getValue();
|
// maps each slot to the modifiers applied when in such slot
|
||||||
String[] slots = allSlots;
|
Map<String, List<StringTag>> slotsToModifiers = new HashMap<>();
|
||||||
if (modifierValue.get("Slot") != null) {
|
for (Tag modifier : attributeModifiers) {
|
||||||
slots = new String[]{(String) modifierValue.get("Slot").getValue()};
|
CompoundTag modifierTag = (CompoundTag) modifier;
|
||||||
|
|
||||||
|
// convert the modifier tag to a lore entry
|
||||||
|
String loreEntry = attributeToLore(modifierTag, language);
|
||||||
|
if (loreEntry == null) {
|
||||||
|
continue; // invalid or failed
|
||||||
}
|
}
|
||||||
for (String slot : slots) {
|
|
||||||
List<Tag> list = slotsToModifiers.get(slot);
|
StringTag loreTag = new StringTag("", loreEntry);
|
||||||
list.add(modifier);
|
StringTag slotTag = modifierTag.get("Slot");
|
||||||
slotsToModifiers.put(slot, list);
|
if (slotTag == null) {
|
||||||
|
// modifier applies to all slots implicitly
|
||||||
|
for (String slot : ALL_SLOTS) {
|
||||||
|
slotsToModifiers.computeIfAbsent(slot, s -> new ArrayList<>()).add(loreTag);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// modifier applies to only the specified slot
|
||||||
|
slotsToModifiers.computeIfAbsent(slotTag.getValue(), s -> new ArrayList<>()).add(loreTag);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (String slot : allSlots) {
|
// iterate through the small array, not the map, so that ordering matches Java Edition
|
||||||
List<Tag> modifiersList = slotsToModifiers.get(slot);
|
for (String slot : ALL_SLOTS) {
|
||||||
if (modifiersList.isEmpty()) continue;
|
List<StringTag> modifiers = slotsToModifiers.get(slot);
|
||||||
|
if (modifiers == null || modifiers.isEmpty()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Declare the slot, e.g. "When in Main Hand"
|
||||||
Component slotComponent = Component.text()
|
Component slotComponent = Component.text()
|
||||||
.resetStyle()
|
.resetStyle()
|
||||||
.color(NamedTextColor.GRAY)
|
.color(NamedTextColor.GRAY)
|
||||||
.append(Component.newline(), Component.translatable("item.modifiers." + slot))
|
.append(Component.newline(), Component.translatable("item.modifiers." + slot))
|
||||||
.build();
|
.build();
|
||||||
listTag.add(new StringTag("", MessageTranslator.convertMessage(slotComponent, language)));
|
lore.add(new StringTag("", MessageTranslator.convertMessage(slotComponent, language)));
|
||||||
|
|
||||||
|
// Then list all the modifiers when used in this slot
|
||||||
for (Tag modifier : modifiersList) {
|
for (StringTag modifier : modifiers) {
|
||||||
Map<String, Tag> modifierValue = (Map) modifier.getValue();
|
lore.add(modifier);
|
||||||
double amount;
|
|
||||||
if (modifierValue.get("Amount") instanceof IntTag intTag) {
|
|
||||||
amount = (double) intTag.getValue();
|
|
||||||
} else if (modifierValue.get("Amount") instanceof DoubleTag doubleTag) {
|
|
||||||
amount = doubleTag.getValue();
|
|
||||||
} else {
|
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
displayTag.put(lore);
|
||||||
|
nbt.put(displayTag);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
private static String attributeToLore(CompoundTag modifier, String language) {
|
||||||
|
Tag amountTag = modifier.get("Amount");
|
||||||
|
if (amountTag == null || !(amountTag.getValue() instanceof Number number)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
double amount = number.doubleValue();
|
||||||
if (amount == 0) {
|
if (amount == 0) {
|
||||||
continue;
|
return null;
|
||||||
}
|
}
|
||||||
ModifierOperation operation = ModifierOperation.from((int) modifierValue.get("Operation").getValue());
|
|
||||||
|
if (!(modifier.get("AttributeName") instanceof StringTag nameTag)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
String name = nameTag.getValue().replace("minecraft:", "");
|
||||||
|
// the namespace does not need to be present, but if it is, the java client ignores it
|
||||||
|
|
||||||
String operationTotal;
|
String operationTotal;
|
||||||
if (operation == ModifierOperation.ADD) {
|
Tag operationTag = modifier.get("Operation");
|
||||||
if (modifierValue.get("Name").equals("knockback_resistance")) {
|
ModifierOperation operation;
|
||||||
|
if (operationTag == null || (operation = ModifierOperation.from((int) operationTag.getValue())) == ModifierOperation.ADD) {
|
||||||
|
if (name.equals("generic.knockback_resistance")) {
|
||||||
amount *= 10;
|
amount *= 10;
|
||||||
}
|
}
|
||||||
operationTotal = decimalFormat.format(amount);
|
operationTotal = ATTRIBUTE_FORMAT.format(amount);
|
||||||
} else if (operation == ModifierOperation.ADD_MULTIPLIED || operation == ModifierOperation.MULTIPLY) {
|
} else if (operation == ModifierOperation.ADD_MULTIPLIED || operation == ModifierOperation.MULTIPLY) {
|
||||||
operationTotal = decimalFormat.format(amount * 100) + "%";
|
operationTotal = ATTRIBUTE_FORMAT.format(amount * 100) + "%";
|
||||||
} else {
|
} else {
|
||||||
continue;
|
GeyserImpl.getInstance().getLogger().warning("Unhandled ModifierOperation while adding item attributes: " + operation);
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
if (amount > 0) {
|
if (amount > 0) {
|
||||||
operationTotal = "+" + operationTotal;
|
operationTotal = "+" + operationTotal;
|
||||||
@ -230,16 +272,10 @@ public final class ItemTranslator {
|
|||||||
Component attributeComponent = Component.text()
|
Component attributeComponent = Component.text()
|
||||||
.resetStyle()
|
.resetStyle()
|
||||||
.color(amount > 0 ? NamedTextColor.BLUE : NamedTextColor.RED)
|
.color(amount > 0 ? NamedTextColor.BLUE : NamedTextColor.RED)
|
||||||
.append(Component.text(operationTotal), Component.text(" "), Component.translatable("attribute.name." + modifierValue.get("Name").getValue()))
|
.append(Component.text(operationTotal + " "), Component.translatable("attribute.name." + name))
|
||||||
.build();
|
.build();
|
||||||
listTag.add(new StringTag("", MessageTranslator.convertMessage(attributeComponent, language)));
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
return MessageTranslator.convertMessage(attributeComponent, language);
|
||||||
|
|
||||||
compoundTag.put(listTag);
|
|
||||||
newNbt.put(compoundTag);
|
|
||||||
return newNbt;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static CompoundTag addAdvancedTooltips(CompoundTag nbt, Item item, String language) {
|
private static CompoundTag addAdvancedTooltips(CompoundTag nbt, Item item, String language) {
|
||||||
@ -519,4 +555,18 @@ public final class ItemTranslator {
|
|||||||
builder.definition(definition);
|
builder.definition(definition);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the NBT of a Java item stack has the given hide flag.
|
||||||
|
*
|
||||||
|
* @param hideFlags the "HideFlags", which may not be null
|
||||||
|
* @param flagMask the flag to check for, as a bit mask
|
||||||
|
* @return true if the flag is present, false if not or if the tag value is not a number
|
||||||
|
*/
|
||||||
|
private static boolean hasFlagPresent(Tag hideFlags, byte flagMask) {
|
||||||
|
if (hideFlags.getValue() instanceof Number flags) {
|
||||||
|
return (flags.byteValue() & flagMask) == flagMask;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Laden…
In neuem Issue referenzieren
Einen Benutzer sperren