Table of Contents

Добавление брони

Вступление

Хотя броня немного сложнее в реализации, чем обычный блок или предмет, как только вы поймете ее логику, броню станет проще реализовать. Чтобы добавить броню, мы сначала создадим класс CustomArmorMaterial, а затем зарегистрируем предметы. Мы также рассмотрим, как их текстурировать. В конце этого документа есть специальная глава, в которой объясняется, как добавить сопротивление откидыванию к броне, поскольку этот метод доступен только через миксин (начиная с 1.16.3).

Пример этого документа можно найти в репозитории мода в GitHub'е.

Создание класса материалов брони

Поскольку новая броня должна быть установлена с новым именем (а также дополнительными функциями, такими как очки брони и прочность), нам придется создать новый класс для нашего CustomArmorMaterial.

Этот класс будет реализовывать ArmorMaterial, и он начнет с присвоения значений точкам брони (называемых PROTECTION_VALUES). Все его следующие аргументы будут использовать @Override.

  1. public class CustomArmorMaterial implements ArmorMaterial {
  2. private static final int[] BASE_DURABILITY = new int[] {13, 15, 16, 11};
  3. private static final int[] PROTECTION_VALUES = new int[] {A, B, C, D};
  4.  
  5. // Здесь A это шлем, B - нагрудник, C - поножи и D - ботинки.
  6. // Для справки, кожа использует {1, 2, 3, 1}, а алмаз/незерит {3, 6, 8, 3}
  7. }

Следующие аргументы определяются таким образом (не беспокойтесь об именах, вы увидите, как мы это реализуем под ними):

И новое значение, введенное на 1.16

Я оставлю все переменные записанными как X или A, B, C, D. С этими аргументами теперь это должно выглядеть примерно так:

  1. public class CustomArmorMaterial implements ArmorMaterial {
  2. private static final int[] BASE_DURABILITY = new int[] {13, 15, 16, 11};
  3. private static final int[] PROTECTION_VALUES = new int[] {A, B, C, D};
  4.  
  5. @Override
  6. public int getDurability(EquipmentSlot slot) {
  7. return BASE_DURABILITY[slot.getEntitySlotId()] * X;
  8. }
  9.  
  10. @Override
  11. public int getProtectionAmount(EquipmentSlot slot) {
  12. return PROTECTION_VALUES[slot.getEntitySlotId()];
  13. }
  14.  
  15. @Override
  16. public int getEnchantability() {
  17. return X;
  18. }
  19.  
  20. @Override
  21. public SoundEvent getEquipSound() {
  22. return SoundEvents.ITEM_ARMOR_EQUIP_X;
  23. }
  24.  
  25. @Override
  26. public Ingredient getRepairIngredient() {
  27. return Ingredient.ofItems(RegisterItems.X);
  28. }
  29.  
  30. @Override
  31. public String getName() {
  32. // Всё должно быть маленькими буквами
  33. return "name";
  34. }
  35.  
  36. @Override
  37. public float getToughness() {
  38. return X.0F;
  39. }
  40.  
  41. @Override
  42. public float getKnockbackResistance() {
  43. return 0.XF;
  44. }
  45. }

Теперь, когда у вас есть основы класса материалов брони, давайте зарегистрируем ваши предметы брони в новом классе, который мы просто назовем RegisterItems.

Создание брони в виде предмета

Мы собираемся создать новый класс под названием RegisterItems для реализации ваших новых элементов брони. Это также будет место, например, для регистрации инструментов, если вы создаете новый предмет, такой как слиток (мы будем называть его Custom_Material). . Эта настройка также поместит предметы на новую вкладку предметов в креативе, но вы можете удалить эту часть.

Синтаксис групп - .group(YourModName.YOUR_MOD_NAME_BUT_IN_CAPS_GROUP). Я буду ссылаться на это как на ExampleMod:

  1. public class RegisterItems {
  2.  
  3. public static final ArmorMaterial CUSTOM_ARMOR_MATERIAL = new CustomArmorMaterial();
  4. public static final Item CUSTOM_MATERIAL = new CustomMaterialItem(new Item.Settings().group(ExampleMod.EXAMPLE_MOD_GROUP));
  5. // Если бы вы создали новый материал, вы бы отметили его именно здесь.
  6. public static final Item CUSTOM_MATERIAL_HELMET = new ArmorItem(CUSTOM_ARMOR_MATERIAL, EquipmentSlot.HEAD, new Item.Settings().group(ExampleMod.EXAMPLE_MOD_GROUP));
  7. public static final Item CUSTOM_MATERIAL_CHESTPLATE = new ArmorItem(CUSTOM_ARMOR_MATERIAL, EquipmentSlot.CHEST, new Item.Settings().group(ExampleMod.EXAMPLE_MOD_GROUP));
  8. public static final Item CUSTOM_MATERIAL_LEGGINGS = new ArmorItem(CUSTOM_ARMOR_MATERIAL, EquipmentSlot.LEGS, new Item.Settings().group(ExampleMod.EXAMPLE_MOD_GROUP));
  9. public static final Item CUSTOM_MATERIAL_BOOTS = new ArmorItem(CUSTOM_ARMOR_MATERIAL, EquipmentSlot.FEET, new Item.Settings().group(ExampleMod.EXAMPLE_MOD_GROUP));
  10.  
  11. }

