Unity实现VR中在黑板上写字效果

本文实例为大家分享了Unity实现VR中在黑板上写字的具体代码,供大家参考,具体内容如下

一、工具

1.开发用的是Unity 5.6.2版本

2.VR中的物理交互用的是VRTK插件,这个插件集成了比较好的物理交互功能;

3.HTC Vive

二、概述

实现的功能: 在一个白板上,用不同颜色的笔,在白板画出任何想要的图形;

因为只是一个初级篇所以只是用两个脚本简单的实现,而且并没有黑板擦等功能 ,也不能两个笔同时画画,这些功能将会在未来的升级篇中写出;

三、知识点

其实这个功能很简单,只是简单的运用Unity Texure2D类中的两个函数:

public void SetPixels32(int x, int y, int blockWidth, int blockHeight, Color32[] colors, int miplevel = 0);

前面4个参数相当于一个矩形,x和y就是矩形的左下角的那个点,blockWidth和blockHeight分别是矩形的宽和高,这个矩形所代表的范围就是blockWidth*blockHeight个像素所在的位置,不妨称这个矩形范围为一个色块;

colors这个参数的大小必须等于blockWidth*blockHeight,因为这个方法就是给坐标(x,y)开始,从左到右,从下到上,一行一行的对矩形范围内的每个像素赋值;

也就是把colors[0]~colors[blockWidth - 1]分别赋值到坐标为(x,y)~(x + blockWidth,y)的像素,以此类推;

最后一个参数,因为我们用的图片把Generate Min Maps这个选项关闭了,所以用默认的可选参数0;

public void Apply(bool updateMipmaps = true, bool makeNoLongerReadable = false);

当对图片改动完成以后,需要调用这个方法,才能让改动真正的应用在图片上;

四、场景搭建

1.画板

在场景中建一个Quad,把它的x和y方向的Scale分别设置为1.92和1.08(或者其它尺寸);注意这个Quad一定要用Mesh Collider作为碰撞体,不然到时候射线获取的纹理坐标有误,并为它设置一个Tag为Board;

2.笔

建一个尺寸合适的笔,创建一个空的子物体,命名为SnapPoint,并设置SnapPoint的Z方向指向笔尖方向,这个子物体就是,手柄拿笔的位置就是,并且保证笔的姿态是相当于人正常拿笔的样子;

3.其它

创建一个放笔的物体,让笔处于比较好拿的位置;

我的场景中代表画板的是WhiteBoard下的Board物体;

五、代码实现功能

这个脚本是挂在代表画板的物体上的:

using System.Linq;
using UnityEngine;

/// <summary>
/// 画板
/// </summary>
public class Board : MonoBehaviour
{
 //当画笔移动速度很快时,为了不出现断断续续的点,所以需要对两个点之间进行插值,lerp就是插值系数
 [Range(0, 1)]
 public float lerp = 0.05f;
 //初始化背景的图片
 public Texture2D initailizeTexture;
 //当前背景的图片
 private Texture2D currentTexture;
 //画笔所在位置映射到画板图片的UV坐标
 private Vector2 paintPos;

 private bool isDrawing = false;//当前画笔是不是正在画板上
 //离开时画笔所在的位置
 private int lastPaintX;
 private int lastPaintY;
 //画笔所代表的色块的大小
 private int painterTipsWidth = 30;
 private int painterTipsHeight = 15;
 //当前画板的背景图片的尺寸
 private int textureWidth;
 private int textureHeight;

 //画笔的颜色
 private Color32[] painterColor;

 private Color32[] currentColor;
 private Color32[] originColor;

 private void Start()
 {
 //获取原始图片的大小
 Texture2D originTexture = GetComponent<MeshRenderer>().material.mainTexture as Texture2D;
 textureWidth = originTexture.width;//1920
 textureHeight = originTexture.height;//1080

 //设置当前图片
 currentTexture = new Texture2D(textureWidth, textureHeight, TextureFormat.RGBA32, false, true);
 currentTexture.SetPixels32(originTexture.GetPixels32());
 currentTexture.Apply();

 //赋值给黑板
 GetComponent<MeshRenderer>().material.mainTexture = currentTexture;

 //初始化画笔的颜色
 painterColor = Enumerable.Repeat<Color32>(new Color32(255, 0, 0, 255), painterTipsWidth * painterTipsHeight).ToArray<Color32>();
 }

 private void LateUpdate()
 {
 //计算当前画笔,所代表的色块的一个起始点
 int texPosX = (int)(paintPos.x * (float)textureWidth - (float)(painterTipsWidth / 2));
 int texPosY = (int)(paintPos.y * (float)textureHeight - (float)(painterTipsHeight / 2));
 if (isDrawing)
 {
  //改变画笔所在的块的像素值
  currentTexture.SetPixels32(texPosX, texPosY, painterTipsWidth, painterTipsHeight, painterColor);
  //如果快速移动画笔的话,会出现断续的现象,所以要插值
  if (lastPaintX != 0 && lastPaintY != 0)
  {
  int lerpCount = (int)(1 / lerp);
  for (int i = 0; i <= lerpCount; i++)
  {
   int x = (int)Mathf.Lerp((float)lastPaintX, (float)texPosX, lerp);
   int y = (int)Mathf.Lerp((float)lastPaintY, (float)texPosY, lerp);
   currentTexture.SetPixels32(x, y, painterTipsWidth, painterTipsHeight, painterColor);
  }
  }
  currentTexture.Apply();
  lastPaintX = texPosX;
  lastPaintY = texPosY;
 }
 else
 {
  lastPaintX = lastPaintY = 0;
 }

 }

