SweepConfigViewModel.cs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562
  1. using Avalonia.Collections;
  2. using Avalonia.Controls;
  3. using Avalonia.Utilities;
  4. using CommunityToolkit.Mvvm.Input;
  5. using OxyPlot;
  6. using OxyPlot.Series;
  7. using Shaker.Models;
  8. using Shaker.Models.Tools;
  9. using System;
  10. using System.Collections.Generic;
  11. using System.Linq;
  12. using System.Linq.Expressions;
  13. using System.Text;
  14. using System.Threading.Tasks;
  15. using System.Windows.Input;
  16. namespace ShakerApp.ViewModels
  17. {
  18. internal class SweepConfigViewModel:ViewModelBase<Shaker.Models.SweepConfigModel>
  19. {
  20. private float interval = 0.1f;
  21. List<OxyPlot.Series.LineSeries> lineSeries = new List<OxyPlot.Series.LineSeries>();
  22. List<DataPoint> velocitydata = new List<DataPoint>();
  23. LineSeries velocity = new LineSeries();
  24. LineSeries displacement = new LineSeries();
  25. List<DataPoint> displacementdata = new List<DataPoint>();
  26. List<OxyColor> oxyColors = new List<OxyColor>() { OxyColors.Green, OxyColors.Red, OxyColors.Red, OxyColors.Yellow, OxyColors.Yellow };
  27. List<LineStyle> lineStyles = new List<LineStyle>() { LineStyle.Solid, LineStyle.Solid, LineStyle.Solid, LineStyle.LongDashDotDot, LineStyle.LongDashDotDot };
  28. List<string> properties = new List<string>() { nameof(SweepData.TargetAcceleration), nameof(SweepData.UpStopAcceleration), nameof(SweepData.DownStopAcceleration), nameof(SweepData.UpWarnAcceleration), nameof(SweepData.DownWarnAcceleration) };
  29. List<SweepData> datas = new List<SweepData>();
  30. private SweepConfigViewModel()
  31. {
  32. OnceSweepTime = Model.OCTToTime(Model.SweepSpeed);
  33. Content = typeof(Views.SweepConfigView);
  34. for(int i=0;i<ShakerConfigViewModel.Instance.AccelerationSensorCount;i++)
  35. {
  36. Channels.Add(new KeyValuePair<int, int>(i+1,i));
  37. }
  38. #region 加速度谱曲线
  39. AccelerationModel.Axes.Add(new OxyPlot.Axes.LogarithmicAxis()
  40. {
  41. MaximumPadding = 0,
  42. MinimumPadding = 0,
  43. Title = App.Current?.FindResource(Shaker.Models.ShakerConstant.FrequencyKey) + "",
  44. Unit = "Hz",
  45. MajorGridlineStyle = LineStyle.Solid,
  46. Position = OxyPlot.Axes.AxisPosition.Bottom,
  47. Key = "Freq"
  48. });
  49. AccelerationModel.Axes.Add(new OxyPlot.Axes.LogarithmicAxis()
  50. {
  51. Key= "Ampt",
  52. Title = App.Current?.FindResource(ShakerConstant.AmptAxisTitleKey)+"",
  53. Unit="g",
  54. MajorGridlineStyle = LineStyle.Solid,
  55. Position = OxyPlot.Axes.AxisPosition.Left,
  56. });
  57. AccelerationModel.Title = App.Current?.FindResource(ShakerConstant.AccelerationSpectrumKey)+ "";
  58. for(int i=0;i<oxyColors.Count;i++)
  59. {
  60. LineSeries line = new LineSeries();
  61. line.Title = App.Current?.FindResource(ShakerConstant.SweepAccelerationSpectrumNames[i])+"";
  62. line.Color = oxyColors[i];
  63. line.StrokeThickness = 1;
  64. line.DataFieldX = nameof(SweepData.Frequency);
  65. line.DataFieldY = properties[i];
  66. line.LineStyle = lineStyles[i];
  67. line.XAxisKey = "Freq";
  68. line.YAxisKey = "Ampt";
  69. lineSeries.Add(line);
  70. AccelerationModel.Series.Add(line);
  71. }
  72. AccelerationModel.Legends.Add(new OxyPlot.Legends.Legend()
  73. {
  74. ShowInvisibleSeries = true,
  75. });
  76. #endregion
  77. #region 速度
  78. VelocityModel.Axes.Add(new OxyPlot.Axes.LogarithmicAxis()
  79. {
  80. MaximumPadding = 0,
  81. MinimumPadding = 0,
  82. Title = App.Current?.FindResource(Shaker.Models.ShakerConstant.FrequencyKey) + "",
  83. Unit = "Hz",
  84. MajorGridlineStyle = LineStyle.Solid,
  85. Position = OxyPlot.Axes.AxisPosition.Bottom,
  86. Key = "Freq"
  87. });
  88. VelocityModel.Title = App.Current?.FindResource(ShakerConstant.VelocityKey) + "";
  89. VelocityModel.Axes.Add(new OxyPlot.Axes.LogarithmicAxis()
  90. {
  91. Key = "Ampt",
  92. Title = App.Current?.FindResource(ShakerConstant.AmptAxisTitleKey) + "",
  93. Unit = "m/s",
  94. MajorGridlineStyle = LineStyle.Solid,
  95. Position = OxyPlot.Axes.AxisPosition.Left,
  96. });
  97. velocity = new LineSeries();
  98. velocity.Title = App.Current?.FindResource(ShakerConstant.VelocityKey) + "";
  99. velocity.Color = oxyColors[0];
  100. velocity.StrokeThickness = 1;
  101. velocity.DataFieldX = nameof(DataPoint.X);
  102. velocity.DataFieldY = nameof(DataPoint.Y);
  103. velocity.LineStyle = lineStyles[0];
  104. velocity.XAxisKey = "Freq";
  105. velocity.YAxisKey = "Ampt";
  106. VelocityModel.Series.Add(velocity);
  107. #endregion
  108. #region 位移
  109. DisplacementModel.Axes.Add(new OxyPlot.Axes.LogarithmicAxis()
  110. {
  111. MaximumPadding = 0,
  112. MinimumPadding = 0,
  113. Title = App.Current?.FindResource(Shaker.Models.ShakerConstant.FrequencyKey) + "",
  114. Unit = "Hz",
  115. MajorGridlineStyle = LineStyle.Solid,
  116. Position = OxyPlot.Axes.AxisPosition.Bottom,
  117. Key = "Freq"
  118. });
  119. DisplacementModel.Title = App.Current?.FindResource(ShakerConstant.DisplacementKey) + "";
  120. DisplacementModel.Axes.Add(new OxyPlot.Axes.LogarithmicAxis()
  121. {
  122. Key = "Ampt",
  123. Title = App.Current?.FindResource(ShakerConstant.AmptAxisTitleKey) + "",
  124. Unit = "mm",
  125. MajorGridlineStyle = LineStyle.Solid,
  126. Position = OxyPlot.Axes.AxisPosition.Left,
  127. });
  128. displacement = new LineSeries();
  129. displacement.Title = App.Current?.FindResource(ShakerConstant.DisplacementKey) + "";
  130. displacement.Color = oxyColors[0];
  131. displacement.StrokeThickness = 1;
  132. displacement.DataFieldX = nameof(DataPoint.X);
  133. displacement.DataFieldY = nameof(DataPoint.Y);
  134. displacement.LineStyle = lineStyles[0];
  135. displacement.XAxisKey = "Freq";
  136. displacement.YAxisKey = "Ampt";
  137. DisplacementModel.Series.Add(displacement);
  138. #endregion
  139. GetEvent(ShakerSettingViewModel.LANGUAGECHANGEDEVENT).Subscrip((_, _) =>
  140. {
  141. AccelerationModel.InvalidatePlot(false);
  142. AccelerationModel.Axes[0].Title = App.Current?.FindResource(ShakerConstant.FrequencyKey) + "";
  143. AccelerationModel.Axes[1].Title = App.Current?.FindResource(ShakerConstant.AmptAxisTitleKey) + "";
  144. AccelerationModel.Title = App.Current?.FindResource(ShakerConstant.AccelerationSpectrumKey) + "";
  145. for(int i=0;i<lineSeries.Count;i++)
  146. {
  147. lineSeries[i].Title = App.Current?.FindResource(ShakerConstant.SweepAccelerationSpectrumNames[i]) + "";
  148. }
  149. AccelerationModel.InvalidatePlot(true);
  150. VelocityModel.InvalidatePlot(false);
  151. VelocityModel.Axes[0].Title = App.Current?.FindResource(ShakerConstant.FrequencyKey) + "";
  152. VelocityModel.Axes[1].Title = App.Current?.FindResource(ShakerConstant.AmptAxisTitleKey) + "";
  153. VelocityModel.Title = App.Current?.FindResource(ShakerConstant.VelocityKey) + "";
  154. velocity.Title = App.Current?.FindResource(ShakerConstant.VelocityKey) + "";
  155. VelocityModel.InvalidatePlot(true);
  156. DisplacementModel.InvalidatePlot(false);
  157. DisplacementModel.Axes[0].Title = App.Current?.FindResource(ShakerConstant.FrequencyKey) + "";
  158. DisplacementModel.Axes[1].Title = App.Current?.FindResource(ShakerConstant.AmptAxisTitleKey) + "";
  159. DisplacementModel.Title = App.Current?.FindResource(ShakerConstant.DisplacementKey) + "";
  160. displacement.Title = App.Current?.FindResource(ShakerConstant.DisplacementKey) + "";
  161. DisplacementModel.InvalidatePlot(true);
  162. });
  163. GetEvent<AllConfig>().Subscrip((sender, args) =>
  164. {
  165. UpDateModel(args.Data.SweepConfig);
  166. CommunicationViewModel.Instance.LocalCommunication?.GetEvent<SweepConfigModel>()?.Subscrip((sender, args) =>
  167. {
  168. UpDateModel(args.Data);
  169. Refresh();
  170. SineMainPageViewModel.Instance.SetRefSpectrum(datas);
  171. CommunicationViewModel.Instance.ServiceCommunication?.GetEvent<SweepConfigModel>()?.Publish(this, Model);
  172. });
  173. });
  174. }
  175. static SweepConfigViewModel()
  176. {
  177. }
  178. public AvaloniaList<KeyValuePair<int,int>> Channels { get; } = new AvaloniaList<KeyValuePair<int,int>>();
  179. [PropertyAssociation(nameof(SweepConfigModel.SelectChannel))]
  180. public int SelectChannel
  181. {
  182. get => Model.SelectChannel;
  183. set
  184. {
  185. SetProperty(ref Model.SelectChannel, value);
  186. for(int i=0;i<ShakerConfigViewModel.Instance.AccelerationSensorCount;i++)
  187. {
  188. ShakerConfigViewModel.Instance.Accelerations[i].Value.Weight = value == i ? 1 : 0;
  189. }
  190. }
  191. }
  192. [PropertyAssociation(nameof(SweepConfigModel.MaxSweepItemCount))]
  193. public int MaxSweepItemCount => Model.MaxSweepItemCount;
  194. [PropertyAssociation(nameof(SweepConfigModel.SweepCount))]
  195. public uint SweepCount { get => Model.SweepCount; set => SetProperty(ref Model.SweepCount, value); }
  196. [PropertyAssociation(nameof(SweepConfigModel.SweepType))]
  197. public SweepType SweepType
  198. {
  199. get => Model.SweepType;
  200. set
  201. {
  202. if (Model.SweepType == value) return;
  203. SetProperty(ref Model.SweepType, value);
  204. Model.OnceSweepTime = Model.OCTToTime(Model.SweepSpeed);
  205. OnPropertyChanged(nameof(OnceSweepTime));
  206. }
  207. }
  208. [PropertyAssociation(nameof(SweepConfigModel.SweepDirection))]
  209. public SweepDirection SweepDirection { get => Model.SweepDirection; set => SetProperty(ref Model.SweepDirection, value); }
  210. [PropertyAssociation(nameof(SweepConfigModel.SignalType))]
  211. public SignalType SignalType
  212. {
  213. get => Model.SignalType;
  214. set
  215. {
  216. SetProperty(ref Model.SignalType, value);
  217. if(value == SignalType.Sweep)
  218. {
  219. OnceSweepTime = Model.OCTToTime(Model.SweepSpeed);
  220. }
  221. }
  222. }
  223. [PropertyAssociation(nameof(SweepConfigModel.StartFrequency))]
  224. public double StartFrequency
  225. {
  226. get => Model.StartFrequency;
  227. set
  228. {
  229. if (value == Model.StartFrequency) return;
  230. SetProperty(ref Model.StartFrequency, value);
  231. if (SignalType == SignalType.Fixed) return;
  232. OnceSweepTime = Model.OCTToTime(Model.SweepSpeed);
  233. }
  234. }
  235. public ICommand MaxLoadCommand => new RelayCommand(MaxLoad);
  236. private void MaxLoad()
  237. {
  238. SweepItems.Clear();
  239. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(1,new SweepItemViewModel(new SweepItemModel()
  240. {
  241. Frequency = ShakerConfigViewModel.Instance.MinFrequency,
  242. Value = ShakerConfigViewModel.Instance.MaxDisplacement,
  243. SweepValueType = SweepValueType.FixedDisplacement,
  244. })));
  245. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(2,new SweepItemViewModel( new SweepItemModel()
  246. {
  247. Frequency = ShakerConfigViewModel.Instance.MinFrequency + interval,
  248. Value = ShakerConfigViewModel.Instance.MaxVelocity,
  249. SweepValueType = SweepValueType.FixedVelocity,
  250. })));
  251. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(3,new SweepItemViewModel( new SweepItemModel()
  252. {
  253. Frequency = ShakerConfigViewModel.Instance.MinFrequency + interval * 2,
  254. Value = ShakerConfigViewModel.Instance.MaxAcceleration,
  255. SweepValueType = SweepValueType.FixedAcceleration,
  256. })));
  257. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(4,new SweepItemViewModel( new SweepItemModel()
  258. {
  259. Frequency = ShakerConfigViewModel.Instance.MaxFrequency,
  260. Value = ShakerConfigViewModel.Instance.MaxAcceleration,
  261. SweepValueType = SweepValueType.FixedAcceleration,
  262. })));
  263. Refresh();
  264. }
  265. public double MaxAcceleration => datas.Count ==0 ?0: datas.Max(x => x.TargetAcceleration);
  266. public double MaxDisplacement => displacementdata.Count == 0 ? 0 : displacementdata.Max(x => x.Y);
  267. public double MaxVelocity => velocitydata.Count == 0 ? 0 : velocitydata.Max(x => x.Y);
  268. public bool AccelerationOverLimit => MaxAcceleration >= ShakerConfigViewModel.Instance.MaxAcceleration;
  269. public bool DisplacementOverLimit => MaxDisplacement > ShakerConfigViewModel.Instance.MaxDisplacement;
  270. public bool VelocityOverLimit => MaxVelocity > ShakerConfigViewModel.Instance.MaxVelocity;
  271. public double AccelerationLoad => Math.Round(MaxAcceleration / ShakerConfigViewModel.Instance.MaxAcceleration * 100f, 2);
  272. public double DisplacementLoad => Math.Round((double)MaxDisplacement / ShakerConfigViewModel.Instance.MaxDisplacement * 100f, 2);
  273. public double VelocityLoad => Math.Round((double)MaxVelocity / ShakerConfigViewModel.Instance.MaxVelocity * 100f, 2);
  274. [PropertyAssociation(nameof(SweepConfigModel.EndFrequency))]
  275. public double EndFrequency
  276. {
  277. get => Model.EndFrequency;
  278. set
  279. {
  280. if (value == Model.EndFrequency) return;
  281. SetProperty(ref Model.EndFrequency, value);
  282. if (SignalType == SignalType.Fixed) return;
  283. OnceSweepTime = Model.OCTToTime(Model.SweepSpeed);
  284. }
  285. }
  286. [PropertyAssociation(nameof(SweepConfigModel.OnceSweepTime))]
  287. public double OnceSweepTime
  288. {
  289. get => Model.OnceSweepTime;
  290. set
  291. {
  292. if (Math.Abs(value - Model.OnceSweepTime) <= Increment || double.IsNaN(value)) return;
  293. SetProperty(ref Model.OnceSweepTime, value);
  294. OnPropertyChanged(nameof(TotalSweepTime));
  295. if (SignalType == SignalType.Fixed) return;
  296. SweepSpeed = Model.TimeToOCT(value);
  297. }
  298. }
  299. private bool VerifySweepItem()
  300. {
  301. if (SweepItems.Count < 2)
  302. {
  303. ShowError(string.Format(App.Current?.FindResource("SweepItemsCountError")+"",2));
  304. return false;
  305. }
  306. Model.SweepItems = SweepItems.Select(x => x.Value.Model).Where(x=>x.Value >0).OrderBy(x => x.Frequency).DistinctBy(x => x.Frequency).ToList();
  307. SweepItems.Clear();
  308. if (Model.SweepItems.Count < 2)
  309. {
  310. for (int i = 0; i < Model.SweepItems.Count; i++)
  311. {
  312. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(i + 1, new SweepItemViewModel(Model.SweepItems[i])));
  313. }
  314. ShowError(string.Format(App.Current?.FindResource("SweepItemsCountError") + "", 2));
  315. return false;
  316. }
  317. else
  318. {
  319. for (int i = 0; i < Model.SweepItems.Count; i++)
  320. {
  321. if (i == 0)
  322. {
  323. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(i + 1, new SweepItemViewModel(Model.SweepItems[i])));
  324. continue;
  325. }
  326. switch (Model.SweepItems[i].SweepValueType)
  327. {
  328. case SweepValueType.DynamicAcceleration:
  329. case SweepValueType.FixedAcceleration:
  330. {
  331. switch(Model.SweepItems[i-1].SweepValueType)
  332. {
  333. case SweepValueType.DynamicAcceleration:
  334. break;
  335. case SweepValueType.FixedAcceleration:
  336. break;
  337. case SweepValueType.FixedDisplacement:
  338. Model.SweepItems[i].Frequency =Math.Round(Math.Sqrt(Model.SweepItems[i].Value*9800 / Model.SweepItems[i - 1].Value) / (2 * Math.PI),2);
  339. break;
  340. case SweepValueType.FixedVelocity:
  341. Model.SweepItems[i].Frequency = Math.Round(Model.SweepItems[i].Value * 9.8f / (2 * MathF.PI * Model.SweepItems[i - 1].Value), 2);
  342. break;
  343. }
  344. }
  345. break;
  346. case SweepValueType.FixedDisplacement:
  347. {
  348. switch(Model.SweepItems[i-1].SweepValueType)
  349. {
  350. case SweepValueType.DynamicAcceleration:
  351. case SweepValueType.FixedAcceleration:
  352. Model.SweepItems[i].Frequency = Math.Round(Math.Sqrt(Model.SweepItems[i-1].Value * 9800 / Model.SweepItems[i].Value) / (2 * Math.PI), 2);
  353. break;
  354. case SweepValueType.FixedVelocity:
  355. Model.SweepItems[i].Frequency = Math.Round(Model.SweepItems[i - 1].Value * 1000 / (2 * Math.PI * Model.SweepItems[i].Value), 2);
  356. break;
  357. case SweepValueType.FixedDisplacement:
  358. break;
  359. }
  360. }
  361. break;
  362. case SweepValueType.FixedVelocity:
  363. {
  364. switch (Model.SweepItems[i - 1].SweepValueType)
  365. {
  366. case SweepValueType.DynamicAcceleration:
  367. case SweepValueType.FixedAcceleration:
  368. Model.SweepItems[i].Frequency = Math.Round(Model.SweepItems[i-1].Value * 9.8f / (2 * Math.PI * Model.SweepItems[i].Value), 2);
  369. break;
  370. case SweepValueType.FixedVelocity:
  371. break;
  372. case SweepValueType.FixedDisplacement:
  373. Model.SweepItems[i].Frequency = Math.Round(Model.SweepItems[i].Value * 1000 / (2 * Math.PI * Model.SweepItems[i-1].Value), 2);
  374. break;
  375. }
  376. }
  377. break;
  378. }
  379. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(i + 1, new SweepItemViewModel(Model.SweepItems[i])));
  380. }
  381. }
  382. return true;
  383. }
  384. public ICommand RefreshCommand => new RelayCommand(Refresh);
  385. private void Refresh()
  386. {
  387. var result = VerifySweepItem();
  388. if (MaxFrequency <= MinFrequency) return;
  389. if (EndFrequency > MaxFrequency) EndFrequency = MaxFrequency;
  390. if (StartFrequency < MinFrequency) StartFrequency = MinFrequency;
  391. if (!result) return;
  392. datas.Clear();
  393. velocitydata.Clear();
  394. displacementdata.Clear();
  395. int count = (int)Math.Ceiling((MaxFrequency - MinFrequency) / interval)+1;
  396. for(int i=0;i<count;i++)
  397. {
  398. double freq = i * interval + MinFrequency;
  399. double acc = 0;
  400. double upstop = 0;
  401. double upwarn = 0;
  402. double downstop = 0;
  403. double downwarn = 0;
  404. Model.CalcAmpt(freq, ref acc, ref upstop, ref upwarn, ref downstop, ref downwarn);
  405. datas.Add(new SweepData(freq,double.NaN,Math.Round(acc,4),upstop,downstop,upwarn,downwarn));
  406. velocitydata.Add(new DataPoint(freq,Math.Round(Shaker.Models.Tools.Tools.AccelerationToVelocity(acc,freq),4)));
  407. displacementdata.Add(new DataPoint(freq,Math.Round(Shaker.Models.Tools.Tools.AccelerationToDisplacement(acc, freq),4)));
  408. }
  409. AccelerationModel.InvalidatePlot(false);
  410. VelocityModel.InvalidatePlot(false);
  411. DisplacementModel.InvalidatePlot(false);
  412. for(int i=0;i<lineSeries.Count;i++)
  413. {
  414. lineSeries[i].ItemsSource = datas;
  415. }
  416. velocity.ItemsSource = velocitydata;
  417. displacement.ItemsSource = displacementdata;
  418. VelocityModel.InvalidatePlot(true);
  419. DisplacementModel.InvalidatePlot(true);
  420. AccelerationModel.InvalidatePlot(true);
  421. OnPropertyChanged(nameof(MaxAcceleration));
  422. OnPropertyChanged(nameof(MaxVelocity));
  423. OnPropertyChanged(nameof(MaxDisplacement));
  424. OnPropertyChanged(nameof(AccelerationLoad));
  425. OnPropertyChanged(nameof(VelocityLoad));
  426. OnPropertyChanged(nameof(DisplacementLoad));
  427. OnPropertyChanged(nameof(AccelerationOverLimit));
  428. OnPropertyChanged(nameof(VelocityOverLimit));
  429. OnPropertyChanged(nameof(DisplacementOverLimit));
  430. OnPropertyChanged(nameof(MaxFrequency));
  431. OnPropertyChanged(nameof(MinFrequency));
  432. }
  433. [PropertyAssociation(nameof(SweepConfigModel.SweepSpeed))]
  434. public double SweepSpeed
  435. {
  436. get => Model.SweepSpeed;
  437. set
  438. {
  439. if (Math.Abs(value - Model.SweepSpeed)<=Increment || double.IsNaN(value)) return;
  440. SetProperty(ref Model.SweepSpeed, value);
  441. OnceSweepTime = Model.OCTToTime(value);
  442. }
  443. }
  444. [PropertyAssociation(nameof(SweepConfigModel.Increment))]
  445. public double Increment => Model.Increment;
  446. [PropertyAssociation(nameof(SweepConfigModel.OnceSweepTime),nameof(SweepConfigModel.SweepCount))]
  447. public double TotalSweepTime => OnceSweepTime * SweepCount;
  448. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  449. public bool AddEnabled => SweepItems.Count < MaxSweepItemCount;
  450. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  451. public bool DeleteEnabled => SweepItems.Count > 0;
  452. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  453. public bool RefreshEnabled => SweepItems.Count >= 2;
  454. [PropertyAssociation(nameof(SweepConfigModel.SweepStartLevel))]
  455. public double SweepStartLevel { get => Model.SweepStartLevel; set => SetProperty(ref Model.SweepStartLevel, value); }
  456. [PropertyAssociation(nameof(SweepConfigModel.CrossoverPoint))]
  457. public double CrossoverPoint { get => Model.CrossoverPoint; set => SetProperty(ref Model.CrossoverPoint, value); }
  458. [PropertyAssociation(nameof(SweepConfigModel.SweepLevelGain))]
  459. public double SweepLevelGain { get => Model.SweepLevelGain; set => SetProperty(ref Model.SweepLevelGain, value); }
  460. [PropertyAssociation(nameof(SweepConfigModel.LowFrequencyMinCorrect))]
  461. public double LowFrequencyMinCorrect { get => Model.LowFrequencyMinCorrect; set => SetProperty(ref Model.LowFrequencyMinCorrect, value); }
  462. [PropertyAssociation(nameof(SweepConfigModel.LowFrequencyMaxCorrect))]
  463. public double LowFrequencyMaxCorrect { get => Model.LowFrequencyMaxCorrect; set => SetProperty(ref Model.LowFrequencyMaxCorrect, value); }
  464. [PropertyAssociation(nameof(SweepConfigModel.HigthFrequencyMinCorrect))]
  465. public double HigthFrequencyMinCorrect { get => Model.HigthFrequencyMinCorrect; set => SetProperty(ref Model.HigthFrequencyMinCorrect, value); }
  466. [PropertyAssociation(nameof(SweepConfigModel.HigthFrequencyMaxCorrect))]
  467. public double HigthFrequencyMaxCorrect { get => Model.HigthFrequencyMaxCorrect; set => SetProperty(ref Model.HigthFrequencyMaxCorrect, value); }
  468. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  469. public AvaloniaList<IndexValueItemViewModel<SweepItemViewModel>> SweepItems { get; } = new AvaloniaList<IndexValueItemViewModel<SweepItemViewModel>>();
  470. public ICommand AddCommand => new RelayCommand(Add);
  471. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  472. public double MaxFrequency => SweepItems.Count ==0 ? ShakerConfigViewModel.Instance.MaxFrequency: SweepItems.Max(x=>x.Value.Frequency);
  473. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  474. public double MinFrequency => SweepItems.Count ==0 ? ShakerConfigViewModel.Instance.MinFrequency: SweepItems.Min(x => x.Value.Frequency);
  475. private void Add()
  476. {
  477. if (SweepItems.Count >= MaxSweepItemCount) return;
  478. Model.SweepItems.Add(new SweepItemModel());
  479. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(SweepItems.Count + 1,new SweepItemViewModel(Model.SweepItems[^1])));
  480. OnPropertyChanged(nameof(AddEnabled));
  481. OnPropertyChanged(nameof(DeleteEnabled));
  482. OnPropertyChanged(nameof(RefreshEnabled));
  483. OnPropertyChanged(nameof(MaxFrequency));
  484. OnPropertyChanged(nameof(MinFrequency));
  485. if (MaxFrequency <= MinFrequency) return;
  486. if (EndFrequency > MaxFrequency) EndFrequency = MaxFrequency;
  487. if (StartFrequency < MinFrequency) StartFrequency = MinFrequency;
  488. }
  489. public ICommand DeleteCommand => new RelayCommand(Delete);
  490. private void Delete()
  491. {
  492. if (SweepItems.Count == 0) return;
  493. Model.SweepItems.RemoveAt(Model.SweepItems.Count - 1);
  494. SweepItems.RemoveAt(SweepItems.Count - 1);
  495. OnPropertyChanged(nameof(AddEnabled));
  496. OnPropertyChanged(nameof(DeleteEnabled));
  497. OnPropertyChanged(nameof(RefreshEnabled));
  498. OnPropertyChanged(nameof(MaxFrequency));
  499. OnPropertyChanged(nameof(MinFrequency));
  500. if (MaxFrequency <= MinFrequency) return;
  501. if (EndFrequency > MaxFrequency) EndFrequency = MaxFrequency;
  502. if (StartFrequency < MinFrequency) StartFrequency = MinFrequency;
  503. }
  504. public override void UpDateModel(SweepConfigModel model)
  505. {
  506. base.UpDateModel(model);
  507. int count = model.SweepItems.Count - SweepItems.Count;
  508. if(count>0)
  509. {
  510. for(int i=0;i<count;i++)
  511. {
  512. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(SweepItems.Count + 1, new SweepItemViewModel()));
  513. }
  514. }
  515. else if(count<0)
  516. {
  517. SweepItems.RemoveRange(SweepItems.Count + count, Math.Abs(count));
  518. }
  519. else
  520. {
  521. }
  522. for(int i=0;i<SweepItems.Count;i++)
  523. {
  524. SweepItems[i].Value.UpDateModel(model.SweepItems[i]);
  525. }
  526. }
  527. protected override void Save()
  528. {
  529. base.Save();
  530. SineMainPageViewModel.Instance.SetRefSpectrum(datas);
  531. CommunicationViewModel.Instance.LocalCommunication?.GetEvent<SweepConfigModel>()?.Publish(this, Model);
  532. CommunicationViewModel.Instance.ServiceCommunication?.GetEvent<SweepConfigModel>()?.Publish(this, Model);
  533. }
  534. public List<SweepData> RefSpectrum => datas;
  535. public OxyPlot.PlotModel AccelerationModel { get; } = new OxyPlot.PlotModel();
  536. public PlotModel DisplacementModel { get; } = new PlotModel();
  537. public PlotModel VelocityModel { get; } = new PlotModel();
  538. public static SweepConfigViewModel Instance { get; } = new SweepConfigViewModel();
  539. }
  540. }