// SPDX-License-Identifier: GPL-3.0-or-later /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Copyright (C) 2013 - 2024, nymea GmbH * Copyright (C) 2024 - 2025, chargebyte austria GmbH * * This file is part of nymea-experience-plugin-airconditioning. * * nymea-experience-plugin-airconditioning is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * nymea-experience-plugin-airconditioning 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with nymea-experience-plugin-airconditioning. If not, see . * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ #include "thermostat.h" #include Q_DECLARE_LOGGING_CATEGORY(dcAirConditioning) Thermostat::Thermostat(ThingManager *thingManager, Thing *thing, QObject *parent): QObject(parent), m_thingManager(thingManager), m_thing(thing) { m_cachedTargetTemperature = m_thing->stateValue("targetTemperature").toDouble(); } Thing *Thermostat::thing() const { return m_thing; } void Thermostat::setTargetTemperature(double targetTemperature, bool force) { qCDebug(dcAirConditioning()) << "setTargetTemp called. Window open:" << m_windowOpen << "force:" << force; m_cachedTargetTemperature = targetTemperature; if (m_windowOpen && !force) { qCDebug(dcAirConditioning()) << "Not setting target temperature on" << m_thing->name() << "because a window is open"; return; } if (m_thing->stateValue("targetTemperature").toDouble() != targetTemperature) { ActionType actionType = m_thing->thingClass().actionTypes().findByName("targetTemperature"); Action action(actionType.id(), m_thing->id(), Action::TriggeredByRule); action.setParams({Param(actionType.id(), targetTemperature)}); qCDebug(dcAirConditioning()) << "Setting target temperature" << targetTemperature << "to" << m_thing->name() << "from" << m_thing->stateValue("targetTemperature").toDouble(); ThingActionInfo *info = m_thingManager->executeAction(action); connect(info, &ThingActionInfo::finished, this, [info, this](){ if (info->status() != Thing::ThingErrorNoError) { qCWarning(dcAirConditioning()) << "Unable to execute targetTemperature action on" << m_thing << info->status() << info->displayMessage(); return; } qCDebug(dcAirConditioning()) << "Target temperature set successfully"; }); } } void Thermostat::setWindowOpen(bool windowOpen) { m_windowOpen = windowOpen; // First check if the device is capable of handling a window open locks if (!m_thing->thingClass().actionTypes().findByName("windowOpen").id().isNull()) { if (m_thing->stateValue("windowOpen").toBool() != windowOpen) { ActionType actionType = m_thing->thingClass().actionTypes().findByName("windowOpen"); Action action(actionType.id(), m_thing->id(), Action::TriggeredByRule); action.setParams({Param(actionType.id(), windowOpen)}); qCDebug(dcAirConditioning()) << "Setting window open" << windowOpen << "to" << m_thing->name(); ThingActionInfo *info = m_thingManager->executeAction(action); connect(info, &ThingActionInfo::finished, this, [info, this](){ if (info->status() != Thing::ThingErrorNoError) { qCWarning(dcAirConditioning()) << "Unable to execute window Open action on" << m_thing << info->status() << info->displayMessage(); return; } }); } return; } // Otherwise see if it can be turned off while the window is open if (m_thing->hasState("power")) { if (m_thing->stateValue("power").toBool() == windowOpen) { ActionType actionType = m_thing->thingClass().actionTypes().findByName("power"); Action action(actionType.id(), m_thing->id(), Action::TriggeredByRule); action.setParams({Param(actionType.id(), !windowOpen)}); qCDebug(dcAirConditioning()) << "Setting power" << !windowOpen << "to" << m_thing->name(); ThingActionInfo *info = m_thingManager->executeAction(action); connect(info, &ThingActionInfo::finished, this, [info, this](){ if (info->status() != Thing::ThingErrorNoError) { qCWarning(dcAirConditioning()) << "Unable to execute power action on" << m_thing << info->status() << info->displayMessage(); return; } }); } } // If nothing works, let's assume it is a very dump radiator thermostat and set the temperature to minimum double temp = windowOpen ? m_thing->state("targetTemperature").minValue().toDouble() : m_cachedTargetTemperature; if (m_thing->stateValue("targetTemperature").toDouble() != temp) { ActionType actionType = m_thing->thingClass().actionTypes().findByName("targetTemperature"); Action action(actionType.id(), m_thing->id(), Action::TriggeredByRule); action.setParams({Param(actionType.id(), temp)}); qCDebug(dcAirConditioning()) << "Setting target temperature (window open control)" << temp << "to" << m_thing->name() << "from" << m_thing->stateValue("targetTemperature").toDouble(); ThingActionInfo *info = m_thingManager->executeAction(action); connect(info, &ThingActionInfo::finished, this, [info, this](){ if (info->status() != Thing::ThingErrorNoError) { qCWarning(dcAirConditioning()) << "Unable to execute targetTemperature action on" << m_thing << info->status() << info->displayMessage(); return; } qCDebug(dcAirConditioning()) << "Target temperature set successfully"; }); } } bool Thermostat::hasTemperatureSensor() const { return m_thing->thingClass().interfaces().contains("temperaturesensor"); } double Thermostat::temperature() const { return m_thing->stateValue("temperature").toDouble(); }