终端一体化运控平台
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

865 lines
37 KiB

  1. using System;
  2. using System.Collections.Generic;
  3. using BPA.Message.Enum;
  4. using BPASmartClient.Device;
  5. using BPASmartClient.EventBus;
  6. using BPASmartClient.Model;
  7. using BPASmartClient.Peripheral;
  8. using static BPASmartClient.EventBus.EventBus;
  9. using BPASmartClient.Helper;
  10. using System.Threading;
  11. using BPASmartClient.Message;
  12. using BPA.Message;
  13. using System.Linq;
  14. using BPASmartClient.Model.PLC;
  15. using System.Threading.Tasks;
  16. using System.Reflection;
  17. using BPASmartClient.MorkSUpgradedVer.Model;
  18. using System.Collections.ObjectModel;
  19. using BPASmartClient.MorkSUpgradedVer.ViewModel;
  20. using BPASmartClient.Business;
  21. using BPASmartClient.Model.小炒机;
  22. using BPA.Models;
  23. using System.Windows.Forms;
  24. using System.Media;
  25. using BPASmartClient.CustomResource;
  26. //using BPA.Helper;
  27. namespace BPASmartClient.MorkSUpgradedVer
  28. {
  29. public class Control_MorkSUpgradedVer : BaseDevice
  30. {
  31. public override DeviceClientType DeviceType => DeviceClientType.MORKS;
  32. GVL_MorkSUpgradedVer mORKS = new GVL_MorkSUpgradedVer();
  33. Alarm alarm = new Alarm();
  34. public override void DoMain()
  35. {
  36. MonitorViewModel.DeviceId = DeviceId;
  37. ServerInit();
  38. DataParse();
  39. Json<MorksPar>.Read();
  40. Json<OrderStatistics>.Read();
  41. if (Json<MorksPar>.Data.parSets == null) Json<MorksPar>.Data.parSets = new ObservableCollection<ParSet>();
  42. if (Json<MorksPar>.Data.parSets.Count < 6)
  43. {
  44. Json<MorksPar>.Data.parSets.Clear();
  45. for (int i = 0; i < 6; i++)
  46. {
  47. Json<MorksPar>.Data.parSets.Add(new ParSet()
  48. {
  49. CheckBoxContext = $"煮面口{i + 1}屏蔽",
  50. Minute = 1,
  51. Second = 0,
  52. IsShield = false,
  53. TextBlockContext = $"煮面口{i + 1}时间设定"
  54. });
  55. }
  56. }
  57. ActionManage.GetInstance.Register(new Action<object[]>((o) =>
  58. {
  59. if (o.Length > 0)
  60. {
  61. Random rd = new Random();
  62. ThreadManage.GetInstance().StartLong(new Action(() =>
  63. {
  64. int NoodleLoc = (int)o[0] == 0 ? rd.Next(1, 6) : (int)o[0];
  65. int BowlLoc = (int)o[1] == 0 ? rd.Next(10, 12) : (int)o[1];
  66. string guid = new Guid().ToString();
  67. mORKS.RBTakeNoodleTask.Enqueue(new OrderLocInfo() { Loc = (ushort)NoodleLoc, SuborderId = guid });
  68. MessageLog.GetInstance.Show($"添加订单:面条位置【{NoodleLoc}】");
  69. mORKS.TakeBowlTask.Enqueue(new OrderLocInfo() { Loc = (ushort)BowlLoc, SuborderId = guid });
  70. MessageLog.GetInstance.Show($"添加订单:碗位置【{BowlLoc}】");
  71. Thread.Sleep(60000);
  72. }), "ForOrder");
  73. }
  74. }), "EnableForOrder");
  75. ActionManage.GetInstance.Register(new Action<object>((o) =>
  76. {
  77. if (o != null && o is WritePar writePar) WriteData(writePar.Address, writePar.Value);
  78. }), "WriteVW");
  79. ActionManage.GetInstance.Register(new Action<object>((o) =>
  80. {
  81. if (o != null && o is WritePar writePar) WriteData(writePar.Address, writePar.Value);
  82. }), "WriteBools");
  83. ActionManage.GetInstance.Register(new Action(() => { DeviceInit(); }), "InitDevice");
  84. }
  85. public override void ResetProgram()
  86. {
  87. mORKS = null;
  88. mORKS = new GVL_MorkSUpgradedVer();
  89. }
  90. public override void Stop()
  91. {
  92. }
  93. private void ServerInit()
  94. {
  95. //物料信息
  96. EventBus.EventBus.GetInstance().Subscribe<MaterialDeliveryEvent>(DeviceId, delegate (IEvent @event, EventCallBackHandle callBack)
  97. {
  98. if (@event == null) return;
  99. if (@event is MaterialDeliveryEvent material)
  100. {
  101. orderMaterialDelivery = material.orderMaterialDelivery;
  102. }
  103. });
  104. //配方数据信息
  105. EventBus.EventBus.GetInstance().Subscribe<RecipeBomEvent>(DeviceId, delegate (IEvent @event, EventCallBackHandle callBack)
  106. {
  107. if (@event == null) return;
  108. if (@event is RecipeBomEvent recipe)
  109. {
  110. recipeBoms = recipe.recipeBoms;
  111. }
  112. });
  113. }
  114. private void OrderChange(string subid, ORDER_STATUS oRDER_STATUS)
  115. {
  116. var res = mORKS.doOrderEvents.FirstOrDefault(p => p.MorkOrder.SuborderId == subid);
  117. string goodName = string.Empty;
  118. string SortNum = string.Empty;
  119. if (res != null)
  120. {
  121. goodName = res.MorkOrder.GoodsName;
  122. SortNum = res.MorkOrder.SortNum.ToString();
  123. }
  124. //if (mORKS.doe.ContainsKey(subid))
  125. //{
  126. // goodName = mORKS.doe[subid].MorkOrder.GoodsName;
  127. // SortNum = mORKS.doe[subid].MorkOrder.SortNum.ToString();
  128. if (!string.IsNullOrEmpty(goodName) && !string.IsNullOrEmpty(SortNum))
  129. {
  130. EventBus.EventBus.GetInstance().Publish(new OrderStatusChangedEvent() { SortNum = SortNum, GoodName = goodName, Status = oRDER_STATUS, SubOrderId = subid, deviceClientType = DeviceType });
  131. var index = DataServer.GetInstance.morkS.MakeOrder.FindIndex(p => p.SortNum == SortNum);
  132. if (index >= 0 && index < DataServer.GetInstance.morkS.MakeOrder.Count)
  133. {
  134. if (oRDER_STATUS == ORDER_STATUS.COMPLETED_COOK)
  135. {
  136. DataServer.GetInstance.morkS.MakeOrder.RemoveAt(index);
  137. DataServer.GetInstance.morkS.MakeOrderOver.Add(new OrderMakeModel()
  138. {
  139. Status = oRDER_STATUS,
  140. GoodName = goodName,
  141. SortNum = SortNum,
  142. StopTime = DateTime.Now.ToString("HH:mm:ss")
  143. });
  144. }
  145. else if (oRDER_STATUS == ORDER_STATUS.COMPLETED_TAKE)
  146. {
  147. var temp = DataServer.GetInstance.morkS.MakeOrderOver.FirstOrDefault(p => p.SortNum == SortNum);
  148. if (temp != null) DataServer.GetInstance.morkS.MakeOrderOver.Remove(temp);
  149. }
  150. else
  151. {
  152. DataServer.GetInstance.morkS.MakeOrder.ElementAt(index).Status = oRDER_STATUS;
  153. }
  154. }
  155. else
  156. {
  157. DataServer.GetInstance.morkS.MakeOrder.Add(new OrderMakeModel()
  158. {
  159. Status = oRDER_STATUS,
  160. GoodName = goodName,
  161. SortNum = SortNum,
  162. StartTime = DateTime.Now.ToString("HH:mm:ss")
  163. });
  164. }
  165. //mORKS.doe.Remove(subid, out _);
  166. }
  167. //}
  168. //var res = mORKS.doOrderEvents.FirstOrDefault(p => p.MorkOrder.SuborderId == subid);
  169. //string goodName = string.Empty;
  170. //string SortNum = string.Empty;
  171. //if (res != null)
  172. //{
  173. // goodName = res.MorkOrder.GoodsName;
  174. // SortNum = res.MorkOrder.SortNum.ToString();
  175. //}
  176. //EventBus.EventBus.GetInstance().Publish(new OrderStatusChangedEvent() { SortNum = SortNum, GoodName = goodName, Status = oRDER_STATUS, SubOrderId = subid, deviceClientType = DeviceType });
  177. //var index = DataServer.GetInstance.morkS.MakeOrder.FindIndex(p => p.SortNum == SortNum);
  178. //if (index >= 0 && index < DataServer.GetInstance.morkS.MakeOrder.Count)
  179. //{
  180. // if (oRDER_STATUS == ORDER_STATUS.COMPLETED_COOK)
  181. // {
  182. // DataServer.GetInstance.morkS.MakeOrder.RemoveAt(index);
  183. // DataServer.GetInstance.morkS.MakeOrderOver.Add(new OrderMakeModel()
  184. // {
  185. // Status = oRDER_STATUS,
  186. // GoodName = goodName,
  187. // SortNum = SortNum,
  188. // StopTime = DateTime.Now.ToString("HH:mm:ss")
  189. // });
  190. // }
  191. // else if (oRDER_STATUS == ORDER_STATUS.COMPLETED_TAKE)
  192. // {
  193. // var temp = DataServer.GetInstance.morkS.MakeOrderOver.FirstOrDefault(p => p.SortNum == SortNum);
  194. // if (temp != null) DataServer.GetInstance.morkS.MakeOrderOver.Remove(temp);
  195. // }
  196. // else
  197. // {
  198. // DataServer.GetInstance.morkS.MakeOrder.ElementAt(index).Status = oRDER_STATUS;
  199. // }
  200. //}
  201. //else
  202. //{
  203. // DataServer.GetInstance.morkS.MakeOrder.Add(new OrderMakeModel()
  204. // {
  205. // Status = oRDER_STATUS,
  206. // GoodName = goodName,
  207. // SortNum = SortNum,
  208. // StartTime = DateTime.Now.ToString("HH:mm:ss")
  209. // });
  210. //}
  211. }
  212. private void GetStatus(string key, Action<object> action)
  213. {
  214. if (peripheralStatus.ContainsKey(key))
  215. {
  216. if (peripheralStatus[key] != null)
  217. {
  218. action?.Invoke(peripheralStatus[key]);
  219. }
  220. }
  221. }
  222. public override void ReadData()
  223. {
  224. DataServer.GetInstance.morkS.Alarm.Clear();
  225. alarms.ForEach(item =>
  226. {
  227. DataServer.GetInstance.morkS.Alarm.Add(new AlarmModel()
  228. {
  229. AlarmTime = $"{item.Date} {item.Time}",
  230. AlarmMs = item.Info
  231. });
  232. });
  233. GetStatus("M0.1", new Action<object>((obj) =>
  234. {
  235. if (obj is bool[] bools && bools.Length > 0 && bools.Length <= 7)
  236. {
  237. Initing = !bools[0];
  238. mORKS.InitComplete = bools[0];
  239. mORKS.MoveScrewRodInitCom = bools[1];
  240. mORKS.SacrificialVesselInitCom = bools[2];
  241. mORKS.CylinderInitCom = bools[3];
  242. mORKS.NoodleCookerInitCom = bools[4];
  243. mORKS.RobotInitCom = bools[5];
  244. mORKS.SiloInitCom = bools[6];
  245. alarm.DeviceNoInit = !mORKS.InitComplete;
  246. alarm.MoveScrewRodNoInit = !mORKS.MoveScrewRodInitCom;
  247. alarm.SacrificialVesselNoInit = !mORKS.SacrificialVesselInitCom;
  248. alarm.CylinderNoInit = !mORKS.CylinderInitCom;
  249. alarm.NoodleCookerNoInit = !mORKS.NoodleCookerInitCom;
  250. alarm.RobotNoInit = !mORKS.RobotInitCom;
  251. alarm.SiloNoInit = !mORKS.SiloInitCom;
  252. }
  253. }));
  254. GetStatus("M10.0", new Action<object>((obj) =>
  255. {
  256. if (obj is bool[] bools && bools.Length > 0 && bools.Length <= 2)
  257. {
  258. mORKS.AllowInvertedFace = bools[0];
  259. mORKS.DiningComplete = bools[1];
  260. }
  261. }));
  262. GetStatus("M10.4", new Action<object>((obj) =>
  263. {
  264. if (obj is bool[] bools && bools.Length > 0 && bools.Length <= 1)
  265. {
  266. mORKS.DropBowlMechanismStatus = bools[0];
  267. }
  268. }));
  269. GetStatus("M12.2", new Action<object>((obj) =>
  270. {
  271. if (obj is bool[] bools && bools.Length > 0 && bools.Length <= 1)
  272. {
  273. mORKS.FixedFlag = bools[0];
  274. }
  275. }));
  276. GetStatus("M13.5", new Action<object>((obj) =>
  277. {
  278. if (obj is bool[] bools && bools.Length > 0 && bools.Length <= 1)
  279. {
  280. mORKS.SiloInPlace = bools[0];
  281. }
  282. }));
  283. GetStatus("M16.7", new Action<object>((obj) =>
  284. {
  285. if (obj is bool[] bools && bools.Length > 0 && bools.Length <= 1)
  286. {
  287. mORKS.RobotTakeNoodleCom = bools[0];
  288. }
  289. }));
  290. GetStatus("M17.4", new Action<object>((obj) =>
  291. {
  292. if (obj is bool[] bools && bools.Length > 0 && bools.Length <= 1)
  293. {
  294. mORKS.RobotStatus = bools[0];
  295. }
  296. }));
  297. GetStatus("M18.0", new Action<object>((obj) =>
  298. {
  299. if (obj is bool[] bools && bools.Length > 0 && bools.Length <= 5)
  300. {
  301. mORKS.SmallBowlYesOrNoCheck = bools[0];
  302. mORKS.LargeBowYesOrNoCheck = bools[1];
  303. mORKS.TurntableLowPosition = bools[2];
  304. mORKS.TurntableHighPosition = bools[3];
  305. alarm.Supply2_LossBowl = !mORKS.SmallBowlYesOrNoCheck;
  306. alarm.Supply1_LossBowl = !mORKS.LargeBowYesOrNoCheck;
  307. }
  308. }));
  309. GetStatus("VW17", new Action<object>((obj) =>
  310. {
  311. if (obj is ushort[] ushorts && ushorts.Length > 0 && ushorts.Length <= 1)
  312. {
  313. var tt = ushorts.UshortsToBytes(true).BytesToUshorts();
  314. for (byte i = 0; i < 6; i++)
  315. {
  316. if (RTrig.GetInstance($"CookNoodleCom{i + 1}").Start(tt[0].GetBitValue((byte)(i + 1))))
  317. {
  318. if (!string.IsNullOrEmpty(mORKS.CookNodelId[i]))
  319. mORKS.CookNoodleCom[i] = true;
  320. }
  321. }
  322. //mORKS.CookNoodleCom[0] = tt[0].GetBitValue(1);
  323. //mORKS.CookNoodleCom[1] = tt[0].GetBitValue(2);
  324. //mORKS.CookNoodleCom[2] = tt[0].GetBitValue(3);
  325. //mORKS.CookNoodleCom[3] = tt[0].GetBitValue(4);
  326. //mORKS.CookNoodleCom[4] = tt[0].GetBitValue(5);
  327. //mORKS.CookNoodleCom[5] = tt[0].GetBitValue(6);
  328. mORKS.Heating = ushorts[0].GetBitValue(15);
  329. mORKS.TemperatureReaches = ushorts[0].GetBitValue(16);
  330. alarm.MachineLowTemperature = !mORKS.TemperatureReaches;
  331. }
  332. }));
  333. GetStatus("VW770", new Action<object>((obj) =>
  334. {
  335. if (obj is ushort[] ushorts && ushorts.Length > 0 && ushorts.Length <= 1)
  336. {
  337. mORKS.CurrentFeedbackLoc = ushorts[0];
  338. }
  339. }));
  340. mORKS.TakeBowlTaskCount = mORKS.TakeBowlTask.Count;
  341. mORKS.RBTakeNoodleTaskCount = mORKS.RBTakeNoodleTask.Count;
  342. for (int i = 0; i < Json<MorksPar>.Data.parSets.Count; i++)
  343. {
  344. mORKS.nsm.ElementAt(i).IsShield = Json<MorksPar>.Data.parSets.ElementAt(i).IsShield;
  345. mORKS.nsm.ElementAt(i).NoodleCookerStatus = mORKS.NoodleCookerStatus[i];
  346. }
  347. }
  348. /// <summary>
  349. /// 数据解析
  350. /// </summary>
  351. private void DataParse()
  352. {
  353. EventBus.EventBus.GetInstance().Subscribe<DoOrderEvent>(DeviceId, delegate (IEvent @event, EventCallBackHandle callBackHandle)
  354. {
  355. if (@event == null) return;
  356. if (@event is DoOrderEvent order)
  357. {
  358. mORKS.doOrderEvents.Add(order);
  359. //mORKS.doe.TryAdd(order.MorkOrder.SuborderId, order);
  360. if (order.MorkOrder.GoodBatchings == null) return;
  361. if (mORKS.HistorySuborderId.Contains(order.MorkOrder.SuborderId)) return;
  362. OrderCount++;
  363. if (DateTime.Now.Subtract(Json<OrderStatistics>.Data.StatisticsTime).Days != 0)
  364. Json<OrderStatistics>.Data.Count = 0;
  365. Json<OrderStatistics>.Data.StatisticsTime = DateTime.Now;
  366. Json<OrderStatistics>.Data.Count++;
  367. Json<OrderStatistics>.Save();
  368. OrderChange(order.MorkOrder.SuborderId, ORDER_STATUS.WAIT);
  369. DeviceProcessLogShow($"接收到{OrderCount}次订单,订单ID:{order.MorkOrder.SuborderId}");
  370. mORKS.HistorySuborderId.Add(order.MorkOrder.SuborderId);
  371. foreach (var item in order.MorkOrder.GoodBatchings)
  372. {
  373. var res = orderMaterialDelivery?.BatchingInfo?.FirstOrDefault(p => p.BatchingId == item.BatchingId);
  374. if (res != null)
  375. {
  376. if (ushort.TryParse(res.BatchingLoc, out ushort loc))
  377. {
  378. if (loc >= 1 && loc <= 5)
  379. {
  380. if (mORKS.RBTakeNoodleTask.FirstOrDefault(p => p.SuborderId == order.MorkOrder.SuborderId) == null)
  381. mORKS.RBTakeNoodleTask.Enqueue(new OrderLocInfo()
  382. {
  383. GoodName = order.MorkOrder.GoodsName,
  384. Loc = ushort.Parse(res.BatchingLoc),
  385. SuborderId = order.MorkOrder.SuborderId,
  386. SortNum = order.MorkOrder.SortNum,
  387. BatchingId = res.BatchingId
  388. });
  389. }
  390. else if (loc >= 10 && loc <= 11)
  391. {
  392. int index = 0;
  393. if (recipeBoms != null)
  394. {
  395. index = Array.FindIndex(recipeBoms.RecipeIds?.ToArray(), p => p.RecipeId == order.MorkOrder.RecipeId);
  396. index++;
  397. }
  398. if (mORKS.TakeBowlTask.FirstOrDefault(p => p.SuborderId == order.MorkOrder.SuborderId) == null)
  399. mORKS.TakeBowlTask.Enqueue(new OrderLocInfo()
  400. {
  401. BatchingId = res.BatchingId,
  402. GoodName = order.MorkOrder.GoodsName,
  403. Loc = ushort.Parse(res.BatchingLoc),
  404. SuborderId = order.MorkOrder.SuborderId,
  405. SortNum = order.MorkOrder.SortNum,
  406. RecipeNumber = (index >= 1 && index <= 10) ? (ushort)index : (ushort)0
  407. });
  408. }
  409. }
  410. }
  411. }
  412. }
  413. });
  414. }
  415. public override void MainTask()
  416. {
  417. mORKS.AllowRun = mORKS.InitComplete;
  418. if (Json<KeepDataBase>.Data.IsVerify)
  419. IsHealth = mORKS.InitComplete;
  420. else
  421. IsHealth = true;
  422. TakeBowlTask();
  423. TakeNoodleTask();
  424. OutNoodleTask();
  425. SingleDetect();
  426. TurntableControl();
  427. }
  428. private void BowlControl(OrderLocInfo orderLocInfo)
  429. {
  430. if (orderLocInfo.Loc >= 10 && orderLocInfo.Loc <= 11)
  431. {
  432. mORKS.TakeBowlId = orderLocInfo.SuborderId;
  433. mORKS.TakeBowName = orderLocInfo.GoodName;
  434. mORKS.TakeBowSortNum = orderLocInfo.SortNum;
  435. TakeBowlControl(orderLocInfo.Loc);
  436. OrderChange(mORKS.TakeBowlId, ORDER_STATUS.COOKING);
  437. DeviceProcessLogShow($"订单【{mORKS.TakeBowlId}】执行取碗控制,位置:[{orderLocInfo.Loc}]");
  438. mORKS.TakeBowlInterlock = true;
  439. }
  440. }
  441. /// <summary>
  442. /// 取碗控制
  443. /// </summary>
  444. private void TakeBowlTask()
  445. {
  446. if (mORKS.AllowRun && mORKS.TakeBowlTask.Count > 0 && !mORKS.DropBowlMechanismStatus && !mORKS.TakeBowlInterlock)
  447. {
  448. ushort BowLoc = 0;
  449. var res = orderMaterialDelivery?.BatchingInfo?.Where(p => p.BatchingId == mORKS.TakeBowlTask.ElementAt(0).BatchingId).ToList();
  450. if (res == null || res?.Count == 0)
  451. {
  452. if (mORKS.TakeBowlTask.TryDequeue(out OrderLocInfo orderLocInfo)) BowlControl(orderLocInfo);
  453. }
  454. else
  455. {
  456. foreach (var item in res)
  457. {
  458. if (ushort.TryParse(item.BatchingLoc, out ushort loc))
  459. {
  460. if (loc == 10 && mORKS.SmallBowlYesOrNoCheck)
  461. {
  462. BowLoc = loc;
  463. break;
  464. }
  465. else if (loc == 11 && mORKS.LargeBowYesOrNoCheck)
  466. {
  467. BowLoc = loc;
  468. break;
  469. }
  470. }
  471. }
  472. if (BowLoc >= 10 && BowLoc <= 11)
  473. {
  474. if (mORKS.TakeBowlTask.TryDequeue(out OrderLocInfo orderLocInfo))
  475. {
  476. orderLocInfo.Loc = BowLoc;
  477. BowlControl(orderLocInfo);
  478. }
  479. }
  480. }
  481. }
  482. }
  483. /// <summary>
  484. /// 转台控制
  485. /// </summary>
  486. private void TurntableControl()
  487. {
  488. if (Global.EnableLocalSimOrder)
  489. {
  490. //不做轮询,直接取面,模拟订单使用
  491. if (mORKS.SiloInPlace && !mORKS.Feeding && mORKS.InitComplete && !mORKS.AllowTakeNoodle && mORKS.RBTakeNoodleTask.Count > 0)
  492. {
  493. if (mORKS.TurntableLowPosition)
  494. {
  495. TurntableStart(mORKS.RBTakeNoodleTask.ElementAt(0).Loc);
  496. mORKS.TurntableLocLists.Clear();
  497. mORKS.AllowTakeNoodle = true;
  498. DeviceProcessLogShow($"控制机器人去转台【{mORKS.RBTakeNoodleTask.ElementAt(0).Loc}】号位置取面");
  499. }
  500. }
  501. }
  502. else
  503. {
  504. //正常轮询
  505. if (Delay.GetInstance("到位检测延时").Start(mORKS.SiloInPlace, 2))
  506. {
  507. if (mORKS.SiloInPlace && !mORKS.Feeding && mORKS.InitComplete && !mORKS.AllowTakeNoodle && mORKS.RBTakeNoodleTask.Count > 0)
  508. {
  509. var result = orderMaterialDelivery.BatchingInfo.Where(p => p.BatchingId == mORKS.RBTakeNoodleTask.ElementAt(0).BatchingId).ToList();
  510. if (result != null)
  511. {
  512. var res = result.FirstOrDefault(P => P.BatchingLoc == mORKS.CurrentFeedbackLoc.ToString());
  513. if (mORKS.TurntableLowPosition && res != null)
  514. {
  515. TurntableStart(mORKS.CurrentFeedbackLoc);
  516. mORKS.TurntableLocLists.Clear();
  517. mORKS.AllowTakeNoodle = true;
  518. DeviceProcessLogShow($"控制机器人去转台【{mORKS.CurrentFeedbackLoc}】号位置取面");
  519. }
  520. else
  521. {
  522. if (!mORKS.TurntableInterlock)
  523. {
  524. foreach (var item in result)
  525. {
  526. if (ushort.TryParse(item.BatchingLoc, out ushort loc))
  527. {
  528. if (mORKS.CurrentFeedbackLoc != loc && !mORKS.TurntableLocLists.Contains(loc))
  529. {
  530. if (!mORKS.TurntableLowPosition)
  531. {
  532. DeviceProcessLogShow($"执行了转台启动互锁信号复位");
  533. }
  534. TurntableStart(loc);
  535. DeviceProcessLogShow($"没有物料检测的启动转台控制,转台位置:[{loc}]");
  536. break;
  537. }
  538. else if (mORKS.CurrentFeedbackLoc == loc && !mORKS.TurntableLocLists.Contains(loc)) mORKS.TurntableLocLists.Add(loc);
  539. }
  540. }
  541. }
  542. }
  543. }
  544. else DeviceProcessLogShow("未找到可用的物料信息");
  545. }
  546. }
  547. }
  548. //补料中检测
  549. if (RTrig.GetInstance("mORKS.Feeding").Start(mORKS.Feeding))
  550. {
  551. mORKS.AllowTakeNoodle = false;
  552. mORKS.TakeNoodleInterlock = false;
  553. }
  554. //转台到位检测
  555. if (RTrig.GetInstance("TurntableInPlace").Start(mORKS.SiloInPlace && mORKS.CurrentLoc == mORKS.CurrentFeedbackLoc))
  556. {
  557. mORKS.TurntableInterlock = false;
  558. DeviceProcessLogShow("转台到位检测");
  559. }
  560. //补料完成检测
  561. if (RTrig.GetInstance("FeedComplete").Start(mORKS.FeedComplete))
  562. {
  563. if (!mORKS.AllowTakeNoodle && mORKS.TurntableLocLists.Count > 0)
  564. {
  565. mORKS.TurntableLocLists.Clear();
  566. mORKS.TurntableInterlock = false;
  567. DeviceProcessLogShow("补料完成检测");
  568. }
  569. }
  570. }
  571. /// <summary>
  572. /// 取面任务
  573. /// </summary>
  574. private void TakeNoodleTask()
  575. {
  576. //取面控制
  577. if (mORKS.AllowRun && mORKS.RobotStatus && !mORKS.Feeding && !mORKS.RobotTaskInterlock && mORKS.AllowTakeNoodle && mORKS.SiloInPlace && !mORKS.TakeNoodleInterlock && mORKS.RBTakeNoodleTask.Count > 0)
  578. {
  579. if (mORKS.CurrentLoc == mORKS.CurrentFeedbackLoc)
  580. {
  581. //int loc = Array.FindIndex(mORKS.NoodleCookerStatus, p => p == false);//查找煮面炉空闲位置
  582. int loc = mORKS.nsm.ToList().FindIndex(p => p.NoodleCookerStatus == false && p.IsShield == false);//查找煮面炉空闲位置
  583. if (loc >= 0 && loc <= 5)
  584. {
  585. //if (!Json<MorksPar>.Data.parSets.ElementAt(loc).IsShield)//检查该煮面篮是否被屏蔽
  586. //{
  587. if (mORKS.RBTakeNoodleTask.TryDequeue(out OrderLocInfo orderLocInfo))
  588. {
  589. //写入煮面时间
  590. List<ushort> values = new List<ushort>();
  591. values.Add(Json<MorksPar>.Data.parSets.ElementAt(loc).Minute);
  592. values.Add(Json<MorksPar>.Data.parSets.ElementAt(loc).Second);
  593. WriteData($"VW{324 + (loc * 4)}", values.ToArray());
  594. mORKS.CurrentLoc = 0;
  595. mORKS.CookNodelId[loc] = orderLocInfo.SuborderId;
  596. mORKS.NoodleCookerStatus[loc] = true;
  597. SetFallNoodleLoc((ushort)(loc + 1));
  598. //机器人开始取面
  599. OrderChange(orderLocInfo.SuborderId, ORDER_STATUS.COOKING);
  600. DeviceProcessLogShow($"订单【{orderLocInfo.SuborderId}】,机器人倒面至【{loc + 1}】号煮面栏");
  601. mORKS.TakeNoodleInterlock = true;
  602. }
  603. }
  604. }
  605. }
  606. }
  607. /// <summary>
  608. /// 出餐控制
  609. /// </summary>
  610. private void OutNoodleTask()
  611. {
  612. if (mORKS.AllowInvertedFace && mORKS.RobotTaskInterlock && !mORKS.TakeNoodleInterlock && mORKS.RobotStatus)
  613. {
  614. int loc = Array.FindIndex(mORKS.CookNodelId, p => p == mORKS.IngredientsCompleteId && p.Length > 0);
  615. if (loc >= 0 && loc <= 5)
  616. {
  617. if (mORKS.CookNoodleCom[loc])
  618. {
  619. SetTakeNoodleLoc((ushort)(loc + 1));
  620. mORKS.NoodleCookerStatus[loc] = false;
  621. WriteData($"VW260", (ushort)0);//设置出汤时间
  622. OrderChange(mORKS.IngredientsCompleteId, ORDER_STATUS.COMPLETED_COOK);
  623. DeviceProcessLogShow($"订单【{mORKS.IngredientsCompleteId}】制作完成");
  624. mORKS.CookCompleteFlatBit = true;
  625. mORKS.OutMealId = mORKS.IngredientsCompleteId;
  626. mORKS.OutMealName = mORKS.IngredientsCompleteName;
  627. mORKS.OutMealSortNum = mORKS.IngredientsCompleteSortNum;
  628. mORKS.IngredientsCompleteId = string.Empty;
  629. mORKS.CookNodelId[loc] = string.Empty;
  630. DeviceProcessLogShow($"{loc + 1} 号位置出餐控制,订单ID:{mORKS.OutMealId}");
  631. mORKS.CookNoodleCom[loc] = false;
  632. }
  633. }
  634. }
  635. }
  636. /// <summary>
  637. /// 信号检测
  638. /// </summary>
  639. private void SingleDetect()
  640. {
  641. //允许倒面信号检测
  642. if (RTrig.GetInstance("AllowFallNoodle").Start(mORKS.AllowInvertedFace))
  643. {
  644. mORKS.IngredientsCompleteId = mORKS.TakeBowlId;
  645. mORKS.IngredientsCompleteName = mORKS.TakeBowName;
  646. mORKS.IngredientsCompleteSortNum = mORKS.TakeBowSortNum;
  647. mORKS.TakeBowSortNum = 0;
  648. mORKS.TakeBowlId = string.Empty;
  649. mORKS.TakeBowName = string.Empty;
  650. DeviceProcessLogShow($"碗到位,允许到面,{mORKS.IngredientsCompleteId}");
  651. mORKS.TakeBowlInterlock = false;
  652. }
  653. //取餐完成逻辑处理
  654. if (RTrig.GetInstance("CompleteChange1").Start(mORKS.DiningComplete) && mORKS.CookCompleteFlatBit == true)
  655. {
  656. OrderChange(mORKS.OutMealId, ORDER_STATUS.COMPLETED_TAKE);
  657. DeviceProcessLogShow($"订单【{mORKS.OutMealId}】取餐完成");
  658. WriteData("M10.1", false);
  659. DeviceProcessLogShow($"出餐订单序号【{mORKS.OutMealSortNum}】");
  660. VoiceAPI.Speak(mORKS.OutMealSortNum.ToString());
  661. //DeviceProcessLogShow($"叫号系统通知主题【MORKS/VoiceCall/{DeviceId}】");
  662. //Plugin.GetInstance().GetPlugin<MQTTMgr>().Publish($"MORKS/VoiceCall/{DeviceId}", mORKS.OutMealSortNum.ToString());
  663. mORKS.CookCompleteFlatBit = false;
  664. mORKS.OutMealId = string.Empty;
  665. mORKS.OutMealName = string.Empty;
  666. mORKS.OutMealSortNum = 0;
  667. }
  668. //机器人取面完成信号检测
  669. if (RTrig.GetInstance("TakeNoodleComplete").Start(mORKS.RobotTakeNoodleCom))
  670. {
  671. mORKS.TakeNoodleInterlock = false;
  672. mORKS.AllowTakeNoodle = false;
  673. mORKS.TurntableInterlock = false;
  674. DeviceProcessLogShow("机器人取面完成信号检测");
  675. }
  676. int OutMealRequstCount = mORKS.CookNoodleCom.Where(p => p == true).ToList().Count;
  677. //int mlCount = mORKS.NoodleCookerStatus.Where(p => p == true).ToList().Count - Json<MorksPar>.Data.parSets.Where(x => x.IsShield == true).ToList().Count;
  678. int mlCount = mORKS.nsm.Where(p => p.NoodleCookerStatus == true && p.IsShield == false).ToList().Count;
  679. int index = Array.FindIndex(mORKS.CookNodelId, p => p == mORKS.IngredientsCompleteId);
  680. bool isok = index >= 0 && index < mORKS.CookNoodleCom.Length && mORKS.CookNoodleCom[index];
  681. mORKS.PriorityJudgment = Delay.GetInstance("取餐优先级判断").Start(mORKS.TurntableLocLists.Count > 0 && !mORKS.TurntableLowPosition, 4);
  682. //mORKS.RobotTaskInterlock = OutMealRequstCount > 0 && mORKS.AllowInvertedFace && (mlCount >= 2 || mORKS.RBTakeNoodleTask.Count == 0 || mORKS.PriorityJudgment);
  683. mORKS.RobotTaskInterlock = isok && mORKS.AllowInvertedFace && (mlCount >= 2 || mORKS.RBTakeNoodleTask.Count == 0 || mORKS.PriorityJudgment);
  684. }
  685. /// <summary>
  686. /// 语音提醒取餐
  687. /// </summary>
  688. /// <param name="meal"></param>
  689. private void WaitMeaLSpeak(string meal)
  690. {
  691. //VoiceAPI.m_SystemPlayWav(@"Vioce\电子提示音.wav");
  692. //Thread.Sleep(1000);
  693. //if (meal != null) mORKS.speech.Speak(meal);
  694. //VoiceAPI.m_SystemPlayWav(@"Vioce\取餐通知.wav");
  695. }
  696. #region PLC 控制函数
  697. private void WriteData(string address, object value)
  698. {
  699. EventBus.EventBus.GetInstance().Publish(new WriteModel() { DeviceId = DeviceId, Address = address, Value = value });
  700. }
  701. /// <summary>
  702. /// 设备初始化
  703. /// </summary>
  704. public async void DeviceInit()
  705. {
  706. WriteData("M0.0", true);
  707. await Task.Delay(1000);
  708. WriteData("M0.0", false);
  709. }
  710. /// <summary>
  711. /// 取碗控制
  712. /// </summary>
  713. /// <param name="loc"></param>
  714. private void TakeBowlControl(ushort loc)
  715. {
  716. if (loc == 10)//一次性碗
  717. {
  718. WriteData("M9.1", true);
  719. }
  720. else if (loc == 11)//大碗
  721. {
  722. WriteData("M9.0", true);
  723. }
  724. }
  725. /// <summary>
  726. /// 启动转台
  727. /// </summary>
  728. /// <param name="loc"></param>
  729. private void TurntableStart(ushort loc)
  730. {
  731. if (loc >= 1 && loc <= 5)
  732. {
  733. mORKS.CurrentLoc = loc;
  734. mORKS.TurntableInterlock = true;
  735. mORKS.TurntableLocLists.Add(loc);
  736. WriteData($"M13.{loc - 1}", true);
  737. }
  738. }
  739. /// <summary>
  740. /// 设置倒面位置
  741. /// </summary>
  742. /// <param name="loc"></param>
  743. private void SetFallNoodleLoc(ushort loc)
  744. {
  745. if (loc >= 1 && loc <= 6)
  746. WriteData($"M14.{loc - 1}", true);
  747. }
  748. /// <summary>
  749. /// 设置取面位置
  750. /// </summary>
  751. /// <param name="loc"></param>
  752. private void SetTakeNoodleLoc(ushort loc)
  753. {
  754. if (loc >= 1 && loc <= 6)
  755. WriteData($"M15.{loc - 1}", true);
  756. }
  757. public override void SimOrder()
  758. {
  759. EventBus.EventBus.GetInstance().Subscribe<MorksSimorderModel>(0, delegate (IEvent @event, EventCallBackHandle callBackHandle)
  760. {
  761. if (@event != null && @event is MorksSimorderModel msm)
  762. {
  763. string guid = Guid.NewGuid().ToString();
  764. if (msm.NoodleLoc >= 1 && msm.NoodleLoc <= 5)
  765. {
  766. mORKS.RBTakeNoodleTask.Enqueue(new OrderLocInfo() { Loc = (ushort)msm.NoodleLoc, SuborderId = guid });
  767. MessageLog.GetInstance.Show($"添加订单:面条位置【{(ushort)msm.NoodleLoc}】");
  768. }
  769. if (msm.Bowloc >= 10 && msm.Bowloc <= 11)
  770. {
  771. mORKS.TakeBowlTask.Enqueue(new OrderLocInfo() { Loc = (ushort)msm.Bowloc, SuborderId = guid });
  772. MessageLog.GetInstance.Show($"添加订单:碗位置【{(ushort)msm.Bowloc}】");
  773. }
  774. }
  775. });
  776. }
  777. #endregion
  778. }
  779. }