| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298 | using System;using System.Collections.Generic;using System.Linq;using System.Linq.Expressions;using InABox.Core;using PRSClasses;namespace Comal.Classes{    public class StockMovementDocumentCount : CoreAggregate<StockMovement, StockMovementBatchDocument, Guid>    {        public override Expression<Func<StockMovementBatchDocument, Guid>> Aggregate => x => x.ID;        public override AggregateCalculation Calculation => AggregateCalculation.Count;        public override Dictionary<Expression<Func<StockMovementBatchDocument, object>>, Expression<Func<StockMovement, object>>> Links =>            new Dictionary<Expression<Func<StockMovementBatchDocument, object>>, Expression<Func<StockMovement, object>>>()            {                { StockMovementBatchDocument => StockMovementBatchDocument.EntityLink.ID, StockMovement => StockMovement.Batch.ID }            };    }    public class StockMovementLink : EntityLink<StockMovement>    {        [NullEditor]        public override Guid ID { get; set; }    }        public class StockMovementValueFormula : IFormula<StockMovement, double>    {        public Expression<Func<StockMovement, double>> Value => x => x.Units;        public Expression<Func<StockMovement, double>>[] Modifiers => new Expression<Func<StockMovement, double>>[] { x => x.Cost };        public FormulaOperator Operator => FormulaOperator.Multiply;                public FormulaType Type => FormulaType.Virtual;    }        public class StockMovementQuantityFormula : IFormula<StockMovement, double>    {        public Expression<Func<StockMovement, double>> Value => x => x.Units;        public Expression<Func<StockMovement, double>>[] Modifiers => new Expression<Func<StockMovement, double>>[] { x => x.Dimensions.Value };        public FormulaOperator Operator => FormulaOperator.Multiply;                public FormulaType Type => FormulaType.Virtual;    }    [UserTracking("Warehousing")]    public class StockMovement : StockEntity, IRemotable, IPersistent, IOneToMany<StockLocation>, IOneToMany<Product>,         ILicense<WarehouseLicense>, IStockHolding, IJobMaterial, IExportable, IImportable, IPostable    {        [DateTimeEditor]        [EditorSequence(0)]        [SecondaryIndex]        public DateTime Date { get; set; }        private class ProductLookupGenerator : LookupDefinitionGenerator<Product, StockMovement>        {            public override Filter<Product>? DefineFilter(StockMovement[] items)                => LookupFactory.DefineFilter<Product>().And(x => x.NonStock).IsEqualTo(false);        }        [EditorSequence(1)]        [EntityRelationship(DeleteAction.Cascade)]        [RequiredColumn]        [LookupDefinition(typeof(ProductLookupGenerator))]        public override ProductLink Product { get; set; }                                [EditorSequence(2)]        [RequiredColumn]        [DimensionsEditor(typeof(StockDimensions))]        public override StockDimensions Dimensions { get; set; }        [EditorSequence(3)]        [EntityRelationship(DeleteAction.SetNull)]        public ProductStyleLink Style { get; set; }                // Allowed to be negative.        [DoubleEditor(Summary = Summary.Sum)]        [EditorSequence(4)]        public double Received { get; set; }        [DoubleEditor(Summary = Summary.Sum)]        [EditorSequence(5)]        public double Issued { get; set; }        /// <summary>        /// This indicates the balance of the holding at the current point in time for a stock movement, populated for stock takes.        /// </summary>        [DoubleEditor]        [EditorSequence(6)]        public double Balance { get; set; }        private class StockMovementUnitsFormula : ComplexFormulaGenerator<StockMovement, double>        {            public override IComplexFormulaNode<StockMovement, double> GetFormula() =>                Formula(FormulaOperator.Subtract, Property(x => x.Received), Property(x => x.Issued));        }        [ComplexFormula(typeof(StockMovementUnitsFormula))]        [EditorSequence(7)]        [DoubleEditor(Visible=Visible.Optional, Editable = Editable.Hidden, Summary= Summary.Sum)]        /// <summary>        /// Units = Received - Issued        /// </summary>        public double Units { get; set; }        private class IsRemnantCondition : ComplexFormulaGenerator<StockMovement, bool>        {            public override IComplexFormulaNode<StockMovement, bool> GetFormula() =>                If<double>(                    x => x.Property(x => x.Dimensions.Value),                    Condition.LessThan,                    x => x.Property(x => x.Product.DefaultInstance.Dimensions.Value))                .Then(Constant(true))                .Else(Constant(false));        }        // IsRemnant = Dimensions.Value < Product.Dimensions.Value        [CheckBoxEditor(Editable = Editable.Hidden)]        [ComplexFormula(typeof(IsRemnantCondition))]        [EditorSequence(7)]        public bool IsRemnant { get; set; }                // Qty = Units * Dimensions.Value        [EditorSequence(8)]        [Formula(typeof(StockMovementQuantityFormula))]        [DoubleEditor(Editable = Editable.Hidden, Summary = Summary.Sum)]        public double Qty { get; set; }                        [CurrencyEditor(Visible = Visible.Default)]        [EditorSequence(9)]        public double Cost { get; set; } = 0.0;                [EditorSequence(10)]        [EntityRelationship(DeleteAction.SetNull)]        public StockLocationLink Location { get; set; }                [EditorSequence(11)]        [EntityRelationship(DeleteAction.SetNull)]        public JobLink Job { get; set; }                [MemoEditor]        [EditorSequence(12)]        public string Notes { get; set; }        [EditorSequence(13)]        [EnumLookupEditor(typeof(StockMovementType), Visible = Visible.Default)]        public StockMovementType Type { get; set; }        [EditorSequence(14)]        [EntityRelationship(DeleteAction.SetNull)]        public EmployeeLink Employee { get; set; }                [NullEditor]        public Guid Transaction { get; set; } = Guid.NewGuid();        [NullEditor]        public bool System { get; set; }        [NullEditor]        [Obsolete("Replaced with Type", true)]        public bool IsTransfer { get; set; } = false;        [NullEditor]        public PurchaseOrderItemLink OrderItem { get; set; }        private class JobRequisitionItemLookup : LookupDefinitionGenerator<JobRequisitionItem, StockMovement>        {            public override Columns<JobRequisitionItem> DefineColumns()            {                return Columns.None<JobRequisitionItem>().Add(x => x.Job.JobNumber).Add(x => x.Requisition.Number).Add(x => x.Requisition.Description);            }            public override string FormatDisplay(CoreRow row)            {                var jobNumber = row.Get<JobRequisitionItem, string>(x => x.Job.JobNumber);                var requiNumber = row.Get<JobRequisitionItem, int>(x => x.Requisition.Number);                var requiDesc = row.Get<JobRequisitionItem, string>(x => x.Requisition.Description);                return $"{jobNumber}: #{requiNumber} ({requiDesc})";            }        }        [RequiredColumn]        [LookupDefinition(typeof(JobRequisitionItemLookup))]        public JobRequisitionItemLink JobRequisitionItem { get; set; }        [NullEditor]        public InvoiceLink Invoice { get; set; }        private class JobScopeLookup : LookupDefinitionGenerator<JobScope, StockMovement>        {            public override Filter<JobScope> DefineFilter(StockMovement[] items)            {                var item = items?.Length == 1 ? items[0] : null;                if (item != null)                    return new Filter<JobScope>(x => x.Job.ID).IsEqualTo(item.Job.ID).And(x => x.Status.Approved).IsEqualTo(true);                return new Filter<JobScope>(x => x.ID).None();            }                        public override Columns<StockMovement> DefineFilterColumns()                => Columns.None<StockMovement>().Add(x=>x.Job.ID);        }        [LookupDefinition(typeof(JobScopeLookup))]        [EditorSequence(5)]        [EntityRelationship(DeleteAction.SetNull)]        public JobScopeLink JobScope { get; set; }        public ActualCharge Charge { get; set; }        [Aggregate(typeof(StockMovementDocumentCount))]        [NullEditor]        public int Documents { get; set; }        /// <summary>        ///     Used to Group together movements (particularly images)        ///     when transactions are uploaded from Mobile Devices        /// </summary>        [EntityRelationship(DeleteAction.Cascade)]        public StockMovementBatchLink Batch { get; set; }                [NullEditor]        [Obsolete("Replaced with Dimensions", true)]        public double UnitSize { get; set; }                /// <summary>        /// Value of a stock movement, equal to <see cref="Units"/> * <see cref="Cost"/>.        /// </summary>        [CurrencyEditor(Visible = Visible.Optional, Editable = Editable.Hidden, Summary=Summary.Sum)]        [Formula(typeof(StockMovementValueFormula))]        public double Value { get; set; } = 0.0;        [NullEditor]        [LoggableProperty]        public DateTime Posted { get; set; }        [NullEditor]        [LoggableProperty]        [RequiredColumn]        public PostedStatus PostedStatus { get; set; }        [NullEditor]        public string PostedNote { get; set; }        [NullEditor]        public string PostedReference { get; set; }        static StockMovement()        {            StockEntity.LinkStockDimensions<StockMovement>();            LinkedProperties.Register<StockMovement, ProductStyleLink, Guid>(x=>x.Product.DefaultInstance.Style, x => x.ID, x => x.Style.ID);            //LinkedProperties.Register<StockMovement, ProductLink, double>(x => x.Product, x => x.AverageCost, x => x.Cost);            LinkedProperties.Register<StockMovement, JobScopeLink, Guid>(ass => ass.Job.DefaultScope, scope => scope.ID, ass => ass.JobScope.ID);            LinkedProperties.Register<StockMovement, JobScopeLink, String>(ass => ass.Job.DefaultScope, scope => scope.Number, ass => ass.JobScope.Number);            LinkedProperties.Register<StockMovement, JobScopeLink, String>(ass => ass.Job.DefaultScope, scope => scope.Description, ass => ass.JobScope.Description);        }        private static Column<StockMovement> unitsize = new Column<StockMovement>(x => x.Dimensions.Value);        private static Column<StockMovement> issued = new Column<StockMovement>(x => x.Issued);        private static Column<StockMovement> received = new Column<StockMovement>(x => x.Received);        private bool bChanging;        protected override void DoPropertyChanged(string name, object? before, object? after)        {            if (bChanging)                return;            if (unitsize.IsEqualTo(name))            {                bChanging = true;                Qty = (Received - Issued) * (double)after;                bChanging = false;            }            else if (issued.IsEqualTo(name))            {                bChanging = true;                Units = Received - (double)after;                Qty = Units * Dimensions.Value;                bChanging = false;            }            else if (received.IsEqualTo(name))            {                bChanging = true;                Units = ((double)after - Issued);                Qty = Units * Dimensions.Value;                bChanging = false;            }            base.DoPropertyChanged(name, before, after);        }    }}
 |