2019-04-08 23:13:11 +02:00
|
|
|
// Local-Hyperion includes
|
|
|
|
#include "LedDeviceNanoleaf.h"
|
|
|
|
|
|
|
|
// ssdp discover
|
|
|
|
#include <ssdp/SSDPDiscover.h>
|
|
|
|
|
|
|
|
// Qt includes
|
|
|
|
#include <QEventLoop>
|
|
|
|
#include <QNetworkReply>
|
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
//std includes
|
|
|
|
#include <sstream>
|
|
|
|
#include <iomanip>
|
|
|
|
|
|
|
|
//
|
|
|
|
static const bool verbose = false;
|
|
|
|
static const bool verbose3 = false;
|
|
|
|
|
2019-04-08 23:13:11 +02:00
|
|
|
// Controller configuration settings
|
2020-02-10 15:21:58 +01:00
|
|
|
static const char CONFIG_ADDRESS[] = "host";
|
2019-09-14 22:54:41 +02:00
|
|
|
//static const char CONFIG_PORT[] = "port";
|
|
|
|
static const char CONFIG_AUTH_TOKEN[] ="token";
|
2019-04-08 23:13:11 +02:00
|
|
|
|
|
|
|
// Panel configuration settings
|
2019-09-14 22:54:41 +02:00
|
|
|
static const char PANEL_LAYOUT[] = "layout";
|
|
|
|
static const char PANEL_NUM[] = "numPanels";
|
|
|
|
static const char PANEL_ID[] = "panelId";
|
|
|
|
static const char PANEL_POSITIONDATA[] = "positionData";
|
|
|
|
static const char PANEL_SHAPE_TYPE[] = "shapeType";
|
|
|
|
//static const char PANEL_ORIENTATION[] = "0";
|
|
|
|
static const char PANEL_POS_X[] = "x";
|
|
|
|
static const char PANEL_POS_Y[] = "y";
|
2019-04-08 23:13:11 +02:00
|
|
|
|
|
|
|
// List of State Information
|
2019-09-14 22:54:41 +02:00
|
|
|
static const char STATE_ON[] = "on";
|
|
|
|
static const char STATE_ONOFF_VALUE[] = "value";
|
|
|
|
static const char STATE_VALUE_TRUE[] = "true";
|
|
|
|
static const char STATE_VALUE_FALSE[] = "false";
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-03-26 18:49:44 +01:00
|
|
|
// Device Data elements
|
2019-09-14 22:54:41 +02:00
|
|
|
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";
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-03-26 18:49:44 +01:00
|
|
|
// Nanoleaf Stream Control elements
|
2019-09-14 22:54:41 +02:00
|
|
|
//static const char STREAM_CONTROL_IP[] = "streamControlIpAddr";
|
|
|
|
static const char STREAM_CONTROL_PORT[] = "streamControlPort";
|
|
|
|
//static const char STREAM_CONTROL_PROTOCOL[] = "streamControlProtocol";
|
2019-04-08 23:13:11 +02:00
|
|
|
const quint16 STREAM_CONTROL_DEFAULT_PORT = 60222; //Fixed port for Canvas;
|
|
|
|
|
|
|
|
// Nanoleaf OpenAPI URLs
|
2019-09-14 22:54:41 +02:00
|
|
|
static const char API_DEFAULT_PORT[] = "16021";
|
|
|
|
static const char API_URL_FORMAT[] = "http://%1:%2/api/v1/%3/%4";
|
|
|
|
static const char API_ROOT[] = "";
|
2019-12-08 13:12:01 +01:00
|
|
|
//static const char API_EXT_MODE_STRING_V1[] = "{\"write\" : {\"command\" : \"display\", \"animType\" : \"extControl\"}}";
|
2019-09-14 22:54:41 +02:00
|
|
|
static const char API_EXT_MODE_STRING_V2[] = "{\"write\" : {\"command\" : \"display\", \"animType\" : \"extControl\", \"extControlVersion\" : \"v2\"}}";
|
|
|
|
static const char API_STATE[] ="state";
|
|
|
|
static const char API_PANELLAYOUT[] = "panelLayout";
|
|
|
|
static const char API_EFFECT[] = "effects";
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-03-26 18:49:44 +01:00
|
|
|
// Nanoleaf ssdp services
|
2019-09-14 22:54:41 +02:00
|
|
|
static const char SSDP_CANVAS[] = "nanoleaf:nl29";
|
|
|
|
static const char SSDP_LIGHTPANELS[] = "nanoleaf_aurora:light";
|
2019-04-08 23:13:11 +02:00
|
|
|
const int SSDP_TIMEOUT = 5000; // timout in ms
|
|
|
|
|
|
|
|
// Nanoleaf Panel Shapetypes
|
|
|
|
enum SHAPETYPES {
|
2019-12-08 13:12:01 +01:00
|
|
|
TRIANGLE,
|
|
|
|
RHYTM,
|
|
|
|
SQUARE,
|
|
|
|
CONTROL_SQUARE_PRIMARY,
|
|
|
|
CONTROL_SQUARE_PASSIVE,
|
|
|
|
POWER_SUPPLY,
|
2019-04-08 23:13:11 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
// Nanoleaf external control versions
|
|
|
|
enum EXTCONTROLVERSIONS {
|
2019-12-08 13:12:01 +01:00
|
|
|
EXTCTRLVER_V1 = 1,
|
|
|
|
EXTCTRLVER_V2
|
2019-04-08 23:13:11 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
LedDevice* LedDeviceNanoleaf::construct(const QJsonObject &deviceConfig)
|
|
|
|
{
|
2019-12-08 13:12:01 +01:00
|
|
|
return new LedDeviceNanoleaf(deviceConfig);
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
LedDeviceNanoleaf::~LedDeviceNanoleaf()
|
|
|
|
{
|
|
|
|
_networkmanager->deleteLater();
|
|
|
|
}
|
|
|
|
|
2019-04-08 23:13:11 +02:00
|
|
|
LedDeviceNanoleaf::LedDeviceNanoleaf(const QJsonObject &deviceConfig)
|
2019-12-08 13:12:01 +01:00
|
|
|
: ProviderUdp()
|
2019-04-08 23:13:11 +02:00
|
|
|
{
|
2020-02-10 15:21:58 +01:00
|
|
|
_devConfig = deviceConfig;
|
|
|
|
_deviceReady = false;
|
|
|
|
_networkmanager = nullptr;
|
|
|
|
_extControlVersion = EXTCTRLVER_V2;
|
|
|
|
_panelLedCount = 0;
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
bool LedDeviceNanoleaf::init(const QJsonObject &deviceConfig)
|
|
|
|
{
|
|
|
|
// Overwrite non supported/required features
|
|
|
|
_devConfig["latchTime"] = 0;
|
|
|
|
if (deviceConfig["rewriteTime"].toInt(0) > 0)
|
|
|
|
{
|
|
|
|
Info (_log, "Device Nanoleaf does not require rewrites. Refresh time is ignored.");
|
|
|
|
_devConfig["rewriteTime"] = 0;
|
|
|
|
}
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
DebugIf(verbose, _log, "deviceConfig: [%s]", QString(QJsonDocument(_devConfig).toJson(QJsonDocument::Compact)).toUtf8().constData() );
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
bool isInitOK = LedDevice::init(deviceConfig);
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
if ( isInitOK )
|
|
|
|
{
|
|
|
|
uint configuredLedCount = this->getLedCount();
|
|
|
|
Debug(_log, "DeviceType : %s", QSTRING_CSTR( this->getActiveDeviceType() ));
|
|
|
|
Debug(_log, "LedCount : %u", configuredLedCount);
|
|
|
|
Debug(_log, "ColorOrder : %s", QSTRING_CSTR( this->getColorOrder() ));
|
|
|
|
Debug(_log, "RefreshTime : %d", _refresh_timer_interval);
|
|
|
|
Debug(_log, "LatchTime : %d", this->getLatchTime());
|
|
|
|
|
|
|
|
//Set hostname as per configuration and_defaultHost default port
|
|
|
|
_hostname = deviceConfig[ CONFIG_ADDRESS ].toString();
|
|
|
|
_api_port = API_DEFAULT_PORT;
|
|
|
|
_auth_token = deviceConfig[ CONFIG_AUTH_TOKEN ].toString();
|
|
|
|
|
|
|
|
//If host not configured then discover device
|
|
|
|
if ( _hostname.isEmpty() )
|
|
|
|
{
|
|
|
|
//Discover Nanoleaf device
|
2020-03-26 18:49:44 +01:00
|
|
|
if ( !discoverDevice() )
|
2020-02-10 15:21:58 +01:00
|
|
|
{
|
|
|
|
this->setInError("No target IP defined nor Nanoleaf device was discovered");
|
|
|
|
return false;
|
|
|
|
}
|
2019-12-08 13:12:01 +01:00
|
|
|
}
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
// Set UDP streaming port
|
|
|
|
_devConfig["host"] = _hostname;
|
|
|
|
_devConfig["port"] = STREAM_CONTROL_DEFAULT_PORT;
|
|
|
|
isInitOK = ProviderUdp::init(_devConfig);
|
|
|
|
|
|
|
|
Debug(_log, "Hostname/IP : %s", QSTRING_CSTR( _hostname ));
|
|
|
|
Debug(_log, "Port : %d", _port);
|
|
|
|
}
|
|
|
|
return isInitOK;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LedDeviceNanoleaf::initLeds()
|
|
|
|
{
|
|
|
|
bool isInitOK = true;
|
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
//Get Nanoleaf device details and configuration
|
|
|
|
_networkmanager = new QNetworkAccessManager();
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
// Read Panel count and panel Ids
|
|
|
|
QString url = getUrl(_hostname, _api_port, _auth_token, API_ROOT );
|
|
|
|
QJsonDocument doc = getJson( url );
|
2020-02-10 15:21:58 +01:00
|
|
|
if ( this->isInError() )
|
|
|
|
{
|
|
|
|
isInitOK = false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QJsonObject jsonAllPanelInfo = doc.object();
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
QString deviceName = jsonAllPanelInfo[DEV_DATA_NAME].toString();
|
|
|
|
_deviceModel = jsonAllPanelInfo[DEV_DATA_MODEL].toString();
|
|
|
|
QString deviceManufacturer = jsonAllPanelInfo[DEV_DATA_MANUFACTURER].toString();
|
|
|
|
_deviceFirmwareVersion = jsonAllPanelInfo[DEV_DATA_FIRMWAREVERSION].toString();
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
Debug(_log, "Name : %s", QSTRING_CSTR( deviceName ));
|
|
|
|
Debug(_log, "Model : %s", QSTRING_CSTR( _deviceModel ));
|
|
|
|
Debug(_log, "Manufacturer : %s", QSTRING_CSTR( deviceManufacturer ));
|
|
|
|
Debug(_log, "FirmwareVersion: %s", QSTRING_CSTR( _deviceFirmwareVersion));
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
// Get panel details from /panelLayout/layout
|
|
|
|
QJsonObject jsonPanelLayout = jsonAllPanelInfo[API_PANELLAYOUT].toObject();
|
|
|
|
QJsonObject jsonLayout = jsonPanelLayout[PANEL_LAYOUT].toObject();
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
uint panelNum = static_cast<uint>(jsonLayout[PANEL_NUM].toInt());
|
|
|
|
QJsonArray positionData = jsonLayout[PANEL_POSITIONDATA].toArray();
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
std::map<uint, std::map<uint, uint>> panelMap;
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
// Loop over all children.
|
|
|
|
foreach (const QJsonValue & value, positionData)
|
|
|
|
{
|
|
|
|
QJsonObject panelObj = value.toObject();
|
2019-12-08 13:12:01 +01:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
uint panelId = static_cast<uint>(panelObj[PANEL_ID].toInt());
|
|
|
|
uint panelX = static_cast<uint>(panelObj[PANEL_POS_X].toInt());
|
|
|
|
uint panelY = static_cast<uint>(panelObj[PANEL_POS_Y].toInt());
|
|
|
|
uint panelshapeType = static_cast<uint>(panelObj[PANEL_SHAPE_TYPE].toInt());
|
|
|
|
//uint panelOrientation = static_cast<uint>(panelObj[PANEL_ORIENTATION].toInt());
|
2019-12-08 13:12:01 +01:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
DebugIf(verbose, _log, "Panel [%u] (%u,%u) - Type: [%u]", panelId, panelX, panelY, panelshapeType );
|
2019-12-08 13:12:01 +01:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
// Skip Rhythm panels
|
|
|
|
if ( panelshapeType != RHYTM )
|
|
|
|
{
|
|
|
|
panelMap[panelY][panelX] = panelId;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // Reset non support/required features
|
|
|
|
Info(_log, "Rhythm panel skipped.");
|
|
|
|
}
|
2019-12-08 13:12:01 +01:00
|
|
|
}
|
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
// Sort panels top down, left right
|
|
|
|
for(auto posY = panelMap.crbegin(); posY != panelMap.crend(); ++posY)
|
|
|
|
{
|
|
|
|
// posY.first is the first key
|
|
|
|
for(auto const &posX : posY->second)
|
|
|
|
{
|
|
|
|
// posX.first is the second key, posX.second is the data
|
|
|
|
DebugIf(verbose3, _log, "panelMap[%u][%u]=%u", posY->first, posX.first, posX.second );
|
|
|
|
_panelIds.push_back(posX.second);
|
|
|
|
}
|
2019-12-08 13:12:01 +01:00
|
|
|
}
|
2020-02-10 15:21:58 +01:00
|
|
|
this->_panelLedCount = static_cast<uint>(_panelIds.size());
|
|
|
|
_devConfig["hardwareLedCount"] = static_cast<int>(_panelLedCount);
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
Debug(_log, "PanelsNum : %u", panelNum);
|
|
|
|
Debug(_log, "PanelLedCount : %u", _panelLedCount);
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
// Check. if enough panelds were found.
|
|
|
|
uint configuredLedCount = this->getLedCount();
|
|
|
|
if (_panelLedCount < configuredLedCount )
|
|
|
|
{
|
|
|
|
QString errorReason = QString("Not enough panels [%1] for configured LEDs [%2] found!")
|
|
|
|
.arg(_panelLedCount)
|
|
|
|
.arg(configuredLedCount);
|
|
|
|
this->setInError(errorReason);
|
|
|
|
isInitOK = false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ( _panelLedCount > this->getLedCount() )
|
|
|
|
{
|
|
|
|
Warning(_log, "Nanoleaf: More panels [%u] than configured LEDs [%u].", _panelLedCount, configuredLedCount );
|
|
|
|
}
|
2019-12-08 13:12:01 +01:00
|
|
|
}
|
|
|
|
}
|
2020-02-10 15:21:58 +01:00
|
|
|
return isInitOK;
|
|
|
|
}
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
int LedDeviceNanoleaf::open()
|
|
|
|
{
|
|
|
|
int retval = -1;
|
|
|
|
_deviceReady = false;
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
if ( init(_devConfig) )
|
|
|
|
{
|
2020-04-23 19:05:27 +02:00
|
|
|
if ( !initNetwork() )
|
2020-02-10 15:21:58 +01:00
|
|
|
{
|
2020-04-23 19:05:27 +02:00
|
|
|
this->setInError( "UDP Network error!" );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ( initLeds() )
|
|
|
|
{
|
|
|
|
_deviceReady = true;
|
|
|
|
setEnable(true);
|
|
|
|
retval = 0;
|
|
|
|
}
|
2020-02-10 15:21:58 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return retval;
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
2020-03-26 18:49:44 +01:00
|
|
|
bool LedDeviceNanoleaf::discoverDevice()
|
2020-02-10 15:21:58 +01:00
|
|
|
{
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
bool isDeviceFound (false);
|
|
|
|
// device searching by ssdp
|
|
|
|
QString address;
|
|
|
|
SSDPDiscover discover;
|
|
|
|
|
|
|
|
// Discover Canvas device
|
|
|
|
address = discover.getFirstService(STY_WEBSERVER, SSDP_CANVAS, SSDP_TIMEOUT);
|
|
|
|
|
|
|
|
//No Canvas device not found
|
|
|
|
if ( address.isEmpty() ) {
|
|
|
|
// Discover Light Panels (Aurora) device
|
|
|
|
address = discover.getFirstService(STY_WEBSERVER, SSDP_LIGHTPANELS, SSDP_TIMEOUT);
|
|
|
|
|
|
|
|
if ( address.isEmpty() ) {
|
|
|
|
Warning(_log, "No Nanoleaf device discovered");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Canvas or Light Panels found
|
|
|
|
if ( ! address.isEmpty() ) {
|
|
|
|
Info(_log, "Nanoleaf device discovered at [%s]", QSTRING_CSTR( address ));
|
|
|
|
isDeviceFound = true;
|
|
|
|
QStringList addressparts = address.split(":", QString::SkipEmptyParts);
|
|
|
|
_hostname = addressparts[0];
|
|
|
|
_api_port = addressparts[1];
|
|
|
|
}
|
|
|
|
return isDeviceFound;
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
|
|
|
|
QJsonDocument LedDeviceNanoleaf::changeToExternalControlMode()
|
|
|
|
{
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
QString url = getUrl(_hostname, _api_port, _auth_token, API_EFFECT );
|
|
|
|
QJsonDocument jsonDoc;
|
|
|
|
|
|
|
|
_extControlVersion = EXTCTRLVER_V2;
|
|
|
|
//Enable UDP Mode v2
|
|
|
|
jsonDoc= putJson(url, API_EXT_MODE_STRING_V2);
|
|
|
|
|
|
|
|
return jsonDoc;
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
QString LedDeviceNanoleaf::getUrl(QString host, QString port, QString auth_token, QString endpoint) const {
|
2019-09-14 22:54:41 +02:00
|
|
|
return QString(API_URL_FORMAT).arg(host, port, auth_token, endpoint);
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
QJsonDocument LedDeviceNanoleaf::getJson(QString url)
|
|
|
|
{
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
Debug(_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;
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
QJsonDocument LedDeviceNanoleaf::putJson(QString url, QString json)
|
|
|
|
{
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
Debug(_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;
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
QJsonDocument LedDeviceNanoleaf::handleReply(QNetworkReply* const &reply )
|
|
|
|
{
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
QJsonDocument jsonDoc;
|
|
|
|
|
|
|
|
int httpStatusCode = reply->attribute( QNetworkRequest::HttpStatusCodeAttribute ).toInt();
|
|
|
|
Debug(_log, "Reply.httpStatusCode [%d]", httpStatusCode );
|
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
if(reply->error() == QNetworkReply::NoError)
|
2019-12-08 13:12:01 +01:00
|
|
|
{
|
|
|
|
if ( httpStatusCode != 204 ){
|
|
|
|
QByteArray response = reply->readAll();
|
|
|
|
QJsonParseError error;
|
|
|
|
jsonDoc = QJsonDocument::fromJson(response, &error);
|
|
|
|
if (error.error != QJsonParseError::NoError)
|
|
|
|
{
|
2020-02-10 15:21:58 +01:00
|
|
|
this->setInError ( "Got invalid response" );
|
2019-12-08 13:12:01 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
//Debug
|
|
|
|
QString strJson(jsonDoc.toJson(QJsonDocument::Compact));
|
|
|
|
DebugIf(verbose, _log, "Reply: [%s]", strJson.toUtf8().constData() );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
QString errorReason;
|
|
|
|
if ( httpStatusCode > 0 ) {
|
|
|
|
QString httpReason = reply->attribute( QNetworkRequest::HttpReasonPhraseAttribute ).toString();
|
|
|
|
QString advise;
|
|
|
|
switch ( httpStatusCode ) {
|
2020-02-10 15:21:58 +01:00
|
|
|
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;
|
2019-12-08 13:12:01 +01:00
|
|
|
}
|
2020-02-10 15:21:58 +01:00
|
|
|
errorReason = QString ("%1:%2 [%3 %4] - %5").arg(_hostname, _api_port, QString(httpStatusCode) , httpReason, advise);
|
2019-12-08 13:12:01 +01:00
|
|
|
}
|
2019-09-14 22:54:41 +02:00
|
|
|
else {
|
|
|
|
errorReason = QString ("%1:%2 - %3").arg(_hostname, _api_port, reply->errorString());
|
2019-12-08 13:12:01 +01:00
|
|
|
}
|
2020-02-10 15:21:58 +01:00
|
|
|
this->setInError ( errorReason );
|
2019-12-08 13:12:01 +01:00
|
|
|
}
|
|
|
|
// Return response
|
|
|
|
return jsonDoc;
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int LedDeviceNanoleaf::write(const std::vector<ColorRgb> & ledValues)
|
|
|
|
{
|
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
int retVal = 0;
|
|
|
|
uint udpBufferSize;
|
|
|
|
|
|
|
|
//
|
|
|
|
// nPanels 2B
|
|
|
|
// panelID 2B
|
|
|
|
// <R> <G> <B> 3B
|
|
|
|
// <W> 1B
|
|
|
|
// tranitionTime 2B
|
|
|
|
//
|
|
|
|
// Note: Nanoleaf Light Panels (Aurora) now support External Control V2 (tested with FW 3.2.0)
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-09-14 22:54:41 +02:00
|
|
|
udpBufferSize = _panelLedCount * 8 + 2;
|
|
|
|
std::vector<uint8_t> udpbuffer;
|
|
|
|
udpbuffer.resize(udpBufferSize);
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
uchar lowByte; // lower byte
|
|
|
|
uchar highByte; // upper byte
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
uint i=0;
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
// Set number of panels
|
2019-09-14 22:54:41 +02:00
|
|
|
highByte = static_cast<uchar>(_panelLedCount >>8 );
|
|
|
|
lowByte = static_cast<uchar>(_panelLedCount & 0xFF);
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
udpbuffer[i++] = highByte;
|
|
|
|
udpbuffer[i++] = lowByte;
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
ColorRgb color;
|
2019-09-14 22:54:41 +02:00
|
|
|
for ( uint panelCounter=0; panelCounter < _panelLedCount; panelCounter++ )
|
2019-12-08 13:12:01 +01:00
|
|
|
{
|
2019-09-14 22:54:41 +02:00
|
|
|
uint panelID = _panelIds[panelCounter];
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-09-14 22:54:41 +02:00
|
|
|
highByte = static_cast<uchar>(panelID >>8 );
|
|
|
|
lowByte = static_cast<uchar>(panelID & 0xFF);
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
// Set panels configured
|
2020-02-10 15:21:58 +01:00
|
|
|
if( panelCounter < this->getLedCount() ) {
|
2019-09-14 22:54:41 +02:00
|
|
|
color = static_cast<ColorRgb>(ledValues.at(panelCounter));
|
2019-12-08 13:12:01 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Set panels not configed to black;
|
|
|
|
color = ColorRgb::BLACK;
|
|
|
|
DebugIf(verbose3, _log, "[%u] >= panelLedCount [%u] => Set to BLACK", panelCounter, _panelLedCount );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set panelID
|
|
|
|
udpbuffer[i++] = highByte;
|
|
|
|
udpbuffer[i++] = lowByte;
|
|
|
|
|
|
|
|
// Set panel's color LEDs
|
|
|
|
udpbuffer[i++] = color.red;
|
|
|
|
udpbuffer[i++] = color.green;
|
|
|
|
udpbuffer[i++] = color.blue;
|
|
|
|
|
|
|
|
// Set white LED
|
|
|
|
udpbuffer[i++] = 0; // W not set manually
|
|
|
|
|
|
|
|
// Set transition time
|
|
|
|
unsigned char tranitionTime = 1; // currently fixed at value 1 which corresponds to 100ms
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-09-14 22:54:41 +02:00
|
|
|
highByte = static_cast<uchar>(tranitionTime >>8 );
|
|
|
|
lowByte = static_cast<uchar>(tranitionTime & 0xFF);
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
udpbuffer[i++] = highByte;
|
|
|
|
udpbuffer[i++] = lowByte;
|
|
|
|
DebugIf(verbose3, _log, "[%u] Color: {%u,%u,%u}", panelCounter, color.red, color.green, color.blue );
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-12-08 13:12:01 +01:00
|
|
|
}
|
|
|
|
DebugIf(verbose3, _log, "UDP-Address [%s], UDP-Port [%u], udpBufferSize[%u], Bytes to send [%u]", QSTRING_CSTR(_address.toString()), _port, udpBufferSize, i);
|
|
|
|
DebugIf(verbose3, _log, "[%s]", uint8_vector_to_hex_string(udpbuffer).c_str() );
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2019-09-14 22:54:41 +02:00
|
|
|
retVal &= writeBytes( i , udpbuffer.data());
|
2019-12-08 13:12:01 +01:00
|
|
|
DebugIf(verbose3, _log, "writeBytes(): [%d]",retVal);
|
|
|
|
return retVal;
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
QString LedDeviceNanoleaf::getOnOffRequest (bool isOn ) const
|
|
|
|
{
|
2019-12-08 13:12:01 +01:00
|
|
|
QString state = isOn ? STATE_VALUE_TRUE : STATE_VALUE_FALSE;
|
2019-09-14 22:54:41 +02:00
|
|
|
return QString( "{\"%1\":{\"%2\":%3}}" ).arg(STATE_ON, STATE_ONOFF_VALUE, state);
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
int LedDeviceNanoleaf::switchOn()
|
|
|
|
{
|
|
|
|
if ( _deviceReady)
|
|
|
|
{
|
|
|
|
// Set Nanoleaf to External Control (UDP) mode
|
|
|
|
Debug(_log, "Set Nanoleaf to External Control (UDP) streaming mode");
|
|
|
|
QJsonDocument responseDoc = changeToExternalControlMode();
|
|
|
|
// Resolve port for Ligh Panels
|
|
|
|
QJsonObject jsonStreamControllInfo = responseDoc.object();
|
|
|
|
if ( ! jsonStreamControllInfo.isEmpty() ) {
|
|
|
|
_port = static_cast<uchar>(jsonStreamControllInfo[STREAM_CONTROL_PORT].toInt());
|
|
|
|
}
|
2019-09-14 22:54:41 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
//Switch on Nanoleaf device
|
|
|
|
QString url = getUrl(_hostname, _api_port, _auth_token, API_STATE );
|
|
|
|
putJson(url, this->getOnOffRequest(true) );
|
2019-09-14 22:54:41 +02:00
|
|
|
}
|
2019-12-08 13:12:01 +01:00
|
|
|
return 0;
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
int LedDeviceNanoleaf::switchOff()
|
|
|
|
{
|
2019-12-08 13:12:01 +01:00
|
|
|
//Set all LEDs to Black
|
2020-02-10 15:21:58 +01:00
|
|
|
int rc = LedDevice::switchOff();
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
if ( _deviceReady)
|
|
|
|
{
|
|
|
|
//Switch off Nanoleaf device physically
|
|
|
|
QString url = getUrl(_hostname, _api_port, _auth_token, API_STATE );
|
|
|
|
putJson(url, getOnOffRequest(false) );
|
|
|
|
}
|
2019-12-08 13:12:01 +01:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string LedDeviceNanoleaf:: uint8_vector_to_hex_string( const std::vector<uint8_t>& buffer ) const
|
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << std::hex << std::setfill('0');
|
|
|
|
std::vector<uint8_t>::const_iterator it;
|
2019-04-08 23:13:11 +02:00
|
|
|
|
2020-02-10 15:21:58 +01:00
|
|
|
for (it = buffer.begin(); it != buffer.end(); ++it)
|
2019-12-08 13:12:01 +01:00
|
|
|
{
|
|
|
|
ss << " " << std::setw(2) << static_cast<unsigned>(*it);
|
|
|
|
}
|
|
|
|
return ss.str();
|
2019-04-08 23:13:11 +02:00
|
|
|
}
|