ESPHome 2026.5.0
Loading...
Searching...
No Matches
light_call.cpp
Go to the documentation of this file.
1#include <cinttypes>
2
3#include "light_call.h"
4#include "light_state.h"
5#include "esphome/core/log.h"
8
9namespace esphome::light {
10
11static const char *const TAG = "light";
12
13// Cold-path logger; caller handles the clamp so the in-range hot path avoids
14// the spill/reload around the call.
15static void log_value_out_of_range(const char *name, float value, const LogString *param_name, float min, float max) {
16 ESP_LOGW(TAG, "'%s': %s value %.2f is out of range [%.1f - %.1f]", name, LOG_STR_ARG(param_name), value, min, max);
17}
18
19#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_WARN
20static void log_feature_not_supported(const char *name, const LogString *feature) {
21 ESP_LOGW(TAG, "'%s': %s not supported", name, LOG_STR_ARG(feature));
22}
23
24static void log_color_mode_not_supported(const char *name, const LogString *feature) {
25 ESP_LOGW(TAG, "'%s': color mode does not support setting %s", name, LOG_STR_ARG(feature));
26}
27
28static void log_invalid_parameter(const char *name, const LogString *message) {
29 ESP_LOGW(TAG, "'%s': %s", name, LOG_STR_ARG(message));
30}
31#else
32#define log_feature_not_supported(name, feature)
33#define log_color_mode_not_supported(name, feature)
34#define log_invalid_parameter(name, message)
35#endif
36
37// Macro to reduce repetitive setter code
38#define IMPLEMENT_LIGHT_CALL_SETTER(name, type, flag) \
39 LightCall &LightCall::set_##name(optional<type>(name)) { \
40 if ((name).has_value()) { \
41 this->name##_ = (name).value(); \
42 } \
43 this->set_flag_(flag, (name).has_value()); \
44 return *this; \
45 } \
46 LightCall &LightCall::set_##name(type name) { \
47 this->name##_ = name; \
48 this->set_flag_(flag); \
49 return *this; \
50 }
51
52// Color mode human-readable strings indexed by ColorModeBitPolicy::to_bit() (0-9)
53// Index 0 is Unknown (for ColorMode::UNKNOWN), also used as fallback for out-of-range
54PROGMEM_STRING_TABLE(ColorModeHumanStrings, "Unknown", "On/Off", "Brightness", "White", "Color temperature",
55 "Cold/warm white", "RGB", "RGBW", "RGB + color temperature", "RGB + cold/warm white");
56
57// Indices 0-7 match FieldFlags bits 0-7; index 8 is color_temperature.
58// PROGMEM_STRING_TABLE is constexpr-init (no RAM guard variable).
59PROGMEM_STRING_TABLE(ValidateFieldNames, "Brightness", "Color brightness", "Red", "Green", "Blue", "White",
60 "Cold white", "Warm white", "Color temperature");
61static constexpr uint8_t VALIDATE_CT_INDEX = 8;
62
63static const LogString *color_mode_to_human(ColorMode color_mode) {
64 return ColorModeHumanStrings::get_log_str(ColorModeBitPolicy::to_bit(color_mode), 0);
65}
66
67// Helper to log percentage values
68#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
69static void log_percent(const LogString *param, float value) {
70 ESP_LOGV(TAG, " %s: %.0f%%", LOG_STR_ARG(param), value * 100.0f);
71}
72#else
73#define log_percent(param, value)
74#endif
75
77 const char *name = this->parent_->get_name().c_str();
78 LightColorValues v = this->validate_();
79 const bool publish = this->get_publish_();
80
81 if (publish) {
82 ESP_LOGV(TAG, "'%s' Setting:", name);
83
84 // Only print color mode when it's being changed
85 ColorMode current_color_mode = this->parent_->remote_values.get_color_mode();
86 ColorMode target_color_mode = this->has_color_mode() ? this->color_mode_ : current_color_mode;
87 if (target_color_mode != current_color_mode) {
88 ESP_LOGV(TAG, " Color mode: %s", LOG_STR_ARG(color_mode_to_human(v.get_color_mode())));
89 }
90
91 // Only print state when it's being changed
92 bool current_state = this->parent_->remote_values.is_on();
93 bool target_state = this->has_state() ? this->state_ : current_state;
94 if (target_state != current_state) {
95 ESP_LOGV(TAG, " State: %s", ONOFF(v.is_on()));
96 }
97
98 if (this->has_brightness()) {
99 log_percent(LOG_STR("Brightness"), v.get_brightness());
100 }
101
102 if (this->has_color_brightness()) {
103 log_percent(LOG_STR("Color brightness"), v.get_color_brightness());
104 }
105 if (this->has_red() || this->has_green() || this->has_blue()) {
106 ESP_LOGV(TAG, " Red: %.0f%%, Green: %.0f%%, Blue: %.0f%%", v.get_red() * 100.0f, v.get_green() * 100.0f,
107 v.get_blue() * 100.0f);
108 }
109
110 if (this->has_white()) {
111 log_percent(LOG_STR("White"), v.get_white());
112 }
113 if (this->has_color_temperature()) {
114 ESP_LOGV(TAG, " Color temperature: %.1f mireds", v.get_color_temperature());
115 }
116
117 if (this->has_cold_white() || this->has_warm_white()) {
118 ESP_LOGV(TAG, " Cold white: %.0f%%, warm white: %.0f%%", v.get_cold_white() * 100.0f,
119 v.get_warm_white() * 100.0f);
120 }
121 }
122
123 if (this->has_flash_()) {
124 // FLASH
125 if (publish) {
126 ESP_LOGV(TAG, " Flash length: %.1fs", this->flash_length_ / 1e3f);
127 }
128
129 this->parent_->start_flash_(v, this->flash_length_, publish);
130 } else if (this->has_transition_()) {
131 // TRANSITION
132 if (publish) {
133 ESP_LOGV(TAG, " Transition length: %.1fs", this->transition_length_ / 1e3f);
134 }
135
136 // Special case: Transition and effect can be set when turning off
137 if (this->has_effect_()) {
138 if (publish) {
139 ESP_LOGV(TAG, " Effect: 'None'");
140 }
141 this->parent_->stop_effect_();
142 }
143
144 this->parent_->start_transition_(v, this->transition_length_, publish);
145
146 } else if (this->has_effect_()) {
147 // EFFECT
148 StringRef effect_s;
149 if (this->effect_ == 0u) {
150 effect_s = StringRef::from_lit("None");
151 } else {
152 effect_s = this->parent_->effects_[this->effect_ - 1]->get_name();
153 }
154
155 if (publish) {
156 ESP_LOGV(TAG, " Effect: '%.*s'", (int) effect_s.size(), effect_s.c_str());
157 }
158
159 this->parent_->start_effect_(this->effect_);
160
161 // Also set light color values when starting an effect
162 // For example to turn off the light
163 this->parent_->set_immediately_(v, true);
164 } else {
165 // INSTANT CHANGE
166 this->parent_->set_immediately_(v, publish);
167 }
168
170 for (auto *listener : *this->parent_->target_state_reached_listeners_) {
171 listener->on_light_target_state_reached();
172 }
173 }
174 if (publish) {
175 this->parent_->publish_state();
176 }
177 if (this->get_save_()) {
179 }
180}
181
182void LightCall::log_and_clear_unsupported_(FieldFlags flag, const LogString *feature, bool use_color_mode_log) {
183 auto *name = this->parent_->get_name().c_str();
184 if (use_color_mode_log) {
185 log_color_mode_not_supported(name, feature);
186 } else {
187 log_feature_not_supported(name, feature);
188 }
189 this->clear_flag_(flag);
190}
191
193 auto *name = this->parent_->get_name().c_str();
194 auto traits = this->parent_->get_traits();
195
196 // Color mode check
197 if (this->has_color_mode() && !traits.supports_color_mode(this->color_mode_)) {
198 ESP_LOGW(TAG, "'%s' does not support color mode %s", name, LOG_STR_ARG(color_mode_to_human(this->color_mode_)));
200 }
201
202 // Ensure there is always a color mode set
203 if (!this->has_color_mode()) {
204 this->color_mode_ = this->compute_color_mode_(traits);
206 }
207 auto color_mode = this->color_mode_;
208
209 // Transform calls that use non-native parameters for the current mode.
210 this->transform_parameters_(traits);
211
212 // Business logic adjustments before validation
213 // Flag whether an explicit turn off was requested, in which case we'll also stop the effect.
214 bool explicit_turn_off_request = this->has_state() && !this->state_;
215
216 // Turn off when brightness is set to zero, and reset brightness (so that it has nonzero brightness when turned on).
217 if (this->has_brightness() && this->brightness_ == 0.0f) {
218 this->state_ = false;
220 if (color_mode & ColorCapability::BRIGHTNESS) {
221 // Reset brightness so the light has nonzero brightness when turned back on.
222 this->brightness_ = 1.0f;
223 } else {
224 // Light doesn't support brightness; clear the flag to avoid a spurious
225 // "brightness not supported" warning during capability validation.
227 }
228 }
229
230 // Set color brightness to 100% if currently zero and a color is set.
231 if ((this->has_red() || this->has_green() || this->has_blue()) && !this->has_color_brightness() &&
232 this->parent_->remote_values.get_color_brightness() == 0.0f) {
233 this->color_brightness_ = 1.0f;
235 }
236
237 // Capability validation
238 if (this->has_brightness() && this->brightness_ > 0.0f && !(color_mode & ColorCapability::BRIGHTNESS))
239 this->log_and_clear_unsupported_(FLAG_HAS_BRIGHTNESS, LOG_STR("brightness"), false);
240
241 // Transition length possible check
242 if (this->has_transition_() && this->transition_length_ != 0 && !(color_mode & ColorCapability::BRIGHTNESS))
243 this->log_and_clear_unsupported_(FLAG_HAS_TRANSITION, LOG_STR("transitions"), false);
244
245 if (this->has_color_brightness() && this->color_brightness_ > 0.0f && !(color_mode & ColorCapability::RGB))
246 this->log_and_clear_unsupported_(FLAG_HAS_COLOR_BRIGHTNESS, LOG_STR("RGB brightness"), true);
247
248 // RGB exists check
249 if (((this->has_red() && this->red_ > 0.0f) || (this->has_green() && this->green_ > 0.0f) ||
250 (this->has_blue() && this->blue_ > 0.0f)) &&
251 !(color_mode & ColorCapability::RGB)) {
252 log_color_mode_not_supported(name, LOG_STR("RGB color"));
256 }
257
258 // White value exists check
259 if (this->has_white() && this->white_ > 0.0f &&
260 !(color_mode & ColorCapability::WHITE || color_mode & ColorCapability::COLD_WARM_WHITE))
261 this->log_and_clear_unsupported_(FLAG_HAS_WHITE, LOG_STR("white value"), true);
262
263 // Color temperature exists check
264 if (this->has_color_temperature() &&
266 this->log_and_clear_unsupported_(FLAG_HAS_COLOR_TEMPERATURE, LOG_STR("color temperature"), true);
267
268 // Cold/warm white value exists check
269 if (((this->has_cold_white() && this->cold_white_ > 0.0f) || (this->has_warm_white() && this->warm_white_ > 0.0f)) &&
270 !(color_mode & ColorCapability::COLD_WARM_WHITE)) {
271 log_color_mode_not_supported(name, LOG_STR("cold/warm white value"));
274 }
275
276 // Create color values and validate+apply ranges in one step to eliminate duplicate checks
277 auto v = this->parent_->remote_values;
278 if (this->has_color_mode())
280 if (this->has_state())
281 v.set_state(this->state_);
282
283 // FieldFlags bits 0-7 must match unit_fields_ array indices.
284 static_assert(FLAG_HAS_BRIGHTNESS == 1u << 0 && FLAG_HAS_COLOR_BRIGHTNESS == 1u << 1 && FLAG_HAS_RED == 1u << 2 &&
285 FLAG_HAS_GREEN == 1u << 3 && FLAG_HAS_BLUE == 1u << 4 && FLAG_HAS_WHITE == 1u << 5 &&
286 FLAG_HAS_COLD_WHITE == 1u << 6 && FLAG_HAS_WARM_WHITE == 1u << 7,
287 "FieldFlags bits 0-7 must match unit_fields_ indices");
288
289 // Iterate set bits only (ctz + clear-lowest) — HA can drive perform()
290 // at high frequency so the hot path is O(popcount).
291 unsigned active = this->flags_ & CLAMP_FLAGS_MASK;
292 while (active != 0) {
293 unsigned bit = __builtin_ctz(active);
294 active &= active - 1; // clear lowest set bit
295 float &value = this->unit_fields_[bit];
296 if (float_out_of_unit_range(value)) {
297 log_value_out_of_range(name, value, ValidateFieldNames::get_log_str(bit, 0), 0.0f, 1.0f);
298 value = clamp_unit_float(value);
299 }
300 v.unit_fields_[bit] = value;
301 }
302
303 // color_temperature: runtime range from traits.
304 if (this->has_color_temperature()) {
305 const float ct_min = traits.get_min_mireds();
306 const float ct_max = traits.get_max_mireds();
308 log_value_out_of_range(name, this->color_temperature_, ValidateFieldNames::get_log_str(VALIDATE_CT_INDEX, 0),
309 ct_min, ct_max);
310 this->color_temperature_ = clamp(this->color_temperature_, ct_min, ct_max);
311 }
312 v.color_temperature_ = this->color_temperature_;
313 }
314
315 v.normalize_color();
316
317 // Flash length check
318 if (this->has_flash_() && this->flash_length_ == 0) {
319 log_invalid_parameter(name, LOG_STR("flash length must be >0"));
321 }
322
323 // validate transition length/flash length/effect not used at the same time
324 bool supports_transition = color_mode & ColorCapability::BRIGHTNESS;
325
326 // If effect is already active, remove effect start
327 if (this->has_effect_() && this->effect_ == this->parent_->active_effect_index_) {
329 }
330
331 // validate effect index
332 if (this->has_effect_() && this->effect_ > this->parent_->effects_.size()) {
333 ESP_LOGW(TAG, "'%s': invalid effect index %" PRIu32, name, this->effect_);
335 }
336
337 if (this->has_effect_() && (this->has_transition_() || this->has_flash_())) {
338 log_invalid_parameter(name, LOG_STR("effect cannot be used with transition/flash"));
341 }
342
343 if (this->has_flash_() && this->has_transition_()) {
344 log_invalid_parameter(name, LOG_STR("flash cannot be used with transition"));
346 }
347
348 if (!this->has_transition_() && !this->has_flash_() && (!this->has_effect_() || this->effect_ == 0) &&
349 supports_transition) {
350 // nothing specified and light supports transitions, set default transition length
353 }
354
355 if (this->has_transition_() && this->transition_length_ == 0) {
356 // 0 transition is interpreted as no transition (instant change)
358 }
359
360 if (this->has_transition_() && !supports_transition)
361 this->log_and_clear_unsupported_(FLAG_HAS_TRANSITION, LOG_STR("transitions"), false);
362
363 // If not a flash and turning the light off, then disable the light
364 // Do not use light color values directly, so that effects can set 0% brightness
365 // Reason: When user turns off the light in frontend, the effect should also stop
366 bool target_state = this->has_state() ? this->state_ : v.is_on();
367 if (!this->has_flash_() && !target_state) {
368 if (this->has_effect_()) {
369 log_invalid_parameter(name, LOG_STR("cannot start effect when turning off"));
371 } else if (this->parent_->active_effect_index_ != 0 && explicit_turn_off_request) {
372 // Auto turn off effect
373 this->effect_ = 0;
375 }
376 }
377
378 // Disable saving for flashes
379 if (this->has_flash_())
380 this->clear_flag_(FLAG_SAVE);
381
382 return v;
383}
385 // Allow CWWW modes to be set with a white value and/or color temperature.
386 // This is used in three cases in HA:
387 // - CW/WW lights, which set the "brightness" and "color_temperature"
388 // - RGBWW lights with color_interlock=true, which also sets "brightness" and
389 // "color_temperature" (without color_interlock, CW/WW are set directly)
390 // - Legacy Home Assistant (pre-colormode), which sets "white" and "color_temperature"
391
392 // Cache min/max mireds to avoid repeated calls
393 const float min_mireds = traits.get_min_mireds();
394 const float max_mireds = traits.get_max_mireds();
395
396 if (((this->has_white() && this->white_ > 0.0f) || this->has_color_temperature()) && //
398 !(this->color_mode_ & ColorCapability::WHITE) && //
400 min_mireds > 0.0f && max_mireds > 0.0f) {
401 ESP_LOGV(TAG, "'%s': setting cold/warm white channels using white/color temperature values",
402 this->parent_->get_name().c_str());
403 // Only compute cold_white/warm_white from color_temperature if they're not already explicitly set.
404 // This is important for state restoration, where both color_temperature and cold_white/warm_white
405 // are restored from flash - we want to preserve the saved cold_white/warm_white values.
406 if (this->has_color_temperature() && !this->has_cold_white() && !this->has_warm_white()) {
407 const float color_temp = clamp(this->color_temperature_, min_mireds, max_mireds);
408 const float range = max_mireds - min_mireds;
409 const float ww_fraction = (color_temp - min_mireds) / range;
410 const float cw_fraction = 1.0f - ww_fraction;
411 const float max_cw_ww = std::max(ww_fraction, cw_fraction);
412 this->cold_white_ = this->parent_->gamma_uncorrect_lut(cw_fraction / max_cw_ww);
413 this->warm_white_ = this->parent_->gamma_uncorrect_lut(ww_fraction / max_cw_ww);
416 }
417 if (this->has_white()) {
418 this->brightness_ = this->white_;
420 }
421 }
422}
424 auto supported_modes = traits.get_supported_color_modes();
425 int supported_count = supported_modes.size();
426
427 // Some lights don't support any color modes (e.g. monochromatic light), leave it at unknown.
428 if (supported_count == 0)
429 return ColorMode::UNKNOWN;
430
431 // In the common case of lights supporting only a single mode, use that one.
432 if (supported_count == 1)
433 return *supported_modes.begin();
434
435 // Don't change if the light is being turned off.
436 ColorMode current_mode = this->parent_->remote_values.get_color_mode();
437 if (this->has_state() && !this->state_)
438 return current_mode;
439
440 // If no color mode is specified, we try to guess the color mode. This is needed for backward compatibility to
441 // pre-colormode clients and automations, but also for the MQTT API, where HA doesn't let us know which color mode
442 // was used for some reason.
443 // Compute intersection of suitable and supported modes using bitwise AND
444 color_mode_bitmask_t intersection = this->get_suitable_color_modes_mask_() & supported_modes.get_mask();
445
446 // Don't change if the current mode is in the intersection (suitable AND supported)
447 if (ColorModeMask::mask_contains(intersection, current_mode)) {
448 ESP_LOGV(TAG, "'%s': color mode not specified; retaining %s", this->parent_->get_name().c_str(),
449 LOG_STR_ARG(color_mode_to_human(current_mode)));
450 return current_mode;
451 }
452
453 // Use the preferred suitable mode.
454 if (intersection != 0) {
456 ESP_LOGV(TAG, "'%s': color mode not specified; using %s", this->parent_->get_name().c_str(),
457 LOG_STR_ARG(color_mode_to_human(mode)));
458 return mode;
459 }
460
461 // There's no supported mode for this call, so warn, use the current more or a mode at random and let validation strip
462 // out whatever we don't support.
463 auto color_mode = current_mode != ColorMode::UNKNOWN ? current_mode : *supported_modes.begin();
464 ESP_LOGW(TAG, "'%s': no suitable color mode supported; defaulting to %s", this->parent_->get_name().c_str(),
465 LOG_STR_ARG(color_mode_to_human(color_mode)));
466 return color_mode;
467}
468// PROGMEM lookup table for get_suitable_color_modes_mask_().
469// Maps 4-bit key (white | ct<<1 | cwww<<2 | rgb<<3) to color mode bitmask.
470// Packed into uint8_t by right-shifting by PACK_SHIFT since the lower bits
471// (UNKNOWN, ON_OFF, BRIGHTNESS) are never present in suitable mode masks.
472static constexpr unsigned PACK_SHIFT = ColorModeBitPolicy::to_bit(ColorMode::WHITE);
473// clang-format off
474static constexpr uint8_t SUITABLE_COLOR_MODES[] PROGMEM = {
475 // [0] none - all modes with brightness
478 ColorMode::COLD_WARM_WHITE}).get_mask() >> PACK_SHIFT),
479 // [1] white only
482 // [2] ct only
485 // [3] white + ct
487 ColorMode::RGB_COLD_WARM_WHITE}).get_mask() >> PACK_SHIFT),
488 // [4] cwww only
489 static_cast<uint8_t>(ColorModeMask({ColorMode::COLD_WARM_WHITE,
490 ColorMode::RGB_COLD_WARM_WHITE}).get_mask() >> PACK_SHIFT),
491 0, // [5] white + cwww (conflicting)
492 0, // [6] ct + cwww (conflicting)
493 0, // [7] white + ct + cwww (conflicting)
494 // [8] rgb only
496 ColorMode::RGB_COLD_WARM_WHITE}).get_mask() >> PACK_SHIFT),
497 // [9] rgb + white
499 ColorMode::RGB_COLD_WARM_WHITE}).get_mask() >> PACK_SHIFT),
500 // [10] rgb + ct
502 ColorMode::RGB_COLD_WARM_WHITE}).get_mask() >> PACK_SHIFT),
503 // [11] rgb + white + ct
505 ColorMode::RGB_COLD_WARM_WHITE}).get_mask() >> PACK_SHIFT),
506 // [12] rgb + cwww
507 static_cast<uint8_t>(ColorModeMask({ColorMode::RGB_COLD_WARM_WHITE}).get_mask() >> PACK_SHIFT),
508 0, // [13] rgb + white + cwww (conflicting)
509 0, // [14] rgb + ct + cwww (conflicting)
510 0, // [15] all (conflicting)
511};
512// clang-format on
513
515 bool has_white = this->has_white() && this->white_ > 0.0f;
516 bool has_ct = this->has_color_temperature();
517 bool has_cwww =
518 (this->has_cold_white() && this->cold_white_ > 0.0f) || (this->has_warm_white() && this->warm_white_ > 0.0f);
519 bool has_rgb = (this->has_color_brightness() && this->color_brightness_ > 0.0f) ||
520 (this->has_red() || this->has_green() || this->has_blue());
521
522 // Build key from flags: [rgb][cwww][ct][white]
523 uint8_t key = has_white | (has_ct << 1) | (has_cwww << 2) | (has_rgb << 3);
524 return static_cast<color_mode_bitmask_t>(progmem_read_byte(&SUITABLE_COLOR_MODES[key])) << PACK_SHIFT;
525}
526
527LightCall &LightCall::set_effect(const char *effect, size_t len) {
528 if (len == 4 && strncasecmp(effect, "none", 4) == 0) {
529 this->set_effect(uint32_t{0});
530 return *this;
531 }
532
533 bool found = false;
534 StringRef effect_ref(effect, len);
535 for (uint32_t i = 0; i < this->parent_->effects_.size(); i++) {
536 if (str_equals_case_insensitive(effect_ref, this->parent_->effects_[i]->get_name())) {
537 this->set_effect(i + 1);
538 found = true;
539 break;
540 }
541 }
542 if (!found) {
543 ESP_LOGW(TAG, "'%s': no such effect '%.*s'", this->parent_->get_name().c_str(), (int) len, effect);
544 }
545 return *this;
546}
566 this->set_transition_length(transition_length);
567 return *this;
568}
571 this->set_brightness(brightness);
572 return *this;
573}
575 if (this->parent_->get_traits().supports_color_mode(color_mode))
576 this->set_color_mode(color_mode);
577 return *this;
578}
581 this->set_color_brightness(brightness);
582 return *this;
583}
586 this->set_red(red);
587 return *this;
588}
591 this->set_green(green);
592 return *this;
593}
596 this->set_blue(blue);
597 return *this;
598}
601 this->set_white(white);
602 return *this;
603}
612 this->set_cold_white(cold_white);
613 return *this;
614}
617 this->set_warm_white(warm_white);
618 return *this;
619}
620IMPLEMENT_LIGHT_CALL_SETTER(state, bool, FLAG_HAS_STATE)
621IMPLEMENT_LIGHT_CALL_SETTER(transition_length, uint32_t, FLAG_HAS_TRANSITION)
622IMPLEMENT_LIGHT_CALL_SETTER(flash_length, uint32_t, FLAG_HAS_FLASH)
623IMPLEMENT_LIGHT_CALL_SETTER(brightness, float, FLAG_HAS_BRIGHTNESS)
624IMPLEMENT_LIGHT_CALL_SETTER(color_mode, ColorMode, FLAG_HAS_COLOR_MODE)
625IMPLEMENT_LIGHT_CALL_SETTER(color_brightness, float, FLAG_HAS_COLOR_BRIGHTNESS)
626IMPLEMENT_LIGHT_CALL_SETTER(red, float, FLAG_HAS_RED)
627IMPLEMENT_LIGHT_CALL_SETTER(green, float, FLAG_HAS_GREEN)
628IMPLEMENT_LIGHT_CALL_SETTER(blue, float, FLAG_HAS_BLUE)
629IMPLEMENT_LIGHT_CALL_SETTER(white, float, FLAG_HAS_WHITE)
630IMPLEMENT_LIGHT_CALL_SETTER(color_temperature, float, FLAG_HAS_COLOR_TEMPERATURE)
631IMPLEMENT_LIGHT_CALL_SETTER(cold_white, float, FLAG_HAS_COLD_WHITE)
632IMPLEMENT_LIGHT_CALL_SETTER(warm_white, float, FLAG_HAS_WARM_WHITE)
633LightCall &LightCall::set_effect(optional<std::string> effect) {
634 if (effect.has_value())
635 this->set_effect(*effect);
636 return *this;
637}
639 this->effect_ = effect_number;
641 return *this;
642}
643LightCall &LightCall::set_effect(optional<uint32_t> effect_number) {
644 if (effect_number.has_value()) {
645 this->effect_ = effect_number.value();
646 }
647 this->set_flag_(FLAG_HAS_EFFECT, effect_number.has_value());
648 return *this;
649}
651 this->set_flag_(FLAG_PUBLISH, publish);
652 return *this;
653}
655 this->set_flag_(FLAG_SAVE, save);
656 return *this;
657}
658LightCall &LightCall::set_rgb(float red, float green, float blue) {
659 this->set_red(red);
660 this->set_green(green);
661 this->set_blue(blue);
662 return *this;
663}
664LightCall &LightCall::set_rgbw(float red, float green, float blue, float white) {
665 this->set_rgb(red, green, blue);
666 this->set_white(white);
667 return *this;
668}
669
670} // namespace esphome::light
BedjetMode mode
BedJet operating mode.
const StringRef & get_name() const
Definition entity_base.h:71
static constexpr ColorMode first_value_from_mask(bitmask_t mask)
constexpr bitmask_t get_mask() const
Get the raw bitmask value for optimized operations.
constexpr size_t size() const
Count the number of values in the set.
static constexpr bool mask_contains(bitmask_t mask, ColorMode value)
StringRef is a reference to a string owned by something else.
Definition string_ref.h:26
constexpr const char * c_str() const
Definition string_ref.h:73
constexpr size_type size() const
Definition string_ref.h:74
static constexpr StringRef from_lit(const CharT(&s)[N])
Definition string_ref.h:50
This class represents a requested change in a light state.
Definition light_call.h:22
bool has_color_mode() const
Definition light_call.h:157
bool has_color_temperature() const
Definition light_call.h:154
LightCall & set_color_mode_if_supported(ColorMode color_mode)
Set the color mode of the light, if this mode is supported.
LightCall & set_color_temperature(optional< float > color_temperature)
Set the color temperature of the light in mireds for CWWW or RGBWW lights.
LightCall & set_publish(bool publish)
Set whether this light call should trigger a publish state.
void log_and_clear_unsupported_(FieldFlags flag, const LogString *feature, bool use_color_mode_log)
LightCall & set_color_brightness(optional< float > brightness)
Set the color brightness of the light from 0.0 (no color) to 1.0 (fully on)
bool has_warm_white() const
Definition light_call.h:156
void set_flag_(FieldFlags flag, bool value=true) ESPHOME_ALWAYS_INLINE
Definition light_call.h:226
bool has_brightness() const
Definition light_call.h:148
LightCall & set_rgb(float red, float green, float blue)
Set the RGB color of the light by RGB values.
bool has_cold_white() const
Definition light_call.h:155
void transform_parameters_(const LightTraits &traits)
Some color modes also can be set using non-native parameters, transform those calls.
LightCall & set_transition_length_if_supported(uint32_t transition_length)
Set the transition length property if the light supports transitions.
bool has_color_brightness() const
Definition light_call.h:149
LightCall & set_red_if_supported(float red)
Set the red property if the light supports RGB.
LightCall & set_effect(optional< std::string > effect)
Set the effect of the light by its name.
LightCall & set_color_brightness_if_supported(float brightness)
Set the color brightness property if the light supports RGBW.
LightCall & set_white(optional< float > white)
Set the white value value of the light from 0.0 to 1.0 for RGBW[W] lights.
LightCall & set_green(optional< float > green)
Set the green RGB value of the light from 0.0 to 1.0.
LightCall & set_green_if_supported(float green)
Set the green property if the light supports RGB.
LightCall & set_warm_white(optional< float > warm_white)
Set the warm white value of the light from 0.0 to 1.0.
LightCall & set_rgbw(float red, float green, float blue, float white)
Set the RGBW color of the light by RGB values.
LightCall & set_save(bool save)
Set whether this light call should trigger a save state to recover them at startup....
LightCall & set_color_temperature_if_supported(float color_temperature)
Set the color_temperature property if the light supports color temperature.
void clear_flag_(FieldFlags flag) ESPHOME_ALWAYS_INLINE
Definition light_call.h:235
LightCall & set_blue(optional< float > blue)
Set the blue RGB value of the light from 0.0 to 1.0.
LightCall & set_cold_white(optional< float > cold_white)
Set the cold white value of the light from 0.0 to 1.0.
LightCall & set_red(optional< float > red)
Set the red RGB value of the light from 0.0 to 1.0.
ColorMode get_active_color_mode_()
Get the currently targeted, or active if none set, color mode.
LightCall & set_white_if_supported(float white)
Set the white property if the light supports RGB.
LightCall & set_cold_white_if_supported(float cold_white)
Set the cold white property if the light supports cold white output.
LightCall & set_warm_white_if_supported(float warm_white)
Set the warm white property if the light supports cold white output.
ColorMode compute_color_mode_(const LightTraits &traits)
static constexpr uint16_t CLAMP_FLAGS_MASK
Definition light_call.h:217
LightCall & set_brightness_if_supported(float brightness)
Set the brightness property if the light supports brightness.
LightColorValues validate_()
Validate all properties and return the target light color values.
LightCall & set_brightness(optional< float > brightness)
Set the target brightness of the light from 0.0 (fully off) to 1.0 (fully on)
LightCall & set_blue_if_supported(float blue)
Set the blue property if the light supports RGB.
LightCall & from_light_color_values(const LightColorValues &values)
LightCall & set_state(optional< bool > state)
Set the binary ON/OFF state of the light.
LightCall & set_color_mode(optional< ColorMode > color_mode)
Set the color mode of the light.
color_mode_bitmask_t get_suitable_color_modes_mask_()
Get potential color modes bitmask for this light call.
LightCall & set_transition_length(optional< uint32_t > transition_length)
Set the transition length of this call in milliseconds.
This class represents the color state for a light object.
void set_color_mode(ColorMode color_mode)
Set the color mode of these light color values.
float get_brightness() const
Get the brightness property of these light color values. In range 0.0 to 1.0.
float get_blue() const
Get the blue property of these light color values. In range 0.0 to 1.0.
float get_white() const
Get the white property of these light color values. In range 0.0 to 1.0.
float get_color_temperature() const
Get the color temperature property of these light color values in mired.
float get_cold_white() const
Get the cold white property of these light color values. In range 0.0 to 1.0.
bool is_on() const
Get the binary true/false state of these light color values.
float get_green() const
Get the green property of these light color values. In range 0.0 to 1.0.
float get_warm_white() const
Get the warm white property of these light color values. In range 0.0 to 1.0.
ColorMode get_color_mode() const
Get the color mode of these light color values.
float get_red() const
Get the red property of these light color values. In range 0.0 to 1.0.
float get_color_brightness() const
Get the color brightness property of these light color values. In range 0.0 to 1.0.
FixedVector< LightEffect * > effects_
List of effects for this light.
void start_effect_(uint32_t effect_index)
Internal method to start an effect with the given index.
void stop_effect_()
Internal method to stop the current effect (if one is active).
LightColorValues remote_values
The remote color values reported to the frontend.
void save_remote_values_()
Internal method to save the current remote_values to the preferences.
void set_immediately_(const LightColorValues &target, bool set_remote_values)
Internal method to set the color values to target immediately (with no transition).
float gamma_uncorrect_lut(float value) const
Reverse gamma correction by binary-searching the forward LUT.
void publish_state()
Publish the currently active state to the frontend.
uint32_t active_effect_index_
Value for storing the index of the currently active effect. 0 if no effect is active.
void start_flash_(const LightColorValues &target, uint32_t length, bool set_remote_values)
Internal method to start a flash for the specified amount of time.
void start_transition_(const LightColorValues &target, uint32_t length, bool set_remote_values)
Internal method to start a transition to the target color with the given length.
uint32_t default_transition_length_
Default transition length for all transitions in ms.
std::unique_ptr< std::vector< LightTargetStateReachedListener * > > target_state_reached_listeners_
Listeners for target state reached.
This class is used to represent the capabilities of a light.
Definition light_traits.h:9
bool supports_color_mode(ColorMode color_mode) const
ColorModeMask get_supported_color_modes() const
const char * message
Definition component.cpp:35
bool state
Definition fan.h:2
Range range
Definition msa3xx.h:0
IMPLEMENT_LIGHT_CALL_SETTER(state, bool, FLAG_HAS_STATE) IMPLEMENT_LIGHT_CALL_SETTER(transition_length
float clamp_unit_float(float x)
PROGMEM_STRING_TABLE(ColorModeHumanStrings, "Unknown", "On/Off", "Brightness", "White", "Color temperature", "Cold/warm white", "RGB", "RGBW", "RGB + color temperature", "RGB + cold/warm white")
FiniteSetMask< ColorMode, ColorModeBitPolicy > ColorModeMask
Definition color_mode.h:147
bool float_out_of_unit_range(float x)
uint16_t color_mode_bitmask_t
Definition color_mode.h:108
ColorMode
Color modes are a combination of color capabilities that can be used at the same time.
Definition color_mode.h:49
@ RGB_COLD_WARM_WHITE
RGB color output, and separate cold and warm white outputs.
@ UNKNOWN
No color mode configured (cannot be a supported mode, only active when light is off).
@ RGB_WHITE
RGB color output and a separate white output.
@ RGB_COLOR_TEMPERATURE
RGB color output and a separate white output with controllable color temperature.
@ RGB
RGB color output.
@ COLOR_TEMPERATURE
Controllable color temperature output.
@ WHITE
White output only (use only if the light also has another color mode such as RGB).
@ COLD_WARM_WHITE
Cold and warm white output with individually controllable brightness.
@ BRIGHTNESS
Master brightness of the light can be controlled.
@ RGB
Color can be controlled using RGB format (includes a brightness control for the color).
@ COLOR_TEMPERATURE
Color temperature can be controlled.
@ WHITE
Brightness of white channel can be controlled separately from other channels.
@ COLD_WARM_WHITE
Brightness of cold and warm white output can be controlled.
std::string size_t len
bool str_equals_case_insensitive(const std::string &a, const std::string &b)
Compare strings for equality in case-insensitive manner.
Definition helpers.cpp:202
uint8_t progmem_read_byte(const uint8_t *addr)
Definition hal.h:42
static constexpr unsigned to_bit(ColorMode mode)
Definition color_mode.h:131