2013-07-26 22:38:34 +02:00
2013-11-11 10:00:37 +01:00
// STL includes
# include <cassert>
2016-06-17 01:25:40 +02:00
# include <exception>
2016-07-01 23:20:41 +02:00
# include <sstream>
2013-11-11 10:00:37 +01:00
2013-08-21 16:25:27 +02:00
// QT includes
2013-08-14 17:02:09 +02:00
# include <QDateTime>
2013-11-29 23:22:49 +01:00
# include <QThread>
2013-11-22 11:48:10 +01:00
# include <QRegExp>
# include <QString>
# include <QStringList>
2016-09-14 13:51:28 +02:00
# include <QCryptographicHash>
# include <QFile>
2013-08-14 17:02:09 +02:00
2013-07-26 22:38:34 +02:00
// JsonSchema include
# include <utils/jsonschema/JsonFactory.h>
// hyperion include
# include <hyperion/Hyperion.h>
2013-08-15 21:11:02 +02:00
# include <hyperion/ImageProcessorFactory.h>
2015-12-22 10:01:47 +01:00
# include <hyperion/ColorTransform.h>
# include <hyperion/ColorCorrection.h>
2016-04-02 00:04:11 +02:00
# include <hyperion/ColorAdjustment.h>
2013-07-26 22:38:34 +02:00
2013-12-17 19:50:15 +01:00
// Leddevice includes
# include <leddevice/LedDevice.h>
# include <leddevice/LedDeviceFactory.h>
2013-08-13 11:10:45 +02:00
2013-11-20 09:25:49 +01:00
# include "MultiColorTransform.h"
2015-12-22 10:01:47 +01:00
# include "MultiColorCorrection.h"
2016-04-02 00:04:11 +02:00
# include "MultiColorAdjustment.h"
2013-10-27 18:04:37 +01:00
# include "LinearColorSmoothing.h"
2013-11-24 16:10:48 +01:00
// effect engine includes
# include <effectengine/EffectEngine.h>
2016-06-17 01:25:40 +02:00
Hyperion * Hyperion : : _hyperion = nullptr ;
2016-09-25 21:59:31 +02:00
Hyperion * Hyperion : : initInstance ( const Json : : Value & jsonConfig , const QJsonObject & qjsonConfig , const std : : string configFile ) // REMOVE jsonConfig variable when the conversion from jsonCPP to QtJSON is finished
2016-06-17 01:25:40 +02:00
{
if ( Hyperion : : _hyperion ! = nullptr )
throw std : : runtime_error ( " Hyperion::initInstance can be called only one time " ) ;
2016-09-25 21:59:31 +02:00
Hyperion : : _hyperion = new Hyperion ( jsonConfig , qjsonConfig , configFile ) ;
2016-06-17 01:25:40 +02:00
return Hyperion : : _hyperion ;
}
Hyperion * Hyperion : : getInstance ( )
{
if ( Hyperion : : _hyperion = = nullptr )
throw std : : runtime_error ( " Hyperion::getInstance used without call of Hyperion::initInstance before " ) ;
return Hyperion : : _hyperion ;
}
2013-07-26 22:38:34 +02:00
2016-09-25 21:59:31 +02:00
ColorOrder Hyperion : : createColorOrder ( const QJsonObject & deviceConfig )
2013-11-04 20:52:57 +01:00
{
2016-09-25 21:59:31 +02:00
return stringToColorOrder ( deviceConfig [ " colorOrder " ] . toString ( " rgb " ) ) ;
2013-11-04 20:52:57 +01:00
}
2016-09-25 21:59:31 +02:00
ColorTransform * Hyperion : : createColorTransform ( const QJsonObject & transformConfig )
2013-11-20 09:25:49 +01:00
{
2016-09-25 21:59:31 +02:00
const std : : string id = transformConfig [ " id " ] . toString ( " default " ) . toStdString ( ) ;
2013-11-22 11:48:10 +01:00
2016-09-25 21:59:31 +02:00
RgbChannelTransform * redTransform = createRgbChannelTransform ( transformConfig [ " red " ] . toObject ( ) ) ;
RgbChannelTransform * greenTransform = createRgbChannelTransform ( transformConfig [ " green " ] . toObject ( ) ) ;
RgbChannelTransform * blueTransform = createRgbChannelTransform ( transformConfig [ " blue " ] . toObject ( ) ) ;
2013-11-22 11:48:10 +01:00
2016-09-25 21:59:31 +02:00
HsvTransform * hsvTransform = createHsvTransform ( transformConfig [ " hsv " ] . toObject ( ) ) ;
HslTransform * hslTransform = createHslTransform ( transformConfig [ " hsl " ] . toObject ( ) ) ;
2013-11-20 09:25:49 +01:00
2013-11-22 11:48:10 +01:00
ColorTransform * transform = new ColorTransform ( ) ;
transform - > _id = id ;
transform - > _rgbRedTransform = * redTransform ;
transform - > _rgbGreenTransform = * greenTransform ;
transform - > _rgbBlueTransform = * blueTransform ;
transform - > _hsvTransform = * hsvTransform ;
2015-12-22 10:01:47 +01:00
transform - > _hslTransform = * hslTransform ;
2013-11-20 09:25:49 +01:00
2013-11-22 11:48:10 +01:00
// Cleanup the allocated individual transforms
delete redTransform ;
delete greenTransform ;
delete blueTransform ;
delete hsvTransform ;
2015-12-22 10:01:47 +01:00
delete hslTransform ;
2013-11-20 09:25:49 +01:00
2013-11-22 11:48:10 +01:00
return transform ;
}
2016-04-02 00:04:11 +02:00
2016-09-25 21:59:31 +02:00
ColorCorrection * Hyperion : : createColorCorrection ( const QJsonObject & correctionConfig )
2015-12-22 10:01:47 +01:00
{
2016-09-25 21:59:31 +02:00
const std : : string id = correctionConfig [ " id " ] . toString ( " default " ) . toStdString ( ) ;
2015-12-22 10:01:47 +01:00
2016-09-25 21:59:31 +02:00
RgbChannelAdjustment * rgbCorrection = createRgbChannelCorrection ( correctionConfig [ " correctionValues " ] . toObject ( ) ) ;
2015-12-22 10:01:47 +01:00
ColorCorrection * correction = new ColorCorrection ( ) ;
correction - > _id = id ;
correction - > _rgbCorrection = * rgbCorrection ;
// Cleanup the allocated individual transforms
delete rgbCorrection ;
return correction ;
}
2016-04-02 00:04:11 +02:00
2016-09-25 21:59:31 +02:00
ColorAdjustment * Hyperion : : createColorAdjustment ( const QJsonObject & adjustmentConfig )
2016-04-02 00:04:11 +02:00
{
2016-09-25 21:59:31 +02:00
const std : : string id = adjustmentConfig [ " id " ] . toString ( " default " ) . toStdString ( ) ;
2016-04-02 00:04:11 +02:00
2016-09-25 21:59:31 +02:00
RgbChannelAdjustment * redAdjustment = createRgbChannelAdjustment ( adjustmentConfig [ " pureRed " ] . toObject ( ) , RED ) ;
RgbChannelAdjustment * greenAdjustment = createRgbChannelAdjustment ( adjustmentConfig [ " pureGreen " ] . toObject ( ) , GREEN ) ;
RgbChannelAdjustment * blueAdjustment = createRgbChannelAdjustment ( adjustmentConfig [ " pureBlue " ] . toObject ( ) , BLUE ) ;
2016-04-02 00:04:11 +02:00
ColorAdjustment * adjustment = new ColorAdjustment ( ) ;
adjustment - > _id = id ;
adjustment - > _rgbRedAdjustment = * redAdjustment ;
adjustment - > _rgbGreenAdjustment = * greenAdjustment ;
adjustment - > _rgbBlueAdjustment = * blueAdjustment ;
// Cleanup the allocated individual adjustments
delete redAdjustment ;
delete greenAdjustment ;
delete blueAdjustment ;
return adjustment ;
}
2016-09-25 21:59:31 +02:00
MultiColorTransform * Hyperion : : createLedColorsTransform ( const unsigned ledCnt , const QJsonObject & colorConfig )
{
2013-11-22 11:48:10 +01:00
// Create the result, the transforms are added to this
MultiColorTransform * transform = new MultiColorTransform ( ledCnt ) ;
2016-07-01 23:20:41 +02:00
Logger * log = Logger : : getInstance ( " Core " ) ;
2016-09-25 21:59:31 +02:00
const QJsonValue transformConfig = colorConfig [ " transform " ] ;
2013-11-22 11:48:10 +01:00
if ( transformConfig . isNull ( ) )
{
// Old style color transformation config (just one for all leds)
ColorTransform * colorTransform = createColorTransform ( colorConfig ) ;
transform - > addTransform ( colorTransform ) ;
transform - > setTransformForLed ( colorTransform - > _id , 0 , ledCnt - 1 ) ;
}
2016-09-25 21:59:31 +02:00
else if ( transformConfig . isObject ( ) )
2013-11-22 11:48:10 +01:00
{
2016-09-25 21:59:31 +02:00
ColorTransform * colorTransform = createColorTransform ( transformConfig . toObject ( ) ) ;
2013-11-22 11:48:10 +01:00
transform - > addTransform ( colorTransform ) ;
transform - > setTransformForLed ( colorTransform - > _id , 0 , ledCnt - 1 ) ;
2013-11-20 09:25:49 +01:00
}
2016-09-25 21:59:31 +02:00
else if ( transformConfig . isArray ( ) )
2013-11-20 09:25:49 +01:00
{
2013-11-22 11:48:10 +01:00
const QRegExp overallExp ( " ([0-9]+( \\ -[0-9]+) ? ) ( , [ ] * ( [ 0 - 9 ] + ( \ \ - [ 0 - 9 ] + ) ? ) ) * " ) ;
2013-11-20 09:25:49 +01:00
2016-09-25 21:59:31 +02:00
const QJsonArray & transformConfigArray = transformConfig . toArray ( ) ;
for ( signed i = 0 ; i < transformConfigArray . size ( ) ; + + i )
2013-11-22 11:48:10 +01:00
{
2016-09-25 21:59:31 +02:00
const QJsonObject & config = transformConfigArray [ i ] . toObject ( ) ;
2013-11-22 11:48:10 +01:00
ColorTransform * colorTransform = createColorTransform ( config ) ;
transform - > addTransform ( colorTransform ) ;
2016-09-25 21:59:31 +02:00
const QString ledIndicesStr = config [ " leds " ] . toString ( " " ) . trimmed ( ) ;
2013-12-13 22:58:09 +01:00
if ( ledIndicesStr . compare ( " * " ) = = 0 )
{
// Special case for indices '*' => all leds
transform - > setTransformForLed ( colorTransform - > _id , 0 , ledCnt - 1 ) ;
2016-07-01 23:20:41 +02:00
Info ( log , " ColorTransform '%s' => [0; %d] " , colorTransform - > _id . c_str ( ) , ledCnt - 1 ) ;
2013-12-13 22:58:09 +01:00
continue ;
}
2013-11-22 11:48:10 +01:00
if ( ! overallExp . exactMatch ( ledIndicesStr ) )
{
2016-07-01 23:20:41 +02:00
Error ( log , " Given led indices %d not correct format: %s " , i , ledIndicesStr . toStdString ( ) . c_str ( ) ) ;
2013-11-22 11:48:10 +01:00
continue ;
}
2016-07-01 23:20:41 +02:00
std : : stringstream ss ;
2013-11-22 11:48:10 +01:00
const QStringList ledIndexList = ledIndicesStr . split ( " , " ) ;
for ( int i = 0 ; i < ledIndexList . size ( ) ; + + i ) {
if ( i > 0 )
{
2016-07-10 23:14:23 +02:00
ss < < " , " ;
2013-11-22 11:48:10 +01:00
}
if ( ledIndexList [ i ] . contains ( " - " ) )
{
QStringList ledIndices = ledIndexList [ i ] . split ( " - " ) ;
int startInd = ledIndices [ 0 ] . toInt ( ) ;
int endInd = ledIndices [ 1 ] . toInt ( ) ;
transform - > setTransformForLed ( colorTransform - > _id , startInd , endInd ) ;
2016-07-10 23:14:23 +02:00
ss < < startInd < < " - " < < endInd ;
2013-11-22 11:48:10 +01:00
}
else
{
int index = ledIndexList [ i ] . toInt ( ) ;
transform - > setTransformForLed ( colorTransform - > _id , index , index ) ;
2016-07-10 23:14:23 +02:00
ss < < index ;
2013-11-22 11:48:10 +01:00
}
}
2016-07-01 23:20:41 +02:00
Info ( log , " ColorTransform '%s' => [%s] " , colorTransform - > _id . c_str ( ) , ss . str ( ) . c_str ( ) ) ;
2013-11-22 11:48:10 +01:00
}
2013-11-20 09:25:49 +01:00
}
return transform ;
}
2016-09-25 21:59:31 +02:00
MultiColorCorrection * Hyperion : : createLedColorsTemperature ( const unsigned ledCnt , const QJsonObject & colorConfig )
2015-12-22 10:01:47 +01:00
{
// Create the result, the corrections are added to this
MultiColorCorrection * correction = new MultiColorCorrection ( ledCnt ) ;
2016-07-01 23:20:41 +02:00
Logger * log = Logger : : getInstance ( " Core " ) ;
2015-12-22 10:01:47 +01:00
2016-09-25 21:59:31 +02:00
const QString jsonKey = colorConfig . contains ( " temperature " ) ? " temperature " : " correction " ;
const QJsonValue correctionConfig = colorConfig [ jsonKey ] ;
2015-12-22 10:01:47 +01:00
if ( correctionConfig . isNull ( ) )
{
// Old style color correction config (just one for all leds)
ColorCorrection * colorCorrection = createColorCorrection ( colorConfig ) ;
correction - > addCorrection ( colorCorrection ) ;
correction - > setCorrectionForLed ( colorCorrection - > _id , 0 , ledCnt - 1 ) ;
}
2016-09-25 21:59:31 +02:00
else if ( correctionConfig . isObject ( ) )
2015-12-22 10:01:47 +01:00
{
2016-09-25 21:59:31 +02:00
ColorCorrection * colorCorrection = createColorCorrection ( correctionConfig . toObject ( ) ) ;
2015-12-22 10:01:47 +01:00
correction - > addCorrection ( colorCorrection ) ;
correction - > setCorrectionForLed ( colorCorrection - > _id , 0 , ledCnt - 1 ) ;
}
2016-09-25 21:59:31 +02:00
else if ( correctionConfig . isArray ( ) )
2015-12-22 10:01:47 +01:00
{
const QRegExp overallExp ( " ([0-9]+( \\ -[0-9]+) ? ) ( , [ ] * ( [ 0 - 9 ] + ( \ \ - [ 0 - 9 ] + ) ? ) ) * " ) ;
2016-09-25 21:59:31 +02:00
const QJsonArray & correctionConfigArray = correctionConfig . toArray ( ) ;
for ( signed i = 0 ; i < correctionConfigArray . size ( ) ; + + i )
2015-12-22 10:01:47 +01:00
{
2016-09-25 21:59:31 +02:00
const QJsonObject & config = correctionConfigArray . at ( i ) . toObject ( ) ;
2015-12-22 10:01:47 +01:00
ColorCorrection * colorCorrection = createColorCorrection ( config ) ;
correction - > addCorrection ( colorCorrection ) ;
2016-09-25 21:59:31 +02:00
const QString ledIndicesStr = config [ " leds " ] . toString ( " " ) . trimmed ( ) ;
2015-12-22 10:01:47 +01:00
if ( ledIndicesStr . compare ( " * " ) = = 0 )
{
// Special case for indices '*' => all leds
correction - > setCorrectionForLed ( colorCorrection - > _id , 0 , ledCnt - 1 ) ;
2016-07-01 23:20:41 +02:00
Info ( log , " ColorTemperature '%s' => [0; %d] " , colorCorrection - > _id . c_str ( ) , ledCnt - 1 ) ;
2015-12-22 10:01:47 +01:00
continue ;
}
if ( ! overallExp . exactMatch ( ledIndicesStr ) )
{
2016-07-01 23:20:41 +02:00
Error ( log , " Given led indices %d not correct format: %s " , i , ledIndicesStr . toStdString ( ) . c_str ( ) ) ;
2015-12-22 10:01:47 +01:00
continue ;
}
2016-07-01 23:20:41 +02:00
std : : stringstream ss ;
2015-12-22 10:01:47 +01:00
const QStringList ledIndexList = ledIndicesStr . split ( " , " ) ;
for ( int i = 0 ; i < ledIndexList . size ( ) ; + + i ) {
if ( i > 0 )
{
2016-07-01 23:20:41 +02:00
ss < < " , " ;
2015-12-22 10:01:47 +01:00
}
if ( ledIndexList [ i ] . contains ( " - " ) )
{
QStringList ledIndices = ledIndexList [ i ] . split ( " - " ) ;
int startInd = ledIndices [ 0 ] . toInt ( ) ;
int endInd = ledIndices [ 1 ] . toInt ( ) ;
correction - > setCorrectionForLed ( colorCorrection - > _id , startInd , endInd ) ;
2016-07-01 23:20:41 +02:00
ss < < startInd < < " - " < < endInd ;
2015-12-22 10:01:47 +01:00
}
else
{
int index = ledIndexList [ i ] . toInt ( ) ;
correction - > setCorrectionForLed ( colorCorrection - > _id , index , index ) ;
2016-07-01 23:20:41 +02:00
ss < < index ;
2015-12-22 10:01:47 +01:00
}
}
2016-07-01 23:20:41 +02:00
Info ( log , " ColorTemperature '%s' => [%s] " , colorCorrection - > _id . c_str ( ) , ss . str ( ) . c_str ( ) ) ;
2015-12-22 10:01:47 +01:00
}
}
return correction ;
}
2016-07-01 23:20:41 +02:00
2016-09-25 21:59:31 +02:00
MultiColorAdjustment * Hyperion : : createLedColorsAdjustment ( const unsigned ledCnt , const QJsonObject & colorConfig )
2016-04-02 00:04:11 +02:00
{
// Create the result, the transforms are added to this
MultiColorAdjustment * adjustment = new MultiColorAdjustment ( ledCnt ) ;
2016-07-01 23:20:41 +02:00
Logger * log = Logger : : getInstance ( " Core " ) ;
2016-04-02 00:04:11 +02:00
2016-09-25 21:59:31 +02:00
const QJsonValue adjustmentConfig = colorConfig [ " channelAdjustment " ] ;
2016-04-02 00:04:11 +02:00
if ( adjustmentConfig . isNull ( ) )
{
// Old style color transformation config (just one for all leds)
ColorAdjustment * colorAdjustment = createColorAdjustment ( colorConfig ) ;
adjustment - > addAdjustment ( colorAdjustment ) ;
adjustment - > setAdjustmentForLed ( colorAdjustment - > _id , 0 , ledCnt - 1 ) ;
}
2016-09-25 21:59:31 +02:00
else if ( adjustmentConfig . isObject ( ) )
2016-04-02 00:04:11 +02:00
{
2016-09-25 21:59:31 +02:00
ColorAdjustment * colorAdjustment = createColorAdjustment ( adjustmentConfig . toObject ( ) ) ;
2016-04-02 00:04:11 +02:00
adjustment - > addAdjustment ( colorAdjustment ) ;
adjustment - > setAdjustmentForLed ( colorAdjustment - > _id , 0 , ledCnt - 1 ) ;
}
2016-09-25 21:59:31 +02:00
else if ( adjustmentConfig . isArray ( ) )
2016-04-02 00:04:11 +02:00
{
const QRegExp overallExp ( " ([0-9]+( \\ -[0-9]+) ? ) ( , [ ] * ( [ 0 - 9 ] + ( \ \ - [ 0 - 9 ] + ) ? ) ) * " ) ;
2016-09-25 21:59:31 +02:00
const QJsonArray & adjustmentConfigArray = adjustmentConfig . toArray ( ) ;
for ( signed i = 0 ; i < adjustmentConfigArray . size ( ) ; + + i )
2016-04-02 00:04:11 +02:00
{
2016-09-25 21:59:31 +02:00
const QJsonObject & config = adjustmentConfigArray . at ( i ) . toObject ( ) ;
2016-04-02 00:04:11 +02:00
ColorAdjustment * colorAdjustment = createColorAdjustment ( config ) ;
adjustment - > addAdjustment ( colorAdjustment ) ;
2016-09-25 21:59:31 +02:00
const QString ledIndicesStr = config [ " leds " ] . toString ( " " ) . trimmed ( ) ;
2016-04-02 00:04:11 +02:00
if ( ledIndicesStr . compare ( " * " ) = = 0 )
{
// Special case for indices '*' => all leds
adjustment - > setAdjustmentForLed ( colorAdjustment - > _id , 0 , ledCnt - 1 ) ;
2016-07-01 23:20:41 +02:00
Info ( log , " ColorAdjustment '%s' => [0; %d] " , colorAdjustment - > _id . c_str ( ) , ledCnt - 1 ) ;
2016-04-02 00:04:11 +02:00
continue ;
}
if ( ! overallExp . exactMatch ( ledIndicesStr ) )
{
2016-07-01 23:20:41 +02:00
Error ( log , " Given led indices %d not correct format: %s " , i , ledIndicesStr . toStdString ( ) . c_str ( ) ) ;
2016-04-02 00:04:11 +02:00
continue ;
}
2016-07-01 23:20:41 +02:00
std : : stringstream ss ;
2016-04-02 00:04:11 +02:00
const QStringList ledIndexList = ledIndicesStr . split ( " , " ) ;
for ( int i = 0 ; i < ledIndexList . size ( ) ; + + i ) {
if ( i > 0 )
{
2016-07-01 23:20:41 +02:00
ss < < " , " ;
2016-04-02 00:04:11 +02:00
}
if ( ledIndexList [ i ] . contains ( " - " ) )
{
QStringList ledIndices = ledIndexList [ i ] . split ( " - " ) ;
int startInd = ledIndices [ 0 ] . toInt ( ) ;
int endInd = ledIndices [ 1 ] . toInt ( ) ;
adjustment - > setAdjustmentForLed ( colorAdjustment - > _id , startInd , endInd ) ;
2016-07-01 23:20:41 +02:00
ss < < startInd < < " - " < < endInd ;
2016-04-02 00:04:11 +02:00
}
else
{
int index = ledIndexList [ i ] . toInt ( ) ;
adjustment - > setAdjustmentForLed ( colorAdjustment - > _id , index , index ) ;
2016-07-01 23:20:41 +02:00
ss < < index ;
2016-04-02 00:04:11 +02:00
}
}
2016-07-01 23:20:41 +02:00
Info ( log , " ColorAdjustment '%s' => [%s] " , colorAdjustment - > _id . c_str ( ) , ss . str ( ) . c_str ( ) ) ;
2016-04-02 00:04:11 +02:00
}
}
return adjustment ;
}
2016-09-25 21:59:31 +02:00
HsvTransform * Hyperion : : createHsvTransform ( const QJsonObject & hsvConfig )
2013-08-21 21:50:17 +02:00
{
2016-09-25 21:59:31 +02:00
const double saturationGain = hsvConfig [ " saturationGain " ] . toDouble ( 1.0 ) ;
const double valueGain = hsvConfig [ " valueGain " ] . toDouble ( 1.0 ) ;
2013-10-17 09:44:42 +02:00
return new HsvTransform ( saturationGain , valueGain ) ;
2013-08-21 21:50:17 +02:00
}
2016-09-25 21:59:31 +02:00
HslTransform * Hyperion : : createHslTransform ( const QJsonObject & hslConfig )
2015-12-22 10:01:47 +01:00
{
2016-09-25 21:59:31 +02:00
const double saturationGain = hslConfig [ " saturationGain " ] . toDouble ( 1.0 ) ;
const double luminanceGain = hslConfig [ " luminanceGain " ] . toDouble ( 1.0 ) ;
const double luminanceMinimum = hslConfig [ " luminanceMinimum " ] . toDouble ( 0.0 ) ;
2015-12-22 10:01:47 +01:00
2016-05-23 00:00:48 +02:00
return new HslTransform ( saturationGain , luminanceGain , luminanceMinimum ) ;
2015-12-22 10:01:47 +01:00
}
2016-09-25 21:59:31 +02:00
RgbChannelTransform * Hyperion : : createRgbChannelTransform ( const QJsonObject & colorConfig )
2013-08-13 11:10:45 +02:00
{
2016-09-25 21:59:31 +02:00
const double threshold = colorConfig [ " threshold " ] . toDouble ( 0.0 ) ;
const double gamma = colorConfig [ " gamma " ] . toDouble ( 1.0 ) ;
const double blacklevel = colorConfig [ " blacklevel " ] . toDouble ( 0.0 ) ;
const double whitelevel = colorConfig [ " whitelevel " ] . toDouble ( 1.0 ) ;
2013-08-13 11:10:45 +02:00
2013-11-19 21:17:59 +01:00
RgbChannelTransform * transform = new RgbChannelTransform ( threshold , gamma , blacklevel , whitelevel ) ;
2013-08-13 11:10:45 +02:00
return transform ;
}
2013-10-27 18:04:37 +01:00
2016-09-25 21:59:31 +02:00
RgbChannelAdjustment * Hyperion : : createRgbChannelCorrection ( const QJsonObject & colorConfig )
2015-12-22 10:01:47 +01:00
{
2016-09-25 21:59:31 +02:00
const int varR = colorConfig [ " red " ] . toInt ( 255 ) ;
const int varG = colorConfig [ " green " ] . toInt ( 255 ) ;
const int varB = colorConfig [ " blue " ] . toInt ( 255 ) ;
2015-12-22 10:01:47 +01:00
2016-07-04 00:43:41 +02:00
RgbChannelAdjustment * correction = new RgbChannelAdjustment ( varR , varG , varB ) ;
2015-12-22 10:01:47 +01:00
return correction ;
}
2016-09-25 21:59:31 +02:00
RgbChannelAdjustment * Hyperion : : createRgbChannelAdjustment ( const QJsonObject & colorConfig , const RgbChannel color )
2016-04-02 00:04:11 +02:00
{
2016-05-26 07:01:10 +02:00
int varR = 0 , varG = 0 , varB = 0 ;
2016-04-02 00:04:11 +02:00
if ( color = = RED )
{
2016-09-25 21:59:31 +02:00
varR = colorConfig [ " redChannel " ] . toInt ( 255 ) ;
varG = colorConfig [ " greenChannel " ] . toInt ( 0 ) ;
varB = colorConfig [ " blueChannel " ] . toInt ( 0 ) ;
2016-04-02 00:04:11 +02:00
}
else if ( color = = GREEN )
{
2016-09-25 21:59:31 +02:00
varR = colorConfig [ " redChannel " ] . toInt ( 0 ) ;
varG = colorConfig [ " greenChannel " ] . toInt ( 255 ) ;
varB = colorConfig [ " blueChannel " ] . toInt ( 0 ) ;
2016-04-02 00:04:11 +02:00
}
else if ( color = = BLUE )
{
2016-09-25 21:59:31 +02:00
varR = colorConfig [ " redChannel " ] . toInt ( 0 ) ;
varG = colorConfig [ " greenChannel " ] . toInt ( 0 ) ;
varB = colorConfig [ " blueChannel " ] . toInt ( 255 ) ;
2016-04-02 00:04:11 +02:00
}
RgbChannelAdjustment * adjustment = new RgbChannelAdjustment ( varR , varG , varB ) ;
return adjustment ;
}
2016-09-25 21:59:31 +02:00
LedString Hyperion : : createLedString ( const QJsonValue & ledsConfig , const ColorOrder deviceOrder )
2013-08-13 11:10:45 +02:00
{
LedString ledString ;
2016-09-25 21:59:31 +02:00
const QString deviceOrderStr = colorOrderToString ( deviceOrder ) ;
const QJsonArray & ledConfigArray = ledsConfig . toArray ( ) ;
int maxLedId = ledConfigArray . size ( ) ;
for ( signed i = 0 ; i < ledConfigArray . size ( ) ; + + i )
2013-08-13 11:10:45 +02:00
{
2016-09-25 21:59:31 +02:00
const QJsonObject & index = ledConfigArray [ i ] . toObject ( ) ;
2013-08-13 11:10:45 +02:00
Led led ;
2016-09-25 21:59:31 +02:00
led . index = index [ " index " ] . toInt ( ) ;
led . clone = index [ " clone " ] . toInt ( - 1 ) ;
2016-08-08 00:17:00 +02:00
if ( led . clone < - 1 | | led . clone > = maxLedId )
{
Warning ( Logger : : getInstance ( " Core " ) , " LED %d: clone index of %d is out of range, clone ignored " , led . index , led . clone ) ;
led . clone = - 1 ;
}
2015-02-23 16:11:33 +01:00
2016-08-08 00:17:00 +02:00
if ( led . clone < 0 )
2013-10-16 23:19:40 +02:00
{
2016-09-25 21:59:31 +02:00
const QJsonObject & hscanConfig = ledConfigArray [ i ] . toObject ( ) [ " hscan " ] . toObject ( ) ;
const QJsonObject & vscanConfig = ledConfigArray [ i ] . toObject ( ) [ " vscan " ] . toObject ( ) ;
led . minX_frac = std : : max ( 0.0 , std : : min ( 1.0 , hscanConfig [ " minimum " ] . toDouble ( ) ) ) ;
led . maxX_frac = std : : max ( 0.0 , std : : min ( 1.0 , hscanConfig [ " maximum " ] . toDouble ( ) ) ) ;
led . minY_frac = std : : max ( 0.0 , std : : min ( 1.0 , vscanConfig [ " minimum " ] . toDouble ( ) ) ) ;
led . maxY_frac = std : : max ( 0.0 , std : : min ( 1.0 , vscanConfig [ " maximum " ] . toDouble ( ) ) ) ;
2016-08-08 00:17:00 +02:00
// Fix if the user swapped min and max
if ( led . minX_frac > led . maxX_frac )
{
std : : swap ( led . minX_frac , led . maxX_frac ) ;
}
if ( led . minY_frac > led . maxY_frac )
{
std : : swap ( led . minY_frac , led . maxY_frac ) ;
}
// Get the order of the rgb channels for this led (default is device order)
2016-09-25 21:59:31 +02:00
led . colorOrder = stringToColorOrder ( index [ " colorOrder " ] . toString ( deviceOrderStr ) ) ;
2016-08-08 00:17:00 +02:00
ledString . leds ( ) . push_back ( led ) ;
2013-10-16 23:19:40 +02:00
}
2016-08-08 00:17:00 +02:00
}
// Make sure the leds are sorted (on their indices)
std : : sort ( ledString . leds ( ) . begin ( ) , ledString . leds ( ) . end ( ) , [ ] ( const Led & lhs , const Led & rhs ) { return lhs . index < rhs . index ; } ) ;
return ledString ;
}
2016-09-25 21:59:31 +02:00
LedString Hyperion : : createLedStringClone ( const QJsonValue & ledsConfig , const ColorOrder deviceOrder )
2016-08-08 00:17:00 +02:00
{
LedString ledString ;
2016-09-25 21:59:31 +02:00
const QString deviceOrderStr = colorOrderToString ( deviceOrder ) ;
const QJsonArray & ledConfigArray = ledsConfig . toArray ( ) ;
int maxLedId = ledConfigArray . size ( ) ;
for ( signed i = 0 ; i < ledConfigArray . size ( ) ; + + i )
2016-08-08 00:17:00 +02:00
{
2016-09-25 21:59:31 +02:00
const QJsonObject & index = ledConfigArray [ i ] . toObject ( ) ;
2016-08-08 00:17:00 +02:00
Led led ;
2016-09-25 21:59:31 +02:00
led . index = index [ " index " ] . toInt ( ) ;
led . clone = index [ " clone " ] . toInt ( - 1 ) ;
2016-08-08 00:17:00 +02:00
if ( led . clone < - 1 | | led . clone > = maxLedId )
2013-10-16 23:19:40 +02:00
{
2016-08-08 00:17:00 +02:00
Warning ( Logger : : getInstance ( " Core " ) , " LED %d: clone index of %d is out of range, clone ignored " , led . index , led . clone ) ;
led . clone = - 1 ;
2013-10-16 23:19:40 +02:00
}
2016-08-08 00:17:00 +02:00
if ( led . clone > = 0 )
{
Debug ( Logger : : getInstance ( " Core " ) , " LED %d: clone from led %d " , led . index , led . clone ) ;
led . minX_frac = 0 ;
led . maxX_frac = 0 ;
led . minY_frac = 0 ;
led . maxY_frac = 0 ;
// Get the order of the rgb channels for this led (default is device order)
2016-09-25 21:59:31 +02:00
led . colorOrder = stringToColorOrder ( index [ " colorOrder " ] . toString ( deviceOrderStr ) ) ;
2016-08-08 00:17:00 +02:00
ledString . leds ( ) . push_back ( led ) ;
}
2015-02-23 16:11:33 +01:00
2013-08-13 11:10:45 +02:00
}
2013-11-05 16:05:00 +01:00
// Make sure the leds are sorted (on their indices)
std : : sort ( ledString . leds ( ) . begin ( ) , ledString . leds ( ) . end ( ) , [ ] ( const Led & lhs , const Led & rhs ) { return lhs . index < rhs . index ; } ) ;
2013-08-13 11:10:45 +02:00
return ledString ;
}
2016-09-25 21:59:31 +02:00
QSize Hyperion : : getLedLayoutGridSize ( const QJsonValue & ledsConfig )
2016-09-21 22:01:50 +02:00
{
std : : vector < int > midPointsX ;
std : : vector < int > midPointsY ;
2016-09-25 21:59:31 +02:00
const QJsonArray & ledConfigArray = ledsConfig . toArray ( ) ;
2016-09-21 22:01:50 +02:00
2016-09-25 21:59:31 +02:00
for ( signed i = 0 ; i < ledConfigArray . size ( ) ; + + i )
2016-09-21 22:01:50 +02:00
{
2016-09-25 21:59:31 +02:00
const QJsonObject & index = ledConfigArray [ i ] . toObject ( ) ;
if ( index [ " clone " ] . toInt ( - 1 ) < 0 )
2016-09-21 22:01:50 +02:00
{
2016-09-25 21:59:31 +02:00
const QJsonObject & hscanConfig = ledConfigArray [ i ] . toObject ( ) [ " hscan " ] . toObject ( ) ;
const QJsonObject & vscanConfig = ledConfigArray [ i ] . toObject ( ) [ " vscan " ] . toObject ( ) ;
double minX_frac = std : : max ( 0.0 , std : : min ( 1.0 , hscanConfig [ " minimum " ] . toDouble ( ) ) ) ;
double maxX_frac = std : : max ( 0.0 , std : : min ( 1.0 , hscanConfig [ " maximum " ] . toDouble ( ) ) ) ;
double minY_frac = std : : max ( 0.0 , std : : min ( 1.0 , vscanConfig [ " minimum " ] . toDouble ( ) ) ) ;
double maxY_frac = std : : max ( 0.0 , std : : min ( 1.0 , vscanConfig [ " maximum " ] . toDouble ( ) ) ) ;
2016-09-21 22:01:50 +02:00
// Fix if the user swapped min and max
if ( minX_frac > maxX_frac )
{
std : : swap ( minX_frac , maxX_frac ) ;
}
if ( minY_frac > maxY_frac )
{
std : : swap ( minY_frac , maxY_frac ) ;
}
// calculate mid point and make grid calculation
midPointsX . push_back ( int ( 1000.0 * ( minX_frac + maxX_frac ) / 2.0 ) ) ;
midPointsY . push_back ( int ( 1000.0 * ( minY_frac + maxY_frac ) / 2.0 ) ) ;
}
}
// remove duplicates
std : : sort ( midPointsX . begin ( ) , midPointsX . end ( ) ) ;
midPointsX . erase ( std : : unique ( midPointsX . begin ( ) , midPointsX . end ( ) ) , midPointsX . end ( ) ) ;
std : : sort ( midPointsY . begin ( ) , midPointsY . end ( ) ) ;
midPointsY . erase ( std : : unique ( midPointsY . begin ( ) , midPointsY . end ( ) ) , midPointsY . end ( ) ) ;
QSize gridSize ( midPointsX . size ( ) , midPointsY . size ( ) ) ;
Debug ( Logger : : getInstance ( " Core " ) , " led layout grid: %dx%d " , gridSize . width ( ) , gridSize . height ( ) ) ;
return gridSize ;
}
2016-09-25 21:59:31 +02:00
LinearColorSmoothing * Hyperion : : createColorSmoothing ( const QJsonObject & smoothingConfig , LedDevice * leddevice )
2016-08-11 07:13:55 +02:00
{
2016-07-12 12:46:54 +02:00
Logger * log = Logger : : getInstance ( " Core " ) ;
2016-09-25 21:59:31 +02:00
std : : string type = smoothingConfig [ " type " ] . toString ( " linear " ) . toStdString ( ) ;
2013-10-27 21:06:35 +01:00
std : : transform ( type . begin ( ) , type . end ( ) , type . begin ( ) , : : tolower ) ;
2016-09-13 11:51:16 +02:00
LinearColorSmoothing * device = nullptr ;
2016-07-13 11:18:12 +02:00
type = " linear " ; // TODO currently hardcoded type, delete it if we have more types
2016-09-08 16:32:42 +02:00
2016-07-13 11:18:12 +02:00
if ( type = = " linear " )
2013-10-27 21:06:35 +01:00
{
2016-09-08 16:32:42 +02:00
Info ( log , " Creating linear smoothing " ) ;
device = new LinearColorSmoothing (
leddevice ,
2016-09-25 21:59:31 +02:00
smoothingConfig [ " updateFrequency " ] . toDouble ( 25.0 ) ,
smoothingConfig [ " time_ms " ] . toInt ( 200 ) ,
smoothingConfig [ " updateDelay " ] . toInt ( 0 ) ,
smoothingConfig [ " continuousOutput " ] . toBool ( true )
2016-07-13 11:18:12 +02:00
) ;
2013-10-27 21:06:35 +01:00
}
2016-09-08 16:32:42 +02:00
else
{
Error ( log , " Smoothing disabled, because of unknown type '%s'. " , type . c_str ( ) ) ;
}
2016-09-25 21:59:31 +02:00
device - > setEnable ( smoothingConfig [ " enable " ] . toBool ( true ) ) ;
2016-09-08 16:32:42 +02:00
InfoIf ( ! device - > enabled ( ) , log , " Smoothing disabled " ) ;
2013-10-27 21:06:35 +01:00
2016-09-14 17:28:03 +02:00
assert ( device ! = nullptr ) ;
2016-09-08 16:32:42 +02:00
return device ;
2013-10-27 18:04:37 +01:00
}
2016-09-25 21:59:31 +02:00
MessageForwarder * Hyperion : : createMessageForwarder ( const QJsonObject & forwarderConfig )
2016-03-09 19:45:00 +01:00
{
MessageForwarder * forwarder = new MessageForwarder ( ) ;
2016-09-25 21:59:31 +02:00
if ( ! forwarderConfig . isEmpty ( ) & & forwarderConfig [ " enable " ] . toBool ( true ) )
2016-03-09 19:45:00 +01:00
{
2016-09-25 21:59:31 +02:00
if ( ! forwarderConfig [ " json " ] . isNull ( ) & & forwarderConfig [ " json " ] . isArray ( ) )
2016-03-09 19:45:00 +01:00
{
2016-09-25 21:59:31 +02:00
const QJsonArray & addr = forwarderConfig [ " json " ] . toArray ( ) ;
for ( signed i = 0 ; i < addr . size ( ) ; + + i )
2016-03-09 19:45:00 +01:00
{
2016-09-25 21:59:31 +02:00
Info ( Logger : : getInstance ( " Core " ) , " Json forward to %s " , addr . at ( i ) . toString ( ) . toStdString ( ) . c_str ( ) ) ;
forwarder - > addJsonSlave ( addr [ i ] . toString ( ) . toStdString ( ) ) ;
2016-03-09 19:45:00 +01:00
}
}
2016-09-25 21:59:31 +02:00
if ( ! forwarderConfig [ " proto " ] . isNull ( ) & & forwarderConfig [ " proto " ] . isArray ( ) )
2016-03-09 19:45:00 +01:00
{
2016-09-25 21:59:31 +02:00
const QJsonArray & addr = forwarderConfig [ " proto " ] . toArray ( ) ;
for ( signed i = 0 ; i < addr . size ( ) ; + + i )
2016-03-09 19:45:00 +01:00
{
2016-09-25 21:59:31 +02:00
Info ( Logger : : getInstance ( " Core " ) , " Proto forward to %s " , addr . at ( i ) . toString ( ) . toStdString ( ) . c_str ( ) ) ;
forwarder - > addProtoSlave ( addr [ i ] . toString ( ) . toStdString ( ) ) ;
2016-03-09 19:45:00 +01:00
}
}
}
return forwarder ;
}
MessageForwarder * Hyperion : : getForwarder ( )
{
return _messageForwarder ;
}
2013-11-04 20:52:57 +01:00
2016-09-25 21:59:31 +02:00
Hyperion : : Hyperion ( const Json : : Value & jsonConfig , const QJsonObject & qjsonConfig , const std : : string configFile )
: _ledString ( createLedString ( qjsonConfig [ " leds " ] , createColorOrder ( qjsonConfig [ " device " ] . toObject ( ) ) ) )
, _ledStringClone ( createLedStringClone ( qjsonConfig [ " leds " ] , createColorOrder ( qjsonConfig [ " device " ] . toObject ( ) ) ) )
2016-07-15 23:08:55 +02:00
, _muxer ( _ledString . leds ( ) . size ( ) )
2016-09-25 21:59:31 +02:00
, _raw2ledTransform ( createLedColorsTransform ( _ledString . leds ( ) . size ( ) , qjsonConfig [ " color " ] . toObject ( ) ) )
, _raw2ledTemperature ( createLedColorsTemperature ( _ledString . leds ( ) . size ( ) , qjsonConfig [ " color " ] . toObject ( ) ) )
, _raw2ledAdjustment ( createLedColorsAdjustment ( _ledString . leds ( ) . size ( ) , qjsonConfig [ " color " ] . toObject ( ) ) )
2016-07-15 23:08:55 +02:00
, _effectEngine ( nullptr )
2016-09-25 21:59:31 +02:00
, _messageForwarder ( createMessageForwarder ( qjsonConfig [ " forwarder " ] . toObject ( ) ) )
2016-07-15 23:08:55 +02:00
, _jsonConfig ( jsonConfig )
, _configFile ( configFile )
, _timer ( )
, _log ( Logger : : getInstance ( " Core " ) )
, _hwLedCount ( _ledString . leds ( ) . size ( ) )
2016-07-31 22:21:35 +02:00
, _sourceAutoSelectEnabled ( true )
2016-09-14 13:51:28 +02:00
, _configHash ( )
2016-09-25 21:59:31 +02:00
, _ledGridSize ( getLedLayoutGridSize ( qjsonConfig [ " leds " ] ) )
2013-07-26 22:38:34 +02:00
{
2016-08-06 08:28:42 +02:00
registerPriority ( " Off " , PriorityMuxer : : LOWEST_PRIORITY ) ;
2016-04-02 00:04:11 +02:00
if ( ! _raw2ledAdjustment - > verifyAdjustments ( ) )
{
2016-07-15 23:08:55 +02:00
throw std : : runtime_error ( " Color adjustment incorrectly set " ) ;
2016-04-02 00:04:11 +02:00
}
2015-12-22 10:06:04 +01:00
if ( ! _raw2ledTemperature - > verifyCorrections ( ) )
{
2016-07-15 23:08:55 +02:00
throw std : : runtime_error ( " Color temperature incorrectly set " ) ;
2015-12-22 10:06:04 +01:00
}
2013-11-22 11:48:10 +01:00
if ( ! _raw2ledTransform - > verifyTransforms ( ) )
{
2016-07-15 23:08:55 +02:00
throw std : : runtime_error ( " Color transformation incorrectly set " ) ;
2013-11-22 11:48:10 +01:00
}
2016-07-30 13:07:53 +02:00
// set color correction activity state
2016-09-25 21:59:31 +02:00
const QJsonObject & color = qjsonConfig [ " color " ] . toObject ( ) ;
_transformEnabled = color [ " transform_enable " ] . toBool ( true ) ;
_adjustmentEnabled = color [ " channelAdjustment_enable " ] . toBool ( true ) ;
_temperatureEnabled = color [ " temperature_enable " ] . toBool ( true ) ;
2016-07-30 13:07:53 +02:00
InfoIf ( ! _transformEnabled , _log , " Color transformation disabled " ) ;
InfoIf ( ! _adjustmentEnabled , _log , " Color adjustment disabled " ) ;
InfoIf ( ! _temperatureEnabled , _log , " Color temperature disabled " ) ;
2013-10-27 18:04:37 +01:00
// initialize the image processor factory
2014-01-20 20:46:38 +01:00
ImageProcessorFactory : : getInstance ( ) . init (
_ledString ,
2016-09-25 21:59:31 +02:00
qjsonConfig [ " blackborderdetector " ] . toObject ( )
2016-03-09 19:57:03 +01:00
) ;
2016-09-07 20:10:37 +02:00
getComponentRegister ( ) . componentStateChanged ( hyperion : : COMP_FORWARDER , _messageForwarder - > forwardingEnabled ( ) ) ;
2016-09-08 16:32:42 +02:00
// initialize leddevices
2016-10-08 08:14:36 +02:00
_device = LedDeviceFactory : : construct ( jsonConfig [ " device " ] , _hwLedCount ) ;
2016-09-25 21:59:31 +02:00
_deviceSmooth = createColorSmoothing ( qjsonConfig [ " smoothing " ] . toObject ( ) , _device ) ;
2016-09-08 16:32:42 +02:00
getComponentRegister ( ) . componentStateChanged ( hyperion : : COMP_SMOOTHING , _deviceSmooth - > componentState ( ) ) ;
2013-10-27 18:04:37 +01:00
// setup the timer
2013-08-14 17:02:09 +02:00
_timer . setSingleShot ( true ) ;
QObject : : connect ( & _timer , SIGNAL ( timeout ( ) ) , this , SLOT ( update ( ) ) ) ;
2013-11-24 16:10:48 +01:00
// create the effect engine
2016-08-08 00:17:00 +02:00
_effectEngine = new EffectEngine ( this , jsonConfig [ " effects " ] ) ;
2016-07-01 23:20:41 +02:00
2016-09-25 21:59:31 +02:00
const QJsonObject & device = qjsonConfig [ " device " ] . toObject ( ) ;
unsigned int hwLedCount = device [ " ledCount " ] . toInt ( getLedCount ( ) ) ;
2016-07-01 23:20:41 +02:00
_hwLedCount = std : : max ( hwLedCount , getLedCount ( ) ) ;
Debug ( _log , " configured leds: %d hw leds: %d " , getLedCount ( ) , _hwLedCount ) ;
WarningIf ( hwLedCount < getLedCount ( ) , _log , " more leds configured than available. check 'ledCount' in 'device' section " ) ;
2013-11-24 16:10:48 +01:00
2016-09-14 13:51:28 +02:00
// initialize hash of current config
configModified ( ) ;
2013-08-18 13:33:56 +02:00
// initialize the leds
update ( ) ;
2013-07-26 22:38:34 +02:00
}
Hyperion : : ~ Hyperion ( )
{
2013-10-27 18:04:37 +01:00
// switch off all leds
clearall ( ) ;
_device - > switchOff ( ) ;
2016-07-01 23:20:41 +02:00
// delete components on exit of hyperion core
2013-11-24 16:10:48 +01:00
delete _effectEngine ;
2013-08-18 13:33:56 +02:00
delete _device ;
2013-11-20 09:25:49 +01:00
delete _raw2ledTransform ;
2015-12-22 10:06:04 +01:00
delete _raw2ledTemperature ;
2016-04-02 00:04:11 +02:00
delete _raw2ledAdjustment ;
2016-03-09 19:45:00 +01:00
delete _messageForwarder ;
2013-07-26 22:38:34 +02:00
}
2013-08-14 10:54:49 +02:00
unsigned Hyperion : : getLedCount ( ) const
{
2013-08-18 13:33:56 +02:00
return _ledString . leds ( ) . size ( ) ;
2013-08-14 10:54:49 +02:00
}
2016-09-14 13:51:28 +02:00
bool Hyperion : : configModified ( )
{
2016-09-15 20:42:58 +02:00
bool isModified = false ;
2016-09-14 13:51:28 +02:00
QFile f ( _configFile . c_str ( ) ) ;
if ( f . open ( QFile : : ReadOnly ) )
{
QCryptographicHash hash ( QCryptographicHash : : Sha1 ) ;
2016-09-15 20:42:58 +02:00
if ( hash . addData ( & f ) )
{
if ( _configHash . size ( ) = = 0 )
2016-09-14 13:51:28 +02:00
{
2016-09-15 20:42:58 +02:00
_configHash = hash . result ( ) ;
2016-09-14 13:51:28 +02:00
}
2016-09-15 20:42:58 +02:00
else
{
isModified = _configHash ! = hash . result ( ) ;
}
}
2016-09-14 13:51:28 +02:00
}
f . close ( ) ;
2016-09-15 20:42:58 +02:00
return isModified ;
2016-09-14 13:51:28 +02:00
}
2016-07-15 23:08:55 +02:00
void Hyperion : : registerPriority ( const std : : string name , const int priority )
{
Info ( _log , " Register new input source named '%s' for priority channel '%d' " , name . c_str ( ) , priority ) ;
for ( auto const & entry : _priorityRegister )
{
WarningIf ( ( entry . first ! = name & & entry . second = = priority ) , _log ,
" Input source '%s' uses same priority channel (%d) as '%s'. " , name . c_str ( ) , priority , entry . first . c_str ( ) ) ;
}
_priorityRegister . emplace ( name , priority ) ;
}
void Hyperion : : unRegisterPriority ( const std : : string name )
{
Info ( _log , " Unregister input source named '%s' from priority register " , name . c_str ( ) ) ;
_priorityRegister . erase ( name ) ;
}
2016-07-31 22:21:35 +02:00
void Hyperion : : setSourceAutoSelectEnabled ( bool enabled )
{
_sourceAutoSelectEnabled = enabled ;
if ( ! _sourceAutoSelectEnabled )
{
setCurrentSourcePriority ( _muxer . getCurrentPriority ( ) ) ;
}
DebugIf ( ! _sourceAutoSelectEnabled , _log , " source auto select is disabled " ) ;
InfoIf ( _sourceAutoSelectEnabled , _log , " set current input source to auto select " ) ;
}
bool Hyperion : : setCurrentSourcePriority ( int priority )
{
bool priorityValid = _muxer . hasPriority ( priority ) ;
if ( priorityValid )
{
DebugIf ( _sourceAutoSelectEnabled , _log , " source auto select is disabled " ) ;
_sourceAutoSelectEnabled = false ;
_currentSourcePriority = priority ;
Info ( _log , " set current input source to priority channel %d " , _currentSourcePriority ) ;
}
return priorityValid ;
}
2016-08-11 07:13:55 +02:00
void Hyperion : : setComponentState ( const hyperion : : Components component , const bool state )
2016-08-04 13:10:53 +02:00
{
2016-09-08 16:32:42 +02:00
if ( component = = hyperion : : COMP_SMOOTHING )
{
_deviceSmooth - > setEnable ( state ) ;
getComponentRegister ( ) . componentStateChanged ( hyperion : : COMP_SMOOTHING , _deviceSmooth - > componentState ( ) ) ;
}
else
{
emit componentStateChanged ( component , state ) ;
}
2016-08-04 13:10:53 +02:00
}
2016-07-15 23:08:55 +02:00
2013-12-08 17:45:26 +01:00
void Hyperion : : setColor ( int priority , const ColorRgb & color , const int timeout_ms , bool clearEffects )
2013-07-26 22:38:34 +02:00
{
2013-08-18 13:33:56 +02:00
// create led output
2013-11-11 10:00:37 +01:00
std : : vector < ColorRgb > ledColors ( _ledString . leds ( ) . size ( ) , color ) ;
2013-08-18 13:33:56 +02:00
// set colors
2013-12-08 17:45:26 +01:00
setColors ( priority , ledColors , timeout_ms , clearEffects ) ;
2013-08-18 13:33:56 +02:00
}
2013-07-26 22:38:34 +02:00
2013-12-08 17:45:26 +01:00
void Hyperion : : setColors ( int priority , const std : : vector < ColorRgb > & ledColors , const int timeout_ms , bool clearEffects )
2013-07-26 22:38:34 +02:00
{
2013-12-08 17:45:26 +01:00
// clear effects if this call does not come from an effect
if ( clearEffects )
{
_effectEngine - > channelCleared ( priority ) ;
}
2013-08-14 17:02:09 +02:00
if ( timeout_ms > 0 )
{
const uint64_t timeoutTime = QDateTime : : currentMSecsSinceEpoch ( ) + timeout_ms ;
2013-08-18 13:33:56 +02:00
_muxer . setInput ( priority , ledColors , timeoutTime ) ;
2013-08-14 17:02:09 +02:00
}
else
{
2013-08-18 13:33:56 +02:00
_muxer . setInput ( priority , ledColors ) ;
2013-08-14 17:02:09 +02:00
}
2013-07-26 22:38:34 +02:00
2016-09-30 19:26:31 +02:00
if ( ! _sourceAutoSelectEnabled | | priority = = _muxer . getCurrentPriority ( ) )
2013-07-26 22:38:34 +02:00
{
2013-08-14 17:02:09 +02:00
update ( ) ;
}
}
2013-11-22 11:48:10 +01:00
const std : : vector < std : : string > & Hyperion : : getTransformIds ( ) const
2013-08-19 19:57:35 +02:00
{
2013-11-22 11:48:10 +01:00
return _raw2ledTransform - > getTransformIds ( ) ;
}
2013-08-21 21:50:17 +02:00
2015-12-22 10:06:04 +01:00
const std : : vector < std : : string > & Hyperion : : getTemperatureIds ( ) const
{
return _raw2ledTemperature - > getCorrectionIds ( ) ;
}
2016-04-02 00:04:11 +02:00
const std : : vector < std : : string > & Hyperion : : getAdjustmentIds ( ) const
{
return _raw2ledAdjustment - > getAdjustmentIds ( ) ;
}
2013-11-22 11:48:10 +01:00
ColorTransform * Hyperion : : getTransform ( const std : : string & id )
{
return _raw2ledTransform - > getTransform ( id ) ;
2013-08-19 19:57:35 +02:00
}
2015-12-22 10:06:04 +01:00
ColorCorrection * Hyperion : : getTemperature ( const std : : string & id )
{
return _raw2ledTemperature - > getCorrection ( id ) ;
}
2016-04-02 00:04:11 +02:00
ColorAdjustment * Hyperion : : getAdjustment ( const std : : string & id )
{
return _raw2ledAdjustment - > getAdjustment ( id ) ;
}
2013-12-03 22:56:46 +01:00
void Hyperion : : transformsUpdated ( )
2013-08-19 19:57:35 +02:00
{
2013-08-21 21:50:17 +02:00
update ( ) ;
2013-08-19 19:57:35 +02:00
}
2015-12-22 10:01:47 +01:00
void Hyperion : : correctionsUpdated ( )
{
update ( ) ;
}
2016-03-10 18:13:46 +01:00
void Hyperion : : temperaturesUpdated ( )
{
update ( ) ;
}
2016-04-02 00:04:11 +02:00
void Hyperion : : adjustmentsUpdated ( )
{
update ( ) ;
}
2013-08-18 13:33:56 +02:00
void Hyperion : : clear ( int priority )
{
if ( _muxer . hasPriority ( priority ) )
{
_muxer . clearInput ( priority ) ;
// update leds if necessary
2013-12-28 08:41:23 +01:00
if ( priority < _muxer . getCurrentPriority ( ) )
2013-08-18 13:33:56 +02:00
{
update ( ) ;
}
}
2013-11-29 23:22:49 +01:00
// send clear signal to the effect engine
// (outside the check so the effect gets cleared even when the effect is not sending colors)
_effectEngine - > channelCleared ( priority ) ;
2013-08-18 13:33:56 +02:00
}
void Hyperion : : clearall ( )
{
_muxer . clearAll ( ) ;
// update leds
update ( ) ;
2013-11-24 16:10:48 +01:00
// send clearall signal to the effect engine
_effectEngine - > allChannelsCleared ( ) ;
2013-08-18 13:33:56 +02:00
}
2016-05-30 22:39:12 +02:00
int Hyperion : : getCurrentPriority ( ) const
{
2016-07-31 22:21:35 +02:00
return _sourceAutoSelectEnabled | | ! _muxer . hasPriority ( _currentSourcePriority ) ? _muxer . getCurrentPriority ( ) : _currentSourcePriority ;
2016-05-30 22:39:12 +02:00
}
2013-08-19 20:33:36 +02:00
QList < int > Hyperion : : getActivePriorities ( ) const
{
return _muxer . getPriorities ( ) ;
}
const Hyperion : : InputInfo & Hyperion : : getPriorityInfo ( const int priority ) const
{
return _muxer . getInputInfo ( priority ) ;
}
2013-12-01 14:09:01 +01:00
const std : : list < EffectDefinition > & Hyperion : : getEffects ( ) const
2013-11-24 16:10:48 +01:00
{
return _effectEngine - > getEffects ( ) ;
}
2016-04-24 17:07:31 +02:00
const std : : list < ActiveEffectDefinition > & Hyperion : : getActiveEffects ( )
{
return _effectEngine - > getActiveEffects ( ) ;
}
2013-11-24 16:10:48 +01:00
int Hyperion : : setEffect ( const std : : string & effectName , int priority , int timeout )
{
return _effectEngine - > runEffect ( effectName , priority , timeout ) ;
}
2013-12-01 14:09:01 +01:00
int Hyperion : : setEffect ( const std : : string & effectName , const Json : : Value & args , int priority , int timeout )
{
return _effectEngine - > runEffect ( effectName , args , priority , timeout ) ;
}
2013-08-14 17:02:09 +02:00
void Hyperion : : update ( )
{
// Update the muxer, cleaning obsolete priorities
2013-08-18 13:33:56 +02:00
_muxer . setCurrentTime ( QDateTime : : currentMSecsSinceEpoch ( ) ) ;
2013-08-14 17:02:09 +02:00
// Obtain the current priority channel
2016-07-31 22:21:35 +02:00
int priority = _sourceAutoSelectEnabled | | ! _muxer . hasPriority ( _currentSourcePriority ) ? _muxer . getCurrentPriority ( ) : _currentSourcePriority ;
const PriorityMuxer : : InputInfo & priorityInfo = _muxer . getInputInfo ( priority ) ;
2013-08-14 17:02:09 +02:00
2016-07-01 23:20:41 +02:00
// copy ledcolors to local buffer
_ledBuffer . reserve ( _hwLedCount ) ;
_ledBuffer = priorityInfo . ledColors ;
2015-12-22 10:01:47 +01:00
// Apply the correction and the transform to each led and color-channel
2016-04-02 00:04:11 +02:00
// Avoid applying correction, the same task is performed by adjustment
2016-07-30 13:07:53 +02:00
if ( _transformEnabled ) _raw2ledTransform - > applyTransform ( _ledBuffer ) ;
if ( _adjustmentEnabled ) _raw2ledAdjustment - > applyAdjustment ( _ledBuffer ) ;
if ( _temperatureEnabled ) _raw2ledTemperature - > applyCorrection ( _ledBuffer ) ;
2016-07-01 23:20:41 +02:00
2016-10-08 08:14:36 +02:00
// init colororder vector, if empty
2016-08-08 00:17:00 +02:00
if ( _ledStringColorOrder . empty ( ) )
{
for ( Led & led : _ledString . leds ( ) )
{
_ledStringColorOrder . push_back ( led . colorOrder ) ;
}
for ( Led & led : _ledStringClone . leds ( ) )
{
_ledStringColorOrder . insert ( _ledStringColorOrder . begin ( ) + led . index , led . colorOrder ) ;
}
}
// insert cloned leds into buffer
for ( Led & led : _ledStringClone . leds ( ) )
{
_ledBuffer . insert ( _ledBuffer . begin ( ) + led . index , _ledBuffer . at ( led . clone ) ) ;
}
2015-01-01 19:31:04 +01:00
int i = 0 ;
2016-07-01 23:20:41 +02:00
for ( ColorRgb & color : _ledBuffer )
2013-08-21 21:50:17 +02:00
{
2016-08-08 00:17:00 +02:00
//const ColorOrder ledColorOrder = leds.at(i).colorOrder;
2013-11-04 20:52:57 +01:00
// correct the color byte order
2016-08-08 00:17:00 +02:00
switch ( _ledStringColorOrder . at ( i ) )
2013-10-27 10:18:31 +01:00
{
2013-11-04 20:52:57 +01:00
case ORDER_RGB :
// leave as it is
break ;
case ORDER_BGR :
2013-10-27 10:18:31 +01:00
std : : swap ( color . red , color . blue ) ;
2013-11-04 20:52:57 +01:00
break ;
case ORDER_RBG :
std : : swap ( color . green , color . blue ) ;
break ;
case ORDER_GRB :
std : : swap ( color . red , color . green ) ;
break ;
case ORDER_GBR :
{
2016-05-22 00:30:53 +02:00
std : : swap ( color . red , color . green ) ;
std : : swap ( color . green , color . blue ) ;
2013-11-04 20:52:57 +01:00
break ;
}
case ORDER_BRG :
{
2016-05-22 00:30:53 +02:00
std : : swap ( color . red , color . blue ) ;
std : : swap ( color . green , color . blue ) ;
2013-11-04 20:52:57 +01:00
break ;
}
2013-10-27 10:18:31 +01:00
}
2015-01-01 19:31:04 +01:00
i + + ;
2013-08-21 21:50:17 +02:00
}
2013-08-14 17:02:09 +02:00
2016-07-01 23:20:41 +02:00
if ( _hwLedCount > _ledBuffer . size ( ) )
{
_ledBuffer . resize ( _hwLedCount , ColorRgb : : BLACK ) ;
}
2013-08-14 17:02:09 +02:00
// Write the data to the device
2016-09-08 16:32:42 +02:00
if ( _deviceSmooth - > enabled ( ) )
2016-09-23 08:49:22 +02:00
_deviceSmooth - > setLedValues ( _ledBuffer ) ;
2016-09-08 16:32:42 +02:00
else
2016-09-23 08:49:22 +02:00
_device - > setLedValues ( _ledBuffer ) ;
2013-08-14 17:02:09 +02:00
// Start the timeout-timer
if ( priorityInfo . timeoutTime_ms = = - 1 )
{
_timer . stop ( ) ;
}
else
{
int timeout_ms = std : : max ( 0 , int ( priorityInfo . timeoutTime_ms - QDateTime : : currentMSecsSinceEpoch ( ) ) ) ;
_timer . start ( timeout_ms ) ;
2013-07-26 22:38:34 +02:00
}
}