mirror of
				https://github.com/home-assistant/core.git
				synced 2025-11-04 00:19:31 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			238 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			238 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
"""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, QAdapt, SleepTimeout
 | 
						|
from aioairzone.const import (
 | 
						|
    API_COLD_ANGLE,
 | 
						|
    API_HEAT_ANGLE,
 | 
						|
    API_MODE,
 | 
						|
    API_Q_ADAPT,
 | 
						|
    API_SLEEP,
 | 
						|
    AZD_COLD_ANGLE,
 | 
						|
    AZD_HEAT_ANGLE,
 | 
						|
    AZD_MASTER,
 | 
						|
    AZD_MODE,
 | 
						|
    AZD_MODES,
 | 
						|
    AZD_Q_ADAPT,
 | 
						|
    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 AddConfigEntryEntitiesCallback
 | 
						|
 | 
						|
from .coordinator import AirzoneConfigEntry, 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,
 | 
						|
}
 | 
						|
 | 
						|
Q_ADAPT_DICT: Final[dict[str, int]] = {
 | 
						|
    "standard": QAdapt.STANDARD,
 | 
						|
    "power": QAdapt.POWER,
 | 
						|
    "silence": QAdapt.SILENCE,
 | 
						|
    "minimum": QAdapt.MINIMUM,
 | 
						|
    "maximum": QAdapt.MAXIMUM,
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
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",
 | 
						|
    ),
 | 
						|
    AirzoneSelectDescription(
 | 
						|
        api_param=API_Q_ADAPT,
 | 
						|
        entity_category=EntityCategory.CONFIG,
 | 
						|
        key=AZD_Q_ADAPT,
 | 
						|
        options=list(Q_ADAPT_DICT),
 | 
						|
        options_dict=Q_ADAPT_DICT,
 | 
						|
        translation_key="q_adapt",
 | 
						|
    ),
 | 
						|
)
 | 
						|
 | 
						|
 | 
						|
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: AddConfigEntryEntitiesCallback,
 | 
						|
) -> 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})
 |