zh_cn:tutorial:inventory
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
zh_cn:tutorial:inventory [2019/12/19 05:03] – lightcolour | zh_cn:tutorial:inventory [2024/08/27 03:03] (current) – solidblock | ||
---|---|---|---|
Line 1: | Line 1: | ||
- | ====== | + | ====== |
- | Every type of block in Minecraft is represented by a singular | + | 阅读本教程之前,请确保已经做好了[[zh_cn: |
- | This makes it impossible to change a specific block's state by simply changing the '' | + | |
- | as every other block of that type will be affected! | + | 将物品存储在 BlockEntity(方块实体)中的标准方法是使其成为 |
- | But, what if you //do// want to give a singular block state, so it can change based on some condition? | + | |
- | This is what '' | + | 这个教程是用于 1.21 的,对于旧版本,一些方法可能改变。 |
- | Say we wanted a block to have a hardness of '' | + | |
- | it would become harder and gain a hardness of '' | + | ===== 实现 Inventory 接口 ===== |
- | + | ||
- | First we define the boolean property of the block - whether or not it is hard (careful not to import the wrong BooleanProperty!): | + | '' |
- | <code java> | + | |
- | public | + | < |
- | | + | /** |
- | } | + | * 一个简单的 {@code Inventory} 实现,仅有默认的方法和物品列表的 getter。 |
- | </code> | + | * |
- | Then we need to register the property by overriding '' | + | * Originally by Juuz |
- | <code java> | + | */ |
- | public class MyBlock extends Block { | + | public |
- | [...] | + | |
+ | /** | ||
+ | * 从此物品栏中检索物品。 | ||
+ | * 每次被调用时必须返回相同实例。 | ||
+ | */ | ||
+ | DefaultedList< | ||
+ | |||
+ | /** | ||
+ | * 从物品列表创建物品栏。 | ||
+ | */ | ||
+ | static | ||
+ | return () -> items; | ||
+ | } | ||
+ | |||
+ | | ||
+ | * 根据指定的尺寸创建新的物品栏。 | ||
+ | */ | ||
+ | | ||
+ | | ||
+ | } | ||
+ | |||
+ | /** | ||
+ | * 返回物品栏的大小。 | ||
+ | */ | ||
@Override | @Override | ||
- | | + | |
- | | + | |
} | } | ||
| | ||
- | } | + | |
- | </code> | + | * 检查物品栏是否为空。 |
- | Then we need to set the default state of our property in the block constructor: | + | * @return true,如果物品栏仅有一个空堆,否则为true。 |
- | <code java> | + | */ |
- | public class MyBlock extends Block { | + | |
- | | + | |
- | | + | |
- | | + | |
- | | + | if (!stack.isEmpty()) { |
+ | return false; | ||
+ | } | ||
+ | } | ||
+ | return true; | ||
} | } | ||
| | ||
+ | /** | ||
+ | * 检索槽位中的物品。 | ||
+ | */ | ||
+ | @Override | ||
+ | default ItemStack getStack(int slot) { | ||
+ | return getItems().get(slot); | ||
+ | } | ||
+ | | ||
+ | /** | ||
+ | * 从物品栏槽位移除物品。 | ||
+ | * @param slot 从该槽位移除。 | ||
+ | * @param count 需要移除的物品个数。如果槽位中的物品少于需要的,则将其全部取出。 | ||
+ | */ | ||
+ | @Override | ||
+ | default ItemStack removeStack(int slot, int count) { | ||
+ | ItemStack result = Inventories.splitStack(getItems(), | ||
+ | if (!result.isEmpty()) { | ||
+ | markDirty(); | ||
+ | } | ||
+ | return result; | ||
+ | } | ||
+ | | ||
+ | /** | ||
+ | * 从物品栏槽位移除所有物品。 | ||
+ | * @param slot 从该槽位移除。 | ||
+ | */ | ||
+ | @Override | ||
+ | default ItemStack removeStack(int slot) { | ||
+ | return Inventories.removeStack(getItems(), | ||
+ | } | ||
+ | | ||
+ | /** | ||
+ | * 将物品栏槽位中的当前物品堆替换为提供的物品堆。 | ||
+ | * @param slot 替换该槽位的物品堆。 | ||
+ | * @param stack 替换后新的物品堆。如果堆对于此物品栏过大({@link Inventory# | ||
+ | */ | ||
+ | @Override | ||
+ | default void setStack(int slot, ItemStack stack) { | ||
+ | getItems().set(slot, | ||
+ | if (stack.getCount() > getMaxCountPerStack()) { | ||
+ | stack.setCount(getMaxCountPerStack()); | ||
+ | } | ||
+ | } | ||
+ | | ||
+ | /** | ||
+ | * 清除物品栏。 | ||
+ | */ | ||
+ | @Override | ||
+ | default void clear() { | ||
+ | getItems().clear(); | ||
+ | } | ||
+ | | ||
+ | /** | ||
+ | * 将方块状态标记为脏。 | ||
+ | * 更改物品栏之后必须调用,所以游戏正确地储存物品栏内容并提取邻近方块物品栏改变。 | ||
+ | | ||
+ | @Override | ||
+ | default void markDirty() { | ||
+ | // 需要行为时,覆盖此方法。 | ||
+ | } | ||
+ | | ||
+ | /** | ||
+ | * @return true 如果玩家可以使用物品栏,否则为 false。i | ||
+ | | ||
+ | @Override | ||
+ | default boolean canPlayerUse(PlayerEntity player) { | ||
+ | return true; | ||
+ | } | ||
} | } | ||
</ | </ | ||
- | (To set multiple properties, chain '' | ||
- | Now, to set the property we need to call | + | 现在在你的 |
- | + | < | |
- | (Replace | + | public class DemoBlockEntity |
- | <code java> | + | |
- | public class MyBlock | + | |
- | | + | |
@Override | @Override | ||
- | public | + | public |
- | world.setBlockState(pos, | + | return |
- | return | + | |
} | } | ||
+ | [...] | ||
+ | |||
} | } | ||
+ | |||
</ | </ | ||
- | And to use the property we call '' | + | 我们还需要将物品栏保存到 NBT 并从那里加载。'' |
- | <code java> | + | < |
- | public class MyBlock | + | public class DemoBlockEntity |
[...] | [...] | ||
@Override | @Override | ||
- | public | + | public |
- | | + | |
- | | + | |
- | | + | } |
+ | |||
+ | @Override | ||
+ | public NbtCompound writeNbt(NbtCompound nbt, RegistryWrapper.WrapperLookup registryLookup) { | ||
+ | Inventories.writeNbt(nbt, | ||
+ | return | ||
} | } | ||
} | } | ||
</ | </ | ||
- | ==== Adding models for your blockstates | + | ===== 从物品栏(或任何物品栏)中提取和放入 ===== |
- | You can also make the texture and model of your block change based on the state. This is done through a JSON file called a Blockstate JSON. All blocks need a blockstate JSON, whether they have multiple states or not, but the contents of the JSON can be as simple or complex as you like. If you want to change the textures of your block based on the state, you //will// need multiple models. | + | 我们覆盖方块类中的 '' |
- | Let's say you register an instance of '' | + | 首先我们处理第一个槽位,如果是空的。玩家如果拿着物品,则会将拿着的物品放入。物品进入第一个槽位,如果是空的,或者进入第二个槽位,如果第一个是空的,或者如果第二个是空的,我们则会输出与物品栏有关的信息。 |
- | < | + | 注意我们将 '' |
- | { | + | < |
- | " | + | public class DemoBlock extends BlockWithEntity { |
- | "" | + | [...] |
+ | @Override | ||
+ | protected ItemActionResult onUseWithItem(ItemStack stack, BlockState state, World world, BlockPos pos, PlayerEntity player, Hand hand, BlockHitResult hit) { | ||
+ | if (world.isClient) return ActionResult.SUCCESS; | ||
+ | |||
+ | if (!(world.getBlockEntity(pos) instanceof DemoBlockEntity blockEntity)) { | ||
+ | return ItemActionResult.PASS_TO_DEFAULT_BLOCK_INTERACTION; | ||
+ | } | ||
+ | |||
+ | if (!player.getStackInHand(hand).isEmpty()) { | ||
+ | | ||
+ | if (blockEntity.getStack(0).isEmpty()) { | ||
+ | | ||
+ | blockEntity.setStack(0, player.getStackInHand(hand).copy()); | ||
+ | // 从玩家手中移除物品堆 | ||
+ | player.getStackInHand(hand).setCount(0); | ||
+ | } else if (blockEntity.getStack(1).isEmpty()) | ||
+ | | ||
+ | player.getStackInHand(hand).setCount(0); | ||
+ | } else { | ||
+ | // 如果物品栏满,提醒玩家 | ||
+ | player.sendMessage(Text.literal("物品栏满!第一个槽位是") | ||
+ | .append(blockEntity.getStack(0).getName()) | ||
+ | .append(",第二个槽位是") | ||
+ | .append(blockEntity.getStack(1).getName())); | ||
+ | | ||
+ | } | ||
+ | return ItemActionResult.SUCCESS; | ||
} | } | ||
} | } | ||
</ | </ | ||
- | Let's break this simple example down. There are a couple important parts to this JSON: | + | 玩家不持有物品时,我们将采取相反的行为。我们将从第二个槽位中取出物品,如果第二个是空的就第一个。如果第一个也是空的,就不做任何事情。 |
- | - The ''" | + | <code java DemoBlock.java> |
- | - A variant named ''""'' | + | public class DemoBlock extends BlockWithEntity { |
- | - The object assigned to the ''""'' | + | [...] |
- | - The ''" | + | @Override |
- | + | | |
- | If you //do// want to have different models for each blockstate, you'd want to add multiple variants. For the same '' | + | |
+ | | ||
+ | ... | ||
+ | } else { | ||
+ | // 如果玩家手中没有东西,就依次给玩家方块实体中的物品 | ||
- | <code JavaScript resources/assets/mymod/blockstates/my_block.json> | + | |
- | { | + | if (!blockEntity.getStack(1).isEmpty()) { |
- | " | + | |
- | " | + | player.getInventory().offerOrDrop(blockEntity.getStack(1)); |
- | | + | // 从物品栏移除物品堆 |
+ | | ||
+ | } else if (!blockEntity.getStack(0).isEmpty()) | ||
+ | | ||
+ | blockEntity.removeStack(0); | ||
+ | } else { | ||
+ | return ItemActionResult.PASS_TO_DEFAULT_BLOCK_INTERACTION; | ||
+ | | ||
+ | } | ||
+ | |||
+ | return ItemActionResult.SUCCESS; | ||
} | } | ||
} | } | ||
</ | </ | ||
- | In this JSON, there are two variants, one for each possibility of the '' | + | ===== 实现 SidedInventory 接口 ===== |
+ | 如果你希望有基于与方块不同的面(漏斗或者其他模组)进行交互的不同逻辑,你可以实现 SidedInventory 接口。如果说你想使得方块不能从上侧插入,可以这样做: | ||
- | Variants are based on possible permutations of the properties added to your block. A property can be totally ignored in the blockstate JSON if you want, like in the first blockstate JSON where we ignored the '' | + | <code java DemoBlockEntity.java> |
+ | public class DemoBlockEntity extends BlockEntity implements ImplementedInventory, SidedInventory { | ||
+ | [...] | ||
+ | @Override | ||
+ | public int[] getInvAvailableSlots(Direction side) { | ||
+ | | ||
+ | return IntStream.of(getItems().size()).toArray(); | ||
+ | } | ||
- | This is only a simple introduction to blockstate JSONs. All of the tricks you can do with blockstate and model JSONs are documented on the [[https:// | + | @Override |
- | ==== A note about performance ==== | + | public boolean canInsert(int slot, ItemStack stack, Direction direction) { |
- | Every possible state of a block is registered at the start of the game. This means that if you have 14 boolean | + | return direction |
+ | } | ||
+ | |||
+ | @Override | ||
+ | public | ||
+ | return true; | ||
+ | } | ||
+ | } | ||
+ | </ |
zh_cn/tutorial/inventory.1576731825.txt.gz · Last modified: 2019/12/19 05:03 by lightcolour