介绍:
雷达图是以从同一点开始的轴上表示的三个或更多个定量变量的二维图表的形式显示多变量数据的图形方法,它利用代表各项数据占比的顶点位置连接成面状,直观的反应当前事物的一个能力倾向。
例如下图:我们能直观的看到该玩家对于(输出,发育)两项的能力比较好?
雷达图基本原理介绍:
(这里我们以正五边形的雷达背景图为例,其他正多边形原理相同)
1.我们首先需要具备一张美术的雷达背景图,之后在背景图下创建Image;
2.之后重写image的OnPopulateMesh(VertexHelper vh)函数,清空所有顶点信息(VertexHelper.Clear() 清空后该image就不会进行显示);
3.之后我们需要构建五边形的5个顶点(初始位置在正五边形的五个顶点位置),之后创建外部系数控制5项数据的大小,得到5个顶点的最终显示位置,之后3个顶点构成面即可;
4.重新给VertexHelper对象添加顶点信息,三角面信息,我们原始的Image就会显示为五边形形状 来表现数据倾向。
大致来说:
我们游戏中的雷达图都是接收了数据源,计算得到该数据源每项占各项最大限度的比值,之后将原本的各项顶点的位置Position与比值相乘,得到新的顶点位置Position,再利用OnPopulateMesh(VertexHelper vh)重新绘制显示面。
Image重新绘制图形API:
了解:Image类继承自MaskableGraphic,实现了ISerializationCallbackReceiver, ILayoutElement, ICanvasRaycastFilter这三个接口。最关键的是MaskableGraphic类,MaskableGraphic负责绘制逻辑,MaskableGraphic继承自Graphic,Graphic里有个OnPopulateMesh函数,这正是我们需要的函数。
以正五边形为例
步骤1:
首先我们应该知道雷达背景图(所代表的圆)的半径radius,由于背景图的image尺寸不同,因此我们可以通过自己手动测量得到。
手动测量演示:
在背景图下创建一个物体,位置居于雷达图的中心,记录其posY1,向上拖到最顶端的顶点位置处,记录其posY2,利用
posY2-posY1即可得到radius
步骤2.
利用半径以及弧度求五边形雷达图最顶端的5个顶点坐标。
原理:5个顶点都是基于雷达图中心进行的一个坐标偏移,因此我们可以通过sin()*radius,cos()*radius求出这个偏移量,例如下图一。
图一:(黑线即五边形以及对应的圆, 红粗线为五边形对应的圆的半径)
注意:
由于在c#和unity中 Mathf.Cos()和Mathf.Sin() 传参时传递的为弧度(radian),我们需要把角度转换为弧度计算。
首先整圆的弧度为 2*pi ,因此我们可以计算出五边形每条边对应圆的弧度,即 (2*pi)/5 ,之后我们以1号顶点为起始点,逐渐加上
(2*pi)/5得到2,3,4,5号顶点的弧度。
即:
1号起始顶点对应弧度(整圆的1/4) =2pi / 4 ;
2号顶点对应弧度 =(2pi / 4) +(2pi) / 5 * 1;
3号顶点对应弧度 =(2pi / 4) +(2pi) / 5 * 2;
4号顶点对应弧度 =(2pi / 4) +(2pi) / 5 * 3;
5号顶点对应弧度 =(2pi / 4) +(2*pi) / 5 * 4;
之后有了五个弧度,即可直接求得5个顶点坐标:
private void SetPointPos() //设置点坐标
{
float radian = 2 * Mathf.PI / _pointCount; //计算出当前多边形的每一段对应填充圆的弧长
float radius = 130; //暂定当前正多边形的所填充圆的半径为100.
float curRadian = 2 * Mathf.PI / 4.0f; //计算出初始弧度位置 为正五边形的正上方 (起点选择与坐标轴延伸线相接的地方)
for (int i = 0; i < _pointCount; i++)
{
float x = Mathf.Cos(curRadian) * radius;
float y = Mathf.Sin(curRadian) * radius;
curRadian += radian;
_points[i].anchoredPosition = new Vector2(x, y);
}
}
步骤3.
设置雷达图的各项数据比例系数,得到每项数据对应雷达图的沿半径比例的准确顶点位置
各项数据比例系数是从外部传递的,我们不需要管,我们的目的是如何让雷达图按照这些数据给显示出来。
原理:
红圆圈是我们按照比例系数重新计算的顶点坐标,黑圆圈是我们的五个原始顶点坐标,我们不难看出,如果比例系数为1时,这项数据就是已经到顶了(顶点位置依旧不变),当小于1时,我们只需要将我们的顶点坐标与比例系数(radio)相乘就可以得到新的顶点坐标。
例如:1号顶点原本是黑圆圈位置,乘上比例系数radio=0.7的位置,是在黑圆圈与中心的连线的十分之七处。
步骤4
创建5个雷达图顶点UIVertex和一个雷达图中心点UIVertex
当我们知道了重新进行计算的5个雷达图顶点位置之后,我们直接创建5个顶点即可,创建顶点格式如下
UIVertex uiVertex=new UIVertex();
uiVertex.color = base.color; //这里指基类Image的颜色
uiVertex.position = pointPos; //雷达图顶点坐标
uiVertex.uv0=Vector2.one; //uv坐标 由于雷达图都是纯色显示,不依赖Texture,我们直接默认Vector2.one即可
注意:中心点UIVertex的坐标为 雷达背景图中心点位置 。
步骤5.
利用OnPopulateMesh(VertexHelper vertexHelper) 重新添加顶点信息以及三角面信息。
原理:
image的画面展示都是由OnPopulateMesh方法完成的,其中VertexHelper 是承载顶点信息与面信息的类,我们将它的数据清理掉后,添加我们的面和顶点信息,我们就能渲染出一个雷达图了。
protected override void OnPopulateMesh(VertexHelper vertexHelper)
{
vertexHelper.Clear();
AddVerts(vertexHelper); //添加全部顶点信息
AddTriangles(vertexHelper); //添加全部面信息
}
添加顶点api:VertexHelper.AddVert( Vertex vert)
添加面api: VertexHelper.AddTriangle(int index0,int index1,int index2)
VertexHelper是通过AddTriangle接口接受三角形信息:public void AddTriangle(int idx0, int idx1, int idx2)
接口的传入参数并不是UIVertex类型,而是int类型的索引值。哪来的索引?还记得之前往VertexHelper传入了一堆顶点吗?按照传入顺序,第一个顶点,索引记为0,依次类推。每次传入三个顶点的索引,就记录下了一个三角形。
需要注意:
GPU 默认是做backface culling(背面剔除)的,GPU只渲染正对屏幕的三角面片,当GPU认为某个三角面片是背对屏幕时,直接丢弃该三角面片,不做渲染。那么GPU怎么判断我们传入的某个三角形是正对屏幕,还是背对屏幕?答案是通过三个顶点的时针顺序,当三个顶点是呈顺时针时,判定为正对屏幕;呈逆时针时,判定为背对屏幕。
下图中:左边的图中指定顶点的顺序是顺时针的,右边是逆时针的。
因此:我们添加顶点时以 (1,0,2)(2,0,3)(3,0,4).。。。这样顺时针添加即可完成添加面的操作
步骤6. 项目展示:
求github小星星哈,谢谢啦?
Radarchart项目工程链接
(https://github.com/HanxianshengGame/Radarchart)
参考 scene Example 4与Script Example4 进行学习:
--------------------------------------------------------我是有底线的-------------------------------------------------------------
感谢能够观看博客的各位Unity开发爱好者们,有问题发表评论呐,★,°:.☆( ̄▽ ̄)/$:.°★ 。