Unity 模拟高度尺系统开发详解——实现拖动、范围限制、碰撞吸附与本地坐标轴选择

article/2025/6/21 7:14:55

内容将会持续更新,有错误的地方欢迎指正,谢谢!
 

Unity 模拟高度尺系统开发详解——实现拖动、范围限制、碰撞吸附与本地坐标轴选择
     
TechX 坚持将创新的科技带给世界!

拥有更好的学习体验 —— 不断努力,不断进步,不断探索
TechX —— 心探索、心进取!

助力快速掌握 物理引擎 学习

为初学者节省宝贵的学习时间,避免困惑!


前言:

  在Unity开发中,构建一个可交互的高度尺控制器是许多3D工具类项目中的常见需求。本文将基于最新的 HeightGaugeController.cs 脚本,详细介绍如何通过 鼠标拖拽、坐标转换、触发器碰撞检测和轴向限制 实现一个完整的高度尺系统。


文章目录

  • 🧭 一、项目背景与目标
  • 🛠️ 二、对象层级结构与组件配置
    • 1. 场景层级结构建议如下:
    • 2. 关键组件说明
  • 🔧 三、核心功能解析
    • 1. 拖动逻辑
    • 2. 限定上下范围
    • 3. 碰撞检测并吸附表面
    • 4. 使用本地坐标,支持轴向选择
    • 5. 屏幕坐标 → 世界坐标 → 本地坐标转换(关键逻辑)
  • 📐 四、核心函数 GetMeasurePos() 解析
  • 📌 五、变量管理与状态同步
  • 💡 六、自定义组件:MeasurementObject.cs
  • ⚙️ 七、完整代码清单(含注释)
  • 八、项目地址


🧭 一、项目背景与目标


在工业仿真、VR/AR 教学或虚拟装配场景中,我们经常需要模拟现实中的“高度尺”、“卡尺”等测量工具。

本教程将带你一步步实现一个 Unity 中的高度尺模拟系统 ,支持:

✅ 鼠标拖动爪子
✅ 精准限制移动范围(上下限)
✅ 碰撞物体后自动吸附表面
✅ 支持 X/Y/Z 轴向选择
✅ 使用本地坐标系,确保旋转不影响方向
✅ 屏幕坐标 → 世界坐标 → 本地坐标准确转换
✅ 对象层级结构与设置说明

在这里插入图片描述



🛠️ 二、对象层级结构与组件配置


1. 场景层级结构建议如下:


HeightGauge(空对象)
├──Scaleplate
│	├── Claw (爪子)  
│   │──	├── BoxCollider(勾选 isTrigger)
│   └──	└── HeightGaugeController.cs  
│	│──	LowestLevel (底部刻度点)  
│   └── └── Transform.localPosition.y = 0(作为参考点)  
│	├── HighestLevel (顶部刻度点)  
│   └── └── Transform.localPosition.y = 10(作为上限)  
MeasurementObject (被测物体)  └── Collider + MeasurementObject.cs(定义接触面位置)

2. 关键组件说明


BoxCollider (勾选 isTrigger )用于触发检测,判断是否点击或碰到物体
HeightGaugeController.cs核心脚本,控制拖动、限制、吸附、坐标转换
MeasurementObject.cs提供触碰时的表面位置

在这里插入图片描述


🔧 三、核心功能解析


1. 拖动逻辑


private bool RaycastGrabClaw()
{Ray ray = Camera.main.ScreenPointToRay(Input.mousePosition);RaycastHit hit;if (Physics.Raycast(ray, out hit) && hit.transform == transform){return true;}return false;
}

原理说明:

  • 使用射线检测判断是否点击到了当前爪子对象
  • 如果返回 true,表示可以开始拖动
  • 记录初始鼠标位置和爪子位置,用于后续计算偏移量

2. 限定上下范围


newPosition.y = Mathf.Clamp(newPosition.y, lowestLevel.localPosition.y, highestLevel.localPosition.y);

