2025-02-14 10:04:29 -03:00
|
|
|
|
using CommunityToolkit.Mvvm.ComponentModel;
|
2024-09-12 11:43:39 -03:00
|
|
|
|
using CtrEditor.FuncionesBase;
|
2024-06-28 14:47:08 -03:00
|
|
|
|
using LibS7Adv;
|
2024-09-12 11:43:39 -03:00
|
|
|
|
using Newtonsoft.Json;
|
2025-02-13 10:00:47 -03:00
|
|
|
|
using System.Diagnostics;
|
2024-05-11 11:58:55 -03:00
|
|
|
|
using System.Windows;
|
|
|
|
|
using System.Windows.Controls;
|
|
|
|
|
using System.Windows.Media;
|
|
|
|
|
|
|
|
|
|
namespace CtrEditor.ObjetosSim
|
|
|
|
|
{
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Interaction logic for ucVMmotorSim.xaml
|
|
|
|
|
/// </summary>
|
2024-05-14 07:04:22 -03:00
|
|
|
|
///
|
|
|
|
|
|
2024-05-22 14:21:39 -03:00
|
|
|
|
public partial class osVMmotorSim : osBase, IosBase
|
2024-05-11 11:58:55 -03:00
|
|
|
|
{
|
2024-05-14 07:04:22 -03:00
|
|
|
|
|
2024-05-11 11:58:55 -03:00
|
|
|
|
// Otros datos y métodos relevantes para la simulación
|
2024-06-28 14:47:08 -03:00
|
|
|
|
|
2024-05-18 05:53:04 -03:00
|
|
|
|
private VMSimMotor motState = new VMSimMotor();
|
2024-05-14 12:10:32 -03:00
|
|
|
|
|
2024-05-19 16:38:57 -03:00
|
|
|
|
public static string NombreClase()
|
|
|
|
|
{
|
|
|
|
|
return "VetroMeccanica Motor";
|
|
|
|
|
}
|
2024-05-22 14:21:39 -03:00
|
|
|
|
private string nombre = NombreClase();
|
|
|
|
|
public override string Nombre
|
2024-05-18 05:53:04 -03:00
|
|
|
|
{
|
2024-05-22 14:21:39 -03:00
|
|
|
|
get => nombre;
|
|
|
|
|
set => SetProperty(ref nombre, value);
|
2024-05-18 05:53:04 -03:00
|
|
|
|
}
|
2024-05-11 11:58:55 -03:00
|
|
|
|
|
2024-05-22 14:21:39 -03:00
|
|
|
|
[JsonIgnore]
|
|
|
|
|
[ObservableProperty]
|
|
|
|
|
public ImageSource imageSource_oculta;
|
2024-05-11 11:58:55 -03:00
|
|
|
|
|
2024-05-22 14:21:39 -03:00
|
|
|
|
[ObservableProperty]
|
|
|
|
|
public float tamano;
|
2024-06-30 09:32:32 -03:00
|
|
|
|
|
|
|
|
|
public override void OnResize(float Delta_Width, float Delta_Height)
|
|
|
|
|
{
|
|
|
|
|
Tamano += Delta_Width + Delta_Height;
|
|
|
|
|
}
|
|
|
|
|
|
2025-02-13 10:00:47 -03:00
|
|
|
|
[ObservableProperty]
|
2025-02-14 10:04:29 -03:00
|
|
|
|
public float refresh_Time_ms;
|
2025-02-13 10:00:47 -03:00
|
|
|
|
|
2024-05-22 14:21:39 -03:00
|
|
|
|
[ObservableProperty]
|
2024-09-12 11:43:39 -03:00
|
|
|
|
public float proporcional_Speed;
|
2024-05-27 05:34:20 -03:00
|
|
|
|
|
|
|
|
|
[ObservableProperty]
|
|
|
|
|
public float max_Speed_for_Ramp;
|
|
|
|
|
|
2024-06-28 14:47:08 -03:00
|
|
|
|
[ObservableProperty]
|
|
|
|
|
bool vFD_Trip_NC;
|
2024-05-14 12:10:32 -03:00
|
|
|
|
|
2024-05-22 14:21:39 -03:00
|
|
|
|
[ObservableProperty]
|
|
|
|
|
public float tiempoRampa;
|
2024-05-14 12:10:32 -03:00
|
|
|
|
|
2024-05-22 14:21:39 -03:00
|
|
|
|
partial void OnTiempoRampaChanged(float value)
|
2024-05-14 07:04:22 -03:00
|
|
|
|
{
|
2024-05-22 14:21:39 -03:00
|
|
|
|
if (value < 0.1f)
|
|
|
|
|
value = 0.1f;
|
|
|
|
|
tiempoRampa = value;
|
2024-05-14 07:04:22 -03:00
|
|
|
|
}
|
|
|
|
|
|
2024-05-22 14:21:39 -03:00
|
|
|
|
[ObservableProperty]
|
2024-06-28 14:47:08 -03:00
|
|
|
|
bool encendido;
|
2024-05-22 14:21:39 -03:00
|
|
|
|
[ObservableProperty]
|
2024-06-28 14:47:08 -03:00
|
|
|
|
int pLC_NumeroMotor;
|
|
|
|
|
|
|
|
|
|
partial void OnPLC_NumeroMotorChanged(int value)
|
|
|
|
|
{
|
2024-09-12 11:43:39 -03:00
|
|
|
|
if (PLC_DB_Motor == 0)
|
2024-06-28 14:47:08 -03:00
|
|
|
|
{
|
|
|
|
|
PLC_DB_Motor = PLC_NumeroMotor - 30 + 300;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[ObservableProperty]
|
|
|
|
|
int pLC_DB_Motor;
|
|
|
|
|
|
|
|
|
|
|
2024-05-22 14:21:39 -03:00
|
|
|
|
[ObservableProperty]
|
|
|
|
|
public float ratio;
|
2024-05-11 11:58:55 -03:00
|
|
|
|
|
2024-05-22 14:21:39 -03:00
|
|
|
|
[ObservableProperty]
|
|
|
|
|
public float velocidad;
|
2024-05-11 11:58:55 -03:00
|
|
|
|
|
2025-01-04 06:34:19 -03:00
|
|
|
|
[ObservableProperty]
|
|
|
|
|
public bool sentido_contrario;
|
|
|
|
|
|
|
|
|
|
|
2024-05-22 14:21:39 -03:00
|
|
|
|
partial void OnVelocidadChanged(float value)
|
2024-05-11 11:58:55 -03:00
|
|
|
|
{
|
2024-05-22 14:21:39 -03:00
|
|
|
|
if (value > 0)
|
|
|
|
|
ImageSource_oculta = ImageFromPath("/imagenes/motorVerde.png");
|
|
|
|
|
else
|
|
|
|
|
ImageSource_oculta = ImageFromPath("/imagenes/motorNegro.png");
|
2024-05-14 07:04:22 -03:00
|
|
|
|
}
|
|
|
|
|
|
2024-05-11 11:58:55 -03:00
|
|
|
|
public osVMmotorSim()
|
|
|
|
|
{
|
|
|
|
|
Tamano = 0.30f;
|
2024-05-11 15:55:44 -03:00
|
|
|
|
PLC_NumeroMotor = 31;
|
2024-09-12 11:43:39 -03:00
|
|
|
|
Proporcional_Speed = 100;
|
2024-05-27 05:34:20 -03:00
|
|
|
|
Max_Speed_for_Ramp = 100;
|
2024-05-14 12:10:32 -03:00
|
|
|
|
TiempoRampa = 3;
|
2024-05-22 14:21:39 -03:00
|
|
|
|
ImageSource_oculta = ImageFromPath("/imagenes/motor2.png");
|
2025-02-13 10:00:47 -03:00
|
|
|
|
Refresh_Time_ms = 500;
|
2024-06-28 14:47:08 -03:00
|
|
|
|
}
|
2024-05-11 11:58:55 -03:00
|
|
|
|
|
2024-05-14 03:15:54 -03:00
|
|
|
|
public override void UpdateGeometryStart()
|
2024-05-11 11:58:55 -03:00
|
|
|
|
{
|
|
|
|
|
// Se llama antes de la simulacion
|
|
|
|
|
|
|
|
|
|
}
|
2025-02-13 10:00:47 -03:00
|
|
|
|
|
|
|
|
|
[JsonIgnore]
|
|
|
|
|
private float elapsedTimeAccumulator = 0;
|
|
|
|
|
|
|
|
|
|
public override void UpdatePLC(PLCViewModel plc, int TotalMilliseconds)
|
2024-06-28 14:47:08 -03:00
|
|
|
|
{
|
2025-02-13 10:00:47 -03:00
|
|
|
|
|
|
|
|
|
elapsedTimeAccumulator += TotalMilliseconds;
|
|
|
|
|
float randomFactor = (float)(new Random().NextDouble() * 0.1); // 10% random factor
|
|
|
|
|
float adjustedRefreshTime = Refresh_Time_ms * (1 + randomFactor);
|
|
|
|
|
|
|
|
|
|
if (elapsedTimeAccumulator >= adjustedRefreshTime)
|
|
|
|
|
{
|
|
|
|
|
motState.UpdatePLC(plc, this, TotalMilliseconds);
|
|
|
|
|
elapsedTimeAccumulator = 0;
|
|
|
|
|
}
|
2024-09-12 11:43:39 -03:00
|
|
|
|
Velocidad = (Proporcional_Speed / 100) * (motState.STATUS_VFD_ACT_Speed_Hz / 10);
|
2025-01-04 06:34:19 -03:00
|
|
|
|
Sentido_contrario = motState.OUT_Reversal;
|
2025-02-13 10:00:47 -03:00
|
|
|
|
|
2024-05-14 12:10:32 -03:00
|
|
|
|
}
|
|
|
|
|
|
2025-02-13 10:00:47 -03:00
|
|
|
|
public override void UpdateControl(int TotalMilliseconds)
|
2024-05-11 11:58:55 -03:00
|
|
|
|
{
|
2024-05-18 05:53:04 -03:00
|
|
|
|
// Calculamos la velocidad
|
2025-02-13 10:00:47 -03:00
|
|
|
|
motState.UpdateSpeed(Max_Speed_for_Ramp, TiempoRampa, TotalMilliseconds);
|
2024-05-11 15:55:44 -03:00
|
|
|
|
|
2024-05-11 11:58:55 -03:00
|
|
|
|
}
|
2024-05-18 05:53:04 -03:00
|
|
|
|
|
2024-05-14 03:15:54 -03:00
|
|
|
|
public override void ucLoaded()
|
|
|
|
|
{
|
|
|
|
|
// El UserControl ya se ha cargado y podemos obtener las coordenadas para
|
|
|
|
|
// crear el objeto de simulacion
|
2024-06-09 12:33:09 -03:00
|
|
|
|
base.ucLoaded();
|
2024-06-04 12:33:00 -03:00
|
|
|
|
OnVelocidadChanged(Velocidad);
|
2024-05-18 06:49:02 -03:00
|
|
|
|
|
2024-05-14 03:15:54 -03:00
|
|
|
|
}
|
2024-05-11 11:58:55 -03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public partial class ucVMmotorSim : UserControl, IDataContainer
|
|
|
|
|
{
|
|
|
|
|
public osBase? Datos { get; set; }
|
2025-02-13 10:00:47 -03:00
|
|
|
|
public int zIndex_fromFrames { get; set; }
|
2024-05-11 11:58:55 -03:00
|
|
|
|
|
|
|
|
|
public ucVMmotorSim()
|
|
|
|
|
{
|
|
|
|
|
InitializeComponent();
|
2024-05-14 07:04:22 -03:00
|
|
|
|
this.Loaded += OnLoaded;
|
2024-05-18 09:58:41 -03:00
|
|
|
|
this.Unloaded += OnUnloaded;
|
2024-05-14 07:04:22 -03:00
|
|
|
|
}
|
|
|
|
|
private void OnLoaded(object sender, RoutedEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
Datos?.ucLoaded();
|
2024-05-11 11:58:55 -03:00
|
|
|
|
}
|
2024-05-18 09:58:41 -03:00
|
|
|
|
private void OnUnloaded(object sender, RoutedEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
Datos?.ucUnLoaded();
|
|
|
|
|
}
|
2024-05-11 11:58:55 -03:00
|
|
|
|
public void Highlight(bool State) { }
|
2025-02-13 10:00:47 -03:00
|
|
|
|
public ZIndexEnum ZIndex_Base()
|
2024-05-11 11:58:55 -03:00
|
|
|
|
{
|
2024-06-30 13:17:44 -03:00
|
|
|
|
return ZIndexEnum.Estaticos;
|
2024-05-11 11:58:55 -03:00
|
|
|
|
}
|
2024-06-30 13:17:44 -03:00
|
|
|
|
|
2024-05-11 11:58:55 -03:00
|
|
|
|
}
|
2024-05-18 05:53:04 -03:00
|
|
|
|
|
|
|
|
|
public class VMSimMotor
|
|
|
|
|
{
|
|
|
|
|
public bool _STATUS_VFD_Ready;
|
|
|
|
|
public float STATUS_VFD_ACT_Speed_Hz;
|
|
|
|
|
public bool Motor_Running;
|
|
|
|
|
public bool STATUS_VFD_Trip;
|
|
|
|
|
public bool STATUS_VFD_Warning;
|
|
|
|
|
public bool STATUS_VFD_Coasting;
|
|
|
|
|
public bool OUT_Run;
|
|
|
|
|
public bool OUT_Stop;
|
|
|
|
|
public bool OUT_Reversal;
|
|
|
|
|
public float OUT_OUT_VFD_REQ_Speed_Hz;
|
|
|
|
|
|
2025-02-13 10:00:47 -03:00
|
|
|
|
public void UpdatePLC(PLCViewModel plc, osVMmotorSim Data, int TotalMilliseconds)
|
2024-05-18 05:53:04 -03:00
|
|
|
|
{
|
2024-06-28 14:47:08 -03:00
|
|
|
|
var DB_Motor = Data.PLC_DB_Motor;
|
2024-05-18 05:53:04 -03:00
|
|
|
|
|
2024-06-28 14:47:08 -03:00
|
|
|
|
if (DB_Motor == 0)
|
|
|
|
|
return;
|
2024-05-18 05:53:04 -03:00
|
|
|
|
|
2025-02-14 10:04:29 -03:00
|
|
|
|
// Read ControlWord in one operation
|
|
|
|
|
int controlWord = plc.LeerTagDInt($"\"DB MotorSimulate\".Motors[{DB_Motor}].ControlWord") ?? 0;
|
|
|
|
|
var control = VMMotorBitPacker.UnpackControlWord(controlWord);
|
2024-05-18 05:53:04 -03:00
|
|
|
|
|
2025-02-14 10:04:29 -03:00
|
|
|
|
// Update local state from ControlWord
|
|
|
|
|
OUT_Run = control.run;
|
|
|
|
|
OUT_Stop = control.stop;
|
|
|
|
|
OUT_Reversal = control.reversal;
|
|
|
|
|
OUT_OUT_VFD_REQ_Speed_Hz = control.reqSpeedHz;
|
2024-06-28 14:47:08 -03:00
|
|
|
|
|
2025-02-14 10:04:29 -03:00
|
|
|
|
// Update motor state based on enable status
|
2024-06-28 14:47:08 -03:00
|
|
|
|
if (Data.Encendido)
|
2024-05-18 05:53:04 -03:00
|
|
|
|
{
|
|
|
|
|
_STATUS_VFD_Ready = true;
|
2025-02-14 10:04:29 -03:00
|
|
|
|
Motor_Running = OUT_Run && !OUT_Stop;
|
2024-05-18 05:53:04 -03:00
|
|
|
|
STATUS_VFD_Trip = false;
|
|
|
|
|
STATUS_VFD_Warning = false;
|
|
|
|
|
STATUS_VFD_Coasting = false;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
_STATUS_VFD_Ready = false;
|
|
|
|
|
Motor_Running = false;
|
|
|
|
|
STATUS_VFD_Trip = true;
|
|
|
|
|
STATUS_VFD_Warning = false;
|
|
|
|
|
STATUS_VFD_Coasting = false;
|
|
|
|
|
}
|
|
|
|
|
|
2024-06-28 14:47:08 -03:00
|
|
|
|
if (Data.VFD_Trip_NC)
|
|
|
|
|
STATUS_VFD_Trip = !STATUS_VFD_Trip;
|
|
|
|
|
|
2025-02-14 10:04:29 -03:00
|
|
|
|
// Pack all status bits and speed into StatusWord
|
|
|
|
|
int statusWord = VMMotorBitPacker.PackStatusWord(
|
|
|
|
|
_STATUS_VFD_Ready,
|
|
|
|
|
Motor_Running,
|
|
|
|
|
STATUS_VFD_Trip,
|
|
|
|
|
STATUS_VFD_Warning,
|
|
|
|
|
STATUS_VFD_Coasting,
|
|
|
|
|
(short)STATUS_VFD_ACT_Speed_Hz
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Write StatusWord in one operation
|
|
|
|
|
plc.EscribirTagDInt($"\"DB MotorSimulate\".Motors[{DB_Motor}].StatusWord", statusWord);
|
2024-05-18 05:53:04 -03:00
|
|
|
|
}
|
|
|
|
|
|
2025-02-13 10:00:47 -03:00
|
|
|
|
private float CalcSpeedRamp(float max_Speed_for_Ramp, float TiempoRampa, float actual, float expected, int TotalMilliseconds)
|
2024-05-18 05:53:04 -03:00
|
|
|
|
{
|
2025-02-13 10:00:47 -03:00
|
|
|
|
float hzIncrementsRamp = (max_Speed_for_Ramp * 10) / (TiempoRampa * (1000.0f / TotalMilliseconds));
|
2024-05-18 05:53:04 -03:00
|
|
|
|
float delta = expected - actual;
|
|
|
|
|
// Conrtolar si la diferencia no es mayor de lo que falta
|
|
|
|
|
if (Math.Abs(hzIncrementsRamp) > Math.Abs(delta))
|
|
|
|
|
hzIncrementsRamp = Math.Abs(delta);
|
|
|
|
|
if (delta < 0)
|
|
|
|
|
return -hzIncrementsRamp;
|
|
|
|
|
else
|
|
|
|
|
return hzIncrementsRamp;
|
|
|
|
|
}
|
|
|
|
|
|
2025-02-13 10:00:47 -03:00
|
|
|
|
public void UpdateSpeed(float max_Speed_for_Ramp, float TiempoRampa, int TotalMilliseconds)
|
2024-05-18 05:53:04 -03:00
|
|
|
|
{
|
|
|
|
|
// Calculamos la velocidad
|
2025-02-13 10:00:47 -03:00
|
|
|
|
STATUS_VFD_ACT_Speed_Hz += CalcSpeedRamp(max_Speed_for_Ramp, TiempoRampa, STATUS_VFD_ACT_Speed_Hz, OUT_OUT_VFD_REQ_Speed_Hz, TotalMilliseconds);
|
2024-05-18 05:53:04 -03:00
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-02-13 10:00:47 -03:00
|
|
|
|
|
|
|
|
|
|
2024-05-11 11:58:55 -03:00
|
|
|
|
}
|