Unity Graphic功能,实现UGUI上三角形,四边形,圆环的绘制

2023-10-29 22:31

本文主要是介绍Unity Graphic功能,实现UGUI上三角形,四边形,圆环的绘制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

这篇简单的纪录下利用Graphic类,实现UGUI圆环的绘制。效果图如下:

github目录:https://github.com/luckyWjr/Demo

 

Unity如何绘制图形

我们知道一个图形是由N个顶点,互相连成线,然后填充起来。如三角形有三个顶点,四边形有四个,而圆形可以理解为很多很多个顶点。Unity绘制图形的时候同样需要知道这些顶点信息,而区别在于这些看起来无缝连接的形状,在Untiy中是由一个个三角形拼接起来的。

也就是说,Unity只能绘制出三角形,然后由一个个三角形来组成更复杂的形状。例如四边形是两个三角形组成(下图中是ABC和ADC两个三角形,当然也可以是ABD和BDC两个组成),五边形则是三个,而圆则可以由N个三角形组成,如下图:

要绘制三角形那就需要三个顶点,例如上图中的ABC,Unity提供了UIVertex类来纪录顶点的信息,例如坐标,颜色,uv,顶点法线等。除了三个顶点信息外,我们还需要知道其绘制顺序,如是A->B->C还是A->C->B或者其他。这个顺序会影响到三角形面的朝向(Unity是左手坐标系,绘制方向则是左手手指弯曲的方向,面的朝向即左手大拇指朝向,因此ABC的顺序面朝向屏幕前的我们,ACB的顺序则是朝向屏幕后)由于UGUI默认的Shader:UI/Default,设置了Cuff Off,也就是双面渲染,所以我们看不出差别。

注:Unity内置Shader下载路径:https://unity3d.com/get-unity/download/archive

 

Graphic

前面讲了大致的理念,那么知道顶点信息后如何实现图形的绘制,就是由Graphic类来帮助我们实现。如下图,我们需要自定义一个类继承Graphic,然后重新其OnPopulateMesh方法。(UIGI中的Image也是继承于此)

public class Triangle : Graphic
{protected override void OnPopulateMesh(VertexHelper vh){}
}

注:文件名和类名必须相同,否则在给GameObject挂该组件的时候,会报错:cant add script component because the script class cannot be found...

注:若想要支持RectMask2D功能,则改为继承MaskableGraphic即可

在OnPopulateMesh方法中提供了VertexHelper参数,通过它我们就可以实现添加顶点等操作了。

添加顶点

VertexHelper.AddVert(Vector3 position, Color32 color, Vector2 uv0)

参数是顶点坐标,颜色以及uv,也可以直接添加我们前面提到的UIVertex:

VertexHelper.AddVert(UIVertex v)

在VertexHelper中会有如下List来存放这些顶点的相关信息,添加一个顶点对应的List.Count + 1

List<Vector3> m_Positions;
List<Color32> m_Colors;
List<Vector2> m_Uv0S;
List<Vector2> m_Uv1S;
List<Vector2> m_Uv2S;
List<Vector2> m_Uv3S;
List<Vector3> m_Normals;
List<Vector4> m_Tangents;

添加三角面

VertexHelper.AddTriangle(int idx0, int idx1, int idx2)

参数是三个顶点存储在VertexHelper中的下标,同时也代表了其绘制顺序,idx0 -> idx1 -> idx2。

在VertexHelper中有如下List来存放这些下标,添加一个三角面对应的List.Count + 3

private List<int> m_Indices;

当前顶点数量

VertexHelper.currentVertCount

对应 m_Positions.Count,添加了几个顶点数量就是几

当前索引数量

VertexHelper.currentIndexCount

对应 m_Indices.Count,添加了几次三角面数量就是N * 3

因此正常的绘制一个四边形,有四个顶点,两个三角面,因此currentVertCount = 4,currentIndexCount = 6

清除所有顶点及索引信息

VertexHelper.Clear()

设置贴图

除了重写OnPopulateMesh方法外,我们还可以重写mainTexture属性来设置贴图,例如

[SerializeField] Sprite m_image;    
public override Texture mainTexture => m_image == null ? s_WhiteTexture : m_image.texture;

s_WhiteTexture即是Texture2D.whiteTexture,也就是纯白的图。

SetDirty

如果我们在运行时动态修改了一些属性,需要重新绘制。例如修改了贴图,或者绘制圆的时候,修改了填充比例。我们可以调用下面方法来触发。

