Skip to content
是封面

简单、轻量、优雅——dc装饰模型支持库

引言

Decoration Creator Kit(dc),是一个面向MC高版本(1.20.5+)的装饰模型的支持库。它支持开发者使用至少两条指令,在游戏内注册一个可交互的模型。

该支持库可以支持开发者很简单的定义模型的各种属性和交互事件。注册的模型可以像方块一样直接放置、互动和破坏,也能像实体一样移动、旋转,以及对玩家的互动做出各种反馈。因此,该支持库十分适合装饰向模型的作者使用。

关于dc的诞生……

这个项目最初来自于24年3月份的一个委托,委托人希望我制作一个能够支持自定义模型放置、移动和旋转的数据包。实际上,在更早开发CAM的时候,我便考虑过制作装饰模型的可能。

我曾经规划过一个“雕塑扩展包”,内容便是各种各样的装饰性雕塑。不过随着CAM项目的终止,这个企划便也没有下文了。

因此,我开始设想制作一个统一的框架,能够将模型的放置、破坏和交互统一管理,同时又拥有比较好的性能。正巧,mj在当时引入了展示实体和交互实体,而函数宏的引入也大大方便了动态参数的传递,在这些便利下,我花了数周时间研究调试,便将dc的首个版本完成了。顺便,最初的委托便是森罗物语数据包版的前身,而当前的森罗数据包也依赖于dc。

dc的特色和优势

dc的目标是在易用性、轻量级、高性能、高自由度之间取得平衡。因此,dc的模型具备基本的交互功能,同时也支持引入外部函数和谓词来简单地扩展。dc拥有非常简单清晰的索引注册结构,并且引入了mcdoc支持索引注册的自动补全,更加方便开发者注册模型,减少记忆成本。

一个典型的dc模型由一个标记实体、一个展示实体和一个交互实体组成。如果它具有碰撞箱或光源属性,还可能具有其他的辅助方块或实体。它们通过uid计分板进行逻辑连接。

dc使用索引方法存储模型数据。在手动执行install指令时,支持库会解析并存储模型的索引数据,在模型生成时,标记实体会直接拉取存储中的索引数据并分配给展示实体和交互实体。如此,我们便不需在模型物品中存储大量数据,同时减少了轮询操作,保证了常态下的性能。

dc的使用方法

索引(Index)

dc采用索引的方式读取模型数据,在添加模型之前,需要注册对应的索引。

索引注册

索引的注册为创建一个函数,按照如下结构写入数据:

mcfunction
data modify storage dc:index input.<name> set value <data>
data modify storage dc:index keylist append value "<name>"

其中 <name> 为需要注册的索引名称, <data> 为索引数据,为一复合标签,格式如下:

注:键前方的括号为键值类型,在compound后面带有冒号的为结构模板,只在第一次出现时解析其结构,后面则省略;