实现逻辑:

  • 在 GetMeasurePos() 中对新位置做 Mathf.Clamp() 限制
  • 确保不会超出标尺上下限
  • 支持 X/Y/Z 轴自由切换

3. 碰撞检测并吸附表面


使用 Trigger 系统来检测是否接触到测量物体,并记录其表面位置:
private void OnTriggerStay(Collider other)
{if (((1 << other.gameObject.layer) & collisionLayer.value) != 0){isClawColliding = true;MeasurementObject measurement = other.GetComponent<MeasurementObject>();colliderPosition = measurement.surface.position;}
}private void OnTriggerExit(Collider other)
{if (((1 << other.gameObject.layer) & collisionLayer.value) != 0){isClawColliding = false;colliderPosition = Vector3.zero;}
}

吸附逻辑:

  • 判断是否是向下移动
    -若发生碰撞,则更新位置为接触点
  • 不允许继续下移,但允许上移
if (newPosition.y <= clawTargetPos.y && isClawColliding)
{newPosition.y = clawTargetPos.y;
}

4. 使用本地坐标,支持轴向选择


public enum MeasurementAxis
{X,Y,Z
}

根据枚举值动态选择轴向:

switch (measurementAxis)
{case MeasurementAxis.X:// X 方向移动逻辑break;case MeasurementAxis.Y:// Y 方向移动逻辑break;case MeasurementAxis.Z:// Z 方向移动逻辑break;
}

优点:

  • 爪子即使旋转也不会影响移动方向
  • 支持横版、竖版等多种测量方式

5. 屏幕坐标 → 世界坐标 → 本地坐标转换(关键逻辑)


float GetDepthInCameraSpace()
{return Vector3.Dot(transform.position - Camera.main.transform.position, Camera.main.transform.forward);
}Vector3 ScenePointToLocalPoint()
{float depth = GetDepthInCameraSpace();Vector3 worldPos = Camera.main.ScreenToWorldPoint(new Vector3(Input.mousePosition.x, Input.mousePosition.y, depth));return WordPointToLocalPoint(worldPos);
}private Vector3 WordPointToLocalPoint(Vector3 worldPosition)
{return transform.parent.InverseTransformPoint(worldPosition);
}

转换流程:

  • 获取相机到爪子的 Z 轴投影作为深度值
  • 使用 ScreenToWorldPoint 转换屏幕坐标为世界坐标
  • 再通过 InverseTransformPoint 转换为父级下的本地坐标

📌 注意:

  • 深度值不能直接用 Input.mousePosition.z,而是要用相机空间中的 Z 投影
  • 这样才能保证坐标转换准确无误


📐 四、核心函数 GetMeasurePos() 解析


private Vector3 GetMeasurePos()
{Vector3 newPosition = Vector3.zero;switch (measurementAxis){case MeasurementAxis.X:newPosition = clawInitialPos + new Vector3(delta.x, 0, 0);newPosition.x = Mathf.Clamp(newPosition.x, lowestLevel.localPosition.x, highestLevel.localPosition.x);if (newPosition.x <= clawTargetPos.x && isClawColliding){newPosition.x = clawTargetPos.x;}break;case MeasurementAxis.Y:newPosition = clawInitialPos + new Vector3(0, delta.y, 0);newPosition.y = Mathf.Clamp(newPosition.y, lowestLevel.localPosition.y, highestLevel.localPosition.y);if (newPosition.y <= clawTargetPos.y && isClawColliding){newPosition.y = clawTargetPos.y;}break;case MeasurementAxis.Z:newPosition = clawInitialPos + new Vector3(0, 0, delta.z);newPosition.z = Mathf.Clamp(newPosition.z, lowestLevel.localPosition.z, highestLevel.localPosition.z);if (newPosition.z <= clawTargetPos.z && isClawColliding){newPosition.z = clawTargetPos.z;}break;}return newPosition;
}

