using System; using System.Collections.Generic; using Artemis.Core.Events; using Artemis.Core.Plugins.Models; using Artemis.Core.RGB.NET; using Artemis.Core.Services.Interfaces; using RGB.NET.Brushes; using RGB.NET.Core; using RGB.NET.Groups; using Serilog; namespace Artemis.Core.Services { /// /// Provides wrapped access the RGB.NET /// public class RgbService : IRgbService, IDisposable { private readonly List _loadedDevices; private readonly ILogger _logger; private readonly PluginSetting _renderScaleSetting; private readonly PluginSetting _targetFrameRateSetting; private readonly TimerUpdateTrigger _updateTrigger; private ListLedGroup _surfaceLedGroup; internal RgbService(ILogger logger, ISettingsService settingsService) { _logger = logger; _renderScaleSetting = settingsService.GetSetting("Core.RenderScale", 1.0); _targetFrameRateSetting = settingsService.GetSetting("Core.TargetFrameRate", 25); Surface = RGBSurface.Instance; // Let's throw these for now Surface.Exception += SurfaceOnException; _renderScaleSetting.SettingChanged += RenderScaleSettingOnSettingChanged; _targetFrameRateSetting.SettingChanged += TargetFrameRateSettingOnSettingChanged; _loadedDevices = new List(); _updateTrigger = new TimerUpdateTrigger {UpdateFrequency = 1.0 / _targetFrameRateSetting.Value}; Surface.RegisterUpdateTrigger(_updateTrigger); } /// public RGBSurface Surface { get; set; } public BitmapBrush BitmapBrush { get; private set; } public IReadOnlyCollection LoadedDevices => _loadedDevices.AsReadOnly(); public void AddDeviceProvider(IRGBDeviceProvider deviceProvider) { Surface.LoadDevices(deviceProvider); if (deviceProvider.Devices == null) { _logger.Warning("RgbDevice provider {deviceProvider} has no devices", deviceProvider.GetType().Name); return; } foreach (var surfaceDevice in deviceProvider.Devices) { if (!_loadedDevices.Contains(surfaceDevice)) { _loadedDevices.Add(surfaceDevice); OnDeviceLoaded(new DeviceEventArgs(surfaceDevice)); } else OnDeviceReloaded(new DeviceEventArgs(surfaceDevice)); } } public void Dispose() { Surface.UnregisterUpdateTrigger(_updateTrigger); _updateTrigger.Dispose(); Surface.Dispose(); } private void RenderScaleSettingOnSettingChanged(object sender, EventArgs e) { UpdateSurfaceLedGroup(); } private void TargetFrameRateSettingOnSettingChanged(object sender, EventArgs e) { _updateTrigger.UpdateFrequency = 1.0 / _targetFrameRateSetting.Value; } private void SurfaceOnException(ExceptionEventArgs args) { _logger.Warning("Surface threw e"); throw args.Exception; } #region Events public event EventHandler DeviceLoaded; public event EventHandler DeviceReloaded; public void UpdateSurfaceLedGroup() { if (_surfaceLedGroup == null) { // Apply the application wide brush and decorator BitmapBrush = new BitmapBrush(new Scale(_renderScaleSetting.Value)); _surfaceLedGroup = new ListLedGroup(Surface.Leds) { Brush = BitmapBrush }; return; } lock (_surfaceLedGroup) { // Clean up the old background _surfaceLedGroup.Detach(); // Apply the application wide brush and decorator BitmapBrush.Scale = new Scale(_renderScaleSetting.Value); _surfaceLedGroup = new ListLedGroup(Surface.Leds) { Brush = BitmapBrush }; } lock (BitmapBrush) { } } private void OnDeviceLoaded(DeviceEventArgs e) { DeviceLoaded?.Invoke(this, e); } private void OnDeviceReloaded(DeviceEventArgs e) { DeviceReloaded?.Invoke(this, e); } #endregion } }