RawImage显示多边形或圆形贴图

本文介绍了一种在Unity中使用RawImage组件显示圆形图标的方法,通过自定义UICircle类,可以灵活调整圆形的段数,实现从多边形到圆形的平滑过渡。这种方法避免了使用shader带来的不便,适用于需要在不同形状间切换图标的应用场景。

RawImage显示多边形或圆形贴图

RawImage显示多边形或圆形贴图

项目功能需要,便在网上找到了雨松大神的一篇文章。亲测有效,做个记录。

原文章连接:
UGUI研究院之RawImage显示多边形或圆形贴图(二十六)

正文

项目中可能有些icon.有的地方显示矩形,有的地方又要显示圆形。
网上我看有人用shader实现了,我觉得使用起来不太方便。后来参考了一个工程 https://bitbucket.org/ddreaper/unity-ui-extensions

把sprite或者texture拽上去。在这里设置多边形的数量,我测试36就已经很圆了。。

UGUI研究院之RawImage显示多边形或圆形贴图(二十六) - 雨松MOMO程序研究院 - 3
然后就是代码

using System;
using UnityEngine;
using UnityEngine.UI;
 
public class UICircle : RawImage
{
	const int FILL_PERCENT = 100;
	float thickness = 5;
 
	[SerializeField][Range(4,360)]
	int _segments = 36;
 
	public int segments 
	{
		get { return _segments;}
		set
			{ 
				if (_segments != value)
				 {
					_segments = value; 
					SetVerticesDirty ();
					#if UNITY_EDITOR
					UnityEditor.EditorUtility.SetDirty (transform);
					#endif
				}
			}
	}
 
	protected override void OnRectTransformDimensionsChange ()
	{
		base.OnRectTransformDimensionsChange ();
		this.thickness = (float)Mathf.Clamp (this.thickness, 0, rectTransform.rect.width / 2);
	}
    
	protected override void OnPopulateMesh(VertexHelper vh)
    {
        float outer = -rectTransform.pivot.x * rectTransform.rect.width;
        float inner = -rectTransform.pivot.x * rectTransform.rect.width + this.thickness;
 
        vh.Clear();
 
        Vector2 prevX = Vector2.zero;
        Vector2 prevY = Vector2.zero;
        Vector2 uv0 = new Vector2(0, 0);
        Vector2 uv1 = new Vector2(0, 1);
        Vector2 uv2 = new Vector2(1, 1);
        Vector2 uv3 = new Vector2(1, 0);
        Vector2 pos0;
        Vector2 pos1;
        Vector2 pos2;
        Vector2 pos3;
 
        float tw = rectTransform.rect.width;
        float th = rectTransform.rect.height;
 
        float angleByStep = (FILL_PERCENT / 100f * (Mathf.PI * 2f)) / segments;
        float currentAngle = 0f;
        for (int i = 0; i < segments + 1; i++)
        {
            float c = Mathf.Cos(currentAngle);
            float s = Mathf.Sin(currentAngle);
            StepThroughPointsAndFill(outer, inner, ref prevX, ref prevY, out pos0, out pos1, out pos2, out pos3, c, s);
            uv0 = new Vector2(pos0.x / tw + 0.5f, pos0.y / th + 0.5f);
            uv1 = new Vector2(pos1.x / tw + 0.5f, pos1.y / th + 0.5f);
            uv2 = new Vector2(pos2.x / tw + 0.5f, pos2.y / th + 0.5f);
            uv3 = new Vector2(pos3.x / tw + 0.5f, pos3.y / th + 0.5f);
            vh.AddUIVertexQuad(SetVbo(new[] { pos0, pos1, pos2, pos3 }, new[] { uv0, uv1, uv2, uv3 }));
            currentAngle += angleByStep;
        }
    }
 
