yanjinhui
2025-10-18 d48e486d63ae2a0fc454c27c79b8f12115e4bcbe
н¨Îļþ¼Ð/WIDESEA_WMSServer/WIDESEA_SquareCabinServices/DeliveryOrderServices.cs
@@ -176,288 +176,11 @@
        //}
        public WebResponseContent GetUpstreamOutOrder()
        {
            var responseContent = new WebResponseContent();
            try
            {
                var url = "http://121.37.118.63:80/GYZ2/95fck/outOrder";
                var requestData = new
                {
                    searchDate = "2022-10-10 20:45:16"
                };
                var result = HttpHelper.Post(url, requestData.ToJsonString());
                var response = JsonConvert.DeserializeObject<UpstreamResponse<UpstramOutOrderInfo>>(result);
                if (response.resultCode != "0")
                {
                    SendErrorToUpstream(3, "", response.resultMsg ?? "上游接口返回失败", "");
                    return responseContent.Error(response.resultMsg ?? "上游接口返回失败");
                }
                if (response.data == null || !response.data.Any())
                {
                    return responseContent.OK("无新出库单数据");
                }
                // å·²å­˜åœ¨å‡ºåº“单号
                var existingOutOrderNos = BaseDal.Db.Queryable<Dt_DeliveryOrder>()
                    .Select(x => x.Out_no)
                    .ToList();
                // æ–°å‡ºåº“单
                var newOutOrders = response.data
                    .Where(outorder => !existingOutOrderNos.Contains(outorder.order_no))
                    .ToList();
                if (!newOutOrders.Any())
                {
                    return responseContent.OK("所有出库单已存在,无需新增");
                }
                Db.Ado.BeginTran();
                try
                {
                    List<Dt_DeliveryOrder> _DeliveryOrders = new List<Dt_DeliveryOrder>();
                    foreach (var outorder in newOutOrders)
                    {
                        var entityOrder = new Dt_DeliveryOrder
                        {
                            Out_no = outorder.order_no,
                            Out_type = outorder.order_type,
                            Client_no = outorder.client_no,
                            Client_name = outorder.client_name,
                            Account_time = outorder.account_time,
                            Warehouse_no = outorder.warehouse_no,
                            OutStatus = "未完成",
                            Details = new List<Dt_DeliveryOrderDetail>()
                        };
                        // éåŽ†å‡ºåº“æ˜Žç»†
                        foreach (var item in outorder.details)
                        {
                            //  1️⃣ è½¬ä¸ºæ­£æ•°ï¼ˆä¸Šæ¸¸å¯èƒ½ä¼ è´Ÿæ•°ï¼‰
                            item.order_qty = Math.Abs(item.order_qty);
                            decimal orderQty = (decimal)item.order_qty;
                            //  2️⃣ æŸ¥ç‰©æ–™åŸºç¡€ä¿¡æ¯ï¼ˆèŽ·å–ç®±è§„ï¼‰
                            var medication = BaseDal.Db.Queryable<Dt_MedicineGoods>()
                                .Where(m => m.Goods_no == item.goods_no)
                                .First();
                            if (medication == null)
                            {
                                SendErrorToUpstream(3, "", $"找不到物料信息:{item.goods_no}", "");
                                continue;
                            }
                            decimal boxQty = medication.BoxQty <= 0 ? 1 : medication.BoxQty;
                            //  3️⃣ è®¡ç®—整件与散件数量
                            var fullBoxes = (int)(orderQty / boxQty);   // æ•´ä»¶ç®±æ•°
                            var partialQty = orderQty % boxQty;         // æ•£ä»¶æ•°é‡
                            // è‹¥ä¸Šæ¸¸ç»™äº†æ‰¹æ¬¡å·ï¼Œåˆ™ä¼˜å…ˆåŒ¹é…
                            string requestedBatch = string.IsNullOrEmpty(item.batch_num) ? null : item.batch_num;
                            // helper: æŸ¥è¯¢åº“存函数(按入库时间升序)
                            Func<int, List<Dt_InventoryInfo>> queryInventoryByStockStatus = (stockStatus) =>
                            {
                                var q = BaseDal.Db.Queryable<Dt_InventoryInfo>()
                                    .Where(i => i.MaterielCode == item.goods_no &&
                                                (i.StockQuantity - i.OutboundQuantity) > 0 &&
                                                i.StockStatus == stockStatus);
                                if (!string.IsNullOrEmpty(requestedBatch))
                                    q = q.Where(i => i.BatchNo == requestedBatch);
                                return q.OrderBy(i => i.InDate).ToList();
                            };
                            // 4️⃣分配散件(优先立库 Status=0)
                            decimal remainingPartial = partialQty;
                            if (remainingPartial > 0)
                            {
                                var invList_ly = queryInventoryByStockStatus(1);
                                foreach (var inv in invList_ly)
                                {
                                    if (remainingPartial <= 0) break;
                                    decimal available = (decimal)(inv.StockQuantity - inv.OutboundQuantity);
                                    if (available <= 0) continue;
                                    decimal use = Math.Min(available, remainingPartial);
                                    var detail = new Dt_DeliveryOrderDetail
                                    {
                                        Goods_no = item.goods_no,
                                        Order_qty = use,
                                        Batch_num = inv.BatchNo,
                                        Exp_date = inv.ValidityPeriod,
                                        OotDetailStatus = "新建",
                                        Status = 0, //立库
                                        Reservoirarea = inv.LocationCode
                                    };
                                    entityOrder.Details.Add(detail);
                                    inv.OutboundQuantity += use;
                                    BaseDal.Db.Updateable(inv).ExecuteCommand();
                                    remainingPartial -= use;
                                }
                                // ç«‹åº“不够 â†’ å¹³åº“è¡¥ (Status=2)
                                if (remainingPartial > 0)
                                {
                                    var invList_pk = queryInventoryByStockStatus(2);
                                    foreach (var inv in invList_pk)
                                    {
                                        if (remainingPartial <= 0) break;
                                        decimal available = (decimal)(inv.StockQuantity - inv.OutboundQuantity);
                                        if (available <= 0) continue;
                                        decimal use = Math.Min(available, remainingPartial);
                                        var detail = new Dt_DeliveryOrderDetail
                                        {
                                            Goods_no = item.goods_no,
                                            Order_qty = use,
                                            Batch_num = inv.BatchNo,
                                            Exp_date = inv.ValidityPeriod,
                                            OotDetailStatus = "新建",
                                            Status = 2, //平库
                                            Reservoirarea = inv.LocationCode
                                        };
                                        entityOrder.Details.Add(detail);
                                        inv.OutboundQuantity += use;
                                        BaseDal.Db.Updateable(inv).ExecuteCommand();
                                        remainingPartial -= use;
                                    }
                                }
                                if (remainingPartial > 0)
                                {
                                    SendErrorToUpstream(3, "", $"出库单[{outorder.order_no}]物料[{item.goods_no}]散件库存不足,未分配:{remainingPartial}", "");
                                }
                            }
                            //分配整件(优先平库 Status=2)
                            int remainingFullBoxes = fullBoxes;
                            if (remainingFullBoxes > 0)
                            {
                                var invList_pk = queryInventoryByStockStatus(2);
                                foreach (var inv in invList_pk)
                                {
                                    if (remainingFullBoxes <= 0) break;
                                    decimal available = (decimal)(inv.StockQuantity - inv.OutboundQuantity);
                                    if (available < boxQty) continue;
                                    int canProvideBoxes = (int)(available / boxQty);
                                    if (canProvideBoxes <= 0) continue;
                                    int useBoxes = Math.Min(canProvideBoxes, remainingFullBoxes);
                                    decimal useQty = useBoxes * boxQty;
                                    var detail = new Dt_DeliveryOrderDetail
                                    {
                                        Goods_no = item.goods_no,
                                        Order_qty = useQty,
                                        Batch_num = inv.BatchNo,
                                        Exp_date = inv.ValidityPeriod,
                                        OotDetailStatus = "新建",
                                        Status = 2, //平库
                                        Reservoirarea = inv.LocationCode
                                    };
                                    entityOrder.Details.Add(detail);
                                    inv.OutboundQuantity += useQty;
                                    BaseDal.Db.Updateable(inv).ExecuteCommand();
                                    remainingFullBoxes -= useBoxes;
                                }
                                // å¹³åº“不够 â†’ ç«‹åº“è¡¥ (Status=1)
                                if (remainingFullBoxes > 0)
                                {
                                    var invList_ly = queryInventoryByStockStatus(1);
                                    foreach (var inv in invList_ly)
                                    {
                                        if (remainingFullBoxes <= 0) break;
                                        decimal available = (decimal)(inv.StockQuantity - inv.OutboundQuantity);
                                        if (available < boxQty) continue;
                                        int canProvideBoxes = (int)(available / boxQty);
                                        if (canProvideBoxes <= 0) continue;
                                        int useBoxes = Math.Min(canProvideBoxes, remainingFullBoxes);
                                        decimal useQty = useBoxes * boxQty;
                                        var detail = new Dt_DeliveryOrderDetail
                                        {
                                            Goods_no = item.goods_no,
                                            Order_qty = useQty,
                                            Batch_num = inv.BatchNo,
                                            Exp_date = inv.ValidityPeriod,
                                            OotDetailStatus = "新建",
                                            Status = 0, //立库(补整箱)
                                            Reservoirarea = inv.LocationCode
                                        };
                                        entityOrder.Details.Add(detail);
                                        inv.OutboundQuantity += useQty;
                                        BaseDal.Db.Updateable(inv).ExecuteCommand();
                                        remainingFullBoxes -= useBoxes;
                                    }
                                }
                                if (remainingFullBoxes > 0)
                                {
                                    decimal unfilledQty = remainingFullBoxes * boxQty;
                                    SendErrorToUpstream(3, "", $"出库单[{outorder.order_no}]物料[{item.goods_no}]整箱库存不足,未分配数量:{unfilledQty}", "");
                                }
                            }
                        }
                        _DeliveryOrders.Add(entityOrder);
                    }
                    // æ’入主表+明细
                    BaseDal.Db.InsertNav(_DeliveryOrders)
                        .Include(x => x.Details)
                        .ExecuteCommand();
                    Db.Ado.CommitTran();
                    return responseContent.OK("同步出库单成功");
                }
                catch (Exception ex)
                {
                    Db.Ado.RollbackTran();
                    SendErrorToUpstream(3, "", ex.Message, "");
                    return responseContent.Error("同步失败: " + ex.Message);
                }
            }
            catch (Exception ex)
            {
                SendErrorToUpstream(3, "", ex.Message, "");
                return responseContent.Error("同步失败: " + ex.Message);
            }
        }
        /// <summary>
        /// è¿‡æ»¤æ²¡æœ‰å¯¹å•†å“è¿›è¡Œç»´æŠ¤çš„出库单进行过滤
        /// </summary>
        /// <returns></returns>
        public WebResponseContent GetUpstreamOutOrder2()
        public WebResponseContent GetUpstreamOutOrder()
        {
            var responseContent = new WebResponseContent();
            try
@@ -525,12 +248,15 @@
                        {
                            //  1️⃣ è½¬ä¸ºæ­£æ•°ï¼ˆä¸Šæ¸¸å¯èƒ½ä¼ è´Ÿæ•°ï¼‰
                            item.order_qty = Math.Abs(item.order_qty);
                            //出库数量
                            decimal orderQty = (decimal)item.order_qty;
                            //  2️⃣ æŸ¥ç‰©æ–™åŸºç¡€ä¿¡æ¯ï¼ˆèŽ·å–ç®±è§„ï¼‰
                            var medication = BaseDal.Db.Queryable<Dt_MaterielInfo>()
                                .Where(m => m.MaterielCode == item.goods_no)
                                .First();
                            #region æ£€æŸ¥è¿™ä¸ªå‡ºåº“单中的物料信息是否存在
                            //如果物料信息不存在,跳过整个入库单
                            if (medication == null)
                            {
@@ -547,12 +273,15 @@
                                break;
                            }
                            if (medication == null)
                            {
                                SendErrorToUpstream(3, "", $"找不到物料信息:{item.goods_no}", "");
                                continue;
                            }
                            #endregion
                            // å¦‚果箱规小于等于0,使用默认值1,否则就用实际数量
                            decimal boxQty = medication.BoxQty <= 0 ? 1 : medication.BoxQty;
                            //  3️⃣ è®¡ç®—整件与散件数量
@@ -563,34 +292,61 @@
                            string requestedBatch = string.IsNullOrEmpty(item.batch_num) ? null : item.batch_num;
                            // helper: æŸ¥è¯¢åº“存函数(按入库时间升序)
                            #region ä½¿ç”¨åº“存信息中的stockStatus来区分平库和立库
                            // å®šä¹‰ä¸€ä¸ªå§”托变量:输入int参数,返回List<Dt_InventoryInfo>
                            Func<int, List<Dt_InventoryInfo>> queryInventoryByStockStatus = (stockStatus) =>
                            {
                                // 1. åˆ›å»ºåŸºç¡€æŸ¥è¯¢
                                var q = BaseDal.Db.Queryable<Dt_InventoryInfo>()
                                    .Where(i => i.MaterielCode == item.goods_no &&           // ç‰©æ–™ç¼–号匹配
                                                (i.StockQuantity - i.OutboundQuantity) > 0 && // å¯ç”¨åº“å­˜>0
                                                i.StockStatus == stockStatus);               // åº“存状态匹配
                            //Func<int, List<Dt_InventoryInfo>> queryInventoryByStockStatus = (stockStatus) =>
                            //{
                            //    // 1. åˆ›å»ºåŸºç¡€æŸ¥è¯¢
                            //    var q = BaseDal.Db.Queryable<Dt_InventoryInfo>()
                            //        .Where(i => i.MaterielCode == item.goods_no &&           // ç‰©æ–™ç¼–号匹配
                            //                    (i.StockQuantity - i.OutboundQuantity) > 0 && // å¯ç”¨åº“å­˜>0
                            //                    i.StockStatus == stockStatus);               // åº“存状态匹配
                                // 2. æ¡ä»¶ç­›é€‰ï¼ˆå¦‚果有批次号要求)
                            //    // 2. æ¡ä»¶ç­›é€‰ï¼ˆå¦‚果有批次号要求)
                            //    if (!string.IsNullOrEmpty(requestedBatch))
                            //        q = q.Where(i => i.BatchNo == requestedBatch);
                            //    // 3. æ‰§è¡ŒæŸ¥è¯¢å¹¶è¿”回结果
                            //    return q.OrderBy(i => i.InDate).ToList();  // æŒ‰å…¥åº“时间排序
                            //};
                            #endregion
                            #region ä½¿ç”¨åº“房编号来区分是平库和立库
                            // helper: æŸ¥è¯¢åº“存函数(按入库时间升序)
                            // ç”¨åº“房编号区分立库和平库:立库 WarehouseCode == "001" è¾“å…¥true就是立库否则就是平库
                            Func<bool, List<Dt_InventoryInfo>> queryInventoryByWarehouseType = (isLiku) =>
                            {
                                var q = BaseDal.Db.Queryable<Dt_InventoryInfo>()
                                    .Where(i => i.MaterielCode == item.goods_no &&
                                                (i.StockQuantity - i.OutboundQuantity) > 0);
                                if (isLiku)
                                    q = q.Where(i => i.WarehouseCode == "001"); // ç«‹åº“
                                else
                                    q = q.Where(i => i.WarehouseCode != "001"); // å¹³åº“
                                if (!string.IsNullOrEmpty(requestedBatch))
                                    q = q.Where(i => i.BatchNo == requestedBatch);
                                // 3. æ‰§è¡ŒæŸ¥è¯¢å¹¶è¿”回结果
                                return q.OrderBy(i => i.InDate).ToList();  // æŒ‰å…¥åº“时间排序
                                return q.OrderBy(i => i.InDate).ToList();
                            };
                            // 4️⃣分配散件(优先立库 Status=0)
                            decimal remainingPartial = partialQty;
                            #endregion
                            // 4️⃣分配散件(优先立库 WarehouseCode == "001)
                            decimal remainingPartial = partialQty; //散件
                            if (remainingPartial > 0)
                            {
                                var invList_ly = queryInventoryByStockStatus(0);
                                //返回一个List<Dt_InventoryInfo>stockStatus==0的库存列表
                                var invList_ly = queryInventoryByWarehouseType(true); //立库 WarehouseCode == "001"
                                foreach (var inv in invList_ly)
                                {
                                    if (remainingPartial <= 0) break;
                                    //可用库存数量
                                    decimal available = (decimal)(inv.StockQuantity - inv.OutboundQuantity);
                                    if (available <= 0) continue;
                                    //表示「当前还需要出库的数量」或「待分配数量」。
                                    decimal use = Math.Min(available, remainingPartial);
                                    var detail = new Dt_DeliveryOrderDetail
@@ -601,7 +357,7 @@
                                        Exp_date = inv.ValidityPeriod,
                                        OotDetailStatus = "新建",
                                        Status = 0, //立库
                                        Reservoirarea = inv.LocationCode
                                        Reservoirarea = inv.WarehouseCode
                                    };
                                    entityOrder.Details.Add(detail);
@@ -614,7 +370,7 @@
                                // ç«‹åº“不够 â†’ å¹³åº“è¡¥ (Status=2)
                                if (remainingPartial > 0)
                                {
                                    var invList_pk = queryInventoryByStockStatus(2);
                                    var invList_pk = queryInventoryByWarehouseType(false); //平库
                                    foreach (var inv in invList_pk)
                                    {
                                        if (remainingPartial <= 0) break;
@@ -632,7 +388,7 @@
                                            Exp_date = inv.ValidityPeriod,
                                            OotDetailStatus = "新建",
                                            Status = 2, //平库
                                            Reservoirarea = inv.LocationCode
                                            Reservoirarea = inv.WarehouseCode
                                        };
                                        entityOrder.Details.Add(detail);
@@ -650,10 +406,10 @@
                            }
                            //分配整件(优先平库 Status=2)
                            int remainingFullBoxes = fullBoxes;
                            int remainingFullBoxes = fullBoxes;//整件箱数
                            if (remainingFullBoxes > 0)
                            {
                                var invList_pk = queryInventoryByStockStatus(2);
                                var invList_pk = queryInventoryByWarehouseType(true);
                                foreach (var inv in invList_pk)
                                {
                                    if (remainingFullBoxes <= 0) break;
@@ -675,7 +431,7 @@
                                        Exp_date = inv.ValidityPeriod,
                                        OotDetailStatus = "新建",
                                        Status = 2, //平库
                                        Reservoirarea = inv.LocationCode
                                        Reservoirarea = inv.WarehouseCode
                                    };
                                    entityOrder.Details.Add(detail);
@@ -685,10 +441,10 @@
                                    remainingFullBoxes -= useBoxes;
                                }
                                // å¹³åº“不够 â†’ ç«‹åº“è¡¥ (Status=1)
                                // å¹³åº“不够 â†’ ç«‹åº“è¡¥ (Status=0)
                                if (remainingFullBoxes > 0)
                                {
                                    var invList_ly = queryInventoryByStockStatus(1);
                                    var invList_ly = queryInventoryByWarehouseType(true); //立库
                                    foreach (var inv in invList_ly)
                                    {
                                        if (remainingFullBoxes <= 0) break;
@@ -843,7 +599,6 @@
                              .SetColumns(d => new Dt_DeliveryOrderDetail { Status = 1, OotDetailStatus = "已完成" })
                              .Where(d => d.DeliveryOrderId == order.Id && d.Status == 0)
                              .ExecuteCommand();
                            Console.WriteLine($"订单 {order.Out_no} æŽ¨é€æˆåŠŸ");
                        }
                        else
@@ -1132,8 +887,9 @@
            }
            return content;
        }
        /// <summary>
        /// æŸ¥è¯¢å‡ºåº“/盘点单详情
        /// æŸ¥è¯¢å‡ºåº“/盘点单详情 çœ‹å‡ºåº“单明细。
        /// </summary>
        /// <param name="pageNo"></param>
        /// <param name="orderNo"></param>
@@ -1151,6 +907,7 @@
            content.OK(data: cabinOrderDetails);
            return content;
        }
        public WebResponseContent OutFinish(SaveModel saveModel)
        {
            WebResponseContent content = new WebResponseContent();
@@ -1244,6 +1001,12 @@
            return content;
        }
        /// <summary>
        /// å¹³åº“人工拣料出库处理(即人工扫码出库时调用)
        /// </summary>
        /// <param name="saveModel"></param>
        /// <returns></returns>
        public WebResponseContent MatPicking(SaveModel saveModel)
        {
            WebResponseContent content = new WebResponseContent();