流程说明:

  • 获取增量 :delta = mouseCurrentPos - mouseInitialPos
  • 计算新位置 :基于初始位置 + 增量
  • 限制范围 :使用 Mathf.Clamp() 防止越界
  • 碰撞处理 :如果是向下移动且发生碰撞,禁止进一步下移


📌 五、变量管理与状态同步


为了避免松开鼠标后仍执行一次赋值导致误动作,加入了状态同步机制:

private void ResetVariables()
{mouseCurentPos = Vector3.zero;mouseInitialPos = Vector3.zero;clawInitialPos = Vector3.zero;clawTargetPos = Vector3.zero;delta = Vector3.zero;
}

并在松开鼠标时调用:

if (Input.GetMouseButtonUp(0))
{isDragging = false;ResetVariables();
}


💡 六、自定义组件:MeasurementObject.cs


为了让测量物体能提供“表面位置”,我们创建一个辅助类:

public class MeasurementObject : MonoBehaviour
{public Transform surface; // 表面位置(如物体顶部)
}

你可以把这个组件挂在测量物体上,并设置一个 Transform 来代表“接触面”的位置。



⚙️ 七、完整代码清单(含注释)


以下是完整 C# 脚本,已加入详细注释,方便理解和复用。

文件名:HeightGaugeController.cs

