Bedrock Wiki
  • QQ
  • 原站
新手入门指南
  • Guide
    • 1. 简介
      指南
    • 2. 附加组件详解
    • 3. 软件与准备工作
    • 4. 项目设置
    • 5. 创建自定义物品
    • 6. 创建自定义实体
    • 7. Blockbench:建模、贴图与动画制作
    • 8. 添加战利品表、生成规则与合成配方
  • Extra
    • a. 理解JSON
    • b. 下载示例包
    • c. 故障排除
      帮助
    • d. 高级清单文件指南
    • e. 格式版本
    • f. Android 项目设置
JSON UI
  • General
    • JSON UI 入门指南
      指南
    • 最佳实践
      指南
  • Tutorials
    • Aseprite 动画
    • 保留标题文本
      中级
    • 修改服务器表单
      中级
    • 字符串与数字转换
      中级
    • 按钮与开关
      新手
    • 添加HUD界面元素
      新手
  • Documentation
    • JSON UI 文档
Meta
  • Style Guide
  • 使用JSON模式(Schemas)
  • 实用链接
  • 版本控制
  • 附加包性能优化
  • Q&A
    • GameTest 问答集 2021/08/06
    • 世界生成问答 2024/11/15
    • 延迟渲染技术预览问答 2024/02/23
    • 方块与物品问答 2024/08/30
    • 脚本与编辑器问答 2023/09/22
NBT
  • General
    • .mcstructure
  • Tutorials
    • 扩展结构限制
      简单
    • 教育版中的实验功能
      简单
  • NBT in Depth
    • NBT 库列表
      专家
    • NBT读取示例
      专家
    • 关于NBT(命名二进制标签)
      专家
世界生成
  • General
    • 世界生成入门
      指南
      实验性
    • 特性类型
      实验性
    • 生物群系
      指南
      实验性
  • Tutorials
    • 特征(Feature)的方块条件
      实验性
    • 生成地表区块
      实验性
    • 生成自定义矿石
      实验性
    • 生成自定义结构
      实验性
    • 高度图噪声地形
      实验性
  • Documentation
    • 生物群系标签
动画控制器
  • 动画控制器入门指南
    指南
  • 实体命令
    中级
  • AFK检测器
  • 将Molang变量转换为计分板数值
  • 死亡指令
  • 重生指令
命令
  • General
    • 命令方块入门指南
    • 函数
    • NBT 命令
    • 坐标系
    • 方块状态
    • 理解目标选择器
    • 记分板操作
  • Commands
    • Execute
      简单
    • Playanimation
    • 伤害
    • 播放音效
  • On Event Systems
    • 玩家首次加入时
    • On Player Join
    • 玩家离开时触发
    • 玩家死亡事件
    • 玩家重生事件系统
    • 首次加载世界时
  • Scoreboard Systems
    • 实体计数器
    • 计分板计时器
    • 比较与获取分数
  • Techniques
    • 执行逻辑门
    • MBE - Max的方块实体
    • FMBE - 创建显示实体的新方法
    • 视线检测
    • 移动状态检测
    • 轨道摄像机
  • Useful Creations
    • 多人位置重排系统
      函数
    • 自定义合成台
      简单
实体
  • General
    • Intro to Entities BP
      指南
      新手
    • 实体资源包入门
      指南
      新手
    • 实体问题排查指南
      帮助
    • NPC对话系统
      中级
    • 实体事件
      新手
    • 实体属性
    • 渲染控制器
      新手
    • 生成规则
  • Tutorials
    • 任意坐标系间的坐标转换(世界、实体、骨骼)
      中级
    • 创建船只
      中级
    • 实体手持物品
      中级
    • 实体攻击机制
      中级
    • 实体睡眠机制
      中级
    • 实体碰撞体
      中级
    • 实体移动
    • 实体计时器
      中级
    • 无敌实体
      新手
    • 村庄机制实现指南
    • 检测其他实体
      中级
    • 生成已驯服的实体
      脚本
      中级
    • 视线检测实体
      中级
    • 禁用队友伤害
      中级
    • 范围效果云入门指南
      中级
    • 虚拟实体
      新手
    • 飞行实体控制
      中级
  • Documentation
    • Vanilla Usage Spawn Rules
    • 原版使用组件
    • 抛射物
    • 虚拟组件
    • 运行时标识符
    • 非生物实体运行时标识符
战利品、配方与交易
  • General
    • 交易行为
  • Documentation
    • 战利品表
    • 交易表
    • 合成配方
    • 物品函数
  • Tutorials
    • 随机化结构战利品
      简单
