mirror of
https://github.com/PepperDash/Essentials.git
synced 2026-02-12 11:15:08 +00:00
286 lines
9.4 KiB
C#
286 lines
9.4 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using Crestron.SimplSharp;
|
|
using Crestron.SimplSharpPro.DeviceSupport;
|
|
using PepperDash.Core;
|
|
using PepperDash.Essentials.Core;
|
|
using PepperDash.Essentials.Core.Config;
|
|
using PepperDash.Essentials.Core.Bridges;
|
|
using PepperDash.Essentials.Core.Lighting;
|
|
using LightingBase = PepperDash.Essentials.Core.Lighting.LightingBase;
|
|
|
|
namespace PepperDash.Essentials.Devices.Common.Environment.Lutron
|
|
{
|
|
public class LutronQuantumArea : LightingBase, ILightingMasterRaiseLower, ICommunicationMonitor
|
|
{
|
|
public IBasicCommunication Communication { get; private set; }
|
|
public CommunicationGather PortGather { get; private set; }
|
|
public StatusMonitorBase CommunicationMonitor { get; private set; }
|
|
|
|
CTimer SubscribeAfterLogin;
|
|
|
|
public string IntegrationId;
|
|
string Username;
|
|
string Password;
|
|
|
|
const string Delimiter = "\x0d\x0a";
|
|
const string Set = "#";
|
|
const string Get = "?";
|
|
|
|
public LutronQuantumArea(string key, string name, IBasicCommunication comm, LutronQuantumPropertiesConfig props)
|
|
: base(key, name)
|
|
{
|
|
Communication = comm;
|
|
|
|
IntegrationId = props.IntegrationId;
|
|
|
|
if (props.Control.Method != eControlMethod.Com)
|
|
{
|
|
|
|
Username = props.Control.TcpSshProperties.Username;
|
|
Password = props.Control.TcpSshProperties.Password;
|
|
}
|
|
|
|
|
|
LightingScenes = props.Scenes;
|
|
|
|
var socket = comm as ISocketStatus;
|
|
if (socket != null)
|
|
{
|
|
// IP Control
|
|
socket.ConnectionChange += new EventHandler<GenericSocketStatusChageEventArgs>(socket_ConnectionChange);
|
|
}
|
|
else
|
|
{
|
|
// RS-232 Control
|
|
}
|
|
|
|
Communication.TextReceived += new EventHandler<GenericCommMethodReceiveTextArgs>(Communication_TextReceived);
|
|
|
|
PortGather = new CommunicationGather(Communication, Delimiter);
|
|
PortGather.LineReceived += new EventHandler<GenericCommMethodReceiveTextArgs>(PortGather_LineReceived);
|
|
|
|
if (props.CommunicationMonitorProperties != null)
|
|
{
|
|
CommunicationMonitor = new GenericCommunicationMonitor(this, Communication, props.CommunicationMonitorProperties);
|
|
}
|
|
else
|
|
{
|
|
CommunicationMonitor = new GenericCommunicationMonitor(this, Communication, 120000, 120000, 300000, "?ETHERNET,0\x0d\x0a");
|
|
}
|
|
}
|
|
|
|
public override bool CustomActivate()
|
|
{
|
|
Communication.Connect();
|
|
CommunicationMonitor.StatusChange += (o, a) => { Debug.Console(2, this, "Communication monitor state: {0}", CommunicationMonitor.Status); };
|
|
CommunicationMonitor.Start();
|
|
|
|
return true;
|
|
}
|
|
|
|
public override void LinkToApi(BasicTriList trilist, uint joinStart, string joinMapKey, EiscApiAdvanced bridge)
|
|
{
|
|
var joinMap = LinkLightingToApi(this, trilist, joinStart, joinMapKey, bridge);
|
|
|
|
CommunicationMonitor.IsOnlineFeedback.LinkInputSig(trilist.BooleanInput[joinMap.IsOnline.JoinNumber]);
|
|
trilist.SetStringSigAction(joinMap.IntegrationIdSet.JoinNumber , s => IntegrationId = s);
|
|
}
|
|
|
|
void socket_ConnectionChange(object sender, GenericSocketStatusChageEventArgs e)
|
|
{
|
|
Debug.Console(2, this, "Socket Status Change: {0}", e.Client.ClientStatus.ToString());
|
|
|
|
if (e.Client.IsConnected)
|
|
{
|
|
// Tasks on connect
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Checks for responses that do not contain the delimiter
|
|
/// </summary>
|
|
/// <param name="sender"></param>
|
|
/// <param name="args"></param>
|
|
void Communication_TextReceived(object sender, GenericCommMethodReceiveTextArgs args)
|
|
{
|
|
Debug.Console(2, this, "Text Received: '{0}'", args.Text);
|
|
|
|
if (args.Text.Contains("login:"))
|
|
{
|
|
// Login
|
|
SendLine(Username);
|
|
}
|
|
else if (args.Text.Contains("password:"))
|
|
{
|
|
// Login
|
|
SendLine(Password);
|
|
SubscribeAfterLogin = new CTimer(x => SubscribeToFeedback(), null, 5000);
|
|
|
|
}
|
|
else if (args.Text.Contains("Access Granted"))
|
|
{
|
|
if (SubscribeAfterLogin != null)
|
|
{
|
|
SubscribeAfterLogin.Stop();
|
|
}
|
|
SubscribeToFeedback();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Handles all responses that contain the delimiter
|
|
/// </summary>
|
|
/// <param name="sender"></param>
|
|
/// <param name="args"></param>
|
|
void PortGather_LineReceived(object sender, GenericCommMethodReceiveTextArgs args)
|
|
{
|
|
Debug.Console(2, this, "Line Received: '{0}'", args.Text);
|
|
|
|
try
|
|
{
|
|
if (args.Text.Contains("~AREA"))
|
|
{
|
|
var response = args.Text.Split(',');
|
|
|
|
var integrationId = response[1];
|
|
|
|
if (integrationId != IntegrationId)
|
|
{
|
|
Debug.Console(2, this, "Response is not for correct Integration ID");
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
var action = Int32.Parse(response[2]);
|
|
|
|
switch (action)
|
|
{
|
|
case (int)eAction.Scene:
|
|
{
|
|
var scene = response[3];
|
|
CurrentLightingScene = LightingScenes.FirstOrDefault(s => s.ID.Equals(scene));
|
|
|
|
OnLightingSceneChange();
|
|
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Debug.Console(2, this, "Error parsing response:\n{0}", e);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Subscribes to feedback
|
|
/// </summary>
|
|
public void SubscribeToFeedback()
|
|
{
|
|
Debug.Console(1, "Sending Monitoring Subscriptions");
|
|
SendLine("#MONITORING,6,1");
|
|
SendLine("#MONITORING,8,1");
|
|
SendLine("#MONITORING,5,2");
|
|
}
|
|
|
|
/// <summary>
|
|
/// Recalls the specified scene
|
|
/// </summary>
|
|
/// <param name="scene"></param>
|
|
///
|
|
|
|
public override void SelectScene(LightingScene scene)
|
|
{
|
|
Debug.Console(1, this, "Selecting Scene: '{0}'", scene.Name);
|
|
SendLine(string.Format("{0}AREA,{1},{2},{3}", Set, IntegrationId, (int)eAction.Scene, scene.ID));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Begins raising the lights in the area
|
|
/// </summary>
|
|
public void MasterRaise()
|
|
{
|
|
SendLine(string.Format("{0}AREA,{1},{2}", Set, IntegrationId, (int)eAction.Raise));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Begins lowering the lights in the area
|
|
/// </summary>
|
|
public void MasterLower()
|
|
{
|
|
SendLine(string.Format("{0}AREA,{1},{2}", Set, IntegrationId, (int)eAction.Lower));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Stops the current raise/lower action
|
|
/// </summary>
|
|
public void MasterRaiseLowerStop()
|
|
{
|
|
SendLine(string.Format("{0}AREA,{1},{2}", Set, IntegrationId, (int)eAction.Stop));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Appends the delimiter and sends the string
|
|
/// </summary>
|
|
/// <param name="s"></param>
|
|
public void SendLine(string s)
|
|
{
|
|
Communication.SendText(s + Delimiter);
|
|
}
|
|
}
|
|
|
|
public enum eAction : int
|
|
{
|
|
SetLevel = 1,
|
|
Raise = 2,
|
|
Lower = 3,
|
|
Stop = 4,
|
|
Scene = 6,
|
|
DaylightMode = 7,
|
|
OccupancyState = 8,
|
|
OccupancyMode = 9,
|
|
OccupiedLevelOrScene = 12,
|
|
UnoccupiedLevelOrScene = 13,
|
|
HyperionShaddowSensorOverrideState = 26,
|
|
HyperionBrightnessSensorOverrideStatue = 27
|
|
}
|
|
|
|
public class LutronQuantumPropertiesConfig
|
|
{
|
|
public CommunicationMonitorConfig CommunicationMonitorProperties { get; set; }
|
|
public ControlPropertiesConfig Control { get; set; }
|
|
|
|
public string IntegrationId { get; set; }
|
|
public List<LightingScene> Scenes { get; set; }
|
|
|
|
// Moved to use existing properties in Control object
|
|
// public string Username { get; set; }
|
|
// public string Password { get; set; }
|
|
}
|
|
|
|
public class LutronQuantumAreaFactory : EssentialsDeviceFactory<LutronQuantumArea>
|
|
{
|
|
public LutronQuantumAreaFactory()
|
|
{
|
|
TypeNames = new List<string>() { "lutronqs" };
|
|
}
|
|
|
|
public override EssentialsDevice BuildDevice(DeviceConfig dc)
|
|
{
|
|
Debug.Console(1, "Factory Attempting to create new LutronQuantumArea Device");
|
|
var comm = CommFactory.CreateCommForDevice(dc);
|
|
|
|
var props = Newtonsoft.Json.JsonConvert.DeserializeObject<Environment.Lutron.LutronQuantumPropertiesConfig>(dc.Properties.ToString());
|
|
|
|
return new LutronQuantumArea(dc.Key, dc.Name, comm, props);
|
|
}
|
|
}
|
|
|
|
} |