Unity 进阶 之 AR/VR 3D空间场景中Laser镭射线拖拽UI实现问题的简单整理

news2025/1/26 14:24:21

Unity 进阶 之 AR/VR 3D场景中Laser镭射线拖拽UI实现问题的简单整理

目录

Unity 进阶 之 AR/VR 3D场景中Laser镭射线拖拽UI实现问题的简单整理

一、简单介绍

二、实现原理

三、注意事项

四、效果预览

五、简单实现步骤

常规拖拽

常规拖拽在3D空间拖拽位置跳动问题

解决常规拖拽在3D空间拖拽位置跳动问题的简单方法

六、关键代码


一、简单介绍

Unity中的一些知识点整理。

本节简单介绍在Unity开发中的,在AR/VR 开发中,有时候需要在 3D 空间进行 UI 元素的拖拽功能,或者即把UI元素从一个面板位置拖拽到另一个面板位置,有时候Laser 镭射线拖拽 UI 元素,不小心拖出屏幕的时候,常规的拖拽操作就会出现位置错误,如果您也遇到了类似情况,这里提供一个简单思路方法,提供参考,如果你有新的方式也可以留言,多谢。

二、实现原理

1、常规的 UI 拖拽功能,一般使用 RectTransformUtility.ScreenPointToWorldPointInRectangle 或者 RectTransformUtility.ScreenPointToLocalPointInRectangle 进行位置的计算

1、RectTransformUtility.ScreenPointToWorldPointInRectangle(this.gameObject.GetComponent<RectTransform>(), eventData.position, eventData.enterEventCamera, out pos);

2、RectTransformUtility.ScreenPointToLocalPointInRectangle(m_canvsGo.GetComponent<RectTransform>(), eventData.position, eventData.enterEventCamera, out pos);

2、3D 空间场景(AR/VR)中的 UI 拖拽,一般常规功能也可以适用,这里介绍使用 Laser 镭射线交互得到的 eventData.position 转为世界坐标的位置赋值给 UI 使用,来实现 3D 空间场景(AR/VR)中的 UI 拖拽

        Vector3 pos = eventData.position;
        pos.z = 1080;
        this.transform.position = Camera.main.ScreenToWorldPoint(pos) + m_OffsetPos;

三、注意事项

1、拖拽 UI 的 时候,为了避免移动的突兀,在开始移动的时候注意添加一个移动偏差值,在正式移动的时候做位置赋值的时候注意添加上偏移量,其中 pos.z 的数值是, Canvas 位置 z 相对Camera 位置的 z (默认camera 位置 Vector.zero 方向朝前z,Canvas 在 z 向位置)

1、移动开始时计算出移动偏差值
        Vector3 pos = eventData.position;
        pos.z = 1080;
        m_OffsetPos = this.transform.position - Camera.main.ScreenToWorldPoint(pos);
2、移动的时候添加上移动的偏差值
        Vector3 pos = eventData.position;
        pos.z = 1080;
        this.transform.position = Camera.main.ScreenToWorldPoint(pos) + m_OffsetPos;

四、效果预览

五、简单实现步骤

常规拖拽

1、打开 Unity 创建场景,在 场景中添加移动的 UI -Image

2、创建脚本,实现 Image 常规的拖拽功能

3、把脚本挂载到场景 Image 上

4、运行场景,效果如图

常规拖拽在3D空间拖拽位置跳动问题

5、切换到 3D 空间场景中,使用Laser镭射线,代替鼠标交互移动 UI Image,效果如图

(Laser镭射线交互当,不小心移动到屏幕外,常规的拖拽移动UI的方法会出现位置跳动)

解决常规拖拽在3D空间拖拽位置跳动问题的简单方法

6、根据常规拖拽出现的位置跳动问题,这里使用把获取到的 eventData.position 转为 世界坐标赋值给移动 UI - Image ,代码如下

7、在  3D 空间场景中,使用Laser镭射线交互,效果如下

六、关键代码

1、Drag.cs

using UnityEngine;
using UnityEngine.EventSystems;