using UnityEngine;
public enum MeasurementAxis
{X,Y,Z
}public class HeightGaugeController : MonoBehaviour
{// 底部标尺的 Transformpublic Transform lowestLevel;// 顶部标尺的 Transformpublic Transform highestLevel;// 测量轴public MeasurementAxis measurementAxis = MeasurementAxis.Y;// 用于指定哪些层可以被击中public LayerMask collisionLayer;// 鼠标初始位置private Vector3 mouseInitialPos;// 鼠标当前目标位置private Vector3 mouseCurentPos;// 增量private Vector3 delta;// 爪子初始位置private Vector3 clawInitialPos;// 爪子目标位置private Vector3 clawTargetPos;private bool isDragging = false;public bool isClawColliding = false;private Vector3 colliderPosition; // 爪子碰撞器位置public float measureHeight; // 爪子高度void Update(){if (Input.GetMouseButtonDown(0)){if (RaycastGrabClaw()){isDragging = true;mouseInitialPos = ScenePointToLocalPoint();clawInitialPos = transform.localPosition;}}if (isDragging && Input.GetMouseButton(0)){mouseCurentPos = ScenePointToLocalPoint();delta = mouseCurentPos - mouseInitialPos;clawTargetPos = GetMeasurePos();transform.localPosition = clawTargetPos;measureHeight = (transform.localPosition - lowestLevel.localPosition).y;}if (Input.GetMouseButtonUp(0)){isDragging = false;ResetVariables();}}private void ResetVariables(){mouseCurentPos = Vector3.zero;  // 重置目标位置mouseInitialPos = Vector3.zero; // 重置初始位置clawInitialPos = Vector3.zero;  // 重置初始爪子位置clawTargetPos = Vector3.zero;   // 重置当前目标位置delta = Vector3.zero;           // 重置增量}/// <summary>/// 检测鼠标是否点击了爪子,并返回是否成功点击。/// </summary>/// <returns></returns>private bool RaycastGrabClaw(){// 检测鼠标是否点击了爪子Ray ray = Camera.main.ScreenPointToRay(Input.mousePosition);RaycastHit hit;if (Physics.Raycast(ray, out hit) && hit.transform == transform){return true;}return false;}/// <summary>/// 这里把屏幕坐标转换为世界坐标,使用target在相机空间中的 Z 轴深度。/// </summary>/// <param name="target"></param>Vector3 ScenePointToLocalPoint(){Vector3 currentScreenPos = Input.mousePosition;float depth = GetDepthInCameraSpace(); // 使用相同深度Vector3 worldPos = Camera.main.ScreenToWorldPoint(new Vector3(currentScreenPos.x, currentScreenPos.y, depth));return WordPointToLocalPoint(worldPos);}/// <summary>/// 获取物体在相机本地空间中的 Z 值(这才是 ScreenToWorldPoint 需要的深度)/// </summary>/// <param name="target"></param>/// <returns></returns>float GetDepthInCameraSpace(){return Vector3.Dot(transform.position - Camera.main.transform.position, Camera.main.transform.forward);}/// <summary>/// 把世界坐标转换为本地坐标/// </summary>/// <param name="worldPosition"></param>/// <returns></returns>private Vector3 WordPointToLocalPoint(Vector3 worldPosition){return transform.parent.InverseTransformPoint(worldPosition);}private Vector3 GetMeasurePos(){Vector3 newPosition = Vector3.zero;switch (measurementAxis){case MeasurementAxis.X:newPosition = clawInitialPos + new Vector3(delta.x, 0, 0);// 限制范围  newPosition.x = Mathf.Clamp(newPosition.x, lowestLevel.localPosition.x, highestLevel.localPosition.x);// 判断是否向下移动,并且碰撞了测量物体  if (newPosition.x <= clawTargetPos.x && isClawColliding){clawTargetPos = WordPointToLocalPoint(colliderPosition);// 如果是向下移动并且碰到物体,不允许继续下移  newPosition.x = clawTargetPos.x;}break;case MeasurementAxis.Y:newPosition = clawInitialPos + new Vector3(0, delta.y, 0);// 限制范围  newPosition.y = Mathf.Clamp(newPosition.y, lowestLevel.localPosition.y, highestLevel.localPosition.y);// 判断是否向下移动,并且碰撞了测量物体  if (newPosition.y <= clawTargetPos.y && isClawColliding){clawTargetPos = WordPointToLocalPoint(colliderPosition);// 如果是向下移动并且碰到物体,不允许继续下移  newPosition.y = clawTargetPos.y;}break;case MeasurementAxis.Z:newPosition = clawInitialPos + new Vector3(0, 0, delta.z);// 限制范围  newPosition.z = Mathf.Clamp(newPosition.z, lowestLevel.localPosition.z, highestLevel.localPosition.z);// 判断是否向下移动,并且碰撞了测量物体  if (newPosition.z <= clawTargetPos.z && isClawColliding){clawTargetPos = WordPointToLocalPoint(colliderPosition);// 如果是向下移动并且碰到物体,不允许继续下移  newPosition.z = clawTargetPos.z;}break;}return newPosition;}private void OnTriggerStay(Collider other){if (((1 << other.gameObject.layer) & collisionLayer.value) != 0){isClawColliding = true;MeasurementObject measurement = other.GetComponent<MeasurementObject>();colliderPosition = measurement.surface.position;}}private void OnTriggerExit(Collider other){if (((1 << other.gameObject.layer) & collisionLayer.value) != 0){isClawColliding = false;colliderPosition = Vector3.zero;}}
}

在这里插入图片描述



八、项目地址


以下是项目地址,有需要的小伙伴门可以自取:

https://download.csdn.net/download/caiprogram123/90943926





TechX —— 心探索、心进取!

每一次跌倒都是一次成长

每一次努力都是一次进步


END
感谢您阅读本篇博客!希望这篇内容对您有所帮助。如果您有任何问题或意见,或者想要了解更多关于本主题的信息,欢迎在评论区留言与我交流。我会非常乐意与大家讨论和分享更多有趣的内容。
如果您喜欢本博客,请点赞和分享给更多的朋友,让更多人受益。同时,您也可以关注我的博客,以便及时获取最新的更新和文章。
在未来的写作中,我将继续努力,分享更多有趣、实用的内容。再次感谢大家的支持和鼓励,期待与您在下一篇博客再见!

http://www.hkcw.cn/article/XNawRHmLml.shtml

相关文章

Spark核心:单跳转换率计算全解析

