frmMain.cs 48 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028
  1. using com.hnshituo.core.webapp.vo;
  2. using Newtonsoft.Json;
  3. using System;
  4. using System.Collections.Generic;
  5. using System.ComponentModel;
  6. using System.Data;
  7. using System.Diagnostics;
  8. using System.Drawing;
  9. using System.Drawing.Drawing2D;
  10. using System.IO;
  11. using System.Linq;
  12. using System.Net.NetworkInformation;
  13. using System.Runtime.InteropServices;
  14. using System.Text;
  15. using System.Threading;
  16. using System.Threading.Tasks;
  17. using System.Windows.Forms;
  18. namespace CarLocalMeter
  19. {
  20. public partial class frmMain : Form
  21. {
  22. public frmMain()
  23. {
  24. InitializeComponent();
  25. }
  26. Log lg = Log.GetInstance();
  27. private LED_Control led_controler;
  28. //List<CamerEs> lc = new List<CamerEs>();
  29. /// <summary>
  30. /// 硬盘录像机图片抓拍通道
  31. /// </summary>
  32. List<int> lc = new List<int>();
  33. /// <summary>
  34. /// 服务端委托获取
  35. /// </summary>
  36. private PreTrackScaleService PreTrackScaleService = new PreTrackScaleService();
  37. RESTfulResult<List<PreTrackScale>> rmScaleListCarTimer = new RESTfulResult<List<PreTrackScale>>();
  38. private MeterWorkCarActualFirstService workCarActualFirstService = new MeterWorkCarActualFirstService(); //一次计量数据
  39. RESTfulResult<List<MeterWorkCarActualFirst>> RESTfulResultOutTimer = new RESTfulResult<List<MeterWorkCarActualFirst>>();
  40. PlcCls plc = new PlcCls();
  41. /// <summary>
  42. /// redis操作类
  43. /// </summary>
  44. RedisOption redis = new RedisOption();
  45. /// <summary>
  46. /// 图片操作类
  47. /// </summary>
  48. ImageOption imgControl = new ImageOption();
  49. /// <summary>
  50. /// 重量采集类
  51. /// </summary>
  52. MoxaCls mx = new MoxaCls();
  53. /// <summary>
  54. /// 语音播放
  55. /// </summary>
  56. VoicePlay vicPlayClass = new VoicePlay();
  57. /// <summary>
  58. /// 摄像头车号识别枪机
  59. /// </summary>
  60. CarNoCls carCls = new CarNoCls();
  61. /// <summary>
  62. /// 摄像头识别球机
  63. /// </summary>
  64. CarNoCls carCls2 = new CarNoCls();
  65. /// <summary>
  66. /// rfid车号识别
  67. /// </summary>
  68. RfIdCarNo rfid = new RfIdCarNo();
  69. /// <summary>
  70. /// Ping服务器服务是否通
  71. /// </summary>
  72. PingNetwork ping = new PingNetwork();
  73. /// <summary>
  74. /// 本地离线计量的数据上传
  75. /// </summary>
  76. DbUpload upload = new DbUpload();
  77. /// <summary>
  78. /// 以下是硬盘录像机的控制
  79. /// </summary>
  80. bool bVoice = false;
  81. CamerEs ce = new CamerEs(AppConfigCache.voicePlayIp, $"{AppConfigCache.voicePlayPort}", AppConfigCache.voicePlayUid, AppConfigCache.voicePlayPwd);
  82. List<int> voPlay = new List<int>();
  83. int iPic = 0;
  84. private void frmMain_Load(object sender, EventArgs e)
  85. {
  86. CacleCls.updateRfidInfoTime = DateTime.Now;
  87. btnSave.Enabled = false;
  88. if (!string.IsNullOrEmpty(AppConfigCache.ledIp))
  89. {
  90. led_controler = new LED_Control(AppConfigCache.ledIp);
  91. }
  92. #region 读取音频文件名称
  93. List<string> ls = new List<string>();
  94. DirectoryInfo folder = new DirectoryInfo(AppConfigCache.path + "\\Sound");
  95. foreach (FileInfo file in folder.GetFiles("*.wav"))
  96. {
  97. ls.Add(file.Name.Replace(".wav", ""));
  98. }
  99. CacleCls.voiceInfo = ls;
  100. #endregion
  101. string[] strTd1 = AppConfigCache.channelOther.Split(',');
  102. string[] strTd2 = AppConfigCache.channelVis.Split(',');
  103. string[] strAll = new string[strTd1.Length + strTd2.Length];
  104. strTd1.CopyTo(strAll, 0);
  105. strTd2.CopyTo(strAll, strTd1.Length);
  106. List<string> lcs = strAll.Distinct().OrderBy(s => s).ToList(); //
  107. lc.Add(AppConfigCache.channelCarNo);
  108. foreach (string str in lcs)
  109. {
  110. if ($"{AppConfigCache.channel}" != str && $"{AppConfigCache.channelCarNo}" != str)
  111. {
  112. lc.Add(Convert.ToInt32(str)); //抓拍通道与数组中不相同的时候写入,因为我们仪表抓拍的通道需单独进行处理
  113. }
  114. }
  115. plImgShow.Visible = false;
  116. lbPointName.Text = AppConfigCache.pointName;
  117. foreach (string str in AppConfigCache.channelVis.Split(','))
  118. {
  119. voPlay.Add(Convert.ToInt32(str));
  120. }
  121. ce.Connection();
  122. foreach(int i in voPlay)
  123. {
  124. PictureBox pb = new PictureBox();
  125. pb.Dock = DockStyle.Top;
  126. pb.DoubleClick += new EventHandler(PictureBoxDoubleClick);
  127. pb.Height = AppConfigCache.channelVisHeight;
  128. pb.Name = $"pb{i}";
  129. panel1.Controls.Add(pb);
  130. ce.RealPlay(pb, i);
  131. }
  132. if (AppConfigCache.plcStart == "1") plc.Start();
  133. //*
  134. ping.start();
  135. upload.start();
  136. mx.start();
  137. rfid.Start();
  138. imgControl.Start();
  139. carCls.Login(AppConfigCache.voiceCarNoIp, AppConfigCache.voiceCarNoPort, AppConfigCache.voiceCarNoUid, AppConfigCache.voiceCarNoPwd);
  140. //*/
  141. //carCls2.Login(AppConfigCache.voiceCarNoIp2, AppConfigCache.voiceCarNoPort2, AppConfigCache.voiceCarNoUid2, AppConfigCache.voiceCarNoPwd2, false);
  142. blThreadFlag = true;
  143. DataCollectThread = new Thread(new ThreadStart(DataCollect));
  144. start();
  145. }
  146. #region 本地主逻辑判断线程
  147. bool blThreadFlag;//数据采集线程开关
  148. Thread DataCollectThread = null;//采集进程
  149. bool getPreInfo = false, getFirst = false, flagCaption = false, isVoiceDownCar = false;
  150. string strOldCarNo = "";
  151. bool isMeasureSuccess = false;
  152. int voiceNum = 0;
  153. int iImgCnt = 0;
  154. private void start() => DataCollectThread.Start();
  155. private void stop() => blThreadFlag = false;
  156. private void DataCollect()
  157. {
  158. if (!blThreadFlag) return;
  159. int iError = 0;
  160. while (blThreadFlag)
  161. {
  162. try
  163. {
  164. this.panel3.Invoke(new Action(() => {
  165. if (!CacleCls.isLock)
  166. {
  167. ucWeightT1.setWgt(Math.Round((double)CacleCls.weight / 1000, 2)); //2021年5月10日界面显示单位,调整为T
  168. ucWeightT1.setStable(CacleCls.isWd != 1 ? true : false);
  169. ucWeightT1.setExceed(CacleCls.weight > AppConfigCache.maxRange * 1000 ? false : true);
  170. if (cbCarNo.Text.Trim() != "" && txtCarNo.Text.Trim() != "")
  171. {
  172. CacleCls.lockCarNo = cbCarNo.Text.Trim() + txtCarNo.Text.Trim();
  173. }
  174. else if (!string.IsNullOrEmpty(CacleCls.rfidCarNo))
  175. {
  176. CacleCls.lockCarNo = CacleCls.rfidCarNo;
  177. }
  178. else if (!string.IsNullOrEmpty(CacleCls.voicCarNo))
  179. {
  180. CacleCls.lockCarNo = CacleCls.voicCarNo;
  181. }
  182. else if (!string.IsNullOrEmpty(CacleCls.voicCarNo2))
  183. {
  184. CacleCls.lockCarNo = CacleCls.voicCarNo2;
  185. }
  186. if (!string.IsNullOrEmpty(CacleCls.lockCarNo) && cbCarNo.Text.Trim() == "" && txtCarNo.Text.Trim() == "")
  187. {
  188. cbCarNo.Text = CacleCls.lockCarNo.Substring(0, 1);
  189. if (CacleCls.lockCarNo.Length > 1)
  190. {
  191. txtCarNo.Text = CacleCls.lockCarNo.Substring(1);
  192. }
  193. }
  194. if (strOldCarNo != CacleCls.lockCarNo)
  195. {
  196. strOldCarNo = CacleCls.lockCarNo;
  197. getPreInfo = false;
  198. getFirst = false;
  199. }
  200. }
  201. iError = 1;
  202. //是否能连上远程服务器,这里只是服务器,不是服务器上的服务端
  203. pbLineOn.Load(CacleCls.serverFlag ? (AppConfigCache.path + "\\image\\icon\\green.gif") : (AppConfigCache.path + "\\image\\icon\\red.gif"));
  204. iError = 2;
  205. if (CacleCls.weight > 500)
  206. {
  207. iError = 21;
  208. iImgCnt = 0;
  209. CacleCls.isClear = false;
  210. #region 一些必要验证及提醒
  211. if (!CacleCls.isLock)
  212. {
  213. //重量最后一位不为0也是超量程
  214. if ((AppConfigCache.maxRange * 1000 < CacleCls.weight) || CacleCls.weight.ToString().Substring(CacleCls.weight.ToString().Length - 1, 1) != "0")
  215. {
  216. //超量程
  217. vicPlayClass.GetVoicePlay(VoiceEnum.车辆超重, CacleCls.lockCarNo);
  218. txtMsgInfo.Text = "当前重量已超量程,秤体量程为【" + AppConfigCache.maxRange + "】T,不允许进行自助计量操作!";
  219. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "当前重量已超量程,秤体量程为【" + AppConfigCache.maxRange + "】T,不允许进行自助计量操作!");
  220. CacleCls.isLock = false;
  221. return;
  222. }
  223. iError = 22;
  224. if (CacleCls.isWd == 0 && !string.IsNullOrEmpty(CacleCls.lockCarNo))
  225. {
  226. if ((CacleCls.topJg == "1" || CacleCls.bottomJg == "1"))
  227. {
  228. //重量稳定,且对射验证不合格
  229. vicPlayClass.GetVoicePlay(VoiceEnum.车辆未停到位, CacleCls.lockCarNo);
  230. txtMsgInfo.Text = "车辆超出秤台,请调整车位2";
  231. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "车辆超出秤台,请调整车位2");
  232. CacleCls.isLock = false;
  233. return;
  234. }
  235. else if (CacleCls.rightGs == "1")
  236. {
  237. //重量稳定,且右光栅不合格
  238. vicPlayClass.GetVoicePlay(VoiceEnum.车辆未停到位, CacleCls.lockCarNo);
  239. txtMsgInfo.Text = "车辆超出秤台,请调整车位3";
  240. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "车辆超出秤台,请调整车位3");
  241. CacleCls.isLock = false;
  242. return;
  243. }
  244. else if (CacleCls.leftGs == "1")
  245. {
  246. //重量稳定,且左光栅不合格
  247. vicPlayClass.GetVoicePlay(VoiceEnum.车辆未停到位, CacleCls.lockCarNo);
  248. txtMsgInfo.Text = "车辆超出秤台,请调整车位4";
  249. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "车辆超出秤台,请调整车位4");
  250. CacleCls.isLock = false;
  251. return;
  252. }
  253. }
  254. iError = 23;
  255. /*
  256. if (PbCache.isOvertimeAlarm)
  257. {
  258. //停留超时提醒,不作为计量卡控的条件
  259. if (!isPlayOvertimeVoice && !PbCache.isLockFrm)
  260. {
  261. vicPlayClass.GetVoicePlay("停留超时", PbCache.collect.carno);
  262. led_controler.setStaticLineMsg(PbCache.sportInfo.ledIp, "停留超时");
  263. txtMsgInfo.Text = "停留超时";
  264. PbCache.ResultMessage = "停留超时";
  265. isPlayOvertimeVoice = true;
  266. return;
  267. }
  268. }
  269. //*/
  270. #region 一次计量数据跟预报获取一次,有可能稳定特别快导致一次计量数据跟预报还未获取到,所以我这改为去保存按钮处获取
  271. /*
  272. if (!string.IsNullOrEmpty(CacleCls.lockCarNo) && CacleCls.serverFlag)
  273. {
  274. //查询一次计量数据
  275. if (!getFirst)
  276. {
  277. RESTfulResultOutTimer = workCarActualFirstService.doQueryWf(new MeterWorkCarActualFirst { valueFlag = "1", carNo = CacleCls.lockCarNo });
  278. if (RESTfulResultOutTimer.Succeed)
  279. {
  280. if (RESTfulResultOutTimer.Data != null && RESTfulResultOutTimer.Data.Count > 0)
  281. {
  282. CacleCls.firstDb = RESTfulResultOutTimer.Data[0];
  283. if (CacleCls.firstDb.predictionNo != null && !CacleCls.firstDb.predictionNo.Contains("_"))
  284. {
  285. txtMATTER_NAME.Text = CacleCls.firstDb.matterName;
  286. txtFORWARDING_UNIT_NAME.Text = CacleCls.firstDb.forwardingUnitName;
  287. txtRECEIVING_UINT_NAME.Text = CacleCls.firstDb.receivingUintName;
  288. txtMETER_TYPE.Text = CacleCls.firstDb.meterTypeName;
  289. }
  290. }
  291. }
  292. getFirst = true;
  293. }
  294. iError = 24;
  295. //根据网络状态及车号以及是否存在一次计量数据,查询一次预报信息,然后将getPreInfo改为true
  296. if (!getPreInfo)
  297. {
  298. if (CacleCls.firstDb == null || string.IsNullOrEmpty(CacleCls.firstDb?.predictionNo))
  299. {
  300. //如果没有一次信息或者一次信息里面没带预报信息,则查询一次预报信息
  301. rmScaleListCarTimer = PreTrackScaleService.doQueryByFlag(new PreTrackScale { lineDesc = "1", carNo = CacleCls.lockCarNo });
  302. if (rmScaleListCarTimer.Succeed)
  303. {
  304. if (rmScaleListCarTimer.Data != null && rmScaleListCarTimer.Data.Count == 1)
  305. {
  306. CacleCls.preTrackScale = rmScaleListCarTimer.Data[0];
  307. txtMATTER_NAME.Text = CacleCls.preTrackScale.matterName;
  308. txtFORWARDING_UNIT_NAME.Text = CacleCls.preTrackScale.forwardingUnitName;
  309. txtRECEIVING_UINT_NAME.Text = CacleCls.preTrackScale.receivingUintName;
  310. txtMETER_TYPE.Text = CacleCls.preTrackScale.meterTypeName;
  311. }
  312. }
  313. }
  314. getPreInfo = true;
  315. }
  316. iError = 25;
  317. }
  318. //*/
  319. #endregion
  320. //进行计量称重条件的判断 车号不为空、重量稳定、车辆已停到位
  321. if (CacleCls.isWd == 0 && CacleCls.leftGs != "1" && CacleCls.rightGs != "1" && CacleCls.topJg != "1" && CacleCls.bottomJg != "1")
  322. {
  323. iError = 11;
  324. if (!string.IsNullOrEmpty(CacleCls.lockCarNo))
  325. {
  326. if (!isVoiceDownCar)
  327. {
  328. txtMsgInfo.Text = "";
  329. isVoiceDownCar = true;
  330. vicPlayClass.GetVoicePlay(VoiceEnum.开始计量, CacleCls.lockCarNo);//只播放一次
  331. }
  332. //if(!btnSave.Enabled) btnSave.Enabled = true;
  333. if (!CacleCls.isJg)
  334. {
  335. lg.WriteLog(LogType.SystemLog, CacleCls.lockCarNo + "开始保存:" + DateTime.Now.ToLongTimeString());
  336. btnSave_Click(null, null);
  337. }
  338. }
  339. else
  340. {
  341. txtMsgInfo.Text = "车号未识别";
  342. vicPlayClass.GetVoicePlay(VoiceEnum.车号未识别, "");
  343. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "车号未识别");
  344. }
  345. iError = 12;
  346. }
  347. }
  348. #endregion
  349. //计量完成后进行语音播报,提示离开秤台,播报5次
  350. if (isMeasureSuccess)
  351. {
  352. if (voiceNum < AppConfigCache.voiceNum)
  353. {
  354. if (CacleCls.voiceOver)
  355. {
  356. vicPlayClass.GetVoicePlay(VoiceEnum.计量完成, CacleCls.lockCarNo);
  357. voiceNum++;
  358. }
  359. }
  360. return;
  361. }
  362. iError = 13;
  363. }
  364. else
  365. {
  366. iError = 31;
  367. voiceNum = 0;
  368. isMeasureSuccess = false;
  369. lockWgtImg = CacleCls.weight;
  370. getFirst = false;
  371. getPreInfo = false;
  372. flagCaption = false;
  373. isVoiceDownCar = false;
  374. strGuid = "";
  375. if (!CacleCls.isClear)
  376. {
  377. iError = 32;
  378. CacleCls.Clear(); //清理一次数据
  379. btnSave.Enabled = false;
  380. cbCarNo.Text = "";
  381. cbCarNo.SelectedIndex = -1;
  382. txtCarNo.Text = "";
  383. txtMATTER_NAME.Text = "";
  384. txtFORWARDING_UNIT_NAME.Text = "";
  385. txtRECEIVING_UINT_NAME.Text = "";
  386. txtMETER_TYPE.Text = "";
  387. cbCarNo.Enabled = true;
  388. txtCarNo.Enabled = true;
  389. txtMsgInfo.Text = "";
  390. txtGroess.Text = "";
  391. txtTare.Text = "";
  392. txtNet.Text = "";
  393. iError = 33;
  394. //释放一次内存
  395. FlushMemory();
  396. iError = 34;
  397. //
  398. imgControl.ZipFiles();
  399. iError = 35;
  400. }
  401. }
  402. }));
  403. }
  404. catch (Exception ex)
  405. {
  406. //在关闭程序后,由于线程还会跑一下,所以会有一个错误“调用方法时发生错误。 目标线程不再存在” 这个不影响
  407. lg.WriteLog(LogType.SystemLog, $"主逻辑判断线程,其i值为:{iError},异常信息为:{ex.Message}");
  408. }
  409. Thread.Sleep(AppConfigCache.sleepTime);
  410. }
  411. }
  412. #endregion
  413. int lockWgtImg = 0;
  414. string strGuid = "";
  415. private void btnSave_Click(object sender, EventArgs e)
  416. {
  417. try
  418. {
  419. #region 下面是业务逻辑代码
  420. string msgInfo = "";
  421. if (txtCarNo.Focused) return; //如果正在修改车号,就必须等光标离开后再保存,否则容易导致车号输入了一半保存了
  422. if (CacleCls.isWd != 0) return;
  423. ucWeightT1.setStable(true); // 重量稳定
  424. ucWeightT1.setWgt(Math.Round((double)CacleCls.weight / 1000, 2)); // 最新重量
  425. CacleCls.lockWgt = CacleCls.weight;
  426. if (!CacleCls.lockWgt.ToString().EndsWith("0")) //个位不是0则说明超量程了
  427. {
  428. //txtMsgInfo.Text = "仪表重量[" + CacleCls.lockWgt + "]不是以0结尾,禁止计量操作!";
  429. vicPlayClass.GetVoicePlay(VoiceEnum.车辆超重, CacleCls.lockCarNo);
  430. txtMsgInfo.Text = "当前重量已超量程,秤体量程为【" + AppConfigCache.maxRange + "】T,不允许进行自助计量操作!";
  431. return;
  432. }
  433. //接管状态点的保存
  434. if (CacleCls.isJg)
  435. {
  436. if (string.IsNullOrEmpty(CacleCls.lockCarNo))
  437. {
  438. txtMsgInfo.Text = "没有车号,无法保存";
  439. return;
  440. }
  441. else if (CacleCls.lockWgt < 500)
  442. {
  443. txtMsgInfo.Text = "重量过小,无法保存";
  444. return;
  445. }
  446. else if (CacleCls.isWd != 0)
  447. {
  448. txtMsgInfo.Text = "重量不稳定,无法保存";
  449. return;
  450. }
  451. else if (CacleCls.leftGs == "1" || CacleCls.rightGs == "1" || CacleCls.topJg == "1" || CacleCls.bottomJg == "1")
  452. {
  453. txtMsgInfo.Text = "激光被遮挡,车辆未停到位,无法保存";
  454. return;
  455. }
  456. else
  457. {
  458. lg.WriteLog(LogType.SystemLog, CacleCls.lockCarNo + "接管状态下进行的保存:" + DateTime.Now.ToLongTimeString());
  459. }
  460. }
  461. //存在零点,且不为接管状态则禁止计量
  462. if (CacleCls.isZeroState && !CacleCls.isJg)
  463. {
  464. vicPlayClass.GetVoicePlay(VoiceEnum.仪表未归零, CacleCls.lockCarNo);
  465. txtMsgInfo.Text = "仪表未归零,无法计量,请联系大厅"; //btnSave_Click
  466. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "仪表未归零,无法计量");
  467. return;
  468. }
  469. if (!CacleCls.isJg && CacleCls.firstDb != null)
  470. {
  471. if (Math.Abs(CacleCls.firstDb.meterWeight.Value - CacleCls.lockWgt) < AppConfigCache.differenceWgt)
  472. {
  473. vicPlayClass.GetVoicePlay(VoiceEnum.洁净重量过小, CacleCls.lockCarNo);
  474. txtMsgInfo.Text = "计量失败,洁净重量过小";
  475. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "计量失败,洁净重量过小");
  476. return;
  477. }
  478. }
  479. #region 获取一次数据跟车号
  480. if (!string.IsNullOrEmpty(CacleCls.lockCarNo) && CacleCls.serverFlag)
  481. {
  482. //查询一次计量数据
  483. if (!getFirst)
  484. {
  485. RESTfulResultOutTimer = workCarActualFirstService.doQueryWf(new MeterWorkCarActualFirst { valueFlag = "1", carNo = CacleCls.lockCarNo });
  486. if (RESTfulResultOutTimer.Succeed)
  487. {
  488. if (RESTfulResultOutTimer.Data != null && RESTfulResultOutTimer.Data.Count > 0)
  489. {
  490. if (RESTfulResultOutTimer.Data.Count == 1)
  491. {
  492. CacleCls.firstDb = RESTfulResultOutTimer.Data[0];
  493. if (CacleCls.firstDb.predictionNo != null && !CacleCls.firstDb.predictionNo.Contains("_"))
  494. {
  495. txtMATTER_NAME.Text = CacleCls.firstDb.matterName;
  496. txtFORWARDING_UNIT_NAME.Text = CacleCls.firstDb.forwardingUnitName;
  497. txtRECEIVING_UINT_NAME.Text = CacleCls.firstDb.receivingUintName;
  498. txtMETER_TYPE.Text = CacleCls.firstDb.meterTypeName;
  499. }
  500. }
  501. else
  502. {
  503. vicPlayClass.GetVoicePlay(VoiceEnum.程序处理异常, CacleCls.lockCarNo);
  504. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "计量失败,请联系管理员");
  505. txtMsgInfo.Text = $"存在{RESTfulResultOutTimer.Data.Count}条未结净的一次计量数据";
  506. lg.WriteLog(LogType.serverLog, $"存在{RESTfulResultOutTimer.Data.Count}条未结净的一次计量数据");
  507. CacleCls.isLock = false;
  508. return;
  509. }
  510. }
  511. }
  512. else
  513. {
  514. vicPlayClass.GetVoicePlay(VoiceEnum.程序处理异常, CacleCls.lockCarNo);
  515. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "计量失败,请联系管理员");
  516. txtMsgInfo.Text = "获取一次计量数据失败:" + RESTfulResultOutTimer.Message;
  517. lg.WriteLog(LogType.serverLog, "获取一次计量数据失败:" + RESTfulResultOutTimer.Message);
  518. CacleCls.isLock = false;
  519. return;
  520. }
  521. getFirst = true;
  522. }
  523. //根据网络状态及车号以及是否存在一次计量数据,查询一次预报信息,然后将getPreInfo改为true
  524. if (!getPreInfo)
  525. {
  526. if (CacleCls.firstDb != null)
  527. {
  528. //做结净的时候,才查询一次预报信息
  529. rmScaleListCarTimer = PreTrackScaleService.doQueryByFlag(new PreTrackScale { lineDesc = "1", carNo = CacleCls.lockCarNo });
  530. if (rmScaleListCarTimer.Succeed)
  531. {
  532. if (rmScaleListCarTimer.Data != null && rmScaleListCarTimer.Data.Count > 0)
  533. {
  534. CacleCls.preTrackScale = rmScaleListCarTimer.Data[0];
  535. txtMATTER_NAME.Text = CacleCls.preTrackScale.matterName;
  536. txtFORWARDING_UNIT_NAME.Text = CacleCls.preTrackScale.forwardingUnitName;
  537. txtRECEIVING_UINT_NAME.Text = CacleCls.preTrackScale.receivingUintName;
  538. txtMETER_TYPE.Text = CacleCls.preTrackScale.meterTypeName;
  539. }
  540. }
  541. }
  542. getPreInfo = true;
  543. }
  544. }
  545. #endregion
  546. lg.WriteLog(LogType.SystemLog, $"锁定车号:{CacleCls.lockCarNo},锁定重量:{CacleCls.lockWgt},rfid编号:{CacleCls.rfidEpc},rfid车号:{CacleCls.rfidCarNo},枪机:{CacleCls.voicCarNo},球机:{CacleCls.voicCarNo2}");
  547. txtMsgInfo.Text = "称重保存中......";
  548. CacleCls.isLock = true;
  549. cbCarNo.Enabled = false;
  550. txtCarNo.Enabled = false;
  551. strGuid = strGuid == "" ? Guid.NewGuid().ToString() : strGuid;
  552. #region 截图
  553. if (!flagCaption)
  554. {
  555. for (int i = 0; i < lc.Count(); i++)
  556. {
  557. if (i == 0)
  558. {
  559. carCls.CapturePictrue2($"{AppConfigCache.path}/imgShort/formalImg/{AppConfigCache.pointNo}_C{strGuid}_tempImg_{i + 2}.jpg");
  560. continue;
  561. }
  562. //从_2开始,因为_1是仪表图片 2是车号快照抓拍
  563. ce.CapPic((uint)lc[i], $"{AppConfigCache.path}/imgShort/formalImg/{AppConfigCache.pointNo}_C{strGuid}_tempImg_{i + 2}.jpg");
  564. }
  565. flagCaption = true;
  566. }
  567. if (lockWgtImg != CacleCls.lockWgt)
  568. {
  569. lockWgtImg = CacleCls.lockWgt;
  570. foreach (Control cn in panel1.Controls)
  571. {
  572. if (cn.Name.Substring(cn.Name.Length - 1) == $"{AppConfigCache.channel}")
  573. {
  574. if (cn is PictureBox)
  575. {
  576. Bitmap bit = new Bitmap(cn.Width, cn.Height);//实例化一个和窗体一样大的bitmap
  577. Graphics g = Graphics.FromImage(bit);
  578. g.CompositingQuality = CompositingQuality.HighQuality;//质量设为最高 HighQuality
  579. g.CopyFromScreen(panel1.Left, panel1.Top + 25, 0, 0, new Size(cn.Width, cn.Height));//保存整个窗体为图片
  580. bit.Save($"{AppConfigCache.path}/imgShort/formalImg/{AppConfigCache.pointNo}_C{strGuid}_tempImg_1.jpg");//默认保存格式为PNG,保存成jpg格式质量不是很好
  581. bit.Dispose();
  582. break;
  583. }
  584. }
  585. }
  586. #region
  587. /* //下面的代码可实现抓放大的图,不过我这为了时间更准,抓的左侧显示的图
  588. if (iPic != 0)
  589. {
  590. ce.StopRealPlay(iPic);
  591. }
  592. ce.RealPlay(pictureShow, AppConfigCache.channel);
  593. plImgShow.Visible = true;
  594. Thread.Sleep(200);
  595. Bitmap bit = new Bitmap(plImgShow.Width, plImgShow.Height);//实例化一个和窗体一样大的bitmap
  596. Graphics g = Graphics.FromImage(bit);
  597. g.CompositingQuality = CompositingQuality.HighQuality;//质量设为最高 HighQuality
  598. g.CopyFromScreen(plImgShow.Left, plImgShow.Top + 25, 0, 0, new Size(plImgShow.Width, plImgShow.Height));//保存整个窗体为图片
  599. bit.Save($"{AppConfigCache.path}/imgShort/formalImg/{AppConfigCache.pointNo}_C{strGuid}_tempImg_1.jpg");//默认保存格式为PNG,保存成jpg格式质量不是很好
  600. bit.Dispose();
  601. plImgShow.Visible = false;
  602. ce.StopRealPlay(AppConfigCache.channel);
  603. //*/
  604. #endregion
  605. }
  606. #endregion
  607. if (CacleCls.serverFlag) //网络正常
  608. {
  609. //没有接管的情况下
  610. if (!CacleCls.isJg && CacleCls.firstDb != null && Math.Abs(CacleCls.firstDb.meterWeight.Value - AppConfigCache.differenceWgt) < AppConfigCache.differenceWgt)
  611. {
  612. string str = "第一次计量重量【" + CacleCls.firstDb.meterWeight.Value / 1000 + "T】和第二次计量重量【" + CacleCls.lockWgt / 1000 + "T】接近,小于结净最小值【" + AppConfigCache.differenceWgt / 1000 + " T】";
  613. vicPlayClass.GetVoicePlay(VoiceEnum.两次计量重量接近, CacleCls.lockCarNo);
  614. led_controler.setStaticLineMsg(AppConfigCache.ledIp, $"上次重量与当,前重量接近,需操作工确认");
  615. txtMsgInfo.Text = str;
  616. lg.WriteLog(LogType.serverLog, str);
  617. CacleCls.isLock = false;
  618. return;
  619. }
  620. //一次计量
  621. if (CacleCls.firstDb == null)
  622. {
  623. //进行计量,不管有没有一次计量数据,都调用这个,如果服务端判断有一次计量数据则进行结净,没有则做一次计量操作
  624. RESTfulResult<string> rmRst = workCarActualFirstService.doAddFirstDjDb(CacleCls.preTrackScale, new MeterWorkCarActualFirst
  625. {
  626. baseSpotNo = AppConfigCache.pointNo,
  627. baseSpotName = AppConfigCache.pointName,
  628. carNo = CacleCls.lockCarNo,
  629. createTime = DateTime.Now,
  630. meterWeight = CacleCls.lockWgt,
  631. createManNo = "system",
  632. createManName = "system",
  633. valueFlag = "1",
  634. meterMode = "2",
  635. isPreScale = "0",
  636. dataSource = "1",
  637. checkFlag = "1"
  638. });
  639. if (rmRst.Succeed)
  640. {
  641. isMeasureSuccess = true;
  642. CacleCls.actualNo = rmRst.Data;
  643. CacleCls.tempNo = strGuid;
  644. }
  645. else
  646. {
  647. //计量失败,播报失败,然后写led屏幕及日志
  648. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "计量失败,请联系管理员");
  649. vicPlayClass.GetVoicePlay(VoiceEnum.程序处理异常, CacleCls.lockCarNo);
  650. txtMsgInfo.Text = "计量失败,请求服务异常:" + rmRst.ResultMessage;
  651. lg.WriteLog(LogType.serverLog, "计量失败,请求服务异常:" + rmRst.ResultMessage);
  652. CacleCls.isLock = false;
  653. return;
  654. }
  655. }
  656. else //结净操作
  657. {
  658. RESTfulResult<string> rmRst = workCarActualFirstService.doAddNet(CacleCls.preTrackScale, CacleCls.firstDb, new MeterWorkCarActualFirst
  659. {
  660. baseSpotNo = AppConfigCache.pointNo,
  661. baseSpotName = AppConfigCache.pointName,
  662. carNo = CacleCls.lockCarNo,
  663. createTime = DateTime.Now,
  664. meterWeight = CacleCls.lockWgt,
  665. createManNo = "system",
  666. createManName = "system",
  667. valueFlag = "1",
  668. meterMode = "2",
  669. isPreScale = "0",
  670. dataSource = "1",
  671. checkFlag = "1"
  672. });
  673. if (rmRst.Succeed)
  674. {
  675. isMeasureSuccess = true;
  676. CacleCls.actualNo = rmRst.Data;
  677. CacleCls.tempNo = strGuid;
  678. }
  679. else
  680. {
  681. //计量失败,播报失败,然后写led屏幕及日志
  682. vicPlayClass.GetVoicePlay(VoiceEnum.程序处理异常, CacleCls.lockCarNo);
  683. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "计量失败,请联系管理员");
  684. txtMsgInfo.Text = "计量失败,请求服务异常:" + rmRst.ResultMessage;
  685. lg.WriteLog(LogType.serverLog, "计量失败,请求服务异常:" + rmRst.ResultMessage);
  686. CacleCls.isLock = false;
  687. return;
  688. }
  689. }
  690. }
  691. else
  692. {
  693. //写本地文件
  694. bool flag = FileOption.WriterDb(new ActualFirstModel { actualFirstNo = strGuid, baseSpotNo = AppConfigCache.pointNo, baseSpotName = AppConfigCache.pointName, carNo = CacleCls.lockCarNo, createTime = DateTime.Now, meterWeight = CacleCls.lockWgt }, out msgInfo);
  695. if (flag)
  696. {
  697. isMeasureSuccess = true;
  698. }
  699. else
  700. {
  701. vicPlayClass.GetVoicePlay(VoiceEnum.程序处理异常, CacleCls.lockCarNo);
  702. led_controler.setStaticLineMsg(AppConfigCache.ledIp, "计量失败,请联系管理员");
  703. txtMsgInfo.Text = "计量失败写入,本地文件异常:" + msgInfo;
  704. lg.WriteLog(LogType.serverLog, "计量失败,写入本地文件异常:" + msgInfo);
  705. CacleCls.isLock = false;
  706. return;
  707. }
  708. }
  709. //计量成功
  710. if (isMeasureSuccess)
  711. {
  712. txtMsgInfo.Text = "计量完成";
  713. #region 毛皮净赋值
  714. if (CacleCls.firstDb != null && CacleCls.firstDb.meterWeight > 0 && CacleCls.lockWgt > 0)
  715. {
  716. if (CacleCls.firstDb.meterWeight > CacleCls.lockWgt)
  717. {
  718. txtGroess.Text = $"{CacleCls.firstDb.meterWeight / 1000}";
  719. txtTare.Text = $"{CacleCls.lockWgt / 1000}";
  720. txtNet.Text = $"{(CacleCls.firstDb.meterWeight - CacleCls.lockWgt) / 1000}";
  721. }
  722. else
  723. {
  724. txtGroess.Text = $"{CacleCls.lockWgt / 1000}";
  725. txtTare.Text = $"{CacleCls.firstDb.meterWeight / 1000}";
  726. txtNet.Text = $"{(CacleCls.lockWgt - CacleCls.firstDb.meterWeight) / 1000}";
  727. }
  728. }
  729. if (!string.IsNullOrEmpty(CacleCls.LEDResultMessage))
  730. {
  731. led_controler.setStaticLineMsg(AppConfigCache.ledIp, CacleCls.LEDResultMessage);
  732. }
  733. #endregion
  734. //屏幕截图操作
  735. _ = BeginInvoke(new Action(() => { shotImageScreen(strGuid); }));
  736. BindTable();
  737. //只需要控制plc即可,其他的不管
  738. if (AppConfigCache.plcStart == "1")
  739. {
  740. plc.DataCollectWrite(AppConfigCache.dz, AppConfigCache.dzValue);
  741. }
  742. }
  743. #endregion
  744. }
  745. catch (Exception ex)
  746. {
  747. vicPlayClass.GetVoicePlay(VoiceEnum.程序处理异常, CacleCls.lockCarNo);
  748. lg.WriteLog(LogType.SystemLog, "计量失败1:" + ex.Message);
  749. CacleCls.isLock = false;
  750. }
  751. }
  752. private void BindTable()
  753. {
  754. bool bReadFlag = false;
  755. string strRtMsgInfo = "";
  756. List<ActualFirstModel> lm = FileOption.ReadDb<ActualFirstModel>(ref bReadFlag, ref strRtMsgInfo);
  757. DataTable dtV = dataTableFirst.Clone();
  758. if (lm != null && lm.Count > 0)
  759. {
  760. dtV = lm.ListToDataTable<ActualFirstModel>();
  761. }
  762. ClsControlPack.CopyDataToDatatable(ref dtV, ref this.dataTableFirst, true);
  763. ClsControlPack.RefreshAndAutoSize(ultraGridFirst);
  764. }
  765. private void btnOpenVoice_Click(object sender, EventArgs e)
  766. {
  767. //*
  768. if (btnOpenVoice.Text == "打开对讲")
  769. {
  770. VoiceOpen();
  771. }
  772. else
  773. {
  774. VoiceClose();
  775. }
  776. //*/
  777. }
  778. private void frmMain_FormClosing(object sender, FormClosingEventArgs e)
  779. {
  780. stop();
  781. try { mx?.ClosingCollect(); } catch { }
  782. try { rfid?.ClosingCollect(); } catch { }
  783. try { if (bVoice) ce?.StopTalk(); } catch { }
  784. foreach (int i in voPlay)
  785. {
  786. try { ce?.StopRealPlay(i); } catch { }
  787. }
  788. try { plc?.Stop(); } catch { }
  789. try { ping?.stop(); } catch { }
  790. try { upload?.stop(); } catch { }
  791. try { carCls?.LoginOut(); } catch { }
  792. //try { carCls2?.LoginOut(); } catch { }
  793. try { imgControl?.Stop(); } catch { }
  794. }
  795. /// <summary>
  796. /// 必须先打开视频,然后再打开语音
  797. /// </summary>
  798. private void VoiceOpen()
  799. {
  800. try
  801. {
  802. if (ce.StartTalk())
  803. {
  804. btnOpenVoice.Text = "关闭对讲";
  805. bVoice = true;
  806. }
  807. }
  808. catch (Exception ex)
  809. {
  810. txtMsgInfo.Text = "打开对讲失败:" + ex.Message;
  811. }
  812. }
  813. /// <summary>
  814. /// 关闭语音对讲
  815. /// </summary>
  816. private void VoiceClose()
  817. {
  818. try
  819. {
  820. if (ce.StopTalk())
  821. {
  822. btnOpenVoice.Text = "打开对讲";
  823. bVoice = false;
  824. }
  825. }
  826. catch (Exception ex)
  827. {
  828. txtMsgInfo.Text = "关闭对讲失败:" + ex.Message;
  829. }
  830. }
  831. private void btnExport_Click(object sender, EventArgs e)
  832. {
  833. ClsControlPack.ExportDataWithSaveDialog2(ref ultraGridFirst, DateTime.Now.ToString("yyyyMMddHHmmss"));
  834. }
  835. private void cbCarNo_SelectedIndexChanged(object sender, EventArgs e)
  836. {
  837. CacleCls.isLock = false;
  838. }
  839. private void cbJg_CheckedChanged(object sender, EventArgs e)
  840. {
  841. CacleCls.isJg = cbJg.Checked;
  842. btnSave.Enabled = cbJg.Checked;
  843. }
  844. private void txtCarNo_TextChanged(object sender, EventArgs e)
  845. {
  846. CacleCls.isLock = false;
  847. }
  848. private void PictureBoxDoubleClick(object sender, EventArgs e)
  849. {
  850. if (iPic != 0)
  851. {
  852. ce.StopRealPlay(iPic);
  853. }
  854. PictureBox picture = (PictureBox)sender;
  855. iPic = Convert.ToInt32(picture.Name.Substring(picture.Name.Length - 1, 1));
  856. plImgShow.Visible = true;
  857. ce.RealPlay(pictureShow, iPic);
  858. }
  859. private void pictureShow_DoubleClick(object sender, EventArgs e)
  860. {
  861. plImgShow.Visible = false;
  862. ce.StopRealPlay(iPic);
  863. iPic = 0;
  864. }
  865. private void txtCarNo_KeyDown(object sender, KeyEventArgs e)
  866. {
  867. if (e.KeyCode == Keys.Enter) label3.Focus();
  868. }
  869. private void btnOpenDz_Click(object sender, EventArgs e)
  870. {
  871. if (AppConfigCache.plcStart == "1")
  872. {
  873. plc.DataCollectWrite(AppConfigCache.dz, AppConfigCache.dzValue);
  874. }
  875. else
  876. {
  877. MessageBox.Show("当前PLC处于未连接状态,请先配置连接再操作");
  878. }
  879. }
  880. /// <summary>
  881. /// 终端截图(临时截图,使用临时ID存储)
  882. /// </summary>
  883. public void shotImageScreen(string strActualFirstNo)
  884. {
  885. try
  886. {
  887. #region 截取图片信息
  888. //截取屏幕信息
  889. Point screenPoint = plCaptionMain.PointToScreen(new Point());
  890. Rectangle rect = new Rectangle(screenPoint, plCaptionMain.Size);
  891. Image img = new Bitmap(plCaptionMain.Width, plCaptionMain.Height);
  892. Graphics g = Graphics.FromImage(img);
  893. g.CopyFromScreen(rect.X - 1, rect.Y - 1, 0, 0, rect.Size);//"D://file/1.jpg"
  894. img.Save(string.Format("{0}imgShort\\formalImg\\{1}_{2}_tempImg_{3}.jpg",
  895. AppDomain.CurrentDomain.SetupInformation.ApplicationBase,
  896. AppConfigCache.pointNo, "C" + strActualFirstNo, 7), System.Drawing.Imaging.ImageFormat.Jpeg);
  897. //lg.WriteLog(35, "完成截图 成功标识:" + PbCache.shotSuccess.ToString() + "车号:" + PbCache.lockCarNo + ";重量:" + PbCache.lockWgt + ";时间:" + DateTime.Now.ToLongTimeString());
  898. #endregion 截取图片信息
  899. }
  900. catch (Exception ex)
  901. {
  902. lg.WriteLog(LogType.SystemLog, "计量完成,但图片截取失败,编号:" + strActualFirstNo);
  903. }
  904. }
  905. #region 释放内存
  906. [DllImport("kernel32.dll")]
  907. private static extern bool SetProcessWorkingSetSize(IntPtr process, int minSize, int maxSize);
  908. //刷新存储器
  909. private static void FlushMemory()
  910. {
  911. GC.Collect();
  912. GC.WaitForPendingFinalizers();
  913. if (Environment.OSVersion.Platform == PlatformID.Win32NT)
  914. {
  915. SetProcessWorkingSetSize(Process.GetCurrentProcess().Handle, -1, -1);
  916. }
  917. }
  918. #endregion
  919. }
  920. }