StockMovement.cs 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Linq.Expressions;
  5. using InABox.Core;
  6. using PRSClasses;
  7. namespace Comal.Classes
  8. {
  9. public class StockMovementDocumentCount : CoreAggregate<StockMovement, StockMovementBatchDocument, Guid>
  10. {
  11. public override Expression<Func<StockMovementBatchDocument, Guid>> Aggregate => x => x.ID;
  12. public override AggregateCalculation Calculation => AggregateCalculation.Count;
  13. public override Dictionary<Expression<Func<StockMovementBatchDocument, object>>, Expression<Func<StockMovement, object>>> Links =>
  14. new Dictionary<Expression<Func<StockMovementBatchDocument, object>>, Expression<Func<StockMovement, object>>>()
  15. {
  16. { StockMovementBatchDocument => StockMovementBatchDocument.EntityLink.ID, StockMovement => StockMovement.Batch.ID }
  17. };
  18. }
  19. public class StockMovementLink : EntityLink<StockMovement>
  20. {
  21. [NullEditor]
  22. public override Guid ID { get; set; }
  23. }
  24. public class StockMovementUnitsFormula : IFormula<StockMovement, double>
  25. {
  26. public Expression<Func<StockMovement, double>> Value => x => x.Received;
  27. public Expression<Func<StockMovement, double>>[] Modifiers => new Expression<Func<StockMovement, double>>[] { x => x.Issued };
  28. public FormulaOperator Operator => FormulaOperator.Subtract;
  29. public FormulaType Type => FormulaType.Virtual;
  30. }
  31. public class StockMovementValueFormula : IFormula<StockMovement, double>
  32. {
  33. public Expression<Func<StockMovement, double>> Value => x => x.Units;
  34. public Expression<Func<StockMovement, double>>[] Modifiers => new Expression<Func<StockMovement, double>>[] { x => x.Cost };
  35. public FormulaOperator Operator => FormulaOperator.Multiply;
  36. public FormulaType Type => FormulaType.Virtual;
  37. }
  38. public class StockMovementQuantityFormula : IFormula<StockMovement, double>
  39. {
  40. public Expression<Func<StockMovement, double>> Value => x => x.Units;
  41. public Expression<Func<StockMovement, double>>[] Modifiers => new Expression<Func<StockMovement, double>>[] { x => x.Dimensions.Value };
  42. public FormulaOperator Operator => FormulaOperator.Multiply;
  43. public FormulaType Type => FormulaType.Virtual;
  44. }
  45. public class StockMovementIsRemnantCondition : ICondition<StockHolding, double, object>
  46. {
  47. public Expression<Func<StockHolding, double>> Left => x => x.Dimensions.Value;
  48. public Condition Condition => Condition.LessThan;
  49. public Expression<Func<StockHolding, double>> Right => x => x.Product.DefaultInstance.Dimensions.Value;
  50. public Expression<Func<StockHolding, object>> True => x => true;
  51. public Expression<Func<StockHolding, object>> False => x => null;
  52. public ConditionType Type => ConditionType.Virtual;
  53. }
  54. [UserTracking("Warehousing")]
  55. public class StockMovement : StockEntity, IRemotable, IPersistent, IOneToMany<StockLocation>, IOneToMany<Product>,
  56. ILicense<WarehouseLicense>, IStockHolding, IJobMaterial, IExportable, IImportable, IPostable
  57. {
  58. [DateTimeEditor]
  59. [EditorSequence(0)]
  60. public DateTime Date { get; set; }
  61. private class ProductLookupGenerator : LookupDefinitionGenerator<Product, StockMovement>
  62. {
  63. public override Filter<Product>? DefineFilter(StockMovement[] items)
  64. => LookupFactory.DefineFilter<Product>().And(x => x.NonStock).IsEqualTo(false);
  65. }
  66. [EditorSequence(1)]
  67. [EntityRelationship(DeleteAction.Cascade)]
  68. [RequiredColumn]
  69. [LookupDefinition(typeof(ProductLookupGenerator))]
  70. public override ProductLink Product { get; set; }
  71. [EditorSequence(2)]
  72. [RequiredColumn]
  73. [DimensionsEditor(typeof(StockDimensions))]
  74. public override StockDimensions Dimensions { get; set; }
  75. [EditorSequence(3)]
  76. [EntityRelationship(DeleteAction.SetNull)]
  77. public ProductStyleLink Style { get; set; }
  78. [DoubleEditor(Summary = Summary.Sum)]
  79. [EditorSequence(4)]
  80. public double Received { get; set; }
  81. [DoubleEditor(Summary = Summary.Sum)]
  82. [EditorSequence(5)]
  83. public double Issued { get; set; }
  84. /// <summary>
  85. /// This indicates the balance of the holding at the current point in time for a stock movement, populated for stock takes.
  86. /// </summary>
  87. [DoubleEditor]
  88. [EditorSequence(6)]
  89. public double Balance { get; set; }
  90. // Units = Received - Issued
  91. [Formula(typeof(StockMovementUnitsFormula))]
  92. [EditorSequence(7)]
  93. [DoubleEditor(Visible=Visible.Optional, Editable = Editable.Hidden, Summary= Summary.Sum)]
  94. public double Units { get; set; }
  95. // IsRemnant = Dimensions.Value < Product.Dimensions.Value
  96. [CheckBoxEditor(Editable = Editable.Hidden)]
  97. [Condition(typeof(StockMovementIsRemnantCondition))]
  98. [EditorSequence(7)]
  99. public bool IsRemnant { get; set; }
  100. // Qty = Units * Dimensions.Value
  101. [EditorSequence(8)]
  102. [Formula(typeof(StockMovementQuantityFormula))]
  103. [DoubleEditor(Editable = Editable.Hidden, Summary = Summary.Sum)]
  104. public double Qty { get; set; }
  105. [CurrencyEditor(Visible = Visible.Default)]
  106. [EditorSequence(9)]
  107. public double Cost { get; set; } = 0.0;
  108. [EditorSequence(10)]
  109. [EntityRelationship(DeleteAction.SetNull)]
  110. public StockLocationLink Location { get; set; }
  111. [EditorSequence(11)]
  112. [EntityRelationship(DeleteAction.SetNull)]
  113. public JobLink Job { get; set; }
  114. [MemoEditor]
  115. [EditorSequence(12)]
  116. public string Notes { get; set; }
  117. [EditorSequence(13)]
  118. [EnumLookupEditor(typeof(StockMovementType),Editable = Editable.Hidden, Visible = Visible.Default)]
  119. public StockMovementType Type { get; set; }
  120. [EditorSequence(14)]
  121. [EntityRelationship(DeleteAction.SetNull)]
  122. public EmployeeLink Employee { get; set; }
  123. [NullEditor]
  124. public Guid Transaction { get; set; } = Guid.NewGuid();
  125. [NullEditor]
  126. public bool System { get; set; }
  127. [NullEditor]
  128. [Obsolete("Replaced with Type", true)]
  129. public bool IsTransfer { get; set; } = false;
  130. [NullEditor]
  131. public PurchaseOrderItemLink OrderItem { get; set; }
  132. [NullEditor]
  133. [RequiredColumn]
  134. public JobRequisitionItemLink JobRequisitionItem { get; set; }
  135. [Aggregate(typeof(StockMovementDocumentCount))]
  136. [NullEditor]
  137. public int Documents { get; set; }
  138. /// <summary>
  139. /// Used to Group together movements (particularly images)
  140. /// when transactions are uploaded from Mobile Devices
  141. /// </summary>
  142. [EntityRelationship(DeleteAction.Cascade)]
  143. public StockMovementBatchLink Batch { get; set; }
  144. [NullEditor]
  145. [Obsolete("Replaced with Dimensions", true)]
  146. public double UnitSize { get; set; }
  147. [CurrencyEditor(Visible = Visible.Optional, Editable = Editable.Hidden, Summary=Summary.Sum)]
  148. [Formula(typeof(StockMovementValueFormula))]
  149. public double Value { get; set; } = 0.0;
  150. [NullEditor]
  151. [LoggableProperty]
  152. public DateTime Posted { get; set; }
  153. [NullEditor]
  154. [LoggableProperty]
  155. [RequiredColumn]
  156. public PostedStatus PostedStatus { get; set; }
  157. [NullEditor]
  158. public string PostedNote { get; set; }
  159. [NullEditor]
  160. public string PostedReference { get; set; }
  161. static StockMovement()
  162. {
  163. StockEntity.LinkStockDimensions<StockMovement>();
  164. LinkedProperties.Register<StockMovement, ProductStyleLink, Guid>(x=>x.Product.DefaultInstance.Style, x => x.ID, x => x.Style.ID);
  165. //LinkedProperties.Register<StockMovement, ProductLink, double>(x => x.Product, x => x.AverageCost, x => x.Cost);
  166. }
  167. private static Column<StockMovement> unitsize = new Column<StockMovement>(x => x.Dimensions.Value);
  168. private static Column<StockMovement> issued = new Column<StockMovement>(x => x.Issued);
  169. private static Column<StockMovement> received = new Column<StockMovement>(x => x.Received);
  170. private bool bChanging;
  171. protected override void DoPropertyChanged(string name, object? before, object? after)
  172. {
  173. if (bChanging)
  174. return;
  175. if (unitsize.IsEqualTo(name))
  176. {
  177. bChanging = true;
  178. Qty = (Received - Issued) * (double)after;
  179. bChanging = false;
  180. }
  181. else if (issued.IsEqualTo(name))
  182. {
  183. bChanging = true;
  184. Units = Received - (double)after;
  185. Qty = Units * Dimensions.Value;
  186. bChanging = false;
  187. }
  188. else if (received.IsEqualTo(name))
  189. {
  190. bChanging = true;
  191. Units = ((double)after - Issued);
  192. Qty = Units * Dimensions.Value;
  193. bChanging = false;
  194. }
  195. else
  196. base.DoPropertyChanged(name, before, after);
  197. }
  198. }
  199. }