RadarCoord.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493
  1. using System.Collections.Generic;
  2. using UnityEngine;
  3. using UnityEngine.UI;
  4. namespace XCharts.Runtime
  5. {
  6. /// <summary>
  7. /// Radar coordinate conponnet for radar charts.
  8. /// 雷达图坐标系组件,只适用于雷达图。
  9. /// </summary>
  10. [System.Serializable]
  11. [ComponentHandler(typeof(RadarCoordHandler), true)]
  12. [CoordOptions(typeof(RadarCoord))]
  13. public class RadarCoord : CoordSystem, ISerieContainer
  14. {
  15. /// <summary>
  16. /// Radar render type, in which 'Polygon' and 'Circle' are supported.
  17. /// ||雷达图绘制类型,支持 'Polygon' 和 'Circle'。
  18. /// </summary>
  19. public enum Shape
  20. {
  21. Polygon,
  22. Circle
  23. }
  24. /// <summary>
  25. /// The position type of radar.
  26. /// ||显示位置。
  27. /// </summary>
  28. public enum PositionType
  29. {
  30. /// <summary>
  31. /// Display at the vertex.
  32. /// ||显示在顶点处。
  33. /// </summary>
  34. Vertice,
  35. /// <summary>
  36. /// Display at the middle of line.
  37. /// ||显示在两者之间。
  38. /// </summary>
  39. Between,
  40. }
  41. /// <summary>
  42. /// Indicator of radar chart, which is used to assign multiple variables(dimensions) in radar chart.
  43. /// ||雷达图的指示器,用来指定雷达图中的多个变量(维度)。
  44. /// </summary>
  45. [System.Serializable]
  46. public class Indicator
  47. {
  48. [SerializeField] private string m_Name;
  49. [SerializeField] private double m_Max;
  50. [SerializeField] private double m_Min;
  51. [SerializeField] private double[] m_Range = new double[2] { 0, 0 };
  52. /// <summary>
  53. /// The name of indicator.
  54. /// ||指示器名称。
  55. /// </summary>
  56. public string name { get { return m_Name; } set { m_Name = value; } }
  57. /// <summary>
  58. /// The maximum value of indicator, with default value of 0, but we recommend to set it manually.
  59. /// ||指示器的最大值,默认为 0 无限制。
  60. /// </summary>
  61. public double max { get { return m_Max; } set { m_Max = value; } }
  62. /// <summary>
  63. /// The minimum value of indicator, with default value of 0.
  64. /// ||指示器的最小值,默认为 0 无限制。
  65. /// </summary>
  66. public double min { get { return m_Min; } set { m_Min = value; } }
  67. /// <summary>
  68. /// the text conponent of indicator.
  69. /// ||指示器的文本组件。
  70. /// </summary>
  71. public Text text { get; set; }
  72. /// <summary>
  73. /// Normal range. When the value is outside this range, the display color is automatically changed.
  74. /// ||正常值范围。当数值不在这个范围时,会自动变更显示颜色。
  75. /// </summary>
  76. public double[] range
  77. {
  78. get { return m_Range; }
  79. set { if (value != null && value.Length == 2) { m_Range = value; } }
  80. }
  81. public bool IsInRange(double value)
  82. {
  83. if (m_Range == null || m_Range.Length < 2) return true;
  84. if (m_Range[0] != 0 || m_Range[1] != 0)
  85. return value >= m_Range[0] && value <= m_Range[1];
  86. else
  87. return true;
  88. }
  89. }
  90. [SerializeField] private bool m_Show;
  91. [SerializeField] private Shape m_Shape;
  92. [SerializeField] private float m_Radius = 100;
  93. [SerializeField] private int m_SplitNumber = 5;
  94. [SerializeField] private float[] m_Center = new float[2] { 0.5f, 0.5f };
  95. [SerializeField] private AxisLine m_AxisLine = AxisLine.defaultAxisLine;
  96. [SerializeField] private AxisName m_AxisName = AxisName.defaultAxisName;
  97. [SerializeField] private AxisSplitLine m_SplitLine = AxisSplitLine.defaultSplitLine;
  98. [SerializeField] private AxisSplitArea m_SplitArea = AxisSplitArea.defaultSplitArea;
  99. [SerializeField] private bool m_Indicator = true;
  100. [SerializeField] private PositionType m_PositionType = PositionType.Vertice;
  101. [SerializeField] private float m_IndicatorGap = 10;
  102. [SerializeField] private double m_CeilRate = 0;
  103. [SerializeField] private bool m_IsAxisTooltip;
  104. [SerializeField] private Color32 m_OutRangeColor = Color.red;
  105. [SerializeField] private bool m_ConnectCenter = false;
  106. [SerializeField] private bool m_LineGradient = true;
  107. [SerializeField][Since("v3.4.0")] private float m_StartAngle;
  108. [SerializeField][Since("v3.8.0")] private int m_GridIndex = -1;
  109. [SerializeField] private List<Indicator> m_IndicatorList = new List<Indicator>();
  110. public RadarCoordContext context = new RadarCoordContext();
  111. /// <summary>
  112. /// [default:true]
  113. /// Set this to false to prevent the radar from showing.
  114. /// ||是否显示雷达坐标系组件。
  115. /// </summary>
  116. public bool show { get { return m_Show; } set { if (PropertyUtil.SetStruct(ref m_Show, value)) SetComponentDirty(); } }
  117. /// <summary>
  118. /// Index of layout component that serie uses. Default is -1 means not use layout, otherwise use the first layout component.
  119. /// ||所使用的 layout 组件的 index。 默认为-1不指定index, 当为大于或等于0时, 为第一个layout组件的第index个格子。
  120. /// </summary>
  121. public int gridIndex
  122. {
  123. get { return m_GridIndex; }
  124. set { if (PropertyUtil.SetStruct(ref m_GridIndex, value)) SetVerticesDirty(); }
  125. }
  126. /// <summary>
  127. /// Radar render type, in which 'Polygon' and 'Circle' are supported.
  128. /// ||雷达图绘制类型,支持 'Polygon' 和 'Circle'。
  129. /// </summary>
  130. public Shape shape
  131. {
  132. get { return m_Shape; }
  133. set { if (PropertyUtil.SetStruct(ref m_Shape, value)) SetAllDirty(); }
  134. }
  135. /// <summary>
  136. /// the radius of radar.
  137. /// ||雷达图的半径。
  138. /// </summary>
  139. public float radius
  140. {
  141. get { return m_Radius; }
  142. set { if (PropertyUtil.SetStruct(ref m_Radius, value)) SetAllDirty(); }
  143. }
  144. /// <summary>
  145. /// Segments of indicator axis.
  146. /// ||指示器轴的分割段数。
  147. /// </summary>
  148. public int splitNumber
  149. {
  150. get { return m_SplitNumber; }
  151. set { if (PropertyUtil.SetStruct(ref m_SplitNumber, value)) SetAllDirty(); }
  152. }
  153. /// <summary>
  154. /// the center of radar chart.
  155. /// ||雷达图的中心点。数组的第一项是横坐标,第二项是纵坐标。
  156. /// 当值为0-1之间时表示百分比,设置成百分比时第一项是相对于容器宽度,第二项是相对于容器高度。
  157. /// </summary>
  158. public float[] center
  159. {
  160. get { return m_Center; }
  161. set { if (value != null) { m_Center = value; SetAllDirty(); } }
  162. }
  163. /// <summary>
  164. /// axis line.
  165. /// ||轴线。
  166. /// </summary>
  167. public AxisLine axisLine
  168. {
  169. get { return m_AxisLine; }
  170. set { if (PropertyUtil.SetClass(ref m_AxisLine, value, true)) SetAllDirty(); }
  171. }
  172. /// <summary>
  173. /// Name options for radar indicators.
  174. /// ||雷达图每个指示器名称的配置项。
  175. /// </summary>
  176. public AxisName axisName
  177. {
  178. get { return m_AxisName; }
  179. set { if (PropertyUtil.SetClass(ref m_AxisName, value, true)) SetAllDirty(); }
  180. }
  181. /// <summary>
  182. /// split line.
  183. /// ||分割线。
  184. /// </summary>
  185. public AxisSplitLine splitLine
  186. {
  187. get { return m_SplitLine; }
  188. set { if (PropertyUtil.SetClass(ref m_SplitLine, value, true)) SetAllDirty(); }
  189. }
  190. /// <summary>
  191. /// Split area of axis in grid area.
  192. /// ||分割区域。
  193. /// </summary>
  194. public AxisSplitArea splitArea
  195. {
  196. get { return m_SplitArea; }
  197. set { if (PropertyUtil.SetClass(ref m_SplitArea, value, true)) SetAllDirty(); }
  198. }
  199. /// <summary>
  200. /// Whether to show indicator.
  201. /// ||是否显示指示器。
  202. /// </summary>
  203. public bool indicator
  204. {
  205. get { return m_Indicator; }
  206. set { if (PropertyUtil.SetStruct(ref m_Indicator, value)) SetComponentDirty(); }
  207. }
  208. /// <summary>
  209. /// The gap of indicator and radar.
  210. /// ||指示器和雷达的间距。
  211. /// </summary>
  212. public float indicatorGap
  213. {
  214. get { return m_IndicatorGap; }
  215. set { if (PropertyUtil.SetStruct(ref m_IndicatorGap, value)) SetComponentDirty(); }
  216. }
  217. /// <summary>
  218. /// The ratio of maximum and minimum values rounded upward. The default is 0, which is automatically calculated.
  219. /// ||最大最小值向上取整的倍率。默认为0时自动计算。
  220. /// </summary>
  221. public double ceilRate
  222. {
  223. get { return m_CeilRate; }
  224. set { if (PropertyUtil.SetStruct(ref m_CeilRate, value < 0 ? 0 : value)) SetAllDirty(); }
  225. }
  226. /// <summary>
  227. /// 是否Tooltip显示轴线上的所有数据。
  228. /// </summary>
  229. public bool isAxisTooltip
  230. {
  231. get { return m_IsAxisTooltip; }
  232. set { if (PropertyUtil.SetStruct(ref m_IsAxisTooltip, value)) SetAllDirty(); }
  233. }
  234. /// <summary>
  235. /// The position type of indicator.
  236. /// ||显示位置类型。
  237. /// </summary>
  238. public PositionType positionType
  239. {
  240. get { return m_PositionType; }
  241. set { if (PropertyUtil.SetStruct(ref m_PositionType, value)) SetAllDirty(); }
  242. }
  243. /// <summary>
  244. /// The color displayed when data out of range.
  245. /// ||数值超出范围时显示的颜色。
  246. /// </summary>
  247. public Color32 outRangeColor
  248. {
  249. get { return m_OutRangeColor; }
  250. set { if (PropertyUtil.SetStruct(ref m_OutRangeColor, value)) SetAllDirty(); }
  251. }
  252. /// <summary>
  253. /// Whether serie data connect to radar center with line.
  254. /// ||数值是否连线到中心点。
  255. /// </summary>
  256. public bool connectCenter
  257. {
  258. get { return m_ConnectCenter; }
  259. set { if (PropertyUtil.SetStruct(ref m_ConnectCenter, value)) SetAllDirty(); }
  260. }
  261. /// <summary>
  262. /// Whether need gradient for data line.
  263. /// ||数值线段是否需要渐变。
  264. /// </summary>
  265. public bool lineGradient
  266. {
  267. get { return m_LineGradient; }
  268. set { if (PropertyUtil.SetStruct(ref m_LineGradient, value)) SetAllDirty(); }
  269. }
  270. /// <summary>
  271. /// 起始角度。和时钟一样,12点钟位置是0度,顺时针到360度。
  272. /// </summary>
  273. public float startAngle
  274. {
  275. get { return m_StartAngle; }
  276. set { if (PropertyUtil.SetStruct(ref m_StartAngle, value)) SetVerticesDirty(); }
  277. }
  278. /// <summary>
  279. /// the indicator list.
  280. /// ||指示器列表。
  281. /// </summary>
  282. public List<Indicator> indicatorList { get { return m_IndicatorList; } }
  283. public bool IsPointerEnter()
  284. {
  285. return context.isPointerEnter;
  286. }
  287. public override void SetDefaultValue()
  288. {
  289. m_Show = true;
  290. m_GridIndex = -1;
  291. m_Shape = Shape.Polygon;
  292. m_Radius = 0.35f;
  293. m_SplitNumber = 5;
  294. m_Indicator = true;
  295. m_IndicatorList = new List<Indicator>(5)
  296. {
  297. new Indicator() { name = "indicator1", max = 0 },
  298. new Indicator() { name = "indicator2", max = 0 },
  299. new Indicator() { name = "indicator3", max = 0 },
  300. new Indicator() { name = "indicator4", max = 0 },
  301. new Indicator() { name = "indicator5", max = 0 },
  302. };
  303. center[0] = 0.5f;
  304. center[1] = 0.4f;
  305. splitLine.show = true;
  306. splitArea.show = true;
  307. axisName.show = true;
  308. axisName.name = null;
  309. }
  310. private bool IsEqualsIndicatorList(List<Indicator> indicators1, List<Indicator> indicators2)
  311. {
  312. if (indicators1.Count != indicators2.Count) return false;
  313. for (int i = 0; i < indicators1.Count; i++)
  314. {
  315. var indicator1 = indicators1[i];
  316. var indicator2 = indicators2[i];
  317. if (!indicator1.Equals(indicator2)) return false;
  318. }
  319. return true;
  320. }
  321. public bool IsInIndicatorRange(int index, double value)
  322. {
  323. var indicator = GetIndicator(index);
  324. return indicator == null ? true : indicator.IsInRange(value);
  325. }
  326. public double GetIndicatorMin(int index)
  327. {
  328. if (index >= 0 && index < m_IndicatorList.Count)
  329. {
  330. return m_IndicatorList[index].min;
  331. }
  332. return 0;
  333. }
  334. public double GetIndicatorMax(int index)
  335. {
  336. if (index >= 0 && index < m_IndicatorList.Count)
  337. {
  338. return m_IndicatorList[index].max;
  339. }
  340. return 0;
  341. }
  342. internal void UpdateRadarCenter(BaseChart chart)
  343. {
  344. if (center.Length < 2) return;
  345. var chartPosition = chart.chartPosition;
  346. var chartWidth = chart.chartWidth;
  347. var chartHeight = chart.chartHeight;
  348. if (gridIndex >= 0)
  349. {
  350. var layout = chart.GetChartComponent<GridLayout>(0);
  351. if (layout != null)
  352. {
  353. layout.UpdateRuntimeData(chart);
  354. layout.UpdateGridContext(gridIndex, ref chartPosition, ref chartWidth, ref chartHeight);
  355. }
  356. }
  357. var centerX = center[0] <= 1 ? chartWidth * center[0] : center[0];
  358. var centerY = center[1] <= 1 ? chartHeight * center[1] : center[1];
  359. context.center = chartPosition + new Vector3(centerX, centerY);
  360. if (radius <= 0)
  361. {
  362. context.radius = 0;
  363. }
  364. else if (radius <= 1)
  365. {
  366. context.radius = Mathf.Min(chartWidth, chartHeight) * radius;
  367. }
  368. else
  369. {
  370. context.radius = radius;
  371. }
  372. if (shape == RadarCoord.Shape.Polygon && positionType == PositionType.Between)
  373. {
  374. var angle = Mathf.PI / indicatorList.Count;
  375. context.dataRadius = context.radius * Mathf.Cos(angle);
  376. }
  377. else
  378. {
  379. context.dataRadius = context.radius;
  380. }
  381. }
  382. public Vector3 GetIndicatorPosition(int index)
  383. {
  384. int indicatorNum = indicatorList.Count;
  385. var angle = 0f;
  386. switch (positionType)
  387. {
  388. case PositionType.Vertice:
  389. angle = 2 * Mathf.PI / indicatorNum * index;
  390. break;
  391. case PositionType.Between:
  392. angle = 2 * Mathf.PI / indicatorNum * (index + 0.5f);
  393. break;
  394. }
  395. angle += startAngle * Mathf.PI / 180;
  396. var x = context.center.x + (context.radius + indicatorGap) * Mathf.Sin(angle);
  397. var y = context.center.y + (context.radius + indicatorGap) * Mathf.Cos(angle);
  398. return new Vector3(x, y);
  399. }
  400. public void AddIndicator(RadarCoord.Indicator indicator)
  401. {
  402. indicatorList.Add(indicator);
  403. SetAllDirty();
  404. }
  405. public RadarCoord.Indicator AddIndicator(string name, double min, double max)
  406. {
  407. var indicator = new RadarCoord.Indicator();
  408. indicator.name = name;
  409. indicator.min = min;
  410. indicator.max = max;
  411. indicatorList.Add(indicator);
  412. SetAllDirty();
  413. return indicator;
  414. }
  415. [Since("v3.3.0")]
  416. public void AddIndicatorList(List<string> nameList, double min = 0, double max = 0)
  417. {
  418. foreach (var name in nameList)
  419. AddIndicator(name, min, max);
  420. }
  421. public bool UpdateIndicator(int indicatorIndex, string name, double min, double max)
  422. {
  423. var indicator = GetIndicator(indicatorIndex);
  424. if (indicator == null) return false;
  425. indicator.name = name;
  426. indicator.min = min;
  427. indicator.max = max;
  428. SetAllDirty();
  429. return true;
  430. }
  431. public RadarCoord.Indicator GetIndicator(int indicatorIndex)
  432. {
  433. if (indicatorIndex < 0 || indicatorIndex > indicatorList.Count - 1) return null;
  434. return indicatorList[indicatorIndex];
  435. }
  436. public string GetIndicatorName(int indicatorIndex)
  437. {
  438. var indicator = GetIndicator(indicatorIndex);
  439. if (indicator == null) return string.Empty;
  440. return indicator.name;
  441. }
  442. public override void ClearData()
  443. {
  444. indicatorList.Clear();
  445. }
  446. public string GetFormatterIndicatorContent(int indicatorIndex)
  447. {
  448. var indicator = GetIndicator(indicatorIndex);
  449. if (indicator == null)
  450. return string.Empty;
  451. else
  452. return GetFormatterIndicatorContent(indicator.name);
  453. }
  454. public string GetFormatterIndicatorContent(string indicatorName)
  455. {
  456. if (string.IsNullOrEmpty(indicatorName))
  457. return indicatorName;
  458. if (string.IsNullOrEmpty(m_AxisName.labelStyle.formatter))
  459. {
  460. return indicatorName;
  461. }
  462. else
  463. {
  464. var content = m_AxisName.labelStyle.formatter;
  465. FormatterHelper.ReplaceAxisLabelContent(ref content, indicatorName);
  466. return content;
  467. }
  468. }
  469. }
  470. }