175 lines
6.0 KiB
Python
175 lines
6.0 KiB
Python
"""Support for Tuya (de)humidifiers."""
|
|
from __future__ import annotations
|
|
|
|
from dataclasses import dataclass
|
|
|
|
from tuya_iot import TuyaDevice, TuyaDeviceManager
|
|
|
|
from homeassistant.components.humidifier import (
|
|
DEVICE_CLASS_DEHUMIDIFIER,
|
|
DEVICE_CLASS_HUMIDIFIER,
|
|
SUPPORT_MODES,
|
|
HumidifierEntity,
|
|
HumidifierEntityDescription,
|
|
)
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.core import HomeAssistant, callback
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from . import HomeAssistantTuyaData
|
|
from .base import EnumTypeData, IntegerTypeData, TuyaEntity
|
|
from .const import DOMAIN, TUYA_DISCOVERY_NEW, DPCode
|
|
|
|
|
|
@dataclass
|
|
class TuyaHumidifierEntityDescription(HumidifierEntityDescription):
|
|
"""Describe an Tuya (de)humidifier entity."""
|
|
|
|
# DPCode, to use. If None, the key will be used as DPCode
|
|
dpcode: DPCode | tuple[DPCode, ...] | None = None
|
|
|
|
humidity: DPCode | None = None
|
|
|
|
|
|
HUMIDIFIERS: dict[str, TuyaHumidifierEntityDescription] = {
|
|
# Dehumidifier
|
|
# https://developer.tuya.com/en/docs/iot/categorycs?id=Kaiuz1vcz4dha
|
|
"cs": TuyaHumidifierEntityDescription(
|
|
key=DPCode.SWITCH,
|
|
dpcode=(DPCode.SWITCH, DPCode.SWITCH_SPRAY),
|
|
humidity=DPCode.DEHUMIDITY_SET_VALUE,
|
|
device_class=DEVICE_CLASS_DEHUMIDIFIER,
|
|
),
|
|
# Humidifier
|
|
# https://developer.tuya.com/en/docs/iot/categoryjsq?id=Kaiuz1smr440b
|
|
"jsq": TuyaHumidifierEntityDescription(
|
|
key=DPCode.SWITCH,
|
|
humidity=DPCode.HUMIDITY_SET,
|
|
device_class=DEVICE_CLASS_HUMIDIFIER,
|
|
),
|
|
}
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
|
) -> None:
|
|
"""Set up Tuya (de)humidifier dynamically through Tuya discovery."""
|
|
hass_data: HomeAssistantTuyaData = hass.data[DOMAIN][entry.entry_id]
|
|
|
|
@callback
|
|
def async_discover_device(device_ids: list[str]) -> None:
|
|
"""Discover and add a discovered Tuya (de)humidifier."""
|
|
entities: list[TuyaHumidifierEntity] = []
|
|
for device_id in device_ids:
|
|
device = hass_data.device_manager.device_map[device_id]
|
|
if description := HUMIDIFIERS.get(device.category):
|
|
entities.append(
|
|
TuyaHumidifierEntity(device, hass_data.device_manager, description)
|
|
)
|
|
async_add_entities(entities)
|
|
|
|
async_discover_device([*hass_data.device_manager.device_map])
|
|
|
|
entry.async_on_unload(
|
|
async_dispatcher_connect(hass, TUYA_DISCOVERY_NEW, async_discover_device)
|
|
)
|
|
|
|
|
|
class TuyaHumidifierEntity(TuyaEntity, HumidifierEntity):
|
|
"""Tuya (de)humidifier Device."""
|
|
|
|
_set_humidity_type: IntegerTypeData | None = None
|
|
_switch_dpcode: DPCode | None = None
|
|
entity_description: TuyaHumidifierEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
device: TuyaDevice,
|
|
device_manager: TuyaDeviceManager,
|
|
description: TuyaHumidifierEntityDescription,
|
|
) -> None:
|
|
"""Init Tuya (de)humidier."""
|
|
super().__init__(device, device_manager)
|
|
self.entity_description = description
|
|
self._attr_unique_id = f"{super().unique_id}{description.key}"
|
|
self._attr_supported_features = 0
|
|
|
|
# Determine main switch DPCode
|
|
possible_dpcodes = description.dpcode or description.key
|
|
if isinstance(possible_dpcodes, DPCode) and possible_dpcodes in device.function:
|
|
self._switch_dpcode = possible_dpcodes
|
|
elif isinstance(possible_dpcodes, tuple):
|
|
self._switch_dpcode = next(
|
|
(dpcode for dpcode in possible_dpcodes if dpcode in device.function),
|
|
None,
|
|
)
|
|
|
|
# Determine humidity parameters
|
|
if description.humidity in device.status_range:
|
|
type_data = IntegerTypeData.from_json(
|
|
device.status_range[description.humidity].values
|
|
)
|
|
self._set_humidity_type = type_data
|
|
self._attr_min_humidity = int(type_data.min_scaled)
|
|
self._attr_max_humidity = int(type_data.max_scaled)
|
|
|
|
# Determine mode support and provided modes
|
|
if DPCode.MODE in device.function:
|
|
self._attr_supported_features |= SUPPORT_MODES
|
|
self._attr_available_modes = EnumTypeData.from_json(
|
|
device.function[DPCode.MODE].values
|
|
).range
|
|
|
|
@property
|
|
def is_on(self) -> bool:
|
|
"""Return the device is on or off."""
|
|
if self._switch_dpcode is None:
|
|
return False
|
|
return self.device.status.get(self._switch_dpcode, False)
|
|
|
|
@property
|
|
def mode(self) -> str | None:
|
|
"""Return the current mode."""
|
|
return self.device.status.get(DPCode.MODE)
|
|
|
|
@property
|
|
def target_humidity(self) -> int | None:
|
|
"""Return the humidity we try to reach."""
|
|
if self._set_humidity_type is None:
|
|
return None
|
|
|
|
humidity = self.device.status.get(self.entity_description.humidity)
|
|
if humidity is None:
|
|
return None
|
|
|
|
return round(self._set_humidity_type.scale_value(humidity))
|
|
|
|
def turn_on(self, **kwargs):
|
|
"""Turn the device on."""
|
|
self._send_command([{"code": self._switch_dpcode, "value": True}])
|
|
|
|
def turn_off(self, **kwargs):
|
|
"""Turn the device off."""
|
|
self._send_command([{"code": self._switch_dpcode, "value": False}])
|
|
|
|
def set_humidity(self, humidity):
|
|
"""Set new target humidity."""
|
|
if self._set_humidity_type is None:
|
|
raise RuntimeError(
|
|
"Cannot set humidity, device doesn't provide methods to set it"
|
|
)
|
|
|
|
self._send_command(
|
|
[
|
|
{
|
|
"code": self.entity_description.humidity,
|
|
"value": self._set_humidity_type.scale_value(humidity),
|
|
}
|
|
]
|
|
)
|
|
|
|
def set_mode(self, mode):
|
|
"""Set new target preset mode."""
|
|
self._send_command([{"code": DPCode.MODE, "value": mode}])
|