索引数据
compound索引数据
  • stringtype: ( <"regular"|"hitbox"|"fixed"|"light"> )(默认为regular)模型类型。hitbox类型拥有碰撞箱,需要额外设置碰撞箱参数,fixed类型拥有1格完整碰撞箱,不可移动和缩放,交互框大小固定为11。light类型可以提供光照,其他与fixed类型相似,不可移动和缩放,交互框大小固定为11。只有类型相同的模型才可互相转换。
  • compoundextra_data:typeregular时,需要提供的额外数据。

      当type"hitbox"时:

    • floatwidth:碰撞箱宽度;
    • floatoffset:碰撞箱的偏移量,可为负值;

    type"light"时:

    • bytelevel:(值域0-15,默认值15)光源亮度;
    • boolhitbox: (默认0)是否有碰撞箱,设置为1时拥有一个碰撞箱。
  • stringtemplate:(可选)模板名称;
  • compounditem: 显示的模型对应的物品数据,格式与一般物品格式相同,无slot标签。
    物品共通标签
    • stringid:物品的id
    • intcount:物品的数量
    • compoundcomponents:物品组件格式
      • any物品组件ID:一个物品组件
      • ...
  • stringloot_table:战利品表,可以替换上述的item字段,从战利品表中导入物品数据。会与item中数据合并,如有同名标签,item覆盖loot_table中标签。
  • float_listmodsize: 三维数组,物品展示实体的长宽高,等效于展示实体中的scale属性
  • compoundinteractsize: 交互实体尺寸:
    • floatheight: 交互实体的高度;
    • floatwidth: 交互实体的宽度;
  • compoundprop: 模型的额外属性,可以在此指定一些模型特性。
    • boolheight_adaption:设为1时,垂直旋转时碰撞箱高度随之调整;
  • compoundevents:互动事件设置。
    • listconstruct:放置时默认执行的事件,如播放声音等;
      • compound一个事件,不带条件参数。
        • stringevent: 事件名,详细信息见下;
        • compoundargs: 参数,具体格式随事件而决定(部分事件若不带参数,则此项不存在)
    • listupdate:在模型更新时执行的事件,旋转、缩放、转换等事件会触发更新行为
      • compound一个事件 ,不带条件参数。
        • stringevent: 事件名,详细信息见下;
        • compoundargs: 参数,具体格式随事件而决定(部分事件若不带参数,则此项不存在)
    • compoundleft_click:左键点击事件
      • compoundfallback:criteria中的事件判据没有符合的时,应用的默认事件。一个事件 ,不带条件参数。
        • stringevent: 事件名,详细信息见下;
        • compoundargs: 参数,具体格式随事件而决定(部分事件若不带参数,则此项不存在)
      • listcriteria:条件事件,当玩家手持相应物品点击模型时,触发的事件
        • compound一个事件
          • stringevent: 事件名,详细信息见下;
          • compoundargs: 参数,具体格式随事件而决定(部分事件若不带参数,则此项不存在)
          • compounditem:触发该事件要求玩家主手持有的物品,id可为物品或物品标签;
            • 结构略
          • stringpreidcate:compoundlist谓词条件,触发该事件需要点击的玩家满足谓词要求的条件,可内联定义;
    • compoundright_click:右键点击事件
      • compoundfallback:criteria中的事件判据没有符合的时,应用的默认事件。一个事件 ,不带条件参数。
        • stringevent: 事件名,详细信息见下;
        • compoundargs: 参数,具体格式随事件而决定(部分事件若不带参数,则此项不存在)
      • listcriteria:条件事件,当玩家手持相应物品点击模型时,触发的事件
        • compound一个事件
          • stringevent: 事件名,详细信息见下;
          • compoundargs: 参数,具体格式随事件而决定(部分事件若不带参数,则此项不存在)
          • compounditem:触发该事件要求玩家主手持有的物品,id可为物品或物品标签;
            • 结构略
          • stringpreidcate:compoundlist谓词条件,触发该事件需要点击的玩家满足谓词要求的条件,可内联定义;

此后,在自己的数据包下建立一个名为dc的命名空间,并创建一个名为index的函数标签(dc/tags/functions/index.json),将索引注册函数加入即可。

索引加载

在注册完索引后,该索引还未构建完成,还不可使用。此时需要加载索引。 执行/reload指令,然后执行/function dc:api/install指令加载索引。 动态构建可能需要花费一定时间,请耐心等待。

索引使用

加载完成后即可使用该索引生成模型。 可以以任何方式生成一个带有指定数据的标记(marker)或物品展示框(item_frame)(仅支持1.21.5+)(即带有dc_place标签,data中指定索引),即可生成指定的自定义模型。

mcfunction
/give @s minecraft:cow_spawn_egg[entity_data={id:"minecraft:marker",Tags:["dc_place"],data:{index:"#YOURINDEX#"}}]
mcfunction
/give @s minecraft:item_frame[entity_data={id:"minecraft:item_frame",Tags:["dc_place"],data:{index:"#YOURINDEX#"}}]

模型类型

