mirror of
https://github.com/esphome/esphome.git
synced 2024-11-30 18:54:14 +01:00
Merge branch 'dev' into dev
This commit is contained in:
commit
83bdbc9cc6
13 changed files with 225 additions and 59 deletions
|
@ -20,6 +20,8 @@ from esphome.const import (
|
||||||
CONF_DEASSERT_RTS_DTR,
|
CONF_DEASSERT_RTS_DTR,
|
||||||
CONF_DISABLED,
|
CONF_DISABLED,
|
||||||
CONF_ESPHOME,
|
CONF_ESPHOME,
|
||||||
|
CONF_LEVEL,
|
||||||
|
CONF_LOG_TOPIC,
|
||||||
CONF_LOGGER,
|
CONF_LOGGER,
|
||||||
CONF_MDNS,
|
CONF_MDNS,
|
||||||
CONF_MQTT,
|
CONF_MQTT,
|
||||||
|
@ -30,6 +32,7 @@ from esphome.const import (
|
||||||
CONF_PLATFORMIO_OPTIONS,
|
CONF_PLATFORMIO_OPTIONS,
|
||||||
CONF_PORT,
|
CONF_PORT,
|
||||||
CONF_SUBSTITUTIONS,
|
CONF_SUBSTITUTIONS,
|
||||||
|
CONF_TOPIC,
|
||||||
PLATFORM_BK72XX,
|
PLATFORM_BK72XX,
|
||||||
PLATFORM_ESP32,
|
PLATFORM_ESP32,
|
||||||
PLATFORM_ESP8266,
|
PLATFORM_ESP8266,
|
||||||
|
@ -95,8 +98,12 @@ def choose_upload_log_host(
|
||||||
options.append((f"Over The Air ({CORE.address})", CORE.address))
|
options.append((f"Over The Air ({CORE.address})", CORE.address))
|
||||||
if default == "OTA":
|
if default == "OTA":
|
||||||
return CORE.address
|
return CORE.address
|
||||||
if show_mqtt and CONF_MQTT in CORE.config:
|
if (
|
||||||
options.append((f"MQTT ({CORE.config['mqtt'][CONF_BROKER]})", "MQTT"))
|
show_mqtt
|
||||||
|
and (mqtt_config := CORE.config.get(CONF_MQTT))
|
||||||
|
and mqtt_logging_enabled(mqtt_config)
|
||||||
|
):
|
||||||
|
options.append((f"MQTT ({mqtt_config[CONF_BROKER]})", "MQTT"))
|
||||||
if default == "OTA":
|
if default == "OTA":
|
||||||
return "MQTT"
|
return "MQTT"
|
||||||
if default is not None:
|
if default is not None:
|
||||||
|
@ -106,6 +113,17 @@ def choose_upload_log_host(
|
||||||
return choose_prompt(options, purpose=purpose)
|
return choose_prompt(options, purpose=purpose)
|
||||||
|
|
||||||
|
|
||||||
|
def mqtt_logging_enabled(mqtt_config):
|
||||||
|
log_topic = mqtt_config[CONF_LOG_TOPIC]
|
||||||
|
if log_topic is None:
|
||||||
|
return False
|
||||||
|
if CONF_TOPIC not in log_topic:
|
||||||
|
return False
|
||||||
|
if log_topic.get(CONF_LEVEL, None) == "NONE":
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
def get_port_type(port):
|
def get_port_type(port):
|
||||||
if port.startswith("/") or port.startswith("COM"):
|
if port.startswith("/") or port.startswith("COM"):
|
||||||
return "SERIAL"
|
return "SERIAL"
|
||||||
|
|
|
@ -7,6 +7,7 @@ import esphome.config_validation as cv
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
CONF_AUTO_CLEAR_ENABLED,
|
CONF_AUTO_CLEAR_ENABLED,
|
||||||
CONF_BUFFER_SIZE,
|
CONF_BUFFER_SIZE,
|
||||||
|
CONF_GROUP,
|
||||||
CONF_ID,
|
CONF_ID,
|
||||||
CONF_LAMBDA,
|
CONF_LAMBDA,
|
||||||
CONF_ON_IDLE,
|
CONF_ON_IDLE,
|
||||||
|
@ -23,9 +24,15 @@ from esphome.helpers import write_file_if_changed
|
||||||
from . import defines as df, helpers, lv_validation as lvalid
|
from . import defines as df, helpers, lv_validation as lvalid
|
||||||
from .automation import disp_update, focused_widgets, update_to_code
|
from .automation import disp_update, focused_widgets, update_to_code
|
||||||
from .defines import add_define
|
from .defines import add_define
|
||||||
from .encoders import ENCODERS_CONFIG, encoders_to_code, initial_focus_to_code
|
from .encoders import (
|
||||||
|
ENCODERS_CONFIG,
|
||||||
|
encoders_to_code,
|
||||||
|
get_default_group,
|
||||||
|
initial_focus_to_code,
|
||||||
|
)
|
||||||
from .gradient import GRADIENT_SCHEMA, gradients_to_code
|
from .gradient import GRADIENT_SCHEMA, gradients_to_code
|
||||||
from .hello_world import get_hello_world
|
from .hello_world import get_hello_world
|
||||||
|
from .keypads import KEYPADS_CONFIG, keypads_to_code
|
||||||
from .lv_validation import lv_bool, lv_images_used
|
from .lv_validation import lv_bool, lv_images_used
|
||||||
from .lvcode import LvContext, LvglComponent, lvgl_static
|
from .lvcode import LvContext, LvglComponent, lvgl_static
|
||||||
from .schemas import (
|
from .schemas import (
|
||||||
|
@ -158,6 +165,13 @@ def multi_conf_validate(configs: list[dict]):
|
||||||
display_list = [disp for disps in displays for disp in disps]
|
display_list = [disp for disps in displays for disp in disps]
|
||||||
if len(display_list) != len(set(display_list)):
|
if len(display_list) != len(set(display_list)):
|
||||||
raise cv.Invalid("A display ID may be used in only one LVGL instance")
|
raise cv.Invalid("A display ID may be used in only one LVGL instance")
|
||||||
|
for config in configs:
|
||||||
|
for item in (df.CONF_ENCODERS, df.CONF_KEYPADS):
|
||||||
|
for enc in config.get(item, ()):
|
||||||
|
if CONF_GROUP not in enc:
|
||||||
|
raise cv.Invalid(
|
||||||
|
f"'{item}' must have an explicit group set when using multiple LVGL instances"
|
||||||
|
)
|
||||||
base_config = configs[0]
|
base_config = configs[0]
|
||||||
for config in configs[1:]:
|
for config in configs[1:]:
|
||||||
for item in (
|
for item in (
|
||||||
|
@ -173,6 +187,7 @@ def multi_conf_validate(configs: list[dict]):
|
||||||
|
|
||||||
|
|
||||||
def final_validation(configs):
|
def final_validation(configs):
|
||||||
|
if len(configs) != 1:
|
||||||
multi_conf_validate(configs)
|
multi_conf_validate(configs)
|
||||||
global_config = full_config.get()
|
global_config = full_config.get()
|
||||||
for config in configs:
|
for config in configs:
|
||||||
|
@ -275,6 +290,7 @@ async def to_code(configs):
|
||||||
else:
|
else:
|
||||||
add_define("LV_FONT_DEFAULT", await lvalid.lv_font.process(default_font))
|
add_define("LV_FONT_DEFAULT", await lvalid.lv_font.process(default_font))
|
||||||
cg.add(lvgl_static.esphome_lvgl_init())
|
cg.add(lvgl_static.esphome_lvgl_init())
|
||||||
|
default_group = get_default_group(config_0)
|
||||||
|
|
||||||
for config in configs:
|
for config in configs:
|
||||||
frac = config[CONF_BUFFER_SIZE]
|
frac = config[CONF_BUFFER_SIZE]
|
||||||
|
@ -303,7 +319,8 @@ async def to_code(configs):
|
||||||
lv_scr_act = get_scr_act(lv_component)
|
lv_scr_act = get_scr_act(lv_component)
|
||||||
async with LvContext():
|
async with LvContext():
|
||||||
await touchscreens_to_code(lv_component, config)
|
await touchscreens_to_code(lv_component, config)
|
||||||
await encoders_to_code(lv_component, config)
|
await encoders_to_code(lv_component, config, default_group)
|
||||||
|
await keypads_to_code(lv_component, config, default_group)
|
||||||
await theme_to_code(config)
|
await theme_to_code(config)
|
||||||
await styles_to_code(config)
|
await styles_to_code(config)
|
||||||
await gradients_to_code(config)
|
await gradients_to_code(config)
|
||||||
|
@ -430,6 +447,7 @@ LVGL_SCHEMA = (
|
||||||
cv.Optional(df.CONF_GRADIENTS): GRADIENT_SCHEMA,
|
cv.Optional(df.CONF_GRADIENTS): GRADIENT_SCHEMA,
|
||||||
cv.Optional(df.CONF_TOUCHSCREENS, default=None): touchscreen_schema,
|
cv.Optional(df.CONF_TOUCHSCREENS, default=None): touchscreen_schema,
|
||||||
cv.Optional(df.CONF_ENCODERS, default=None): ENCODERS_CONFIG,
|
cv.Optional(df.CONF_ENCODERS, default=None): ENCODERS_CONFIG,
|
||||||
|
cv.Optional(df.CONF_KEYPADS, default=None): KEYPADS_CONFIG,
|
||||||
cv.GenerateID(df.CONF_DEFAULT_GROUP): cv.declare_id(lv_group_t),
|
cv.GenerateID(df.CONF_DEFAULT_GROUP): cv.declare_id(lv_group_t),
|
||||||
cv.Optional(df.CONF_RESUME_ON_INPUT, default=True): cv.boolean,
|
cv.Optional(df.CONF_RESUME_ON_INPUT, default=True): cv.boolean,
|
||||||
}
|
}
|
||||||
|
|
|
@ -438,6 +438,7 @@ CONF_HEADER_MODE = "header_mode"
|
||||||
CONF_HOME = "home"
|
CONF_HOME = "home"
|
||||||
CONF_INITIAL_FOCUS = "initial_focus"
|
CONF_INITIAL_FOCUS = "initial_focus"
|
||||||
CONF_KEY_CODE = "key_code"
|
CONF_KEY_CODE = "key_code"
|
||||||
|
CONF_KEYPADS = "keypads"
|
||||||
CONF_LAYOUT = "layout"
|
CONF_LAYOUT = "layout"
|
||||||
CONF_LEFT_BUTTON = "left_button"
|
CONF_LEFT_BUTTON = "left_button"
|
||||||
CONF_LINE_WIDTH = "line_width"
|
CONF_LINE_WIDTH = "line_width"
|
||||||
|
|
|
@ -17,7 +17,7 @@ from .defines import (
|
||||||
from .helpers import lvgl_components_required, requires_component
|
from .helpers import lvgl_components_required, requires_component
|
||||||
from .lvcode import lv, lv_add, lv_assign, lv_expr, lv_Pvariable
|
from .lvcode import lv, lv_add, lv_assign, lv_expr, lv_Pvariable
|
||||||
from .schemas import ENCODER_SCHEMA
|
from .schemas import ENCODER_SCHEMA
|
||||||
from .types import lv_group_t, lv_indev_type_t
|
from .types import lv_group_t, lv_indev_type_t, lv_key_t
|
||||||
|
|
||||||
ENCODERS_CONFIG = cv.ensure_list(
|
ENCODERS_CONFIG = cv.ensure_list(
|
||||||
ENCODER_SCHEMA.extend(
|
ENCODER_SCHEMA.extend(
|
||||||
|
@ -39,10 +39,13 @@ ENCODERS_CONFIG = cv.ensure_list(
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
async def encoders_to_code(var, config):
|
def get_default_group(config):
|
||||||
default_group = lv_Pvariable(lv_group_t, config[CONF_DEFAULT_GROUP])
|
default_group = cg.Pvariable(config[CONF_DEFAULT_GROUP], lv_expr.group_create())
|
||||||
lv_assign(default_group, lv_expr.group_create())
|
cg.add(lv.group_set_default(default_group))
|
||||||
lv.group_set_default(default_group)
|
return default_group
|
||||||
|
|
||||||
|
|
||||||
|
async def encoders_to_code(var, config, default_group):
|
||||||
for enc_conf in config[CONF_ENCODERS]:
|
for enc_conf in config[CONF_ENCODERS]:
|
||||||
lvgl_components_required.add("KEY_LISTENER")
|
lvgl_components_required.add("KEY_LISTENER")
|
||||||
lpt = enc_conf[CONF_LONG_PRESS_TIME].total_milliseconds
|
lpt = enc_conf[CONF_LONG_PRESS_TIME].total_milliseconds
|
||||||
|
@ -54,14 +57,14 @@ async def encoders_to_code(var, config):
|
||||||
if sensor_config := enc_conf.get(CONF_SENSOR):
|
if sensor_config := enc_conf.get(CONF_SENSOR):
|
||||||
if isinstance(sensor_config, dict):
|
if isinstance(sensor_config, dict):
|
||||||
b_sensor = await cg.get_variable(sensor_config[CONF_LEFT_BUTTON])
|
b_sensor = await cg.get_variable(sensor_config[CONF_LEFT_BUTTON])
|
||||||
cg.add(listener.set_left_button(b_sensor))
|
cg.add(listener.add_button(b_sensor, lv_key_t.LV_KEY_LEFT))
|
||||||
b_sensor = await cg.get_variable(sensor_config[CONF_RIGHT_BUTTON])
|
b_sensor = await cg.get_variable(sensor_config[CONF_RIGHT_BUTTON])
|
||||||
cg.add(listener.set_right_button(b_sensor))
|
cg.add(listener.add_button(b_sensor, lv_key_t.LV_KEY_RIGHT))
|
||||||
else:
|
else:
|
||||||
sensor_config = await cg.get_variable(sensor_config)
|
sensor_config = await cg.get_variable(sensor_config)
|
||||||
lv_add(listener.set_sensor(sensor_config))
|
lv_add(listener.set_sensor(sensor_config))
|
||||||
b_sensor = await cg.get_variable(enc_conf[CONF_ENTER_BUTTON])
|
b_sensor = await cg.get_variable(enc_conf[CONF_ENTER_BUTTON])
|
||||||
cg.add(listener.set_enter_button(b_sensor))
|
cg.add(listener.add_button(b_sensor, lv_key_t.LV_KEY_ENTER))
|
||||||
if group := enc_conf.get(CONF_GROUP):
|
if group := enc_conf.get(CONF_GROUP):
|
||||||
group = lv_Pvariable(lv_group_t, group)
|
group = lv_Pvariable(lv_group_t, group)
|
||||||
lv_assign(group, lv_expr.group_create())
|
lv_assign(group, lv_expr.group_create())
|
||||||
|
|
77
esphome/components/lvgl/keypads.py
Normal file
77
esphome/components/lvgl/keypads.py
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
import esphome.codegen as cg
|
||||||
|
from esphome.components.binary_sensor import BinarySensor
|
||||||
|
import esphome.config_validation as cv
|
||||||
|
from esphome.const import CONF_GROUP, CONF_ID
|
||||||
|
|
||||||
|
from .defines import (
|
||||||
|
CONF_ENCODERS,
|
||||||
|
CONF_INITIAL_FOCUS,
|
||||||
|
CONF_KEYPADS,
|
||||||
|
CONF_LONG_PRESS_REPEAT_TIME,
|
||||||
|
CONF_LONG_PRESS_TIME,
|
||||||
|
literal,
|
||||||
|
)
|
||||||
|
from .helpers import lvgl_components_required
|
||||||
|
from .lvcode import lv, lv_assign, lv_expr, lv_Pvariable
|
||||||
|
from .schemas import ENCODER_SCHEMA
|
||||||
|
from .types import lv_group_t, lv_indev_type_t
|
||||||
|
|
||||||
|
KEYPAD_KEYS = (
|
||||||
|
"up",
|
||||||
|
"down",
|
||||||
|
"right",
|
||||||
|
"left",
|
||||||
|
"esc",
|
||||||
|
"del",
|
||||||
|
"backspace",
|
||||||
|
"enter",
|
||||||
|
"next",
|
||||||
|
"prev",
|
||||||
|
"home",
|
||||||
|
"end",
|
||||||
|
"0",
|
||||||
|
"1",
|
||||||
|
"2",
|
||||||
|
"3",
|
||||||
|
"4",
|
||||||
|
"5",
|
||||||
|
"6",
|
||||||
|
"7",
|
||||||
|
"8",
|
||||||
|
"9",
|
||||||
|
"#",
|
||||||
|
"*",
|
||||||
|
)
|
||||||
|
|
||||||
|
KEYPADS_CONFIG = cv.ensure_list(
|
||||||
|
ENCODER_SCHEMA.extend(
|
||||||
|
{cv.Optional(key): cv.use_id(BinarySensor) for key in KEYPAD_KEYS}
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def keypads_to_code(var, config, default_group):
|
||||||
|
for enc_conf in config[CONF_KEYPADS]:
|
||||||
|
lvgl_components_required.add("KEY_LISTENER")
|
||||||
|
lpt = enc_conf[CONF_LONG_PRESS_TIME].total_milliseconds
|
||||||
|
lprt = enc_conf[CONF_LONG_PRESS_REPEAT_TIME].total_milliseconds
|
||||||
|
listener = cg.new_Pvariable(
|
||||||
|
enc_conf[CONF_ID], lv_indev_type_t.LV_INDEV_TYPE_KEYPAD, lpt, lprt
|
||||||
|
)
|
||||||
|
await cg.register_parented(listener, var)
|
||||||
|
for key in [x for x in enc_conf if x in KEYPAD_KEYS]:
|
||||||
|
b_sensor = await cg.get_variable(enc_conf[key])
|
||||||
|
cg.add(listener.add_button(b_sensor, literal(f"LV_KEY_{key.upper()}")))
|
||||||
|
if group := enc_conf.get(CONF_GROUP):
|
||||||
|
group = lv_Pvariable(lv_group_t, group)
|
||||||
|
lv_assign(group, lv_expr.group_create())
|
||||||
|
else:
|
||||||
|
group = default_group
|
||||||
|
lv.indev_set_group(lv_expr.indev_drv_register(listener.get_drv()), group)
|
||||||
|
|
||||||
|
|
||||||
|
async def initial_focus_to_code(config):
|
||||||
|
for enc_conf in config[CONF_ENCODERS]:
|
||||||
|
if default_focus := enc_conf.get(CONF_INITIAL_FOCUS):
|
||||||
|
obj = await cg.get_variable(default_focus)
|
||||||
|
lv.group_focus_obj(obj)
|
|
@ -256,15 +256,8 @@ class LVEncoderListener : public Parented<LvglComponent> {
|
||||||
LVEncoderListener(lv_indev_type_t type, uint16_t lpt, uint16_t lprt);
|
LVEncoderListener(lv_indev_type_t type, uint16_t lpt, uint16_t lprt);
|
||||||
|
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
void set_left_button(binary_sensor::BinarySensor *left_button) {
|
void add_button(binary_sensor::BinarySensor *button, lv_key_t key) {
|
||||||
left_button->add_on_state_callback([this](bool state) { this->event(LV_KEY_LEFT, state); });
|
button->add_on_state_callback([this, key](bool state) { this->event(key, state); });
|
||||||
}
|
|
||||||
void set_right_button(binary_sensor::BinarySensor *right_button) {
|
|
||||||
right_button->add_on_state_callback([this](bool state) { this->event(LV_KEY_RIGHT, state); });
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_enter_button(binary_sensor::BinarySensor *enter_button) {
|
|
||||||
enter_button->add_on_state_callback([this](bool state) { this->event(LV_KEY_ENTER, state); });
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
|
@ -40,6 +40,7 @@ void_ptr = cg.void.operator("ptr")
|
||||||
lv_coord_t = cg.global_ns.namespace("lv_coord_t")
|
lv_coord_t = cg.global_ns.namespace("lv_coord_t")
|
||||||
lv_event_code_t = cg.global_ns.enum("lv_event_code_t")
|
lv_event_code_t = cg.global_ns.enum("lv_event_code_t")
|
||||||
lv_indev_type_t = cg.global_ns.enum("lv_indev_type_t")
|
lv_indev_type_t = cg.global_ns.enum("lv_indev_type_t")
|
||||||
|
lv_key_t = cg.global_ns.enum("lv_key_t")
|
||||||
FontEngine = lvgl_ns.class_("FontEngine")
|
FontEngine = lvgl_ns.class_("FontEngine")
|
||||||
IdleTrigger = lvgl_ns.class_("IdleTrigger", automation.Trigger.template())
|
IdleTrigger = lvgl_ns.class_("IdleTrigger", automation.Trigger.template())
|
||||||
PauseTrigger = lvgl_ns.class_("PauseTrigger", automation.Trigger.template())
|
PauseTrigger = lvgl_ns.class_("PauseTrigger", automation.Trigger.template())
|
||||||
|
|
|
@ -3,6 +3,8 @@
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
#include "air_conditioner.h"
|
#include "air_conditioner.h"
|
||||||
#include "ac_adapter.h"
|
#include "ac_adapter.h"
|
||||||
|
#include <cmath>
|
||||||
|
#include <cstdint>
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace midea {
|
namespace midea {
|
||||||
|
@ -121,7 +123,21 @@ void AirConditioner::dump_config() {
|
||||||
|
|
||||||
void AirConditioner::do_follow_me(float temperature, bool beeper) {
|
void AirConditioner::do_follow_me(float temperature, bool beeper) {
|
||||||
#ifdef USE_REMOTE_TRANSMITTER
|
#ifdef USE_REMOTE_TRANSMITTER
|
||||||
IrFollowMeData data(static_cast<uint8_t>(lroundf(temperature)), beeper);
|
// Check if temperature is finite (not NaN or infinite)
|
||||||
|
if (!std::isfinite(temperature)) {
|
||||||
|
ESP_LOGW(Constants::TAG, "Follow me action requires a finite temperature, got: %f", temperature);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Round and convert temperature to long, then clamp and convert it to uint8_t
|
||||||
|
uint8_t temp_uint8 =
|
||||||
|
static_cast<uint8_t>(std::max(0L, std::min(static_cast<long>(UINT8_MAX), std::lroundf(temperature))));
|
||||||
|
|
||||||
|
ESP_LOGD(Constants::TAG, "Follow me action called with temperature: %f °C, rounded to: %u °C", temperature,
|
||||||
|
temp_uint8);
|
||||||
|
|
||||||
|
// Create and transmit the data
|
||||||
|
IrFollowMeData data(temp_uint8, beeper);
|
||||||
this->transmitter_.transmit(data);
|
this->transmitter_.transmit(data);
|
||||||
#else
|
#else
|
||||||
ESP_LOGW(Constants::TAG, "Action needs remote_transmitter component");
|
ESP_LOGW(Constants::TAG, "Action needs remote_transmitter component");
|
||||||
|
|
|
@ -335,19 +335,28 @@ def sensor_schema(
|
||||||
return SENSOR_SCHEMA.extend(schema)
|
return SENSOR_SCHEMA.extend(schema)
|
||||||
|
|
||||||
|
|
||||||
@FILTER_REGISTRY.register("offset", OffsetFilter, cv.float_)
|
@FILTER_REGISTRY.register("offset", OffsetFilter, cv.templatable(cv.float_))
|
||||||
async def offset_filter_to_code(config, filter_id):
|
async def offset_filter_to_code(config, filter_id):
|
||||||
return cg.new_Pvariable(filter_id, config)
|
template_ = await cg.templatable(config, [], float)
|
||||||
|
return cg.new_Pvariable(filter_id, template_)
|
||||||
|
|
||||||
|
|
||||||
@FILTER_REGISTRY.register("multiply", MultiplyFilter, cv.float_)
|
@FILTER_REGISTRY.register("multiply", MultiplyFilter, cv.templatable(cv.float_))
|
||||||
async def multiply_filter_to_code(config, filter_id):
|
async def multiply_filter_to_code(config, filter_id):
|
||||||
return cg.new_Pvariable(filter_id, config)
|
template_ = await cg.templatable(config, [], float)
|
||||||
|
return cg.new_Pvariable(filter_id, template_)
|
||||||
|
|
||||||
|
|
||||||
@FILTER_REGISTRY.register("filter_out", FilterOutValueFilter, cv.float_)
|
@FILTER_REGISTRY.register(
|
||||||
|
"filter_out",
|
||||||
|
FilterOutValueFilter,
|
||||||
|
cv.Any(cv.templatable(cv.float_), [cv.templatable(cv.float_)]),
|
||||||
|
)
|
||||||
async def filter_out_filter_to_code(config, filter_id):
|
async def filter_out_filter_to_code(config, filter_id):
|
||||||
return cg.new_Pvariable(filter_id, config)
|
if not isinstance(config, list):
|
||||||
|
config = [config]
|
||||||
|
template_ = [await cg.templatable(x, [], float) for x in config]
|
||||||
|
return cg.new_Pvariable(filter_id, template_)
|
||||||
|
|
||||||
|
|
||||||
QUANTILE_SCHEMA = cv.All(
|
QUANTILE_SCHEMA = cv.All(
|
||||||
|
@ -573,7 +582,7 @@ async def heartbeat_filter_to_code(config, filter_id):
|
||||||
TIMEOUT_SCHEMA = cv.maybe_simple_value(
|
TIMEOUT_SCHEMA = cv.maybe_simple_value(
|
||||||
{
|
{
|
||||||
cv.Required(CONF_TIMEOUT): cv.positive_time_period_milliseconds,
|
cv.Required(CONF_TIMEOUT): cv.positive_time_period_milliseconds,
|
||||||
cv.Optional(CONF_VALUE, default="nan"): cv.float_,
|
cv.Optional(CONF_VALUE, default="nan"): cv.templatable(cv.float_),
|
||||||
},
|
},
|
||||||
key=CONF_TIMEOUT,
|
key=CONF_TIMEOUT,
|
||||||
)
|
)
|
||||||
|
@ -581,7 +590,8 @@ TIMEOUT_SCHEMA = cv.maybe_simple_value(
|
||||||
|
|
||||||
@FILTER_REGISTRY.register("timeout", TimeoutFilter, TIMEOUT_SCHEMA)
|
@FILTER_REGISTRY.register("timeout", TimeoutFilter, TIMEOUT_SCHEMA)
|
||||||
async def timeout_filter_to_code(config, filter_id):
|
async def timeout_filter_to_code(config, filter_id):
|
||||||
var = cg.new_Pvariable(filter_id, config[CONF_TIMEOUT], config[CONF_VALUE])
|
template_ = await cg.templatable(config[CONF_VALUE], [], float)
|
||||||
|
var = cg.new_Pvariable(filter_id, config[CONF_TIMEOUT], template_)
|
||||||
await cg.register_component(var, {})
|
await cg.register_component(var, {})
|
||||||
return var
|
return var
|
||||||
|
|
||||||
|
|
|
@ -288,37 +288,37 @@ optional<float> LambdaFilter::new_value(float value) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// OffsetFilter
|
// OffsetFilter
|
||||||
OffsetFilter::OffsetFilter(float offset) : offset_(offset) {}
|
OffsetFilter::OffsetFilter(TemplatableValue<float> offset) : offset_(std::move(offset)) {}
|
||||||
|
|
||||||
optional<float> OffsetFilter::new_value(float value) { return value + this->offset_; }
|
optional<float> OffsetFilter::new_value(float value) { return value + this->offset_.value(); }
|
||||||
|
|
||||||
// MultiplyFilter
|
// MultiplyFilter
|
||||||
MultiplyFilter::MultiplyFilter(float multiplier) : multiplier_(multiplier) {}
|
MultiplyFilter::MultiplyFilter(TemplatableValue<float> multiplier) : multiplier_(std::move(multiplier)) {}
|
||||||
|
|
||||||
optional<float> MultiplyFilter::new_value(float value) { return value * this->multiplier_; }
|
optional<float> MultiplyFilter::new_value(float value) { return value * this->multiplier_.value(); }
|
||||||
|
|
||||||
// FilterOutValueFilter
|
// FilterOutValueFilter
|
||||||
FilterOutValueFilter::FilterOutValueFilter(float value_to_filter_out) : value_to_filter_out_(value_to_filter_out) {}
|
FilterOutValueFilter::FilterOutValueFilter(std::vector<TemplatableValue<float>> values_to_filter_out)
|
||||||
|
: values_to_filter_out_(std::move(values_to_filter_out)) {}
|
||||||
|
|
||||||
optional<float> FilterOutValueFilter::new_value(float value) {
|
optional<float> FilterOutValueFilter::new_value(float value) {
|
||||||
if (std::isnan(this->value_to_filter_out_)) {
|
|
||||||
if (std::isnan(value)) {
|
|
||||||
return {};
|
|
||||||
} else {
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
int8_t accuracy = this->parent_->get_accuracy_decimals();
|
int8_t accuracy = this->parent_->get_accuracy_decimals();
|
||||||
float accuracy_mult = powf(10.0f, accuracy);
|
float accuracy_mult = powf(10.0f, accuracy);
|
||||||
float rounded_filter_out = roundf(accuracy_mult * this->value_to_filter_out_);
|
for (auto filter_value : this->values_to_filter_out_) {
|
||||||
|
if (std::isnan(filter_value.value())) {
|
||||||
|
if (std::isnan(value)) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
float rounded_filter_out = roundf(accuracy_mult * filter_value.value());
|
||||||
float rounded_value = roundf(accuracy_mult * value);
|
float rounded_value = roundf(accuracy_mult * value);
|
||||||
if (rounded_filter_out == rounded_value) {
|
if (rounded_filter_out == rounded_value) {
|
||||||
return {};
|
return {};
|
||||||
} else {
|
}
|
||||||
|
}
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// ThrottleFilter
|
// ThrottleFilter
|
||||||
ThrottleFilter::ThrottleFilter(uint32_t min_time_between_inputs) : min_time_between_inputs_(min_time_between_inputs) {}
|
ThrottleFilter::ThrottleFilter(uint32_t min_time_between_inputs) : min_time_between_inputs_(min_time_between_inputs) {}
|
||||||
|
@ -383,11 +383,12 @@ void OrFilter::initialize(Sensor *parent, Filter *next) {
|
||||||
|
|
||||||
// TimeoutFilter
|
// TimeoutFilter
|
||||||
optional<float> TimeoutFilter::new_value(float value) {
|
optional<float> TimeoutFilter::new_value(float value) {
|
||||||
this->set_timeout("timeout", this->time_period_, [this]() { this->output(this->value_); });
|
this->set_timeout("timeout", this->time_period_, [this]() { this->output(this->value_.value()); });
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
TimeoutFilter::TimeoutFilter(uint32_t time_period, float new_value) : time_period_(time_period), value_(new_value) {}
|
TimeoutFilter::TimeoutFilter(uint32_t time_period, TemplatableValue<float> new_value)
|
||||||
|
: time_period_(time_period), value_(std::move(new_value)) {}
|
||||||
float TimeoutFilter::get_setup_priority() const { return setup_priority::HARDWARE; }
|
float TimeoutFilter::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||||
|
|
||||||
// DebounceFilter
|
// DebounceFilter
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include "esphome/core/component.h"
|
#include "esphome/core/component.h"
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
|
#include "esphome/core/automation.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace sensor {
|
namespace sensor {
|
||||||
|
@ -273,34 +274,33 @@ class LambdaFilter : public Filter {
|
||||||
/// A simple filter that adds `offset` to each value it receives.
|
/// A simple filter that adds `offset` to each value it receives.
|
||||||
class OffsetFilter : public Filter {
|
class OffsetFilter : public Filter {
|
||||||
public:
|
public:
|
||||||
explicit OffsetFilter(float offset);
|
explicit OffsetFilter(TemplatableValue<float> offset);
|
||||||
|
|
||||||
optional<float> new_value(float value) override;
|
optional<float> new_value(float value) override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
float offset_;
|
TemplatableValue<float> offset_;
|
||||||
};
|
};
|
||||||
|
|
||||||
/// A simple filter that multiplies to each value it receives by `multiplier`.
|
/// A simple filter that multiplies to each value it receives by `multiplier`.
|
||||||
class MultiplyFilter : public Filter {
|
class MultiplyFilter : public Filter {
|
||||||
public:
|
public:
|
||||||
explicit MultiplyFilter(float multiplier);
|
explicit MultiplyFilter(TemplatableValue<float> multiplier);
|
||||||
|
|
||||||
optional<float> new_value(float value) override;
|
optional<float> new_value(float value) override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
float multiplier_;
|
TemplatableValue<float> multiplier_;
|
||||||
};
|
};
|
||||||
|
|
||||||
/// A simple filter that only forwards the filter chain if it doesn't receive `value_to_filter_out`.
|
/// A simple filter that only forwards the filter chain if it doesn't receive `value_to_filter_out`.
|
||||||
class FilterOutValueFilter : public Filter {
|
class FilterOutValueFilter : public Filter {
|
||||||
public:
|
public:
|
||||||
explicit FilterOutValueFilter(float value_to_filter_out);
|
explicit FilterOutValueFilter(std::vector<TemplatableValue<float>> values_to_filter_out);
|
||||||
|
|
||||||
optional<float> new_value(float value) override;
|
optional<float> new_value(float value) override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
float value_to_filter_out_;
|
std::vector<TemplatableValue<float>> values_to_filter_out_;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ThrottleFilter : public Filter {
|
class ThrottleFilter : public Filter {
|
||||||
|
@ -316,8 +316,7 @@ class ThrottleFilter : public Filter {
|
||||||
|
|
||||||
class TimeoutFilter : public Filter, public Component {
|
class TimeoutFilter : public Filter, public Component {
|
||||||
public:
|
public:
|
||||||
explicit TimeoutFilter(uint32_t time_period, float new_value);
|
explicit TimeoutFilter(uint32_t time_period, TemplatableValue<float> new_value);
|
||||||
void set_value(float new_value) { this->value_ = new_value; }
|
|
||||||
|
|
||||||
optional<float> new_value(float value) override;
|
optional<float> new_value(float value) override;
|
||||||
|
|
||||||
|
@ -325,7 +324,7 @@ class TimeoutFilter : public Filter, public Component {
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
uint32_t time_period_;
|
uint32_t time_period_;
|
||||||
float value_;
|
TemplatableValue<float> value_;
|
||||||
};
|
};
|
||||||
|
|
||||||
class DebounceFilter : public Filter, public Component {
|
class DebounceFilter : public Filter, public Component {
|
||||||
|
|
|
@ -11,6 +11,12 @@ substitutions:
|
||||||
check: "\U000F012C"
|
check: "\U000F012C"
|
||||||
arrow_down: "\U000F004B"
|
arrow_down: "\U000F004B"
|
||||||
|
|
||||||
|
binary_sensor:
|
||||||
|
- id: enter_sensor
|
||||||
|
platform: template
|
||||||
|
- id: left_sensor
|
||||||
|
platform: template
|
||||||
|
|
||||||
lvgl:
|
lvgl:
|
||||||
log_level: debug
|
log_level: debug
|
||||||
resume_on_input: true
|
resume_on_input: true
|
||||||
|
@ -93,6 +99,10 @@ lvgl:
|
||||||
- touchscreen_id: tft_touch
|
- touchscreen_id: tft_touch
|
||||||
long_press_repeat_time: 200ms
|
long_press_repeat_time: 200ms
|
||||||
long_press_time: 500ms
|
long_press_time: 500ms
|
||||||
|
keypads:
|
||||||
|
- initial_focus: button_button
|
||||||
|
enter: enter_sensor
|
||||||
|
next: left_sensor
|
||||||
|
|
||||||
msgboxes:
|
msgboxes:
|
||||||
- id: message_box
|
- id: message_box
|
||||||
|
|
|
@ -9,6 +9,25 @@ sensor:
|
||||||
return 0.0;
|
return 0.0;
|
||||||
}
|
}
|
||||||
update_interval: 60s
|
update_interval: 60s
|
||||||
|
filters:
|
||||||
|
- offset: 10
|
||||||
|
- multiply: 1
|
||||||
|
- offset: !lambda return 10;
|
||||||
|
- multiply: !lambda return 2;
|
||||||
|
- filter_out:
|
||||||
|
- 10
|
||||||
|
- 20
|
||||||
|
- !lambda return 10;
|
||||||
|
- filter_out: 10
|
||||||
|
- filter_out: !lambda return NAN;
|
||||||
|
- timeout:
|
||||||
|
timeout: 10s
|
||||||
|
value: !lambda return 10;
|
||||||
|
- timeout:
|
||||||
|
timeout: 1h
|
||||||
|
value: 20.0
|
||||||
|
- timeout:
|
||||||
|
timeout: 1d
|
||||||
|
|
||||||
esphome:
|
esphome:
|
||||||
on_boot:
|
on_boot:
|
||||||
|
|
Loading…
Reference in a new issue