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