2013-10-27 18:04:37 +01:00
|
|
|
// Qt includes
|
|
|
|
#include <QDateTime>
|
|
|
|
|
|
|
|
#include "LinearColorSmoothing.h"
|
2016-09-07 20:10:37 +02:00
|
|
|
#include <hyperion/Hyperion.h>
|
2013-10-27 18:04:37 +01:00
|
|
|
|
2016-10-08 08:14:36 +02:00
|
|
|
#include <cmath>
|
|
|
|
|
2016-08-11 07:13:55 +02:00
|
|
|
using namespace hyperion;
|
|
|
|
|
2017-08-04 12:01:45 +02:00
|
|
|
// ledUpdateFrequency_hz = 0 > cause divide by zero!
|
2016-07-13 11:18:12 +02:00
|
|
|
LinearColorSmoothing::LinearColorSmoothing( LedDevice * ledDevice, double ledUpdateFrequency_hz, int settlingTime_ms, unsigned updateDelay, bool continuousOutput)
|
2016-08-14 10:46:44 +02:00
|
|
|
: LedDevice()
|
2016-07-13 11:18:12 +02:00
|
|
|
, _ledDevice(ledDevice)
|
|
|
|
, _updateInterval(1000 / ledUpdateFrequency_hz)
|
|
|
|
, _settlingTime(settlingTime_ms)
|
|
|
|
, _timer()
|
|
|
|
, _outputDelay(updateDelay)
|
|
|
|
, _writeToLedsEnable(true)
|
|
|
|
, _continuousOutput(continuousOutput)
|
2017-03-30 06:20:20 +02:00
|
|
|
, _pause(false)
|
2017-08-04 12:01:45 +02:00
|
|
|
, _currentConfigId(0)
|
2013-10-27 18:04:37 +01:00
|
|
|
{
|
2016-08-11 07:13:55 +02:00
|
|
|
_log = Logger::getInstance("Smoothing");
|
2013-10-27 18:04:37 +01:00
|
|
|
_timer.setSingleShot(false);
|
|
|
|
_timer.setInterval(_updateInterval);
|
|
|
|
|
2017-08-04 12:01:45 +02:00
|
|
|
selectConfig( addConfig(_settlingTime, ledUpdateFrequency_hz, updateDelay) );
|
|
|
|
|
|
|
|
// add pause on cfg 1
|
|
|
|
SMOOTHING_CFG cfg = {true, 100, 50, 0};
|
|
|
|
_cfgList.append(cfg);
|
|
|
|
Info( _log, "smoothing cfg %d: pause", _cfgList.count()-1);
|
|
|
|
|
2013-10-27 18:04:37 +01:00
|
|
|
connect(&_timer, SIGNAL(timeout()), this, SLOT(updateLeds()));
|
|
|
|
}
|
|
|
|
|
|
|
|
LinearColorSmoothing::~LinearColorSmoothing()
|
|
|
|
{
|
2014-09-30 21:00:42 +02:00
|
|
|
// Make sure to switch off the underlying led-device (because switchOff is no longer forwarded)
|
|
|
|
_ledDevice->switchOff();
|
2013-10-27 18:04:37 +01:00
|
|
|
delete _ledDevice;
|
|
|
|
}
|
|
|
|
|
2013-11-11 09:00:37 +00:00
|
|
|
int LinearColorSmoothing::write(const std::vector<ColorRgb> &ledValues)
|
2013-10-27 18:04:37 +01:00
|
|
|
{
|
2016-09-08 16:32:42 +02:00
|
|
|
// received a new target color
|
|
|
|
if (_previousValues.empty())
|
2013-10-27 18:04:37 +01:00
|
|
|
{
|
2016-09-08 16:32:42 +02:00
|
|
|
// not initialized yet
|
|
|
|
_targetTime = QDateTime::currentMSecsSinceEpoch() + _settlingTime;
|
|
|
|
_targetValues = ledValues;
|
|
|
|
|
|
|
|
_previousTime = QDateTime::currentMSecsSinceEpoch();
|
|
|
|
_previousValues = ledValues;
|
|
|
|
_timer.start();
|
2013-10-27 18:04:37 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-09-08 16:32:42 +02:00
|
|
|
_targetTime = QDateTime::currentMSecsSinceEpoch() + _settlingTime;
|
|
|
|
memcpy(_targetValues.data(), ledValues.data(), ledValues.size() * sizeof(ColorRgb));
|
2013-10-27 18:04:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int LinearColorSmoothing::switchOff()
|
|
|
|
{
|
2014-09-25 22:05:01 +02:00
|
|
|
// We will keep updating the leds (but with pure-black)
|
|
|
|
|
2014-09-22 21:28:38 +02:00
|
|
|
// Clear the smoothing parameters
|
|
|
|
std::fill(_targetValues.begin(), _targetValues.end(), ColorRgb::BLACK);
|
2013-10-27 18:04:37 +01:00
|
|
|
_targetTime = 0;
|
|
|
|
|
2014-09-22 21:28:38 +02:00
|
|
|
// Erase the output-queue
|
|
|
|
for (unsigned i=0; i<_outputQueue.size(); ++i)
|
|
|
|
{
|
|
|
|
_outputQueue.push_back(_targetValues);
|
|
|
|
_outputQueue.pop_front();
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2013-10-27 18:04:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void LinearColorSmoothing::updateLeds()
|
|
|
|
{
|
|
|
|
int64_t now = QDateTime::currentMSecsSinceEpoch();
|
|
|
|
int deltaTime = _targetTime - now;
|
|
|
|
|
|
|
|
if (deltaTime < 0)
|
|
|
|
{
|
2013-11-11 09:00:37 +00:00
|
|
|
memcpy(_previousValues.data(), _targetValues.data(), _targetValues.size() * sizeof(ColorRgb));
|
2013-10-27 18:04:37 +01:00
|
|
|
_previousTime = now;
|
|
|
|
|
2014-09-22 21:28:38 +02:00
|
|
|
queueColors(_previousValues);
|
2016-07-13 11:18:12 +02:00
|
|
|
_writeToLedsEnable = _continuousOutput;
|
2013-10-27 18:04:37 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-05-26 07:01:10 +02:00
|
|
|
_writeToLedsEnable = true;
|
2013-10-27 18:04:37 +01:00
|
|
|
float k = 1.0f - 1.0f * deltaTime / (_targetTime - _previousTime);
|
|
|
|
|
2016-10-08 08:14:36 +02:00
|
|
|
int reddif = 0, greendif = 0, bluedif = 0;
|
2013-10-27 18:04:37 +01:00
|
|
|
|
2016-10-08 08:14:36 +02:00
|
|
|
for (size_t i = 0; i < _previousValues.size(); ++i)
|
|
|
|
{
|
|
|
|
ColorRgb & prev = _previousValues[i];
|
|
|
|
ColorRgb & target = _targetValues[i];
|
2016-10-04 22:13:29 +02:00
|
|
|
|
2016-10-08 08:14:36 +02:00
|
|
|
reddif = target.red - prev.red;
|
|
|
|
greendif = target.green - prev.green;
|
|
|
|
bluedif = target.blue - prev.blue;
|
2016-10-04 22:13:29 +02:00
|
|
|
|
2016-10-08 08:14:36 +02:00
|
|
|
prev.red += (reddif < 0 ? -1:1) * std::ceil(k * std::abs(reddif));
|
|
|
|
prev.green += (greendif < 0 ? -1:1) * std::ceil(k * std::abs(greendif));
|
|
|
|
prev.blue += (bluedif < 0 ? -1:1) * std::ceil(k * std::abs(bluedif));
|
|
|
|
}
|
2013-10-27 18:04:37 +01:00
|
|
|
_previousTime = now;
|
|
|
|
|
2014-09-22 21:28:38 +02:00
|
|
|
queueColors(_previousValues);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void LinearColorSmoothing::queueColors(const std::vector<ColorRgb> & ledColors)
|
|
|
|
{
|
|
|
|
if (_outputDelay == 0)
|
|
|
|
{
|
2014-09-25 22:05:01 +02:00
|
|
|
// No output delay => immediate write
|
2017-03-30 06:20:20 +02:00
|
|
|
if ( _writeToLedsEnable && !_pause)
|
2016-09-23 08:49:22 +02:00
|
|
|
_ledDevice->setLedValues(ledColors);
|
2014-09-22 21:28:38 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-07-13 11:18:12 +02:00
|
|
|
// Push new colors in the delay-buffer
|
|
|
|
if ( _writeToLedsEnable )
|
|
|
|
_outputQueue.push_back(ledColors);
|
|
|
|
|
2014-09-25 22:05:01 +02:00
|
|
|
// If the delay-buffer is filled pop the front and write to device
|
2016-07-13 11:18:12 +02:00
|
|
|
if (_outputQueue.size() > 0 )
|
2014-09-22 21:28:38 +02:00
|
|
|
{
|
2016-07-13 11:18:12 +02:00
|
|
|
if ( _outputQueue.size() > _outputDelay || !_writeToLedsEnable )
|
|
|
|
{
|
2017-03-30 06:20:20 +02:00
|
|
|
if (!_pause)
|
|
|
|
{
|
|
|
|
_ledDevice->setLedValues(_outputQueue.front());
|
|
|
|
}
|
2016-07-13 11:18:12 +02:00
|
|
|
_outputQueue.pop_front();
|
|
|
|
}
|
2014-09-22 21:28:38 +02:00
|
|
|
}
|
2013-10-27 18:04:37 +01:00
|
|
|
}
|
|
|
|
}
|
2016-08-11 07:13:55 +02:00
|
|
|
|
2016-09-08 16:32:42 +02:00
|
|
|
|
|
|
|
void LinearColorSmoothing::setEnable(bool enable)
|
2016-08-11 07:13:55 +02:00
|
|
|
{
|
2017-03-21 17:55:46 +01:00
|
|
|
LedDevice::setEnable(enable);
|
|
|
|
|
2016-09-08 18:07:57 +02:00
|
|
|
if (!enable)
|
|
|
|
{
|
|
|
|
_timer.stop();
|
|
|
|
_previousValues.clear();
|
|
|
|
}
|
2016-08-11 07:13:55 +02:00
|
|
|
}
|
2017-03-30 06:20:20 +02:00
|
|
|
|
|
|
|
void LinearColorSmoothing::setPause(bool pause)
|
|
|
|
{
|
|
|
|
_pause = pause;
|
|
|
|
}
|
|
|
|
|
2017-08-04 12:01:45 +02:00
|
|
|
unsigned LinearColorSmoothing::addConfig(int settlingTime_ms, double ledUpdateFrequency_hz, unsigned updateDelay)
|
|
|
|
{
|
|
|
|
SMOOTHING_CFG cfg = {false, settlingTime_ms, int64_t(1000.0/ledUpdateFrequency_hz), updateDelay};
|
|
|
|
_cfgList.append(cfg);
|
|
|
|
|
|
|
|
Info( _log, "smoothing cfg %d: interval: %d ms, settlingTime: %d ms, updateDelay: %d frames", _cfgList.count()-1, cfg.updateInterval, cfg.settlingTime, cfg.outputDelay );
|
|
|
|
return _cfgList.count() - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LinearColorSmoothing::selectConfig(unsigned cfg)
|
|
|
|
{
|
|
|
|
if (_currentConfigId == cfg)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( cfg < (unsigned)_cfgList.count())
|
|
|
|
{
|
|
|
|
_settlingTime = _cfgList[cfg].settlingTime;
|
|
|
|
_outputDelay = _cfgList[cfg].outputDelay;
|
|
|
|
_pause = _cfgList[cfg].pause;
|
|
|
|
|
|
|
|
if (_cfgList[cfg].updateInterval != _updateInterval)
|
|
|
|
{
|
|
|
|
_timer.stop();
|
|
|
|
_updateInterval = _cfgList[cfg].updateInterval;
|
|
|
|
_timer.setInterval(_updateInterval);
|
|
|
|
_timer.start();
|
|
|
|
}
|
|
|
|
_currentConfigId = cfg;
|
|
|
|
InfoIf( enabled() && !_pause, _log, "set smoothing cfg: %d, interval: %d ms, settlingTime: %d ms, updateDelay: %d frames", _currentConfigId, _updateInterval, _settlingTime, _outputDelay );
|
|
|
|
InfoIf( _pause, _log, "set smoothing cfg: %d, pause", _currentConfigId );
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// reset to default
|
|
|
|
_currentConfigId = 0;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|