RequisitionItemGrid.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366
  1. using System;
  2. using System.Collections.Generic;
  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 InABox.WPF;
  13. using Microsoft.Office.Interop.Outlook;
  14. using Syncfusion.Windows.Tools.Controls;
  15. using Exception = System.Exception;
  16. namespace PRSDesktop
  17. {
  18. public class RequisitionItemGrid : DynamicDataGrid<RequisitionItem>
  19. {
  20. //public int Boxes { get; set; }
  21. private static readonly BitmapImage tick = PRSDesktop.Resources.tick.AsBitmapImage();
  22. public RequisitionItemGrid()
  23. {
  24. HiddenColumns.Add(x => x.Code);
  25. HiddenColumns.Add(x => x.BarCode);
  26. HiddenColumns.Add(x => x.RequisitionLink.ID);
  27. HiddenColumns.Add(x => x.RequisitionLink.JobLink.ID);
  28. HiddenColumns.Add(x => x.RequisitionLink.JobScope.ID);
  29. HiddenColumns.Add(x => x.RequisitionLink.Filled);
  30. HiddenColumns.Add(x => x.RequisitionLink.Archived);
  31. HiddenColumns.Add(x => x.Product.ID);
  32. HiddenColumns.Add(x => x.Product.Deleted);
  33. HiddenColumns.Add(x => x.Product.DefaultInstance.Style.ID);
  34. foreach(var column in new Columns<RequisitionItem>()
  35. .AddDimensionsColumns(x => x.Product.DefaultInstance.Dimensions, Dimensions.ColumnsType.Data)
  36. .GetColumns())
  37. {
  38. HiddenColumns.Add(column);
  39. }
  40. //HiddenColumns.Add(x => x.Product.Units.ID);
  41. HiddenColumns.Add(x => x.Product.NonStock);
  42. HiddenColumns.Add(x => x.Location.ID);
  43. HiddenColumns.Add(x => x.Location.Code);
  44. HiddenColumns.Add(x => x.Location.Description);
  45. HiddenColumns.Add(x => x.Location.Deleted);
  46. HiddenColumns.Add(x => x.Style.ID);
  47. HiddenColumns.Add(x => x.Style.Code);
  48. HiddenColumns.Add(x => x.Picked);
  49. ActionColumns.Add(new DynamicImageColumn(Pick_Image, Pick_Click)
  50. {
  51. ToolTip = Pick_ToolTip
  52. });
  53. ActionColumns.Add(new DynamicMenuColumn(SelectHolding,
  54. (row) => (row.Get<RequisitionItem, Guid>(c => c.Product.ID) == Guid.Empty) || row.Get<RequisitionItem, bool>(c => c.Product.NonStock) == true
  55. ? DynamicMenuStatus.Hidden
  56. : DynamicMenuStatus.Enabled)
  57. );
  58. }
  59. private FrameworkElement? Pick_ToolTip(DynamicActionColumn column, CoreRow? row)
  60. {
  61. string content;
  62. if (row is null)
  63. {
  64. content = "Has this item been picked?";
  65. }
  66. else
  67. {
  68. if(!row.Get<RequisitionItem, DateTime>(x => x.Picked).IsEmpty())
  69. {
  70. content = "This item has been picked";
  71. }
  72. else
  73. {
  74. content = "This item has not been picked";
  75. }
  76. }
  77. return column.TextToolTip(content);
  78. }
  79. private BitmapImage? Pick_Image(CoreRow? row)
  80. {
  81. return row is null
  82. ? tick
  83. : (row.Get<RequisitionItem, DateTime>(x => x.Picked).IsEmpty()
  84. ? null
  85. : tick);
  86. }
  87. private bool Pick_Click(CoreRow? row)
  88. {
  89. if(row is null)
  90. {
  91. return false;
  92. }
  93. var reqItem = row.ToObject<RequisitionItem>();
  94. return ProcessItems(reqItem.Picked.IsEmpty() ? DateTime.Now : DateTime.MinValue, CoreUtils.One(reqItem));
  95. }
  96. private static bool ProcessItems(DateTime picked, IEnumerable<RequisitionItem> items)
  97. {
  98. var list = new List<RequisitionItem>();
  99. foreach (var item in items)
  100. {
  101. item.Picked = picked;
  102. if (!picked.IsEmpty())
  103. {
  104. var quantity = item.Quantity;
  105. if (!DoubleEdit.Execute("Enter actual quantity picked:", 0, double.MaxValue, ref quantity))
  106. {
  107. continue;
  108. }
  109. item.ActualQuantity = quantity;
  110. }
  111. else
  112. item.ActualQuantity = 0;
  113. list.Add(item);
  114. }
  115. if(list.Count > 0)
  116. {
  117. string audittrail;
  118. if (picked == DateTime.MinValue)
  119. audittrail = "Item unpicked";
  120. else
  121. audittrail = "Item picked " + picked.ToString("dd MMM yy");
  122. Client.Save(list, audittrail);
  123. return true;
  124. }
  125. else
  126. {
  127. return false;
  128. }
  129. }
  130. protected override void DoReconfigure(FluentList<DynamicGridOption> options)
  131. {
  132. base.DoReconfigure(options);
  133. options.BeginUpdate()
  134. .Add(DynamicGridOption.RecordCount)
  135. .Add(DynamicGridOption.SelectColumns)
  136. .Add(DynamicGridOption.AddRows)
  137. .Add(DynamicGridOption.EditRows)
  138. .Add(DynamicGridOption.DeleteRows)
  139. .Add(DynamicGridOption.FilterRows)
  140. .Add(DynamicGridOption.MultiSelect)
  141. .Add(DynamicGridOption.DirectEdit)
  142. .Add(DynamicGridOption.DragTarget)
  143. .EndUpdate();
  144. }
  145. private bool CanAddItems() =>
  146. Requisition is not null && Requisition.ID != Guid.Empty && Requisition.Filled.IsEmpty() && Security.CanEdit<Requisition>() && Security.CanEdit<RequisitionItem>();
  147. protected override void HandleDragOver(object sender, DragEventArgs e)
  148. {
  149. base.HandleDragOver(sender, e);
  150. if (e.Data.GetDataPresent(typeof(Product)))
  151. {
  152. if (e.Data.GetData(typeof(Product)) is Product product)
  153. {
  154. if (!CanAddItems())
  155. {
  156. e.Effects = DragDropEffects.None;
  157. }
  158. }
  159. }
  160. }
  161. protected override void HandleDragDrop(object sender, DragEventArgs e)
  162. {
  163. base.HandleDragDrop(sender, e);
  164. if (e.Data.GetDataPresent(typeof(Product)))
  165. {
  166. if(e.Data.GetData(typeof(Product)) is Product product)
  167. {
  168. if(CanAddItems())
  169. {
  170. int quantity = 1;
  171. if (NumberEdit.Execute("Enter Quantity:", 1, int.MaxValue, ref quantity))
  172. {
  173. var item = CreateItem();
  174. item.Product.ID = product.ID;
  175. item.Product.Synchronise(product);
  176. item.Quantity = quantity;
  177. item.Picked = DateTime.MinValue;
  178. SaveItem(item);
  179. Refresh(false, true);
  180. }
  181. }
  182. }
  183. }
  184. }
  185. private void SelectHolding(DynamicMenuColumn column, CoreRow? row)
  186. {
  187. if(row is null)
  188. {
  189. return;
  190. }
  191. var locations = new List<Guid>();
  192. var holdings = Client.Query(
  193. new Filter<StockHolding>(x => x.Product.ID).IsEqualTo(row.Get<RequisitionItem, Guid>(c => c.Product.ID)),
  194. new Columns<StockHolding>(x => x.Location.ID)
  195. .Add(x => x.Units));
  196. foreach (var holding in holdings.Rows)
  197. {
  198. var qty = holding.Get<StockHolding, double>(c => c.Units);
  199. if (!CoreUtils.IsEffectivelyEqual(qty, 0.0F) && qty > 0)
  200. locations.Add(holding.Get<StockHolding, Guid>(x => x.Location.ID));
  201. }
  202. if (locations.Count == 0)
  203. {
  204. MessageWindow.ShowMessage("No valid holdings found for product", "No holdings");
  205. return;
  206. }
  207. var selection = new MultiSelectDialog<StockHolding>
  208. (
  209. new Filter<StockHolding>(x => x.Product.ID).IsEqualTo(row.Get<RequisitionItem, Guid>(c => c.Product.ID))
  210. .And(x => x.Location.ID).InList(locations.ToArray())
  211. ,
  212. new Columns<StockHolding>(
  213. x => x.Job.Name,
  214. x => x.Job.JobNumber,
  215. x => x.Units,
  216. x => x.Dimensions.UnitSize,
  217. x => x.Dimensions.Height,
  218. x => x.Dimensions.Width,
  219. x => x.Dimensions.Weight,
  220. x => x.Dimensions.Quantity,
  221. x => x.Dimensions.Length,
  222. x => x.Dimensions.Unit.ID,
  223. x => x.Dimensions.Unit.Format,
  224. x => x.Dimensions.Unit.Formula,
  225. x => x.Dimensions.Unit.HasHeight,
  226. x => x.Dimensions.Unit.HasWeight,
  227. x => x.Dimensions.Unit.HasWidth,
  228. x => x.Dimensions.Unit.HasQuantity,
  229. x => x.Dimensions.Unit.HasLength,
  230. x => x.Style.ID,
  231. x => x.Style.Code,
  232. x => x.Style.Description,
  233. x => x.Location.ID,
  234. x => x.Location.Code,
  235. x => x.Location.Description,
  236. x => x.Location.Area.Code,
  237. x => x.Location.Area.Description
  238. ), false);
  239. if (selection.ShowDialog("Units", "0", Syncfusion.Data.FilterType.GreaterThan) == true)
  240. SelectLocation(selection.Data().Rows.FirstOrDefault(), row);
  241. }
  242. private void SelectLocation(CoreRow holdingrow, CoreRow itemrow)
  243. {
  244. var item = itemrow.ToObject<RequisitionItem>();
  245. var holding = holdingrow.ToObject<StockHolding>();
  246. item.Location.ID = holding.Location.ID;
  247. item.Location.Code = holding.Location.Code;
  248. item.Dimensions.CopyFrom(holding.Dimensions, true);
  249. item.Style.ID = holding.Style.ID;
  250. item.Style.Code = holding.Style.Code;
  251. new Client<RequisitionItem>().Save(item, "Changed due to stock holding selection");
  252. Data.LoadRow(itemrow, item);
  253. InvalidateRow(itemrow);
  254. }
  255. public Requisition? Requisition { get; set; }
  256. protected override void Reload(Filters<RequisitionItem> criteria, Columns<RequisitionItem> columns, ref SortOrder<RequisitionItem>? sort,
  257. Action<CoreTable?, Exception?> action)
  258. {
  259. criteria.Add(
  260. new Filter<RequisitionItem>(x => x.RequisitionLink.ID).IsEqualTo(Requisition != null ? Requisition.ID : CoreUtils.FullGuid));
  261. sort = new SortOrder<RequisitionItem>(x => x.Created);
  262. base.Reload(
  263. criteria,
  264. columns,
  265. ref sort,
  266. action
  267. );
  268. }
  269. protected override bool CanDeleteItems(CoreRow[] rows)
  270. {
  271. if (Requisition == null || Requisition.ID.Equals(Guid.Empty))
  272. {
  273. MessageBox.Show("Please select a Requisition first!");
  274. return false;
  275. }
  276. if (!Requisition.Filled.IsEmpty())
  277. {
  278. MessageBox.Show("Cannot Modify a Completed Requisition");
  279. return false;
  280. }
  281. return base.CanDeleteItems(rows);
  282. }
  283. public override bool EditItems(RequisitionItem[] items, Func<Type, CoreTable?>? PageDataHandler, bool PreloadPages = false)
  284. {
  285. if (Requisition == null || Requisition.ID.Equals(Guid.Empty))
  286. {
  287. MessageBox.Show("Please select a Requisition first!");
  288. return false;
  289. }
  290. if (!Requisition.StockUpdated.IsEmpty())
  291. {
  292. MessageBox.Show("Cannot Edit Items after Stock Holdings have been updated!");
  293. return false;
  294. }
  295. return base.EditItems(items, PageDataHandler, PreloadPages);
  296. }
  297. protected override RequisitionItem CreateItem()
  298. {
  299. var item = base.CreateItem();
  300. item.RequisitionLink.ID = Requisition?.ID ?? Guid.Empty;
  301. item.RequisitionLink.Synchronise(Requisition);
  302. item.Quantity = 1;
  303. return item;
  304. }
  305. protected override void DoAdd(bool OpenEditorOnDirectEdit = false)
  306. {
  307. if (Requisition == null || Requisition.ID.Equals(Guid.Empty))
  308. {
  309. MessageBox.Show("Please select a Requisition first!");
  310. return;
  311. }
  312. if (!Requisition.Filled.IsEmpty())
  313. {
  314. MessageBox.Show("Cannot Add Items to a Completed Requisition");
  315. return;
  316. }
  317. base.DoAdd();
  318. }
  319. }
  320. }