mirror of
https://github.com/CalcProgrammer1/OpenRGB.git
synced 2025-12-24 07:47:49 -05:00
233 lines
7.1 KiB
C++
233 lines
7.1 KiB
C++
/*---------------------------------------------------------*\
|
|
| RGBController_DMX.cpp |
|
|
| |
|
|
| RGBController for DMX devices |
|
|
| |
|
|
| Adam Honse (CalcProgrammer1) 30 Apr 2023 |
|
|
| |
|
|
| This file is part of the OpenRGB project |
|
|
| SPDX-License-Identifier: GPL-2.0-only |
|
|
\*---------------------------------------------------------*/
|
|
|
|
#include <math.h>
|
|
#include "LogManager.h"
|
|
#include "RGBController_DMX.h"
|
|
|
|
using namespace std::chrono_literals;
|
|
|
|
/**------------------------------------------------------------------*\
|
|
@name DMX Devices
|
|
@category LEDStrip
|
|
@type Serial
|
|
@save :x:
|
|
@direct :white_check_mark:
|
|
@effects :x:
|
|
@detectors DetectDMXControllers
|
|
@comment
|
|
\*-------------------------------------------------------------------*/
|
|
|
|
RGBController_DMX::RGBController_DMX(std::vector<DMXDevice> device_list)
|
|
{
|
|
devices = device_list;
|
|
|
|
name = "DMX Device Group";
|
|
type = DEVICE_TYPE_LEDSTRIP;
|
|
description = "DMX Device";
|
|
location = "DMX: " + devices[0].port;
|
|
|
|
/*-----------------------------------------*\
|
|
| If this controller only represents a |
|
|
| single device, use the device name for the|
|
|
| controller name |
|
|
\*-----------------------------------------*/
|
|
if(devices.size() == 1)
|
|
{
|
|
name = devices[0].name;
|
|
}
|
|
|
|
/*-----------------------------------------*\
|
|
| Open OpenDMX port |
|
|
\*-----------------------------------------*/
|
|
port = new serial_port(devices[0].port.c_str(), 250000, SERIAL_PORT_PARITY_NONE, SERIAL_PORT_SIZE_8, SERIAL_PORT_STOP_BITS_2, false);
|
|
|
|
/*-----------------------------------------*\
|
|
| Clear the RTS signal, which enables the |
|
|
| OpenDMX RS-485 drive enable |
|
|
\*-----------------------------------------*/
|
|
port->serial_set_rts(false);
|
|
|
|
/*-----------------------------------------*\
|
|
| Set up modes |
|
|
\*-----------------------------------------*/
|
|
mode Direct;
|
|
Direct.name = "Direct";
|
|
Direct.value = 0;
|
|
Direct.flags = MODE_FLAG_HAS_PER_LED_COLOR | MODE_FLAG_HAS_BRIGHTNESS;
|
|
Direct.color_mode = MODE_COLORS_PER_LED;
|
|
Direct.brightness = 255;
|
|
Direct.brightness_min = 0;
|
|
Direct.brightness_max = 255;
|
|
modes.push_back(Direct);
|
|
|
|
keepalive_delay = 0ms;
|
|
|
|
SetupZones();
|
|
|
|
for (std::size_t device_idx = 0; device_idx < devices.size(); device_idx++)
|
|
{
|
|
/*-----------------------------------------*\
|
|
| Update keepalive delay |
|
|
\*-----------------------------------------*/
|
|
if(devices[device_idx].keepalive_time > 0)
|
|
{
|
|
if(keepalive_delay.count() == 0 || keepalive_delay.count() > devices[device_idx].keepalive_time)
|
|
{
|
|
keepalive_delay = std::chrono::milliseconds(devices[device_idx].keepalive_time);
|
|
}
|
|
}
|
|
}
|
|
|
|
if(keepalive_delay.count() > 0)
|
|
{
|
|
keepalive_thread_run = 1;
|
|
keepalive_thread = new std::thread(&RGBController_DMX::KeepaliveThreadFunction, this);
|
|
}
|
|
else
|
|
{
|
|
keepalive_thread_run = 0;
|
|
keepalive_thread = nullptr;
|
|
}
|
|
}
|
|
|
|
RGBController_DMX::~RGBController_DMX()
|
|
{
|
|
if(keepalive_thread != nullptr)
|
|
{
|
|
keepalive_thread_run = 0;
|
|
keepalive_thread->join();
|
|
delete keepalive_thread;
|
|
}
|
|
|
|
/*---------------------------------------------------------*\
|
|
| Delete the matrix map |
|
|
\*---------------------------------------------------------*/
|
|
for(unsigned int zone_index = 0; zone_index < zones.size(); zone_index++)
|
|
{
|
|
if(zones[zone_index].matrix_map != NULL)
|
|
{
|
|
if(zones[zone_index].matrix_map->map != NULL)
|
|
{
|
|
delete zones[zone_index].matrix_map->map;
|
|
}
|
|
|
|
delete zones[zone_index].matrix_map;
|
|
}
|
|
}
|
|
}
|
|
|
|
void RGBController_DMX::SetupZones()
|
|
{
|
|
/*-----------------------------------------*\
|
|
| Add Zones |
|
|
\*-----------------------------------------*/
|
|
for(std::size_t zone_idx = 0; zone_idx < devices.size(); zone_idx++)
|
|
{
|
|
zone led_zone;
|
|
led_zone.name = devices[zone_idx].name;
|
|
led_zone.type = ZONE_TYPE_SINGLE;
|
|
led_zone.leds_min = 1;
|
|
led_zone.leds_max = 1;
|
|
led_zone.leds_count = 1;
|
|
led_zone.matrix_map = NULL;
|
|
|
|
zones.push_back(led_zone);
|
|
}
|
|
|
|
/*-----------------------------------------*\
|
|
| Add LEDs |
|
|
\*-----------------------------------------*/
|
|
for(std::size_t zone_idx = 0; zone_idx < zones.size(); zone_idx++)
|
|
{
|
|
for(std::size_t led_idx = 0; led_idx < zones[zone_idx].leds_count; led_idx++)
|
|
{
|
|
led new_led;
|
|
|
|
new_led.name = zones[zone_idx].name + " LED ";
|
|
new_led.name.append(std::to_string(led_idx));
|
|
|
|
leds.push_back(new_led);
|
|
}
|
|
}
|
|
|
|
SetupColors();
|
|
}
|
|
|
|
void RGBController_DMX::ResizeZone(int /*zone*/, int /*new_size*/)
|
|
{
|
|
/*---------------------------------------------------------*\
|
|
| This device does not support resizing zones |
|
|
\*---------------------------------------------------------*/
|
|
}
|
|
|
|
void RGBController_DMX::DeviceUpdateLEDs()
|
|
{
|
|
last_update_time = std::chrono::steady_clock::now();
|
|
|
|
unsigned char dmx_data[513];
|
|
|
|
memset(dmx_data, 0, sizeof(dmx_data));
|
|
|
|
for(unsigned int device_idx = 0; device_idx < devices.size(); device_idx++)
|
|
{
|
|
if(devices[device_idx].brightness_channel > 0)
|
|
{
|
|
dmx_data[devices[device_idx].brightness_channel] = modes[0].brightness;
|
|
}
|
|
|
|
if(devices[device_idx].red_channel > 0)
|
|
{
|
|
dmx_data[devices[device_idx].red_channel] = RGBGetRValue(colors[device_idx]);
|
|
}
|
|
|
|
if(devices[device_idx].green_channel > 0)
|
|
{
|
|
dmx_data[devices[device_idx].green_channel] = RGBGetGValue(colors[device_idx]);
|
|
}
|
|
|
|
if(devices[device_idx].blue_channel > 0)
|
|
{
|
|
dmx_data[devices[device_idx].blue_channel] = RGBGetBValue(colors[device_idx]);
|
|
}
|
|
}
|
|
|
|
port->serial_break();
|
|
port->serial_write((char*)&dmx_data, sizeof(dmx_data));
|
|
}
|
|
|
|
void RGBController_DMX::UpdateZoneLEDs(int /*zone*/)
|
|
{
|
|
DeviceUpdateLEDs();
|
|
}
|
|
|
|
void RGBController_DMX::UpdateSingleLED(int /*led*/)
|
|
{
|
|
DeviceUpdateLEDs();
|
|
}
|
|
|
|
void RGBController_DMX::DeviceUpdateMode()
|
|
{
|
|
|
|
}
|
|
|
|
void RGBController_DMX::KeepaliveThreadFunction()
|
|
{
|
|
while(keepalive_thread_run.load())
|
|
{
|
|
if((std::chrono::steady_clock::now() - last_update_time) > ( keepalive_delay * 0.95f ) )
|
|
{
|
|
UpdateLEDs();
|
|
}
|
|
std::this_thread::sleep_for(keepalive_delay / 2);
|
|
}
|
|
}
|