JobRequisitionReviewGrid.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. using Comal.Classes;
  2. using InABox.Clients;
  3. using InABox.Configuration;
  4. using InABox.Core;
  5. using InABox.DynamicGrid;
  6. using InABox.Wpf;
  7. using InABox.WPF;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Drawing;
  11. using System.Linq;
  12. using System.Windows;
  13. using System.Windows.Controls;
  14. namespace PRSDesktop;
  15. public class JobRequisitionReviewUserSettings : IUserConfigurationSettings
  16. {
  17. [Obsolete]
  18. private CoreFilterDefinition? _currentFilter;
  19. [Obsolete]
  20. public CoreFilterDefinition? CurrentFilter
  21. {
  22. get => _currentFilter;
  23. set
  24. {
  25. if (value is not null)
  26. {
  27. Filters = new DynamicGridSelectedFilterSettings(new List<CoreFilterDefinition> { value }, false, null);
  28. }
  29. }
  30. }
  31. public DynamicGridSelectedFilterSettings Filters { get; set; } = new();
  32. }
  33. public delegate void JobRequiItemSelect(CoreRow[] rows);
  34. public delegate void GridRefresh();
  35. public class JobRequisitionReviewGrid : DynamicDataGrid<JobRequisitionItem>
  36. {
  37. private readonly JobRequisitionReviewUserSettings FilterSettings = new JobRequisitionReviewUserSettings();
  38. public JobRequisitionReviewGrid()
  39. {
  40. FilterSettings = new UserConfiguration<JobRequisitionReviewUserSettings>().Load();
  41. FilterComponent.SetSettings(FilterSettings.Filters, false);
  42. HiddenColumns.Add(x => x.ID);
  43. HiddenColumns.Add(x => x.Product.ID);
  44. HiddenColumns.Add(x => x.Product.Code);
  45. HiddenColumns.Add(x => x.Product.Group.ID);
  46. HiddenColumns.Add(x => x.Product.Group.Code);
  47. HiddenColumns.Add(x => x.Product.Group.Description);
  48. HiddenColumns.Add(x => x.Style.ID);
  49. HiddenColumns.Add(x => x.Style.Code);
  50. HiddenColumns.Add(x => x.Style.Description);
  51. HiddenColumns.Add(x => x.Status);
  52. HiddenColumns.Add(x => x.Requisition.ID);
  53. HiddenColumns.Add(x => x.Requisition.Job.ID);
  54. HiddenColumns.Add(x => x.Requisition.Job.JobNumber);
  55. HiddenColumns.Add(x => x.Requisition.Job.Name);
  56. HiddenColumns.Add(x => x.Requisition.Number);
  57. HiddenColumns.Add(x => x.Job.ID);
  58. HiddenColumns.Add(x => x.Job.Name);
  59. HiddenColumns.Add(x => x.Job.JobNumber);
  60. HiddenColumns.Add(x => x.Dimensions.UnitSize);
  61. HiddenColumns.Add(x => x.Dimensions.Length);
  62. HiddenColumns.Add(x => x.Dimensions.Width);
  63. HiddenColumns.Add(x => x.Dimensions.Height);
  64. HiddenColumns.Add(x => x.Dimensions.Weight);
  65. HiddenColumns.Add(x => x.Dimensions.Quantity);
  66. HiddenColumns.Add(x => x.Dimensions.Value);
  67. HiddenColumns.Add(x => x.Dimensions.Unit.ID);
  68. HiddenColumns.Add(x => x.Dimensions.Unit.HasLength);
  69. HiddenColumns.Add(x => x.Dimensions.Unit.HasHeight);
  70. HiddenColumns.Add(x => x.Dimensions.Unit.HasWidth);
  71. HiddenColumns.Add(x => x.Dimensions.Unit.HasWeight);
  72. HiddenColumns.Add(x => x.Dimensions.Unit.HasQuantity);
  73. HiddenColumns.Add(x => x.Dimensions.Unit.Formula);
  74. HiddenColumns.Add(x => x.Dimensions.Unit.Format);
  75. HiddenColumns.Add(x => x.Dimensions.Unit.Code);
  76. HiddenColumns.Add(x => x.Dimensions.Unit.Description);
  77. if (Security.CanEdit<JobRequisitionItem>())
  78. ActionColumns.Add(new DynamicMenuColumn(BuildMenu));
  79. ColumnsTag = "JobRequisitionReview";
  80. FilterComponent.OnFiltersSelected += GridOnFilterSelected;
  81. }
  82. protected override void DoReconfigure(FluentList<DynamicGridOption> options)
  83. {
  84. base.DoReconfigure(options);
  85. options.BeginUpdate()
  86. .AddRange(
  87. DynamicGridOption.FilterRows,
  88. DynamicGridOption.SelectColumns,
  89. DynamicGridOption.RecordCount,
  90. DynamicGridOption.DragSource)
  91. .Remove(DynamicGridOption.AddRows)
  92. .Remove(DynamicGridOption.ImportData)
  93. .Remove(DynamicGridOption.ExportData)
  94. .Remove(DynamicGridOption.Print)
  95. .Remove(DynamicGridOption.ShowHelp)
  96. .EndUpdate();
  97. }
  98. private void GridOnFilterSelected(DynamicGridSelectedFilterSettings settings)
  99. {
  100. new UserConfiguration<JobRequisitionReviewUserSettings>().Save(new JobRequisitionReviewUserSettings { Filters = settings });
  101. Refresh(false, true);
  102. }
  103. public override DynamicGridColumns GenerateColumns()
  104. {
  105. var columns = new DynamicGridColumns();
  106. columns.Add<JobRequisitionItem, DateTime>(x => x.Created, 80, "Date", "", Alignment.MiddleLeft);
  107. columns.Add<JobRequisitionItem, string>(x => x.Requisition.Job.JobNumber, 70, "Job", "", Alignment.MiddleLeft);
  108. columns.Add<JobRequisitionItem, int>(x => x.Requisition.Number, 50, "NO.", "", Alignment.MiddleLeft);
  109. columns.Add<JobRequisitionItem, string>(x => x.Product.Code, 70, "Code", "", Alignment.MiddleLeft);
  110. columns.Add<JobRequisitionItem, string>(x => x.Product.Name, 200, "Product Name", "", Alignment.MiddleLeft);
  111. columns.Add<JobRequisitionItem, string>(x => x.Style.Description, 150, "Style", "", Alignment.MiddleLeft);
  112. columns.Add<JobRequisitionItem, double>(x => x.Qty, 50, "Qty", "", Alignment.MiddleLeft);
  113. columns.Add<JobRequisitionItem, string>(x => x.Dimensions.UnitSize, 50, "Size", "", Alignment.MiddleLeft);
  114. columns.Add<JobRequisitionItem, string>(x => x.PurchaseOrderNumbers, 80, "PO Numbers", "", Alignment.MiddleLeft);
  115. columns.Add<JobRequisitionItem, string>(x => x.Notes, 300, "Notes", "", Alignment.MiddleLeft);
  116. return columns;
  117. }
  118. #region Action Column Buttons
  119. private bool CheckValidAction(JobRequisitionItem item)
  120. {
  121. bool valid = true;
  122. if (item.Status == JobRequisitionItemStatus.Allocated)
  123. {
  124. MessageWindow.ShowMessage("Item has already been reserved!", "Error", image: MessageWindow.WarningImage);
  125. return false;
  126. }
  127. else if (item.Status == JobRequisitionItemStatus.OnOrder)
  128. {
  129. MessageWindow.ShowMessage("Item is already on order!", "Error", image: MessageWindow.WarningImage);
  130. return false;
  131. }
  132. return valid;
  133. }
  134. private void SplitLine(JobRequisitionItem item, double oldItemQty, double newItemQty, string notes)
  135. {
  136. List<JobRequisitionItem> items = new List<JobRequisitionItem>();
  137. JobRequisitionItem newItem = new JobRequisitionItem();
  138. newItem.Requisition.ID = item.Requisition.ID;
  139. newItem.Requisition.Job.ID = item.Requisition.Job.ID;
  140. newItem.Requisition.Job.JobNumber = item.Requisition.Job.JobNumber;
  141. newItem.Requisition.Job.Name = item.Requisition.Job.Name;
  142. newItem.Product.ID = item.Product.ID;
  143. newItem.Product.Name = item.Product.Name;
  144. newItem.Product.Code = item.Product.Code;
  145. newItem.Product.Group.ID = item.Product.Group.ID;
  146. newItem.Product.Group.Description = item.Product.Group.Description;
  147. newItem.Dimensions.CopyFrom(item.Dimensions);
  148. newItem.Style.ID = item.Style.ID;
  149. newItem.Style.Description = item.Style.Description;
  150. newItem.Style.Code = item.Style.Code;
  151. newItem.Notes = item.Notes + Environment.NewLine + notes;
  152. item.Notes = newItem.Notes;
  153. item.Qty = oldItemQty;
  154. newItem.Qty = newItemQty;
  155. items.Add(newItem);
  156. items.Add(item);
  157. Client.Save(items, "Split lines from Job Requi Item Review Dashboard");
  158. MessageWindow.ShowMessage("Line split - original line Qty is now " + item.Qty + ". New line Qty is " + newItem.Qty, "Success");
  159. Refresh(false, true);
  160. }
  161. private void SplitLine_Clicked(CoreRow? row)
  162. {
  163. if (row is null) return;
  164. var item = row.ToObject<JobRequisitionItem>();
  165. if (CheckValidAction(item))
  166. {
  167. int units = Convert.ToInt32(item.Qty);
  168. if (NumberEdit.Execute("Enter amount to split", 1, units, ref units))
  169. {
  170. SplitLine(item, item.Qty - units, units, "Line split");
  171. }
  172. }
  173. }
  174. private void Archive_Clicked(CoreRow? row)
  175. {
  176. if (row is null) return;
  177. var item = row.ToObject<JobRequisitionItem>();
  178. if(item.Status != JobRequisitionItemStatus.Allocated)
  179. {
  180. var result = MessageWindow.New()
  181. .Message($"This requisition item is not fully allocated; its current status is {item.Status}. Are you sure you wish to archive this item?")
  182. .Title("Confirm Archive")
  183. .AddYesButton("Archive")
  184. .AddCancelButton()
  185. .Display().Result;
  186. if(result != MessageWindowResult.Yes)
  187. {
  188. return;
  189. }
  190. }
  191. item.Archived = DateTime.Now;
  192. item.Notes += Environment.NewLine + "Line marked as Archived by " + App.EmployeeName + " on " + DateTime.Now.ToString("dd MMM yy");
  193. Client.Save(item, "Updated From Job Requisition Review Dashboard");
  194. Refresh(false, true);
  195. }
  196. private void OrderRequired_Clicked(CoreRow? row)
  197. {
  198. if (row is null) return;
  199. var item = row.ToObject<JobRequisitionItem>();
  200. if (CheckValidAction(item))
  201. {
  202. item.OrderRequired = DateTime.Now;
  203. item.Notes += Environment.NewLine + "Line marked as Order Required by " + App.EmployeeName + " on " + DateTime.Now.ToString("dd MMM yy");
  204. Client.Save(item, "Updated From Job Requisition Review Dashboard");
  205. Refresh(false, true);
  206. }
  207. }
  208. #endregion
  209. protected override void Reload(Filters<JobRequisitionItem> criteria, Columns<JobRequisitionItem> columns, ref SortOrder<JobRequisitionItem>? sort, Action<CoreTable?, Exception?> action)
  210. {
  211. criteria.Add(new Filter<JobRequisitionItem>(x => x.Requisition.Approved).IsNotEqualTo(DateTime.MinValue));
  212. criteria.Add(new Filter<JobRequisitionItem>(x => x.Archived).IsEqualTo(DateTime.MinValue));
  213. sort = new SortOrder<JobRequisitionItem>(x => x.Requisition.Number, SortDirection.Descending);
  214. base.Reload(criteria, columns, ref sort, action);
  215. }
  216. private void BuildMenu(DynamicMenuColumn column, CoreRow? row)
  217. {
  218. column.AddItem("Order Required", PRSDesktop.Resources.purchase, OrderRequired_Clicked);
  219. column.AddItem("Split Line", PRSDesktop.Resources.split, SplitLine_Clicked);
  220. column.AddItem("Archive", PRSDesktop.Resources.archive, Archive_Clicked);
  221. }
  222. protected override DragDropEffects OnRowsDragStart(CoreRow[] rows)
  223. {
  224. // Only allow dragging the selected rows.
  225. var selected = SelectedRows.Select(x => x.Get<JobRequisitionItem, Guid>(x => x.ID)).ToHashSet();
  226. var draggedRows = rows.Where(x => selected.Contains(x.Get<JobRequisitionItem, Guid>(x => x.ID))).ToArray();
  227. if(draggedRows.Length == 0)
  228. {
  229. return DragDropEffects.None;
  230. }
  231. else
  232. {
  233. return base.OnRowsDragStart(draggedRows);
  234. }
  235. }
  236. }
  237. public class JobRequiReviewDashboardFilterItem
  238. {
  239. public Guid SupplierID { get; set; }
  240. public Guid ProductID { get; set; }
  241. public string Text { get; set; }
  242. public JobRequiReviewDashboardFilterItem()
  243. {
  244. SupplierID = Guid.Empty;
  245. ProductID = Guid.Empty;
  246. Text = "";
  247. }
  248. }