文档
  • Shared Constructs
  • Molang 查询详解
  • Vanilla Materials
    专家
  • 声音定义
  • 文件类型
  • 材质配置文件说明
    专家
  • 菜单分类
  • 资源包文件夹结构
  • 雾效ID
  • 高级Molang指南
方块
  • General
    • 方块入门指南
      指南
      新手
    • 方块组件
    • Block Tags
    • 方块状态
    • Block Traits
    • 方块排列组合
    • 方块事件
      脚本
    • 方块事件迁移指南
      帮助
    • 方块物品化
      中级
    • 方块问题排查指南
      帮助
  • Visuals
    • 方块剔除
      中级
    • 方块模型
      指南
      新手
      简单
    • 方块着色
      简单
    • 方块纹理动画
      中级
    • 方块纹理变体
      中级
  • Tutorials
    • Precise Interaction
      专家
      脚本
    • Precise Rotation
      专家
      脚本
    • 伪方块
      中级
    • 可旋转方块
    • 应用持续效果
      简单
      脚本
    • 矿石战利品表
      简单
      脚本
    • 规避状态值限制
      专家
  • Vanilla Re-Creations
    • 自定义作物
      中级
      脚本
    • 自定义活板门
      中级
      脚本
    • 自定义玻璃
      新手
      简单
      已弃用
    • 自定义釉面陶瓦
      简单
  • Documentation
    • 原版方块模型
      新手
    • 方块形状
    • 方块格式历史
    • 方块音效
服务器
  • Software
    • Bedrock Server Software
  • Protocols
    • Bedrock Protocol
    • NetherNet 协议
    • RakNet 协议
概念
  • contents.json
  • Molang
    中级
  • Rawtext
  • textures_list.json
  • 命名空间
  • 子包
  • 文本与本地化
  • 着色器
  • 纹理图集
    中级
  • 表情符号与特殊字符
  • 覆盖资源
    中级
  • 音效
    中级
物品
  • General
    • 物品入门指南
      指南
      新手
    • 物品组件
    • 物品标签
    • 物品事件
      脚本
    • Item Event Migration
      帮助
    • 物品问题排查指南
      帮助
  • Tutorials
    • Custom Pottery Sherds
    • 可投掷物品
      中级
    • 生成物品
      中级
    • 自定义武器
      简单
    • 自定义盔甲
    • 自定义食物
      简单
      脚本
    • 通过装备物品执行命令
      实验性
      中级
    • 高分辨率物品
  • Documentation
    • 附魔
    • Numerical Item IDs
    • Vanilla Usage Components
    • 原版物品标识符
      已弃用
    • 可附着物
      新手
    • 物品格式历史记录
视觉效果
  • General
    • 实体视觉效果简介
      指南
    • 基岩版建模指南
    • 动画中的特效
    • 基于数学的动画
      中级
    • 材质
      专家
    • 材质创作
      专家
    • 皮肤包制作指南
    • 自定义死亡动画
      中级
  • Tutorials
    • Glowing Entity Texture
    • 受伤动画
      中级
    • 实体纹理动画
      中级
    • 栓绳位置调整
      简单
    • 玩家几何模型
      新手
    • 移除实体阴影
      中级
    • 重绘生成蛋纹理
      新手
  • Ideas
    • 结构展示技巧
粒子效果
  • General
    • 粒子效果入门
      指南
  • Tutorials
    • 禁用粒子效果
      新手
  • Documentation
    • 原版粒子效果
脚本编写
  • General
    • 脚本编程入门
    • 什么是Script API?
    • API 模块
  • Tutorials
    • GameTests
      实验性
    • 简易聊天命令
      实验性
    • 脚本核心功能
    • 脚本表单
      实验性
    • 脚本请求API
      实验性
    • 阻止方块放置
  • Documentation
    • JavaScript 问题排查指南
    • Script Resources
    • Script Watchdog
      实验性
    • TypeScript
    • 引擎环境
虚拟现实
  • General
    • 启用VR模式
      指南
    • 配置资源包
      专家
  • Tutorials
    • 编辑你的第一个模型
      专家

任意坐标系间的坐标转换(世界、实体、骨骼)

intermediate
任意坐标系间的坐标转换(世界、实体、骨骼)
  • 概述
  • 背景知识
    • 矩阵
  • 开始前的世界坐标系认知
  • 实体操作实战
    • 1: 逆平移
    • 2: 逆旋转Z
    • 3: 逆旋转Y
    • 4: 逆旋转X
    • 5: 逆缩放

概述 ​

