##################################################################################################### ##### NSPANEL ESPHOME created by Blackymas - https://github.com/Blackymas/NSPanel_HA_Blueprint ##### ##### ESPHOME CORE ##### ##### PLEASE only make changes if it is necessary and also the required knowledge is available. ##### ##### For normal use with the Blueprint, no changes are necessary. ##### ##################################################################################################### --- substitutions: ############################## ## Change only in your ## ## local yaml substitutions ## device_name: NSPanel name: ${device_name} friendly_name: ${device_name} ota_password: ${wifi_password} temp_units: "°C" invalid_cooldown: "100ms" bytes_per_char: "1" ##### DON'T CHANGE THIS ###### version: "4.3.3d1" ############################## ##### External components ##### external_components: - source: # type: local # path: packages/Blackymas/components type: git url: https://github.com/Blackymas/NSPanel_HA_Blueprint ref: dev # To do: Change to tag when releasing components: - nspanel_ha_blueprint refresh: 300s - source: type: git url: https://github.com/edwardtfn/esphome ref: nextion-v432 components: - nextion # Change this when that PR#6192 gets released (2024.4?) refresh: 300s - source: type: git url: https://github.com/edwardtfn/esphome ref: adc-sampling components: - adc # Change this when PR#6330 is merged (2024.4?) refresh: 300s ##### ESPHOME CONFIGURATION ##### esphome: name: ${name} friendly_name: ${friendly_name} min_version: 2023.12.0 platformio_options: build_flags: - -Wno-missing-field-initializers - -D NSPANEL_HA_BLUEPRINT_CORE on_boot: - priority: 600.0 # This is where most sensors are set up. then: - lambda: |- if (isnan(blueprint_status->raw_state)) blueprint_status->publish_state(0); device_name->publish_state("${name}"); notification_label->publish_state(""); notification_text->publish_state(""); notification_unread->turn_off(); - wait_until: condition: - lambda: return (not isnan(stoi(baud_rate->state))); timeout: 60s - lambda: if (stoi(baud_rate->state) != tf_uart->get_baud_rate()) set_baud_rate->execute(stoi(baud_rate->state), true); - wait_until: condition: - lambda: return disp1->is_setup(); timeout: 60s - if: condition: - lambda: return (not disp1->is_detected()); then: - switch.turn_off: screen_power - delay: 2s - switch.turn_on: screen_power on_shutdown: - priority: 0 then: - lambda: |- // Make it unavailable to blueprint calls nextion_init->publish_state(false); // Update Wi-Fi icon disp1->set_component_text("home.wifi_icon", "\uE708"); // Update Wi-Fi icon color disp1->set_component_font_color("home.wifi_icon", 63488); - priority: 600.0 then: - switch.turn_off: screen_power esp32: board: esp32dev framework: type: esp-idf psram: id: ext_ram ##### WIFI SETUP ##### wifi: id: wifi_component power_save_mode: LIGHT networks: - id: wifi_default ssid: ${wifi_ssid} password: ${wifi_password} on_connect: then: - script.execute: watchdog on_disconnect: then: - script.execute: watchdog ##### OTA PASSWORD ##### ota: id: ota_std password: ${ota_password} safe_mode: true reboot_timeout: 3min num_attempts: 3 ##### Adds custom library for NSPanel HA Blueprint project nspanel_ha_blueprint: ##### LOGGER ##### logger: id: logger_std baud_rate: 0 ##### ENABLE RINGTONE MUSIC SUPPORT ##### rtttl: id: buzzer output: buzzer_out ##### CONFIGURE INTERNAL BUZZER ##### output: ##### BUZZER FOR PLAYING RINGTONES ##### - id: buzzer_out platform: ledc pin: number: 21 ##### UART FOR NEXTION DISPLAY ##### uart: - id: tf_uart tx_pin: 16 rx_pin: 17 baud_rate: 115200 ##### Keeps time display updated ##### time: - id: time_provider platform: homeassistant on_time: - seconds: 0 then: - lambda: |- refresh_datetime->execute(); refresh_relays->execute(3); refresh_hardware_buttons_bars->execute(3); refresh_wifi_icon->execute(); - seconds: 30 then: - script.execute: watchdog on_time_sync: then: - logger.log: "System clock synchronized" - script.execute: refresh_datetime json: # Can be replaced by web_server ##### START - API CONFIGURATION ##### api: id: api_server reboot_timeout: 60min on_client_connected: then: - script.execute: watchdog on_client_disconnected: then: - script.execute: watchdog services: # Dynamically configures button properties on a specified page, enhancing UI interactivity by allowing updates to button appearance and behavior based on given parameters. - service: button # yamllint disable-line rule:indentation variables: page: string # Identifier of the page where the button is located. id: string # Unique identifier for the button. state: bool # Determines the button's state, influencing background and other visual aspects. icon: string # Icon codepoint from HASwitchPlate Material Design Icons. Example: "\uE6E8" for mdi:lightbulb-on-outline. icon_color: int[] # RGB color array for the icon. icon_font: int # Nextion font identifier for the icon, default is 8. bri: string # Brightness level or other dynamic info to be displayed close to the icon. label: string # Main text label for the button. then: - lambda: |- if (page == current_page->state and !id(is_uploading_tft)) { disp1->set_component_picc((id + "pic").c_str(), state ? 47 : 46); disp1->set_component_picc((id + "bri").c_str(), state ? 47 : 46); disp1->set_component_picc((id + "text").c_str(), state ? 47 : 46); disp1->set_component_picc((id + "icon").c_str(), state ? 47 : 46); disp1->set_component_font((id + "icon").c_str(), icon_font); disp1->set_component_foreground_color((id + "bri").c_str(), state ? 10597 : 65535); disp1->set_component_foreground_color((id + "text").c_str(), state ? 10597 : 65535); disp1->set_component_font_color((id + "icon").c_str(), rgbTo565(icon_color)); disp1->set_component_text((id + "icon").c_str(), icon.c_str()); display_wrapped_text->execute((id + "text").c_str(), label.c_str(), 10); disp1->set_component_text((id + "bri").c_str(), (strcmp(bri.c_str(), "0") == 0) ? " " : bri.c_str()); set_component_visibility->execute((id + "pic").c_str(), true); set_component_visibility->execute((id + "icon").c_str(), true); set_component_visibility->execute((id + "text").c_str(), true); set_component_visibility->execute((id + "bri").c_str(), true); set_component_visibility->execute(id.c_str(), true); } # Sends custom commands directly to the display for dynamic interactions and updates. - service: command variables: cmd: string # Command string to be sent. Refer to the Nextion Instruction Set for supported commands: https://nextion.tech/instruction-set/ then: - lambda: |- if (!id(is_uploading_tft)) disp1->send_command_printf("%s", cmd.c_str()); # Changes the foreground color of a specified component on the display. - service: component_color variables: id: string # Identifier of the component to change color. Ensure this matches the component's ID in your display layout. color: int[] # New color for the component, specified as an RGB array (e.g., [255, 0, 0] for red). then: - lambda: |- if (!id(is_uploading_tft)) disp1->set_component_font_color(id.c_str(), rgbTo565(color)); # Updates the text of a specified component on the display. - service: component_text variables: id: string # Identifier of the component. Ensure it matches the component's ID in your display layout. txt: string # New text content to be displayed. Supports both static and dynamic content. then: - lambda: |- if (!id(is_uploading_tft)) disp1->set_component_text(id.c_str(), txt.c_str()); # Updates the value of a specified component on the display. - service: component_val variables: id: string # Identifier of the component to update. Must match the component's ID in the display layout. val: int # New integer value to set for the component. Adjust based on the data type you're displaying. then: - lambda: |- if (!id(is_uploading_tft)) disp1->set_component_value(id.c_str(), val); # Hides or shows a specified component on the display. - service: components_visibility variables: ids: string[] # Identifier of the component to be hidden/shown. Ensure this matches the component's ID in your display layout. visible: bool # Set to true to show the component, or false to hide it. then: - lambda: |- if (!id(is_uploading_tft)) { for (const std::string& component_id : ids) { if (!id(is_uploading_tft) and !component_id.empty()) { set_component_visibility->execute(component_id.c_str(), visible); } } } # Displays detailed information for a specific entity. - service: entity_details_show variables: entity: string # The ID of the entity for which details are shown. Supports "embedded_climate" for built-in climate control. back_page: string # Specifies the page to return to. Accepts "home" or "buttonpage01" to "buttonpage04". then: - lambda: |- if (!id(is_uploading_tft)) { HomeAssistantEntity entity_id = extractHomeAssistantEntity(entity); if (entity_id.domain != "invalid" or entity == "embedded_climate") { detailed_entity->publish_state(entity); if (entity_id.domain == "alarm_control_panel") entity_id.domain = "alarm"; goto_page->execute(entity_id.domain.c_str()); disp1->send_command_printf("back_page_id=%" PRIu8, get_page_id(back_page.c_str())); if (entity_id.domain == "climate") disp1->set_component_value("embedded", (entity == "embedded_climate") ? 1 : 0); } } # Hardware Button State Indication Service - service: hw_button_state variables: button_mask: int # Bitwise value for buttons: 1 for "left button", 2 for "right button", 3 for both buttons state: bool # State for the button(s) indication: true for active, false for inactive then: - lambda: |- if (!id(is_uploading_tft)) { // Updates the visual state indication for hardware buttons // Use bitwise AND to check specific bits if (button_mask & 1) // Checks if the least significant bit is set (left button) update_bitwise_setting(id(buttons_settings), state, ButtonSettings::ButtonLeft_State); if (button_mask & 2) // Checks if the second least significant bit is set (right button) update_bitwise_setting(id(buttons_settings), state, ButtonSettings::ButtonRight_State); // Refreshes the indication bars on the display refresh_hardware_buttons_bars->execute(button_mask); } # Icon Service - service: icon variables: id: string # Identifier of the component. See "Screen components" in the documentation. icon: string # Icon codepoint, e.g., "/uE6E8" for `mdi:lightbulb-on-outline`. icon_color: int[] # RGB color array for the icon, e.g., [0, 255, 0] for green. visible: bool # Set to `true` for visible or `false` for hidden. then: - lambda: |- if (!id(is_uploading_tft) and !id.empty()) { disp1->set_component_text(id.c_str(), visible ? icon.c_str() : ""); if (icon_color.size() == 3) disp1->set_component_font_color(id.c_str(), rgbTo565(icon_color)); set_component_visibility->execute(id.c_str(), visible); } # Transfers global settings from the blueprint to ESPHome, configuring the necessary parameters for optimal operation. - service: init_global variables: blueprint_version: string # Version of the blueprint in use. ent_value_xcen: int # Alignment of values on entities pages (0 for right (default), 1 for center or 2 for left). mui_please_confirm: string # Localized message for confirmation prompts. mui_unavailable: string # Localized message indicating unavailability. screensaver_time: bool # Toggles the screensaver time display. screensaver_time_font: int # Specifies the font id for the screensaver time display. screensaver_time_color: int[] # RGB color for the screensaver time display, e.g., [165, 42, 42] for reddish-brown. decimal_separator: string # The char to be used as decimal separator. # bytes_per_char: init then: - if: condition: - lambda: return (!id(is_uploading_tft)); then: - script.execute: id: global_settings blueprint_version: !lambda return blueprint_version; ent_value_xcen: !lambda return ent_value_xcen; mui_please_confirm: !lambda return mui_please_confirm; mui_unavailable: !lambda return mui_unavailable; screensaver_time: !lambda return screensaver_time; screensaver_time_font: !lambda return screensaver_time_font; screensaver_time_color: !lambda return screensaver_time_color; decimal_separator: !lambda return decimal_separator; # bytes_per_char: !lambda return bytes_per_char; - script.wait: global_settings - lambda: blueprint_status->publish_state(int(blueprint_status->raw_state) | (1 << 5)); # Configures NSPanel hardware (buttons, relays, etc.) settings - service: init_hardware variables: relay1_local_control: bool # Enable/disable local control for Relay 1. relay1_icon: string # Icon for Relay 1 (e.g., "\uE3A5" for mdi:numeric-1-box-outline). relay1_icon_color: int[] # RGB color array for Relay 1's icon. relay1_fallback: bool # Fallback state for Relay 1 in case of communication loss. relay2_local_control: bool # Enable/disable local control for Relay 2. relay2_icon: string # Icon for Relay 2 (e.g., "\uE3A8" for mdi:numeric-2-box-outline). relay2_icon_color: int[] # RGB color array for Relay 2's icon. relay2_fallback: bool # Fallback state for Relay 2 in case of communication loss. button_left: bool # Enable/disable left button status visualization. button_right: bool # Enable/disable right button status visualization. button_bar_pages: int # As uint representing the list of pages where the buttons bars will be visible button_bar_color_on: int[] # RGB color array for the hardware button bar when the status is `On`. button_bar_color_off: int[] # RGB color array for the hardware button bar when the status is `Off`. embedded_climate: bool # Indicates if climate control is integrated. embedded_climate_friendly_name: string # Friendly name for the climate control feature. embedded_indoor_temperature: bool # Enables indoor temperature display. then: - if: condition: - lambda: return (!id(is_uploading_tft)); then: - script.execute: id: init_hardware_climate embedded_climate: !lambda return embedded_climate; embedded_climate_friendly_name: !lambda return embedded_climate_friendly_name; embedded_indoor_temperature: !lambda return embedded_indoor_temperature; - lambda: |- if (!id(is_uploading_tft)) { using namespace esphome::display; // Relay settings update_bitwise_setting(id(relay_settings), relay1_local_control, RelaySettings::Relay1_Local); update_bitwise_setting(id(relay_settings), relay1_fallback, RelaySettings::Relay1_Fallback); update_bitwise_setting(id(relay_settings), relay2_local_control, RelaySettings::Relay2_Local); update_bitwise_setting(id(relay_settings), relay2_fallback, RelaySettings::Relay2_Fallback); // Relay icons if (not relay1_icon.empty()) copyStringToCharArray(id(home_relay1_icon), relay1_icon); if (not relay2_icon.empty()) copyStringToCharArray(id(home_relay2_icon), relay2_icon); // Relay icon's colors if (relay1_icon_color.size() == 3) { id(home_relay1_icon_color) = rgbTo565(relay1_icon_color); disp1->set_component_font_color("home.chip_relay1", id(home_relay1_icon_color)); } if (relay2_icon_color.size() == 3) { id(home_relay2_icon_color) = rgbTo565(relay2_icon_color); disp1->set_component_font_color("home.chip_relay2", id(home_relay2_icon_color)); } // Buttons bars settings id(buttons_bars_pages) = button_bar_pages; update_bitwise_setting(id(buttons_settings), button_left, ButtonSettings::ButtonLeft_Enabled); update_bitwise_setting(id(buttons_settings), button_right, ButtonSettings::ButtonRight_Enabled); if (button_bar_color_on.size() == 3) id(buttons_color_on) = rgbTo565(button_bar_color_on); if (button_bar_color_off.size() == 3) id(buttons_color_off) = rgbTo565(button_bar_color_off); // Refresh relays display refresh_relays->execute(3); refresh_hardware_buttons_bars->execute(3); } blueprint_status->publish_state(int(blueprint_status->raw_state) | (1 << 4)); # Sets up the "Home" page in ESPHome with customized settings and UI elements as defined in the project blueprint. - service: init_page_home variables: date_color: int[] # RGB color array for the date display. time_format: string # Time display format string, utilizing standard formatting symbols. time_color: int[] # RGB color array for the time display. meridiem: string[] # Optional array for AM/PM labels if included in time format. chip_font: int # Font Id for chip icons displayed on the "Home" page. custom_buttons_font: int # Font Id for icons on custom buttons. qrcode: bool # Enable/disable flag for QR code button display. qrcode_icon: string # Icon codepoint for QR code button, sourced from HASwitchPlate Material Design Icons. qrcode_icon_color: int[] # RGB color array for QR code button icon. entities_pages_icon: string # Icon codepoint for entities page button, sourced from HASwitchPlate Material Design Icons. entities_pages_icon_color: int[] # RGB color array for entities page button icon. utilities: bool # Enable/disable flag for utilities page button display. utilities_icon: string # Icon codepoint for utilities page button, sourced from HASwitchPlate Material Design Icons. utilities_icon_color: int[] # RGB color array for utilities page button icon. outdoor_temp_font: int # Font Id for outdoor temperature indication on the "Home" page. then: - lambda: |- if (!id(is_uploading_tft)) { using namespace esphome::display; // Localization id(mui_time_format) = time_format; if (meridiem.size() == 2) { id(mui_meridiem)[0] = meridiem[0]; id(mui_meridiem)[1] = meridiem[1]; } // Date/Time colors id(home_date_color) = rgbTo565(date_color); id(home_time_color) = rgbTo565(time_color); disp1->set_component_font_color("home.date", id(home_date_color)); disp1->set_component_font_color("home.time", id(home_time_color)); // Chips icon size disp1->set_component_font("home.chip_relay1", chip_font); disp1->set_component_font("home.chip_relay2", chip_font); disp1->set_component_font("home.chip_climate", chip_font); for (int i = 1; i <= 7; ++i) { disp1->send_command_printf("home.chip%02d.font=%" PRIi32, i, chip_font); } disp1->set_component_font("home.wifi_icon", chip_font); id(home_chip_font_id) = chip_font; // Custom buttons icon size id(home_custom_buttons_font_id) = custom_buttons_font; for (int i = 1; i <= 7; ++i) { disp1->send_command_printf("home.button%02d.font=%" PRIu8, i, id(home_custom_buttons_font_id)); } disp1->set_component_font("home.bt_notific", id(home_custom_buttons_font_id)); disp1->set_component_font("home.bt_qrcode", id(home_custom_buttons_font_id)); disp1->set_component_font("home.bt_entities", id(home_custom_buttons_font_id)); // Outdoor temperature font size disp1->set_component_font("home.outdoor_temp", outdoor_temp_font); // QRCode button set_component_visibility->execute("home.bt_qrcode", qrcode); disp1->set_component_text("home.bt_qrcode", qrcode_icon.c_str()); disp1->set_component_font_color("home.bt_qrcode", rgbTo565(qrcode_icon_color)); // Entities pages button disp1->set_component_text("home.bt_entities", entities_pages_icon.c_str()); disp1->set_component_font_color("home.bt_entities", rgbTo565(entities_pages_icon_color)); // Utilities button disp1->send_command_printf("is_utilities=%i", utilities ? 1 : 0); disp1->set_component_text("home.bt_utilities", utilities_icon.c_str()); disp1->set_component_font_color("home.bt_utilities", rgbTo565(utilities_icon_color)); blueprint_status->publish_state(int(blueprint_status->raw_state) | (1 << 1)); } # Populates the "Settings" page with user-configurable options, aligning with the project's blueprint for a cohesive and intuitive settings interface. - service: init_page_settings variables: reboot: string # Label for the reboot button, directing users on restarting the device. brightness: string # Caption for brightness adjustment controls. bright: string # Text label for the high brightness level slider, signaling a brighter screen option. dim: string # Text label for the dim brightness level slider, signaling a lower light option for energy saving. then: - lambda: |- if (!id(is_uploading_tft)) { if (not reboot.empty()) disp1->set_component_text_printf("settings.lbl_reboot", " %s", reboot.c_str()); disp1->set_component_text_printf("settings.lbl_brightness", " %s", brightness.c_str()); display_wrapped_text->execute("settings.lbl_bright", bright.c_str(), display_mode->state == 2 ? 25 : 10); display_wrapped_text->execute("settings.lbl_dim", dim.c_str(), display_mode->state == 2 ? 25 : 10); blueprint_status->publish_state(int(blueprint_status->raw_state) | (1 << 3)); } # This service removes any displayed notifications from the screen, helping to keep the user interface clean and focused on its primary functions. - service: notification_clear then: - lambda: |- if (!id(is_uploading_tft)) { if (current_page->state == "notification") goto_page->execute("home"); notification_label->publish_state(""); notification_text->publish_state(""); notification_unread->turn_off(); set_component_visibility->execute("home.bt_notific", false); } # Displays a notification message on the screen, useful for alerts or informational updates. - service: notification_show variables: label: string # Title or label for the notification, displayed in a prominent format. message: string # Detailed message or content of the notification. Include `\r` to insert a line break, allowing for custom formatting. then: - lambda: |- if (!id(is_uploading_tft)) { set_component_visibility->execute("home.bt_notific", true); goto_page->execute("notification"); timer_reset_all->execute(); disp1->set_component_text("notification.notifi_label", label.c_str()); display_wrapped_text->execute("notification.notifi_text01", message.c_str(), display_mode->state == 2 ? 23 : 32); notification_label->publish_state(label.c_str()); notification_text->publish_state(message.c_str()); notification_unread->turn_on(); if (notification_sound->state) buzzer->play("two short:d=4,o=5,b=100:16e6,16e6"); } # Updates the alarm settings page with current state and configuration, integrating with the panel's interface. - service: page_alarm variables: page_title: string # Title for the alarm settings page, displayed prominently at the top. state: string # Current state of the alarm system (e.g., "armed_home", "disarmed"). supported_features: int # Bitmask representing the alarm system's supported features, determining available controls on the page. code_format: string # Format required for the alarm code (numeric, alphanumeric). code_arm_required: bool # Indicates if a code is needed to arm the system. entity: string # Entity ID for the alarm system, enabling state updates and control. mui_alarm: string[] # Localized text for alarm control buttons (e.g., Arm, Disarm), allowing for a multilingual interface. then: - lambda: |- // Is page Alarm visible? if (current_page->state == "alarm" and !id(is_uploading_tft)) // To do: This page constructor should be moved to Blueprint { // Update alarm page detailed_entity->publish_state(entity); // Alarm page - Header update_alarm_icon->execute("icon_state", state.c_str()); if (page_title.find("\\r") != std::string::npos) { page_title = page_title.replace(page_title.find("\\r"), 2, " "); } disp1->set_component_text("page_label", page_title.c_str()); disp1->set_component_text("code_format", code_format.c_str()); disp1->set_component_text("code_arm_req", code_arm_required ? "1" : "0"); // Alarm page - Button's text display_wrapped_text->execute("bt_home_text", mui_alarm[0].c_str(), 10); display_wrapped_text->execute("bt_away_text", mui_alarm[1].c_str(), 10); display_wrapped_text->execute("bt_night_text", mui_alarm[2].c_str(), 10); display_wrapped_text->execute("bt_vacat_text", mui_alarm[3].c_str(), 10); display_wrapped_text->execute("bt_bypass_text", mui_alarm[4].c_str(), 10); display_wrapped_text->execute("bt_disarm_text", mui_alarm[5].c_str(), 10); // Alarm page - Buttons if (supported_features & 1 or state == "armed_home") // Alarm - Button - Home { disp1->set_component_pic("bt_home_pic", (state == "armed_home") ? 43 : 42); disp1->set_component_background_color("bt_home_text", (state == "armed_home") ? 19818 : 52857); disp1->set_component_background_color("bt_home_icon", (state == "armed_home") ? 19818 : 52857); disp1->set_component_font_color("bt_home_text", (state == "armed_home") ? 65535 : 0); disp1->set_component_font_color("bt_home_icon", (state == "armed_home") ? 65535 : 0); set_component_visibility->execute("bt_home", (state != "armed_home")); } if (supported_features & 2 or state == "armed_away") // Alarm - Button - Away { disp1->set_component_pic("bt_away_pic", (state == "armed_away") ? 43 : 42); disp1->set_component_background_color("bt_away_text", (state == "armed_away") ? 19818 : 52857); disp1->set_component_background_color("bt_away_icon", (state == "armed_away") ? 19818 : 52857); disp1->set_component_font_color("bt_away_text", (state == "armed_away") ? 65535 : 0); disp1->set_component_font_color("bt_away_icon", (state == "armed_away") ? 65535 : 0); set_component_visibility->execute("bt_away", (state != "armed_away")); } if (supported_features & 4 or state == "armed_night") // Alarm - Button - Night { disp1->set_component_pic("bt_night_pic", (state == "armed_night") ? 43 : 42); disp1->set_component_background_color("bt_night_text", (state == "armed_night") ? 19818 : 52857); disp1->set_component_background_color("bt_night_icon", (state == "armed_night") ? 19818 : 52857); disp1->set_component_font_color("bt_night_text", (state == "armed_night") ? 65535 : 0); disp1->set_component_font_color("bt_night_icon", (state == "armed_night") ? 65535 : 0); set_component_visibility->execute("bt_night", (state != "armed_night")); } if (supported_features & 32 or state == "armed_vacation") // Alarm - Button - Vacation { disp1->set_component_pic("bt_vacat_pic", (state == "armed_vacation") ? 43 : 42); disp1->set_component_background_color("bt_vacat_text", (state == "armed_vacation") ? 19818 : 52857); disp1->set_component_background_color("bt_vacat_icon", (state == "armed_vacation") ? 19818 : 52857); disp1->set_component_font_color("bt_vacat_text", (state == "armed_vacation") ? 65535 : 0); disp1->set_component_font_color("bt_vacat_icon", (state == "armed_vacation") ? 65535 : 0); set_component_visibility->execute("bt_vacat", (state != "armed_vacation")); } if (supported_features & 16 or state == "armed_bypass") // Alarm - Button - Custom bypass { disp1->set_component_pic("bt_bypass_pic", (state == "armed_bypass") ? 43 : 42); disp1->set_component_background_color("bt_bypass_text", (state == "armed_bypass") ? 19818 : 52857); disp1->set_component_background_color("bt_bypass_icon", (state == "armed_bypass") ? 19818 : 52857); disp1->set_component_font_color("bt_bypass_text", (state == "armed_bypass") ? 65535 : 0); disp1->set_component_font_color("bt_bypass_icon", (state == "armed_bypass") ? 65535 : 0); set_component_visibility->execute("bt_bypass", (state != "armed_bypass")); } if ( true ) // Alarm - Button - Disarm { disp1->set_component_pic("bt_disarm_pic", (state == "disarmed") ? 43 : 42); disp1->set_component_background_color("bt_disarm_text", (state == "disarmed") ? 19818 : 52857); disp1->set_component_background_color("bt_disarm_icon", (state == "disarmed") ? 19818 : 52857); disp1->set_component_font_color("bt_disarm_text", (state == "disarmed") ? 65535 : 0); disp1->set_component_font_color("bt_disarm_icon", (state == "disarmed") ? 65535 : 0); set_component_visibility->execute("bt_disarm", (state != "disarmed")); } } # Dynamically updates the climate page with the latest climate control settings and status. - service: page_climate variables: current_temp: float # Current temperature reading. supported_features: int # Bitmask indicating the supported features of the climate device, such as temperature control (1) and fan mode (4). target_temp: float # Desired target temperature setting. target_temp_high: float # Upper limit of the target temperature range for devices supporting ranges. target_temp_low: float # Lower limit of the target temperature range. temp_step: int # Temperature adjustment step size, indicating the granularity of changes (multiplied by 10 for precision). total_steps: int # Total adjustment steps available, derived from the device's temperature range and step size. temp_offset: int # Calibration offset applied to the temperature reading (multiplied by 10 for precision). climate_icon: string # Icon codepoint representing the current climate status, chosen from HASwitchPlate Material Design Icons. embedded_climate: bool # Indicates if climate control is integrated into the interface. entity: string # Entity ID of the climate device, allowing for direct control and status updates. then: - if: condition: lambda: return !id(is_uploading_tft); then: - lambda: if (current_page->state == "climate") detailed_entity->publish_state(entity); - script.execute: id: set_climate current_temp: !lambda return current_temp; supported_features: !lambda return supported_features; target_temp: !lambda return target_temp; target_temp_high: !lambda return target_temp_high; target_temp_low: !lambda return target_temp_low; temp_step: !lambda return temp_step; total_steps: !lambda return total_steps; temp_offset: !lambda return temp_offset; climate_icon: !lambda return climate_icon; embedded_climate: !lambda return embedded_climate; # Dynamically updates the media player page with current state and media information. - service: page_media_player variables: entity: string # Entity ID of the media player, used for state updates and control. state: string # Current playback state of the media player (e.g., "playing", "paused", "stopped"). is_volume_muted: bool # Indicates if the media volume is currently muted. friendly_name: string # Display name of the media player, shown as the page title. volume_level: int # Current volume level, typically expressed as a percentage. media_title: string # Title of the currently playing media. media_artist: string # Artist of the currently playing media. media_duration: float # Total duration of the current media in seconds. media_position: float # Current playback position in the media in seconds. media_position_delta: float # Time elapsed since the last media position update in seconds. supported_features: int # Bitmask indicating the media player's supported features (e.g., play, pause, volume control). then: - lambda: |- if (current_page->state == "media_player" and !id(is_uploading_tft)) { detailed_entity->publish_state(entity); disp1->set_component_text("page_label", friendly_name.c_str()); display_wrapped_text->execute("track", media_title.c_str(), display_mode->state == 2 ? 16 : 27); display_wrapped_text->execute("artist", media_artist.c_str(), display_mode->state == 2 ? 26 : 40); // on/off button if (supported_features & 128 and state == "off") { //TURN_ON disp1->set_component_foreground_color("bt_on_off", 65535); set_component_visibility->execute("bt_on_off", true); } else if (supported_features & 256 and state != "off") { //TURN_OFF disp1->set_component_foreground_color("bt_on_off", 10597); set_component_visibility->execute("bt_on_off", true); } else set_component_visibility->execute("bt_on_off", false); // play/pause button if ((supported_features & 512 or supported_features & 16384) and state != "playing" and state != "off") { //PLAY_MEDIA+PLAY disp1->set_component_text("bt_play_pause", "\uE409"); // mdi:play set_component_visibility->execute("bt_play_pause", true); } else if (supported_features & 1 and state == "playing" ) { //PAUSE disp1->set_component_text("bt_play_pause", "\uE3E3"); // mdi:pause set_component_visibility->execute("bt_play_pause", true); } else set_component_visibility->execute("bt_play_pause", false); // bt_prev button - PREVIOUS_TRACK set_component_visibility->execute("bt_prev", (supported_features & 16 and state != "off")); // bt_next button - NEXT_TRACK set_component_visibility->execute("bt_next", (supported_features & 32 and state != "off")); // Stop button - STOP //set_component_visibility->execute("bt_stop", (supported_features & 4096 and (state == "playing" or state == "paused"))); // mute/unmute button - VOLUME_MUTE disp1->set_component_value("is_muted", is_volume_muted ? 1 : 0); if (supported_features & 8 and is_volume_muted) { // unmute disp1->set_component_text("bt_mute", "\uEE07"); // mdi:volume-variant-off set_component_visibility->execute("bt_mute", true); } else if (supported_features & 8) { // mute disp1->set_component_text("bt_mute", "\uE57E"); // mdi:volume-low set_component_visibility->execute("bt_mute", true); } else set_component_visibility->execute("bt_mute", false); // VOLUME_SET if (supported_features & 4) { if (volume_level != id(last_volume_level)) { id(last_volume_level) = volume_level; disp1->set_component_text_printf("vol_text", "%" PRIu32 "%%", volume_level); disp1->set_component_value("vol_slider", volume_level); } set_component_visibility->execute("vol_slider", true); set_component_visibility->execute("bt_vol_down", true); set_component_visibility->execute("bt_vol_up", true); set_component_visibility->execute("vol_text", true); } else { set_component_visibility->execute("vol_slider", false); set_component_visibility->execute("bt_vol_down", false); set_component_visibility->execute("bt_vol_up", false); set_component_visibility->execute("vol_text", false); } if (media_duration > 0) { if (media_duration != id(last_media_duration) or media_position != id(last_media_position)) { id(last_media_duration) = media_duration; id(last_media_position) = media_position; disp1->set_component_value("prg_current", int(round(min(media_position + media_position_delta, media_duration)))); } disp1->set_component_value("prg_total", int(round(media_duration))); disp1->send_command_printf("prg_timer.en=%i", (state == "playing") ? 1 : 0); set_component_visibility->execute("time_current", true); set_component_visibility->execute("time_total", true); set_component_visibility->execute("time_progress", true); } else { disp1->send_command_printf("prg_timer.en=0"); set_component_visibility->execute("time_current", false); set_component_visibility->execute("time_total", false); set_component_visibility->execute("time_progress", false); } } # Dynamically displays QR codes on the ESPHome UI for sharing information such as WiFi passwords or website links. - service: qrcode variables: title: string # Heading or title for the QR code, offering context or instructions. qrcode: string # Data or URL to be encoded into the QR code. show: bool # Flag to immediately display the QR code page upon service invocation. then: - lambda: |- if (!id(is_uploading_tft)) { set_component_visibility->execute("home.bt_qrcode", !(qrcode.empty())); disp1->set_component_text("qrcode.qrcode_label", title.c_str()); disp1->set_component_text("qrcode.qrcode_value", qrcode.c_str()); if (show) goto_page->execute("qrcode"); blueprint_status->publish_state(int(blueprint_status->raw_state) | (1 << 2)); } # Plays melodies encoded in RTTTL format, suitable for audio feedback, notifications, or simple tunes. - service: rtttl_play variables: tone: string # The RTTTL string for the melody to be played. It should follow the RTTTL format, including the melody's name, default settings, and a sequence of notes. then: - lambda: if (!id(is_uploading_tft)) buzzer->play(tone); # Utilities group refresh - service: utilities_group_refresh variables: group_id: string value1: string value2: string direction: int then: - lambda: |- if (!id(is_uploading_tft) and !group_id.empty()) { uint8_t id = findUtilitiesGroupIndex(group_id.c_str()); if (id != UINT8_MAX) { // Update Value 1 if (!value1.empty() and strcmp(value1.c_str(), UtilitiesGroups[id].value1) != 0) { copyStringToCharArray(UtilitiesGroups[id].value1, value1); disp1->set_component_text(group_id.c_str(), value1.c_str()); } // Update Value 2 if (!value2.empty() and strcmp(value2.c_str(), UtilitiesGroups[id].value2) != 0) { copyStringToCharArray(UtilitiesGroups[id].value2, value2); disp1->set_component_text((group_id + "b").c_str(), value2.c_str()); } // Update direction if (group_id != "grid" and !isnan(direction) and direction != UtilitiesGroups[id].direction) { UtilitiesGroups[id].direction = direction; disp1->set_component_value((group_id + "_line_d").c_str(), direction); } } } # Updates an entity to display specific values with dynamic icons, names, and color codes. - service: value variables: id: string # Identifier of the entity. See "Screen components" for entity IDs. icon: string # Icon codepoint (e.g., "/uE6E8" for mdi:thermometer) from HASwitchPlate Material Design Icons. icon_color: int[] # RGB color array for the icon (e.g., [255, 0, 0] for red). name: string # Display name for the entity (e.g., "Temperature"). value: string # Actual value to display (e.g., "75°F"). value_color: int[] # RGB color array for the value text (e.g., [255, 255, 0] for yellow). then: - lambda: |- if (!id(is_uploading_tft) and !(id.empty())) { using namespace esphome::display; if (!(icon.empty())) disp1->set_component_text((id + "_icon").c_str(), icon.c_str()); if (icon_color.size() == 3) disp1->set_component_font_color((id + "_icon").c_str(), rgbTo565(icon_color)); if (!(name.empty())) disp1->set_component_text((id + "_label").c_str(), name.c_str()); if (!(value.empty())) disp1->set_component_text(id.c_str(), adjustDecimalSeparator(value, id(mui_decimal_separator)).c_str()); if (value_color.size() == 3) disp1->set_component_font_color(id.c_str(), rgbTo565(value_color)); if (current_page->state.find("entitypage") == 0 and !(value.empty())) { // Adjust value's font on entities pages // Adjusted length starts at 0 float adjusted_length = 0.0; // Iterate over each character in the string for (char const &c: value) { // Check if character is a space or other specified exceptions if (display_charset->state == 2 or std::string(" iljtIf'-,;:!.\"|()[]{}").find(c) != std::string::npos) { adjusted_length += 0.5; // Count these as half } else { adjusted_length += 1.0; // Count all other characters as 1 } } // Decide which font to use based on adjusted length if (adjusted_length > 8.0 and adjusted_length <= 12.0) { disp1->set_component_font(id.c_str(), 1); } else if (adjusted_length > 12.0) { disp1->set_component_font(id.c_str(), 0); } } } # Wake Up Service - service: wake_up variables: reset_timer: bool # Determines whether to reset the sleep and dimming timers upon waking the display. then: - lambda: |- if (!id(is_uploading_tft)) { if (current_page->state == "screensaver") goto_page->execute(wakeup_page_name->state.c_str()); if (reset_timer) timer_reset_all->execute(); else { timer_sleep->execute(); timer_dim->execute(); } } # yamllint enable rule:comments-indentation ##### START - DISPLAY START CONFIGURATION ##### display: - id: disp1 platform: nextion uart_id: tf_uart on_setup: lambda: |- if (!id(is_uploading_tft)) { nextion_init->publish_state(true); version_tft->update(); } on_page: lambda: |- if (!id(is_uploading_tft)) { if (current_page->state != page_names[x] or x == 9) { current_page->publish_state(page_names[x]); } } on_touch: lambda: |- if (!id(is_uploading_tft)) { timer_reset_all->execute(); switch (page_id) { case 0: // Boot switch (component_id) { case 4: // Reboot button if (!touch_event) { // Release App.safe_reboot(); } break; } break; case 1: // Home switch (component_id) { case 4: // indr_temp case 27: // indr_temp_icon if (!touch_event) { // Release detailed_entity->publish_state((id(is_embedded_thermostat)) ? "embedded_climate" : ""); disp1->set_component_value("climate.embedded", id(is_embedded_thermostat) ? 1 : 0); goto_page->execute("climate"); } break; } break; case 8: // Settings switch (component_id) { case 9: // Reboot button if (!touch_event) { // Release App.safe_reboot(); } break; } break; case 10: // light switch (component_id) { case 34: // power_button if (!touch_event) { // Release ha_call_service->execute("light.toggle", "", "", detailed_entity->state.c_str()); } break; } break; case 16: // notification switch (component_id) { case 7: // bt_accept if (!touch_event) { // Release notification_label->publish_state(""); notification_text->publish_state(""); notification_unread->turn_off(); goto_page->execute("home"); set_component_visibility->execute("home.bt_notific", false); } break; case 8: // bt_clear if (!touch_event) { // Release notification_unread->turn_off(); goto_page->execute("home"); } break; } break; case 22: // fan switch (component_id) { case 17: // bt_oscillate if (!touch_event) { // Release ha_call_service->execute("fan.oscillate", "oscillating", "toggle", detailed_entity->state.c_str()); } break; } break; } } ##### START - GLOBALS CONFIGURATION ##### globals: - id: mui_bytes_per_char type: uint8_t restore_value: true initial_value: ${bytes_per_char} ###### Buttons settings ###### # Bit # Settings # # 0 # Left Bt - Enabled # # 1 # Left Bt - State # # 2 # reserved # # 3 # reserved # # 4 # Right Bt - Enabled # # 5 # Right Bt - State # # 6 # reserved # # 7 # reserved # ############################## - id: buttons_settings type: uint8_t restore_value: false initial_value: '0' - id: buttons_color_on type: uint16_t restore_value: true initial_value: '7519' - id: buttons_color_off type: uint16_t restore_value: true initial_value: '10597' - id: buttons_bars_pages type: uint32_t restore_value: true initial_value: '1' ####### Relay settings ####### # Bit # Settings # # 0 # Relay 1 - Local # # 1 # Relay 1 - Fallback # # 2 # reserved # # 3 # reserved # # 4 # Relay 2 - Local # # 5 # Relay 2 - Fallback # # 6 # reserved # # 7 # reserved # ############################## - id: relay_settings type: uint8_t restore_value: true initial_value: '0' ##### Relay icons ##### - id: home_relay1_icon type: char[4] restore_value: true initial_value: '' - id: home_relay1_icon_color type: uint16_t restore_value: true initial_value: '65535' - id: home_relay2_icon type: char[4] restore_value: true initial_value: '' - id: home_relay2_icon_color type: uint16_t restore_value: true initial_value: '65535' ##### Is uploading TFT ##### - id: is_uploading_tft type: bool restore_value: false initial_value: 'false' ##### Media Player ##### ###### Last volume level from Home Assistant ###### - id: last_volume_level type: uint8_t restore_value: false initial_value: '0' ###### Last duration from Home Assistant ###### - id: last_media_duration type: uint restore_value: false initial_value: '0' ###### Last duration from Home Assistant ###### - id: last_media_position type: uint restore_value: false initial_value: '0' ##### Add-on Climate ##### ##### Is embedded thermostat set as main climate entity? ##### - id: is_embedded_thermostat type: bool restore_value: true initial_value: 'false' ##### Is embedded sensor used for indoor temperature? ##### - id: embedded_indoor_temp type: bool restore_value: true initial_value: 'true' ##### Date/time formats ##### - id: home_date_color type: uint16_t restore_value: true initial_value: '65535' - id: mui_time_format type: std::string restore_value: true max_restore_data_length: 15 initial_value: '"%H:%M"' - id: home_time_color type: uint16_t restore_value: true initial_value: '65535' - id: mui_meridiem type: std::array restore_value: false initial_value: '{"AM", "PM"}' #### Localization (MUI) #### - id: mui_please_confirm_global type: std::string restore_value: true initial_value: '"Please confirm"' - id: mui_unavailable_global type: std::string restore_value: true initial_value: '"Unavailable"' - id: mui_decimal_separator type: char restore_value: true initial_value: "'.'" ##### Chips ##### - id: home_chip_font_id type: uint8_t restore_value: true initial_value: '7' #### Custom buttons #### - id: home_custom_buttons_font_id type: uint8_t restore_value: true initial_value: '8' ##### Screensaver ##### - id: screensaver_display_time type: bool restore_value: true initial_value: 'false' - id: screensaver_display_time_font type: uint8_t restore_value: true initial_value: '6' - id: screensaver_display_time_color type: uint16_t restore_value: true initial_value: '16904' - id: page_entity_value_horizontal_alignment type: uint8_t restore_value: false initial_value: '1' # Horizontal alignment:0-Left;1-Center;2-Right ##### START - BINARY SENSOR CONFIGURATION ##### binary_sensor: ###### LEFT BUTTON BELOW DISPLAY TO TOGGLE RELAY##### - name: Left Button platform: gpio id: left_button pin: number: 14 inverted: true on_multi_click: - timing: &long_click-timing - ON for at least 0.8s invalid_cooldown: ${invalid_cooldown} then: - logger.log: "Left button - Long click" - script.execute: id: ha_button page: !lambda return current_page->state.c_str(); component: "hw_bt_left" command: "long_click" - timing: &short_click-timing - ON for at most 0.8s invalid_cooldown: ${invalid_cooldown} then: - logger.log: "Left button - Short click" - if: condition: or: - lambda: return (id(relay_settings) & RelaySettings::Relay1_Local); - and: - lambda: return (id(relay_settings) & RelaySettings::Relay1_Fallback); - or: - not: - api.connected: - not: - wifi.connected: then: - switch.toggle: relay_1 - script.execute: id: ha_button page: !lambda return current_page->state.c_str(); component: "hw_bt_left" command: "short_click" ##### RIGHT BUTTON BELOW DISPLAY TO TOGGLE RELAY ##### - name: Right Button platform: gpio id: right_button pin: number: 27 inverted: true on_multi_click: - timing: *long_click-timing invalid_cooldown: ${invalid_cooldown} then: - logger.log: "Right button - Long click" - script.execute: id: ha_button page: !lambda return current_page->state.c_str(); component: "hw_bt_right" command: "long_click" - timing: *short_click-timing invalid_cooldown: ${invalid_cooldown} then: - logger.log: "Right button - Short click" - if: condition: or: - lambda: return (id(relay_settings) & RelaySettings::Relay2_Local); - and: - lambda: return (id(relay_settings) & RelaySettings::Relay2_Fallback); - or: - not: - api.connected: - not: - wifi.connected: then: - switch.toggle: relay_2 - script.execute: id: ha_button page: !lambda return current_page->state.c_str(); component: "hw_bt_right" command: "short_click" ## Delays initial info from HA to the display ##### - name: Nextion display id: nextion_init platform: template device_class: connectivity publish_initial_state: true entity_category: diagnostic icon: mdi:tablet-dashboard lambda: |- return disp1->is_setup(); ##### START - BUTTON CONFIGURATION ##### button: ###### Factory Reset button ##### - name: Factory reset platform: factory_reset id: nspanel_factory_reset internal: false disabled_by_default: true icon: mdi:restart-alert ###### REBOOT BUTTON ##### - name: Restart platform: restart id: restart_nspanel ###### Power cycle Nextion Display ###### - name: Nextion display - Power cycle id: screen_power_cycle platform: template internal: false disabled_by_default: true icon: mdi:power-cycle entity_category: diagnostic on_press: - switch.turn_off: screen_power - delay: 1s - switch.turn_on: screen_power ##### START - NUMBER CONFIGURATION ##### number: ##### SCREEN BRIGHTNESS ##### - id: display_brightness name: Display Brightness platform: template entity_category: config unit_of_measurement: '%' min_value: 1 max_value: 100 initial_value: 100 step: 1 restore_value: true optimistic: true on_value: then: - lambda: |- disp1->send_command_printf("brightness=%i", int(x)); disp1->set_component_value("settings.brightslider", int(x)); if (current_page->state != "screensaver") { set_brightness->execute(x); timer_dim->execute(); timer_sleep->execute(); if (current_page->state == "settings") disp1->set_component_text_printf("bright_text", "%i%%", int(x)); } ##### SCREEN BRIGHTNESS DIMMED DOWN ##### - id: display_dim_brightness name: Display Brightness Dimdown platform: template entity_category: config unit_of_measurement: '%' min_value: 1 max_value: 100 initial_value: 25 step: 1 restore_value: true optimistic: true on_value: then: - lambda: |- disp1->send_command_printf("brightness_dim=%i", int(x)); disp1->set_component_value("settings.dimslider", int(x)); if (current_page->state != "screensaver" and current_brightness->state != x) { set_brightness->execute(x); timer_sleep->execute(); if (current_page->state == "settings") disp1->set_component_text_printf("dim_text", "%i%%", int(x)); } ##### SCREEN BRIGHTNESS SLEEP ##### - id: display_sleep_brightness name: Display Brightness Sleep platform: template entity_category: config unit_of_measurement: '%' min_value: 0 max_value: 100 initial_value: 0 step: 1 restore_value: true optimistic: true on_value: then: - lambda: |- disp1->send_command_printf("brightness_sleep=%i", int(x)); page_screensaver->execute(); ##### Temperature Correction ##### - id: temperature_correction name: Temperature Correction platform: template entity_category: config unit_of_measurement: °C min_value: -10 max_value: 10 initial_value: 0 step: 0.1 mode: box restore_value: true internal: false optimistic: true on_value: - logger.log: Temperature correction changed. - delay: 1s - lambda: temp_nspanel->publish_state(temp_nspanel->raw_state); ##### Timers settings ##### - name: Timeout Page platform: template id: timeout_page entity_category: config min_value: 0 max_value: 86400 initial_value: 15 step: 1 restore_value: true optimistic: true icon: mdi:timer unit_of_measurement: "s" on_value: - lambda: timer_page->execute(); - name: Timeout Dimming platform: template id: timeout_dim entity_category: config min_value: 0 max_value: 86400 initial_value: 30 step: 1 restore_value: true optimistic: true icon: mdi:timer unit_of_measurement: "s" on_value: - lambda: timer_dim->execute(); - name: Timeout Sleep platform: template id: timeout_sleep entity_category: config min_value: 0 max_value: 86400 initial_value: 60 step: 1 restore_value: true optimistic: true icon: mdi:timer unit_of_measurement: "s" on_value: - lambda: |- timer_dim->execute(); timer_sleep->execute(); ##### START - SELECT CONFIGURATION ##### select: - id: baud_rate name: Baud rate platform: template options: - "2400" - "4800" - "9600" - "19200" - "31250" - "38400" - "57600" - "115200" - "230400" - "250000" - "256000" - "512000" - "921600" initial_option: "115200" optimistic: true restore_value: true internal: false entity_category: config disabled_by_default: true icon: mdi:swap-horizontal on_value: - lambda: set_baud_rate->execute(stoi(x), true); - id: wakeup_page_name name: Wake-up page platform: template options: - buttonpage01 - buttonpage02 - buttonpage03 - buttonpage04 - climate - entitypage01 - entitypage02 - entitypage03 - entitypage04 - home - qrcode - utilities initial_option: home optimistic: true restore_value: true internal: false entity_category: config icon: mdi:page-next-outline on_value: - lambda: |- page_screensaver->execute(); ##### START - SENSOR CONFIGURATION ##### sensor: ##### Blueprint status ##### # Bit # Settings step # # 0 # reserved # # 1 # page_home # # 2 # qrcode # # 3 # page_settings # # 4 # relay_settings # # 5 # global_settings # # 6 # reserved # # 7 # reserved # ############################## - id: blueprint_status name: Blueprint platform: template unit_of_measurement: "%" accuracy_decimals: 1 entity_category: diagnostic icon: mdi:link-variant internal: false disabled_by_default: false filters: - lambda: |- if (!isnan(x) and x>0) return (x / 62) * 100.0f; else return 0; on_value: then: - script.execute: refresh_wifi_icon ##### INTERNAL TEMPERATURE SENSOR, ADC VALUE ##### - id: ntc_source platform: adc pin: 38 attenuation: 11db samples: 4 update_interval: 10s filters: - sliding_window_moving_average: window_size: 6 send_every: 1 ##### INTERNAL TEMPERATURE SENSOR, adc reading converted to resistance (calculation)##### - id: resistance_sensor platform: resistance sensor: ntc_source configuration: DOWNSTREAM resistor: 11.2kOhm ##### INTERNAL TEMPERATURE SENSOR, resistance to temperature (calculation) ##### - id: temp_nspanel name: Temperature platform: ntc sensor: resistance_sensor unit_of_measurement: °C internal: false calibration: b_constant: 3950 reference_temperature: 25°C reference_resistance: 10kOhm filters: - lambda: |- return x + temperature_correction->state; on_value: then: # Show panel's temperature if API or Wi-Fi are out - lambda: display_embedded_temp->execute(); ###### Display Brightness GET VALUE FROM NSPanel SLIDER ##### - id: brightslider name: brightness Slider platform: nextion variable_name: brightslider internal: true on_value: then: - number.set: id: display_brightness value: !lambda return int(x); - lambda: |- timer_reset_all->execute(); ###### Display DIM Brightness GET VALUE FROM NSPanel SLIDER ##### - id: dimslider name: dim brightness slider platform: nextion variable_name: dimslider internal: true on_value: then: - number.set: id: display_dim_brightness value: !lambda return int(x); - lambda: |- timer_reset_all->execute(); ###### Display Brightness - Current value (%) ##### - id: current_brightness name: Display Current brightness platform: nextion variable_name: dim precision: 0 accuracy_decimals: 0 unit_of_measurement: "%" icon: mdi:brightness-percent internal: false disabled_by_default: false ##### Charset (1 = International (original), 2 = CJK languages) - id: display_charset name: Display charset platform: nextion variable_name: charset precision: 0 accuracy_decimals: 0 internal: false icon: mdi:translate entity_category: diagnostic ##### Display mode (1 = EU, 2 = US, 3 = US Landscape) - id: display_mode name: Display mode platform: nextion variable_name: display_mode precision: 0 accuracy_decimals: 0 internal: false icon: mdi:phone-rotate-portrait entity_category: diagnostic ##### Wi-Fi Signal stregth - name: RSSI id: wifi_rssi platform: wifi_signal internal: false disabled_by_default: false icon: mdi:wifi entity_category: diagnostic ##### START - SWITCH CONFIGURATION ##### switch: ##### Notification unread ##### - name: Notification unread platform: template id: notification_unread entity_category: config optimistic: true restore_mode: ALWAYS_OFF on_turn_on: lambda: |- esphome::api::CustomAPIDevice ha_event; ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "notification_changed"}, {"component", "notification_unread"}, {"action", "turn_on"} }); on_turn_off: lambda: |- esphome::api::CustomAPIDevice ha_event; ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "notification_changed"}, {"component", "notification_unread"}, {"action", "turn_off"} }); ##### Notification sound ##### - name: Notification sound platform: template id: notification_sound entity_category: config optimistic: true restore_mode: RESTORE_DEFAULT_OFF ##### PHYSICAL SWITCH 1 ##### - name: Relay 1 platform: gpio id: relay_1 pin: number: 22 restore_mode: RESTORE_DEFAULT_OFF on_turn_on: then: - lambda: |- if (id(relay_settings) & RelaySettings::Relay1_Local) update_bitwise_setting(id(buttons_settings), true, ButtonSettings::ButtonLeft_State); refresh_relays->execute(1); refresh_hardware_buttons_bars->execute(1); on_turn_off: then: - lambda: |- if (id(relay_settings) & RelaySettings::Relay1_Local) update_bitwise_setting(id(buttons_settings), false, ButtonSettings::ButtonLeft_State); refresh_relays->execute(1); refresh_hardware_buttons_bars->execute(1); ##### PHYSICAL SWITCH 2 ###### - name: Relay 2 platform: gpio id: relay_2 pin: number: 19 restore_mode: RESTORE_DEFAULT_OFF on_turn_on: then: - lambda: |- if (id(relay_settings) & RelaySettings::Relay2_Local) update_bitwise_setting(id(buttons_settings), true, ButtonSettings::ButtonRight_State); refresh_relays->execute(2); refresh_hardware_buttons_bars->execute(2); on_turn_off: then: - lambda: |- if (id(relay_settings) & RelaySettings::Relay2_Local) update_bitwise_setting(id(buttons_settings), false, ButtonSettings::ButtonRight_State); refresh_relays->execute(2); refresh_hardware_buttons_bars->execute(2); ##### DISPLAY ALWAYS ON ##### - name: Nextion display - Power platform: gpio id: screen_power entity_category: diagnostic pin: number: 4 inverted: true restore_mode: ALWAYS_ON internal: true disabled_by_default: false on_turn_off: - lambda: |- nextion_init->publish_state(false); ##### START - TEXT SENSOR CONFIGURATION ##### text_sensor: ##### Device name - Used by bluepring to find service's names ##### - id: device_name name: Device Name platform: template icon: mdi:identifier entity_category: diagnostic internal: false disabled_by_default: false lambda: return {"${name}"}; filters: - lambda: |- std::string result; bool last_was_underscore = false; for (char& c : x) { if (isalnum(c)) { result += tolower(c); // Add alphanumeric characters as lowercase last_was_underscore = false; } else if (!last_was_underscore) { // Replace non-alphanumeric with '_' but avoid consecutive '_' result += '_'; last_was_underscore = true; } } return result; ##### Entity Id of the entity displayed on the detailed pages - id: detailed_entity name: Detailed Entity platform: template icon: mdi:tablet-dashboard internal: false disabled_by_default: false ##### Current page name ##### - id: current_page name: Current Page platform: nextion component_name: current_page icon: mdi:tablet-dashboard internal: false disabled_by_default: false filters: - lambda: |- x = x.c_str(); x.shrink_to_fit(); return x; on_value: lambda: |- if (!id(is_uploading_tft)) { page_changed->execute(); } - id: notification_label name: Notification Label platform: template on_value: lambda: |- esphome::api::CustomAPIDevice ha_event; ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "notification_changed"}, {"component", "notification_label"}, {"action", "new_value"} }); - id: notification_text name: Notification Text platform: template on_value: lambda: |- esphome::api::CustomAPIDevice ha_event; ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "notification_changed"}, {"component", "notification_text"}, {"action", "new_value"} }); ##### NSPanel event - Execute actions from ESPHome - NO push to HA ##### - id: disp1_local_event name: NSPanel local event platform: nextion nextion_id: disp1 component_name: localevent internal: true on_value: then: - lambda: |- DynamicJsonDocument json(1024); DeserializationError error = deserializeJson(json, x.c_str()); if (error) { ESP_LOGE("text_sensor.disp1_local_event", "Error parsing json: %s", x.c_str()); ESP_LOGE("text_sensor.disp1_local_event", "Error: %s", error.c_str()); } else { const std::string page = json["page"]; const std::string event = json["event"]; const std::string component = json["component"]; const std::string key = json["key"]; const std::string value = json["value"]; esphome::api::CustomAPIDevice ha_event; // Send event to Home Assistant if (event == "short_click" or event == "long_click") { ha_button->execute(page.c_str(), component.c_str(), event.c_str()); } else if (page == "light" or page == "climate") { // Generic event ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "generic"}, {"page", page.c_str()}, {"component", component.c_str()}, {"event", event.c_str()}, {"value", value.c_str()}, {"entity", detailed_entity->state.c_str()} }); } // page based actions if (page == "alarm") { const std::string code_format = json["code_format"]; const std::string code_arm_req = json["code_arm_req"]; const std::string title = json["mui"]; if (code_format == "number" and (key == "disarm" or code_arm_req == "1")) { goto_page->execute("keyb_num"); disp1->set_component_value("keyb_num.page_id", get_page_id("alarm")); //Calling from Alarm page disp1->set_component_text("keyb_num.domain", page.c_str()); disp1->set_component_text("keyb_num.key", key.c_str()); disp1->set_component_text("keyb_num.value", value.c_str()); disp1->set_component_text("keyb_num.entity", detailed_entity->state.c_str()); disp1->set_component_text("keyb_num.title", title.c_str()); } else service_call_alarm_control_panel->execute(detailed_entity->state.c_str(), key.c_str(), code_format.c_str(), ""); } else if (page == "climate") { const uint8_t embedded = json["embedded"]; change_climate_state->execute(embedded == 1, key.c_str(), value.c_str()); } else if (page == "cover") { if (key == "position") ha_call_service->execute("cover.set_cover_position", key.c_str(), value.c_str(), detailed_entity->state.c_str()); else ha_call_service->execute(("cover." + key).c_str(), "", "", detailed_entity->state.c_str()); } else if (page == "fan") { if (key == "stop" or value == "0") ha_call_service->execute("fan.turn_off", "", "", detailed_entity->state.c_str()); else ha_call_service->execute("fan.turn_on", key.c_str(), value.c_str(), detailed_entity->state.c_str()); } else if (page == "keyb_num") { const std::string base_domain = json["base_domain"]; if (base_domain == "alarm") { const std::string code_format = json["code_format"]; const std::string pin = json["pin"]; service_call_alarm_control_panel->execute(detailed_entity->state.c_str(), key.c_str(), code_format.c_str(), pin.c_str()); } goto_page->execute(base_domain.empty() ? "home" : base_domain.c_str()); } else if (page == "light") { if (key == "brightness_pct" or key == "color_temp") { ha_call_service->execute("light.turn_on", key.c_str(), value.c_str(), detailed_entity->state.c_str()); } else if (component == "rgb_color") { JsonArray rgb_color = json["value"]; if (rgb_color.size() == 3) { ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "service_call"}, {"service", "light.turn_on"}, {"key", "rgb_color"}, {"red",to_string(rgb_color[0].as())}, {"green",to_string(rgb_color[1].as())}, {"blue",to_string(rgb_color[2].as())}, {"entity", detailed_entity->state.c_str()} }); } } } else if (page == "media_player") { if (key == "volume_mute") { ha_call_service->execute("media_player.volume_mute", "is_volume_muted", value.c_str(), detailed_entity->state.c_str()); } else if (key == "volume_set") { const float value_float = json["value"].as(); ha_call_service->execute("media_player.volume_set", "volume_level", to_string(value_float / 100.0f), detailed_entity->state.c_str()); } else if (!key.empty()) { ha_call_service->execute((std::string("media_player.") + key.c_str()), "", "", detailed_entity->state.c_str()); } } } ##### Versioning ##### - id: version_blueprint name: Version Blueprint platform: template entity_category: diagnostic icon: mdi:tag-text-outline internal: false update_interval: never on_value: - lambda: |- check_versions->execute(); - id: version_tft name: Version TFT platform: nextion component_name: boot.tft_version entity_category: diagnostic icon: mdi:tag-text-outline internal: false update_interval: never on_value: - lambda: |- check_versions->execute(); ### Scripts ###### script: - id: boot_event mode: restart parameters: init: bool then: - lambda: |- if (init) { esphome::api::CustomAPIDevice ha_event; ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "boot"}, {"step", "start"} }); } - while: condition: - lambda: return (blueprint_status->state < 99); then: - delay: 10s - lambda: |- if (blueprint_status->state < 99) { esphome::api::CustomAPIDevice ha_event; ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "boot"}, {"step", "timeout"} }); } - id: boot_progress mode: restart parameters: step: uint then: - lambda: |- timer_reset_all->execute(); if (step == 10) { ESP_LOGD("script.boot_progress", "Progress: Completed"); disp1->set_component_value("boot.progress", 100); } else { step = std::min(100, int(round((blueprint_status->state + (step*10))/2))); ESP_LOGD("script.boot_progress", "Progress: %i%%", step); disp1->set_component_value("boot.progress", step); } if (current_page->state == "boot" and !isnan(display_charset->state) and !isnan(display_mode->state) and !version_tft->state.empty()) disp1->send_command_printf("tm_esphome.en=0"); - id: change_climate_state mode: restart parameters: embedded: bool key: string value: string then: - lambda: |- if (!id(is_uploading_tft)) { if (not embedded) { if (key == "temperature" or key == "target_temp_high" or key == "target_temp_low") ha_call_service->execute("climate.set_temperature", key.c_str(), to_string(stof(value) / 10), detailed_entity->state.c_str()); else if (key == "hvac_mode") ha_call_service->execute("climate.set_hvac_mode", key.c_str(), value.c_str(), detailed_entity->state.c_str()); } } - id: check_versions mode: restart then: - wait_until: condition: - lambda: |- return (compare_versions("${version}", version_tft->state.c_str()) and compare_versions("${version}", version_blueprint->state.c_str())); timeout: 60s - lambda: |- if (!id(is_uploading_tft)) { ESP_LOGD("script.check_versions", "Versions:"); ESP_LOGD("script.check_versions", " ESPHome: ${version}"); ESP_LOGD("script.check_versions", " TFT: %s", version_tft->state.c_str()); if (not compare_versions("${version}", version_tft->state.c_str())) ESP_LOGE("script.check_versions", "TFT version mismatch!"); ESP_LOGD("script.check_versions", " Blueprint: %s", version_blueprint->state.c_str()); if (not compare_versions("${version}", version_blueprint->state.c_str())) ESP_LOGE("script.check_versions", "Blueprint version mismatch!"); esphome::api::CustomAPIDevice ha_event; ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "version"}, {"tft", version_tft->state.c_str()}, {"esphome", "${version}"}, {"blueprint", version_blueprint->state.c_str()} }); } - id: display_embedded_temp mode: restart then: - lambda: |- if (id(embedded_indoor_temp) or !wifi_component->is_connected() or !api_server->is_connected()) { float unit_based_temperature = id(temp_nspanel).state; char buffer[15]; // Buffer for formatted temperature string if ("${temp_units}"[0] == 'F' || "${temp_units}"[0] == 'f' || "${temp_units}"[1] == 'F' || "${temp_units}"[1] == 'f') { unit_based_temperature = (unit_based_temperature * 9.0 / 5.0) + 32; // Convert to Fahrenheit if necessary snprintf(buffer, sizeof(buffer), "%.0f${temp_units}", unit_based_temperature); // Fahrenheit with no decimal } else { snprintf(buffer, sizeof(buffer), "%.1f${temp_units}", unit_based_temperature); // Celsius with one decimal } id(disp1)->set_component_text("home.indr_temp", adjustDecimalSeparator(buffer, id(mui_decimal_separator)).c_str()); } - id: display_wrapped_text mode: parallel max_runs: 5 parameters: component: string text_to_display: string line_length_limit: uint then: - lambda: |- int startPos = 0; int endPos = 0; std::string wrappedText = ""; if (text_to_display.find("\\r") != std::string::npos) { wrappedText = text_to_display; } else { while (startPos < text_to_display.length()) { while (text_to_display[startPos] == ' ' and startPos < text_to_display.length()) { startPos++; } int endPos = startPos + (line_length_limit * id(mui_bytes_per_char)); if (endPos >= text_to_display.length()) endPos = text_to_display.length(); else { while (endPos > startPos && text_to_display[endPos] != ' ') { endPos--; } if (endPos == startPos) endPos = startPos + (line_length_limit * id(mui_bytes_per_char)); // Handle case of long word } wrappedText += text_to_display.substr(startPos, endPos-startPos); if (endPos < text_to_display.length()) { while (text_to_display[endPos] == ' ') { endPos--; } if (endPos >= startPos) wrappedText += "\\r"; } startPos = endPos + 1; // Skip the space while (text_to_display[startPos] == ' ' and startPos < text_to_display.length()) { startPos++; } } } disp1->set_component_text(component.c_str(), wrappedText.c_str()); - id: global_settings mode: restart parameters: blueprint_version: string ent_value_xcen: int mui_please_confirm: string mui_unavailable: string screensaver_time: bool screensaver_time_font: int screensaver_time_color: int32_t[] decimal_separator: string # bytes_per_char: int then: - lambda: |- if (!id(is_uploading_tft)) { // Blueprint version version_blueprint->publish_state(blueprint_version.c_str()); disp1->set_component_text("boot.bluep_version", blueprint_version.c_str()); check_versions->execute(); // MUI strings id(mui_please_confirm_global) = mui_please_confirm; id(mui_unavailable_global) = mui_unavailable; // id(mui_bytes_per_char) = bytes_per_char; // Screen saver page (sleep) id(screensaver_display_time) = screensaver_time; id(screensaver_display_time_font) = screensaver_time_font; id(screensaver_display_time_color) = rgbTo565(screensaver_time_color); page_screensaver->execute(); // Entities pages alignment id(page_entity_value_horizontal_alignment) = ent_value_xcen; // Decimal separator if (not decimal_separator.empty()) id(mui_decimal_separator) = decimal_separator[0]; if (current_page->state != "boot") { // Update current page page_changed->execute(); } } - id: goto_page mode: restart parameters: page: string then: - lambda: |- if (current_page->state != page) { disp1->goto_page(page.c_str()); } - id: ha_button mode: parallel parameters: page: string component: string command: string then: - lambda: |- if (id(is_uploading_tft)) ha_button->stop(); timer_reset_all->execute(); esphome::api::CustomAPIDevice ha_event; ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "button_click"}, {"page", page}, {"component", component}, {"command", command} }); - id: ha_call_service mode: restart parameters: service: string key: string value: string entity: string then: - lambda: |- if (!id(is_uploading_tft) and !service.empty()) { esphome::api::CustomAPIDevice ha_event; ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "service_call"}, {"service", service}, {"key", key}, {"value", value}, {"entity", entity} }); } - id: init_hardware_climate mode: restart parameters: embedded_climate: bool # Indicates if climate control is integrated. embedded_climate_friendly_name: string # Friendly name for the climate control feature. embedded_indoor_temperature: bool # Enables indoor temperature display. then: - lambda: |- // Embedded thermostat id(is_embedded_thermostat) = embedded_climate; // Indoor temperature id(embedded_indoor_temp) = embedded_indoor_temperature; display_embedded_temp->execute(); - id: page_alarm mode: restart then: # There's nothing here so far - id: page_blank mode: restart then: - lambda: |- ESP_LOGW("script.page_blank", "Construct blank page"); disp1->set_component_text("esp_version", "ESP: ${version}"); // ESPHome version #ifdef ARDUINO disp1->set_component_text("framework", "Arduino"); #elif defined(USE_ESP_IDF) disp1->set_component_text("framework", "ESP-IDF"); #endif disp1->send_command_printf("tm_esphome.en=0"); - id: page_boot mode: single then: - logger.log: Page boot called - lambda: |- boot_progress->execute(0); set_brightness->execute(100); update_tft_info->execute(); disp1->set_component_text("boot.esph_version", "${version}"); // ESPHome version boot_progress->execute(1); #ifdef ARDUINO disp1->set_component_text("framework", "Arduino"); #elif defined(USE_ESP_IDF) disp1->set_component_text("framework", "ESP-IDF"); #endif boot_progress->execute(2); - logger.log: Wait for Wi-Fi - wait_until: condition: - lambda: return (wifi_component->is_connected()); timeout: 10s - if: condition: - lambda: return (wifi_component->is_connected()); then: # Wi-Fi connected - logger.log: Wi-Fi connected - lambda: |- boot_progress->execute(3); if (current_page->state == "boot") { #if ESPHOME_VERSION_CODE < VERSION_CODE(2024, 3, 0) // Code for ESPHome earlier than v2024.3.0 disp1->set_component_text("ip_addr", network::get_ip_address().str().c_str()); #else // Code for ESPHome v2024.3.0 or newer disp1->set_component_text("ip_addr", network::get_ip_addresses()[0].str().c_str()); #endif // ESPHome version based code set_brightness->execute(100); } - logger.log: Wait for API - wait_until: condition: - lambda: return (api_server->is_connected()); timeout: 10s - if: condition: - lambda: return (api_server->is_connected()); then: # API connected - logger.log: API connected - lambda: |- boot_progress->execute(4); if (blueprint_status->state <= 99) boot_event->execute(true); - wait_until: condition: - lambda: return (wifi_component->is_connected() and api_server->is_connected() and blueprint_status->state > 99); timeout: 5s - lambda: |- boot_progress->execute(5); disp1->send_command_printf("brightness=%i", int(display_brightness->state)); disp1->set_component_value("settings.brightslider", int(display_brightness->state)); disp1->send_command_printf("brightness_dim=%i", int(display_dim_brightness->state)); disp1->set_component_value("settings.dimslider", int(display_dim_brightness->state)); disp1->send_command_printf("brightness_sleep=%i", int(display_sleep_brightness->state)); disp1->send_command_printf("wakeup_page_id=%" PRIu8, get_page_id(wakeup_page_name->state.c_str())); boot_progress->execute(6); nextion_init->publish_state(disp1->is_setup()); if (api_server->is_connected() and disp1->is_setup()) { esphome::api::CustomAPIDevice ha_event; ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "boot"}, {"step", "nextion_init"} }); } boot_progress->execute(7); // Chips icon size for (int i = 1; i <= 7; ++i) { disp1->send_command_printf("home.chip%02d.font=%i", i, id(home_chip_font_id)); } // Custom buttons icon size for (int i = 1; i <= 7; ++i) { disp1->send_command_printf("home.button%02d.font=%i", i, id(home_custom_buttons_font_id)); } disp1->set_component_font("home.bt_notific", id(home_custom_buttons_font_id)); disp1->set_component_font("home.bt_qrcode", id(home_custom_buttons_font_id)); disp1->set_component_font("home.bt_entities", id(home_custom_buttons_font_id)); disp1->set_component_font("home.wifi_icon", id(home_chip_font_id)); disp1->set_component_font_color("home.chip_relay1", id(home_relay1_icon_color)); disp1->set_component_font_color("home.chip_relay2", id(home_relay2_icon_color)); boot_progress->execute(8); - wait_until: condition: - lambda: return (wifi_component->is_connected() and api_server->is_connected() and blueprint_status->state > 99); timeout: 10s - lambda: boot_progress->execute(9); - delay: 5s - lambda: boot_progress->execute(10); - delay: 1s - lambda: |- if (notification_sound->state) buzzer->play("two short:d=4,o=5,b=100:16e6,16e6"); set_brightness->execute(display_brightness->state); ESP_LOGD("script.page_boot", "Jump to wake-up page: %s", wakeup_page_name->state.c_str()); goto_page->execute(wakeup_page_name->state.c_str()); - id: page_buttonpage mode: restart parameters: page_number: uint then: # There's nothing here so far - id: page_buttonpage01 mode: restart then: - script.execute: id: page_buttonpage page_number: 1 - id: page_buttonpage02 mode: restart then: - script.execute: id: page_buttonpage page_number: 2 - id: page_buttonpage03 mode: restart then: - script.execute: id: page_buttonpage page_number: 3 - id: page_buttonpage04 mode: restart then: - script.execute: id: page_buttonpage page_number: 4 - id: page_changed mode: restart then: - lambda: |- // Report new page to logs ESP_LOGD("script.page_changed", "New page: %s", current_page->state.c_str()); if (current_page->state.empty()) { goto_page->execute("boot"); page_changed->stop(); } // Reset globals if (current_page->state != "alarm" && current_page->state != "climate" && current_page->state != "cover" && current_page->state != "fan" && current_page->state != "light" && current_page->state != "media_player" && current_page->state != "confirm" && current_page->state != "keyb_num") { detailed_entity->publish_state(""); disp1->send_command_printf("back_page_id=1"); } if (current_page->state != "media_player") { id(last_volume_level) = 0; id(last_media_duration) = 0; id(last_media_position) = 0; } if (!detailed_entity->state.empty()) ESP_LOGD("script.page_changed", "Entity shown: %s", detailed_entity->state.c_str()); // Update buttons bars on screen refresh_hardware_buttons_bars->execute(3); // Reset timers if (current_page->state != "screensaver") timer_reset_all->execute(); // Report new page to Home Assistant esphome::api::CustomAPIDevice ha_event; ha_event.fire_homeassistant_event("esphome.nspanel_ha_blueprint", { {"device_name", device_name->state.c_str()}, {"type", "page_changed"}, {"page", current_page->state.c_str()}, {"entity", detailed_entity->state.c_str()} }); // Call page constructor if (current_page->state == "alarm") page_alarm->execute(); else if (current_page->state == "blank") page_blank->execute(); else if (current_page->state == "boot") page_boot->execute(); else if (current_page->state == "buttonpage01") page_buttonpage01->execute(); else if (current_page->state == "buttonpage02") page_buttonpage02->execute(); else if (current_page->state == "buttonpage03") page_buttonpage03->execute(); else if (current_page->state == "buttonpage04") page_buttonpage04->execute(); else if (current_page->state == "climate") page_climate->execute(); else if (current_page->state == "confirm") page_confirm->execute(); else if (current_page->state == "cover") page_cover->execute(); else if (current_page->state == "entitypage01") page_entitypage01->execute(); else if (current_page->state == "entitypage02") page_entitypage02->execute(); else if (current_page->state == "entitypage03") page_entitypage03->execute(); else if (current_page->state == "entitypage04") page_entitypage04->execute(); else if (current_page->state == "fan") page_fan->execute(); else if (current_page->state == "home") page_home->execute(); else if (current_page->state == "keyb_num") page_keyb_num->execute(); else if (current_page->state == "light") page_light->execute(); else if (current_page->state == "media_player") page_media_player->execute(); else if (current_page->state == "notification") page_notification->execute(); else if (current_page->state == "qrcode") page_qrcode->execute(); else if (current_page->state == "screensaver") page_screensaver->execute(); else if (current_page->state == "settings") page_settings->execute(); else if (current_page->state == "utilities") page_utilities->execute(); else if (current_page->state == "weather01") page_weather01->execute(); else if (current_page->state == "weather02") page_weather02->execute(); else if (current_page->state == "weather03") page_weather03->execute(); else if (current_page->state == "weather04") page_weather04->execute(); else if (current_page->state == "weather05") page_weather05->execute(); - delay: 1s - lambda: refresh_hardware_buttons_bars->execute(3); - id: page_climate mode: restart then: # There's nothing here so far - id: page_confirm mode: restart then: - lambda: |- if (!id(is_uploading_tft)) display_wrapped_text->execute("confirm.title", id(mui_please_confirm_global).c_str(), 15); - id: page_cover mode: restart then: # There's nothing here so far - id: page_entitypage mode: restart parameters: page_number: uint then: - lambda: |- if (current_page->state.find("entitypage") == 0) { // Set value alignment if (id(page_entity_value_horizontal_alignment) != 1) { for (int i = 1; i <= 8; ++i) { disp1->send_command_printf("value%02d.xcen=%" PRIu8, i, id(page_entity_value_horizontal_alignment)); } } } - id: page_entitypage01 mode: restart then: - script.execute: id: page_entitypage page_number: 1 - id: page_entitypage02 mode: restart then: - script.execute: id: page_entitypage page_number: 2 - id: page_entitypage03 mode: restart then: - script.execute: id: page_entitypage page_number: 3 - id: page_entitypage04 mode: restart then: - script.execute: id: page_entitypage page_number: 4 - id: page_fan mode: restart then: # There's nothing here so far - id: page_home mode: restart then: - lambda: refresh_relays->execute(3); refresh_wifi_icon->execute(); - id: page_keyb_num mode: restart then: # There's nothing here so far - id: page_light mode: restart then: # There's nothing here so far - id: page_media_player mode: restart then: # There's nothing here so far - id: page_notification mode: restart then: - lambda: |- disp1->set_component_text("notification.notifi_label", notification_label->state.c_str()); display_wrapped_text->execute("notification.notifi_text01", notification_text->state.c_str(), display_mode->state == 2 ? 23 : 32); - id: page_qrcode mode: restart then: # There's nothing here so far - id: page_screensaver mode: restart then: - lambda: |- if (current_page->state == "screensaver" and !id(is_uploading_tft)) { disp1->send_command_printf("wakeup_page_id=%" PRIu8, get_page_id(wakeup_page_name->state.c_str())); if (id(screensaver_display_time)) { disp1->set_component_font("screensaver.text", id(screensaver_display_time_font)); disp1->set_component_font_color("screensaver.text", id(screensaver_display_time_color)); set_component_visibility->execute("screensaver.text", true); refresh_datetime->execute(); } set_brightness->execute(display_sleep_brightness->state); } - id: page_settings mode: restart then: - lambda: |- set_component_visibility->execute("page_settings.lbl_sleep", false); set_component_visibility->execute("page_settings.bt_sleep", false); - id: page_utilities mode: restart then: - lambda: resetUtilitiesGroups(); - id: page_weather mode: restart parameters: page_number: uint then: # There's nothing here so far - id: page_weather01 mode: restart then: - script.execute: id: page_weather page_number: 1 - id: page_weather02 mode: restart then: - script.execute: id: page_weather page_number: 2 - id: page_weather03 mode: restart then: - script.execute: id: page_weather page_number: 3 - id: page_weather04 mode: restart then: - script.execute: id: page_weather page_number: 4 - id: page_weather05 mode: restart then: - script.execute: id: page_weather page_number: 5 - id: refresh_datetime mode: restart then: - lambda: |- std::string time_format_str = id(mui_time_format); if (time_format_str.find("%-H") != std::string::npos) { time_format_str = time_format_str.replace(time_format_str.find("%-H"), sizeof("%-H")-1, to_string((int)(id(time_provider).now().hour))); } if (time_format_str.find("%-I") != std::string::npos) { if (id(time_provider).now().hour>12) { time_format_str = time_format_str.replace(time_format_str.find("%-I"), sizeof("%-I")-1, to_string((int)(id(time_provider).now().hour-12))); } else if (id(time_provider).now().hour==0) { time_format_str = time_format_str.replace(time_format_str.find("%-I"), sizeof("%-I")-1, "12"); } else { time_format_str = time_format_str.replace(time_format_str.find("%-I"), sizeof("%-I")-1, to_string((int)(id(time_provider).now().hour))); } } std::string meridiem_text = (id(time_provider).now().hour<12) ? id(mui_meridiem)[0] : id(mui_meridiem)[1]; if (current_page->state == "screensaver" and id(screensaver_display_time)) { std::string time_format_str_sleep = time_format_str; if (time_format_str_sleep.find("%p") != std::string::npos) time_format_str_sleep.replace(time_format_str_sleep.find("%p"), sizeof("%p")-1, meridiem_text.c_str()); disp1->set_component_text("text", id(time_provider).now().strftime(time_format_str_sleep).c_str()); } disp1->set_component_text("home.meridiem", (time_format_str.find("%p") != std::string::npos) ? meridiem_text.c_str() : " "); disp1->set_component_text("home.time", id(time_provider).now().strftime(time_format_str).c_str()); - id: refresh_hardware_buttons_bars mode: restart parameters: button_mask: uint8_t then: - lambda: |- #if ESPHOME_LOG_LEVEL > ESPHOME_LOG_LEVEL_DEBUG ESP_LOGV("script.refresh_hardware_buttons_bars", "Page: %s", current_page->state.c_str()); ESP_LOGV("script.refresh_hardware_buttons_bars", "Page id: %i", get_page_id(current_page->state.c_str())); ESP_LOGV("script.refresh_hardware_buttons_bars", "buttons_bars_pages: %i", id(buttons_bars_pages)); ESP_LOGV("script.refresh_hardware_buttons_bars", "relay_settings: %i", id(relay_settings)); ESP_LOGV("script.refresh_hardware_buttons_bars", "button_mask: %i", button_mask); #endif if (!id(is_uploading_tft) and ((id(buttons_bars_pages) & (1 << get_page_id(current_page->state.c_str()))) != 0)) { switch (int(display_mode->state)) { case 1: // EU model if (button_mask & 1 and id(buttons_settings) & ButtonSettings::ButtonLeft_Enabled) { // Left button disp1->fill_area(48, 307, 118, 3, (id(buttons_settings) & ButtonSettings::ButtonLeft_State) ? id(buttons_color_on) : id(buttons_color_off)); disp1->fill_area(47, 308, 120, 1, (id(buttons_settings) & ButtonSettings::ButtonLeft_State) ? id(buttons_color_on) : id(buttons_color_off)); } if (button_mask & 2 and id(buttons_settings) & ButtonSettings::ButtonRight_Enabled) { // Right button disp1->fill_area(289, 307, 118, 3, (id(buttons_settings) & ButtonSettings::ButtonRight_State) ? id(buttons_color_on) : id(buttons_color_off)); disp1->fill_area(288, 308, 120, 1, (id(buttons_settings) & ButtonSettings::ButtonRight_State) ? id(buttons_color_on) : id(buttons_color_off)); } break; case 2: // US Portrait if (button_mask & 1 and id(buttons_settings) & ButtonSettings::ButtonLeft_Enabled) { // Left button disp1->fill_area(17, 466, 118, 3, (id(buttons_settings) & ButtonSettings::ButtonLeft_State) ? id(buttons_color_on) : id(buttons_color_off)); disp1->fill_area(16, 467, 120, 1, (id(buttons_settings) & ButtonSettings::ButtonLeft_State) ? id(buttons_color_on) : id(buttons_color_off)); } if (button_mask & 2 and id(buttons_settings) & ButtonSettings::ButtonRight_Enabled) { // Right button disp1->fill_area(184, 466, 118, 3, (id(buttons_settings) & ButtonSettings::ButtonRight_State) ? id(buttons_color_on) : id(buttons_color_off)); disp1->fill_area(183, 467, 120, 1, (id(buttons_settings) & ButtonSettings::ButtonRight_State) ? id(buttons_color_on) : id(buttons_color_off)); } break; case 3: // US Landscape if (button_mask & 1 and id(buttons_settings) & ButtonSettings::ButtonLeft_Enabled) { // Left button disp1->fill_area(467, 174, 3, 118, (id(buttons_settings) & ButtonSettings::ButtonLeft_State) ? id(buttons_color_on) : id(buttons_color_off)); disp1->fill_area(468, 173, 1, 120, (id(buttons_settings) & ButtonSettings::ButtonLeft_State) ? id(buttons_color_on) : id(buttons_color_off)); } if (button_mask & 2 and id(buttons_settings) & ButtonSettings::ButtonRight_Enabled) { // Right button disp1->fill_area(467, 28, 3, 118, (id(buttons_settings) & ButtonSettings::ButtonRight_State) ? id(buttons_color_on) : id(buttons_color_off)); disp1->fill_area(468, 27, 1, 120, (id(buttons_settings) & ButtonSettings::ButtonRight_State) ? id(buttons_color_on) : id(buttons_color_off)); } break; } } - id: refresh_relays mode: restart parameters: relay_mask: uint8_t then: - lambda: |- // Chips - Relays if (!id(is_uploading_tft) and relay_mask & 1) disp1->set_component_text("home.chip_relay1", (relay_1->state) ? id(home_relay1_icon) : "\uFFFF"); if (!id(is_uploading_tft) and relay_mask & 2) disp1->set_component_text("home.chip_relay2", (relay_2->state) ? id(home_relay2_icon) : "\uFFFF"); - id: refresh_wifi_icon mode: restart then: - lambda: |- if (!id(is_uploading_tft)) { disp1->send_command_printf("api=%i", (wifi_component->is_connected() and api_server->is_connected() and blueprint_status->state > 99) ? 1 : 0); // Update Wi-Fi icon color disp1->set_component_font_color("home.wifi_icon", (blueprint_status->state > 99) ? (wifi_rssi->state > -70 ? 33808 : 64992) : 63488); // Update Wi-Fi icon disp1->set_component_text("home.wifi_icon", wifi_component->is_connected() ? (api_server->is_connected() ? ((blueprint_status->state > 99) ? "\uE5A8" : // mdi:wifi - All right! "\uE7CF") : // mdi:home-assistant - Blueprint is out "\uF256") : // mdi:api-off "\uE5A9"); // mdi:wifi-off } - id: service_call_alarm_control_panel mode: restart parameters: entity: string key: string code_format: string pin: string then: - lambda: |- std::string service = ""; if (key == "home") service = "alarm_control_panel.alarm_arm_home"; else if (key == "away") service = "alarm_control_panel.alarm_arm_away"; else if (key == "night") service = "alarm_control_panel.alarm_arm_night"; else if (key == "vacation") service = "alarm_control_panel.alarm_arm_vacation"; else if (key == "bypass") service = "alarm_control_panel.alarm_arm_custom_bypass"; else if (key == "disarm") service = "alarm_control_panel.alarm_disarm"; if (not service.empty()) { HomeassistantServiceResponse resp; HomeassistantServiceMap resp_kv; resp.service = service.c_str(); resp_kv.key = "entity_id"; resp_kv.value = entity.c_str(); resp.data.push_back(resp_kv); if (not pin.empty()) { resp_kv.key = "code"; resp_kv.value = pin.c_str(); resp.data.push_back(resp_kv); } api_server->send_homeassistant_service_call(resp); } - id: set_baud_rate mode: restart parameters: baud_rate: uint32_t definitive: bool then: - if: condition: - lambda: return (tf_uart->get_baud_rate() != baud_rate); then: - lambda: |- ESP_LOGD("script.set_baud_rate", "Baud rate changing from %" PRIu32 " to %" PRIu32 " bps", tf_uart->get_baud_rate(), baud_rate); ESP_LOGD("script.set_baud_rate", "Flush UART"); - wait_until: condition: - lambda: return (tf_uart->available() < 1); timeout: 5s - lambda: |- ESP_LOGD("script.set_baud_rate", "Sending instruction '%s=%" PRIu32 "' to Nextion", definitive ? "bauds" : "baud", baud_rate); disp1->send_command_printf("%s=%" PRIu32, definitive ? "bauds" : "baud", baud_rate); ESP_LOGD("script.set_baud_rate", "Flush UART"); - wait_until: condition: - lambda: return (tf_uart->available() < 1); timeout: 5s - lambda: |- ESP_LOGD("script.set_baud_rate", "Set ESPHome new baud rate to %" PRIu32 " bps", baud_rate); tf_uart->set_baud_rate(baud_rate); tf_uart->load_settings(); ESP_LOGD("script.set_baud_rate", "Current baud rate: %" PRIu32 " bps", tf_uart->get_baud_rate()); - id: set_brightness mode: restart parameters: brightness: float then: - if: condition: - lambda: return (!id(is_uploading_tft)); then: - lambda: |- if (!id(is_uploading_tft)) { if (brightness == display_brightness->state and current_page->state != "boot" and current_page->state != "screensaver") disp1->send_command_printf("wakeup_timer.en=1"); else disp1->set_backlight_brightness(brightness / 100.0f); current_brightness->update(); } - delay: 5s - lambda: if (!id(is_uploading_tft)) current_brightness->update(); - id: set_climate mode: restart parameters: current_temp: float supported_features: int target_temp: float target_temp_high: float target_temp_low: float temp_step: uint total_steps: uint temp_offset: int climate_icon: string embedded_climate: bool then: - lambda: |- if (!id(is_uploading_tft) and current_page->state == "climate") { bool useDecimal = (temp_step % 10 != 0); char buffer[15]; disp1->send_command_printf("climateslider.maxval=%i", total_steps); disp1->send_command_printf("slider_high.maxval=%i", total_steps); disp1->send_command_printf("slider_low.maxval=%i", total_steps); disp1->set_component_value("temp_offset", temp_offset); disp1->set_component_value("temp_step", temp_step); char dec_separator_str[2] = {id(mui_decimal_separator), '\0'}; disp1->set_component_text("dec_separator", dec_separator_str); set_component_visibility->execute("current_temp", true); if (current_temp > -999) { snprintf(buffer, sizeof(buffer), (useDecimal) ? "%.1f°" : "%.0f°", current_temp); disp1->set_component_text("current_temp", adjustDecimalSeparator(buffer, id(mui_decimal_separator)).c_str()); } else disp1->set_component_text("current_temp", id(mui_unavailable_global).c_str()); if (target_temp > -999) { // Target temp enabled disp1->set_component_value("active_slider", 0); snprintf(buffer, sizeof(buffer), (useDecimal) ? "%.1f°" : "%.0f°", target_temp); disp1->set_component_text("target_high", adjustDecimalSeparator(buffer, id(mui_decimal_separator)).c_str()); disp1->set_component_value("climateslider", round(((10*target_temp) - temp_offset) / temp_step)); set_component_visibility->execute("slider_high", false); set_component_visibility->execute("slider_low", false); set_component_visibility->execute("target_low", false); set_component_visibility->execute("target_high", true); set_component_visibility->execute("climateslider", true); } else { set_component_visibility->execute("climate.slider_high", false); if (target_temp_low > -999) { // Target temp low enabled disp1->set_component_value("active_slider", 2); snprintf(buffer, sizeof(buffer), (useDecimal) ? "%.1f°" : "%.0f°", target_temp_low); disp1->set_component_text("target_low", adjustDecimalSeparator(buffer, id(mui_decimal_separator)).c_str()); disp1->set_component_value("slider_low", round(((10*target_temp_low) - temp_offset) / temp_step)); set_component_visibility->execute("target_low", true); set_component_visibility->execute("slider_low", true); } else { set_component_visibility->execute("target_low", false); set_component_visibility->execute("slider_low", false); } if (target_temp_high > -999) { // Target temp high enabled disp1->set_component_value("active_slider", 1); snprintf(buffer, sizeof(buffer), (useDecimal) ? "%.1f°" : "%.0f°", target_temp_high); disp1->set_component_text("target_high", adjustDecimalSeparator(buffer, id(mui_decimal_separator)).c_str()); disp1->set_component_value("slider_high", round(((10*target_temp_high) - temp_offset) / temp_step)); set_component_visibility->execute("target_high", true); set_component_visibility->execute("slider_high", true); } else { set_component_visibility->execute("target_high", false); set_component_visibility->execute("slider_high", false); } } if (target_temp > -999 or target_temp_high > -999 or target_temp_low > -999) { disp1->set_component_text("target_icon", climate_icon.c_str()); set_component_visibility->execute("target_icon", true); set_component_visibility->execute("decrease_temp", true); set_component_visibility->execute("increase_temp", true); } else { set_component_visibility->execute("target_icon", false); set_component_visibility->execute("decrease_temp", false); set_component_visibility->execute("increase_temp", false); } disp1->set_component_value("embedded", (embedded_climate) ? 1 : 0); } - id: set_component_visibility mode: queued max_runs: 15 parameters: component_id: string show: bool then: - lambda: |- NextionComponent component = extractNextionComponent(component_id, current_page->state); if (component.is_current_page) disp1->send_command_printf("vis %s,%i", component.component_id, show ? 1 : 0); - id: stop_all mode: restart then: - lambda: |- boot_event->stop(); boot_progress->stop(); change_climate_state->stop(); check_versions->stop(); display_embedded_temp->stop(); display_wrapped_text->stop(); global_settings->stop(); ha_button->stop(); ha_call_service->stop(); init_hardware_climate->stop(); page_alarm->stop(); page_blank->stop(); page_boot->stop(); page_buttonpage01->stop(); page_buttonpage02->stop(); page_buttonpage03->stop(); page_buttonpage04->stop(); page_buttonpage->stop(); page_climate->stop(); page_changed->stop(); page_confirm->stop(); page_cover->stop(); page_entitypage01->stop(); page_entitypage02->stop(); page_entitypage03->stop(); page_entitypage04->stop(); page_entitypage->stop(); page_fan->stop(); page_home->stop(); page_keyb_num->stop(); page_light->stop(); page_media_player->stop(); page_notification->stop(); page_qrcode->stop(); page_screensaver->stop(); page_settings->stop(); page_utilities->stop(); page_weather01->stop(); page_weather02->stop(); page_weather03->stop(); page_weather04->stop(); page_weather05->stop(); page_weather->stop(); refresh_datetime->stop(); refresh_relays->stop(); refresh_wifi_icon->stop(); service_call_alarm_control_panel->stop(); set_baud_rate->stop(); set_brightness->stop(); set_climate->stop(); timer_dim->stop(); timer_page->stop(); timer_reset_all->stop(); timer_sleep->stop(); update_alarm_icon->stop(); update_climate_icon->stop(); update_tft_info->stop(); watchdog->stop(); ###### Timers ###### - id: timer_reset_all # Global timer reset - Triggered with a touch on the screen mode: restart then: - lambda: |- timer_page->execute(); timer_dim->execute(); timer_sleep->execute(); - id: timer_page # Handles the fallback to home page after a timeout mode: restart then: - if: condition: - lambda: |- return (timeout_page->state >= 1 and current_page->state != "boot" and current_page->state != "confirm" and current_page->state != "home" and current_page->state != "notification" and current_page->state != "screensaver"); then: - delay: !lambda return (int(timeout_page->state) *1000); - lambda: |- #if ESPHOME_LOG_LEVEL > ESPHOME_LOG_LEVEL_DEBUG ESP_LOGV("script.timer_page", "Timed out on page: %s", current_page->state.c_str()); #endif if (timeout_page->state >= 1 and current_page->state != "boot" and current_page->state != "confirm" and current_page->state != "home" and current_page->state != "notification" and current_page->state != "screensaver") { ESP_LOGD("script.timer_page", "Fallback to page Home"); goto_page->execute("home"); } - id: timer_dim # Handles the brightness dimming after a timeout mode: restart then: - lambda: |- if (current_brightness->state <= display_dim_brightness->state and current_page->state != "screensaver" and current_page->state != "boot") { ESP_LOGD("script.timer_dim", "Waking up on page: %s", current_page->state.c_str()); set_brightness->execute(display_brightness->state); } - if: condition: - lambda: return (timeout_dim->state >= 1); then: - delay: !lambda return (int(timeout_dim->state) *1000); - lambda: |- if (current_page->state != "screensaver" and current_page->state != "boot" and timeout_dim->state >= 1) { set_brightness->execute(display_dim_brightness->state); } - id: timer_sleep # Handles the sleep (go to screensaver page) after a timeout mode: restart then: - if: condition: - lambda: |- return (timeout_sleep->state >= 1 and current_page->state != "screensaver" and current_page->state != "boot"); then: - delay: !lambda return (int(timeout_sleep->state) *1000); - lambda: |- if (current_page->state != "screensaver" and current_page->state != "boot" and timeout_sleep->state >= 1) { ESP_LOGD("script.timer_sleep", "Going to sleep from page %s", current_page->state.c_str()); goto_page->execute("screensaver"); set_brightness->execute(display_sleep_brightness->state); } - id: update_alarm_icon # To do: Move to blueprint mode: restart parameters: component: string state: string then: - lambda: |- std::string alarm_icon = "\uEECC"; //mdi:shield-alert-outline int alarm_color = 65535; if (state == "disarmed") { alarm_icon = "\uE99B"; //mdi:shield-off-outline alarm_color = 65535; } else if (state == "armed_home") { alarm_icon = "\uECCA"; //mdi:shield-home-outline alarm_color = 19818; } else if (state == "armed_away") { alarm_icon = "\uECCB"; //mdi:shield-lock-outline alarm_color = 19818; } else if (state == "armed_night") { alarm_icon = "\uF828"; //mdi:shield-moon-outline alarm_color = 19818; } else if (state == "armed_vacation") { alarm_icon = "\uECC6"; //mdi:shield-airplane-outline alarm_color = 19818; } else if (state == "armed_custom_bypass") { alarm_icon = "\uE77F"; //mdi:shield-half-full alarm_color = 19818; } else if (state == "pending" or state == "arming") { alarm_icon = "\uE498"; //mdi:shield-outline alarm_color = 65024; } else if (state == "disarming") { alarm_icon = "\uE99B"; //mdi:shield-off-outline alarm_color = 65024; } else if (state == "triggered") { alarm_icon = "\uEECC"; //mdi:shield-alert-outline alarm_color = 63488; } disp1->set_component_text(component.c_str(), alarm_icon.c_str()); disp1->set_component_font_color(component.c_str(), alarm_color); - id: update_climate_icon mode: restart parameters: component: string action: uint mode: uint then: - lambda: |- switch (action) // CLIMATE_ACTION_OFF = 0, CLIMATE_ACTION_COOLING = 2, CLIMATE_ACTION_HEATING = 3, CLIMATE_ACTION_IDLE = 4, CLIMATE_ACTION_DRYING = 5, CLIMATE_ACTION_FAN = 6 { case 0: //CLIMATE_ACTION_OFF switch (mode) // CLIMATE_MODE_OFF = 0, CLIMATE_MODE_HEAT_COOL = 1, CLIMATE_MODE_COOL = 2, CLIMATE_MODE_HEAT = 3, CLIMATE_MODE_FAN_ONLY = 4, CLIMATE_MODE_DRY = 5, CLIMATE_MODE_AUTO = 6 { case 0: //CLIMATE_MODE_OFF disp1->set_component_text(component.c_str(), "\uFFFF"); // (E424) Don't show icon when off disp1->set_component_font_color(component.c_str(), 35921); // grey (off) break; case 1: //CLIMATE_MODE_HEAT_COOL disp1->set_component_text(component.c_str(), "\uE069"); // mdi:autorenew disp1->set_component_font_color(component.c_str(), 35921); // grey (off) break; case 2: //CLIMATE_MODE_COOL disp1->set_component_text(component.c_str(), "\uE716"); // mdi:snowflake disp1->set_component_font_color(component.c_str(), 35921); // grey (off) break; case 3: //CLIMATE_MODE_HEAT disp1->set_component_text(component.c_str(), "\uE237"); // mdi:fire disp1->set_component_font_color(component.c_str(), 35921); // grey (off) break; case 4: //CLIMATE_MODE_FAN_ONLY disp1->set_component_text(component.c_str(), "\uE20F"); // mdi:fan disp1->set_component_font_color(component.c_str(), 35921); // grey (off) break; case 5: //CLIMATE_MODE_DRY disp1->set_component_text(component.c_str(), "\uE58D"); // mdi:water-percent disp1->set_component_font_color(component.c_str(), 35921); // grey (off) break; case 6: //CLIMATE_MODE_AUTO disp1->set_component_text(component.c_str(), "\uEE8D"); // mdi:calendar-sync disp1->set_component_font_color(component.c_str(), 35921); // grey (off) break; } break; case 2: //CLIMATE_ACTION_COOLING disp1->set_component_text(component.c_str(), "\uE716"); // mdi:snowflake disp1->set_component_font_color(component.c_str(), 1055); // blue break; case 3: //CLIMATE_ACTION_HEATING disp1->set_component_text(component.c_str(), "\uE237"); // mdi:fire disp1->set_component_font_color(component.c_str(), 64164); // deep-orange break; case 4: //CLIMATE_ACTION_IDLE disp1->set_component_text(component.c_str(), "\uE50E"); // mdi:thermometer disp1->set_component_font_color(component.c_str(), 35921); // grey (off) break; case 5: //CLIMATE_ACTION_DRYING disp1->set_component_text(component.c_str(), "\uE58D"); // mdi:water-percent disp1->set_component_font_color(component.c_str(), 64704); // orange break; case 6: //CLIMATE_ACTION_FAN disp1->set_component_text(component.c_str(), "\uE20F"); // mdi:fan disp1->set_component_font_color(component.c_str(), 1530); // cyan break; } - id: update_tft_info mode: restart then: - while: condition: - lambda: return (isnan(display_charset->state) or isnan(display_mode->state) or version_tft->state.empty()); then: - lambda: |- ESP_LOGD("script.update_tft_info", "Updating TFT info"); if (isnan(display_charset->state)) display_charset->update(); if (isnan(display_mode->state)) display_mode->update(); if (version_tft->state.empty()) version_tft->update(); - wait_until: condition: - lambda: return (!isnan(display_charset->state) and !isnan(display_mode->state) and !(version_tft->state.empty())); timeout: 10s - id: watchdog mode: restart then: - lambda: |- #include static const char *const TAG = "script.watchdog"; if (id(is_uploading_tft)) { ESP_LOGW(TAG, "TFT upload in progress"); } else { // report Wi-Fi status bool wifi_connected = wifi_component->is_connected(); if (wifi_connected) { float rssi = wifi_rssi->state; const char *rssi_status = "Unknown"; // Use const char* to avoid dynamic memory allocation if (rssi > -50) rssi_status = "Excellent"; else if (rssi > -60) rssi_status = "Good"; else if (rssi > -70) rssi_status = "Fair"; else if (rssi > -80) rssi_status = "Weak"; else rssi_status = "Poor"; if (rssi > -70) ESP_LOGI(TAG, "Wi-Fi: %s (%.0f dBm)", rssi_status, rssi); else if (rssi > -80) ESP_LOGW(TAG, "Wi-Fi: %s (%.0f dBm)", rssi_status, rssi); else ESP_LOGE(TAG, "Wi-Fi: %s (%.0f dBm)", rssi_status, rssi); } else { ESP_LOGW(TAG, "Wi-Fi: DISCONNECTED"); ESP_LOGI(TAG, "Retrying Wi-Fi connection"); wifi_component->retry_connect(); } // report API status bool api_connected = api_server->is_connected(); if (api_connected) { ESP_LOGI(TAG, "API: Connected"); } else { ESP_LOGW(TAG, "API: DISCONNECTED"); blueprint_status->publish_state(0); if (current_page->state != "blank" and current_page->state != "boot" and current_page->state != "home" and current_page->state != "screensaver" and current_page->state != "settings" and current_page->state != "qrcode") { ESP_LOGI(TAG, "Fallback to page Home"); disp1->goto_page("home"); } } if (!wifi_connected or !api_connected) blueprint_status->publish_state(0); // Report blueprint version ESP_LOGI(TAG, "Blueprint:"); if (blueprint_status->state > 99) { ESP_LOGI(TAG, " Version: %s", version_blueprint->state.c_str()); ESP_LOGI(TAG, " Init steps: %i (%0.1f%%)", int(blueprint_status->raw_state), blueprint_status->state); } else { ESP_LOGW(TAG, " Init steps: %i (%0.1f%%)", int(blueprint_status->raw_state), blueprint_status->state); ESP_LOGW(TAG, " State: %s", (wifi_connected and api_connected) ? "Pending" : "DISCONNECTED"); ESP_LOGI(TAG, "Requesting blueprint settings"); boot_event->execute(false); } // Report ESPHome ESP_LOGI(TAG, "ESPHome:"); ESP_LOGI(TAG, " Version: ${version}"); ESP_LOGI(TAG, " Compiler: %s", ESPHOME_VERSION); // Report framework #ifdef ARDUINO ESP_LOGI(TAG, " Framework: Arduino"); #elif defined(USE_ESP_IDF) ESP_LOGI(TAG, " Framework: ESP-IDF"); #endif // Report memory const size_t internal_heap_size = heap_caps_get_total_size(MALLOC_CAP_INTERNAL); const size_t internal_heap_size_free = heap_caps_get_free_size(MALLOC_CAP_INTERNAL); const size_t psram_heap_size = heap_caps_get_total_size(MALLOC_CAP_SPIRAM); const size_t psram_heap_size_free = heap_caps_get_free_size(MALLOC_CAP_SPIRAM); ESP_LOGI(TAG, " Free heap:"); if (internal_heap_size != 0) { ESP_LOGI(TAG, " Internal: %7d bytes (%0.1f%%)", internal_heap_size_free, ((float)internal_heap_size_free / internal_heap_size) * 100.0f); } else { ESP_LOGI(TAG, " Internal: %7d bytes", internal_heap_size_free); } if (psram_heap_size != 0) { ESP_LOGI(TAG, " PSRAM: %7d bytes (%0.1f%%)", psram_heap_size_free, ((float)psram_heap_size_free / psram_heap_size) * 100.0f); } else { ESP_LOGI(TAG, " PSRAM: %7d bytes", psram_heap_size_free); } // Report UART ESP_LOGI(TAG, "UART:"); ESP_LOGI(TAG, " Baud rate: %" PRIu32 " bps", tf_uart->get_baud_rate()); ESP_LOGI(TAG, " Queue size: %d", tf_uart->available()); // Report Nextion status nextion_init->publish_state(nextion_init->state and disp1->is_setup()); ESP_LOGI(TAG, "Nextion:"); ESP_LOGI(TAG, " Queue size: %d", disp1->queue_size()); if (disp1->is_setup()) ESP_LOGI(TAG, " Is setup: True"); else { ESP_LOGW(TAG, " Is setup: False"); ESP_LOGW(TAG, " Is detected: %s", YESNO(disp1->is_detected())); //exit_reparse->execute(); } if (nextion_init->state) { ESP_LOGI(TAG, " Init: True"); } else ESP_LOGW(TAG, " Init: False"); if (version_tft->state.empty()) ESP_LOGW(TAG, " TFT: UNKNOWN"); else ESP_LOGI(TAG, " TFT: %s", version_tft->state.c_str()); ESP_LOGI(TAG, "Packages:"); #ifdef NSPANEL_HA_BLUEPRINT_CORE ESP_LOGI(TAG, " - Core"); #endif #ifdef NSPANEL_HA_BLUEPRINT_ADVANCED ESP_LOGI(TAG, " - Advanced"); #endif #ifdef NSPANEL_HA_BLUEPRINT_ADDON_UPLOAD_TFT ESP_LOGI(TAG, " - Upload TFT"); #endif #ifdef NSPANEL_HA_BLUEPRINT_ADDON_CLIMATE_BASE ESP_LOGI(TAG, " - Climate - Base"); #endif #ifdef NSPANEL_HA_BLUEPRINT_ADDON_CLIMATE_COOL ESP_LOGI(TAG, " - Climate - Cool"); #endif #ifdef NSPANEL_HA_BLUEPRINT_ADDON_CLIMATE_DUAL ESP_LOGI(TAG, " - Climate - Dual"); #endif #ifdef NSPANEL_HA_BLUEPRINT_ADDON_CLIMATE_HEAT ESP_LOGI(TAG, " - Climate - Heat"); #endif #if defined(USE_BLUETOOTH_PROXY) || defined(USE_WEBSERVER) || defined(USE_CAPTIVE_PORTAL) ESP_LOGI(TAG, "Non-standard components:"); #ifdef USE_CAPTIVE_PORTAL ESP_LOGI(TAG, " - Captive portal"); #endif #ifdef USE_BLUETOOTH_PROXY ESP_LOGI(TAG, " - Bluetooth proxy"); #endif #ifdef USE_WEBSERVER ESP_LOGI(TAG, " - Web server"); #endif #endif refresh_wifi_icon->execute(); } ...