//调用后会在下一帧重新执行OnPopulateMesh
SetVerticesDirty();//调用后会在下一帧重新设置material以及Texture
SetMaterialDirty();//调用后会重新布局
SetLayoutDirty();//以上全部调用
SetAllDirty();

 

绘制一个三角形

根据前面的介绍,要绘制一个三角形就很简单了,添加三个顶点,添加一个三角面即可

[ExecuteInEditMode]
public class Triangle : Graphic
{public Vector2 positionA;public Vector2 positionB;public Vector2 positionC;protected override void OnPopulateMesh(VertexHelper vh){vh.Clear();vh.AddVert(positionA, Color.white, Vector2.zero);vh.AddVert(positionB, Color.red, Vector2.zero);vh.AddVert(positionC, Color.green, Vector2.zero);vh.AddTriangle(0, 1, 2);}
}

在Canvas中添加一个GameObject,挂上我们的组件,然后随便设置三个点的坐标即可,效果如下

 

绘制一个四边形

同样的四边形也就很简单了,四个顶点,两个三角面。不过在这边顺便讲一下前面没有提到的uv属性,先来看完整的代码

[ExecuteInEditMode]
public class Quadrangle : Graphic
{public Vector2 positionLeftTop;public Vector2 positionRightTop;public Vector2 positionRightBottom;public Vector2 positionLeftBottom;[SerializeField] Sprite m_image;public override Texture mainTexture => m_image == null ? s_WhiteTexture : m_image.texture;UIVertex[] m_vertexes = new UIVertex[4];Vector2[] m_uvs = new Vector2[4];protected override void Start(){m_uvs[0] = new Vector2(0, 1);m_uvs[1] = new Vector2(1, 1);m_uvs[2] = new Vector2(1, 0);m_uvs[3] = new Vector2(0, 0);}protected override void OnPopulateMesh(VertexHelper vh){vh.Clear();for (int i = 0; i < 4; i++){m_vertexes[i].color = color;m_vertexes[i].uv0 = m_uvs[i];}m_vertexes[0].position = positionLeftTop;m_vertexes[1].position = positionRightTop;m_vertexes[2].position = positionRightBottom;m_vertexes[3].position = positionLeftBottom;vh.AddVert(m_vertexes[0]);vh.AddVert(m_vertexes[1]);vh.AddVert(m_vertexes[2]);vh.AddVert(m_vertexes[3]);vh.AddTriangle(0, 2, 1);vh.AddTriangle(0, 3, 2);}
}

相比之前,我们添加了Sprite的设置,同时也为顶点添加了uv属性,效果图如下

正方形(就像Image组件了):   不规则形状:

 

顶点的uv属性

uv简单来说就是一个二维坐标(u和v的取值范围都是 0-1 ),代表着一张贴图每个像素的位置信息。uv(0,0)就代表图片的左下角,uv(1,1)代表图片的右上角。这样上述代码m_uvs中的四个uv值就很好理解了,分别是左上,右上,右下,左下。

接着我们的正方形正好是四个顶点,每个顶点设置相应的uv值,那么在该顶点上就会显示图片中相应uv的像素信息了。两个顶点之间的颜色同样也对应着图片中两点之间的像素信息。(顶点的color值要设置为Graphic中的color属性)

 

绘制圆和圆环

圆的话就是由N个三角形组成,圆环则是把这些组成圆的三角形切了一刀,也就是变成了一个四边形。由于代码写了注释,这里就不详细介绍了,效果图见文章最上方。

