mirror of
https://github.com/esphome/esphome.git
synced 2025-10-24 04:33:49 +01:00
Reduce LightCall memory usage by 50 bytes per call (#9333)
This commit is contained in:
@@ -97,12 +97,12 @@ class AddressableLight : public LightOutput, public Component {
|
|||||||
}
|
}
|
||||||
virtual ESPColorView get_view_internal(int32_t index) const = 0;
|
virtual ESPColorView get_view_internal(int32_t index) const = 0;
|
||||||
|
|
||||||
bool effect_active_{false};
|
|
||||||
ESPColorCorrection correction_{};
|
ESPColorCorrection correction_{};
|
||||||
|
LightState *state_parent_{nullptr};
|
||||||
#ifdef USE_POWER_SUPPLY
|
#ifdef USE_POWER_SUPPLY
|
||||||
power_supply::PowerSupplyRequester power_;
|
power_supply::PowerSupplyRequester power_;
|
||||||
#endif
|
#endif
|
||||||
LightState *state_parent_{nullptr};
|
bool effect_active_{false};
|
||||||
};
|
};
|
||||||
|
|
||||||
class AddressableLightTransformer : public LightTransitionTransformer {
|
class AddressableLightTransformer : public LightTransitionTransformer {
|
||||||
@@ -114,9 +114,9 @@ class AddressableLightTransformer : public LightTransitionTransformer {
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
AddressableLight &light_;
|
AddressableLight &light_;
|
||||||
Color target_color_{};
|
|
||||||
float last_transition_progress_{0.0f};
|
float last_transition_progress_{0.0f};
|
||||||
float accumulated_alpha_{0.0f};
|
float accumulated_alpha_{0.0f};
|
||||||
|
Color target_color_{};
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace light
|
} // namespace light
|
||||||
|
@@ -69,8 +69,8 @@ class ESPColorCorrection {
|
|||||||
protected:
|
protected:
|
||||||
uint8_t gamma_table_[256];
|
uint8_t gamma_table_[256];
|
||||||
uint8_t gamma_reverse_table_[256];
|
uint8_t gamma_reverse_table_[256];
|
||||||
uint8_t local_brightness_{255};
|
|
||||||
Color max_brightness_;
|
Color max_brightness_;
|
||||||
|
uint8_t local_brightness_{255};
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace light
|
} // namespace light
|
||||||
|
@@ -2,12 +2,28 @@
|
|||||||
#include "light_call.h"
|
#include "light_call.h"
|
||||||
#include "light_state.h"
|
#include "light_state.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
#include "esphome/core/optional.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace light {
|
namespace light {
|
||||||
|
|
||||||
static const char *const TAG = "light";
|
static const char *const TAG = "light";
|
||||||
|
|
||||||
|
// Macro to reduce repetitive setter code
|
||||||
|
#define IMPLEMENT_LIGHT_CALL_SETTER(name, type, flag) \
|
||||||
|
LightCall &LightCall::set_##name(optional<type>(name)) { \
|
||||||
|
if ((name).has_value()) { \
|
||||||
|
this->name##_ = (name).value(); \
|
||||||
|
} \
|
||||||
|
this->set_flag_(flag, (name).has_value()); \
|
||||||
|
return *this; \
|
||||||
|
} \
|
||||||
|
LightCall &LightCall::set_##name(type name) { \
|
||||||
|
this->name##_ = name; \
|
||||||
|
this->set_flag_(flag, true); \
|
||||||
|
return *this; \
|
||||||
|
}
|
||||||
|
|
||||||
static const LogString *color_mode_to_human(ColorMode color_mode) {
|
static const LogString *color_mode_to_human(ColorMode color_mode) {
|
||||||
if (color_mode == ColorMode::UNKNOWN)
|
if (color_mode == ColorMode::UNKNOWN)
|
||||||
return LOG_STR("Unknown");
|
return LOG_STR("Unknown");
|
||||||
@@ -32,41 +48,43 @@ void LightCall::perform() {
|
|||||||
const char *name = this->parent_->get_name().c_str();
|
const char *name = this->parent_->get_name().c_str();
|
||||||
LightColorValues v = this->validate_();
|
LightColorValues v = this->validate_();
|
||||||
|
|
||||||
if (this->publish_) {
|
if (this->get_publish_()) {
|
||||||
ESP_LOGD(TAG, "'%s' Setting:", name);
|
ESP_LOGD(TAG, "'%s' Setting:", name);
|
||||||
|
|
||||||
// Only print color mode when it's being changed
|
// Only print color mode when it's being changed
|
||||||
ColorMode current_color_mode = this->parent_->remote_values.get_color_mode();
|
ColorMode current_color_mode = this->parent_->remote_values.get_color_mode();
|
||||||
if (this->color_mode_.value_or(current_color_mode) != current_color_mode) {
|
ColorMode target_color_mode = this->has_color_mode() ? this->color_mode_ : current_color_mode;
|
||||||
|
if (target_color_mode != current_color_mode) {
|
||||||
ESP_LOGD(TAG, " Color mode: %s", LOG_STR_ARG(color_mode_to_human(v.get_color_mode())));
|
ESP_LOGD(TAG, " Color mode: %s", LOG_STR_ARG(color_mode_to_human(v.get_color_mode())));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Only print state when it's being changed
|
// Only print state when it's being changed
|
||||||
bool current_state = this->parent_->remote_values.is_on();
|
bool current_state = this->parent_->remote_values.is_on();
|
||||||
if (this->state_.value_or(current_state) != current_state) {
|
bool target_state = this->has_state() ? this->state_ : current_state;
|
||||||
|
if (target_state != current_state) {
|
||||||
ESP_LOGD(TAG, " State: %s", ONOFF(v.is_on()));
|
ESP_LOGD(TAG, " State: %s", ONOFF(v.is_on()));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->brightness_.has_value()) {
|
if (this->has_brightness()) {
|
||||||
ESP_LOGD(TAG, " Brightness: %.0f%%", v.get_brightness() * 100.0f);
|
ESP_LOGD(TAG, " Brightness: %.0f%%", v.get_brightness() * 100.0f);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->color_brightness_.has_value()) {
|
if (this->has_color_brightness()) {
|
||||||
ESP_LOGD(TAG, " Color brightness: %.0f%%", v.get_color_brightness() * 100.0f);
|
ESP_LOGD(TAG, " Color brightness: %.0f%%", v.get_color_brightness() * 100.0f);
|
||||||
}
|
}
|
||||||
if (this->red_.has_value() || this->green_.has_value() || this->blue_.has_value()) {
|
if (this->has_red() || this->has_green() || this->has_blue()) {
|
||||||
ESP_LOGD(TAG, " Red: %.0f%%, Green: %.0f%%, Blue: %.0f%%", v.get_red() * 100.0f, v.get_green() * 100.0f,
|
ESP_LOGD(TAG, " Red: %.0f%%, Green: %.0f%%, Blue: %.0f%%", v.get_red() * 100.0f, v.get_green() * 100.0f,
|
||||||
v.get_blue() * 100.0f);
|
v.get_blue() * 100.0f);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->white_.has_value()) {
|
if (this->has_white()) {
|
||||||
ESP_LOGD(TAG, " White: %.0f%%", v.get_white() * 100.0f);
|
ESP_LOGD(TAG, " White: %.0f%%", v.get_white() * 100.0f);
|
||||||
}
|
}
|
||||||
if (this->color_temperature_.has_value()) {
|
if (this->has_color_temperature()) {
|
||||||
ESP_LOGD(TAG, " Color temperature: %.1f mireds", v.get_color_temperature());
|
ESP_LOGD(TAG, " Color temperature: %.1f mireds", v.get_color_temperature());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->cold_white_.has_value() || this->warm_white_.has_value()) {
|
if (this->has_cold_white() || this->has_warm_white()) {
|
||||||
ESP_LOGD(TAG, " Cold white: %.0f%%, warm white: %.0f%%", v.get_cold_white() * 100.0f,
|
ESP_LOGD(TAG, " Cold white: %.0f%%, warm white: %.0f%%", v.get_cold_white() * 100.0f,
|
||||||
v.get_warm_white() * 100.0f);
|
v.get_warm_white() * 100.0f);
|
||||||
}
|
}
|
||||||
@@ -74,58 +92,57 @@ void LightCall::perform() {
|
|||||||
|
|
||||||
if (this->has_flash_()) {
|
if (this->has_flash_()) {
|
||||||
// FLASH
|
// FLASH
|
||||||
if (this->publish_) {
|
if (this->get_publish_()) {
|
||||||
ESP_LOGD(TAG, " Flash length: %.1fs", *this->flash_length_ / 1e3f);
|
ESP_LOGD(TAG, " Flash length: %.1fs", this->flash_length_ / 1e3f);
|
||||||
}
|
}
|
||||||
|
|
||||||
this->parent_->start_flash_(v, *this->flash_length_, this->publish_);
|
this->parent_->start_flash_(v, this->flash_length_, this->get_publish_());
|
||||||
} else if (this->has_transition_()) {
|
} else if (this->has_transition_()) {
|
||||||
// TRANSITION
|
// TRANSITION
|
||||||
if (this->publish_) {
|
if (this->get_publish_()) {
|
||||||
ESP_LOGD(TAG, " Transition length: %.1fs", *this->transition_length_ / 1e3f);
|
ESP_LOGD(TAG, " Transition length: %.1fs", this->transition_length_ / 1e3f);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Special case: Transition and effect can be set when turning off
|
// Special case: Transition and effect can be set when turning off
|
||||||
if (this->has_effect_()) {
|
if (this->has_effect_()) {
|
||||||
if (this->publish_) {
|
if (this->get_publish_()) {
|
||||||
ESP_LOGD(TAG, " Effect: 'None'");
|
ESP_LOGD(TAG, " Effect: 'None'");
|
||||||
}
|
}
|
||||||
this->parent_->stop_effect_();
|
this->parent_->stop_effect_();
|
||||||
}
|
}
|
||||||
|
|
||||||
this->parent_->start_transition_(v, *this->transition_length_, this->publish_);
|
this->parent_->start_transition_(v, this->transition_length_, this->get_publish_());
|
||||||
|
|
||||||
} else if (this->has_effect_()) {
|
} else if (this->has_effect_()) {
|
||||||
// EFFECT
|
// EFFECT
|
||||||
auto effect = this->effect_;
|
|
||||||
const char *effect_s;
|
const char *effect_s;
|
||||||
if (effect == 0u) {
|
if (this->effect_ == 0u) {
|
||||||
effect_s = "None";
|
effect_s = "None";
|
||||||
} else {
|
} else {
|
||||||
effect_s = this->parent_->effects_[*this->effect_ - 1]->get_name().c_str();
|
effect_s = this->parent_->effects_[this->effect_ - 1]->get_name().c_str();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->publish_) {
|
if (this->get_publish_()) {
|
||||||
ESP_LOGD(TAG, " Effect: '%s'", effect_s);
|
ESP_LOGD(TAG, " Effect: '%s'", effect_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
this->parent_->start_effect_(*this->effect_);
|
this->parent_->start_effect_(this->effect_);
|
||||||
|
|
||||||
// Also set light color values when starting an effect
|
// Also set light color values when starting an effect
|
||||||
// For example to turn off the light
|
// For example to turn off the light
|
||||||
this->parent_->set_immediately_(v, true);
|
this->parent_->set_immediately_(v, true);
|
||||||
} else {
|
} else {
|
||||||
// INSTANT CHANGE
|
// INSTANT CHANGE
|
||||||
this->parent_->set_immediately_(v, this->publish_);
|
this->parent_->set_immediately_(v, this->get_publish_());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!this->has_transition_()) {
|
if (!this->has_transition_()) {
|
||||||
this->parent_->target_state_reached_callback_.call();
|
this->parent_->target_state_reached_callback_.call();
|
||||||
}
|
}
|
||||||
if (this->publish_) {
|
if (this->get_publish_()) {
|
||||||
this->parent_->publish_state();
|
this->parent_->publish_state();
|
||||||
}
|
}
|
||||||
if (this->save_) {
|
if (this->get_save_()) {
|
||||||
this->parent_->save_remote_values_();
|
this->parent_->save_remote_values_();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -135,82 +152,80 @@ LightColorValues LightCall::validate_() {
|
|||||||
auto traits = this->parent_->get_traits();
|
auto traits = this->parent_->get_traits();
|
||||||
|
|
||||||
// Color mode check
|
// Color mode check
|
||||||
if (this->color_mode_.has_value() && !traits.supports_color_mode(this->color_mode_.value())) {
|
if (this->has_color_mode() && !traits.supports_color_mode(this->color_mode_)) {
|
||||||
ESP_LOGW(TAG, "'%s' does not support color mode %s", name,
|
ESP_LOGW(TAG, "'%s' does not support color mode %s", name, LOG_STR_ARG(color_mode_to_human(this->color_mode_)));
|
||||||
LOG_STR_ARG(color_mode_to_human(this->color_mode_.value())));
|
this->set_flag_(FLAG_HAS_COLOR_MODE, false);
|
||||||
this->color_mode_.reset();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure there is always a color mode set
|
// Ensure there is always a color mode set
|
||||||
if (!this->color_mode_.has_value()) {
|
if (!this->has_color_mode()) {
|
||||||
this->color_mode_ = this->compute_color_mode_();
|
this->color_mode_ = this->compute_color_mode_();
|
||||||
|
this->set_flag_(FLAG_HAS_COLOR_MODE, true);
|
||||||
}
|
}
|
||||||
auto color_mode = *this->color_mode_;
|
auto color_mode = this->color_mode_;
|
||||||
|
|
||||||
// Transform calls that use non-native parameters for the current mode.
|
// Transform calls that use non-native parameters for the current mode.
|
||||||
this->transform_parameters_();
|
this->transform_parameters_();
|
||||||
|
|
||||||
// Brightness exists check
|
// Brightness exists check
|
||||||
if (this->brightness_.has_value() && *this->brightness_ > 0.0f && !(color_mode & ColorCapability::BRIGHTNESS)) {
|
if (this->has_brightness() && this->brightness_ > 0.0f && !(color_mode & ColorCapability::BRIGHTNESS)) {
|
||||||
ESP_LOGW(TAG, "'%s': setting brightness not supported", name);
|
ESP_LOGW(TAG, "'%s': setting brightness not supported", name);
|
||||||
this->brightness_.reset();
|
this->set_flag_(FLAG_HAS_BRIGHTNESS, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Transition length possible check
|
// Transition length possible check
|
||||||
if (this->transition_length_.has_value() && *this->transition_length_ != 0 &&
|
if (this->has_transition_() && this->transition_length_ != 0 && !(color_mode & ColorCapability::BRIGHTNESS)) {
|
||||||
!(color_mode & ColorCapability::BRIGHTNESS)) {
|
|
||||||
ESP_LOGW(TAG, "'%s': transitions not supported", name);
|
ESP_LOGW(TAG, "'%s': transitions not supported", name);
|
||||||
this->transition_length_.reset();
|
this->set_flag_(FLAG_HAS_TRANSITION, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Color brightness exists check
|
// Color brightness exists check
|
||||||
if (this->color_brightness_.has_value() && *this->color_brightness_ > 0.0f && !(color_mode & ColorCapability::RGB)) {
|
if (this->has_color_brightness() && this->color_brightness_ > 0.0f && !(color_mode & ColorCapability::RGB)) {
|
||||||
ESP_LOGW(TAG, "'%s': color mode does not support setting RGB brightness", name);
|
ESP_LOGW(TAG, "'%s': color mode does not support setting RGB brightness", name);
|
||||||
this->color_brightness_.reset();
|
this->set_flag_(FLAG_HAS_COLOR_BRIGHTNESS, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// RGB exists check
|
// RGB exists check
|
||||||
if ((this->red_.has_value() && *this->red_ > 0.0f) || (this->green_.has_value() && *this->green_ > 0.0f) ||
|
if ((this->has_red() && this->red_ > 0.0f) || (this->has_green() && this->green_ > 0.0f) ||
|
||||||
(this->blue_.has_value() && *this->blue_ > 0.0f)) {
|
(this->has_blue() && this->blue_ > 0.0f)) {
|
||||||
if (!(color_mode & ColorCapability::RGB)) {
|
if (!(color_mode & ColorCapability::RGB)) {
|
||||||
ESP_LOGW(TAG, "'%s': color mode does not support setting RGB color", name);
|
ESP_LOGW(TAG, "'%s': color mode does not support setting RGB color", name);
|
||||||
this->red_.reset();
|
this->set_flag_(FLAG_HAS_RED, false);
|
||||||
this->green_.reset();
|
this->set_flag_(FLAG_HAS_GREEN, false);
|
||||||
this->blue_.reset();
|
this->set_flag_(FLAG_HAS_BLUE, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// White value exists check
|
// White value exists check
|
||||||
if (this->white_.has_value() && *this->white_ > 0.0f &&
|
if (this->has_white() && this->white_ > 0.0f &&
|
||||||
!(color_mode & ColorCapability::WHITE || color_mode & ColorCapability::COLD_WARM_WHITE)) {
|
!(color_mode & ColorCapability::WHITE || color_mode & ColorCapability::COLD_WARM_WHITE)) {
|
||||||
ESP_LOGW(TAG, "'%s': color mode does not support setting white value", name);
|
ESP_LOGW(TAG, "'%s': color mode does not support setting white value", name);
|
||||||
this->white_.reset();
|
this->set_flag_(FLAG_HAS_WHITE, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Color temperature exists check
|
// Color temperature exists check
|
||||||
if (this->color_temperature_.has_value() &&
|
if (this->has_color_temperature() &&
|
||||||
!(color_mode & ColorCapability::COLOR_TEMPERATURE || color_mode & ColorCapability::COLD_WARM_WHITE)) {
|
!(color_mode & ColorCapability::COLOR_TEMPERATURE || color_mode & ColorCapability::COLD_WARM_WHITE)) {
|
||||||
ESP_LOGW(TAG, "'%s': color mode does not support setting color temperature", name);
|
ESP_LOGW(TAG, "'%s': color mode does not support setting color temperature", name);
|
||||||
this->color_temperature_.reset();
|
this->set_flag_(FLAG_HAS_COLOR_TEMPERATURE, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cold/warm white value exists check
|
// Cold/warm white value exists check
|
||||||
if ((this->cold_white_.has_value() && *this->cold_white_ > 0.0f) ||
|
if ((this->has_cold_white() && this->cold_white_ > 0.0f) || (this->has_warm_white() && this->warm_white_ > 0.0f)) {
|
||||||
(this->warm_white_.has_value() && *this->warm_white_ > 0.0f)) {
|
|
||||||
if (!(color_mode & ColorCapability::COLD_WARM_WHITE)) {
|
if (!(color_mode & ColorCapability::COLD_WARM_WHITE)) {
|
||||||
ESP_LOGW(TAG, "'%s': color mode does not support setting cold/warm white value", name);
|
ESP_LOGW(TAG, "'%s': color mode does not support setting cold/warm white value", name);
|
||||||
this->cold_white_.reset();
|
this->set_flag_(FLAG_HAS_COLD_WHITE, false);
|
||||||
this->warm_white_.reset();
|
this->set_flag_(FLAG_HAS_WARM_WHITE, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#define VALIDATE_RANGE_(name_, upper_name, min, max) \
|
#define VALIDATE_RANGE_(name_, upper_name, min, max) \
|
||||||
if (name_##_.has_value()) { \
|
if (this->has_##name_()) { \
|
||||||
auto val = *name_##_; \
|
auto val = this->name_##_; \
|
||||||
if (val < (min) || val > (max)) { \
|
if (val < (min) || val > (max)) { \
|
||||||
ESP_LOGW(TAG, "'%s': %s value %.2f is out of range [%.1f - %.1f]", name, LOG_STR_LITERAL(upper_name), val, \
|
ESP_LOGW(TAG, "'%s': %s value %.2f is out of range [%.1f - %.1f]", name, LOG_STR_LITERAL(upper_name), val, \
|
||||||
(min), (max)); \
|
(min), (max)); \
|
||||||
name_##_ = clamp(val, (min), (max)); \
|
this->name_##_ = clamp(val, (min), (max)); \
|
||||||
} \
|
} \
|
||||||
}
|
}
|
||||||
#define VALIDATE_RANGE(name, upper_name) VALIDATE_RANGE_(name, upper_name, 0.0f, 1.0f)
|
#define VALIDATE_RANGE(name, upper_name) VALIDATE_RANGE_(name, upper_name, 0.0f, 1.0f)
|
||||||
@@ -227,110 +242,116 @@ LightColorValues LightCall::validate_() {
|
|||||||
VALIDATE_RANGE_(color_temperature, "Color temperature", traits.get_min_mireds(), traits.get_max_mireds())
|
VALIDATE_RANGE_(color_temperature, "Color temperature", traits.get_min_mireds(), traits.get_max_mireds())
|
||||||
|
|
||||||
// Flag whether an explicit turn off was requested, in which case we'll also stop the effect.
|
// Flag whether an explicit turn off was requested, in which case we'll also stop the effect.
|
||||||
bool explicit_turn_off_request = this->state_.has_value() && !*this->state_;
|
bool explicit_turn_off_request = this->has_state() && !this->state_;
|
||||||
|
|
||||||
// Turn off when brightness is set to zero, and reset brightness (so that it has nonzero brightness when turned on).
|
// Turn off when brightness is set to zero, and reset brightness (so that it has nonzero brightness when turned on).
|
||||||
if (this->brightness_.has_value() && *this->brightness_ == 0.0f) {
|
if (this->has_brightness() && this->brightness_ == 0.0f) {
|
||||||
this->state_ = optional<float>(false);
|
this->state_ = false;
|
||||||
this->brightness_ = optional<float>(1.0f);
|
this->set_flag_(FLAG_HAS_STATE, true);
|
||||||
|
this->brightness_ = 1.0f;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set color brightness to 100% if currently zero and a color is set.
|
// Set color brightness to 100% if currently zero and a color is set.
|
||||||
if (this->red_.has_value() || this->green_.has_value() || this->blue_.has_value()) {
|
if (this->has_red() || this->has_green() || this->has_blue()) {
|
||||||
if (!this->color_brightness_.has_value() && this->parent_->remote_values.get_color_brightness() == 0.0f)
|
if (!this->has_color_brightness() && this->parent_->remote_values.get_color_brightness() == 0.0f) {
|
||||||
this->color_brightness_ = optional<float>(1.0f);
|
this->color_brightness_ = 1.0f;
|
||||||
|
this->set_flag_(FLAG_HAS_COLOR_BRIGHTNESS, true);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create color values for the light with this call applied.
|
// Create color values for the light with this call applied.
|
||||||
auto v = this->parent_->remote_values;
|
auto v = this->parent_->remote_values;
|
||||||
if (this->color_mode_.has_value())
|
if (this->has_color_mode())
|
||||||
v.set_color_mode(*this->color_mode_);
|
v.set_color_mode(this->color_mode_);
|
||||||
if (this->state_.has_value())
|
if (this->has_state())
|
||||||
v.set_state(*this->state_);
|
v.set_state(this->state_);
|
||||||
if (this->brightness_.has_value())
|
if (this->has_brightness())
|
||||||
v.set_brightness(*this->brightness_);
|
v.set_brightness(this->brightness_);
|
||||||
if (this->color_brightness_.has_value())
|
if (this->has_color_brightness())
|
||||||
v.set_color_brightness(*this->color_brightness_);
|
v.set_color_brightness(this->color_brightness_);
|
||||||
if (this->red_.has_value())
|
if (this->has_red())
|
||||||
v.set_red(*this->red_);
|
v.set_red(this->red_);
|
||||||
if (this->green_.has_value())
|
if (this->has_green())
|
||||||
v.set_green(*this->green_);
|
v.set_green(this->green_);
|
||||||
if (this->blue_.has_value())
|
if (this->has_blue())
|
||||||
v.set_blue(*this->blue_);
|
v.set_blue(this->blue_);
|
||||||
if (this->white_.has_value())
|
if (this->has_white())
|
||||||
v.set_white(*this->white_);
|
v.set_white(this->white_);
|
||||||
if (this->color_temperature_.has_value())
|
if (this->has_color_temperature())
|
||||||
v.set_color_temperature(*this->color_temperature_);
|
v.set_color_temperature(this->color_temperature_);
|
||||||
if (this->cold_white_.has_value())
|
if (this->has_cold_white())
|
||||||
v.set_cold_white(*this->cold_white_);
|
v.set_cold_white(this->cold_white_);
|
||||||
if (this->warm_white_.has_value())
|
if (this->has_warm_white())
|
||||||
v.set_warm_white(*this->warm_white_);
|
v.set_warm_white(this->warm_white_);
|
||||||
|
|
||||||
v.normalize_color();
|
v.normalize_color();
|
||||||
|
|
||||||
// Flash length check
|
// Flash length check
|
||||||
if (this->has_flash_() && *this->flash_length_ == 0) {
|
if (this->has_flash_() && this->flash_length_ == 0) {
|
||||||
ESP_LOGW(TAG, "'%s': flash length must be greater than zero", name);
|
ESP_LOGW(TAG, "'%s': flash length must be greater than zero", name);
|
||||||
this->flash_length_.reset();
|
this->set_flag_(FLAG_HAS_FLASH, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// validate transition length/flash length/effect not used at the same time
|
// validate transition length/flash length/effect not used at the same time
|
||||||
bool supports_transition = color_mode & ColorCapability::BRIGHTNESS;
|
bool supports_transition = color_mode & ColorCapability::BRIGHTNESS;
|
||||||
|
|
||||||
// If effect is already active, remove effect start
|
// If effect is already active, remove effect start
|
||||||
if (this->has_effect_() && *this->effect_ == this->parent_->active_effect_index_) {
|
if (this->has_effect_() && this->effect_ == this->parent_->active_effect_index_) {
|
||||||
this->effect_.reset();
|
this->set_flag_(FLAG_HAS_EFFECT, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// validate effect index
|
// validate effect index
|
||||||
if (this->has_effect_() && *this->effect_ > this->parent_->effects_.size()) {
|
if (this->has_effect_() && this->effect_ > this->parent_->effects_.size()) {
|
||||||
ESP_LOGW(TAG, "'%s': invalid effect index %" PRIu32, name, *this->effect_);
|
ESP_LOGW(TAG, "'%s': invalid effect index %" PRIu32, name, this->effect_);
|
||||||
this->effect_.reset();
|
this->set_flag_(FLAG_HAS_EFFECT, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->has_effect_() && (this->has_transition_() || this->has_flash_())) {
|
if (this->has_effect_() && (this->has_transition_() || this->has_flash_())) {
|
||||||
ESP_LOGW(TAG, "'%s': effect cannot be used with transition/flash", name);
|
ESP_LOGW(TAG, "'%s': effect cannot be used with transition/flash", name);
|
||||||
this->transition_length_.reset();
|
this->set_flag_(FLAG_HAS_TRANSITION, false);
|
||||||
this->flash_length_.reset();
|
this->set_flag_(FLAG_HAS_FLASH, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->has_flash_() && this->has_transition_()) {
|
if (this->has_flash_() && this->has_transition_()) {
|
||||||
ESP_LOGW(TAG, "'%s': flash cannot be used with transition", name);
|
ESP_LOGW(TAG, "'%s': flash cannot be used with transition", name);
|
||||||
this->transition_length_.reset();
|
this->set_flag_(FLAG_HAS_TRANSITION, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!this->has_transition_() && !this->has_flash_() && (!this->has_effect_() || *this->effect_ == 0) &&
|
if (!this->has_transition_() && !this->has_flash_() && (!this->has_effect_() || this->effect_ == 0) &&
|
||||||
supports_transition) {
|
supports_transition) {
|
||||||
// nothing specified and light supports transitions, set default transition length
|
// nothing specified and light supports transitions, set default transition length
|
||||||
this->transition_length_ = this->parent_->default_transition_length_;
|
this->transition_length_ = this->parent_->default_transition_length_;
|
||||||
|
this->set_flag_(FLAG_HAS_TRANSITION, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->transition_length_.value_or(0) == 0) {
|
if (this->has_transition_() && this->transition_length_ == 0) {
|
||||||
// 0 transition is interpreted as no transition (instant change)
|
// 0 transition is interpreted as no transition (instant change)
|
||||||
this->transition_length_.reset();
|
this->set_flag_(FLAG_HAS_TRANSITION, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->has_transition_() && !supports_transition) {
|
if (this->has_transition_() && !supports_transition) {
|
||||||
ESP_LOGW(TAG, "'%s': transitions not supported", name);
|
ESP_LOGW(TAG, "'%s': transitions not supported", name);
|
||||||
this->transition_length_.reset();
|
this->set_flag_(FLAG_HAS_TRANSITION, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
// If not a flash and turning the light off, then disable the light
|
// If not a flash and turning the light off, then disable the light
|
||||||
// Do not use light color values directly, so that effects can set 0% brightness
|
// Do not use light color values directly, so that effects can set 0% brightness
|
||||||
// Reason: When user turns off the light in frontend, the effect should also stop
|
// Reason: When user turns off the light in frontend, the effect should also stop
|
||||||
if (!this->has_flash_() && !this->state_.value_or(v.is_on())) {
|
bool target_state = this->has_state() ? this->state_ : v.is_on();
|
||||||
|
if (!this->has_flash_() && !target_state) {
|
||||||
if (this->has_effect_()) {
|
if (this->has_effect_()) {
|
||||||
ESP_LOGW(TAG, "'%s': cannot start effect when turning off", name);
|
ESP_LOGW(TAG, "'%s': cannot start effect when turning off", name);
|
||||||
this->effect_.reset();
|
this->set_flag_(FLAG_HAS_EFFECT, false);
|
||||||
} else if (this->parent_->active_effect_index_ != 0 && explicit_turn_off_request) {
|
} else if (this->parent_->active_effect_index_ != 0 && explicit_turn_off_request) {
|
||||||
// Auto turn off effect
|
// Auto turn off effect
|
||||||
this->effect_ = 0;
|
this->effect_ = 0;
|
||||||
|
this->set_flag_(FLAG_HAS_EFFECT, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Disable saving for flashes
|
// Disable saving for flashes
|
||||||
if (this->has_flash_())
|
if (this->has_flash_())
|
||||||
this->save_ = false;
|
this->set_flag_(FLAG_SAVE, false);
|
||||||
|
|
||||||
return v;
|
return v;
|
||||||
}
|
}
|
||||||
@@ -343,24 +364,27 @@ void LightCall::transform_parameters_() {
|
|||||||
// - RGBWW lights with color_interlock=true, which also sets "brightness" and
|
// - RGBWW lights with color_interlock=true, which also sets "brightness" and
|
||||||
// "color_temperature" (without color_interlock, CW/WW are set directly)
|
// "color_temperature" (without color_interlock, CW/WW are set directly)
|
||||||
// - Legacy Home Assistant (pre-colormode), which sets "white" and "color_temperature"
|
// - Legacy Home Assistant (pre-colormode), which sets "white" and "color_temperature"
|
||||||
if (((this->white_.has_value() && *this->white_ > 0.0f) || this->color_temperature_.has_value()) && //
|
if (((this->has_white() && this->white_ > 0.0f) || this->has_color_temperature()) && //
|
||||||
(*this->color_mode_ & ColorCapability::COLD_WARM_WHITE) && //
|
(this->color_mode_ & ColorCapability::COLD_WARM_WHITE) && //
|
||||||
!(*this->color_mode_ & ColorCapability::WHITE) && //
|
!(this->color_mode_ & ColorCapability::WHITE) && //
|
||||||
!(*this->color_mode_ & ColorCapability::COLOR_TEMPERATURE) && //
|
!(this->color_mode_ & ColorCapability::COLOR_TEMPERATURE) && //
|
||||||
traits.get_min_mireds() > 0.0f && traits.get_max_mireds() > 0.0f) {
|
traits.get_min_mireds() > 0.0f && traits.get_max_mireds() > 0.0f) {
|
||||||
ESP_LOGD(TAG, "'%s': setting cold/warm white channels using white/color temperature values",
|
ESP_LOGD(TAG, "'%s': setting cold/warm white channels using white/color temperature values",
|
||||||
this->parent_->get_name().c_str());
|
this->parent_->get_name().c_str());
|
||||||
if (this->color_temperature_.has_value()) {
|
if (this->has_color_temperature()) {
|
||||||
const float color_temp = clamp(*this->color_temperature_, traits.get_min_mireds(), traits.get_max_mireds());
|
const float color_temp = clamp(this->color_temperature_, traits.get_min_mireds(), traits.get_max_mireds());
|
||||||
const float ww_fraction =
|
const float ww_fraction =
|
||||||
(color_temp - traits.get_min_mireds()) / (traits.get_max_mireds() - traits.get_min_mireds());
|
(color_temp - traits.get_min_mireds()) / (traits.get_max_mireds() - traits.get_min_mireds());
|
||||||
const float cw_fraction = 1.0f - ww_fraction;
|
const float cw_fraction = 1.0f - ww_fraction;
|
||||||
const float max_cw_ww = std::max(ww_fraction, cw_fraction);
|
const float max_cw_ww = std::max(ww_fraction, cw_fraction);
|
||||||
this->cold_white_ = gamma_uncorrect(cw_fraction / max_cw_ww, this->parent_->get_gamma_correct());
|
this->cold_white_ = gamma_uncorrect(cw_fraction / max_cw_ww, this->parent_->get_gamma_correct());
|
||||||
this->warm_white_ = gamma_uncorrect(ww_fraction / max_cw_ww, this->parent_->get_gamma_correct());
|
this->warm_white_ = gamma_uncorrect(ww_fraction / max_cw_ww, this->parent_->get_gamma_correct());
|
||||||
|
this->set_flag_(FLAG_HAS_COLD_WHITE, true);
|
||||||
|
this->set_flag_(FLAG_HAS_WARM_WHITE, true);
|
||||||
}
|
}
|
||||||
if (this->white_.has_value()) {
|
if (this->has_white()) {
|
||||||
this->brightness_ = *this->white_;
|
this->brightness_ = this->white_;
|
||||||
|
this->set_flag_(FLAG_HAS_BRIGHTNESS, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -378,7 +402,7 @@ ColorMode LightCall::compute_color_mode_() {
|
|||||||
|
|
||||||
// Don't change if the light is being turned off.
|
// Don't change if the light is being turned off.
|
||||||
ColorMode current_mode = this->parent_->remote_values.get_color_mode();
|
ColorMode current_mode = this->parent_->remote_values.get_color_mode();
|
||||||
if (this->state_.has_value() && !*this->state_)
|
if (this->has_state() && !this->state_)
|
||||||
return current_mode;
|
return current_mode;
|
||||||
|
|
||||||
// If no color mode is specified, we try to guess the color mode. This is needed for backward compatibility to
|
// If no color mode is specified, we try to guess the color mode. This is needed for backward compatibility to
|
||||||
@@ -411,12 +435,12 @@ ColorMode LightCall::compute_color_mode_() {
|
|||||||
return color_mode;
|
return color_mode;
|
||||||
}
|
}
|
||||||
std::set<ColorMode> LightCall::get_suitable_color_modes_() {
|
std::set<ColorMode> LightCall::get_suitable_color_modes_() {
|
||||||
bool has_white = this->white_.has_value() && *this->white_ > 0.0f;
|
bool has_white = this->has_white() && this->white_ > 0.0f;
|
||||||
bool has_ct = this->color_temperature_.has_value();
|
bool has_ct = this->has_color_temperature();
|
||||||
bool has_cwww = (this->cold_white_.has_value() && *this->cold_white_ > 0.0f) ||
|
bool has_cwww =
|
||||||
(this->warm_white_.has_value() && *this->warm_white_ > 0.0f);
|
(this->has_cold_white() && this->cold_white_ > 0.0f) || (this->has_warm_white() && this->warm_white_ > 0.0f);
|
||||||
bool has_rgb = (this->color_brightness_.has_value() && *this->color_brightness_ > 0.0f) ||
|
bool has_rgb = (this->has_color_brightness() && this->color_brightness_ > 0.0f) ||
|
||||||
(this->red_.has_value() || this->green_.has_value() || this->blue_.has_value());
|
(this->has_red() || this->has_green() || this->has_blue());
|
||||||
|
|
||||||
#define KEY(white, ct, cwww, rgb) ((white) << 0 | (ct) << 1 | (cwww) << 2 | (rgb) << 3)
|
#define KEY(white, ct, cwww, rgb) ((white) << 0 | (ct) << 1 | (cwww) << 2 | (rgb) << 3)
|
||||||
#define ENTRY(white, ct, cwww, rgb, ...) \
|
#define ENTRY(white, ct, cwww, rgb, ...) \
|
||||||
@@ -491,7 +515,7 @@ LightCall &LightCall::from_light_color_values(const LightColorValues &values) {
|
|||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
ColorMode LightCall::get_active_color_mode_() {
|
ColorMode LightCall::get_active_color_mode_() {
|
||||||
return this->color_mode_.value_or(this->parent_->remote_values.get_color_mode());
|
return this->has_color_mode() ? this->color_mode_ : this->parent_->remote_values.get_color_mode();
|
||||||
}
|
}
|
||||||
LightCall &LightCall::set_transition_length_if_supported(uint32_t transition_length) {
|
LightCall &LightCall::set_transition_length_if_supported(uint32_t transition_length) {
|
||||||
if (this->get_active_color_mode_() & ColorCapability::BRIGHTNESS)
|
if (this->get_active_color_mode_() & ColorCapability::BRIGHTNESS)
|
||||||
@@ -505,7 +529,7 @@ LightCall &LightCall::set_brightness_if_supported(float brightness) {
|
|||||||
}
|
}
|
||||||
LightCall &LightCall::set_color_mode_if_supported(ColorMode color_mode) {
|
LightCall &LightCall::set_color_mode_if_supported(ColorMode color_mode) {
|
||||||
if (this->parent_->get_traits().supports_color_mode(color_mode))
|
if (this->parent_->get_traits().supports_color_mode(color_mode))
|
||||||
this->color_mode_ = color_mode;
|
this->set_color_mode(color_mode);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
LightCall &LightCall::set_color_brightness_if_supported(float brightness) {
|
LightCall &LightCall::set_color_brightness_if_supported(float brightness) {
|
||||||
@@ -549,110 +573,19 @@ LightCall &LightCall::set_warm_white_if_supported(float warm_white) {
|
|||||||
this->set_warm_white(warm_white);
|
this->set_warm_white(warm_white);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
LightCall &LightCall::set_state(optional<bool> state) {
|
IMPLEMENT_LIGHT_CALL_SETTER(state, bool, FLAG_HAS_STATE)
|
||||||
this->state_ = state;
|
IMPLEMENT_LIGHT_CALL_SETTER(transition_length, uint32_t, FLAG_HAS_TRANSITION)
|
||||||
return *this;
|
IMPLEMENT_LIGHT_CALL_SETTER(flash_length, uint32_t, FLAG_HAS_FLASH)
|
||||||
}
|
IMPLEMENT_LIGHT_CALL_SETTER(brightness, float, FLAG_HAS_BRIGHTNESS)
|
||||||
LightCall &LightCall::set_state(bool state) {
|
IMPLEMENT_LIGHT_CALL_SETTER(color_mode, ColorMode, FLAG_HAS_COLOR_MODE)
|
||||||
this->state_ = state;
|
IMPLEMENT_LIGHT_CALL_SETTER(color_brightness, float, FLAG_HAS_COLOR_BRIGHTNESS)
|
||||||
return *this;
|
IMPLEMENT_LIGHT_CALL_SETTER(red, float, FLAG_HAS_RED)
|
||||||
}
|
IMPLEMENT_LIGHT_CALL_SETTER(green, float, FLAG_HAS_GREEN)
|
||||||
LightCall &LightCall::set_transition_length(optional<uint32_t> transition_length) {
|
IMPLEMENT_LIGHT_CALL_SETTER(blue, float, FLAG_HAS_BLUE)
|
||||||
this->transition_length_ = transition_length;
|
IMPLEMENT_LIGHT_CALL_SETTER(white, float, FLAG_HAS_WHITE)
|
||||||
return *this;
|
IMPLEMENT_LIGHT_CALL_SETTER(color_temperature, float, FLAG_HAS_COLOR_TEMPERATURE)
|
||||||
}
|
IMPLEMENT_LIGHT_CALL_SETTER(cold_white, float, FLAG_HAS_COLD_WHITE)
|
||||||
LightCall &LightCall::set_transition_length(uint32_t transition_length) {
|
IMPLEMENT_LIGHT_CALL_SETTER(warm_white, float, FLAG_HAS_WARM_WHITE)
|
||||||
this->transition_length_ = transition_length;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_flash_length(optional<uint32_t> flash_length) {
|
|
||||||
this->flash_length_ = flash_length;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_flash_length(uint32_t flash_length) {
|
|
||||||
this->flash_length_ = flash_length;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_brightness(optional<float> brightness) {
|
|
||||||
this->brightness_ = brightness;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_brightness(float brightness) {
|
|
||||||
this->brightness_ = brightness;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_color_mode(optional<ColorMode> color_mode) {
|
|
||||||
this->color_mode_ = color_mode;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_color_mode(ColorMode color_mode) {
|
|
||||||
this->color_mode_ = color_mode;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_color_brightness(optional<float> brightness) {
|
|
||||||
this->color_brightness_ = brightness;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_color_brightness(float brightness) {
|
|
||||||
this->color_brightness_ = brightness;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_red(optional<float> red) {
|
|
||||||
this->red_ = red;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_red(float red) {
|
|
||||||
this->red_ = red;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_green(optional<float> green) {
|
|
||||||
this->green_ = green;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_green(float green) {
|
|
||||||
this->green_ = green;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_blue(optional<float> blue) {
|
|
||||||
this->blue_ = blue;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_blue(float blue) {
|
|
||||||
this->blue_ = blue;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_white(optional<float> white) {
|
|
||||||
this->white_ = white;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_white(float white) {
|
|
||||||
this->white_ = white;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_color_temperature(optional<float> color_temperature) {
|
|
||||||
this->color_temperature_ = color_temperature;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_color_temperature(float color_temperature) {
|
|
||||||
this->color_temperature_ = color_temperature;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_cold_white(optional<float> cold_white) {
|
|
||||||
this->cold_white_ = cold_white;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_cold_white(float cold_white) {
|
|
||||||
this->cold_white_ = cold_white;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_warm_white(optional<float> warm_white) {
|
|
||||||
this->warm_white_ = warm_white;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_warm_white(float warm_white) {
|
|
||||||
this->warm_white_ = warm_white;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
LightCall &LightCall::set_effect(optional<std::string> effect) {
|
LightCall &LightCall::set_effect(optional<std::string> effect) {
|
||||||
if (effect.has_value())
|
if (effect.has_value())
|
||||||
this->set_effect(*effect);
|
this->set_effect(*effect);
|
||||||
@@ -660,18 +593,22 @@ LightCall &LightCall::set_effect(optional<std::string> effect) {
|
|||||||
}
|
}
|
||||||
LightCall &LightCall::set_effect(uint32_t effect_number) {
|
LightCall &LightCall::set_effect(uint32_t effect_number) {
|
||||||
this->effect_ = effect_number;
|
this->effect_ = effect_number;
|
||||||
|
this->set_flag_(FLAG_HAS_EFFECT, true);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
LightCall &LightCall::set_effect(optional<uint32_t> effect_number) {
|
LightCall &LightCall::set_effect(optional<uint32_t> effect_number) {
|
||||||
this->effect_ = effect_number;
|
if (effect_number.has_value()) {
|
||||||
|
this->effect_ = effect_number.value();
|
||||||
|
}
|
||||||
|
this->set_flag_(FLAG_HAS_EFFECT, effect_number.has_value());
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
LightCall &LightCall::set_publish(bool publish) {
|
LightCall &LightCall::set_publish(bool publish) {
|
||||||
this->publish_ = publish;
|
this->set_flag_(FLAG_PUBLISH, publish);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
LightCall &LightCall::set_save(bool save) {
|
LightCall &LightCall::set_save(bool save) {
|
||||||
this->save_ = save;
|
this->set_flag_(FLAG_SAVE, save);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
LightCall &LightCall::set_rgb(float red, float green, float blue) {
|
LightCall &LightCall::set_rgb(float red, float green, float blue) {
|
||||||
|
@@ -1,6 +1,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "esphome/core/optional.h"
|
|
||||||
#include "light_color_values.h"
|
#include "light_color_values.h"
|
||||||
#include <set>
|
#include <set>
|
||||||
|
|
||||||
@@ -10,6 +9,11 @@ namespace light {
|
|||||||
class LightState;
|
class LightState;
|
||||||
|
|
||||||
/** This class represents a requested change in a light state.
|
/** This class represents a requested change in a light state.
|
||||||
|
*
|
||||||
|
* Light state changes are tracked using a bitfield flags_ to minimize memory usage.
|
||||||
|
* Each possible light property has a flag indicating whether it has been set.
|
||||||
|
* This design keeps LightCall at ~56 bytes to minimize heap fragmentation on
|
||||||
|
* ESP8266 and other memory-constrained devices.
|
||||||
*/
|
*/
|
||||||
class LightCall {
|
class LightCall {
|
||||||
public:
|
public:
|
||||||
@@ -131,6 +135,19 @@ class LightCall {
|
|||||||
/// Set whether this light call should trigger a save state to recover them at startup..
|
/// Set whether this light call should trigger a save state to recover them at startup..
|
||||||
LightCall &set_save(bool save);
|
LightCall &set_save(bool save);
|
||||||
|
|
||||||
|
// Getter methods to check if values are set
|
||||||
|
bool has_state() const { return (flags_ & FLAG_HAS_STATE) != 0; }
|
||||||
|
bool has_brightness() const { return (flags_ & FLAG_HAS_BRIGHTNESS) != 0; }
|
||||||
|
bool has_color_brightness() const { return (flags_ & FLAG_HAS_COLOR_BRIGHTNESS) != 0; }
|
||||||
|
bool has_red() const { return (flags_ & FLAG_HAS_RED) != 0; }
|
||||||
|
bool has_green() const { return (flags_ & FLAG_HAS_GREEN) != 0; }
|
||||||
|
bool has_blue() const { return (flags_ & FLAG_HAS_BLUE) != 0; }
|
||||||
|
bool has_white() const { return (flags_ & FLAG_HAS_WHITE) != 0; }
|
||||||
|
bool has_color_temperature() const { return (flags_ & FLAG_HAS_COLOR_TEMPERATURE) != 0; }
|
||||||
|
bool has_cold_white() const { return (flags_ & FLAG_HAS_COLD_WHITE) != 0; }
|
||||||
|
bool has_warm_white() const { return (flags_ & FLAG_HAS_WARM_WHITE) != 0; }
|
||||||
|
bool has_color_mode() const { return (flags_ & FLAG_HAS_COLOR_MODE) != 0; }
|
||||||
|
|
||||||
/** Set the RGB color of the light by RGB values.
|
/** Set the RGB color of the light by RGB values.
|
||||||
*
|
*
|
||||||
* Please note that this only changes the color of the light, not the brightness.
|
* Please note that this only changes the color of the light, not the brightness.
|
||||||
@@ -170,27 +187,62 @@ class LightCall {
|
|||||||
/// Some color modes also can be set using non-native parameters, transform those calls.
|
/// Some color modes also can be set using non-native parameters, transform those calls.
|
||||||
void transform_parameters_();
|
void transform_parameters_();
|
||||||
|
|
||||||
bool has_transition_() { return this->transition_length_.has_value(); }
|
// Bitfield flags - each flag indicates whether a corresponding value has been set.
|
||||||
bool has_flash_() { return this->flash_length_.has_value(); }
|
enum FieldFlags : uint16_t {
|
||||||
bool has_effect_() { return this->effect_.has_value(); }
|
FLAG_HAS_STATE = 1 << 0,
|
||||||
|
FLAG_HAS_TRANSITION = 1 << 1,
|
||||||
|
FLAG_HAS_FLASH = 1 << 2,
|
||||||
|
FLAG_HAS_EFFECT = 1 << 3,
|
||||||
|
FLAG_HAS_BRIGHTNESS = 1 << 4,
|
||||||
|
FLAG_HAS_COLOR_BRIGHTNESS = 1 << 5,
|
||||||
|
FLAG_HAS_RED = 1 << 6,
|
||||||
|
FLAG_HAS_GREEN = 1 << 7,
|
||||||
|
FLAG_HAS_BLUE = 1 << 8,
|
||||||
|
FLAG_HAS_WHITE = 1 << 9,
|
||||||
|
FLAG_HAS_COLOR_TEMPERATURE = 1 << 10,
|
||||||
|
FLAG_HAS_COLD_WHITE = 1 << 11,
|
||||||
|
FLAG_HAS_WARM_WHITE = 1 << 12,
|
||||||
|
FLAG_HAS_COLOR_MODE = 1 << 13,
|
||||||
|
FLAG_PUBLISH = 1 << 14,
|
||||||
|
FLAG_SAVE = 1 << 15,
|
||||||
|
};
|
||||||
|
|
||||||
|
bool has_transition_() { return (this->flags_ & FLAG_HAS_TRANSITION) != 0; }
|
||||||
|
bool has_flash_() { return (this->flags_ & FLAG_HAS_FLASH) != 0; }
|
||||||
|
bool has_effect_() { return (this->flags_ & FLAG_HAS_EFFECT) != 0; }
|
||||||
|
bool get_publish_() { return (this->flags_ & FLAG_PUBLISH) != 0; }
|
||||||
|
bool get_save_() { return (this->flags_ & FLAG_SAVE) != 0; }
|
||||||
|
|
||||||
|
// Helper to set flag
|
||||||
|
void set_flag_(FieldFlags flag, bool value) {
|
||||||
|
if (value) {
|
||||||
|
this->flags_ |= flag;
|
||||||
|
} else {
|
||||||
|
this->flags_ &= ~flag;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
LightState *parent_;
|
LightState *parent_;
|
||||||
optional<bool> state_;
|
|
||||||
optional<uint32_t> transition_length_;
|
// Light state values - use flags_ to check if a value has been set.
|
||||||
optional<uint32_t> flash_length_;
|
// Group 4-byte aligned members first
|
||||||
optional<ColorMode> color_mode_;
|
uint32_t transition_length_;
|
||||||
optional<float> brightness_;
|
uint32_t flash_length_;
|
||||||
optional<float> color_brightness_;
|
uint32_t effect_;
|
||||||
optional<float> red_;
|
float brightness_;
|
||||||
optional<float> green_;
|
float color_brightness_;
|
||||||
optional<float> blue_;
|
float red_;
|
||||||
optional<float> white_;
|
float green_;
|
||||||
optional<float> color_temperature_;
|
float blue_;
|
||||||
optional<float> cold_white_;
|
float white_;
|
||||||
optional<float> warm_white_;
|
float color_temperature_;
|
||||||
optional<uint32_t> effect_;
|
float cold_white_;
|
||||||
bool publish_{true};
|
float warm_white_;
|
||||||
bool save_{true};
|
|
||||||
|
// Smaller members at the end for better packing
|
||||||
|
uint16_t flags_{FLAG_PUBLISH | FLAG_SAVE}; // Tracks which values are set
|
||||||
|
ColorMode color_mode_;
|
||||||
|
bool state_;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace light
|
} // namespace light
|
||||||
|
@@ -46,8 +46,7 @@ class LightColorValues {
|
|||||||
public:
|
public:
|
||||||
/// Construct the LightColorValues with all attributes enabled, but state set to off.
|
/// Construct the LightColorValues with all attributes enabled, but state set to off.
|
||||||
LightColorValues()
|
LightColorValues()
|
||||||
: color_mode_(ColorMode::UNKNOWN),
|
: state_(0.0f),
|
||||||
state_(0.0f),
|
|
||||||
brightness_(1.0f),
|
brightness_(1.0f),
|
||||||
color_brightness_(1.0f),
|
color_brightness_(1.0f),
|
||||||
red_(1.0f),
|
red_(1.0f),
|
||||||
@@ -56,7 +55,8 @@ class LightColorValues {
|
|||||||
white_(1.0f),
|
white_(1.0f),
|
||||||
color_temperature_{0.0f},
|
color_temperature_{0.0f},
|
||||||
cold_white_{1.0f},
|
cold_white_{1.0f},
|
||||||
warm_white_{1.0f} {}
|
warm_white_{1.0f},
|
||||||
|
color_mode_(ColorMode::UNKNOWN) {}
|
||||||
|
|
||||||
LightColorValues(ColorMode color_mode, float state, float brightness, float color_brightness, float red, float green,
|
LightColorValues(ColorMode color_mode, float state, float brightness, float color_brightness, float red, float green,
|
||||||
float blue, float white, float color_temperature, float cold_white, float warm_white) {
|
float blue, float white, float color_temperature, float cold_white, float warm_white) {
|
||||||
@@ -292,7 +292,6 @@ class LightColorValues {
|
|||||||
void set_warm_white(float warm_white) { this->warm_white_ = clamp(warm_white, 0.0f, 1.0f); }
|
void set_warm_white(float warm_white) { this->warm_white_ = clamp(warm_white, 0.0f, 1.0f); }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
ColorMode color_mode_;
|
|
||||||
float state_; ///< ON / OFF, float for transition
|
float state_; ///< ON / OFF, float for transition
|
||||||
float brightness_;
|
float brightness_;
|
||||||
float color_brightness_;
|
float color_brightness_;
|
||||||
@@ -303,6 +302,7 @@ class LightColorValues {
|
|||||||
float color_temperature_; ///< Color Temperature in Mired
|
float color_temperature_; ///< Color Temperature in Mired
|
||||||
float cold_white_;
|
float cold_white_;
|
||||||
float warm_white_;
|
float warm_white_;
|
||||||
|
ColorMode color_mode_;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace light
|
} // namespace light
|
||||||
|
@@ -31,9 +31,7 @@ enum LightRestoreMode : uint8_t {
|
|||||||
struct LightStateRTCState {
|
struct LightStateRTCState {
|
||||||
LightStateRTCState(ColorMode color_mode, bool state, float brightness, float color_brightness, float red, float green,
|
LightStateRTCState(ColorMode color_mode, bool state, float brightness, float color_brightness, float red, float green,
|
||||||
float blue, float white, float color_temp, float cold_white, float warm_white)
|
float blue, float white, float color_temp, float cold_white, float warm_white)
|
||||||
: color_mode(color_mode),
|
: brightness(brightness),
|
||||||
state(state),
|
|
||||||
brightness(brightness),
|
|
||||||
color_brightness(color_brightness),
|
color_brightness(color_brightness),
|
||||||
red(red),
|
red(red),
|
||||||
green(green),
|
green(green),
|
||||||
@@ -41,10 +39,12 @@ struct LightStateRTCState {
|
|||||||
white(white),
|
white(white),
|
||||||
color_temp(color_temp),
|
color_temp(color_temp),
|
||||||
cold_white(cold_white),
|
cold_white(cold_white),
|
||||||
warm_white(warm_white) {}
|
warm_white(warm_white),
|
||||||
|
effect(0),
|
||||||
|
color_mode(color_mode),
|
||||||
|
state(state) {}
|
||||||
LightStateRTCState() = default;
|
LightStateRTCState() = default;
|
||||||
ColorMode color_mode{ColorMode::UNKNOWN};
|
// Group 4-byte aligned members first
|
||||||
bool state{false};
|
|
||||||
float brightness{1.0f};
|
float brightness{1.0f};
|
||||||
float color_brightness{1.0f};
|
float color_brightness{1.0f};
|
||||||
float red{1.0f};
|
float red{1.0f};
|
||||||
@@ -55,6 +55,9 @@ struct LightStateRTCState {
|
|||||||
float cold_white{1.0f};
|
float cold_white{1.0f};
|
||||||
float warm_white{1.0f};
|
float warm_white{1.0f};
|
||||||
uint32_t effect{0};
|
uint32_t effect{0};
|
||||||
|
// Group smaller members at the end
|
||||||
|
ColorMode color_mode{ColorMode::UNKNOWN};
|
||||||
|
bool state{false};
|
||||||
};
|
};
|
||||||
|
|
||||||
/** This class represents the communication layer between the front-end MQTT layer and the
|
/** This class represents the communication layer between the front-end MQTT layer and the
|
||||||
@@ -216,6 +219,8 @@ class LightState : public EntityBase, public Component {
|
|||||||
std::unique_ptr<LightTransformer> transformer_{nullptr};
|
std::unique_ptr<LightTransformer> transformer_{nullptr};
|
||||||
/// List of effects for this light.
|
/// List of effects for this light.
|
||||||
std::vector<LightEffect *> effects_;
|
std::vector<LightEffect *> effects_;
|
||||||
|
/// Object used to store the persisted values of the light.
|
||||||
|
ESPPreferenceObject rtc_;
|
||||||
/// Value for storing the index of the currently active effect. 0 if no effect is active
|
/// Value for storing the index of the currently active effect. 0 if no effect is active
|
||||||
uint32_t active_effect_index_{};
|
uint32_t active_effect_index_{};
|
||||||
/// Default transition length for all transitions in ms.
|
/// Default transition length for all transitions in ms.
|
||||||
@@ -224,15 +229,11 @@ class LightState : public EntityBase, public Component {
|
|||||||
uint32_t flash_transition_length_{};
|
uint32_t flash_transition_length_{};
|
||||||
/// Gamma correction factor for the light.
|
/// Gamma correction factor for the light.
|
||||||
float gamma_correct_{};
|
float gamma_correct_{};
|
||||||
|
|
||||||
/// Whether the light value should be written in the next cycle.
|
/// Whether the light value should be written in the next cycle.
|
||||||
bool next_write_{true};
|
bool next_write_{true};
|
||||||
// for effects, true if a transformer (transition) is active.
|
// for effects, true if a transformer (transition) is active.
|
||||||
bool is_transformer_active_ = false;
|
bool is_transformer_active_ = false;
|
||||||
|
|
||||||
/// Object used to store the persisted values of the light.
|
|
||||||
ESPPreferenceObject rtc_;
|
|
||||||
|
|
||||||
/** Callback to call when new values for the frontend are available.
|
/** Callback to call when new values for the frontend are available.
|
||||||
*
|
*
|
||||||
* "Remote values" are light color values that are reported to the frontend and have a lower
|
* "Remote values" are light color values that are reported to the frontend and have a lower
|
||||||
|
@@ -59,9 +59,9 @@ class LightTransitionTransformer : public LightTransformer {
|
|||||||
// transition from 0 to 1 on x = [0, 1]
|
// transition from 0 to 1 on x = [0, 1]
|
||||||
static float smoothed_progress(float x) { return x * x * x * (x * (x * 6.0f - 15.0f) + 10.0f); }
|
static float smoothed_progress(float x) { return x * x * x * (x * (x * 6.0f - 15.0f) + 10.0f); }
|
||||||
|
|
||||||
bool changing_color_mode_{false};
|
|
||||||
LightColorValues end_values_{};
|
LightColorValues end_values_{};
|
||||||
LightColorValues intermediate_values_{};
|
LightColorValues intermediate_values_{};
|
||||||
|
bool changing_color_mode_{false};
|
||||||
};
|
};
|
||||||
|
|
||||||
class LightFlashTransformer : public LightTransformer {
|
class LightFlashTransformer : public LightTransformer {
|
||||||
@@ -117,8 +117,8 @@ class LightFlashTransformer : public LightTransformer {
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
LightState &state_;
|
LightState &state_;
|
||||||
uint32_t transition_length_;
|
|
||||||
std::unique_ptr<LightTransformer> transformer_{nullptr};
|
std::unique_ptr<LightTransformer> transformer_{nullptr};
|
||||||
|
uint32_t transition_length_;
|
||||||
bool begun_lightstate_restore_;
|
bool begun_lightstate_restore_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
80
tests/integration/fixtures/light_calls.yaml
Normal file
80
tests/integration/fixtures/light_calls.yaml
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
esphome:
|
||||||
|
name: light-calls-test
|
||||||
|
host:
|
||||||
|
api: # Port will be automatically injected
|
||||||
|
logger:
|
||||||
|
level: DEBUG
|
||||||
|
|
||||||
|
# Test outputs for RGBCW light
|
||||||
|
output:
|
||||||
|
- platform: template
|
||||||
|
id: test_red
|
||||||
|
type: float
|
||||||
|
write_action:
|
||||||
|
- logger.log:
|
||||||
|
format: "Red output: %.2f"
|
||||||
|
args: [state]
|
||||||
|
- platform: template
|
||||||
|
id: test_green
|
||||||
|
type: float
|
||||||
|
write_action:
|
||||||
|
- logger.log:
|
||||||
|
format: "Green output: %.2f"
|
||||||
|
args: [state]
|
||||||
|
- platform: template
|
||||||
|
id: test_blue
|
||||||
|
type: float
|
||||||
|
write_action:
|
||||||
|
- logger.log:
|
||||||
|
format: "Blue output: %.2f"
|
||||||
|
args: [state]
|
||||||
|
- platform: template
|
||||||
|
id: test_cold_white
|
||||||
|
type: float
|
||||||
|
write_action:
|
||||||
|
- logger.log:
|
||||||
|
format: "Cold white output: %.2f"
|
||||||
|
args: [state]
|
||||||
|
- platform: template
|
||||||
|
id: test_warm_white
|
||||||
|
type: float
|
||||||
|
write_action:
|
||||||
|
- logger.log:
|
||||||
|
format: "Warm white output: %.2f"
|
||||||
|
args: [state]
|
||||||
|
|
||||||
|
light:
|
||||||
|
- platform: rgbww
|
||||||
|
name: "Test RGBCW Light"
|
||||||
|
id: test_light
|
||||||
|
red: test_red
|
||||||
|
green: test_green
|
||||||
|
blue: test_blue
|
||||||
|
cold_white: test_cold_white
|
||||||
|
warm_white: test_warm_white
|
||||||
|
cold_white_color_temperature: 6536 K
|
||||||
|
warm_white_color_temperature: 2000 K
|
||||||
|
constant_brightness: true
|
||||||
|
effects:
|
||||||
|
- random:
|
||||||
|
name: "Random Effect"
|
||||||
|
transition_length: 100ms
|
||||||
|
update_interval: 200ms
|
||||||
|
- strobe:
|
||||||
|
name: "Strobe Effect"
|
||||||
|
- pulse:
|
||||||
|
name: "Pulse Effect"
|
||||||
|
transition_length: 100ms
|
||||||
|
|
||||||
|
# Additional lights to test memory with multiple instances
|
||||||
|
- platform: rgb
|
||||||
|
name: "Test RGB Light"
|
||||||
|
id: test_rgb_light
|
||||||
|
red: test_red
|
||||||
|
green: test_green
|
||||||
|
blue: test_blue
|
||||||
|
|
||||||
|
- platform: binary
|
||||||
|
name: "Test Binary Light"
|
||||||
|
id: test_binary_light
|
||||||
|
output: test_red
|
189
tests/integration/test_light_calls.py
Normal file
189
tests/integration/test_light_calls.py
Normal file
@@ -0,0 +1,189 @@
|
|||||||
|
"""Integration test for all light call combinations.
|
||||||
|
|
||||||
|
Tests that LightCall handles all possible light operations correctly
|
||||||
|
including RGB, color temperature, effects, transitions, and flash.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import asyncio
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from .types import APIClientConnectedFactory, RunCompiledFunction
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.asyncio
|
||||||
|
async def test_light_calls(
|
||||||
|
yaml_config: str,
|
||||||
|
run_compiled: RunCompiledFunction,
|
||||||
|
api_client_connected: APIClientConnectedFactory,
|
||||||
|
) -> None:
|
||||||
|
"""Test all possible LightCall operations and combinations."""
|
||||||
|
async with run_compiled(yaml_config), api_client_connected() as client:
|
||||||
|
# Track state changes with futures
|
||||||
|
state_futures: dict[int, asyncio.Future[Any]] = {}
|
||||||
|
states: dict[int, Any] = {}
|
||||||
|
|
||||||
|
def on_state(state: Any) -> None:
|
||||||
|
states[state.key] = state
|
||||||
|
if state.key in state_futures and not state_futures[state.key].done():
|
||||||
|
state_futures[state.key].set_result(state)
|
||||||
|
|
||||||
|
client.subscribe_states(on_state)
|
||||||
|
|
||||||
|
# Get the light entities
|
||||||
|
entities = await client.list_entities_services()
|
||||||
|
lights = [e for e in entities[0] if e.object_id.startswith("test_")]
|
||||||
|
assert len(lights) >= 2 # Should have RGBCW and RGB lights
|
||||||
|
|
||||||
|
rgbcw_light = next(light for light in lights if "RGBCW" in light.name)
|
||||||
|
rgb_light = next(light for light in lights if "RGB Light" in light.name)
|
||||||
|
|
||||||
|
async def wait_for_state_change(key: int, timeout: float = 1.0) -> Any:
|
||||||
|
"""Wait for a state change for the given entity key."""
|
||||||
|
loop = asyncio.get_event_loop()
|
||||||
|
state_futures[key] = loop.create_future()
|
||||||
|
try:
|
||||||
|
return await asyncio.wait_for(state_futures[key], timeout)
|
||||||
|
finally:
|
||||||
|
state_futures.pop(key, None)
|
||||||
|
|
||||||
|
# Test all individual parameters first
|
||||||
|
|
||||||
|
# Test 1: state only
|
||||||
|
client.light_command(key=rgbcw_light.key, state=True)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.state is True
|
||||||
|
|
||||||
|
# Test 2: brightness only
|
||||||
|
client.light_command(key=rgbcw_light.key, brightness=0.5)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.brightness == pytest.approx(0.5)
|
||||||
|
|
||||||
|
# Test 3: color_brightness only
|
||||||
|
client.light_command(key=rgbcw_light.key, color_brightness=0.8)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.color_brightness == pytest.approx(0.8)
|
||||||
|
|
||||||
|
# Test 4-7: RGB values must be set together via rgb parameter
|
||||||
|
client.light_command(key=rgbcw_light.key, rgb=(0.7, 0.3, 0.9))
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.red == pytest.approx(0.7, abs=0.1)
|
||||||
|
assert state.green == pytest.approx(0.3, abs=0.1)
|
||||||
|
assert state.blue == pytest.approx(0.9, abs=0.1)
|
||||||
|
|
||||||
|
# Test 7: white value
|
||||||
|
client.light_command(key=rgbcw_light.key, white=0.6)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
# White might need more tolerance or might not be directly settable
|
||||||
|
if hasattr(state, "white"):
|
||||||
|
assert state.white == pytest.approx(0.6, abs=0.1)
|
||||||
|
|
||||||
|
# Test 8: color_temperature only
|
||||||
|
client.light_command(key=rgbcw_light.key, color_temperature=300)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.color_temperature == pytest.approx(300)
|
||||||
|
|
||||||
|
# Test 9: cold_white only
|
||||||
|
client.light_command(key=rgbcw_light.key, cold_white=0.8)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.cold_white == pytest.approx(0.8)
|
||||||
|
|
||||||
|
# Test 10: warm_white only
|
||||||
|
client.light_command(key=rgbcw_light.key, warm_white=0.2)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.warm_white == pytest.approx(0.2)
|
||||||
|
|
||||||
|
# Test 11: transition_length with state change
|
||||||
|
client.light_command(key=rgbcw_light.key, state=False, transition_length=0.1)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.state is False
|
||||||
|
|
||||||
|
# Test 12: flash_length
|
||||||
|
client.light_command(key=rgbcw_light.key, state=True, flash_length=0.2)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
# Flash starts
|
||||||
|
assert state.state is True
|
||||||
|
# Wait for flash to end
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
|
||||||
|
# Test 13: effect only
|
||||||
|
# First ensure light is on
|
||||||
|
client.light_command(key=rgbcw_light.key, state=True)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
# Now set effect
|
||||||
|
client.light_command(key=rgbcw_light.key, effect="Random Effect")
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.effect == "Random Effect"
|
||||||
|
|
||||||
|
# Test 14: stop effect
|
||||||
|
client.light_command(key=rgbcw_light.key, effect="None")
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.effect == "None"
|
||||||
|
|
||||||
|
# Test 15: color_mode parameter
|
||||||
|
client.light_command(
|
||||||
|
key=rgbcw_light.key, state=True, color_mode=5
|
||||||
|
) # COLD_WARM_WHITE
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.state is True
|
||||||
|
|
||||||
|
# Now test common combinations
|
||||||
|
|
||||||
|
# Test 16: RGB combination (set_rgb) - RGB values get normalized
|
||||||
|
client.light_command(key=rgbcw_light.key, rgb=(1.0, 0.0, 0.5))
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
# RGB values get normalized - in this case red is already 1.0
|
||||||
|
assert state.red == pytest.approx(1.0, abs=0.1)
|
||||||
|
assert state.green == pytest.approx(0.0, abs=0.1)
|
||||||
|
assert state.blue == pytest.approx(0.5, abs=0.1)
|
||||||
|
|
||||||
|
# Test 17: Multiple RGB changes to test transitions
|
||||||
|
client.light_command(key=rgbcw_light.key, rgb=(0.2, 0.8, 0.4))
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
# RGB values get normalized so green (highest) becomes 1.0
|
||||||
|
# Expected: (0.2/0.8, 0.8/0.8, 0.4/0.8) = (0.25, 1.0, 0.5)
|
||||||
|
assert state.red == pytest.approx(0.25, abs=0.01)
|
||||||
|
assert state.green == pytest.approx(1.0, abs=0.01)
|
||||||
|
assert state.blue == pytest.approx(0.5, abs=0.01)
|
||||||
|
|
||||||
|
# Test 18: State + brightness + transition
|
||||||
|
client.light_command(
|
||||||
|
key=rgbcw_light.key, state=True, brightness=0.7, transition_length=0.1
|
||||||
|
)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.state is True
|
||||||
|
assert state.brightness == pytest.approx(0.7)
|
||||||
|
|
||||||
|
# Test 19: RGB + brightness + color_brightness
|
||||||
|
client.light_command(
|
||||||
|
key=rgb_light.key,
|
||||||
|
state=True,
|
||||||
|
brightness=0.8,
|
||||||
|
color_brightness=0.9,
|
||||||
|
rgb=(0.2, 0.4, 0.6),
|
||||||
|
)
|
||||||
|
state = await wait_for_state_change(rgb_light.key)
|
||||||
|
assert state.state is True
|
||||||
|
assert state.brightness == pytest.approx(0.8)
|
||||||
|
|
||||||
|
# Test 20: Color temp + cold/warm white
|
||||||
|
client.light_command(
|
||||||
|
key=rgbcw_light.key, color_temperature=250, cold_white=0.7, warm_white=0.3
|
||||||
|
)
|
||||||
|
state = await wait_for_state_change(rgbcw_light.key)
|
||||||
|
assert state.color_temperature == pytest.approx(250)
|
||||||
|
|
||||||
|
# Test 21: Turn RGB light off
|
||||||
|
client.light_command(key=rgb_light.key, state=False)
|
||||||
|
state = await wait_for_state_change(rgb_light.key)
|
||||||
|
assert state.state is False
|
||||||
|
|
||||||
|
# Final cleanup - turn all lights off
|
||||||
|
for light in lights:
|
||||||
|
client.light_command(
|
||||||
|
key=light.key,
|
||||||
|
state=False,
|
||||||
|
)
|
||||||
|
state = await wait_for_state_change(light.key)
|
||||||
|
assert state.state is False
|
Reference in New Issue
Block a user