目录 代码功能解释与问题分析 关键问题分析 修正与拓展方案 1. 修正分子计算逻辑 2. 修正分母计算逻辑 3. 完善转换率计算 4. 优化代码结构 5. 性能优化 修正后的代码示例 关键改进点说明 测试与验证建议 package core.reqimport org.apache.spark.rdd.RDD import o…

基于STM32单片机CO气体检测

基于STM32单片机CO检测 &#xff08;仿真&#xff0b;程序&#xff0b;原理图&#xff09; 功能介绍 具体功能&#xff1a; 1.MQ-7传感器检测CO气体浓度&#xff1b; 2.LCD1602实时显示气体浓度及上限值&#xff1b; 3.气体浓度超过设定对应上限值&#xff0c;电机转动&…

MySQL事务

事务&#xff08;Transaction&#xff09;是数据库管理系统中一组操作的集合&#xff0c;作为一个单元要么全部成功&#xff0c;要么全部失败&#xff0c;确保数据的一致性和完整性。它像一个“原子操作单元”&#xff0c;遵循ACID原则&#xff08;原子性、一致性、隔离性、持久…

C# 反射与特性:深入探索运行时类型系统与元数据编程

在C#开发中&#xff0c;我们通常编写静态类型的代码——编译器在编译时就知道所有类型信息。然而&#xff0c;.NET框架提供了一套强大的机制&#xff0c;允许我们在运行时检查、发现和使用类型信息&#xff0c;这就是反射(Reflection)。而与反射密切相关的另一项技术是特性(Att…

腾讯面试手撕题:返回行递增有序矩阵第k小的元素

题目 给定一个n行n列的矩阵&#xff0c;这个矩阵的每一行是递增有序的&#xff0c;求这个矩阵中第k小的元素。 解答 优解 基于二分查找和按行统计小于等于目标值的元素个数。算法的时间复杂度为&#xff0c;其中D是矩阵中元素值域的范围&#xff08;即最大值与最小值的差&a…

【PostgreSQL 02】PostgreSQL数据类型革命:JSON、数组与地理信息让你的应用飞起来

PostgreSQL数据类型革命&#xff1a;JSON、数组与地理信息让你的应用飞起来 关键词 PostgreSQL高级数据类型, JSONB, 数组类型, PostGIS, 地理信息系统, NoSQL, 文档数据库, 空间数据, 数据库设计, PostgreSQL扩展 摘要 PostgreSQL的高级数据类型是其区别于传统关系数据库的核心…

[Windows] 剪映 视频编辑处理

附链接&#xff1a;夸克网盘分享&#xff08;点击蓝色字体自行保存下载&#xff09;

NW994NX734美光固态闪存NX737NX740

NW994NX734美光固态闪存NX737NX740 在数字化浪潮汹涌澎湃的今天&#xff0c;数据存储技术如同一座坚实的基石&#xff0c;支撑着科技世界的大厦。美光固态闪存以其卓越的性能和创新的技术&#xff0c;在存储领域占据着重要的地位。本文将深入剖析NW994、NX734、NX737以及NX740…

C# 类和继承(使用基类的引用)

使用基类的引用 派生类的实例由基类的实例和派生类新增的成员组成。派生类的引用指向整个类对象&#xff0c;包括 基类部分。 如果有一个派生类对象的引用&#xff0c;就可以获取该对象基类部分的引用&#xff08;使用类型转换运算符把 该引用转换为基类类型&#xff09;。类…

VMvare 创建虚拟机 安装CentOS7,配置静态IP地址

创建虚拟机 安装CentOS7 设置网络模式 设置静态ip vim /etc/sysconfig/network-scripts/ifcfg-ens33 systemctl restart network

python:PyMOL 能处理 *.pdb 文件吗?

PyMOL 完全可以打开并处理 PDB&#xff08;Protein Data Bank&#xff09;文件&#xff0c;这是 PyMOL 最主要的功能之一。PDB 格式是结构生物学领域的标准文件格式&#xff0c;专门用于存储生物大分子&#xff08;如蛋白质、核酸&#xff09;的三维结构数据。 在 PyMOL 中打开…

