我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Unity使用LineRender实现签名效果

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Unity使用LineRender实现签名效果

本文为大家分享了Unity制作签名功能的具体代码,供大家参考,具体内容如下

前言:项目中需要做一个签名的功能,同时需要两个两个屏幕进行显示,但是都是在UI上,从网上查了大量资料。

找到两种方法:

1、修改图片像素点  但是是马赛克效果,不满足需求
2、使用LineRenderer 的3D签名制作出2D效果

改像素点:

先上代码


using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
 
public class Test : ObjBase
{
 
    public GameObject m_obj;
    private Texture2D m_tex;
    public Color m_color;
    public int size = 3;
    private Color[] m_textureColorsStart;
 
 
    public RawImage showImg;
    void Start()
    {
        if (Display.displays.Length > 1)
            Display.displays[1].Activate();
        if (Display.displays.Length > 2)
            Display.displays[2].Activate();
        m_tex = m_obj.GetComponent<MeshRenderer>().material.mainTexture as Texture2D;
        //从纹理中获取像素颜色
        m_textureColorsStart = m_tex.GetPixels();
        Debug.Log(m_tex.name);
    }
 
 
    void Update()
    {
        //Vector3 oldPos=Vector3.zero;
        //oldPos = Input.mousePosition;
        //Ray ray = uiCam.ScreenPointToRay(Input.mousePosition);
        Ray ray = Camera.main.ScreenPointToRay(Input.mousePosition);
        RaycastHit hit;
        if (Input.GetMouseButton(0))
        {
            // float m_magnitude = (Input.mousePosition - oldPos).magnitude;
            // Vector3 dir = Input.mousePosition - oldPos;  
            if (Physics.Raycast(ray, out hit))
            {
                //在碰撞位置处的UV纹理坐标。
                Vector2 pixelUV = hit.textureCoord;
                //以像素为单位的纹理宽度
                pixelUV.x *= m_tex.width;
                pixelUV.y *= m_tex.height;
                //贴图UV坐标以右上角为原点
                for (float i = pixelUV.x - 1; i < pixelUV.x + size; i++)
                {
                    for (float j = pixelUV.y - 1; j < pixelUV.y + size; j++)
                    {
                        m_tex.SetPixel((int)i, (int)j, m_color);
                    }
                }
                Debug.Log(pixelUV);
                m_tex.Apply();
                showImg.texture = m_tex;
            }
        }
        if (Input.GetKeyDown(KeyCode.Space))
        {
            //还原
            m_tex.SetPixels(m_textureColorsStart);
            m_tex.Apply();
        }
 
 
        //在处理鼠标按下的记录下位置,抬起的时候记录下位置,取2个位置中间的位置发射射线
        //if (Input.GetMouseButtonDown(0))
        //{
 
        //}
        //if (Input.GetMouseButtonUp(0))
        //{
 
        //}
    }
 
    public void OnClick()
    {
       
        showImg.texture = m_tex;
    }
}

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
 
public class ObjBase : MonoBehaviour
{
    
 
    public bool IsShow
    {
        get { return gameObject.activeSelf; }
    }
 
    // Use this for initialization
    void Start()
    {
 
    }
 
    /// <summary>
    /// 显示
    /// </summary>
    /// <param name="parameter"></param>
    public virtual void Show(object parameter = null)
    {
       
        gameObject.SetActive(true);
    }
 
    /// <summary>
    /// 隐藏
    /// </summary>
    /// <param name="parameter"></param>
    public virtual void Hide(object parameter = null)
    {
        gameObject.SetActive(false);
    }
 
   
 
}

Test脚本是用来修改像素点的,ObjBase只是一个根父类,控制显示和隐藏。

测试场景用的Quad,通过读取他的mainTexture对应的像素,进行修改,UI中的话就是将一张图片转成Texture2D形式,通过读取像素点,进行修改即可,同时还可以实现同步效果。

项目中的Hierarchy窗口设置:

项目需求:使用了两个画布,MainCamera照射Quad,两个UI相机分别照射两个画布,画布的Render Mode设置为Screen Space -Camera格式。GameObject挂载脚本,Quad用来修改其上的图片的像素点。

效果图:

使用LineRenderer   3D划线方法实现2D签名效果:

先上代码:


using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
using System.Text;
using System.IO;
using UnityEngine.EventSystems;
 
public class Test5 : MonoBehaviour {
 
    public GameObject drawObj;
    private bool beginDraw;
    private GameObject obj;
    public Transform parent;
    public RawImage rawImg;
    public Camera UICam;
    public Camera main;//主相机和UI相机共同照射到的地方进行截图
    Color[] colors;
    Texture2D myTexture2D;
    public RawImage photo;
 
    public RawImage showImg;
    [SerializeField] private string _name;
 
    public RectTransform canvas1;
 
