using BPA.Message;
using BPA.Message.Enum;
using BPASmartClient.Helper;
using BPASmartClient.Message;
using BPASmartClient.Model;
using BPASmartClient.Peripheral;
using System;
using System.Collections.Concurrent;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Collections.ObjectModel;
using BPASmartClient.Model.单片机;
using BPASmartClient.EventBus;
using BPA.Models;
namespace BPASmartClient.Device
{
///
/// 设备基类
///
public abstract class BaseDevice : IDevice
{
public BaseDevice()
{
}
#region 属性
///
/// 订单物料信息
///
public OrderMaterialDelivery orderMaterialDelivery { get; set; } = new OrderMaterialDelivery();
///
/// 配方数据信息
///
public RecipeBoms recipeBoms { get; set; } = new RecipeBoms();
///
/// 设备ID
///
public int DeviceId { get; set; }
///
/// 设备所有状态
///
public DeviceStatus Status { get; set; } = new DeviceStatus();
///
/// 设备名称
///
public string Name { get; set; }
///
/// 当前订单数量
///
protected int OrderCount { get; set; }
///
/// 设备初始化中
///
protected bool Initing { get; set; }
///
/// 设备类型
///
public abstract DeviceClientType DeviceType { get; }
///
/// 是否忙碌
///
public bool IsBusy { get; protected set; }
///
/// 是否健康
///
public bool IsHealth { get; protected set; }
///
/// 设备运行日志
///
public List Log { get; set; } = new List();
///
/// 设备运行告警与错误
///
public List Error { get; set; } = new List();
///
/// mork_F暂用余量列表
///
public List BatchingInfos { get; set; } = new List();
///
/// 设备变量监控
///
public ObservableCollection variableMonitors { get; set; } = new ObservableCollection();
///
/// 外设状态,硬件设备数据
///
protected ConcurrentDictionary peripheralStatus = new ConcurrentDictionary();
protected
///
/// 外设设备集合
///
private List peripherals;
///
/// <炒锅>:<外设状态,硬件设备数据>的键值对
///
protected Dictionary> dicPort2peripheralStatus = new Dictionary>();
public Action AddErrorAction { get; set; }
public Action DeleteErrorAction { get; set; }
public List alarms { get; set; } = new List();
public IAlarm InterfaceAlarm { get; set; }
public AlarmHelper alarmHelper { get; set; } = new AlarmHelper();
public IStatus InterfaceStatus { get; set; }
public ObservableCollection variables { get; set; } = new ObservableCollection();
#endregion
///
/// 写控制
///
///
///
public void WriteControl(string address, object value)
{
if (peripherals != null)
{
for (int i = 0; i < peripherals.Count; i++)
{
peripherals.ElementAt(i).WriteData(address, value);
}
}
}
///
/// 多设备分开写控制
///
///
///
public void WriteControlExact(string address, object value, int i)
{
if (peripherals != null)
{
if (peripherals.Count > i)
{
peripherals.ElementAt(i).WriteData(address, value);
}
}
}
///
/// 设备过程日志显示
///
///
public void DeviceProcessLogShow(string info)
{
Log.Insert(0, new { Time = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"), Type = "流程", Text = info });
MessageLog.GetInstance.DeviceProcessLogShow(DeviceId.ToString(), info);
if (Log.Count > 100) { Log.RemoveAt(Log.Count - 1); }
}
public void Initliaze()
{
}
public void Initliaze(List peripherals)
{
peripherals.ForEach(p =>
{
p.DeviceId = this.DeviceId;
p.Init();
});
this.peripherals = peripherals;
}
public virtual void StartMain()
{
ThreadManage.GetInstance().StartLong(new Action(() =>
{
int i = 0;
foreach (var peripheral in peripherals)
{
string TypeName = peripheral.GetType().FullName.Replace("BPASmartClient.", "");
Status.Update($"{TypeName}.IsConnected", peripheral.IsConnected);
Status.Update($"{TypeName}.IsWork", peripheral.IsWork);
//做为炒锅与状态字典的新数据
ConcurrentDictionary newPeripheralStatus = new ConcurrentDictionary();
foreach (var key in peripheral.GetAllStatus().Keys)
{
peripheralStatus[key] = peripheral.GetAllStatus()[key];
//新的硬件设备数据存储
newPeripheralStatus[key] = peripheral.GetAllStatus()[key];
if (TypeName != "PLC.PLCMachine")
{
Status.Update($"{TypeName}.{key}", peripheral.GetAllStatus()[key]);
}
}
if (dicPort2peripheralStatus.ContainsKey(i))
{
dicPort2peripheralStatus[i] = newPeripheralStatus;
}
else
{
//将存储的新硬件设备数据放入字典中,i是作为炒锅编号。
dicPort2peripheralStatus.Add(i, newPeripheralStatus);
}
i++;
}
if (AddErrorAction != null && DeleteErrorAction != null)
{
foreach (var item in Status.GetStatusT())
{
if (item.Name == "Warning" || item.Name == "Fault")
{
if (item.Status != "无故障" && item.Status != "无警告" && item.Status != "未发生故障")
{
var res = Error?.FirstOrDefault(p => p.GetType().GetProperty("Text").GetValue(p).ToString() == item.Ms);
if (res == null)
{
object obj = new
{
Time = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
Type = item.Name == "Warning" ? "警告" : "故障",
Text = item.Ms
};
Error.Add(obj);
AddErrorAction?.Invoke(DeviceId, obj);
}
}
else
{
var res = Error?.FirstOrDefault(p => p.GetType().GetProperty("Text").GetValue(p).ToString().Contains(item.id));
if (res != null)
{
Error.Remove(res);
DeleteErrorAction?.Invoke(DeviceId, res);
}
}
}
}
}
Thread.Sleep(100);
}), $"GetAllStatus:{DeviceId}");
DoMain();
SimOrder();
GetGvlStatus();
InitResetTask();
InitTask();
}
private void ResetStatus()
{
this.GetType().GetFields(BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic).ToList().ForEach(item =>
{
var res = item.FieldType.GetInterfaces();
if (res != null)
{
foreach (var faces in res)
{
if (faces.Name == "IStatus") InterfaceStatus = item.GetValue(this) as IStatus;
}
}
});
}
private void GetGvlStatus()
{
this.GetType().GetFields(BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic).ToList().ForEach(item =>
{
var res = item.FieldType.GetInterfaces();
if (res != null)
{
foreach (var faces in res)
{
if (faces.Name == "IStatus")
{
InterfaceStatus = item.GetValue(this) as IStatus;
GetMonitorData(InterfaceStatus);
}
else if (faces.Name == "IAlarm")
{
InterfaceAlarm = item.GetValue(this) as IAlarm;
alarmHelper.AddAction += new Action((s) =>
{
var res = alarmHelper.Alarms.FirstOrDefault(p => p.Info == s);
if (res != null)
{
object obj = new
{
Time = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
Type = res.Grade,
Text = res.Info
};
Error.Insert(0, obj);
AddErrorAction?.Invoke(DeviceId, obj);
}
});
alarmHelper.RemoveAction += new Action((s) =>
{
var res = Error.FirstOrDefault(p => p.GetType().GetProperty("Text").GetValue(p).ToString() == s);
if (res != null && Error.Contains(res))
{
Error.Remove(res);
DeleteErrorAction?.Invoke(DeviceId, res);
}
});
ThreadManage.GetInstance().StartLong(new Action(() =>
{
AlarmMonitoring();
Thread.Sleep(500);
}), $"报警检测监控:{DeviceId}");
}
}
}
});
}
///
/// 报警监控
///
///
///
private void AlarmMonitoring()
{
if (InterfaceAlarm == null) return;
foreach (var item in InterfaceAlarm.GetType().GetProperties())
{
var res = item.GetValue(InterfaceAlarm);
if (res != null && res is bool blen)
{
if (item.CustomAttributes.Count() > 0 && item.CustomAttributes.ElementAt(0)?.ConstructorArguments.Count() > 0)
{
var info = item.GetCustomAttribute().AlarmInfo;
if (info != null) alarmHelper.EdgeAlarm(blen, $"{info.ToString()}-{DeviceId}");
}
}
}
}
private void InitResetTask()
{
#region 复位程序
ThreadManage.GetInstance().StartLong(new Action(() =>
{
if (RTrig.GetInstance($"ResetProgram:{DeviceId}").Start(Initing))
{
ThreadManage.GetInstance().StopTask($"MainTask:{DeviceId}", new Action(() =>
{
ThreadManage.GetInstance().StopTask($"ReadData:{DeviceId}", new Action(() =>
{
ThreadManage.GetInstance().StopTask($"GvlStatusMonitor:{DeviceId}", new Action(() =>
{
ActionManage.GetInstance.Send("ClearOrders");
ResetProgram();
ResetStatus();
InitTask();
}));
}));
}));
}
Thread.Sleep(10);
}), $"ResetProgram:{DeviceId}");
#endregion
}
private void InitTask()
{
#region 数据读取
ThreadManage.GetInstance().StartLong(new Action(() =>
{
ReadData();
Thread.Sleep(10);
}), $"ReadData:{DeviceId}", true);
#endregion
#region 任务流程
ThreadManage.GetInstance().StartLong(new Action(() =>
{
MainTask();
Thread.Sleep(10);
}), $"MainTask:{DeviceId}", true);
#endregion
#region 设备状态监控
ThreadManage.GetInstance().StartLong(new Action(() =>
{
UpdateValue(InterfaceStatus);
Thread.Sleep(1000);
}), $"GvlStatusMonitor:{DeviceId}");
#endregion
}
///
/// 获取监控信息
///
private void GetMonitorData(IStatus status)
{
if (status == null) return;
foreach (var item in status.GetType().GetProperties())
{
if (item.CustomAttributes.Count() > 0)
{
var attributeName = item.CustomAttributes.FirstOrDefault(p => p.AttributeType.Name == "VariableMonitorAttribute");
if (attributeName == null) return;
var plcadd = item.GetCustomAttribute()?.PLCAddress;
var modadd = item.GetCustomAttribute()?.ModbusTcpAddress;
var notes = item.GetCustomAttribute()?.Notes;
if (item.PropertyType?.BaseType?.Name == "Array")
{
if (plcadd?.Length > 0)
{
var arrayRes = item.GetValue(status, null);
if (arrayRes != null && arrayRes is Array arr)
{
for (int i = 0; i < arr.Length; i++)
{
var res = variableMonitors.FirstOrDefault(p => p.VarName == $"{item.Name}_{i + 1}");
if (res == null)
{
string[] plc = plcadd?.Substring(1).Split('.');
string TempPlcAddress = string.Empty;
if (plc?.Length == 2)
{
int add = int.Parse(plc[1]);
int firstAdd = int.Parse(plc[0]);
if (add >= 0 && add < 7)
{
add += i;
}
else if (add >= 7)
{
add = 0;
firstAdd++;
}
plc[0] = firstAdd.ToString();
plc[1] = add.ToString();
TempPlcAddress = $"M{plc[0]}.{plc[1]}";
}
variableMonitors.Add(new VariableMonitor()
{
Id = variableMonitors.Count,
VarName = $"{item.Name}_{i + 1}",
Notes = $"{notes}_{i + 1}",
ModbusTcpAddress = $"{int.Parse(modadd) + i}",
PLCAddress = TempPlcAddress,
});
}
}
}
}
else
{
var arrayRes = item.GetValue(status, null);
if (arrayRes != null && arrayRes is Array arr)
{
for (int i = 0; i < arr.Length; i++)
{
var res = variableMonitors.FirstOrDefault(p => p.VarName == $"{item.Name}_{i + 1}");
if (res == null)
{
variableMonitors.Add(new VariableMonitor()
{
Id = variableMonitors.Count,
VarName = $"{item.Name}_{i + 1}",
Notes = $"{notes}_{i + 1}",
});
}
}
}
}
}
else
{
var res = variableMonitors.FirstOrDefault(p => p.VarName == item.Name);
if (res == null)
{
variableMonitors.Add(new VariableMonitor()
{
Id = variableMonitors.Count,
VarName = item.Name,
Notes = notes,
ModbusTcpAddress = modadd,
PLCAddress = plcadd,
});
}
}
}
}
}
///
/// 更新数据
///
///
public void UpdateValue(IStatus status)
{
if (status == null) return;
foreach (var item in status.GetType().GetProperties())
{
if (item.CustomAttributes.Count() > 0)
{
if (item.PropertyType?.BaseType?.Name == "Array")
{
var arrayRes = item.GetValue(status);
if (arrayRes != null && arrayRes is Array arr)
{
for (int i = 0; i < arr.Length; i++)
{
int index = Array.FindIndex(variableMonitors.ToArray(), p => p.VarName == $"{item.Name}_{i + 1}");
if (index >= 0 && index < variableMonitors.Count)
{
variableMonitors.ElementAt(index).CurrentValue = arr.GetValue(i)?.ToString();
}
}
}
}
else
{
int index = Array.FindIndex(variableMonitors.ToArray(), p => p.VarName == item.Name);
if (index >= 0 && index < variableMonitors.Count)
{
variableMonitors.ElementAt(index).CurrentValue = item.GetValue(status)?.ToString();
}
}
}
}
}
public abstract void DoMain();
public abstract void Stop();
///
/// 数据读取
///
public abstract void ReadData();
///
/// 主流程控制
///
public abstract void MainTask();
///
/// 复位程序
///
public abstract void ResetProgram();
///
/// 模拟订单
///
public abstract void SimOrder();
public object GetError()
{
return new { data = Error };
}
public object GetLog()
{
return new { data = Log };
}
public object GetVariableMonitor()
{
return new { data = variableMonitors };
}
///
/// 获取某个对象中的属性值
///
///
///
///
public object GetPropertyValue(object info, string field)
{
if (info == null) return null;
Type t = info.GetType();
IEnumerable property = from pi in t.GetProperties() where pi.Name.ToLower() == field.ToLower() select pi;
return property.First().GetValue(info, null);
}
}
}