草庐IT

Unity绘制圆和缓动雷达图

x-zhi 2023-03-28 原文

Unity绘制圆和缓动雷达图

之前在做UI模块的时候遇到过需要做雷达图的效果,所以简单复习了一下

关于网格绘制

我们都知道Unity绘制图形是通过Mesh网格添加顶点进行绘制,那么知道顶点信息后如何实现图形的绘制,就是由Graphic类来帮助我们实现。因此我们需要自定义一个类继承Graphic,然后重写其OnPopulateMesh方法,将其挂载到UI物体上。

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

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

OnPopulateMesh方法中提供了VertexHelper参数,这是Unity的顶点辅助类,通过它我们可以轻松实现添加顶点和绘制等操作了。

添加顶点

//参数是顶点坐标,颜色以及uv坐标
new VertexHelper().AddVert(Vector3 position, Color32 color, Vector2 uv0)

顶点的绘制

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

绘制圆

对于Unity中的网格,我们只能通过一个个三角形拼接成不同的形状,包括Unity本身的物体网格也是如此,因此,画圆可以通过弧度和半径,以及圆的边长去绘制多个三角形,拼接而成一个圆

这是对画圆的一些总结

画圆的代码如下:

using UnityEngine;
using UnityEngine.UI;

public class RadarMap : Graphic
{
    public Texture2D texture;
    public Color _color = Color.white;
    public int n;
    public float r;
    public override Texture mainTexture//赋值UI图片
    {
        get
        {
            if (texture != null)
            {
                return texture;//如果有外部精灵则返回外部精灵的texture
            }
            if (material != null && material.mainTexture != null)
            {
                return material.mainTexture;//如果有材质则返回材质上的texture
            }
            return s_WhiteTexture;//什么都没有的情况返回默认的texture
        }
    }
    protected override void OnPopulateMesh(VertexHelper vh)
    {
        Rect rect = rectTransform.rect;
        r = rect.width < rect.height ? rect.width / 2 : rect.height / 2;//取当前最小的宽或高,取一半获得半径
        if (n >= 3)
        {
            vh.Clear();
            vh.AddVert(Vector3.zero, color, new Vector2(0.5f, 0.5f));//添加圆心点
            float ang = 2 * Mathf.PI / n;
            for (int i = 0; i < n; i++)
            {
                float x = Mathf.Sin(i * ang) * r;
                float y = Mathf.Cos(i * ang) * r;

                float uvx = (x + r) / (2 * r);//通过圆上点+半径/直径求出uv坐标
                float uvy = (y + r) / (2 * r);
                vh.AddVert(new Vector3(x, y, 0), color, new Vector3(uvx, uvy));//添加顶点
                if (i == 0)
                {
                    vh.AddTriangle(0, n, 1);//第一次绘制最后一部分
                }
                else
                {
                    vh.AddTriangle(0, i, i + 1);//依次绘制
                }
            }
        }
    }
}

画出圆的效果

绘制雷达图

根据绘制圆我们可以发现,绘制雷达图是同理的,同样是添加一个圆心点并根据半径和弧度求出圆上的顶点,只是雷达图每个顶点的位置都跟雷达图中的数值有关。因此,我们只需要添加一个数组,并将画圆的代码稍作修改就可以了

以下是雷达图的绘制代码:

using UnityEngine;
using UnityEngine.UI;
public class RadarMap : Graphic
{
    public Texture2D texture;
    public Color _color = Color.white;
    public float[] arr = new float[0];
    public float r;
    public override Texture mainTexture
    {
        get
        {
            if (texture != null)
            {
                return texture;//如果有外部精灵则返回外部精灵的texture
            }
            if (material != null && material.mainTexture != null)
            {
                return material.mainTexture;//如果有材质则返回材质上的texture
            }
            return s_WhiteTexture;//什么都没有的情况返回默认的texture
        }
    }
    //生成无背景雷达图 这里的生成方法和生成圆的方法是一致的
    protected override void OnPopulateMesh(VertexHelper vh)
    {
        Rect rect = rectTransform.rect;
        int n = arr.Length;
        //取当前最小的宽或高,取一半获得半径
        r = rect.width < rect.height ? rect.width / 2 : rect.height / 2;
        if (n >= 3)
        {
            vh.Clear();
            //添加圆心点
            vh.AddVert(Vector3.zero, color, new Vector2(0.5f, 0.5f));
            float ang = 2 * Mathf.PI / n;
            for (int i = 0; i < n; i++)
            {
                //通过数组中数据的值为雷达图赋值,最大值为半径
                float x = Mathf.Sin(i * ang) * (arr[i] < r ? arr[i] : r);
                float y = Mathf.Cos(i * ang) * (arr[i] < r ? arr[i] : r);
                //通过圆上点+半径/直径求出uv坐标
                float uvx = (x + r) / (2 * r);
                float uvy = (y + r) / (2 * r);
                //添加顶点
                vh.AddVert(new Vector3(x, y, 0), color, new Vector3(uvx, uvy));
                if (i == 0)
                {
                    //第一次绘制最后一部分
                    vh.AddTriangle(0, n, 1);
                }
                else
                {
                    //依次绘制
                    vh.AddTriangle(0, i, i + 1);
                }
            }
        }
    }
}