[ExecuteInEditMode]
// Changed to maskableGraphic so it can be masked with RectMask2D
public class Annulus : MaskableGraphic
{public enum ShapeType{Annulus,//圆环Circle,//圆}[SerializeField] Sprite m_image;public ShapeType shapeType;public float innerRadius = 10;//圆环内径,为0即是圆public float outerRadius = 20;//圆环外径[Range(0, 1)] [SerializeField] float m_fillAmount;//填充值[Range(0, 720)] public int segments = 360;//片数,越大锯齿越不明显[SerializeField] Image.Origin360 m_originType;//填充的起点public bool m_isClockwise = true;//填充方向,是否是顺时针填充public override Texture mainTexture => m_image == null ? s_WhiteTexture : m_image.texture;float m_originRadian = -1;//根据m_originType设置相关弧度(-1表示还没设置过对应的值)public float fillAmount{get => m_fillAmount;set{m_fillAmount = value;SetVerticesDirty();}}public Sprite image{get => m_image;set{if (m_image == value)return;m_image = value;SetVerticesDirty();SetMaterialDirty();}}public Image.Origin360 originType{get => m_originType;set{if (m_originType == value)return;m_originType = value;SetOriginRadian();SetVerticesDirty();}}public bool isClockwise{get => m_isClockwise;set{if (m_isClockwise != value){m_isClockwise = value;SetVerticesDirty();}}}UIVertex[] m_vertexes = new UIVertex[4];Vector2[] m_uvs = new Vector2[4];Vector2[] m_positions = new Vector2[4];protected override void Start(){if (m_originRadian == -1)SetOriginRadian();m_uvs[0] = new Vector2(0, 1);m_uvs[1] = new Vector2(1, 1);m_uvs[2] = new Vector2(1, 0);m_uvs[3] = new Vector2(0, 0);}protected override void OnPopulateMesh(VertexHelper vh){vh.Clear();//m_fillAmount == 0,什么也不绘制if (m_fillAmount == 0) return;#if UNITY_EDITORSetOriginRadian();
#endif//每个面片的角度float degrees = 360f / segments;//需要绘制的面片数量int count = (int)(segments * m_fillAmount);float cos = Mathf.Cos(m_originRadian);float sin = Mathf.Sin(m_originRadian);//计算外环起点,例如m_originRadian = 0,x = -outerRadius,y = 0,所以起点是Left(九点钟方向)float x = -outerRadius * cos;float y = outerRadius * sin;Vector2 originOuter = new Vector2(x, y);//计算内环起点x = -innerRadius * cos;y = innerRadius * sin;Vector2 originInner = new Vector2(x, y);for (int i = 1; i <= count; i++){//m_positions[0] 当前面片的外环起点m_positions[0] = originOuter;//当前面片的弧度 + 起始弧度 = 终止弧度float endRadian = i * degrees * Mathf.Deg2Rad * (isClockwise ? 1 : -1) + m_originRadian;cos = Mathf.Cos(endRadian);sin = Mathf.Sin(endRadian);//m_positions[1] 当前面片的外环终点m_positions[1] = new Vector2(-outerRadius * cos, outerRadius * sin);//m_positions[2] 当前面片的内环终点//m_positions[3] 当前面片的内环起点if (shapeType == ShapeType.Annulus){m_positions[2] = new Vector2(-innerRadius * cos, innerRadius * sin);m_positions[3] = originInner;}else{m_positions[2] = Vector2.zero;m_positions[3] = Vector2.zero;}// 设置顶点的颜色坐标以及uvfor (int j = 0; j < 4; j++){m_vertexes[j].color = color;m_vertexes[j].position = m_positions[j];m_vertexes[j].uv0 = m_uvs[j];}//当前顶点数量int vertCount = vh.currentVertCount;//如果是圆只需要添加三个顶点,创建一个三角面vh.AddVert(m_vertexes[0]);vh.AddVert(m_vertexes[1]);vh.AddVert(m_vertexes[2]);//参数即三角面的顶点绘制顺序vh.AddTriangle(vertCount, vertCount + 2, vertCount + 1);// 如果是圆环就需要添加第四个顶点,并再创建一个三角面if (shapeType == ShapeType.Annulus){vh.AddVert(m_vertexes[3]);vh.AddTriangle(vertCount, vertCount + 3, vertCount + 2);}//当前面片的终点就是下个面片的起点originOuter = m_positions[1];originInner = m_positions[2];}}//m_originType改变的时候需要重新设置m_originRadianvoid SetOriginRadian(){switch (m_originType){case Image.Origin360.Left:m_originRadian = 0 * Mathf.Deg2Rad;break;case Image.Origin360.Top:m_originRadian = 90 * Mathf.Deg2Rad;break;case Image.Origin360.Right:m_originRadian = 180 * Mathf.Deg2Rad;break;case Image.Origin360.Bottom:m_originRadian = 270 * Mathf.Deg2Rad;break;}}
}

知道原理,我们就可以进行一些魔改,例如fillmount小于1的时候,用别的颜色来填充等,大家可以自由发挥。

这篇关于Unity Graphic功能,实现UGUI上三角形,四边形,圆环的绘制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/303787

相关文章

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

【WebGPU Unleashed】1.1 绘制三角形

一部2024新的WebGPU教程,作者Shi Yan。内容很好,翻译过来与大家共享,内容上会有改动,加上自己的理解。更多精彩内容尽在 dt.sim3d.cn ,关注公众号【sky的数孪技术】,技术交流、源码下载请添加微信号:digital_twin123 在 3D 渲染领域,三角形是最基本的绘制元素。在这里,我们将学习如何绘制单个三角形。接下来我们将制作一个简单的着色器来定义三角形内的像素

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、