JobRequisitionItemGrid.cs 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Reactive.Linq;
  5. using System.Windows;
  6. using System.Windows.Controls;
  7. using Comal.Classes;
  8. using InABox.Clients;
  9. using InABox.Core;
  10. using InABox.DynamicGrid;
  11. using InABox.Wpf;
  12. using InABox.WPF;
  13. namespace PRSDesktop;
  14. internal class JobRequisitionItemGrid : DynamicDataGrid<JobRequisitionItem>, IMasterDetailControl<JobRequisition,JobRequisitionItem>, ISpecificGrid
  15. {
  16. public JobRequisition? Master { get; set; }
  17. public Filter<JobRequisitionItem> MasterDetailFilter => Master != null
  18. ? new Filter<JobRequisitionItem>(x => x.Requisition.ID).IsEqualTo(Master.ID)
  19. : new Filter<JobRequisitionItem>().None();
  20. private Button CancelItemsButton;
  21. public JobRequisitionItemGrid()
  22. {
  23. CancelItemsButton = AddButton("Cancel Items", PRSDesktop.Resources.disabled.AsBitmapImage(), CancelItems);
  24. CancelItemsButton.Visibility = Security.CanEdit<JobRequisition>() && Security.CanEdit<JobRequisitionItem>()
  25. ? System.Windows.Visibility.Visible
  26. : System.Windows.Visibility.Hidden;
  27. }
  28. protected override void SelectItems(CoreRow[]? rows)
  29. {
  30. base.SelectItems(rows);
  31. CancelItemsButton.IsEnabled = rows is not null && rows.Length > 0;
  32. }
  33. private bool CancelItems(Button button, CoreRow[] rows)
  34. {
  35. if(rows.Length == 0)
  36. {
  37. MessageWindow.ShowMessage("Please select at least one item to cancel.", "Select items");
  38. return false;
  39. }
  40. // Reloading so I can ensure the correct columns without having to add hidden columns to JobRequisitionGrid.
  41. var oldRequi = Client.Query(
  42. new Filter<JobRequisition>(x => x.ID).IsEqualTo(Master?.ID ?? Guid.Empty),
  43. new Columns<JobRequisition>(x => x.Description)
  44. .Add(x => x.Number)
  45. .Add(x => x.DueDate))
  46. .ToObjects<JobRequisition>()
  47. .First();
  48. var oldRequiItems = Client.Query(
  49. new Filter<JobRequisitionItem>(x => x.ID).InList(rows.Select(x => x.Get<JobRequisitionItem, Guid>(x => x.ID)).ToArray()),
  50. new Columns<JobRequisitionItem>(x => x.Qty)
  51. .Add(x => x.Sequence)
  52. .Add(x => x.Job.ID)
  53. .Add(x => x.Product.ID)
  54. .Add(x => x.Product.Code)
  55. .Add(x => x.Style.ID)
  56. .Add(x => x.Style.Code)
  57. .Add(x => x.Style.Description)
  58. .Add(x => x.Dimensions.Unit.ID)
  59. .Add(x => x.Dimensions.Quantity)
  60. .Add(x => x.Dimensions.Length)
  61. .Add(x => x.Dimensions.Width)
  62. .Add(x => x.Dimensions.Height)
  63. .Add(x => x.Dimensions.Weight)
  64. .Add(x => x.Dimensions.UnitSize)
  65. .Add(x => x.Supplier.ID));
  66. var requisition = new JobRequisition
  67. {
  68. Description = $"Adjustment Requisition for Requisition {oldRequi.Number}",
  69. DueDate = oldRequi.DueDate
  70. };
  71. requisition.Job.ID = Master?.Job.ID ?? Guid.Empty;
  72. requisition.Job.Synchronise(Master?.Job ?? new JobLink());
  73. var requiItems = new List<JobRequisitionItem>();
  74. foreach(var oldItem in oldRequiItems.ToObjects<JobRequisitionItem>())
  75. {
  76. var newItem = new JobRequisitionItem
  77. {
  78. Notes = "Adjustment Requisition item",
  79. Qty = -oldItem.Qty,
  80. Sequence = oldItem.Sequence
  81. };
  82. newItem.Job.ID = requisition.Job.ID;
  83. newItem.Product.ID = oldItem.Product.ID;
  84. newItem.Style.ID = oldItem.Style.ID;
  85. newItem.Supplier.ID = oldItem.Supplier.ID;
  86. requiItems.Add(newItem);
  87. }
  88. var grid = DynamicGridUtils.CreateDynamicGrid(typeof(DynamicGrid<>), typeof(JobRequisition));
  89. if (grid.EditItems(new JobRequisition[] { requisition }, t =>
  90. {
  91. if (t == typeof(JobRequisitionItem))
  92. {
  93. var table = new CoreTable();
  94. table.LoadColumns(new Columns<JobRequisitionItem>().Default(
  95. ColumnType.IncludeOptional,
  96. ColumnType.IncludeEditable,
  97. ColumnType.IncludeAggregates,
  98. ColumnType.IncludeForeignKeys,
  99. ColumnType.IncludeFormulae));
  100. table.LoadRows(requiItems);
  101. return table;
  102. }
  103. return null;
  104. }))
  105. {
  106. MessageWindow.ShowMessage($"Created requisition {requisition.Number}", "Created Requisition");
  107. return true;
  108. }
  109. else
  110. {
  111. return false;
  112. }
  113. }
  114. protected override void DoReconfigure(FluentList<DynamicGridOption> options)
  115. {
  116. base.DoReconfigure(options);
  117. options.AddRange(
  118. DynamicGridOption.RecordCount,
  119. DynamicGridOption.SelectColumns,
  120. DynamicGridOption.FilterRows,
  121. DynamicGridOption.MultiSelect
  122. );
  123. }
  124. protected override void Reload(Filters<JobRequisitionItem> criteria, Columns<JobRequisitionItem> columns,
  125. ref SortOrder<JobRequisitionItem>? sort,
  126. Action<CoreTable?, Exception?> action)
  127. {
  128. criteria.Add(MasterDetailFilter);
  129. base.Reload(criteria, columns, ref sort, action);
  130. }
  131. protected override bool CanCreateItems()
  132. {
  133. return base.CanCreateItems() && (Master?.ID ?? Guid.Empty) != Guid.Empty;
  134. }
  135. protected override JobRequisitionItem CreateItem()
  136. {
  137. var result = base.CreateItem();
  138. result.Requisition.ID = Master?.ID ?? Guid.Empty;
  139. result.Requisition.Synchronise(Master ?? new JobRequisition());
  140. result.Job.ID = Master?.Job.ID ?? Guid.Empty;
  141. result.Job.Synchronise(Master?.Job ?? new JobLink());
  142. result.Qty = 1;
  143. return result;
  144. }
  145. protected override void OnAfterEditorValueChanged(DynamicEditorGrid? grid, JobRequisitionItem[] items, AfterEditorValueChangedArgs args, Dictionary<string, object?> changes)
  146. {
  147. base.OnAfterEditorValueChanged(grid, items, args, changes);
  148. if (args.ColumnName.Equals("Product.ID") || args.ColumnName.Equals("Dimensions") || args.ColumnName.StartsWith("Dimensions.") || args.ColumnName.Equals("Style.ID") || args.ColumnName.Equals("Supplier.ID"))
  149. {
  150. JobRequisitionItem.UpdateCosts(
  151. items,
  152. changes
  153. );
  154. }
  155. }
  156. public override DynamicGridColumns GenerateColumns()
  157. {
  158. var columns = new DynamicGridColumns();
  159. columns.Add<JobRequisitionItem, DateTime>(x => x.Created, 80, "Date", "", Alignment.MiddleLeft);
  160. columns.Add<JobRequisitionItem, string>(x => x.Requisition.Job.JobNumber, 70, "Job", "", Alignment.MiddleLeft);
  161. columns.Add<JobRequisitionItem, int>(x => x.Requisition.Number, 50, "NO.", "", Alignment.MiddleLeft);
  162. columns.Add<JobRequisitionItem, string>(x => x.Product.Code, 70, "Code", "", Alignment.MiddleLeft);
  163. columns.Add<JobRequisitionItem, string>(x => x.Product.Name, 200, "Product Name", "", Alignment.MiddleLeft);
  164. columns.Add<JobRequisitionItem, string>(x => x.Style.Description, 150, "Style", "", Alignment.MiddleLeft);
  165. columns.Add<JobRequisitionItem, double>(x => x.Qty, 50, "Qty", "", Alignment.MiddleLeft);
  166. columns.Add<JobRequisitionItem, string>(x => x.Dimensions.UnitSize, 50, "Size", "", Alignment.MiddleLeft);
  167. columns.Add<JobRequisitionItem, string>(x => x.PurchaseOrderNumbers, 80, "PO Numbers", "", Alignment.MiddleLeft);
  168. columns.Add<JobRequisitionItem, JobRequisitionItemStatus>(x => x.Status, 80, "Status", "", Alignment.MiddleLeft);
  169. columns.Add<JobRequisitionItem, string>(x => x.Notes, 300, "Notes", "", Alignment.MiddleLeft);
  170. columns.AddRange(base.GenerateColumns());
  171. return columns;
  172. }
  173. }