ReservationManagementItemGrid.cs 15 KB

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