RecordLineChart.xaml.cs 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. using Microsoft.Win32;
  2. using ScottPlot;
  3. using ScottPlot.Plottables;
  4. using ScottPlot.WPF;
  5. using SWRIS.Dtos;
  6. using System;
  7. using System.Collections.Generic;
  8. using System.Linq;
  9. using System.Windows;
  10. using System.Windows.Controls;
  11. namespace SWRIS.Controls
  12. {
  13. /// <summary>
  14. /// 动态折线图用户控件(基于 ScottPlot.WPF 的高性能版本)
  15. /// </summary>
  16. public partial class RecordLineChart : UserControl
  17. {
  18. private readonly Color[] _colors = new Color[] {
  19. new Color("#70a1d7"), new Color("#a1de93"),
  20. new Color("#f7f48b"), new Color("#f47c7c"),
  21. new Color("#c264fe"), new Color("#00fff5"),
  22. new Color("#f8b195"), new Color("#7a08fa") };
  23. private const double MAXDAMAGE = 128; // 最大损伤值
  24. private SignalXY[] _sensorSignals;
  25. private SignalXY _mergeSignal;
  26. private Scatter[] _makerScatters;
  27. private AxisLimits _axisLimit;
  28. public RecordLineChart()
  29. {
  30. InitializeComponent();
  31. InitializePlot();
  32. }
  33. private void InitializePlot()
  34. {
  35. WpfPlot.Menu?.Clear();
  36. WpfPlot.Menu.Add("保存图片", pt =>
  37. {
  38. OpenSaveImageDialog(pt);
  39. });
  40. WpfPlot.Menu.Add("重置视图", pt =>
  41. {
  42. WpfPlot.Plot.Axes.SetLimits(_axisLimit);
  43. WpfPlot.Refresh();
  44. });
  45. var plot = WpfPlot.Plot;
  46. plot.Clear();
  47. plot.DataBackground.Color = Colors.Transparent;
  48. plot.FigureBackground.Color = Colors.Transparent;
  49. plot.Grid.MajorLineColor = new Color(59, 59, 123).WithOpacity(0.6);
  50. plot.Grid.XAxis.TickLabelStyle.ForeColor = new Color(151, 166, 212);
  51. plot.Grid.YAxis.TickLabelStyle.ForeColor = new Color(151, 166, 212);
  52. plot.Grid.XAxis.FrameLineStyle.Color = new Color(53, 53, 112);
  53. plot.Grid.XAxis.FrameLineStyle.Width = 3;
  54. plot.Grid.YAxis.FrameLineStyle.Color = new Color(53, 53, 112);
  55. plot.Grid.YAxis.FrameLineStyle.Width = 3;
  56. plot.Grid.XAxis.MajorTickStyle.Color = new Color(53, 53, 112);
  57. plot.Grid.XAxis.MajorTickStyle.Width = 1;
  58. plot.Grid.YAxis.MajorTickStyle.Color = new Color(53, 53, 112);
  59. plot.Grid.YAxis.MajorTickStyle.Width = 1;
  60. plot.Grid.YAxis.MinorTickStyle.Color = new Color(53, 53, 112);
  61. plot.Grid.XAxis.MinorTickStyle.Color = new Color(53, 53, 112);
  62. plot.Axes.Hairline(false);
  63. }
  64. public void AddDataPoints((double[] Positions, ushort[,] Damages) dataPoints, List<DamageDto> alertDamages)
  65. {
  66. if (dataPoints.Damages == null || dataPoints.Positions == null)
  67. {
  68. return;
  69. }
  70. int sensorCount = dataPoints.Damages.GetLength(0);
  71. int sampleCount = dataPoints.Damages.GetLength(1);
  72. double startPoint = dataPoints.Positions.First();
  73. double endPoint = dataPoints.Positions.Last();
  74. _sensorSignals = new SignalXY[sensorCount];
  75. double[] mergeData = new double[sampleCount];
  76. // 先计算所有传感器的合并数据
  77. for (int j = 0; j < sampleCount; j++)
  78. {
  79. double sum = 0;
  80. for (int i = 0; i < sensorCount; i++)
  81. {
  82. sum += dataPoints.Damages[i, j];
  83. }
  84. mergeData[j] = sum / sensorCount; // 计算平均值
  85. }
  86. AddMaker(dataPoints.Positions, mergeData, alertDamages);
  87. // 然后处理每个传感器的数据
  88. for (int i = 0; i < sensorCount; i++)
  89. {
  90. ushort[] sensorData = new ushort[sampleCount];
  91. for (int j = 0; j < sampleCount; j++)
  92. {
  93. sensorData[j] = dataPoints.Damages[i, j];
  94. }
  95. var signalPlot = WpfPlot.Plot.Add.SignalXY(dataPoints.Positions, sensorData);
  96. signalPlot.LineWidth = 1;
  97. signalPlot.Color = _colors[i];
  98. signalPlot.IsVisible = false;
  99. _sensorSignals[i] = signalPlot;
  100. }
  101. _mergeSignal = WpfPlot.Plot.Add.SignalXY(dataPoints.Positions, mergeData);
  102. _mergeSignal.LineWidth = 2;
  103. _mergeSignal.Color = new Color("#9664F3");
  104. //差值
  105. double offset = (endPoint - startPoint) * 0.1;
  106. _axisLimit = new AxisLimits(startPoint - offset, endPoint + offset, 0, MAXDAMAGE);
  107. WpfPlot.Plot.Axes.SetLimits(_axisLimit);
  108. WpfPlot.Refresh();
  109. }
  110. public void SignalPlotVisible(int id, bool isVisible)
  111. {
  112. if (id > _sensorSignals.Length) return;
  113. if (id == 0)
  114. {
  115. _mergeSignal.IsVisible = isVisible;
  116. foreach (var maker in _makerScatters)
  117. {
  118. maker.IsVisible = isVisible;
  119. }
  120. }
  121. else
  122. {
  123. _sensorSignals[id - 1].IsVisible = isVisible;
  124. }
  125. WpfPlot.Refresh();
  126. }
  127. private void AddMaker(double[] dataPoints, double[] mergeData, List<DamageDto> alertDamages)
  128. {
  129. int markerCount = alertDamages.Count;
  130. _makerScatters = new Scatter[markerCount];
  131. for (int i = 0; i < markerCount; i++)
  132. {
  133. int index = FindNearestIndex(dataPoints, alertDamages[i].DamagePoint);
  134. if (index < 0 || index >= dataPoints.Length)
  135. continue;
  136. var marker = WpfPlot.Plot.Add.Scatter(dataPoints[index], mergeData[index]);
  137. marker.LineWidth = 0; // 不显示连线
  138. marker.MarkerSize = 10;
  139. marker.MarkerShape = MarkerShape.OpenCircle;
  140. marker.MarkerLineWidth = 2;
  141. switch (alertDamages[i].DamageLevel)
  142. {
  143. case Enums.DamageLevel.Mild:
  144. marker.Color = Color.FromHex("#00FF78");
  145. break;
  146. case Enums.DamageLevel.Light:
  147. marker.Color = Color.FromHex("#FFF000");
  148. break;
  149. case Enums.DamageLevel.Moderate:
  150. marker.Color = Color.FromHex("#FF9400");
  151. break;
  152. case Enums.DamageLevel.Severe:
  153. marker.Color = Color.FromHex("#FF6C00");
  154. break;
  155. case Enums.DamageLevel.Critical:
  156. marker.Color = Color.FromHex("#FF0000");
  157. break;
  158. case Enums.DamageLevel.ExceededLimit:
  159. marker.Color = Color.FromHex("#FF008A");
  160. break;
  161. }
  162. _makerScatters[i] = marker;
  163. }
  164. }
  165. private int FindNearestIndex(double[] positions, double target)
  166. {
  167. if (positions == null || positions.Length == 0)
  168. return -1;
  169. int index = Array.BinarySearch(positions, target);
  170. // 如果找到精确匹配
  171. if (index >= 0)
  172. return index;
  173. // 如果没找到,BinarySearch 返回下一个更大元素的补码
  174. int largerIndex = ~index;
  175. // 处理边界情况
  176. if (largerIndex == 0)
  177. return 0;
  178. if (largerIndex == positions.Length)
  179. return positions.Length - 1;
  180. // 比较左右两个元素哪个更接近
  181. double diffLeft = Math.Abs(positions[largerIndex - 1] - target);
  182. double diffRight = Math.Abs(positions[largerIndex] - target);
  183. return diffLeft < diffRight ? largerIndex - 1 : largerIndex;
  184. }
  185. public void OpenSaveImageDialog(Plot plot)
  186. {
  187. SaveFileDialog saveFileDialog = new SaveFileDialog
  188. {
  189. Filter = "PNG Files (*.png)|*.png|JPEG Files (*.jpg, *.jpeg)|*.jpg;*.jpeg|BMP Files (*.bmp)|*.bmp|WebP Files (*.webp)|*.webp|SVG Files (*.svg)|*.svg|All files (*.*)|*.*"
  190. };
  191. bool? flag = saveFileDialog.ShowDialog();
  192. if (flag.HasValue && flag == true && !string.IsNullOrEmpty(saveFileDialog.FileName))
  193. {
  194. ImageFormat format;
  195. try
  196. {
  197. format = ImageFormats.FromFilename(saveFileDialog.FileName);
  198. }
  199. catch (ArgumentException)
  200. {
  201. MessageBox.Show("Unsupported image file format", "ERROR", MessageBoxButton.OK, MessageBoxImage.Hand);
  202. return;
  203. }
  204. try
  205. {
  206. PixelSize size = plot.RenderManager.LastRender.FigureRect.Size;
  207. plot.Save(saveFileDialog.FileName, (int)size.Width, (int)size.Height, format);
  208. }
  209. catch (Exception)
  210. {
  211. MessageBox.Show("Image save failed", "ERROR", MessageBoxButton.OK, MessageBoxImage.Hand);
  212. }
  213. }
  214. }
  215. }
  216. }