 /// <summary>
 /// 设置当前画笔所在的UV位置
 /// </summary>
 /// <param name="x"></param>
 /// <param name="y"></param>
 public void SetPainterPositon(float x, float y)
 {
 paintPos.Set(x, y);
 }

 /// <summary>
 /// 画笔当前是不是在画画
 /// </summary>
 public bool IsDrawing
 {
 get
 {
  return isDrawing;
 }
 set
 {
  isDrawing = value;
 }
 }

 /// <summary>
 /// 使用当前正在画板上的画笔的颜色
 /// </summary>
 /// <param name="color"></param>
 public void SetPainterColor(Color32 color)
 {
 if (!painterColor[0].IsEqual(color))
 {
  for (int i = 0; i < painterColor.Length; i++)
  {
  painterColor[i] = color;
  }
 }
 }

}
public static class MethodExtention
{
 /// <summary>
 /// 用于比较两个Color32类型是不是同种颜色
 /// </summary>
 /// <param name="origin"></param>
 /// <param name="compare"></param>
 /// <returns></returns>
 public static bool IsEqual(this Color32 origin, Color32 compare)
 {
 if (origin.g == compare.g && origin.r == compare.r)
 {
  if (origin.a == compare.a && origin.b == compare.b)
  {
  return true;
  }
 }
 return false;
 }
}

下面这个脚本是挂在画笔上的:

using UnityEngine;

public class Painter : MonoBehaviour
{
 /// <summary>
 /// 画笔的颜色
 /// </summary>
 public Color32 penColor;

 public Transform rayOrigin;

 private RaycastHit hitInfo;
 //这个画笔是不是正在被手柄抓着
 private bool IsGrabbing;
 private static Board board;//设置成类型的成员,而不是类型实例的成员,因为所有画笔都是用的同一个board

 private void Start()
 {
 //将画笔部件设置为画笔的颜色,用于识别这个画笔的颜色
 foreach (var renderer in GetComponentsInChildren<MeshRenderer>())
 {
  if (renderer.transform == transform)
  {
  continue;
  }
  renderer.material.color = penColor;
 }
 if (!board)
 {
  board = FindObjectOfType<Board>();
 }

 }

 private void Update()
 {
 Ray r = new Ray(rayOrigin.position, rayOrigin.forward);
 if (Physics.Raycast(r, out hitInfo, 0.1f))
 {
  if (hitInfo.collider.tag == "Board")
  {
  //设置画笔所在位置对应画板图片的UV坐标
  board.SetPainterPositon(hitInfo.textureCoord.x, hitInfo.textureCoord.y);
  //当前笔的颜色
  board.SetPainterColor(penColor);
  board.IsDrawing = true;
  IsGrabbing = true;
  }
 }
 else if(IsGrabbing)
 {
  board.IsDrawing = false;
  IsGrabbing = false;
 }
 }

}

六、等待完善的地方

1.画笔所能画的最小点是有大小的,也就是SetPixels参数中的blockWidth*blockHeight的大小,当这个画笔在画板的边缘的时候,那么这个画笔所能画的色块的矩形范围就到图片之外去了,这会引起未处理异常;

2.同时只有一个笔能在画板上画画;

3.没有黑板擦功能;

4.没有颜色混合功能;

5.画笔是纯粹的颜色,其实可以用一个图片设置画笔的形状;

6.笔可以穿透画板

这些问题都将在升级篇中完善;

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

(0)