Теперь, когда ваши элементы созданы должным образом, давайте зарегистрируем их и дадим им правильные имена. Вашим первым параметром будет ваше пространство имен, которое является вашим идентификатором мода, а затем следующим - имя, которое вы хотите присвоить своему элементу.

Мы напишем это прямо под вашим последним ArmorItem.

  1. public static void register() {
  2. Registry.register(Registry.ITEM, new Identifier("examplemod", "custom_material"), CUSTOM_MATERIAL);
  3. Registry.register(Registry.ITEM, new Identifier("examplemod", "custom_material_helmet"), CUSTOM_MATERIAL_HELMET);
  4. Registry.register(Registry.ITEM, new Identifier("examplemod", "custom_material_chestplate"), CUSTOM_MATERIAL_CHESTPLATE);
  5. Registry.register(Registry.ITEM, new Identifier("examplemod", "custom_material_leggings"), CUSTOM_MATERIAL_LEGGINGS);
  6. Registry.register(Registry.ITEM, new Identifier("examplemod", "custom_material_boots"), CUSTOM_MATERIAL_BOOTS);
  7. }

Ваши доспехи готовы. Теперь мы просто вызовем регистрацию в нашем основном классе (и запишем новую группу).

  1. public static final ItemGroup EXAMPLE_MOD_GROUP = FabricItemGroupBuilder.create(
  2. new Identifier("examplemod", "example_mod_group"))
  3. .icon(() -> new ItemStack(RegisterItems.CUSTOM_MATERIAL)) // Здесь в качестве значка используется модель нового материала, который вы создали, но вы можете ссылаться на все, что вам нравится
  4. .build();
  5.  
  6. @Override
  7. public void onInitialize() {
  8. RegisterItems.register();
  9. }

Вот и все! Теперь ваша броня должна существовать в игре, все еще нетекстурированная, но присутствующая и способная быть выданной с помощью /give.

Теперь мы будем присваивать текстуры каждой детали.

Текстурирование

Мы предполагаем, что у вас

Нужно назначить их каждому предмету брони.

Следующий код должен быть одинаковым со всеми предметами брони, меняется только то, какую часть мы используем. Мы будем использовать шлем для нашего примера.

resources/assets/examplemod/models/item/custom_material_helmet.json
{
	"parent": "item/generated",
	"textures": {
		"layer0": "examplemod:item/custom_material_helmet"
	}
}

Повторите со всеми предметами брони.

Чтобы придать вашей броне текстуру, просто поместите X_layer_1.png и X_layer_2.png (где X - аргумент getName, который вы выбрали в своем классе материалов брони) в resources/assets/minecraft/textures/models/armor.

Если вы следовали всем инструкциям, то теперь у вас должен быть полный комплект брони!

Добавление сопротивления откидыванию

И вот пришли к самому проклятому!

Mojang решили, что они не только собираются жестко запрограммировать getKnockbackResistance, но и сделают его неизменяемым! Забавная штука.

Чтобы обойти это, мы собираемся сделать миксин, который войдет в ArmorItem. Если это ваш первый раз, здесь о том, как зарегистрировать миксины в вашем fabric.mod.json

Мы создадим класс под названием ArmorItemMixin и напишем:

@Mixin (ArmorItem.class)
public abstract class ArmorItemMixin {
 
}

Теперь мы должны создать @Shadow, чтобы изменить knockbackResistance, который является EntityAttribute

@Mixin (ArmorItem.class)
public abstract class ArmorItemMixin {
	@Shadow @Final private static UUID[] MODIFIERS;
	@Shadow @Final @Mutable private Multimap<EntityAttribute, EntityAttributeModifier> attributeModifiers;
	@Shadow @Final protected float knockbackResistance;
}

Затем мы внедряем через @Inject наш GENERIC_KNOCKBACK_RESISTANCE в конструктор ArmorMaterial.

@Mixin (ArmorItem.class)
public abstract class ArmorItemMixin {
 
    @Shadow @Final private static UUID[] MODIFIERS;
    @Shadow @Final @Mutable private Multimap<EntityAttribute, EntityAttributeModifier> attributeModifiers;
    @Shadow @Final protected float knockbackResistance;
 
    @Inject(method = "<init>", at = @At(value = "RETURN"))
    private void constructor(ArmorMaterial material, EquipmentSlot slot, Item.Settings settings, CallbackInfo ci) {
        UUID uUID = MODIFIERS[slot.getEntitySlotId()];
 
        if (material == RegisterItems.customArmorMaterial) {
            ImmutableMultimap.Builder<EntityAttribute, EntityAttributeModifier> builder = ImmutableMultimap.builder();
 
            this.attributeModifiers.forEach(builder::put);
 
            builder.put(
                    EntityAttributes.GENERIC_KNOCKBACK_RESISTANCE,
                    new EntityAttributeModifier(uUID,
                            "Armor knockback resistance",
                            this.knockbackResistance,
                            EntityAttributeModifier.Operation.ADDITION
                    )
            );
 
            this.attributeModifiers = builder.build();
        }
    }
 
}

Теперь ваша броня имеет значение сопротивления откидыванию, которое вы присвоили ей еще в CustomArmorMaterial.