    public void SaveFile()
    {
        Camera mainCam;
        GameObject cam = Camera.main.gameObject;
 
        if (cam)
        {
            mainCam = cam.GetComponent<Camera>();
        }
        else
        {
            return;
        }
 
        RenderTexture renderTex;
 
        renderTex = new RenderTexture(Screen.width, Screen.height, 24);
        mainCam.targetTexture = renderTex;
        mainCam.Render();
 
         myTexture2D = new Texture2D(renderTex.width, renderTex.height);
        RenderTexture.active = renderTex;
        myTexture2D.ReadPixels(new Rect(0, 0, renderTex.width, renderTex.height), 0, 0);
        
       
        myTexture2D.Apply();
        byte[] bytes = myTexture2D.EncodeToJPG();
 
        myTexture2D.Compress(true);
        myTexture2D.Apply();
        RenderTexture.active = null;
 
        File.WriteAllBytes(Application.dataPath + "/StreamingAssets/TextureTemp.png", bytes);
        mainCam.targetTexture = null;
        GameObject.Destroy(renderTex);
    }
 
    public void OnClick()
    {
       
        main.rect = new Rect(0, 0, 1, 1);
       CaptureCamera( main,new Rect(Screen.width * 0f, Screen.height * 0f, Screen.width * 1f, Screen.height * 1f));
 
        
    }
 
 
    /// <summary>  
    /// 对相机截图。   
    /// </summary>  
    /// <returns>The screenshot2.</returns>  
    /// <param name="camera">Camera.要被截屏的相机</param>  
    /// <param name="rect">Rect.截屏的区域</param>  
    Texture2D CaptureCamera(Camera camera,Rect rect)
    {
        // 创建一个RenderTexture对象  
        RenderTexture rt = new RenderTexture((int)rect.width, (int)rect.height, 0);
        // 临时设置相关相机的targetTexture为rt, 并手动渲染相关相机  
        camera.targetTexture = rt;
        camera.Render();
        //ps: --- 如果这样加上第二个相机,可以实现只截图某几个指定的相机一起看到的图像。  
         //camera2.targetTexture = rt;  
        // camera2.Render();  
        //ps: -------------------------------------------------------------------  
 
        // 激活这个rt, 并从中中读取像素。  
        RenderTexture.active = rt;
        Texture2D screenShot = new Texture2D((int)rect.width, (int)rect.height, TextureFormat.RGB24, false);
        screenShot.ReadPixels(rect, 0, 0);// 注:这个时候,它是从RenderTexture.active中读取像素  
        screenShot.Apply();
 
        // 重置相关参数,以使用camera继续在屏幕上显示  
        camera.targetTexture = null;
       // camera2.targetTexture = null;
        //ps: camera2.targetTexture = null;  
        RenderTexture.active = null; // JC: added to avoid errors  
        GameObject.Destroy(rt);
        // 最后将这些纹理数据,成一个png图片文件  
        byte[] bytes = screenShot.EncodeToPNG();
        string filename = Application.dataPath + string.Format("/Screenshot_{0}.png", _name);
        System.IO.File.WriteAllBytes(filename, bytes);
        Debug.Log(string.Format("截屏了一张照片: {0}", filename));
        showImg.texture = screenShot;
        main.rect = new Rect(0.25f, 0.35f, 0.5f, 0.5f);
        return screenShot;
    }
   
 
    void Start () {
        if (Display.displays.Length > 1)
            Display.displays[1].Activate();
        if (Display.displays.Length > 2)
            Display.displays[2].Activate();
    }
 
 // Update is called once per frame
 void Update () {
        if (Input.GetMouseButtonDown(0))
        {
            beginDraw = true;
            obj = Instantiate(drawObj) as GameObject;
            obj.transform.parent = parent;
        }
        if (Input.GetMouseButtonUp(0))
        {
            beginDraw = false;
        }
 
        if (beginDraw)
        {
            Vector3 position = new Vector3(Input.mousePosition.x, Input.mousePosition.y, 10f);
            position = Camera.main.ScreenToWorldPoint(position);
            //Vector3 localPoint;
            //if(RectTransformUtility.ScreenPointToWorldPointInRectangle(canvas1, position, null, out localPoint))
            //{
            //    position = localPoint;
            //}
           
 
            DrawText dt = obj.GetComponent<DrawText>();
            dt.points.Add(position);
            dt.Draw();
            dt.line.startColor = Color.yellow;
            dt.line.endColor = Color.yellow;
            dt.line.startWidth = 0.03f;
            dt.line.endWidth = 0.03f;
        }
 
    }
}

Test5是划线和截取签名的操作,绑定在空物体上,OnClick函数绑定在按钮上

Line:制作签名预制体


using System.Collections;
using System.Collections.Generic;
using UnityEngine;
 
public class DrawText : MonoBehaviour {
 
 
    public List<Vector3> points = new List<Vector3>();
    public  LineRenderer line;
    private void Awake()
    {
        line = GetComponent<LineRenderer>();
    }
    public  void Draw()
    {
        line.positionCount = points.Count;
        for (int i = 0; i < points.Count; i++)
        {
            line.SetPosition(i, points[i]);
            line.startWidth =2f;
            line.endWidth =2f;
        }
    }
    // Use this for initialization
    void Start () {
  
 }
 
