unity VR项目3DUI:人物头部旁“说话框”,功能:1.永远朝向相机 2.打字效果 3.UI不会被模型遮挡 4.切换位置 5.文字自动根据内容扩充

news2025/1/11 8:05:58

提示:文章有错误的地方,还望诸位大神不吝指教!

文章目录

  • 前言
  • 一、UI搭建
    • 1.创建基节点
    • 2.创建Canvas
    • 3.添加永远看向相机代码
    • 4.创建UI背景,设置相关操作
      • 1.锚点设置
      • 2.添加组件:Vertical Layout Group、Content Size Fitter
      • 3.添加文本
  • 二、UI不被模型遮挡
  • 三、打字效果代码
  • 总结


前言

仅适用于VRAR项目!
需求:
人物说话时:头旁边出现说话内容,且根据内容长短,以打字效果的方式,进行向下自动扩充,并且UI复用,可以多人物使用。
功能:
1.永远朝向相机
2.打字效果
3.UI不会被模型遮挡
4.切换位置
5.文字自动根据内容扩充
效果:
在这里插入图片描述
在这里插入图片描述

一、UI搭建

1.创建基节点

创建空物体作为UI的父节点,改名字:UIDialog,因为是3dUI,大小和高度根据自己的情况调节即可
在这里插入图片描述

2.创建Canvas

因为是VR项目,所以就用XR创建Canvas:在Hierarchy面板中右击鼠标–>XR–>UICanvas
在这里插入图片描述

3.添加永远看向相机代码

在这里插入图片描述


using UnityEngine;

public class LookatPlayer : MonoBehaviour
{
    Camera MainCamera;
    void Awake()
    {
        MainCamera = Camera.main;
        if (MainCamera == null)
        {
            Debug.LogError("没有找到MainCamera");
            return;
        }
    }


    void Update()
    {
        this.transform.LookAt(MainCamera.transform);
        // this.transform.forward = Vector3.forward;
        this.transform.localRotation = Quaternion.Euler(new Vector3(0, this.transform.localRotation.eulerAngles.y, 0));
    }
}

4.创建UI背景,设置相关操作

1.锚点设置

把Pivot的Y改为:1,即可。作业:扩张时会向下,扩展。
在这里插入图片描述

2.添加组件:Vertical Layout Group、Content Size Fitter

Vertical Layout Group:竖向排列 (横向排列:Horizontal Layout Group)
Content Size Fitter:根据子物体内容多少扩展(可根据需求设置:横向或者竖向,如下图)
在这里插入图片描述

3.添加文本

我选择添加旧版文本:Text (Legacy),并没有使用Text (TMP),原因是后面要做一个UI不被模型遮挡的效果。
但是TextMeshPro - Text (UI)组件效果更好
在这里插入图片描述

二、UI不被模型遮挡

如果UI设置跟随,或者当前位置模型太多,会出现UI被模型挡住,看不全UI面板的内容。
所有Text、Image都需要指定该shader
在这里插入图片描述
在这里插入图片描述
效果图:
在这里插入图片描述

使用Shader修改渲染层级

// Unity built-in shader source. Copyright (c) 2016 Unity Technologies. MIT license (see license.txt)

