2013-08-13 11:10:45 +02:00
# pragma once
// STL includes
# include <vector>
# include <cstdint>
// QT includes
# include <QMap>
2017-06-17 23:29:04 +02:00
# include <QObject>
2017-09-01 08:50:37 +02:00
# include <QMap>
# include <QVector>
2013-08-13 11:10:45 +02:00
// Utils includes
2013-11-11 10:00:37 +01:00
# include <utils/ColorRgb.h>
2018-12-27 23:11:32 +01:00
# include <utils/Image.h>
2016-10-10 18:29:54 +02:00
# include <utils/Components.h>
2013-08-13 11:10:45 +02:00
2017-08-04 12:01:45 +02:00
// global defines
# define SMOOTHING_MODE_DEFAULT 0
# define SMOOTHING_MODE_PAUSE 1
2018-12-27 23:11:32 +01:00
class QTimer ;
class Logger ;
2017-08-04 12:01:45 +02:00
2013-09-06 21:26:58 +02:00
///
2018-12-27 23:11:32 +01:00
/// The PriorityMuxer handles the priority channels. Led values input/ images are written to the priority map
2013-09-06 21:26:58 +02:00
/// and the muxer keeps track of all active priorities. The current priority can be queried and per
2018-12-27 23:11:32 +01:00
/// priority the led colors. Handles also manual/auto selection mode, provides a lot of signals to hook into priority related events
2013-09-06 21:26:58 +02:00
///
2017-06-17 23:29:04 +02:00
class PriorityMuxer : public QObject
2013-08-13 11:10:45 +02:00
{
2017-06-17 23:29:04 +02:00
Q_OBJECT
2013-08-13 11:10:45 +02:00
public :
2013-09-06 21:26:58 +02:00
///
/// The information structure for a single priority channel
///
2013-08-13 11:10:45 +02:00
struct InputInfo
{
2013-09-06 21:26:58 +02:00
/// The priority of this channel
2013-08-13 11:10:45 +02:00
int priority ;
2013-09-06 21:26:58 +02:00
/// The absolute timeout of the channel
2013-08-13 11:10:45 +02:00
int64_t timeoutTime_ms ;
2013-09-06 21:26:58 +02:00
/// The colors for each led of the channel
2013-11-11 10:00:37 +01:00
std : : vector < ColorRgb > ledColors ;
2018-12-27 23:11:32 +01:00
/// The raw Image (size should be preprocessed!)
Image < ColorRgb > image ;
2017-02-28 17:53:41 +01:00
/// The component
2016-10-10 18:29:54 +02:00
hyperion : : Components componentId ;
2017-02-28 17:53:41 +01:00
/// Who set it
QString origin ;
2018-12-27 23:11:32 +01:00
/// id of smoothing config
2017-08-04 12:01:45 +02:00
unsigned smooth_cfg ;
2018-12-27 23:11:32 +01:00
/// specific owner description
QString owner ;
2013-08-13 11:10:45 +02:00
} ;
2021-03-19 22:52:41 +01:00
//Foreground and Background priorities
const static int FG_PRIORITY ;
const static int BG_PRIORITY ;
2016-08-06 08:28:42 +02:00
/// The lowest possible priority, which is used when no priority channels are active
2017-09-01 08:50:37 +02:00
const static int LOWEST_PRIORITY ;
2016-08-06 08:28:42 +02:00
2013-09-06 21:26:58 +02:00
///
2020-11-14 17:58:56 +01:00
/// Constructs the PriorityMuxer for the given number of LEDs (used to switch to black when
2013-09-06 21:26:58 +02:00
/// there are no priority channels
///
2020-11-14 17:58:56 +01:00
/// @param ledCount The number of LEDs
2013-09-06 21:26:58 +02:00
///
2020-07-19 15:13:41 +02:00
PriorityMuxer ( int ledCount , QObject * parent ) ;
2013-08-13 11:10:45 +02:00
2013-09-06 21:26:58 +02:00
///
/// Destructor
///
2020-08-08 23:12:43 +02:00
~ PriorityMuxer ( ) override ;
2013-08-13 11:10:45 +02:00
2018-12-27 23:11:32 +01:00
///
/// @brief Start/Stop the PriorityMuxer update timer; On disabled no priority and timeout updates will be performend
/// @param enable The new state
///
2020-08-08 13:09:15 +02:00
void setEnable ( bool enable ) ;
2018-12-27 23:11:32 +01:00
/// @brief Enable or disable auto source selection
/// @param enable True if it should be enabled else false
/// @param update True to update _currentPriority - INTERNAL usage.
/// @return True if changed has been applied, false if the state is unchanged
///
2020-08-08 13:09:15 +02:00
bool setSourceAutoSelectEnabled ( bool enabel , bool update = true ) ;
2018-12-27 23:11:32 +01:00
///
/// @brief Get the state of source auto selection
/// @return True if enabled, else false
///
2020-11-14 17:58:56 +01:00
bool isSourceAutoSelectEnabled ( ) const { return _sourceAutoSelectEnabled ; }
2018-12-27 23:11:32 +01:00
///
2020-11-14 17:58:56 +01:00
/// @brief Overwrite current lowest priority with manual selection; On success disables auto selection
2018-12-27 23:11:32 +01:00
/// @param priority The
/// @return True on success, false if priority not found
///
2020-11-14 17:58:56 +01:00
bool setPriority ( int priority ) ;
2018-12-27 23:11:32 +01:00
///
2020-11-14 17:58:56 +01:00
/// @brief Update all LED-Colors with min length of >= 1 to fit the new led length
/// @param[in] ledCount The count of LEDs
2018-12-27 23:11:32 +01:00
///
2020-08-08 13:09:15 +02:00
void updateLedColorsLength ( int ledCount ) ;
2018-12-27 23:11:32 +01:00
2013-09-06 21:26:58 +02:00
///
/// Returns the current priority
///
/// @return The current priority
///
2019-02-03 14:36:57 +01:00
int getCurrentPriority ( ) const { return _currentPriority ; }
2013-08-13 11:10:45 +02:00
2020-09-14 17:19:14 +02:00
///
/// Returns the previous priority before current priority
///
/// @return The previous priority
///
int getPreviousPriority ( ) const { return _previousPriority ; }
2013-09-06 21:26:58 +02:00
///
/// Returns the state (enabled/disabled) of a specific priority channel
/// @param priority The priority channel
/// @return True if the priority channel exists else false
///
2020-08-08 13:09:15 +02:00
bool hasPriority ( int priority ) const ;
2013-08-13 11:10:45 +02:00
2013-09-06 21:26:58 +02:00
///
/// Returns the number of active priorities
///
/// @return The list with active priorities
///
2013-08-13 11:10:45 +02:00
QList < int > getPriorities ( ) const ;
2013-09-06 21:26:58 +02:00
///
2018-12-27 23:11:32 +01:00
/// Returns the information of a specified priority channel.
/// If a priority is no longer available the _lowestPriorityInfo (255) is returned
2013-09-06 21:26:58 +02:00
///
/// @param priority The priority channel
///
/// @return The information for the specified priority channel
///
2020-08-08 13:09:15 +02:00
InputInfo getInputInfo ( int priority ) const ;
2018-12-27 23:11:32 +01:00
///
/// @brief Register a new input by priority, the priority is not active (timeout -100 isn't muxer recognized) until you start to update the data with setInput()
/// A repeated call to update the base data of a known priority won't overwrite their current timeout
/// @param[in] priority The priority of the channel
/// @param[in] component The component of the channel
/// @param[in] origin Who set the channel (CustomString@IP)
2020-11-14 17:58:56 +01:00
/// @param[in] owner Specific owner string, might be empty
2018-12-27 23:11:32 +01:00
/// @param[in] smooth_cfg The smooth id to use
2013-09-06 21:26:58 +02:00
///
2020-08-08 13:09:15 +02:00
void registerInput ( int priority , hyperion : : Components component , const QString & origin = " System " , const QString & owner = " " , unsigned smooth_cfg = SMOOTHING_MODE_DEFAULT ) ;
2013-08-13 11:10:45 +02:00
2013-09-06 21:26:58 +02:00
///
2020-11-14 17:58:56 +01:00
/// @brief Update the current color of a priority (previous registered with registerInput())
2018-12-27 23:11:32 +01:00
/// @param priority The priority to update
/// @param ledColors The colors
/// @param timeout_ms The new timeout (defaults to -1 endless)
/// @return True on success, false when priority is not found
2013-09-06 21:26:58 +02:00
///
2020-08-08 13:09:15 +02:00
bool setInput ( int priority , const std : : vector < ColorRgb > & ledColors , int64_t timeout_ms = - 1 ) ;
2018-12-27 23:11:32 +01:00
2013-09-06 21:26:58 +02:00
///
2018-12-27 23:11:32 +01:00
/// @brief Update the current image of a priority (prev registered with registerInput())
/// @param priority The priority to update
/// @param image The new image
/// @param timeout_ms The new timeout (defaults to -1 endless)
/// @return True on success, false when priority is not found
///
2020-08-08 13:09:15 +02:00
bool setInputImage ( int priority , const Image < ColorRgb > & image , int64_t timeout_ms = - 1 ) ;
2013-08-13 11:10:45 +02:00
2018-12-28 18:12:45 +01:00
///
/// @brief Set the given priority to inactive
/// @param priority The priority
/// @return True on success false if not found
///
2020-11-14 17:58:56 +01:00
bool setInputInactive ( int priority ) ;
2018-12-28 18:12:45 +01:00
2013-09-06 21:26:58 +02:00
///
2018-12-27 23:11:32 +01:00
/// Clears the specified priority channel and update _currentPriority on success
2013-09-06 21:26:58 +02:00
///
/// @param[in] priority The priority of the channel to clear
2018-12-27 23:11:32 +01:00
/// @return True if priority has been cleared else false (not found)
2013-09-06 21:26:58 +02:00
///
2020-11-14 17:58:56 +01:00
bool clearInput ( int priority ) ;
2013-08-13 11:10:45 +02:00
2013-09-06 21:26:58 +02:00
///
/// Clears all priority channels
///
2017-08-12 07:55:32 +02:00
void clearAll ( bool forceClearAll = false ) ;
2013-08-13 11:10:45 +02:00
2019-07-21 19:06:47 +02:00
///
2020-11-14 17:58:56 +01:00
/// @brief Queue a manual push where muxer doesn't recognize them (e.g. continuous single color pushes)
2019-07-21 19:06:47 +02:00
///
2020-08-08 23:12:43 +02:00
void queuePush ( ) { emit timeRunner ( ) ; }
2019-07-21 19:06:47 +02:00
2018-12-27 23:11:32 +01:00
signals :
2013-09-06 21:26:58 +02:00
///
2018-12-27 23:11:32 +01:00
/// @brief Signal which emits when a effect or color with timeout > -1 is running, once per second
///
void timeRunner ( ) ;
2017-06-17 23:29:04 +02:00
///
2018-12-27 23:11:32 +01:00
/// @brief Emits whenever the visible priority has changed
2019-02-03 14:36:57 +01:00
/// @param priority The new visible priority
2018-12-27 23:11:32 +01:00
///
2020-08-08 13:09:15 +02:00
void visiblePriorityChanged ( quint8 priority ) ;
2018-12-27 23:11:32 +01:00
2020-02-15 22:47:27 +01:00
///
/// @brief Emits whenever the current visible component changed
/// @param comp The new component
///
2020-08-08 13:09:15 +02:00
void visibleComponentChanged ( hyperion : : Components comp ) ;
2020-02-15 22:47:27 +01:00
2018-12-27 23:11:32 +01:00
///
/// @brief Emits whenever something changes which influences the priorities listing
/// Emits also in 1s interval when a COLOR or EFFECT is running with a timeout > -1
2017-06-17 23:29:04 +02:00
///
2020-08-08 13:09:15 +02:00
void prioritiesChanged ( ) ;
2018-12-27 23:11:32 +01:00
///
/// internal used signal to resolve treading issues with timer
///
void signalTimeTrigger ( ) ;
2017-06-17 23:29:04 +02:00
private slots :
///
2018-12-27 23:11:32 +01:00
/// Slot which is called to adapt to 1s interval for signal timeRunner() / prioritiesChanged()
2017-06-17 23:29:04 +02:00
///
2018-12-27 23:11:32 +01:00
void timeTrigger ( ) ;
///
/// Updates the current time. Channels with a configured time out will be checked and cleared if
/// required.
///
2020-08-08 13:09:15 +02:00
void setCurrentTime ( ) ;
2017-06-17 23:29:04 +02:00
2013-08-13 11:10:45 +02:00
private :
2020-02-15 22:47:27 +01:00
///
/// @brief Get the component of the given priority
/// @return The component
///
2020-08-08 23:12:43 +02:00
hyperion : : Components getComponentOfPriority ( int priority ) const ;
2020-02-15 22:47:27 +01:00
2018-12-27 23:11:32 +01:00
/// Logger instance
Logger * _log ;
2013-09-06 21:26:58 +02:00
/// The current priority (lowest value in _activeInputs)
2013-08-18 13:33:56 +02:00
int _currentPriority ;
2013-08-13 11:10:45 +02:00
2020-09-14 17:19:14 +02:00
/// The previous priority before current priority
int _previousPriority ;
2018-12-27 23:11:32 +01:00
/// The manual select priority set with setPriority
int _manualSelectedPriority ;
2020-02-15 22:47:27 +01:00
// The last visible component
hyperion : : Components _prevVisComp = hyperion : : COMP_INVALID ;
2013-09-06 21:26:58 +02:00
/// The mapping from priority channel to led-information
2013-08-18 13:33:56 +02:00
QMap < int , InputInfo > _activeInputs ;
2013-08-13 11:10:45 +02:00
2013-09-06 21:26:58 +02:00
/// The information of the lowest priority channel
2013-08-18 13:33:56 +02:00
InputInfo _lowestPriorityInfo ;
2018-12-27 23:11:32 +01:00
// Reflect the state of auto select
bool _sourceAutoSelectEnabled ;
// Timer to update Muxer times independent
QTimer * _updateTimer ;
QTimer * _timer ;
QTimer * _blockTimer ;
2013-08-13 11:10:45 +02:00
} ;