 // Update is called once per frame
 void Update () {
  
 }
}

Draw Text脚本挂在预制体Line上,Line 添加LineRenderer组件,同时Material中加入自己创建的材质球

项目需求:Hierarchy窗口设置

和上面一种方法一样,也是两个画布,两个UI相机,同时需要一个MainCamera

parent为空物体,用来作为根节点,将签名时实时生成的预制体放在其下面,作为子节点,方便后面进行销毁,重新签名。

重点:

第二种方法使用的是特定相机照射画面进行截图,Test5中的CaptureCamera方法就是截取主相机照射到的画面。由于签名不能进行全屏进行截图,只能部分截图,类似相面的画面

下面会有一些常规的功能按钮,重新签名,保存签名等等操作,这些操作就是在UI上进行签名。

所以,通过修改MainCamera的Viewport Rect窗口来进行截图,同时能够实现正常的签名操作。

MainCamera的Viewport Rect设置:

运行刚开始:

通过设置这个属性,可以使签名界面呈现上一个图的效果,前面是UI层,后面是3D层。

然而在截屏图的时候如果始终保持Viewport Rect是上面的设置,则截图的时候仍把周围的黑色部分也截取出来,刚开始以为特定相机照射截图只截取Viewport Rect中的图像,后来测试是周围的所有黑色部分也截取了,这样就不满足要求。

所以,在代码中签字 的时候保持上面的设置,截图之前main.rect = new Rect(0, 0, 1, 1);设置成全屏,截好之后重新回复成原来的设置  main.rect = new Rect(0.25f, 0.35f, 0.5f, 0.5f);,截图完成之后将签名图片赋值给第二个屏幕画布中的RawImage进行展示。

达到效果。结合UI实际签名过程中

中间的白色部分,通过设置MainCamera中的Camera组件中的Background(设置为白色)以及天空盒(Windows->Lighting->Settings->Scene->Skybox Material设置为空),设置为需要的颜色。UI制作的时候需要签名的部分制作成透明的即可。

效果图:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程网。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Unity使用LineRender实现签名效果

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

如何使用Flutter实现手写签名效果

Flutter插件提供了用于绘制平滑签名的签名板,下面这篇文章主要给大家介绍了关于如何使用Flutter实现手写签名效果的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考下
2022-12-24

使用canvas怎么实现一个手写签名效果

这期内容当中小编将会给大家带来有关使用canvas怎么实现一个手写签名效果,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。HTML代码:
2023-06-09

用Unity实现使用鼠标旋转物体效果

本篇内容主要讲解“用Unity实现使用鼠标旋转物体效果”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“用Unity实现使用鼠标旋转物体效果”吧!本文实例为大家分享了Unity使用鼠标旋转物体效果的
2023-06-20

怎么用JavaScript实现电子签名效果

本篇内容主要讲解“怎么用JavaScript实现电子签名效果”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么用JavaScript实现电子签名效果”吧!步骤一:创建HTML和CSS首先,我们需
2023-07-05

3分钟教你用JavaScript实现电子签名效果

电子签名已经成为现代商业中不可或缺的一部分,它可以提高业务流程的效率和安全性。本文将介绍如何使用HTML5的canvas元素和JavaScript在前端实现电子签名,需要的可以参考一下
2023-05-14

利用unity怎么实现一个翻页效果

这期内容当中小编将会给大家带来有关利用unity怎么实现一个翻页效果,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。创建物体ToPanel;添加组件ScrollRect,在下面创建一个空物体用来装需要移动的
2023-06-06

怎么利用CSS实现书签效果

小编给大家分享一下怎么利用CSS实现书签效果,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!实现的效果图如下:示例代码如下:
2023-06-08

Unity中怎么利用ScrollView实现自动吸附效果

Unity中怎么利用ScrollView实现自动吸附效果,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。通过使用UGUI的拖拽接口,在拖拽结束时比较当前滑动框的Normalize
2023-06-20

Vue怎么引入sign-canvas实现签名画板效果

这篇“Vue怎么引入sign-canvas实现签名画板效果”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Vue怎么引入sig
2023-07-05

Unity中怎么利用Shader实现一个3D翻页效果

本篇文章给大家分享的是有关Unity中怎么利用Shader实现一个3D翻页效果,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。效果图:原理:Shader顶点动画在顶点着色器进行对
2023-06-20

uniapp怎么实现微信小程序的电子签名效果

今天小编给大家分享一下uniapp怎么实现微信小程序的电子签名效果的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。开发框架:u
2023-06-30

接口签名如何使用Java实现

这篇文章主要介绍了接口签名如何使用Java实现,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。java实现接口签名为了保证数据传输的安全性,跟其他系统进行数据交互时,双方应该约
2023-06-15

Python使用Pygame实现时钟效果

本文实例为大家分享了Python使用Pygame实现时钟效果的具体代码,供大家参考,具体内容如下import pygame,sys,math,random from pygame.locals import * from datetime
2022-06-02

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录