etm-powersync-plugins/zigbee-generic-lights/integrationpluginzigbeegenericlights.cpp
2020-11-16 15:15:36 +01:00

1081 lines
54 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*
*
* Copyright 2013 - 2020, nymea GmbH
* Contact: contact@nymea.io
* This file is part of nymea.
* This project including source code and documentation is protected by
* copyright law, and remains the property of nymea GmbH. All rights, including
* reproduction, publication, editing and translation, are reserved. The use of
* this project is subject to the terms of a license agreement to be concluded
* with nymea GmbH in accordance with the terms of use of nymea GmbH, available
* under https://nymea.io/license
*
* GNU Lesser General Public License Usage
* Alternatively, this project may be redistributed and/or modified under the
* terms of the GNU Lesser General Public License as published by the Free
* Software Foundation; version 3. This project is distributed in the hope that
* it will be useful, but WITHOUT ANY WARRANTY; without even the implied
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this project. If not, see <https://www.gnu.org/licenses/>.
*
* For any further details and any questions please contact us under
* contact@nymea.io or see our FAQ/Licensing Information on
* https://nymea.io/license/faq
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
#include "integrationpluginzigbeegenericlights.h"
#include "plugininfo.h"
#include "hardware/zigbee/zigbeehardwareresource.h"
#include <QDebug>
#include <zigbeeutils.h>
IntegrationPluginZigbeeGenericLights::IntegrationPluginZigbeeGenericLights()
{
// Common thing params map
m_ieeeAddressParamTypeIds[onOffLightThingClassId] = onOffLightThingIeeeAddressParamTypeId;
m_ieeeAddressParamTypeIds[dimmableLightThingClassId] = dimmableLightThingIeeeAddressParamTypeId;
m_ieeeAddressParamTypeIds[colorTemperatureLightThingClassId] = colorTemperatureLightThingIeeeAddressParamTypeId;
m_ieeeAddressParamTypeIds[colorLightThingClassId] = colorLightThingIeeeAddressParamTypeId;
m_networkUuidParamTypeIds[onOffLightThingClassId] = onOffLightThingNetworkUuidParamTypeId;
m_networkUuidParamTypeIds[dimmableLightThingClassId] = dimmableLightThingNetworkUuidParamTypeId;
m_networkUuidParamTypeIds[colorTemperatureLightThingClassId] = colorTemperatureLightThingNetworkUuidParamTypeId;
m_networkUuidParamTypeIds[colorLightThingClassId] = colorLightThingNetworkUuidParamTypeId;
m_endpointIdParamTypeIds[onOffLightThingClassId] = onOffLightThingEndpointIdParamTypeId;
m_endpointIdParamTypeIds[dimmableLightThingClassId] = dimmableLightThingEndpointIdParamTypeId;
m_endpointIdParamTypeIds[colorTemperatureLightThingClassId] = colorTemperatureLightThingEndpointIdParamTypeId;
m_endpointIdParamTypeIds[colorLightThingClassId] = colorLightThingEndpointIdParamTypeId;
m_manufacturerIdParamTypeIds[onOffLightThingClassId] = onOffLightThingManufacturerParamTypeId;
m_manufacturerIdParamTypeIds[dimmableLightThingClassId] = dimmableLightThingManufacturerParamTypeId;
m_manufacturerIdParamTypeIds[colorTemperatureLightThingClassId] = colorTemperatureLightThingManufacturerParamTypeId;
m_manufacturerIdParamTypeIds[colorLightThingClassId] = colorLightThingManufacturerParamTypeId;
m_modelIdParamTypeIds[onOffLightThingClassId] = onOffLightThingModelParamTypeId;
m_modelIdParamTypeIds[dimmableLightThingClassId] = dimmableLightThingModelParamTypeId;
m_modelIdParamTypeIds[colorTemperatureLightThingClassId] = colorTemperatureLightThingModelParamTypeId;
m_modelIdParamTypeIds[colorLightThingClassId] = colorLightThingModelParamTypeId;
// Common sates map
m_connectedStateTypeIds[onOffLightThingClassId] = onOffLightConnectedStateTypeId;
m_connectedStateTypeIds[dimmableLightThingClassId] = dimmableLightConnectedStateTypeId;
m_connectedStateTypeIds[colorTemperatureLightThingClassId] = colorTemperatureLightConnectedStateTypeId;
m_connectedStateTypeIds[colorLightThingClassId] = colorLightConnectedStateTypeId;
m_signalStrengthStateTypeIds[onOffLightThingClassId] = onOffLightSignalStrengthStateTypeId;
m_signalStrengthStateTypeIds[dimmableLightThingClassId] = dimmableLightSignalStrengthStateTypeId;
m_signalStrengthStateTypeIds[colorTemperatureLightThingClassId] = colorTemperatureLightSignalStrengthStateTypeId;
m_signalStrengthStateTypeIds[colorLightThingClassId] = colorLightSignalStrengthStateTypeId;
m_versionStateTypeIds[onOffLightThingClassId] = onOffLightVersionStateTypeId;
m_versionStateTypeIds[dimmableLightThingClassId] = dimmableLightVersionStateTypeId;
m_versionStateTypeIds[colorTemperatureLightThingClassId] = colorTemperatureLightVersionStateTypeId;
m_versionStateTypeIds[colorLightThingClassId] = colorLightVersionStateTypeId;
}
QString IntegrationPluginZigbeeGenericLights::name() const
{
return "Generic lights";
}
bool IntegrationPluginZigbeeGenericLights::handleNode(ZigbeeNode *node, const QUuid &networkUuid)
{
foreach (ZigbeeNodeEndpoint *endpoint, node->endpoints()) {
if ((endpoint->profile() == Zigbee::ZigbeeProfile::ZigbeeProfileLightLink &&
endpoint->deviceId() == Zigbee::LightLinkDevice::LightLinkDeviceOnOffLight) ||
(endpoint->profile() == Zigbee::ZigbeeProfile::ZigbeeProfileHomeAutomation &&
endpoint->deviceId() == Zigbee::HomeAutomationDeviceOnOffLight)) {
qCDebug(dcZigbeeGenericLights()) << "Handeling on/off light for" << node << endpoint;
createLightThing(onOffLightThingClassId, networkUuid, node, endpoint);
return true;
}
if ((endpoint->profile() == Zigbee::ZigbeeProfile::ZigbeeProfileLightLink &&
endpoint->deviceId() == Zigbee::LightLinkDevice::LightLinkDeviceDimmableLight) ||
(endpoint->profile() == Zigbee::ZigbeeProfile::ZigbeeProfileHomeAutomation &&
endpoint->deviceId() == Zigbee::HomeAutomationDeviceDimmableLight)) {
qCDebug(dcZigbeeGenericLights()) << "Handeling dimmable light for" << node << endpoint;
createLightThing(dimmableLightThingClassId, networkUuid, node, endpoint);
return true;
}
if ((endpoint->profile() == Zigbee::ZigbeeProfileLightLink &&
endpoint->deviceId() == Zigbee::LightLinkDeviceColourTemperatureLight) ||
(endpoint->profile() == Zigbee::ZigbeeProfileHomeAutomation &&
endpoint->deviceId() == Zigbee::HomeAutomationDeviceColourTemperatureLight)) {
qCDebug(dcZigbeeGenericLights()) << "Handeling color temperature light for" << node << endpoint;
createLightThing(colorTemperatureLightThingClassId, networkUuid, node, endpoint);
return true;
}
if ((endpoint->profile() == Zigbee::ZigbeeProfileLightLink && endpoint->deviceId() == Zigbee::LightLinkDeviceColourLight) ||
(endpoint->profile() == Zigbee::ZigbeeProfileLightLink && endpoint->deviceId() == Zigbee::LightLinkDeviceExtendedColourLight) ||
(endpoint->profile() == Zigbee::ZigbeeProfileHomeAutomation && endpoint->deviceId() == Zigbee::HomeAutomationDeviceExtendedColourLight)) {
qCDebug(dcZigbeeGenericLights()) << "Handeling color light for" << node << endpoint;
createLightThing(colorLightThingClassId, networkUuid, node, endpoint);
return true;
}
}
return false;
}
void IntegrationPluginZigbeeGenericLights::handleRemoveNode(ZigbeeNode *node, const QUuid &networkUuid)
{
Q_UNUSED(networkUuid)
if (m_thingNodes.values().contains(node)) {
Thing *thing = m_thingNodes.key(node);
qCDebug(dcZigbeeGenericLights()) << node << "for" << thing << "has left the network.";
m_thingNodes.remove(thing);
emit autoThingDisappeared(thing->id());
if (m_colorTemperatureRanges.contains(thing)) {
m_colorTemperatureRanges.remove(thing);
}
}
}
void IntegrationPluginZigbeeGenericLights::init()
{
hardwareManager()->zigbeeResource()->registerHandler(this, ZigbeeHardwareResource::HandlerTypeCatchAll);
}
void IntegrationPluginZigbeeGenericLights::setupThing(ThingSetupInfo *info)
{
Thing *thing = info->thing();
// Get the node for this thing
QUuid networkUuid = thing->paramValue(m_networkUuidParamTypeIds.value(thing->thingClassId())).toUuid();
ZigbeeAddress zigbeeAddress = ZigbeeAddress(thing->paramValue(m_ieeeAddressParamTypeIds.value(thing->thingClassId())).toString());
ZigbeeNode *node = hardwareManager()->zigbeeResource()->claimNode(this, networkUuid, zigbeeAddress);
if (!node) {
qCWarning(dcZigbeeGenericLights()) << "Zigbee node for" << info->thing()->name() << "not found.´";
info->finish(Thing::ThingErrorHardwareNotAvailable);
return;
}
// Store the node thing association for removing
m_thingNodes.insert(thing, node);
// Get the endpoint for this thing
ZigbeeNodeEndpoint *endpoint = findEndpoint(thing);
if (!endpoint) {
qCWarning(dcZigbeeGenericLights()) << "Unable to get the endpoint from node" << node << "for" << thing;
info->finish(Thing::ThingErrorSetupFailed);
return;
}
// General thing setup
// Update connected state
thing->setStateValue(m_connectedStateTypeIds.value(thing->thingClassId()), node->reachable());
connect(node, &ZigbeeNode::reachableChanged, thing, [thing, this](bool reachable){
thing->setStateValue(m_connectedStateTypeIds.value(thing->thingClassId()), reachable);
});
// Update signal strength
thing->setStateValue(m_signalStrengthStateTypeIds.value(thing->thingClassId()), qRound(node->lqi() * 100.0 / 255.0));
connect(node, &ZigbeeNode::lqiChanged, thing, [this, thing](quint8 lqi){
uint signalStrength = qRound(lqi * 100.0 / 255.0);
qCDebug(dcZigbeeGenericLights()) << thing << "signal strength changed" << signalStrength << "%";
thing->setStateValue(m_signalStrengthStateTypeIds.value(thing->thingClassId()), signalStrength);
});
// Set the version
thing->setStateValue(m_versionStateTypeIds.value(thing->thingClassId()), endpoint->softwareBuildId());
// Thing specific setup
// On/Off light
if (thing->thingClassId() == onOffLightThingClassId) {
// Get the on/off cluster
ZigbeeClusterOnOff *onOffCluster = endpoint->inputCluster<ZigbeeClusterOnOff>(ZigbeeClusterLibrary::ClusterIdOnOff);
if (!onOffCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find on/off cluster for" << thing << "in" << node;
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
if (onOffCluster->hasAttribute(ZigbeeClusterOnOff::AttributeOnOff)) {
thing->setStateValue(onOffLightPowerStateTypeId, onOffCluster->power());
}
// Update the power state if the node power value changes
connect(onOffCluster, &ZigbeeClusterOnOff::powerChanged, thing, [thing](bool power){
qCDebug(dcZigbeeGenericLights()) << thing << "power state changed" << power;
thing->setStateValue(onOffLightPowerStateTypeId, power);
});
// Read the power state once the node gets reachable
connect(node, &ZigbeeNode::reachableChanged, thing, [thing, this](bool reachable){
if (reachable) {
readLightPowerState(thing);
}
});
}
// Dimmable light
if (thing->thingClassId() == dimmableLightThingClassId) {
// Get the on/off cluster
ZigbeeClusterOnOff *onOffCluster = endpoint->inputCluster<ZigbeeClusterOnOff>(ZigbeeClusterLibrary::ClusterIdOnOff);
if (!onOffCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find on/off cluster for" << thing << "in" << node;
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Only set the state if the cluster actually has the attribute
if (onOffCluster->hasAttribute(ZigbeeClusterOnOff::AttributeOnOff)) {
thing->setStateValue(dimmableLightPowerStateTypeId, onOffCluster->power());
}
// Update the power state if the node power value changes
connect(onOffCluster, &ZigbeeClusterOnOff::powerChanged, thing, [thing](bool power){
qCDebug(dcZigbeeGenericLights()) << thing << "power state changed" << power;
thing->setStateValue(dimmableLightPowerStateTypeId, power);
});
// Get the level cluster
ZigbeeClusterLevelControl *levelCluster = endpoint->inputCluster<ZigbeeClusterLevelControl>(ZigbeeClusterLibrary::ClusterIdLevelControl);
if (!levelCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find level cluster for" << thing << "in" << node;
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Only set the state if the cluster actually has the attribute
if (levelCluster->hasAttribute(ZigbeeClusterLevelControl::AttributeCurrentLevel)) {
int percentage = qRound(levelCluster->currentLevel() * 100.0 / 255.0);
thing->setStateValue(dimmableLightBrightnessStateTypeId, percentage);
}
connect(levelCluster, &ZigbeeClusterLevelControl::currentLevelChanged, thing, [thing](quint8 level){
int percentage = qRound(level * 100.0 / 255.0);
qCDebug(dcZigbeeGenericLights()) << thing << "level state changed" << level << percentage << "%";
thing->setStateValue(dimmableLightBrightnessStateTypeId, percentage);
});
// Read the states once the node gets reachable
connect(node, &ZigbeeNode::reachableChanged, thing, [thing, this](bool reachable){
if (reachable) {
readLightPowerState(thing);
readLightLevelState(thing);
}
});
}
// Color temperature light
if (thing->thingClassId() == colorTemperatureLightThingClassId) {
// Get the on/off cluster
ZigbeeClusterOnOff *onOffCluster = endpoint->inputCluster<ZigbeeClusterOnOff>(ZigbeeClusterLibrary::ClusterIdOnOff);
if (!onOffCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find on/off cluster for" << thing << "in" << node;
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Only set the state if the cluster actually has the attribute
if (onOffCluster->hasAttribute(ZigbeeClusterOnOff::AttributeOnOff)) {
thing->setStateValue(colorTemperatureLightPowerStateTypeId, onOffCluster->power());
}
// Update the power state if the node power value changes
connect(onOffCluster, &ZigbeeClusterOnOff::powerChanged, thing, [thing](bool power){
qCDebug(dcZigbeeGenericLights()) << thing << "power state changed" << power;
thing->setStateValue(colorTemperatureLightPowerStateTypeId, power);
});
// Get the level cluster
ZigbeeClusterLevelControl *levelCluster = endpoint->inputCluster<ZigbeeClusterLevelControl>(ZigbeeClusterLibrary::ClusterIdLevelControl);
if (!levelCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find level cluster for" << thing << "in" << node;
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Only set the state if the cluster actually has the attribute
if (levelCluster->hasAttribute(ZigbeeClusterLevelControl::AttributeCurrentLevel)) {
int percentage = qRound(levelCluster->currentLevel() * 100.0 / 255.0);
thing->setStateValue(colorTemperatureLightBrightnessStateTypeId, percentage);
}
connect(levelCluster, &ZigbeeClusterLevelControl::currentLevelChanged, thing, [thing](quint8 level){
int percentage = qRound(level * 100.0 / 255.0);
qCDebug(dcZigbeeGenericLights()) << thing << "level state changed" << level << percentage << "%";
thing->setStateValue(colorTemperatureLightBrightnessStateTypeId, percentage);
});
// Get color cluster
ZigbeeClusterColorControl *colorCluster = endpoint->inputCluster<ZigbeeClusterColorControl>(ZigbeeClusterLibrary::ClusterIdColorControl);
if (!colorCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find color cluster for" << thing << "in" << node;
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Only set the state if the cluster actually has the attribute
if (colorCluster->hasAttribute(ZigbeeClusterColorControl::AttributeColorTemperatureMireds)) {
int mappedValue = mapColorTemperatureToScaledValue(thing, colorCluster->colorTemperatureMireds());
thing->setStateValue(colorTemperatureLightColorTemperatureStateTypeId, mappedValue);
}
connect(colorCluster, &ZigbeeClusterColorControl::colorTemperatureMiredsChanged, thing, [this, thing](quint16 colorTemperatureMired){
qCDebug(dcZigbeeGenericLights()) << "Actual color temperature is" << colorTemperatureMired << "mireds";
int mappedValue = mapColorTemperatureToScaledValue(thing, colorTemperatureMired);
qCDebug(dcZigbeeGenericLights()) << "Mapped color temperature is" << mappedValue;
thing->setStateValue(colorTemperatureLightColorTemperatureStateTypeId, mappedValue);
});
// Read the states once the node gets reachable
connect(node, &ZigbeeNode::reachableChanged, thing, [thing, this](bool reachable){
if (reachable) {
readColorTemperatureRange(thing);
readLightPowerState(thing);
readLightLevelState(thing);
readLightColorTemperatureState(thing);
}
});
}
// Color temperature light
if (thing->thingClassId() == colorLightThingClassId) {
// Get the on/off cluster
ZigbeeClusterOnOff *onOffCluster = endpoint->inputCluster<ZigbeeClusterOnOff>(ZigbeeClusterLibrary::ClusterIdOnOff);
if (!onOffCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find on/off cluster for" << thing << "in" << node;
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Only set the state if the cluster actually has the attribute
if (onOffCluster->hasAttribute(ZigbeeClusterOnOff::AttributeOnOff)) {
thing->setStateValue(colorLightPowerStateTypeId, onOffCluster->power());
}
// Update the power state if the node power value changes
connect(onOffCluster, &ZigbeeClusterOnOff::powerChanged, thing, [thing](bool power){
qCDebug(dcZigbeeGenericLights()) << thing << "power state changed" << power;
thing->setStateValue(colorLightPowerStateTypeId, power);
});
// Get the level cluster
ZigbeeClusterLevelControl *levelCluster = endpoint->inputCluster<ZigbeeClusterLevelControl>(ZigbeeClusterLibrary::ClusterIdLevelControl);
if (!levelCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find level cluster for" << thing << "in" << node;
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Only set the state if the cluster actually has the attribute
if (levelCluster->hasAttribute(ZigbeeClusterLevelControl::AttributeCurrentLevel)) {
int percentage = qRound(levelCluster->currentLevel() * 100.0 / 255.0);
thing->setStateValue(colorLightBrightnessStateTypeId, percentage);
}
connect(levelCluster, &ZigbeeClusterLevelControl::currentLevelChanged, thing, [thing](quint8 level){
int percentage = qRound(level * 100.0 / 255.0);
qCDebug(dcZigbeeGenericLights()) << thing << "level state changed" << level << percentage << "%";
thing->setStateValue(colorLightBrightnessStateTypeId, percentage);
});
// Get color cluster
ZigbeeClusterColorControl *colorCluster = endpoint->inputCluster<ZigbeeClusterColorControl>(ZigbeeClusterLibrary::ClusterIdColorControl);
if (!colorCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find color cluster for" << thing << "in" << node;
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Only set the state if the cluster actually has the attribute
if (colorCluster->hasAttribute(ZigbeeClusterColorControl::AttributeColorTemperatureMireds)) {
int mappedValue = mapColorTemperatureToScaledValue(thing, colorCluster->colorTemperatureMireds());
thing->setStateValue(colorLightColorTemperatureStateTypeId, mappedValue);
}
connect(colorCluster, &ZigbeeClusterColorControl::colorTemperatureMiredsChanged, thing, [this, thing](quint16 colorTemperatureMired){
qCDebug(dcZigbeeGenericLights()) << "Actual color temperature is" << colorTemperatureMired << "mireds";
int mappedValue = mapColorTemperatureToScaledValue(thing, colorTemperatureMired);
qCDebug(dcZigbeeGenericLights()) << "Mapped color temperature is" << mappedValue;
thing->setStateValue(colorLightColorTemperatureStateTypeId, mappedValue);
});
// Read the states once the node gets reachable
connect(node, &ZigbeeNode::reachableChanged, thing, [thing, this](bool reachable){
if (reachable) {
// Note: this will also read the color temperature range if available
readColorCapabilities(thing);
readLightPowerState(thing);
readLightLevelState(thing);
readLightColorXYState(thing);
}
});
}
info->finish(Thing::ThingErrorNoError);
}
void IntegrationPluginZigbeeGenericLights::executeAction(ThingActionInfo *info)
{
if (!hardwareManager()->zigbeeResource()->available()) {
info->finish(Thing::ThingErrorHardwareNotAvailable);
return;
}
// Get the node
Thing *thing = info->thing();
ZigbeeNode *node = m_thingNodes.value(thing);
if (!node->reachable()) {
info->finish(Thing::ThingErrorHardwareNotAvailable);
return;
}
// Get the endpoint
ZigbeeNodeEndpoint *endpoint = findEndpoint(thing);
if (!endpoint) {
info->finish(Thing::ThingErrorHardwareNotAvailable);
return;
}
// On/Off light
if (thing->thingClassId() == onOffLightThingClassId) {
if (info->action().actionTypeId() == onOffLightAlertActionTypeId) {
executeAlertAction(info, endpoint);
return;
}
if (info->action().actionTypeId() == onOffLightPowerActionTypeId) {
// Get the on/off server cluster from the endpoint
bool power = info->action().param(onOffLightPowerActionPowerParamTypeId).value().toBool();
executePowerAction(info, endpoint, onOffLightPowerStateTypeId, power);
return;
}
}
// Dimmable light
if (thing->thingClassId() == dimmableLightThingClassId) {
if (info->action().actionTypeId() == dimmableLightAlertActionTypeId) {
executeAlertAction(info, endpoint);
return;
}
if (info->action().actionTypeId() == dimmableLightPowerActionTypeId) {
bool power = info->action().param(dimmableLightPowerActionPowerParamTypeId).value().toBool();
executePowerAction(info, endpoint, dimmableLightPowerStateTypeId, power);
return;
}
if (info->action().actionTypeId() == dimmableLightBrightnessActionTypeId) {
int brightness = info->action().param(dimmableLightBrightnessActionBrightnessParamTypeId).value().toInt();
quint8 level = static_cast<quint8>(qRound(255.0 * brightness / 100.0));
executeBrightnessAction(info, endpoint, dimmableLightPowerStateTypeId, dimmableLightBrightnessStateTypeId, brightness, level);
return;
}
}
// Color temperature light
if (thing->thingClassId() == colorTemperatureLightThingClassId) {
if (info->action().actionTypeId() == colorTemperatureLightAlertActionTypeId) {
executeAlertAction(info, endpoint);
return;
}
if (info->action().actionTypeId() == colorTemperatureLightPowerActionTypeId) {
bool power = info->action().param(colorTemperatureLightPowerActionPowerParamTypeId).value().toBool();
executePowerAction(info, endpoint, colorTemperatureLightPowerStateTypeId, power);
return;
}
if (info->action().actionTypeId() == colorTemperatureLightBrightnessActionTypeId) {
int brightness = info->action().param(colorTemperatureLightBrightnessActionBrightnessParamTypeId).value().toInt();
quint8 level = static_cast<quint8>(qRound(255.0 * brightness / 100.0));
executeBrightnessAction(info, endpoint, colorTemperatureLightPowerStateTypeId, colorTemperatureLightBrightnessStateTypeId, brightness, level);
return;
}
if (info->action().actionTypeId() == colorTemperatureLightColorTemperatureActionTypeId) {
int colorTemperatureScaled = info->action().param(colorTemperatureLightColorTemperatureActionColorTemperatureParamTypeId).value().toInt();
executeColorTemperatureAction(info, endpoint, colorTemperatureLightColorTemperatureStateTypeId, colorTemperatureScaled);
return;
}
}
// Color light
if (thing->thingClassId() == colorLightThingClassId) {
if (info->action().actionTypeId() == colorLightAlertActionTypeId) {
executeAlertAction(info, endpoint);
return;
}
if (info->action().actionTypeId() == colorLightPowerActionTypeId) {
bool power = info->action().param(colorLightPowerActionPowerParamTypeId).value().toBool();
executePowerAction(info, endpoint, colorLightPowerStateTypeId, power);
return;
}
if (info->action().actionTypeId() == colorLightBrightnessActionTypeId) {
int brightness = info->action().param(colorLightBrightnessActionBrightnessParamTypeId).value().toInt();
quint8 level = static_cast<quint8>(qRound(255.0 * brightness / 100.0));
executeBrightnessAction(info, endpoint, colorLightPowerStateTypeId, colorLightBrightnessStateTypeId, brightness, level);
return;
}
if (info->action().actionTypeId() == colorLightColorTemperatureActionTypeId) {
int colorTemperatureScaled = info->action().param(colorLightColorTemperatureActionColorTemperatureParamTypeId).value().toInt();
if (m_colorCapabilities[thing].testFlag(ZigbeeClusterColorControl::ColorCapabilityColorTemperature)) {
// Native color temperature available
executeColorTemperatureAction(info, endpoint, colorLightColorTemperatureStateTypeId, colorTemperatureScaled);
return;
}
// Note: there is no color temperature capability, we have to emulate the color using default min/max values
// Convert the scaled value into the min/max color temperature interval
quint16 colorTemperature = mapScaledValueToColorTemperature(thing, colorTemperatureScaled);
qCDebug(dcZigbeeGenericLights()) << "Mapping action value" << colorTemperatureScaled << "to the color temperature in the range of [" << m_colorTemperatureRanges[thing].minValue << "," << m_colorTemperatureRanges[thing].maxValue << "] -->" << colorTemperature << "mired";
// Note: the color temperature command/attribute is not supported. Using xy colors to interpolate the temperature colors
QColor temperatureColor = ZigbeeUtils::interpolateColorFromColorTemperature(colorTemperature, m_colorTemperatureRanges[thing].minValue, m_colorTemperatureRanges[thing].maxValue);
QPoint temperatureColorXyInt = ZigbeeUtils::convertColorToXYInt(temperatureColor);
qCDebug(dcZigbeeGenericLights()) << "Mapping interpolated value" << temperatureColor << "mired to the xy color" << temperatureColorXyInt;
// Set color
ZigbeeClusterColorControl *colorCluster = endpoint->inputCluster<ZigbeeClusterColorControl>(ZigbeeClusterLibrary::ClusterIdColorControl);
if (!colorCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find color control cluster for" << thing << "in" << m_thingNodes.value(thing);
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
ZigbeeClusterReply *reply = colorCluster->commandMoveToColor(temperatureColorXyInt.x(), temperatureColorXyInt.y(), 2);
connect(reply, &ZigbeeClusterReply::finished, info, [=](){
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
info->finish(Thing::ThingErrorHardwareFailure);
} else {
qCDebug(dcZigbeeGenericLights()) << "Set color temperature" << colorTemperature << "mired finished successfully" << "(scalled" << colorTemperatureScaled << ")";
thing->setStateValue(colorLightColorTemperatureStateTypeId, colorTemperatureScaled);
info->finish(Thing::ThingErrorNoError);
}
});
return;
}
if (info->action().actionTypeId() == colorLightColorActionTypeId) {
QColor color = info->action().param(colorLightColorActionColorParamTypeId).value().value<QColor>();
QPoint xyColorInt = ZigbeeUtils::convertColorToXYInt(color);
qCDebug(dcZigbeeGenericLights()) << "Set color" << color.toRgb() << xyColorInt;
executeColorAction(info, endpoint, colorLightColorStateTypeId, color);
return;
}
}
info->finish(Thing::ThingErrorUnsupportedFeature);
}
void IntegrationPluginZigbeeGenericLights::thingRemoved(Thing *thing)
{
ZigbeeNode *node = m_thingNodes.take(thing);
if (node) {
QUuid networkUuid = thing->paramValue(m_networkUuidParamTypeIds.value(thing->thingClassId())).toUuid();
hardwareManager()->zigbeeResource()->removeNodeFromNetwork(networkUuid, node);
}
if (m_colorTemperatureRanges.contains(thing)) {
m_colorTemperatureRanges.remove(thing);
}
}
ZigbeeNodeEndpoint *IntegrationPluginZigbeeGenericLights::findEndpoint(Thing *thing)
{
ZigbeeNode *node = m_thingNodes.value(thing);
if (!node) {
qCWarning(dcZigbeeGenericLights()) << "Could not find the node for" << thing;
return nullptr;
}
quint8 endpointId = thing->paramValue(m_endpointIdParamTypeIds.value(thing->thingClassId())).toUInt();
return node->getEndpoint(endpointId);
}
void IntegrationPluginZigbeeGenericLights::createLightThing(const ThingClassId &thingClassId, const QUuid &networkUuid, ZigbeeNode *node, ZigbeeNodeEndpoint *endpoint)
{
ThingDescriptor descriptor(thingClassId);
QString deviceClassName = supportedThings().findById(thingClassId).displayName();
descriptor.setTitle(QString("%1 (%2 - %3)").arg(deviceClassName).arg(endpoint->manufacturerName()).arg(endpoint->modelIdentifier()));
ParamList params;
params.append(Param(m_networkUuidParamTypeIds[thingClassId], networkUuid.toString()));
params.append(Param(m_ieeeAddressParamTypeIds[thingClassId], node->extendedAddress().toString()));
params.append(Param(m_endpointIdParamTypeIds[thingClassId], endpoint->endpointId()));
params.append(Param(m_modelIdParamTypeIds[thingClassId], endpoint->modelIdentifier()));
params.append(Param(m_manufacturerIdParamTypeIds[thingClassId], endpoint->manufacturerName()));
descriptor.setParams(params);
emit autoThingsAppeared({descriptor});
}
void IntegrationPluginZigbeeGenericLights::executeAlertAction(ThingActionInfo *info, ZigbeeNodeEndpoint *endpoint)
{
Thing *thing = info->thing();
ZigbeeClusterIdentify *identifyCluster = endpoint->inputCluster<ZigbeeClusterIdentify>(ZigbeeClusterLibrary::ClusterIdIdentify);
if (!identifyCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find identify cluster for" << thing << "in" << m_thingNodes.value(thing);
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Send the command trough the network
ZigbeeClusterReply *reply = identifyCluster->identify(2);
connect(reply, &ZigbeeClusterReply::finished, this, [reply, info](){
// Note: reply will be deleted automatically
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
info->finish(Thing::ThingErrorHardwareFailure);
} else {
info->finish(Thing::ThingErrorNoError);
}
});
}
void IntegrationPluginZigbeeGenericLights::executePowerAction(ThingActionInfo *info, ZigbeeNodeEndpoint *endpoint, const StateTypeId &powerStateTypeId, bool power)
{
Thing *thing = info->thing();
ZigbeeClusterOnOff *onOffCluster = endpoint->inputCluster<ZigbeeClusterOnOff>(ZigbeeClusterLibrary::ClusterIdOnOff);
if (!onOffCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find on/off cluster for" << thing << "in" << m_thingNodes.value(thing);
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Send the command trough the network
qCDebug(dcZigbeeGenericLights()) << "Set power for" << info->thing() << "to" << power;
ZigbeeClusterReply *reply = (power ? onOffCluster->commandOn() : onOffCluster->commandOff());
connect(reply, &ZigbeeClusterReply::finished, info, [=](){
// Note: reply will be deleted automatically
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
qCWarning(dcZigbeeGenericLights()) << "Failed to set power on" << thing << reply->error();
info->finish(Thing::ThingErrorHardwareFailure);
} else {
info->finish(Thing::ThingErrorNoError);
qCDebug(dcZigbeeGenericLights()) << "Set power finished successfully for" << thing;
thing->setStateValue(powerStateTypeId, power);
}
});
}
void IntegrationPluginZigbeeGenericLights::executeBrightnessAction(ThingActionInfo *info, ZigbeeNodeEndpoint *endpoint, const StateTypeId &powerStateTypeId, const StateTypeId &brightnessStateTypeId, int brightness, quint8 level)
{
Thing *thing = info->thing();
ZigbeeClusterLevelControl *levelCluster = endpoint->inputCluster<ZigbeeClusterLevelControl>(ZigbeeClusterLibrary::ClusterIdLevelControl);
if (!levelCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find level control cluster for" << thing << "in" << m_thingNodes.value(thing);
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
ZigbeeClusterReply *reply = levelCluster->commandMoveToLevelWithOnOff(level);
connect(reply, &ZigbeeClusterReply::finished, info, [=](){
// Note: reply will be deleted automatically
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
info->finish(Thing::ThingErrorHardwareFailure);
} else {
info->finish(Thing::ThingErrorNoError);
thing->setStateValue(powerStateTypeId, (brightness > 0 ? true : false));
thing->setStateValue(brightnessStateTypeId, brightness);
}
});
}
void IntegrationPluginZigbeeGenericLights::executeColorTemperatureAction(ThingActionInfo *info, ZigbeeNodeEndpoint *endpoint, const StateTypeId &colorTemperatureStateTypeId, int colorTemperatureScaled)
{
Thing *thing = info->thing();
ZigbeeClusterColorControl *colorCluster = endpoint->inputCluster<ZigbeeClusterColorControl>(ZigbeeClusterLibrary::ClusterIdColorControl);
if (!colorCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find color control cluster for" << thing << "in" << m_thingNodes.value(thing);
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Scale the value to the actual color temperature of this lamp
quint16 colorTemperature = mapScaledValueToColorTemperature(thing, colorTemperatureScaled);
// Note: time unit is 1/10 s
ZigbeeClusterReply *reply = colorCluster->commandMoveToColorTemperature(colorTemperature, 5);
connect(reply, &ZigbeeClusterReply::finished, info, [=](){
// Note: reply will be deleted automatically
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
info->finish(Thing::ThingErrorHardwareFailure);
} else {
info->finish(Thing::ThingErrorNoError);
thing->setStateValue(colorTemperatureStateTypeId, colorTemperatureScaled);
}
});
}
void IntegrationPluginZigbeeGenericLights::executeColorAction(ThingActionInfo *info, ZigbeeNodeEndpoint *endpoint, const StateTypeId &colorStateTypeId, const QColor &color)
{
Thing *thing = info->thing();
ZigbeeClusterColorControl *colorCluster = endpoint->inputCluster<ZigbeeClusterColorControl>(ZigbeeClusterLibrary::ClusterIdColorControl);
if (!colorCluster) {
qCWarning(dcZigbeeGenericLights()) << "Could not find color control cluster for" << thing << "in" << m_thingNodes.value(thing);
info->finish(Thing::ThingErrorHardwareFailure);
return;
}
// Note: time unit is 1/10 s
QPoint colorXyInt = ZigbeeUtils::convertColorToXYInt(color);
ZigbeeClusterReply *reply = colorCluster->commandMoveToColor(colorXyInt.x(), colorXyInt.y(), 2);
connect(reply, &ZigbeeClusterReply::finished, info, [=](){
// Note: reply will be deleted automatically
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
info->finish(Thing::ThingErrorHardwareFailure);
} else {
info->finish(Thing::ThingErrorNoError);
thing->setStateValue(colorStateTypeId, color);
}
});
}
void IntegrationPluginZigbeeGenericLights::readLightPowerState(Thing *thing)
{
// Get the node
ZigbeeNode *node = m_thingNodes.value(thing);
if (!node->reachable())
return;
// Get the endpoint
ZigbeeNodeEndpoint *endpoint = findEndpoint(thing);
if (!endpoint)
return;
// Get the on/off server cluster from the endpoint
ZigbeeClusterOnOff *onOffCluster = endpoint->inputCluster<ZigbeeClusterOnOff>(ZigbeeClusterLibrary::ClusterIdOnOff);
if (!onOffCluster)
return;
qCDebug(dcZigbeeGenericLights()) << "Reading on/off power value for" << thing << "on" << node;
ZigbeeClusterReply *reply = onOffCluster->readAttributes({ZigbeeClusterOnOff::AttributeOnOff});
connect(reply, &ZigbeeClusterReply::finished, thing, [thing, reply](){
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read on/off cluster attribute from" << thing << reply->error();
return;
}
});
}
void IntegrationPluginZigbeeGenericLights::readLightLevelState(Thing *thing)
{
// Get the node
ZigbeeNode *node = m_thingNodes.value(thing);
if (!node->reachable())
return;
// Get the endpoint
ZigbeeNodeEndpoint *endpoint = findEndpoint(thing);
if (!endpoint)
return;
// Get the level server cluster from the endpoint
ZigbeeClusterLevelControl *levelCluster = endpoint->inputCluster<ZigbeeClusterLevelControl>(ZigbeeClusterLibrary::ClusterIdLevelControl);
if (!levelCluster)
return;
qCDebug(dcZigbeeGenericLights()) << "Reading level value for" << thing << "on" << node;
ZigbeeClusterReply *reply = levelCluster->readAttributes({ZigbeeClusterLevelControl::AttributeCurrentLevel});
connect(reply, &ZigbeeClusterReply::finished, thing, [thing, reply](){
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read level cluster attribute from" << thing << reply->error();
return;
}
});
}
void IntegrationPluginZigbeeGenericLights::readLightColorTemperatureState(Thing *thing)
{
ZigbeeNodeEndpoint *endpoint = findEndpoint(thing);
if (!endpoint) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color temperature for" << thing << "because the node could not be found";
return;
}
// Get the color server cluster from the endpoint
ZigbeeClusterColorControl *colorCluster = endpoint->inputCluster<ZigbeeClusterColorControl>(ZigbeeClusterLibrary::ClusterIdColorControl);
if (!colorCluster) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color temperature for" << thing << "because the color cluster could not be found on" << endpoint;
return;
}
ZigbeeClusterReply *reply = colorCluster->readAttributes({ZigbeeClusterColorControl::AttributeColorTemperatureMireds});
connect(reply, &ZigbeeClusterReply::finished, thing, [=](){
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read ColorControl cluster attribute color temperature" << reply->error();
return;
}
});
}
void IntegrationPluginZigbeeGenericLights::readLightColorXYState(Thing *thing)
{
ZigbeeNodeEndpoint *endpoint = findEndpoint(thing);
if (!endpoint) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color xy for" << thing << "because the node could not be found";
return;
}
// Get the color server cluster from the endpoint
ZigbeeClusterColorControl *colorCluster = endpoint->inputCluster<ZigbeeClusterColorControl>(ZigbeeClusterLibrary::ClusterIdColorControl);
if (!colorCluster) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color xy for" << thing << "because the color cluster could not be found on" << endpoint;
return;
}
ZigbeeClusterReply *reply = colorCluster->readAttributes({ZigbeeClusterColorControl::AttributeCurrentX, ZigbeeClusterColorControl::AttributeCurrentY});
connect(reply, &ZigbeeClusterReply::finished, thing, [=](){
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read ColorControl cluster attribute" << reply->error();
return;
}
// Note: the attribute gets updated internally and the state gets updated with the currentLevelChanged signal
qCDebug(dcZigbeeGenericLights()) << "Reading ColorControl cluster attribute color x and y finished successfully";
QList<ZigbeeClusterLibrary::ReadAttributeStatusRecord> attributeStatusRecords = ZigbeeClusterLibrary::parseAttributeStatusRecords(reply->responseFrame().payload);
if (attributeStatusRecords.count() != 2) {
qCWarning(dcZigbeeGenericLights()) << "Did not receive color x and y attribute values from" << thing;
return;
}
// Parse the attribute status records and calculate the color
quint16 currentX = 0; quint16 currentY = 0;
foreach (const ZigbeeClusterLibrary::ReadAttributeStatusRecord &attributeStatusRecord, attributeStatusRecords) {
qCDebug(dcZigbeeGenericLights()) << "Received read attribute status record" << thing << attributeStatusRecord;
if (attributeStatusRecord.attributeId == ZigbeeClusterColorControl::AttributeCurrentX) {
bool valueOk = false;
currentX = attributeStatusRecord.dataType.toUInt16(&valueOk);
if (!valueOk) {
qCWarning(dcZigbeeGenericLights()) << "Failed to convert color x attribute values from" << thing << attributeStatusRecord;
return;
}
continue;
}
if (attributeStatusRecord.attributeId == ZigbeeClusterColorControl::AttributeCurrentY) {
bool valueOk = false;
currentY = attributeStatusRecord.dataType.toUInt16(&valueOk);
if (!valueOk) {
qCWarning(dcZigbeeGenericLights()) << "Failed to convert color y attribute values from" << thing << attributeStatusRecord;
return;
}
continue;
}
}
// Set the current color
QColor color = ZigbeeUtils::convertXYToColor(currentX, currentY);
qCDebug(dcZigbeeGenericLights()) << "Current color" << color.toRgb() << QPoint(currentX, currentY);
thing->setStateValue(colorLightColorStateTypeId, color);
});
}
void IntegrationPluginZigbeeGenericLights::readColorCapabilities(Thing *thing)
{
ZigbeeNodeEndpoint *endpoint = findEndpoint(thing);
if (!endpoint) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color capabilities for" << thing << "because the node could not be found";
return;
}
// Get the color server cluster from the endpoint
ZigbeeClusterColorControl *colorCluster = endpoint->inputCluster<ZigbeeClusterColorControl>(ZigbeeClusterLibrary::ClusterIdColorControl);
if (!colorCluster) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color color capabilities for" << thing << "because the color cluster could not be found on" << endpoint;
return;
}
// Check if we know already the color capabilities for this lamp
if (colorCluster->hasAttribute(ZigbeeClusterColorControl::AttributeColorCapabilities)) {
m_colorCapabilities[thing] = colorCluster->colorCapabilities();
qCDebug(dcZigbeeGenericLights()) << "Found cached color capabilities for" << thing << colorCluster->colorCapabilities();
processColorCapabilities(thing);
return;
}
// We have to read the color capabilities
ZigbeeClusterReply *reply = colorCluster->readAttributes({ZigbeeClusterColorControl::AttributeColorCapabilities});
connect(reply, &ZigbeeClusterReply::finished, thing, [=](){
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color capabilitie for" << thing << "because the node could not be found";
return;
}
m_colorCapabilities[thing] = colorCluster->colorCapabilities();
processColorCapabilities(thing);
});
}
void IntegrationPluginZigbeeGenericLights::processColorCapabilities(Thing *thing)
{
// Fetch all information required depending on the capabilities
qCDebug(dcZigbeeGenericLights()) << "Loading information depending on the lamp capabilities" << thing << m_colorCapabilities[thing];
if (m_colorCapabilities[thing].testFlag(ZigbeeClusterColorControl::ColorCapabilityColorTemperature)) {
qCDebug(dcZigbeeGenericLights()) << "The lamp is capable of native controlling the color temperature";
// Read min/max value, otherwise emulate the color temperature using the color map
readColorTemperatureRange(thing);
} else {
qCDebug(dcZigbeeGenericLights()) << "The lamp has no native color temperature capability, emulating them using color map.";
// TODO: continue with color fetching (xy, hsv, gamut values)
qCDebug(dcZigbeeGenericLights()) << "Lamp capabilities information complete";
}
}
void IntegrationPluginZigbeeGenericLights::readColorTemperatureRange(Thing *thing)
{
ZigbeeNodeEndpoint *endpoint = findEndpoint(thing);
if (!endpoint) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color temperature range for" << thing << "because the node could not be found";
return;
}
// Get the color server cluster from the endpoint
ZigbeeClusterColorControl *colorCluster = endpoint->inputCluster<ZigbeeClusterColorControl>(ZigbeeClusterLibrary::ClusterIdColorControl);
if (!colorCluster) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color temperature range for" << thing << "because the color cluster could not be found on" << endpoint;
return;
}
// Check if we can use the cached values from the database
if (readCachedColorTemperatureRange(thing, colorCluster)) {
qCDebug(dcZigbeeGenericLights()) << "Using cached color temperature mireds interval for mapping" << thing << "[" << m_colorTemperatureRanges[thing].minValue << "," << m_colorTemperatureRanges[thing].maxValue << "] mired";
return;
}
// Init with default values
m_colorTemperatureRanges[thing] = ColorTemperatureRange();
// We need to read them from the lamp
ZigbeeClusterReply *reply = colorCluster->readAttributes({ZigbeeClusterColorControl::AttributeColorTempPhysicalMinMireds, ZigbeeClusterColorControl::AttributeColorTempPhysicalMaxMireds});
connect(reply, &ZigbeeClusterReply::finished, thing, [=](){
if (reply->error() != ZigbeeClusterReply::ErrorNoError) {
qCWarning(dcZigbeeGenericLights()) << "Reading color temperature range attributes finished with error" << reply->error();
qCWarning(dcZigbeeGenericLights()) << "Failed to read color temperature min/max interval values. Using default values for" << thing << "[" << m_colorTemperatureRanges[thing].minValue << "," << m_colorTemperatureRanges[thing].maxValue << "] mired";
return;
}
QList<ZigbeeClusterLibrary::ReadAttributeStatusRecord> attributeStatusRecords = ZigbeeClusterLibrary::parseAttributeStatusRecords(reply->responseFrame().payload);
if (attributeStatusRecords.count() != 2) {
qCWarning(dcZigbeeGenericLights()) << "Did not receive temperature min/max interval values from" << thing;
qCWarning(dcZigbeeGenericLights()) << "Using default values for" << thing << "[" << m_colorTemperatureRanges[thing].minValue << "," << m_colorTemperatureRanges[thing].maxValue << "] mired" ;
return;
}
// Parse the attribute status records
foreach (const ZigbeeClusterLibrary::ReadAttributeStatusRecord &attributeStatusRecord, attributeStatusRecords) {
if (attributeStatusRecord.attributeId == ZigbeeClusterColorControl::AttributeColorTempPhysicalMinMireds) {
bool valueOk = false;
quint16 minMiredsValue = attributeStatusRecord.dataType.toUInt16(&valueOk);
if (!valueOk) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color temperature min mireds attribute value and convert it" << attributeStatusRecord;
break;
}
m_colorTemperatureRanges[thing].minValue = minMiredsValue;
}
if (attributeStatusRecord.attributeId == ZigbeeClusterColorControl::AttributeColorTempPhysicalMaxMireds) {
bool valueOk = false;
quint16 maxMiredsValue = attributeStatusRecord.dataType.toUInt16(&valueOk);
if (!valueOk) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color temperature max mireds attribute value and convert it" << attributeStatusRecord;
break;
}
m_colorTemperatureRanges[thing].maxValue = maxMiredsValue;
}
}
qCDebug(dcZigbeeGenericLights()) << "Using lamp specific color temperature mireds interval for mapping" << thing << "[" << m_colorTemperatureRanges[thing].minValue << "," << m_colorTemperatureRanges[thing].maxValue << "] mired";
});
}
bool IntegrationPluginZigbeeGenericLights::readCachedColorTemperatureRange(Thing *thing, ZigbeeClusterColorControl *colorCluster)
{
if (colorCluster->hasAttribute(ZigbeeClusterColorControl::AttributeColorTempPhysicalMinMireds) && colorCluster->hasAttribute(ZigbeeClusterColorControl::AttributeColorTempPhysicalMaxMireds)) {
ZigbeeClusterAttribute minMiredsAttribute = colorCluster->attribute(ZigbeeClusterColorControl::AttributeColorTempPhysicalMinMireds);
bool valueOk = false;
quint16 minMiredsValue = minMiredsAttribute.dataType().toUInt16(&valueOk);
if (!valueOk) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color temperature min mireds attribute value and convert it" << minMiredsAttribute;
return false;
}
ZigbeeClusterAttribute maxMiredsAttribute = colorCluster->attribute(ZigbeeClusterColorControl::AttributeColorTempPhysicalMaxMireds);
quint16 maxMiredsValue = maxMiredsAttribute.dataType().toUInt16(&valueOk);
if (!valueOk) {
qCWarning(dcZigbeeGenericLights()) << "Failed to read color temperature max mireds attribute value and convert it" << maxMiredsAttribute;
return false;
}
ColorTemperatureRange range;
range.minValue = minMiredsValue;
range.maxValue = maxMiredsValue;
m_colorTemperatureRanges[thing] = range;
return true;
}
return false;
}
quint16 IntegrationPluginZigbeeGenericLights::mapScaledValueToColorTemperature(Thing *thing, int scaledColorTemperature)
{
// Make sure we have values to scale
if (!m_colorTemperatureRanges.contains(thing)) {
m_colorTemperatureRanges[thing] = ColorTemperatureRange();
}
double percentage = static_cast<double>((scaledColorTemperature - m_minScaleValue)) / (m_maxScaleValue - m_minScaleValue);
//qCDebug(dcZigbeeGenericLights()) << "Mapping color temperature value" << scaledColorTemperature << "between" << m_minScaleValue << m_maxScaleValue << "is" << percentage * 100 << "%";
double mappedValue = (m_colorTemperatureRanges[thing].maxValue - m_colorTemperatureRanges[thing].minValue) * percentage + m_colorTemperatureRanges[thing].minValue;
//qCDebug(dcZigbeeGenericLights()) << "Mapping color temperature value" << scaledColorTemperature << "is" << mappedValue << "mireds";
return static_cast<quint16>(qRound(mappedValue));
}
int IntegrationPluginZigbeeGenericLights::mapColorTemperatureToScaledValue(Thing *thing, quint16 colorTemperature)
{
// Make sure we have values to scale
if (!m_colorTemperatureRanges.contains(thing)) {
m_colorTemperatureRanges[thing] = ColorTemperatureRange();
}
double percentage = static_cast<double>((colorTemperature - m_colorTemperatureRanges[thing].minValue)) / (m_colorTemperatureRanges[thing].maxValue - m_colorTemperatureRanges[thing].minValue);
//qCDebug(dcZigbee()) << "Mapping color temperature value" << colorTemperature << "mirred" << m_minColorTemperature << m_maxColorTemperature << "is" << percentage * 100 << "%";
double mappedValue = (m_maxScaleValue - m_minScaleValue) * percentage + m_minScaleValue;
//qCDebug(dcZigbee()) << "Mapping color temperature value" << colorTemperature << "results into the scaled value of" << mappedValue;
return static_cast<int>(qRound(mappedValue));
}