public class Drag : MonoBehaviour, IBeginDragHandler, IEndDragHandler, IDragHandler
{
    private Vector3 m_OffsetPos;



    public void OnBeginDrag(PointerEventData eventData)
    {
        Debug.Log("OnBeginDrag");
        Vector3 pos;
        RectTransformUtility.ScreenPointToWorldPointInRectangle(gameObject.GetComponent<RectTransform>(), eventData.position, eventData.enterEventCamera, out pos);
        m_OffsetPos = this.gameObject.GetComponent<RectTransform>().position - pos;
    }

    public void OnEndDrag(PointerEventData eventData)
    {
        Debug.Log("OnEndDrag");
    }

    public void OnDrag(PointerEventData eventData)
    {
        Debug.Log("OnDrag");
        Vector3 pos;
        RectTransformUtility.ScreenPointToWorldPointInRectangle(this.gameObject.GetComponent<RectTransform>(), eventData.position, eventData.enterEventCamera, out pos);
        this.gameObject.GetComponent<RectTransform>().position = pos + m_OffsetPos;
    }
}

2、DragItem

using UnityEngine;
using UnityEngine.EventSystems;

public class DragItem : MonoBehaviour, IBeginDragHandler, IEndDragHandler, IDragHandler
{
    private Vector3 m_OffsetPos;

    public void OnBeginDrag(PointerEventData eventData)
    {
        Debug.Log("OnBeginDrag");
        // 方法一:
        //Vector3 pos;
        //RectTransformUtility.ScreenPointToWorldPointInRectangle(gameObject.GetComponent<RectTransform>(), eventData.position, eventData.enterEventCamera, out pos);
        //m_OffsetPos = this.gameObject.GetComponent<RectTransform>().position - pos;

        // 方法二:略有问题,暂时不用
        //Vector2 pos;
        //RectTransformUtility.ScreenPointToLocalPointInRectangle(gameObject.GetComponent<RectTransform>(), eventData.position, eventData.enterEventCamera, out pos);
        //Vector3 tmp = this.gameObject.GetComponent<RectTransform>().position;
        //m_OffsetPos.x = tmp.x - pos.x;
        //m_OffsetPos.y = tmp.y - pos.y;
        //m_OffsetPos.z = tmp.z;

        // 方法三:
        Vector3 pos = eventData.position;
        pos.z = 1080; // 根据 Canvas 与 Camera 实际距离赋值
        m_OffsetPos = this.transform.position - Camera.main.ScreenToWorldPoint(pos);

    }

    public void OnEndDrag(PointerEventData eventData)
    {
        Debug.Log("OnEndDrag");
    }