【数据治理】要点整理-信息技术数据质量评价指标-GB/T36344-2018

导读&#xff1a;指标为数据质量评估提供了一套系统化、标准化的框架&#xff0c;涵盖规范性、完整性、准确性、一致性、时效性、可访问性六大核心指标&#xff0c;助力组织提升数据处理效率、支持决策制定及业务流程优化&#xff0c;确保数据在数据生存周期各阶段的质量可控。…

【Redis】hash 类型

hash 一. hash 类型介绍二. hash 命令hset、hgethexists、hdelhkeys、hvals、hgetallhmset、hmgethlen、hstrlen、hsetnxhincrby、hincrbyfloat 三. hash 命令小结四. hash 内部编码方式五. hash 的应用场景缓存功能缓存方式对比 一. hash 类型介绍 哈希表在日常开发中&#x…

ubuntu/windows系统下如何让.desktop/.exe文件 在开机的时候自动运行

目录 1&#xff0c;​​让 .desktop 文件在 Ubuntu 开机时自动启动​ 1.1 创建 autostart 目录&#xff08;如果不存在&#xff09;​ ​ 1.2 将 .desktop 文件复制到 autostart 目录​ ​ 1.3 确保 .desktop 文件有可执行权限​ 2,windows 2.1 打开「启动」文件夹​…

1-Wire 一线式总线:从原理到实战,玩转 DS18B20 温度采集

引言 在嵌入式系统中&#xff0c;通信总线是连接 CPU 与外设的桥梁。从 I2C、SPI 到 UART&#xff0c;每种总线都有其独特的应用场景。而本文要介绍的1-Wire 一线式总线&#xff0c;以其极简的硬件设计和独特的通信协议&#xff0c;在温度采集、身份识别等领域大放异彩。本文将…

有机黑鸡蛋与普通鸡蛋:差异剖析与选购指南

在我们的日常饮食结构里&#xff0c;鸡蛋始终占据着不可或缺的位置&#xff0c;是人们获取营养的重要来源。如今&#xff0c;市场上鸡蛋种类丰富&#xff0c;除了常见的普通鸡蛋&#xff0c;有机黑鸡蛋也逐渐崭露头角&#xff0c;其价格通常略高于普通鸡蛋。这两者究竟存在哪些…

Fastapi 学习使用

Fastapi 学习使用 Fastapi 可以用来快速搭建 Web 应用来进行接口的搭建。 参考文章&#xff1a;https://blog.csdn.net/liudadaxuexi/article/details/141062582 参考文章&#xff1a;https://blog.csdn.net/jcgeneral/article/details/146505880 参考文章&#xff1a;http…

数字化转型进阶:精读41页华为数字化转型实践【附全文阅读】

该文档聚焦华为数字化转型实践&#xff0c;核心内容如下&#xff1a; 转型本质与目标&#xff1a;数字化转型是通过数字技术穿透业务&#xff0c;实现物理世界与数字世界的融合&#xff0c;目标是支撑主业成功、提升体验与效率、探索模式创新。华为以 “平台 服务” 为核心&am…

共享内存-systemV

01. 共享内存简述 共享内存是一个允许多个进程直接访问同一块物理内存区域的进程通信工具&#xff0c;因其本身不涉及用户态与核心态之间转换&#xff0c;故效率最佳。为了使用一个共享内存段&#xff0c;一般需要以下几个步骤&#xff1a; 调用shmget()创建一个新共享内存段…

大语言模型值ollama使用(1)

ollama为本地调用大语言模型提供了便捷的方式。下面列举如何在windows系统中快捷调用ollama。 winR打开运行框&#xff0c;输入cmd 1、输入ollama list 显示已下载模型 2、输入ollama pull llama3 下载llama3模型 3、 输入 ollama run llama3 运行模型 4、其他 ollama li…