ScheduleItemGrid.cs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. using System.Collections.Generic;
  2. using System.IO;
  3. using System.Linq;
  4. using System.Windows;
  5. using System.Windows.Controls;
  6. using System.Windows.Media.Imaging;
  7. using Comal.Classes;
  8. using InABox.Clients;
  9. using InABox.Core;
  10. using InABox.DynamicGrid;
  11. using InABox.WPF;
  12. using Microsoft.Win32;
  13. using Syncfusion.Linq;
  14. using Button = System.Windows.Controls.Button;
  15. using MessageBox = System.Windows.MessageBox;
  16. using OpenFileDialog = Microsoft.Win32.OpenFileDialog;
  17. using SaveFileDialog = Microsoft.Win32.SaveFileDialog;
  18. namespace PRS.Shared;
  19. internal class ScheduleItemGrid : DynamicDataGrid<Schedule>//, IDefaultGrid
  20. {
  21. private readonly BitmapImage disabled = PRS.Shared.Resources.disabled.AsBitmapImage();
  22. private readonly BitmapImage tick = PRS.Shared.Resources.tick.AsBitmapImage();
  23. protected override void Init()
  24. {
  25. base.Init();
  26. ActionColumns.Add(new DynamicTickColumn<Schedule, bool>(x => x.Active, tick, tick, disabled, CheckClick));
  27. HiddenColumns.Add(x => x.Active);
  28. HiddenColumns.Add(x => x.Title);
  29. HiddenColumns.Add(x => x.DocumentClass);
  30. AddButton("Export", PRS.Shared.Resources.download.AsBitmapImage(), SaveSchedules);
  31. AddButton("Import", PRS.Shared.Resources.upload.AsBitmapImage(), LoadSchedules);
  32. }
  33. protected override void DoReconfigure(DynamicGridOptions options)
  34. {
  35. base.DoReconfigure(options);
  36. options.RecordCount = true;
  37. options.SelectColumns = true;
  38. options.MultiSelect = true;
  39. }
  40. public CoreTable Schedules { get; set; }
  41. public Type DocumentType { get; set; }
  42. public Guid DocumentID { get; set; }
  43. private bool LoadSchedules(Button sender, CoreRow[] rows)
  44. {
  45. if (rows.Length != 1)
  46. {
  47. MessageBox.Show("Please select only one row to process");
  48. return false;
  49. }
  50. var row = rows.First();
  51. var dlg = new OpenFileDialog();
  52. dlg.Filter = "PRS Schedule Files (*.schedule)|*.schedule";
  53. if (dlg.ShowDialog() == true)
  54. {
  55. Progress.Show("");
  56. var json = File.ReadAllText(dlg.FileName);
  57. Schedule[] schedules = { };
  58. try
  59. {
  60. schedules = Serialization.Deserialize<Schedule[]>(json);
  61. }
  62. catch
  63. {
  64. Progress.Close();
  65. MessageBox.Show("[" + Path.GetFileName(dlg.FileName) + "] is not a valid schedule file!");
  66. return false;
  67. }
  68. if (!schedules.Any())
  69. {
  70. Progress.Close();
  71. MessageBox.Show("[" + Path.GetFileName(dlg.FileName) + "] does not contain any schedules!");
  72. return false;
  73. }
  74. foreach (var schedule in schedules)
  75. {
  76. schedule.DocumentID = DocumentID;
  77. schedule.DocumentClass = DocumentType.EntityName();
  78. schedule.ID = Guid.Empty;
  79. schedule.Active = false;
  80. schedule.DueDate = DateTime.MinValue;
  81. }
  82. new Client<Schedule>().Save(schedules, "Imported from [" + Path.GetFileName(dlg.FileName) + "]");
  83. Progress.Close();
  84. MessageBox.Show(string.Format("{0} schedules loaded from [{1}]", schedules.Length, Path.GetFileName(dlg.FileName)));
  85. }
  86. return true;
  87. }
  88. private bool SaveSchedules(Button sender, CoreRow[] rows)
  89. {
  90. if (rows.Any())
  91. {
  92. MessageBox.Show("Please select at least one schedule before Exporting!");
  93. return false;
  94. }
  95. var names = new List<string>();
  96. rows.ForEach(r => names.Add(r.Get<Schedule, string>(c => c.Title)));
  97. var filename = DocumentType.Name + " (" + string.Join(" + ", names) + ")";
  98. Path.GetInvalidFileNameChars().ForEach(c => filename = filename.Replace(c.ToString(), ""));
  99. Path.GetInvalidPathChars().ForEach(c => filename = filename.Replace(c.ToString(), ""));
  100. var dlg = new SaveFileDialog();
  101. dlg.Filter = "PRS Schedule Files (*.schedule)|*.schedule";
  102. dlg.FileName = filename + ".schedule";
  103. dlg.AddExtension = false;
  104. if (dlg.ShowDialog() == true)
  105. {
  106. Progress.Show("");
  107. Filter<Schedule> filter = null;
  108. foreach (var schedule in SelectedRows)
  109. if (filter == null)
  110. filter = new Filter<Schedule>(x => x.ID).IsEqualTo(schedule.Get<Schedule, Guid>(x => x.ID));
  111. else
  112. filter = filter.Or(x => x.ID).IsEqualTo(schedule.Get<Schedule, Guid>(x => x.ID));
  113. var schedules = new Client<Schedule>().Load(filter);
  114. foreach (var schedule in schedules)
  115. {
  116. schedule.DocumentID = DocumentID;
  117. schedule.DocumentClass = DocumentType.EntityName();
  118. schedule.ID = Guid.Empty;
  119. schedule.Active = false;
  120. schedule.DueDate = DateTime.MinValue;
  121. }
  122. var json = Serialization.Serialize(schedules);
  123. File.WriteAllText(dlg.FileName + ".schedule", json);
  124. Progress.Close();
  125. MessageBox.Show(string.Format("{0} schedules saved to [{1}]", SelectedRows.Length, Path.GetFileName(dlg.FileName)));
  126. }
  127. return false;
  128. }
  129. private bool CheckClick(CoreRow row)
  130. {
  131. var Due = row.Get<Schedule, DateTime>(x => x.DueDate);
  132. if (Due.Equals(DateTime.MinValue))
  133. {
  134. MessageBox.Show("Schedule must have a due date!");
  135. return false;
  136. }
  137. using (var client = new Client<Schedule>())
  138. {
  139. var schedule = client.Query(
  140. new Filter<Schedule>(x => x.ID).IsEqualTo(row.Get<Schedule, Guid>(x => x.ID)),
  141. Columns.Required<Schedule>().Add(x => x.Active))
  142. .ToObjects<Schedule>().First();
  143. schedule.Active = !schedule.Active;
  144. client.Save(schedule, schedule.Active ? "Activated Schedule" : "Disabled Schedule");
  145. }
  146. return true;
  147. }
  148. protected override void Reload(
  149. Filters<Schedule> criteria, Columns<Schedule> columns, ref SortOrder<Schedule>? sort,
  150. CancellationToken token, Action<CoreTable?, Exception?> action)
  151. {
  152. criteria.Add(new Filter<Schedule>(x => x.DocumentID).IsEqualTo(DocumentID));
  153. sort = new SortOrder<Schedule>(x => x.DueDate);
  154. base.Reload(criteria, columns, ref sort, token, action);
  155. }
  156. public override Schedule CreateItem()
  157. {
  158. if(DocumentType is null)
  159. {
  160. throw new Exception("Cannot create item when DocumentType is null.");
  161. }
  162. var schedule = base.CreateItem();
  163. schedule.DocumentClass = DocumentType.EntityName();
  164. schedule.DocumentID = DocumentID;
  165. if (DocumentType == typeof(CustomModule) || DocumentType == typeof(ScheduledScript))
  166. schedule.ScheduleType = ScheduleType.None;
  167. return schedule;
  168. }
  169. public override void SaveItem(Schedule item)
  170. {
  171. if(DocumentType is not null)
  172. {
  173. item.DocumentClass = DocumentType.EntityName();
  174. }
  175. else
  176. {
  177. if (string.IsNullOrWhiteSpace(item.DocumentClass))
  178. {
  179. throw new Exception("Cannot save item when DocumentType is null.");
  180. }
  181. }
  182. base.SaveItem(item);
  183. }
  184. protected override void DoReconfigureEditors(DynamicEditorGrid grid, Schedule[] items)
  185. {
  186. base.DoReconfigureEditors(grid, items);
  187. var frequency = grid.FindEditor("Frequency");
  188. if (frequency != null)
  189. {
  190. var freq = (int)frequency.GetValue("Frequency");
  191. var period = grid.FindEditor("Period");
  192. period?.SetEnabled(freq > 0);
  193. var due = grid.FindEditor("DueDate");
  194. due?.SetEnabled(freq > 0);
  195. }
  196. var threshold = grid.FindEditor("Threshold");
  197. if (threshold != null)
  198. {
  199. var thresh = (int)threshold.GetValue("Threshold");
  200. var trigger = grid.FindEditor("Trigger");
  201. trigger?.SetEnabled(thresh > 0);
  202. var next = grid.FindEditor("DueThreshold");
  203. next?.SetEnabled(thresh > 0);
  204. }
  205. var scheduleTypeEditor = grid.FindEditor("ScheduleType");
  206. if(scheduleTypeEditor != null)
  207. {
  208. var scheduleType = (ScheduleType)scheduleTypeEditor.GetValue("ScheduleType");
  209. var taskTypeEditor = grid.FindEditor(nameof(Schedule.KanbanType));
  210. taskTypeEditor?.SetEnabled(scheduleType == ScheduleType.Task);
  211. }
  212. }
  213. protected override BaseEditor? GetEditor(object item, DynamicGridColumn column)
  214. {
  215. var types = new List<Type> { typeof(CustomModule), typeof(ScheduledScript), typeof(Employee), typeof(Equipment) };
  216. var columns = new List<string> { "ScheduleType" };
  217. var schedule = (Schedule)item;
  218. var documentType = schedule.DocumentType();
  219. if (documentType == typeof(CustomModule) || documentType == typeof(ScheduledScript))
  220. columns.AddRange(new[]
  221. {
  222. "Description", "LeadTime", "EmployeeLink.ID", "ManagerLink.ID", "Report.ID", "Threshold", "Trigger", "DueThreshold", "Rollover",
  223. "QAForm"
  224. });
  225. else if (documentType == typeof(Employee))
  226. columns.AddRange(new[] { "Description", "EmployeeLink.ID", "Threshold", "Trigger", "DueThreshold" });
  227. else if (documentType == typeof(Customer))
  228. columns.AddRange(new[] { "Threshold", "Trigger", "DueThreshold" });
  229. if (types.Contains(documentType) && columns.Contains(column.ColumnName))
  230. return new NullEditor();
  231. return base.GetEditor(item, column);
  232. }
  233. public override DynamicEditorPages LoadEditorPages(Schedule item)
  234. {
  235. var pages = base.LoadEditorPages(item);
  236. foreach (var page in pages.ToArray())
  237. {
  238. if (page is IDynamicOneToManyGrid<Schedule, Kanban> && item.ScheduleType != ScheduleType.Task)
  239. pages = new DynamicEditorPages(pages.Where(x => x != page));
  240. else if (page is IDynamicOneToManyGrid<Schedule, Job> && item.ScheduleType != ScheduleType.Job)
  241. pages = new DynamicEditorPages(pages.Where(x => x != page));
  242. }
  243. return pages;
  244. }
  245. protected override void DefineLookups(ILookupEditorControl sender, Schedule[] items, bool async = true)
  246. {
  247. base.DefineLookups(sender, items, async);
  248. }
  249. }