186 lines
7.4 KiB
C++
186 lines
7.4 KiB
C++
// SPDX-License-Identifier: LGPL-3.0-or-later
|
|
|
|
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
*
|
|
* Copyright (C) 2013 - 2024, nymea GmbH
|
|
* Copyright (C) 2024 - 2025, chargebyte austria GmbH
|
|
*
|
|
* This file is part of libnymea-app.
|
|
*
|
|
* libnymea-app is free software: you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public License
|
|
* as published by the Free Software Foundation, either version 3
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* libnymea-app 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 libnymea-app. If not, see <https://www.gnu.org/licenses/>.
|
|
*
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
#include "thinggroup.h"
|
|
#include "thingmanager.h"
|
|
#include "thingsproxy.h"
|
|
#include "types/statetypes.h"
|
|
#include "types/actiontype.h"
|
|
|
|
ThingGroup::ThingGroup(ThingManager *thingManager, ThingClass *thingClass, ThingsProxy *things, QObject *parent):
|
|
Thing(thingManager, thingClass, QUuid::createUuid(), parent),
|
|
m_things(things)
|
|
{
|
|
thingClass->setParent(this);
|
|
|
|
States *states = new States(this);
|
|
for (int i = 0; i < thingClass->stateTypes()->rowCount(); i++) {
|
|
StateType *st = thingClass->stateTypes()->get(i);
|
|
State *state = new State(id(), st->id(), QVariant(), this);
|
|
qDebug() << "Adding state" << st->name() << st->minValue() << st->maxValue();
|
|
states->addState(state);
|
|
}
|
|
setStates(states);
|
|
syncStates();
|
|
setName(thingClass->displayName());
|
|
|
|
connect(things, &ThingsProxy::dataChanged, this, [this](const QModelIndex &/*topLeft*/, const QModelIndex &/*bottomRight*/, const QVector<int> &/*roles*/){
|
|
syncStates();
|
|
});
|
|
|
|
connect(m_thingManager, &ThingManager::executeActionReply, this, [this](int commandId, Thing::ThingError error, const QString &displayMessage){
|
|
// This should maybe check the params and create a sensible group result instead of just forwarding the result of the last reply
|
|
qDebug() << "action reply:" << commandId;
|
|
foreach (int id, m_pendingGroupActions.keys()) {
|
|
if (m_pendingGroupActions.value(id).contains(commandId)) {
|
|
m_pendingGroupActions[id].removeAll(commandId);
|
|
if (m_pendingGroupActions[id].isEmpty()) {
|
|
m_pendingGroupActions.remove(id);
|
|
emit executeActionReply(id, error, displayMessage);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
});
|
|
|
|
}
|
|
|
|
int ThingGroup::executeAction(const QString &actionName, const QVariantList ¶ms)
|
|
{
|
|
QList<int> pendingIds;
|
|
|
|
ActionType *groupActionType = m_thingClass->actionTypes()->findByName(actionName);
|
|
if (!groupActionType) {
|
|
qWarning() << "Group has no action" << actionName;
|
|
return -1;
|
|
}
|
|
|
|
qDebug() << "Execute action for group:" << this;
|
|
for (int i = 0; i < m_things->rowCount(); i++) {
|
|
Thing *thing = m_things->get(i);
|
|
if (thing->setupStatus() != Thing::ThingSetupStatusComplete) {
|
|
continue;
|
|
}
|
|
ActionType *actionType = thing->thingClass()->actionTypes()->findByName(actionName);
|
|
if (!actionType) {
|
|
qWarning() << "Cannot send action to thing" << thing->name() << "because according action can't be found";
|
|
continue;
|
|
}
|
|
|
|
|
|
QVariantList finalParams;
|
|
foreach (const QVariant ¶mVariant, params) {
|
|
QString paramName = paramVariant.toMap().value("paramName").toString();
|
|
ParamType *groupParamType = groupActionType->paramTypes()->findByName(paramName);
|
|
if (!groupParamType) {
|
|
qWarning() << "Not adding param" << paramName << "to action" << actionName << "because group action param can't be found";
|
|
continue;
|
|
}
|
|
ParamType *paramType = actionType->paramTypes()->findByName(paramName);
|
|
if (!paramType) {
|
|
qWarning() << "Not adding param" << paramName << "to action" << actionName << "because according action params can't be found";
|
|
continue;
|
|
}
|
|
|
|
QVariantMap finalParam;
|
|
finalParam.insert("paramTypeId", paramType->id());
|
|
finalParam.insert("value", mapValue(paramVariant.toMap().value("value"), groupParamType, paramType));
|
|
finalParams.append(finalParam);
|
|
}
|
|
|
|
qDebug() << "Initial params" << params;
|
|
qDebug() << "Executing" << thing->id() << actionType->name() << finalParams;
|
|
int id = m_thingManager->executeAction(thing->id(), actionType->id(), finalParams);
|
|
pendingIds.append(id);
|
|
}
|
|
m_pendingGroupActions.insert(++m_idCounter, pendingIds);
|
|
return m_idCounter;
|
|
}
|
|
|
|
void ThingGroup::syncStates()
|
|
{
|
|
for (int i = 0; i < thingClass()->stateTypes()->rowCount(); i++) {
|
|
StateType *stateType = thingClass()->stateTypes()->get(i);
|
|
State *state = states()->getState(stateType->id());
|
|
|
|
qDebug() << "syncing state" << stateType->name() << stateType->type();
|
|
|
|
QVariant value;
|
|
int count = 0;
|
|
for (int j = 0; j < m_things->rowCount(); j++) {
|
|
Thing *d = m_things->get(j);
|
|
// Skip things that don't have the required state
|
|
StateType *ds = d->thingClass()->stateTypes()->findByName(stateType->name());
|
|
if (!ds) {
|
|
continue;
|
|
}
|
|
|
|
// Skip disconnected things
|
|
StateType *connectedStateType = d->thingClass()->stateTypes()->findByName("connected");
|
|
if (connectedStateType) {
|
|
if (!d->stateValue(connectedStateType->id()).toBool()) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (stateType->type().toLower() == "bool") {
|
|
if (d->stateValue(ds->id()).toBool()) {
|
|
value = true;
|
|
break;
|
|
}
|
|
} else if (stateType->type().toLower() == "int") {
|
|
value = value.toInt() + d->stateValue(ds->id()).toInt();
|
|
count++;
|
|
} else if (stateType->type().toLower() == "qcolor") {
|
|
value = d->stateValue(ds->id());
|
|
break;
|
|
}
|
|
}
|
|
if (count > 0) {
|
|
value = value.toDouble() / count;
|
|
}
|
|
state->setValue(value);
|
|
}
|
|
}
|
|
|
|
QVariant ThingGroup::mapValue(const QVariant &value, ParamType *fromParamType, ParamType *toParamType) const
|
|
{
|
|
qDebug() << "Mapping:" << value << fromParamType->minValue() << fromParamType->maxValue() << toParamType->minValue() << toParamType->maxValue();
|
|
|
|
if (!fromParamType->minValue().isValid()
|
|
|| !fromParamType->maxValue().isValid()
|
|
|| !toParamType->minValue().isValid()
|
|
|| !toParamType->maxValue().isValid()) {
|
|
return value;
|
|
}
|
|
double fromMin = fromParamType->minValue().toDouble();
|
|
double fromMax = fromParamType->maxValue().toDouble();
|
|
double toMin = toParamType->minValue().toDouble();
|
|
double toMax = toParamType->maxValue().toDouble();
|
|
double fromValue = value.toDouble();
|
|
double fromPercent = (fromValue - fromMin) / (fromMax - fromMin);
|
|
double toValue = toMin + (toMax - toMin) * fromPercent;
|
|
return toValue;
|
|
}
|