RGVWorks.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403
  1. using DBHelper;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using WCS.Core;
  6. using WCS.Entity;
  7. using WCS.Entity.Protocol.RGV;
  8. using WCS.Service.Extensions;
  9. using WCS.Service.Handlers;
  10. namespace WCS.Service.Works.RGV
  11. {
  12. [WorkTitle(typeof(RGVHandler), "直穿RGV")]
  13. public class 直穿RGV : DeviceWork<RGVDevice>
  14. {
  15. private readonly string ConvGroup_1030 = "G1030";
  16. private readonly string RGV8 = "RGV8";
  17. protected override void Do(RGVDevice rgv)
  18. {
  19. rgv.EX(rgv =>
  20. {
  21. //RGV是自动且空闲的
  22. if (rgv.Data2.WorkMode != RGVMode.自动) throw new DoException($"{rgv.Data2.WorkMode}");
  23. if (rgv.Data2.SystemStatus != RGVRunStatus.空闲) throw new DoException($"{rgv.Data2.SystemStatus}");
  24. //RGV状态为 RGV到站&&任务完成 时程序认为RGV当前不在取货点
  25. if (rgv.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.RGV到站) && rgv.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.任务完成))
  26. {
  27. //下达一个目标为1030的移动任务
  28. rgv.Data.TaskID_1 = 1030;
  29. rgv.Data.DestPosition_1 = 1030;
  30. rgv.Data.TaskType_1 = RGVTaskType.移动;
  31. rgv.Data.Trigger_1++;
  32. }
  33. var obj = Device.Find(ConvGroup_1030).Create<StationDeviceGroup>();
  34. obj.WhetherToExecute();
  35. //筛选出有任务号和起始及目标地址的设备
  36. var dev = obj.RGVGetTaskedDevice() ?? throw new WarnException("无可用任务");
  37. DB.Do(db =>
  38. {
  39. var taskids = dev.Select(v => v.Data2.Tasknum);
  40. var tasks = db.Default.Set<WCS_TASK>().Where(p => taskids.Contains(p.ID));
  41. if (tasks.GroupBy(p => p.TaskGroupKey).Count() > 1)
  42. throw new WarnException("任务组ID不一致");
  43. if (tasks.GroupBy(p => p.TaskGroupKey).Count() == 0)
  44. throw new WarnException("无任务组ID");
  45. var gw1 = obj.Items.ToArray()[0];
  46. var gw2 = obj.Items.ToArray()[1];
  47. if (gw1.Data2.Tasknum != 0 && gw2.Data2.Tasknum != 0)
  48. {
  49. if (gw1.Data2.Goodsend != gw2.Data2.Goodsend) throw new WarnException($"{obj.Entity.CODE}目标地址不一致");
  50. }
  51. rgv.Data.TaskID_1 = gw1.Data2.Tasknum;
  52. rgv.Data.TaskID_2 = gw2.Data2.Tasknum;
  53. if (gw1.Data2.Tasknum != 0)
  54. {
  55. rgv.Data.StartPosition_1 = gw1.Entity.CODE.ToShort();
  56. rgv.Data.DestPosition_1 = gw1.Data2.Goodsend;
  57. }
  58. else
  59. {
  60. rgv.Data.StartPosition_1 = gw1.Entity.CODE.ToShort();
  61. rgv.Data.DestPosition_1 = gw2.Data2.Goodsend;
  62. }
  63. rgv.Data.Trigger_1++;
  64. foreach (var task in tasks)
  65. {
  66. var msg = $"下达从{rgv.Data.StartPosition_1}移动至{rgv.Data.DestPosition_1}的RGV PLC指令。";
  67. msg += $"[{task.ID}][{rgv.Data.StartPosition_1}][{rgv.Data.DestPosition_1}[{rgv.Data.Trigger_1}]";
  68. task.CreateStatusLog(db, msg, this.GetType());
  69. }
  70. });
  71. });
  72. }
  73. protected override bool SelectDevice(WCS_DEVICE dev)
  74. {
  75. return dev.CODE == RGV8;
  76. }
  77. }
  78. [WorkTitle(typeof(RGVHandler), "涂布环穿")]
  79. public class 涂布环穿 : DeviceWork<RGVDevice>
  80. {
  81. protected override void Do(RGVDevice obj)
  82. {
  83. //11号站台是取货点,但不是待命点,如需取货,从九号站台调车
  84. //2号站台是一个待命点,但是为了保证入库取货效率,所有后面会跟一个车
  85. //如果3号站台的小车拦住了后一个小车的放货任务,检测一次最近的空车距离值,如果大于或等于618500并且小于被拦住小车的位置值再进行调车,用于避免无效空跑
  86. obj.EX(obj =>
  87. {
  88. if (obj.Data2.Trigger_1 != obj.Data.Trigger_1) throw new WarnException($"等待执行任务{obj.Data2.TaskID_1}--{obj.Data2.TaskID_2}");
  89. if (obj.Data2.WorkMode != RGVMode.自动) throw new DoException(obj.Data2.WorkMode.ToString());
  90. if (obj.Data2.SystemStatus != RGVRunStatus.空闲) throw new DoException(obj.Data2.SystemStatus.ToString());
  91. //RGV当前是否刚刚完成取货任务,等待放货
  92. if (obj.IsPut())
  93. {
  94. if (!obj.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) throw new WarnException("RGV无光电,无法放货,请检查实际情况");
  95. //取小车上的任务
  96. DB.Do(db =>
  97. {
  98. List<int> taskids = new List<int>() { obj.Data2.TaskID_1, obj.Data2.TaskID_2 };
  99. var tasks = db.Default.Set<WCS_TASK>().Where(p => taskids.Contains(p.ID));
  100. if (tasks.GroupBy(p => p.TaskGroupKey).Count() > 1)
  101. throw new WarnException("任务组ID不一致");
  102. if (!tasks.GroupBy(p => p.TaskGroupKey).Any())
  103. throw new WarnException("无任务组ID");
  104. var destStation = Device.Find(tasks.FirstOrDefault().ADDRNEXT).Create<StationDeviceGroup>();
  105. obj.Put(destStation, obj.Data2.TaskID_1, obj.Data2.TaskID_2);
  106. });
  107. return;
  108. }
  109. var pickStation = obj.CurrentStation();
  110. //RGV是否在任何一个站台
  111. if (pickStation != null)
  112. {
  113. //是否是取货站台
  114. if (pickStation.Entity.Is(DF.涂布RGV取货设备组))
  115. {
  116. // 筛选出有任务号和起始及目标地址的设备
  117. var devs = pickStation.RGVGetTaskedDevice();
  118. //是否需要取货
  119. if (devs != null && devs.Count > 0)
  120. {
  121. //开始下达取货任务
  122. DB.Do(db =>
  123. {
  124. if (obj.Data2.TaskType_1 != RGVTaskType.移动) throw new WarnException($"RGV正在执行{obj.Data2.TaskType_1}");
  125. if (obj.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) throw new WarnException("RGV有光电,无法取货,请检查实际情况");
  126. if (!obj.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.RGV到站)) throw new WarnException("RGV无到站状态,请检查RGV实际状态");
  127. var taskids = devs.Select(v => v.Data2.Tasknum);
  128. var tasks = db.Default.Set<WCS_TASK>().Where(p => taskids.Contains(p.ID));
  129. if (tasks.GroupBy(p => p.TaskGroupKey).Count() > 1)
  130. throw new WarnException("任务组ID不一致");
  131. if (!tasks.GroupBy(p => p.TaskGroupKey).Any())
  132. throw new WarnException("无任务组ID");
  133. var gw1 = pickStation.Items.ToArray()[0];
  134. var gw2 = pickStation.Items.ToArray()[1];
  135. if (gw1.Data2.Tasknum != 0 && gw2.Data2.Tasknum != 0)
  136. {
  137. if (gw1.Data2.Goodsend != gw2.Data2.Goodsend) throw new WarnException($"{obj.Entity.CODE}目标地址不一致");
  138. }
  139. obj.Data.TaskID_1 = gw1.Data2.Tasknum;
  140. obj.Data.TaskID_2 = gw2.Data2.Tasknum;
  141. obj.Data.TaskType_1 = RGVTaskType.取货;
  142. obj.Data.DestPosition_1 = pickStation.Entity.CODE.Replace("G", "").ToShort();
  143. obj.Data.Trigger_1++;
  144. foreach (var task in tasks)
  145. {
  146. var msg = $"下达从{obj.Data.StartPosition_1}移动至{obj.Data.DestPosition_1}的RGV PLC指令。";
  147. msg += $"[{task.ID}][{obj.Data.StartPosition_1}][{obj.Data.DestPosition_1}[{obj.Data.Trigger_1}]";
  148. task.CreateStatusLog(db, msg, this.GetType());
  149. }
  150. });
  151. return;
  152. }
  153. else if (pickStation.Entity.CODE == "G9")
  154. {
  155. //九站台的小车需要额外检测一下11站台是否需要取货
  156. var station11 = Device.Find("G11").Create<StationDeviceGroup>();
  157. var station11Dev = station11.RGVGetTaskedDevice();
  158. if (station11Dev != null && station11Dev.Count > 0)
  159. {
  160. obj.Move(station11);
  161. return;
  162. }
  163. }
  164. //找到自己的后一个小车
  165. var afterRgv = obj.After();
  166. //自己是否阻挡了该小车
  167. if (afterRgv.Data2.WorkMode == RGVMode.自动 && afterRgv.Data2.SystemStatus != RGVRunStatus.空闲 && obj.StopedByMe(afterRgv))
  168. {
  169. if (pickStation.Entity.CODE == "G2")
  170. {
  171. //此站台因需要进行排队,因此是否需要赶车取决于是否拦住了两个小车的任务
  172. var afterRgv1 = afterRgv.After();
  173. //如果是空闲或者没有拦住他,就不执行赶车
  174. if (afterRgv1.Data2.WorkMode == RGVMode.自动 && (afterRgv1.Data2.SystemStatus == RGVRunStatus.空闲 || !afterRgv.StopedByMe(afterRgv1))) return;
  175. }
  176. else if (pickStation.Entity.CODE == "G3")
  177. {
  178. //为了减少空跑,需要被阻拦的小车后面的小车任务类型为移动并且位置值要大于618500小于被阻拦小车的位置值
  179. var afterRgv1 = afterRgv.After();
  180. if (afterRgv1.Data2.WorkMode == RGVMode.自动 && afterRgv.Data2.TaskType_1 == RGVTaskType.放货 && !(afterRgv1.Position < afterRgv.Position && afterRgv1.Position > 618500)) return;
  181. }
  182. //找到当前所在站台的下一个取货点
  183. var beforeStation = Device.Where(v => v.Is(DF.涂布RGV取货设备组) && v.CODE != pickStation.Entity.CODE)
  184. .Select(v => v.Create<StationDeviceGroup>())
  185. .OrderBy(v => pickStation.Distance(v))
  186. .FirstOrDefault();
  187. //写入移动任务
  188. obj.Move(beforeStation);
  189. return;
  190. }
  191. else if (pickStation.Entity.CODE == "G11")
  192. {
  193. obj.Move(Device.Find("G2").Create<StationDeviceGroup>());
  194. return;
  195. }
  196. else
  197. {
  198. //找到自己前面的 有货 没有小车 没有正在过来的小车 距离自己最近的 的站台
  199. var station = Device.Where(v => v.Is(DF.涂布RGV取货设备组))
  200. .Select(v => v.Create<StationDeviceGroup>())
  201. .Where(v => v.Position > obj.Position) //前面的
  202. .Where(v => v.CurrentRGV() == null)
  203. .Where(v => !v.RgvList.Any(c => c.Data2.SystemStatus != RGVRunStatus.空闲 && c.Data2.DestPosition_1 == v.Entity.CODE.GetShortCode()))
  204. .Where(v =>
  205. {
  206. // 筛选出有任务号和起始及目标地址的设备
  207. var b = pickStation.RGVGetTaskedDevice();
  208. //是否需要取货
  209. if (b != null && b.Count > 0) return true;
  210. else return false;
  211. })
  212. .OrderBy(v => obj.Distance(v)).FirstOrDefault();//取当前小车最近
  213. if (obj.Data2.TaskType_1 == RGVTaskType.移动 && obj.Distance(station) > 100000)
  214. {
  215. obj.Move(station);
  216. return;
  217. }
  218. }
  219. }
  220. //是否在一个放货站台
  221. if (pickStation.Entity.Is(DF.涂布RGV放货设备组))
  222. {
  223. pickStation = obj.BeforeStation();
  224. obj.Move(pickStation);
  225. }
  226. return;
  227. }
  228. //此时RGV即没有等待执行的放货任务,也不在任何一个取货点,因此需要调往最近的一个取货点
  229. //找到距离这个RGV最近的一个取货点
  230. //必须所有RGV都是空闲状态时才可以进行初始化
  231. if (obj.RGVList.Any(v => v.Data2.SystemStatus != RGVRunStatus.空闲)) return;
  232. pickStation = obj.BeforeStation();
  233. obj.Move(pickStation);
  234. });
  235. }
  236. protected override bool SelectDevice(WCS_DEVICE dev)
  237. {
  238. return dev.Is(DF.涂布RGV);
  239. }
  240. }
  241. [WorkTitle(typeof(RGVHandler), "BOPP环穿")]
  242. public class BOPP环穿 : DeviceWork<RGVDevice>
  243. {
  244. protected override void Do(RGVDevice obj)
  245. {
  246. obj.EX(obj =>
  247. {
  248. if (obj.Data2.Trigger_1 != obj.Data.Trigger_1) throw new WarnException($"等待执行任务{obj.Data2.TaskID_1}--{obj.Data2.TaskID_2}");
  249. if (obj.Data2.WorkMode != RGVMode.自动) throw new DoException(obj.Data2.WorkMode.ToString());
  250. if (obj.Data2.SystemStatus != RGVRunStatus.空闲) throw new DoException(obj.Data2.SystemStatus.ToString());
  251. //RGV当前是否刚刚完成取货任务,等待放货
  252. if (obj.IsPut())
  253. {
  254. if (!obj.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) throw new WarnException("RGV无光电,无法放货,请检查实际情况");
  255. //取小车上的任务
  256. DB.Do(db =>
  257. {
  258. List<int> taskids = new List<int>() { obj.Data2.TaskID_1, obj.Data2.TaskID_2 };
  259. var tasks = db.Default.Set<WCS_TASK>().Where(p => taskids.Contains(p.ID));
  260. if (tasks.GroupBy(p => p.TaskGroupKey).Count() > 1)
  261. throw new WarnException("任务组ID不一致");
  262. if (!tasks.GroupBy(p => p.TaskGroupKey).Any())
  263. throw new WarnException("无任务组ID");
  264. var destStation = Device.Find(tasks.FirstOrDefault().ADDRNEXT).Create<StationDeviceGroup>();
  265. obj.Put(destStation, obj.Data2.TaskID_1, obj.Data2.TaskID_2);
  266. });
  267. return;
  268. }
  269. var pickStation = obj.CurrentStation();
  270. //RGV是否在任何一个站台
  271. if (pickStation != null)
  272. {
  273. //是否是取货站台
  274. if (pickStation.Entity.Is(DF.BOPPRGV取货设备组))
  275. {
  276. // 筛选出有任务号和起始及目标地址的设备
  277. var devs = pickStation.RGVGetTaskedDevice();
  278. //是否需要取货
  279. if (devs != null && devs.Count() > 0)
  280. {
  281. //开始下达取货任务
  282. DB.Do(db =>
  283. {
  284. if (obj.Data2.TaskType_1 != RGVTaskType.移动) throw new WarnException($"RGV正在执行{obj.Data2.TaskType_1}");
  285. if (obj.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) throw new WarnException("RGV有光电,无法取货,请检查实际情况");
  286. if (!obj.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.RGV到站)) throw new WarnException("RGV无到站状态,请检查RGV实际状态");
  287. var taskids = devs.Select(v => v.Data2.Tasknum);
  288. var tasks = db.Default.Set<WCS_TASK>().Where(p => taskids.Contains(p.ID));
  289. if (tasks.GroupBy(p => p.TaskGroupKey).Count() > 1)
  290. throw new WarnException("任务组ID不一致");
  291. if (!tasks.GroupBy(p => p.TaskGroupKey).Any())
  292. throw new WarnException("无任务组ID");
  293. var gw1 = pickStation.Items.ToArray()[0];
  294. var gw2 = pickStation.Items.ToArray()[1];
  295. if (gw1.Data2.Tasknum != 0 && gw2.Data2.Tasknum != 0)
  296. {
  297. if (gw1.Data2.Goodsend != gw2.Data2.Goodsend) throw new WarnException($"{obj.Entity.CODE}目标地址不一致");
  298. }
  299. obj.Pick(pickStation, gw1.Data2.Tasknum, gw2.Data2.Tasknum);
  300. foreach (var task in tasks)
  301. {
  302. var msg = $"下达从{obj.Data.StartPosition_1}移动至{obj.Data.DestPosition_1}的RGV PLC指令。";
  303. msg += $"[{task.ID}][{obj.Data.StartPosition_1}][{obj.Data.DestPosition_1}[{obj.Data.Trigger_1}]";
  304. task.CreateStatusLog(db, msg, this.GetType());
  305. }
  306. });
  307. return;
  308. }
  309. ////找到自己的后一个小车
  310. //var afterRgv = obj.After();
  311. ////自己是否阻挡了该小车
  312. //if (afterRgv.Data2.WorkMode == RGVMode.自动 && afterRgv.Data2.SystemStatus != RGVRunStatus.空闲 && obj.StopedByMe(afterRgv))
  313. //{
  314. // //找到当前所在站台的下一个取货点
  315. // var beforeStation = Device.Where(v => v.Is(DF.BOPPRGV取货设备组) && v.CODE != pickStation.Entity.CODE)
  316. // .Select(v => v.Create<StationDeviceGroup>())
  317. // .OrderBy(v => pickStation.Distance(v))
  318. // .FirstOrDefault();
  319. // //写入移动任务
  320. // obj.Move(beforeStation);
  321. // return;
  322. //}
  323. //else
  324. //{
  325. // //取前一个取货点
  326. // pickStation = obj.BeforeStation();
  327. // //前一个取货点的小车
  328. // var rgv = pickStation.CurrentRGV();
  329. // //前一个取货点没有车 且没有非空闲目的地为前一个取货点的小车
  330. // if (rgv == null && !obj.RGVList.Any(v => v.Data2.SystemStatus != RGVRunStatus.空闲 && v.Data2.DestPosition_1 == pickStation.Entity.CODE.Replace("G", "").ToShort()))
  331. // {
  332. // obj.Move(pickStation);
  333. // return;
  334. // }
  335. //}
  336. //计算当前RGV拦住小车的数量
  337. var max = obj.RGVList.Count(v => v.Data2.WorkMode == RGVMode.自动 && v.Data2.SystemStatus != RGVRunStatus.空闲 && obj.StopedByMe(v));
  338. if (pickStation.Entity.CODE == "G19" && max > 2)
  339. {
  340. obj.Move(Device.Find("G23").Create<StationDeviceGroup>());
  341. }
  342. else if (pickStation.Entity.CODE == "G23" && max > 3)
  343. {
  344. obj.Move(Device.Find("G19").Create<StationDeviceGroup>());
  345. }
  346. }
  347. //是否在一个放货站台
  348. if (pickStation.Entity.Is(DF.BOPPRGV放货设备组))
  349. {
  350. pickStation = obj.BeforeStation();
  351. obj.Move(pickStation);
  352. }
  353. return;
  354. }
  355. //此时RGV即没有等待执行的放货任务,也不在任何一个取货点,因此需要调往最近的一个取货点
  356. //找到距离这个RGV最近的一个取货点
  357. //必须所有RGV都是空闲状态时才可以进行初始化
  358. if (obj.RGVList.Any(v => v.Data2.SystemStatus != RGVRunStatus.空闲 || v.Data2.WorkMode != RGVMode.自动)) return;
  359. pickStation = obj.BeforeStation();
  360. obj.Move(pickStation);
  361. });
  362. }
  363. protected override bool SelectDevice(WCS_DEVICE dev)
  364. {
  365. return dev.Is(DF.BOPPRGV);
  366. }
  367. }
  368. }