最后的效果如下

缓动则通过Mathf.Lerp的差值函数实现,也可以直接使用公式( max - min ) * p + min

代码如下:

//缓动代码块,这里的数组和集合已经赋值过
public RadarMap radarMap;
bool isDrow = false;
List<float> arr = new List<float>();
float[] arry;
float MaxTime = 3f;//缓动最大时间
float nowTime = 0f;//缓动开始时间
private void Update()
{
    if (isDrow)
    {
        nowTime += Time.deltaTime;
        for (int i = 0; i < arr.Count; i++)
        {
            //差值函数,始终在当前点和目标点中间取一个参数
            arry[i] = Mathf.Lerp(0, arr[i], nowTime / MaxTime);
            //当达到了这个值以后,循环结束,清空缓存原始数据的集合
            if (i == arr.Count - 1 && arr[i] - arry[i] <= 0.1f)
            {
                Mathf.Round(arry[i]);
                arr.Clear();
                isDrow = false;
                nowTime = 0f;
            }
        }
        //清除脏数据
        radarMap.SetAllDirty();
        //将每次算出的差值赋给RadarMap
        radarMap.arr = arry;
    }
}

有关Unity绘制圆和缓动雷达图的更多相关文章

  1. Unity 热更新技术 | (三) Lua语言基本介绍及下载安装 - 2

    ?博客主页:https://xiaoy.blog.csdn.net?本文由呆呆敲代码的小Y原创,首发于CSDN??学习专栏推荐:Unity系统学习专栏?游戏制作专栏推荐:游戏制作?Unity实战100例专栏推荐:Unity实战100例教程?欢迎点赞?收藏⭐留言?如有错误敬请指正!?未来很长,值得我们全力奔赴更美好的生活✨------------------❤️分割线❤️-------------------------

  2. FOHEART H1数据手套驱动Optitrack光学动捕双手运动(Unity3D) - 2

    本教程将在Unity3D中混合Optitrack与数据手套的数据流,在人体运动的基础上,添加双手手指部分的运动。双手手背的角度仍由Optitrack提供,数据手套提供双手手指的角度。 01  客户端软件分别安装MotiveBody与MotionVenus并校准人体与数据手套。MotiveBodyMotionVenus数据手套使用、校准流程参照:https://gitee.com/foheart_1/foheart-h1-data-summary.git02  数据转发打开MotiveBody软件的Streaming,开始向Unity3D广播数据;MotionVenus中设置->选项选择Unit

  3. unity---接入Admob - 2

    目录1.AdmobSDK下载地址2.将下载好的unityPackagesdk导入到unity里​编辑 3.解析依赖到项目中

  4. Unity 3D 制作开关门动画,旋转门制作,推拉门制作,门把手动画制作 - 2

    Unity自动旋转动画1.开门需要门把手先动,门再动2.关门需要门先动,门把手再动3.中途播放过程中不可以再次进行操作觉得太复杂?查看我的文章开关门简易进阶版效果:如果这个门可以直接打开的话,就不需要放置"门把手"如果门把手还有钥匙需要旋转,那就可以把钥匙放在门把手的"门把手",理论上是可以无限套娃的可调整参数有:角度,反向,轴向,速度运行时点击Test进行测试自己写的代码比较垃圾,命名与结构比较拉,高手轻点喷,新手有类似的需求可以拿去做参考上代码usingSystem.Collections;usingSystem.Collections.Generic;usingUnityEngine;u

  5. Unity Shader 学习笔记(5)Shader变体、Shader属性定义技巧、自定义材质面板 - 2

    写在之前Shader变体、Shader属性定义技巧、自定义材质面板,这三个知识点任何一个单拿出来都是一套知识体系,不能一概而论,本文章目的在于将学习和实际工作中遇见的问题进行总结,类似于网络笔记之用,方便后续回顾查看,如有以偏概全、不祥不尽之处,还望海涵。1、Shader变体先看一段代码......Properties{ [KeywordEnum(on,off)]USL_USE_COL("IsUseColorMixTex?",int)=0 [Toggle(IS_RED_ON)]_IsRed("IsRed?",int)=0}......//中间省略,后续会有完整代码 #pragmamulti_c

  6. 三分钟集成 TapTap 防沉迷 SDK(Unity 版) - 2

    三分钟集成Tap防沉迷SDK(Unity版)一、SDK介绍基于国家对上线所有游戏必须增加防沉迷功能的政策下,TapTap推出防沉迷SDK,供游戏开发者进行接入;允许未成年用户在周五、六、日以及法定节假日晚上8:00-9:00进行游戏,防沉谜时间段进入游戏会弹窗进行提示!开发环境要求:Unity2019.4或更高版本iOS10或更高版本Android5.0(APIlevel21)或更高版本🔗Unity集成Demo参考链接🔗UnityTapSDK功能体验APK下载链接二、集成前准备1.创建应用进入开发者后台,按照提示开始创建应用;2.开通服务在使用TDS实名认证和防沉迷服务之前,需要在上面创建的应

  7. 【Unity大气散射】GAMES104:3A中如何实现大气散射 - 2

    写在前面前两天学习并整理的大气散射基础知识:【Unity大气渲染】关于单次大气散射的理论知识,收获了很多,但不得不承认的是,这其实已经是最早的、90年代的非常古老的方法了,后来也出现了一些优化性的计算思路和方法。因此,我打算先不急着跟各种教程在Unity中实现大气散射,而是再花时间来看看最近的游戏是如何去实现大气渲染的:06.游戏中地形大气和云的渲染(下)|GAMES104-现代游戏引擎:从入门到实践接下来就跟着GAMES104讲地形大气和云渲染的部分学习并做简单的记录,涉及到之前没提到的Mie散射也只选择直接截图PPT的方式记录啦!毕竟对于做作品来说,之后实现出来才是重要的~当然,May佬的

  8. ruby - 使用 Ruby 编写 Unity 游戏 - 2

    所以我看到unity支持c#、JS和Boo。我可以学习其中一个,但我想制作一个“编译器”或类似的东西,让我可以编写ruby​​代码并输出JS代码或制作一个可以被Unity编译器读取的层。这有可能吗?我愿意在这方面投入很多时间并且有相当多的经验。 最佳答案 如果您的问题实际上是“我如何将Ruby编译为JavaScript”,那么这更容易回答:Opal:RubytoJavaScriptcompiler但是,学习其中一种受支持的语言会更好。当运行的是用另一种语言解释的代码时,很难调试“您的”代码。

  9. Unity常用文件夹 - 2

    1.Scenes游戏场景文件夹用于放置unity的场景文件 2.Plugins插件文件夹用于放置unity的依赖文件,例如dll 3.Scripts脚本文件夹用于放置unity的c#脚本文件 4.Resources游戏资源文件夹用于放置unity的各种游戏资源,比如images,prefabs,同时只有放到Resources文件夹的游戏资源才能使用Resource.load(资源路径不加后缀)加载到游戏内存中进行使用 5.EditorUnity编辑器扩展脚本文件夹usingUnityEditor;这个名称空间就是Unity编辑器的名称空间这个名称空间提供了扩展Unity编辑器的各种类 【你所有

  10. Unity数据可视化图表插件XCharts3.0发布 - 2

    Unity数据可视化图表插件XCharts3.0发布历时8个多月,业余时间,断断续续,XCharts3.0总算发布了。如果要打个满意度,我给3.0版本来个80分。对于代码框架结构设计的调整改动,基本符合预期,甚是满意。相比之前的1.0和2.0版本,我认为3.0才是一个拿得出手给广大开发者使用的版本。1.0发布的时候,很兴奋,从0.1到1.0,也磨了一年,真的等不及想给大家试用了,还特地写过一篇文章以示庆祝。那个时候,1.0虽然还还不够完善,功能也不够丰富,但它是XCharts的开始,没有1.0,也就没有后面的2.0和3.0。后面的2.0发布,做了很多改进和优化,随着版本迭代,慢慢的发现有不少硬

随机推荐