    public void OnDrag(PointerEventData eventData)
    {
        // 方法一:
        //Debug.Log("OnDrag eventData.position " + eventData.position.ToString());
        //Vector3 pos;
        //RectTransformUtility.ScreenPointToWorldPointInRectangle(this.gameObject.GetComponent<RectTransform>(), eventData.position, eventData.enterEventCamera, out pos);
        //this.gameObject.GetComponent<RectTransform>().position = pos + m_OffsetPos;

        // 方法二:略有问题,暂时不用
        //Vector2 pos;
        //RectTransformUtility.ScreenPointToLocalPointInRectangle(gameObject.GetComponent<RectTransform>(), eventData.position, eventData.enterEventCamera, out pos);
        //this.gameObject.GetComponent<RectTransform>().position = new Vector3(pos.x+m_OffsetPos.x,pos.y+m_OffsetPos.y,m_OffsetPos.z);

        // 方法三:
        Vector3 pos = eventData.position;
        pos.z = 1080;// 根据 Canvas 与 Camera 实际距离赋值
        this.transform.position = Camera.main.ScreenToWorldPoint(pos) + m_OffsetPos;

        Debug.Log("OnDrag pos " + pos.ToString());
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/334002.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C++003-C++变量和数据类型2

文章目录C003-C变量和数据类型2C数据类型数据类型及定义数据类型及类型大小浮点数据的使用**题目描述**&#xff1a;求阴影面积**题目描述**&#xff1a;计算园的周长和面积如何取消科学计数法表示浮点数iomanip的作用比较多:布尔变量的使用隐式转换与显式转换**题目描述**&…

JAVA环境变量配置步骤及测试(JDK的下载 安装 环境配置教程)

一&#xff1a;JDK的下载、安装和配置1、输入下载地址&#xff1a;https://www.oracle.com/downloads/往下滑&#xff0c;看到Developer Downloads &#xff0c;点击 java点击 Java (JDK) for Developers选择你想要下载的jdk版本4、 点击 JDK Downloads &#xff0c;往下拉&…

树莓派系统创建指南

无意中发现一个落灰的树莓派 故事便开始了…… 准备工作 树莓派 3B一张大于 8G 的 micro SD 卡一个读卡器HDMI 显示器及连接线、键盘、鼠标等外围设备 系统镜像下载 推荐两个树莓派镜像下载网站 树莓派官方网站&#xff1a;https://www.raspberrypi.com/software/树莓派实…

python+request+pytest+pytest-html集成的API自动化测试框架

对于框架任何问题&#xff0c;欢迎联系我&#xff01; 需要框架源码的&#xff0c;请私聊我&#xff01; 一、框架架构 二、项目目录结构 ├────.gitignore ├────case_utils/ 测试用例相关的工具类&#xff0c;包括HTTP请求封装及用例数据处理&#xff0c;用例文件生…

常用的辅助类2(StringBuilder、StringBuffer、处理时间相关的类、对象比较器)

Java知识点总结&#xff1a;想看的可以从这里进入 目录7.7、字符串相关类7.8、时间处理7.8.1、JDK8前7.8.2、JDK8后1、时间日期类2、格式化日期3、其他7.9、对象比较器7.7、字符串相关类 String&#xff1a;JDK1.0出现&#xff0c;字符串类&#xff0c;被final修饰其值不可改。…

30个HTML+CSS前端开发案例(三)

30个HTMLCSS前端开发案例&#xff08;11-15&#xff09;小米上称右侧悬浮菜单实现代码效果图自动轮播图效果实现代码效果图小米商城二级下拉菜单效果实现代码效果图时间轴效果实现代码效果图QQ音乐排行榜效果实现代码效果图资源包获取小米上称右侧悬浮菜单 实现代码 <!DOC…

idea中使用Git

目录 一、在idea中配置Git 1、打开settings&#xff0c;搜索git&#xff0c;找到本地上的git安装目录&#xff0c;选择git.exe 2、本地git安装目录 二、获取Git 1、本地初始化仓库 2、选中项目这层目录&#xff0c;点击确定 2、从远程仓库克隆 三、本地仓库操作 1、将文…

阶段二4_常用API之StringBuilder

一. StringBuilder类概述 概述 : StringBuilder 是一个可变的字符串类&#xff0c;我们可以把它看成是一个容器&#xff0c;这里的可变指的是 StringBuilder 对象中的内容是可变的。 作用&#xff1a; 提高供字符串的操作效率 案例&#xff1a; /**证明StringBuilder提高供字…

api接口详解大全(看这篇就足以了)

api接口详解大全?优秀的设计是产品变得卓越的原因设计API意味着提供有效的接口&#xff0c;可以帮助API使用者更好地了解、使用和集成&#xff0c;同时帮助人们有效地维护它每个产品都需要使用手册&#xff0c;API也不例外在API领域&#xff0c;可以将设计视为服务器和客户端之…

函数/任意波形发生器 DG5072 技术资料

函数/任意波形发生器 DG5072 DG5000人性化的界面设计和键盘布局&#xff0c;给用户带来非凡体验&#xff1b;丰富的标准配置接口&#xff0c;可轻松实现仪器远程控制&#xff0c;为用户提供更多解决方案。 产品特性 4.3英寸16M真彩TFT液晶显示屏 350 MHz、250MHz、100 MHz或70…

Pacemaker详解、pcs命令详解和参数说明、centos8或bclinux8.2离线安装pcs以及搭建pcs、pcs的使用说明

文章目录Pacemaker详解一、前言二、 Pacemaker概述1、Pacemaker介绍2、pacemaker的服务模式。3、Pacemaker的架构4、Pacemake内部组件三、Pacemaker集群管理工具pcspcs说明最为常用的管理命令四、Pacemaker集群资源管理1、集群资源代理常用的命令方法2、集群资源约束3、集群资源…

国家政策鼓励使用电子保函和银行函证,君子签助推函证数字化建设

近日&#xff0c;国家发改委发文&#xff0c;推动电子保函应用&#xff0c;降低电子保函费用&#xff1b;财政部会同银保监会发文&#xff0c;开展数字化函证&#xff0c;有效提升函证效率和效果。政策的出台有助于加快推进普及电子函证应用。 发改委&#xff1a;鼓励使用电子…

PMP考试前两个月开始备考时间足够吗?

简单不代表报名费便宜&#xff0c;但报名费贵是肯定会难的&#xff0c;不然从何而来的含金量一说&#xff1f;花钱就能买到的话估计现在全中国持有pmp的一抓一大把。 时间完全足够的。 相信很多朋友都了解过&#xff0c;pmp的备考时间基本上是在2~3个月&#xff0c;2个月最为…

shell的变量和引用

文章目录二、变量和引用2.1 什么是变量2.2变量的命名2.3 变量的类型2.3.1 根据数据类型分类2.3.2 根据作用域分类2.4 变量的定义2.5 shell中的引用2.6 变量的运算练习&#xff1a;二、变量和引用 在程序设计语言中&#xff0c;变量是一个非常重要的概念。也是初学者在进行Shel…

计讯物联智慧灯杆网关数字赋能智慧公园,点亮城市新未来

方案背景 …… 公园作为城市景观与生态要素的空间载体&#xff0c;是市民远离尘嚣、缓解压力的理想休闲地。随着人们生活水平的提高&#xff0c;公园的信息化与智能化成为公园建设的必然方向。计讯物联依托于物联网技术&#xff0c;以绿色发展理念为指导&#xff0c;充分考虑…

/proc/cpuinfo详解

在分析该文件输出之前&#xff0c;我们先理解下几个重要的概念&#xff1a;物理CPU、CPU核心数、逻辑CPU。 物理CPU数&#xff08;physical id&#xff09;&#xff1a;主板上实际插入的cpu数量&#xff0c;不重复的 physical id 有几个就有多少个物理CPU。 CPU核心数&#xf…

95. 不同的二叉搜索树 II

95. 不同的二叉搜索树 II题目算法设计&#xff1a;深度优先搜索题目 传送门&#xff1a;https://leetcode.cn/problems/unique-binary-search-trees-ii/ 算法设计&#xff1a;深度优先搜索 二叉树子问题分解 根节点 左右子树的子问题。 根节点的子问题&#xff1a;循环历遍…

高压放大器由哪些部分组成?如何验证它性能的好坏?

虽然很多电子工程师经常使用高压放大器&#xff0c;但是对于高压功率放大器的组成和使用都不太清楚&#xff0c;下面由安泰测试给大家介绍高压放大器的组成部分以及如何验证功率放大器的性能好坏。 一、高压放大器的介绍 高压放大器是一种理想的功率放大器&#xff0c;可以放…

企业版快速开发框架源码 Spring Cloud快速开发脚手架源码

一款 Java 语言基于Spring Cloud Alibaba、SpringBoot、Mybatis、Seata、Sentinel、RabbitMQ、FastDFS/MinIO、SkyWalking等主要框架和中间件&#xff0c;精心打造的一款模块化、插件化、高性能的快速开发框架&#xff0c;可用于快速搭建后台管理系统。 私信了解更多&#xff…

(九)docker复杂安装-安装mysql主从复制

目录 前提&#xff1a; 一、新建主服务器容器实例3307 二、进入/mydata/mysql-master/conf目录下新建my.cnf 三、修改完配置后重启master实例 四、进入mysql-master容器并测试 五、master容器实例内创建数据同步用户 六、新建从服务器容器实例3308 七、进入/mydata/mys…