StockHolding.cs 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Linq.Expressions;
  5. using InABox.Clients;
  6. using InABox.Core;
  7. using PRSClasses;
  8. namespace Comal.Classes
  9. {
  10. public class StockHoldingLastStocktake : CoreAggregate<StockHolding, StockMovement, DateTime>
  11. {
  12. public override Expression<Func<StockMovement, DateTime>> Aggregate => x => x.Date;
  13. public override Filter<StockMovement> Filter => new Filter<StockMovement>(x => x.Type)
  14. .IsEqualTo(StockMovementType.StockTake);
  15. public override Dictionary<Expression<Func<StockMovement, object>>, Expression<Func<StockHolding, object>>> Links =>
  16. new Dictionary<Expression<Func<StockMovement, object>>, Expression<Func<StockHolding, object>>>()
  17. {
  18. { StockMovement => StockMovement.Product.ID, StockHolding => StockHolding.Product.ID },
  19. { StockMovement => StockMovement.Location.ID, StockHolding => StockHolding.Location.ID },
  20. { StockMovement => StockMovement.Style.ID, StockHolding => StockHolding.Style.ID },
  21. { StockMovement => StockMovement.Job.ID, StockHolding => StockHolding.Job.ID },
  22. { StockMovement => StockMovement.Dimensions.Unit.ID, StockHolding => StockHolding.Dimensions.Unit.ID },
  23. { StockMovement => StockMovement.Dimensions.UnitSize, StockHolding => StockHolding.Dimensions.UnitSize },
  24. };
  25. public override AggregateCalculation Calculation => AggregateCalculation.Maximum;
  26. }
  27. [UserTracking(typeof(StockMovement))]
  28. [Unrecoverable]
  29. public class StockHolding : StockEntity, IRemotable, IPersistent, IOneToMany<StockLocation>, IOneToMany<Product>,
  30. IStockHolding, ILicense<WarehouseLicense>
  31. {
  32. [Editable(Editable.Disabled)]
  33. [EditorSequence(1)]
  34. public StockLocationLink Location { get; set; }
  35. private class ProductLookupGenerator : LookupDefinitionGenerator<Product, StockHolding>
  36. {
  37. public override Filter<Product>? DefineFilter(StockHolding[] items)
  38. => LookupFactory.DefineFilter<Product>().And(x => x.NonStock).IsEqualTo(false);
  39. }
  40. [Editable(Editable.Disabled)]
  41. [EditorSequence(2)]
  42. [LookupDefinition(typeof(ProductLookupGenerator))]
  43. public override ProductLink Product { get; set; }
  44. [DimensionsEditor(typeof(StockDimensions), AllowEditingUnit = false)]
  45. [Editable(Editable.Disabled)]
  46. [EditorSequence(3)]
  47. public override StockDimensions Dimensions { get; set; }
  48. [Editable(Editable.Disabled)]
  49. [EditorSequence(4)]
  50. public ProductStyleLink Style { get; set; }
  51. [Editable(Editable.Disabled)]
  52. [EditorSequence(4)]
  53. public JobLink Job { get; set; }
  54. [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
  55. [EditorSequence(5)]
  56. public double Units { get; set; }
  57. [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
  58. [EditorSequence(6)]
  59. public double Qty { get; set; }
  60. [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
  61. [EditorSequence(7)]
  62. public double Weight { get; set; }
  63. [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
  64. [EditorSequence(8)]
  65. public double Value { get; set; }
  66. [DoubleEditor(Editable = Editable.Disabled)]
  67. [EditorSequence(9)]
  68. public double AverageValue { get; set; }
  69. [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
  70. [EditorSequence(10)]
  71. public double Available { get; set; }
  72. [Aggregate(typeof(StockHoldingLastStocktake))]
  73. [DateEditor(Editable = Editable.Disabled)]
  74. [EditorSequence(11)]
  75. public DateTime LastStockTake { get; set; }
  76. public static Column<IStockHolding>[] Columns => new Column<IStockHolding>[]
  77. {
  78. new Column<IStockHolding>(x => x.Job.ID),
  79. new Column<IStockHolding>(x => x.Location.ID),
  80. new Column<IStockHolding>(x => x.Product.ID),
  81. new Column<IStockHolding>(x => x.Style.ID),
  82. new Column<IStockHolding>(x => x.Dimensions.Unit.ID),
  83. new Column<IStockHolding>(x => x.Dimensions.Quantity),
  84. new Column<IStockHolding>(x => x.Dimensions.Length),
  85. new Column<IStockHolding>(x => x.Dimensions.Width),
  86. new Column<IStockHolding>(x => x.Dimensions.Height),
  87. new Column<IStockHolding>(x => x.Dimensions.Weight),
  88. };
  89. public static Filter<StockMovement>? GetFilter(IStockHolding holding)
  90. {
  91. var filter = new Filters<StockMovement>();
  92. foreach(var column in Columns)
  93. {
  94. filter.Add(new Filter<StockMovement>(column.Cast<StockMovement>()).IsEqualTo(CoreUtils.GetPropertyValue(holding, column.Property)));
  95. }
  96. return filter.Combine();
  97. }
  98. }
  99. public static class StockHoldingExtensions
  100. {
  101. /// <summary>
  102. /// Create a new stock movement from an <see cref="IStockHolding"/>, copying across the "key" properties;
  103. /// that is, the job, product, style, location and dimensions.
  104. /// </summary>
  105. /// <param name="holding"></param>
  106. /// <returns></returns>
  107. public static StockMovement CreateMovement(this IStockHolding holding)
  108. {
  109. var movement = new StockMovement();
  110. movement.Job.ID = holding.Job.ID;
  111. movement.Job.Synchronise(holding.Job);
  112. movement.Product.ID = holding.Product.ID;
  113. movement.Product.Synchronise(holding.Product);
  114. movement.Style.ID = holding.Style.ID;
  115. movement.Style.Synchronise(holding.Style);
  116. movement.Location.ID = holding.Location.ID;
  117. movement.Location.Synchronise(holding.Location);
  118. movement.Dimensions.CopyFrom(holding.Dimensions);
  119. return movement;
  120. }
  121. public static IEnumerable<StockHolding> GroupMovements(IEnumerable<StockMovement> movements)
  122. {
  123. var grouped = new List<StockHolding>();
  124. var toGroup = movements.AsList();
  125. while (toGroup.Count > 0)
  126. {
  127. var first = toGroup.First();
  128. var selected = toGroup.Where(x => x.IsEqualTo(first)).ToList();
  129. var holding = grouped.FirstOrDefault(x => x.IsEqualTo(first));
  130. if (holding == null)
  131. {
  132. holding = new StockHolding();
  133. holding.Location.ID = first.Location.ID;
  134. holding.Product.ID = first.Product.ID;
  135. holding.Style.ID = first.Style.ID;
  136. holding.Job.ID = first.Job.ID;
  137. holding.Dimensions.CopyFrom(first.Dimensions);
  138. }
  139. holding.Recalculate(selected);
  140. toGroup.RemoveAll(x => selected.Any(s => s.ID == x.ID));
  141. }
  142. return grouped;
  143. }
  144. public static bool IsEqualTo(this IStockHolding h1, IStockHolding h2)
  145. {
  146. return h1.Product.ID == h2.Product.ID
  147. && h1.Location.ID == h2.Location.ID
  148. && h1.Job.ID == h2.Job.ID
  149. && h1.Style.ID == h2.Style.ID
  150. && h1.Dimensions.Unit.ID == h2.Dimensions.Unit.ID
  151. && h1.Dimensions.Length.IsEffectivelyEqual(h2.Dimensions.Length)
  152. && h1.Dimensions.Width.IsEffectivelyEqual(h2.Dimensions.Width)
  153. && h1.Dimensions.Height.IsEffectivelyEqual(h2.Dimensions.Height)
  154. && h1.Dimensions.Quantity.IsEffectivelyEqual(h2.Dimensions.Quantity)
  155. && h1.Dimensions.Weight.IsEffectivelyEqual(h2.Dimensions.Weight);
  156. }
  157. public static void Recalculate(this StockHolding holding, IEnumerable<StockMovement> movements)
  158. {
  159. movements = movements.AsIList();
  160. var units = movements.Sum(x => x.Units);
  161. var cost = movements.Select(x => x.Units * x.Cost).Sum();
  162. var available = movements.Where(x => x.JobRequisitionItem.ID == Guid.Empty).Sum(x => x.Units);
  163. holding.Units = units;
  164. holding.Available = available;
  165. holding.Qty = movements.Sum(x => x.Units * x.Dimensions.Value);
  166. holding.Value = cost;
  167. holding.AverageValue = units.IsEffectivelyEqual(0.0F) ? 0.0d : cost / units;
  168. holding.Weight = holding.Qty * holding.Dimensions.Weight;
  169. }
  170. public static IEnumerable<JobRequisitionItem> LoadRequisitionItems(this StockHolding holding)
  171. {
  172. var items = new Client<JobRequisitionItem>().Query(
  173. new Filter<JobRequisitionItem>(x => x.ID).InQuery(StockHolding.GetFilter(holding), x => x.JobRequisitionItem.ID),
  174. new Columns<JobRequisitionItem>(x => x.ID)
  175. .Add(x => x.Job.JobNumber)
  176. .Add(x => x.Requisition.Number)
  177. .Add(x => x.Requisition.Description)
  178. .Add(x => x.Qty))
  179. .ToObjects<JobRequisitionItem>();
  180. if (!holding.Available.IsEffectivelyEqual(0.0F))
  181. {
  182. var requi = new JobRequisitionItem() { Qty = holding.Available };
  183. requi.Requisition.Description = "Unallocated Items";
  184. items = CoreUtils.One(requi).Concat(items);
  185. }
  186. return items;
  187. }
  188. }
  189. }