########################################################################################## ########################################################################################## # Title: DOWNSTAIRS KITCHEN - OVER PANTRY LEDS # Hardware: Sinilink MOSFET Board XY-WFMS (ESP8266) — sometimes listed as “XY-VFMS” # https://devices.esphome.io/devices/Sinilink-XY-VFMS # Repo: https://home.fox.co.nz/gitea/zorruno/zorruno-homeassistant/src/branch/master/esphome/esp-downstairskitchleds.yaml # # v1.4 - 2025-08-22 Improved power loss/on actions # v1.3 - 2025-08-22 Added a "max on time” setting (1-48 h, 0 = no limit) # v1.2 - 2025-08-21 Added defaults to “Device Specific Settings” in substitutions & a PWM % view # v1.1 - 2025-08-18 Full tidy-up as general-purpose LED strip controller # v1.0 - 2025-08-17 First setup (and replacement of Tasmota) # # ------------------------------------------ # DEVICE GPIO (Sinilink XY-WFMS) # ------------------------------------------ # GPIO02 Blue LED (used for ESPHome status) # GPIO04 MOSFET output (0 V when switched) and Red LED # GPIO12 Toggle button # GPIO13 Green LED (used to display fading status) # # ------------------------------------------ # OPERATION (as of v1.4) # ------------------------------------------ # 1. General-purpose LED controller. # 2. Designed for the Sinilink XY-WFMS board with a MOSFET output (claimed 5A, 5-36 V DC). # 3. Global setting for MAX % output to extend LED life. # 4. Minimum output setting; switches fully OFF at/below the minimum to avoid low-PWM flicker. # 5. PWM frequency is set to 500 Hz by default. You can increase it, but higher values caused # resets on this device. On ESP32 you can run much higher (~40 kHz). # 6. Min/Max output settings are not exposed in Home Assistant/MQTT by default, but can be. # With a 1 MB flash, space is tight and only minimal optimisation has been done so far. # 7. PACKAGES include common items: network settings, diagnostic entities, MQTT, and SNTP (optional). # 8. Default behaviour is to fade slowly up to full at power-up (so it can run with no network). # 9. The green LED flashes while fading (different patterns for up/down). The red LED follows the # output (it shares the MOSFET GPIO). # 10. Fade timing scales with the configured values (proportionally when starting mid-brightness). # 11. Useful 3D-printed case: https://cults3d.com/en/3d-model/tool/snapfit-enclosure-for-esp8266-sinilink-xy-wfms-5v-36v-mosfet-switch-module # 12. Exposed in Home Assistant/MQTT: # - Startup action # - Fade Up / Fade Down / Fade Stop buttons # - Fade Up/Down switch # - Normal On/Off switch (quick ramp up/down) # - Fade up/down times (0-60s) # - Output % (pre-gamma) and PWM % (post-gamma) # - Output Set (1-100, respects min/max) # - Device diagnostics (from the included package) # - Maximum 'on' time before automatic fade-down (1-48 h, 0 = no limit) # ########################################################################################## ########################################################################################## ########################################################################################## # SPECIFIC DEVICE VARIABLE SUBSTITUTIONS # If NOT using a secrets file, just replace these with the passwords etc (in quotes) ########################################################################################## substitutions: # Device Naming device_name: "esp-downstairskitchleds" friendly_name: "Downstairs Kitchen LEDs" description_comment: "Downstairs Kitchen Over Pantry LEDs :: Sinilink XY-WFMS" device_area: "Downstairs Kitchen" # Allows the ESP device to be automatically linked to an 'Area' in Home Assistant. # Project Naming project_name: "Sinilink.XY-WFMS" # Project details project_version: "v1.4" # Project version denotes release of the YAML file, allowing checking of deployed vs latest version # Passwords & Secrets api_key: !secret esp-api_key ota_pass: !secret esp-ota_pass static_ip_address: !secret esp-downstairskitchleds_ip # Unfortunately, you can't use substitutions inside secret names mqtt_local_command_main_topic: !secret mqtt_local_command_main_topic mqtt_local_status_main_topic: !secret mqtt_local_status_main_topic # MQTT LOCAL Controls mqtt_local_device_name: "downstairskitchen-pantryleds" mqtt_local_command_topic: "${mqtt_local_command_main_topic}/${mqtt_local_device_name}" # Topic we will use to command this locally without HA mqtt_local_status_topic: "${mqtt_local_status_main_topic}/${mqtt_local_device_name}" # Topic we will use to view status locally without HA mqtt_local_device_command_ON: "ON" mqtt_local_device_command_OFF: "OFF" # Device Specific Settings log_level: "NONE" # Define logging level: NONE, ERROR, WARN, INFO, DEBUG (default), VERBOSE, VERY_VERBOSE update_interval: "20s" # Update time for general sensors, etc. led_gamma: "1.2" # Gamma from 1.2-3 is sensible to normalise the LED fading vs PWM minimum_led_output: "1" # % If at this value or below, we'll switch it completely off maximum_led_output: "90" # % Maximum output; it is sometimes nice to limit the output for longevity or aesthetics max_on_default_hours: "6" # The maximum time the LEDs will be on, in case they get left on. 0 = no automatic turn-off ########################################################################################## # PACKAGES: Included Common Packages # https://esphome.io/components/packages.html ########################################################################################## packages: common_wifi: !include file: common/network_common.yaml vars: local_device_name: "${device_name}" local_static_ip_address: "${static_ip_address}" local_ota_pass: "${ota_pass}" common_api: !include #file: common/api_common.yaml file: common/api_common_noencryption.yaml vars: local_api_key: "${api_key}" #common_webportal: !include # file: common/webportal_common.yaml common_mqtt: !include file: common/mqtt_common.yaml vars: local_device_name: "${device_name}" #common_sntp: !include # file: common/sntp_common.yaml common_general_sensors: !include #file: common/sensors_common.yaml file: common/sensors_common_lite.yaml vars: local_friendly_name: "${friendly_name}" local_update_interval: "${update_interval}" ########################################################################################## # ESPHome CORE CONFIGURATION # https://esphome.io/components/esphome.html ########################################################################################## esphome: name: "${device_name}" friendly_name: "${friendly_name}" comment: "${description_comment}" area: "${device_area}" on_boot: priority: -200 then: # Keep the HA dropdown in sync with the stored mode - lambda: |- switch (id(restart_mode)) { case 0: id(restart_action).publish_state("Fade up to full"); break; case 1: id(restart_action).publish_state("Restore Brightness"); break; case 2: default: id(restart_action).publish_state("Remain Off"); break; } # Mode 0: Fade up to full (obeys fade settings; no on/off flicker) - if: condition: lambda: 'return id(restart_mode) == 0;' then: - lambda: 'id(ramp_switch_target_on) = true;' - script.execute: ramp_on_script # Mode 1: Restore Brightness using the light's normal defaults # (uses default_transition_length and on_turn_on handlers) - if: condition: lambda: 'return id(restart_mode) == 1;' then: - lambda: |- float target = id(last_brightness_pct); if (target < 0.0f) target = 0.0f; if (target > 100.0f) target = 100.0f; // Gently clamp to min/max caps to avoid an immediate post-on_state correction. const float minp = (float) id(min_brightness_pct); const float maxp = (float) id(max_brightness_pct); if (target > 0.0f && target < minp) target = minp; if (target > maxp) target = maxp; id(suppress_slider_sync) = true; if (target <= 0.0f) { id(ramp_switch_target_on) = false; auto call = id(mosfet_leds).make_call(); call.set_state(false); call.set_transition_length(0); call.perform(); } else { id(ramp_switch_target_on) = true; auto call = id(mosfet_leds).make_call(); call.set_state(true); call.set_brightness(target / 100.0f); // No transition_length here: use light.default_transition_length. call.perform(); } - delay: 300ms - lambda: 'id(suppress_slider_sync) = false;' # Mode 2: Remain Off (no blip; stays off) - if: condition: lambda: 'return id(restart_mode) == 2;' then: - script.stop: ramp_on_script - script.stop: ramp_off_script - lambda: 'id(ramp_switch_target_on) = false;' - light.turn_off: id: mosfet_leds transition_length: 0s platformio_options: build_unflags: - -flto build_flags: - -fno-lto - -Wl,--gc-sections - -ffunction-sections - -fdata-sections - -DNDEBUG ########################################################################################## # ESP PLATFORM AND FRAMEWORK # https://esphome.io/components/esp8266.html # https://esphome.io/components/esp32.html ########################################################################################## esp8266: board: esp01_1m restore_from_flash: true # restore some values on reboot #preferences: # flash_write_interval: 5min mdns: disabled: false # Disabling will make the build file smaller (and it is still available via static IP) ########################################################################################## # GLOBAL VARIABLES # https://esphome.io/components/globals.html ########################################################################################## globals: # Minimum Brightness % for LEDs (will switch off if <=) - id: min_brightness_pct type: int restore_value: true initial_value: "${minimum_led_output}" # start/finish at X% # Maximum Brightness % for LEDs (should never go beyond this) - id: max_brightness_pct type: int restore_value: false initial_value: "${maximum_led_output}" # hard cap; never exceed this # The maximum time the lights will stay on, in hours. Just in case they are left on. 0 = forever - id: max_on_hours type: int restore_value: true initial_value: '${max_on_default_hours}' # Default Fading Up Time (Selectable and will be retained) - id: ramp_up_ms # fade-in when turned ON type: int restore_value: true initial_value: '5000' # 5 s # Default Fading Down Time (Selectable and will be retained) - id: ramp_down_ms # fade-out when turned OFF type: int restore_value: true initial_value: '10000' # 10 s # Action on Restart. (0=Fade full, 1=Restore brightness, 2=Remain off) - id: restart_mode type: int restore_value: true initial_value: '0' # default = Fade Up to Full (so can be deployed with no other setup) # Determine last fade direction. # true when you asked the light to end up ON (Ramp Up) # false when you asked the light to end up OFF (Ramp Down) - id: ramp_switch_target_on type: bool restore_value: true initial_value: 'false' # Prevent jitter when adjusting the slider - id: suppress_slider_sync type: bool restore_value: false initial_value: 'false' # actual 0..100 seen last time, for restart - id: last_brightness_pct type: float restore_value: true initial_value: '0.0' # last published "Output Set (0-100)" integer - id: last_set_pos type: int restore_value: false initial_value: '-1' # helper to keep blink time == transition time - id: last_ramp_ms type: int restore_value: false initial_value: '0' ########################################################################################## # LOGGER COMPONENT # https://esphome.io/components/logger.html # Logs all log messages through the serial port and through MQTT topics. ########################################################################################## logger: level: "${log_level}" # INFO Level suggested, or DEBUG for testing baud_rate: 0 # set to 0 for no logging via UART, needed if you are using it for other serial things (eg PZEM, Serial control) ########################################################################################## # MQTT COMMANDS # This adds device-specific MQTT command triggers to the common MQTT configuration. ########################################################################################## mqtt: on_message: # Light control to ramp up - topic: "${mqtt_local_command_topic}/light/set" payload: "${mqtt_local_device_command_ON}" then: - switch.turn_on: mosfet_ramp_switch # Light control to ramp up - topic: "${mqtt_local_command_topic}/light/set" payload: "${mqtt_local_device_command_OFF}" then: - switch.turn_off: mosfet_ramp_switch ######################################################################################### # STATUS LED # https://esphome.io/components/status_led.html ######################################################################################### # SINILINK: Status LED Blue LED on GPIO2, active-low ######################################################################################### status_led: pin: number: GPIO2 inverted: true ########################################################################################## # SWITCH COMPONENT # https://esphome.io/components/switch/ ########################################################################################## switch: # Ramp-aware ON/OFF for HA (asymmetric, eased; no bounce) - platform: template id: mosfet_ramp_switch name: "${friendly_name} Fade Up/Down" icon: mdi:led-strip-variant lambda: |- return id(ramp_switch_target_on); turn_on_action: - lambda: 'id(ramp_switch_target_on) = true;' - script.stop: ramp_off_script - script.execute: ramp_on_script turn_off_action: - lambda: 'id(ramp_switch_target_on) = false;' - script.stop: ramp_on_script - script.execute: ramp_off_script ################################################################################################# # BUTTON COMPONENT # https://esphome.io/components/button/index.html ################################################################################################# button: # Start ramping UP (from current level) - platform: template id: fade_up_button name: "${friendly_name} Fade Up" icon: mdi:arrow-up-bold on_press: - lambda: |- id(ramp_switch_target_on) = true; id(mosfet_ramp_switch).publish_state(true); // reflect in HA immediately - script.stop: ramp_off_script - script.execute: ramp_on_script # Start ramping DOWN (from current level) - platform: template id: fade_down_button name: "${friendly_name} Fade Down" icon: mdi:arrow-down-bold on_press: - lambda: |- id(ramp_switch_target_on) = false; id(mosfet_ramp_switch).publish_state(false); // reflect in HA immediately - script.stop: ramp_on_script - script.execute: ramp_off_script # STOP any ramping (hold current brightness) - platform: template id: fade_stop_button name: "${friendly_name} Fade Stop" icon: mdi:pause on_press: # Stop any pending scripts (and their delayed turn_off) - script.stop: ramp_on_script - script.stop: ramp_off_script - script.stop: led_flash_up - script.stop: led_flash_down - output.turn_off: green_led_out # Cancel the light's transition by commanding the current level with 0 ms, # but DO NOT change the ramp switch state/flag. - lambda: |- const auto &cv = id(mosfet_leds).current_values; if (cv.is_on()) { auto call = id(mosfet_leds).make_call(); call.set_state(true); call.set_brightness(cv.get_brightness()); call.set_transition_length(0); call.perform(); } ######################################################################################### # SELECT COMPONENT # https://esphome.io/components/select/index.html ######################################################################################### select: - platform: template id: restart_action name: "${friendly_name} Restart Action" icon: mdi:restart optimistic: true options: - "Fade up to full" - "Restore Brightness" - "Remain Off" initial_option: "Restore Brightness" set_action: - lambda: |- if (x == "Fade up to full") { id(restart_mode) = 0; } else if (x == "Restore Brightness") { id(restart_mode) = 1; } else { id(restart_mode) = 2; } ######################################################################################### # BINARY SENSORS # https://esphome.io/components/binary_sensor/ ######################################################################################### binary_sensor: - platform: gpio id: btn_gpio12 name: "${friendly_name} Button" pin: number: GPIO12 mode: input: true pullup: true inverted: true filters: - delayed_on: 20ms - delayed_off: 20ms on_press: - if: condition: lambda: 'return id(ramp_switch_target_on);' then: # Target is currently ON → press should go OFF (start ramp-down) - lambda: |- id(ramp_switch_target_on) = false; id(mosfet_ramp_switch).publish_state(false); // reflect in HA immediately - script.stop: ramp_on_script - script.execute: ramp_off_script else: # Target is currently OFF → press should go ON (start ramp-up) - lambda: |- id(ramp_switch_target_on) = true; id(mosfet_ramp_switch).publish_state(true); // reflect in HA immediately - script.stop: ramp_off_script - script.execute: ramp_on_script ########################################################################################## # SENSOR COMPONENT # https://esphome.io/components/sensor/ ########################################################################################## sensor: - platform: template id: mosfet_output_pct name: "${friendly_name} Output (%)" unit_of_measurement: "%" icon: mdi:percent accuracy_decimals: 0 update_interval: 250ms # consider 200ms if you want fewer updates lambda: |- const auto &cv = id(mosfet_leds).current_values; return cv.is_on() ? (cv.get_brightness() * 100.0f) : 0.0f; on_value: then: - lambda: |- // Remember latest actual output (0..100) for "Restore Brightness" id(last_brightness_pct) = x; // If not suppressing sync, update the 0..100 slider only when its INT changes if (!id(suppress_slider_sync)) { float actual = x; // actual % float minp = (float) id(min_brightness_pct); float maxp = (float) id(max_brightness_pct); if (maxp <= minp) maxp = minp + 1.0f; float pos = (actual <= 0.0f) ? 0.0f : ((actual - minp) * 100.0f / (maxp - minp)); if (pos < 0.0f) pos = 0.0f; if (pos > 100.0f) pos = 100.0f; int pos_i = (int) floorf(pos + 0.5f); if (pos_i != id(last_set_pos)) { id(last_set_pos) = pos_i; id(led_output_set_pct).publish_state(pos_i); } } - platform: template id: mosfet_output_pwm_pct name: "${friendly_name} Output PWM (%)" unit_of_measurement: "%" icon: mdi:square-wave accuracy_decimals: 1 update_interval: 250ms lambda: |- const auto &cv = id(mosfet_leds).current_values; if (!cv.is_on()) return 0.0f; const float lin = cv.get_brightness(); // 0..1 (linear brightness) const float gamma = atof("${led_gamma}"); // parse substitution string → float float pwm = powf(lin, gamma); // approx PWM duty after gamma if (pwm < 0.0f) pwm = 0.0f; if (pwm > 1.0f) pwm = 1.0f; return pwm * 100.0f; ########################################################################################## # OUTPUT COMPONENT # https://esphome.io/components/light/index.html ########################################################################################## # An OUTPUT can be binary (0,1) or float, which is any value between 0 and 1. # PWM Outputs such as "ledc" are float. https://esphome.io/components/output/ledc.html ########################################################################################## output: - platform: esp8266_pwm id: mosfet_pwm pin: GPIO4 frequency: 500 Hz # high frequency to avoid audible/visible artifacts - platform: gpio id: green_led_out # Green LED pin: number: GPIO13 inverted: false ########################################################################################## # LIGHT COMPONENT # https://esphome.io/components/light/ ########################################################################################## light: - platform: monochromatic id: mosfet_leds name: "${friendly_name}" output: mosfet_pwm restore_mode: RESTORE_DEFAULT_OFF default_transition_length: 2s icon: mdi:led-strip-variant gamma_correct: "${led_gamma}" on_turn_on: - mqtt.publish: topic: "${mqtt_local_status_topic}/light/state" payload: "${mqtt_local_device_command_ON}" retain: true - lambda: 'id(ramp_switch_target_on) = true;' - script.stop: max_on_watchdog - if: condition: lambda: 'return id(max_on_hours) > 0;' then: - script.execute: max_on_watchdog on_turn_off: - mqtt.publish: topic: "${mqtt_local_status_topic}/light/state" payload: "${mqtt_local_device_command_OFF}" retain: true - lambda: 'id(ramp_switch_target_on) = false;' - script.stop: max_on_watchdog on_state: - lambda: |- const float cap = id(max_brightness_pct) / 100.0f; const auto &cv = id(mosfet_leds).current_values; if (cv.is_on() && cv.get_brightness() > cap + 0.001f) { auto call = id(mosfet_leds).make_call(); call.set_state(true); call.set_brightness(cap); call.set_transition_length(0); call.perform(); } ########################################################################################## # NUMBER COMPONENT # https://esphome.io/components/number/ ########################################################################################## number: - platform: template id: cfg_ramp_up_s name: "${friendly_name} Fade Up Time (s)" entity_category: config unit_of_measurement: s icon: mdi:timer-sand mode: slider min_value: 0 max_value: 60 step: 1 lambda: |- return (float) id(ramp_up_ms) / 1000.0f; set_action: - lambda: |- int secs = (int) floorf(x + 0.5f); if (secs < 0) secs = 0; if (secs > 60) secs = 60; id(ramp_up_ms) = secs * 1000; id(cfg_ramp_up_s).publish_state((float) secs); - platform: template id: cfg_ramp_down_s name: "${friendly_name} Fade Down Time (s)" entity_category: config unit_of_measurement: s icon: mdi:timer-sand-complete mode: slider min_value: 0 max_value: 60 step: 1 lambda: |- return (float) id(ramp_down_ms) / 1000.0f; set_action: - lambda: |- int secs = (int) floorf(x + 0.5f); if (secs < 0) secs = 0; if (secs > 60) secs = 60; id(ramp_down_ms) = secs * 1000; id(cfg_ramp_down_s).publish_state((float) secs); - platform: template id: led_output_set_pct name: "${friendly_name} Output Set (0-100)" icon: mdi:tune mode: slider min_value: 0 max_value: 100 step: 1 # Show current position mapped into 0..100 across [min..max] lambda: |- const auto &cv = id(mosfet_leds).current_values; float actual = cv.is_on() ? (cv.get_brightness() * 100.0f) : 0.0f; // 0..100 actual float minp = (float) id(min_brightness_pct); float maxp = (float) id(max_brightness_pct); if (maxp <= minp) maxp = minp + 1.0f; // avoid div/0 if (actual <= 0.0f) return 0.0f; // when OFF, show 0 float pos = (actual - minp) * 100.0f / (maxp - minp); if (pos < 0.0f) pos = 0.0f; if (pos > 100.0f) pos = 100.0f; return floorf(pos + 0.5f); // integer set_action: - if: condition: lambda: 'return x <= 0.0f;' then: # 0 means OFF - lambda: 'id(suppress_slider_sync) = true;' - script.stop: ramp_on_script - script.stop: ramp_off_script - light.turn_off: id: mosfet_leds transition_length: 200ms - lambda: |- id(ramp_switch_target_on) = false; id(led_output_set_pct).publish_state(0); - delay: 400ms - lambda: 'id(suppress_slider_sync) = false;' else: # Map 1..100 - [min..max] and set ON - lambda: |- id(suppress_slider_sync) = true; float pos = x; // 0..100 if (pos < 1.0f) pos = 1.0f; // 0 is OFF if (pos > 100.0f) pos = 100.0f; id(led_output_set_pct).publish_state((int) floorf(pos + 0.5f)); - script.stop: ramp_off_script - script.stop: ramp_on_script - light.turn_on: id: mosfet_leds brightness: !lambda |- float pos = id(led_output_set_pct).state; // 1..100 float minp = (float) id(min_brightness_pct); float maxp = (float) id(max_brightness_pct); if (maxp <= minp) maxp = minp + 1.0f; float out_pct = minp + (pos * (maxp - minp) / 100.0f); if (out_pct > maxp) out_pct = maxp; return out_pct / 100.0f; transition_length: 250ms - lambda: 'id(ramp_switch_target_on) = true;' - delay: 400ms - lambda: 'id(suppress_slider_sync) = false;' - platform: template id: cfg_max_on_hours name: "${friendly_name} Max On (h)" entity_category: config unit_of_measurement: h icon: mdi:timer-cog mode: slider min_value: 0 max_value: 48 step: 1 lambda: |- return (float) id(max_on_hours); set_action: - lambda: |- int hrs = (int) x; if (hrs < 0) hrs = 0; if (hrs > 48) hrs = 48; id(max_on_hours) = hrs; id(cfg_max_on_hours).publish_state((float) hrs); - if: condition: lambda: 'return id(mosfet_leds).current_values.is_on();' then: - script.stop: max_on_watchdog - if: condition: lambda: 'return id(max_on_hours) > 0;' then: - script.execute: max_on_watchdog ########################################################################################## # SCRIPT COMPONENT # https://esphome.io/components/script.html # Scripts can be executed nearly anywhere in your device configuration with a single call. ########################################################################################## script: # Blink pattern while ramping UP: quick double-blink, pause, repeat - id: led_flash_up mode: restart then: - while: condition: lambda: 'return true;' then: - output.turn_on: green_led_out - delay: 100ms - output.turn_off: green_led_out - delay: 100ms - output.turn_on: green_led_out - delay: 100ms - output.turn_off: green_led_out - delay: 400ms # Blink pattern while ramping DOWN: steady slow blink - id: led_flash_down mode: restart then: - while: condition: lambda: 'return true;' then: - output.turn_on: green_led_out - delay: 250ms - output.turn_off: green_led_out - delay: 250ms # Script: ramp up from current level. Obey global max. - id: ramp_on_script mode: restart then: - script.stop: ramp_off_script - script.stop: led_flash_down - script.execute: led_flash_up # Ensure we start at at least the floor without a visible "pop". - if: condition: lambda: |- const auto &cv = id(mosfet_leds).current_values; const float floor = id(min_brightness_pct) / 100.0f; return (!cv.is_on()) || (cv.get_brightness() + 0.0005f < floor); then: - light.turn_on: id: mosfet_leds brightness: !lambda 'return id(min_brightness_pct) / 100.0f;' transition_length: 80ms # Ramp from current (>= floor) to cap over a fraction of ramp_up_ms. - light.turn_on: id: mosfet_leds brightness: !lambda 'return id(max_brightness_pct) / 100.0f;' transition_length: !lambda |- const auto &cv = id(mosfet_leds).current_values; const float floor = id(min_brightness_pct) / 100.0f; const float cap = id(max_brightness_pct) / 100.0f; float curr = cv.is_on() ? cv.get_brightness() : 0.0f; if (curr < floor) curr = floor; if (curr > cap) curr = cap; float frac = (cap - curr) / (cap - floor); if (frac < 0.0f) frac = 0.0f; if (frac > 1.0f) frac = 1.0f; id(last_ramp_ms) = (int) (id(ramp_up_ms) * frac); return (uint32_t) id(last_ramp_ms); - delay: !lambda 'return (uint32_t) id(last_ramp_ms);' - script.stop: led_flash_up - output.turn_off: green_led_out # Script: ramp down from current level to floor, then cleanly cut to OFF - id: ramp_off_script mode: restart then: - script.stop: ramp_on_script - script.stop: led_flash_up - script.execute: led_flash_down - light.turn_on: id: mosfet_leds brightness: !lambda 'return id(min_brightness_pct) / 100.0f;' transition_length: !lambda |- const auto &cv = id(mosfet_leds).current_values; const float floor = id(min_brightness_pct) / 100.0f; float curr = cv.is_on() ? cv.get_brightness() : 0.0f; if (curr < floor) curr = floor; float frac = (curr - floor) / (1.0f - floor); if (frac < 0.0f) frac = 0.0f; if (frac > 1.0f) frac = 1.0f; id(last_ramp_ms) = (int) (id(ramp_down_ms) * frac); return (uint32_t) id(last_ramp_ms); - delay: !lambda 'return (uint32_t) id(last_ramp_ms);' - light.turn_off: id: mosfet_leds transition_length: 150ms - delay: 150ms - script.stop: led_flash_down - output.turn_off: green_led_out - lambda: |- auto call = id(mosfet_leds).make_call(); call.set_state(false); call.set_brightness(id(max_brightness_pct) / 100.0f); call.perform(); - id: max_on_watchdog mode: restart then: - if: condition: lambda: 'return id(max_on_hours) > 0;' then: - delay: !lambda 'return (uint32_t) (id(max_on_hours) * 3600000UL);' - if: condition: lambda: 'return id(mosfet_leds).current_values.is_on();' then: - lambda: |- id(ramp_switch_target_on) = false; id(mosfet_ramp_switch).publish_state(false); - script.stop: ramp_on_script - script.execute: ramp_off_script