RadarHandler.cs 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529
  1. using System.Collections.Generic;
  2. using UnityEngine;
  3. using UnityEngine.UI;
  4. using XUGL;
  5. namespace XCharts.Runtime
  6. {
  7. [UnityEngine.Scripting.Preserve]
  8. internal sealed class RadarHandler : SerieHandler<Radar>
  9. {
  10. private RadarCoord m_RadarCoord;
  11. public override void Update()
  12. {
  13. base.Update();
  14. }
  15. public override void DrawSerie(VertexHelper vh)
  16. {
  17. if (!serie.show) return;
  18. switch (serie.radarType)
  19. {
  20. case RadarType.Multiple:
  21. DrawMutipleRadar(vh);
  22. break;
  23. case RadarType.Single:
  24. DrawSingleRadar(vh);
  25. break;
  26. }
  27. }
  28. public override void UpdateTooltipSerieParams(int dataIndex, bool showCategory, string category,
  29. string marker, string itemFormatter, string numericFormatter, string ignoreDataDefaultContent,
  30. ref List<SerieParams> paramList, ref string title)
  31. {
  32. if (!serie.context.pointerEnter)
  33. return;
  34. dataIndex = serie.context.pointerItemDataIndex;
  35. if (dataIndex < 0)
  36. return;
  37. var radar = chart.GetChartComponent<RadarCoord>(serie.radarIndex);
  38. if (radar == null)
  39. return;
  40. if (serie.radarType == RadarType.Single)
  41. {
  42. var colorIndex1 = serie.colorByData ? dataIndex : serie.context.colorIndex;
  43. category = radar.GetIndicatorName(dataIndex);
  44. UpdateItemSerieParams(ref paramList, ref title, dataIndex, category,
  45. marker, itemFormatter, numericFormatter, ignoreDataDefaultContent, 1, colorIndex1);
  46. return;
  47. }
  48. var serieData = serie.GetSerieData(dataIndex);
  49. if (serieData == null)
  50. return;
  51. Color32 color, toColor;
  52. var colorIndex = serie.colorByData ? chart.GetLegendRealShowNameIndex(serieData.legendName) : serie.context.colorIndex;
  53. SerieHelper.GetItemColor(out color, out toColor, serie, serieData, chart.theme, colorIndex, SerieState.Normal);
  54. title = serieData.name;
  55. for (int i = 0; i < serieData.data.Count; i++)
  56. {
  57. var indicator = radar.GetIndicator(i);
  58. if (indicator == null) continue;
  59. var param = new SerieParams();
  60. param.serieName = serie.serieName;
  61. param.serieIndex = serie.index;
  62. param.dimension = i;
  63. param.serieData = serieData;
  64. param.dataCount = serie.dataCount;
  65. param.value = serieData.GetData(i);
  66. param.total = indicator.max;
  67. param.color = color;
  68. param.category = radar.GetIndicatorName(i);
  69. param.marker = SerieHelper.GetItemMarker(serie, serieData, marker);
  70. param.itemFormatter = SerieHelper.GetItemFormatter(serie, serieData, itemFormatter);
  71. param.numericFormatter = SerieHelper.GetNumericFormatter(serie, serieData, numericFormatter);
  72. param.columns.Clear();
  73. param.columns.Add(param.marker);
  74. param.columns.Add(indicator.name);
  75. param.columns.Add(ChartCached.NumberToStr(serieData.GetData(i), param.numericFormatter));
  76. paramList.Add(param);
  77. }
  78. }
  79. public override void UpdateSerieContext()
  80. {
  81. var needCheck = m_LegendEnter ||
  82. (chart.isPointerInChart && (m_RadarCoord != null && m_RadarCoord.IsPointerEnter()));
  83. var needInteract = false;
  84. if (!needCheck)
  85. {
  86. if (m_LastCheckContextFlag != needCheck)
  87. {
  88. m_LastCheckContextFlag = needCheck;
  89. serie.context.pointerItemDataIndex = -1;
  90. serie.context.pointerItemDataDimension = -1;
  91. serie.context.pointerEnter = false;
  92. foreach (var serieData in serie.data)
  93. {
  94. serieData.context.highlight = false;
  95. serieData.interact.Reset();
  96. }
  97. chart.RefreshPainter(serie);
  98. }
  99. return;
  100. }
  101. m_LastCheckContextFlag = needCheck;
  102. serie.highlight = false;
  103. serie.context.pointerEnter = false;
  104. serie.context.pointerItemDataIndex = -1;
  105. serie.context.pointerItemDataDimension = -1;
  106. var areaStyle = serie.areaStyle;
  107. var themeSymbolSize = chart.theme.serie.lineSymbolSize;
  108. switch (serie.radarType)
  109. {
  110. case RadarType.Multiple:
  111. for (int i = 0; i < serie.data.Count; i++)
  112. {
  113. var serieData = serie.data[i];
  114. var symbol = SerieHelper.GetSerieSymbol(serie, serieData);
  115. var symbolSize = symbol.GetSize(serieData.data, chart.theme.serie.lineSymbolSize);
  116. if (m_LegendEnter)
  117. {
  118. serieData.context.highlight = true;
  119. serieData.interact.SetValue(ref needInteract, serie.animation.interaction.GetRadius(symbolSize));
  120. }
  121. else
  122. {
  123. serieData.context.highlight = false;
  124. for (int n = 0; n < serieData.context.dataPoints.Count; n++)
  125. {
  126. var pos = serieData.context.dataPoints[n];
  127. if (Vector3.Distance(chart.pointerPos, pos) < symbolSize * 2)
  128. {
  129. serie.highlight = true;
  130. serie.context.pointerEnter = true;
  131. serie.context.pointerItemDataIndex = i;
  132. serie.context.pointerItemDataDimension = n;
  133. serieData.context.highlight = true;
  134. break;
  135. }
  136. }
  137. if (!serieData.context.highlight && areaStyle != null)
  138. {
  139. var center = m_RadarCoord.context.center;
  140. var dataPoints = serieData.context.dataPoints;
  141. for (int n = 0; n < dataPoints.Count; n++)
  142. {
  143. var p1 = dataPoints[n];
  144. var p2 = n >= dataPoints.Count - 1 ? dataPoints[0] : dataPoints[n + 1];
  145. if (UGLHelper.IsPointInTriangle(p1, center, p2, chart.pointerPos))
  146. {
  147. serie.highlight = true;
  148. serie.context.pointerEnter = true;
  149. serie.context.pointerItemDataIndex = i;
  150. serie.context.pointerItemDataDimension = n;
  151. serieData.context.highlight = true;
  152. break;
  153. }
  154. }
  155. }
  156. if (serieData.context.highlight)
  157. serieData.interact.SetValue(ref needInteract, serie.animation.interaction.GetRadius(symbolSize));
  158. else
  159. serieData.interact.SetValue(ref needInteract, symbolSize);
  160. }
  161. }
  162. break;
  163. case RadarType.Single:
  164. needInteract = false;
  165. for (int i = 0; i < serie.data.Count; i++)
  166. {
  167. var serieData = serie.data[i];
  168. var size = SerieHelper.GetSysmbolSize(serie, serieData, themeSymbolSize);
  169. if (Vector3.Distance(chart.pointerPos, serieData.context.position) < size * 2)
  170. {
  171. serie.context.pointerEnter = true;
  172. serie.context.pointerItemDataIndex = i;
  173. serie.context.pointerItemDataDimension = 1;
  174. serieData.context.highlight = true;
  175. needInteract = true;
  176. }
  177. else
  178. {
  179. serieData.context.highlight = false;
  180. }
  181. }
  182. if (!serie.context.pointerEnter && areaStyle != null)
  183. {
  184. var center = m_RadarCoord.context.center;
  185. var dataPoints = serie.data;
  186. for (int n = 0; n < dataPoints.Count; n++)
  187. {
  188. var p1 = dataPoints[n];
  189. var p2 = n >= dataPoints.Count - 1 ? dataPoints[0] : dataPoints[n + 1];
  190. if (UGLHelper.IsPointInTriangle(p1.context.position, center, p2.context.position, chart.pointerPos))
  191. {
  192. serie.context.pointerEnter = true;
  193. serie.context.pointerItemDataIndex = n;
  194. serie.context.pointerItemDataDimension = 1;
  195. p1.context.highlight = true;
  196. needInteract = true;
  197. break;
  198. }
  199. }
  200. }
  201. break;
  202. }
  203. if (needInteract)
  204. {
  205. chart.RefreshPainter(serie);
  206. }
  207. }
  208. private void DrawMutipleRadar(VertexHelper vh)
  209. {
  210. if (!serie.show) return;
  211. m_RadarCoord = chart.GetChartComponent<RadarCoord>(serie.radarIndex);
  212. if (m_RadarCoord == null) return;
  213. serie.containerIndex = m_RadarCoord.index;
  214. serie.containterInstanceId = m_RadarCoord.instanceId;
  215. var startPoint = Vector3.zero;
  216. var toPoint = Vector3.zero;
  217. var firstPoint = Vector3.zero;
  218. var indicatorNum = m_RadarCoord.indicatorList.Count;
  219. var angle = 2 * Mathf.PI / indicatorNum;
  220. var centerPos = m_RadarCoord.context.center;
  221. serie.animation.InitProgress(0, 1);
  222. if (!serie.show || serie.animation.HasFadeOut())
  223. {
  224. return;
  225. }
  226. var rate = serie.animation.GetCurrRate();
  227. var dataChanging = false;
  228. var interacting = false;
  229. SerieHelper.GetAllMinMaxData(serie, m_RadarCoord.ceilRate);
  230. Color32 areaColor, areaToColor;
  231. var startAngle = m_RadarCoord.startAngle * Mathf.PI / 180;
  232. var interactDuration = serie.animation.GetInteractionDuration();
  233. for (int j = 0; j < serie.data.Count; j++)
  234. {
  235. var serieData = serie.data[j];
  236. string dataName = serieData.name;
  237. if (!serieData.show)
  238. {
  239. continue;
  240. }
  241. var serieState = SerieHelper.GetSerieState(serie, serieData, true);
  242. var lineStyle = SerieHelper.GetLineStyle(serie, serieData);
  243. var symbol = SerieHelper.GetSerieSymbol(serie, serieData, serieState);
  244. var colorIndex = serie.colorByData ? chart.GetLegendRealShowNameIndex(serieData.legendName) : serie.context.colorIndex;
  245. var showArea = SerieHelper.GetAreaColor(out areaColor, out areaToColor, serie, serieData, chart.theme, colorIndex);
  246. var lineColor = SerieHelper.GetLineColor(serie, serieData, chart.theme, colorIndex);
  247. var lineWidth = lineStyle.GetWidth(chart.theme.serie.lineWidth);
  248. int dataCount = m_RadarCoord.indicatorList.Count;
  249. serieData.context.dataPoints.Clear();
  250. for (int n = 0; n < dataCount; n++)
  251. {
  252. if (n >= serieData.data.Count) break;
  253. var min = m_RadarCoord.GetIndicatorMin(n);
  254. var max = m_RadarCoord.GetIndicatorMax(n);
  255. var value = serieData.GetCurrData(n, serie.animation);
  256. if (serieData.IsDataChanged()) dataChanging = true;
  257. if (max == 0)
  258. {
  259. if (serie.data.Count > 1)
  260. {
  261. SerieHelper.GetMinMaxData(serie, n, out min, out max);
  262. min = ChartHelper.GetMinDivisibleValue(min, 0);
  263. max = ChartHelper.GetMaxDivisibleValue(max, 0);
  264. if (min > 0) min = 0;
  265. }
  266. else
  267. {
  268. max = serie.context.dataMax;
  269. }
  270. }
  271. if (max - min == 0) continue;
  272. var radius = (float) (m_RadarCoord.context.dataRadius * (value - min) / (max - min));
  273. var currAngle = startAngle + (n + (m_RadarCoord.positionType == RadarCoord.PositionType.Between ? 0.5f : 0)) * angle;
  274. radius *= rate;
  275. if (n == 0)
  276. {
  277. startPoint = new Vector3(centerPos.x + radius * Mathf.Sin(currAngle),
  278. centerPos.y + radius * Mathf.Cos(currAngle));
  279. firstPoint = startPoint;
  280. }
  281. else
  282. {
  283. toPoint = new Vector3(centerPos.x + radius * Mathf.Sin(currAngle),
  284. centerPos.y + radius * Mathf.Cos(currAngle));
  285. if (showArea && !serie.smooth)
  286. {
  287. UGL.DrawTriangle(vh, startPoint, toPoint, centerPos, areaColor, areaColor, areaToColor);
  288. }
  289. if (lineStyle.show && !serie.smooth)
  290. {
  291. ChartDrawer.DrawLineStyle(vh, lineStyle.type, lineWidth, startPoint, toPoint, lineColor);
  292. }
  293. startPoint = toPoint;
  294. }
  295. serieData.context.dataPoints.Add(startPoint);
  296. }
  297. if (showArea && !serie.smooth)
  298. {
  299. UGL.DrawTriangle(vh, startPoint, firstPoint, centerPos, areaColor, areaColor, areaToColor);
  300. }
  301. if (lineStyle.show && !serie.smooth)
  302. {
  303. ChartDrawer.DrawLineStyle(vh, lineStyle.type, lineWidth, startPoint, firstPoint, lineColor);
  304. }
  305. if (serie.smooth)
  306. {
  307. UGL.DrawCurves(vh, serieData.context.dataPoints, lineWidth, lineColor,
  308. chart.settings.lineSmoothStyle,
  309. chart.settings.lineSmoothness,
  310. UGL.Direction.Random,
  311. float.NaN, true);
  312. }
  313. if (symbol.show && symbol.type != SymbolType.None)
  314. {
  315. float symbolBorder = 0f;
  316. float[] cornerRadius = null;
  317. Color32 symbolColor, symbolToColor, symbolEmptyColor, borderColor;
  318. for (int m = 0; m < serieData.context.dataPoints.Count; m++)
  319. {
  320. var point = serieData.context.dataPoints[m];
  321. var symbolSize = 0f;
  322. if (!serieData.interact.TryGetValue(ref symbolSize, ref interacting, interactDuration))
  323. {
  324. symbolSize = SerieHelper.GetSysmbolSize(serie, serieData, chart.theme.serie.lineSymbolSize, serieState);
  325. serieData.interact.SetValue(ref interacting, symbolSize);
  326. symbolSize = serie.animation.GetSysmbolSize(symbolSize);
  327. }
  328. SerieHelper.GetItemColor(out symbolColor, out symbolToColor, out symbolEmptyColor, serie, serieData, chart.theme, colorIndex, serieState);
  329. SerieHelper.GetSymbolInfo(out borderColor, out symbolBorder, out cornerRadius, serie, serieData, chart.theme, serieState);
  330. chart.DrawSymbol(vh, symbol.type, symbolSize, symbolBorder, point, symbolColor,
  331. symbolToColor, symbolEmptyColor, borderColor, symbol.gap, cornerRadius);
  332. }
  333. }
  334. }
  335. if (!serie.animation.IsFinish())
  336. {
  337. serie.animation.CheckProgress(1);
  338. chart.RefreshPainter(serie);
  339. }
  340. if (dataChanging || interacting)
  341. {
  342. chart.RefreshPainter(serie);
  343. }
  344. }
  345. private void DrawSingleRadar(VertexHelper vh)
  346. {
  347. m_RadarCoord = chart.GetChartComponent<RadarCoord>(serie.radarIndex);
  348. if (m_RadarCoord == null)
  349. return;
  350. var indicatorNum = m_RadarCoord.indicatorList.Count;
  351. var angle = 2 * Mathf.PI / indicatorNum;
  352. var centerPos = m_RadarCoord.context.center;
  353. serie.animation.InitProgress(0, 1);
  354. if (!serie.show || serie.animation.HasFadeOut())
  355. {
  356. return;
  357. }
  358. var startPoint = Vector3.zero;
  359. var toPoint = Vector3.zero;
  360. var firstPoint = Vector3.zero;
  361. var lastColor = ColorUtil.clearColor32;
  362. var firstColor = ColorUtil.clearColor32;
  363. var rate = serie.animation.GetCurrRate();
  364. var dataChanging = false;
  365. var startIndex = GetStartShowIndex(serie);
  366. var endIndex = GetEndShowIndex(serie);
  367. var startAngle = m_RadarCoord.startAngle * Mathf.PI / 180;
  368. SerieHelper.UpdateMinMaxData(serie, 1, m_RadarCoord.ceilRate);
  369. for (int j = 0; j < serie.data.Count; j++)
  370. {
  371. var serieData = serie.data[j];
  372. string dataName = serieData.name;
  373. if (!serieData.show)
  374. {
  375. serieData.context.labelPosition = Vector3.zero;
  376. continue;
  377. }
  378. var lineStyle = SerieHelper.GetLineStyle(serie, serieData);
  379. Color32 areaColor, areaToColor;
  380. var colorIndex = serie.colorByData ? j : serie.context.colorIndex;
  381. var showArea = SerieHelper.GetAreaColor(out areaColor, out areaToColor, serie, serieData, chart.theme, colorIndex - 1);
  382. var lineColor = SerieHelper.GetLineColor(serie, serieData, chart.theme, colorIndex);
  383. int dataCount = m_RadarCoord.indicatorList.Count;
  384. var index = serieData.index;
  385. var p = m_RadarCoord.context.center;
  386. var max = m_RadarCoord.GetIndicatorMax(index);
  387. var value = serieData.GetCurrData(1, serie.animation);
  388. if (serieData.IsDataChanged()) dataChanging = true;
  389. if (max == 0)
  390. {
  391. max = serie.context.dataMax;
  392. }
  393. if (!m_RadarCoord.IsInIndicatorRange(j, serieData.GetData(1)))
  394. {
  395. lineColor = m_RadarCoord.outRangeColor;
  396. }
  397. var radius = (float) (max < 0 ? m_RadarCoord.context.dataRadius - m_RadarCoord.context.dataRadius * value / max :
  398. m_RadarCoord.context.dataRadius * value / max);
  399. var currAngle = startAngle + (index + (m_RadarCoord.positionType == RadarCoord.PositionType.Between ? 0.5f : 0)) * angle;
  400. radius *= rate;
  401. if (index == startIndex)
  402. {
  403. startPoint = new Vector3(p.x + radius * Mathf.Sin(currAngle),
  404. p.y + radius * Mathf.Cos(currAngle));
  405. firstPoint = startPoint;
  406. lastColor = lineColor;
  407. firstColor = lineColor;
  408. }
  409. else
  410. {
  411. toPoint = new Vector3(p.x + radius * Mathf.Sin(currAngle),
  412. p.y + radius * Mathf.Cos(currAngle));
  413. if (showArea && !serie.smooth)
  414. {
  415. UGL.DrawTriangle(vh, startPoint, toPoint, p, areaColor, areaColor, areaToColor);
  416. }
  417. if (lineStyle.show && !serie.smooth)
  418. {
  419. if (m_RadarCoord.connectCenter)
  420. ChartDrawer.DrawLineStyle(vh, lineStyle, startPoint, centerPos,
  421. chart.theme.serie.lineWidth, LineStyle.Type.Solid, lastColor, lastColor);
  422. ChartDrawer.DrawLineStyle(vh, lineStyle, startPoint, toPoint, chart.theme.serie.lineWidth,
  423. LineStyle.Type.Solid, m_RadarCoord.lineGradient ? lastColor : lineColor, lineColor);
  424. }
  425. startPoint = toPoint;
  426. lastColor = lineColor;
  427. }
  428. serie.context.dataPoints.Add(startPoint);
  429. serie.context.dataIndexs.Add(serieData.index);
  430. serieData.context.position = startPoint;
  431. serieData.context.labelPosition = startPoint;
  432. if (showArea && j == endIndex && !serie.smooth)
  433. {
  434. SerieHelper.GetAreaColor(out areaColor, out areaToColor, serie, serieData, chart.theme, colorIndex);
  435. UGL.DrawTriangle(vh, startPoint, firstPoint, centerPos, areaColor, areaColor, areaToColor);
  436. }
  437. if (lineStyle.show && j == endIndex && !serie.smooth)
  438. {
  439. if (m_RadarCoord.connectCenter)
  440. ChartDrawer.DrawLineStyle(vh, lineStyle, startPoint, centerPos,
  441. chart.theme.serie.lineWidth, LineStyle.Type.Solid, lastColor, lastColor);
  442. ChartDrawer.DrawLineStyle(vh, lineStyle, startPoint, firstPoint, chart.theme.serie.lineWidth,
  443. LineStyle.Type.Solid, lineColor, m_RadarCoord.lineGradient ? firstColor : lineColor);
  444. }
  445. }
  446. if (serie.smooth)
  447. {
  448. var lineWidth = serie.lineStyle.GetWidth(chart.theme.serie.lineWidth);
  449. var lineColor = SerieHelper.GetLineColor(serie, null, chart.theme, serie.context.colorIndex);
  450. UGL.DrawCurves(vh, serie.context.dataPoints, lineWidth, lineColor,
  451. chart.settings.lineSmoothStyle,
  452. chart.settings.lineSmoothness,
  453. UGL.Direction.Random,
  454. float.NaN, true);
  455. }
  456. if (serie.symbol.show && serie.symbol.type != SymbolType.None)
  457. {
  458. float symbolBorder = 0f;
  459. float[] cornerRadius = null;
  460. Color32 symbolColor, symbolToColor, symbolEmptyColor, borderColor;
  461. for (int j = 0; j < serie.data.Count; j++)
  462. {
  463. var serieData = serie.data[j];
  464. if (!serieData.show) continue;
  465. var state = SerieHelper.GetSerieState(serie, serieData);
  466. var symbolSize = SerieHelper.GetSysmbolSize(serie, serieData, chart.theme.serie.lineSymbolSize, state);
  467. var colorIndex = serie.colorByData ? serieData.index : serie.context.colorIndex;
  468. SerieHelper.GetItemColor(out symbolColor, out symbolToColor, out symbolEmptyColor, serie, serieData, chart.theme, colorIndex, state);
  469. SerieHelper.GetSymbolInfo(out borderColor, out symbolBorder, out cornerRadius, serie, serieData, chart.theme, state);
  470. if (!m_RadarCoord.IsInIndicatorRange(j, serieData.GetData(1)))
  471. {
  472. symbolColor = m_RadarCoord.outRangeColor;
  473. symbolToColor = m_RadarCoord.outRangeColor;
  474. }
  475. chart.DrawSymbol(vh, serie.symbol.type, symbolSize, symbolBorder, serieData.context.labelPosition, symbolColor,
  476. symbolToColor, symbolEmptyColor, borderColor, serie.symbol.gap, cornerRadius);
  477. }
  478. }
  479. if (!serie.animation.IsFinish())
  480. {
  481. serie.animation.CheckProgress(1);
  482. chart.RefreshPainter(serie);
  483. }
  484. if (dataChanging)
  485. {
  486. chart.RefreshPainter(serie);
  487. }
  488. }
  489. private int GetStartShowIndex(Serie serie)
  490. {
  491. for (int i = 0; i < serie.dataCount; i++)
  492. {
  493. if (serie.data[i].show) return i;
  494. }
  495. return 0;
  496. }
  497. private int GetEndShowIndex(Serie serie)
  498. {
  499. for (int i = serie.dataCount - 1; i >= 0; i--)
  500. {
  501. if (serie.data[i].show) return i;
  502. }
  503. return 0;
  504. }
  505. }
  506. }