| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335 | 
							- using System;
 
- using System.Collections.Generic;
 
- using System.Linq;
 
- using System.Linq.Expressions;
 
- using InABox.Core;
 
- using PRSClasses;
 
- namespace Comal.Classes
 
- {
 
-     
 
-     // public class StockHoldingUnitAggregate : CoreAggregate<StockHolding, StockMovement, double>
 
-     // {
 
-     //     public override Expression<Func<StockMovement, double>> Aggregate => x => x.Units;
 
-     //
 
-     //     public override Dictionary<Expression<Func<StockMovement, object>>, Expression<Func<StockHolding, object>>> Links =>
 
-     //         new Dictionary<Expression<Func<StockMovement, object>>, Expression<Func<StockHolding, object>>>()
 
-     //         {
 
-     //             { StockMovement => StockMovement.Product.ID, StockHolding => StockHolding.Product.ID },
 
-     //             { StockMovement => StockMovement.Job.ID, StockHolding => StockHolding.Job.ID },
 
-     //             { StockMovement => StockMovement.Location.ID, StockHolding => StockHolding.Location.ID },
 
-     //             { StockMovement => StockMovement.Style.ID, StockHolding => StockHolding.Style.ID },
 
-     //             { StockMovement => StockMovement.Dimensions.Unit.ID, StockHolding => StockHolding.Dimensions.Unit.ID },
 
-     //             { StockMovement => StockMovement.Dimensions.Quantity, StockHolding => StockHolding.Dimensions.Quantity },
 
-     //             { StockMovement => StockMovement.Dimensions.Length, StockHolding => StockHolding.Dimensions.Length },
 
-     //             { StockMovement => StockMovement.Dimensions.Width, StockHolding => StockHolding.Dimensions.Width },
 
-     //             { StockMovement => StockMovement.Dimensions.Height, StockHolding => StockHolding.Dimensions.Height },
 
-     //             { StockMovement => StockMovement.Dimensions.Weight, StockHolding => StockHolding.Dimensions.Weight },
 
-     //         };
 
-     //
 
-     //     public override AggregateCalculation Calculation => AggregateCalculation.Sum;
 
-     // }
 
-     //
 
-     // public class StockHoldingValueAggregate : CoreAggregate<StockHolding, StockMovement, double>
 
-     // {
 
-     //     public override Expression<Func<StockMovement, double>> Aggregate => x => x.Value;
 
-     //
 
-     //     public override Dictionary<Expression<Func<StockMovement, object>>, Expression<Func<StockHolding, object>>> Links =>
 
-     //         new Dictionary<Expression<Func<StockMovement, object>>, Expression<Func<StockHolding, object>>>()
 
-     //         {
 
-     //             { StockMovement => StockMovement.Product.ID, StockHolding => StockHolding.Product.ID },
 
-     //             { StockMovement => StockMovement.Job.ID, StockHolding => StockHolding.Job.ID },
 
-     //             { StockMovement => StockMovement.Location.ID, StockHolding => StockHolding.Location.ID },
 
-     //             { StockMovement => StockMovement.Style.ID, StockHolding => StockHolding.Style.ID },
 
-     //             { StockMovement => StockMovement.Dimensions.Unit.ID, StockHolding => StockHolding.Dimensions.Unit.ID },
 
-     //             { StockMovement => StockMovement.Dimensions.Quantity, StockHolding => StockHolding.Dimensions.Quantity },
 
-     //             { StockMovement => StockMovement.Dimensions.Length, StockHolding => StockHolding.Dimensions.Length },
 
-     //             { StockMovement => StockMovement.Dimensions.Width, StockHolding => StockHolding.Dimensions.Width },
 
-     //             { StockMovement => StockMovement.Dimensions.Height, StockHolding => StockHolding.Dimensions.Height },
 
-     //             { StockMovement => StockMovement.Dimensions.Weight, StockHolding => StockHolding.Dimensions.Weight },
 
-     //         };
 
-     //
 
-     //     public override AggregateCalculation Calculation => AggregateCalculation.Sum;
 
-     // }
 
-     //
 
-     // public class StockHoldingAverageValueFormula : IFormula<StockHolding, double>
 
-     // {
 
-     //     public Expression<Func<StockHolding, double>> Value => x => x.Value;
 
-     //
 
-     //     public Expression<Func<StockHolding, double>>[] Modifiers => new Expression<Func<StockHolding, double>>[] { x => x.Units };
 
-     //
 
-     //     public FormulaOperator Operator => FormulaOperator.Divide;
 
-     //     
 
-     //     public FormulaType Type => FormulaType.Virtual;
 
-     // }
 
-     //
 
-     // public class StockHoldingQuantityAggregate : CoreAggregate<StockHolding, StockMovement, double>
 
-     // {
 
-     //     public override Expression<Func<StockMovement, double>> Aggregate => x => x.Qty;
 
-     //
 
-     //     public override Dictionary<Expression<Func<StockMovement, object>>, Expression<Func<StockHolding, object>>> Links =>
 
-     //         new Dictionary<Expression<Func<StockMovement, object>>, Expression<Func<StockHolding, object>>>()
 
-     //         {
 
-     //             { StockMovement => StockMovement.Product.ID, StockHolding => StockHolding.Product.ID },
 
-     //             { StockMovement => StockMovement.Job.ID, StockHolding => StockHolding.Job.ID },
 
-     //             { StockMovement => StockMovement.Location.ID, StockHolding => StockHolding.Location.ID },
 
-     //             { StockMovement => StockMovement.Style.ID, StockHolding => StockHolding.Style.ID },
 
-     //             { StockMovement => StockMovement.Dimensions.Unit.ID, StockHolding => StockHolding.Dimensions.Unit.ID },
 
-     //             { StockMovement => StockMovement.Dimensions.Quantity, StockHolding => StockHolding.Dimensions.Quantity },
 
-     //             { StockMovement => StockMovement.Dimensions.Length, StockHolding => StockHolding.Dimensions.Length },
 
-     //             { StockMovement => StockMovement.Dimensions.Width, StockHolding => StockHolding.Dimensions.Width },
 
-     //             { StockMovement => StockMovement.Dimensions.Height, StockHolding => StockHolding.Dimensions.Height },
 
-     //             { StockMovement => StockMovement.Dimensions.Weight, StockHolding => StockHolding.Dimensions.Weight },
 
-     //         };
 
-     //
 
-     //     public override AggregateCalculation Calculation => AggregateCalculation.Sum;
 
-     // }
 
-     // public class StockHoldingAvailableAggregate : CoreAggregate<StockHolding, StockMovement, double>
 
-     // {
 
-     //     public override Expression<Func<StockMovement, double>> Aggregate => x => x.Units;
 
-     //
 
-     //     public override Dictionary<Expression<Func<StockMovement, object>>, Expression<Func<StockHolding, object>>> Links =>
 
-     //         new Dictionary<Expression<Func<StockMovement, object>>, Expression<Func<StockHolding, object>>>()
 
-     //         {
 
-     //             { StockMovement => StockMovement.Product.ID, StockHolding => StockHolding.Product.ID },
 
-     //             { StockMovement => StockMovement.Job.ID, StockHolding => StockHolding.Job.ID },
 
-     //             { StockMovement => StockMovement.Location.ID, StockHolding => StockHolding.Location.ID },
 
-     //             { StockMovement => StockMovement.Style.ID, StockHolding => StockHolding.Style.ID },
 
-     //             { StockMovement => StockMovement.Dimensions.Unit.ID, StockHolding => StockHolding.Dimensions.Unit.ID },
 
-     //             { StockMovement => StockMovement.Dimensions.Quantity, StockHolding => StockHolding.Dimensions.Quantity },
 
-     //             { StockMovement => StockMovement.Dimensions.Length, StockHolding => StockHolding.Dimensions.Length },
 
-     //             { StockMovement => StockMovement.Dimensions.Width, StockHolding => StockHolding.Dimensions.Width },
 
-     //             { StockMovement => StockMovement.Dimensions.Height, StockHolding => StockHolding.Dimensions.Height },
 
-     //             { StockMovement => StockMovement.Dimensions.Weight, StockHolding => StockHolding.Dimensions.Weight },
 
-     //         };
 
-     //
 
-     //     public override AggregateCalculation Calculation => AggregateCalculation.Sum;
 
-     //
 
-     //     public override Filter<StockMovement>? Filter => new Filter<StockMovement>(x => x.JobRequisitionItem.ID).IsEqualTo(Guid.Empty);
 
-     // }
 
-     
 
-     // public class StockHoldingWeightFormula : IFormula<StockHolding, double>
 
-     // {
 
-     //     public Expression<Func<StockHolding, double>> Value => x => x.Qty;
 
-     //
 
-     //     public Expression<Func<StockHolding, double>>[] Modifiers => new Expression<Func<StockHolding, double>>[] { x => x.Dimensions.Weight };
 
-     //
 
-     //     public FormulaOperator Operator => FormulaOperator.Multiply;
 
-     //     
 
-     //     public FormulaType Type => FormulaType.Virtual;
 
-     // }
 
-     //
 
-     // public class StockHoldingIsRemnantCondition : ICondition<StockHolding, double, object>
 
-     // {
 
-     //     public Expression<Func<StockHolding, double>> Left => x => x.Dimensions.Value;
 
-     //
 
-     //     public Condition Condition => Condition.LessThan;
 
-     //
 
-     //     public Expression<Func<StockHolding, double>> Right => x => x.Product.DefaultInstance.Dimensions.Value;
 
-     //
 
-     //     public Expression<Func<StockHolding, object>> True => x => true;
 
-     //
 
-     //     public Expression<Func<StockHolding, object>> False => x => null;
 
-     //     
 
-     //     public ConditionType Type => ConditionType.Virtual;
 
-     // }
 
-     
 
-     // public class StockHoldingUnionGenerator : AutoEntityUnionGenerator<IStockHolding>
 
-     // {
 
-     //     protected override void Configure()
 
-     //     {
 
-     //         AddTable<StockMovement>();
 
-     //     }
 
-     //
 
-     //     public override bool Distinct => true;
 
-     //
 
-     //     public override Column<IStockHolding>[] IDColumns => Columns;
 
-     //
 
-     //     public static Column<IStockHolding>[] Columns => new Column<IStockHolding>[]
 
-     //     {
 
-     //         new Column<IStockHolding>(x => x.Job.ID),
 
-     //         new Column<IStockHolding>(x => x.Location.ID),
 
-     //         new Column<IStockHolding>(x => x.Product.ID),
 
-     //         new Column<IStockHolding>(x => x.Style.ID),
 
-     //         new Column<IStockHolding>(x => x.Dimensions.Unit.ID),
 
-     //         new Column<IStockHolding>(x => x.Dimensions.Quantity),
 
-     //         new Column<IStockHolding>(x => x.Dimensions.Length),
 
-     //         new Column<IStockHolding>(x => x.Dimensions.Width),
 
-     //         new Column<IStockHolding>(x => x.Dimensions.Height),
 
-     //         new Column<IStockHolding>(x => x.Dimensions.Weight),
 
-     //     };
 
-     //
 
-     //     public static Filter<StockMovement>? GetFilter(IStockHolding holding)
 
-     //     {
 
-     //         var filter = new Filters<StockMovement>();
 
-     //
 
-     //         foreach(var column in Columns)
 
-     //         {
 
-     //             filter.Add(new Filter<StockMovement>(column.Cast<StockMovement>()).IsEqualTo(CoreUtils.GetPropertyValue(holding, column.Property)));
 
-     //         }
 
-     //
 
-     //         return filter.Combine();
 
-     //     }
 
-     // }
 
-     //
 
-     // [UserTracking(typeof(StockMovement))]
 
-     // [AutoEntity(typeof(StockHoldingUnionGenerator))]
 
-     // public class StockHoldingView : StockEntity, IRemotable, IPersistent, IOneToMany<StockLocation>, IOneToMany<Product>, 
 
-     //     IStockHolding, ILicense<WarehouseLicense>
 
-     // {
 
-     //     
 
-     //     public override ProductLink Product { get; set; }
 
-     //
 
-     //     public ProductStyleLink Style { get; set; }
 
-     //
 
-     //     public StockLocationLink Location { get; set; }
 
-     //
 
-     //     public JobLink Job { get; set; }
 
-     //
 
-     //     [RequiredColumn]
 
-     //     [DimensionsEditor(typeof(StockDimensions), AllowEditingUnit = false)]
 
-     //     public override StockDimensions Dimensions { get; set; }
 
-     //
 
-     //     [Condition(typeof(StockHoldingIsRemnantCondition))]
 
-     //     [NullEditor]
 
-     //     public bool IsRemnant { get; set; }
 
-     //     
 
-     //     [Aggregate(typeof(StockHoldingUnitAggregate))]
 
-     //     [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
 
-     //     public double Units { get; set; }
 
-     //     
 
-     //     [Aggregate(typeof(StockHoldingQuantityAggregate))]
 
-     //     [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
 
-     //     public double Qty { get; set; }
 
-     //     
 
-     //     [Formula(typeof(StockHoldingWeightFormula))]
 
-     //     [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
 
-     //     public double Weight { get; set; }
 
-     //     
 
-     //     [Aggregate(typeof(StockHoldingValueAggregate))]
 
-     //     [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
 
-     //     public double Value { get; set; }
 
-     //     
 
-     //     [Formula(typeof(StockHoldingAverageValueFormula))]
 
-     //     [DoubleEditor(Editable = Editable.Disabled)]
 
-     //     public double AverageValue { get; set; }
 
-     // }
 
-     
 
-     // FV 6/2/24: This used to be a view (see above), but we were running into performance issues trying to
 
-     // get aggregates fof stock locations (ie show me how many holdings in a given location was taking upwrds of
 
-     // 20 seconds.  Have moved update logic to StockHoldingStore / StockMovementStore so updates are a bit slower,
 
-     // but reads are now much faster.
 
-     [UserTracking(typeof(StockMovement))]
 
-     public class StockHolding : StockEntity, IRemotable, IPersistent, IOneToMany<StockLocation>, IOneToMany<Product>, 
 
-         IStockHolding, ILicense<WarehouseLicense>
 
-     {
 
-         
 
-         [Editable(Editable.Disabled)]
 
-         [EditorSequence(1)]
 
-         public StockLocationLink Location { get; set; }
 
-         
 
-         private class ProductLookupGenerator : LookupDefinitionGenerator<Product, StockHolding>
 
-         {
 
-             public override Filter<Product>? DefineFilter(StockHolding[] items)
 
-                 => LookupFactory.DefineFilter<Product>().And(x => x.NonStock).IsEqualTo(false);
 
-         }
 
-         [Editable(Editable.Disabled)]
 
-         [EditorSequence(2)]
 
-         [LookupDefinition(typeof(ProductLookupGenerator))]
 
-         public override ProductLink Product { get; set; }
 
-         
 
-         [DimensionsEditor(typeof(StockDimensions), AllowEditingUnit = false)]
 
-         [Editable(Editable.Disabled)]
 
-         [EditorSequence(3)]
 
-         public override StockDimensions Dimensions { get; set; }
 
-         [Editable(Editable.Disabled)]
 
-         [EditorSequence(4)]
 
-         public ProductStyleLink Style { get; set; }
 
-         
 
-         [Editable(Editable.Disabled)]
 
-         [EditorSequence(4)]
 
-         public JobLink Job { get; set; }
 
-         
 
-         [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
 
-         [EditorSequence(5)]
 
-         public double Units { get; set; }
 
-         
 
-         [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
 
-         [EditorSequence(6)]
 
-         public double Qty { get; set; }
 
-         
 
-         [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
 
-         [EditorSequence(7)]
 
-         public double Weight { get; set; }
 
-         
 
-         [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
 
-         [EditorSequence(8)]
 
-         public double Value { get; set; }
 
-         
 
-         [DoubleEditor(Editable = Editable.Disabled)]
 
-         [EditorSequence(9)]
 
-         public double AverageValue { get; set; }
 
-         
 
-         [DoubleEditor(Editable = Editable.Disabled, Summary = Summary.Sum)]
 
-         [EditorSequence(10)]
 
-         public double Available { get; set; }
 
-         
 
-         public static Column<IStockHolding>[] Columns => new Column<IStockHolding>[]
 
-         {
 
-             new Column<IStockHolding>(x => x.Job.ID),
 
-             new Column<IStockHolding>(x => x.Location.ID),
 
-             new Column<IStockHolding>(x => x.Product.ID),
 
-             new Column<IStockHolding>(x => x.Style.ID),
 
-             new Column<IStockHolding>(x => x.Dimensions.Unit.ID),
 
-             new Column<IStockHolding>(x => x.Dimensions.Quantity),
 
-             new Column<IStockHolding>(x => x.Dimensions.Length),
 
-             new Column<IStockHolding>(x => x.Dimensions.Width),
 
-             new Column<IStockHolding>(x => x.Dimensions.Height),
 
-             new Column<IStockHolding>(x => x.Dimensions.Weight),
 
-         };
 
-         
 
-         public static Filter<StockMovement>? GetFilter(IStockHolding holding)
 
-         {
 
-             var filter = new Filters<StockMovement>();
 
-         
 
-             foreach(var column in Columns)
 
-             {
 
-                 filter.Add(new Filter<StockMovement>(column.Cast<StockMovement>()).IsEqualTo(CoreUtils.GetPropertyValue(holding, column.Property)));
 
-             }
 
-         
 
-             return filter.Combine();
 
-         }
 
-     }
 
-     public static class StockHoldingExtensions
 
-     {
 
-         public static bool IsEqualTo(this IStockHolding h1, IStockHolding h2)
 
-         {
 
-             return h1.Product.ID == h2.Product.ID
 
-                 && h1.Location.ID == h2.Location.ID
 
-                 && h1.Job.ID == h2.Job.ID
 
-                 && h1.Style.ID == h2.Style.ID
 
-                 && h1.Dimensions.Unit.ID == h2.Dimensions.Unit.ID
 
-                 && h1.Dimensions.Length.IsEffectivelyEqual(h2.Dimensions.Length)
 
-                 && h1.Dimensions.Width.IsEffectivelyEqual(h2.Dimensions.Width)
 
-                 && h1.Dimensions.Height.IsEffectivelyEqual(h2.Dimensions.Height)
 
-                 && h1.Dimensions.Quantity.IsEffectivelyEqual(h2.Dimensions.Quantity)
 
-                 && h1.Dimensions.Weight.IsEffectivelyEqual(h2.Dimensions.Weight);
 
-         }
 
-         public static void Recalculate(this StockHolding holding, IEnumerable<StockMovement> movements)
 
-         {
 
-             movements = movements.AsIList();
 
-             var units = movements.Sum(x => x.Units);
 
-             var cost = movements.Select(x => x.Units * x.Cost).Sum();
 
-             var available = movements.Where(x => x.JobRequisitionItem.ID == Guid.Empty).Sum(x => x.Units);
 
-             holding.Units = units;
 
-             holding.AverageValue = units.IsEffectivelyEqual(0.0F) ? 0.0d : cost / units;
 
-             holding.Available = available;
 
-         }
 
-     }
 
-     
 
- }
 
 
  |