2019-11-28 09:42:17 +00:00
|
|
|
"""Support for Somfy Covers."""
|
2020-10-26 22:15:39 +00:00
|
|
|
|
2019-11-06 18:55:56 +00:00
|
|
|
from pymfy.api.devices.blind import Blind
|
2019-12-09 11:07:10 +00:00
|
|
|
from pymfy.api.devices.category import Category
|
2019-06-11 15:45:34 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
from homeassistant.components.cover import (
|
|
|
|
ATTR_POSITION,
|
|
|
|
ATTR_TILT_POSITION,
|
2020-10-26 22:15:39 +00:00
|
|
|
DEVICE_CLASS_BLIND,
|
|
|
|
DEVICE_CLASS_SHUTTER,
|
2020-11-23 18:13:52 +00:00
|
|
|
SUPPORT_CLOSE,
|
|
|
|
SUPPORT_CLOSE_TILT,
|
|
|
|
SUPPORT_OPEN,
|
|
|
|
SUPPORT_OPEN_TILT,
|
|
|
|
SUPPORT_SET_POSITION,
|
|
|
|
SUPPORT_SET_TILT_POSITION,
|
|
|
|
SUPPORT_STOP,
|
|
|
|
SUPPORT_STOP_TILT,
|
2020-04-25 16:07:15 +00:00
|
|
|
CoverEntity,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2021-02-12 22:25:15 +00:00
|
|
|
from homeassistant.const import CONF_OPTIMISTIC, STATE_CLOSED, STATE_OPEN
|
2020-10-26 22:15:39 +00:00
|
|
|
from homeassistant.helpers.restore_state import RestoreEntity
|
2019-12-09 11:07:10 +00:00
|
|
|
|
2021-07-06 16:48:48 +00:00
|
|
|
from .const import COORDINATOR, DOMAIN
|
|
|
|
from .entity import SomfyEntity
|
2019-06-11 15:45:34 +00:00
|
|
|
|
2020-10-26 22:15:39 +00:00
|
|
|
BLIND_DEVICE_CATEGORIES = {Category.INTERIOR_BLIND.value, Category.EXTERIOR_BLIND.value}
|
|
|
|
SHUTTER_DEVICE_CATEGORIES = {Category.EXTERIOR_BLIND.value}
|
|
|
|
SUPPORTED_CATEGORIES = {
|
|
|
|
Category.ROLLER_SHUTTER.value,
|
|
|
|
Category.INTERIOR_BLIND.value,
|
|
|
|
Category.EXTERIOR_BLIND.value,
|
|
|
|
}
|
|
|
|
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
async def async_setup_entry(hass, config_entry, async_add_entities):
|
|
|
|
"""Set up the Somfy cover platform."""
|
2020-12-24 19:42:56 +00:00
|
|
|
domain_data = hass.data[DOMAIN]
|
|
|
|
coordinator = domain_data[COORDINATOR]
|
2019-06-11 15:45:34 +00:00
|
|
|
|
2020-12-24 19:42:56 +00:00
|
|
|
covers = [
|
2021-07-06 16:48:48 +00:00
|
|
|
SomfyCover(coordinator, device_id, domain_data[CONF_OPTIMISTIC])
|
2020-12-24 19:42:56 +00:00
|
|
|
for device_id, device in coordinator.data.items()
|
|
|
|
if SUPPORTED_CATEGORIES & set(device.categories)
|
|
|
|
]
|
2019-06-11 15:45:34 +00:00
|
|
|
|
2020-12-24 19:42:56 +00:00
|
|
|
async_add_entities(covers)
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
|
2020-10-26 22:15:39 +00:00
|
|
|
class SomfyCover(SomfyEntity, RestoreEntity, CoverEntity):
|
2019-06-11 15:45:34 +00:00
|
|
|
"""Representation of a Somfy cover device."""
|
|
|
|
|
2021-07-06 16:48:48 +00:00
|
|
|
def __init__(self, coordinator, device_id, optimistic):
|
2019-06-11 15:45:34 +00:00
|
|
|
"""Initialize the Somfy device."""
|
2021-07-06 16:48:48 +00:00
|
|
|
super().__init__(coordinator, device_id)
|
2020-10-27 04:42:12 +00:00
|
|
|
self.categories = set(self.device.categories)
|
2020-03-01 15:42:26 +00:00
|
|
|
self.optimistic = optimistic
|
|
|
|
self._closed = None
|
2020-10-26 22:15:39 +00:00
|
|
|
self._is_opening = None
|
|
|
|
self._is_closing = None
|
2020-12-15 15:04:35 +00:00
|
|
|
self._cover = None
|
2020-10-28 00:57:22 +00:00
|
|
|
self._create_device()
|
2019-06-11 15:45:34 +00:00
|
|
|
|
2020-11-23 18:13:52 +00:00
|
|
|
def _create_device(self) -> Blind:
|
2019-06-11 15:45:34 +00:00
|
|
|
"""Update the device with the latest data."""
|
2021-07-06 16:48:48 +00:00
|
|
|
self._cover = Blind(self.device, self.coordinator.client)
|
2019-06-11 15:45:34 +00:00
|
|
|
|
2020-11-23 18:13:52 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self) -> int:
|
|
|
|
"""Flag supported features."""
|
|
|
|
supported_features = 0
|
|
|
|
if self.has_capability("open"):
|
|
|
|
supported_features |= SUPPORT_OPEN
|
|
|
|
if self.has_capability("close"):
|
|
|
|
supported_features |= SUPPORT_CLOSE
|
|
|
|
if self.has_capability("stop"):
|
|
|
|
supported_features |= SUPPORT_STOP
|
|
|
|
if self.has_capability("position"):
|
|
|
|
supported_features |= SUPPORT_SET_POSITION
|
|
|
|
if self.has_capability("rotation"):
|
|
|
|
supported_features |= (
|
|
|
|
SUPPORT_OPEN_TILT
|
|
|
|
| SUPPORT_CLOSE_TILT
|
|
|
|
| SUPPORT_STOP_TILT
|
|
|
|
| SUPPORT_SET_TILT_POSITION
|
|
|
|
)
|
|
|
|
|
|
|
|
return supported_features
|
|
|
|
|
2020-10-27 13:08:32 +00:00
|
|
|
async def async_close_cover(self, **kwargs):
|
2019-06-11 15:45:34 +00:00
|
|
|
"""Close the cover."""
|
2020-10-26 22:15:39 +00:00
|
|
|
self._is_closing = True
|
2020-10-27 13:08:32 +00:00
|
|
|
self.async_write_ha_state()
|
2020-10-26 22:15:39 +00:00
|
|
|
try:
|
|
|
|
# Blocks until the close command is sent
|
2020-12-15 15:04:35 +00:00
|
|
|
await self.hass.async_add_executor_job(self._cover.close)
|
2020-03-01 15:42:26 +00:00
|
|
|
self._closed = True
|
2020-10-26 22:15:39 +00:00
|
|
|
finally:
|
|
|
|
self._is_closing = None
|
2020-10-27 13:08:32 +00:00
|
|
|
self.async_write_ha_state()
|
2019-06-11 15:45:34 +00:00
|
|
|
|
2020-10-27 13:08:32 +00:00
|
|
|
async def async_open_cover(self, **kwargs):
|
2019-06-11 15:45:34 +00:00
|
|
|
"""Open the cover."""
|
2020-10-26 22:15:39 +00:00
|
|
|
self._is_opening = True
|
2020-10-27 13:08:32 +00:00
|
|
|
self.async_write_ha_state()
|
2020-10-26 22:15:39 +00:00
|
|
|
try:
|
|
|
|
# Blocks until the open command is sent
|
2020-12-15 15:04:35 +00:00
|
|
|
await self.hass.async_add_executor_job(self._cover.open)
|
2020-03-01 15:42:26 +00:00
|
|
|
self._closed = False
|
2020-10-26 22:15:39 +00:00
|
|
|
finally:
|
|
|
|
self._is_opening = None
|
2020-10-27 13:08:32 +00:00
|
|
|
self.async_write_ha_state()
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
def stop_cover(self, **kwargs):
|
|
|
|
"""Stop the cover."""
|
2020-12-15 15:04:35 +00:00
|
|
|
self._cover.stop()
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
def set_cover_position(self, **kwargs):
|
|
|
|
"""Move the cover shutter to a specific position."""
|
2020-12-15 15:04:35 +00:00
|
|
|
self._cover.set_position(100 - kwargs[ATTR_POSITION])
|
2019-06-11 15:45:34 +00:00
|
|
|
|
2020-10-26 22:15:39 +00:00
|
|
|
@property
|
|
|
|
def device_class(self):
|
|
|
|
"""Return the device class."""
|
|
|
|
if self.categories & BLIND_DEVICE_CATEGORIES:
|
|
|
|
return DEVICE_CLASS_BLIND
|
|
|
|
if self.categories & SHUTTER_DEVICE_CATEGORIES:
|
|
|
|
return DEVICE_CLASS_SHUTTER
|
|
|
|
return None
|
|
|
|
|
2019-06-11 15:45:34 +00:00
|
|
|
@property
|
|
|
|
def current_cover_position(self):
|
|
|
|
"""Return the current position of cover shutter."""
|
2020-11-23 18:13:52 +00:00
|
|
|
if not self.has_state("position"):
|
|
|
|
return None
|
2020-12-15 15:04:35 +00:00
|
|
|
return 100 - self._cover.get_position()
|
2019-06-11 15:45:34 +00:00
|
|
|
|
2020-10-26 22:15:39 +00:00
|
|
|
@property
|
|
|
|
def is_opening(self):
|
|
|
|
"""Return if the cover is opening."""
|
|
|
|
if not self.optimistic:
|
|
|
|
return None
|
|
|
|
return self._is_opening
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_closing(self):
|
|
|
|
"""Return if the cover is closing."""
|
|
|
|
if not self.optimistic:
|
|
|
|
return None
|
|
|
|
return self._is_closing
|
|
|
|
|
2019-06-11 15:45:34 +00:00
|
|
|
@property
|
2020-11-23 18:13:52 +00:00
|
|
|
def is_closed(self) -> bool:
|
2019-06-11 15:45:34 +00:00
|
|
|
"""Return if the cover is closed."""
|
|
|
|
is_closed = None
|
2020-11-23 18:13:52 +00:00
|
|
|
if self.has_state("position"):
|
2020-12-15 15:04:35 +00:00
|
|
|
is_closed = self._cover.is_closed()
|
2020-03-01 15:42:26 +00:00
|
|
|
elif self.optimistic:
|
|
|
|
is_closed = self._closed
|
2019-06-11 15:45:34 +00:00
|
|
|
return is_closed
|
|
|
|
|
|
|
|
@property
|
2020-11-23 18:13:52 +00:00
|
|
|
def current_cover_tilt_position(self) -> int:
|
2019-06-11 15:45:34 +00:00
|
|
|
"""Return current position of cover tilt.
|
|
|
|
|
|
|
|
None is unknown, 0 is closed, 100 is fully open.
|
|
|
|
"""
|
2020-11-23 18:13:52 +00:00
|
|
|
if not self.has_state("orientation"):
|
|
|
|
return None
|
2020-12-15 15:04:35 +00:00
|
|
|
return 100 - self._cover.orientation
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
def set_cover_tilt_position(self, **kwargs):
|
|
|
|
"""Move the cover tilt to a specific position."""
|
2020-12-15 15:04:35 +00:00
|
|
|
self._cover.orientation = 100 - kwargs[ATTR_TILT_POSITION]
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
def open_cover_tilt(self, **kwargs):
|
|
|
|
"""Open the cover tilt."""
|
2020-12-15 15:04:35 +00:00
|
|
|
self._cover.orientation = 0
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
def close_cover_tilt(self, **kwargs):
|
|
|
|
"""Close the cover tilt."""
|
2020-12-15 15:04:35 +00:00
|
|
|
self._cover.orientation = 100
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
def stop_cover_tilt(self, **kwargs):
|
|
|
|
"""Stop the cover."""
|
2020-12-15 15:04:35 +00:00
|
|
|
self._cover.stop()
|
2020-10-26 22:15:39 +00:00
|
|
|
|
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""Complete the initialization."""
|
|
|
|
await super().async_added_to_hass()
|
2020-10-27 04:42:12 +00:00
|
|
|
if not self.optimistic:
|
|
|
|
return
|
|
|
|
# Restore the last state if we use optimistic
|
|
|
|
last_state = await self.async_get_last_state()
|
|
|
|
|
|
|
|
if last_state is not None and last_state.state in (
|
|
|
|
STATE_OPEN,
|
|
|
|
STATE_CLOSED,
|
|
|
|
):
|
|
|
|
self._closed = last_state.state == STATE_CLOSED
|