Vehicle.cs 104 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987
  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using System.Diagnostics;
  5. using System.Linq;
  6. using System.Runtime.Remoting.Messaging;
  7. using System.Threading;
  8. using System.Threading.Tasks;
  9. using FluentResults;
  10. using GSG.NET.Concurrent;
  11. using GSG.NET.Extensions;
  12. using GSG.NET.LINQ;
  13. using GSG.NET.Logging;
  14. using GSG.NET.Quartz;
  15. using GSG.NET.Utils;
  16. using OHV.Common.Events;
  17. using OHV.Common.Model;
  18. using OHV.Common.Shareds;
  19. using OHV.SqliteDAL;
  20. using Prism.Events;
  21. using VehicleControlSystem.ControlLayer.Actuator.Cylinder;
  22. using VehicleControlSystem.ControlLayer.Drive;
  23. using VehicleControlSystem.ControlLayer.IO;
  24. using VehicleControlSystem.ControlLayer.MQ;
  25. using VehicleControlSystem.ControlLayer.Serial.BatteryTabos;
  26. using VehicleControlSystem.ControlLayer.Serial.DataModel;
  27. using VehicleControlSystem.Managers;
  28. using static VehicleControlSystem.ControlLayer.Drive.GSIDrive;
  29. namespace VehicleControlSystem.ControlLayer
  30. {
  31. public class VehicleRefObjects
  32. {
  33. public GSIDrive Drive { get; set; }
  34. public Clamp Clamp { get; set; }
  35. public Conveyor Conveyor { get; set; }
  36. public EzIO IO { get; set; }
  37. public Steering Steering { get; set; }
  38. public BMUManager BMUManager { get; set; }
  39. public ZmqManager ZmqManager { get; set; }
  40. public List<Alarm> Alarms { get; set; }
  41. public HostManager HostManager { get; set; }
  42. public SqliteManager Sqlite { get; set; }
  43. public AutoManager AutoManager { get; set; }
  44. }
  45. /// <summary>
  46. /// Control Layer 의 자원을 여기서 사용하자.
  47. /// </summary>
  48. public class Vehicle : ControlObjectBase, IDisposable
  49. {
  50. /// <summary>
  51. /// OCS Report Code
  52. /// 목적지에 도착해서 Load, Unload 시 발생하는 Alarm
  53. /// </summary>
  54. public enum eFailCode
  55. {
  56. Load_PortHasNotCarrier = 1,
  57. Load_VehicleHasCarrier,
  58. Unload_PortHasCarrier,
  59. Unload_VehicleHasNotCarrier,
  60. LoadPIOInterlockTimeout,
  61. UnlaodPIOInterlockTimeout,
  62. }
  63. static Logger logger = Logger.GetLogger();
  64. static Logger loggerPIO = Logger.GetLogger("PIO");
  65. #region Properties
  66. /// <summary>
  67. /// Tag 위치
  68. /// </summary>
  69. private int currentTag = 0;
  70. public int CurrentTag
  71. {
  72. get { return currentTag; }
  73. set
  74. {
  75. if (SetField(ref this.currentTag, value))
  76. {
  77. //var info = this.refObjects.Sqlite.VehicleInfoDAL.GetAll().FirstOrDefault();
  78. //info.CurrentTag = value.ToString();
  79. //this.refObjects.Sqlite.VehicleInfoDAL.Update( info );
  80. this.OnCurrentTagChanged?.Invoke(value);
  81. }
  82. }
  83. }
  84. /// <summary>
  85. /// Scale Value
  86. /// </summary>
  87. private double currentPosition;
  88. public double CurrentPosition
  89. {
  90. get { return currentPosition; }
  91. set
  92. {
  93. if (SetField(ref this.currentPosition, value))
  94. {
  95. }
  96. }
  97. }
  98. private double currentSpeed;
  99. public double CurrentSpeed
  100. {
  101. get { return currentSpeed; }
  102. set { SetField(ref this.currentSpeed, value); }
  103. }
  104. private double currentTorque;
  105. public double CurrentTorque
  106. {
  107. get { return currentTorque; }
  108. set { SetField(ref this.currentTorque, value); }
  109. }
  110. private bool isContain;
  111. public bool IsContain
  112. {
  113. get { return isContain; }
  114. set { SetField(ref this.isContain, value); }
  115. }
  116. eClampState _clampState;
  117. public eClampState ClampState
  118. {
  119. get { return this._clampState; }
  120. set { this.SetField(ref this._clampState, value); }
  121. }
  122. private eSteeringState steeringState;
  123. public eSteeringState SteeringState
  124. {
  125. get { return steeringState; }
  126. set
  127. {
  128. if (SetField(ref this.steeringState, value))
  129. {
  130. this.refObjects.ZmqManager.SetCurrentSteeringState(value);
  131. }
  132. }
  133. }
  134. private int _obstacleDrive;
  135. public int ObstacleDrive { get { return this._obstacleDrive; } set { SetField(ref this._obstacleDrive, value); } }
  136. private int _obstacleCurve;
  137. public int ObstacleCurve { get { return this._obstacleCurve; } set { SetField(ref this._obstacleCurve, value); } }
  138. private int obstacleCurrent;
  139. public int ObstacleCurrent { get { return this.ObstacleCurrent; } set { SetField(ref this.obstacleCurrent, value); } }
  140. private eObstacleState obstacleState = eObstacleState.Normal;
  141. public eObstacleState ObstacleStateProperty
  142. {
  143. get { return obstacleState; }
  144. set
  145. {
  146. if (SetField(ref this.obstacleState, value))
  147. {
  148. if (value == eObstacleState.Blocked)
  149. this.VehicleStateProperty = eVehicleState.Blocked;
  150. }
  151. }
  152. }
  153. private eVehicleState vehicleState = eVehicleState.None;
  154. public eVehicleState VehicleStateProperty
  155. {
  156. get { return vehicleState; }
  157. set
  158. {
  159. if (SetField(ref this.vehicleState, value))
  160. {
  161. var info = this.refObjects.Sqlite.VehicleInfoDAL.GetAll().FirstOrDefault();
  162. info.VehicleState = value;
  163. //this.refObjects.Sqlite.VehicleInfoDAL.Update( info );
  164. }
  165. }
  166. }
  167. private eMachineMode machineMode = eMachineMode.LocalMode;
  168. public eMachineMode MachineMode
  169. {
  170. get { return machineMode; }
  171. set
  172. {
  173. if (SetField(ref this.machineMode, value))
  174. {
  175. var info = this.refObjects.Sqlite.VehicleInfoDAL.GetAll().FirstOrDefault();
  176. info.MachineMode = value;
  177. //this.refObjects.Sqlite.VehicleInfoDAL.Update( info );
  178. }
  179. }
  180. }
  181. private eVehicleOnRailState vehicleOnRailState = eVehicleOnRailState.None;
  182. public eVehicleOnRailState VehicleOnRailState
  183. {
  184. get => this.vehicleOnRailState;
  185. set
  186. {
  187. if (SetField(ref this.vehicleOnRailState, value))
  188. {
  189. var info = refObjects.Sqlite.VehicleInfoDAL.GetAll().FirstOrDefault();
  190. info.VehicleOnRailState = value;
  191. refObjects.Sqlite.VehicleInfoDAL.Update(info);
  192. if (value == eVehicleOnRailState.Install)
  193. refObjects.HostManager.Send_Inline();
  194. else
  195. refObjects.HostManager.Send_Outline();
  196. }
  197. }
  198. }
  199. private int obstaclePattern;
  200. public int ObstaclePattern
  201. {
  202. get { return obstaclePattern; }
  203. set { SetField(ref this.obstaclePattern, value); }
  204. }
  205. private DriveState frontDriveState;
  206. public DriveState FrontDriveState
  207. {
  208. get { return frontDriveState; }
  209. set { SetField(ref this.frontDriveState, value); }
  210. }
  211. private DriveState rearDriveState;
  212. public DriveState RearDriveState
  213. {
  214. get { return rearDriveState; }
  215. set { SetField(ref this.rearDriveState, value); }
  216. }
  217. private eVehicleAutoReadyState autoReadyState;
  218. public eVehicleAutoReadyState AutoReadyState
  219. {
  220. get { return autoReadyState; }
  221. set { SetField(ref this.autoReadyState, value); }
  222. }
  223. //이동
  224. public bool Busy
  225. {
  226. get
  227. {
  228. return this.CurrentSubCommand == null ? false : true;
  229. }
  230. set { }
  231. }
  232. public bool IsMoving { get => this.refObjects.Drive.IsDriveMoving(); }
  233. private bool batteryChargeState;
  234. public bool BatteryChargeState
  235. {
  236. get { return batteryChargeState; }
  237. set
  238. {
  239. if (SetField(ref batteryChargeState, value))
  240. {
  241. if (value)
  242. {
  243. this.OnChargingStart?.Invoke();
  244. LockUtils.Wait(1000);
  245. this.OnCharging?.Invoke();
  246. //QuartzUtils.Invoke("BatterLogger", QuartzUtils.GetExpnMinute(1), () => { batteryLogger.I($"SOC - {this.BatteryStateOfCharge} / Current - {this.BatteryCurrent} / Voltage - {this.BatteryVoltage} / Capacity - {this.BatteryCapacity} / Energy - {this.BatteryEnergy}"); });
  247. }
  248. else
  249. {
  250. //QuartzUtils.StopSchedule("BatterLogger");
  251. this.OnChargingFull?.Invoke();
  252. }
  253. }
  254. }
  255. }
  256. #region Battery Property
  257. double batteryVoltage;
  258. public double BatteryVoltage
  259. {
  260. get { return this.batteryVoltage; }
  261. set
  262. {
  263. //var d = Math.Truncate( value * 10 ) / 10;
  264. //this.SetField( ref this.batteryVoltage, d );
  265. this.SetField(ref this.batteryVoltage, value);
  266. }
  267. }
  268. double batteryCurrent;
  269. public double BatteryCurrent
  270. {
  271. get { return this.batteryCurrent; }
  272. set { this.SetField(ref this.batteryCurrent, value); }
  273. }
  274. double batteryState;
  275. public double BatteryState
  276. {
  277. get { return this.batteryState; }
  278. set { this.SetField(ref this.batteryState, value); }
  279. }
  280. double batteryChargeTime;
  281. public double BatteryChargeTime
  282. {
  283. get { return this.batteryChargeTime; }
  284. set { this.SetField(ref this.batteryChargeTime, value); }
  285. }
  286. double batteryDisChargeTime;
  287. public double BatteryDisChargeTime
  288. {
  289. get { return this.batteryDisChargeTime; }
  290. set { this.SetField(ref this.batteryDisChargeTime, value); }
  291. }
  292. double batteryStateOfCharge;
  293. public double BatteryStateOfCharge
  294. {
  295. get { return this.batteryStateOfCharge; }
  296. set { if (this.SetField(ref this.batteryStateOfCharge, value)) { this.BatteryChargeStateLED(value); } }
  297. }
  298. double batteryStateOfHealth;
  299. public double BatteryStateOfHealth
  300. {
  301. get { return this.batteryStateOfHealth; }
  302. set
  303. {
  304. if (this.SetField(ref this.batteryStateOfHealth, value)) { }
  305. }
  306. }
  307. double batteryCapacity;
  308. public double BatteryCapacity
  309. {
  310. get { return this.batteryCapacity; }
  311. set { this.SetField(ref this.batteryCapacity, value); }
  312. }
  313. double batteryEnergy;
  314. public double BatteryEnergy
  315. {
  316. get { return this.batteryEnergy; }
  317. set
  318. {
  319. var d = Math.Truncate(value * 10) / 10;
  320. this.SetField(ref this.batteryEnergy, d);
  321. }
  322. }
  323. double batteryTemperature;
  324. public double BatteryTemperature
  325. {
  326. get { return this.batteryTemperature; }
  327. set { this.SetField(ref this.batteryTemperature, value); }
  328. }
  329. bool batteryIsConnect;
  330. public bool BatteryIsConnect
  331. {
  332. get
  333. {
  334. return batteryIsConnect;
  335. }
  336. set { this.SetField(ref this.batteryIsConnect, value); }
  337. }
  338. #endregion
  339. public bool IsError
  340. {
  341. get
  342. {
  343. if (refObjects.AutoManager.ActiveAlarms.Count > 0)
  344. return true;
  345. else
  346. return false;
  347. }
  348. }
  349. public SubCmd CurrentSubCommand { get; private set; }
  350. double frontLoadFactor = 0;
  351. public double FrontLoadFactor
  352. {
  353. get { return this.frontLoadFactor; }
  354. set { SetField(ref this.frontLoadFactor, value); }
  355. }
  356. double frontRpm = 0;
  357. public double FrontRpm
  358. {
  359. get { return this.frontRpm; }
  360. set { SetField(ref this.frontRpm, value); }
  361. }
  362. double frontSpeed = 0;
  363. public double FrontSpeed
  364. {
  365. get { return this.frontSpeed; }
  366. set { SetField(ref this.frontSpeed, value); }
  367. }
  368. double frontTorque = 0;
  369. public double FrontTorque
  370. {
  371. get { return this.frontTorque; }
  372. set { SetField(ref this.frontTorque, value); }
  373. }
  374. double rearLoadFactor = 0;
  375. public double RearLoadFactor
  376. {
  377. get { return this.rearLoadFactor; }
  378. set { SetField(ref this.rearLoadFactor, value); }
  379. }
  380. double rearRpm = 0;
  381. public double RearRpm
  382. {
  383. get { return this.rearRpm; }
  384. set { SetField(ref this.rearRpm, value); }
  385. }
  386. double rearSpeed = 0;
  387. public double RearSpeed
  388. {
  389. get { return this.rearSpeed; }
  390. set { SetField(ref this.rearSpeed, value); }
  391. }
  392. double rearTorque = 0;
  393. public double RearTorque
  394. {
  395. get { return this.rearTorque; }
  396. set { SetField(ref this.rearTorque, value); }
  397. }
  398. private double linearSpeed;
  399. public double LinearSpeed
  400. {
  401. get { return linearSpeed; }
  402. set { SetField(ref this.linearSpeed, value); }
  403. }
  404. private double curveSpeed;
  405. public double CurveSpeed
  406. {
  407. get { return curveSpeed; }
  408. set { SetField(ref this.curveSpeed, value); }
  409. }
  410. private double accelSpeed;
  411. public double AccelSpeed
  412. {
  413. get { return accelSpeed; }
  414. set { SetField(ref this.accelSpeed, value); }
  415. }
  416. private double decelSpeed;
  417. public double DecelSpeed
  418. {
  419. get { return decelSpeed; }
  420. set { SetField(ref this.decelSpeed, value); }
  421. }
  422. private double creepSpeed;
  423. public double CreepSpeed
  424. {
  425. get { return creepSpeed; }
  426. set { SetField(ref this.creepSpeed, value); }
  427. }
  428. private double creepDistance;
  429. public double CreepDistance
  430. {
  431. get { return creepDistance; }
  432. set { SetField(ref this.creepDistance, value); }
  433. }
  434. private double jogSpeed;
  435. public double JogSPeed
  436. {
  437. get { return jogSpeed; }
  438. set { SetField(ref this.jogSpeed, value); }
  439. }
  440. private bool isCanStanbyLocation;
  441. public bool IsCanStanbyLocation
  442. {
  443. get { return isCanStanbyLocation; }
  444. set { this.SetField(ref this.isCanStanbyLocation, value); }
  445. }
  446. public bool IsAutoRun { get; internal set; } = false;
  447. #endregion
  448. #region Event
  449. public event Action OnMoveReady;
  450. public event Action OnMoving;
  451. public event Action OnMoveFinish;
  452. public event Action OnChargingStart;
  453. public event Action OnCharging;
  454. public event Action OnChargingFull;
  455. //public event Action<double> OnBatteryVelueChanged;
  456. public event Action<bool> OnPIOStart;
  457. public event Action<bool> OnConveyorStart;
  458. public event Action<bool> OnConveyorStop;
  459. public event Action<bool> OnCarrierDetected;
  460. public event Action OnLoadComplete;
  461. public event Action OnUnloadComplete;
  462. public event Action<int> OnCurrentTagChanged;
  463. public event Action<eFailCode> OnFailReport;
  464. #endregion
  465. #region List.
  466. List<ICylinder> cylinders = new List<ICylinder>();
  467. List<string> obstacleBitList = new List<string>();
  468. #endregion
  469. VehicleRefObjects refObjects = null;
  470. //SqliteManager sql = null;
  471. //AutoManager autoManager = null;
  472. ThreadCancel cancel = new ThreadCancel();
  473. TaskCancel taskCancel = new TaskCancel();
  474. TaskCancel taskMoveCancel = new TaskCancel();
  475. IEventAggregator eventAggregator;
  476. public Vehicle(IEventAggregator ea)
  477. {
  478. this.obstacleBitList.AddRange(new string[]
  479. {
  480. "OUT_OBSTRUCTION_PATTERN_00",
  481. "OUT_OBSTRUCTION_PATTERN_01",
  482. "OUT_OBSTRUCTION_PATTERN_02",
  483. "OUT_OBSTRUCTION_PATTERN_03",
  484. "OUT_OBSTRUCTION_PATTERN_04",
  485. });
  486. this.eventAggregator = ea;
  487. this.eventAggregator.GetEvent<DriveControlPubSubEvent>().Unsubscribe(ReceiveDriveControlEvent);
  488. this.eventAggregator.GetEvent<DriveControlPubSubEvent>().Subscribe(ReceiveDriveControlEvent);
  489. this.eventAggregator.GetEvent<ObstacleControlPubSubEvent>().Unsubscribe(ObstacleReceiveEvent);
  490. this.eventAggregator.GetEvent<ObstacleControlPubSubEvent>().Subscribe(ObstacleReceiveEvent);
  491. }
  492. public void DependencyInjection(VehicleRefObjects vehicleRefObjects)
  493. {
  494. this.refObjects = vehicleRefObjects;
  495. this.refObjects.IO.OnChangedIO += IO_OnChangedIO;
  496. this.refObjects.BMUManager.OnConnect += BMUManager_OnConnect;
  497. this.refObjects.BMUManager.OnDisconnect += BMUManager_OnDisconnect;
  498. this.refObjects.BMUManager.OnChangedReceivedData += BMUManager_OnChangedReceivedData;
  499. this.refObjects.BMUManager.OnFirstColtd += BMUManager_OnFirstColtd;
  500. this.refObjects.Drive.PropertyChanged += Motion_PropertyChanged;
  501. this.refObjects.Clamp.PropertyChanged += Clamp_PropertyChanged;
  502. this.refObjects.Steering.OnSteeringError += Steering_OnSteeringError;
  503. this.refObjects.Steering.PropertyChanged += Steering_PropertyChanged;
  504. this.refObjects.ZmqManager.PropertyChanged += ZmqManager_PropertyChanged;
  505. this.refObjects.ZmqManager.OnDriveAlarmBitArrayChg += ZmqManager_OnDriveAlarmBitArrayChg;
  506. this.refObjects.AutoManager.OnOperationModeChanged += AutoManager_OnOperationModeChanged;
  507. }
  508. private void ZmqManager_OnDriveAlarmBitArrayChg(BitArray obj)
  509. {
  510. for (int i = 0; i < obj.Count; i++)
  511. {
  512. if (obj[i])
  513. this.OccurVehicleAlarm(i + 1000);
  514. else
  515. this.refObjects.AutoManager.ClearAlarm(i + 1000);
  516. }
  517. if ( this.refObjects.AutoManager.ActiveAlarms.Count <= 0)
  518. this.VehicleStateProperty = eVehicleState.Idle;
  519. }
  520. private void ObstacleReceiveEvent(ObstacleControlEventArgs obj)
  521. {
  522. if (this.refObjects.AutoManager.OperationModeProperty != eOperatationMode.ManualMode)
  523. return;
  524. if (obj.EventDir == ObstacleControlEventArgs.eEventDir.ToBack)
  525. {
  526. switch (obj.ControlKind)
  527. {
  528. case ObstacleControlEventArgs.eControlKind.NONE:
  529. break;
  530. case ObstacleControlEventArgs.eControlKind.DRIVE:
  531. //this.ChgObstacleDetectPattern( new Random().Next( 0 , 40 ) );
  532. break;
  533. case ObstacleControlEventArgs.eControlKind.CURVE:
  534. //this.ChgObstacleDetectPattern( new Random().Next( 0 , 40 ) );
  535. break;
  536. case ObstacleControlEventArgs.eControlKind.STATE:
  537. //var value = this.GetObstacleDetectPattern();
  538. break;
  539. case ObstacleControlEventArgs.eControlKind.INFO:
  540. {
  541. var msg = new ObstacleControlEventArgs
  542. {
  543. ControlKind = ObstacleControlEventArgs.eControlKind.INFO,
  544. Drive = this.ObstacleDrive,
  545. Curve = this.ObstacleCurve,
  546. Current = this.GetObstacleDetectPattern(),
  547. ObstacleState = this.ObstacleStateProperty.ToString()
  548. };
  549. this.ObstacleControlEventPublish(msg);
  550. }
  551. break;
  552. case ObstacleControlEventArgs.eControlKind.SAVE:
  553. {
  554. this.ObstacleCurve = obj.Curve;
  555. this.ObstacleDrive = obj.Drive;
  556. var reply = new ObstacleControlEventArgs
  557. {
  558. ControlKind = ObstacleControlEventArgs.eControlKind.SAVE
  559. };
  560. reply.Result = Results.Ok(ObstacleControlEventArgs.eControlKind.SAVE);
  561. this.ObstacleControlEventPublish(reply);
  562. }
  563. break;
  564. }
  565. }
  566. }
  567. private void ObstacleControlEventPublish(ObstacleControlEventArgs args)
  568. {
  569. args.EventDir = ObstacleControlEventArgs.eEventDir.ToFront;
  570. this.eventAggregator.GetEvent<ObstacleControlPubSubEvent>().Publish(args);
  571. }
  572. private void ReceiveDriveControlEvent(DriveControlEventArgs args)
  573. {
  574. var msg = args;
  575. if (msg.EventDir == DriveControlEventArgs.eEventDir.ToBack)
  576. {
  577. switch (msg.ControlKind)
  578. {
  579. case DriveControlEventArgs.eControlKind.MOVE:
  580. this.ReqMoveToPos(args);
  581. break;
  582. case DriveControlEventArgs.eControlKind.STOP:
  583. break;
  584. case DriveControlEventArgs.eControlKind.Steering:
  585. if ( msg.MoveDir == DriveControlEventArgs.eMoveDir.LEFT )
  586. this.refObjects.Steering.RequestControl( eSteeringState.Left );
  587. else
  588. this.refObjects.Steering.RequestControl( eSteeringState.Right );
  589. break;
  590. case DriveControlEventArgs.eControlKind.SteeringState:
  591. {
  592. var reply = new DriveControlEventArgs();
  593. reply.ControlKind = DriveControlEventArgs.eControlKind.SteeringState;
  594. if (this.refObjects.Steering.IsLeft())
  595. {
  596. reply.Args = eSteeringState.Left;
  597. reply.Result = FluentResults.Results.Ok();
  598. }
  599. else if (this.refObjects.Steering.IsRight())
  600. {
  601. reply.Args = eSteeringState.Right;
  602. reply.Result = FluentResults.Results.Ok();
  603. }
  604. else
  605. reply.Result = FluentResults.Results.Fail(new Error());
  606. this.DriveControlEventPublish(reply);
  607. }
  608. break;
  609. case DriveControlEventArgs.eControlKind.ReqCurrentPos:
  610. //this.ReqCurrentPos();
  611. break;
  612. case DriveControlEventArgs.eControlKind.ReqStopCurrentPos:
  613. //this.taskCancel.Cancel();
  614. //this.taskCancel.WaitAll();
  615. break;
  616. case DriveControlEventArgs.eControlKind.FaultReset:
  617. this.ReqFaultReset(args);
  618. break;
  619. case DriveControlEventArgs.eControlKind.DriveON:
  620. this.ReqDriveOn(args);
  621. break;
  622. case DriveControlEventArgs.eControlKind.DriveOFF:
  623. this.ReqDriveOff(args);
  624. break;
  625. case DriveControlEventArgs.eControlKind.JOG:
  626. this.ReqJog(args);
  627. break;
  628. case DriveControlEventArgs.eControlKind.VehicleState:
  629. ReqVehicleState(args);
  630. break;
  631. default:
  632. break;
  633. }
  634. }
  635. }
  636. public void ReqConveyorMove(string dir)
  637. {
  638. if (dir.Equals("CW"))
  639. this.refObjects.Conveyor.OnOffConveyor(true, false);
  640. else if (dir.Equals("CCW"))
  641. this.refObjects.Conveyor.OnOffConveyor(true, true);
  642. else if (dir.Equals("STOP"))
  643. this.refObjects.Conveyor.OnOffConveyor(false, false);
  644. }
  645. private void DriveControlEventPublish(DriveControlEventArgs args)
  646. {
  647. args.EventDir = DriveControlEventArgs.eEventDir.ToFront;
  648. this.eventAggregator.GetEvent<DriveControlPubSubEvent>().Publish(args);
  649. }
  650. public void Init()
  651. {
  652. ThreadStart();
  653. //TimerUtils.Once(5000, () => { this.CurrentPosition = 1000; });
  654. var v = refObjects.Sqlite.VehicleInfoDAL.GetAll().FirstOrDefault();
  655. v.CurrentTag = "0000";
  656. v.VehicleState = eVehicleState.None;
  657. v.MachineMode = eMachineMode.LocalMode;
  658. this.VehicleOnRailState = v.VehicleOnRailState;
  659. this.CurrentTag = this.refObjects.ZmqManager.CurrentPointNo;
  660. //refObjects.Sqlite.VehicleInfoDAL.Update( v );
  661. }
  662. public int InitializationVehicle()
  663. {
  664. #if SIMULATION
  665. this.VehicleStateProperty = eVehicleState.Idle;
  666. return 0;
  667. #else
  668. int result = 0;
  669. //PIOClear();
  670. //PIOSensorOff();
  671. result = this.refObjects.Clamp.IsAllAxisOriginReturn();
  672. if (result != ConstInt.EXECUTE_SUCCESS)
  673. {
  674. this.refObjects.Clamp.AllOriginReturn();
  675. LockUtils.Wait(2000);
  676. }
  677. if (this.refObjects.Conveyor.IsDetectedCenter()) //자제가 있으면 Lock
  678. result = this.refObjects.Clamp.Lock_Sync();
  679. else
  680. result = this.refObjects.Clamp.Unlock_Sync();
  681. result = 0;
  682. if (result != 0)
  683. {
  684. this.VehicleStateProperty = eVehicleState.Abnormal;
  685. return result;
  686. }
  687. if (this.refObjects.Drive.IsErrorOn)
  688. return 22;
  689. if (IsChargeCylinderForword())
  690. {
  691. result = this.StopBatteryCharge();
  692. if (result != 0)
  693. return result;
  694. }
  695. this.VehicleStateProperty = eVehicleState.Idle;
  696. return result;
  697. #endif
  698. }
  699. public void Dispose()
  700. {
  701. this.taskCancel.Cancel();
  702. this.cancel.Cancel();
  703. this.cancel.StopWaitAll();
  704. }
  705. #region Request Method
  706. private void ReqVehicleState(DriveControlEventArgs args)
  707. {
  708. VehicleInfo state = new VehicleInfo();
  709. state.CurrentPosition = this.CurrentPosition;
  710. state.CurrentSpeed = this.CurrentSpeed;
  711. state.CurrentTag = this.CurrentTag.ToString();
  712. state.CurrentTorque = this.CurrentTorque;
  713. var msg = new DriveControlEventArgs();
  714. msg.ControlKind = args.ControlKind;
  715. msg.Args = state;
  716. DriveControlEventPublish(msg);
  717. }
  718. private void ReqMoveToPos(DriveControlEventArgs args)
  719. {
  720. //var result = drive.Move( args.PositionTag );
  721. //this.MoveTo( "1111" );
  722. var reply = new DriveControlEventArgs();
  723. int targetTag = args.TargetRouteID;
  724. var route = this.refObjects.Sqlite.RouteDal.Get(x => x.Name.Equals(targetTag)).FirstOrDefault();
  725. if (route == null)
  726. {
  727. reply.Result = Results.Fail("Not Found Route");
  728. this.DriveControlEventPublish(reply);
  729. return;
  730. }
  731. var subCommand = new SubCmd();
  732. subCommand.TargetID = route.Name;
  733. this.MoveTo(subCommand);
  734. reply.Result = Results.Ok("Position Move");
  735. this.DriveControlEventPublish(reply);
  736. }
  737. void ReqFaultReset(DriveControlEventArgs _args)
  738. {
  739. var msg = new DriveControlEventArgs
  740. {
  741. ControlKind = DriveControlEventArgs.eControlKind.FaultReset
  742. };
  743. var ll = this.refObjects.ZmqManager.SetDriveControl(eDriveControl.ResetFault);
  744. if (ll == 38)
  745. {
  746. logger.E($"Drive Fault Reset Error");
  747. msg.Result = Results.Fail("Drive Fault Reset Error");
  748. }
  749. else
  750. msg.Result = Results.Ok();
  751. this.DriveControlEventPublish(msg);
  752. }
  753. void ReqJog(DriveControlEventArgs _args)
  754. {
  755. if (_args.JogDir == DriveControlEventArgs.eJogMoveDir.Positive)
  756. this.refObjects.Drive.JogForword();
  757. else
  758. this.refObjects.Drive.JogBackword();
  759. }
  760. Logger batteryLogger = Logger.GetLogger("BatteryLogger");
  761. void ReqCurrentPos()
  762. {
  763. var task = Task.Factory.StartNew(() =>
  764. {
  765. while (!this.taskCancel.Canceled)
  766. {
  767. LockUtils.Wait(1000);
  768. batteryLogger.I($"SOC - {this.BatteryStateOfCharge} / Current - {this.BatteryCurrent} / Voltage - {this.BatteryVoltage} / Capacity - {this.BatteryCapacity} / Energy - {this.BatteryEnergy}");
  769. //var msg = new DriveControlEventArgs
  770. //{
  771. // EventDir = DriveControlEventArgs.eEventDir.ToFront,
  772. // ControlKind = DriveControlEventArgs.eControlKind.ReqCurrentPos,
  773. // CurrentPosition = new Random().Next( 0, 1000 ),
  774. //};
  775. //this.DriveControlEventPublish( msg );
  776. }
  777. });
  778. this.taskCancel.Add(task);
  779. }
  780. void ReqDriveOn(DriveControlEventArgs _args)
  781. {
  782. var msg = new DriveControlEventArgs
  783. {
  784. ControlKind = DriveControlEventArgs.eControlKind.DriveON
  785. };
  786. var ll = this.refObjects.ZmqManager.SetDriveControl(eDriveControl.ServoOn);
  787. if (ll == 38)
  788. {
  789. logger.E($"Drive On Error");
  790. msg.Result = Results.Fail("Drive On Error");
  791. }
  792. else
  793. msg.Result = Results.Ok("Drive On");
  794. this.DriveControlEventPublish(msg);
  795. }
  796. void ReqDriveOff(DriveControlEventArgs _args)
  797. {
  798. var ll = this.refObjects.ZmqManager.SetDriveControl(eDriveControl.ServoOff);
  799. if (ll == 38)
  800. {
  801. logger.E($"Servo Off Error");
  802. }
  803. var msg = new DriveControlEventArgs
  804. {
  805. ControlKind = DriveControlEventArgs.eControlKind.DriveOFF
  806. };
  807. msg.Result = Results.Ok("Drive On");
  808. this.DriveControlEventPublish(msg);
  809. }
  810. #endregion
  811. #region Thread
  812. void ThreadStart()
  813. {
  814. this.cancel.AddGo(new Action(this.ThreadJob));
  815. //this.cancel.AddGo(new Action(this._ThVehicleStateCheck));
  816. //this.cancel.AddGo(new Action(this._ThVehicleStateCheck));
  817. }
  818. //장애물 감지 Thread
  819. //장애물 감지 패턴 변경도 여기 하자.
  820. private void _ThVehicleStateCheck()
  821. {
  822. while (!this.cancel.Canceled)
  823. {
  824. try
  825. {
  826. //if ( this.autoManager.OperationModeProperty == eOperatationMode.AutoMode )
  827. //this.CheckObstacle();
  828. //this.CheckIOState();
  829. }
  830. catch (ThreadInterruptedException threadInterruptedException)
  831. {
  832. logger.E(threadInterruptedException);
  833. }
  834. catch (Exception exception)
  835. {
  836. logger.E(exception);
  837. }
  838. finally
  839. {
  840. LockUtils.Wait(5);
  841. }
  842. }
  843. logger.D("Vehicle - _ThObstacleChecker Dispose");
  844. }
  845. void ThreadJob()
  846. {
  847. Thread.CurrentThread.Name = $"{this.GetType().Name}Job";
  848. logger.D($"Vehicle - Process Started {Thread.CurrentThread.Name}");
  849. while (!this.taskCancel.Canceled)
  850. {
  851. try
  852. {
  853. switch (this.refObjects.AutoManager.OperationModeProperty)
  854. {
  855. case eOperatationMode.ManualMode:
  856. break;
  857. case eOperatationMode.AutoMode:
  858. {
  859. switch (this.refObjects.AutoManager.AutoModeStateProperty)
  860. {
  861. case eAutoModeState.ErrorStop:
  862. break;
  863. case eAutoModeState.StartRun:
  864. this.IsAutoRun = true;
  865. break;
  866. case eAutoModeState.Run:
  867. this.DoWork();
  868. break;
  869. case eAutoModeState.Stop:
  870. this.IsAutoRun = false;
  871. break;
  872. case eAutoModeState.WaitStop:
  873. break;
  874. default:
  875. break;
  876. }
  877. }
  878. break;
  879. case eOperatationMode.InitialMode:
  880. break;
  881. default:
  882. break;
  883. }
  884. }
  885. catch (Exception ex)
  886. {
  887. logger.E(ex);
  888. }
  889. finally
  890. {
  891. LockUtils.Wait(500);
  892. }
  893. }
  894. logger.D("Vehicle - Process Disposed");
  895. }
  896. void DoWork()
  897. {
  898. //2020.08.18.Kang. 상위 연결이 안되어 있으면 SubCmd 실행을 안하고 대기함.
  899. if (!this.refObjects.HostManager.IsConnected)
  900. return;
  901. var subCmd = this.refObjects.Sqlite.SubCmdDAL.GetAll().FirstOrDefault();
  902. if (subCmd == null) return;
  903. if (!this.refObjects.Sqlite.CommandDAL.GetAll().Any(x => x.CommandID.Equals(subCmd.CmdID)))
  904. {
  905. if (subCmd.CmdType == SubCmd.eCmdType.Auto) //자동 명령중 Main Command 가 없으면 삭제.
  906. {
  907. this.refObjects.Sqlite.SubCmdDAL.Delete(subCmd.ID);
  908. logger.I($"SubCmd Deleted Main Command Not Exist - ID={subCmd.ID}, CommandID={subCmd.CmdID}");
  909. return;
  910. }
  911. }
  912. switch (subCmd.Type)
  913. {
  914. case eSubCommandType.Move:
  915. this.CurrentSubCommand = subCmd;
  916. this.Move(subCmd);
  917. break;
  918. case eSubCommandType.Load:
  919. this.CurrentSubCommand = subCmd;
  920. this.LoadCarrier(subCmd);
  921. break;
  922. case eSubCommandType.Unload:
  923. this.CurrentSubCommand = subCmd;
  924. this.UnloadCarrier(subCmd);
  925. break;
  926. case eSubCommandType.Charge:
  927. this.CurrentSubCommand = subCmd;
  928. this.BatteryCharge(subCmd);
  929. break;
  930. case eSubCommandType.Cancel:
  931. this.CurrentSubCommand = subCmd;
  932. CancelCommand(subCmd);
  933. break;
  934. default:
  935. break;
  936. }
  937. }
  938. /// <summary>
  939. /// Scheduler 가 주는 Sub Command 를 이용하여 동작하자.
  940. /// </summary>
  941. public void _ThSubCmdWorker()
  942. {
  943. while (!this.cancel.Canceled)
  944. {
  945. try
  946. {
  947. if (this.refObjects.AutoManager.AutoModeStateProperty != eAutoModeState.Run)
  948. continue;
  949. var subCmd = this.refObjects.Sqlite.SubCmdDAL.GetAll().FirstOrDefault();
  950. if (subCmd == null) continue;
  951. if (!this.refObjects.Sqlite.CommandDAL.GetAll().Any(x => x.CommandID.Equals(subCmd.CmdID)))
  952. {
  953. if (subCmd.CmdType == SubCmd.eCmdType.Auto) //자동 명령중 Main Command 가 없으면 삭제.
  954. {
  955. this.refObjects.Sqlite.SubCmdDAL.Delete(subCmd.ID);
  956. logger.I($"SubCmd Deleted Main Command Not Exist - ID={subCmd.ID}, CommandID={subCmd.CmdID}");
  957. continue;
  958. }
  959. }
  960. switch (subCmd.Type)
  961. {
  962. case eSubCommandType.Move:
  963. this.CurrentSubCommand = subCmd;
  964. this.Move(subCmd);
  965. break;
  966. case eSubCommandType.Load:
  967. this.CurrentSubCommand = subCmd;
  968. this.LoadCarrier(subCmd);
  969. break;
  970. case eSubCommandType.Unload:
  971. this.CurrentSubCommand = subCmd;
  972. this.UnloadCarrier(subCmd);
  973. break;
  974. case eSubCommandType.Charge:
  975. this.CurrentSubCommand = subCmd;
  976. this.BatteryCharge(subCmd);
  977. break;
  978. case eSubCommandType.Cancel:
  979. this.CurrentSubCommand = subCmd;
  980. CancelCommand(subCmd);
  981. break;
  982. default:
  983. break;
  984. }
  985. }
  986. catch (Exception exception)
  987. {
  988. logger.E(exception);
  989. }
  990. finally
  991. {
  992. LockUtils.Wait(500);
  993. }
  994. }
  995. logger.D("Vehicle - _ThSubCmdWorker Dispose");
  996. }
  997. private void CancelCommand(SubCmd subCmd)
  998. {
  999. this.refObjects.Drive.CancelCommand(subCmd);
  1000. var cmd = this.refObjects.Sqlite.CommandDAL.GetById(subCmd.CmdID);
  1001. cmd.State = eCommandState.Complete;
  1002. this.refObjects.Sqlite.CommandDAL.Update(cmd);
  1003. this.refObjects.Sqlite.SubCmdDAL.Clean();
  1004. logger.D($"[Cancel Command] {cmd.TargetID} / {cmd.Type}");
  1005. }
  1006. #endregion
  1007. #region Control Action Method
  1008. public void EStop()
  1009. {
  1010. this.refObjects.Conveyor.OnOffConveyor(false);
  1011. //Clamp EStop
  1012. //this.refObjects.Clamp.ClampEStop();
  1013. this.refObjects.Drive.EStop();
  1014. //this.OccurVehicleAlarm( 23 );
  1015. }
  1016. #region For Moving
  1017. void Move(SubCmd sub)
  1018. {
  1019. logger.I($"[SubCommand Start] - Target : {sub.TargetID} / Type : {sub.Type}");
  1020. //Move 시작 시 충전 중이면 충전 중지 실행.
  1021. int result = 0;
  1022. if (this.IsChargeCylinderForword())
  1023. {
  1024. result = this.StopBatteryCharge();
  1025. if (result != 0)
  1026. {
  1027. OccurVehicleAlarm(result);
  1028. return;
  1029. }
  1030. }
  1031. var iTarget = Convert.ToInt32(sub.TargetID);
  1032. if (iTarget != CurrentTag) //하나의 목표 위치로 왔을때 현재 위치와 동일 하면 이동안함.
  1033. {
  1034. result = this.MoveTo(sub);
  1035. if (result != ConstInt.EXECUTE_SUCCESS)
  1036. {
  1037. OccurVehicleAlarm(result);
  1038. }
  1039. }
  1040. //result = this.MoveTo(sub);
  1041. //if (result != ConstInt.EXECUTE_SUCCESS)
  1042. //{
  1043. // OccurVehicleAlarm(result);
  1044. // return;
  1045. //}
  1046. //2020.07.29. Kang Move 이후 OCS 연결이 없으면 Load/Unlaod 동작을 하지 말자.
  1047. //if ( !this.refObjects.HostManager.IsConnected )
  1048. //{
  1049. // var cmd = this.refObjects.Sqlite.CommandDAL.GetById( sub.CmdID );
  1050. // cmd.IsSecondCommanded = true;
  1051. // cmd.State = eCommandState.Complete;
  1052. // this.refObjects.Sqlite.CommandDAL.Update( cmd );
  1053. //}
  1054. LockUtils.Wait(500);
  1055. this.refObjects.Sqlite.SubCmdDAL.Clean();
  1056. //this.taskMoveCancel.Cancel();
  1057. //this.taskMoveCancel.WaitAll();
  1058. //this.CheckOCSConnectionState();
  1059. logger.I($"[SubCommand Complete] - Target : {sub.TargetID} / Type : {sub.Type}");
  1060. }
  1061. int MoveTo(SubCmd sub)
  1062. {
  1063. PIOClear();
  1064. PIOSensorOff();
  1065. if (this.VehicleStateProperty == eVehicleState.Idle)
  1066. {
  1067. this.OnMoveReady?.Invoke();
  1068. var moveReadyBuzzerTime = Convert.ToInt32(this.refObjects.Sqlite.ConfigDal.GetById(ConstString.BuzzerStartReadyTime).Value);
  1069. Thread.Sleep(moveReadyBuzzerTime);
  1070. this.VehicleStateProperty = eVehicleState.Move;
  1071. this.ObstaclePatternChange(this.refObjects.ZmqManager.SegmentID);
  1072. }
  1073. this.OnMoving?.Invoke();
  1074. //이전에 있던 작업들 종료 및 삭제
  1075. //this.taskMoveCancel.Cancel();
  1076. //this.taskMoveCancel.WaitAll();
  1077. //this.taskMoveCancel.Add(CheckCrossPoint());
  1078. this.VehicleStateProperty = eVehicleState.Move;
  1079. int result = this.refObjects.Drive.MoveToPoint(sub, 100);
  1080. if (result != ConstInt.EXECUTE_SUCCESS)
  1081. return result;
  1082. result = this.Wait4MoveDone();
  1083. if (result != ConstInt.EXECUTE_SUCCESS)
  1084. return result;
  1085. //Drive 에서 정지 확인 후 상태 변경
  1086. if (this.refObjects.Drive.IsDriveStop())
  1087. {
  1088. this.OnMoveFinish?.Invoke();
  1089. this.VehicleStateProperty = eVehicleState.Idle;
  1090. logger.D("Move Finish");
  1091. AddCurrentDriveDistance();
  1092. //2020.08.13. Kang. 주행이 끝나면 장애물감지센서의 패턴을 특정 패턴으로 변경(나성권B 요청)
  1093. var pattern = Convert.ToInt32(this.refObjects.Sqlite.ConfigDal.GetById(ConstString.StandbyPattern).Value);
  1094. this.ChgObstacleDetectPattern(pattern);
  1095. }
  1096. return result;
  1097. }
  1098. int Wait4MoveDone()
  1099. {
  1100. int waitTime = 10 * ConstUtils.ONE_MINUTE; //설정 할 수있게.
  1101. long st = SwUtils.CurrentTimeMillis;
  1102. bool isComingStop = false;
  1103. while (true)
  1104. {
  1105. LockUtils.Wait(50); //2020.07.29. Kang 빠르게 확인 할 필요가 없으므로 5ms -> 50ms 로 변경
  1106. if (SwUtils.Gt(st, waitTime))
  1107. {
  1108. logger.D("Wait4MoveDone Time Over");
  1109. return 39;
  1110. }
  1111. //이동중 메인 명력이 없어진다면 정지 후
  1112. if (null == this.refObjects.Sqlite.CommandDAL.GetById(this.CurrentSubCommand.CmdID))
  1113. {
  1114. logger.D("[Wait Move Done] - 메인 명령 사라짐");
  1115. var cmd = this.refObjects.Sqlite.CommandDAL.GetAll();
  1116. if (cmd == null)
  1117. {
  1118. logger.D("[Wait Move Done] - Main Command not Exist Motion Stop");
  1119. this.refObjects.Drive.Stop();
  1120. return 40;
  1121. }
  1122. else
  1123. {
  1124. logger.D("[Wait Move Done] - Main Command not Exist Motion command, New Command Exist");
  1125. break;
  1126. }
  1127. }
  1128. if (this.refObjects.Drive.IsDriveFault)
  1129. return 45;
  1130. //2020.08.04. Kang. 드라이브에서 알람이 발생하여 Auto Run 상태 가 아니면 기다리지 말고 Thread 종료를 위해.
  1131. if (this.refObjects.AutoManager.AutoModeStateProperty != eAutoModeState.Run)
  1132. {
  1133. if (!isComingStop) //stop 명령을 한번만 내리기 위해.
  1134. {
  1135. isComingStop = true;
  1136. this.refObjects.Drive.Stop();
  1137. }
  1138. }
  1139. if (this.refObjects.Drive.IsDriveStop()) //Move Stop
  1140. break;
  1141. }
  1142. return ConstInt.EXECUTE_SUCCESS;
  1143. }
  1144. Task CheckCrossPoint()
  1145. {
  1146. var task = Task.Run(() =>
  1147. {
  1148. long sTime = SwUtils.CurrentTimeMillis;
  1149. while (!this.taskMoveCancel.Canceled)
  1150. {
  1151. Thread.Sleep(10);
  1152. //ToDo: approach Cross Point Check
  1153. //ToDo: Obstacle Laser Sensor Pattern Change Method 구현 필요.
  1154. }
  1155. });
  1156. return task;
  1157. }
  1158. #endregion
  1159. public bool LoadCarrier(SubCmd sub)
  1160. {
  1161. logger.I($"[SubCommand Start] - Target : {sub.TargetID} / Type : {sub.Type}");
  1162. this.VehicleStateProperty = eVehicleState.Load;
  1163. var targetNo = Convert.ToInt32(sub.TargetID);
  1164. if (this.CurrentTag != targetNo)
  1165. {
  1166. this.OccurVehicleAlarm(20);
  1167. return false; //Alarm
  1168. }
  1169. //var route = sql.RouteDal.GetRoute( sub.TargetID );
  1170. //if ( !CorrectPosition( route, this.CurrentPosition ) )
  1171. //{
  1172. // this.OccurVehicleAlarm( 20 );
  1173. // return false; //Alarm
  1174. //}
  1175. int result = 0;
  1176. result = this.PIOAndLoad(sub.TargetID);
  1177. if (result != 0)
  1178. {
  1179. this.OccurVehicleAlarm(result);
  1180. return false;
  1181. }
  1182. this.PIOSensorOff();
  1183. //Load, Unload 가 끝나면 메인 Command 를 완료 했다고 판단.
  1184. logger.I($"[SubCommand Complete] - Target : {sub.TargetID} / Type : {sub.Type}");
  1185. this.refObjects.Sqlite.SubCmdDAL.Clean();
  1186. var cmd = this.refObjects.Sqlite.CommandDAL.GetById(sub.CmdID);
  1187. if (cmd != null)
  1188. {
  1189. cmd.State = eCommandState.Complete;
  1190. this.refObjects.Sqlite.CommandDAL.Update(cmd);
  1191. }
  1192. //LockUtils.Wait( 1000 );
  1193. //this.OnLoadComplete?.Invoke(); //일찍 주면 다음 명령을 500ms 안에 주는 현상 있음. 그러니까 천천히 주자
  1194. this.VehicleStateProperty = eVehicleState.Idle;
  1195. return true;
  1196. }
  1197. public bool UnloadCarrier(SubCmd sub)
  1198. {
  1199. logger.I($"[SubCommand Start] - Target : {sub.TargetID} / Type : {sub.Type}");
  1200. var targetNo = Convert.ToInt32(sub.TargetID);
  1201. if (this.CurrentTag != targetNo)
  1202. {
  1203. this.OccurVehicleAlarm(21);
  1204. return false; //Alarm
  1205. }
  1206. //var route = sql.RouteDal.GetRoute( sub.TargetID );
  1207. //if ( !CorrectPosition( route, this.CurrentPosition ) )
  1208. //{
  1209. // this.OccurVehicleAlarm( 21 );
  1210. // return false; //Alarm
  1211. //}
  1212. //PIO 내부로 이동.
  1213. //int result = this.clamp.Unlock_Sync();
  1214. //if ( result != 0 )
  1215. //{
  1216. // this.OccurVehicleAlarm( result );
  1217. // return false;
  1218. //}
  1219. this.VehicleStateProperty = eVehicleState.Unload;
  1220. int result = 0;
  1221. result = this.PIOAndUnload(sub.TargetID);
  1222. if (result != 0)
  1223. {
  1224. this.refObjects.IO.OutputOn("OUT_PIO_SENSOR_ONOFF");
  1225. this.OccurVehicleAlarm(result);
  1226. return false;
  1227. }
  1228. this.PIOSensorOff();
  1229. logger.I($"[SubCommand Complete] - Target : {sub.TargetID} / Type : {sub.Type}");
  1230. this.refObjects.Sqlite.SubCmdDAL.Clean();
  1231. var cmd = this.refObjects.Sqlite.CommandDAL.GetById(sub.CmdID);
  1232. if (cmd != null)
  1233. {
  1234. cmd.State = eCommandState.Complete;
  1235. this.refObjects.Sqlite.CommandDAL.Update(cmd);
  1236. }
  1237. //LockUtils.Wait( 1000 );
  1238. //this.OnUnloadComplete?.Invoke(); //일찍 주면 다음 명령을 500ms 안에 주는 현상 있음. 그러니까 천천히 주자
  1239. this.VehicleStateProperty = eVehicleState.Idle;
  1240. return true;
  1241. }
  1242. public void BatteryCharge(SubCmd subCmd)
  1243. {
  1244. int result = 0;
  1245. result = this.StartBatteryCharge();
  1246. if (result != ConstInt.EXECUTE_SUCCESS)
  1247. this.OccurVehicleAlarm(result);
  1248. this.refObjects.Sqlite.SubCmdDAL.Clean();
  1249. var cmd = this.refObjects.Sqlite.CommandDAL.GetById(subCmd.CmdID);
  1250. if (cmd == null)
  1251. return;
  1252. cmd.State = eCommandState.Complete;
  1253. this.refObjects.Sqlite.CommandDAL.Update(cmd);
  1254. }
  1255. /// <summary>
  1256. /// Battery Charge
  1257. /// 충전 시 PIO 를 해야 함.
  1258. /// </summary>
  1259. /// <param name="sub"></param>
  1260. /// <returns></returns>
  1261. int PIOBatteryCharge(SubCmd sub)
  1262. {
  1263. var route = this.refObjects.Sqlite.RouteDal.GetById(sub.TargetID);
  1264. if (!CorrectPosition(route, this.CurrentPosition))
  1265. {
  1266. this.OccurVehicleAlarm(21);
  1267. return 0; //Alarm
  1268. }
  1269. var pioTimeout = CastTo<int>.From<string>(this.refObjects.Sqlite.ConfigDal.GetById(ConstString.PIOTimeOut).Value);
  1270. PIOClear();
  1271. loggerPIO.I($"Start Charge PIO - [{sub.TargetID}]");
  1272. if (!this.refObjects.IO.IsOn("IN_PIO_READY"))
  1273. {
  1274. loggerPIO.E("[Port] - 1 Ready not On");
  1275. this.OccurVehicleAlarm(25);
  1276. return 0;
  1277. }
  1278. this.refObjects.IO.WriteOutputIO("OUT_PIO_READY", true);
  1279. loggerPIO.I("[Vehicle] - 1 Ready On");
  1280. if (!this.refObjects.IO.WaitChangeInputIO(true, pioTimeout, "IN_PIO_RECEIVE_RUN"))
  1281. {
  1282. PIOClear();
  1283. loggerPIO.E("[Port] - 2 Receive CV Run Timeout");
  1284. this.OccurVehicleAlarm(26);
  1285. return 0;
  1286. }
  1287. this.refObjects.IO.WriteOutputIO("OUT_PIO_SENDING_RUN", true);
  1288. loggerPIO.I("[Vehicle] - 2 Send Run On");
  1289. var sTime = SwUtils.CurrentTimeMillis;
  1290. while (true)
  1291. {
  1292. Thread.Sleep(5);
  1293. if (!this.refObjects.IO.IsOn("IN_PIO_READY") || this.refObjects.IO.IsOn("IN_PIO_RECEIVE_RUN"))
  1294. break;
  1295. if (null == this.refObjects.Sqlite.CommandDAL.GetById(this.CurrentSubCommand.CmdID))
  1296. {
  1297. PIOClear();
  1298. logger.D("[Wait Charging] - 메인 명령 사라짐");
  1299. break;
  1300. }
  1301. }
  1302. if (!this.refObjects.IO.WaitChangeInputIO(true, pioTimeout, "IN_PIO_RECEIVE_COMPLITE"))
  1303. {
  1304. PIOClear();
  1305. loggerPIO.E("[Port] - 3 Receive Complete Timeout");
  1306. this.OccurVehicleAlarm(26);
  1307. return 0;
  1308. }
  1309. PIOClear();
  1310. this.refObjects.IO.WriteOutputIO("OUT_PIO_SEND_COMPLITE", true);
  1311. Thread.Sleep(1000);
  1312. this.refObjects.IO.WriteOutputIO("OUT_PIO_SEND_COMPLITE", false);
  1313. return 0;
  1314. }
  1315. public int StartBatteryCharge()
  1316. {
  1317. #if SIMULATION
  1318. this.PIOSensorOn();
  1319. var pioTimeout = CastTo<int>.From<string>( refObjects.Sqlite.ConfigDal.GetById( ConstString.PIOTimeOut ).Value );
  1320. PIOClear();
  1321. loggerPIO.I( $"Start Battery Charge PIO" );
  1322. //if ( !this.iO.WaitChangeInputIO( true, pioTimeout, "IN_PIO_SENDABLE" ) )
  1323. //{
  1324. // PIOClear();
  1325. // loggerPIO.E( "[Port] - 4 Ready Time Out" );
  1326. // return 34;
  1327. //}
  1328. Thread.Sleep( 1000 );
  1329. loggerPIO.E( "[Port] - 4 Ready On" );
  1330. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECEIVABLE", true );
  1331. loggerPIO.I( "[Vehicle] - 4 Receivable" );
  1332. //if ( !this.iO.WaitChangeInputIO( true, 20000, "IN_PIO_SEND_RUN" ) )
  1333. //{
  1334. // PIOClear();
  1335. // loggerPIO.E( "[Port] - 5 Sending Run Time Out" );
  1336. // return 35;
  1337. //}
  1338. Thread.Sleep( 1000 );
  1339. loggerPIO.I( "[Port] - 5 Sending Run On" );
  1340. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECEIVE_RUN", true );
  1341. loggerPIO.I( "[Vehicle] - 5 Receive Run On" );
  1342. this.VehicleStateProperty = eVehicleState.Charge;
  1343. this.OnChargingStart?.Invoke();
  1344. Thread.Sleep( 1000 );
  1345. this.OnCharging?.Invoke();
  1346. return 0;
  1347. #else
  1348. PIOClear();
  1349. this.PIOSensorOn();
  1350. var pioTimeout = Convert.ToInt32(this.refObjects.Sqlite.ConfigDal.GetById(ConstString.PIOTimeOut).Value);
  1351. loggerPIO.I($"Start Battery Charge PIO");
  1352. if (!this.refObjects.IO.WaitChangeInputIO(true, pioTimeout, "IN_PIO_SENDABLE"))
  1353. {
  1354. PIOClear();
  1355. loggerPIO.E("[Port] - 4 Ready Time Out");
  1356. return 34;
  1357. }
  1358. loggerPIO.I("[Port] - 4 Ready On");
  1359. this.refObjects.IO.WriteOutputIO("OUT_PIO_RECEIVABLE", true);
  1360. loggerPIO.I("[Vehicle] - 4 Receivable");
  1361. if (!this.refObjects.IO.WaitChangeInputIO(true, 5000, "IN_PIO_SEND_RUN"))
  1362. {
  1363. PIOClear();
  1364. loggerPIO.E("[Port] - 5 Sending Run Time Out");
  1365. return 35;
  1366. }
  1367. loggerPIO.I("[Port] - 5 Sending Run On");
  1368. long sTime = SwUtils.CurrentTimeMillis;
  1369. while (!IsChargeCylinderForword())
  1370. {
  1371. LockUtils.Wait(10);
  1372. if (SwUtils.Gt(sTime, 5000))
  1373. {
  1374. PIOClear();
  1375. return 45; //Alarm ID 추가.
  1376. }
  1377. }
  1378. this.refObjects.IO.WriteOutputIO("OUT_PIO_RECEIVE_RUN", true);
  1379. loggerPIO.I("[Vehicle] - 5 Receive Run On");
  1380. //this.VehicleStateProperty = eVehicleState.Charge;
  1381. return 0;
  1382. #endif
  1383. }
  1384. public int StopBatteryCharge()
  1385. {
  1386. #if SIMULATION
  1387. loggerPIO.I( $"Stop Battery Charge PIO" );
  1388. var pioTimeout = CastTo<int>.From<string>( refObjects.Sqlite.ConfigDal.GetById( ConstString.PIOTimeOut ).Value );
  1389. this.PIOClear();
  1390. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECIVE_COMPLITE", true );
  1391. loggerPIO.I( "[Vehicle] Receive Complete On" );
  1392. //if ( !this.iO.WaitChangeInputIO( true, 20000, "IN_PIO_SEND_COMPLITE" ) )
  1393. //{
  1394. // this.iO.WriteOutputIO( "OUT_PIO_RECIVE_COMPLITE", false );
  1395. // loggerPIO.E( "[Port] IN_PIO_SEND_COMPLITE On Time Out" );
  1396. // return 36;
  1397. //}
  1398. Thread.Sleep( 1000 );
  1399. loggerPIO.I( "[Port] Send Complete On" );
  1400. Thread.Sleep( 1000 );
  1401. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECIVE_COMPLITE", false );
  1402. this.OnChargingFull?.Invoke();
  1403. this.PIOSensorOff();
  1404. this.VehicleStateProperty = eVehicleState.Idle;
  1405. return 0;
  1406. #else
  1407. loggerPIO.I($"Stop Battery Charge PIO");
  1408. var pioTimeout = Convert.ToInt32(this.refObjects.Sqlite.ConfigDal.GetById(ConstString.PIOTimeOut).Value);
  1409. this.PIOClear();
  1410. this.refObjects.IO.WriteOutputIO("OUT_PIO_RECIVE_COMPLITE", true);
  1411. loggerPIO.I("[Vehicle] Receive Complete On");
  1412. if (!this.refObjects.IO.WaitChangeInputIO(true, 5000, "IN_PIO_SEND_COMPLITE"))
  1413. {
  1414. this.refObjects.IO.WriteOutputIO("OUT_PIO_RECIVE_COMPLITE", false);
  1415. loggerPIO.E("[Port] IN_PIO_SEND_COMPLITE On Time Out");
  1416. return 36;
  1417. }
  1418. loggerPIO.I("[Port] Send Complete On");
  1419. long sTime = SwUtils.CurrentTimeMillis;
  1420. while (IsChargeCylinderForword())
  1421. {
  1422. LockUtils.Wait(10);
  1423. if (SwUtils.Gt(sTime, 5000))
  1424. {
  1425. PIOClear();
  1426. return 44; //Alarm ID 추가.
  1427. }
  1428. }
  1429. Thread.Sleep(1000);
  1430. this.refObjects.IO.WriteOutputIO("OUT_PIO_RECIVE_COMPLITE", false);
  1431. this.PIOSensorOff();
  1432. //this.OnChargingFull?.Invoke();
  1433. //this.VehicleStateProperty = eVehicleState.Idle;
  1434. return 0;
  1435. #endif
  1436. }
  1437. #endregion
  1438. #region Check Method
  1439. bool CheckObstacle()
  1440. {
  1441. if (this.refObjects.IO.IsOff("IN_OBSTRUCTION_DETECT_SAFETY") || this.refObjects.IO.IsOff("IN_OBSTRUCTION_DETECT_ERROR"))
  1442. {
  1443. this.ObstacleStateProperty = eObstacleState.Abnormal;
  1444. //this.OccurVehicleAlarm( 42 );
  1445. }
  1446. else if (this.refObjects.IO.IsOff("IN_OBSTRUCTION_DETECT_STOP"))
  1447. {
  1448. this.ObstacleStateProperty = eObstacleState.Blocked;
  1449. }
  1450. else if (this.refObjects.IO.IsOff("IN_OBSTRUCTION_DETECT_SLOW"))
  1451. {
  1452. this.ObstacleStateProperty = eObstacleState.Decelerate;
  1453. }
  1454. else
  1455. {
  1456. this.ObstacleStateProperty = eObstacleState.Normal;
  1457. }
  1458. this.refObjects.Drive.SetObstacleState(this.ObstacleStateProperty);
  1459. return false;
  1460. }
  1461. #endregion
  1462. #region Mechanical Method
  1463. #region IO
  1464. /// <summary>
  1465. /// Out Put 을 On 이면 Sensor Off
  1466. /// </summary>
  1467. public void PIOSensorOn()
  1468. {
  1469. this.refObjects.IO.OutputOff( "OUT_PIO_SENSOR_ONOFF" );
  1470. LockUtils.Wait(500); //Sensor 가 커지고 안정 되는것을 기다리자.
  1471. }
  1472. /// <summary>
  1473. /// Out Put 을 Off 이면 Sensor On
  1474. /// </summary>
  1475. public void PIOSensorOff() => this.refObjects.IO.OutputOn( "OUT_PIO_SENSOR_ONOFF" );
  1476. public bool IsChargeCylinderForword() => this.refObjects.IO.IsOn( "IN_CHARGE_CYL_FWD_DETECT" );
  1477. /// <summary>
  1478. /// 충전 중일때 MTL PIO Bit 가 살아 있다
  1479. /// 이걸 보고 판단 하자. 충전 단자 전진 상태임.
  1480. /// </summary>
  1481. /// <returns></returns>
  1482. public bool CheckBatteryCharge()
  1483. {
  1484. if ( this.VehicleStateProperty == eVehicleState.Charge || IsChargeCylinderForword() )
  1485. return true;
  1486. return false;
  1487. }
  1488. #endregion
  1489. #region Conveyor
  1490. public void ConveyorOff() => this.refObjects.Conveyor.OnOffConveyor( false );
  1491. public int ConveyorLoad() => this.refObjects.Conveyor.ConveyorLoad();
  1492. public int ConveyorUnload() => this.refObjects.Conveyor.ConveyorUnload();
  1493. public void PIOClear()
  1494. {
  1495. string[] pio = { "OUT_PIO_READY", "OUT_PIO_SENDING_RUN", "OUT_PIO_SEND_COMPLITE", "OUT_PIO_RECEIVABLE", "OUT_PIO_RECEIVE_RUN", "OUT_PIO_RECIVE_COMPLITE", "OUT_PIO_INTERLOCK" };
  1496. pio.FwEach( x => { this.refObjects.IO.OutputOff( x ); } );
  1497. }
  1498. public int PIOAndLoad( string targetName )
  1499. {
  1500. #if SIMULATION
  1501. PIOClear();
  1502. loggerPIO.I( $"Start Load PIO - [{targetName}]" );
  1503. this.OnPIOStart?.Invoke( true );
  1504. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECEIVE_RUN", true );
  1505. loggerPIO.I( "[Vehicle] - 4 Receive Run On" );
  1506. Thread.Sleep( 1000 );//상대 IO 기다린다 생각.
  1507. loggerPIO.E( "[Port] - 4 Ready On" );
  1508. //Conveyor Start
  1509. loggerPIO.I( "[Vehicle] - Conveyor Run" );
  1510. this.OnConveyorStart?.Invoke( true );
  1511. Thread.Sleep( 10000 );//Conveyor 구동
  1512. this.OnCarrierDetected?.Invoke( true );
  1513. PIOClear();
  1514. Thread.Sleep( 1000 );
  1515. this.OnConveyorStop?.Invoke( true );
  1516. Thread.Sleep( 1000 );
  1517. this.OnLoadComplete?.Invoke();
  1518. #else
  1519. loggerPIO.I( $"Start Load PIO - [{targetName}]" );
  1520. PIOClear();
  1521. this.PIOSensorOn();
  1522. int result = 0;
  1523. var pioTimeout = Convert.ToInt32( this.refObjects.Sqlite.ConfigDal.GetById( ConstString.PIOTimeOut ).Value );
  1524. result = this.refObjects.Clamp.Unlock_Sync();
  1525. if ( result != 0 )
  1526. return result;
  1527. //if (this.refObjects.Conveyor.IsInverterError())
  1528. // return 16;
  1529. if ( this.refObjects.Conveyor.IsDetectedCenter() )
  1530. {
  1531. this.OnFailReport?.Invoke( eFailCode.Load_VehicleHasCarrier );
  1532. return 0;
  1533. }
  1534. if ( !this.refObjects.IO.WaitChangeInputIO( true, 40 * ConstUtils.ONE_SECOND, "IN_LIFTER_POSITION_DETECT" ) )
  1535. {
  1536. loggerPIO.E( "[Port] - Lift Position Check Error" );
  1537. TimerUtils.Once( 1000, this.OnFailReport, eFailCode.LoadPIOInterlockTimeout );
  1538. //this.OnFailReport?.Invoke( eFailCode.LoadPIOInterlockTimeout );
  1539. return 0; //14
  1540. }
  1541. //if ( !this.refObjects.Conveyor.IsLifterPositinCheck() )
  1542. //{
  1543. // loggerPIO.E( "[Port] - Lift Position Check Error" );
  1544. // this.OnFailReport?.Invoke( eFailCode.LoadPIOInterlockTimeout );
  1545. // return 0; //14
  1546. //}
  1547. //Todo: Sensor Setting 이 후 주석 풀기.
  1548. //if ( !this.IsLifterDuplication() )
  1549. //{
  1550. // this.OnFailReport?.Invoke( eFailCode.Load_PortHasNotCarrier );
  1551. // return 0;
  1552. //}
  1553. this.OnPIOStart?.Invoke( true );
  1554. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECEIVABLE", true );
  1555. loggerPIO.I( "[Vehicle] - 4 Receivable" );
  1556. if ( !this.refObjects.IO.WaitChangeInputIO( true, 20 * ConstUtils.ONE_SECOND, "IN_PIO_SENDABLE" ) )
  1557. {
  1558. PIOClear();
  1559. loggerPIO.E( "[Port] - 4 Ready Time Out" );
  1560. TimerUtils.Once( 1000, this.OnFailReport, eFailCode.LoadPIOInterlockTimeout );
  1561. //this.OnFailReport?.Invoke( eFailCode.LoadPIOInterlockTimeout );
  1562. return 0;
  1563. }
  1564. loggerPIO.I( "[Port] - 4 Ready On" );
  1565. //this.conveyor.SetConveyorSpeed( true );
  1566. result = this.refObjects.Conveyor.OnOffConveyor( true, true );
  1567. if ( result != ConstInt.EXECUTE_SUCCESS )
  1568. return result;
  1569. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECEIVE_RUN", true, 1000 ); //1Sec 이후 On
  1570. loggerPIO.I( "[Vehicle] - Conveyor Run" );
  1571. this.OnConveyorStart?.Invoke( true );
  1572. if ( !this.refObjects.IO.WaitChangeInputIO( true, pioTimeout, "IN_PIO_SEND_RUN" ) )
  1573. {
  1574. this.refObjects.Conveyor.OnOffConveyor( false, true );
  1575. PIOClear();
  1576. loggerPIO.E( "[Port] - 5 Sending Run Time Out" );
  1577. TimerUtils.Once( 1000, this.OnFailReport, eFailCode.LoadPIOInterlockTimeout );
  1578. //this.OnFailReport?.Invoke( eFailCode.LoadPIOInterlockTimeout );
  1579. return 0;
  1580. }
  1581. loggerPIO.I( "[Port] - 5 Sending Run On" );
  1582. bool isStartDetected = false;
  1583. var sTime = SwUtils.CurrentTimeMillis;
  1584. while ( true )
  1585. {
  1586. LockUtils.Wait( 50 );
  1587. if ( SwUtils.Gt( sTime, 30 * ConstUtils.ONE_SECOND ) )
  1588. {
  1589. this.refObjects.IO.WriteOutputIO( "OUT_PIO_INTERLOCK", true );
  1590. this.refObjects.Conveyor.OnOffConveyor( false, true );
  1591. loggerPIO.E( "[Vehicle] Conveyor Wait Time Out" );
  1592. //TimerUtils.Once( 1000, this.OnFailReport, eFailCode.LoadPIOInterlockTimeout );
  1593. return 10; //Conveyor Moving Timeout
  1594. }
  1595. //if ( this.refObjects.Conveyor.IsDetectedLoadStart() && !isStartDetected )
  1596. // isStartDetected = true;
  1597. //if ( !this.refObjects.Conveyor.IsDetectedLoadStart() && isStartDetected ) { }
  1598. //this.conveyor.SetConveyorSpeed( false );
  1599. if ( this.refObjects.Conveyor.IsDetectedLoadStop() ) break;
  1600. if ( this.refObjects.Conveyor.IsPIOInterLockOn() )
  1601. {
  1602. this.refObjects.Conveyor.OnOffConveyor( false ); //Stop
  1603. loggerPIO.E( "[Port] PIO InterLock On " );
  1604. return 19; //
  1605. }
  1606. }
  1607. if ( this.refObjects.Conveyor.IsDetectedCenter() )
  1608. this.OnCarrierDetected?.Invoke( true );
  1609. this.refObjects.Conveyor.OnOffConveyor( false ); //Stop
  1610. PIOClear();
  1611. this.OnConveyorStop?.Invoke( true );
  1612. loggerPIO.I( "[Vehicle] Conveyor Stop" );
  1613. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECIVE_COMPLITE", true );
  1614. loggerPIO.I( "[Vehicle] Receive Complete On" );
  1615. if ( !this.refObjects.IO.WaitChangeInputIO( true, pioTimeout, "IN_PIO_SEND_COMPLITE" ) )
  1616. {
  1617. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECIVE_COMPLITE", false );
  1618. loggerPIO.E( "[Port] IN_PIO_SEND_COMPLITE On Time Out" );
  1619. }
  1620. loggerPIO.I( "[Port] Send Complete On" );
  1621. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECIVE_COMPLITE", false, 1000 );
  1622. loggerPIO.I( $"End Load PIO - [{targetName}]" );
  1623. result = this.refObjects.Clamp.Lock_Sync();
  1624. if ( result != 0 )
  1625. {
  1626. this.OccurVehicleAlarm( result );
  1627. return result;
  1628. }
  1629. #endif
  1630. //2020.08.18.Kang. Conveyor 작동후 OCS 연결이 끊어지면 알람 처리
  1631. //if (!this.refObjects.HostManager.IsConnected)
  1632. // return 43;
  1633. TimerUtils.Once(2000, this.OnLoadComplete);
  1634. return 0;
  1635. }
  1636. public int PIOAndUnload( string targetName )
  1637. {
  1638. #if SIMULATION
  1639. PIOClear();
  1640. loggerPIO.I( $"Start Unload PIO - [{targetName}]" );
  1641. this.OnPIOStart?.Invoke( false );
  1642. Thread.Sleep( 1000 );
  1643. this.refObjects.IO.WriteOutputIO( "OUT_PIO_READY", true );
  1644. loggerPIO.I( "[Vehicle] - 1 Ready On" );
  1645. Thread.Sleep( 1000 );
  1646. this.OnConveyorStart?.Invoke( false );
  1647. Thread.Sleep( 10000 );
  1648. this.refObjects.Conveyor.OnOffConveyor( false ); //Stop
  1649. this.OnConveyorStop?.Invoke( false );
  1650. PIOClear();
  1651. Thread.Sleep( 1000 );
  1652. this.OnUnloadComplete?.Invoke();
  1653. #else
  1654. loggerPIO.I( $"Start Unload PIO -[{targetName}]" );
  1655. PIOClear();
  1656. this.PIOSensorOn();
  1657. int result = 0;
  1658. var pioTimeout = Convert.ToInt32( this.refObjects.Sqlite.ConfigDal.GetById( ConstString.PIOTimeOut ).Value );
  1659. if ( this.refObjects.Conveyor.IsInverterError() )
  1660. return 16;
  1661. //if ( this.refObjects.Conveyor.IsLifterDuplication() )
  1662. //{
  1663. // this.OnFailReport?.Invoke( eFailCode.Unload_PortHasCarrier );
  1664. // return 0;
  1665. //}
  1666. if ( !this.refObjects.Conveyor.IsDetectedCenter() )
  1667. {
  1668. TimerUtils.Once( 1000, this.OnFailReport, eFailCode.Unload_VehicleHasNotCarrier );
  1669. //this.OnFailReport?.Invoke( eFailCode.Unload_VehicleHasNotCarrier );
  1670. return 0;
  1671. }
  1672. if ( !this.refObjects.IO.WaitChangeInputIO( true, 40 * ConstUtils.ONE_SECOND, "IN_LIFTER_POSITION_DETECT" ) )
  1673. {
  1674. loggerPIO.E( "[Port] - Lift Position Check Error" );
  1675. TimerUtils.Once( 1000, this.OnFailReport, eFailCode.UnlaodPIOInterlockTimeout );
  1676. //this.OnFailReport?.Invoke( eFailCode.UnlaodPIOInterlockTimeout );
  1677. return 0;
  1678. }
  1679. //if ( !this.refObjects.Conveyor.IsLifterPositinCheck() )
  1680. //{
  1681. // loggerPIO.E( "[Port] - Lift Position Check Error" );
  1682. // this.OnFailReport?.Invoke( eFailCode.UnlaodPIOInterlockTimeout ); //Unloading Fail Code 를 Time Out 으로 하자. 정의 된 코드 없음.
  1683. // return 0; //13 -> Alarm Code 보고 없음. 경알람 상태 이므로 한번 보고 후 정상 동작
  1684. //}
  1685. PIOClear();
  1686. this.OnPIOStart?.Invoke( false );
  1687. if ( !this.refObjects.IO.WaitChangeInputIO( true, 20 * ConstUtils.ONE_SECOND, "IN_PIO_READY" ) )
  1688. {
  1689. loggerPIO.E( "[Port] - 1 Ready not On" );
  1690. TimerUtils.Once( 1000, this.OnFailReport, eFailCode.UnlaodPIOInterlockTimeout );
  1691. //this.OnFailReport?.Invoke( eFailCode.UnlaodPIOInterlockTimeout );
  1692. return 0;
  1693. }
  1694. this.refObjects.IO.WriteOutputIO( "OUT_PIO_READY", true );
  1695. loggerPIO.I( "[Vehicle] - 1 Ready On" );
  1696. if ( !this.refObjects.IO.WaitChangeInputIO( true, 10 * ConstUtils.ONE_SECOND, "IN_PIO_RECEIVE_RUN" ) )
  1697. {
  1698. PIOClear();
  1699. loggerPIO.E( "[Port] - 2 Receive CV Run Timeout" );
  1700. TimerUtils.Once( 1000, this.OnFailReport, eFailCode.UnlaodPIOInterlockTimeout );
  1701. //this.OnFailReport?.Invoke( eFailCode.UnlaodPIOInterlockTimeout );
  1702. return 0;
  1703. }
  1704. loggerPIO.I( "[Port] - 2 Receive CV Run On" );
  1705. result = this.refObjects.Clamp.Unlock_Sync();
  1706. if ( result != 0 )
  1707. return result;
  1708. this.refObjects.IO.WriteOutputIO( "OUT_PIO_SENDING_RUN", true );
  1709. loggerPIO.I( "[Vehicle] - 2 Send Run On" );
  1710. //this.conveyor.SetConveyorSpeed( true );
  1711. this.refObjects.Conveyor.OnOffConveyor( true );
  1712. this.OnConveyorStart?.Invoke( false );
  1713. var sTime = SwUtils.CurrentTimeMillis;
  1714. while ( true )
  1715. {
  1716. LockUtils.Wait(50);
  1717. if ( SwUtils.Gt( sTime, 30 * ConstUtils.ONE_SECOND ) )
  1718. {
  1719. this.refObjects.IO.WriteOutputIO( "OUT_PIO_INTERLOCK", true );
  1720. this.refObjects.Conveyor.OnOffConveyor( false, true );
  1721. loggerPIO.E( "[Port] Conveyor Wait Time Out" );
  1722. //TimerUtils.Once( 1000, this.OnFailReport, eFailCode.UnlaodPIOInterlockTimeout );
  1723. return 12; // 2020.07.18 Kang. 센서에 걸리지 않아도 중간에 걸려 있을 가능성 때문에 알람 처리.
  1724. //if ( this.refObjects.Conveyor.IsDetectedLoadStart() || this.refObjects.Conveyor.IsDetectedCenter() ) //중간에 걸려 있다고 생각해서 알람 처리.
  1725. // return 12; //Conveyor Moving Timeout
  1726. //else
  1727. //{
  1728. // if ( this.refObjects.Conveyor.IsDetectedCenter() ) //Time Out 후 아직 Tray 가 있으면 다시 Clamp Lock.
  1729. // {
  1730. // result = this.refObjects.Clamp.Unlock_Sync();
  1731. // if ( result != 0 )
  1732. // {
  1733. // this.OccurVehicleAlarm( result );
  1734. // return result;
  1735. // }
  1736. // }
  1737. // else
  1738. // return 0;
  1739. //}
  1740. }
  1741. if ( !this.refObjects.Conveyor.IsDetectedLoadStart() && !this.refObjects.Conveyor.IsDetectedCenter() )
  1742. {
  1743. if ( this.refObjects.IO.IsOn( "IN_PIO_RECEIVE_COMPLITE" ) )
  1744. {
  1745. loggerPIO.I( "[Port] - 3 Receive Complete On" );
  1746. break;
  1747. }
  1748. }
  1749. if ( this.refObjects.Conveyor.IsPIOInterLockOn() )
  1750. {
  1751. this.refObjects.Conveyor.OnOffConveyor( false ); //Stop
  1752. loggerPIO.E( "[Port] PIO InterLock On " );
  1753. return 19; //
  1754. }
  1755. }
  1756. if ( !this.refObjects.Conveyor.IsDetectedCenter() )
  1757. this.OnCarrierDetected?.Invoke( false );
  1758. this.refObjects.Conveyor.OnOffConveyor( false ); //Stop
  1759. this.OnConveyorStop?.Invoke( false );
  1760. PIOClear();
  1761. this.refObjects.IO.WriteOutputIO( "OUT_PIO_SEND_COMPLITE", true );
  1762. Thread.Sleep( 1000 );
  1763. this.refObjects.IO.WriteOutputIO( "OUT_PIO_SEND_COMPLITE", false );
  1764. loggerPIO.I( "[Vehicle] - 3 Send Complete OnOff" );
  1765. #endif
  1766. loggerPIO.I( $"End Unload PIO - [{targetName}]" );
  1767. //2020.08.18.Kang. Conveyor 작동후 OCS 연결이 끊어지면 알람 처리
  1768. //if (!this.refObjects.HostManager.IsConnected)
  1769. // return 43;
  1770. TimerUtils.Once(2000, this.OnUnloadComplete);
  1771. return 0;
  1772. }
  1773. public int ManualPIOAndLoad()
  1774. {
  1775. loggerPIO.I( $"Start Load PIO - [{this.CurrentTag}]" );
  1776. PIOClear();
  1777. this.PIOSensorOn();
  1778. int result = ConstInt.EXECUTE_SUCCESS;
  1779. var pioTimeout = Convert.ToInt32( this.refObjects.Sqlite.ConfigDal.GetById( ConstString.PIOTimeOut ).Value );
  1780. result = this.refObjects.Clamp.Unlock_Sync();
  1781. if ( result != 0 )
  1782. {
  1783. this.OccurVehicleAlarm( result );
  1784. return result;
  1785. }
  1786. if ( this.refObjects.Conveyor.IsInverterError() )
  1787. return 16;
  1788. if ( this.refObjects.Conveyor.IsDetectedCenter() )
  1789. return 9;
  1790. if ( !this.refObjects.Conveyor.IsLifterPositinCheck() )
  1791. {
  1792. loggerPIO.E( "[Port] - Lift Position Check Error" );
  1793. return 14;
  1794. }
  1795. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECEIVABLE", true );
  1796. loggerPIO.I( "[Vehicle] - 4 Receivable" );
  1797. if ( !this.refObjects.IO.WaitChangeInputIO( true, 1 * ConstUtils.ONE_SECOND, "IN_PIO_SENDABLE" ) )
  1798. {
  1799. PIOClear();
  1800. loggerPIO.E( "[Port] - 4 Ready Time Out" );
  1801. TimerUtils.Once( 1000, this.OnFailReport, eFailCode.LoadPIOInterlockTimeout );
  1802. //this.OnFailReport?.Invoke( eFailCode.LoadPIOInterlockTimeout );
  1803. return 18;
  1804. }
  1805. loggerPIO.I( "[Port] - 4 Ready On" );
  1806. //this.conveyor.SetConveyorSpeed( true );
  1807. this.refObjects.Conveyor.OnOffConveyor( true, true );
  1808. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECEIVE_RUN", true, 1000 ); //1Sec 이후 On
  1809. loggerPIO.I( "[Vehicle] - Conveyor Run" );
  1810. this.OnConveyorStart?.Invoke( true );
  1811. if ( !this.refObjects.IO.WaitChangeInputIO( true, pioTimeout, "IN_PIO_SEND_RUN" ) )
  1812. {
  1813. this.refObjects.Conveyor.OnOffConveyor( false, true );
  1814. PIOClear();
  1815. loggerPIO.E( "[Port] - 5 Sending Run Time Out" );
  1816. return 18;
  1817. }
  1818. loggerPIO.I( "[Port] - 5 Sending Run On" );
  1819. bool isStartDetected = false;
  1820. var sTime = SwUtils.CurrentTimeMillis;
  1821. while ( true )
  1822. {
  1823. LockUtils.Wait( 10 );
  1824. if ( SwUtils.Gt( sTime, 20 * ConstUtils.ONE_SECOND ) )
  1825. {
  1826. PIOClear();
  1827. this.refObjects.Conveyor.OnOffConveyor( false, true );
  1828. loggerPIO.E( "[Vehicle] Conveyor Wait Time Out" );
  1829. return 10; //Conveyor Time Out
  1830. }
  1831. if ( this.refObjects.Conveyor.IsDetectedLoadStart() && !isStartDetected )
  1832. isStartDetected = true;
  1833. if ( !this.refObjects.Conveyor.IsDetectedLoadStart() && isStartDetected ) { }
  1834. if ( this.refObjects.Conveyor.IsDetectedLoadStop() ) break;
  1835. if ( this.refObjects.Conveyor.IsPIOInterLockOn() )
  1836. {
  1837. PIOClear();
  1838. this.refObjects.Conveyor.OnOffConveyor( false ); //Stop
  1839. loggerPIO.E( "[Port] PIO InterLock On " );
  1840. return 19; //
  1841. }
  1842. }
  1843. this.refObjects.Conveyor.OnOffConveyor( false ); //Stop
  1844. PIOClear();
  1845. loggerPIO.I( "[Vehicle] Conveyor Stop" );
  1846. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECIVE_COMPLITE", true );
  1847. loggerPIO.I( "[Vehicle] Receive Complete On" );
  1848. if ( !this.refObjects.IO.WaitChangeInputIO( true, pioTimeout, "IN_PIO_SEND_COMPLITE" ) )
  1849. {
  1850. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECIVE_COMPLITE", false );
  1851. loggerPIO.E( "[Port] IN_PIO_SEND_COMPLITE On Time Out" );
  1852. }
  1853. loggerPIO.I( "[Port] Send Complete On" );
  1854. this.refObjects.IO.WriteOutputIO( "OUT_PIO_RECIVE_COMPLITE", false, 1000 );
  1855. loggerPIO.I( $"End Load PIO - [{this.CurrentTag}]" );
  1856. result = this.refObjects.Clamp.Lock_Sync();
  1857. if ( result != 0 )
  1858. return result;
  1859. return result;
  1860. }
  1861. public int ManualPIOAndUnlaod()
  1862. {
  1863. loggerPIO.I( $"Start Unload PIO -[{this.CurrentTag}]" );
  1864. PIOClear();
  1865. this.PIOSensorOn();
  1866. int result = 0;
  1867. var pioTimeout = Convert.ToInt32( this.refObjects.Sqlite.ConfigDal.GetById( ConstString.PIOTimeOut ).Value );
  1868. if ( this.refObjects.Conveyor.IsInverterError() )
  1869. return 16;
  1870. if ( !this.refObjects.Conveyor.IsDetectedCenter() )
  1871. {
  1872. return 11;
  1873. }
  1874. if ( !this.refObjects.Conveyor.IsLifterPositinCheck() )
  1875. {
  1876. loggerPIO.E( "[Port] - Lift Position Check Error" );
  1877. return 13;
  1878. }
  1879. if ( !this.refObjects.IO.WaitChangeInputIO( true, 1 * ConstUtils.ONE_SECOND, "IN_PIO_READY" ) )
  1880. {
  1881. loggerPIO.E( "[Port] - 1 Ready not On" );
  1882. return 17;
  1883. }
  1884. this.refObjects.IO.WriteOutputIO( "OUT_PIO_READY", true );
  1885. loggerPIO.I( "[Vehicle] - 1 Ready On" );
  1886. if ( !this.refObjects.IO.WaitChangeInputIO( true, 3 * ConstUtils.ONE_SECOND, "IN_PIO_RECEIVE_RUN" ) )
  1887. {
  1888. PIOClear();
  1889. loggerPIO.E( "[Port] - 2 Receive CV Run Timeout" );
  1890. return 0;
  1891. }
  1892. loggerPIO.I( "[Port] - 2 Receive CV Run On" );
  1893. result = this.refObjects.Clamp.Unlock_Sync();
  1894. if ( result != 0 )
  1895. return result;
  1896. this.refObjects.IO.WriteOutputIO( "OUT_PIO_SENDING_RUN", true );
  1897. loggerPIO.I( "[Vehicle] - 2 Send Run On" );
  1898. this.refObjects.Conveyor.OnOffConveyor( true );
  1899. var sTime = SwUtils.CurrentTimeMillis;
  1900. while ( true )
  1901. {
  1902. if ( SwUtils.Gt( sTime, 20 * ConstUtils.ONE_SECOND ) )
  1903. {
  1904. PIOClear();
  1905. this.refObjects.Conveyor.OnOffConveyor( false, true );
  1906. loggerPIO.E( "[Port] Conveyor Wait Time Out" );
  1907. return 12; //Conveyor Moving Timeout
  1908. }
  1909. if ( !this.refObjects.Conveyor.IsDetectedLoadStart() && !this.refObjects.Conveyor.IsDetectedCenter() )
  1910. {
  1911. if ( this.refObjects.IO.IsOn( "IN_PIO_RECEIVE_COMPLITE" ) )
  1912. {
  1913. loggerPIO.I( "[Port] - 3 Receive Complete On" );
  1914. break;
  1915. }
  1916. }
  1917. }
  1918. this.refObjects.Conveyor.OnOffConveyor( false ); //Stop
  1919. PIOClear();
  1920. this.refObjects.IO.WriteOutputIO( "OUT_PIO_SEND_COMPLITE", true );
  1921. LockUtils.Wait( 1000 );
  1922. this.refObjects.IO.WriteOutputIO( "OUT_PIO_SEND_COMPLITE", false );
  1923. loggerPIO.I( "[Vehicle] - 3 Send Complete OnOff" );
  1924. loggerPIO.I( $"End Unload PIO - [{CurrentTag}]" );
  1925. return result;
  1926. }
  1927. #endregion
  1928. #endregion
  1929. #region Help Method
  1930. void AddCurrentDriveDistance()
  1931. {
  1932. var cD = refObjects.ZmqManager.GetDriveDistance();
  1933. if (cD == 0)
  1934. return;
  1935. var info = refObjects.Sqlite.VehicleInfoDAL.GetAll().FirstOrDefault();
  1936. if ((info.TripDistance += (cD * 0.001)) >= double.MaxValue)
  1937. {
  1938. info.TripDistance = 0d;
  1939. logger.E("Trip Distance Limit Over - Zero Set!");
  1940. }
  1941. if ((info.CumulativeDistance += (cD * 0.001)) >= double.MaxValue)
  1942. {
  1943. info.CumulativeDistance = 0d;
  1944. logger.E("Cumulative Distance Limit Over - Zero Set!");
  1945. }
  1946. refObjects.Sqlite.VehicleInfoDAL.Update(info);
  1947. }
  1948. Task taskHostConnection = null;
  1949. /// <summary>
  1950. /// Move 이후 OCS 연결을 확인 후 Disconnected 상태이면 5분 상태 확인 후 Alarm 발생.
  1951. /// </summary>
  1952. void CheckOCSConnectionState()
  1953. {
  1954. if ( this.taskHostConnection != null )
  1955. return;
  1956. if ( refObjects.HostManager.IsConnected )
  1957. return;
  1958. this.taskHostConnection = new Task( () =>
  1959. {
  1960. long sTime = SwUtils.CurrentTimeMillis;
  1961. while ( !refObjects.HostManager.IsConnected )
  1962. {
  1963. LockUtils.Wait( 100 );
  1964. if ( SwUtils.Gt( sTime, 30 * ConstUtils.ONE_SECOND ) )
  1965. {
  1966. OccurVehicleAlarm( 43 );
  1967. break;
  1968. }
  1969. }
  1970. } );
  1971. this.taskHostConnection = null;
  1972. }
  1973. void BatteryChargeStateLED( double percent )
  1974. {
  1975. if ( percent > 60 )
  1976. {
  1977. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_00", true );
  1978. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_01", true );
  1979. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_02", true );
  1980. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_03", true );
  1981. }
  1982. else if ( percent > 45 )
  1983. {
  1984. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_00", true );
  1985. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_01", true );
  1986. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_02", true );
  1987. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_03", false );
  1988. }
  1989. else if ( percent > 30 )
  1990. {
  1991. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_00", true );
  1992. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_01", true );
  1993. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_02", false );
  1994. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_03", false );
  1995. }
  1996. else
  1997. {
  1998. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_00", true );
  1999. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_01", false );
  2000. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_02", false );
  2001. this.refObjects.IO.WriteOutputIO( "OUT_BATTERY_LED_03", false );
  2002. }
  2003. }
  2004. /// <summary>
  2005. /// 현재 Point 가 변경될때 장애물 감지 센서의 패턴을 변경한다.
  2006. /// </summary>
  2007. /// <param name="v"></param>
  2008. private void ObstaclePatternChange( int v )
  2009. {
  2010. //2020.08.13. Kang 주행 중이 아닐때는 패턴 변경 안함.(나성권B 요청)
  2011. if (this.refObjects.Drive.IsDriveStop())
  2012. return;
  2013. if ( !RouteManager.Instance.Segments.Any( s => s.ID == v ) )
  2014. {
  2015. logger.E( "Current Tag Not Exist RouteManager Point List" );
  2016. return;
  2017. }
  2018. var obstacle = RouteManager.Instance.Obstacles.Where( o => o.segmentID == v ).Single();
  2019. this.ChgObstacleDetectPattern( obstacle.fieldset );
  2020. logger.D( $"[Obstacle Pattern Chg] - {obstacle.fieldset}" );
  2021. }
  2022. /// <summary>
  2023. /// 현재 좌표 값이 등록된 Route 에 맞는 위치인지 확인한다.
  2024. /// 판단 기준은 Route 에 Tolerance 범위를 사용.
  2025. /// </summary>
  2026. /// <param name="route"></param>
  2027. /// <param name="currentPosition"></param>
  2028. /// <returns></returns>
  2029. bool CorrectPosition( Route route, double currentPosition )
  2030. {
  2031. var rScale = route.ScaleValue;
  2032. var rTolerance = route.ScaleTolerance;
  2033. var result = currentPosition - rScale;
  2034. if ( rTolerance < Math.Abs( result ) )
  2035. return false;
  2036. return true;
  2037. }
  2038. /// <summary>
  2039. /// if no is zero, Laser Off
  2040. /// bit Off, On, On, On,On Area1
  2041. /// </summary>
  2042. /// <param name="no"> 0 == Off Laser</param>
  2043. /// <returns></returns>
  2044. public bool ChgObstacleDetectPattern( int no )
  2045. {
  2046. var bitArray = BitUtils.ChgBitArray( no );
  2047. int bitIndex = 0;
  2048. this.obstacleBitList.ForEach( b =>
  2049. {
  2050. if ( bitArray[bitIndex] )
  2051. this.refObjects.IO.OutputOff( b );
  2052. else
  2053. this.refObjects.IO.OutputOn( b );
  2054. bitIndex++;
  2055. } );
  2056. ObstaclePattern = no;
  2057. return true;
  2058. }
  2059. public int GetObstacleDetectPattern()
  2060. {
  2061. int bitIndex = 0;
  2062. BitArray bitArray = new BitArray( this.obstacleBitList.Count );
  2063. this.obstacleBitList.ForEach( b =>
  2064. {
  2065. if ( this.refObjects.IO.IsOn( b, false ) )
  2066. bitArray.Set( bitIndex, false );
  2067. else
  2068. bitArray.Set( bitIndex, true );
  2069. bitIndex++;
  2070. } );
  2071. return BitUtils.ChgInt32( bitArray );
  2072. }
  2073. /// <summary>
  2074. /// Vehicle 이동 및 동작 중 Alarm 발생 시 처리
  2075. /// </summary>
  2076. /// <param name="alarmID"></param>
  2077. public void OccurVehicleAlarm( int alarmID )
  2078. {
  2079. logger.D($"[{this.GetType().Name}] - Occur Alarm Method- {alarmID}");
  2080. var alarm = refObjects.Alarms.FirstOrDefault( x=>x.AlarmId == alarmID );
  2081. if ( alarm != null )
  2082. {
  2083. if ( alarm.Level == eAlarmLevel.Fault )
  2084. {
  2085. this.MachineMode = eMachineMode.LocalMode;
  2086. this.VehicleStateProperty = eVehicleState.Abnormal;
  2087. this.ConveyorOff();
  2088. }
  2089. }
  2090. this.refObjects.AutoManager.ProcessAlarm( alarmID );
  2091. }
  2092. public void SetObstaclePattern( ObstacleControlEventArgs.eControlKind state, int value )
  2093. {
  2094. if ( state == ObstacleControlEventArgs.eControlKind.DRIVE )
  2095. {
  2096. this.ObstacleDrive = value;
  2097. ChgObstacleDetectPattern( this.ObstacleDrive );
  2098. }
  2099. else if ( state == ObstacleControlEventArgs.eControlKind.CURVE )
  2100. {
  2101. this.ObstacleCurve = value;
  2102. ChgObstacleDetectPattern( this.ObstacleCurve );
  2103. }
  2104. else
  2105. return;
  2106. }
  2107. public eSteeringState GetESteeringState() => this.refObjects.Steering.GetSteeringState();
  2108. #endregion
  2109. #region Event Subscribe
  2110. private void BMUManager_OnDisconnect( string obj )
  2111. {
  2112. this.BatteryIsConnect = false;
  2113. this.OccurVehicleAlarm( 32 );
  2114. }
  2115. private void BMUManager_OnConnect( string obj )
  2116. {
  2117. this.BatteryIsConnect = true;
  2118. //ReqCurrentPos();
  2119. }
  2120. private void BMUManager_OnFirstColtd( List<ReceivedData> obj )
  2121. {
  2122. }
  2123. private void BMUManager_OnChangedReceivedData( Serial.DataModel.ReceivedData obj )
  2124. {
  2125. var kind = CastTo<eDataKind>.From<Enum>( obj.DataKind );
  2126. switch ( kind )
  2127. {
  2128. case eDataKind.Voltage:
  2129. this.BatteryVoltage = (double)obj.Value;
  2130. break;
  2131. case eDataKind.Current:
  2132. this.BatteryCurrent = (double)obj.Value;
  2133. //if ( BatteryCurrent > 3 && IsChargeCylinderForword() )
  2134. // this.BatteryChargeState = true;
  2135. //else
  2136. // this.BatteryChargeState = false;
  2137. break;
  2138. case eDataKind.BatteryState:
  2139. this.BatteryState = (double)obj.Value;
  2140. break;
  2141. case eDataKind.ChargeCompleteTime:
  2142. this.BatteryChargeTime = (double)obj.Value;
  2143. break;
  2144. case eDataKind.DisChargeCompleteTime:
  2145. this.BatteryDisChargeTime = (double)obj.Value;
  2146. break;
  2147. case eDataKind.SOC:
  2148. this.BatteryStateOfCharge = (double)obj.Value;
  2149. break;
  2150. case eDataKind.SOH:
  2151. this.BatteryStateOfHealth = (double)obj.Value;
  2152. break;
  2153. case eDataKind.ResidualCapacity:
  2154. this.BatteryCapacity = (double)obj.Value;
  2155. break;
  2156. case eDataKind.ResidualEnergy:
  2157. this.BatteryEnergy = (double)obj.Value;
  2158. break;
  2159. case eDataKind.Temperature:
  2160. this.BatteryTemperature = (double)obj.Value;
  2161. break;
  2162. default:
  2163. break;
  2164. }
  2165. }
  2166. private void AutoManager_OnOperationModeChanged( eOperatationMode obj )
  2167. {
  2168. //this.refObjects.Drive.SetDriveOperationMode(obj);
  2169. }
  2170. private void ZmqManager_PropertyChanged( object sender, System.ComponentModel.PropertyChangedEventArgs e )
  2171. {
  2172. var property = sender.GetType().GetProperty( e.PropertyName );
  2173. var newValue = property.GetValue( sender, null );
  2174. switch ( e.PropertyName )
  2175. {
  2176. case "SegmentID":
  2177. {
  2178. var v = CastTo<int>.From<object>( newValue );
  2179. this.ObstaclePatternChange( v );
  2180. }
  2181. break;
  2182. case "RequestSteering":
  2183. {
  2184. var v = CastTo<eSteeringState>.From<object>( newValue );
  2185. switch ( v )
  2186. {
  2187. case eSteeringState.None:
  2188. break;
  2189. case eSteeringState.Left:
  2190. this.refObjects.Steering.RequestControl( eSteeringState.Left );
  2191. break;
  2192. case eSteeringState.Right:
  2193. this.refObjects.Steering.RequestControl( eSteeringState.Right );
  2194. break;
  2195. default:
  2196. break;
  2197. }
  2198. logger.D( $"[Request Steering] - {v}" );
  2199. }
  2200. break;
  2201. case "CurrentPointNo":
  2202. {
  2203. var v = CastTo<int>.From<object>( newValue );
  2204. logger.D( $"{this.CurrentTag} -> {v}" );
  2205. this.CurrentTag = v;
  2206. }
  2207. break;
  2208. case "CurrentMCR":
  2209. break;
  2210. case "RearDriveState":
  2211. {
  2212. this.RearDriveState = CastTo<DriveState>.From<object>( newValue );
  2213. }
  2214. break;
  2215. case "FrontLoadFactor":
  2216. {
  2217. this.FrontLoadFactor = CastTo<double>.From<object>( newValue );
  2218. this.FrontTorque = Math.Truncate( ( ( FrontLoadFactor * 1.9 ) / 1000 ) * 100 ) / 100;
  2219. }
  2220. break;
  2221. case "FrontRPM":
  2222. {
  2223. this.FrontRpm = CastTo<double>.From<object>( newValue );
  2224. var ll = ( ( ( ( FrontRpm / 60 ) * ( 2 * Math.PI ) ) * 0.06 ) / 10 );
  2225. this.FrontSpeed = Math.Truncate( ll * 100 ) / 100;
  2226. }
  2227. break;
  2228. case "RearLoadFactor":
  2229. {
  2230. this.RearLoadFactor = CastTo<double>.From<object>( newValue );
  2231. this.RearTorque = Math.Truncate( ( ( RearLoadFactor * 1.9 ) / 1000 ) * 100 ) / 100;
  2232. }
  2233. break;
  2234. case "RearRPM":
  2235. {
  2236. this.RearRpm = CastTo<double>.From<object>( newValue );
  2237. var ll = ( ( ( ( RearRpm / 60 ) * ( 2 * Math.PI ) ) * 0.06 ) / 10 );
  2238. this.RearSpeed = Math.Truncate( ll * 100 ) / 100;
  2239. }
  2240. break;
  2241. case "IsReqConnected":
  2242. {
  2243. var isConnected = CastTo<bool>.From<object>( newValue );
  2244. if ( !isConnected )
  2245. OccurVehicleAlarm( 44 );
  2246. }
  2247. break;
  2248. //case "LinearSpeed":
  2249. // {
  2250. // this.LinearSpeed = CastTo<double>.From<object>(newValue);
  2251. // }
  2252. // break;
  2253. //case "CurveSpeed":
  2254. // {
  2255. // this.CurveSpeed = CastTo<double>.From<object>(newValue);
  2256. // }
  2257. // break;
  2258. //case "JogSPeed":
  2259. // {
  2260. // this.JogSPeed = CastTo<double>.From<object>(newValue);
  2261. // }
  2262. // break;
  2263. //case "AccelSpeed":
  2264. // {
  2265. // this.AccelSpeed = CastTo<double>.From<object>(newValue);
  2266. // }
  2267. // break;
  2268. //case "DecelSpeed":
  2269. // {
  2270. // this.DecelSpeed = CastTo<double>.From<object>(newValue);
  2271. // }
  2272. // break;
  2273. //case "CreepSpeed":
  2274. // {
  2275. // this.CreepSpeed = CastTo<double>.From<object>(newValue);
  2276. // }
  2277. // break;
  2278. //case "CreepDistance":
  2279. // {
  2280. // this.CreepDistance = CastTo<double>.From<object>(newValue);
  2281. // }
  2282. // break;
  2283. //case "IsCanStanbyLocation":
  2284. // this.IsCanStanbyLocation = CastTo<bool>.From<object>(newValue);
  2285. // break;
  2286. default:
  2287. break;
  2288. }
  2289. }
  2290. private void Motion_PropertyChanged( object sender, System.ComponentModel.PropertyChangedEventArgs e )
  2291. {
  2292. var property = sender.GetType().GetProperty( e.PropertyName );
  2293. var newValue = property.GetValue( sender, null );
  2294. switch ( e.PropertyName )
  2295. {
  2296. case "CurrentPos":
  2297. {
  2298. var v = CastTo<double>.From<object>( newValue );
  2299. this.CurrentPosition = v;
  2300. }
  2301. break;
  2302. //case "CurrentTag":
  2303. // {
  2304. // var v = CastTo<int>.From<object>( newValue );
  2305. // logger.D( $"{this.CurrentTag} -> {v}" );
  2306. // this.CurrentTag = v;
  2307. // }
  2308. // break;
  2309. case "CurrentSpeed":
  2310. {
  2311. var v = CastTo<double>.From<object>( newValue );
  2312. this.CurrentSpeed = v;
  2313. }
  2314. break;
  2315. case "CurrentTorque":
  2316. {
  2317. var v = CastTo<double>.From<object>( newValue );
  2318. this.CurrentTorque = v;
  2319. }
  2320. break;
  2321. case "ReqSteeringState":
  2322. //{
  2323. // var v = CastTo<eSteeringState>.From<object>( newValue );
  2324. // switch ( v )
  2325. // {
  2326. // case eSteeringState.None:
  2327. // break;
  2328. // case eSteeringState.Left:
  2329. // this.refObjects.Steering.ControlSteering( true );
  2330. // break;
  2331. // case eSteeringState.Right:
  2332. // this.refObjects.Steering.ControlSteering();
  2333. // break;
  2334. // default:
  2335. // break;
  2336. // }
  2337. //}
  2338. break;
  2339. case "DriveServoState":
  2340. {
  2341. var v = CastTo<eDriveServoState>.From<object>( newValue );
  2342. if ( v == eDriveServoState.ServoOn )
  2343. this.refObjects.IO.OutputOn( "OUT_DRIVE_BRAKE_OFF" );
  2344. else
  2345. this.refObjects.IO.OutputOff( "OUT_DRIVE_BRAKE_OFF" );
  2346. }
  2347. break;
  2348. case "FrontDriveState":
  2349. {
  2350. var v = CastTo<DriveState>.From<object>( newValue );
  2351. this.FrontDriveState = v;
  2352. }
  2353. break;
  2354. case "RearDriveState":
  2355. {
  2356. var v = CastTo<DriveState>.From<object>( newValue );
  2357. this.RearDriveState = v;
  2358. }
  2359. break;
  2360. default:
  2361. break;
  2362. }
  2363. }
  2364. private void Steering_PropertyChanged( object sender, System.ComponentModel.PropertyChangedEventArgs e )
  2365. {
  2366. var property = sender.GetType().GetProperty( e.PropertyName );
  2367. var newValue = property.GetValue( sender, null );
  2368. //Todo: 나중에 Test 하자
  2369. //var ownPropperty = this.GetType().GetProperty(e.PropertyName);
  2370. if ( e.PropertyName.Equals( "SteeringState" ) )
  2371. {
  2372. var v = CastTo<eSteeringState>.From<object>( newValue );
  2373. this.SteeringState = v;
  2374. logger.D( $"steering - {v}" );
  2375. }
  2376. }
  2377. private void Clamp_PropertyChanged( object sender, System.ComponentModel.PropertyChangedEventArgs e )
  2378. {
  2379. var property = sender.GetType().GetProperty( e.PropertyName );
  2380. var newValue = property.GetValue( sender, null );
  2381. if ( e.PropertyName.Equals( "ClampState" ) )
  2382. {
  2383. var v = CastTo<eClampState>.From<object>( newValue );
  2384. this.ClampState = v;
  2385. }
  2386. }
  2387. private void IO_OnChangedIO( BitBlock bit )
  2388. {
  2389. switch ( bit.Tag )
  2390. {
  2391. case "IN_CV_DETECT_01":
  2392. this.IsContain = bit.IsBitOn;
  2393. this.refObjects.ZmqManager.Publish( "Contain", this.IsContain.ToString() );
  2394. break;
  2395. case "IN_MC_ON":
  2396. if ( bit.IsBitOn )
  2397. {
  2398. if (!this.refObjects.IO.IsOn("IN_EMS_SW") && this.refObjects.IO.IsOn("IN_CP_ON_SAFETY") && this.refObjects.IO.IsOn("IN_CP_ON_24V"))
  2399. {
  2400. //this.VehicleStateProperty = eVehicleState.Idle;
  2401. }
  2402. }
  2403. else
  2404. {
  2405. if ( this.ObstacleStateProperty == eObstacleState.Abnormal )
  2406. this.OccurVehicleAlarm( 42 );
  2407. else
  2408. this.OccurVehicleAlarm( 29 );
  2409. }
  2410. break;
  2411. case "IN_EMS_SW":
  2412. if ( bit.IsBitOn )
  2413. this.OccurVehicleAlarm( 28 );
  2414. break;
  2415. case "IN_CP_ON_SAFETY":
  2416. if ( !bit.IsBitOn )
  2417. this.OccurVehicleAlarm( 31 );
  2418. break;
  2419. case "IN_CP_ON_24V":
  2420. if ( !bit.IsBitOn )
  2421. this.OccurVehicleAlarm( 30 );
  2422. break;
  2423. case "IN_OBSTRUCTION_DETECT_ERROR":
  2424. case "IN_OBSTRUCTION_DETECT_SAFETY":
  2425. case "IN_OBSTRUCTION_DETECT_STOP":
  2426. case "IN_OBSTRUCTION_DETECT_SLOW":
  2427. CheckObstacle();
  2428. break;
  2429. case "IN_BUMPER_DETECT":
  2430. if ( bit.IsBitOn )
  2431. this.refObjects.ZmqManager.SetSafetyBumper();
  2432. break;
  2433. case "IN_CHARGE_CYL_FWD_DETECT":
  2434. if ( bit.IsBitOn )
  2435. this.BatteryChargeState = true;
  2436. else
  2437. this.BatteryChargeState = false;
  2438. break;
  2439. case "IN_VEHICLE_FRONT_DETECT":
  2440. this.refObjects.ZmqManager.SetFrontDetectedVehicle(bit.IsBitOn);
  2441. break;
  2442. case "IN_F_STEERING_DETECT_LEFT":
  2443. case "IN_F_STEERING_DETECT_RIGHT":
  2444. case "IN_R_STEERING_DETECT_LEFT":
  2445. case "IN_R_STEERING_DETECT_RIGHT":
  2446. var frontLeft = this.refObjects.IO.IsOn("IN_F_STEERING_DETECT_LEFT");
  2447. var frontRight = this.refObjects.IO.IsOn("IN_F_STEERING_DETECT_RIGHT");
  2448. var rearLeft = this.refObjects.IO.IsOn("IN_R_STEERING_DETECT_LEFT");
  2449. var rearRight = this.refObjects.IO.IsOn("IN_R_STEERING_DETECT_RIGHT");
  2450. this.refObjects.ZmqManager.SetSteerDetectSensorState(frontLeft, frontRight, rearLeft, rearRight);
  2451. logger.D($"[STEERING DETECTED] - {bit.Tag} / {bit.IsBitOn}");
  2452. break;
  2453. default:
  2454. break;
  2455. }
  2456. }
  2457. private void Steering_OnSteeringError( object sender, int e )
  2458. {
  2459. if ( e != 0 )
  2460. {
  2461. logger.E( $"[Steering] - Control Error {e}" );
  2462. this.OccurVehicleAlarm( e );
  2463. }
  2464. else
  2465. {
  2466. var msg = new DriveControlEventArgs()
  2467. {
  2468. EventDir = DriveControlEventArgs.eEventDir.ToFront,
  2469. ControlKind = DriveControlEventArgs.eControlKind.Steering,
  2470. Result = FluentResults.Results.Ok<DriveControlEventArgs.eMoveDir>( DriveControlEventArgs.eMoveDir.LEFT ),
  2471. };
  2472. this.eventAggregator.GetEvent<DriveControlPubSubEvent>().Publish( msg );
  2473. }
  2474. }
  2475. #endregion
  2476. }
  2477. }