using Aitex.Core.RT.DataCenter;
using Aitex.Core.RT.Device;
using Aitex.Core.RT.Log;
using Aitex.Core.Util;
using MECF.Framework.Common.Persistent.Reservoirs;
using MECF.Framework.Common.ToolLayout;
using CyberX8_Core;
using CyberX8_RT.Devices.LinMot;
using CyberX8_RT.Devices.PowerSupplier;
using Aitex.Core.RT.OperationCenter;
using CyberX8_RT.Modules;
using CyberX8_RT.Modules.Metal;
namespace CyberX8_RT.Devices.Metal
{
    public class MetalCellDevice : BaseDevice, IDevice
    {
        #region 常量 
        private const string PERSISTENT_VALUE = "PersistentValue";
        private const string AUTO = "Auto";
        private const string MANUAL = "Manual";
        private const string STRATUS = "Stratus";
        private const string DISABLED = "Disabled";
        #endregion
        #region 内部变量
        /// 
        /// 操作当前状态
        /// 
        protected RState _status;
        /// 
        /// 持久化数据
        /// 
        protected MetalPersistentValue _persistentValue;
        /// 
        /// A面PowerSupplier
        /// 
        protected CellPowerSupplier _sideAPowerSupplier;
        /// 
        /// B面PowerSupplier
        /// 
        protected CellPowerSupplier _sideBPowerSupplier;
        /// 
        /// Linmot
        /// 
        protected LinMotAxis _linmotAxis;
        /// 
        /// Metal项
        /// 
        private MetalItem _metalItem;
        #endregion
        #region 属性
        /// 
        /// 状态
        /// 
        public RState Status { get { return _status; } }
        /// 
        /// 是否禁用
        /// 
        public bool IsDisable { get { return _persistentValue == null || _persistentValue.OperatingMode == DISABLED; } }
        /// 
        /// clamp off状态
        /// 
        public bool ClampOff
        {
            get
            {
                if (_metalItem.SubType == STRATUS)
                {
                    StandardHotMetalDevice metalDevice = DEVICE.GetDevice(Module.ToString());
                    return !metalDevice.MetalDeviceData.WaferHolderClamp;
                }
                else
                {
                    CompactMembranMetalDevice metalDevice = DEVICE.GetDevice(Module.ToString());
                    return !metalDevice.MetalDeviceData.WHClamp && metalDevice.MetalDeviceData.WHUnclamp;
                }
            }
        }
        /// 
        /// 操作模式
        /// 
        public string OperationMode { get { return _persistentValue.OperatingMode; } }
        /// 
        /// 工程模式
        /// 
        public string EngineerMode { get { return _persistentValue.RecipeOperatingMode; } }
        /// 
        /// A面PowerSupplier
        /// 
        public CellPowerSupplier SideAPowerSupplier { get { return _sideAPowerSupplier; } }
        /// 
        /// B面PowerSupplier
        /// 
        public CellPowerSupplier SideBPowerSupplier { get { return _sideBPowerSupplier; } }
        /// 
        /// 是否为Auto
        /// 
        public bool IsAuto { get { return _persistentValue != null ? _persistentValue.OperatingMode == AUTO : false; } }
        /// 
        /// 是否为Auto
        /// 
        public bool IsManual { get { return _persistentValue != null ? _persistentValue.OperatingMode == MANUAL : false; } }
        /// 
        /// linmot motor on 
        /// 
        public bool IsLinmotMotorOn { get { return _linmotAxis != null ? _linmotAxis.IsMotorOn : false; } }
        #endregion
        /// 
        /// 构造函数
        /// 
        /// 
        public MetalCellDevice(string moduleName) : base(moduleName, moduleName, moduleName, moduleName)
        {
        }
        /// 
        /// 初始化
        /// 
        /// 
        public virtual bool Initialize()
        {
            InitializeParameter();
            SubscribeData();
            InitializeOperation();
            return true;
        }
        /// 
        /// 定时器执行
        /// 
        public virtual bool OnTimer(int interval)
        {
            return true;
        }
        /// 
        /// 初始化参数
        /// 
        private void InitializeParameter()
        {
            _persistentValue = MetalPersistentManager.Instance.GetMetalPersistentValue(Module);
            if(_persistentValue == null)
            {
                LOG.WriteLog(eEvent.ERR_RESERVOIR, Module, "Persistent Value Object is not exist");
            }
            _metalItem = MetalItemManager.Instance.GetMetalItem(Module);
            if (_metalItem != null)
            {
                _sideAPowerSupplier = DEVICE.GetDevice(_metalItem.PlatingPowerSupplyAID);
                _sideBPowerSupplier = DEVICE.GetDevice(_metalItem.PlatingPowerSupplyBID);
                _linmotAxis = DEVICE.GetDevice(_metalItem.LinmotID);
            }
        }
        /// 
        /// 订阅数据
        /// 
        private void SubscribeData()
        {
            DATA.Subscribe($"{Module}.{PERSISTENT_VALUE}", () => _persistentValue, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideAPowerSupplierData", () => _sideAPowerSupplier.PowerSupplierData, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideBPowerSupplierData", () => _sideBPowerSupplier.PowerSupplierData, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideAPowerSupplier.ID", () => _sideAPowerSupplier.Module, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideAPowerSupplier.IsConnected", () => _sideAPowerSupplier.IsConnected, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideAPowerSupplier.Voltage", () => _sideAPowerSupplier.PowerSupplierData.Voltage, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideAPowerSupplier.Current", () => _sideAPowerSupplier.PowerSupplierData.Current, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideAPowerSupplier.SetPoint", () => _sideAPowerSupplier.PowerSupplierData.SetPoint, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideAPowerSupplier.RunModel", () => _sideAPowerSupplier.PowerSupplierData.PowerRunModelContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideAPowerSupplier.PowerControl", () => _sideAPowerSupplier.PowerSupplierData.PowerControlContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideAPowerSupplier.PowerStatus", () => _sideAPowerSupplier.PowerSupplierData.PowerStatusContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideAPowerSupplier.Enable", () => _sideBPowerSupplier.PowerSupplierData.Enabled, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideBPowerSupplier.ID", () => _sideBPowerSupplier.Module, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideBPowerSupplier.IsConnected", () => _sideAPowerSupplier.IsConnected, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideBPowerSupplier.Voltage", () => _sideBPowerSupplier.PowerSupplierData.Voltage, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideBPowerSupplier.Current", () => _sideBPowerSupplier.PowerSupplierData.Current, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideBPowerSupplier.SetPoint", () => _sideBPowerSupplier.PowerSupplierData.SetPoint, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideBPowerSupplier.RunModel", () => _sideBPowerSupplier.PowerSupplierData.PowerRunModelContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideBPowerSupplier.PowerControl", () => _sideBPowerSupplier.PowerSupplierData.PowerControlContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideBPowerSupplier.PowerStatus", () => _sideBPowerSupplier.PowerSupplierData.PowerStatusContent, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.SideBPowerSupplier.Enable", () => _sideBPowerSupplier.PowerSupplierData.Enabled, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.Linmot.ID", () => _linmotAxis.Module, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.Linmot.IsMotorOn", () => _linmotAxis.IsMotorOn, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.Linmot.IsError", () => _linmotAxis.IsError, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.Linmot.IsSwitchOn", () => _linmotAxis.IsSwitchOn, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.Linmot.CurveSpeed", () => _linmotAxis.CurveSpeed, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.Linmot.ErrorCode", () => _linmotAxis.ErrorCode, SubscriptionAttribute.FLAG.IgnoreSaveDB);
            DATA.Subscribe($"{Module}.Linmot.CurrentPosition", () => _linmotAxis.CurrentPosition, SubscriptionAttribute.FLAG.IgnoreSaveDB);
        }
        /// 
        /// 初始化操作
        /// 
        protected virtual void InitializeOperation()
        {
            OP.Subscribe($"{Module}.DisabledAction", DisabledOperation);
            OP.Subscribe($"{Module}.ManualAction", ManualOperation);
            OP.Subscribe($"{Module}.AutoAction", AutoOperation);
            OP.Subscribe($"{Module}.EngineeringModeAction", EngineeringModeOperation);
            OP.Subscribe($"{Module}.ProductionModeAction", ProductionModeOperation);
            OP.Subscribe($"{Module}.SetMetalWaferSize", (cmd, args) => { return SetMetalWaferSize(cmd, args); });
        }
        /// 
        /// 开始Curve
        /// 
        /// 
        /// 
        public bool StartCurveMotion(int speed)
        {
            if (_linmotAxis != null)
            {
                return _linmotAxis.StartCurve(speed);
            }
            else
            {
                LOG.WriteLog(eEvent.ERR_METAL, Module, "linmot is null");
                return false;
            }
        }
        /// 
        /// 开始Curve
        /// 
        /// 
        /// 
        public bool ChangeCurveSpeedMotion(int speed)
        {
            if (_linmotAxis != null)
            {
                return _linmotAxis.ChangeCurveSpeed(speed);
            }
            else
            {
                LOG.WriteLog(eEvent.ERR_METAL, Module, "linmot is null");
                return false;
            }
        }
        /// 
        /// ResetLinmot
        /// 
        /// 
        public bool ResetLinmot()
        {
            if (_linmotAxis != null)
            {
                return _linmotAxis.ResetOperation("", false);
            }
            else
            {
                LOG.WriteLog(eEvent.ERR_METAL, Module, "linmot is null");
                return false;
            }
        }
        /// 
        /// 检验Linmot Routine状态是否为结束状态
        /// 
        /// 
        public bool CheckLinmotRoutineEnd()
        {
            if (_linmotAxis != null)
            {
                return _linmotAxis.Status == RState.End;
            }
            else
            {
                LOG.WriteLog(eEvent.ERR_METAL, Module, "linmot is null");
                return false;
            }
        }
        /// 
        /// 检验Linmot Routine状态是否为错误状态
        /// 
        /// 
        public bool CheckLinmotRoutineError()
        {
            if (_linmotAxis != null)
            {
                return _linmotAxis.Status == RState.Failed || _linmotAxis.Status == RState.Timeout;
            }
            else
            {
                return false;
            }
        }
        /// 
        /// 停止Linmot
        /// 
        /// 
        public bool StopLinmot()
        {
            if (_linmotAxis != null)
            {
                return _linmotAxis.StopOperation("", null);
            }
            else
            {
                LOG.WriteLog(eEvent.ERR_METAL, Module, "linmot is null");
                return false;
            }
        }
        #region Operation
        /// 
        /// DisabledAction
        /// 
        /// 
        /// 
        /// 
        public bool DisabledOperation(string cmd, object[] args)
        {
            string currentOperation = "Disabled";
            MetalEntity metalEntity = Singleton.Instance.GetModule(Module);
            if (metalEntity != null && _persistentValue != null && _persistentValue.OperatingMode != currentOperation)
            {
                string preOperation = _persistentValue.OperatingMode;
                if (metalEntity.IsBusy)
                {
                    LOG.WriteLog(eEvent.ERR_METAL, Module, $"{Module} is Busy, can't switch to Disabled mode");
                    return false;
                }
                metalEntity.EnterInit();
                _persistentValue.OperatingMode = currentOperation;
                LOG.WriteLog(eEvent.INFO_METAL, Module, $"Operating mode is switched from {preOperation} to {currentOperation}");
            }
            MetalPersistentManager.Instance.UpdatePersistentValue(Module);
            return true;
        }
        /// 
        /// ManualAction
        /// 
        /// 
        /// 
        /// 
        public bool ManualOperation(string cmd, object[] args)
        {
            string currentOperation = "Manual";
            MetalEntity metalEntity = Singleton.Instance.GetModule(Module);
            if (metalEntity != null && _persistentValue != null && _persistentValue.OperatingMode != currentOperation)
            {
                string preOperation = _persistentValue.OperatingMode;
                if (metalEntity.IsBusy)
                {
                    LOG.WriteLog(eEvent.ERR_METAL, Module, $"{Module} is Busy, can't switch to Manual mode");
                    return false;
                }
                metalEntity.EnterInit();
                _persistentValue.OperatingMode = currentOperation;
                LOG.WriteLog(eEvent.INFO_METAL, Module, $"Operating mode is switched from {preOperation} to {currentOperation}");
            }
            MetalPersistentManager.Instance.UpdatePersistentValue(Module);
            return true;
        }
        /// 
        /// AutoAction
        /// 
        /// 
        /// 
        /// 
        public bool AutoOperation(string cmd, object[] args)
        {
            string currentOperation = "Auto";
            MetalEntity metalEntity = Singleton.Instance.GetModule(Module);
            if (metalEntity != null && _persistentValue != null && _persistentValue.OperatingMode != currentOperation)
            {
                string preOperation = _persistentValue.OperatingMode;
                if (metalEntity.IsBusy)
                {
                    LOG.WriteLog(eEvent.ERR_METAL, Module, $"{Module} is Busy, can't switch to Auto mode");
                    return false;
                }
                metalEntity.EnterInit();
                _persistentValue.OperatingMode = currentOperation;
                LOG.WriteLog(eEvent.INFO_METAL, Module, $"Operating mode is switched from {preOperation} to {currentOperation}");
            }
            MetalPersistentManager.Instance.UpdatePersistentValue(Module);
            return true;
        }
        /// 
        /// EngineeringModeAction
        /// 
        /// 
        /// 
        /// 
        private bool EngineeringModeOperation(string cmd, object[] args)
        {
            string currentRecipeOperation = "Engineering";
            if (_persistentValue != null)
            {
                _persistentValue.RecipeOperatingMode = currentRecipeOperation;
            }
            MetalPersistentManager.Instance.UpdatePersistentValue(Module);
            return true;
        }
        /// 
        /// ProductionAction
        /// 
        /// 
        /// 
        /// 
        private bool ProductionModeOperation(string cmd, object[] args)
        {
            string currentRecipeOperation = "Production";
            if (_persistentValue != null)
            {
                _persistentValue.RecipeOperatingMode = currentRecipeOperation;
            }
            MetalPersistentManager.Instance.UpdatePersistentValue(Module);
            return true;
        }
        private bool SetMetalWaferSize(string cmd, object[] args)
        {
            string metalWaferSize = args[0] as string;
            if (_persistentValue != null)
            {
                _persistentValue.MetalWaferSize = int.Parse(metalWaferSize);
            }
            MetalPersistentManager.Instance.UpdatePersistentValue(Module);
            return true;
        }
        #endregion
        #region Clamp
        /// 
        /// Clamp Off
        /// 
        /// 
        public bool WaferHolderClampOff()
        {
            if (_metalItem.SubType == STRATUS)
            {
                StandardHotMetalDevice metalDevice = DEVICE.GetDevice(Module.ToString());
                return metalDevice.WaferHolderClampOff("", null);
            }
            else
            {
                CompactMembranMetalDevice metalDevice = DEVICE.GetDevice(Module.ToString());
                return metalDevice.WaferHolderUnclampOn("", null);
            }
        }
        #endregion
        public virtual void Monitor()
        {
        }
        public virtual void Reset()
        {
        }
        public virtual void Terminate()
        {
        }
    }
}