RGVSystems.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. using ServiceCenter.Extensions;
  2. using ServiceCenter.Logs;
  3. using ServiceCenter.SqlSugars;
  4. using System.ComponentModel;
  5. using WCS.Core;
  6. using WCS.Entity;
  7. using WCS.Entity.Protocol.Protocol.BCR;
  8. using WCS.Entity.Protocol.Protocol.RGV;
  9. using WCS.Entity.Protocol.Protocol.Robot;
  10. using WCS.Entity.Protocol.Protocol.Station;
  11. using WCS.Entity.Protocol.Protocol.Truss;
  12. using WCS.WorkEngineering.Extensions;
  13. using WCS.WorkEngineering.Worlds;
  14. using DeviceFlags = WCS.WorkEngineering.Extensions.DeviceFlags;
  15. using TaskStatus = WCS.Entity.TaskStatus;
  16. namespace WCS.WorkEngineering.Systems
  17. {
  18. /// <summary>
  19. /// Agv交互系统
  20. /// </summary>
  21. [BelongTo(typeof(RgvWorld))]
  22. [Description("RGV交互系统")]
  23. public class RGVSystems : DeviceSystem<Device<IRGV520, IRGV521, IBCR81>>
  24. {
  25. protected override bool ParallelDo => true;
  26. protected override bool SaveLogsToFile => true;
  27. /// <summary>
  28. /// 取货点设备集合
  29. /// </summary>
  30. private readonly Dictionary<string, List<Device<IStation520, IStation521, IStation523>>> _pickUpDevices = new();
  31. public RGVSystems()
  32. {
  33. //获取所有的巷道集合
  34. var rgvList = Device.All.Where(v => v.HasFlag(DeviceFlags.RGV));
  35. foreach (var rgv in rgvList)
  36. {
  37. _pickUpDevices.Add(rgv.Code, rgv.Sources.Where(v => v.HasFlag(DeviceFlags.输送机)).Select(v => new Device<IStation520, IStation521, IStation523>(v, this.World)).ToList());
  38. }
  39. }
  40. public override void Do(Device<IRGV520, IRGV521, IBCR81> obj)
  41. {
  42. //var dev1 = new Device<IStation520>(Device.All.First(x => x.Code == "1602"), World);
  43. //dev1.Data.TaskNumber = 0;
  44. //dev1.Data.GoodsEnd=0;
  45. if (obj.Data.VoucherNo != obj.Data2.VoucherNo) throw new KnownException($"凭证号不一致,DB520:{obj.Data.VoucherNo}-DB521:{obj.Data2.VoucherNo}", LogLevelEnum.High);
  46. if (obj.Data2.WorkMode != RGVWorkMode.Automatic) return;
  47. if (obj.Data.RES1 == 1) return; //wcs任务完成确认信号未清除
  48. if (obj.Data2.Status.HasFlag(RGVStatus.Taskfinishi))
  49. {
  50. switch (obj.Data2.CmdType)
  51. {
  52. case RGVCmdType.PickGoods: //单独取货任务完成,默认只有空托盘才会下发单独取货任务
  53. //开始申请读码信息
  54. var bcrCode = obj.Data3.GetBCRCode();
  55. int taskNumber = 0;
  56. SqlSugarHelper.Do(_db =>
  57. {
  58. var db = _db.Default;
  59. var dev = Device.All.First(x => x.Code == obj.Data2.StartPosition.ToString());
  60. if (dev.HasFlag(DeviceFlags.拆盘机09))
  61. {
  62. //开始绑定任务,并下发新的任务信息到小车
  63. WCS_Palletizing palletizingInfo = db.Queryable<WCS_Palletizing>().Single(x => x.Id == obj.Data2.TaskNumber);
  64. if (palletizingInfo == null) return;
  65. palletizingInfo.PalleCode = bcrCode;
  66. db.Updateable(palletizingInfo).ExecuteCommand();
  67. taskNumber = palletizingInfo.Id;
  68. }
  69. else if (dev.HasFlag(DeviceFlags.拆盘机非09))
  70. {
  71. //开始处理对应的搬运任务信息
  72. var task = db.Queryable<WCS_TaskInfo>().First(x => x.Type == TaskType.Delivery && x.ID == obj.Data2.TaskNumber && x.AddrTo == obj.Data2.DestPosition.ToString());
  73. if (task == null) throw new KnownException($"未找到对应的任务{obj.Data2.TaskNumber}", LogLevelEnum.Mid);
  74. task.BarCode = bcrCode;
  75. db.Updateable(task).ExecuteCommand();
  76. task.AddWCS_TASK_DTL(db, obj.Entity.Code, obj.Data2.DestPosition.ToString(), $"环形库码垛位{obj.Data2.DestPosition}搬运任务绑定条码信息{bcrCode}");
  77. taskNumber = task.ID;
  78. }
  79. });
  80. if (taskNumber == 0) return;
  81. //清空起始点信息
  82. var staDev = new Device<IStation520>(Device.All.FirstOrDefault(x => x.Code == obj.Data2.DestPosition.ToString())!, World);
  83. staDev.Data.TaskNumber = 0;
  84. staDev.Data.GoodsStart = 0;
  85. staDev.Data.GoodsEnd = 0;
  86. obj.Data2.TaskNumber = taskNumber;
  87. obj.Data.RES1 = 1;
  88. break;
  89. case RGVCmdType.PutGoods:
  90. WCS_TaskInfo finishiTask = null;
  91. var startPosition = Device.All.Where(x => x.Code == obj.Data2.StartPosition.ToString()).Select(x => new Device<IStation520, IStation521, IStation523>(x, World)).FirstOrDefault();
  92. var destPosition = Device.All.FirstOrDefault(x => x.Code == obj.Data2.DestPosition.ToString());
  93. var isPalletizing = destPosition!.HasFlag(DeviceFlags.桁架码垛位, DeviceFlags.环形库码垛工位);
  94. short countQty = 0;
  95. short shortCode = 0;
  96. SqlSugarHelper.Do(_db =>
  97. {
  98. var db = _db.Default;
  99. if (isPalletizing)
  100. {
  101. if (destPosition.HasFlag(DeviceFlags.桁架码垛位))
  102. {
  103. var palletizingInfo = db.Queryable<WCS_Palletizing>().First(x => x.Id == obj.Data.TaskNumber);
  104. countQty = palletizingInfo.CountQty.ToShort();
  105. shortCode = palletizingInfo.ShortCode;
  106. }
  107. else if (destPosition.HasFlag(DeviceFlags.环形库码垛工位))
  108. {
  109. var deliveryTask = db.Queryable<WCS_TaskInfo>().First(x => x.ID == obj.Data.TaskNumber);
  110. countQty = deliveryTask.FullQty;
  111. shortCode = deliveryTask.PalletType;
  112. deliveryTask.Status = TaskStatus.RgvCompleted;
  113. deliveryTask.EditTime = DateTime.Now;
  114. deliveryTask.LastInteractionPoint = obj.Entity.Code;
  115. db.Updateable(deliveryTask).ExecuteCommand();
  116. deliveryTask.AddWCS_TASK_DTL(db, deliveryTask.AddrTo, $"RGV任务执行结束");
  117. }
  118. }
  119. });
  120. //初始化起始点信息
  121. startPosition.Data.TaskNumber = 0;
  122. startPosition.Data.GoodsEnd = 0;
  123. //目标地址是码垛工位
  124. if (isPalletizing)
  125. {
  126. if (destPosition.HasFlag(DeviceFlags.桁架码垛位))
  127. {
  128. var dest = new Device<ITruss530>(destPosition!, World);
  129. dest.Data.MaxQuantity = countQty;
  130. dest.Data.Type = shortCode;
  131. dest.Data.VoucherNo++;
  132. }
  133. else if (destPosition.HasFlag(DeviceFlags.环形库码垛工位))
  134. {
  135. var dest = new Device<IRobot530>(destPosition!, World);
  136. dest.Data.MaxQuantity = countQty;
  137. dest.Data.Type = shortCode;
  138. dest.Data.VoucherNo++;
  139. }
  140. }
  141. obj.Data.RES1 = 1;
  142. break;
  143. case RGVCmdType.Move:
  144. obj.Data.RES1 = 1;
  145. break;
  146. case RGVCmdType.ChangePutGoods:
  147. break;
  148. case RGVCmdType.ChangePickGoods:
  149. break;
  150. case RGVCmdType.PickPutGoods:
  151. staDev = new Device<IStation520, IStation521>(Device.All.FirstOrDefault(x => x.Code == obj.Data.DestPosition.ToString())!, World);
  152. staDev.Data.TaskNumber = 0;
  153. staDev.Data.GoodsStart = 0;
  154. staDev.Data.GoodsEnd = 0;
  155. obj.Data.RES1 = 1;
  156. break;
  157. default:
  158. throw new ArgumentOutOfRangeException();
  159. }
  160. return;
  161. }
  162. if (obj.Data2.SystemStatus != RGVSystemStatus.空闲) return;
  163. if (obj.Data2.Status.HasFlag(RGVStatus.RES1))
  164. {
  165. obj.Data.TaskNumber = obj.Data.TaskNumber;
  166. obj.Data.CmdType = RGVCmdType.Move;
  167. obj.Data.DestPosition = 1668;
  168. obj.Data.VoucherNo++;
  169. return;
  170. }
  171. if (obj.Data2.CmdType == RGVCmdType.PickGoods && !obj.Data2.Status.HasFlag(RGVStatus.Taskfinishi))
  172. {
  173. if (obj.Data2.Status.HasFlag(RGVStatus.PH_Status))
  174. {
  175. obj.Data.TaskNumber = obj.Data.TaskNumber;
  176. obj.Data.CmdType = RGVCmdType.PutGoods;
  177. obj.Data.StartPosition = obj.Data2.StartPosition;
  178. obj.Data.DestPosition = obj.Data2.DestPosition;
  179. obj.Data.VoucherNo++;
  180. return;
  181. }
  182. }
  183. var pickUpDevices = _pickUpDevices.FirstOrDefault(x => x.Key == obj.Entity.Code).Value;
  184. //有货且需要搬运货物的站台
  185. var devs = pickUpDevices.Where(v => v.Data3.Status.HasFlag(StationStatus.PH_Status) && v.Data.TaskNumber != 0)
  186. .Where(v => v.Entity.Code.ToShort() != v.Data.GoodsEnd && v.Data.GoodsEnd != 0)
  187. .ToList();
  188. //筛选出目标站台无货的站台
  189. var putDev = obj.Entity.Targets.Where(x => x.HasFlag(DeviceFlags.输送机))
  190. .Select(x => new Device<IStation520, IStation521, IStation523>(x, World))
  191. .Where(x => !x.Data3.Status.HasFlag(StationStatus.PH_Status))
  192. .Select(x => x.Entity.Code.ToShort());
  193. var dev = devs.OrderBy(x => x.Entity.Code).FirstOrDefault(x => putDev.Contains(x.Data.GoodsEnd));
  194. if (dev == null) return;
  195. //区分任务是拆盘机到码垛工位,还是码垛工位到拆盘机
  196. if (dev.Entity.HasFlag(DeviceFlags.拆盘机09, DeviceFlags.拆盘机非09))
  197. {
  198. obj.Data.TaskNumber = dev.Data.TaskNumber;
  199. obj.Data.CmdType = RGVCmdType.PickGoods;
  200. obj.Data.StartPosition = dev.Entity.Code.ToShort();
  201. obj.Data.DestPosition = dev.Data.GoodsEnd;
  202. obj.Data.VoucherNo++;
  203. }
  204. else //非拆盘机起始任务
  205. {
  206. //站台中的任务号
  207. WCS_TaskInfo task = null;
  208. SqlSugarHelper.Do(_db =>
  209. {
  210. var db = _db.Default;
  211. var taskInfo = db.Queryable<WCS_TaskInfo>().First(p => p.ID == dev.Data.TaskNumber && p.Status == TaskStatus.WaitingToExecute);
  212. if (taskInfo == null) return;
  213. taskInfo.Status = TaskStatus.RgvExecution;
  214. taskInfo.AddrNext = obj.Entity.Code;
  215. taskInfo.EditWho = "WCS";
  216. taskInfo.EditTime = DateTime.Now; ;
  217. db.Updateable(taskInfo).ExecuteCommand();
  218. taskInfo.AddWCS_TASK_DTL(db, dev.Entity.Code, obj.Entity.Code, $"任务分配至{obj.Entity.Code}");
  219. task = taskInfo;
  220. });
  221. if (task == null) return;
  222. obj.Data.TaskNumber = task.ID;
  223. obj.Data.CmdType = RGVCmdType.PickPutGoods;
  224. obj.Data.StartPosition = dev.Entity.Code.ToShort();
  225. obj.Data.DestPosition = dev.Data.GoodsEnd;
  226. obj.Data.VoucherNo++;
  227. }
  228. }
  229. public override bool Select(Device dev)
  230. {
  231. return dev.Code == "RGV1";
  232. //return dev.HasFlag(Extensions.DeviceFlags.RGV);
  233. }
  234. }
  235. }