本文将探讨如何在Minecraft中实现不同坐标系间的转换。这种转换在以下场景中尤为重要:

  • 使用细长立方体构建3D线条来可视化空间时,需要将世界坐标转换为实体骨骼坐标
  • 实现精确头部追踪时,需要测量实体特定关节到目标的夹角
  • 从武器尖端发射投射物时
  • 为肢体解算IK链以匹配地面接触点时

背景知识 ​

在深入指南前,我们需要了解几个基础概念。

矩阵 ​

在图形处理中,我们通常将数据转换为矩阵形式,因为通过矩阵变换大量顶点(如网格顶点)效率极高。游戏开发中除了矩阵,还会使用四元数或Minecraft中的欧拉旋转等表示方式,但矩阵是最基础的理解起点。

一个4x4矩阵对初学者可能看起来像天书:

但实际上,3D变换矩阵通常只包含位置偏移和由"轴向向量"表示的旋转,这些向量分别描述X、Y、Z轴的三分量向量。

在3x3矩阵中,方向向量是单位长度向量(除非有缩放),这些向量的x,y,z分量定义了该空间的轴向:

教科书中的3x3矩阵通常按列排列:

[ X轴.x  Y轴.x  Z轴.x ]
[ X轴.y  Y轴.y  Z轴.y ]
[ X轴.z  Y轴.z  Z轴.z ]
1
2
3

实际编程中,数据存储顺序始终为<X轴.x, X轴.y, X轴.z, Y轴.x, Y轴.y, Y轴.z, Z轴.x, Z轴.y, Z轴.z>。行优先或列优先存储(行列优先顺序维基)只影响乘法运算顺序:

矩阵乘法始终是行×列。

行优先的向量矩阵乘法为:行向量 * 矩阵 = 行向量:

列优先要实现相同效果需要逆序:

这个区别非常重要,特别是在查阅在线资料时需要理解变换顺序的影响。

因此,如果我们有一个相对于右手的局部坐标,要转换为世界坐标,需要依次经过右手→右肘→右肩→脊柱→骨盆→根骨骼→实体等变换。用列优先表示就是:

实体 * 根骨骼 * 骨盆 * 脊柱0..脊柱N * 右肩 * 右肘 * 右手 * 点;
1

对于复杂变换,强烈建议用"从...到..."的命名方式明确变换空间。例如:

世界到实体 * 实体到根骨骼 * 根骨骼到骨盆 * 骨盆到脊柱0..脊柱(N-1)到脊柱N * 脊柱N到右肩 * 右肩到右肘 * 右肘到右手 * 右手点;
1

这种命名方式能清晰表明所处的"空间"。上述过程称为"局部空间"变换,每个关节都相对于父关节。如果我们将从根骨骼到右手的所有变换相乘,结果仍是右手变换,只是处于"角色或实体空间"。要转换到世界空间还需乘以世界到实体 * 实体到右手 = 世界到右手。

严格来说,"到"的命名有些混淆,因为实体变换实际上是实体到世界,但由于从右向左应用,世界到实体从右向左读就是实体到世界。另一种命名是世界来自实体 * 实体来自根骨骼。这种相对命名的优势在于可以通过以下方式验证数学正确性:

A到B = A到某物 * 某物到B
          ^^^^^^^^^^^^^^^^^^^^^
1
2

标记区域必须始终匹配。合并后能得到自然的变换名称。

回到Minecraft,目前游戏没有提供可相乘的变换矩阵,因此不涉及行列优先问题。你只需按顺序应用获得的变换即可。

无论是使用TRS(变换旋转缩放组合对象)、四元数、矩阵还是欧拉角,涉及旋转时顺序至关重要。先旋转A再旋转B,与先旋转B再旋转A结果不同。

开始前的世界坐标系认知 ​

让我们思考标准无旋转状态下的轴向向量。通过游戏移动可以确定正X、Y、Z方向:初次生成时面朝北方,前进增加Z值,跳跃增加Y值,向左平移增加X值。这是右手坐标系(手指指向一个轴,弯曲指向相邻轴,拇指指向第三个轴:XY→Z,YZ→X,ZX→Y)。

实体操作实战 ​

创建实体时,建议先在Blockbench中制作一个简单的三轴坐标系:

需要注意几个特殊现象:

  1. Blockbench的"北东南西"标签方向与Minecraft世界坐标系有180度旋转差异。实体应面朝"北方",即游戏世界坐标的负Z方向
  2. 创建骨骼时,动画选项卡中的移动手柄在X正方向拖动会产生负值。+X动画方向实际为西,+Y仍向上,+Z仍向南。相比世界坐标,实体需翻转Z轴并使用左手坐标系
  3. 实体存在16倍的缩放因子,世界中1单位方块等于实体的16单位

让我们将三轴模型分组到骨骼下,并复制一组用于世界定位:

