StockHolding.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  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. }.AddRange(Dimensions.GetLinks<StockMovement, StockHolding>(x => x.Dimensions, x => x.Dimensions));
  23. public override AggregateCalculation Calculation => AggregateCalculation.Maximum;
  24. }
  25. [UserTracking(typeof(StockMovement))]
  26. [Unrecoverable]
  27. public class StockHolding : StockEntity, IRemotable, IPersistent, IOneToMany<StockLocation>, IOneToMany<Product>,
  28. IStockHolding, ILicense<WarehouseLicense>
  29. {
  30. [Editable(Editable.Disabled)]
  31. [EditorSequence(1)]
  32. public StockLocationLink Location { get; set; }
  33. private class ProductLookupGenerator : LookupDefinitionGenerator<Product, StockHolding>
  34. {
  35. public override Filter<Product>? DefineFilter(StockHolding[] items)
  36. => LookupFactory.DefineFilter<Product>().And(x => x.NonStock).IsEqualTo(false);
  37. }
  38. [Editable(Editable.Disabled)]
  39. [EditorSequence(2)]
  40. [LookupDefinition(typeof(ProductLookupGenerator))]
  41. public override ProductLink Product { get; set; }
  42. [DimensionsEditor(typeof(StockDimensions))]
  43. [Editable(Editable.Disabled)]
  44. [EditorSequence(3)]
  45. public override StockDimensions Dimensions { get; set; }
  46. [Editable(Editable.Disabled)]
  47. [EditorSequence(4)]
  48. public ProductStyleLink Style { get; set; }
  49. [Editable(Editable.Disabled)]
  50. [EditorSequence(4)]
  51. public JobLink Job { get; set; }
  52. [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
  53. [EditorSequence(5)]
  54. public double Units { get; set; }
  55. [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
  56. [EditorSequence(6)]
  57. public double Qty { get; set; }
  58. [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
  59. [EditorSequence(7)]
  60. public double Weight { get; set; }
  61. [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
  62. [EditorSequence(8)]
  63. public double Value { get; set; }
  64. [DoubleEditor(Editable = Editable.Disabled)]
  65. [EditorSequence(9)]
  66. public double AverageValue { get; set; }
  67. [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
  68. [EditorSequence(10)]
  69. public double Available { get; set; }
  70. [Formula(typeof(StockHoldingAllocatedFormula))]
  71. [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
  72. [EditorSequence(11)]
  73. public double Allocated { get; set; }
  74. [Aggregate(typeof(StockHoldingLastStocktake))]
  75. [DateEditor(Editable = Editable.Disabled)]
  76. [EditorSequence(11)]
  77. public DateTime LastStockTake { get; set; }
  78. public static Column<IStockHolding>[] Columns => new Column<IStockHolding>[]
  79. {
  80. new Column<IStockHolding>(x => x.Job.ID),
  81. new Column<IStockHolding>(x => x.Location.ID),
  82. new Column<IStockHolding>(x => x.Product.ID),
  83. new Column<IStockHolding>(x => x.Style.ID),
  84. new Column<IStockHolding>(x => x.Dimensions.Unit.ID),
  85. new Column<IStockHolding>(x => x.Dimensions.Quantity),
  86. new Column<IStockHolding>(x => x.Dimensions.Length),
  87. new Column<IStockHolding>(x => x.Dimensions.Width),
  88. new Column<IStockHolding>(x => x.Dimensions.Height),
  89. new Column<IStockHolding>(x => x.Dimensions.Weight),
  90. };
  91. public static Filter<StockMovement>? GetFilter(IStockHolding holding)
  92. {
  93. var filter = new Filters<StockMovement>();
  94. foreach(var column in Columns)
  95. {
  96. filter.Add(new Filter<StockMovement>(column.Cast<StockMovement>()).IsEqualTo(CoreUtils.GetPropertyValue(holding, column.Property)));
  97. }
  98. return filter.Combine();
  99. }
  100. }
  101. internal class StockHoldingAllocatedFormula : IFormula<StockHolding, double>
  102. {
  103. public Expression<Func<StockHolding, double>> Value => x => x.Qty;
  104. public FormulaOperator Operator => FormulaOperator.Subtract;
  105. public Expression<Func<StockHolding, double>>[] Modifiers => new Expression<Func<StockHolding, double>>[]
  106. {
  107. x => x.Available
  108. };
  109. public FormulaType Type => FormulaType.Virtual;
  110. }
  111. public static class StockHoldingExtensions
  112. {
  113. /// <summary>
  114. /// Create a new stock movement from an <see cref="IStockHolding"/>, copying across the "key" properties;
  115. /// that is, the job, product, style, location and dimensions.
  116. /// </summary>
  117. /// <param name="holding"></param>
  118. /// <returns></returns>
  119. public static StockMovement CreateMovement(this IStockHolding holding)
  120. {
  121. var movement = new StockMovement();
  122. movement.Date = DateTime.Now;
  123. movement.Job.ID = holding.Job.ID;
  124. movement.Job.Synchronise(holding.Job);
  125. movement.Product.ID = holding.Product.ID;
  126. movement.Product.Synchronise(holding.Product);
  127. movement.Style.ID = holding.Style.ID;
  128. movement.Style.Synchronise(holding.Style);
  129. movement.Location.ID = holding.Location.ID;
  130. movement.Location.Synchronise(holding.Location);
  131. movement.Dimensions.CopyFrom(holding.Dimensions);
  132. return movement;
  133. }
  134. public static IEnumerable<StockHolding> GroupMovements(IEnumerable<StockMovement> movements)
  135. {
  136. var grouped = new List<StockHolding>();
  137. var toGroup = movements.AsList();
  138. while (toGroup.Count > 0)
  139. {
  140. var first = toGroup.First();
  141. var selected = toGroup.Where(x => x.IsEqualTo(first)).ToList();
  142. var holding = grouped.FirstOrDefault(x => x.IsEqualTo(first));
  143. if (holding == null)
  144. {
  145. holding = new StockHolding();
  146. holding.Location.ID = first.Location.ID;
  147. holding.Product.ID = first.Product.ID;
  148. holding.Style.ID = first.Style.ID;
  149. holding.Job.ID = first.Job.ID;
  150. holding.Dimensions.CopyFrom(first.Dimensions);
  151. }
  152. holding.Recalculate(selected);
  153. toGroup.RemoveAll(x => selected.Any(s => s.ID == x.ID));
  154. }
  155. return grouped;
  156. }
  157. public static bool IsEqualTo(this IStockHolding h1, IStockHolding h2)
  158. {
  159. return h1.Product.ID == h2.Product.ID
  160. && h1.Location.ID == h2.Location.ID
  161. && h1.Job.ID == h2.Job.ID
  162. && h1.Style.ID == h2.Style.ID
  163. && h1.Dimensions.Unit.ID == h2.Dimensions.Unit.ID
  164. && h1.Dimensions.Length.IsEffectivelyEqual(h2.Dimensions.Length)
  165. && h1.Dimensions.Width.IsEffectivelyEqual(h2.Dimensions.Width)
  166. && h1.Dimensions.Height.IsEffectivelyEqual(h2.Dimensions.Height)
  167. && h1.Dimensions.Quantity.IsEffectivelyEqual(h2.Dimensions.Quantity)
  168. && h1.Dimensions.Weight.IsEffectivelyEqual(h2.Dimensions.Weight);
  169. }
  170. public static void Recalculate(this StockHolding holding, IEnumerable<StockMovement> movements)
  171. {
  172. movements = movements.AsIList();
  173. var units = movements.Sum(x => x.Units);
  174. var cost = movements.Select(x => x.Units * x.Cost).Sum();
  175. var available = movements.Where(x => x.JobRequisitionItem.ID == Guid.Empty).Sum(x => x.Units);
  176. holding.Units = units;
  177. holding.Available = available;
  178. holding.Qty = movements.Sum(x => x.Units * x.Dimensions.Value);
  179. holding.Value = cost;
  180. holding.AverageValue = units.IsEffectivelyEqual(0.0F) ? 0.0d : cost / units;
  181. holding.Weight = holding.Qty * holding.Dimensions.Weight;
  182. }
  183. public static IEnumerable<JobRequisitionItem> LoadRequisitionItems(this StockHolding holding, bool alwaysshowunallocated = false)
  184. {
  185. var items = new Client<JobRequisitionItem>().Query(
  186. new Filter<JobRequisitionItem>(x => x.ID).InQuery(StockHolding.GetFilter(holding), x => x.JobRequisitionItem.ID),
  187. Columns.None<JobRequisitionItem>().Add(x => x.ID)
  188. .Add(x => x.Job.ID)
  189. .Add(x => x.Job.JobNumber)
  190. .Add(x => x.Job.Name)
  191. .Add(x => x.Requisition.Number)
  192. .Add(x => x.Requisition.Description)
  193. .Add(x => x.Qty))
  194. .ToObjects<JobRequisitionItem>();
  195. if (holding.Available > 0 || alwaysshowunallocated)
  196. {
  197. var requi = new JobRequisitionItem() { Qty = holding.Available };
  198. requi.Requisition.Description = "Unallocated Items";
  199. items = CoreUtils.One(requi).Concat(items);
  200. }
  201. return items;
  202. }
  203. public static IEnumerable<StockMovement> AdjustValue(this StockHolding holding, double unitvalue, StockMovementBatch batch)
  204. {
  205. List<StockMovement> _result = new List<StockMovement>();
  206. var movements = Client.Query(
  207. new Filter<StockMovement>(x => x.Location.ID).IsEqualTo(holding.Location.ID)
  208. .And(x => x.Style.ID).IsEqualTo(holding.Style.ID)
  209. .And(x => x.Dimensions).DimensionEquals(holding.Dimensions)
  210. .And(x => x.Job.ID).IsEqualTo(holding.Job.ID),
  211. Columns.Required<StockMovement>().Add(x=>x.Units)
  212. ).Rows.ToObjects<StockMovement>().ToArray();
  213. var _allocations = movements.GroupBy(x => x.JobRequisitionItem.ID);
  214. foreach (var _allocation in _allocations)
  215. {
  216. var _units = _allocation.Sum(x => x.Units);
  217. if (!_units.IsEffectivelyEqual(0.0))
  218. {
  219. var _transout = holding.CreateMovement();
  220. _transout.Employee.ID = batch.Employee.ID;
  221. _transout.Issued = _units;
  222. _transout.Cost = holding.AverageValue;
  223. _transout.Type = StockMovementType.TransferOut;
  224. _transout.JobRequisitionItem.ID = _allocation.Key;
  225. _transout.Batch.ID = batch.ID;
  226. _transout.Notes = $"Adjusting Average Value from ${holding.AverageValue:F2} to ${unitvalue:F2}";
  227. _result.Add(_transout);
  228. var _transin = holding.CreateMovement();
  229. _transin.Date = _transout.Date.AddTicks(1);
  230. _transout.Employee.ID = batch.Employee.ID;
  231. _transin.Received = _units;
  232. _transin.Cost = unitvalue;
  233. _transin.Type = StockMovementType.TransferIn;
  234. _transin.Transaction = _transout.Transaction;
  235. _transin.JobRequisitionItem.ID = _allocation.Key;
  236. _transin.Batch.ID = batch.ID;
  237. _transin.Notes = $"Adjusting Average Value from ${holding.AverageValue:F2} to ${unitvalue:F2}";
  238. _result.Add(_transin);
  239. }
  240. }
  241. return _result;
  242. }
  243. // public static IEnumerable<Tuple<Guid,double>> GetAllocations(this StockHolding holding, bool alwaysshowunallocated)
  244. // {
  245. // var table = new Client<StockMovement>().Query(
  246. // StockHolding.GetFilter(holding),
  247. // new Columns<StockMovement>(x => x.Units)
  248. // .Add(x => x.Location.ID)
  249. // .Add(x => x.Product.ID)
  250. // .Add(x => x.Style.ID)
  251. // .AddDimensionsColumns(x => x.Dimensions)
  252. // .Add(x => x.Cost)
  253. // .Add(x => x.OrderItem.ID)
  254. // .Add(x => x.JobRequisitionItem.ID)
  255. // );
  256. //
  257. // var movements = table
  258. // .ToObjects<StockMovement>();
  259. //
  260. // var groups = movements
  261. // .GroupBy(x => new
  262. // {
  263. // Location = x.Location.ID,
  264. // Product = x.Product.ID,
  265. // Style = x.Style.ID,
  266. // x.Dimensions,
  267. // x.Cost,
  268. // OrderItem = x.OrderItem.ID,
  269. // JobRequisitionItem = x.JobRequisitionItem.ID
  270. // });
  271. //
  272. // var result = groups
  273. // .Select(x => new Tuple<Guid, double>(
  274. // x.Key.JobRequisitionItem,
  275. // x.Sum(x => x.Units))
  276. // ).ToList();
  277. //
  278. // if (alwaysshowunallocated || !holding.Available.IsEffectivelyEqual(0))
  279. // result.Add(new Tuple<Guid, double>(Guid.Empty,holding.Available));
  280. //
  281. // return result;
  282. //
  283. // }
  284. }
  285. }