From 09b2ad0527ee21319e8ade67a62f191e863be577 Mon Sep 17 00:00:00 2001 From: Pieter Kokx Date: Sat, 16 Nov 2024 15:00:44 +0100 Subject: [PATCH] Added changing DUCO fan modes --- esphome/components/duco/select/__init__.py | 32 ++++++ esphome/components/duco/select/select.cpp | 120 +++++++++++++++++++++ esphome/components/duco/select/select.h | 45 ++++++++ 3 files changed, 197 insertions(+) create mode 100644 esphome/components/duco/select/__init__.py create mode 100644 esphome/components/duco/select/select.cpp create mode 100644 esphome/components/duco/select/select.h diff --git a/esphome/components/duco/select/__init__.py b/esphome/components/duco/select/__init__.py new file mode 100644 index 0000000000..a7a0f208fb --- /dev/null +++ b/esphome/components/duco/select/__init__.py @@ -0,0 +1,32 @@ +import esphome.codegen as cg +from esphome.components import select +import esphome.config_validation as cv +from esphome.const import CONF_ID + +from .. import CONF_DUCO_ID, DUCO_COMPONENT_SCHEMA + +DEPENDENCIES = ["duco"] +CODEOWNERS = ["@kokx"] + +DUCO_MODE_OPTIONS = ["AUTO", "MAN1", "MAN2", "MAN3", "EMPT", "CNT1", "CNT2", "CNT3"] + +duco_ns = cg.esphome_ns.namespace("duco") +DucoSelect = duco_ns.class_("DucoSelect", cg.PollingComponent, select.Select) + +CONFIG_SCHEMA = cv.All( + select.select_schema(DucoSelect) + .extend(cv.COMPONENT_SCHEMA) + .extend(cv.polling_component_schema("5s")) + .extend(DUCO_COMPONENT_SCHEMA) + .extend({cv.GenerateID(): cv.declare_id(DucoSelect)}) +) + + +async def to_code(config): + var = cg.new_Pvariable(config[CONF_ID]) + await cg.register_component(var, config) + + await select.register_select(var, config, options=DUCO_MODE_OPTIONS) + + parent = await cg.get_variable(config[CONF_DUCO_ID]) + cg.add(parent.add_sensor_item(var)) diff --git a/esphome/components/duco/select/select.cpp b/esphome/components/duco/select/select.cpp new file mode 100644 index 0000000000..744971d4f2 --- /dev/null +++ b/esphome/components/duco/select/select.cpp @@ -0,0 +1,120 @@ +#include "select.h" +#include "../duco.h" +#include + +namespace esphome { +namespace duco { + +static const char *const TAG = "duco select"; + +const std::string DucoSelect::MODE_AUTO = "AUTO"; +const std::string DucoSelect::MODE_MAN1 = "MAN1"; +const std::string DucoSelect::MODE_MAN2 = "MAN2"; +const std::string DucoSelect::MODE_MAN3 = "MAN3"; +const std::string DucoSelect::MODE_EMPT = "EMPT"; +const std::string DucoSelect::MODE_CNT1 = "CNT1"; +const std::string DucoSelect::MODE_CNT2 = "CNT2"; +const std::string DucoSelect::MODE_CNT3 = "CNT3"; + +const uint8_t DucoSelect::MODE_CODE_AUTO = 0x00; +const uint8_t DucoSelect::MODE_CODE_MAN1 = 0x04; +const uint8_t DucoSelect::MODE_CODE_MAN2 = 0x05; +const uint8_t DucoSelect::MODE_CODE_MAN3 = 0x06; +const uint8_t DucoSelect::MODE_CODE_EMPT = 0x07; +const uint8_t DucoSelect::MODE_CODE_CNT1 = 0x08; +const uint8_t DucoSelect::MODE_CODE_CNT2 = 0x09; +const uint8_t DucoSelect::MODE_CODE_CNT3 = 0x0a; + +void DucoSelect::setup() {} + +void DucoSelect::update() { + // ask for current mode + ESP_LOGD(TAG, "Ask for current mode"); + + // ask for information from node 1 + std::vector message = {0x02, 0x01}; + this->parent_->send(0x0c, message, this); +} + +float DucoSelect::get_setup_priority() const { + // After DUCO + return setup_priority::BUS - 2.0f; +} + +const std::string code_to_string(uint8_t mode) { + switch (mode) { + case DucoSelect::MODE_CODE_MAN1: + return DucoSelect::MODE_MAN1; + case DucoSelect::MODE_CODE_MAN2: + return DucoSelect::MODE_MAN2; + case DucoSelect::MODE_CODE_MAN3: + return DucoSelect::MODE_MAN3; + case DucoSelect::MODE_CODE_EMPT: + return DucoSelect::MODE_EMPT; + case DucoSelect::MODE_CODE_CNT1: + return DucoSelect::MODE_CNT1; + case DucoSelect::MODE_CODE_CNT2: + return DucoSelect::MODE_CNT2; + case DucoSelect::MODE_CODE_CNT3: + return DucoSelect::MODE_CNT3; + case DucoSelect::MODE_CODE_AUTO: + default: + return DucoSelect::MODE_AUTO; + } + return DucoSelect::MODE_AUTO; +} + +uint8_t string_to_code(std::string mode) { + if (mode == DucoSelect::MODE_MAN1) { + return DucoSelect::MODE_CODE_MAN1; + } + if (mode == DucoSelect::MODE_MAN2) { + return DucoSelect::MODE_CODE_MAN2; + } + if (mode == DucoSelect::MODE_MAN3) { + return DucoSelect::MODE_CODE_MAN3; + } + if (mode == DucoSelect::MODE_EMPT) { + return DucoSelect::MODE_CODE_EMPT; + } + if (mode == DucoSelect::MODE_CNT1) { + return DucoSelect::MODE_CODE_CNT1; + } + if (mode == DucoSelect::MODE_CNT2) { + return DucoSelect::MODE_CODE_CNT2; + } + if (mode == DucoSelect::MODE_CNT3) { + return DucoSelect::MODE_CODE_CNT3; + } + if (mode == DucoSelect::MODE_AUTO) { + return DucoSelect::MODE_CODE_AUTO; + } + return DucoSelect::MODE_CODE_AUTO; +} + +void DucoSelect::receive_response(std::vector message) { + if (message[1] == 0x0e && message[3] != 0x01) { + // mode response received, parse it + auto mode = code_to_string(message[3]); + + publish_state(mode); + + ESP_LOGD(TAG, "Current mode: %s", mode.c_str()); + + // do not wait for new messages with the same ID + this->parent_->stop_waiting(message[2]); + } + if (message[1] == 0x0e && message[3] == 0x01) { + this->parent_->stop_waiting(message[2]); + } +} + +void DucoSelect::control(const std::string &value) { + ESP_LOGD(TAG, "TODO: Set value %s", value.c_str()); + std::vector message = {0x04, 0x01, string_to_code(value)}; + + this->parent_->send(0x0c, message, this); +} + +} // namespace duco +} // namespace esphome diff --git a/esphome/components/duco/select/select.h b/esphome/components/duco/select/select.h new file mode 100644 index 0000000000..a6bcaa3ae2 --- /dev/null +++ b/esphome/components/duco/select/select.h @@ -0,0 +1,45 @@ +#pragma once + +#include "esphome/core/log.h" +#include "esphome/core/component.h" +#include "esphome/components/select/select.h" +#include "../duco.h" + +namespace esphome { +namespace duco { + +class DucoSelect : public DucoDevice, public PollingComponent, public select::Select { + public: + static const std::string MODE_AUTO; + static const std::string MODE_MAN1; + static const std::string MODE_MAN2; + static const std::string MODE_MAN3; + static const std::string MODE_EMPT; + static const std::string MODE_CNT1; + static const std::string MODE_CNT2; + static const std::string MODE_CNT3; + + static const uint8_t MODE_CODE_AUTO; + static const uint8_t MODE_CODE_MAN1; + static const uint8_t MODE_CODE_MAN2; + static const uint8_t MODE_CODE_MAN3; + static const uint8_t MODE_CODE_EMPT; + static const uint8_t MODE_CODE_CNT1; + static const uint8_t MODE_CODE_CNT2; + static const uint8_t MODE_CODE_CNT3; + + void setup() override; + void update() override; + + float get_setup_priority() const override; + + void receive_response(std::vector message) override; + + void control(const std::string &value) override; + + protected: + uint8_t attempts = 0; +}; + +} // namespace duco +} // namespace esphome