| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999 | using Aitex.Core.RT.DataCenter;using Aitex.Core.RT.Device;using Aitex.Core.RT.Log;using Aitex.Core.RT.OperationCenter;using Aitex.Core.RT.Routine;using Aitex.Core.UI.Control;using Aitex.Core.Util;using MECF.Framework.Common.Beckhoff.IOAxis;using MECF.Framework.Common.Beckhoff.ModuleIO;using MECF.Framework.Common.CommonData;using MECF.Framework.Common.CommonData.Metal;using MECF.Framework.Common.Persistent.Reservoirs;using MECF.Framework.Common.TwinCat;using CyberX8_Core;using CyberX8_RT.Devices.Prewet;using System;using System.Collections.Generic;using System.Globalization;using System.Linq;using System.Reflection;using System.Text;using System.Threading.Tasks;using MECF.Framework.Common.ToolLayout;using CyberX8_RT.Devices.Reservoir;using MECF.Framework.Common.Layout;using Aitex.Core.RT.SCCore;using System.Diagnostics;using MECF.Framework.Common.IOCore;namespace CyberX8_RT.Devices.Metal{    public class CompactMembranMetalDevice : MetalCellDevice    {        private enum ANCellStatus        {            Unknow,            Filling,            Emptying,            Full,            Empty        }        private enum ANCellOperation        {            None,            Fill,            Drain        }        #region 常量         private const string SIDE_A = "SideA";        private const string SIDE_B = "SideB";        private const string AN_A_PINENABLE="ANAPinEnable";        private const string AN_B_PINENABLE = "ANBPinEnable";        private const string AN_A_POUTENABLE = "ANAPoutEnable";        private const string AN_B_POUTENABLE = "ANBPoutEnable";        private const string CELL_FLOW = "CellFlow";        private const string WH_CLAMP = "WHClamp";        private const string WH_UNCLAMP = "WHUnclamp";        private const string CELL_FLOW_VALVE = "CellFlowValve";        private const string COUNTER_VALUE = "CounterValue";        private const string COUNTER_START = "Start";        private const string COUNTER_STOP = "Stop";        private const string COUNTER_RESET = "Reset";        private const string AN_A_CELL_FLOW = "ANACellFlow";        private const string AN_B_CELL_FLOW = "ANBCellFlow";        #endregion        #region 内部变量        /// <summary>        /// 设备数据        /// </summary>        private CompactMembranMetalDeviceData _metalDeviceData =new CompactMembranMetalDeviceData();        /// <summary>        /// 变量是否初始化字典        /// </summary>        private Dictionary<string, bool> _variableInitializeDic = new Dictionary<string, bool>();        /// <summary>        /// 阳极A面cellFlow        /// </summary>        private CounterFlowData _anACellFlow = new CounterFlowData();        /// <summary>        /// 阳极B面cellFlow        /// </summary>        private CounterFlowData _anBCellFlow = new CounterFlowData();        /// <summary>        /// Counter字典        /// </summary>        private Dictionary<string, CounterFlowData> _nameCounterFlowData = new Dictionary<string, CounterFlowData>();        /// <summary>        /// 当前阳极A面操作        /// </summary>        private ANCellOperation _currentANACellOperation;        /// <summary>        /// 当前阳机B面操作        /// </summary>        private ANCellOperation _currentANBCellOperation;        /// <summary>        /// AN Fill Routine        /// </summary>        private CompactMembranFillRoutine _anANACellFillRoutine;        /// <summary>        /// 阳机B面Fill Routine        /// </summary>        private CompactMembranFillRoutine _anANBCellFillRoutine;        /// <summary>        /// 阳极A面流量总数值         /// </summary>        private double _anATotalFillFlow = 0;        /// <summary>        /// 阳极B面流量总数值         /// </summary>        private double _anBTotalFillFlow = 0;        /// <summary>        /// 阳极A面Drain时间        /// </summary>        private DateTime _anADrainTime = DateTime.Now;        /// <summary>        /// 阳极B面Drain时间        /// </summary>        private DateTime _anBDrainTime = DateTime.Now;        /// <summary>        /// Flow Valve计时        /// </summary>        private Stopwatch _flowValveStopWatch = new Stopwatch();        #endregion        #region 属性        /// <summary>        /// 设备数据        /// </summary>        public CompactMembranMetalDeviceData MetalDeviceData { get { return _metalDeviceData; } }        /// <summary>        /// 阳极A面CellFlow        /// </summary>        public CounterFlowData ANACellFlow { get { return _anACellFlow; } }        /// <summary>        /// 阳极B面CellFlow        /// </summary>        public CounterFlowData ANBCellFlow { get { return _anBCellFlow; } }        /// <summary>        /// Flow Valve稳定状态        /// </summary>        public bool FlowValveStable { get { return _metalDeviceData.CellFlowValve && _flowValveStopWatch.ElapsedMilliseconds >= 3000; } }        #endregion        /// <summary>        /// 构造函数        /// </summary>        /// <param name="moduleName"></param>        public CompactMembranMetalDevice(string moduleName) : base(moduleName)        {        }                /// <summary>        /// 初始化        /// </summary>        /// <returns></returns>        public override bool Initialize()        {            base.Initialize();            InitializeRoutine();            SubscribeValueAction();            SubscribeData();            return true;        }        /// <summary>        /// 初始化Routine        /// </summary>        private void InitializeRoutine()        {            _anANACellFillRoutine = new CompactMembranFillRoutine(Module,"A");            _anANBCellFillRoutine = new CompactMembranFillRoutine(Module,"B");        }        /// <summary>        /// 订阅数据        /// </summary>        private void SubscribeData()        {            DATA.Subscribe($"{Module}.MetalData", () => _metalDeviceData, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.SideAFlow",()=>ANACellFlow.CounterValue, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.SideAFlowStatus", () => ANACellFlow.Status, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.SideBFlow", () => ANBCellFlow.CounterValue, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.SideBFlowStatus",()=>ANBCellFlow.Status, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.CellFlow",()=>MetalDeviceData.CellFlow, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.ANBPinEnable",()=>_metalDeviceData.ANBPinEnable, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.ANAPinEnable", () => _metalDeviceData.ANAPinEnable, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.ANAPoutEnable", () => _metalDeviceData.ANAPoutEnable, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.ANBPoutEnable", () => _metalDeviceData.ANBPoutEnable, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.CellFlowEnable", () => _metalDeviceData.CellFlowValve, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.WaferShuttleClamped", () => _metalDeviceData.WHClamp, SubscriptionAttribute.FLAG.IgnoreSaveDB);            DATA.Subscribe($"{Module}.WaferShuttleUnclamped", () => _metalDeviceData.WHUnclamp, SubscriptionAttribute.FLAG.IgnoreSaveDB);        }        /// <summary>        /// 订阅变量数值发生变化        /// </summary>        private void SubscribeValueAction()        {            BeckhoffIoSubscribeUpdateVariable(AN_A_PINENABLE);            BeckhoffIoSubscribeUpdateVariable(AN_B_PINENABLE);            BeckhoffIoSubscribeUpdateVariable(AN_A_POUTENABLE);            BeckhoffIoSubscribeUpdateVariable(AN_B_POUTENABLE);            BeckhoffIoSubscribeUpdateVariable(CELL_FLOW);            BeckhoffIoSubscribeUpdateVariable(WH_CLAMP);            BeckhoffIoSubscribeUpdateVariable(WH_UNCLAMP);            BeckhoffIoSubscribeUpdateVariable(CELL_FLOW_VALVE);            BeckhoffCounterSubscribeUpdateVariable(AN_A_CELL_FLOW,ANACellFlow);            BeckhoffCounter anACellFlowCounter = BeckhoffCounterManager.Instance.GetBeckhoffCounter($"{Module}.{AN_A_CELL_FLOW}");            if(anACellFlowCounter!=null)            {                ANACellFlow.Period = anACellFlowCounter.Period;            }            BeckhoffCounterSubscribeUpdateVariable(AN_B_CELL_FLOW,ANBCellFlow);            BeckhoffCounter anBCellFlowCounter = BeckhoffCounterManager.Instance.GetBeckhoffCounter($"{Module}.{AN_B_CELL_FLOW}");            if (anACellFlowCounter != null)            {                ANBCellFlow.Period = anBCellFlowCounter.Period;            }        }        /// <summary>        /// 订阅Operation        /// </summary>        protected override void InitializeOperation()        {            base.InitializeOperation();            OP.Subscribe($"{Module}.ANAFillOn",AnSideAFillOn);            OP.Subscribe($"{Module}.ANAFillOff", AnSideAFillOff);            OP.Subscribe($"{Module}.ANBFillOn", AnSideBFillOn);            OP.Subscribe($"{Module}.ANBFillOff", AnSideBFillOff);            OP.Subscribe($"{Module}.ANADrainOn", AnSideADrainOn);            OP.Subscribe($"{Module}.ANADrainOff", AnSideADrainOff);            OP.Subscribe($"{Module}.ANBDrainOn", AnSideBDrainOn);            OP.Subscribe($"{Module}.ANBDrainOff", AnSideBDrainOff);            OP.Subscribe($"{Module}.WHClampOn", WHClampOn);            OP.Subscribe($"{Module}.WHClampOff", WHClampOff);            OP.Subscribe($"{Module}.WHUnclampOn", WHUnClampOn);            OP.Subscribe($"{Module}.WHUnclampOff", WHUnClampOff);            OP.Subscribe($"{Module}.CellFlowValveOn", CellFlowValveOn);            OP.Subscribe($"{Module}.CellFlowValveOff", CellFlowValveOff);            OP.Subscribe($"{Module}.WaferHolderClampOn", WaferHolderClampOn);            OP.Subscribe($"{Module}.WaferHolderUnclampOn", WaferHolderUnclampOn);            OP.Subscribe($"{Module}.WaferHolderDisconnect", WaferHolderDisConnect);            OP.Subscribe($"{Module}.A.Fill", ANSideAFillOperation);            OP.Subscribe($"{Module}.B.Fill", ANSideBFillOperation);            OP.Subscribe($"{Module}.A.Stop", ANSideAStopOperation);            OP.Subscribe($"{Module}.B.Stop", ANSideBStopOperation);            OP.Subscribe($"{Module}.A.Drain", ANSideADrainOperation);            OP.Subscribe($"{Module}.B.Drain", ANSideBDrainOperation);        }        /// <summary>        /// 订阅IO变量        /// </summary>        /// <param name="variable"></param>        private void BeckhoffIoSubscribeUpdateVariable(string variable)        {            _variableInitializeDic[variable] = false;            IOModuleManager.Instance.SubscribeModuleVariable(Module, variable, UpdateVariableValue);        }        /// <summary>        /// 更新变量数值        /// </summary>        /// <param name="variable"></param>        /// <param name="value"></param>        private void UpdateVariableValue(string variable, object value)        {            if (!_metalDeviceData.IsDataInitialized)            {                _metalDeviceData.IsDataInitialized = true;            }            PropertyInfo property = _metalDeviceData.GetType().GetProperty(variable);            if (property != null)            {                property.SetValue(_metalDeviceData, value);            }            if (_variableInitializeDic.ContainsKey(variable) && !_variableInitializeDic[variable])            {                _variableInitializeDic[variable] = true;            }            if (variable == CELL_FLOW_VALVE)            {                bool bValue = (bool)value;                if (bValue)                {                    _flowValveStopWatch.Restart();                }            }        }        /// <summary>        /// 订阅Counter变量        /// </summary>        /// <param name="variable"></param>        private void BeckhoffCounterSubscribeUpdateVariable(string variable,CounterFlowData counterFlowData)        {            _nameCounterFlowData[$"{Module}.{variable}"] = counterFlowData;            BeckhoffCounterManager.Instance.SubscribeModuleVariable($"{Module}.{variable}", COUNTER_VALUE, UpdateCounterVariableValue);            BeckhoffCounterManager.Instance.SubscribeModuleVariable($"{Module}.{variable}", COUNTER_START, UpdateCounterVariableValue);            BeckhoffCounterManager.Instance.SubscribeModuleVariable($"{Module}.{variable}", COUNTER_STOP, UpdateCounterVariableValue);            BeckhoffCounterManager.Instance.SubscribeModuleVariable($"{Module}.{variable}", COUNTER_RESET, UpdateCounterVariableValue);        }        /// <summary>        /// 更新变量数值        /// </summary>        /// <param name="variable"></param>        /// <param name="value"></param>        private void UpdateCounterVariableValue(string variable, object value)        {            string[] strAry = variable.Split('.');            string lastVariable = strAry[strAry.Length - 1];            PropertyInfo property = null;            string key = variable.Replace($".{lastVariable}", "");            if(_nameCounterFlowData.ContainsKey(key))            {                CounterFlowData counterFlowData = _nameCounterFlowData[key];                if (counterFlowData != null)                {                    property = counterFlowData.GetType().GetProperty(lastVariable);                    if (property != null)                    {                        property.SetValue(counterFlowData, value);                    }                }            }        }        #region Fill        /// <summary>        /// AN A面Fill On        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        public bool AnSideAFillOn(string cmd, object[] param)        {            string ioName= BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_A_PINENABLE}");            string countName= BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_A_CELL_FLOW}");            bool result= BeckhoffCounterManager.Instance.StartCounter(countName);            if (result)            {                return IOModuleManager.Instance.WriteIoValue(ioName, true);            }            else            {                return false;            }        }        /// <summary>        /// AN A面Fill Off        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        public bool AnSideAFillOff(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_A_PINENABLE}");            string countName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_A_CELL_FLOW}");            bool result = BeckhoffCounterManager.Instance.StopCounter(countName);            if(result)            {                result= BeckhoffCounterManager.Instance.ResetCounter(countName,0);                if(result)                {                    return IOModuleManager.Instance.WriteIoValue(ioName, false);                }            }            return false;        }        /// <summary>        /// AN B面Fill On        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        public bool AnSideBFillOn(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_B_PINENABLE}");            string countName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_B_CELL_FLOW}");            bool result = BeckhoffCounterManager.Instance.StartCounter(countName);            if (result)            {                return IOModuleManager.Instance.WriteIoValue(ioName, true);            }            else            {                return false;            }        }        /// <summary>        /// AN B面Fill Off        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        public bool AnSideBFillOff(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_B_PINENABLE}");            string countName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_B_CELL_FLOW}");            bool result = BeckhoffCounterManager.Instance.StopCounter(countName);            if (result)            {                result = BeckhoffCounterManager.Instance.ResetCounter(countName, 0);                if (result)                {                    return IOModuleManager.Instance.WriteIoValue(ioName, false);                }            }            return false;        }        #endregion        #region Drain        /// <summary>        /// AN A面Drain On        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        private bool AnSideADrainOn(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_A_POUTENABLE}");            return IOModuleManager.Instance.WriteIoValue(ioName, true);        }        /// <summary>        /// AN A面Drain Off        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        public bool AnSideADrainOff(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_A_POUTENABLE}");            return IOModuleManager.Instance.WriteIoValue(ioName, false);        }        /// <summary>        /// AN B面Drain On        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        private bool AnSideBDrainOn(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_B_POUTENABLE}");            return IOModuleManager.Instance.WriteIoValue(ioName, true);        }        /// <summary>        /// AN A面Drain Off        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        public bool AnSideBDrainOff(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{AN_B_POUTENABLE}");            return IOModuleManager.Instance.WriteIoValue(ioName, false);        }        #endregion        #region WH Clamp        /// <summary>        /// WH Clamp On        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        private bool WHClampOn(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{WH_CLAMP}");            return IOModuleManager.Instance.WriteIoValue(ioName, true);        }        /// <summary>        /// WH Clamp Off        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        private bool WHClampOff(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{WH_CLAMP}");            return IOModuleManager.Instance.WriteIoValue(ioName, false);        }        /// <summary>        /// WH UnClamp On        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        private bool WHUnClampOn(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{WH_UNCLAMP}");            return IOModuleManager.Instance.WriteIoValue(ioName, true);        }        /// <summary>        /// WH UnClamp Off        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        private bool WHUnClampOff(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{WH_UNCLAMP}");            return IOModuleManager.Instance.WriteIoValue(ioName, false);        }        /// <summary>        /// Wafer Holder Clamp On        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        public bool WaferHolderClampOn(string cmd, object[] param)        {            string unclampIOName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{WH_UNCLAMP}");            bool result = IOModuleManager.Instance.WriteIoValue(unclampIOName, false);            if(result)            {                string clampIOName= BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{WH_CLAMP}");                return IOModuleManager.Instance.WriteIoValue(clampIOName, true);            }            return false;        }        /// <summary>        /// Wafer Holder Clamp On        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        public bool WaferHolderUnclampOn(string cmd, object[] param)        {            string clampIOName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{WH_CLAMP}");            bool result = IOModuleManager.Instance.WriteIoValue(clampIOName, false);            if (result)            {                string unclampIOName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{WH_UNCLAMP}");                return IOModuleManager.Instance.WriteIoValue(unclampIOName, true);            }            return false;        }        /// <summary>        /// Wafer Holder Disconnect        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        public bool WaferHolderDisConnect(string cmd, object[] param)        {            string clampIOName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{WH_CLAMP}");            bool result = IOModuleManager.Instance.WriteIoValue(clampIOName, false);            if (result)            {                string unclampIOName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{WH_UNCLAMP}");                return IOModuleManager.Instance.WriteIoValue(unclampIOName, false);            }            return false;        }        #endregion        #region Cell Flow Valve        /// <summary>        /// Cell Flow On        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        public bool CellFlowValveOn(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{CELL_FLOW_VALVE}");            return IOModuleManager.Instance.WriteIoValue(ioName, true);        }        /// <summary>        /// Cell Flow Off        /// </summary>        /// <param name="cmd"></param>        /// <param name="param"></param>        /// <returns></returns>        public bool CellFlowValveOff(string cmd, object[] param)        {            string ioName = BeckhoffModuleIOManager.Instance.GetIoNameByInnerModuleName($"{Module}.{CELL_FLOW_VALVE}");            return IOModuleManager.Instance.WriteIoValue(ioName, false);        }        #endregion        #region AN Fill        /// <summary>        /// 阳极A面Fill        /// </summary>        /// <param name="cmd"></param>        /// <param name="objects"></param>        /// <returns></returns>        private bool ANSideAFillOperation(string cmd, object[] objects)        {            if(_currentANACellOperation!=ANCellOperation.None)            {                LOG.WriteLog(eEvent.ERR_METAL, Module, $"{Module} current is {_currentANACellOperation},cannot execute AN Side A Fill");                return false;            }            bool result = _anANACellFillRoutine.Start() == RState.Running;            if (result)            {                _currentANACellOperation = ANCellOperation.Fill;            }            return result;        }        /// <summary>        /// 阳极B面Fill        /// </summary>        /// <param name="cmd"></param>        /// <param name="objects"></param>        /// <returns></returns>        private bool ANSideBFillOperation(string cmd, object[] objects)        {            if (_currentANBCellOperation != ANCellOperation.None)            {                LOG.WriteLog(eEvent.ERR_METAL, Module, $"{Module} current is {_currentANBCellOperation},cannot execute AN Side B Fill");                return false;            }            bool result = _anANBCellFillRoutine.Start() == RState.Running;            if (result)            {                _currentANBCellOperation = ANCellOperation.Fill;            }            return result;        }        #endregion        #region AN Drain        /// <summary>        /// 阳极A面Drain        /// </summary>        /// <param name="cmd"></param>        /// <param name="objects"></param>        /// <returns></returns>        private bool ANSideADrainOperation(string cmd, object[] objects)        {            if (_currentANACellOperation != ANCellOperation.None)            {                LOG.WriteLog(eEvent.ERR_METAL, Module, $"{Module} current is {_currentANACellOperation},cannot execute AN Side A Drain");                return false;            }            bool result = true;            CompactMembranReservoirDevice reservoirDevice = GetReservoirDevice();            if(CheckOtherMetalDeviceDrainStatus(reservoirDevice,"A"))            {                return false;            }            if (reservoirDevice != null && reservoirDevice.ReservoirData.ANADrainPump == 0)            {                result = reservoirDevice.AnADrainPumpOn("", null);            }            if (result)            {                result = AnSideADrainOn("", null);                if (result)                {                    ANACellFlow.Status = ANCellStatus.Emptying.ToString();                    _anATotalFillFlow = 0;                    _anADrainTime=DateTime.Now;                }                else                {                    reservoirDevice.AnADrainPump(0);                }            }            return result;        }        /// <summary>        /// 其他Metal Drain是否打开        /// </summary>        /// <param name="reservoirDevice"></param>        /// <param name="side"></param>        /// <returns></returns>        private bool CheckOtherMetalDeviceDrainStatus(CompactMembranReservoirDevice reservoirDevice,string side)        {            if(reservoirDevice==null)            {                return false;            }            ReservoirItem reservoirItem = ReservoirItemManager.Instance.GetReservoirItem(reservoirDevice.Module);            if(reservoirItem!=null)            {                foreach(var item in reservoirItem.MetalCells)                {                    if(item.ModuleName!=Module)                    {                        CompactMembranMetalDevice metalDevice = DEVICE.GetDevice<CompactMembranMetalDevice>(item.ModuleName);                        if(metalDevice!=null)                        {                            if(side=="A"&&metalDevice.MetalDeviceData.ANAPoutEnable)                            {                                LOG.WriteLog(eEvent.ERR_METAL, Module, $"{item.ModuleName} A Drain is on");                                return true;                            }                            if (side == "B" && metalDevice.MetalDeviceData.ANBPoutEnable)                            {                                LOG.WriteLog(eEvent.ERR_METAL, Module, $"{item.ModuleName} B Drain is on");                                return true;                            }                        }                    }                }            }            return false;        }        /// <summary>        /// 阳极A面Drain        /// </summary>        /// <param name="cmd"></param>        /// <param name="objects"></param>        /// <returns></returns>        private bool ANSideBDrainOperation(string cmd, object[] objects)        {            if (_currentANBCellOperation != ANCellOperation.None)            {                LOG.WriteLog(eEvent.ERR_METAL, Module, $"{Module} current is {_currentANBCellOperation},cannot execute AN Side A Drain");                return false;            }            bool result = true;            CompactMembranReservoirDevice reservoirDevice = GetReservoirDevice();            if (CheckOtherMetalDeviceDrainStatus(reservoirDevice, "B"))            {                return false;            }            if(reservoirDevice!=null&&reservoirDevice.ReservoirData.ANBDrainPump==0)            {                result= reservoirDevice.AnBDrainPumpOn("", null);            }            if (result)            {                result = AnSideBDrainOn("", null);                if (result)                {                    ANBCellFlow.Status = ANCellStatus.Emptying.ToString();                    _anBTotalFillFlow = 0;                    _anBDrainTime=DateTime.Now;                }                else                {                    reservoirDevice.AnBDrainPump(0);                }            }            return result;        }        #endregion        #region AN Stop        /// <summary>        /// 阳面A停止         /// </summary>        /// <param name="cmd"></param>        /// <param name="objs"></param>        /// <returns></returns>        private bool ANSideAStopOperation(string cmd, object[] objs)        {            CompactMembranReservoirDevice reservoirDevice = GetReservoirDevice();            if (reservoirDevice != null)            {                if (reservoirDevice.ReservoirData.ANPump != 0)                {                    reservoirDevice.AnPump(0);                }                if (reservoirDevice.ReservoirData.ANADrainPump != 0)                {                    reservoirDevice.AnADrainPump(0);                }            }            if (MetalDeviceData.ANAPinEnable)            {                AnSideAFillOff("", null);            }            if (MetalDeviceData.ANAPoutEnable)            {                AnSideADrainOff("", null);            }                        ANACellFlow.Status = ANCellStatus.Unknow.ToString();            _anATotalFillFlow = 0;            return true;        }        /// <summary>        /// 阳面A停止         /// </summary>        /// <param name="cmd"></param>        /// <param name="objs"></param>        /// <returns></returns>        private bool ANSideBStopOperation(string cmd, object[] objs)        {            CompactMembranReservoirDevice reservoirDevice = GetReservoirDevice();            if (reservoirDevice != null)            {                if (reservoirDevice.ReservoirData.ANPump != 0)                {                    reservoirDevice.AnPump(0);                }                if(reservoirDevice.ReservoirData.ANBDrainPump!=0)                {                    reservoirDevice.AnBDrainPump(0);                }            }            if(MetalDeviceData.ANBPinEnable)            {                AnSideBFillOff("", null);            }            if(MetalDeviceData.ANBPoutEnable)            {                AnSideBDrainOff("", null);            }            ANBCellFlow.Status = ANCellStatus.Unknow.ToString();            _anBTotalFillFlow = 0;            return true;        }        #endregion        #region Timer定时器        /// <summary>        /// 定时器        /// </summary>        /// <returns></returns>        public override bool OnTimer(int interval)        {            if (_currentANACellOperation == ANCellOperation.Fill)            {                SideFillRoutineMonitor(_anANACellFillRoutine, "A", ANACellFlow, ref _currentANACellOperation);            }            if (_currentANBCellOperation == ANCellOperation.Fill)            {                SideFillRoutineMonitor(_anANBCellFillRoutine, "B", ANBCellFlow, ref _currentANBCellOperation);            }            //正在Filing            if (ANACellFlow.Status == ANCellStatus.Filling.ToString())            {                JudgeFillFull(ANACellFlow,interval,ref _anATotalFillFlow);            }            if (ANBCellFlow.Status == ANCellStatus.Filling.ToString())            {                JudgeFillFull(ANBCellFlow,interval,ref _anBTotalFillFlow);            }            if (ANACellFlow.Status == ANCellStatus.Emptying.ToString())            {                JudgeDrainEmpty("A");            }            if (ANBCellFlow.Status == ANCellStatus.Emptying.ToString())            {                JudgeDrainEmpty("B");            }            return true;        }        /// <summary>        /// 单面Fill Routine监控        /// </summary>        /// <param name="routine"></param>        /// <param name="side"></param>        /// <param name="operation"></param>        private void SideFillRoutineMonitor(IRoutine routine, string side,CounterFlowData counterFlowData, ref ANCellOperation operation)        {            RState state = routine.Monitor();            if (state == RState.Failed || state == RState.Timeout)            {                CloseReservoirPumpAndMetalFill(side);                operation = ANCellOperation.None;            }            else if (state == RState.End)            {                counterFlowData.Status = ANCellStatus.Filling.ToString();                operation = ANCellOperation.None;            }        }        /// <summary>        /// 判定是否fill Full        /// </summary>        private void JudgeFillFull(CounterFlowData counterFlowData,int interval,ref double totalFillFlow)        {            double anodeFillVolume = SC.GetValue<double>($"Metal.AnodeFillVolume");            totalFillFlow += counterFlowData.CounterValue / 60 * ((double)interval / 1000);            if (totalFillFlow >= anodeFillVolume)            {                counterFlowData.Status = "Full";            }        }        /// <summary>        /// 判定是否Drain Empty        /// </summary>        /// <param name="counterFlowData"></param>        private void JudgeDrainEmpty(string side)        {            if (side == "A")            {                JudgeSideADrain(ANACellFlow, _anADrainTime);                if(ANACellFlow.Status==ANCellStatus.Empty.ToString())                {                    CompactMembranReservoirDevice reservoirDevice = GetReservoirDevice();                    if (reservoirDevice != null)                    {                        reservoirDevice.AnADrainPump(0);                    }                    AnSideADrainOff("", null);                }            }            else            {                JudgeSideADrain(ANBCellFlow, _anBDrainTime);                if (ANBCellFlow.Status == ANCellStatus.Empty.ToString())                {                    CompactMembranReservoirDevice reservoirDevice = GetReservoirDevice();                    if (reservoirDevice != null)                    {                        reservoirDevice.AnBDrainPump(0);                    }                    AnSideBDrainOff("", null);                }            }        }        /// <summary>        /// 判定单面Drain的情况        /// </summary>        /// <param name="counterFlowData"></param>        /// <param name="drainDateTime"></param>        private void JudgeSideADrain(CounterFlowData counterFlowData, DateTime drainDateTime)        {            double anodeDrainTime = SC.GetValue<double>("Metal.AnodeDrainTime");            if (DateTime.Now.Subtract(drainDateTime).TotalMilliseconds >= anodeDrainTime * 1000)            {                counterFlowData.Status = ANCellStatus.Empty.ToString();            }        }        /// <summary>        /// 关闭Reservoir Pump和Metal Fill        /// </summary>        private void CloseReservoirPumpAndMetalFill(string side)        {            CompactMembranReservoirDevice reservoirDevice = GetReservoirDevice();            if (reservoirDevice != null)            {                reservoirDevice.AnPump(0);                if (side == "A")                {                    AnSideAFillOff("", null);                }                else                {                    AnSideBDrainOff("", null);                }            }        }        /// <summary>        /// 获取Reservoir设备        /// </summary>        /// <returns></returns>        private CompactMembranReservoirDevice GetReservoirDevice()        {            string reservoir = ReservoirItemManager.Instance.GetReservoirByMetal(Module);            return DEVICE.GetDevice<CompactMembranReservoirDevice>(reservoir);        }        #endregion        /// <summary>        /// Enter Disabled Operation        /// </summary>        /// <returns></returns>        public void EnterDisabledOperation()        {            ANSideAStopOperation("", null);            ANSideBStopOperation("", null);            if (_metalDeviceData.CellFlowValve)            {                CellFlowValveOff("", null);            }                    }        #region 设备接口        public override void Monitor()        {        }        public override void Reset()        {        }        public override void Terminate()        {        }        #endregion    }}
 |