using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Diagnostics; using System.Linq; using System.Net.NetworkInformation; using System.Text; using System.Text.RegularExpressions; using System.Threading.Tasks; using System.Xml.Linq; using DH.Commons.Base; using DH.Commons.Enums; using DH.Commons.Models; using HslCommunication; using HslCommunication.Enthernet; using HslCommunication.Profinet.XINJE; using OpenCvSharp; namespace DH.Devices.PLC { public class XinJEPLCTcpNet : PLCBase { private static XinJEPLCTcpNet _instance; public static XinJEPLCTcpNet Instance { get { if (_instance == null) _instance = new XinJEPLCTcpNet(); return _instance; } } private XinJETcpNet TcpNet = new XinJETcpNet(); public override bool PLCConnect() { try { TcpNet.IpAddress = IP; TcpNet.Port = Port; TcpNet.ConnectTimeOut = 5000; TcpNet.ReceiveTimeOut = 10000; TcpNet.SleepTime = 0; TcpNet.AddressStartWithZero = true; TcpNet.IsStringReverse = false; TcpNet.Station = 1; switch (PLCType) { case EnumPLCType.信捷XD网口: TcpNet.Series = XinJESeries.XD; TcpNet.DataFormat = HslCommunication.Core.DataFormat.CDAB; break; case EnumPLCType.信捷XC网口: TcpNet.Series = XinJESeries.XC; TcpNet.DataFormat = HslCommunication.Core.DataFormat.CDAB; break; } OperateResult ret = TcpNet.ConnectServer(); if (ret.IsSuccess) { Connected = true; //初始化流程 InitProcess(); return true; } else { Connected = false; throw new Exception($"{IP}:{Port}PLC连接失败!"); } } catch(Exception ex) { Connected = false; throw new Exception($"{IP}:{Port}PLC连接失败!失败原因:{ex.ToString()}"); } } public override Int16 ReadInt16(string address) { try { if (Connected) { // 读取Int变量 var result = TcpNet.ReadInt16(address); if (result.IsSuccess) { return result.Content; } else { throw new Exception($"PLC操作读取int16失败,地址{address}"); } } else { throw new Exception($"PLC未连接,地址{address}"); } } catch (Exception ex) { throw new Exception($"PLC操作读取int16失败,地址{address},失败原因:{ex.ToString()}"); } } public override Int32 ReadInt32(string address) { try { if (Connected) { // 读取Int变量 var result = TcpNet.ReadInt32(address); if (result.IsSuccess) { return result.Content; } else { throw new Exception($"PLC操作读取int32失败,地址{address}"); } } else { throw new Exception($"PLC未连接,地址{address}"); } } catch (Exception ex) { throw new Exception($"PLC操作读取int32失败,地址{address},失败原因:{ex.ToString()}"); } } public override UInt16 ReadUInt16(string address) { try { if (Connected) { // 读取Int变量 var result = TcpNet.ReadUInt16(address); if (result.IsSuccess) { return result.Content; } else { throw new Exception($"PLC操作读取uint16失败,地址{address}"); } } else { throw new Exception($"PLC未连接,地址{address}"); } } catch (Exception ex) { throw new Exception($"PLC操作读取uint16失败,地址{address},失败原因:{ex.ToString()}"); } } public override UInt32 ReadUInt32(string address) { try { if (Connected) { // 读取Int变量 var result = TcpNet.ReadUInt32(address); if (result.IsSuccess) { return result.Content; } else { throw new Exception($"PLC操作uint32失败,地址{address}"); } } else { throw new Exception($"PLC未连接,地址{address}"); } } catch (Exception ex) { throw new Exception($"PLC操作读取uint32失败,地址{address},失败原因:{ex.ToString()}"); } } public override float ReadFloat(string address) { try { if (Connected) { // 读取Float变量 var result = TcpNet.ReadFloat(address); if (result.IsSuccess) { return result.Content; } else { throw new Exception($"PLC操作读取float失败,地址{address}"); } } else { throw new Exception($"PLC未连接,地址{address}"); } } catch (Exception ex) { throw new Exception($"PLC操作读取float失败,地址{address},失败原因:{ex.ToString()}"); } } public override bool ReadBool(string address) { try { if (Connected) { // 读取Bool变量 var result = TcpNet.ReadBool(address); if (result.IsSuccess) { return result.Content; } else { throw new Exception($"PLC操作读取bool失败,地址{address}"); } } else { throw new Exception($"PLC未连接,地址{address}"); } } catch (Exception ex) { throw new Exception($"PLC操作读取bool失败,地址{address},失败原因:{ex.ToString()}"); } } /// /// 写单独地址 uint32 值 /// /// 地址 /// 要写入的 int 值 /// 是否等待回复 public override bool WriteUInt32(string address, UInt32 writeValue, bool waitForReply = true) { if (Connected) { if (string.IsNullOrEmpty(address)) { return false; } int repeatTime = 3; do { try { var result = TcpNet.Write(address, writeValue); if (result.IsSuccess) { return true; } } catch (Exception ex) { repeatTime--; if (repeatTime <= 0) { throw new Exception($"PLC操作写入uint32失败,地址{address},失败原因:{ex.ToString()}"); } } } while (repeatTime > 0); } else { throw new Exception($"PLC未连接,地址{address}"); } return false; } /// /// 写单独地址 uint16 值 /// /// 地址 /// 要写入的 int 值 /// 是否等待回复 public override bool WriteUInt16(string address, UInt16 writeValue, bool waitForReply = true) { if (Connected) { if (string.IsNullOrEmpty(address)) { return false; } int repeatTime = 3; do { try { var result = TcpNet.Write(address, writeValue); if (result.IsSuccess) { return true; } } catch (Exception ex) { repeatTime--; if (repeatTime <= 0) { throw new Exception($"PLC操作写入uint16失败,地址{address},失败原因:{ex.ToString()}"); } } } while (repeatTime > 0); } else { throw new Exception($"PLC未连接,地址{address}"); } return false; } /// /// 写单独地址 int32 值 /// /// 地址 /// 要写入的 int 值 /// 是否等待回复 public override bool WriteInt32(string address, Int32 writeValue, bool waitForReply = true) { if (Connected) { if (string.IsNullOrEmpty(address)) { return false; } int repeatTime = 3; do { try { var result = TcpNet.Write(address, writeValue); if (result.IsSuccess) { return true; } } catch (Exception ex) { repeatTime--; if (repeatTime <= 0) { throw new Exception($"PLC操作写入int32失败,地址{address},失败原因:{ex.ToString()}"); } } } while (repeatTime > 0); } else { throw new Exception($"PLC未连接,地址{address}"); } return false; } /// /// 写单独地址 int16 值 /// /// 地址 /// 要写入的 int 值 /// 是否等待回复 public override bool WriteInt16(string address, Int16 writeValue, bool waitForReply = true) { if (Connected) { if (string.IsNullOrEmpty(address)) { return false; } int repeatTime = 3; do { try { var result = TcpNet.Write(address, writeValue); if (result.IsSuccess) { return true; } } catch (Exception ex) { repeatTime--; if (repeatTime <= 0) { throw new Exception($"PLC操作写入int16失败,地址{address},失败原因:{ex.ToString()}"); } } } while (repeatTime > 0); } else { throw new Exception($"PLC未连接,地址{address}"); } return false; } /// /// 写单独地址 float 值 /// /// 地址 /// 要写入的 float 值 /// 是否等待回复 public override bool WriteFloat(string address, float writeValue, bool waitForReply = true) { if (Connected) { if (string.IsNullOrEmpty(address)) { return false; } int repeatTime = 3; do { try { var result = TcpNet.Write(address, writeValue); if (result.IsSuccess) { return true; } } catch (Exception ex) { repeatTime--; if (repeatTime <= 0) { throw new Exception($"PLC操作写入float失败,地址{address},失败原因:{ex.ToString()}"); } } } while (repeatTime > 0); } else { throw new Exception($"PLC未连接,地址{address}"); } return false; } /// /// 写单独地址 bool 值 /// /// 地址 /// 要写入的 bool 值 /// 是否等待回复 public override bool WriteBool(string address, bool writeValue, bool waitForReply = true) { if(Connected) { if (string.IsNullOrEmpty(address)) { return false; } int repeatTime = 3; do { try { var result = TcpNet.Write(address, writeValue); if (result.IsSuccess) { return true; } } catch (Exception ex) { repeatTime--; if (repeatTime <= 0) { throw new Exception($"PLC操作写入bool失败,地址{address},失败原因:{ex.ToString()}"); } } } while (repeatTime > 0); Thread.Sleep(30); } else { throw new Exception($"PLC未连接,地址{address}"); } return false; } public override bool PLCDisConnect() { if (Connected) { var res = TcpNet.ConnectClose(); if (res.IsSuccess) { Connected = false; return true; } return false; } return false; } private void MonitorPieces() { ThreadStart ts = new ThreadStart(MonitorPiecesImpl); Thread th = new Thread(ts); th.Priority = ThreadPriority.AboveNormal; th.IsBackground = false; th.Start(); } public TaskFactory _taskFactory = new TaskFactory(TaskCreationOptions.LongRunning, TaskContinuationOptions.LongRunning); private Dictionary piecesCountDic = new Dictionary(); private uint piecesCount = 0; /// /// int,int 轴号 捕获位置 /// public event Action OnNewPieces; public void NewPieces(int axisIndex, uint pieceNumber) { _taskFactory.StartNew(() => { Thread.CurrentThread.Priority = ThreadPriority.Highest; OnNewPieces?.Invoke(axisIndex, pieceNumber); }); } public async Task HeartbeatAsync1() { PLCItem? pLCItem = PLCItemList.FirstOrDefault(u => u.Name == "心跳地址"); while (Connected) { if (pLCItem == null) return; WriteBool(pLCItem.Address, true); await Task.Delay(900); // 非阻塞,等待1秒 } } /// /// 入料监听 /// /// private void MonitorPiecesImpl() { PLCItem pLCItem= PLCItemList.FirstOrDefault(u => u.Name == "产品计数"); if (pLCItem == null) return; string Count = pLCItem.Address; DateTime startTime = DateTime.Now; DateTime endTime = startTime; TimeSpan timeSpan = endTime - startTime; Thread.CurrentThread.Priority = ThreadPriority.AboveNormal; //while (CurrentState != DeviceState.DSClose && CurrentState != DeviceState.DSExcept && CurrentState != DeviceState.DSUninit) while (Connected) { Stopwatch sw = new Stopwatch(); uint tmpPieceNumber = 0; sw.Start(); // var ret = TcpNet.ReadUInt16("D1016"); var ret = TcpNet.ReadUInt32(Count); sw.Stop(); if (ret.IsSuccess) { tmpPieceNumber = ret.Content; } if (ret.IsSuccess && ret.Content > piecesCount) { sw.Start(); // Console.WriteLine($"{DateTime.Now.ToString("HH:mm:ss.fff")} 板卡{station}产品入列触发{tmpPieceNumber}"); //LogAsync(DateTime.Now, LogLevel.Information, $"转盘{0}产品入列 {piecesCountDic[0]} size:{sum}"); if (tmpPieceNumber != piecesCount + 1) { //LogAsync(DateTime.Now, LogLevel.Information, $"入列触发丢失\t{tmpPieceNumber}"); // Console.WriteLine($"{DateTime.Now.ToString("HH:mm:ss.fff")}\t板卡{station}产品入列触发丢失,{piecesCountDic[station]}\t{tmpPieceNumber}"); } piecesCount = tmpPieceNumber; //NewPieces(ai, piecesCountDic[station]); NewPieces(1, piecesCount); sw.Stop(); startTime = DateTime.Now; //if (idalarm) //{ // idalarm = false; // AlarmVibration(false); //} } Thread.Sleep(1); } } public void InitProcess() { //启用心跳 OpenHeartbeat(true); //状态复位 StatusReset(); //关闭定位 VisionPos(false); //写入流程加载点位配置 InitProcessAction(); //计数清零 CountToZero(); //停止转盘 TurnStart(false); //转盘使能 TurnEnable(true); //开启入料监听 MonitorPieces(); } public void StartProcess() { //状态复位 StatusReset(); //关闭定位 VisionPos(false); //写入流程启动点位配置 StartProcessAction(); //计数清零 CountToZero(); //转盘启动 TurnStart(true); } public void CloseProcess() { StatusReset(); VisionPos(false); CountToZero(); TurnStart(false); TurnEnable(false); OpenHeartbeat(false); PLCDisConnect(); } public void InitProcessAction() => ProcessAction(ConfigModel.GlobalList?.FirstOrDefault()?.InitProcessList?.ToList() ?? new List()); public void StartProcessAction() => ProcessAction(ConfigModel.GlobalList?.FirstOrDefault()?.StartProcessList?.ToList() ?? new List()); public void StopProcessAction() => ProcessAction(ConfigModel.GlobalList?.FirstOrDefault()?.StopProcessList?.ToList() ?? new List()); public void StartResetAction() => ProcessAction(ConfigModel.GlobalList?.FirstOrDefault()?.StartResetList?.ToList() ?? new List()); public void StopResetAction() => ProcessAction(ConfigModel.GlobalList?.FirstOrDefault()?.StopResetList?.ToList() ?? new List()); // 处理方法:无需再检查 null private void ProcessAction(List items) { if (items.Count == 0) return; foreach (var item in items.OrderBy(x => x.StartIndex)) { switch (item.Type) { case EnumPLCDataType.单字整型: WriteUInt16(item.Address, ushort.Parse(item.Value)); break; case EnumPLCDataType.双字整型: WriteUInt32(item.Address, uint.Parse(item.Value)); break; case EnumPLCDataType.浮点型: WriteFloat(item.Address, float.Parse(item.Value)); break; case EnumPLCDataType.布尔型: WriteBool(item.Address, item.Value.Equals("true", StringComparison.OrdinalIgnoreCase)); break; } } } /// /// 状态复位 /// public void StatusReset() { PLCItem? pLCItem = PLCItemList.FirstOrDefault(u => u.Name == "状态复位"); if (pLCItem == null) return; WriteBool(pLCItem.Address, false); } /// /// 开启关闭视觉定位 /// public void VisionPos(bool b) { PLCItem? pLCItem = PLCItemList.FirstOrDefault(u => u.Name == "启用定位"); if (pLCItem == null) return; WriteBool(pLCItem.Address, b); } /// /// 转盘启停 /// public void TurnStart(bool b) { PLCItem? DiskRunItem = PLCItemList.FirstOrDefault(u => u.Name == "转盘启动"); if (DiskRunItem == null) return; WriteBool(DiskRunItem.Address, b); Thread.Sleep(30); piecesCount = 0; } /// /// 转盘使能 /// /// public void TurnEnable(bool b) { PLCItem? DiskOpenItem = PLCItemList.FirstOrDefault(u => u.Name == "转盘使能"); if (DiskOpenItem == null) return; WriteBool(DiskOpenItem.Address, b); Thread.Sleep(30); } /// /// 计数清零 /// public void CountToZero() { PLCItem? pLCItem = PLCItemList.FirstOrDefault(u => u.Name == "计数清零"); if (pLCItem == null) return; WriteBool(pLCItem.Address, true); Thread.Sleep(30); } public void RedLight(bool b) { PLCItem? pLCItem = PLCItemList.FirstOrDefault(u => u.Name == "指示灯红"); if (pLCItem == null) return; WriteBool(pLCItem.Address, b); Thread.Sleep(30); } public void GreenLight(bool b) { PLCItem? pLCItem = PLCItemList.FirstOrDefault(u => u.Name == "指示灯绿"); if (pLCItem == null) return; WriteBool(pLCItem.Address, b); Thread.Sleep(30); } public void YellowLight(bool b) { PLCItem? pLCItem = PLCItemList.FirstOrDefault(u => u.Name == "指示灯黄"); if (pLCItem == null) return; WriteBool(pLCItem.Address, b); Thread.Sleep(30); } public void Buzzer(bool b) { PLCItem? pLCItem = PLCItemList.FirstOrDefault(u => u.Name == "蜂鸣器"); if (pLCItem == null) return; WriteBool(pLCItem.Address, b); Thread.Sleep(30); } /// /// 挡料电机操作 /// true: 顺时针 /// False: 逆时针 /// /// public void FeedingMotor( bool direction) { // 设置最大等待时间,假设为 3 秒 int timeout = 3000; int elapsedTime = 0; int checkInterval = 100; // 每次检查等待 100ms PLCItem pLCItem = PLCItemList.FirstOrDefault(u => u.Name == "挡料电机回原点"); if (pLCItem == null) return; PLCItem zerospeeditem = PLCItemList.FirstOrDefault(u => u.Name == "挡料电机回原点速度"); if (zerospeeditem == null) return; PLCItem CunSpeed = PLCItemList.FirstOrDefault(u => u.Name == "挡料电机速度"); if (CunSpeed == null) return; PLCItem CunClockwiseItem = PLCItemList.FirstOrDefault(u => u.Name == "挡料电机顺时针"); if (CunClockwiseItem == null) return; PLCItem CunCounterclockwiseItem = PLCItemList.FirstOrDefault(u => u.Name == "挡料电机逆时针"); if (CunCounterclockwiseItem == null) return; PLCItem CunPosItem = PLCItemList.FirstOrDefault(u => u.Name == "挡料电机位置"); if (CunPosItem == null) return; string CunToZero = pLCItem.Type + pLCItem.Address; string CunToZeroSpeed = zerospeeditem.Type + zerospeeditem.Address; string CunSpeedadress = CunSpeed.Type + CunSpeed.Address; string CunClockwise = CunClockwiseItem.Type + CunClockwiseItem.Address; string CunCounterclockwise = CunCounterclockwiseItem.Type + CunCounterclockwiseItem.Address; string CunPos = CunPosItem.Type + CunPosItem.Address; UInt16 zerospeed = UInt16.Parse(zerospeeditem.Value); UInt16 cunSpeed = UInt16.Parse(CunSpeed.Value); UInt16 u = UInt16.Parse(CunPosItem.Value); // WriteBool(CountToZero, true); // 检查是否不在原点,如果不在,则回原点 if (!ReadBool(CunToZero)) { WriteUInt16(CunToZeroSpeed, zerospeed); // 速度 Thread.Sleep(30); // 发送回原点指令 WriteBool(CunToZero, true); Thread.Sleep(1000); // 给设备一些时间响应 // 等待回到原点 while (!ReadBool(CunToZero)) { if (elapsedTime >= timeout) { break; } Thread.Sleep(checkInterval); elapsedTime += checkInterval; } } // 无论是刚回到原点还是已经在原点,执行目标位置、速度和方向设置 WriteUInt16(CunSpeedadress,cunSpeed); Thread.Sleep(2000); string dir = string.Empty; if (direction) { WriteBool(CunClockwise, true); // 顺时针转动 dir = "顺时针"; } else { WriteBool(CunCounterclockwise, true); // 逆时针转动 dir = "逆时针"; } Thread.Sleep(30); WriteUInt16(CunPos, u); // 目标位置 Thread.Sleep(2000); } /// /// 转盘清料 /// /// public void TurnClear(bool b) { PLCItem? pLCItem = PLCItemList.FirstOrDefault(u => u.Name == "转盘清料"); if (pLCItem == null) return; WriteBool(pLCItem.Address, b); Thread.Sleep(30); } /// /// 开启心跳 /// /// public void OpenHeartbeat(bool b) { PLCItem? pLCItem = PLCItemList.FirstOrDefault(u => u.Name == "启用心跳"); if (pLCItem == null) return; WriteBool(pLCItem.Address, b); Thread.Sleep(30); if (b) { //开启心跳 Task.Run(async () => await HeartbeatAsync1()); } } int currentRegister = 0; public void Blowing(uint productNumber, UInt16 value) { int Register = (int)((productNumber - 1) % 30); currentRegister = 411 + Register; WriteUInt16($"D{currentRegister}", value); } } }