import esphome.codegen as cg import esphome.config_validation as cv from esphome.components import climate from esphome.components.logger import HARDWARE_UART_TO_SERIAL from esphome.const import ( CONF_ID, CONF_HARDWARE_UART, CONF_BAUD_RATE, CONF_RX_PIN, CONF_TX_PIN, CONF_UPDATE_INTERVAL, CONF_MODE, CONF_FAN_MODE, CONF_SWING_MODE, ) from esphome.core import CORE, coroutine AUTO_LOAD = ["climate"] CONF_SUPPORTS = "supports" DEFAULT_CLIMATE_MODES = ["HEAT_COOL", "COOL", "HEAT", "DRY", "FAN_ONLY"] DEFAULT_FAN_MODES = ["AUTO", "DIFFUSE", "LOW", "MEDIUM", "MIDDLE", "HIGH"] DEFAULT_SWING_MODES = ["OFF", "VERTICAL"] # Remote temperature timeout configuration CONF_REMOTE_OPERATING_TIMEOUT = "remote_temperature_operating_timeout_minutes" CONF_REMOTE_IDLE_TIMEOUT = "remote_temperature_idle_timeout_minutes" CONF_REMOTE_PING_TIMEOUT = "remote_temperature_ping_timeout_minutes" MitsubishiHeatPump = cg.global_ns.class_( "MitsubishiHeatPump", climate.Climate, cg.PollingComponent ) def valid_uart(uart): if CORE.is_esp8266: uarts = ["UART0"] # UART1 is tx-only elif CORE.is_esp32: uarts = ["UART0", "UART1", "UART2"] else: raise NotImplementedError return cv.one_of(*uarts, upper=True)(uart) CONFIG_SCHEMA = climate.CLIMATE_SCHEMA.extend( { cv.GenerateID(): cv.declare_id(MitsubishiHeatPump), cv.Optional(CONF_HARDWARE_UART, default="UART0"): valid_uart, cv.Optional(CONF_BAUD_RATE): cv.positive_int, cv.Optional(CONF_REMOTE_OPERATING_TIMEOUT): cv.positive_int, cv.Optional(CONF_REMOTE_IDLE_TIMEOUT): cv.positive_int, cv.Optional(CONF_REMOTE_PING_TIMEOUT): cv.positive_int, cv.Optional(CONF_RX_PIN): cv.positive_int, cv.Optional(CONF_TX_PIN): cv.positive_int, # If polling interval is greater than 9 seconds, the HeatPump library # reconnects, but doesn't then follow up with our data request. cv.Optional(CONF_UPDATE_INTERVAL, default="500ms"): cv.All( cv.update_interval, cv.Range(max=cv.TimePeriod(milliseconds=9000)) ), # Optionally override the supported ClimateTraits. cv.Optional(CONF_SUPPORTS, default={}): cv.Schema( { cv.Optional(CONF_MODE, default=DEFAULT_CLIMATE_MODES): cv.ensure_list(climate.validate_climate_mode), cv.Optional(CONF_FAN_MODE, default=DEFAULT_FAN_MODES): cv.ensure_list(climate.validate_climate_fan_mode), cv.Optional(CONF_SWING_MODE, default=DEFAULT_SWING_MODES): cv.ensure_list(climate.validate_climate_swing_mode), } ), } ).extend(cv.COMPONENT_SCHEMA) @coroutine def to_code(config): serial = HARDWARE_UART_TO_SERIAL[config[CONF_HARDWARE_UART]] var = cg.new_Pvariable(config[CONF_ID], cg.RawExpression(f"&{serial}")) if CONF_BAUD_RATE in config: cg.add(var.set_baud_rate(config[CONF_BAUD_RATE])) if CONF_RX_PIN in config: cg.add(var.set_rx_pin(config[CONF_RX_PIN])) if CONF_TX_PIN in config: cg.add(var.set_tx_pin(config[CONF_TX_PIN])) if CONF_REMOTE_OPERATING_TIMEOUT in config: cg.add(var.set_remote_operating_timeout_minutes(config[CONF_REMOTE_OPERATING_TIMEOUT])) if CONF_REMOTE_IDLE_TIMEOUT in config: cg.add(var.set_remote_idle_timeout_minutes(config[CONF_REMOTE_IDLE_TIMEOUT])) if CONF_REMOTE_PING_TIMEOUT in config: cg.add(var.set_remote_ping_timeout_minutes(config[CONF_REMOTE_PING_TIMEOUT])) supports = config[CONF_SUPPORTS] traits = var.config_traits() for mode in supports[CONF_MODE]: if mode == "OFF": continue cg.add(traits.add_supported_mode(climate.CLIMATE_MODES[mode])) for mode in supports[CONF_FAN_MODE]: cg.add(traits.add_supported_fan_mode(climate.CLIMATE_FAN_MODES[mode])) for mode in supports[CONF_SWING_MODE]: cg.add(traits.add_supported_swing_mode( climate.CLIMATE_SWING_MODES[mode] )) yield cg.register_component(var, config) yield climate.register_climate(var, config) cg.add_library( name="HeatPump", repository="https://github.com/SwiCago/HeatPump", version="d6a29134401d7caae1b8fca9c452c8eb92af60c5", )