| using System; | 
| using System.Collections.Generic; | 
| using System.Linq; | 
| using System.Text; | 
| using System.Threading.Tasks; | 
| using WIDESEA_Common.CommonEnum; | 
| using WIDESEA_Common.LocationEnum; | 
| using WIDESEA_Core.BaseServices; | 
| using WIDESEA_Core.Helper; | 
| using WIDESEA_IBasicRepository; | 
| using WIDESEA_IBasicService; | 
| using WIDESEA_Model.Models; | 
|   | 
| namespace WIDESEA_BasicService | 
| { | 
|     public partial class LocationInfoService | 
|     { | 
|   | 
|         double weightValue_GM = 0.25; | 
|   | 
|         private readonly static object _locker_GM = new object(); | 
|         static List<LocationCache> locationCaches_GM = new List<LocationCache>(); | 
|   | 
|         /// <summary> | 
|         /// 干膜仓货位分配 | 
|         /// </summary> | 
|         /// <param name="roadwayNo">巷道号</param> | 
|         /// <param name="palletType"> | 
|         /// 托盘类型  | 
|         /// </param> | 
|         /// <returns></returns> | 
|         public Dt_LocationInfo? AssignLocation_GM(string roadwayNo, int palletType, Dt_PalletTypeInfo palletTypeInfo, string beRelocationCode = "") | 
|         { | 
|             lock (_locker_GM) | 
|             { | 
|                 List<LocationCache> removeItems = locationCaches_GM.Where(x => (DateTime.Now - x.DateTime).TotalMinutes > 5).ToList(); | 
|                 int count = removeItems.Count; | 
|                 for (int i = 0; i < count; i++) | 
|                 { | 
|                     locationCaches_GM.Remove(removeItems[i]); | 
|                 } | 
|   | 
|                 List<string> lockLocationCodes = locationCaches_GM.Select(x => x.LocationCode).ToList(); | 
|   | 
|                 List<Dt_LocationInfo> locationInfos = Repository.QueryData(x => x.RoadwayNo == roadwayNo); | 
|                 //获取所有货位数据 | 
|                 List<Dt_LocationInfo> locationInfoDepth = Repository.QueryData(x => x.RoadwayNo == roadwayNo); | 
|                 if (locationInfos == null || locationInfos.Count == 0) | 
|                 { | 
|                     throw new Exception($"未找到该巷道的货位信息,巷道号:{roadwayNo}"); | 
|                 } | 
|   | 
|                 if (!string.IsNullOrEmpty(beRelocationCode)) | 
|                 { | 
|                     Dt_LocationInfo? beRelocation = locationInfos.FirstOrDefault(x => x.LocationCode == beRelocationCode); | 
|                     if (beRelocation == null) | 
|                     { | 
|                         throw new Exception($"未找到货位信息"); | 
|                     } | 
|                     int maxDepth = locationInfos.Max(x => x.Depth); | 
|                     //int mathCurrentRow = beRelocation.Row - Convert.ToInt32(Math.Ceiling(beRelocation.Row / maxDepth / 2.0)) * maxDepth * 2; | 
|                     //if (mathCurrentRow <= maxDepth) | 
|                     //{ | 
|                     //    locationInfos = locationInfos.Where(x => x.Row - Convert.ToInt32(Math.Ceiling(x.Row / maxDepth / 2.0)) * maxDepth * 2 <= maxDepth).ToList(); | 
|                          | 
|                     //} | 
|                     //else | 
|                     //{ | 
|                     //    locationInfos = locationInfos.Where(x => x.Row - Convert.ToInt32(Math.Ceiling(x.Row / maxDepth / 2.0)) * maxDepth * 2 > maxDepth).ToList(); | 
|                          | 
|                     //} | 
|                     if (beRelocation.Row <= maxDepth) | 
|                     { | 
|                         locationInfos = locationInfos.Where(x => x.Row <= maxDepth).ToList(); | 
|                     } | 
|                     else if (beRelocation.Row > maxDepth) | 
|                     { | 
|                         locationInfos = locationInfos.Where(x => x.Row > maxDepth).ToList(); | 
|                     } | 
|                 } | 
|   | 
|                 //已定义货位类型的货位 | 
|                 List<Dt_LocationInfo> definedTypeLocations = locationInfos.Where(x => x.LocationType == palletType.ObjToInt()).ToList(); | 
|   | 
|                 //未定义类型的空货位 | 
|                 List<Dt_LocationInfo> undefinedTypeEmptyLocations = locationInfos.Where(x => (x.EnableStatus == EnableStatusEnum.Normal.ObjToInt() || x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt()) && x.LocationStatus == LocationStatusEnum.Free.ObjToInt() && x.LocationType == LocationTypeEnum.Undefined.ObjToInt() && !lockLocationCodes.Contains(x.LocationCode)).OrderByDescending(x => x.Depth).ThenBy(x => x.Layer).ThenBy(x => x.Column).ThenBy(x => x.Row).ToList(); | 
|   | 
|                 List<Dt_LocationInfo> definedTypeEmptyLocations = locationInfos.Where(x => (x.EnableStatus == EnableStatusEnum.Normal.ObjToInt() || x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt()) && x.LocationStatus == LocationStatusEnum.Free.ObjToInt() && x.LocationType == palletType.ObjToInt() && !lockLocationCodes.Contains(x.LocationCode)).OrderByDescending(x => x.Depth).ThenBy(x => x.Layer).ThenBy(x => x.Column).ThenBy(x => x.Row).ToList(); | 
|   | 
|                 if (definedTypeEmptyLocations.Any()) | 
|                 { | 
|                     for (int i = 0; i < definedTypeEmptyLocations.Count; i++) | 
|                     { | 
|                         Dt_LocationInfo definedTypeEmptyLocation = definedTypeEmptyLocations[i]; | 
|                         Dt_LocationInfo? locationInfo = GetUsableLocation_GM(locationInfos, definedTypeEmptyLocation, palletType, palletTypeInfo); | 
|                         if (locationInfo != null) | 
|                         { | 
|                             if (locationInfo.Depth<locationInfoDepth.Max(x=>x.Depth)) | 
|                             { | 
|                                 Dt_LocationInfo? locationInfoExist = null; | 
|                                 int Column = locationInfo.Column; | 
|                                 if (Column % 2 == 0) | 
|                                 { | 
|                                     Column -= 1; | 
|                                 } | 
|                                 if (locationInfo.Row - locationInfo.Depth == locationInfo.Depth) | 
|                                 { | 
|                                     locationInfoExist = locationInfoDepth.FirstOrDefault(x => x.Row == (locationInfo.Row-1) && x.Column== Column && x.Layer==locationInfo.Layer); | 
|                                 } | 
|                                 else | 
|                                 { | 
|                                     locationInfoExist = locationInfoDepth.FirstOrDefault(x => x.Row == (locationInfo.Row + 1) && x.Column == Column && x.Layer == locationInfo.Layer); | 
|                                 } | 
|                                 //获取深位货位类型 | 
|                                 if (locationInfoExist==null) | 
|                                 { | 
|                                     continue; | 
|                                 } | 
|                                 Dt_PalletTypeInfo palletTypeInfoDepth = _basicRepository.PalletTypeInfoRepository.QueryFirst(x => x.WarehouseId == locationInfoExist.WarehouseId && x.PalletType == locationInfoExist.LocationType); | 
|                                 if ((palletTypeInfoDepth != null && palletTypeInfoDepth.LocaitonCount != palletTypeInfo.LocaitonCount) || locationInfoExist.LocationStatus < LocationStatusEnum.InStock.ObjToInt()) | 
|                                 { | 
|                                     continue; | 
|                                 } | 
|                             } | 
|                             else | 
|                             { | 
|                                 Dt_LocationInfo? locationInfoExist = null; | 
|                                 int Column = locationInfo.Column; | 
|                                 if (Column % 2 == 0) | 
|                                 { | 
|                                     Column -= 1; | 
|                                 } | 
|                                 if (locationInfo.Row % 2 == 0) | 
|                                 { | 
|                                     locationInfoExist = locationInfoDepth.FirstOrDefault(x => x.Row == (locationInfo.Row - 1) && x.Column == Column && x.Layer == locationInfo.Layer); | 
|                                 } | 
|                                 else | 
|                                 { | 
|                                     locationInfoExist = locationInfoDepth.FirstOrDefault(x => x.Row == (locationInfo.Row + 1) && x.Column == Column && x.Layer == locationInfo.Layer); | 
|                                 } | 
|                                 //获取深位货位类型 | 
|                                 if (locationInfoExist == null) | 
|                                 { | 
|                                     continue; | 
|                                 } | 
|                                 Dt_PalletTypeInfo palletTypeInfoDepth = _basicRepository.PalletTypeInfoRepository.QueryFirst(x => x.WarehouseId == locationInfoExist.WarehouseId && x.PalletType == locationInfoExist.LocationType); | 
|                                 if (palletTypeInfoDepth != null && palletTypeInfoDepth.LocaitonCount != palletTypeInfo.LocaitonCount) | 
|                                 { | 
|                                     continue; | 
|                                 } | 
|                                 if (locationInfoExist.LocationStatus!=LocationStatusEnum.Free.ObjToInt() || locationInfoExist.EnableStatus!= EnableStatusEnum.Normal.ObjToInt()) | 
|                                 { | 
|                                     continue; | 
|                                 } | 
|                             } | 
|                             locationCaches_GM.Add(new LocationCache { LocationCode = locationInfo.LocationCode, DateTime = DateTime.Now }); | 
|                             return locationInfo; | 
|                         } | 
|                     } | 
|                 } | 
|                 if ((locationInfos.Count * weightValue_GM >= definedTypeLocations.Count && undefinedTypeEmptyLocations.Any()) || !definedTypeEmptyLocations.Any())//如果已定义类型货位未超过比例,且有未定义类型的货位 | 
|                 { | 
|                     if (palletTypeInfo.LocaitonCount == 2) | 
|                     { | 
|                         if (palletTypeInfo.IsOdd) | 
|                             undefinedTypeEmptyLocations = undefinedTypeEmptyLocations.Where(x => x.Column % 2 == 1).ToList(); | 
|                         else | 
|                             undefinedTypeEmptyLocations = undefinedTypeEmptyLocations.Where(x => x.Column % 2 == 0).ToList(); | 
|                     } | 
|                     for (int i = 0; i < undefinedTypeEmptyLocations.Count; i++) | 
|                     { | 
|                         Dt_LocationInfo undefinedTypeEmptyLocation = undefinedTypeEmptyLocations[i]; | 
|                         Dt_LocationInfo? locationInfo = GetUsableLocation_GM(locationInfos, undefinedTypeEmptyLocation, palletType, palletTypeInfo); | 
|                         if (locationInfo != null) | 
|                         { | 
|                             if (locationInfo.Depth < locationInfoDepth.Max(x => x.Depth)) | 
|                             { | 
|                                 Dt_LocationInfo? locationInfoExist = null; | 
|                                 int Column = locationInfo.Column; | 
|                                 if (Column % 2 == 0) | 
|                                 { | 
|                                     Column -= 1; | 
|                                 } | 
|                                 if (locationInfo.Row - locationInfo.Depth == locationInfo.Depth) | 
|                                 { | 
|                                     locationInfoExist = locationInfoDepth.FirstOrDefault(x => x.Row == (locationInfo.Row - 1) && x.Column == Column && x.Layer == locationInfo.Layer); | 
|                                 } | 
|                                 else | 
|                                 { | 
|                                     locationInfoExist = locationInfoDepth.FirstOrDefault(x => x.Row == (locationInfo.Row + 1) && x.Column == Column && x.Layer == locationInfo.Layer); | 
|                                 } | 
|                                 //获取深位货位类型 | 
|                                 if (locationInfoExist == null) | 
|                                 { | 
|                                     continue; | 
|                                 } | 
|                                 Dt_PalletTypeInfo palletTypeInfoDepth = _basicRepository.PalletTypeInfoRepository.QueryFirst(x => x.WarehouseId == locationInfoExist.WarehouseId && x.PalletType == locationInfoExist.LocationType); | 
|                                 if (palletTypeInfoDepth != null && palletTypeInfoDepth.LocaitonCount != palletTypeInfo.LocaitonCount || locationInfoExist.LocationStatus < LocationStatusEnum.InStock.ObjToInt()) | 
|                                 { | 
|                                     continue; | 
|                                 } | 
|                             } | 
|                             else | 
|                             { | 
|                                 Dt_LocationInfo? locationInfoExist = null; | 
|                                 int Column = locationInfo.Column; | 
|                                 if (Column % 2 == 0) | 
|                                 { | 
|                                     Column -= 1; | 
|                                 } | 
|                                 if (locationInfo.Row % 2 == 0) | 
|                                 { | 
|                                     locationInfoExist = locationInfoDepth.FirstOrDefault(x => x.Row == (locationInfo.Row - 1) && x.Column == Column && x.Layer == locationInfo.Layer); | 
|                                 } | 
|                                 else | 
|                                 { | 
|                                     locationInfoExist = locationInfoDepth.FirstOrDefault(x => x.Row == (locationInfo.Row + 1) && x.Column == Column && x.Layer == locationInfo.Layer); | 
|                                 } | 
|                                 //获取深位货位类型 | 
|                                 if (locationInfoExist == null) | 
|                                 { | 
|                                     continue; | 
|                                 } | 
|                                 if (locationInfoExist.LocationType > 0) | 
|                                 { | 
|                                     Dt_PalletTypeInfo palletTypeInfoDepth = _basicRepository.PalletTypeInfoRepository.QueryFirst(x => x.WarehouseId == locationInfoExist.WarehouseId && x.PalletType == locationInfoExist.LocationType); | 
|                                     if (palletTypeInfoDepth != null && palletTypeInfoDepth.LocaitonCount != palletTypeInfo.LocaitonCount) | 
|                                     { | 
|                                         continue; | 
|                                     } | 
|                                 } | 
|                                 if (locationInfoExist.LocationStatus != LocationStatusEnum.Free.ObjToInt() || locationInfoExist.EnableStatus != EnableStatusEnum.Normal.ObjToInt()) | 
|                                 { | 
|                                     continue; | 
|                                 } | 
|                                  | 
|                                  | 
|                             } | 
|                             locationCaches_GM.Add(new LocationCache { LocationCode = locationInfo.LocationCode, DateTime = DateTime.Now }); | 
|                             return locationInfo; | 
|                         } | 
|                     } | 
|                 } | 
|                 ////判断如果已定义货位类型的货位和未定义类型的空货位都为空释放满足条件的货位 | 
|                 //if (!definedTypeEmptyLocations.Any() && !undefinedTypeEmptyLocations.Any()) | 
|                 //{ | 
|   | 
|                 //} | 
|                 return null; | 
|             } | 
|         } | 
|   | 
|         /// <summary> | 
|         /// 获取可用货位(测试架仓) | 
|         /// </summary> | 
|         /// <param name="locationInfos"></param> | 
|         /// <param name="emptyLocation"></param> | 
|         /// <param name="palletType"></param> | 
|         /// <returns></returns> | 
|         private Dt_LocationInfo? GetUsableLocation_GM(List<Dt_LocationInfo> locationInfos, Dt_LocationInfo emptyLocation, int palletType, Dt_PalletTypeInfo palletTypeInfo) | 
|         { | 
|             if (palletTypeInfo.LocaitonCount == 2) | 
|             { | 
|                 if (palletTypeInfo.IsOdd && emptyLocation.Column % 2 != 1 || !palletTypeInfo.IsOdd && emptyLocation.Column % 2 != 0) | 
|                     return null; | 
|   | 
|                 Dt_LocationInfo? nearLocation = locationInfos.FirstOrDefault(x => x.Row == emptyLocation.Row && x.Layer == emptyLocation.Layer && x.Depth == emptyLocation.Depth && x.Column == emptyLocation.Column + 1); | 
|                 if (nearLocation != null && DepthLocationIsEmpty_BC(locationInfos, nearLocation) != null) | 
|                 { | 
|                     Dt_LocationInfo? locationInfo = DepthLocationIsEmpty_BC(locationInfos, emptyLocation); | 
|                     if (locationInfo != null) | 
|                     { | 
|                         return locationInfo; | 
|                     } | 
|                 } | 
|             } | 
|             else | 
|             { | 
|                 Dt_LocationInfo? locationInfo = DepthLocationIsEmpty_BC(locationInfos, emptyLocation); | 
|                 if (locationInfo != null) | 
|                 { | 
|                     return locationInfo; | 
|                 } | 
|             } | 
|             return null; | 
|         } | 
|   | 
|   | 
|         /// <summary> | 
|         /// 判断不同深度的同组货位状态是否为空闲空位(测试架仓) | 
|         /// </summary> | 
|         /// <param name="locationInfos"></param> | 
|         /// <param name="emptyLocation"></param> | 
|         /// <returns></returns> | 
|         private Dt_LocationInfo? DepthLocationIsEmpty_GM(List<Dt_LocationInfo> locationInfos, Dt_LocationInfo emptyLocation) | 
|         { | 
|             List<Dt_LocationInfo> locations = GetGroupLocations(locationInfos, emptyLocation); | 
|   | 
|             bool moreDepthFlag = false; | 
|             bool littleDepthFlag = false; | 
|   | 
|             if (emptyLocation.LocationType == 0) | 
|             { | 
|                 List<Dt_LocationInfo> moreDepth = locations.Where(x => x.Depth > emptyLocation.Depth).ToList(); | 
|                 moreDepthFlag = moreDepth.FirstOrDefault(x => x.LocationStatus != LocationStatusEnum.InStock.ObjToInt() && (x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt() || x.EnableStatus == EnableStatusEnum.Normal.ObjToInt()) && x.LocationType != 0) == null;//查询大于当前货位深度的集合里是否有状态不为有货的货位,如果是true,则表示深货位有未被使用的情况 | 
|   | 
|                 List<Dt_LocationInfo> littleDepth = locations.Where(x => x.Depth <= emptyLocation.Depth).ToList(); | 
|                 littleDepthFlag = littleDepth.FirstOrDefault(x => x.LocationStatus != LocationStatusEnum.Free.ObjToInt() && (x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt() || x.EnableStatus == EnableStatusEnum.Normal.ObjToInt()) && x.LocationType != 0) == null; //查询小于当前货位深度的集合里是否有状态不为空,且禁用状态不为禁用以及只入的货位,如果是true,则表示浅货位被使用或者被禁用的情况 | 
|             } | 
|             else | 
|             { | 
|                 List<Dt_LocationInfo> moreDepth = locations.Where(x => x.Depth > emptyLocation.Depth).ToList(); | 
|                 moreDepthFlag = moreDepth.FirstOrDefault(x => x.LocationStatus != LocationStatusEnum.InStock.ObjToInt() && (x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt() || x.EnableStatus == EnableStatusEnum.Normal.ObjToInt()) && x.LocationType != emptyLocation.LocationType) == null;//查询大于当前货位深度的集合里是否有状态不为有货的货位,如果是true,则表示深货位有未被使用的情况 | 
|   | 
|                 List<Dt_LocationInfo> littleDepth = locations.Where(x => x.Depth <= emptyLocation.Depth).ToList(); | 
|                 littleDepthFlag = littleDepth.FirstOrDefault(x => x.LocationStatus != LocationStatusEnum.Free.ObjToInt() && (x.EnableStatus == EnableStatusEnum.OnlyIn.ObjToInt() || x.EnableStatus == EnableStatusEnum.Normal.ObjToInt()) && x.LocationType != emptyLocation.LocationType) == null; //查询小于当前货位深度的集合里是否有状态不为空,且禁用状态不为禁用以及只入的货位,如果是true,则表示浅货位被使用或者被禁用的情况 | 
|             } | 
|   | 
|             if (moreDepthFlag && littleDepthFlag) | 
|             { | 
|                 return emptyLocation; | 
|             } | 
|   | 
|             return null; | 
|         } | 
|     } | 
| } |