操作步骤:

  1. 进入动画选项卡
  1. 创建新动画
  1. 为移动器添加位置关键帧
  1. 测试X轴移动确认异常现象
  1. 通过pre_animation脚本变量设置位置

可以参考基础机器人示例来初始化实体。行为端只需最简配置:

json
"minecraft:physics": {},
"minecraft:collision_box": {},
1
2

实体端只需支持动画播放:

json
"animations": {
    "myAnim": "animation.tut_transform.move"
},
"scripts": {
    "pre_animation": [
        "// 待填充内容"
    ],
    "animate": [
        "myAnim"
    ]
}
1
2
3
4
5
6
7
8
9
10
11

脚本内容如下:

json
"
v.target.x = 10;
v.target.y = q.position(1);
v.target.z = 10;

v.target.x = v.target.x - q.position(0);
v.target.y = v.target.y - q.position(1);
v.target.z = v.target.z - q.position(2);

t.cos_yaw = math.cos(q.body_y_rotation);
t.sin_yaw = math.sin(q.body_y_rotation);
t.x = v.target.x;
v.target.x=t.cos_yaw * t.x + t.sin_yaw * v.target.z;
v.target.z=-t.sin_yaw * t.x + t.cos_yaw * v.target.z;

v.target.x = v.target.x * 16;
v.target.y = v.target.y * 16;
v.target.z = -v.target.z * 16;
"
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

代码解析:

pre_animation在动画前执行,用于设置位置。以下是硬编码的世界坐标(10,y,10)转换过程:

v.target.x = 10;
v.target.y = q.position(1);
v.target.z = 10;
1
2
3

我们实际上是在应用"TRS"(平移、旋转、缩放)变换来实现空间转换。数学上正向变换堆栈为:

平移 * 旋转Z * 旋转Y * 旋转X * 缩放 * 点;
1

但这里我们需要的是世界到实体的逆变换。对于不可交换的数学运算,逆运算应用规则为:

逆(A*B) = 逆(B) * 逆(A)
1

因此对向量应用以下逆序操作:

  1. 逆平移
  2. 逆旋转Z
  3. 逆旋转Y
  4. 逆旋转X
  5. 逆缩放

数学表达式为:

逆缩放 * 逆旋转X * 逆旋转Y * 逆旋转Z * 逆平移 * 点;
1

1: 逆平移 ​

正向变换是实体相对位置加上实体位置,逆变换则是减去实体位置:

v.target_x = v.target_x - q.position(0);
v.target_y = v.target_y - q.position(1);
v.target_z = v.target_z - q.position(2);
1
2
3

2: 逆旋转Z ​

目前实体似乎只能通过控制器调整俯仰和偏航,没有Z旋转,故跳过。

3: 逆旋转Y ​

通过q.body_y_rotation查询实体偏航。正旋转使角色左转。向量旋转公式使用sin和cos,但符号很重要。初始面向世界+Z,左转时+X轴先正后负。Z轴则变为负值。临时变量t.x保存目标值:

t.cos_yaw = math.cos(q.body_y_rotation);
t.sin_yaw = math.sin(q.body_y_rotation);
t.x = v.target_x;
v.target_x=t.cos_yaw * t.x + t.sin_yaw * v.target_z;
v.target_z=-t.sin_yaw * t.x + t.cos_yaw * v.target_z;
1
2
3
4
5

更通用的写法是:

新第一轴 = cos(角) * 第一轴 - sin(角) * 第二轴;
新第二轴 = sin(角) * 第一轴 + cos(角) * 第二轴;
1
2

其中第一、第二轴是旋转轴的垂直轴,按右手顺序:XY、YZ或ZX。

4: 逆旋转X ​

实体可能有俯仰旋转,但实践中较少见,故跳过。后续骨骼变换部分会提供更多相关信息。

5: 逆缩放 ​

最后将世界单位转换为实体单位(乘以16)。从实体到世界则是除以16。由于Blockbench动画中X轴表现与预期相反(但实际与世界坐标系一致),而Z轴仍相反,故在缩放步骤对Z取反:

v.target.x = v.target.x * 16;
v.target.y = v.target.y * 16;
v.target.z = -v.target.z * 16;
1
2
3

贡献者

编辑 任意坐标系间的坐标转换(世界、实体、骨骼)

Bedrock Wiki by Bedrock OSS ,Translate by 8aka-Team

"Minecraft"是Mojang AB的注册商标。

Bedrock OSS、Bedrock Wiki以及 bedrock.dev 与Microsoft及Mojang AB不存在任何隶属关系。

  • 隐私政策
  • 加入QQ社区
  • 参与贡献指南
  • 访问代码仓库