UGUI—能力雷达地图
作者:追风剑情 发布于:2025-4-23 23:45 分类:Unity3d
一、工程截图
二、代码
using System;
using UnityEngine;
using UnityEngine.UI;
/// <summary>
/// 能力雷达地图
/// </summary>
[ExecuteInEditMode]
public class UIAbilityRadarGraph : MonoBehaviour
{
[Serializable]
public class AbilityVector
{
public RectTransform rectTransform;
[Range(0.01f, 1.0f)]
public float value;
}
//特别注意:数组里的对象一定要顺时针排列
[Tooltip("能力向量对象")]
public AbilityVector[] abilityVectors;
[Tooltip("内部填充颜色")]
public Color innerColor = Color.blue;
[Tooltip("边框颜色")]
public Color borderColor = Color.red;
[Tooltip("边框宽度(单位:像素)")]
public int borderWidth = 10;
private RectTransform rectTransform;
private RawImage rawImage;
private float radius;
private Vector2[] directions;
private void Awake()
{
rectTransform = this.GetComponent<RectTransform>();
rawImage = GetComponent<RawImage>();
radius = rectTransform.sizeDelta.x / 2;
UpdateAbility();
}
private void Start()
{
Draw(directions, innerColor, borderColor);
}
private void UpdateAbility()
{
if (abilityVectors != null && abilityVectors.Length > 0)
{
directions = new Vector2[abilityVectors.Length];
for (int i = 0; i < abilityVectors.Length; i++)
{
var ab = abilityVectors[i];
var pos = ab.rectTransform.anchoredPosition;
directions[i] = pos.normalized * ab.value;
}
}
}
/// <summary>
/// 绘制能力雷达图
/// </summary>
/// <param name="values">能力值数组,取值范围[0,1]</param>
public void Draw(float[] values)
{
if (abilityVectors == null)
return;
for (int i = 0; i < abilityVectors.Length; i++)
{
if (i >= values.Length)
break;
float value = values[i];
value = Mathf.Clamp(value, 0.1f, 1.0f);
abilityVectors[i].value = value;
}
UpdateAbility();
Draw(directions, innerColor, borderColor);
}
/// <summary>
/// 绘制能力雷达图
/// </summary>
/// <param name="directions">能力方向向量</param>
/// <param name="innerColor">内部颜色</param>
/// <param name="borderColor">边框颜色</param>
public void Draw(Vector2[] directions, Color innerColor, Color borderColor)
{
if (directions == null || directions.Length <= 2)
return;
//归一化
foreach (var dv in directions)
dv.Normalize();
//创建Texture2D
RectTransform rt = rectTransform;
if (rt == null)
return;
int w = (int)rt.sizeDelta.x;
int h = (int)rt.sizeDelta.y;
Texture2D tex = new Texture2D(w, h, TextureFormat.ARGB32, false);
//构造边向量
Vector2[] edges = new Vector2[directions.Length];
for(int i=1; i<directions.Length; i++)
{
Vector2 v0 = directions[i-1];
Vector2 v1 = directions[i];
//顺时针构造边向量
edges[i-1] = v1 - v0;
}
//构造最后一条边向量
Vector2 firstV = directions[0];
Vector2 lastV = directions[directions.Length-1];
edges[edges.Length-1] = firstV - lastV;
Vector2 v = Vector2.zero;
//遍历Texture2D
for (int y = 0; y < h; y++)
{
for (int x = 0; x < w; x++)
{
//将区间转为 [-0.5, 0.5]
v.x = (float)x / (float)w - 0.5f;
v.y = (float)y / (float)h - 0.5f;
//将区间转为 [-1, 1]
v.x *= 2;
v.y *= 2;
bool inner = IsInner(v, directions, edges);
bool innerBorder = IsInnerBoder(v, directions, edges);
if (inner)
{
Color c = innerBorder ? borderColor : innerColor;
tex.SetPixel(x, y, c);
}
else
{
tex.SetPixel(x, y, Color.clear);
}
}
}
tex.Apply();
rawImage.texture = tex;
}
// 判断像素点是否在多边形内部
private bool IsInner(Vector2 v, Vector2[] directions, Vector2[] edges)
{
for(int i = 0; i < edges.Length; i++)
{
Vector2 ev = edges[i];
Vector2 dv = directions[i];
Vector2 pv = v - dv;
if (CrossProduct(ev, pv) > 0)
return false;
}
return true;
}
// 向量叉乘
private float CrossProduct(Vector2 v1, Vector2 v2)
{
return v1.x * v2.y - v2.x * v1.y;
}
// 是否在border内
private bool IsInnerBoder(Vector2 v, Vector2[] directions, Vector2[] edges)
{
float d = MinEdgeDistance(v, directions, edges);
return d <= borderWidth;
}
// 计算离边缘的最小距离
private float MinEdgeDistance(Vector2 v, Vector2[] directions, Vector2[] edges)
{
float min_d = float.MaxValue;
for (int i = 0; i < edges.Length; i++)
{
Vector2 ev = edges[i];
Vector2 dv = directions[i];
Vector2 pv = v - dv;
float d = Distance(ev, pv);
if (d < min_d) min_d = d;
}
return min_d;
}
// 计算离指定边的距离
private float Distance(Vector2 edge, Vector2 pv)
{
float angle = Vector2.Angle(edge, pv);
//弧度角
float radian = angle * Mathf.Deg2Rad;
//此时的d是归一化的
float d = pv.magnitude * Mathf.Sin(radian);
//转成像素尺寸
d = d * radius;
return d;
}
// Inspector上的值发生变化时调用
private void OnValidate()
{
UpdateAbility();
Draw(directions, innerColor, borderColor);
}
}
运行效果
标签: Unity3d
日历
最新文章
随机文章
热门文章
分类
存档
- 2025年11月(1)
- 2025年9月(3)
- 2025年7月(4)
- 2025年6月(5)
- 2025年5月(1)
- 2025年4月(5)
- 2025年3月(4)
- 2025年2月(3)
- 2025年1月(1)
- 2024年12月(5)
- 2024年11月(5)
- 2024年10月(5)
- 2024年9月(3)
- 2024年8月(3)
- 2024年7月(11)
- 2024年6月(3)
- 2024年5月(9)
- 2024年4月(10)
- 2024年3月(11)
- 2024年2月(24)
- 2024年1月(12)
- 2023年12月(3)
- 2023年11月(9)
- 2023年10月(7)
- 2023年9月(2)
- 2023年8月(7)
- 2023年7月(9)
- 2023年6月(6)
- 2023年5月(7)
- 2023年4月(11)
- 2023年3月(6)
- 2023年2月(11)
- 2023年1月(8)
- 2022年12月(2)
- 2022年11月(4)
- 2022年10月(10)
- 2022年9月(2)
- 2022年8月(13)
- 2022年7月(7)
- 2022年6月(11)
- 2022年5月(18)
- 2022年4月(29)
- 2022年3月(5)
- 2022年2月(6)
- 2022年1月(8)
- 2021年12月(5)
- 2021年11月(3)
- 2021年10月(4)
- 2021年9月(9)
- 2021年8月(14)
- 2021年7月(8)
- 2021年6月(5)
- 2021年5月(2)
- 2021年4月(3)
- 2021年3月(7)
- 2021年2月(2)
- 2021年1月(8)
- 2020年12月(7)
- 2020年11月(2)
- 2020年10月(6)
- 2020年9月(9)
- 2020年8月(10)
- 2020年7月(9)
- 2020年6月(18)
- 2020年5月(4)
- 2020年4月(25)
- 2020年3月(38)
- 2020年1月(21)
- 2019年12月(13)
- 2019年11月(29)
- 2019年10月(44)
- 2019年9月(17)
- 2019年8月(18)
- 2019年7月(25)
- 2019年6月(25)
- 2019年5月(17)
- 2019年4月(10)
- 2019年3月(36)
- 2019年2月(35)
- 2019年1月(28)
- 2018年12月(30)
- 2018年11月(22)
- 2018年10月(4)
- 2018年9月(7)
- 2018年8月(13)
- 2018年7月(13)
- 2018年6月(6)
- 2018年5月(5)
- 2018年4月(13)
- 2018年3月(5)
- 2018年2月(3)
- 2018年1月(8)
- 2017年12月(35)
- 2017年11月(17)
- 2017年10月(16)
- 2017年9月(17)
- 2017年8月(20)
- 2017年7月(34)
- 2017年6月(17)
- 2017年5月(15)
- 2017年4月(32)
- 2017年3月(8)
- 2017年2月(2)
- 2017年1月(5)
- 2016年12月(14)
- 2016年11月(26)
- 2016年10月(12)
- 2016年9月(25)
- 2016年8月(32)
- 2016年7月(14)
- 2016年6月(21)
- 2016年5月(17)
- 2016年4月(13)
- 2016年3月(8)
- 2016年2月(8)
- 2016年1月(18)
- 2015年12月(13)
- 2015年11月(15)
- 2015年10月(12)
- 2015年9月(18)
- 2015年8月(21)
- 2015年7月(35)
- 2015年6月(13)
- 2015年5月(9)
- 2015年4月(4)
- 2015年3月(5)
- 2015年2月(4)
- 2015年1月(13)
- 2014年12月(7)
- 2014年11月(5)
- 2014年10月(4)
- 2014年9月(8)
- 2014年8月(16)
- 2014年7月(26)
- 2014年6月(22)
- 2014年5月(28)
- 2014年4月(15)
友情链接
- Unity官网
- Unity圣典
- Unity在线手册
- Unity中文手册(圣典)
- Unity官方中文论坛
- Unity游戏蛮牛用户文档
- Unity下载存档
- Unity引擎源码下载
- Unity服务
- Unity Ads
- wiki.unity3d
- Visual Studio Code官网
- SenseAR开发文档
- MSDN
- C# 参考
- C# 编程指南
- .NET Framework类库
- .NET 文档
- .NET 开发
- WPF官方文档
- uLua
- xLua
- SharpZipLib
- Protobuf-net
- Protobuf.js
- OpenSSL
- OPEN CASCADE
- JSON
- MessagePack
- C在线工具
- 游戏蛮牛
- GreenVPN
- 聚合数据
- 热云
- 融云
- 腾讯云
- 腾讯开放平台
- 腾讯游戏服务
- 腾讯游戏开发者平台
- 腾讯课堂
- 微信开放平台
- 腾讯实时音视频
- 腾讯即时通信IM
- 微信公众平台技术文档
- 白鹭引擎官网
- 白鹭引擎开放平台
- 白鹭引擎开发文档
- FairyGUI编辑器
- PureMVC-TypeScript
- 讯飞开放平台
- 亲加通讯云
- Cygwin
- Mono开发者联盟
- Scut游戏服务器引擎
- KBEngine游戏服务器引擎
- Photon游戏服务器引擎
- 码云
- SharpSvn
- 腾讯bugly
- 4399原创平台
- 开源中国
- Firebase
- Firebase-Admob-Unity
- google-services-unity
- Firebase SDK for Unity
- Google-Firebase-SDK
- AppsFlyer SDK
- android-repository
- CQASO
- Facebook开发者平台
- gradle下载
- GradleBuildTool下载
- Android Developers
- Google中国开发者
- AndroidDevTools
- Android社区
- Android开发工具
- Google Play Games Services
- Google商店
- Google APIs for Android
- 金钱豹VPN
- TouchSense SDK
- MakeHuman
- Online RSA Key Converter
- Windows UWP应用
- Visual Studio For Unity
- Open CASCADE Technology
- 慕课网
- 阿里云服务器ECS
- 在线免费文字转语音系统
- AI Studio
- 网云穿
- 百度网盘开放平台
- 迅捷画图
- 菜鸟工具
- [CSDN] 程序员研修院
- 华为人脸识别
- 百度AR导航导览SDK
- 海康威视官网
- 海康开放平台
- 海康SDK下载
- git download
- Open CASCADE
- CascadeStudio
交流QQ群
-
Flash游戏设计: 86184192
Unity游戏设计: 171855449
游戏设计订阅号








