mirror of
https://github.com/PepperDash/Essentials.git
synced 2026-01-11 19:44:52 +00:00
654 lines
27 KiB
C#
654 lines
27 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using Crestron.SimplSharp;
|
|
using Crestron.SimplSharp.CrestronIO;
|
|
using Crestron.SimplSharpPro;
|
|
using Crestron.SimplSharpPro.CrestronThread;
|
|
using Crestron.SimplSharpPro.Diagnostics;
|
|
using Crestron.SimplSharp.Reflection;
|
|
|
|
using PepperDash.Core;
|
|
using PepperDash.Essentials.Core;
|
|
using PepperDash.Essentials.Core.Bridges;
|
|
using PepperDash.Essentials.Core.Config;
|
|
using PepperDash.Essentials.Core.Fusion;
|
|
using PepperDash.Essentials.Devices.Common;
|
|
using PepperDash.Essentials.DM;
|
|
using PepperDash.Essentials.Fusion;
|
|
using PepperDash.Essentials.Room.Config;
|
|
|
|
using Newtonsoft.Json;
|
|
using PepperDash.Essentials.Core.DeviceTypeInterfaces;
|
|
|
|
namespace PepperDash.Essentials
|
|
{
|
|
public class ControlSystem : CrestronControlSystem
|
|
{
|
|
HttpLogoServer LogoServer;
|
|
|
|
private CTimer _startTimer;
|
|
private CEvent _initializeEvent;
|
|
private const long StartupTime = 500;
|
|
|
|
public ControlSystem()
|
|
: base()
|
|
{
|
|
Thread.MaxNumberOfUserThreads = 400;
|
|
Global.ControlSystem = this;
|
|
DeviceManager.Initialize(this);
|
|
SecretsManager.Initialize();
|
|
SystemMonitor.ProgramInitialization.ProgramInitializationUnderUserControl = true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Entry point for the program
|
|
/// </summary>
|
|
public override void InitializeSystem()
|
|
{
|
|
_startTimer = new CTimer(StartSystem,StartupTime);
|
|
|
|
|
|
// If the control system is a DMPS type, we need to wait to exit this method until all devices have had time to activate
|
|
// to allow any HD-BaseT DM endpoints to register first.
|
|
if (Global.ControlSystemIsDmpsType)
|
|
{
|
|
Debug.Console(2, "******************* InitializeSystem() Entering **********************");
|
|
|
|
_initializeEvent = new CEvent();
|
|
|
|
DeviceManager.AllDevicesActivated += (o, a) =>
|
|
{
|
|
_initializeEvent.Set();
|
|
Debug.Console(2, "******************* InitializeSystem() Exiting **********************");
|
|
};
|
|
|
|
_initializeEvent.Wait(30000);
|
|
}
|
|
}
|
|
|
|
private void StartSystem(object obj)
|
|
{
|
|
DeterminePlatform();
|
|
|
|
if (Debug.DoNotLoadOnNextBoot)
|
|
{
|
|
CrestronConsole.AddNewConsoleCommand(s => CrestronInvoke.BeginInvoke((o) => GoWithLoad()), "go", "Loads configuration file",
|
|
ConsoleAccessLevelEnum.AccessOperator);
|
|
}
|
|
|
|
CrestronConsole.AddNewConsoleCommand(PluginLoader.ReportAssemblyVersions, "reportversions", "Reports the versions of the loaded assemblies", ConsoleAccessLevelEnum.AccessOperator);
|
|
|
|
CrestronConsole.AddNewConsoleCommand(PepperDash.Essentials.Core.DeviceFactory.GetDeviceFactoryTypes, "gettypes", "Gets the device types that can be built. Accepts a filter string.", ConsoleAccessLevelEnum.AccessOperator);
|
|
|
|
CrestronConsole.AddNewConsoleCommand(BridgeHelper.PrintJoinMap, "getjoinmap", "map(s) for bridge or device on bridge [brKey [devKey]]", ConsoleAccessLevelEnum.AccessOperator);
|
|
|
|
CrestronConsole.AddNewConsoleCommand(s =>
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "CONSOLE MESSAGE: {0}", s);
|
|
}, "appdebugmessage", "Writes message to log", ConsoleAccessLevelEnum.AccessOperator);
|
|
|
|
CrestronConsole.AddNewConsoleCommand(s =>
|
|
{
|
|
foreach (var tl in TieLineCollection.Default)
|
|
CrestronConsole.ConsoleCommandResponse(" {0}\r\n", tl);
|
|
},
|
|
"listtielines", "Prints out all tie lines", ConsoleAccessLevelEnum.AccessOperator);
|
|
|
|
CrestronConsole.AddNewConsoleCommand(s =>
|
|
{
|
|
CrestronConsole.ConsoleCommandResponse
|
|
("Current running configuration. This is the merged system and template configuration");
|
|
CrestronConsole.ConsoleCommandResponse(Newtonsoft.Json.JsonConvert.SerializeObject
|
|
(ConfigReader.ConfigObject, Newtonsoft.Json.Formatting.Indented));
|
|
}, "showconfig", "Shows the current running merged config", ConsoleAccessLevelEnum.AccessOperator);
|
|
|
|
CrestronConsole.AddNewConsoleCommand(s =>
|
|
{
|
|
CrestronConsole.ConsoleCommandResponse("This system can be found at the following URLs:\r\n" +
|
|
"System URL: {0}\r\n" +
|
|
"Template URL: {1}", ConfigReader.ConfigObject.SystemUrl, ConfigReader.ConfigObject.TemplateUrl);
|
|
}, "portalinfo", "Shows portal URLS from configuration", ConsoleAccessLevelEnum.AccessOperator);
|
|
|
|
|
|
CrestronConsole.AddNewConsoleCommand(DeviceManager.GetRoutingPorts,
|
|
"getroutingports", "Reports all routing ports, if any. Requires a device key", ConsoleAccessLevelEnum.AccessOperator);
|
|
|
|
if (!Debug.DoNotLoadOnNextBoot)
|
|
{
|
|
GoWithLoad();
|
|
return;
|
|
}
|
|
|
|
SystemMonitor.ProgramInitialization.ProgramInitializationComplete = true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Determines if the program is running on a processor (appliance) or server (VC-4).
|
|
///
|
|
/// Sets Global.FilePathPrefix and Global.ApplicationDirectoryPathPrefix based on platform
|
|
/// </summary>
|
|
public void DeterminePlatform()
|
|
{
|
|
try
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Determining Platform....");
|
|
|
|
string filePathPrefix;
|
|
|
|
var dirSeparator = Global.DirectorySeparator;
|
|
|
|
string directoryPrefix;
|
|
|
|
directoryPrefix = Crestron.SimplSharp.CrestronIO.Directory.GetApplicationRootDirectory();
|
|
|
|
var fullVersion = Assembly.GetExecutingAssembly().GetCustomAttributes(typeof(AssemblyInformationalVersionAttribute), false);
|
|
|
|
AssemblyInformationalVersionAttribute fullVersionAtt = fullVersion[0] as AssemblyInformationalVersionAttribute;
|
|
|
|
Global.SetAssemblyVersion(fullVersionAtt.InformationalVersion);
|
|
|
|
if (CrestronEnvironment.DevicePlatform != eDevicePlatform.Server) // Handles 3-series running Windows CE OS
|
|
{
|
|
string userFolder;
|
|
string nvramFolder;
|
|
bool is4series = false;
|
|
|
|
if (eCrestronSeries.Series4 == (Global.ProcessorSeries & eCrestronSeries.Series4)) // Handle 4-series
|
|
{
|
|
is4series = true;
|
|
// Set path to user/
|
|
userFolder = "user";
|
|
nvramFolder = "nvram";
|
|
}
|
|
else
|
|
{
|
|
userFolder = "User";
|
|
nvramFolder = "Nvram";
|
|
}
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Starting Essentials v{0} on {1} Appliance", Global.AssemblyVersion, is4series ? "4-series" : "3-series");
|
|
|
|
// Check if User/ProgramX exists
|
|
if (Directory.Exists(Global.ApplicationDirectoryPathPrefix + dirSeparator + userFolder
|
|
+ dirSeparator + string.Format("program{0}", InitialParametersClass.ApplicationNumber)))
|
|
{
|
|
Debug.Console(0, @"{0}/program{1} directory found", userFolder, InitialParametersClass.ApplicationNumber);
|
|
filePathPrefix = directoryPrefix + dirSeparator + userFolder
|
|
+ dirSeparator + string.Format("program{0}", InitialParametersClass.ApplicationNumber) + dirSeparator;
|
|
}
|
|
// Check if Nvram/Programx exists
|
|
else if (Directory.Exists(directoryPrefix + dirSeparator + nvramFolder
|
|
+ dirSeparator + string.Format("program{0}", InitialParametersClass.ApplicationNumber)))
|
|
{
|
|
Debug.Console(0, @"{0}/program{1} directory found", nvramFolder, InitialParametersClass.ApplicationNumber);
|
|
filePathPrefix = directoryPrefix + dirSeparator + nvramFolder
|
|
+ dirSeparator + string.Format("program{0}", InitialParametersClass.ApplicationNumber) + dirSeparator;
|
|
}
|
|
// If neither exists, set path to User/ProgramX
|
|
else
|
|
{
|
|
Debug.Console(0, @"No previous directory found. Using {0}/program{1}", userFolder, InitialParametersClass.ApplicationNumber);
|
|
filePathPrefix = directoryPrefix + dirSeparator + userFolder
|
|
+ dirSeparator + string.Format("program{0}", InitialParametersClass.ApplicationNumber) + dirSeparator;
|
|
}
|
|
}
|
|
else // Handles Linux OS (Virtual Control)
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Starting Essentials v{0} on Virtual Control Server", Global.AssemblyVersion);
|
|
|
|
// Set path to User/
|
|
filePathPrefix = directoryPrefix + dirSeparator + "User" + dirSeparator;
|
|
}
|
|
|
|
Global.SetFilePathPrefix(filePathPrefix);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Debug.Console(0, "Unable to Determine Platform due to Exception: {0}", e.Message);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Begins the process of loading resources including plugins and configuration data
|
|
/// </summary>
|
|
public void GoWithLoad()
|
|
{
|
|
try
|
|
{
|
|
Debug.SetDoNotLoadOnNextBoot(false);
|
|
|
|
PluginLoader.AddProgramAssemblies();
|
|
|
|
new Core.DeviceFactory();
|
|
new Devices.Common.DeviceFactory();
|
|
new DM.DeviceFactory();
|
|
new DeviceFactory();
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Starting Essentials load from configuration");
|
|
|
|
var filesReady = SetupFilesystem();
|
|
if (filesReady)
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Checking for plugins");
|
|
PluginLoader.LoadPlugins();
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Folder structure verified. Loading config...");
|
|
if (!ConfigReader.LoadConfig2())
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Error, "Essentials Load complete with errors");
|
|
return;
|
|
}
|
|
|
|
Load();
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Essentials load complete\r\n" +
|
|
"-------------------------------------------------------------");
|
|
}
|
|
else
|
|
{
|
|
Debug.Console(0,
|
|
@"----------------------------------------------
|
|
------------------------------------------------
|
|
------------------------------------------------
|
|
Essentials file structure setup completed.
|
|
Please load config, sgd and ir files and
|
|
restart program.
|
|
------------------------------------------------
|
|
------------------------------------------------
|
|
------------------------------------------------");
|
|
}
|
|
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Debug.Console(0, "FATAL INITIALIZE ERROR. System is in an inconsistent state:\r\n{0}", e);
|
|
}
|
|
finally
|
|
{
|
|
// Notify the OS that the program intitialization has completed
|
|
SystemMonitor.ProgramInitialization.ProgramInitializationComplete = true;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
/// Verifies filesystem is set up. IR, SGD, and programX folders
|
|
/// </summary>
|
|
bool SetupFilesystem()
|
|
{
|
|
Debug.Console(0, "Verifying and/or creating folder structure");
|
|
var configDir = Global.FilePathPrefix;
|
|
var configExists = Directory.Exists(configDir);
|
|
if (!configExists)
|
|
Directory.Create(configDir);
|
|
|
|
var irDir = Global.FilePathPrefix + "ir";
|
|
if (!Directory.Exists(irDir))
|
|
Directory.Create(irDir);
|
|
|
|
var sgdDir = Global.FilePathPrefix + "sgd";
|
|
if (!Directory.Exists(sgdDir))
|
|
Directory.Create(sgdDir);
|
|
|
|
var pluginDir = Global.FilePathPrefix + "plugins";
|
|
if (!Directory.Exists(pluginDir))
|
|
Directory.Create(pluginDir);
|
|
|
|
return configExists;
|
|
}
|
|
|
|
/// <summary>
|
|
///
|
|
/// </summary>
|
|
public void TearDown()
|
|
{
|
|
Debug.Console(0, "Tearing down existing system");
|
|
DeviceManager.DeactivateAll();
|
|
|
|
TieLineCollection.Default.Clear();
|
|
|
|
foreach (var key in DeviceManager.GetDevices())
|
|
DeviceManager.RemoveDevice(key);
|
|
|
|
Debug.Console(0, "Tear down COMPLETE");
|
|
}
|
|
|
|
/// <summary>
|
|
///
|
|
/// </summary>
|
|
void Load()
|
|
{
|
|
LoadDevices();
|
|
LoadTieLines();
|
|
LoadRooms();
|
|
LoadLogoServer();
|
|
|
|
DeviceManager.ActivateAll();
|
|
|
|
var mobileControl = GetMobileControlDevice();
|
|
|
|
if (mobileControl == null) return;
|
|
|
|
mobileControl.LinkSystemMonitorToAppServer();
|
|
|
|
}
|
|
|
|
/// <summary>
|
|
/// Reads all devices from config and adds them to DeviceManager
|
|
/// </summary>
|
|
public void LoadDevices()
|
|
{
|
|
|
|
// Build the processor wrapper class
|
|
DeviceManager.AddDevice(new PepperDash.Essentials.Core.Devices.CrestronProcessor("processor"));
|
|
|
|
// Add global System Monitor device
|
|
if (CrestronEnvironment.DevicePlatform == eDevicePlatform.Appliance)
|
|
{
|
|
DeviceManager.AddDevice(
|
|
new PepperDash.Essentials.Core.Monitoring.SystemMonitorController("systemMonitor"));
|
|
}
|
|
|
|
foreach (var devConf in ConfigReader.ConfigObject.Devices)
|
|
{
|
|
|
|
try
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Creating device '{0}', type '{1}'", devConf.Key, devConf.Type);
|
|
// Skip this to prevent unnecessary warnings
|
|
if (devConf.Key == "processor")
|
|
{
|
|
var prompt = Global.ControlSystem.ControllerPrompt;
|
|
|
|
var typeMatch = String.Equals(devConf.Type, prompt, StringComparison.OrdinalIgnoreCase) ||
|
|
String.Equals(devConf.Type, prompt.Replace("-", ""), StringComparison.OrdinalIgnoreCase);
|
|
|
|
if (!typeMatch)
|
|
Debug.Console(0,
|
|
"WARNING: Config file defines processor type as '{0}' but actual processor is '{1}'! Some ports may not be available",
|
|
devConf.Type.ToUpper(), Global.ControlSystem.ControllerPrompt.ToUpper());
|
|
|
|
// Check if the processor is a DMPS model
|
|
if (this.ControllerPrompt.IndexOf("dmps", StringComparison.OrdinalIgnoreCase) > -1)
|
|
{
|
|
Debug.Console(2, "Adding DmpsRoutingController for {0} to Device Manager.", this.ControllerPrompt);
|
|
|
|
var propertiesConfig = JsonConvert.DeserializeObject<DM.Config.DmpsRoutingPropertiesConfig>(devConf.Properties.ToString());
|
|
|
|
if(propertiesConfig == null)
|
|
propertiesConfig = new DM.Config.DmpsRoutingPropertiesConfig();
|
|
|
|
DeviceManager.AddDevice(DmpsRoutingController.GetDmpsRoutingController("processor-avRouting", this.ControllerPrompt, propertiesConfig));
|
|
}
|
|
else if (this.ControllerPrompt.IndexOf("mpc3", StringComparison.OrdinalIgnoreCase) > -1)
|
|
{
|
|
Debug.Console(2, "MPC3 processor type detected. Adding Mpc3TouchpanelController.");
|
|
|
|
var butToken = devConf.Properties["buttons"];
|
|
if (butToken != null)
|
|
{
|
|
var buttons = butToken.ToObject<Dictionary<string, Essentials.Core.Touchpanels.KeypadButton>>();
|
|
var tpController = new Essentials.Core.Touchpanels.Mpc3TouchpanelController(devConf.Key, devConf.Name, Global.ControlSystem, buttons);
|
|
DeviceManager.AddDevice(tpController);
|
|
}
|
|
else
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Error, "Error: Unable to deserialize buttons collection for device: {0}", devConf.Key);
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
Debug.Console(2, "************Processor is not DMPS type***************");
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
}
|
|
|
|
// Try local factories first
|
|
IKeyed newDev = null;
|
|
|
|
if (newDev == null)
|
|
newDev = PepperDash.Essentials.Core.DeviceFactory.GetDevice(devConf);
|
|
|
|
if (newDev != null)
|
|
DeviceManager.AddDevice(newDev);
|
|
else
|
|
Debug.Console(0, Debug.ErrorLogLevel.Error, "ERROR: Cannot load unknown device type '{0}', key '{1}'.", devConf.Type, devConf.Key);
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Error, "ERROR: Creating device {0}. Skipping device. \r{1}", devConf.Key, e);
|
|
}
|
|
}
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "All Devices Loaded.");
|
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Helper method to load tie lines. This should run after devices have loaded
|
|
/// </summary>
|
|
public void LoadTieLines()
|
|
{
|
|
// In the future, we can't necessarily just clear here because devices
|
|
// might be making their own internal sources/tie lines
|
|
|
|
var tlc = TieLineCollection.Default;
|
|
//tlc.Clear();
|
|
if (ConfigReader.ConfigObject.TieLines == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
foreach (var tieLineConfig in ConfigReader.ConfigObject.TieLines)
|
|
{
|
|
var newTL = tieLineConfig.GetTieLine();
|
|
if (newTL != null)
|
|
tlc.Add(newTL);
|
|
}
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "All Tie Lines Loaded.");
|
|
|
|
}
|
|
|
|
/// <summary>
|
|
/// Reads all rooms from config and adds them to DeviceManager
|
|
/// </summary>
|
|
public void LoadRooms()
|
|
{
|
|
if (ConfigReader.ConfigObject.Rooms == null)
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Notice: Configuration contains no rooms - Is this intentional? This may be a valid configuration.");
|
|
return;
|
|
}
|
|
|
|
uint fusionIpId = 0xf1;
|
|
|
|
foreach (var roomConfig in ConfigReader.ConfigObject.Rooms)
|
|
{
|
|
var room = EssentialsRoomConfigHelper.GetRoomObject(roomConfig) as IEssentialsRoom;
|
|
if (room != null)
|
|
{
|
|
// default to no join map key
|
|
string fusionJoinMapKey = string.Empty;
|
|
|
|
if (room.Config.Properties["fusion"] != null)
|
|
{
|
|
Debug.Console(2, "Custom Fusion config found. Using custom values");
|
|
|
|
var fusionConfig = room.Config.Properties["fusion"].ToObject<EssentialsRoomFusionConfig>();
|
|
|
|
if (fusionConfig != null)
|
|
{
|
|
fusionIpId = fusionConfig.IpIdInt;
|
|
fusionJoinMapKey = fusionConfig.JoinMapKey;
|
|
}
|
|
}
|
|
|
|
if (room is IEssentialsHuddleSpaceRoom)
|
|
{
|
|
DeviceManager.AddDevice(room);
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Room is EssentialsHuddleSpaceRoom, attempting to add to DeviceManager with Fusion with IP-ID {0:X2}", fusionIpId);
|
|
DeviceManager.AddDevice(new Core.Fusion.EssentialsHuddleSpaceFusionSystemControllerBase(room, fusionIpId, fusionJoinMapKey));
|
|
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Attempting to build Mobile Control Bridge...");
|
|
|
|
CreateMobileControlBridge(room);
|
|
}
|
|
else if (room is IEssentialsHuddleVtc1Room)
|
|
{
|
|
DeviceManager.AddDevice(room);
|
|
|
|
if (!(room is EssentialsCombinedHuddleVtc1Room))
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Room is EssentialsHuddleVtc1Room, attempting to add to DeviceManager with Fusion with IP-ID {0:X2}", fusionIpId);
|
|
DeviceManager.AddDevice(new EssentialsHuddleVtc1FusionController((IEssentialsHuddleVtc1Room)room, fusionIpId, fusionJoinMapKey));
|
|
}
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Attempting to build Mobile Control Bridge...");
|
|
|
|
CreateMobileControlBridge(room);
|
|
}
|
|
else if (room is EssentialsTechRoom)
|
|
{
|
|
DeviceManager.AddDevice(room);
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice,
|
|
"Room is EssentialsTechRoom, Attempting to add to DeviceManager with Fusion with IP-ID {0:X2}", fusionIpId);
|
|
DeviceManager.AddDevice(new EssentialsTechRoomFusionSystemController((EssentialsTechRoom)room, fusionIpId, fusionJoinMapKey));
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Attempting to build Mobile Control Bridge");
|
|
|
|
CreateMobileControlBridge(room);
|
|
}
|
|
else
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Room is NOT EssentialsRoom, attempting to add to DeviceManager w/o Fusion");
|
|
DeviceManager.AddDevice(room);
|
|
}
|
|
|
|
fusionIpId += 1;
|
|
}
|
|
else
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Notice: Cannot create room from config, key '{0}' - Is this intentional? This may be a valid configuration.", roomConfig.Key);
|
|
|
|
}
|
|
}
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "All Rooms Loaded.");
|
|
|
|
}
|
|
|
|
private static void CreateMobileControlBridge(object room)
|
|
{
|
|
var mobileControl = GetMobileControlDevice();
|
|
|
|
if (mobileControl == null) return;
|
|
|
|
var mobileControl3 = mobileControl as IMobileControl3;
|
|
|
|
if (mobileControl3 != null)
|
|
{
|
|
mobileControl3.CreateMobileControlRoomBridge(room as IEssentialsRoom, mobileControl);
|
|
}
|
|
else
|
|
{
|
|
mobileControl.CreateMobileControlRoomBridge(room as EssentialsRoomBase, mobileControl);
|
|
}
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Mobile Control Bridge Added...");
|
|
}
|
|
|
|
private static IMobileControl GetMobileControlDevice()
|
|
{
|
|
var mobileControlList = DeviceManager.AllDevices.OfType<IMobileControl>().ToList();
|
|
|
|
if (mobileControlList.Count > 1)
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Warning,
|
|
"Multiple instances of Mobile Control Server found.");
|
|
|
|
return null;
|
|
}
|
|
|
|
if (mobileControlList.Count > 0)
|
|
{
|
|
return mobileControlList[0];
|
|
}
|
|
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "Mobile Control not enabled for this system");
|
|
return null;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Fires up a logo server if not already running
|
|
/// </summary>
|
|
void LoadLogoServer()
|
|
{
|
|
if (ConfigReader.ConfigObject.Rooms == null)
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "No rooms configured. Bypassing Logo server startup.");
|
|
return;
|
|
}
|
|
|
|
if (
|
|
!ConfigReader.ConfigObject.Rooms.Any(
|
|
CheckRoomConfig))
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "No rooms configured to use system Logo server. Bypassing Logo server startup");
|
|
return;
|
|
}
|
|
|
|
try
|
|
{
|
|
LogoServer = new HttpLogoServer(8080, Global.DirectorySeparator + "html" + Global.DirectorySeparator + "logo");
|
|
}
|
|
catch (Exception)
|
|
{
|
|
Debug.Console(0, Debug.ErrorLogLevel.Notice, "NOTICE: Logo server cannot be started. Likely already running in another program");
|
|
}
|
|
}
|
|
|
|
private bool CheckRoomConfig(DeviceConfig c)
|
|
{
|
|
string logoDark = null;
|
|
string logoLight = null;
|
|
string logo = null;
|
|
|
|
try
|
|
{
|
|
if (c.Properties["logoDark"] != null)
|
|
{
|
|
logoDark = c.Properties["logoDark"].Value<string>("type");
|
|
}
|
|
|
|
if (c.Properties["logoLight"] != null)
|
|
{
|
|
logoLight = c.Properties["logoLight"].Value<string>("type");
|
|
}
|
|
|
|
if (c.Properties["logo"] != null)
|
|
{
|
|
logo = c.Properties["logo"].Value<string>("type");
|
|
}
|
|
|
|
return ((logoDark != null && logoDark == "system") ||
|
|
(logoLight != null && logoLight == "system") || (logo != null && logo == "system"));
|
|
}
|
|
catch
|
|
{
|
|
Debug.Console(1, Debug.ErrorLogLevel.Notice, "Unable to find logo information in any room config");
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|