用户:MashKJo/1.12.2模组开发教程/9.第一个方块

MashKJo留言 | 贡献2024年12月16日 (一) 17:26的版本 →‎添加方块对应的物品形式:​ // Edit via Wikiplus

在开始制作我们的第一个方块前,笔者先区分一下两个概念:方块(Block)和方块的物品形式(ItemBlock)——平时,我们说“方块”,是同时涵盖了这两个概念的:广义上的方块,被放置在世界中时,即为Block;而在物品栏中时,它则是一种特殊的物品,也即ItemBlock。

好,区分了这两个概念后,我们就可以开始了。

添加一个新方块的流程

所有方块都是net.minecraft.block.Block类的实例,该类和Item类不同,它没有零参构造器,我们在构造Block实例时,有2个构造器可供选择:(Material materialIn)(Material materialIn, MapColor mapColorIn)

MapColor用于指示地图上显示该方块所用的颜色,这个类的构造方法是private的,且它要求构造MapColor对象时指定该对象对应的序数,该类有许多静态MapColor字段供你使用,Mojang在这个类里把硬编码体现得淋漓尽致(捂脸)。

Material用于描述方块的“质地”,这个类纯粹用于封装许多和方块有关的信息,它能影响方块的许多属性,这个类的构造方法倒是public的,其中要求传入一个MapColor。不过我们也没必要新建Material,直接复用原版的即可。

上述的第一个构造方法,实际效果为this(materialIn, materialIn.getMaterialMapColor());。读者根据自己的需求自行选择合适的构造器,填入合适的参数就好。

想必读者肯定猜到了,Block类也有很多getter和setter可以用。与Item的类似,Block类的getter牵扯到BlockState这一概念,因此后面再讲,现在只讲setter:

  • setLightOpacity:传入一个int量,用于设定方块的透光度,传入的数值越大透光度越低。完全不透光的方块,该值为255;完全透光则为0;部分透光则在两者之间,如树叶的lightOpacity为1。该值默认由一个三目运算决定,具体为:如果该方块为渲染意义上的完整方块,lightOpacity为255,否则为0
  • setLightLevel:传入一个float量,设置方块的光照等级,注意实际的光照值要把这个传入的参数乘以15。具体设定为多少,读者可以去查阅原版中会发光的方块的源代码,这块是怎么设定的
  • setResistance:传入一个float量,用于设置方块的爆炸抗性,实际抗性值为传入参数乘以3的结果
  • setHardness:传入一个float量,用于设置方块的挖掘硬度,注意这个setter也有可能影响爆炸抗性。特别地,还有一个该setter的特化版:setBlockUnbreakable,会把hardness设定为-1.0F,即让该方块变得像基岩一样不可破坏
  • setUnlocalizedName、setCreativeTab:作用类似于Item类中的同名方法
  • setHarvestLevel:形参列表为(String toolClass, int level),用于设定该方块的挖掘等级和应被什么种类的工具挖掘。实际上这个方法是Forge塞进去的——没错,Minecraft原版甚至没有挖掘等级这一概念,原版中某些方块需要特定等级的工具才能挖掘的例子纯属硬编码的结果。toolClass这个String代表工具类型,如"picakxe""axe"等等。而level的规则是:木制/金制工具的等级为0,石制为1,铁制为2,钻石工具则为3,-1则代表空手挖掘。另外注意:该方法的返回值类型为void,即不符合链式调用规则

例如,笔者新增了一个方块:红宝石块:

src/main/java/net/tutorial_mod/block/BlockRuby.java:

package net.tutorial_mod.block;

import net.minecraft.block.Block;
import net.minecraft.block.material.MapColor;
import net.minecraft.block.material.Material;
import net.tutorial_mod.TutorialMod;

public class BlockRuby extends Block{
   public BlockRuby(){
       super(Material.ROCK, MapColor.RED);
       this.setRegistryName(TutorialMod.MODID, "ruby_block");
       this.setUnlocalizedName("ruby_block");
       this.setCreativeTab(TutorialMod.TutorialModTab);
       this.setHarvestLevel("pickaxe", 2);
   }
}

最后,先实例化,监听RegistryEvent.Register<Block>事件,把我们的新方块注册进游戏。

添加方块对应的物品形式

进入游戏后,读者可能会感到奇怪:创造模式物品栏里,并没有我们的新方块,用/give指令也得不到,只有用/setblock指令才能把该方块放置在世界中。那是因为我们还没有注册该方块的物品形式。

方块的物品形式首先是物品,实际上是ItemBlock——这是Item的一个子类,构造器接受一个Block参数的传入,这种设计很易于理解。我们直接实例化ItemBlock类,再通过监听RegistryEvent.Register<Item>事件即可把它注册进游戏。注意:我们不需要给方块的物品形式的实例调用一系列setter方法,但必须调用setRegistryName——且它的registryName和对应方块的registryName必须相同。

可能会有读者对前面Block类的setUnlocalizedName和setCreativeTab这两个setter感到奇怪——因为其实这两个方法只对物品有意义——毕竟放在世界中的方块需要什么对应的创造模式物品栏?是这样,这两个方法是为ItemBlock服务的。我们用我们的Block对象构造出一个ItemBlock后,该ItemBlock对象中的对应的创造模式物品栏、unlocalizedName这些信息会自动等于对应的Block对象中的对应信息的,这也是为什么我们不需要给ItemBlock调用一大堆的setter。

ItemBlock类中覆写了Item类中的一些方法,导致了ItemBlock对象的通用逻辑为:玩家在世界中对着非空气方块右键时,如果坐标合法(如Y坐标在0~255这个范围内),就会在该处放置一个对应的方块,同时手上的ItemStack的数量这一字段减一。除非你真的有特殊需求,否则你只需要直接实例化ItemBlock类得到方块的物品形式就好了,不用去先继承ItemBlock。

好了,现在再运行游戏,应该可以在创造模式物品栏中看到我们的方块的物品形式了。

为方块和ItemBlock提供模型和材质文件