mirror of
https://github.com/DarthAffe/RGB.NET.git
synced 2025-12-12 17:48:31 +00:00
186 lines
6.1 KiB
C#
186 lines
6.1 KiB
C#
using OpenRGB.NET;
|
|
using RGB.NET.Core;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
|
|
namespace RGB.NET.Devices.OpenRGB;
|
|
|
|
/// <inheritdoc />
|
|
/// <summary>
|
|
/// Represents a device provider responsible for OpenRGB devices.
|
|
/// </summary>
|
|
public sealed class OpenRGBDeviceProvider : AbstractRGBDeviceProvider
|
|
{
|
|
#region Properties & Fields
|
|
|
|
// ReSharper disable once InconsistentNaming
|
|
private static readonly object _lock = new();
|
|
|
|
private readonly List<OpenRgbClient> _clients = [];
|
|
|
|
private static OpenRGBDeviceProvider? _instance;
|
|
|
|
/// <summary>
|
|
/// Gets the singleton <see cref="OpenRGBDeviceProvider"/> instance.
|
|
/// </summary>
|
|
public static OpenRGBDeviceProvider Instance
|
|
{
|
|
get
|
|
{
|
|
lock (_lock)
|
|
return _instance ?? new OpenRGBDeviceProvider();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a list of all defined device-definitions.
|
|
/// </summary>
|
|
public List<OpenRGBServerDefinition> DeviceDefinitions { get; } = [];
|
|
|
|
/// <summary>
|
|
/// Indicates whether all devices will be added, or just the ones with a 'Direct' mode. Defaults to false.
|
|
/// </summary>
|
|
public bool ForceAddAllDevices { get; set; } = false;
|
|
|
|
/// <summary>
|
|
/// Defines which device types will be separated by zones. Defaults to <see cref="RGBDeviceType.LedStripe" /> | <see cref="RGBDeviceType.Mainboard"/> | <see cref="RGBDeviceType.Speaker" />.
|
|
/// </summary>
|
|
public RGBDeviceType PerZoneDeviceFlag { get; } = RGBDeviceType.LedStripe | RGBDeviceType.Mainboard | RGBDeviceType.Speaker;
|
|
|
|
#endregion
|
|
|
|
#region Constructors
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="OpenRGBDeviceProvider"/> class.
|
|
/// </summary>
|
|
/// <exception cref="InvalidOperationException">Thrown if this constructor is called even if there is already an instance of this class.</exception>
|
|
public OpenRGBDeviceProvider()
|
|
{
|
|
lock (_lock)
|
|
{
|
|
if (_instance != null) throw new InvalidOperationException($"There can be only one instance of type {nameof(OpenRGBDeviceProvider)}");
|
|
_instance = this;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Methods
|
|
|
|
/// <summary>
|
|
/// Adds the specified <see cref="OpenRGBServerDefinition" /> to this device-provider.
|
|
/// </summary>
|
|
/// <param name="deviceDefinition">The <see cref="OpenRGBServerDefinition"/> to add.</param>
|
|
// ReSharper disable once UnusedMember.Global
|
|
public void AddDeviceDefinition(OpenRGBServerDefinition deviceDefinition) => DeviceDefinitions.Add(deviceDefinition);
|
|
|
|
/// <inheritdoc />
|
|
protected override void InitializeSDK()
|
|
{
|
|
foreach (OpenRGBServerDefinition? deviceDefinition in DeviceDefinitions)
|
|
{
|
|
try
|
|
{
|
|
OpenRgbClient openRgb = new(ip: deviceDefinition.Ip, port: deviceDefinition.Port, name: deviceDefinition.ClientName, autoConnect: true);
|
|
_clients.Add(openRgb);
|
|
deviceDefinition.Connected = true;
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
deviceDefinition.Connected = false;
|
|
deviceDefinition.LastError = e.Message;
|
|
Throw(e);
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
protected override IEnumerable<IRGBDevice> LoadDevices()
|
|
{
|
|
foreach (OpenRgbClient? openRgb in _clients)
|
|
{
|
|
int deviceCount = openRgb.GetControllerCount();
|
|
|
|
for (int i = 0; i < deviceCount; i++)
|
|
{
|
|
Device device = openRgb.GetControllerData(i);
|
|
|
|
int directModeIndex = Array.FindIndex(device.Modes, d => d.Name == "Direct");
|
|
if (directModeIndex != -1)
|
|
{
|
|
//set the device to direct mode if it has it
|
|
openRgb.UpdateMode(i, directModeIndex);
|
|
}
|
|
else if (!ForceAddAllDevices)
|
|
{
|
|
//if direct mode does not exist
|
|
//and if we're not forcing, continue to the next device.
|
|
continue;
|
|
}
|
|
|
|
if (device.Zones.Length == 0)
|
|
continue;
|
|
if (device.Zones.All(z => z.LedCount == 0))
|
|
continue;
|
|
|
|
OpenRGBUpdateQueue updateQueue = new(GetUpdateTrigger(), i, openRgb, device);
|
|
|
|
bool anyZoneHasSegments = device.Zones.Any(z => z.Segments.Length > 0);
|
|
bool splitDeviceByZones = anyZoneHasSegments || PerZoneDeviceFlag.HasFlag(Helper.GetRgbNetDeviceType(device.Type));
|
|
|
|
if (!splitDeviceByZones)
|
|
{
|
|
yield return new OpenRGBGenericDevice(new OpenRGBDeviceInfo(device), updateQueue);
|
|
continue;
|
|
}
|
|
|
|
int totalLedCount = 0;
|
|
|
|
foreach (Zone zone in device.Zones)
|
|
{
|
|
if (zone.LedCount <= 0)
|
|
continue;
|
|
|
|
if (zone.Segments.Length <= 0)
|
|
{
|
|
yield return new OpenRGBZoneDevice(new OpenRGBDeviceInfo(device), totalLedCount, zone, updateQueue);
|
|
totalLedCount += (int)zone.LedCount;
|
|
}
|
|
else
|
|
{
|
|
foreach (Segment segment in zone.Segments)
|
|
{
|
|
yield return new OpenRGBSegmentDevice(new OpenRGBDeviceInfo(device), totalLedCount, segment, updateQueue);
|
|
totalLedCount += (int)segment.LedCount;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
protected override void Dispose(bool disposing)
|
|
{
|
|
lock (_lock)
|
|
{
|
|
base.Dispose(disposing);
|
|
|
|
foreach (OpenRgbClient client in _clients)
|
|
{
|
|
try { client.Dispose(); }
|
|
catch { /* at least we tried */ }
|
|
}
|
|
|
|
_clients.Clear();
|
|
DeviceDefinitions.Clear();
|
|
|
|
_instance = null;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
}
|