StockHolding.cs 18 KB

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