frmMain.cs 48 KB

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