PieHandler.cs 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736
  1. using System.Collections.Generic;
  2. using UnityEngine;
  3. using UnityEngine.EventSystems;
  4. using UnityEngine.UI;
  5. using XUGL;
  6. namespace XCharts.Runtime
  7. {
  8. [UnityEngine.Scripting.Preserve]
  9. internal sealed class PieHandler : SerieHandler<Pie>
  10. {
  11. public override void Update()
  12. {
  13. base.Update();
  14. }
  15. public override void DrawBase(VertexHelper vh)
  16. {
  17. UpdateRuntimeData(serie);
  18. DrawPieLabelLine(vh, serie, false);
  19. }
  20. public override void DrawSerie(VertexHelper vh)
  21. {
  22. UpdateRuntimeData(serie);
  23. DrawPie(vh, serie);
  24. chart.RefreshBasePainter();
  25. }
  26. public override void DrawUpper(VertexHelper vh)
  27. {
  28. DrawPieLabelLine(vh, serie, true);
  29. }
  30. public override void UpdateTooltipSerieParams(int dataIndex, bool showCategory, string category,
  31. string marker, string itemFormatter, string numericFormatter, string ignoreDataDefaultContent,
  32. ref List<SerieParams> paramList, ref string title)
  33. {
  34. UpdateItemSerieParams(ref paramList, ref title, dataIndex, category,
  35. marker, itemFormatter, numericFormatter, ignoreDataDefaultContent);
  36. }
  37. public override Vector3 GetSerieDataLabelPosition(SerieData serieData, LabelStyle label)
  38. {
  39. var labelLine = SerieHelper.GetSerieLabelLine(serie, serieData);
  40. if (labelLine != null && labelLine.show && serieData.labelObject != null)
  41. {
  42. var currAngle = serieData.context.halfAngle - serie.context.startAngle;
  43. var isLeft = currAngle > 180 || (currAngle == 0 && serieData.context.startAngle > 0);
  44. var textOffset = serieData.labelObject.text.GetPreferredWidth() / 2;
  45. return serieData.context.labelPosition + (isLeft ? Vector3.left : Vector3.right) * textOffset;
  46. }
  47. else
  48. {
  49. return serieData.context.labelPosition;
  50. }
  51. }
  52. public override Vector3 GetSerieDataLabelOffset(SerieData serieData, LabelStyle label)
  53. {
  54. var offset = label.GetOffset(serie.context.insideRadius);
  55. if (label.autoOffset)
  56. {
  57. var currAngle = serieData.context.halfAngle - serie.context.startAngle;
  58. var isLeft = currAngle > 180 || (currAngle == 0 && serieData.context.startAngle > 0);
  59. if (isLeft)
  60. return new Vector3(-offset.x, offset.y, offset.z);
  61. else
  62. return offset;
  63. }
  64. else
  65. {
  66. return offset;
  67. }
  68. }
  69. public override Vector3 GetSerieDataTitlePosition(SerieData serieData, TitleStyle titleStyle)
  70. {
  71. return serie.context.center;
  72. }
  73. public override void OnPointerDown(PointerEventData eventData)
  74. {
  75. if (chart.pointerPos == Vector2.zero) return;
  76. var dataIndex = GetPiePosIndex(serie, chart.pointerPos);
  77. var refresh = false;
  78. if (dataIndex >= 0)
  79. {
  80. refresh = true;
  81. for (int j = 0; j < serie.data.Count; j++)
  82. {
  83. if (j == dataIndex) serie.data[j].context.selected = !serie.data[j].context.selected;
  84. else serie.data[j].context.selected = false;
  85. }
  86. }
  87. if (refresh) chart.RefreshChart();
  88. base.OnPointerDown(eventData);
  89. }
  90. public override int GetPointerItemDataIndex()
  91. {
  92. return GetPiePosIndex(serie, chart.pointerPos);
  93. }
  94. public override void UpdateSerieContext()
  95. {
  96. var needCheck = m_LegendEnter || m_LegendExiting || m_ForceUpdateSerieContext || (chart.isPointerInChart && PointerIsInPieSerie(serie, chart.pointerPos));
  97. var needInteract = false;
  98. var interactEnable = serie.animation.enable && serie.animation.interaction.enable;
  99. Color32 color, toColor;
  100. if (!needCheck)
  101. {
  102. if (m_LastCheckContextFlag != needCheck || m_ForceUpdateSerieContext)
  103. {
  104. serie.context.pointerItemDataIndex = -1;
  105. serie.context.pointerEnter = false;
  106. bool isAllZeroValue1 = SerieHelper.IsAllZeroValue(serie, 1);
  107. var zeroReplaceValue1 = isAllZeroValue1 ? 360 / serie.dataCount : 0;
  108. foreach (var serieData in serie.data)
  109. {
  110. serieData.context.highlight = false;
  111. if (interactEnable)
  112. {
  113. var value = isAllZeroValue1 ? zeroReplaceValue1 : serieData.GetCurrData(1, serie.animation);
  114. var colorIndex = chart.GetLegendRealShowNameIndex(serieData.legendName);
  115. SerieHelper.GetItemColor(out color, out toColor, serie, serieData, chart.theme, colorIndex, SerieState.Normal);
  116. UpdateSerieDataRadius(serieData, value);
  117. serieData.interact.SetValueAndColor(ref needInteract, serieData.context.outsideRadius, color, toColor);
  118. serieData.interact.SetPosition(ref needInteract, serieData.context.offsetCenter);
  119. }
  120. }
  121. if (needInteract)
  122. {
  123. chart.RefreshPainter(serie);
  124. }
  125. else
  126. {
  127. m_LastCheckContextFlag = needCheck;
  128. m_LegendExiting = false;
  129. chart.RefreshPainter(serie);
  130. }
  131. }
  132. return;
  133. }
  134. m_LastCheckContextFlag = needCheck;
  135. var lastPointerItemDataIndex = serie.context.pointerItemDataIndex;
  136. var dataIndex = GetPiePosIndex(serie, chart.pointerPos);
  137. serie.context.pointerItemDataIndex = -1;
  138. serie.context.pointerEnter = dataIndex >= 0;
  139. bool isAllZeroValue = SerieHelper.IsAllZeroValue(serie, 1);
  140. var zeroReplaceValue = isAllZeroValue ? 360 / serie.dataCount : 0;
  141. for (int i = 0; i < serie.dataCount; i++)
  142. {
  143. var serieData = serie.data[i];
  144. var value = isAllZeroValue ? zeroReplaceValue : serieData.GetCurrData(1, serie.animation);
  145. var state = SerieState.Normal;
  146. if (dataIndex == i || (m_LegendEnter && m_LegendEnterIndex == i))
  147. {
  148. serie.context.pointerItemDataIndex = i;
  149. serieData.context.highlight = true;
  150. state = SerieState.Emphasis;
  151. }
  152. else
  153. {
  154. serieData.context.highlight = false;
  155. }
  156. if (interactEnable)
  157. {
  158. UpdateSerieDataRadius(serieData, value);
  159. var colorIndex = chart.GetLegendRealShowNameIndex(serieData.legendName);
  160. SerieHelper.GetItemColor(out color, out toColor, serie, serieData, chart.theme, colorIndex, state);
  161. serieData.interact.SetValueAndColor(ref needInteract, serieData.context.outsideRadius, color, toColor);
  162. serieData.interact.SetPosition(ref needInteract, serieData.context.offsetCenter);
  163. }
  164. }
  165. if (lastPointerItemDataIndex != serie.context.pointerItemDataIndex)
  166. {
  167. needInteract = true;
  168. }
  169. if (needInteract)
  170. {
  171. chart.RefreshPainter(serie);
  172. }
  173. }
  174. private void UpdateRuntimeData(Serie serie)
  175. {
  176. var data = serie.data;
  177. serie.context.dataMax = serie.yMax;
  178. serie.context.startAngle = GetStartAngle(serie);
  179. var runtimePieDataTotal = serie.yTotal;
  180. SerieHelper.UpdateCenter(serie, chart);
  181. float startDegree = serie.context.startAngle;
  182. float totalDegree = 0;
  183. float zeroReplaceValue = 0;
  184. int showdataCount = 0;
  185. foreach (var sd in serie.data)
  186. {
  187. if (sd.show && serie.pieRoseType == RoseType.Area) showdataCount++;
  188. sd.context.canShowLabel = false;
  189. }
  190. bool isAllZeroValue = SerieHelper.IsAllZeroValue(serie, 1);
  191. var dataTotalFilterMinAngle = runtimePieDataTotal;
  192. if (isAllZeroValue)
  193. {
  194. totalDegree = 360;
  195. zeroReplaceValue = totalDegree / data.Count;
  196. serie.context.dataMax = zeroReplaceValue;
  197. runtimePieDataTotal = 360;
  198. dataTotalFilterMinAngle = 360;
  199. }
  200. else
  201. {
  202. dataTotalFilterMinAngle = GetTotalAngle(serie, runtimePieDataTotal, ref totalDegree);
  203. }
  204. if (dataTotalFilterMinAngle == 0)
  205. {
  206. dataTotalFilterMinAngle = 360;
  207. }
  208. for (int n = 0; n < data.Count; n++)
  209. {
  210. var serieData = data[n];
  211. var value = isAllZeroValue ? zeroReplaceValue : serieData.GetCurrData(1, serie.animation);
  212. serieData.context.startAngle = startDegree;
  213. serieData.context.toAngle = startDegree;
  214. serieData.context.halfAngle = startDegree;
  215. serieData.context.currentAngle = startDegree;
  216. if (!serieData.show)
  217. {
  218. continue;
  219. }
  220. float degree = serie.pieRoseType == RoseType.Area ?
  221. (totalDegree / showdataCount) :
  222. (float)(totalDegree * value / dataTotalFilterMinAngle);
  223. if (serie.minAngle > 0 && degree < serie.minAngle) degree = serie.minAngle;
  224. serieData.context.toAngle = startDegree + degree;
  225. var halfDegree = (serieData.context.toAngle - startDegree) / 2;
  226. serieData.context.halfAngle = startDegree + halfDegree;
  227. serieData.context.angle = startDegree + halfDegree;
  228. serieData.context.currentAngle = serie.animation.CheckDetailBreak(serieData.context.toAngle)
  229. ? serie.animation.GetCurrDetail() : serieData.context.toAngle;
  230. serieData.context.insideRadius = serie.context.insideRadius;
  231. serieData.context.canShowLabel = serieData.context.currentAngle >= serieData.context.halfAngle && !serie.IsMinShowLabelValue(value);
  232. UpdateSerieDataRadius(serieData, value);
  233. UpdatePieLabelPosition(serie, serieData);
  234. startDegree = serieData.context.toAngle;
  235. }
  236. AvoidLabelOverlap(serie, chart.theme.common);
  237. }
  238. private void UpdateSerieDataRadius(SerieData serieData, double value)
  239. {
  240. var minChartWidth = Mathf.Min(chart.chartWidth, chart.chartHeight);
  241. var minRadius = serie.minRadius > 0 ? ChartHelper.GetActualValue(serie.minRadius, minChartWidth) : 0;
  242. if (serieData.radius > 0)
  243. {
  244. serieData.context.outsideRadius = ChartHelper.GetActualValue(serieData.radius, minChartWidth);
  245. }
  246. else
  247. {
  248. var minInsideRadius = minRadius > 0 ? minRadius : serie.context.insideRadius;
  249. serieData.context.outsideRadius = serie.pieRoseType > 0 ?
  250. minInsideRadius + (float)((serie.context.outsideRadius - minInsideRadius) * value / serie.context.dataMax) :
  251. serie.context.outsideRadius;
  252. }
  253. if (minRadius > 0 && serieData.context.outsideRadius < minRadius)
  254. {
  255. serieData.context.outsideRadius = minRadius;
  256. }
  257. var offset = 0f;
  258. var interactOffset = serie.animation.interaction.GetOffset(serie.context.outsideRadius);
  259. if (serie.pieClickOffset && (serieData.selected || serieData.context.selected))
  260. {
  261. offset += interactOffset;
  262. }
  263. if (offset > 0)
  264. {
  265. serieData.context.outsideRadius += interactOffset;
  266. var currRad = serieData.context.halfAngle * Mathf.Deg2Rad;
  267. var currSin = Mathf.Sin(currRad);
  268. var currCos = Mathf.Cos(currRad);
  269. serieData.context.offsetRadius = 0;
  270. if (serie.pieClickOffset && (serieData.selected || serieData.context.selected))
  271. {
  272. serieData.context.offsetRadius += interactOffset;
  273. if (serieData.context.insideRadius > 0)
  274. {
  275. serieData.context.insideRadius += interactOffset;
  276. }
  277. }
  278. serieData.context.offsetCenter = new Vector3(
  279. serie.context.center.x + serieData.context.offsetRadius * currSin,
  280. serie.context.center.y + serieData.context.offsetRadius * currCos);
  281. }
  282. else
  283. {
  284. serieData.context.offsetCenter = serie.context.center;
  285. }
  286. if (serieData.context.highlight)
  287. {
  288. serieData.context.outsideRadius = serie.animation.GetInteractionRadius(serieData.context.outsideRadius);
  289. }
  290. var halfRadius = serie.context.insideRadius + (serieData.context.outsideRadius - serie.context.insideRadius) / 2;
  291. serieData.context.position = ChartHelper.GetPosition(serie.context.center, serieData.context.halfAngle, halfRadius);
  292. }
  293. private double GetTotalAngle(Serie serie, double dataTotal, ref float totalAngle)
  294. {
  295. totalAngle = serie.context.startAngle + 360f;
  296. if (serie.minAngle > 0)
  297. {
  298. var rate = serie.minAngle / 360;
  299. var minAngleValue = dataTotal * rate;
  300. foreach (var serieData in serie.data)
  301. {
  302. var value = serieData.GetData(1);
  303. if (value < minAngleValue)
  304. {
  305. totalAngle -= serie.minAngle;
  306. dataTotal -= value;
  307. }
  308. }
  309. return dataTotal;
  310. }
  311. else
  312. {
  313. return dataTotal;
  314. }
  315. }
  316. private void DrawPieCenter(VertexHelper vh, Serie serie, ItemStyle itemStyle, float insideRadius)
  317. {
  318. if (!ChartHelper.IsClearColor(itemStyle.centerColor))
  319. {
  320. var radius = insideRadius - itemStyle.centerGap;
  321. UGL.DrawCricle(vh, serie.context.center, radius, itemStyle.centerColor, chart.settings.cicleSmoothness);
  322. }
  323. }
  324. private void DrawPie(VertexHelper vh, Pie serie)
  325. {
  326. if (!serie.show || serie.animation.HasFadeOut())
  327. {
  328. return;
  329. }
  330. var dataChanging = false;
  331. var interacting = false;
  332. var color = ColorUtil.clearColor32;
  333. var toColor = ColorUtil.clearColor32;
  334. var interactDuration = serie.animation.GetInteractionDuration();
  335. var interactEnable = serie.animation.enable && serie.animation.interaction.enable
  336. && !serie.animation.IsFadeIn() && !serie.animation.IsFadeOut();
  337. var data = serie.data;
  338. serie.animation.InitProgress(0, 360);
  339. if (data.Count == 0)
  340. {
  341. var itemStyle = SerieHelper.GetItemStyle(serie, null);
  342. var fillColor = ChartHelper.IsClearColor(itemStyle.backgroundColor) ?
  343. (Color32)chart.theme.legend.unableColor : itemStyle.backgroundColor;
  344. UGL.DrawDoughnut(vh, serie.context.center, serie.context.insideRadius,
  345. serie.context.outsideRadius, fillColor, fillColor, Color.clear, 0,
  346. 360, itemStyle.borderWidth, itemStyle.borderColor, serie.gap / 2, chart.settings.cicleSmoothness,
  347. false, true, serie.radiusGradient);
  348. }
  349. for (int n = 0; n < data.Count; n++)
  350. {
  351. var serieData = data[n];
  352. if (!serieData.show)
  353. {
  354. continue;
  355. }
  356. if (serieData.IsDataChanged())
  357. dataChanging = true;
  358. var itemStyle = SerieHelper.GetItemStyle(serie, serieData);
  359. var colorIndex = chart.GetLegendRealShowNameIndex(serieData.legendName);
  360. var outsideRadius = 0f;
  361. var needOffset = (serie.pieClickOffset && (serieData.selected || serieData.context.selected));
  362. var offsetCenter = needOffset ? serieData.context.offsetCenter : serie.context.center;
  363. var borderWidth = itemStyle.borderWidth;
  364. var borderColor = itemStyle.borderColor;
  365. var progress = AnimationStyleHelper.CheckDataAnimation(chart, serie, n, 1);
  366. var insideRadius = serieData.context.insideRadius * progress;
  367. if (!interactEnable || !serieData.interact.TryGetValueAndColor(
  368. ref outsideRadius, ref offsetCenter, ref color, ref toColor, ref interacting, interactDuration))
  369. {
  370. SerieHelper.GetItemColor(out color, out toColor, serie, serieData, chart.theme, colorIndex);
  371. outsideRadius = serieData.context.outsideRadius * progress;
  372. if (interactEnable)
  373. {
  374. serieData.interact.SetValueAndColor(ref interacting, outsideRadius, color, toColor);
  375. serieData.interact.SetPosition(ref interacting, offsetCenter);
  376. }
  377. }
  378. var drawEndDegree = serieData.context.currentAngle;
  379. var needRoundCap = serie.roundCap && insideRadius > 0;
  380. UGL.DrawDoughnut(vh, offsetCenter, insideRadius,
  381. outsideRadius, color, toColor, Color.clear, serieData.context.startAngle,
  382. drawEndDegree, borderWidth, borderColor, serie.gap / 2, chart.settings.cicleSmoothness,
  383. needRoundCap, true, serie.radiusGradient);
  384. DrawPieCenter(vh, serie, itemStyle, insideRadius);
  385. if (serie.animation.CheckDetailBreak(serieData.context.toAngle))
  386. break;
  387. }
  388. if (!serie.animation.IsFinish())
  389. {
  390. serie.animation.CheckProgress();
  391. serie.animation.CheckSymbol(serie.symbol.GetSize(null, chart.theme.serie.lineSymbolSize));
  392. chart.RefreshPainter(serie);
  393. }
  394. if (dataChanging || interacting)
  395. {
  396. chart.RefreshPainter(serie);
  397. }
  398. }
  399. private static void UpdatePieLabelPosition(Serie serie, SerieData serieData)
  400. {
  401. if (serieData.labelObject == null) return;
  402. var startAngle = serie.context.startAngle;
  403. var currAngle = serieData.context.halfAngle;
  404. var currRad = currAngle * Mathf.Deg2Rad;
  405. var offsetRadius = serieData.context.offsetRadius;
  406. var insideRadius = serieData.context.insideRadius;
  407. var outsideRadius = serieData.context.outsideRadius;
  408. var serieLabel = SerieHelper.GetSerieLabel(serie, serieData);
  409. var labelLine = SerieHelper.GetSerieLabelLine(serie, serieData);
  410. var center = serieData.context.offsetCenter;
  411. var interact = false;
  412. serieData.interact.TryGetValueAndColor(ref outsideRadius, ref center, ref interact, serie.animation.GetInteractionDuration());
  413. var diffAngle = (currAngle - startAngle) % 360;
  414. var isLeft = diffAngle > 180 || (diffAngle == 0 && serieData.context.startAngle > 0);
  415. switch (serieLabel.position)
  416. {
  417. case LabelStyle.Position.Center:
  418. serieData.context.labelPosition = serie.context.center;
  419. break;
  420. case LabelStyle.Position.Inside:
  421. case LabelStyle.Position.Middle:
  422. var labelRadius = offsetRadius + insideRadius + (outsideRadius - insideRadius) / 2 + serieLabel.distance;
  423. var labelCenter = new Vector2(center.x + labelRadius * Mathf.Sin(currRad),
  424. center.y + labelRadius * Mathf.Cos(currRad));
  425. UpdateLabelPosition(serie, serieData, labelLine, labelCenter, isLeft);
  426. break;
  427. default:
  428. //LabelStyle.Position.Outside
  429. var startPos = new Vector2(center.x + outsideRadius * Mathf.Sin(currRad),
  430. center.y + outsideRadius * Mathf.Cos(currRad));
  431. UpdateLabelPosition(serie, serieData, labelLine, startPos, isLeft);
  432. break;
  433. }
  434. }
  435. private static void UpdateLabelPosition(Serie serie, SerieData serieData, LabelLine labelLine, Vector3 startPosition, bool isLeft)
  436. {
  437. serieData.context.labelLinePosition = startPosition;
  438. if (labelLine == null || !labelLine.show)
  439. {
  440. serieData.context.labelPosition = startPosition;
  441. return;
  442. }
  443. var dire = isLeft ? Vector3.left : Vector3.right;
  444. var rad = Mathf.Deg2Rad * serieData.context.halfAngle;
  445. var lineLength1 = ChartHelper.GetActualValue(labelLine.lineLength1, serie.context.outsideRadius);
  446. var lineLength2 = ChartHelper.GetActualValue(labelLine.lineLength2, serie.context.outsideRadius);
  447. var radius = lineLength1 + serie.context.outsideRadius - serieData.context.outsideRadius;
  448. var pos1 = startPosition;
  449. var pos2 = pos1 + new Vector3(Mathf.Sin(rad) * radius, Mathf.Cos(rad) * radius);
  450. var pos5 = labelLine.lineType == LabelLine.LineType.HorizontalLine
  451. ? pos1 + dire * (radius + lineLength2) + labelLine.GetEndSymbolOffset()
  452. : pos2 + dire * lineLength2 + labelLine.GetEndSymbolOffset();
  453. if (labelLine.lineEndX != 0)
  454. {
  455. pos5.x = serie.context.center.x + (isLeft ? -Mathf.Abs(labelLine.lineEndX) : Mathf.Abs(labelLine.lineEndX));
  456. }
  457. serieData.context.labelLinePosition2 = pos2;
  458. serieData.context.labelPosition = pos5;
  459. }
  460. private void DrawPieLabelLine(VertexHelper vh, Serie serie, bool isTop)
  461. {
  462. foreach (var serieData in serie.data)
  463. {
  464. var serieLabel = SerieHelper.GetSerieLabel(serie, serieData);
  465. var labelLine = SerieHelper.GetSerieLabelLine(serie, serieData);
  466. if (SerieLabelHelper.CanShowLabel(serie, serieData, serieLabel, 1))
  467. {
  468. int colorIndex = chart.m_LegendRealShowName.IndexOf(serieData.name);
  469. if (serieLabel != null && serieLabel.show &&
  470. labelLine != null && labelLine.show)
  471. {
  472. if (serieLabel.position == LabelStyle.Position.Inside || serieLabel.position == LabelStyle.Position.Middle)
  473. {
  474. if (!isTop) continue;
  475. }
  476. else
  477. {
  478. if (isTop && !labelLine.startSymbol.show) continue;
  479. }
  480. var color = ChartHelper.IsClearColor(labelLine.lineColor) ?
  481. chart.theme.GetColor(colorIndex) :
  482. labelLine.lineColor;
  483. switch (labelLine.lineType)
  484. {
  485. case LabelLine.LineType.BrokenLine:
  486. UGL.DrawLine(vh, serieData.context.labelLinePosition, serieData.context.labelLinePosition2,
  487. serieData.context.labelPosition, labelLine.lineWidth, color);
  488. break;
  489. case LabelLine.LineType.Curves:
  490. if (serieData.context.labelLinePosition2 == serieData.context.labelPosition)
  491. {
  492. UGL.DrawCurves(vh, serieData.context.labelLinePosition, serieData.context.labelPosition,
  493. serieData.context.labelLinePosition, (serieData.context.labelLinePosition + serieData.context.labelPosition) * 0.6f,
  494. labelLine.lineWidth, color, chart.settings.lineSmoothness);
  495. }
  496. else
  497. {
  498. UGL.DrawCurves(vh, serieData.context.labelLinePosition, serieData.context.labelPosition,
  499. serieData.context.labelLinePosition, serieData.context.labelLinePosition2,
  500. labelLine.lineWidth, color, chart.settings.lineSmoothness);
  501. }
  502. break;
  503. case LabelLine.LineType.HorizontalLine:
  504. UGL.DrawLine(vh, serieData.context.labelLinePosition, serieData.context.labelPosition,
  505. labelLine.lineWidth, color);
  506. break;
  507. }
  508. DrawLabelLineSymbol(vh, labelLine, serieData.context.labelLinePosition, serieData.context.labelPosition, color);
  509. }
  510. }
  511. }
  512. }
  513. private int GetPiePosIndex(Serie serie, Vector2 local)
  514. {
  515. if (!(serie is Pie))
  516. return -1;
  517. var dist = Vector2.Distance(local, serie.context.center);
  518. var interactOffset = serie.animation.interaction.GetOffset(serie.context.outsideRadius);
  519. var maxRadius = serie.context.outsideRadius + 2 * interactOffset;
  520. if (dist < serie.context.insideRadius || dist > maxRadius)
  521. return -1;
  522. var dir = local - new Vector2(serie.context.center.x, serie.context.center.y);
  523. var angle = ChartHelper.GetAngle360(Vector2.up, dir);
  524. for (int i = 0; i < serie.data.Count; i++)
  525. {
  526. var serieData = serie.data[i];
  527. if (angle >= serieData.context.startAngle && angle <= serieData.context.toAngle)
  528. {
  529. var ndist = (serieData.selected || serieData.context.selected) ?
  530. Vector2.Distance(local, serieData.context.offsetCenter) :
  531. dist;
  532. if (ndist >= serieData.context.insideRadius && ndist <= serieData.context.outsideRadius)
  533. {
  534. return i;
  535. }
  536. }
  537. }
  538. return -1;
  539. }
  540. private bool PointerIsInPieSerie(Serie serie, Vector2 local)
  541. {
  542. if (!(serie is Pie))
  543. return false;
  544. var dist = Vector2.Distance(local, serie.context.center);
  545. if (dist >= serie.context.insideRadius && dist <= serie.context.outsideRadius)
  546. return true;
  547. return false;
  548. }
  549. private float GetStartAngle(Serie serie)
  550. {
  551. return serie.clockwise ? (serie.startAngle + 360) % 360 : 360 - serie.startAngle;
  552. }
  553. private float GetToAngle(Serie serie, float angle)
  554. {
  555. var toAngle = angle + serie.startAngle;
  556. if (!serie.clockwise)
  557. {
  558. toAngle = 360 - angle - serie.startAngle;
  559. }
  560. if (!serie.animation.IsFinish())
  561. {
  562. var currAngle = serie.animation.GetCurrDetail();
  563. if (serie.clockwise)
  564. {
  565. toAngle = toAngle > currAngle ? currAngle : toAngle;
  566. }
  567. else
  568. {
  569. toAngle = toAngle < 360 - currAngle ? 360 - currAngle : toAngle;
  570. }
  571. }
  572. return toAngle;
  573. }
  574. private void AvoidLabelOverlap(Serie serie, ComponentTheme theme)
  575. {
  576. if (!serie.avoidLabelOverlap) return;
  577. var lastCheckPos = Vector3.zero;
  578. var lastX = 0f;
  579. var data = serie.data;
  580. var splitCount = 0;
  581. for (int n = 0; n < data.Count; n++)
  582. {
  583. var serieData = data[n];
  584. if (serieData.context.labelPosition.x != 0 && serieData.context.labelPosition.x < serie.context.center.x)
  585. {
  586. splitCount = n;
  587. break;
  588. }
  589. }
  590. var limitX = float.MinValue;
  591. for (int n = 0; n < splitCount; n++)
  592. {
  593. CheckSerieDataLabel(serie, data[n], splitCount, false, n == splitCount - 1, theme, ref lastCheckPos, ref lastX, ref limitX);
  594. }
  595. lastCheckPos = Vector3.zero;
  596. limitX = float.MaxValue;
  597. for (int n = data.Count - 1; n >= splitCount; n--)
  598. {
  599. CheckSerieDataLabel(serie, data[n], data.Count - splitCount, true, n == splitCount, theme, ref lastCheckPos, ref lastX, ref limitX);
  600. }
  601. }
  602. private void CheckSerieDataLabel(Serie serie, SerieData serieData, int total, bool isLeft, bool isLastOne, ComponentTheme theme,
  603. ref Vector3 lastCheckPos, ref float lastX, ref float limitX)
  604. {
  605. if (!serieData.context.canShowLabel)
  606. {
  607. serieData.SetLabelActive(false);
  608. return;
  609. }
  610. if (!serieData.show) return;
  611. var serieLabel = SerieHelper.GetSerieLabel(serie, serieData);
  612. if (serieLabel == null) return;
  613. if (!serieLabel.show) return;
  614. var labelLine = SerieHelper.GetSerieLabelLine(serie, serieData);
  615. var fontSize = serieData.labelObject.GetHeight();
  616. var lineLength1 = 0f;
  617. var lineLength2 = 0f;
  618. if (labelLine != null && labelLine.show)
  619. {
  620. lineLength1 = ChartHelper.GetActualValue(labelLine.lineLength1, serie.context.outsideRadius);
  621. lineLength2 = ChartHelper.GetActualValue(labelLine.lineLength2, serie.context.outsideRadius);
  622. }
  623. if (lastCheckPos == Vector3.zero)
  624. {
  625. lastCheckPos = serieData.context.labelPosition;
  626. }
  627. else if (serieData.context.labelPosition.x != 0)
  628. {
  629. if (lastCheckPos.y - serieData.context.labelPosition.y < fontSize)
  630. {
  631. var labelRadius = serie.context.outsideRadius + lineLength1;
  632. var y1 = lastCheckPos.y - fontSize;
  633. var cy = serie.context.center.y;
  634. var diff = Mathf.Abs(y1 - cy);
  635. var diffX = labelRadius * labelRadius - diff * diff;
  636. diffX = diffX <= 0 ? 0 : diffX;
  637. var x1 = serie.context.center.x + Mathf.Sqrt(diffX) * (isLeft ? -1 : 1);
  638. var newPos = new Vector3(x1, y1);
  639. serieData.context.labelLinePosition2 = newPos;
  640. if (isLeft)
  641. {
  642. if (x1 < limitX)
  643. {
  644. limitX = x1;
  645. serieData.context.labelPosition = new Vector3(newPos.x - lineLength2, newPos.y);
  646. lastX = serieData.context.labelPosition.x;
  647. }
  648. else
  649. {
  650. serieData.context.labelPosition = new Vector3(lastX, y1);
  651. lastX += 2;
  652. }
  653. }
  654. else
  655. {
  656. if (x1 > limitX)
  657. {
  658. limitX = x1;
  659. serieData.context.labelPosition = new Vector3(newPos.x + lineLength2, newPos.y);
  660. lastX = serieData.context.labelPosition.x;
  661. }
  662. else
  663. {
  664. serieData.context.labelPosition = new Vector3(lastX, y1);
  665. lastX -= 2;
  666. }
  667. }
  668. if (labelLine != null && labelLine.show && labelLine.lineEndX != 0)
  669. {
  670. serieData.context.labelPosition.x = isLeft ? -Mathf.Abs(labelLine.lineEndX) : Mathf.Abs(labelLine.lineEndX);
  671. }
  672. if (!isLastOne && serieData.context.labelPosition.y < serieData.context.labelLinePosition.y)
  673. {
  674. serieData.context.labelLinePosition2 = serieData.context.labelPosition;
  675. }
  676. else
  677. {
  678. if (isLeft && serieData.context.labelLinePosition2.x > serieData.context.labelLinePosition.x)
  679. {
  680. serieData.context.labelLinePosition2.x = serieData.context.labelLinePosition.x;
  681. }
  682. else if (!isLeft && serieData.context.labelLinePosition2.x < serieData.context.labelLinePosition.x)
  683. {
  684. serieData.context.labelLinePosition2.x = serieData.context.labelLinePosition.x;
  685. }
  686. }
  687. }
  688. else
  689. {
  690. lastX = serieData.context.labelPosition.x;
  691. }
  692. lastCheckPos = serieData.context.labelPosition;
  693. UpdateLabelPosition(serieData, serieLabel);
  694. }
  695. }
  696. }
  697. }