2020-09-08 08:35:01 +00:00
|
|
|
"""Cover for Shelly."""
|
2021-07-21 17:11:44 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
from typing import Any, cast
|
|
|
|
|
2021-09-10 21:48:55 +00:00
|
|
|
from aioshelly.block_device import Block
|
2020-09-08 08:35:01 +00:00
|
|
|
|
|
|
|
from homeassistant.components.cover import (
|
|
|
|
ATTR_POSITION,
|
2021-12-16 13:12:23 +00:00
|
|
|
CoverDeviceClass,
|
2020-09-08 08:35:01 +00:00
|
|
|
CoverEntity,
|
2022-04-07 07:38:14 +00:00
|
|
|
CoverEntityFeature,
|
2020-09-08 08:35:01 +00:00
|
|
|
)
|
2021-07-21 17:11:44 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2020-09-08 08:35:01 +00:00
|
|
|
|
2022-03-08 09:58:08 +00:00
|
|
|
from . import BlockDeviceWrapper, RpcDeviceWrapper
|
|
|
|
from .const import BLOCK, DATA_CONFIG_ENTRY, DOMAIN, RPC
|
|
|
|
from .entity import ShellyBlockEntity, ShellyRpcEntity
|
|
|
|
from .utils import get_device_entry_gen, get_rpc_key_ids
|
2020-09-08 08:35:01 +00:00
|
|
|
|
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config_entry: ConfigEntry,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
2022-03-08 09:58:08 +00:00
|
|
|
) -> None:
|
|
|
|
"""Set up switches for device."""
|
|
|
|
if get_device_entry_gen(config_entry) == 2:
|
2022-05-17 17:57:41 +00:00
|
|
|
return async_setup_rpc_entry(hass, config_entry, async_add_entities)
|
2022-03-08 09:58:08 +00:00
|
|
|
|
2022-05-17 17:57:41 +00:00
|
|
|
return async_setup_block_entry(hass, config_entry, async_add_entities)
|
2022-03-08 09:58:08 +00:00
|
|
|
|
|
|
|
|
2022-05-17 17:57:41 +00:00
|
|
|
@callback
|
|
|
|
def async_setup_block_entry(
|
2022-03-08 09:58:08 +00:00
|
|
|
hass: HomeAssistant,
|
|
|
|
config_entry: ConfigEntry,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
2021-07-21 17:11:44 +00:00
|
|
|
) -> None:
|
2020-09-08 08:35:01 +00:00
|
|
|
"""Set up cover for device."""
|
2021-09-13 06:31:35 +00:00
|
|
|
wrapper = hass.data[DOMAIN][DATA_CONFIG_ENTRY][config_entry.entry_id][BLOCK]
|
2020-09-08 08:35:01 +00:00
|
|
|
blocks = [block for block in wrapper.device.blocks if block.type == "roller"]
|
|
|
|
|
|
|
|
if not blocks:
|
|
|
|
return
|
|
|
|
|
2022-03-08 09:58:08 +00:00
|
|
|
async_add_entities(BlockShellyCover(wrapper, block) for block in blocks)
|
2020-09-08 08:35:01 +00:00
|
|
|
|
|
|
|
|
2022-05-17 17:57:41 +00:00
|
|
|
@callback
|
|
|
|
def async_setup_rpc_entry(
|
2022-03-08 09:58:08 +00:00
|
|
|
hass: HomeAssistant,
|
|
|
|
config_entry: ConfigEntry,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
) -> None:
|
|
|
|
"""Set up entities for RPC device."""
|
|
|
|
wrapper = hass.data[DOMAIN][DATA_CONFIG_ENTRY][config_entry.entry_id][RPC]
|
|
|
|
|
|
|
|
cover_key_ids = get_rpc_key_ids(wrapper.device.status, "cover")
|
|
|
|
|
|
|
|
if not cover_key_ids:
|
|
|
|
return
|
|
|
|
|
|
|
|
async_add_entities(RpcShellyCover(wrapper, id_) for id_ in cover_key_ids)
|
|
|
|
|
|
|
|
|
|
|
|
class BlockShellyCover(ShellyBlockEntity, CoverEntity):
|
|
|
|
"""Entity that controls a cover on block based Shelly devices."""
|
2020-09-08 08:35:01 +00:00
|
|
|
|
2021-12-16 13:12:23 +00:00
|
|
|
_attr_device_class = CoverDeviceClass.SHUTTER
|
2021-05-31 08:50:11 +00:00
|
|
|
|
2021-09-13 06:31:35 +00:00
|
|
|
def __init__(self, wrapper: BlockDeviceWrapper, block: Block) -> None:
|
2022-03-08 09:58:08 +00:00
|
|
|
"""Initialize block cover."""
|
2020-09-08 08:35:01 +00:00
|
|
|
super().__init__(wrapper, block)
|
2021-07-21 17:11:44 +00:00
|
|
|
self.control_result: dict[str, Any] | None = None
|
2022-04-07 07:38:14 +00:00
|
|
|
self._attr_supported_features: int = (
|
|
|
|
CoverEntityFeature.OPEN | CoverEntityFeature.CLOSE | CoverEntityFeature.STOP
|
|
|
|
)
|
2020-09-08 08:35:01 +00:00
|
|
|
if self.wrapper.device.settings["rollers"][0]["positioning"]:
|
2022-04-07 07:38:14 +00:00
|
|
|
self._attr_supported_features |= CoverEntityFeature.SET_POSITION
|
2020-09-08 08:35:01 +00:00
|
|
|
|
|
|
|
@property
|
2021-07-21 17:11:44 +00:00
|
|
|
def is_closed(self) -> bool:
|
2020-09-08 08:35:01 +00:00
|
|
|
"""If cover is closed."""
|
|
|
|
if self.control_result:
|
2021-07-21 17:11:44 +00:00
|
|
|
return cast(bool, self.control_result["current_pos"] == 0)
|
2020-09-08 08:35:01 +00:00
|
|
|
|
2021-09-10 21:48:55 +00:00
|
|
|
return cast(int, self.block.rollerPos) == 0
|
2020-09-08 08:35:01 +00:00
|
|
|
|
|
|
|
@property
|
2021-07-21 17:11:44 +00:00
|
|
|
def current_cover_position(self) -> int:
|
2020-09-08 08:35:01 +00:00
|
|
|
"""Position of the cover."""
|
|
|
|
if self.control_result:
|
2021-07-21 17:11:44 +00:00
|
|
|
return cast(int, self.control_result["current_pos"])
|
2020-09-08 08:35:01 +00:00
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
return cast(int, self.block.rollerPos)
|
2020-09-08 08:35:01 +00:00
|
|
|
|
|
|
|
@property
|
2021-07-21 17:11:44 +00:00
|
|
|
def is_closing(self) -> bool:
|
2020-09-08 08:35:01 +00:00
|
|
|
"""Return if the cover is closing."""
|
|
|
|
if self.control_result:
|
2021-07-21 17:11:44 +00:00
|
|
|
return cast(bool, self.control_result["state"] == "close")
|
2020-09-08 08:35:01 +00:00
|
|
|
|
2021-09-10 21:48:55 +00:00
|
|
|
return self.block.roller == "close"
|
2020-09-08 08:35:01 +00:00
|
|
|
|
|
|
|
@property
|
2021-07-21 17:11:44 +00:00
|
|
|
def is_opening(self) -> bool:
|
2020-09-08 08:35:01 +00:00
|
|
|
"""Return if the cover is opening."""
|
|
|
|
if self.control_result:
|
2021-07-21 17:11:44 +00:00
|
|
|
return cast(bool, self.control_result["state"] == "open")
|
2020-09-08 08:35:01 +00:00
|
|
|
|
2021-09-10 21:48:55 +00:00
|
|
|
return self.block.roller == "open"
|
2020-09-08 08:35:01 +00:00
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
async def async_close_cover(self, **kwargs: Any) -> None:
|
2020-09-08 08:35:01 +00:00
|
|
|
"""Close cover."""
|
2021-05-04 16:10:28 +00:00
|
|
|
self.control_result = await self.set_state(go="close")
|
2020-09-08 08:35:01 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
async def async_open_cover(self, **kwargs: Any) -> None:
|
2020-09-08 08:35:01 +00:00
|
|
|
"""Open cover."""
|
2021-05-04 16:10:28 +00:00
|
|
|
self.control_result = await self.set_state(go="open")
|
2020-09-08 08:35:01 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
async def async_set_cover_position(self, **kwargs: Any) -> None:
|
2020-09-08 08:35:01 +00:00
|
|
|
"""Move the cover to a specific position."""
|
2021-05-04 16:10:28 +00:00
|
|
|
self.control_result = await self.set_state(
|
2020-09-08 08:35:01 +00:00
|
|
|
go="to_pos", roller_pos=kwargs[ATTR_POSITION]
|
|
|
|
)
|
|
|
|
self.async_write_ha_state()
|
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
async def async_stop_cover(self, **_kwargs: Any) -> None:
|
2020-09-08 08:35:01 +00:00
|
|
|
"""Stop the cover."""
|
2021-05-04 16:10:28 +00:00
|
|
|
self.control_result = await self.set_state(go="stop")
|
2020-09-08 08:35:01 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
|
|
|
|
@callback
|
2021-07-21 17:11:44 +00:00
|
|
|
def _update_callback(self) -> None:
|
2020-09-08 08:35:01 +00:00
|
|
|
"""When device updates, clear control result that overrides state."""
|
|
|
|
self.control_result = None
|
|
|
|
super()._update_callback()
|
2022-03-08 09:58:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
class RpcShellyCover(ShellyRpcEntity, CoverEntity):
|
|
|
|
"""Entity that controls a cover on RPC based Shelly devices."""
|
|
|
|
|
|
|
|
_attr_device_class = CoverDeviceClass.SHUTTER
|
|
|
|
|
|
|
|
def __init__(self, wrapper: RpcDeviceWrapper, id_: int) -> None:
|
|
|
|
"""Initialize rpc cover."""
|
|
|
|
super().__init__(wrapper, f"cover:{id_}")
|
|
|
|
self._id = id_
|
2022-04-07 07:38:14 +00:00
|
|
|
self._attr_supported_features: int = (
|
|
|
|
CoverEntityFeature.OPEN | CoverEntityFeature.CLOSE | CoverEntityFeature.STOP
|
|
|
|
)
|
2022-03-08 09:58:08 +00:00
|
|
|
if self.status["pos_control"]:
|
2022-04-07 07:38:14 +00:00
|
|
|
self._attr_supported_features |= CoverEntityFeature.SET_POSITION
|
2022-03-08 09:58:08 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_closed(self) -> bool | None:
|
|
|
|
"""If cover is closed."""
|
|
|
|
return cast(bool, self.status["state"] == "closed")
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_cover_position(self) -> int | None:
|
|
|
|
"""Position of the cover."""
|
|
|
|
if not self.status["pos_control"]:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return cast(int, self.status["current_pos"])
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_closing(self) -> bool:
|
|
|
|
"""Return if the cover is closing."""
|
|
|
|
return cast(bool, self.status["state"] == "closing")
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_opening(self) -> bool:
|
|
|
|
"""Return if the cover is opening."""
|
|
|
|
return cast(bool, self.status["state"] == "opening")
|
|
|
|
|
|
|
|
async def async_close_cover(self, **kwargs: Any) -> None:
|
|
|
|
"""Close cover."""
|
|
|
|
await self.call_rpc("Cover.Close", {"id": self._id})
|
|
|
|
|
|
|
|
async def async_open_cover(self, **kwargs: Any) -> None:
|
|
|
|
"""Open cover."""
|
|
|
|
await self.call_rpc("Cover.Open", {"id": self._id})
|
|
|
|
|
|
|
|
async def async_set_cover_position(self, **kwargs: Any) -> None:
|
|
|
|
"""Move the cover to a specific position."""
|
|
|
|
await self.call_rpc(
|
|
|
|
"Cover.GoToPosition", {"id": self._id, "pos": kwargs[ATTR_POSITION]}
|
|
|
|
)
|
|
|
|
|
|
|
|
async def async_stop_cover(self, **_kwargs: Any) -> None:
|
|
|
|
"""Stop the cover."""
|
|
|
|
await self.call_rpc("Cover.Stop", {"id": self._id})
|