frmMain.cs 46 KB

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