Shader "Custom/UIForground"
{
    Properties
    {
        [PerRendererData] _MainTex ("Sprite Texture", 2D) = "white" {}
        _Color ("Tint", Color) = (1,1,1,1)

        _StencilComp ("Stencil Comparison", Float) = 8
        _Stencil ("Stencil ID", Float) = 0
        _StencilOp ("Stencil Operation", Float) = 0
        _StencilWriteMask ("Stencil Write Mask", Float) = 255
        _StencilReadMask ("Stencil Read Mask", Float) = 255

        _ColorMask ("Color Mask", Float) = 15

        [Toggle(UNITY_UI_ALPHACLIP)] _UseUIAlphaClip ("Use Alpha Clip", Float) = 0
    }

    SubShader
    {
        Tags
        {
            "Queue"="Transparent"
            "IgnoreProjector"="True"
            "RenderType"="Transparent"
            "PreviewType"="Plane"
            "CanUseSpriteAtlas"="True"
        }

        Stencil
        {
            Ref [_Stencil]
            Comp [_StencilComp]
            Pass [_StencilOp]
            ReadMask [_StencilReadMask]
            WriteMask [_StencilWriteMask]
        }

        Cull Off
        Lighting Off
        ZWrite Off
        //ZTest [unity_GUIZTestMode]
        ZTest Always
        Blend One OneMinusSrcAlpha
        ColorMask [_ColorMask]

        Pass
        {
            Name "Default"
        CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #pragma target 2.0

            #include "UnityCG.cginc"
            #include "UnityUI.cginc"

            #pragma multi_compile_local _ UNITY_UI_CLIP_RECT
            #pragma multi_compile_local _ UNITY_UI_ALPHACLIP

            struct appdata_t
            {
                float4 vertex   : POSITION;
                float4 color    : COLOR;
                float2 texcoord : TEXCOORD0;
                UNITY_VERTEX_INPUT_INSTANCE_ID
            };

            struct v2f
            {
                float4 vertex   : SV_POSITION;
                fixed4 color    : COLOR;
                float2 texcoord  : TEXCOORD0;
                float4 worldPosition : TEXCOORD1;
                half4  mask : TEXCOORD2;
                UNITY_VERTEX_OUTPUT_STEREO
            };

            sampler2D _MainTex;
            fixed4 _Color;
            fixed4 _TextureSampleAdd;
            float4 _ClipRect;
            float4 _MainTex_ST;
            float _UIMaskSoftnessX;
            float _UIMaskSoftnessY;

            v2f vert(appdata_t v)
            {
                v2f OUT;
                UNITY_SETUP_INSTANCE_ID(v);
                UNITY_INITIALIZE_VERTEX_OUTPUT_STEREO(OUT);
                float4 vPosition = UnityObjectToClipPos(v.vertex);
                OUT.worldPosition = v.vertex;
                OUT.vertex = vPosition;

                float2 pixelSize = vPosition.w;
                pixelSize /= float2(1, 1) * abs(mul((float2x2)UNITY_MATRIX_P, _ScreenParams.xy));

                float4 clampedRect = clamp(_ClipRect, -2e10, 2e10);
                float2 maskUV = (v.vertex.xy - clampedRect.xy) / (clampedRect.zw - clampedRect.xy);
                OUT.texcoord = TRANSFORM_TEX(v.texcoord.xy, _MainTex);
                OUT.mask = half4(v.vertex.xy * 2 - clampedRect.xy - clampedRect.zw, 0.25 / (0.25 * half2(_UIMaskSoftnessX, _UIMaskSoftnessY) + abs(pixelSize.xy)));

                OUT.color = v.color * _Color;
                return OUT;
            }

            fixed4 frag(v2f IN) : SV_Target
            {
                half4 color = IN.color * (tex2D(_MainTex, IN.texcoord) + _TextureSampleAdd);

                #ifdef UNITY_UI_CLIP_RECT
                half2 m = saturate((_ClipRect.zw - _ClipRect.xy - abs(IN.mask.xy)) * IN.mask.zw);
                color.a *= m.x * m.y;
                #endif

                #ifdef UNITY_UI_ALPHACLIP
                clip (color.a - 0.001);
                #endif

                color.rgb *= color.a;

                return color;
            }
        ENDCG
        }
    }
}

三、打字效果代码

using System.Collections;
using System.Collections.Generic;
using Twq;
using UnityEngine;
using UnityEngine.UI;
/// <summary>
/// 对话框(3DUI,可换到任意人物头部位置)
/// </summary>
public class UICS_Dialog : SingletonMono<UICS_Dialog>
{
    public GameObject Mask;
    
    float Speed = 10;
    public  Text textLabel;
    void Start()
    {
        // GetText("作业负责人:进入场景后,弹出界面对任务进行一个介绍,可以视频、文字,点击开始");
    }
    /// <summary>
    /// 获得文字
    /// </summary>
    /// <param name="textToType"></param>
    public void GetText(string text_)
    {
        StartCoroutine(WriteText(text_));
    }
    /// <summary>
    /// 写出来
    /// </summary>
    /// <param name="textToType"></param>
    /// <returns></returns>
    IEnumerator WriteText(string text_)
    {
        float t = 0;//经过的时间
        int charIndex = 0;//字符串索引值
        while (charIndex < text_.Length)
        {
            t += Time.deltaTime * Speed;//简单计时器赋值给t
            charIndex = Mathf.FloorToInt(t);//把t转为int类型赋值给charIndex
            charIndex = Mathf.Clamp(charIndex, 0, text_.Length);
            textLabel.text = text_.Substring(0, charIndex);

            yield return null;
        }
        textLabel.text = text_;
    }