在dc2.0.0以后,引入了三种新的模型类型,其中FixedLight类型为固定模型,推荐通过物品展示框生成。

索引数据继承

在标记实体中可以使用一个inheritance字段,语法和索引注册数据相同,在模型生成的时候会被合并进数据中。可用于特定物品单独设置一些数据。

此处以刷怪蛋为例:

mcfunction
/give @s minecraft:cow_spawn_egg[entity_data={id:"minecraft:marker",Tags:["dc_place"],data:{index:"#YOURINDEX#",inheritance:{item:{id:"apple"}}}}]

事件(Event)

目前支持基础事件:旋转,平移,坐,破坏等。
大部分的参数均可缺省,此时应用默认值。

破坏(destruct)

销毁模型。

参数:

compound 事件。
  • stringevent:事件名,此处为destruct
  • compoundargs:事件参数:
    • stringparticle:破坏模型时产生的粒子;
    • stringsound:破坏模型时播放的音效;
    • anycompounditem:控制非创造模式下破坏模型产生的掉落物:
      · 若不含该参数,则不产生掉落物;
      · 若传入空值或任意字符串,则产生默认掉落物(索引中item指定的物品,此外包含生成该模型的索引值)
      · 若传入一个复合标签,则按照其中规则修改掉落物:
      • stringmode:(add|replace|inherit) 决定物品数据和合并模式(默认为add):
        · add:将下面参数(itemloot_table)中指定的物品数据和索引中的物品数据合并
        · replace:直接使用参数中指定的物品数据,舍弃索引中的物品数据; 注意: 重新放置时依然从索引中获取物品数据;
        · inherit: 舍弃索引中的物品数据,从模型的物品展示实体中动态获取物品数据。
      • compounditem:指定掉落物的物品数据
      • stringloot_table:使用战利品表指定掉落物数据,会与item中数据合并,如有同名标签,item覆盖loot_table中标签;

平移(move)

移动模型。

参数:

compound 事件。
  • stringevent:事件名,此处为move
  • compoundargs:事件参数:
    • stringtype:平移事件类型。接受以下五个值:
      X | Y | Z :绝对路径移动,在指定的轴上平移。
      R : 相对玩家与模型的位置水平移动模型。
      V : 相对玩家与模型的位置垂直移动模型。
    • floatdistance:(-2f~2f)平移的距离。值需要在-2到2之间,超出则会被强制设为最近的边界值。默认值为0f。

若type为"X","Y","Z",则正值向对应轴的正方向移动,负值向对应负方向移动; 若type为"R","V",则正值向远离玩家的方向移动,负值向靠近玩家的方向移动。

旋转(rotate)

旋转模型。

参数:

compound 事件。
  • stringevent:事件名,此处为rotate
  • compoundargs:事件参数:
    • stringtype:平移事件类型。接受以下两个值:
      "H":水平旋转;"V":垂直旋转。默认值为H。
    • floatdistance:(-180f~180f)旋转的角度。值需要在-180到180之间,超出则被强制设为最近的边界值。默认值为0f。

坐(sit)

可以坐上模型。

参数:

compound 事件。
  • stringevent:事件名,此处为sit
  • compoundargs:事件参数:
    • compoundorient:总是为空标签,当存在时,玩家坐上的瞬间朝向会被校正为模型朝向。

声音(sound)

发出声音。

参数:

compound 事件。
  • stringevent:事件名,此处为sound
  • compoundargs:事件参数:
    • stringsound:音效id,播放的音效;

转换(trans)

将一个模型转换为另一个已注册的模型,将会使用对应模型的所有数据覆盖原有数据。 参数:

compound 事件。
  • stringevent:事件名,此处为trans
  • compoundargs:事件参数:
    • string * index:要转换的模型的索引,若该参数不存在,则事件注册将失败;若该参数指向为注册的索引名,则不会做出任何修改。
    • stringfunc:(可选)转换后以标记实体为执行者执行的函数。

