RequisitionGrid.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Threading;
  5. using System.Windows;
  6. using System.Windows.Controls;
  7. using System.Windows.Media;
  8. using System.Windows.Media.Imaging;
  9. using Comal.Classes;
  10. using InABox.Clients;
  11. using InABox.Configuration;
  12. using InABox.Core;
  13. using InABox.DynamicGrid;
  14. using InABox.Wpf;
  15. using InABox.WPF;
  16. using Color = System.Drawing.Color;
  17. namespace PRSDesktop
  18. {
  19. public class RequisitionSettings : IUserConfigurationSettings
  20. {
  21. public DynamicGridSelectedFilterSettings Filters { get; set; } = new();
  22. }
  23. internal class RequisitionGrid : DynamicDataGrid<Requisition>
  24. {
  25. //public Dictionary<Object, Object> _Employees = new Dictionary<object, object>();
  26. //public Dictionary<Object, Object> _Jobs = new Dictionary<object, object>();
  27. private readonly BitmapImage barcode = PRSDesktop.Resources.barcode.AsBitmapImage();
  28. private RequisitionSettings _settings;
  29. private bool bSplitting;
  30. private readonly BitmapImage docs = PRSDesktop.Resources.doc_png.AsBitmapImage();
  31. private readonly BitmapImage forklift = PRSDesktop.Resources.forklift.AsBitmapImage(Color.White);
  32. private readonly BitmapImage printer = PRSDesktop.Resources.printer.AsBitmapImage();
  33. private readonly BitmapImage tick = PRSDesktop.Resources.tick.AsBitmapImage(Color.White);
  34. private readonly BitmapImage truck = PRSDesktop.Resources.truck.AsBitmapImage();
  35. public RequisitionGrid()
  36. {
  37. _settings = new UserConfiguration<RequisitionSettings>().Load();
  38. FilterComponent.SetSettings(_settings.Filters, false);
  39. FilterComponent.OnFiltersSelected += FilterComponent_OnFilterSelected;
  40. ActionColumns.Add(new DynamicImageColumn(DocumentsImage, DocumentsClick) { Position = DynamicActionColumnPosition.Start });
  41. ActionColumns.Add(new DynamicImageColumn(FilledImage));
  42. ActionColumns.Add(new DynamicImageColumn(DeliveryImage));
  43. ActionColumns.Add(new DynamicImageColumn(StockImage));
  44. //ActionColumns.Add(new DynamicImageColumn() { Action = LabelClick, Image = GetLabelImage });
  45. //ActionColumns.Add(new DynamicImageColumn() { Action = DeliveryDocketClick, Image = GetPrinterImage });
  46. HiddenColumns.Add(x => x.JobLink.ID);
  47. HiddenColumns.Add(x => x.JobLink.JobNumber);
  48. HiddenColumns.Add(x => x.JobLink.Name);
  49. HiddenColumns.Add(x => x.JobScope.ID);
  50. HiddenColumns.Add(x => x.Boxes);
  51. HiddenColumns.Add(x => x.Filled);
  52. HiddenColumns.Add(x => x.Title);
  53. HiddenColumns.Add(x => x.Request);
  54. HiddenColumns.Add(x => x.Notes);
  55. HiddenColumns.Add(x => x.Number);
  56. HiddenColumns.Add(x => x.Employee.ID);
  57. HiddenColumns.Add(x => x.Employee.Name);
  58. HiddenColumns.Add(x => x.Due);
  59. HiddenColumns.Add(x => x.RequestedBy.ID);
  60. HiddenColumns.Add(x => x.RequestedBy.Name);
  61. HiddenColumns.Add(x => x.Documents);
  62. HiddenColumns.Add(x => x.Archived);
  63. HiddenColumns.Add(x => x.TakenBy.ID);
  64. HiddenColumns.Add(x => x.TakenBy.Name);
  65. HiddenColumns.Add(x => x.Delivery.ID);
  66. HiddenColumns.Add(x => x.Delivery.Number);
  67. HiddenColumns.Add(x => x.Delivery.Completed);
  68. HiddenColumns.Add(x => x.StockUpdated);
  69. //CoreTable employees = new Client<Employee>().Query(
  70. // null,
  71. // new Columns<Employee>(x => x.ID, x => x.Name),
  72. // new SortOrder<Employee>(x => x.Name)
  73. //);
  74. //foreach (CoreRow row in employees.Rows)
  75. // _Employees[row.Get<Employee, Guid>(x => x.ID)] = row.Get<Employee, String>(x => x.Name);
  76. //Job[] jobs = new Client<Job>().Load(null, new SortOrder<Job>(x => x.JobNumber));
  77. //foreach (Job job in jobs)
  78. //_Jobs[job] = job.ToString();
  79. AddButton("-", PRSDesktop.Resources.box.AsBitmapImage(Color.White), DelBoxClick);
  80. AddButton("+", PRSDesktop.Resources.box.AsBitmapImage(Color.White), AddBoxClick);
  81. AddButton("Split", PRSDesktop.Resources.split.AsBitmapImage(), SplitRequiClick);
  82. //AddButton("Labels", PRSDesktop.Resources.barcode.AsBitmapImage(System.Drawing.Color.White), LabelClick);
  83. //AddButton("Del Dkt", PRSDesktop.Resources.printer.AsBitmapImage(System.Drawing.Color.White), DeliveryDocketClick);
  84. //AddButton("Complete", PRSDesktop.Resources.tick.AsBitmapImage(System.Drawing.Color.White), FilledClick);
  85. OnCustomiseEditor += CustomiseEditor;
  86. }
  87. private void FilterComponent_OnFilterSelected(DynamicGridSelectedFilterSettings settings)
  88. {
  89. _settings.Filters = settings;
  90. new UserConfiguration<RequisitionSettings>().Save(_settings);
  91. }
  92. protected override void DoReconfigure(DynamicGridOptions options)
  93. {
  94. base.DoReconfigure(options);
  95. options.SelectColumns = true;
  96. options.AddRows = true;
  97. options.EditRows = true;
  98. options.FilterRows = true;
  99. if (Security.IsAllowed<CanDeleteStoresRequisitions>())
  100. options.DeleteRows = true;
  101. }
  102. public CoreTable Requisitions { get; private set; }
  103. private void CustomiseEditor(IDynamicEditorForm sender, Requisition[]? items, DynamicGridColumn column, BaseEditor editor)
  104. {
  105. if (column.ColumnName.Equals("Notes"))
  106. {
  107. if (editor is NotesEditor notes)
  108. notes.AlwaysEnabled = bSplitting;
  109. }
  110. else if (column.ColumnName.Equals("Filled"))
  111. {
  112. editor.Editable = Security.IsAllowed<CanSkipRequisitionPhotos>() || (items != null && items.Any() && items.First().Documents > 0)
  113. ? Editable.Enabled
  114. : Editable.Hidden;
  115. }
  116. else if (column.ColumnName.Equals("TakenBy.ID"))
  117. {
  118. editor.Editable = items != null && items.Any() && !items.First().Filled.IsEmpty() ? Editable.Enabled : Editable.Disabled;
  119. }
  120. }
  121. protected override Dictionary<string, object?> EditorValueChanged(IDynamicEditorForm editor, Requisition[] items, string name, object value)
  122. {
  123. var result = base.EditorValueChanged(editor, items, name, value);
  124. if (name.Equals("TakenBy.ID") && (value == null || Equals(value, Guid.Empty)))
  125. editor.SetEditorValue("Archived", DateTime.MinValue);
  126. else if (name.Equals("JobLink.ID"))
  127. {
  128. // false here because a job has a defaultscope
  129. // and we need to load the lookups before we set the default value
  130. var scope = editor.FindEditor("JobScope.ID") as ILookupEditorControl;
  131. if (scope != null)
  132. DefineLookups(scope,items,false);
  133. }
  134. return result;
  135. }
  136. private bool DocumentsClick(CoreRow? arg)
  137. {
  138. if (arg == null)
  139. return false;
  140. var docs = new List<IEntityDocument>();
  141. using (new WaitCursor())
  142. {
  143. var deliveryid = arg.Get<Requisition, Guid>(x => x.ID);
  144. var table = new Client<RequisitionDocument>().Query(
  145. new Filter<RequisitionDocument>(x => x.EntityLink.ID).IsEqualTo(deliveryid)
  146. );
  147. foreach (var row in table.Rows)
  148. docs.Add(row.ToObject<RequisitionDocument>());
  149. }
  150. if (docs.Any())
  151. {
  152. var editor = new DocumentEditor(docs.ToArray());
  153. //editor.PrintAllowed = Security.IsAllowed<CanPrintFactoryFloorDrawings>();
  154. editor.SaveAllowed = Security.IsAllowed<CanSaveFactoryFloorDrawings>();
  155. editor.ShowDialog();
  156. }
  157. else
  158. {
  159. MessageBox.Show("No Documents Available!");
  160. }
  161. return false;
  162. }
  163. private BitmapImage? DocumentsImage(CoreRow? arg)
  164. {
  165. if (arg is null)
  166. return docs;
  167. return arg.Get<Requisition, int>(x => x.Documents) > 0 ? docs : null;
  168. }
  169. private bool SplitRequiClick(Button sender, CoreRow[] rows)
  170. {
  171. var result = false;
  172. if ((rows?.Length ?? 0) != 1)
  173. {
  174. MessageBox.Show("Please select a single Picking List to Split!");
  175. return result;
  176. }
  177. var confirm = MessageWindow.ShowYesNoCancel("Do you wish to move unpicked items to the new picking list?","Confirm");
  178. if (confirm == MessageWindowResult.Cancel)
  179. return false;
  180. bSplitting = true;
  181. if (SplitPickingList(rows![0], confirm == MessageWindowResult.Yes, (r) => EditItems(new Requisition[] { r }) ))
  182. result = true;
  183. bSplitting = false;
  184. return result;
  185. }
  186. public bool SplitPickingList(CoreRow row, bool moveunpickeditems, Func<Requisition,bool> edit, IProgress<string>? progress = null)
  187. {
  188. var oldrequi = row.ToObject<Requisition>();
  189. var newrequi = row.ToObject<Requisition>();
  190. newrequi.ID = Guid.Empty;
  191. newrequi.Number = 0;
  192. newrequi.CommitChanges();
  193. newrequi.Filled = DateTime.MinValue;
  194. List<RequisitionItem> unpicked = moveunpickeditems
  195. ? Client.Query<RequisitionItem>(
  196. new Filter<RequisitionItem>(x => x.RequisitionLink.ID).IsEqualTo(oldrequi.ID)
  197. .And(x => x.ActualQuantity).IsEqualTo(0.0),
  198. Columns.Required<RequisitionItem>()
  199. ).Rows.Select(x => x.ToObject<RequisitionItem>()).ToList()
  200. : new List<RequisitionItem>();
  201. newrequi.Notes = new[]
  202. {
  203. string.Format("Items unavailable on Picking List #{0}:\n{1}{2}",
  204. oldrequi,
  205. string.Join("\n",unpicked.Select(x=>$"{x.Quantity} x {x.Description}")),
  206. string.Join("\n\n==============================", oldrequi.Notes)
  207. )
  208. };
  209. if (edit(newrequi))
  210. {
  211. progress?.Report("Creating new Picking List");
  212. bSplitting = false;
  213. new Client<Requisition>().Save(newrequi, "Created by Splitting Picking List #" + oldrequi.Number);
  214. if (unpicked.Any())
  215. {
  216. progress?.Report("Moving unpicked items");
  217. foreach (var item in unpicked)
  218. item.RequisitionLink.ID = newrequi.ID;
  219. Client.Save(unpicked,$"Moved from {oldrequi.Number} to {newrequi.Number}");
  220. }
  221. progress?.Report("Updating Original Picking List");
  222. var _notes = oldrequi.Notes.ToList();
  223. _notes.Insert(0,
  224. $"Created secondary Picking List #{newrequi.Number} for further action:\n\nOriginal Picking List Notes:\n");
  225. oldrequi.Notes = _notes.ToArray();
  226. new Client<Requisition>().Save(oldrequi, "Split Picking List to #" + newrequi);
  227. return true;
  228. }
  229. return false;
  230. }
  231. private bool AddBoxClick(Button btn, CoreRow[] rows)
  232. {
  233. if (rows.Length != 1)
  234. {
  235. MessageBox.Show("Please select one row to process!");
  236. return false;
  237. }
  238. var row = rows.First();
  239. var id = row.Get<Requisition, Guid>(x => x.ID);
  240. var req = new Client<Requisition>().Load(
  241. new Filter<Requisition>(x => x.ID).IsEqualTo(id)
  242. ).FirstOrDefault();
  243. if (req != null)
  244. {
  245. req.Boxes++;
  246. new Client<Requisition>().Save(req, string.Format("Set Number of Boxes to {0}", req.Boxes));
  247. //OnRequisitionBoxesChanged?.Invoke(req.ID, req.Boxes);
  248. return true;
  249. }
  250. MessageBox.Show("Cannot locate Requisition");
  251. return true;
  252. }
  253. private bool DelBoxClick(Button btn, CoreRow[] rows)
  254. {
  255. if (rows.Length != 1)
  256. {
  257. MessageBox.Show("Please select one row to process!");
  258. return false;
  259. }
  260. var row = rows.First();
  261. var id = row.Get<Requisition, Guid>(x => x.ID);
  262. var req = new Client<Requisition>().Load(
  263. new Filter<Requisition>(x => x.ID).IsEqualTo(id)
  264. ).FirstOrDefault();
  265. if (req != null)
  266. {
  267. if (req.Boxes > 0)
  268. {
  269. req.Boxes--;
  270. new Client<Requisition>().Save(req, string.Format("Set Number of Boxes to {0}", req.Boxes));
  271. //OnRequisitionBoxesChanged?.Invoke(req.ID, req.Boxes);
  272. }
  273. return true;
  274. }
  275. MessageBox.Show("Cannot Locate Requisition!");
  276. return true;
  277. }
  278. public override Requisition CreateItem()
  279. {
  280. var requi = base.CreateItem();
  281. var role = new Client<Role>().Load(new Filter<Role>(x => x.Code).IsEqualTo("STORES")).FirstOrDefault();
  282. if (role != null)
  283. {
  284. var emprole = new Client<EmployeeRole>().Load(new Filter<EmployeeRole>(x => x.RoleLink.ID).IsEqualTo(role.ID)).FirstOrDefault();
  285. if (emprole != null)
  286. requi.Employee.ID = emprole.EmployeeLink.ID;
  287. }
  288. return requi;
  289. }
  290. private BitmapImage? GetBitmapImage(CoreRow? row, BitmapImage image)
  291. {
  292. if (row == null)
  293. return image;
  294. var filled = row.Get<Requisition, DateTime>(x => x.Filled);
  295. return filled.IsEmpty() ? null : image;
  296. }
  297. private BitmapImage? GetLabelImage(CoreRow row)
  298. {
  299. return GetBitmapImage(row, barcode);
  300. }
  301. private BitmapImage? GetPrinterImage(CoreRow row)
  302. {
  303. return GetBitmapImage(row, printer);
  304. }
  305. private BitmapImage? FilledImage(CoreRow? row)
  306. {
  307. if (row == null)
  308. return tick;
  309. var filled = row.Get<Requisition, DateTime>(x => x.Filled);
  310. return filled.IsEmpty() ? null : tick;
  311. }
  312. private BitmapImage? StockImage(CoreRow? row)
  313. {
  314. if (row == null)
  315. return forklift;
  316. var stockupdated = row.Get<Requisition, DateTime>(x => x.StockUpdated);
  317. return stockupdated.IsEmpty() ? null : forklift;
  318. }
  319. private BitmapImage? DeliveryImage(CoreRow? row)
  320. {
  321. if (row == null)
  322. return truck;
  323. var archived = row.Get<Requisition, DateTime>(x => x.Archived);
  324. return archived.IsEmpty() ? null : truck;
  325. }
  326. private void SendNotifications(CoreRow row)
  327. {
  328. var updates = new List<Notification>();
  329. var roles = new Client<EmployeeRole>().Query(new Filter<EmployeeRole>(x => x.RoleLink.Code).IsEqualTo("DELIVERIES"));
  330. foreach (var role in roles.Rows)
  331. {
  332. var notification = new Notification
  333. {
  334. Title = string.Format("Requi #{0} is ready", row.Get<Requisition, int>(x => x.Number)),
  335. Description = "The requisition has been packed and labelled, and is ready to be assigned to a Delivery"
  336. };
  337. notification.Sender.ID = row.Get<Requisition, Guid>(x => x.Employee.ID);
  338. notification.Employee.ID = role.Get<EmployeeRole, Guid>(x => x.EmployeeLink.ID);
  339. notification.Job.ID = row.Get<Requisition, Guid>(x => x.JobLink.ID);
  340. updates.Add(notification);
  341. }
  342. new Client<Notification>().Save(updates, "Sent Notification");
  343. }
  344. protected override void Reload(
  345. Filters<Requisition> criteria, Columns<Requisition> columns, ref SortOrder<Requisition>? sort,
  346. CancellationToken token, Action<CoreTable?, Exception?> action)
  347. {
  348. sort = new SortOrder<Requisition>(x => x.Number, SortDirection.Descending);
  349. base.Reload(criteria, columns, ref sort, token, action);
  350. }
  351. }
  352. }