2016-05-26 23:44:27 +02:00
|
|
|
#pragma once
|
|
|
|
|
2017-03-31 10:17:14 +02:00
|
|
|
// STL includes
|
|
|
|
#include <set>
|
|
|
|
|
2016-05-26 23:44:27 +02:00
|
|
|
// Qt includes
|
|
|
|
#include <QNetworkAccessManager>
|
|
|
|
#include <QTimer>
|
2016-09-23 08:49:22 +02:00
|
|
|
|
2016-05-26 23:44:27 +02:00
|
|
|
// Leddevice includes
|
|
|
|
#include <leddevice/LedDevice.h>
|
|
|
|
|
2017-03-31 10:17:14 +02:00
|
|
|
// Forward declaration
|
|
|
|
struct CiColorTriangle;
|
|
|
|
|
2016-05-26 23:44:27 +02:00
|
|
|
/**
|
|
|
|
* A color point in the color space of the hue system.
|
|
|
|
*/
|
2017-03-31 10:17:14 +02:00
|
|
|
struct CiColor
|
|
|
|
{
|
2016-05-26 23:44:27 +02:00
|
|
|
/// X component.
|
2020-03-26 18:49:44 +01:00
|
|
|
double x;
|
2016-05-26 23:44:27 +02:00
|
|
|
/// Y component.
|
2020-03-26 18:49:44 +01:00
|
|
|
double y;
|
2016-05-26 23:44:27 +02:00
|
|
|
/// The brightness.
|
2020-03-26 18:49:44 +01:00
|
|
|
double bri;
|
2016-05-26 23:44:27 +02:00
|
|
|
|
|
|
|
///
|
|
|
|
/// Converts an RGB color to the Hue xy color space and brightness.
|
|
|
|
/// https://github.com/PhilipsHue/PhilipsHueSDK-iOS-OSX/blob/master/ApplicationDesignNotes/RGB%20to%20xy%20Color%20conversion.md
|
|
|
|
///
|
|
|
|
/// @param red the red component in [0, 1]
|
2017-03-31 10:17:14 +02:00
|
|
|
///
|
2016-05-26 23:44:27 +02:00
|
|
|
/// @param green the green component in [0, 1]
|
2017-03-31 10:17:14 +02:00
|
|
|
///
|
2016-05-26 23:44:27 +02:00
|
|
|
/// @param blue the blue component in [0, 1]
|
|
|
|
///
|
|
|
|
/// @return color point
|
|
|
|
///
|
2020-03-26 18:49:44 +01:00
|
|
|
static CiColor rgbToCiColor(double red, double green, double blue, CiColorTriangle colorSpace);
|
2016-05-26 23:44:27 +02:00
|
|
|
|
|
|
|
///
|
|
|
|
/// @param p the color point to check
|
2017-03-31 10:17:14 +02:00
|
|
|
///
|
2016-05-26 23:44:27 +02:00
|
|
|
/// @return true if the color point is covered by the lamp color space
|
|
|
|
///
|
2017-03-31 10:17:14 +02:00
|
|
|
static bool isPointInLampsReach(CiColor p, CiColorTriangle colorSpace);
|
2016-05-26 23:44:27 +02:00
|
|
|
|
|
|
|
///
|
|
|
|
/// @param p1 point one
|
2017-03-31 10:17:14 +02:00
|
|
|
///
|
2016-05-26 23:44:27 +02:00
|
|
|
/// @param p2 point tow
|
|
|
|
///
|
|
|
|
/// @return the cross product between p1 and p2
|
|
|
|
///
|
2020-03-26 18:49:44 +01:00
|
|
|
static double crossProduct(CiColor p1, CiColor p2);
|
2016-05-26 23:44:27 +02:00
|
|
|
|
|
|
|
///
|
|
|
|
/// @param a reference point one
|
2017-03-31 10:17:14 +02:00
|
|
|
///
|
2016-05-26 23:44:27 +02:00
|
|
|
/// @param b reference point two
|
2017-03-31 10:17:14 +02:00
|
|
|
///
|
2016-05-26 23:44:27 +02:00
|
|
|
/// @param p the point to which the closest point is to be found
|
|
|
|
///
|
|
|
|
/// @return the closest color point of p to a and b
|
|
|
|
///
|
2017-03-31 10:17:14 +02:00
|
|
|
static CiColor getClosestPointToPoint(CiColor a, CiColor b, CiColor p);
|
2016-05-26 23:44:27 +02:00
|
|
|
|
|
|
|
///
|
|
|
|
/// @param p1 point one
|
2017-03-31 10:17:14 +02:00
|
|
|
///
|
2016-05-26 23:44:27 +02:00
|
|
|
/// @param p2 point tow
|
|
|
|
///
|
|
|
|
/// @return the distance between the two points
|
|
|
|
///
|
2020-03-26 18:49:44 +01:00
|
|
|
static double getDistanceBetweenTwoPoints(CiColor p1, CiColor p2);
|
2017-03-31 10:17:14 +02:00
|
|
|
};
|
|
|
|
|
2020-03-26 18:49:44 +01:00
|
|
|
bool operator==(const CiColor& p1, const CiColor& p2);
|
|
|
|
bool operator!=(const CiColor& p1, const CiColor& p2);
|
2017-03-31 10:17:14 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Color triangle to define an available color space for the hue lamps.
|
|
|
|
*/
|
|
|
|
struct CiColorTriangle
|
|
|
|
{
|
|
|
|
CiColor red, green, blue;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Simple class to hold the id, the latest color, the color space and the original state.
|
|
|
|
*/
|
|
|
|
class PhilipsHueLight
|
|
|
|
{
|
|
|
|
|
|
|
|
public:
|
|
|
|
// Hue system model ids (http://www.developers.meethue.com/documentation/supported-lights).
|
|
|
|
// Light strips, color iris, ...
|
|
|
|
static const std::set<QString> GAMUT_A_MODEL_IDS;
|
|
|
|
// Hue bulbs, spots, ...
|
|
|
|
static const std::set<QString> GAMUT_B_MODEL_IDS;
|
|
|
|
// Hue Lightstrip plus, go ...
|
|
|
|
static const std::set<QString> GAMUT_C_MODEL_IDS;
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Constructs the light.
|
|
|
|
///
|
|
|
|
/// @param log the logger
|
|
|
|
/// @param bridge the bridge
|
|
|
|
/// @param id the light id
|
|
|
|
///
|
2020-03-26 18:49:44 +01:00
|
|
|
PhilipsHueLight(Logger* log, unsigned int id, QJsonObject values, unsigned int ledidx);
|
2017-03-31 10:17:14 +02:00
|
|
|
~PhilipsHueLight();
|
|
|
|
|
|
|
|
///
|
|
|
|
/// @param on
|
|
|
|
///
|
2020-03-26 18:49:44 +01:00
|
|
|
void setOnOffState(bool on);
|
2017-03-31 10:17:14 +02:00
|
|
|
|
|
|
|
///
|
|
|
|
/// @param transitionTime the transition time between colors in multiples of 100 ms
|
|
|
|
///
|
2020-03-26 18:49:44 +01:00
|
|
|
void setTransitionTime(unsigned int _transitionTime);
|
2017-03-31 10:17:14 +02:00
|
|
|
|
|
|
|
///
|
|
|
|
/// @param color the color to set
|
|
|
|
///
|
2020-03-26 18:49:44 +01:00
|
|
|
void setColor(const CiColor& _color);
|
|
|
|
|
|
|
|
|
|
|
|
unsigned int getId() const;
|
|
|
|
|
|
|
|
bool getOnOffState() const;
|
|
|
|
unsigned int getTransitionTime() const;
|
2017-03-31 10:17:14 +02:00
|
|
|
CiColor getColor() const;
|
|
|
|
|
|
|
|
///
|
|
|
|
/// @return the color space of the light determined by the model id reported by the bridge.
|
|
|
|
CiColorTriangle getColorSpace() const;
|
|
|
|
|
2020-03-26 18:49:44 +01:00
|
|
|
|
|
|
|
QString getOriginalState();
|
|
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
void saveOriginalState(const QJsonObject& values);
|
|
|
|
|
|
|
|
Logger* _log;
|
|
|
|
/// light id
|
|
|
|
unsigned int _id;
|
|
|
|
unsigned int _ledidx;
|
|
|
|
bool _on;
|
|
|
|
unsigned int _transitionTime;
|
|
|
|
CiColor _color;
|
|
|
|
/// darkes blue color in hue lamp GAMUT = black
|
|
|
|
CiColor _colorBlack;
|
|
|
|
/// The model id of the hue lamp which is used to determine the color space.
|
|
|
|
QString _modelId;
|
|
|
|
QString _lightname;
|
|
|
|
CiColorTriangle _colorSpace;
|
|
|
|
|
|
|
|
/// The json string of the original state.
|
|
|
|
QJsonObject _originalStateJSON;
|
|
|
|
|
|
|
|
QString _originalState;
|
|
|
|
CiColor _originalColor;
|
|
|
|
};
|
|
|
|
|
|
|
|
class LedDevicePhilipsHueBridge : public LedDevice
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
explicit LedDevicePhilipsHueBridge(const QJsonObject &deviceConfig);
|
|
|
|
~LedDevicePhilipsHueBridge();
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Sets configuration
|
|
|
|
///
|
|
|
|
/// @param deviceConfig the json device config
|
|
|
|
/// @return true if success
|
|
|
|
virtual bool init(const QJsonObject &deviceConfig) override;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
///
|
|
|
|
/// @param route the route of the POST request.
|
|
|
|
///
|
|
|
|
/// @param content the content of the POST request.
|
|
|
|
///
|
|
|
|
void post(const QString& route, const QString& content);
|
|
|
|
|
|
|
|
void setLightState(unsigned int lightId = 0, QString state = "");
|
|
|
|
|
|
|
|
const QMap<quint16,QJsonObject>& getLightMap();
|
|
|
|
|
|
|
|
// /// Set device in error state
|
|
|
|
// ///
|
|
|
|
// /// @param errorMsg The error message to be logged
|
|
|
|
// ///
|
|
|
|
// virtual void setInError( const QString& errorMsg) override;
|
|
|
|
|
|
|
|
public slots:
|
|
|
|
///
|
|
|
|
/// Connect to bridge to check availbility and user
|
|
|
|
///
|
|
|
|
virtual int open(void) override;
|
|
|
|
virtual int open( const QString& hostname, const QString& port, const QString& username );
|
|
|
|
|
|
|
|
//signals:
|
|
|
|
// ///
|
|
|
|
// /// Emits with a QMap of current bridge light/value pairs
|
|
|
|
// ///
|
|
|
|
// void newLights(QMap<quint16,QJsonObject> map);
|
|
|
|
protected:
|
|
|
|
|
|
|
|
/// Ip address of the bridge
|
|
|
|
QString _hostname;
|
|
|
|
QString _api_port;
|
|
|
|
/// User name for the API ("newdeveloper")
|
|
|
|
QString _username;
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Discover device via SSDP identifiers
|
|
|
|
///
|
|
|
|
/// @return True, if device was found
|
|
|
|
///
|
|
|
|
bool discoverDevice();
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Get command as url
|
|
|
|
///
|
|
|
|
/// @param host Hostname or IP
|
|
|
|
/// @param port IP-Port
|
|
|
|
/// @param _auth_token Authorization token
|
|
|
|
/// @param Endpoint command for request
|
|
|
|
/// @return Url to execute endpoint/command
|
|
|
|
///
|
|
|
|
QString getUrl(QString host, QString port, QString auth_token, QString endpoint) const;
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Execute GET request
|
|
|
|
///
|
|
|
|
/// @param url GET request for url
|
|
|
|
/// @return Response from device
|
|
|
|
///
|
|
|
|
QJsonDocument getJson(QString url);
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Execute PUT request
|
|
|
|
///
|
|
|
|
/// @param Url for PUT request
|
|
|
|
/// @param json Command for request
|
|
|
|
/// @return Response from device
|
|
|
|
///
|
|
|
|
QJsonDocument putJson(QString url, QString json);
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Handle replys for GET and PUT requests
|
|
|
|
///
|
|
|
|
/// @param reply Network reply
|
|
|
|
/// @return Response for request, if no error
|
|
|
|
///
|
|
|
|
QJsonDocument handleReply(QNetworkReply* const &reply );
|
|
|
|
|
|
|
|
/// QNetworkAccessManager for sending requests.
|
|
|
|
QNetworkAccessManager* _networkmanager;
|
|
|
|
|
|
|
|
//Philips Hue Bridge details
|
|
|
|
QString _deviceModel;
|
|
|
|
QString _deviceFirmwareVersion;
|
|
|
|
QString _deviceAPIVersion;
|
|
|
|
|
|
|
|
uint _api_major;
|
|
|
|
uint _api_minor;
|
|
|
|
uint _api_patch;
|
|
|
|
|
|
|
|
bool _isHueEntertainmentReady;
|
|
|
|
|
|
|
|
QMap<quint16,QJsonObject> _lightsMap;
|
2016-05-26 23:44:27 +02:00
|
|
|
};
|
|
|
|
|
2020-03-26 18:49:44 +01:00
|
|
|
|
|
|
|
|
2016-05-26 23:44:27 +02:00
|
|
|
/**
|
|
|
|
* Implementation for the Philips Hue system.
|
|
|
|
*
|
|
|
|
* To use set the device to "philipshue".
|
|
|
|
* Uses the official Philips Hue API (http://developers.meethue.com).
|
|
|
|
*
|
|
|
|
* @author ntim (github), bimsarck (github)
|
|
|
|
*/
|
2020-03-26 18:49:44 +01:00
|
|
|
class LedDevicePhilipsHue: public LedDevicePhilipsHueBridge
|
2017-03-31 10:17:14 +02:00
|
|
|
{
|
2016-08-14 10:46:44 +02:00
|
|
|
|
|
|
|
Q_OBJECT
|
|
|
|
|
2016-05-26 23:44:27 +02:00
|
|
|
public:
|
|
|
|
///
|
2016-08-23 20:07:12 +02:00
|
|
|
/// Constructs specific LedDevice
|
2016-05-26 23:44:27 +02:00
|
|
|
///
|
2016-08-23 20:07:12 +02:00
|
|
|
/// @param deviceConfig json device config
|
2016-05-26 23:44:27 +02:00
|
|
|
///
|
2020-02-10 15:21:58 +01:00
|
|
|
explicit LedDevicePhilipsHue(const QJsonObject &deviceConfig);
|
2016-05-26 23:44:27 +02:00
|
|
|
|
|
|
|
///
|
|
|
|
/// Destructor of this device
|
|
|
|
///
|
|
|
|
virtual ~LedDevicePhilipsHue();
|
|
|
|
|
2016-08-23 20:07:12 +02:00
|
|
|
/// constructs leddevice
|
2016-10-13 21:59:58 +02:00
|
|
|
static LedDevice* construct(const QJsonObject &deviceConfig);
|
2017-03-31 10:17:14 +02:00
|
|
|
|
2020-03-26 18:49:44 +01:00
|
|
|
///
|
|
|
|
/// Sets configuration
|
|
|
|
///
|
|
|
|
/// @param deviceConfig the json device config
|
|
|
|
/// @return true if success
|
|
|
|
virtual bool init(const QJsonObject &deviceConfig) override;
|
|
|
|
|
|
|
|
/// Switch the device on
|
|
|
|
virtual int switchOn() override;
|
|
|
|
|
|
|
|
/// Switch the device off
|
|
|
|
virtual int switchOff() override;
|
2019-01-01 19:47:07 +01:00
|
|
|
|
2017-09-16 00:18:17 +02:00
|
|
|
/// creates new PhilipsHueLight(s) based on user lightid with bridge feedback
|
|
|
|
///
|
|
|
|
/// @param map Map of lightid/value pairs of bridge
|
|
|
|
///
|
|
|
|
void newLights(QMap<quint16, QJsonObject> map);
|
|
|
|
|
2020-03-26 18:49:44 +01:00
|
|
|
unsigned int getLightsCount() const { return _lightsCount; }
|
|
|
|
void setLightsCount( unsigned int lightsCount);
|
|
|
|
|
|
|
|
void setOnOffState(PhilipsHueLight& light, bool on);
|
|
|
|
void setTransitionTime(PhilipsHueLight& light, unsigned int transitionTime);
|
|
|
|
void setColor(PhilipsHueLight& light, const CiColor& color, double brightnessFactor);
|
|
|
|
void setState(PhilipsHueLight& light, bool on, const CiColor& color, double brightnessFactor, unsigned int transitionTime);
|
|
|
|
|
|
|
|
void restoreOriginalState();
|
|
|
|
|
|
|
|
public slots:
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Closes the output device.
|
|
|
|
/// Includes switching-off the device and stopping refreshes
|
|
|
|
///
|
|
|
|
virtual void close() override;
|
|
|
|
|
|
|
|
private slots:
|
|
|
|
/// creates new PhilipsHueLight(s) based on user lightid with bridge feedback
|
|
|
|
///
|
|
|
|
/// @param map Map of lightid/value pairs of bridge
|
|
|
|
///
|
|
|
|
void updateLights(QMap<quint16, QJsonObject> map);
|
2016-09-23 08:49:22 +02:00
|
|
|
|
2017-03-31 10:17:14 +02:00
|
|
|
protected:
|
2020-03-26 18:49:44 +01:00
|
|
|
|
|
|
|
///
|
|
|
|
/// Opens and initiatialises the output device
|
|
|
|
///
|
|
|
|
/// @return Zero on succes (i.e. device is ready and enabled) else negative
|
|
|
|
///
|
|
|
|
virtual int open() override;
|
|
|
|
|
|
|
|
///
|
|
|
|
/// Get Philips Hue device details and configuration
|
|
|
|
///
|
|
|
|
/// @return True, if Nanoleaf device capabilities fit configuration
|
|
|
|
///
|
|
|
|
bool initLeds();
|
|
|
|
|
2016-05-26 23:44:27 +02:00
|
|
|
///
|
2017-03-31 10:17:14 +02:00
|
|
|
/// Writes the RGB-Color values to the leds.
|
2016-05-26 23:44:27 +02:00
|
|
|
///
|
2017-03-31 10:17:14 +02:00
|
|
|
/// @param[in] ledValues The RGB-color per led
|
2016-05-26 23:44:27 +02:00
|
|
|
///
|
|
|
|
/// @return Zero on success else negative
|
|
|
|
///
|
2020-02-10 15:21:58 +01:00
|
|
|
virtual int write(const std::vector<ColorRgb> & ledValues) override;
|
2016-05-26 23:44:27 +02:00
|
|
|
|
2017-03-31 10:17:14 +02:00
|
|
|
private:
|
2020-03-26 18:49:44 +01:00
|
|
|
|
|
|
|
int writeSingleLights(const std::vector<ColorRgb>& ledValues);
|
2017-09-16 00:18:17 +02:00
|
|
|
|
2016-05-26 23:44:27 +02:00
|
|
|
///
|
2020-03-26 18:49:44 +01:00
|
|
|
bool _switchOffOnBlack;
|
2017-03-31 10:17:14 +02:00
|
|
|
/// The brightness factor to multiply on color change.
|
2020-03-26 18:49:44 +01:00
|
|
|
double _brightnessFactor;
|
|
|
|
/// Transition time in multiples of 100 ms.
|
2017-03-31 10:17:14 +02:00
|
|
|
/// The default of the Hue lights is 400 ms, but we may want it snapier.
|
2020-03-26 18:49:44 +01:00
|
|
|
unsigned int _transitionTime;
|
|
|
|
|
|
|
|
bool _isRestoreOrigState;
|
|
|
|
|
2016-05-26 23:44:27 +02:00
|
|
|
/// Array of the light ids.
|
2020-03-26 18:49:44 +01:00
|
|
|
std::vector<unsigned int> _lightIds;
|
2017-03-31 10:17:14 +02:00
|
|
|
/// Array to save the lamps.
|
2020-03-26 18:49:44 +01:00
|
|
|
std::vector<PhilipsHueLight> _lights;
|
|
|
|
|
|
|
|
unsigned int _lightsCount;
|
|
|
|
|
|
|
|
|
2016-05-26 23:44:27 +02:00
|
|
|
};
|