    private void StepThroughPointsAndFill(float outer, float inner, ref Vector2 prevX, ref Vector2 prevY, out Vector2 pos0, out Vector2 pos1, out Vector2 pos2, out Vector2 pos3, float c, float s)
    {
        pos0 = prevX;
        pos1 = new Vector2(outer * c, outer * s);
 
        pos2 = Vector2.zero;
        pos3 = Vector2.zero;
 
        prevX = pos1;
        prevY = pos2;
    }
	protected UIVertex[] SetVbo(Vector2[] vertices, Vector2[] uvs)
	{
		UIVertex[] vbo = new UIVertex[4];
		for (int i = 0; i < vertices.Length; i++)
		{
			var vert = UIVertex.simpleVert;
			vert.color = color;
			vert.position = vertices[i];
			vert.uv0 = uvs[i];
			vbo[i] = vert;
		}
		return vbo;
	}
}

OK,因为继承的是RawImage所以缺点就是没办法合并批次了。(因为我们项目icon比较多,不想把图集撑得很大,而带来内存的问题,所以暂时没考虑合并批次的问题)
本文固定链接: https://www.xuanyusong.com/archives/4375
根据要求转载注明: 雨松MOMO 2017年04月28日 于 雨松MOMO程序研究院 发表

Unity里,要把RawImage变成圆形,可通过自定义组件的方式实现。以下是具体步骤与示例代码: ### 1. 创建自定义组件类 创建一个继承自`RawImage`的自定义组件类,在这个类里重写`OnPopulateMesh`方法来生成圆形的网格。 ```csharp using UnityEngine; using UnityEngine.UI; public class UICircle : RawImage { public int segments = 36; // 圆形的分段数,数值越大越接近圆形 protected override void OnPopulateMesh(VertexHelper vh) { vh.Clear(); float outer = rectTransform.pivot.x * rectTransform.sizeDelta.x; float inner = outer * 0.9f; // 可调整这个值来控制圆环的宽度 Vector2 prevX = Vector2.zero; Vector2 prevY = Vector2.zero; for (int i = 0; i <= segments; i++) { float c = Mathf.Cos(2 * Mathf.PI * i / segments); float s = Mathf.Sin(2 * Mathf.PI * i / segments); Vector2 pos0; Vector2 pos1; Vector2 pos2; Vector2 pos3; StepThroughPointsAndFill(outer, inner, ref prevX, ref prevY, out pos0, out pos1, out pos2, out pos3, c, s); vh.AddUIVertexQuad(SetVbo(new[] { pos0, pos1, pos2, pos3 }, new[] { new Vector2(0, 0), new Vector2(0, 1), new Vector2(1, 1), new Vector2(1, 0) })); prevX = pos2; prevY = pos3; } } private void StepThroughPointsAndFill(float outer, float inner, ref Vector2 prevX, ref Vector2 prevY, out Vector2 pos0, out Vector2 pos1, out Vector2 pos2, out Vector2 pos3, float c, float s) { pos0 = prevX; pos1 = new Vector2(outer * c, outer * s); pos2 = new Vector2(inner * c, inner * s); pos3 = prevY; } private UIVertex[] SetVbo(Vector2[] vertices, Vector2[] uvs) { UIVertex[] vbo = new UIVertex[4]; for (int i = 0; i < vertices.Length; i++) { var vert = UIVertex.simpleVert; vert.color = color; vert.position = vertices[i]; vert.uv0 = uvs[i]; vbo[i] = vert; } return vbo; } } ``` ### 2. 创建自定义编辑器类 创建一个自定义编辑器类,用于在Inspector面板中调整圆形的分段数。 ```csharp using UnityEditor; using UnityEditor.UI; using UnityEngine; [CustomEditor(typeof(UICircle), true)] [CanEditMultipleObjects] public class UICircleInspector : RawImageEditor { public override void OnInspectorGUI() { base.OnInspectorGUI(); UICircle circle = target as UICircle; circle.segments = Mathf.Clamp(EditorGUILayout.IntField("UICircle多边形", circle.segments), 4, 360); } } ``` ### 3. 使用自定义组件 - 在Unity的Hierarchy面板中创建一个RawImage对象。 - 移除RawImage组件,添加自定义的`UICircle`组件。 - 在Inspector面板中调整`UICircle多边形`的值,数值越大,圆形越平滑。 ### 总结 通过以上步骤,就能在Unity中将RawImage变成圆形。自定义组件类负责生成圆形的网格,自定义编辑器类则方便在Inspector面板中调整圆形的分段数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值