在这个事件的函数中,标记实体可以使用@s选中;绑定的展示实体和交互实体分别具有dc_trans_displaydc_trans_interaction标签。

缩放(scale)

可以放大和缩小模型。

参数:

compound 事件。
  • stringevent:事件名,此处为rotate
  • compoundargs:事件参数:
    • floatscale:模型大小的变化值,正值为放大,负值为缩小。

限制:scale本身没有范围限制,但放大缩小后模型有如下限制: - 模型放大倍数在0.1到10之间 - 交互实体宽高任一不能小于0.1 - 展示实体高不能大于10

备注: 该事件会调用更新模块自动更新模型;

预制事件(pre)

预制事件是一类特殊事件,为带有固定的参数的特定普通事件。因此不需要额外指定arg(指定了也没有效果)。用于简化操作。

在使用中,在events里填写pre/<预制事件名>即可。

compound 事件。
  • stringevent:事件名,此处填写pre/<预制事件名>

目前已有的预制事件:

  • move_r_1px:水平移动1像素(推)
  • move_r_-1px:水平移动-1像素(拉)
  • move_y_1px:竖直移动1像素(上)
  • move_y_-1px:竖直移动-1像素(下)
  • rotate_h_225:水平旋转22.5度(顺时针)
  • rotate_h_-225:水平旋转-22.5度(逆时针)
  • rotate_v_225:垂直旋转22.5度(顺时针)
  • rotate_v_-225:垂直旋转-22.5度(逆时针)
  • scale_02:缩放0.2倍(大)
  • scale_-02:缩放-0.2倍(小)

调试事件:更新(update)

1.更新模型的数据结构到dc的版本,2.并且和当前的索引信息同步

参数:无

备注:

  • 更新完成后会向互动玩家输出一条更新完成的信息,将$silent_updatedc_options计分项设为1可以不输出该信息
  • 使用调试棒左键模型可以触发更新事件

调试事件:信息(info)

向玩家聊天栏输出模型的属性信息,包括该模型的uid,索引,缩放倍数,版本,上次更新时间,当前时间等。

参数:无

备注:

  • 使用调试棒右键模型可以触发信息事件

流程控制事件:组(group)

按照顺序依次执行一系列事件。

参数:

compound 事件。
  • stringevent:事件名,此处为group
  • compoundargs:事件参数:
    • listevents:为一个列表,里面的每一项均为一个事件,不含事件条件字段。
      • compound一个事件。
        • stringevent: 事件名,详细信息见下;
        • compoundargs: 参数,具体格式随事件而决定(部分事件若不带参数,则此项不存在)

备注: 对列表内的事件暂无检查

流程控制事件:随机(random)

在一系列事件中按照权重挑选一个事件执行。

参数:

compound 事件。
  • stringevent:事件名,此处为random
  • compoundargs:事件参数:
    • listevents:为一个列表,里面的每一项均为一个事件,不含事件条件字段。
      • compound一个事件。
        • stringevent: 事件名,详细信息见下;
        • compoundargs: 参数,具体格式随事件而决定(部分事件若不带参数,则此项不存在)
        • intweight:权重。代表该事件有多大的可能被选取。

备注: 对列表内的事件暂无检查

保留字:__nothing__

什么都不做。可在创建空事件时使用。

自定义事件(custom)

此事件支持引用外部函数作为事件执行。

参数:

compound 事件。
  • stringevent:事件名,此处为custom
  • compoundargs:事件参数:
    • stringfunc:自定义事件函数路径。事件定义规范见下。
    • any可选,其他需要传入的参数,随事件类型而定。

自定义事件规范: 自定义事件的路径为一个文件夹,其下包含一个execute.mcfunction文件用于执行,一个可选的check.mcfunction文件用于参数检查,以及其他可选的辅助函数。 在索引构建时,会调用check函数检查参数的合法性;在交互条件触发时,会执行execute函数执行事件。

