core/homeassistant/components/deconz/cover.py

142 lines
4.5 KiB
Python

"""Support for deCONZ covers."""
from __future__ import annotations
from typing import Any, cast
from pydeconz.light import Cover
from homeassistant.components.cover import (
ATTR_POSITION,
ATTR_TILT_POSITION,
DOMAIN,
CoverDeviceClass,
CoverEntity,
CoverEntityFeature,
)
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 .deconz_device import DeconzDevice
from .gateway import DeconzGateway, get_gateway_from_config_entry
DEVICE_CLASS = {
"Level controllable output": CoverDeviceClass.DAMPER,
"Window covering controller": CoverDeviceClass.SHADE,
"Window covering device": CoverDeviceClass.SHADE,
}
async def async_setup_entry(
hass: HomeAssistant,
config_entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up covers for deCONZ component."""
gateway = get_gateway_from_config_entry(hass, config_entry)
gateway.entities[DOMAIN] = set()
@callback
def async_add_cover(lights: list[Cover] | None = None) -> None:
"""Add cover from deCONZ."""
entities = []
if lights is None:
lights = list(gateway.api.lights.covers.values())
for light in lights:
if (
isinstance(light, Cover)
and light.unique_id not in gateway.entities[DOMAIN]
):
entities.append(DeconzCover(light, gateway))
if entities:
async_add_entities(entities)
config_entry.async_on_unload(
async_dispatcher_connect(
hass,
gateway.signal_new_light,
async_add_cover,
)
)
async_add_cover()
class DeconzCover(DeconzDevice, CoverEntity):
"""Representation of a deCONZ cover."""
TYPE = DOMAIN
_device: Cover
def __init__(self, device: Cover, gateway: DeconzGateway) -> None:
"""Set up cover device."""
super().__init__(device, gateway)
self._attr_supported_features = CoverEntityFeature.OPEN
self._attr_supported_features |= CoverEntityFeature.CLOSE
self._attr_supported_features |= CoverEntityFeature.STOP
self._attr_supported_features |= CoverEntityFeature.SET_POSITION
if self._device.tilt is not None:
self._attr_supported_features |= CoverEntityFeature.OPEN_TILT
self._attr_supported_features |= CoverEntityFeature.CLOSE_TILT
self._attr_supported_features |= CoverEntityFeature.STOP_TILT
self._attr_supported_features |= CoverEntityFeature.SET_TILT_POSITION
self._attr_device_class = DEVICE_CLASS.get(self._device.type)
@property
def current_cover_position(self) -> int:
"""Return the current position of the cover."""
return 100 - self._device.lift
@property
def is_closed(self) -> bool:
"""Return if the cover is closed."""
return not self._device.is_open
async def async_set_cover_position(self, **kwargs: Any) -> None:
"""Move the cover to a specific position."""
position = 100 - cast(int, kwargs[ATTR_POSITION])
await self._device.set_position(lift=position)
async def async_open_cover(self, **kwargs: Any) -> None:
"""Open cover."""
await self._device.open()
async def async_close_cover(self, **kwargs: Any) -> None:
"""Close cover."""
await self._device.close()
async def async_stop_cover(self, **kwargs: Any) -> None:
"""Stop cover."""
await self._device.stop()
@property
def current_cover_tilt_position(self) -> int | None:
"""Return the current tilt position of the cover."""
if self._device.tilt is not None:
return 100 - self._device.tilt
return None
async def async_set_cover_tilt_position(self, **kwargs: Any) -> None:
"""Tilt the cover to a specific position."""
position = 100 - cast(int, kwargs[ATTR_TILT_POSITION])
await self._device.set_position(tilt=position)
async def async_open_cover_tilt(self, **kwargs: Any) -> None:
"""Open cover tilt."""
await self._device.set_position(tilt=0)
async def async_close_cover_tilt(self, **kwargs: Any) -> None:
"""Close cover tilt."""
await self._device.set_position(tilt=100)
async def async_stop_cover_tilt(self, **kwargs: Any) -> None:
"""Stop cover tilt."""
await self._device.stop()