SweepConfigViewModel.cs 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576
  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:DisplayViewModelBase<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. if (args.Data.ShakerConfig.AccelerationConfigs.Count>0)
  166. {
  167. int lastselectedchannel = SelectChannel;
  168. Channels.Clear();
  169. for(int i=0;i<args.Data.ShakerConfig.AccelerationConfigs.Count;i++)
  170. {
  171. Channels.Add(new KeyValuePair<int, int>(i + 1, i));
  172. }
  173. if (Channels.Any(x => x.Value == lastselectedchannel))
  174. {
  175. SelectChannel = lastselectedchannel;
  176. }
  177. else SelectChannel = Channels.First().Value;
  178. }
  179. UpDateModel(args.Data.SweepConfig);
  180. CommunicationViewModel.Instance.LocalCommunication?.GetEvent<SweepConfigModel>()?.Subscrip((sender, args) =>
  181. {
  182. UpDateModel(args.Data);
  183. Refresh();
  184. SineMainPageViewModel.Instance.SetRefSpectrum(datas);
  185. CommunicationViewModel.Instance.ServiceCommunication?.GetEvent<SweepConfigModel>()?.Publish(this, Model);
  186. });
  187. });
  188. }
  189. static SweepConfigViewModel()
  190. {
  191. }
  192. public AvaloniaList<KeyValuePair<int,int>> Channels { get; } = new AvaloniaList<KeyValuePair<int,int>>();
  193. [PropertyAssociation(nameof(SweepConfigModel.SelectChannel))]
  194. public int SelectChannel
  195. {
  196. get => Model.SelectChannel;
  197. set
  198. {
  199. SetProperty(ref Model.SelectChannel, value);
  200. for(int i=0;i<ShakerConfigViewModel.Instance.AccelerationSensorCount;i++)
  201. {
  202. ShakerConfigViewModel.Instance.Accelerations[i].Value.Weight = value == i ? 1 : 0;
  203. }
  204. }
  205. }
  206. [PropertyAssociation(nameof(SweepConfigModel.MaxSweepItemCount))]
  207. public int MaxSweepItemCount => Model.MaxSweepItemCount;
  208. [PropertyAssociation(nameof(SweepConfigModel.SweepCount))]
  209. public uint SweepCount { get => Model.SweepCount; set => SetProperty(ref Model.SweepCount, value); }
  210. [PropertyAssociation(nameof(SweepConfigModel.SweepType))]
  211. public SweepType SweepType
  212. {
  213. get => Model.SweepType;
  214. set
  215. {
  216. if (Model.SweepType == value) return;
  217. SetProperty(ref Model.SweepType, value);
  218. Model.OnceSweepTime = Model.OCTToTime(Model.SweepSpeed);
  219. OnPropertyChanged(nameof(OnceSweepTime));
  220. }
  221. }
  222. [PropertyAssociation(nameof(SweepConfigModel.SweepDirection))]
  223. public SweepDirection SweepDirection { get => Model.SweepDirection; set => SetProperty(ref Model.SweepDirection, value); }
  224. [PropertyAssociation(nameof(SweepConfigModel.SignalType))]
  225. public SignalType SignalType
  226. {
  227. get => Model.SignalType;
  228. set
  229. {
  230. SetProperty(ref Model.SignalType, value);
  231. if(value == SignalType.Sweep)
  232. {
  233. OnceSweepTime = Model.OCTToTime(Model.SweepSpeed);
  234. }
  235. }
  236. }
  237. [PropertyAssociation(nameof(SweepConfigModel.StartFrequency))]
  238. public double StartFrequency
  239. {
  240. get => Model.StartFrequency;
  241. set
  242. {
  243. if (value == Model.StartFrequency) return;
  244. SetProperty(ref Model.StartFrequency, value);
  245. if (SignalType == SignalType.Fixed) return;
  246. OnceSweepTime = Model.OCTToTime(Model.SweepSpeed);
  247. }
  248. }
  249. public ICommand MaxLoadCommand => new RelayCommand(MaxLoad);
  250. private void MaxLoad()
  251. {
  252. SweepItems.Clear();
  253. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(1,new SweepItemViewModel(new SweepItemModel()
  254. {
  255. Frequency = ShakerConfigViewModel.Instance.MinFrequency,
  256. Value = ShakerConfigViewModel.Instance.MaxDisplacement,
  257. SweepValueType = SweepValueType.FixedDisplacement,
  258. })));
  259. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(2,new SweepItemViewModel( new SweepItemModel()
  260. {
  261. Frequency = ShakerConfigViewModel.Instance.MinFrequency + interval,
  262. Value = ShakerConfigViewModel.Instance.MaxVelocity,
  263. SweepValueType = SweepValueType.FixedVelocity,
  264. })));
  265. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(3,new SweepItemViewModel( new SweepItemModel()
  266. {
  267. Frequency = ShakerConfigViewModel.Instance.MinFrequency + interval * 2,
  268. Value = ShakerConfigViewModel.Instance.MaxAcceleration,
  269. SweepValueType = SweepValueType.FixedAcceleration,
  270. })));
  271. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(4,new SweepItemViewModel( new SweepItemModel()
  272. {
  273. Frequency = ShakerConfigViewModel.Instance.MaxFrequency,
  274. Value = ShakerConfigViewModel.Instance.MaxAcceleration,
  275. SweepValueType = SweepValueType.FixedAcceleration,
  276. })));
  277. Refresh();
  278. }
  279. public double MaxAcceleration => datas.Count == 0 ? 0 : datas.Max(x => x.TargetAcceleration);
  280. public double MaxDisplacement => displacementdata.Count == 0 ? 0 : displacementdata.Max(x => x.Y);
  281. public double MaxVelocity => velocitydata.Count == 0 ? 0 : velocitydata.Max(x => x.Y);
  282. public bool AccelerationOverLimit => MaxAcceleration >= ShakerConfigViewModel.Instance.MaxAcceleration;
  283. public bool DisplacementOverLimit => MaxDisplacement > ShakerConfigViewModel.Instance.MaxDisplacement;
  284. public bool VelocityOverLimit => MaxVelocity > ShakerConfigViewModel.Instance.MaxVelocity;
  285. public double AccelerationLoad => Math.Round(MaxAcceleration / ShakerConfigViewModel.Instance.MaxAcceleration * 100f, 2);
  286. public double DisplacementLoad => Math.Round((double)MaxDisplacement / ShakerConfigViewModel.Instance.MaxDisplacement * 100f, 2);
  287. public double VelocityLoad => Math.Round((double)MaxVelocity / ShakerConfigViewModel.Instance.MaxVelocity * 100f, 2);
  288. [PropertyAssociation(nameof(SweepConfigModel.EndFrequency))]
  289. public double EndFrequency
  290. {
  291. get => Model.EndFrequency;
  292. set
  293. {
  294. if (value == Model.EndFrequency) return;
  295. SetProperty(ref Model.EndFrequency, value);
  296. if (SignalType == SignalType.Fixed) return;
  297. Model.OnceSweepTime = Model.OCTToTime(Model.SweepSpeed);
  298. }
  299. }
  300. [PropertyAssociation(nameof(SweepConfigModel.OnceSweepTime))]
  301. public double OnceSweepTime
  302. {
  303. get => Model.OnceSweepTime;
  304. set
  305. {
  306. if (Math.Abs(value - Model.OnceSweepTime) <= Increment || double.IsNaN(value)) return;
  307. SetProperty(ref Model.OnceSweepTime, value);
  308. OnPropertyChanged(nameof(TotalSweepTime));
  309. if (SignalType == SignalType.Fixed) return;
  310. Model.SweepSpeed = Model.TimeToOCT(value);
  311. }
  312. }
  313. private bool VerifySweepItem()
  314. {
  315. if (SweepItems.Count < 2)
  316. {
  317. ShowError(string.Format(App.Current?.FindResource("SweepItemsCountError")+"",2));
  318. return false;
  319. }
  320. Model.SweepItems = SweepItems.Select(x => x.Value.Model).Where(x=>x.Value >0).OrderBy(x => x.Frequency).DistinctBy(x => x.Frequency).ToList();
  321. SweepItems.Clear();
  322. if (Model.SweepItems.Count < 2)
  323. {
  324. for (int i = 0; i < Model.SweepItems.Count; i++)
  325. {
  326. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(i + 1, new SweepItemViewModel(Model.SweepItems[i])));
  327. }
  328. ShowError(string.Format(App.Current?.FindResource("SweepItemsCountError") + "", 2));
  329. return false;
  330. }
  331. else
  332. {
  333. for (int i = 0; i < Model.SweepItems.Count; i++)
  334. {
  335. if (i == 0)
  336. {
  337. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(i + 1, new SweepItemViewModel(Model.SweepItems[i])));
  338. continue;
  339. }
  340. switch (Model.SweepItems[i].SweepValueType)
  341. {
  342. case SweepValueType.DynamicAcceleration:
  343. case SweepValueType.FixedAcceleration:
  344. {
  345. switch(Model.SweepItems[i-1].SweepValueType)
  346. {
  347. case SweepValueType.DynamicAcceleration:
  348. break;
  349. case SweepValueType.FixedAcceleration:
  350. break;
  351. case SweepValueType.FixedDisplacement:
  352. Model.SweepItems[i].Frequency =Math.Round(Math.Sqrt(Model.SweepItems[i].Value*9800 / Model.SweepItems[i - 1].Value) / (2 * Math.PI),2);
  353. break;
  354. case SweepValueType.FixedVelocity:
  355. Model.SweepItems[i].Frequency = Math.Round(Model.SweepItems[i].Value * 9.8f / (2 * MathF.PI * Model.SweepItems[i - 1].Value), 2);
  356. break;
  357. }
  358. }
  359. break;
  360. case SweepValueType.FixedDisplacement:
  361. {
  362. switch(Model.SweepItems[i-1].SweepValueType)
  363. {
  364. case SweepValueType.DynamicAcceleration:
  365. case SweepValueType.FixedAcceleration:
  366. Model.SweepItems[i].Frequency = Math.Round(Math.Sqrt(Model.SweepItems[i-1].Value * 9800 / Model.SweepItems[i].Value) / (2 * Math.PI), 2);
  367. break;
  368. case SweepValueType.FixedVelocity:
  369. Model.SweepItems[i].Frequency = Math.Round(Model.SweepItems[i - 1].Value * 1000 / (2 * Math.PI * Model.SweepItems[i].Value), 2);
  370. break;
  371. case SweepValueType.FixedDisplacement:
  372. break;
  373. }
  374. }
  375. break;
  376. case SweepValueType.FixedVelocity:
  377. {
  378. switch (Model.SweepItems[i - 1].SweepValueType)
  379. {
  380. case SweepValueType.DynamicAcceleration:
  381. case SweepValueType.FixedAcceleration:
  382. Model.SweepItems[i].Frequency = Math.Round(Model.SweepItems[i-1].Value * 9.8f / (2 * Math.PI * Model.SweepItems[i].Value), 2);
  383. break;
  384. case SweepValueType.FixedVelocity:
  385. break;
  386. case SweepValueType.FixedDisplacement:
  387. Model.SweepItems[i].Frequency = Math.Round(Model.SweepItems[i].Value * 1000 / (2 * Math.PI * Model.SweepItems[i-1].Value), 2);
  388. break;
  389. }
  390. }
  391. break;
  392. }
  393. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(i + 1, new SweepItemViewModel(Model.SweepItems[i])));
  394. }
  395. }
  396. return true;
  397. }
  398. public ICommand RefreshCommand => new RelayCommand(Refresh);
  399. private void Refresh()
  400. {
  401. var result = VerifySweepItem();
  402. if (MaxFrequency <= MinFrequency) return;
  403. if (EndFrequency > MaxFrequency) EndFrequency = MaxFrequency;
  404. if (StartFrequency < MinFrequency) StartFrequency = MinFrequency;
  405. if (!result) return;
  406. datas.Clear();
  407. velocitydata.Clear();
  408. displacementdata.Clear();
  409. int count = (int)Math.Ceiling((MaxFrequency - MinFrequency) / interval)+1;
  410. for(int i=0;i<count;i++)
  411. {
  412. double freq = i * interval + MinFrequency;
  413. double acc = 0;
  414. double upstop = 0;
  415. double upwarn = 0;
  416. double downstop = 0;
  417. double downwarn = 0;
  418. Model.CalcAmpt(freq, ref acc, ref upstop, ref upwarn, ref downstop, ref downwarn);
  419. datas.Add(new SweepData(freq,double.NaN,Math.Round(acc,4),upstop,downstop,upwarn,downwarn));
  420. velocitydata.Add(new DataPoint(freq,Math.Round(Shaker.Models.Tools.Tools.AccelerationToVelocity(acc,freq),4)));
  421. displacementdata.Add(new DataPoint(freq,Math.Round(Shaker.Models.Tools.Tools.AccelerationToDisplacement(acc, freq),4)));
  422. }
  423. AccelerationModel.InvalidatePlot(false);
  424. VelocityModel.InvalidatePlot(false);
  425. DisplacementModel.InvalidatePlot(false);
  426. for(int i=0;i<lineSeries.Count;i++)
  427. {
  428. lineSeries[i].ItemsSource = datas;
  429. }
  430. velocity.ItemsSource = velocitydata;
  431. displacement.ItemsSource = displacementdata;
  432. VelocityModel.InvalidatePlot(true);
  433. DisplacementModel.InvalidatePlot(true);
  434. AccelerationModel.InvalidatePlot(true);
  435. OnPropertyChanged(nameof(MaxAcceleration));
  436. OnPropertyChanged(nameof(MaxVelocity));
  437. OnPropertyChanged(nameof(MaxDisplacement));
  438. OnPropertyChanged(nameof(AccelerationLoad));
  439. OnPropertyChanged(nameof(VelocityLoad));
  440. OnPropertyChanged(nameof(DisplacementLoad));
  441. OnPropertyChanged(nameof(AccelerationOverLimit));
  442. OnPropertyChanged(nameof(VelocityOverLimit));
  443. OnPropertyChanged(nameof(DisplacementOverLimit));
  444. OnPropertyChanged(nameof(MaxFrequency));
  445. OnPropertyChanged(nameof(MinFrequency));
  446. }
  447. [PropertyAssociation(nameof(SweepConfigModel.SweepSpeed))]
  448. public double SweepSpeed
  449. {
  450. get => Model.SweepSpeed;
  451. set
  452. {
  453. if (Math.Abs(value - Model.SweepSpeed)<=Increment || double.IsNaN(value)) return;
  454. SetProperty(ref Model.SweepSpeed, value);
  455. Model.OnceSweepTime = Model.OCTToTime(value);
  456. }
  457. }
  458. [PropertyAssociation(nameof(SweepConfigModel.Increment))]
  459. public double Increment => Model.Increment;
  460. [PropertyAssociation(nameof(SweepConfigModel.OnceSweepTime),nameof(SweepConfigModel.SweepCount))]
  461. public double TotalSweepTime => OnceSweepTime * SweepCount;
  462. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  463. public bool AddEnabled => SweepItems.Count < MaxSweepItemCount;
  464. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  465. public bool DeleteEnabled => SweepItems.Count > 0;
  466. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  467. public bool RefreshEnabled => SweepItems.Count >= 2;
  468. [PropertyAssociation(nameof(SweepConfigModel.SweepStartLevel))]
  469. public double SweepStartLevel { get => Model.SweepStartLevel; set => SetProperty(ref Model.SweepStartLevel, value); }
  470. [PropertyAssociation(nameof(SweepConfigModel.CrossoverPoint))]
  471. public double CrossoverPoint { get => Model.CrossoverPoint; set => SetProperty(ref Model.CrossoverPoint, value); }
  472. [PropertyAssociation(nameof(SweepConfigModel.SweepLevelGain))]
  473. public double SweepLevelGain { get => Model.SweepLevelGain; set => SetProperty(ref Model.SweepLevelGain, value); }
  474. [PropertyAssociation(nameof(SweepConfigModel.LowFrequencyMinCorrect))]
  475. public double LowFrequencyMinCorrect { get => Model.LowFrequencyMinCorrect; set => SetProperty(ref Model.LowFrequencyMinCorrect, value); }
  476. [PropertyAssociation(nameof(SweepConfigModel.LowFrequencyMaxCorrect))]
  477. public double LowFrequencyMaxCorrect { get => Model.LowFrequencyMaxCorrect; set => SetProperty(ref Model.LowFrequencyMaxCorrect, value); }
  478. [PropertyAssociation(nameof(SweepConfigModel.HigthFrequencyMinCorrect))]
  479. public double HigthFrequencyMinCorrect { get => Model.HigthFrequencyMinCorrect; set => SetProperty(ref Model.HigthFrequencyMinCorrect, value); }
  480. [PropertyAssociation(nameof(SweepConfigModel.HigthFrequencyMaxCorrect))]
  481. public double HigthFrequencyMaxCorrect { get => Model.HigthFrequencyMaxCorrect; set => SetProperty(ref Model.HigthFrequencyMaxCorrect, value); }
  482. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  483. public AvaloniaList<IndexValueItemViewModel<SweepItemViewModel>> SweepItems { get; } = new AvaloniaList<IndexValueItemViewModel<SweepItemViewModel>>();
  484. public ICommand AddCommand => new RelayCommand(Add);
  485. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  486. public double MaxFrequency => SweepItems.Count ==0 ? ShakerConfigViewModel.Instance.MaxFrequency: SweepItems.Max(x=>x.Value.Frequency);
  487. [PropertyAssociation(nameof(SweepConfigModel.SweepItems))]
  488. public double MinFrequency => SweepItems.Count ==0 ? ShakerConfigViewModel.Instance.MinFrequency: SweepItems.Min(x => x.Value.Frequency);
  489. private void Add()
  490. {
  491. if (SweepItems.Count >= MaxSweepItemCount) return;
  492. Model.SweepItems.Add(new SweepItemModel());
  493. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(SweepItems.Count + 1,new SweepItemViewModel(Model.SweepItems[^1])));
  494. OnPropertyChanged(nameof(AddEnabled));
  495. OnPropertyChanged(nameof(DeleteEnabled));
  496. OnPropertyChanged(nameof(RefreshEnabled));
  497. OnPropertyChanged(nameof(MaxFrequency));
  498. OnPropertyChanged(nameof(MinFrequency));
  499. if (MaxFrequency <= MinFrequency) return;
  500. if (EndFrequency > MaxFrequency) EndFrequency = MaxFrequency;
  501. if (StartFrequency < MinFrequency) StartFrequency = MinFrequency;
  502. }
  503. public ICommand DeleteCommand => new RelayCommand(Delete);
  504. private void Delete()
  505. {
  506. if (SweepItems.Count == 0) return;
  507. Model.SweepItems.RemoveAt(Model.SweepItems.Count - 1);
  508. SweepItems.RemoveAt(SweepItems.Count - 1);
  509. OnPropertyChanged(nameof(AddEnabled));
  510. OnPropertyChanged(nameof(DeleteEnabled));
  511. OnPropertyChanged(nameof(RefreshEnabled));
  512. OnPropertyChanged(nameof(MaxFrequency));
  513. OnPropertyChanged(nameof(MinFrequency));
  514. if (MaxFrequency <= MinFrequency) return;
  515. if (EndFrequency > MaxFrequency) EndFrequency = MaxFrequency;
  516. if (StartFrequency < MinFrequency) StartFrequency = MinFrequency;
  517. }
  518. public override void UpDateModel(SweepConfigModel model)
  519. {
  520. base.UpDateModel(model);
  521. int count = model.SweepItems.Count - SweepItems.Count;
  522. if(count>0)
  523. {
  524. for(int i=0;i<count;i++)
  525. {
  526. SweepItems.Add(new IndexValueItemViewModel<SweepItemViewModel>(SweepItems.Count + 1, new SweepItemViewModel()));
  527. }
  528. }
  529. else if(count<0)
  530. {
  531. SweepItems.RemoveRange(SweepItems.Count + count, Math.Abs(count));
  532. }
  533. else
  534. {
  535. }
  536. for(int i=0;i<SweepItems.Count;i++)
  537. {
  538. SweepItems[i].Value.UpDateModel(model.SweepItems[i]);
  539. }
  540. }
  541. protected override void Save()
  542. {
  543. base.Save();
  544. SineMainPageViewModel.Instance.SetRefSpectrum(datas);
  545. CommunicationViewModel.Instance.LocalCommunication?.GetEvent<SweepConfigModel>()?.Publish(this, Model);
  546. CommunicationViewModel.Instance.ServiceCommunication?.GetEvent<SweepConfigModel>()?.Publish(this, Model);
  547. }
  548. public List<SweepData> RefSpectrum => datas;
  549. public OxyPlot.PlotModel AccelerationModel { get; } = new OxyPlot.PlotModel();
  550. public PlotModel DisplacementModel { get; } = new PlotModel();
  551. public PlotModel VelocityModel { get; } = new PlotModel();
  552. public static SweepConfigViewModel Instance { get; } = new SweepConfigViewModel();
  553. }
  554. }