一些你需要知道的接口信息:

  1. 执行函数可以为宏函数,参数会以函数宏的形式传入,可以直接调用; 如果不想使用宏方法调用,可以访问storage dc events.temp.target.args路径。
  2. 检查函数执行时,参数会储存在storage dc:temp event.args路径下,可以检查并修改该路径中的参数信息,完成后,修改过的信息会合并至索引数据下。检测到参数未正确设置等问题时,可以将#checkdc_temp计分项设为1,此时该事件将不会被加入索引的事件列表中。
  3. 本包的每个模型由三个实体组成,一个带有dc_pivot标签的标记实体,一个带有dc_display标签的展示实体,一个带有dc_interaction标签的交互实体; 函数以标记实体为执行者,以其位置为执行位置。此外,展示实体的y坐标比标记实体的高0.5个方块,在移动模型时需要注意。 可以用dc_custom_pivotdc_custom_displaydc_custom_interaction来分别选中交互的模型的标记,展示实体,交互实体。
  4. 执行(任意)事件时,触发点击事件的玩家拥有dc_click_temp标签,可以用此标签找到对应的玩家。

模板(Template)

为辅助构建模型数据的开发工具,为部分的索引数据,在索引构建时会合并到索引数据中。 可以将索引中重复的部分注册为模板,减少重复代码的使用。

模板注册

新建一个函数,按照如下格式写入模板数据:

mcfunction
data modify storage dc:template <name> set value <data>

name为模板名,data为模板数据,为索引的一部分(不完整)数据,格式参照索引数据格式,不包含template选项(即不支持嵌套模板调用),且所有键均为可选。

完成模板注册后,在dc命名空间下建立一个一个名为template的函数标签(dc/tags/functions/template.json),把模板函数加入。 最后执行/reload指令。

模板使用

在注册索引时,把注册好的模板的名字写在索引的template选项下即可。

已有模板

dc内置了两个简单的模板,可以在注册索引时调用。 basic:

snbt
{   
    modsize:[1f,1f,1f],
    interactsize:{height:1f,width:1f},
    events:{
        left_click:{
            fallback:{event:"destruct"},
            criteria:[]
        },
        right_click:{
            fallback:{event:"__nothing__"},
            criteria:[]
        }
    }
}

基础模型模板,左键破坏模型。

default:

snbt
{
    modsize:[1f,1f,1f],
    interactsize:{height:1f,width:1f},
    events:{
        left_click:{
            criteria:[
                {
                    event:"move",
                    item:{id:"minecraft:stick"},
                    args:{distance:-0.2f}
                },
                {
                    event:"rotate",
                    item:{id:"minecraft:shears"},
                    args:{angle:-45.0f}
                },
                {
                    event:"rotate",
                    item:{id:"minecraft:blaze_rod"},
                    args:{type:"V",angle:-22.5f}
                }
            ],
            fallback:{event:"destruct"},
        },
        right_click:{
            criteria:[
                {
                    event:"move",
                    item:{id:"minecraft:stick"},
                    args:{type:"R",distance:0.2f}
                },
                {
                    event:"rotate",
                    item:{id:"minecraft:shears"},
                    args:{angle:45.0f}
                },
                {   
                    event:"rotate",
                    item:{id:"minecraft:blaze_rod"},
                    args:{type:"V",angle:22.5f}
                }
            ],
            fallback:{event:"__nothing__"}
        }
    }
}

默认模型模板,在基础模板基础上,增加了手持木棍移动模型,手持剪刀水平旋转模型,手持烈焰棒垂直旋转模型的事件。其中左右键点击能够施加相反的效果。

设置

dc_options计分板内有一些设置选项。

已有设置:
$auto_install:设为1,在reload时自动加载模型索引数据;
$silent_update:设为1,在更新模型时不再向玩家输出信息;
$silent_register:设为1,在注册新模型时不再向玩家输出信息;

可视化设置界面:输入/function dc:menu/main,呼出设置页面,可以可视化调整上述选项。

Powered by Vitepress and Github Pages