FactoryPackGrid.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Linq.Expressions;
  5. using System.Threading;
  6. using System.Threading.Tasks;
  7. using System.Windows;
  8. using System.Windows.Controls;
  9. using System.Windows.Data;
  10. using System.Windows.Media;
  11. using Comal.Classes;
  12. using InABox.Clients;
  13. using InABox.Configuration;
  14. using InABox.Core;
  15. using InABox.DynamicGrid;
  16. using InABox.WPF;
  17. using Document = InABox.Core.Document;
  18. namespace PRSDesktop;
  19. public class FactoryPackGrid : DynamicDataGrid<StockHolding>
  20. {
  21. public StockArea? Area { get; set; }
  22. public ManufacturingPacket? CurrentPacket { get; set; }
  23. private readonly Dictionary<Guid, byte[]> _images = new();
  24. private readonly DynamicActionColumn _holding;
  25. public FactoryPackGrid()
  26. {
  27. HiddenColumns.Add(x=>x.ID);
  28. HiddenColumns.Add(x=>x.Product.ID);
  29. HiddenColumns.Add(x=>x.Product.Code);
  30. HiddenColumns.Add(x=>x.Product.Name);
  31. HiddenColumns.Add(x=>x.Product.Image.ID);
  32. HiddenColumns.Add(x => x.Dimensions.Unit.ID);
  33. HiddenColumns.Add(x => x.Dimensions.Unit.Formula);
  34. HiddenColumns.Add(x => x.Dimensions.Unit.Format);
  35. HiddenColumns.Add(x=>x.Dimensions.Height);
  36. HiddenColumns.Add(x=>x.Dimensions.Width);
  37. HiddenColumns.Add(x=>x.Dimensions.Length);
  38. HiddenColumns.Add(x=>x.Dimensions.Weight);
  39. HiddenColumns.Add(x=>x.Dimensions.Quantity);
  40. HiddenColumns.Add(x=>x.Dimensions.Value);
  41. HiddenColumns.Add(x=>x.Dimensions.UnitSize);
  42. HiddenColumns.Add(x=>x.Style.ID);
  43. HiddenColumns.Add(x=>x.Style.Code);
  44. HiddenColumns.Add(x=>x.Style.Description);
  45. HiddenColumns.Add(x => x.Job.ID);
  46. HiddenColumns.Add(x=>x.Job.JobNumber);
  47. HiddenColumns.Add(x=>x.Job.Name);
  48. HiddenColumns.Add(x=>x.Location.ID);
  49. HiddenColumns.Add(x=>x.Location.Code);
  50. HiddenColumns.Add(x=>x.Location.Description);
  51. HiddenColumns.Add(x=>x.Units);
  52. HiddenColumns.Add(x => x.Available);
  53. _holding = new DynamicTemplateColumn(PackTemplate) { Width = 0 };
  54. ActionColumns.Add(_holding);
  55. }
  56. protected override void DoReconfigure(DynamicGridOptions options)
  57. {
  58. base.DoReconfigure(options);
  59. options.Clear();
  60. RowHeight = 85;
  61. }
  62. protected override DynamicGridColumns LoadColumns()
  63. {
  64. var columns = new DynamicGridColumns();
  65. return columns;
  66. }
  67. protected override void Reload(
  68. Filters<StockHolding> criteria, Columns<StockHolding> columns, ref SortOrder<StockHolding>? sort,
  69. CancellationToken token, Action<CoreTable?, Exception?> action)
  70. {
  71. Task<CoreTable> imagetask = Task.Run(() =>
  72. {
  73. if (Area == null)
  74. return new Client<Document>().Query(new Filter<Document>().None());
  75. return new Client<Document>().Query(
  76. new Filter<Document>(x => x.ID)
  77. .InQuery(new Filter<StockHolding>(x => x.Location.Area.ID).IsEqualTo(Area.ID),
  78. x => x.Product.Image.ID),
  79. Columns.None<Document>().Add(x => x.ID).Add(x => x.Data)
  80. );
  81. });
  82. criteria.Add(Area == null
  83. ? new Filter<StockHolding>().None()
  84. : new Filter<StockHolding>(x => x.Location.Area.ID).IsEqualTo(Area.ID));
  85. base.Reload(criteria, columns, ref sort, token, (o,e) =>
  86. {
  87. if (o != null)
  88. {
  89. Task.WaitAll(imagetask);
  90. _images.Clear();
  91. imagetask.Result.LoadDictionary<Document, Guid, byte[]>(
  92. _images,
  93. x => x.ID,
  94. x => x.Data
  95. );
  96. }
  97. action(o,e);
  98. });
  99. }
  100. private void CreateStackPanel(Grid grid, int row, int column, FontWeight weight,
  101. params Expression<Func<StockHolding, object>>[] bindings)
  102. {
  103. StackPanel result = new StackPanel()
  104. {
  105. Orientation = Orientation.Horizontal,
  106. HorizontalAlignment = HorizontalAlignment.Center,
  107. };
  108. result.SetValue(Grid.RowProperty,row);
  109. result.SetValue(Grid.ColumnProperty,column);
  110. foreach (var binding in bindings)
  111. {
  112. Label label = new Label()
  113. {
  114. Margin = new Thickness(5,0,0,0),
  115. VerticalContentAlignment = VerticalAlignment.Center,
  116. FontWeight = weight
  117. };
  118. label.Bind(Label.ContentProperty,binding);
  119. result.Children.Add(label);
  120. }
  121. grid.Children.Add(result);
  122. }
  123. private FrameworkElement PackTemplate(CoreRow row)
  124. {
  125. var grid = new Grid();
  126. grid.RowDefinitions.Add(new RowDefinition() { Height = new GridLength(1, GridUnitType.Star) });
  127. grid.RowDefinitions.Add(new RowDefinition() { Height = new GridLength(1, GridUnitType.Star) });
  128. grid.RowDefinitions.Add(new RowDefinition() { Height = new GridLength(1, GridUnitType.Star) });
  129. grid.ColumnDefinitions.Add(new ColumnDefinition() { Width= new GridLength(28, GridUnitType.Pixel)});
  130. grid.ColumnDefinitions.Add(new ColumnDefinition() { Width= new GridLength(1, GridUnitType.Pixel)});
  131. grid.ColumnDefinitions.Add(new ColumnDefinition() { Width= new GridLength(90, GridUnitType.Pixel)});
  132. grid.ColumnDefinitions.Add(new ColumnDefinition() { Width= new GridLength(1, GridUnitType.Star)});
  133. grid.ColumnDefinitions.Add(new ColumnDefinition() { Width= new GridLength(70, GridUnitType.Pixel)});
  134. Label label = new Label()
  135. {
  136. HorizontalAlignment = HorizontalAlignment.Center,
  137. VerticalAlignment = VerticalAlignment.Center,
  138. LayoutTransform = new RotateTransform(-90)
  139. };
  140. label.Bind<StockHolding,String>(Label.ContentProperty, x=>x.Location.Code);
  141. label.SetValue(Grid.RowProperty,0);
  142. label.SetValue(Grid.RowSpanProperty,3);
  143. label.SetValue(Grid.ColumnProperty,0);
  144. grid.Children.Add(label);
  145. Border border = new Border()
  146. {
  147. BorderThickness = new Thickness(0.75),
  148. BorderBrush = new SolidColorBrush(Colors.DarkGray)
  149. };
  150. border.SetValue(Grid.RowProperty,0);
  151. border.SetValue(Grid.RowSpanProperty,3);
  152. border.SetValue(Grid.ColumnProperty,1);
  153. grid.Children.Add(border);
  154. Image image = new Image()
  155. {
  156. Stretch = Stretch.Uniform,
  157. Margin=new Thickness(5)
  158. };
  159. image.Bind<StockHolding,Guid>(Image.SourceProperty, x => x.Product.Image.ID, new GuidToImageSourceConverter() { Images = _images});
  160. image.SetValue(Grid.RowProperty,0);
  161. image.SetValue(Grid.RowSpanProperty,3);
  162. image.SetValue(Grid.ColumnProperty,2);
  163. grid.Children.Add(image);
  164. CreateStackPanel(grid, 0, 3, FontWeights.Bold, x => x.Product.Code, x => x.Product.Name);
  165. CreateStackPanel(grid, 1, 3, FontWeights.Normal, x => x.Dimensions.UnitSize, x => x.Style.Code, x=>x.Style.Description);
  166. CreateStackPanel(grid, 2, 3, FontWeights.Normal, x => x.Job.JobNumber, x => x.Job.Name);
  167. StackPanel buttonContent = new StackPanel()
  168. {
  169. Orientation = Orientation.Vertical,
  170. VerticalAlignment = VerticalAlignment.Center,
  171. HorizontalAlignment = HorizontalAlignment.Center
  172. };
  173. Label qtyLbl = new Label()
  174. {
  175. FontSize = 16,
  176. FontWeight = FontWeights.Bold,
  177. HorizontalAlignment = HorizontalAlignment.Center
  178. };
  179. qtyLbl.Bind<StockHolding, double>(Label.ContentProperty, x => x.Units, null, BindingMode.Default, "{0:F2}");
  180. buttonContent.Children.Add(qtyLbl);
  181. Label useLbl = new Label()
  182. {
  183. HorizontalAlignment = HorizontalAlignment.Center,
  184. Content = "Use"
  185. };
  186. buttonContent.Children.Add(useLbl);
  187. Button button = new Button()
  188. {
  189. Margin = new Thickness(5),
  190. Content = buttonContent
  191. };
  192. button.Bind<StockHolding,Guid>(Image.SourceProperty, x => x.ID);
  193. button.SetValue(Grid.RowProperty,0);
  194. button.SetValue(Grid.RowSpanProperty,4);
  195. button.SetValue(Grid.ColumnProperty,4);
  196. button.Click += UseStock;
  197. grid.Children.Add(button);
  198. return grid;
  199. }
  200. private class FactoryFloorReturn : BaseObject
  201. {
  202. [EditorSequence(1)]
  203. public ProductLink Product { get; set; }
  204. [EditorSequence(2)]
  205. public ProductStyleLink Style { get; set; }
  206. [EditorSequence(3)]
  207. [DimensionsEditor(typeof(StockDimensions))]
  208. public StockDimensions Dimensions { get; set;}
  209. [EditorSequence(4)]
  210. public double Qty { get; set; }
  211. [EditorSequence(5)]
  212. public JobLink Job { get; set; }
  213. }
  214. private class FactoryFloorIssue : BaseObject
  215. {
  216. public JobRequisitionItem[]? RequisitionItems { get; init; }
  217. [EditorSequence(1)]
  218. [ComboLookupEditor(typeof(RequisitionIDGenerator))]
  219. public Guid RequisitionID { get; set; }
  220. private class RequisitionIDGenerator : LookupGenerator<FactoryFloorIssue>
  221. {
  222. public RequisitionIDGenerator(FactoryFloorIssue[] items) : base(items)
  223. {
  224. }
  225. protected override void DoGenerateLookups()
  226. {
  227. var items = Items?.FirstOrDefault()?.RequisitionItems;
  228. if (items == null)
  229. return;
  230. foreach (var item in items)
  231. {
  232. AddValue(item.ID,
  233. item.ID == Guid.Empty
  234. ? $"{item.Requisition.Description}"
  235. : $"{item.Requisition.Number}: {item.Requisition.Description}");
  236. }
  237. }
  238. }
  239. [EditorSequence(2)]
  240. public double Qty { get; set; }
  241. }
  242. public void ReplaceStock()
  243. {
  244. var holdingrow = SelectedRows.FirstOrDefault();
  245. var receive = holdingrow == null
  246. ? new StockMovement()
  247. : holdingrow.ToObject<StockHolding>().CreateMovement();
  248. receive.Type = StockMovementType.Receive;
  249. receive.Date = DateTime.Now;
  250. receive.Employee.ID = App.EmployeeID;
  251. receive.Notes = "Returned from Manufacturing";
  252. receive.Received = 1.0F;
  253. var smg = new DynamicDataGrid<StockMovement>();
  254. smg.OnCustomiseEditor += StockMovementCustomiseEditor;
  255. smg.OnValidate += StockMovementValidate;
  256. smg.OnEditorValueChanged += StockMovementValueChanged;
  257. if (smg.EditItems(new StockMovement[] { receive }))
  258. {
  259. receive.Batch.ID = CheckStockMovementBatch();
  260. new Client<StockMovement>().Save(receive,"Issued Stock to Factory Floor");
  261. Refresh(false,true);
  262. }
  263. }
  264. private Dictionary<string, object?> StockMovementValueChanged(IDynamicEditorForm form, string name, object value)
  265. {
  266. var result = new Dictionary<string, object?>();
  267. if (name.Equals("Location.Job.ID"))
  268. {
  269. var editor = form.FindEditor("Job.ID");
  270. if (!value.Equals(Guid.Empty))
  271. result = DynamicGridUtils.UpdateEditorValue(form.Items, "Job.ID", value);
  272. else
  273. foreach (StockMovement item in form.Items)
  274. result = DynamicGridUtils.UpdateEditorValue(new[] { item }, "Job.ID",
  275. item.Job.HasOriginalValue("ID") ? item.Job.GetOriginalValue(x => x.ID) : item.Job.ID);
  276. editor.IsEnabled = value.Equals(Guid.Empty);
  277. }
  278. return result;
  279. }
  280. private void StockMovementValidate(object sender, StockMovement[] items, List<string> errors)
  281. {
  282. if (items.Any(x => x.Product.ID == Guid.Empty))
  283. errors.Add("Product may not be blank");
  284. if (items.Any(x => x.Location.ID == Guid.Empty))
  285. errors.Add("Location may not be blank");
  286. if (items.Any(x => x is { Received: 0, Issued: 0 }))
  287. errors.Add("Quantity may not be zero");
  288. }
  289. private void StockMovementCustomiseEditor(IDynamicEditorForm sender, StockMovement[]? items, DynamicGridColumn column, BaseEditor editor)
  290. {
  291. if (column.ColumnName.Equals(CoreUtils.GetFullPropertyName<StockMovement,double>(x=>x.Cost,".")))
  292. editor.Editable = Editable.Hidden;
  293. else if (column.ColumnName.Equals(CoreUtils.GetFullPropertyName<StockMovement,DateTime>(x=>x.Date,".")))
  294. editor.Editable = Editable.Hidden;
  295. else if (column.ColumnName.Equals(CoreUtils.GetFullPropertyName<StockMovement,double>(x=>x.Issued,".")))
  296. editor.Editable = Editable.Hidden;
  297. else if (column.ColumnName.Equals(CoreUtils.GetFullPropertyName<StockMovement,Guid>(x=>x.Employee.ID,".")))
  298. editor.Editable = Editable.Hidden;
  299. else if (column.ColumnName.Equals(CoreUtils.GetFullPropertyName<StockMovement,String>(x=>x.Notes,".")))
  300. editor.Editable = Editable.Hidden;
  301. else if (column.ColumnName.Equals(CoreUtils.GetFullPropertyName<StockMovement,Guid>(x=>x.Location.ID,".")))
  302. editor.EditorSequence = 1;
  303. else if (column.ColumnName.Equals(CoreUtils.GetFullPropertyName<StockMovement,Guid>(x=>x.Product.ID,".")))
  304. editor.EditorSequence = 2;
  305. else if (column.ColumnName.Equals(CoreUtils.GetFullPropertyName<StockMovement,Guid>(x=>x.Style.ID,".")))
  306. editor.EditorSequence = 3;
  307. else if (column.ColumnName.Equals(CoreUtils.GetFullPropertyName<StockMovement,Guid>(x=>x.Job.ID,".")))
  308. editor.EditorSequence = 4;
  309. else if (column.ColumnName.Equals(CoreUtils.GetFullPropertyName<StockMovement,StockDimensions>(x=>x.Dimensions,".")))
  310. editor.EditorSequence = 5;
  311. else if (column.ColumnName.Equals(CoreUtils.GetFullPropertyName<StockMovement, double>(x => x.Received, ".")))
  312. {
  313. editor.EditorSequence = 6;
  314. editor.Caption = "Quantity";
  315. }
  316. }
  317. private static Guid CheckStockMovementBatch()
  318. {
  319. var _settings = new LocalConfiguration<FactoryFloorLocalSettings>().Load();
  320. if (_settings.CurrentBatchDate != DateTime.Today)
  321. {
  322. StockMovementBatch batch = new StockMovementBatch
  323. {
  324. Type = StockMovementBatchType.Issue,
  325. TimeStamp = DateTime.Now,
  326. Notes = "Manufacturing Consumption"
  327. };
  328. batch.Employee.ID = App.EmployeeID;
  329. new Client<StockMovementBatch>().Save(batch,"Created on Factory Floor Screen");
  330. _settings.CurrentBatchDate = DateTime.Today;
  331. _settings.CurrentBatchID = batch.ID;
  332. new LocalConfiguration<FactoryFloorLocalSettings>().Save(_settings);
  333. }
  334. return _settings.CurrentBatchID;
  335. }
  336. private void UseStock(object sender, RoutedEventArgs e)
  337. {
  338. if (CurrentPacket == null)
  339. {
  340. MessageBox.Show("Please select a Manufacturing Packet before proceeding");
  341. return;
  342. }
  343. var holdingrow = SelectedRows.FirstOrDefault();
  344. if (holdingrow == null)
  345. {
  346. MessageBox.Show("Please select a Stock Holding before proceeding");
  347. return;
  348. }
  349. var holding = holdingrow.ToObject<StockHolding>();
  350. var requiitems = holding.LoadRequisitionItems(true).ToArray();
  351. var ffi = new FactoryFloorIssue()
  352. {
  353. RequisitionItems = requiitems,
  354. Qty = 1,
  355. RequisitionID = requiitems.FirstOrDefault()?.ID ?? Guid.Empty
  356. };
  357. var ffg = new DynamicItemsListGrid<FactoryFloorIssue>();
  358. ffg.OnValidate += (_, _, errors) =>
  359. {
  360. var id = ffi.RequisitionID;
  361. var requi = ffi.RequisitionItems.FirstOrDefault(x => x.ID == id);
  362. if (requi == null)
  363. errors.Add("Please select an allocation!");
  364. else if (ffi.Qty < 0)
  365. errors.Add($"Qty must not be less than 0");
  366. else if (ffi.Qty > requi.Qty)
  367. errors.Add($"Qty must not exceed {requi.Qty}");
  368. };
  369. if (!ffg.EditItems(new[] { ffi }))
  370. return;
  371. var batchid = CheckStockMovementBatch();
  372. List<StockMovement> updates = new();
  373. Guid transactionid = Guid.NewGuid();
  374. if (CurrentPacket.SetoutLink.JobLink.ID != holding.Job.ID)
  375. {
  376. var transferout = holding.CreateMovement();
  377. transferout.JobRequisitionItem.ID = ffi.RequisitionID;
  378. transferout.Batch.ID = batchid;
  379. transferout.Type = StockMovementType.TransferOut;
  380. transferout.Issued = ffi.Qty;
  381. transferout.Transaction = transactionid;
  382. transferout.System = true;
  383. transferout.Date = DateTime.Now;
  384. transferout.Employee.ID = App.EmployeeID;
  385. updates.Add(transferout);
  386. var transferin = holding.CreateMovement();
  387. transferin.JobRequisitionItem.ID = ffi.RequisitionID;
  388. transferin.Batch.ID = batchid;
  389. transferin.Type = StockMovementType.TransferIn;
  390. transferin.Received = ffi.Qty;
  391. transferin.Job.ID = CurrentPacket.SetoutLink.JobLink.ID;
  392. transferin.Job.Synchronise(CurrentPacket.SetoutLink.JobLink);
  393. transferin.Transaction = transactionid;
  394. transferin.Date = DateTime.Now;
  395. transferin.System = true;
  396. transferin.Employee.ID = App.EmployeeID;
  397. updates.Add(transferin);
  398. }
  399. var issue = holding.CreateMovement();
  400. issue.JobRequisitionItem.ID = ffi.RequisitionID;
  401. issue.Batch.ID = batchid;
  402. issue.Type = StockMovementType.Issue;
  403. issue.Issued = ffi.Qty;
  404. issue.Job.ID = CurrentPacket.SetoutLink.JobLink.ID;
  405. issue.Job.Synchronise(CurrentPacket.SetoutLink.JobLink);
  406. issue.Transaction = transactionid;
  407. issue.Date = DateTime.Now;
  408. issue.Employee.ID = App.EmployeeID;
  409. issue.Notes = "Issued to Manufacturing";
  410. updates.Add(issue);
  411. new Client<StockMovement>().Save(updates,"Issued Stock to Factory Floor", (_,_) => { });
  412. if (issue.JobRequisitionItem.ID == Guid.Empty)
  413. UpdateRow<StockHolding, double>(holdingrow, x=>x.Available, holding.Available - ffi.Qty,false);
  414. UpdateRow<StockHolding,double>(holdingrow, x => x.Units, holding.Units - ffi.Qty);
  415. }
  416. }