    /// <summary>
    /// 设置UI
    /// </summary>
    /// <param name="bo">UI显隐开关</param>
    /// <param name="parentTran">父节点</param>
    /// <param name="textStr">文字内容</param>
    public void SetState(bool bo, Transform parentTran, string textStr)
    {
        this.transform.SetParent(parentTran);//设置父节点
        this.transform.position = new Vector3(0,1.345f,0);//设置固定高度
        this.transform.Rotate(new Vector3(0,0,0)); //设置旋转
        Mask.SetActive(bo);//设置UI显隐
        if (bo)
        {
            GetText(textStr);
        }
    }

}



总结

好记性不如烂笔头!

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

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

相关文章

P-one新增火焰图-为性能测试开启新视野

随着软件业务流程的日益复杂&#xff0c;传统的性能测试方法已经难以满足对性能问题精准定位的需求。测试人员需要一种更加直观、全面的方式来分析软件在运行过程中的性能表现&#xff0c;以便快速准确地找到性能瓶颈并进行优化。因此&#xff0c;我们在性能测试平台P-One中加入…

几分钟教你实现一个酷炫的扫光效果

前言 话不多说&#xff0c;咱们先来看看本篇文章中我们实现的效果。 是不是发现这个效果非常的熟悉&#xff1f;没错&#xff0c;这经常能够在一些电商网站可以看到&#xff0c;那这究竟是怎么实现的呢&#xff1f;接下来由我来带领大家尝试做一个类似这样的效果出来。 实现…

浏览器采集黑屏 问题

chrome://flags/#use-angle Choose ANGLE graphics backend 选择OPENGL 然后重启浏览器 就可以了

美股市场波动与科技股动态

一、美股市场波动 周一&#xff0c;美股三大股指低开高走&#xff0c;但最终收盘时道指跌2.6%&#xff0c;纳指跌3.43%&#xff0c;标普跌3%。美国十年国债收益率涨0.053%&#xff0c;收报3.787%&#xff0c;恐慌指数VIX涨64.9%至38.57。现货黄金跌1.34%&#xff0c;报2409.42…

2024年有哪些好用的文件加密软件?十款常用加密软件推荐

在2024年&#xff0c;随着数据泄露和网络威胁的日益复杂&#xff0c;文件加密软件成为了保护敏感信息不可或缺的工具。无论是个人用户还是企业&#xff0c;选择合适的加密软件都是确保数据安全的重要一环。 1. 安秉加密软件 安秉加密软件专为企业设计&#xff0c;提供全面的信…

SpringBoot框架学习笔记(六):自定义转换器、内容协商 和 Thymeleaf

1 自定义转换器 1.1 基本介绍 &#xff08;1&#xff09;SpringBoot在响应客户端请求时&#xff0c;将提交的数据封装成对象时&#xff0c;使用了内置的转换器&#xff0c;一共提供了124个内置转换器&#xff0c;核心源码&#xff0c;在 GenericConverter 接口的内部类 Conve…

澳洲联储按兵不动,通胀阴霾难散

澳洲联储核心通胀率仍远高于目标水平&#xff0c;经济增长依然强劲&#xff0c;因此维持高利率是必要的。 鹰派立场坚定 澳洲联储的这一决定与全球其他央行的政策走向形成了鲜明对比。许多发达经济体的央行已经开始降息&#xff0c;以应对经济增长放缓的风险。然而&#xff0…

TapData 信创数据源 | 国产信创数据库 OceanBase 数据同步指南,加速国产化进程,推进自主创新建设

随着国家对自主可控的日益重视&#xff0c;目前在各个行业和区域中面临越来越多的国产化&#xff0c;采用有自主知识产权的国产数据库正在成为主流。长期以来&#xff0c;作为拥有纯国产自研背景的 TapData&#xff0c;自是非常重视对于更多国产信创数据库的数据连接器支持&…

WiFi模块无线通信交互,乐鑫ESP32物联网方案,启明云端乐鑫代理商

