frmMain.cs 47 KB

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