From aaa4235cab23cfb102fca4db030a753742369b62 Mon Sep 17 00:00:00 2001
From: LordGrey <48840279+Lord-Grey@users.noreply.github.com>
Date: Thu, 26 Mar 2020 18:49:44 +0100
Subject: [PATCH] refactor: Align Phillips Hue to reworked device handling
(#712)
* Align PhilipsHue (Classic)
* Minor Device corrections
* Have code working with Qt < 5.10
* Fixes on Hue Wizzard
* Fixes on Hue Wizzard
* Calculate Latchtime only for lights updated by hyperion
* Allow to disable restoring original light's state
* Fix - LightIDs / LightMap vectors were not cleared when reopening the device
* Reduce API Calls for state updates by consolidation
---
assets/webconfig/i18n/de.json | 1 +
assets/webconfig/i18n/en.json | 1 +
assets/webconfig/js/wizard.js | 40 +-
include/leddevice/LedDevice.h | 2 +
libsrc/leddevice/LedDevice.cpp | 6 +
.../leddevice/dev_net/LedDeviceNanoleaf.cpp | 25 +-
libsrc/leddevice/dev_net/LedDeviceNanoleaf.h | 2 +-
.../leddevice/dev_net/LedDevicePhilipsHue.cpp | 1108 ++++++++++++-----
.../leddevice/dev_net/LedDevicePhilipsHue.h | 322 +++--
libsrc/leddevice/dev_spi/ProviderSpi.cpp | 2 +-
.../leddevice/schemas/schema-philipshue.json | 9 +-
11 files changed, 1096 insertions(+), 422 deletions(-)
diff --git a/assets/webconfig/i18n/de.json b/assets/webconfig/i18n/de.json
index 479123a8..644049ff 100644
--- a/assets/webconfig/i18n/de.json
+++ b/assets/webconfig/i18n/de.json
@@ -426,6 +426,7 @@
"edt_dev_spec_transistionTime_title": "Übergangszeit",
"edt_dev_spec_switchOffOnBlack_title": "Aus bei schwarz",
"edt_dev_spec_brightnessFactor_title": "Helligkeitsfaktor",
+ "edt_dev_spec_restoreOriginalState_title" : "Lampen Originalzustand wiederhestellen",
"edt_dev_spec_ledType_title": "LED typ",
"edt_dev_spec_uid_title": "UID",
"edt_dev_spec_intervall_title": "Intervall",
diff --git a/assets/webconfig/i18n/en.json b/assets/webconfig/i18n/en.json
index 603d8710..07d9aaea 100644
--- a/assets/webconfig/i18n/en.json
+++ b/assets/webconfig/i18n/en.json
@@ -425,6 +425,7 @@
"edt_dev_spec_transistionTime_title" : "Transition time",
"edt_dev_spec_switchOffOnBlack_title" : "Switch off on black",
"edt_dev_spec_brightnessFactor_title" : "Brightness factor",
+ "edt_dev_spec_restoreOriginalState_title" : "Restore lights' original state",
"edt_dev_spec_ledType_title" : "LED Type",
"edt_dev_spec_uid_title" : "UID",
"edt_dev_spec_intervall_title" : "Interval",
diff --git a/assets/webconfig/js/wizard.js b/assets/webconfig/js/wizard.js
index fb7ab6d2..de33979e 100644
--- a/assets/webconfig/js/wizard.js
+++ b/assets/webconfig/js/wizard.js
@@ -567,23 +567,35 @@ function checkHueBridge(cb,hueUser){
timeout: 2000
})
.done( function( data, textStatus, jqXHR ) {
- if(Array.isArray(data) && data[0].error && data[0].error.type == 4)
- cb(true);
- else if(Array.isArray(data) && data[0].error)
- cb(false);
+ if( Array.isArray(data) && data[0].error)
+ {
+ if ( data[0].error.type == 3 || data[0].error.type == 4)
+ {
+ cb(true, usr);
+ }
+ else
+ {
+ cb(false);
+ }
+ }
else
- cb(true);
+ {
+ cb(true, usr);
+ }
+
})
.fail( function( jqXHR, textStatus ) {
cb(false);
});
}
-function checkUserResult(reply){
+function checkUserResult(reply, usr){
+
if(reply)
{
$('#wiz_hue_usrstate').html("");
$('#wiz_hue_create_user').toggle(false);
+ $('#user').val(usr);
get_hue_lights();
}
else
@@ -640,17 +652,22 @@ function checkBridgeResult(reply){
function identHueId(id, off)
{
- var on = true;
if(off !== true)
+ {
setTimeout(identHueId,1500,id,true);
+ var put_data = '{"on":true,"bri":254,"hue":47000,"sat":254}';
+ }
else
- on = false;
+ {
+ var put_data = '{"on":false}';
+ }
$.ajax({
url: 'http://'+$('#ip').val()+'/api/'+$('#user').val()+'/lights/'+id+'/state',
type: 'PUT',
timeout: 2000,
- data: ' {"on":'+on+', "sat":254, "bri":254,"hue":47000}'
+
+ data: put_data
})
}
@@ -686,7 +703,9 @@ function beginWizardHue()
//check if ip is empty/reachable/search for bridge
if(conf_editor.getEditor("root.specificOptions.output").getValue() == "")
+ {
getHueIPs();
+ }
else
{
var ip = conf_editor.getEditor("root.specificOptions.output").getValue();
@@ -719,6 +738,7 @@ function beginWizardHue()
}
}
+ var ledCount= Object.keys(lightIDs).length;
window.serverConfig.leds = hueLedConfig;
@@ -736,6 +756,7 @@ function beginWizardHue()
d.lightIds = finalLightIds;
d.username = $('#user').val();
d.type = "philipshue";
+ d.hardwareLedCount = ledCount;
d.transitiontime = 1;
d.switchOffOnBlack = true;
@@ -814,6 +835,7 @@ function get_hue_lights(){
for(var lightid in r)
{
+
$('.lidsb').append(createTableRow([lightid+' ('+r[lightid].name+')', '','']));
}
diff --git a/include/leddevice/LedDevice.h b/include/leddevice/LedDevice.h
index a587d5b0..29e5e83a 100644
--- a/include/leddevice/LedDevice.h
+++ b/include/leddevice/LedDevice.h
@@ -59,7 +59,9 @@ public:
unsigned int getLedCount() const { return _ledCount; }
bool enabled() const { return _enabled; }
+
int getLatchTime() const { return _latchTime_ms; }
+ void setLatchTime( int latchTime_ms );
///
/// Check, if device is ready to be used
diff --git a/libsrc/leddevice/LedDevice.cpp b/libsrc/leddevice/LedDevice.cpp
index 0f1e7410..30191903 100644
--- a/libsrc/leddevice/LedDevice.cpp
+++ b/libsrc/leddevice/LedDevice.cpp
@@ -231,6 +231,12 @@ void LedDevice::setLedCount(unsigned int ledCount)
_ledRGBWCount = _ledCount * sizeof(ColorRgbw);
}
+void LedDevice::setLatchTime( int latchTime_ms )
+{
+ _latchTime_ms = latchTime_ms;
+ Debug(_log, "LatchTime updated to %dms", this->getLatchTime());
+}
+
int LedDevice::rewriteLeds()
{
int retval = -1;
diff --git a/libsrc/leddevice/dev_net/LedDeviceNanoleaf.cpp b/libsrc/leddevice/dev_net/LedDeviceNanoleaf.cpp
index 1aad493c..83f162f8 100644
--- a/libsrc/leddevice/dev_net/LedDeviceNanoleaf.cpp
+++ b/libsrc/leddevice/dev_net/LedDeviceNanoleaf.cpp
@@ -37,13 +37,13 @@ static const char STATE_ONOFF_VALUE[] = "value";
static const char STATE_VALUE_TRUE[] = "true";
static const char STATE_VALUE_FALSE[] = "false";
-//Device Data elements
+// Device Data elements
static const char DEV_DATA_NAME[] = "name";
static const char DEV_DATA_MODEL[] = "model";
static const char DEV_DATA_MANUFACTURER[] = "manufacturer";
static const char DEV_DATA_FIRMWAREVERSION[] = "firmwareVersion";
-//Nanoleaf Stream Control elements
+// Nanoleaf Stream Control elements
//static const char STREAM_CONTROL_IP[] = "streamControlIpAddr";
static const char STREAM_CONTROL_PORT[] = "streamControlPort";
//static const char STREAM_CONTROL_PROTOCOL[] = "streamControlProtocol";
@@ -59,7 +59,7 @@ static const char API_STATE[] ="state";
static const char API_PANELLAYOUT[] = "panelLayout";
static const char API_EFFECT[] = "effects";
-//Nanoleaf ssdp services
+// Nanoleaf ssdp services
static const char SSDP_CANVAS[] = "nanoleaf:nl29";
static const char SSDP_LIGHTPANELS[] = "nanoleaf_aurora:light";
const int SSDP_TIMEOUT = 5000; // timout in ms
@@ -132,7 +132,7 @@ bool LedDeviceNanoleaf::init(const QJsonObject &deviceConfig)
if ( _hostname.isEmpty() )
{
//Discover Nanoleaf device
- if ( !discoverNanoleafDevice() )
+ if ( !discoverDevice() )
{
this->setInError("No target IP defined nor Nanoleaf device was discovered");
return false;
@@ -256,24 +256,17 @@ int LedDeviceNanoleaf::open()
if ( init(_devConfig) )
{
- if ( !initNetwork() )
+ if ( initLeds() )
{
- this->setInError( "UDP Network error!" );
- }
- else
- {
- if ( initLeds() )
- {
- _deviceReady = true;
- setEnable(true);
- retval = 0;
- }
+ _deviceReady = true;
+ setEnable(true);
+ retval = 0;
}
}
return retval;
}
-bool LedDeviceNanoleaf::discoverNanoleafDevice()
+bool LedDeviceNanoleaf::discoverDevice()
{
bool isDeviceFound (false);
diff --git a/libsrc/leddevice/dev_net/LedDeviceNanoleaf.h b/libsrc/leddevice/dev_net/LedDeviceNanoleaf.h
index 4fc28734..5ef704f5 100644
--- a/libsrc/leddevice/dev_net/LedDeviceNanoleaf.h
+++ b/libsrc/leddevice/dev_net/LedDeviceNanoleaf.h
@@ -103,7 +103,7 @@ private:
///
/// @return True, if Nanoleaf device was found
///
- bool discoverNanoleafDevice();
+ bool discoverDevice();
///
/// Change Nanoleaf device to External Control (UDP) mode
diff --git a/libsrc/leddevice/dev_net/LedDevicePhilipsHue.cpp b/libsrc/leddevice/dev_net/LedDevicePhilipsHue.cpp
index 83b672cf..8a5d3064 100644
--- a/libsrc/leddevice/dev_net/LedDevicePhilipsHue.cpp
+++ b/libsrc/leddevice/dev_net/LedDevicePhilipsHue.cpp
@@ -1,91 +1,171 @@
// Local-Hyperion includes
#include "LedDevicePhilipsHue.h"
-// qt includes
+// ssdp discover
+#include
+
+// Qt includes
#include
+#include
#include
-bool operator ==(CiColor p1, CiColor p2)
+//
+static const bool verbose = false;
+static const bool verbose3 = false;
+
+// Configuration settings
+static const char CONFIG_ADDRESS[] = "output";
+//static const char CONFIG_PORT[] = "port";
+static const char CONFIG_USERNAME[] ="username";
+static const char CONFIG_BRIGHTNESSFACTOR [] = "brightnessFactor";
+static const char CONFIG_TRANSITIONTIME [] = "transitiontime";
+static const char CONFIG_ON_OFF_BLACK [] = "switchOffOnBlack";
+static const char CONFIG_RESTORE_STATE [] = "restoreOriginalState";
+static const char CONFIG_LIGHTIDS [] = "lightIds";
+
+// Device Data elements
+static const char DEV_DATA_BRIDGEID[] = "bridgeid";
+static const char DEV_DATA_MODEL[] = "modelid";
+static const char DEV_DATA_NAME[] = "name";
+//static const char DEV_DATA_MANUFACTURER[] = "manufacturer";
+static const char DEV_DATA_FIRMWAREVERSION[] = "swversion";
+static const char DEV_DATA_APIVERSION[] = "apiversion";
+
+// Philips Hue OpenAPI URLs
+static const char API_DEFAULT_PORT[] = "80";
+static const char API_URL_FORMAT[] = "http://%1:%2/api/%3/%4";
+static const char API_ROOT[] = "";
+static const char API_STATE[] ="state";
+static const char API_CONFIG[] = "config";
+static const char API_LIGHTS[] = "lights";
+
+// List of resources
+static const char API_XY_COORDINATES[] = "xy";
+static const char API_BRIGHTNESS[] = "bri";
+static const char API_TRANSITIONTIME[] = "transitiontime";
+static const char API_MODEID[] = "modelid";
+
+// List of State Information
+static const char API_STATE_ON[] = "on";
+static const char API_STATE_VALUE_TRUE[] = "true";
+static const char API_STATE_VALUE_FALSE[] = "false";
+
+// List of Error Information
+static const char API_ERROR[] = "error";
+static const char API_ERROR_ADDRESS[] = "address";
+static const char API_ERROR_DESCRIPTION[] = "description";
+static const char API_ERROR_TYPE[]="type";
+
+// Phlips Hue ssdp services
+static const char SSDP_ID[] = "urn:schemas-upnp-org:device:Basic:1";
+const int SSDP_TIMEOUT = 5000; // timout in ms
+
+
+bool operator ==(const CiColor& p1, const CiColor& p2)
{
- return (p1.x == p2.x) && (p1.y == p2.y) && (p1.bri == p2.bri);
+ return ((p1.x == p2.x) && (p1.y == p2.y) && (p1.bri == p2.bri));
}
-bool operator !=(CiColor p1, CiColor p2)
+bool operator != (const CiColor& p1, const CiColor& p2)
{
return !(p1 == p2);
}
-CiColor CiColor::rgbToCiColor(float red, float green, float blue, CiColorTriangle colorSpace)
+CiColor CiColor::rgbToCiColor(double red, double green, double blue, CiColorTriangle colorSpace)
{
- // Apply gamma correction.
- float r = (red > 0.04045f) ? powf((red + 0.055f) / (1.0f + 0.055f), 2.4f) : (red / 12.92f);
- float g = (green > 0.04045f) ? powf((green + 0.055f) / (1.0f + 0.055f), 2.4f) : (green / 12.92f);
- float b = (blue > 0.04045f) ? powf((blue + 0.055f) / (1.0f + 0.055f), 2.4f) : (blue / 12.92f);
- // Convert to XYZ space.
- float X = r * 0.664511f + g * 0.154324f + b * 0.162028f;
- float Y = r * 0.283881f + g * 0.668433f + b * 0.047685f;
- float Z = r * 0.000088f + g * 0.072310f + b * 0.986039f;
- // Convert to x,y space.
- float cx = X / (X + Y + Z);
- float cy = Y / (X + Y + Z);
+ double cx;
+ double cy;
+ double bri;
+
+ if(red + green + blue > 0)
+ {
+
+ // Apply gamma correction.
+ double r = (red > 0.04045) ? pow((red + 0.055) / (1.0 + 0.055), 2.4) : (red / 12.92);
+ double g = (green > 0.04045) ? pow((green + 0.055) / (1.0 + 0.055), 2.4) : (green / 12.92);
+ double b = (blue > 0.04045) ? pow((blue + 0.055) / (1.0 + 0.055), 2.4) : (blue / 12.92);
+
+ // Convert to XYZ space.
+ double X = r * 0.664511 + g * 0.154324 + b * 0.162028;
+ double Y = r * 0.283881 + g * 0.668433 + b * 0.047685;
+ double Z = r * 0.000088 + g * 0.072310 + b * 0.986039;
+
+ cx = X / (X + Y + Z);
+ cy = Y / (X + Y + Z);
+
+ // RGB to HSV/B Conversion before gamma correction V/B for brightness, not Y from XYZ Space.
+ // bri = std::max(std::max(red, green), blue);
+ // RGB to HSV/B Conversion after gamma correction V/B for brightness, not Y from XYZ Space.
+ bri = std::max(r, std::max(g, b));
+ }
+ else
+ {
+ cx = 0.0;
+ cy = 0.0;
+ bri = 0.0;
+ }
+
if (std::isnan(cx))
{
- cx = 0.0f;
+ cx = 0.0;
}
if (std::isnan(cy))
{
- cy = 0.0f;
+ cy = 0.0;
}
- // RGB to HSV/B Conversion after gamma correction use V for brightness, not Y from XYZ Space.
- float bri = fmax(fmax(r, g), b);
- CiColor xy =
- { cx, cy, bri };
- // Check if the given XY value is within the color reach of our lamps.
- if (!isPointInLampsReach(xy, colorSpace))
+ if (std::isnan(bri))
{
- // It seems the color is out of reach let's find the closes color we can produce with our lamp and send this XY value out.
- CiColor pAB = getClosestPointToPoint(colorSpace.red, colorSpace.green, xy);
- CiColor pAC = getClosestPointToPoint(colorSpace.blue, colorSpace.red, xy);
- CiColor pBC = getClosestPointToPoint(colorSpace.green, colorSpace.blue, xy);
- // Get the distances per point and see which point is closer to our Point.
- float dAB = getDistanceBetweenTwoPoints(xy, pAB);
- float dAC = getDistanceBetweenTwoPoints(xy, pAC);
- float dBC = getDistanceBetweenTwoPoints(xy, pBC);
- float lowest = dAB;
- CiColor closestPoint = pAB;
- if (dAC < lowest)
+ bri = 0.0;
+ }
+
+ CiColor xy = { cx, cy, bri };
+
+ if(red + green + blue > 0)
+ {
+ // Check if the given XY value is within the color reach of our lamps.
+ if (!isPointInLampsReach(xy, colorSpace))
{
- lowest = dAC;
- closestPoint = pAC;
+ // It seems the color is out of reach let's find the closes color we can produce with our lamp and send this XY value out.
+ CiColor pAB = getClosestPointToPoint(colorSpace.red, colorSpace.green, xy);
+ CiColor pAC = getClosestPointToPoint(colorSpace.blue, colorSpace.red, xy);
+ CiColor pBC = getClosestPointToPoint(colorSpace.green, colorSpace.blue, xy);
+ // Get the distances per point and see which point is closer to our Point.
+ double dAB = getDistanceBetweenTwoPoints(xy, pAB);
+ double dAC = getDistanceBetweenTwoPoints(xy, pAC);
+ double dBC = getDistanceBetweenTwoPoints(xy, pBC);
+ double lowest = dAB;
+ CiColor closestPoint = pAB;
+ if (dAC < lowest)
+ {
+ lowest = dAC;
+ closestPoint = pAC;
+ }
+ if (dBC < lowest)
+ {
+ //lowest = dBC;
+ closestPoint = pBC;
+ }
+ // Change the xy value to a value which is within the reach of the lamp.
+ xy.x = closestPoint.x;
+ xy.y = closestPoint.y;
}
- if (dBC < lowest)
- {
- lowest = dBC;
- closestPoint = pBC;
- }
- // Change the xy value to a value which is within the reach of the lamp.
- xy.x = closestPoint.x;
- xy.y = closestPoint.y;
}
return xy;
}
-float CiColor::crossProduct(CiColor p1, CiColor p2)
+double CiColor::crossProduct(CiColor p1, CiColor p2)
{
return p1.x * p2.y - p1.y * p2.x;
}
bool CiColor::isPointInLampsReach(CiColor p, CiColorTriangle colorSpace)
{
- CiColor v1 =
- { colorSpace.green.x - colorSpace.red.x, colorSpace.green.y - colorSpace.red.y };
- CiColor v2 =
- { colorSpace.blue.x - colorSpace.red.x, colorSpace.blue.y - colorSpace.red.y };
- CiColor q =
- { p.x - colorSpace.red.x, p.y - colorSpace.red.y };
- float s = crossProduct(q, v2) / crossProduct(v1, v2);
- float t = crossProduct(v1, q) / crossProduct(v1, v2);
- if ((s >= 0.0f) && (t >= 0.0f) && (s + t <= 1.0f))
+ CiColor v1 = { colorSpace.green.x - colorSpace.red.x, colorSpace.green.y - colorSpace.red.y };
+ CiColor v2 = { colorSpace.blue.x - colorSpace.red.x, colorSpace.blue.y - colorSpace.red.y };
+ CiColor q = { p.x - colorSpace.red.x, p.y - colorSpace.red.y };
+ double s = crossProduct(q, v2) / crossProduct(v1, v2);
+ double t = crossProduct(v1, q) / crossProduct(v1, v2);
+ if ((s >= 0.0) && (t >= 0.0) && (s + t <= 1.0))
{
return true;
}
@@ -94,250 +174,522 @@ bool CiColor::isPointInLampsReach(CiColor p, CiColorTriangle colorSpace)
CiColor CiColor::getClosestPointToPoint(CiColor a, CiColor b, CiColor p)
{
- CiColor AP =
- { p.x - a.x, p.y - a.y };
- CiColor AB =
- { b.x - a.x, b.y - a.y };
- float ab2 = AB.x * AB.x + AB.y * AB.y;
- float ap_ab = AP.x * AB.x + AP.y * AB.y;
- float t = ap_ab / ab2;
- if (t < 0.0f)
+ CiColor AP = { p.x - a.x, p.y - a.y };
+ CiColor AB = { b.x - a.x, b.y - a.y };
+ double ab2 = AB.x * AB.x + AB.y * AB.y;
+ double ap_ab = AP.x * AB.x + AP.y * AB.y;
+ double t = ap_ab / ab2;
+ if (t < 0.0)
{
- t = 0.0f;
+ t = 0.0;
}
- else if (t > 1.0f)
+ else if (t > 1.0)
{
- t = 1.0f;
+ t = 1.0;
}
- return
- { a.x + AB.x * t, a.y + AB.y * t};
+ return { a.x + AB.x * t, a.y + AB.y * t };
}
-float CiColor::getDistanceBetweenTwoPoints(CiColor p1, CiColor p2)
+double CiColor::getDistanceBetweenTwoPoints(CiColor p1, CiColor p2)
{
// Horizontal difference.
- float dx = p1.x - p2.x;
+ double dx = p1.x - p2.x;
// Vertical difference.
- float dy = p1.y - p2.y;
+ double dy = p1.y - p2.y;
// Absolute value.
return sqrt(dx * dx + dy * dy);
}
-PhilipsHueBridge::PhilipsHueBridge(Logger* log, QString host, QString username)
- : QObject()
- , _log(log)
- , host(host)
- , username(username)
+LedDevicePhilipsHueBridge::LedDevicePhilipsHueBridge(const QJsonObject &deviceConfig)
+ : LedDevice(deviceConfig)
+ , _networkmanager (nullptr)
+ , _api_major(0)
+ , _api_minor(0)
+ , _api_patch(0)
+ , _isHueEntertainmentReady (false)
{
- // setup reconnection timer
- bTimer.setInterval(5000);
- bTimer.setSingleShot(true);
-
- connect(&bTimer, &QTimer::timeout, this, &PhilipsHueBridge::bConnect);
- connect(&manager, &QNetworkAccessManager::finished, this, &PhilipsHueBridge::resolveReply);
+ _devConfig = deviceConfig;
+ _deviceReady = false;
}
-void PhilipsHueBridge::bConnect(void)
+LedDevicePhilipsHueBridge::~LedDevicePhilipsHueBridge()
{
- if(username.isEmpty() || host.isEmpty())
+ if (_networkmanager != nullptr)
{
- Error(_log,"Username or IP Address is empty!");
+ delete _networkmanager;
+ _networkmanager = nullptr;
+ }
+}
+
+bool LedDevicePhilipsHueBridge::init(const QJsonObject &deviceConfig)
+{
+ // Overwrite non supported/required features
+ _devConfig["latchTime"] = 0;
+ if (deviceConfig["rewriteTime"].toInt(0) > 0)
+ {
+ Info (_log, "Device Philips Hue does not require rewrites. Refresh time is ignored.");
+ _devConfig["rewriteTime"] = 0;
+ }
+
+ DebugIf(verbose, _log, "deviceConfig: [%s]", QString(QJsonDocument(_devConfig).toJson(QJsonDocument::Compact)).toUtf8().constData() );
+
+ bool isInitOK = LedDevice::init(deviceConfig);
+
+ Debug(_log, "DeviceType : %s", QSTRING_CSTR( this->getActiveDeviceType() ));
+ Debug(_log, "LedCount : %u", this->getLedCount());
+ Debug(_log, "ColorOrder : %s", QSTRING_CSTR( this->getColorOrder() ));
+ Debug(_log, "RefreshTime : %d", _refresh_timer_interval);
+ Debug(_log, "LatchTime : %d", this->getLatchTime());
+
+ if ( isInitOK )
+ {
+ //Set hostname as per configuration and_defaultHost default port
+ QString address = deviceConfig[ CONFIG_ADDRESS ].toString();
+
+ if (! address.isEmpty() )
+ {
+ QStringList addressparts = address.split(":", QString::SkipEmptyParts);
+
+ _hostname = addressparts[0];
+ if ( addressparts.size() > 1)
+ {
+ _api_port = addressparts[1];
+ }
+ else
+ {
+ _api_port = API_DEFAULT_PORT;
+ }
+ }
+ _username = deviceConfig[ CONFIG_USERNAME ].toString();
+
+ Debug(_log, "Hostname/IP : %s", QSTRING_CSTR( _hostname ));
+ Debug(_log, "Port : %s", QSTRING_CSTR( _api_port ));
+ }
+ return isInitOK;
+}
+
+int LedDevicePhilipsHueBridge::open( )
+{
+ return open (_hostname,_api_port, _username );
+}
+
+int LedDevicePhilipsHueBridge::open( const QString& hostname, const QString& port, const QString& username )
+{
+ _deviceInError = false;
+ bool isInitOK = true;
+
+ //If host not configured then discover device
+ if ( hostname.isEmpty() )
+ {
+ //Discover Nanoleaf device
+ if ( !discoverDevice() )
+ {
+ this->setInError("No target IP defined nor Philips Hue Bridge was discovered");
+ return false;
+ }
}
else
{
- QString url = QString("http://%1/api/%2").arg(host).arg(username);
- Debug(_log, "Connect to bridge %s", QSTRING_CSTR(url));
-
- QNetworkRequest request(url);
- manager.get(request);
+ _hostname = hostname;
+ _api_port = port;
}
-}
-void PhilipsHueBridge::resolveReply(QNetworkReply* reply)
-{
- // TODO use put request also for network error checking with decent threshold
- if(reply->operation() == QNetworkAccessManager::GetOperation)
+ _username = username;
+
+ //Get Philips Hue Bridge details and configuration
+ if ( _networkmanager == nullptr)
{
- if(reply->error() == QNetworkReply::NoError)
+ _networkmanager = new QNetworkAccessManager();
+ }
+
+ // Read Lights and Light-Ids
+ QString url = getUrl(_hostname, _api_port, _username, API_ROOT );
+ QJsonDocument doc = getJson( url );
+
+ DebugIf(verbose, _log, "doc: [%s]", QString(QJsonDocument(doc).toJson(QJsonDocument::Compact)).toUtf8().constData() );
+
+ if ( this->isInError() )
+ {
+ isInitOK = false;
+ }
+ else
+ {
+ QJsonObject jsonConfigInfo = doc.object()[API_CONFIG].toObject();
+ if ( verbose )
{
- QByteArray response = reply->readAll();
- QJsonParseError error;
- QJsonDocument doc = QJsonDocument::fromJson(response, &error);
- if (error.error != QJsonParseError::NoError)
- {
- Error(_log, "Got invalid response from bridge");
- return;
- }
- // check for authorization
- if(doc.isArray())
- {
- Error(_log, "Authorization failed, username invalid");
- return;
- }
+ std::cout << "jsonConfigInfo: [" << QString(QJsonDocument(jsonConfigInfo).toJson(QJsonDocument::Compact)).toUtf8().constData() << "]" << std::endl;
+ }
- QJsonObject obj = doc.object()["lights"].toObject();
+ QString deviceName = jsonConfigInfo[DEV_DATA_NAME].toString();
+ _deviceModel = jsonConfigInfo[DEV_DATA_MODEL].toString();
+ QString deviceBridgeID = jsonConfigInfo[DEV_DATA_BRIDGEID].toString();
+ _deviceFirmwareVersion = jsonConfigInfo[DEV_DATA_FIRMWAREVERSION].toString();
+ _deviceAPIVersion = jsonConfigInfo[DEV_DATA_APIVERSION].toString();
- if(obj.isEmpty())
- {
- Error(_log, "Bridge has no registered bulbs/stripes");
- return;
- }
+ QStringList apiVersionParts = _deviceAPIVersion.split(".", QString::SkipEmptyParts);
+ if ( !apiVersionParts.isEmpty())
+ {
+ _api_major = apiVersionParts[0].toUInt();
+ _api_minor = apiVersionParts[1].toUInt();
+ _api_patch = apiVersionParts[2].toUInt();
- // get all available light ids and their values
- QStringList keys = obj.keys();
- QMap map;
- for (int i = 0; i < keys.size(); ++i)
+ if ( _api_major > 1 || (_api_major == 1 && _api_minor >= 22) )
{
- map.insert(keys.at(i).toInt(), obj.take(keys.at(i)).toObject());
+ _isHueEntertainmentReady = true;
}
- emit newLights(map);
+ }
+
+ Debug(_log, "Bridge Name : %s", QSTRING_CSTR( deviceName ));
+ Debug(_log, "Model : %s", QSTRING_CSTR( _deviceModel ));
+ Debug(_log, "Bridge-ID : %s", QSTRING_CSTR( deviceBridgeID ));
+ Debug(_log, "SoftwareVersion : %s", QSTRING_CSTR( _deviceFirmwareVersion));
+ Debug(_log, "API-Version : %u.%u.%u", _api_major,_api_minor, _api_patch );
+ Debug(_log, "EntertainmentReady: %d", _isHueEntertainmentReady);
+
+ QJsonObject jsonLightsInfo = doc.object()[API_LIGHTS].toObject();
+
+ DebugIf(verbose, _log, "jsonLightsInfo: [%s]", QString(QJsonDocument(jsonLightsInfo).toJson(QJsonDocument::Compact)).toUtf8().constData() );
+
+ // Get all available light ids and their values
+ QStringList keys = jsonLightsInfo.keys();
+
+ _ledCount = keys.size();
+ _lightsMap.clear();
+
+ for (uint i = 0; i < _ledCount; ++i)
+ {
+ _lightsMap.insert(keys.at(i).toInt(), jsonLightsInfo.take(keys.at(i)).toObject());
+ }
+
+ if (getLedCount() == 0 )
+ {
+ setInError("No light-IDs found at the Philips Hue Bridge");
}
else
{
- Error(_log,"Network Error: %s", QSTRING_CSTR(reply->errorString()));
- bTimer.start();
+ Debug(_log, "Lights found : %u", getLedCount() );
}
}
- reply->deleteLater();
+ return isInitOK;
}
-void PhilipsHueBridge::post(QString route, QString content)
+bool LedDevicePhilipsHueBridge::discoverDevice()
{
- //Debug(_log, "Post %s: %s", QSTRING_CSTR(QString("http://IP/api/USR/%1").arg(route)), QSTRING_CSTR(content));
+ bool isDeviceFound (false);
- QNetworkRequest request(QString("http://%1/api/%2/%3").arg(host).arg(username).arg(route));
- manager.put(request, content.toLatin1());
-}
+ // device searching by ssdp
+ QString address;
+ SSDPDiscover discover;
-const std::set PhilipsHueLight::GAMUT_A_MODEL_IDS =
-{ "LLC001", "LLC005", "LLC006", "LLC007", "LLC010", "LLC011", "LLC012", "LLC013", "LLC014", "LST001" };
-const std::set PhilipsHueLight::GAMUT_B_MODEL_IDS =
-{ "LCT001", "LCT002", "LCT003", "LCT007", "LLM001" };
-const std::set PhilipsHueLight::GAMUT_C_MODEL_IDS =
-{ "LLC020", "LST002", "LCT011", "LCT012", "LCT010", "LCT014", "LCT015", "LCT016", "LCT024" };
-
-PhilipsHueLight::PhilipsHueLight(Logger* log, PhilipsHueBridge* bridge, unsigned int id, QJsonObject values)
- : _log(log)
- , bridge(bridge)
- , id(id)
-{
- // Get state object values which are subject to change.
- if (!values["state"].toObject().contains("on"))
+ // Discover Philips Hue Bridge
+ address = discover.getFirstService(STY_WEBSERVER, SSDP_ID, SSDP_TIMEOUT);
+ if ( address.isEmpty() )
{
- Error(_log, "Got invalid state object from light ID %d", id);
- }
- QJsonObject state;
- state["on"] = values["state"].toObject()["on"];
- on = false;
- if (values["state"].toObject()["on"].toBool())
- {
- state["xy"] = values["state"].toObject()["xy"];
- state["bri"] = values["state"].toObject()["bri"];
- on = true;
-
- color = {
- (float) state["xy"].toArray()[0].toDouble(),
- (float) state["xy"].toArray()[1].toDouble(),
- (float) state["bri"].toDouble() / 255.0f
- };
- transitionTime = values["state"].toObject()["transitiontime"].toInt();
- }
- // Determine the model id.
- modelId = values["modelid"].toString().trimmed().replace("\"", "");
- // Determine the original state.
- originalState = QJsonDocument(state).toJson(QJsonDocument::JsonFormat::Compact).trimmed();
- // Find id in the sets and set the appropriate color space.
- if (GAMUT_A_MODEL_IDS.find(modelId) != GAMUT_A_MODEL_IDS.end())
- {
- Debug(_log, "Recognized model id %s of light ID %d as gamut A", modelId.toStdString().c_str(), id);
- colorSpace.red =
- { 0.704f, 0.296f};
- colorSpace.green =
- { 0.2151f, 0.7106f};
- colorSpace.blue =
- { 0.138f, 0.08f};
- }
- else if (GAMUT_B_MODEL_IDS.find(modelId) != GAMUT_B_MODEL_IDS.end())
- {
- Debug(_log, "Recognized model id %s of light ID %d as gamut B", modelId.toStdString().c_str(), id);
- colorSpace.red =
- { 0.675f, 0.322f};
- colorSpace.green =
- { 0.409f, 0.518f};
- colorSpace.blue =
- { 0.167f, 0.04f};
- }
- else if (GAMUT_C_MODEL_IDS.find(modelId) != GAMUT_C_MODEL_IDS.end())
- {
- Debug(_log, "Recognized model id %s of light ID %d as gamut C", modelId.toStdString().c_str(), id);
- colorSpace.red =
- { 0.6915f, 0.3083f};
- colorSpace.green =
- { 0.17f, 0.7f};
- colorSpace.blue =
- { 0.1532f, 0.0475f};
+ Warning(_log, "No Philips Hue Bridge discovered");
}
else
{
- Warning(_log, "Did not recognize model id %s of light ID %d", modelId.toStdString().c_str(), id);
- colorSpace.red =
- { 1.0f, 0.0f};
- colorSpace.green =
- { 0.0f, 1.0f};
- colorSpace.blue =
- { 0.0f, 0.0f};
+ // Philips Hue Bridge found
+ Info(_log, "Philips Hue Bridge discovered at [%s]", QSTRING_CSTR( address ));
+ isDeviceFound = true;
+ QStringList addressparts = address.split(":", QString::SkipEmptyParts);
+ _hostname = addressparts[0];
+ _api_port = addressparts[1];
+ }
+ return isDeviceFound;
+}
+
+const QMap& LedDevicePhilipsHueBridge::getLightMap(void)
+{
+ return _lightsMap;
+}
+
+QString LedDevicePhilipsHueBridge::getUrl(QString host, QString port, QString auth_token, QString endpoint) const {
+ return QString(API_URL_FORMAT).arg(host, port, auth_token, endpoint);
+}
+
+QJsonDocument LedDevicePhilipsHueBridge::getJson(QString url)
+{
+
+ DebugIf(verbose, _log, "GET: [%s]", QSTRING_CSTR( url ));
+
+ // Perfrom request
+ QNetworkRequest request(url);
+ QNetworkReply* reply = _networkmanager->get(request);
+ // Connect requestFinished signal to quit slot of the loop.
+ QEventLoop loop;
+ loop.connect(reply, SIGNAL(finished()), SLOT(quit()));
+ // Go into the loop until the request is finished.
+ loop.exec();
+
+ QJsonDocument jsonDoc;
+ if(reply->operation() == QNetworkAccessManager::GetOperation)
+ {
+ jsonDoc = handleReply( reply );
+ }
+ // Free space.
+ reply->deleteLater();
+ // Return response
+ return jsonDoc;
+}
+
+QJsonDocument LedDevicePhilipsHueBridge::putJson(QString url, QString json)
+{
+
+ DebugIf(verbose, _log, "PUT: [%s] [%s]", QSTRING_CSTR( url ), QSTRING_CSTR( json ) );
+ // Perfrom request
+ QNetworkRequest request(url);
+ QNetworkReply* reply = _networkmanager->put(request, json.toUtf8());
+ // Connect requestFinished signal to quit slot of the loop.
+ QEventLoop loop;
+ loop.connect(reply, SIGNAL(finished()), SLOT(quit()));
+ // Go into the loop until the request is finished.
+ loop.exec();
+
+ QJsonDocument jsonDoc;
+ if(reply->operation() == QNetworkAccessManager::PutOperation)
+ {
+ jsonDoc = handleReply( reply );
+ }
+ // Free space.
+ reply->deleteLater();
+
+ // Return response
+ return jsonDoc;
+}
+
+QJsonDocument LedDevicePhilipsHueBridge::handleReply(QNetworkReply* const &reply )
+{
+ QJsonDocument jsonDoc;
+
+ int httpStatusCode = reply->attribute( QNetworkRequest::HttpStatusCodeAttribute ).toInt();
+ DebugIf(verbose, _log, "Reply.httpStatusCode [%d]", httpStatusCode );
+ QString errorReason;
+
+ if(reply->error() == QNetworkReply::NoError)
+ {
+ if ( httpStatusCode != 204 ){
+ QByteArray response = reply->readAll();
+ QJsonParseError error;
+ jsonDoc = QJsonDocument::fromJson(response, &error);
+ if (error.error != QJsonParseError::NoError)
+ {
+ this->setInError ( "Got invalid response" );
+ }
+ else
+ {
+ QString strJson(jsonDoc.toJson(QJsonDocument::Compact));
+ DebugIf(verbose, _log, "Reply: [%s]", strJson.toUtf8().constData() );
+
+ QVariantList rspList = jsonDoc.toVariant().toList();
+ if ( !rspList.isEmpty() )
+ {
+ QVariantMap map = rspList.first().toMap();
+ if ( map.contains(API_ERROR) )
+ {
+ // API call failsed to execute an error message was returned
+ QString errorAddress = map.value(API_ERROR).toMap().value(API_ERROR_ADDRESS).toString();
+ QString errorDesc = map.value(API_ERROR).toMap().value(API_ERROR_DESCRIPTION).toString();
+ QString errorType = map.value(API_ERROR).toMap().value(API_ERROR_TYPE).toString();
+
+ Debug(_log, "Error Type : %s", QSTRING_CSTR( errorType ));
+ Debug(_log, "Error Address : %s", QSTRING_CSTR( errorAddress ));
+ Debug(_log, "Error Description : %s", QSTRING_CSTR( errorDesc ));
+
+ errorReason = QString ("(%1) %2, Resource:%3").arg(errorType, errorDesc, errorAddress);
+ this->setInError ( errorReason );
+ }
+ }
+ }
+ }
+ }
+ else
+ {
+ if ( httpStatusCode > 0 ) {
+ QString httpReason = reply->attribute( QNetworkRequest::HttpReasonPhraseAttribute ).toString();
+ QString advise;
+ switch ( httpStatusCode ) {
+ case 400:
+ advise = "Check Request Body";
+ break;
+ case 401:
+ advise = "Check Authentication Token (API Key)";
+ break;
+ case 404:
+ advise = "Check Resource given";
+ break;
+ default:
+ break;
+ }
+ errorReason = QString ("%1:%2 [%3 %4] - %5").arg(_hostname, _api_port, QString(httpStatusCode) , httpReason, advise);
+ }
+ else {
+ errorReason = QString ("%1:%2 - %3").arg(_hostname, _api_port, reply->errorString());
+ }
+ this->setInError ( errorReason );
+ }
+ // Return response
+ return jsonDoc;
+}
+
+void LedDevicePhilipsHueBridge::post(const QString& route, const QString& content)
+{
+ QString url = getUrl(_hostname, _api_port, _username, route );
+ putJson( url, content );
+}
+
+void LedDevicePhilipsHueBridge::setLightState(const unsigned int lightId, QString state)
+{
+ Debug(_log, "SetLightState [%u]: %s", lightId, QSTRING_CSTR(state));
+ post( QString("%1/%2/%3").arg(API_LIGHTS).arg(lightId).arg(API_STATE), state );
+}
+
+const std::set PhilipsHueLight::GAMUT_A_MODEL_IDS =
+ { "LLC001", "LLC005", "LLC006", "LLC007", "LLC010", "LLC011", "LLC012", "LLC013", "LLC014", "LST001" };
+const std::set PhilipsHueLight::GAMUT_B_MODEL_IDS =
+ { "LCT001", "LCT002", "LCT003", "LCT007", "LLM001" };
+const std::set PhilipsHueLight::GAMUT_C_MODEL_IDS =
+ { "LLC020", "LST002", "LCT011", "LCT012", "LCT010", "LCT014", "LCT015", "LCT016", "LCT024" };
+
+PhilipsHueLight::PhilipsHueLight(Logger* log, unsigned int id, QJsonObject values, unsigned int ledidx)
+ : _log(log)
+ , _id(id)
+ , _ledidx(ledidx)
+ , _on(false)
+ , _transitionTime(0)
+ , _colorBlack({0.0, 0.0, 0.0})
+ , _modelId (values[API_MODEID].toString().trimmed().replace("\"", ""))
+{
+ // Find id in the sets and set the appropriate color space.
+ if (GAMUT_A_MODEL_IDS.find(_modelId) != GAMUT_A_MODEL_IDS.end())
+ {
+ Debug(_log, "Recognized model id %s of light ID %d as gamut A", QSTRING_CSTR(_modelId), id);
+ _colorSpace.red = {0.704, 0.296};
+ _colorSpace.green = {0.2151, 0.7106};
+ _colorSpace.blue = {0.138, 0.08};
+ _colorBlack = {0.138, 0.08, 0.0};
+ }
+ else if (GAMUT_B_MODEL_IDS.find(_modelId) != GAMUT_B_MODEL_IDS.end())
+ {
+ Debug(_log, "Recognized model id %s of light ID %d as gamut B", QSTRING_CSTR(_modelId), id);
+ _colorSpace.red = {0.675, 0.322};
+ _colorSpace.green = {0.409, 0.518};
+ _colorSpace.blue = {0.167, 0.04};
+ _colorBlack = {0.167, 0.04, 0.0};
+ }
+ else if (GAMUT_C_MODEL_IDS.find(_modelId) != GAMUT_C_MODEL_IDS.end())
+ {
+ Debug(_log, "Recognized model id %s of light ID %d as gamut C", QSTRING_CSTR(_modelId), id);
+ _colorSpace.red = {0.6915, 0.3083};
+ _colorSpace.green = {0.17, 0.7};
+ _colorSpace.blue = {0.1532, 0.0475};
+ _colorBlack = {0.1532, 0.0475, 0.0};
+ }
+ else
+ {
+ Warning(_log, "Did not recognize model id %s of light ID %d", QSTRING_CSTR(_modelId), id);
+ _colorSpace.red = {1.0, 0.0};
+ _colorSpace.green = {0.0, 1.0};
+ _colorSpace.blue = {0.0, 0.0};
+ _colorBlack = {0.0, 0.0, 0.0};
}
- Info(_log,"Light ID %d created", id);
+ saveOriginalState(values);
+
+ _lightname = values["name"].toString().trimmed().replace("\"", "");
+ Info(_log,"Light ID %d (\"%s\", LED index \"%d\") created", id, QSTRING_CSTR(_lightname), ledidx);
}
PhilipsHueLight::~PhilipsHueLight()
{
- // Restore the original state.
- set(originalState);
}
-void PhilipsHueLight::set(QString state)
+unsigned int PhilipsHueLight::getId() const
{
- bridge->post(QString("lights/%1/state").arg(id), state);
+ return _id;
}
-void PhilipsHueLight::setOn(bool on)
+QString PhilipsHueLight::getOriginalState()
{
- if (this->on != on)
+ return _originalState;
+}
+
+void PhilipsHueLight::saveOriginalState(const QJsonObject& values)
+{
+ // Get state object values which are subject to change.
+ if (!values[API_STATE].toObject().contains("on"))
{
- QString arg = on ? "true" : "false";
- set(QString("{ \"on\": %1 }").arg(arg));
+ Error(_log, "Got invalid state object from light ID %d", _id);
}
- this->on = on;
+ QJsonObject lState = values[API_STATE].toObject();
+ _originalStateJSON = lState;
+
+ QJsonObject state;
+ state["on"] = lState["on"];
+ _originalColor = _colorBlack;
+ QString c;
+ if (state[API_STATE_ON].toBool())
+ {
+ state[API_XY_COORDINATES] = lState[API_XY_COORDINATES];
+ state[API_BRIGHTNESS] = lState[API_BRIGHTNESS];
+ _on = true;
+ _color = {
+ state[API_XY_COORDINATES].toArray()[0].toDouble(),
+ state[API_XY_COORDINATES].toArray()[1].toDouble(),
+ state[API_BRIGHTNESS].toDouble() / 254.0
+ };
+ _originalColor = _color;
+ c = QString("{ \"%1\": [%2, %3], \"%4\": %5 }").arg(API_XY_COORDINATES).arg(_originalColor.x, 0, 'd', 4).arg(_originalColor.y, 0, 'd', 4).arg(API_BRIGHTNESS).arg((_originalColor.bri * 254.0), 0, 'd', 4);
+ DebugIf(verbose, _log, "OriginalColor state on: %s", QSTRING_CSTR(c));
+ _transitionTime = values[API_STATE].toObject()[API_TRANSITIONTIME].toInt();
+ }
+ //Determine the original state.
+ _originalState = QJsonDocument(state).toJson(QJsonDocument::JsonFormat::Compact).trimmed();
+}
+
+void PhilipsHueLight::setOnOffState(bool on)
+{
+ this->_on = on;
}
void PhilipsHueLight::setTransitionTime(unsigned int transitionTime)
{
- if (this->transitionTime != transitionTime)
- {
- set(QString("{ \"transitiontime\": %1 }").arg(transitionTime));
- }
- this->transitionTime = transitionTime;
+ this->_transitionTime = transitionTime;
}
-void PhilipsHueLight::setColor(CiColor color, float brightnessFactor)
+void PhilipsHueLight::setColor(const CiColor& color)
{
- if (this->color != color)
- {
- const int bri = qRound(qMin(254.0f, brightnessFactor * qMax(1.0f, color.bri * 254.0f)));
- set(QString("{ \"xy\": [%1, %2], \"bri\": %3 }").arg(color.x, 0, 'f', 4).arg(color.y, 0, 'f', 4).arg(bri));
- }
- this->color = color;
+ this->_color = color;
+}
+
+bool PhilipsHueLight::getOnOffState() const
+{
+ return _on;
+}
+
+unsigned int PhilipsHueLight::getTransitionTime() const
+{
+ return _transitionTime;
}
CiColor PhilipsHueLight::getColor() const
{
- return color;
+ return _color;
}
CiColorTriangle PhilipsHueLight::getColorSpace() const
{
- return colorSpace;
+ return _colorSpace;
+}
+
+LedDevicePhilipsHue::LedDevicePhilipsHue(const QJsonObject& deviceConfig)
+ : LedDevicePhilipsHueBridge(deviceConfig)
+ , _switchOffOnBlack (false)
+ , _brightnessFactor(1.0)
+ , _transitionTime (1)
+ , _isRestoreOrigState (true)
+{
+ _devConfig = deviceConfig;
+ _deviceReady = false;
}
LedDevice* LedDevicePhilipsHue::construct(const QJsonObject &deviceConfig)
@@ -345,74 +697,132 @@ LedDevice* LedDevicePhilipsHue::construct(const QJsonObject &deviceConfig)
return new LedDevicePhilipsHue(deviceConfig);
}
-LedDevicePhilipsHue::LedDevicePhilipsHue(const QJsonObject& deviceConfig)
- : LedDevice(deviceConfig)
- , _bridge(nullptr)
-{
-
-}
-
LedDevicePhilipsHue::~LedDevicePhilipsHue()
{
- switchOff();
- delete _bridge;
-}
-
-void LedDevicePhilipsHue::start()
-{
- _bridge = new PhilipsHueBridge(_log, _devConfig["output"].toString(), _devConfig["username"].toString());
- _deviceReady = init(_devConfig);
-
- connect(_bridge, &PhilipsHueBridge::newLights, this, &LedDevicePhilipsHue::newLights);
- connect(this, &LedDevice::enableStateChanged, this, &LedDevicePhilipsHue::stateChanged);
}
bool LedDevicePhilipsHue::init(const QJsonObject &deviceConfig)
{
- switchOffOnBlack = deviceConfig["switchOffOnBlack"].toBool(true);
- brightnessFactor = (float) deviceConfig["brightnessFactor"].toDouble(1.0);
- transitionTime = deviceConfig["transitiontime"].toInt(1);
- QJsonArray lArray = deviceConfig["lightIds"].toArray();
+ bool isInitOK = LedDevicePhilipsHueBridge::init(deviceConfig);
- QJsonObject newDC = deviceConfig;
- if(!lArray.empty())
+ if ( isInitOK )
{
- for(const auto i : lArray)
+ // Initiatiale LedDevice configuration and execution environment
+ _switchOffOnBlack = _devConfig[CONFIG_ON_OFF_BLACK].toBool(true);
+ _brightnessFactor = _devConfig[CONFIG_BRIGHTNESSFACTOR].toDouble(1.0);
+ _transitionTime = _devConfig[CONFIG_TRANSITIONTIME].toInt(1);
+ _isRestoreOrigState = _devConfig[CONFIG_RESTORE_STATE].toBool(true);
+ QJsonArray lArray = _devConfig[CONFIG_LIGHTIDS].toArray();
+
+ _lightIds.clear();
+ if(!lArray.empty())
{
- lightIds.push_back(i.toInt());
+ for(const auto i : lArray)
+ {
+ _lightIds.push_back(i.toInt());
+ }
}
- // get light info from bridge
- _bridge->bConnect();
- // adapt latchTime to count of user lightIds (bridge 10Hz max overall)
- newDC.insert("latchTime",QJsonValue(100*(int)lightIds.size()));
+ uint configuredLightsCount = _lightIds.size();
+ Debug(_log, "Off on Black : %d", _switchOffOnBlack );
+ Debug(_log, "Brightness Factor : %f", _brightnessFactor );
+ Debug(_log, "Transition Time : %d", _transitionTime );
+ Debug(_log, "Light IDs defined : %d", configuredLightsCount );
+
+ if ( configuredLightsCount == 0)
+ {
+ setInError("No light-IDs configured");
+ isInitOK = false;
+ }
}
- else
- {
- Error(_log,"No light ID provided, abort");
- }
-
- LedDevice::init(newDC);
-
- return true;
+ return isInitOK;
}
-void LedDevicePhilipsHue::newLights(QMap map)
+bool LedDevicePhilipsHue::initLeds()
{
- if(!lightIds.empty())
+ bool isInitOK = false;
+
+ if ( !isInError() )
+ {
+ updateLights( getLightMap() );
+ // adapt latchTime to count of user lightIds (bridge 10Hz max overall)
+ setLatchTime( static_cast( 100 * getLightsCount() ) );
+ isInitOK = true;
+ }
+
+ return isInitOK;
+}
+
+void LedDevicePhilipsHue::updateLights(QMap map)
+{
+ if(!_lightIds.empty())
{
// search user lightid inside map and create light if found
- lights.clear();
- for(const auto id : lightIds)
+ _lights.clear();
+
+ unsigned int ledidx = 0;
+ _lights.reserve(_lightIds.size());
+ for(const auto id : _lightIds)
{
if (map.contains(id))
{
- lights.push_back(PhilipsHueLight(_log, _bridge, id, map.value(id)));
+ _lights.emplace_back(_log, id, map.value(id), ledidx);
}
else
{
- Error(_log,"Light id %d isn't used on this bridge", id);
+ Warning(_log,"Configured light-ID %d is not available at this bridge", id);
}
+ ledidx++;
+ }
+ setLightsCount ( _lights.size() );
+ }
+}
+
+int LedDevicePhilipsHue::open()
+{
+ int retval = -1;
+ QString errortext;
+ _deviceReady = false;
+
+ // General initialisation and configuration of LedDevice
+ if ( init(_devConfig) )
+ {
+ if ( LedDevicePhilipsHueBridge::open() )
+ // Open/Start LedDevice based on configuration
+ {
+ if ( initLeds() )
+ {
+ // Everything is OK -> enable device
+ _deviceReady = true;
+ setEnable(true);
+ retval = 0;
+ }
+ }
+ }
+ return retval;
+}
+
+void LedDevicePhilipsHue::restoreOriginalState()
+{
+ if(!_lightIds.empty())
+ {
+ for (PhilipsHueLight& light : _lights)
+ {
+ setLightState(light.getId(),light.getOriginalState());
+ }
+ }
+}
+
+void LedDevicePhilipsHue::close()
+{
+ LedDevicePhilipsHueBridge::close();
+
+ if ( _deviceReady)
+ {
+ if ( _isRestoreOrigState )
+ {
+ //Restore Philips Hue devices state
+ restoreOriginalState();
}
}
}
@@ -420,47 +830,139 @@ void LedDevicePhilipsHue::newLights(QMap map)
int LedDevicePhilipsHue::write(const std::vector & ledValues)
{
// lights will be empty sometimes
- if(lights.empty()) return -1;
+ if(_lights.empty()) return -1;
// more lights then leds, stop always
- if(ledValues.size() < lights.size())
+ if(ledValues.size() < getLightsCount() )
{
- Error(_log,"More LightIDs configured than leds, each LightID requires one led!");
+ Error(_log,"More light-IDs configured than leds, each light-ID requires one led!");
return -1;
}
- // Iterate through lights and set colors.
- unsigned int idx = 0;
- for (PhilipsHueLight& light : lights)
- {
- // Get color.
- ColorRgb color = ledValues.at(idx);
- // Scale colors from [0, 255] to [0, 1] and convert to xy space.
- CiColor xy = CiColor::rgbToCiColor(color.red / 255.0f, color.green / 255.0f, color.blue / 255.0f,
- light.getColorSpace());
-
- if (switchOffOnBlack && xy.bri == 0)
- {
- light.setOn(false);
- }
- else
- {
- light.setOn(true);
- // Write color if color has been changed.
- light.setTransitionTime(transitionTime);
- light.setColor(xy, brightnessFactor);
- }
-
- idx++;
- }
+ writeSingleLights (ledValues);
return 0;
}
-void LedDevicePhilipsHue::stateChanged(bool newState)
+int LedDevicePhilipsHue::writeSingleLights(const std::vector& ledValues)
{
- if(newState)
- _bridge->bConnect();
- else
- lights.clear();
+ // Iterate through lights and set colors.
+ unsigned int idx = 0;
+ for (PhilipsHueLight& light : _lights)
+ {
+ // Get color.
+ ColorRgb color = ledValues.at(idx);
+ // Scale colors from [0, 255] to [0, 1] and convert to xy space.
+ CiColor xy = CiColor::rgbToCiColor(color.red / 255.0, color.green / 255.0, color.blue / 255.0, light.getColorSpace());
+
+ if (_switchOffOnBlack && xy.bri == 0.0)
+ {
+ this->setOnOffState(light, false);
+ }
+ else
+ {
+ // Write color if color has been changed.
+ this->setState(light, true, xy, _brightnessFactor,_transitionTime);
+ }
+ idx++;
+ }
+ return 0;
+}
+
+void LedDevicePhilipsHue::setOnOffState(PhilipsHueLight& light, bool on)
+{
+ if (light.getOnOffState() != on)
+ {
+ light.setOnOffState( on );
+ QString state = on ? API_STATE_VALUE_TRUE : API_STATE_VALUE_FALSE;
+ setLightState( light.getId(), QString("{\"%1\": %2 }").arg(API_STATE_ON, state) );
+ }
+}
+
+void LedDevicePhilipsHue::setTransitionTime(PhilipsHueLight& light, unsigned int transitionTime)
+{
+ if (light.getTransitionTime() != transitionTime)
+ {
+ light.setTransitionTime( transitionTime );
+ setLightState( light.getId(), QString("{\"%1\": %2 }").arg(API_TRANSITIONTIME).arg( transitionTime ) );
+ }
+}
+
+void LedDevicePhilipsHue::setColor(PhilipsHueLight& light, const CiColor& color, double brightnessFactor)
+{
+ const int bri = qRound(qMin(254.0, brightnessFactor * qMax(1.0, color.bri * 254.0)));
+ if (light.getColor() != color)
+ {
+ light.setColor( color) ;
+ QString stateCmd = QString("\"%1\":[%2,%3],\"%4\":%5").arg(API_XY_COORDINATES).arg(color.x, 0, 'd', 4).arg(color.y, 0, 'd', 4).arg (API_BRIGHTNESS).arg(bri);
+ setLightState( light.getId(), stateCmd );
+ }
+}
+
+void LedDevicePhilipsHue::setState(PhilipsHueLight& light, bool on, const CiColor& color, double brightnessFactor, unsigned int transitionTime)
+{
+
+ QString stateCmd;
+
+ if (light.getOnOffState() != on)
+ {
+ light.setOnOffState( on );
+ QString state = on ? API_STATE_VALUE_TRUE : API_STATE_VALUE_FALSE;
+ stateCmd += QString("\"%1\":%2,").arg(API_STATE_ON, state);
+ }
+
+ if (light.getTransitionTime() != transitionTime)
+ {
+ light.setTransitionTime( transitionTime );
+ stateCmd += QString("\"%1\":%2,").arg(API_TRANSITIONTIME).arg( transitionTime );
+ }
+
+ const int bri = qRound(qMin(254.0, brightnessFactor * qMax(1.0, color.bri * 254.0)));
+ if (light.getColor() != color)
+ {
+ light.setColor( color) ;
+ stateCmd += QString("\"%1\":[%2,%3],\"%4\":%5").arg(API_XY_COORDINATES).arg(color.x, 0, 'd', 4).arg(color.y, 0, 'd', 4).arg (API_BRIGHTNESS).arg(bri);
+
+ }
+
+ if ( !stateCmd.isEmpty() )
+ {
+ setLightState( light.getId(), "{" + stateCmd + "}" );
+ }
+}
+
+
+void LedDevicePhilipsHue::setLightsCount( unsigned int lightsCount )
+{
+ _lightsCount = lightsCount;
+}
+
+
+int LedDevicePhilipsHue::switchOn()
+{
+ if ( _deviceReady)
+ {
+ //Switch on Philips Hue devices physically
+ for (PhilipsHueLight& light : _lights)
+ {
+ setOnOffState(light,true);
+ }
+ }
+ return 0;
+}
+
+int LedDevicePhilipsHue::switchOff()
+{
+ //Set all LEDs to Black
+ int rc = LedDevice::switchOff();
+
+ if ( _deviceReady)
+ {
+ //Switch off Philips Hue devices physically
+ for (PhilipsHueLight& light : _lights)
+ {
+ setOnOffState(light,false);
+ }
+ }
+ return rc;
}
diff --git a/libsrc/leddevice/dev_net/LedDevicePhilipsHue.h b/libsrc/leddevice/dev_net/LedDevicePhilipsHue.h
index 655fe1cc..96332796 100644
--- a/libsrc/leddevice/dev_net/LedDevicePhilipsHue.h
+++ b/libsrc/leddevice/dev_net/LedDevicePhilipsHue.h
@@ -19,11 +19,11 @@ struct CiColorTriangle;
struct CiColor
{
/// X component.
- float x;
+ double x;
/// Y component.
- float y;
+ double y;
/// The brightness.
- float bri;
+ double bri;
///
/// Converts an RGB color to the Hue xy color space and brightness.
@@ -37,7 +37,7 @@ struct CiColor
///
/// @return color point
///
- static CiColor rgbToCiColor(float red, float green, float blue, CiColorTriangle colorSpace);
+ static CiColor rgbToCiColor(double red, double green, double blue, CiColorTriangle colorSpace);
///
/// @param p the color point to check
@@ -53,7 +53,7 @@ struct CiColor
///
/// @return the cross product between p1 and p2
///
- static float crossProduct(CiColor p1, CiColor p2);
+ static double crossProduct(CiColor p1, CiColor p2);
///
/// @param a reference point one
@@ -73,11 +73,11 @@ struct CiColor
///
/// @return the distance between the two points
///
- static float getDistanceBetweenTwoPoints(CiColor p1, CiColor p2);
+ static double getDistanceBetweenTwoPoints(CiColor p1, CiColor p2);
};
-bool operator==(CiColor p1, CiColor p2);
-bool operator!=(CiColor p1, CiColor p2);
+bool operator==(const CiColor& p1, const CiColor& p2);
+bool operator!=(const CiColor& p1, const CiColor& p2);
/**
* Color triangle to define an available color space for the hue lamps.
@@ -87,74 +87,11 @@ struct CiColorTriangle
CiColor red, green, blue;
};
-class PhilipsHueBridge : public QObject
-{
- Q_OBJECT
-
-private:
- Logger* _log;
- /// QNetworkAccessManager for sending requests.
- QNetworkAccessManager manager;
- /// Ip address of the bridge
- QString host;
- /// User name for the API ("newdeveloper")
- QString username;
- /// Timer for bridge reconnect interval
- QTimer bTimer;
-
-private slots:
- ///
- /// Receive all replies and check for error, schedule reconnect on issues
- /// Emits newLights() on success when triggered from connect()
- ///
- void resolveReply(QNetworkReply* reply);
-
-public slots:
- ///
- /// Connect to bridge to check availbility and user
- ///
- void bConnect(void);
-
-signals:
- ///
- /// Emits with a QMap of current bridge light/value pairs
- ///
- void newLights(QMap map);
-
-public:
- PhilipsHueBridge(Logger* log, QString host, QString username);
-
- ///
- /// @param route the route of the POST request.
- ///
- /// @param content the content of the POST request.
- ///
- void post(QString route, QString content);
-};
-
/**
* Simple class to hold the id, the latest color, the color space and the original state.
*/
class PhilipsHueLight
{
-private:
- Logger* _log;
- PhilipsHueBridge* bridge;
- /// light id
- unsigned int id;
- bool on;
- unsigned int transitionTime;
- CiColor color;
- /// The model id of the hue lamp which is used to determine the color space.
- QString modelId;
- CiColorTriangle colorSpace;
- /// The json string of the original state.
- QString originalState;
-
- ///
- /// @param state the state as json object to set
- ///
- void set(QString state);
public:
// Hue system model ids (http://www.developers.meethue.com/documentation/supported-lights).
@@ -172,32 +109,183 @@ public:
/// @param bridge the bridge
/// @param id the light id
///
- PhilipsHueLight(Logger* log, PhilipsHueBridge* bridge, unsigned int id, QJsonObject values);
+ PhilipsHueLight(Logger* log, unsigned int id, QJsonObject values, unsigned int ledidx);
~PhilipsHueLight();
///
/// @param on
///
- void setOn(bool on);
+ void setOnOffState(bool on);
///
/// @param transitionTime the transition time between colors in multiples of 100 ms
///
- void setTransitionTime(unsigned int transitionTime);
+ void setTransitionTime(unsigned int _transitionTime);
///
/// @param color the color to set
- /// @param brightnessFactor the factor to apply to the CiColor#bri value
///
- void setColor(CiColor color, float brightnessFactor = 1.0f);
+ void setColor(const CiColor& _color);
+
+
+ unsigned int getId() const;
+
+ bool getOnOffState() const;
+ unsigned int getTransitionTime() const;
CiColor getColor() const;
///
/// @return the color space of the light determined by the model id reported by the bridge.
CiColorTriangle getColorSpace() const;
+
+ 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& 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 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 _lightsMap;
+};
+
+
+
/**
* Implementation for the Philips Hue system.
*
@@ -206,7 +294,7 @@ public:
*
* @author ntim (github), bimsarck (github)
*/
-class LedDevicePhilipsHue: public LedDevice
+class LedDevicePhilipsHue: public LedDevicePhilipsHueBridge
{
Q_OBJECT
@@ -227,20 +315,66 @@ public:
/// constructs leddevice
static LedDevice* construct(const QJsonObject &deviceConfig);
-public slots:
- /// thread start
- virtual void start() override;
+ ///
+ /// 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;
-private slots:
/// creates new PhilipsHueLight(s) based on user lightid with bridge feedback
///
/// @param map Map of lightid/value pairs of bridge
///
void newLights(QMap map);
- void stateChanged(bool newState);
+ 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 map);
protected:
+
+ ///
+ /// 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();
+
///
/// Writes the RGB-Color values to the leds.
///
@@ -249,21 +383,27 @@ protected:
/// @return Zero on success else negative
///
virtual int write(const std::vector & ledValues) override;
- bool init(const QJsonObject &deviceConfig) override;
private:
- /// bridge class
- PhilipsHueBridge* _bridge;
+
+ int writeSingleLights(const std::vector& ledValues);
///
- bool switchOffOnBlack;
+ bool _switchOffOnBlack;
/// The brightness factor to multiply on color change.
- float brightnessFactor;
- /// Transition time in multiples of 100 ms.
+ double _brightnessFactor;
+ /// Transition time in multiples of 100 ms.
/// The default of the Hue lights is 400 ms, but we may want it snapier.
- int transitionTime;
+ unsigned int _transitionTime;
+
+ bool _isRestoreOrigState;
+
/// Array of the light ids.
- std::vector lightIds;
+ std::vector _lightIds;
/// Array to save the lamps.
- std::vector lights;
+ std::vector _lights;
+
+ unsigned int _lightsCount;
+
+
};
diff --git a/libsrc/leddevice/dev_spi/ProviderSpi.cpp b/libsrc/leddevice/dev_spi/ProviderSpi.cpp
index 4787484a..588c8ddd 100644
--- a/libsrc/leddevice/dev_spi/ProviderSpi.cpp
+++ b/libsrc/leddevice/dev_spi/ProviderSpi.cpp
@@ -93,7 +93,7 @@ int ProviderSpi::open()
}
if ( retval < 0 )
{
- errortext = QString ("Failed to open device (%1). Error Code: %2").arg(_deviceName, retval);
+ errortext = QString ("Failed to open device (%1). Error Code: %2").arg(_deviceName).arg(retval);
}
}
diff --git a/libsrc/leddevice/schemas/schema-philipshue.json b/libsrc/leddevice/schemas/schema-philipshue.json
index 9ebe806d..d90a85d6 100644
--- a/libsrc/leddevice/schemas/schema-philipshue.json
+++ b/libsrc/leddevice/schemas/schema-philipshue.json
@@ -35,6 +35,13 @@
"maximum" : 10.0,
"propertyOrder" : 5
},
+ "restoreOriginalState": {
+ "type": "boolean",
+ "title":"edt_dev_spec_restoreOriginalState_title",
+ "default" : true,
+ "propertyOrder" : 6
+ },
+
"lightIds": {
"type": "array",
"title":"edt_dev_spec_lightid_title",
@@ -45,7 +52,7 @@
"minimum" : 0,
"title" : "edt_dev_spec_lightid_itemtitle"
},
- "propertyOrder" : 6
+ "propertyOrder" : 7
}
},
"additionalProperties": true