"""Support for the Airzone sensors.""" from __future__ import annotations from collections.abc import Callable from dataclasses import dataclass from typing import Any, Final from aioairzone.common import GrilleAngle, OperationMode, SleepTimeout from aioairzone.const import ( API_COLD_ANGLE, API_HEAT_ANGLE, API_MODE, API_SLEEP, AZD_COLD_ANGLE, AZD_HEAT_ANGLE, AZD_MASTER, AZD_MODE, AZD_MODES, AZD_SLEEP, AZD_ZONES, ) from homeassistant.components.select import SelectEntity, SelectEntityDescription from homeassistant.config_entries import ConfigEntry from homeassistant.const import EntityCategory from homeassistant.core import HomeAssistant, callback from homeassistant.helpers.entity_platform import AddEntitiesCallback from . import AirzoneConfigEntry from .coordinator import AirzoneUpdateCoordinator from .entity import AirzoneEntity, AirzoneZoneEntity @dataclass(frozen=True, kw_only=True) class AirzoneSelectDescription(SelectEntityDescription): """Class to describe an Airzone select entity.""" api_param: str options_dict: dict[str, int] options_fn: Callable[[dict[str, Any], dict[str, int]], list[str]] = ( lambda zone_data, value: list(value) ) GRILLE_ANGLE_DICT: Final[dict[str, int]] = { "90deg": GrilleAngle.DEG_90, "50deg": GrilleAngle.DEG_50, "45deg": GrilleAngle.DEG_45, "40deg": GrilleAngle.DEG_40, } MODE_DICT: Final[dict[str, int]] = { "cool": OperationMode.COOLING, "dry": OperationMode.DRY, "fan": OperationMode.FAN, "heat": OperationMode.HEATING, "heat_cool": OperationMode.AUTO, "stop": OperationMode.STOP, } SLEEP_DICT: Final[dict[str, int]] = { "off": SleepTimeout.SLEEP_OFF, "30m": SleepTimeout.SLEEP_30, "60m": SleepTimeout.SLEEP_60, "90m": SleepTimeout.SLEEP_90, } def main_zone_options( zone_data: dict[str, Any], options: dict[str, int], ) -> list[str]: """Filter available modes.""" modes = zone_data.get(AZD_MODES, []) return [k for k, v in options.items() if v in modes] MAIN_ZONE_SELECT_TYPES: Final[tuple[AirzoneSelectDescription, ...]] = ( AirzoneSelectDescription( api_param=API_MODE, key=AZD_MODE, options_dict=MODE_DICT, options_fn=main_zone_options, translation_key="modes", ), ) ZONE_SELECT_TYPES: Final[tuple[AirzoneSelectDescription, ...]] = ( AirzoneSelectDescription( api_param=API_COLD_ANGLE, entity_category=EntityCategory.CONFIG, key=AZD_COLD_ANGLE, options=list(GRILLE_ANGLE_DICT), options_dict=GRILLE_ANGLE_DICT, translation_key="grille_angles", ), AirzoneSelectDescription( api_param=API_HEAT_ANGLE, entity_category=EntityCategory.CONFIG, key=AZD_HEAT_ANGLE, options=list(GRILLE_ANGLE_DICT), options_dict=GRILLE_ANGLE_DICT, translation_key="heat_angles", ), AirzoneSelectDescription( api_param=API_SLEEP, entity_category=EntityCategory.CONFIG, key=AZD_SLEEP, options=list(SLEEP_DICT), options_dict=SLEEP_DICT, translation_key="sleep_times", ), ) async def async_setup_entry( hass: HomeAssistant, entry: AirzoneConfigEntry, async_add_entities: AddEntitiesCallback, ) -> None: """Add Airzone select from a config_entry.""" coordinator = entry.runtime_data added_zones: set[str] = set() def _async_entity_listener() -> None: """Handle additions of select.""" zones_data = coordinator.data.get(AZD_ZONES, {}) received_zones = set(zones_data) new_zones = received_zones - added_zones if new_zones: entities: list[AirzoneZoneSelect] = [ AirzoneZoneSelect( coordinator, description, entry, system_zone_id, zones_data.get(system_zone_id), ) for system_zone_id in new_zones for description in MAIN_ZONE_SELECT_TYPES if description.key in zones_data.get(system_zone_id) and zones_data.get(system_zone_id).get(AZD_MASTER) is True ] entities += [ AirzoneZoneSelect( coordinator, description, entry, system_zone_id, zones_data.get(system_zone_id), ) for system_zone_id in new_zones for description in ZONE_SELECT_TYPES if description.key in zones_data.get(system_zone_id) ] async_add_entities(entities) added_zones.update(new_zones) entry.async_on_unload(coordinator.async_add_listener(_async_entity_listener)) _async_entity_listener() class AirzoneBaseSelect(AirzoneEntity, SelectEntity): """Define an Airzone select.""" entity_description: AirzoneSelectDescription values_dict: dict[int, str] @callback def _handle_coordinator_update(self) -> None: """Update attributes when the coordinator updates.""" self._async_update_attrs() super()._handle_coordinator_update() def _get_current_option(self) -> str | None: value = self.get_airzone_value(self.entity_description.key) return self.values_dict.get(value) @callback def _async_update_attrs(self) -> None: """Update select attributes.""" self._attr_current_option = self._get_current_option() class AirzoneZoneSelect(AirzoneZoneEntity, AirzoneBaseSelect): """Define an Airzone Zone select.""" def __init__( self, coordinator: AirzoneUpdateCoordinator, description: AirzoneSelectDescription, entry: ConfigEntry, system_zone_id: str, zone_data: dict[str, Any], ) -> None: """Initialize.""" super().__init__(coordinator, entry, system_zone_id, zone_data) self._attr_unique_id = ( f"{self._attr_unique_id}_{system_zone_id}_{description.key}" ) self.entity_description = description self._attr_options = self.entity_description.options_fn( zone_data, description.options_dict ) self.values_dict = {v: k for k, v in description.options_dict.items()} self._async_update_attrs() async def async_select_option(self, option: str) -> None: """Change the selected option.""" param = self.entity_description.api_param value = self.entity_description.options_dict[option] await self._async_update_hvac_params({param: value})