ReservationManagementItemGrid.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409
  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. using System.Windows.Media;
  15. using Brush = System.Windows.Media.Brush;
  16. namespace PRSDesktop;
  17. public delegate void JobRequiItemSelect(CoreRow[] rows);
  18. public delegate void GridRefresh();
  19. public class ReservationManagementItemGrid : DynamicDataGrid<JobRequisitionItem>
  20. {
  21. private readonly ReservationManagementUserSettings _userSettings = new ReservationManagementUserSettings();
  22. private Button ArchiveButton;
  23. public bool ShowColors { get; set; }
  24. public int DueDateAlert { get; set; }
  25. public int DueDateWarning { get; set; }
  26. private class UIComponent : DynamicGridGridUIComponent<JobRequisitionItem>
  27. {
  28. private ReservationManagementItemGrid Grid;
  29. public UIComponent(ReservationManagementItemGrid grid)
  30. {
  31. Grid = grid;
  32. Parent = grid;
  33. }
  34. protected override Brush? GetCellBackground(CoreRow row, string columnname)
  35. {
  36. if (String.Equals(columnname, CoreUtils.GetFullPropertyName<JobRequisitionItem, DateTime>(x => x.Requisition.DueDate,".")))
  37. {
  38. var due = row.Get<JobRequisitionItem, DateTime>(x => x.Requisition.DueDate);
  39. if (Grid.ShowColors && !due.IsEmpty())
  40. {
  41. var background = DateTime.Today > due.Date
  42. ? Colors.Salmon
  43. : DateTime.Today.AddDays(Grid.DueDateWarning) >= due.Date
  44. ? Colors.Orange
  45. : DateTime.Today.AddDays(Grid.DueDateAlert) >= due.Date
  46. ? Colors.LightYellow
  47. : Colors.LightGreen;
  48. return new SolidColorBrush(background);
  49. }
  50. }
  51. return null;
  52. }
  53. }
  54. public ReservationManagementItemGrid()
  55. {
  56. _userSettings = new UserConfiguration<ReservationManagementUserSettings>().Load();
  57. FilterComponent.SetSettings(_userSettings.Filters, false);
  58. HiddenColumns.Add(x => x.ID);
  59. HiddenColumns.Add(x => x.Qty);
  60. HiddenColumns.Add(x => x.InStock);
  61. HiddenColumns.Add(x => x.Product.ID);
  62. HiddenColumns.Add(x => x.Product.Code);
  63. HiddenColumns.Add(x => x.Product.Group.ID);
  64. HiddenColumns.Add(x => x.Product.Group.Code);
  65. HiddenColumns.Add(x => x.Product.Group.Description);
  66. HiddenColumns.Add(x => x.Style.ID);
  67. HiddenColumns.Add(x => x.Style.Code);
  68. HiddenColumns.Add(x => x.Style.Description);
  69. HiddenColumns.Add(x => x.Status);
  70. HiddenColumns.Add(x => x.Requisition.ID);
  71. HiddenColumns.Add(x => x.Requisition.Job.ID);
  72. HiddenColumns.Add(x => x.Requisition.Job.JobNumber);
  73. HiddenColumns.Add(x => x.Requisition.Job.Name);
  74. HiddenColumns.Add(x => x.Requisition.Number);
  75. HiddenColumns.Add(x => x.Requisition.DueDate);
  76. HiddenColumns.Add(x => x.Job.ID);
  77. HiddenColumns.Add(x => x.Job.Name);
  78. HiddenColumns.Add(x => x.Job.JobNumber);
  79. HiddenColumns.Add(x => x.Dimensions.UnitSize);
  80. HiddenColumns.Add(x => x.Dimensions.Length);
  81. HiddenColumns.Add(x => x.Dimensions.Width);
  82. HiddenColumns.Add(x => x.Dimensions.Height);
  83. HiddenColumns.Add(x => x.Dimensions.Weight);
  84. HiddenColumns.Add(x => x.Dimensions.Quantity);
  85. HiddenColumns.Add(x => x.Dimensions.Value);
  86. HiddenColumns.Add(x => x.Dimensions.Unit.ID);
  87. HiddenColumns.Add(x => x.Dimensions.Unit.HasLength);
  88. HiddenColumns.Add(x => x.Dimensions.Unit.HasHeight);
  89. HiddenColumns.Add(x => x.Dimensions.Unit.HasWidth);
  90. HiddenColumns.Add(x => x.Dimensions.Unit.HasWeight);
  91. HiddenColumns.Add(x => x.Dimensions.Unit.HasQuantity);
  92. HiddenColumns.Add(x => x.Dimensions.Unit.Formula);
  93. HiddenColumns.Add(x => x.Dimensions.Unit.Format);
  94. HiddenColumns.Add(x => x.Dimensions.Unit.Code);
  95. HiddenColumns.Add(x => x.Dimensions.Unit.Description);
  96. if (Security.CanEdit<JobRequisitionItem>())
  97. ActionColumns.Add(new DynamicMenuColumn(BuildMenu));
  98. ColumnsTag = "JobRequisitionReview";
  99. FilterComponent.OnFiltersSelected += GridOnFilterSelected;
  100. ArchiveButton = AddButton("Archive", PRSDesktop.Resources.archive.AsBitmapImage(), ArchiveButton_Clicked);
  101. ArchiveButton.IsEnabled = false;
  102. }
  103. protected override IDynamicGridUIComponent<JobRequisitionItem> CreateUIComponent()
  104. {
  105. return new UIComponent(this);
  106. }
  107. protected override void DoReconfigure(FluentList<DynamicGridOption> options)
  108. {
  109. base.DoReconfigure(options);
  110. options.BeginUpdate()
  111. .AddRange(
  112. DynamicGridOption.FilterRows,
  113. DynamicGridOption.SelectColumns,
  114. DynamicGridOption.RecordCount,
  115. DynamicGridOption.DragSource)
  116. .Remove(DynamicGridOption.AddRows)
  117. .Remove(DynamicGridOption.ImportData)
  118. .Remove(DynamicGridOption.ExportData)
  119. .Remove(DynamicGridOption.Print)
  120. .Remove(DynamicGridOption.ShowHelp)
  121. .EndUpdate();
  122. }
  123. private void GridOnFilterSelected(DynamicGridSelectedFilterSettings settings)
  124. {
  125. new UserConfiguration<ReservationManagementUserSettings>().Save(new ReservationManagementUserSettings { Filters = settings });
  126. Refresh(false, true);
  127. }
  128. public override DynamicGridColumns GenerateColumns()
  129. {
  130. var columns = new DynamicGridColumns();
  131. columns.Add<JobRequisitionItem, DateTime>(x => x.Requisition.DueDate, 80, "Due", "", Alignment.MiddleCenter);
  132. columns.Add<JobRequisitionItem, string>(x => x.Requisition.Job.JobNumber, 70, "Job", "", Alignment.MiddleCenter);
  133. columns.Add<JobRequisitionItem, int>(x => x.Requisition.Number, 50, "Requi", "", Alignment.MiddleCenter);
  134. columns.Add<JobRequisitionItem, string>(x => x.Product.Code, 100, "Product Code", "", Alignment.MiddleLeft);
  135. columns.Add<JobRequisitionItem, string>(x => x.Product.Name, 200, "Product Name", "", Alignment.MiddleLeft);
  136. columns.Add<JobRequisitionItem, string>(x => x.Style.Description, 150, "Style", "", Alignment.MiddleLeft);
  137. columns.Add<JobRequisitionItem, double>(x => x.Qty, 50, "Qty", "", Alignment.MiddleCenter);
  138. columns.Add<JobRequisitionItem, string>(x => x.Dimensions.UnitSize, 70, "Size", "", Alignment.MiddleLeft);
  139. columns.Add<JobRequisitionItem, JobRequisitionItemStatus>(x => x.Status, 90, "Status", "", Alignment.MiddleCenter);
  140. columns.Add<JobRequisitionItem, double>(x => x.InStock, 50, "Stk.", "", Alignment.MiddleCenter);
  141. columns.Add<JobRequisitionItem, double>(x => x.OnOrder, 50, "Ord.", "", Alignment.MiddleCenter);
  142. columns.Add<JobRequisitionItem, string>(x => x.PurchaseOrderNumbers, 100, "Orders", "", Alignment.MiddleLeft);
  143. columns.Add<JobRequisitionItem, string>(x => x.Notes, 0, "Notes", "", Alignment.MiddleLeft);
  144. return columns;
  145. }
  146. protected override void SelectItems(CoreRow[]? rows)
  147. {
  148. base.SelectItems(rows);
  149. if(rows?.Any() == true)
  150. {
  151. ArchiveButton.IsEnabled = true;
  152. }
  153. }
  154. #region Action Column Buttons
  155. private void BuildMenu(DynamicMenuColumn column, CoreRow? row)
  156. {
  157. column.AddItem("Order Required", PRSDesktop.Resources.purchase, OrderRequired_Clicked);
  158. column.AddItem("Split Line", PRSDesktop.Resources.split, SplitLine_Clicked);
  159. column.AddItem("Archive", PRSDesktop.Resources.archive, Archive_Clicked);
  160. if (Security.CanView<StockMovement>())
  161. {
  162. column.AddSeparator();
  163. column.AddItem("View Stock Movements", PRSDesktop.Resources.forklift, ViewStockMovements);
  164. }
  165. }
  166. private bool CheckValidAction(JobRequisitionItem item)
  167. {
  168. bool valid = true;
  169. if (item.Status == JobRequisitionItemStatus.Allocated)
  170. {
  171. MessageWindow.ShowMessage("Item has already been reserved!", "Error", image: MessageWindow.WarningImage);
  172. return false;
  173. }
  174. else if (item.Status == JobRequisitionItemStatus.OnOrder)
  175. {
  176. MessageWindow.ShowMessage("Item is already on order!", "Error", image: MessageWindow.WarningImage);
  177. return false;
  178. }
  179. else if (item.InStock >= item.Qty)
  180. {
  181. MessageWindow.ShowMessage("Item is already in stock!", "Error", image: MessageWindow.WarningImage);
  182. return false;
  183. }
  184. return valid;
  185. }
  186. private void SplitLine(JobRequisitionItem item, double oldItemQty, double newItemQty, string notes)
  187. {
  188. var items = new List<JobRequisitionItem>();
  189. var newItem = new JobRequisitionItem();
  190. newItem.Requisition.ID = item.Requisition.ID;
  191. newItem.Requisition.Job.ID = item.Requisition.Job.ID;
  192. newItem.Product.ID = item.Product.ID;
  193. newItem.Dimensions.CopyFrom(item.Dimensions);
  194. newItem.Style.ID = item.Style.ID;
  195. newItem.Notes = item.Notes + Environment.NewLine + notes;
  196. item.Notes = newItem.Notes;
  197. item.Qty = oldItemQty;
  198. newItem.Qty = newItemQty;
  199. items.Add(newItem);
  200. items.Add(item);
  201. Client.Save(items, "Split lines from Job Requi Item Review Dashboard");
  202. MessageWindow.ShowMessage($"Line split - original line Qty is now {item.Qty}. New line Qty is {newItem.Qty}", "Lines split");
  203. Refresh(false, true);
  204. }
  205. private void SplitLine_Clicked(CoreRow? row)
  206. {
  207. if (row is null) return;
  208. var item = row.ToObject<JobRequisitionItem>();
  209. if (CheckValidAction(item))
  210. {
  211. var units = item.Qty - item.InStock;
  212. if(DoubleEdit.Execute("Enter amount to split", 1, units, ref units))
  213. {
  214. SplitLine(item, item.Qty - units, units, "Line split");
  215. }
  216. }
  217. }
  218. private static bool Archive(IEnumerable<JobRequisitionItem> items)
  219. {
  220. var itemsList = items.AsIList();
  221. var toChange = new List<JobRequisitionItem>();
  222. foreach(var item in itemsList)
  223. {
  224. if (item.Status != JobRequisitionItemStatus.Allocated)
  225. {
  226. var win = MessageWindow.New()
  227. .Message($"Requisition item for requisition {item.Requisition.Number} is not fully allocated; " +
  228. $"its current status is {item.Status}. Are you sure you wish to archive this item?")
  229. .Title("Confirm Archive")
  230. .AddYesButton("Archive");
  231. if(itemsList.Count > 1)
  232. {
  233. win.AddNoButton("Skip");
  234. }
  235. var result = win
  236. .AddCancelButton()
  237. .Display().Result;
  238. if (result == MessageWindowResult.Cancel)
  239. {
  240. return false;
  241. }
  242. else if(result == MessageWindowResult.Yes)
  243. {
  244. toChange.Add(item);
  245. }
  246. }
  247. else
  248. {
  249. toChange.Add(item);
  250. }
  251. }
  252. if(toChange.Count == 0)
  253. {
  254. MessageWindow.ShowMessage("No items archived.", "Cancelled");
  255. return false;
  256. }
  257. foreach(var item in toChange)
  258. {
  259. item.Archived = DateTime.Now;
  260. item.Notes += Environment.NewLine + "Line marked as Archived by " + App.EmployeeName + " on " + DateTime.Now.ToString("dd MMM yy");
  261. }
  262. Client.Save(toChange, "Updated From Job Requisition Review Dashboard");
  263. return true;
  264. }
  265. private bool ArchiveButton_Clicked(Button _btn, CoreRow[] rows)
  266. {
  267. return Archive(rows.ToObjects<JobRequisitionItem>());
  268. }
  269. private void Archive_Clicked(CoreRow? row)
  270. {
  271. if (row is null) return;
  272. if (Archive(CoreUtils.One(row.ToObject<JobRequisitionItem>())))
  273. {
  274. Refresh(false, true);
  275. }
  276. }
  277. private void OrderRequired_Clicked(CoreRow? row)
  278. {
  279. if (row is null) return;
  280. var item = row.ToObject<JobRequisitionItem>();
  281. if (CheckValidAction(item))
  282. {
  283. item.OrderRequired = DateTime.Now;
  284. item.Notes += Environment.NewLine + "Line marked as Order Required by " + App.EmployeeName + " on " + DateTime.Now.ToString("dd MMM yy");
  285. Client.Save(item, "Updated From Job Requisition Review Dashboard");
  286. Refresh(false, true);
  287. }
  288. }
  289. private void ViewStockMovements(CoreRow? row)
  290. {
  291. if (row is null) return;
  292. var requiID = row.Get<StockMovement, Guid>(x => x.ID);
  293. DynamicDataGrid<StockMovement> grid;
  294. if (DynamicGridUtils.TryFindDynamicGrid(typeof(DynamicDataGrid<>), typeof(StockMovement), out var gridType))
  295. {
  296. grid = (Activator.CreateInstance(gridType) as DynamicDataGrid<StockMovement>)!;
  297. }
  298. else
  299. {
  300. grid = new DynamicDataGrid<StockMovement>();
  301. grid.OnReconfigure += (options) =>
  302. {
  303. options.Add(DynamicGridOption.SelectColumns);
  304. };
  305. grid.Reconfigure();
  306. }
  307. grid.OnDefineFilter += (t) => new Filter<StockMovement>(x => x.JobRequisitionItem.ID).IsEqualTo(requiID);
  308. var window = DynamicGridUtils.CreateGridWindow("Stock movements", grid);
  309. window.ShowDialog();
  310. }
  311. #endregion
  312. protected override void Reload(Filters<JobRequisitionItem> criteria, Columns<JobRequisitionItem> columns, ref SortOrder<JobRequisitionItem>? sort, Action<CoreTable?, Exception?> action)
  313. {
  314. criteria.Add(new Filter<JobRequisitionItem>(x => x.Requisition.Approved).IsNotEqualTo(DateTime.MinValue));
  315. criteria.Add(new Filter<JobRequisitionItem>(x => x.Archived).IsEqualTo(DateTime.MinValue));
  316. criteria.Add(new Filter<JobRequisitionItem>(x => x.Cancelled).IsEqualTo(DateTime.MinValue));
  317. sort = new SortOrder<JobRequisitionItem>(x => x.Requisition.Number, SortDirection.Descending);
  318. base.Reload(criteria, columns, ref sort, action);
  319. }
  320. protected override DragDropEffects OnRowsDragStart(CoreRow[] rows)
  321. {
  322. // Only allow dragging the selected rows.
  323. var selected = SelectedRows.Select(x => x.Get<JobRequisitionItem, Guid>(x => x.ID)).ToHashSet();
  324. var draggedRows = rows.Where(x => selected.Contains(x.Get<JobRequisitionItem, Guid>(x => x.ID))).ToArray();
  325. if(draggedRows.Length == 0)
  326. {
  327. return DragDropEffects.None;
  328. }
  329. else
  330. {
  331. return base.OnRowsDragStart(draggedRows);
  332. }
  333. }
  334. }
  335. public class JobRequiReviewDashboardFilterItem
  336. {
  337. public Guid SupplierID { get; set; }
  338. public Guid ProductID { get; set; }
  339. public string Text { get; set; }
  340. public JobRequiReviewDashboardFilterItem()
  341. {
  342. SupplierID = Guid.Empty;
  343. ProductID = Guid.Empty;
  344. Text = "";
  345. }
  346. }