相关推荐

  • unity实现屏幕上写字效果

    本文实例为大家分享了unity实现屏幕上写字效果的具体代码,供大家参考,具体内容如下 先建立一个RawImage,然后再在这个图片上加个LineRenderer组件,再建个材质球,把材质球的Shader改成Particles/Additive,把材质球拖给LineRenderer组件的Materials/Element 0(不拖也可以),最后再把代码拖给空物体即可,代码的Target是RawImage,下面的代码 using System.Collections; using System.Co

  • Unity使用LineRender断笔写字

    做多媒体项目时,经常会最后来个客户签名并保存之类的,签名保存之前的博客Unity3d截图方法合集有介绍过了,今天闲着把断笔写字的也贴出来吧,以前用leap motion时尝试用 leap motion演示中的食指写字,当时的写字其实只能一笔画,说白了其实就是个寿命无限长的拖尾,虽然效果不太好,但是很流畅,尝试过用leap motion断笔写字,但是效果不好,很容易误写,然后就产生了此方法,就是鼠标或者触摸屏写字了. 讲一下思路,就是不断的将鼠标的屏幕坐标转换成世界坐标,然后用LineRender

  • Unity实现VR中在黑板上写字效果

    本文实例为大家分享了Unity实现VR中在黑板上写字的具体代码,供大家参考,具体内容如下 一.工具 1.开发用的是Unity 5.6.2版本 2.VR中的物理交互用的是VRTK插件,这个插件集成了比较好的物理交互功能: 3.HTC Vive 二.概述 实现的功能: 在一个白板上,用不同颜色的笔,在白板画出任何想要的图形: 因为只是一个初级篇所以只是用两个脚本简单的实现,而且并没有黑板擦等功能 ,也不能两个笔同时画画,这些功能将会在未来的升级篇中写出: 三.知识点 其实这个功能很简单,只是简单的运

  • java 在图片上写字,两个图片合并的实现方法

    实例如下: package writeimg; import javax.imageio.ImageIO; import java.awt.Color; import java.awt.Font; import java.awt.Graphics2D; import java.awt.image.BufferedImage; import java.io.File; import java.io.IOException; import java.net.URL; public class pic

  • Unity中 ShaderGraph 实现旋涡传送门效果入门级教程(推荐)

    若你的工程还没有进行基础配置,请查看我的博文Unity 之 ShaderGraph入门使用详解,按照步骤操作完成配置即可,还能顺便学习一下基础操作流程哦~ 本文手把手教你制作旋涡效果,入门级教程,还不快动手试试? 一,最终效果展示 老规矩,直接上效果图: 二,创建PBRGraph 在Project面板右键 --> Create --> Shader --> PBRGraph: 重命名:RPAShaderGraph(随意) 双击RPAShaderGraph打开编辑界面 三,实现原理介绍 通

  • java 使用Graphics2D在图片上写字

    目录 使用Graphics2D在图片上写字 Graphics2D 的使用 1.Graphics2D和Graphics的区别 2.setRenderingHint的具体用处 3.Stroke的使用方法 使用Graphics2D在图片上写字 首先jsp页面:src是一个方法地址 然后在Controller如下:写一个生成方法,用流进行读写 生成方法,(关键): 注意:这种方法可行,但是需要计算文字的位置,感觉不太灵活,如果谁有更好的方法请分享一下. Graphics2D 的使用 1.Graphics

  • ASP.NET中FileUpload文件上传控件应用实例

    使用 FileUpload 控件,可以为用户提供一种将文件从用户的计算机发送到服务器的方法.该控件在允许用户上载图片.文本文件或其他文件时很有用.要上载的文件将在回发期间作为浏览器请求的一部分提交给服务器.在文件上载完毕后,您可以用代码管理该文件. 大致了解了一下FileUpload,让我们来看一下FileUpload几个实际应用中问题的处理方法. 1.一次上传多个文件 要一次上传多个文件,我们可以像传单个文件那样对每个文件单独进行处理,除此之外,我们还可以使用HttpFileCollectio

  • Java中实现文件上传下载的三种解决方案(推荐)

    java文件上传与文件下载是程序开发中比较常见的功能,下面通过本文给大家介绍Java中实现文件上传下载的三种解决方案,具体详情如下所示: 第一点:Java代码实现文件上传 FormFile file=manform.getFile(); String newfileName = null; String newpathname=null; String fileAddre="/numUp"; try { InputStream stream = file.getInputStream(

  • iOS中键盘 KeyBoard 上添加工具栏的方法

    iOS中 键盘 KeyBoard 上怎么添加工具栏? 如图中所示 在键盘上面加一条工具栏 大致思路是提前创建好工具栏,在键盘弹出的时候将工具栏显示出来,在键盘消失的时候让工具栏隐藏 上代码 设置两个变量 UIView * _toolView; //工具栏 UITextField *textField;// 输入框 呼出键盘用 创建工具栏 输入框 添加键盘弹出 消失的通知 - (void)viewDidLoad { [super viewDidLoad]; // Do any additional

  • PHP中,文件上传

    在PHP中,文件上传一般是通过move_uploaded_file()来实现的.  bool move_uploaded_file ( string filename, string destination )本函数检查并确保由 filename 指定的文件是合法的上传文件(即通过 PHP 的 HTTP POST 上传机制所上传的).如果文件合法,则将其移动为由 destination 指定的文件. 如果 filename 不是合法的上传文件,不会出现任何操作,move_uploaded_fil

  • 切记ajax中要带上AntiForgeryToken防止CSRF攻击

    经常看到在项目中ajax post数据到服务器不加防伪标记,造成CSRF攻击 在Asp.net Mvc里加入防伪标记很简单在表单中加入Html.AntiForgeryToken()即可. Html.AntiForgeryToken()会生成一对加密的字符串,分别存放在Cookies 和 input 中. 我们在ajax post中也带上AntiForgeryToken @model WebApplication1.Controllers.Person @{ ViewBag.Title = "In

随机推荐