随着物联网(IoT)技术的飞速发展&#xff0c;我们正步入一个智能化、互联化的世界。在这一进程中&#xff0c;无线WiFi模块作为连接物理世界与数字世界的桥梁&#xff0c;扮演着至关重要的角色。 WiFi模块是一种基于WiFi协议的无线模块&#xff0c;它可以实现设备之间的无线通信…

ACM MM 2024 | 比SDXL和DALL-E·3更引人入胜!ReCorD:交互场景生成最新SOTA!

文章链接&#xff1a;https://arxiv.org/pdf/2407.17911 git链接&#xff1a;https://alberthkyhky.github.io/ReCorD/ 亮点直击&#xff1a; 引入了一种新颖的推理框架&#xff0c;将潜在扩散模型&#xff08;LDM&#xff09;与视觉语言模型&#xff08;VLM&#xff09;相结合…

私域成交的关键点

私域运营&#xff0c;私域团购&#xff0c;私域秒杀&#xff0c;私域发售

单张图像降噪Masked and Shuffled Blind Spot Denoising for Real-World Images

文章目录 Masked and Shuffled Blind Spot Denoising for Real-World Images1.噪声的空间相关性2.mask3.loss4.masking ratio 与 spatial correlation的关系5.空间噪声去相关6.Automated selection of the BSD masking ratio7.小结 Masked and Shuffled Blind Spot Denoising f…

深入 Dify 源码,洞察 Dify RAG 切片机制实现细节

背景介绍 最近测试时发现 Dify 的 RAG 分片效果一般&#xff0c;不管是使用之前 深入 Dify 源码&#xff0c;洞察 Dify RAG 核心机制 中有调研过的默认解析还是 Unstructured 解析。因此调研比较了 大量的开源框架 实现了特定格式的结构化解析方案&#xff0c;并与 Dify 现有解…

高通8255 Android Virtio Virtio-IIC 配置方法

目录 一&#xff1a;VirtIO和Passthrough的区别 方法一&#xff1a; passthrough 方法二&#xff1a;virtIO 二&#xff1a;配置逻辑 三&#xff1a;示例Virtio-I2C配置 Virtio-I2C框架 步骤一&#xff1a;QNX IIC资源配置 & 测试 配置 测试 步骤二&#xff1a;B…

干货整理: 什么软件能够监控电脑? 六大好用监控电脑软件抢手推荐

如何保障电脑的安全是许多企业用户关注的焦点。 除了常见的杀毒软件外&#xff0c;电脑监控软件也是一道重要的防线。 这些软件能够实时监控电脑的各项运行状况&#xff0c;及时发现并处理潜在的安全威胁&#xff0c;确保电脑的正常运行和用户数据的安全。 接下来&#xff0…

优思学院|防呆法的十大原理

防呆法&#xff08;Poka-Yoke&#xff09;&#xff0c;又称防错法&#xff08;Mistake-Proofing&#xff09;&#xff0c;是一种通过预防错误的发生来提高工作效率&#xff0c;也是精益管理、六西格玛中常用的管理工具。以下就是防呆法的十大原理&#xff1a; 1. 断根原理 断…

算法_链表专题---持续更新

文章目录 前言两数相加题目要求题目解析代码如下 两两交换链表中的结点题目要求题目解析代码如下 重排链表题目要求题目解析代码如下 合并K个升序链表题目要求题目解析 K个一组翻转链表题目要求题目解析代码如下 前言 本文将记录leetcode链表算法题解&#xff0c;包含题目有&a…

Why Memory Matters?(记忆力为何如此重要?)

What is memory? The general consensus is that memory is a multitude of cognitive systems which allow us to store information for certain periods of time so that we can learn from our past experiences and predict the future. 什么是记忆?人们普遍的共识是&am…

《PostgreSQL 数据库在国内的发展前景》

从DB-engines这张2024年8月的最新排名图上可以看出&#xff0c;PostgreSQL数据库的发展趋势还是非常好的&#xff0c;在国内&#xff0c;PostgreSQL数据库也展现出令人振奋的发展前景&#xff0c;非常明显的一种表现就是腾讯云、人大金仓、阿里云、华为等众多厂商都有基于Postg…

推荐一个uniapp选择文件上传的插件

插件地址&#xff1a;文件选择、文件上传组件&#xff08;图片&#xff0c;视频&#xff0c;文件等&#xff09; - DCloud 插件市场 支持 H5 / App / 微信小程序