Fix equippable slot reading and add support for all consume animations

This commit is contained in:
Eclipse 2024-11-29 15:51:22 +00:00
parent ea6c3c67f0
commit 0922b181f2
No known key found for this signature in database
GPG key ID: 95E6998F82EC938A
2 changed files with 40 additions and 20 deletions

View file

@ -34,7 +34,15 @@ import org.geysermc.mcprotocollib.protocol.data.game.item.component.DataComponen
import org.geysermc.mcprotocollib.protocol.data.game.item.component.Equippable; import org.geysermc.mcprotocollib.protocol.data.game.item.component.Equippable;
import org.geysermc.mcprotocollib.protocol.data.game.level.sound.BuiltinSound; import org.geysermc.mcprotocollib.protocol.data.game.level.sound.BuiltinSound;
import java.util.Map;
public class EquippableReader extends DataComponentReader<Equippable> { public class EquippableReader extends DataComponentReader<Equippable> {
private static final Map<String, EquipmentSlot> SLOTS = Map.of(
"head", EquipmentSlot.HELMET,
"chest", EquipmentSlot.CHESTPLATE,
"legs", EquipmentSlot.LEGGINGS,
"feet", EquipmentSlot.BOOTS
);
public EquippableReader() { public EquippableReader() {
super(DataComponentType.EQUIPPABLE); super(DataComponentType.EQUIPPABLE);
@ -44,12 +52,16 @@ public class EquippableReader extends DataComponentReader<Equippable> {
protected Equippable readDataComponent(@NonNull JsonNode node) throws InvalidCustomMappingsFileException { protected Equippable readDataComponent(@NonNull JsonNode node) throws InvalidCustomMappingsFileException {
requireObject(node); requireObject(node);
JsonNode slot = node.get("slot"); JsonNode slotNode = node.get("slot");
if (slot == null || !slot.isTextual()) { if (slotNode == null) {
throw new InvalidCustomMappingsFileException("Expected slot to be helmet, chestplate, leggings or boots"); throw new InvalidCustomMappingsFileException("Expected slot to be present");
}
EquipmentSlot slot = SLOTS.get(slotNode.asText());
if (slot == null) {
throw new InvalidCustomMappingsFileException("Expected slot to be head, chest, legs or feet");
} }
return new Equippable(EquipmentSlot.valueOf(slot.asText().toUpperCase()), BuiltinSound.ITEM_ARMOR_EQUIP_GENERIC, return new Equippable(slot, BuiltinSound.ITEM_ARMOR_EQUIP_GENERIC,
null, null, null, false, false, false); // Other properties are unused null, null, null, false, false, false); // Other properties are unused
} }
} }

View file

@ -26,7 +26,6 @@
package org.geysermc.geyser.registry.populator; package org.geysermc.geyser.registry.populator;
import com.google.common.collect.Multimap; import com.google.common.collect.Multimap;
import net.kyori.adventure.key.Key;
import org.checkerframework.checker.nullness.qual.Nullable; import org.checkerframework.checker.nullness.qual.Nullable;
import org.cloudburstmc.nbt.NbtMap; import org.cloudburstmc.nbt.NbtMap;
import org.cloudburstmc.nbt.NbtMapBuilder; import org.cloudburstmc.nbt.NbtMapBuilder;
@ -61,6 +60,18 @@ import java.util.Map;
import java.util.Set; import java.util.Set;
public class CustomItemRegistryPopulator_v2 { public class CustomItemRegistryPopulator_v2 {
// In behaviour packs and Java components this is set to a text value, such as "eat" or "drink"; over Bedrock network it's sent as an int.
private static final Map<Consumable.ItemUseAnimation, Integer> BEDROCK_ANIMATIONS = Map.of(
Consumable.ItemUseAnimation.NONE, 0,
Consumable.ItemUseAnimation.EAT, 1,
Consumable.ItemUseAnimation.DRINK, 2,
Consumable.ItemUseAnimation.BLOCK, 3,
Consumable.ItemUseAnimation.BOW, 4,
Consumable.ItemUseAnimation.SPEAR, 6,
Consumable.ItemUseAnimation.CROSSBOW, 9,
Consumable.ItemUseAnimation.SPYGLASS, 10,
Consumable.ItemUseAnimation.BRUSH, 12
);
public static void populate(Map<String, GeyserMappingItem> items, Multimap<String, CustomItemDefinition> customItems, List<NonVanillaCustomItemData> nonVanillaCustomItems /* TODO */) { public static void populate(Map<String, GeyserMappingItem> items, Multimap<String, CustomItemDefinition> customItems, List<NonVanillaCustomItemData> nonVanillaCustomItems /* TODO */) {
// TODO // TODO
@ -256,34 +267,31 @@ public class CustomItemRegistryPopulator_v2 {
case BOOTS -> { case BOOTS -> {
componentBuilder.putString("minecraft:render_offsets", "boots"); componentBuilder.putString("minecraft:render_offsets", "boots");
componentBuilder.putCompound("minecraft:wearable", WearableSlot.FEET.getSlotNbt()); componentBuilder.putCompound("minecraft:wearable", WearableSlot.FEET.getSlotNbt());
componentBuilder.putCompound("minecraft:armor", NbtMap.builder().putInt("protection", protectionValue).build());
itemProperties.putString("enchantable_slot", "armor_feet"); //itemProperties.putString("enchantable_slot", "armor_feet");
itemProperties.putInt("enchantable_value", 15); //itemProperties.putInt("enchantable_value", 15); TODO
} }
case CHESTPLATE -> { case CHESTPLATE -> {
componentBuilder.putString("minecraft:render_offsets", "chestplates"); componentBuilder.putString("minecraft:render_offsets", "chestplates");
componentBuilder.putCompound("minecraft:wearable", WearableSlot.CHEST.getSlotNbt()); componentBuilder.putCompound("minecraft:wearable", WearableSlot.CHEST.getSlotNbt());
componentBuilder.putCompound("minecraft:armor", NbtMap.builder().putInt("protection", protectionValue).build());
itemProperties.putString("enchantable_slot", "armor_torso"); //itemProperties.putString("enchantable_slot", "armor_torso");
itemProperties.putInt("enchantable_value", 15); //itemProperties.putInt("enchantable_value", 15); TODO
} }
case LEGGINGS -> { case LEGGINGS -> {
componentBuilder.putString("minecraft:render_offsets", "leggings"); componentBuilder.putString("minecraft:render_offsets", "leggings");
componentBuilder.putCompound("minecraft:wearable", WearableSlot.LEGS.getSlotNbt()); componentBuilder.putCompound("minecraft:wearable", WearableSlot.LEGS.getSlotNbt());
componentBuilder.putCompound("minecraft:armor", NbtMap.builder().putInt("protection", protectionValue).build());
itemProperties.putString("enchantable_slot", "armor_legs"); //itemProperties.putString("enchantable_slot", "armor_legs");
itemProperties.putInt("enchantable_value", 15); //itemProperties.putInt("enchantable_value", 15); TODO
} }
case HELMET -> { case HELMET -> {
componentBuilder.putString("minecraft:render_offsets", "helmets"); componentBuilder.putString("minecraft:render_offsets", "helmets");
componentBuilder.putCompound("minecraft:wearable", WearableSlot.HEAD.getSlotNbt()); componentBuilder.putCompound("minecraft:wearable", WearableSlot.HEAD.getSlotNbt());
componentBuilder.putCompound("minecraft:armor", NbtMap.builder().putInt("protection", protectionValue).build()); //componentBuilder.putCompound("minecraft:armor", NbtMap.builder().putInt("protection", protectionValue).build());
itemProperties.putString("enchantable_slot", "armor_head"); //itemProperties.putString("enchantable_slot", "armor_head");
itemProperties.putInt("enchantable_value", 15); //itemProperties.putInt("enchantable_value", 15);
} }
} }
} }
@ -291,9 +299,9 @@ public class CustomItemRegistryPopulator_v2 {
private static void computeConsumableProperties(Consumable consumable, boolean canAlwaysEat, NbtMapBuilder itemProperties, NbtMapBuilder componentBuilder) { private static void computeConsumableProperties(Consumable consumable, boolean canAlwaysEat, NbtMapBuilder itemProperties, NbtMapBuilder componentBuilder) {
// this is the duration of the use animation in ticks; note that in behavior packs this is set as a float in seconds, but over the network it is an int in ticks // this is the duration of the use animation in ticks; note that in behavior packs this is set as a float in seconds, but over the network it is an int in ticks
itemProperties.putInt("use_duration", (int) (consumable.consumeSeconds() * 20)); // TODO check and confirm itemProperties.putInt("use_duration", (int) (consumable.consumeSeconds() * 20)); // TODO check and confirm
// this dictates that the item will use the eat or drink animation (in the first person) and play eat or drink sounds
// note that in behavior packs this is set as the string "eat" or "drink", but over the network it as an int, with these values being 1 and 2 respectively itemProperties.putInt("use_animation", BEDROCK_ANIMATIONS.get(consumable.animation()));
itemProperties.putInt("use_animation", 0); // TODO
// this component is required to allow the eat animation to play // this component is required to allow the eat animation to play
componentBuilder.putCompound("minecraft:food", NbtMap.builder().putBoolean("can_always_eat", canAlwaysEat).build()); componentBuilder.putCompound("minecraft:food", NbtMap.builder().putBoolean("can_always_eat", canAlwaysEat).build());
} }