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-04-25 16:07:15 +00:00
|
|
|
CoverEntity,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2020-10-26 22:15:39 +00:00
|
|
|
from homeassistant.const import STATE_CLOSED, STATE_OPEN
|
|
|
|
from homeassistant.helpers.restore_state import RestoreEntity
|
2019-12-09 11:07:10 +00:00
|
|
|
|
2020-10-27 04:42:12 +00:00
|
|
|
from . import SomfyEntity
|
|
|
|
from .const import API, CONF_OPTIMISTIC, COORDINATOR, DOMAIN
|
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."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-06-11 15:45:34 +00:00
|
|
|
def get_covers():
|
|
|
|
"""Retrieve covers."""
|
2020-10-27 04:42:12 +00:00
|
|
|
domain_data = hass.data[DOMAIN]
|
|
|
|
coordinator = domain_data[COORDINATOR]
|
|
|
|
api = domain_data[API]
|
2019-06-11 15:45:34 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
return [
|
2020-10-27 04:42:12 +00:00
|
|
|
SomfyCover(coordinator, device_id, api, domain_data[CONF_OPTIMISTIC])
|
|
|
|
for device_id, device in coordinator.data.items()
|
|
|
|
if SUPPORTED_CATEGORIES & set(device.categories)
|
2019-07-31 19:25:30 +00:00
|
|
|
]
|
2019-06-11 15:45:34 +00:00
|
|
|
|
2020-10-26 22:15:39 +00:00
|
|
|
async_add_entities(await hass.async_add_executor_job(get_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."""
|
|
|
|
|
2020-10-27 04:42:12 +00:00
|
|
|
def __init__(self, coordinator, device_id, api, optimistic):
|
2019-06-11 15:45:34 +00:00
|
|
|
"""Initialize the Somfy device."""
|
2020-10-27 04:42:12 +00:00
|
|
|
super().__init__(coordinator, device_id, api)
|
2019-06-11 15:45:34 +00:00
|
|
|
self.cover = Blind(self.device, self.api)
|
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
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
async def async_update(self):
|
|
|
|
"""Update the device with the latest data."""
|
|
|
|
await super().async_update()
|
|
|
|
self.cover = Blind(self.device, self.api)
|
|
|
|
|
|
|
|
def close_cover(self, **kwargs):
|
|
|
|
"""Close the cover."""
|
2020-10-26 22:15:39 +00:00
|
|
|
self._is_closing = True
|
|
|
|
self.schedule_update_ha_state()
|
|
|
|
try:
|
|
|
|
# Blocks until the close command is sent
|
|
|
|
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
|
|
|
|
self.schedule_update_ha_state()
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
def open_cover(self, **kwargs):
|
|
|
|
"""Open the cover."""
|
2020-10-26 22:15:39 +00:00
|
|
|
self._is_opening = True
|
|
|
|
self.schedule_update_ha_state()
|
|
|
|
try:
|
|
|
|
# Blocks until the open command is sent
|
|
|
|
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
|
|
|
|
self.schedule_update_ha_state()
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
def stop_cover(self, **kwargs):
|
|
|
|
"""Stop the cover."""
|
|
|
|
self.cover.stop()
|
|
|
|
|
|
|
|
def set_cover_position(self, **kwargs):
|
|
|
|
"""Move the cover shutter to a specific position."""
|
|
|
|
self.cover.set_position(100 - kwargs[ATTR_POSITION])
|
|
|
|
|
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."""
|
|
|
|
position = None
|
2019-07-31 19:25:30 +00:00
|
|
|
if self.has_capability("position"):
|
2019-06-11 15:45:34 +00:00
|
|
|
position = 100 - self.cover.get_position()
|
|
|
|
return position
|
|
|
|
|
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
|
|
|
|
def is_closed(self):
|
|
|
|
"""Return if the cover is closed."""
|
|
|
|
is_closed = None
|
2019-07-31 19:25:30 +00:00
|
|
|
if self.has_capability("position"):
|
2019-06-11 15:45:34 +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
|
|
|
|
def current_cover_tilt_position(self):
|
|
|
|
"""Return current position of cover tilt.
|
|
|
|
|
|
|
|
None is unknown, 0 is closed, 100 is fully open.
|
|
|
|
"""
|
|
|
|
orientation = None
|
2019-07-31 19:25:30 +00:00
|
|
|
if self.has_capability("rotation"):
|
2019-06-11 15:45:34 +00:00
|
|
|
orientation = 100 - self.cover.orientation
|
|
|
|
return orientation
|
|
|
|
|
|
|
|
def set_cover_tilt_position(self, **kwargs):
|
|
|
|
"""Move the cover tilt to a specific position."""
|
2020-01-03 10:31:05 +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-01-03 10:31:05 +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-01-03 10:31:05 +00:00
|
|
|
self.cover.orientation = 100
|
2019-06-11 15:45:34 +00:00
|
|
|
|
|
|
|
def stop_cover_tilt(self, **kwargs):
|
|
|
|
"""Stop the cover."""
|
|
|
|
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
|