hOn/custom_components/hon/select.py

341 lines
12 KiB
Python
Raw Normal View History

2023-02-19 01:58:21 +00:00
from __future__ import annotations
2023-04-08 02:44:47 +00:00
import logging
2023-05-27 22:30:08 +00:00
from dataclasses import dataclass
2023-02-19 01:58:21 +00:00
from homeassistant.components.select import SelectEntity, SelectEntityDescription
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import UnitOfTemperature, UnitOfTime, REVOLUTIONS_PER_MINUTE
2023-02-19 01:58:21 +00:00
from homeassistant.core import callback
2023-06-08 18:01:55 +00:00
from homeassistant.helpers.entity import EntityCategory
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from homeassistant.helpers.typing import HomeAssistantType
2023-05-24 22:52:54 +00:00
from . import const
2023-03-05 20:16:09 +00:00
from .const import DOMAIN
2023-06-12 22:14:51 +00:00
from .hon import HonEntity, unique_entities, get_readable
2023-03-03 17:23:30 +00:00
2023-04-08 02:44:47 +00:00
_LOGGER = logging.getLogger(__name__)
2023-05-27 22:30:08 +00:00
2024-01-10 23:41:49 +00:00
@dataclass(frozen=True)
2023-05-27 22:30:08 +00:00
class HonSelectEntityDescription(SelectEntityDescription):
option_list: dict[int, str] | None = None
2023-12-03 16:13:45 +00:00
send_key_only: bool = False
2023-05-27 22:30:08 +00:00
2024-01-10 23:41:49 +00:00
@dataclass(frozen=True)
2023-05-27 22:30:08 +00:00
class HonConfigSelectEntityDescription(SelectEntityDescription):
entity_category: EntityCategory = EntityCategory.CONFIG
option_list: dict[int, str] | None = None
2023-05-27 22:30:08 +00:00
SELECTS: dict[str, tuple[SelectEntityDescription, ...]] = {
2023-02-19 01:58:21 +00:00
"WM": (
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-03-03 17:23:30 +00:00
key="startProgram.spinSpeed",
2023-02-19 01:58:21 +00:00
name="Spin speed",
2023-03-03 17:23:30 +00:00
icon="mdi:numeric",
2023-04-10 17:51:16 +00:00
unit_of_measurement=REVOLUTIONS_PER_MINUTE,
2023-04-23 00:01:14 +00:00
translation_key="spin_speed",
2023-02-19 01:58:21 +00:00
),
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-03-03 17:23:30 +00:00
key="startProgram.temp",
2023-02-19 01:58:21 +00:00
name="Temperature",
2023-03-03 17:23:30 +00:00
icon="mdi:thermometer",
2023-04-10 17:51:16 +00:00
unit_of_measurement=UnitOfTemperature.CELSIUS,
2023-04-23 00:01:14 +00:00
translation_key="temperature",
2023-02-19 01:58:21 +00:00
),
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-03-03 17:23:30 +00:00
key="startProgram.program",
2023-03-10 23:30:38 +00:00
name="Program",
2023-04-22 19:19:32 +00:00
translation_key="programs_wm",
2023-02-19 01:58:21 +00:00
),
2023-07-09 22:20:35 +00:00
HonConfigSelectEntityDescription(
key="startProgram.steamLevel",
name="Steam level",
icon="mdi:weather-dust",
translation_key="steam_level",
option_list=const.STEAM_LEVEL,
),
2023-07-09 22:48:57 +00:00
HonConfigSelectEntityDescription(
key="startProgram.dirtyLevel",
name="Dirty level",
icon="mdi:liquid-spot",
translation_key="dirt_level",
option_list=const.DIRTY_LEVEL,
),
2023-10-02 01:33:45 +00:00
HonConfigSelectEntityDescription(
key="startProgram.extendedStainType",
name="Stain Type",
icon="mdi:liquid-spot",
translation_key="stain_type",
),
2023-03-21 12:00:50 +00:00
),
"TD": (
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-03-21 12:00:50 +00:00
key="startProgram.program",
name="Program",
2023-04-22 19:19:32 +00:00
translation_key="programs_td",
2023-03-21 12:00:50 +00:00
),
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
key="startProgram.dryTimeMM",
name="Dry Time",
icon="mdi:timer",
2023-04-10 17:51:16 +00:00
unit_of_measurement=UnitOfTime.MINUTES,
translation_key="dry_time",
),
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-04-22 21:09:57 +00:00
key="startProgram.dryLevel",
name="Dry level",
icon="mdi:hair-dryer",
translation_key="dry_levels",
option_list=const.TUMBLE_DRYER_DRY_LEVEL,
2023-04-22 21:09:57 +00:00
),
2023-04-07 11:52:55 +00:00
),
"OV": (
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-04-07 11:52:55 +00:00
key="startProgram.program",
name="Program",
2023-04-22 19:19:32 +00:00
translation_key="programs_ov",
2023-04-07 11:52:55 +00:00
),
2023-04-08 02:44:47 +00:00
),
2023-04-16 11:55:08 +00:00
"IH": (
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-04-15 02:27:40 +00:00
key="startProgram.program",
name="Program",
2023-04-22 19:19:32 +00:00
translation_key="programs_ih",
2023-04-15 02:27:40 +00:00
),
),
2023-04-16 19:46:17 +00:00
"DW": (
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-04-16 19:46:17 +00:00
key="startProgram.program",
name="Program",
translation_key="programs_dw",
),
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-05-14 01:17:58 +00:00
key="startProgram.temp",
name="Temperature",
icon="mdi:thermometer",
unit_of_measurement=UnitOfTemperature.CELSIUS,
translation_key="temperature",
),
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-05-14 01:17:58 +00:00
key="startProgram.remainingTime",
name="Remaining Time",
icon="mdi:timer",
unit_of_measurement=UnitOfTime.MINUTES,
translation_key="remaining_time",
),
2023-04-16 19:46:17 +00:00
),
2023-04-26 21:57:44 +00:00
"AC": (
2023-05-27 22:30:08 +00:00
HonSelectEntityDescription(
2023-04-26 21:57:44 +00:00
key="startProgram.program",
name="Program",
translation_key="programs_ac",
),
2023-05-27 22:30:08 +00:00
HonSelectEntityDescription(
2023-05-08 00:05:04 +00:00
key="settings.humanSensingStatus",
2023-04-26 21:57:44 +00:00
name="Eco Pilot",
2023-05-08 00:05:04 +00:00
icon="mdi:run",
2023-04-26 21:57:44 +00:00
translation_key="eco_pilot",
option_list=const.AC_HUMAN_SENSE,
2023-04-26 21:57:44 +00:00
),
HonSelectEntityDescription(
key="settings.windDirectionHorizontal",
name="Fan Direction Horizontal",
icon="mdi:fan",
translation_key="fan_horizontal",
option_list=const.AC_POSITION_HORIZONTAL,
),
HonSelectEntityDescription(
key="settings.windDirectionVertical",
name="Fan Direction Vertical",
icon="mdi:fan",
translation_key="fan_vertical",
option_list=const.AC_POSITION_VERTICAL,
),
2023-04-26 21:57:44 +00:00
),
2023-05-10 16:13:05 +00:00
"REF": (
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-05-10 16:13:05 +00:00
key="startProgram.program",
name="Program",
translation_key="programs_ref",
),
2023-05-27 22:30:08 +00:00
HonConfigSelectEntityDescription(
2023-05-21 18:51:20 +00:00
key="startProgram.zone",
name="Zone",
icon="mdi:radiobox-marked",
translation_key="ref_zones",
),
2023-05-10 16:13:05 +00:00
),
2023-06-20 22:59:00 +00:00
"AP": (
HonSelectEntityDescription(
key="settings.aromaStatus",
name="Diffuser Level",
option_list=const.AP_DIFFUSER_LEVEL,
2023-06-30 23:45:32 +00:00
translation_key="diffuser",
icon="mdi:air-purifier",
2023-06-20 22:59:00 +00:00
),
HonSelectEntityDescription(
key="settings.machMode",
name="Mode",
2023-07-09 22:19:43 +00:00
icon="mdi:play",
2023-06-20 22:59:00 +00:00
option_list=const.AP_MACH_MODE,
2023-06-30 23:45:32 +00:00
translation_key="mode",
2023-06-20 22:59:00 +00:00
),
),
2023-12-03 16:13:45 +00:00
"WH": (
HonSelectEntityDescription(
key="settings.tempSel",
name="Target Temperature",
icon="mdi:thermometer",
unit_of_measurement=UnitOfTemperature.CELSIUS,
translation_key="target_temperature",
send_key_only=True,
),
HonSelectEntityDescription(
key="settings.machMode",
name="Mode",
send_key_only=True,
icon="mdi:information",
option_list=const.WH_MACH_MODE,
translation_key="mach_modes_wh",
),
),
2023-02-19 01:58:21 +00:00
}
2023-05-07 14:39:45 +00:00
SELECTS["WD"] = unique_entities(SELECTS["WM"], SELECTS["TD"])
2023-02-19 01:58:21 +00:00
async def async_setup_entry(
hass: HomeAssistantType, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
) -> None:
2023-05-24 23:30:33 +00:00
entities = []
entity: HonSelectEntity | HonConfigSelectEntity
2023-05-24 23:30:33 +00:00
for device in hass.data[DOMAIN][entry.unique_id].appliances:
for description in SELECTS.get(device.appliance_type, []):
if description.key not in device.available_settings:
continue
2023-05-27 22:30:08 +00:00
if isinstance(description, HonSelectEntityDescription):
entity = HonSelectEntity(hass, entry, device, description)
elif isinstance(description, HonConfigSelectEntityDescription):
entity = HonConfigSelectEntity(hass, entry, device, description)
else:
continue
2023-05-24 23:30:33 +00:00
await entity.coordinator.async_config_entry_first_refresh()
entities.append(entity)
async_add_entities(entities)
2023-02-19 01:58:21 +00:00
2023-06-20 22:23:49 +00:00
class HonConfigSelectEntity(HonEntity, SelectEntity):
entity_description: HonConfigSelectEntityDescription
2023-02-19 01:58:21 +00:00
@property
def current_option(self) -> str | None:
2023-06-15 21:52:56 +00:00
if not (setting := self._device.settings.get(self.entity_description.key)):
2023-02-19 01:58:21 +00:00
return None
2023-06-22 11:18:45 +00:00
value = get_readable(self.entity_description, setting.value)
2023-06-15 21:52:56 +00:00
if value not in self._attr_options:
return None
return str(value)
2023-02-19 01:58:21 +00:00
2023-06-20 22:23:49 +00:00
@property
def options(self) -> list[str]:
setting = self._device.settings.get(self.entity_description.key)
if setting is None:
return []
return [
str(get_readable(self.entity_description, key)) for key in setting.values
]
2023-06-20 22:23:49 +00:00
def _option_to_number(self, option: str, values: list[str]) -> str:
2023-06-15 21:52:56 +00:00
if (options := self.entity_description.option_list) is not None:
2023-07-09 22:21:45 +00:00
return str(
next(
(k for k, v in options.items() if str(k) in values and v == option),
option,
)
2023-06-15 21:52:56 +00:00
)
2023-06-20 22:23:49 +00:00
return option
2023-02-19 01:58:21 +00:00
async def async_select_option(self, option: str) -> None:
2023-06-20 22:23:49 +00:00
setting = self._device.settings[self.entity_description.key]
setting.value = self._option_to_number(option, setting.values)
2023-04-23 18:04:19 +00:00
await self.coordinator.async_refresh()
2023-02-19 01:58:21 +00:00
@callback
def _handle_coordinator_update(self, update: bool = True) -> None:
2023-06-20 22:23:49 +00:00
self._attr_available = self.available
self._attr_options = self.options
self._attr_current_option = self.current_option
2023-06-08 18:01:55 +00:00
if update:
self.async_write_ha_state()
@property
def available(self) -> bool:
"""Return True if entity is available."""
2023-06-20 22:23:49 +00:00
return self._device.settings.get(self.entity_description.key) is not None
2023-05-27 22:30:08 +00:00
class HonSelectEntity(HonEntity, SelectEntity):
2023-06-20 22:23:49 +00:00
entity_description: HonSelectEntityDescription
2023-05-27 22:30:08 +00:00
@property
def current_option(self) -> str | None:
if not (setting := self._device.settings.get(self.entity_description.key)):
return None
value = get_readable(self.entity_description, setting.value)
if value not in self._attr_options:
return None
return str(value)
@property
def options(self) -> list[str]:
setting = self._device.settings.get(self.entity_description.key)
if setting is None:
return []
return [
str(get_readable(self.entity_description, key)) for key in setting.values
]
def _option_to_number(self, option: str, values: list[str]) -> str:
if (options := self.entity_description.option_list) is not None:
return str(
next(
(k for k, v in options.items() if str(k) in values and v == option),
option,
)
)
return option
2023-05-27 22:30:08 +00:00
async def async_select_option(self, option: str) -> None:
2023-06-15 21:52:56 +00:00
setting = self._device.settings[self.entity_description.key]
2023-06-20 22:23:49 +00:00
setting.value = self._option_to_number(option, setting.values)
2023-12-03 16:13:45 +00:00
key_parts = self.entity_description.key.split(".")
command = key_parts[0]
2023-12-03 18:57:27 +00:00
if self.entity_description.send_key_only:
2023-12-03 18:31:46 +00:00
await self._device.commands[command].send_specific([key_parts[1]])
2023-12-03 16:13:45 +00:00
else:
await self._device.commands[command].send()
2023-06-21 17:52:32 +00:00
if command != "settings":
self._device.sync_command(command, "settings")
2023-05-27 22:30:08 +00:00
await self.coordinator.async_refresh()
@property
def available(self) -> bool:
"""Return True if entity is available."""
2023-06-20 22:23:49 +00:00
return (
super().available
2023-06-22 11:18:45 +00:00
and int(self._device.get("remoteCtrValid", 1)) == 1
2023-06-20 22:23:49 +00:00
and self._device.get("attributes.lastConnEvent.category") != "DISCONNECTED"
)
@callback
def _handle_coordinator_update(self, update: bool = True) -> None:
self._attr_available = self.available
self._attr_options = self.options
self._attr_current_option = self.current_option
if update:
self.async_write_ha_state()