2016-08-24 01:23:18 +00:00
|
|
|
"""
|
|
|
|
Support for Cover devices.
|
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/cover/
|
|
|
|
"""
|
2017-01-05 23:16:12 +00:00
|
|
|
from datetime import timedelta
|
2017-02-02 20:39:13 +00:00
|
|
|
import functools as ft
|
2016-08-24 01:23:18 +00:00
|
|
|
import logging
|
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
2017-07-16 17:14:46 +00:00
|
|
|
from homeassistant.loader import bind_hass
|
2016-08-24 01:23:18 +00:00
|
|
|
from homeassistant.helpers.entity_component import EntityComponent
|
|
|
|
from homeassistant.helpers.entity import Entity
|
|
|
|
from homeassistant.helpers.config_validation import PLATFORM_SCHEMA # noqa
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
from homeassistant.components import group
|
2018-03-31 00:22:48 +00:00
|
|
|
from homeassistant.helpers import intent
|
2016-08-24 01:23:18 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
SERVICE_OPEN_COVER, SERVICE_CLOSE_COVER, SERVICE_SET_COVER_POSITION,
|
|
|
|
SERVICE_STOP_COVER, SERVICE_OPEN_COVER_TILT, SERVICE_CLOSE_COVER_TILT,
|
|
|
|
SERVICE_STOP_COVER_TILT, SERVICE_SET_COVER_TILT_POSITION, STATE_OPEN,
|
2019-01-24 07:20:20 +00:00
|
|
|
STATE_CLOSED, STATE_OPENING, STATE_CLOSING, ATTR_ENTITY_ID)
|
2016-08-24 01:23:18 +00:00
|
|
|
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2016-08-24 01:23:18 +00:00
|
|
|
|
|
|
|
DOMAIN = 'cover'
|
2017-06-15 22:52:28 +00:00
|
|
|
DEPENDENCIES = ['group']
|
2017-01-05 23:16:12 +00:00
|
|
|
SCAN_INTERVAL = timedelta(seconds=15)
|
2016-08-24 01:23:18 +00:00
|
|
|
|
2016-09-24 00:13:12 +00:00
|
|
|
GROUP_NAME_ALL_COVERS = 'all covers'
|
2016-09-24 00:14:29 +00:00
|
|
|
ENTITY_ID_ALL_COVERS = group.ENTITY_ID_FORMAT.format('all_covers')
|
2016-08-24 01:23:18 +00:00
|
|
|
|
|
|
|
ENTITY_ID_FORMAT = DOMAIN + '.{}'
|
|
|
|
|
2017-02-12 19:08:06 +00:00
|
|
|
DEVICE_CLASSES = [
|
2018-09-21 17:59:20 +00:00
|
|
|
'damper',
|
2017-02-12 19:08:06 +00:00
|
|
|
'garage', # Garage door control
|
2018-09-21 17:59:20 +00:00
|
|
|
'window', # Window control
|
2017-02-12 19:08:06 +00:00
|
|
|
]
|
|
|
|
|
2017-06-19 05:30:39 +00:00
|
|
|
DEVICE_CLASSES_SCHEMA = vol.All(vol.Lower, vol.In(DEVICE_CLASSES))
|
|
|
|
|
2017-02-19 02:11:03 +00:00
|
|
|
SUPPORT_OPEN = 1
|
|
|
|
SUPPORT_CLOSE = 2
|
|
|
|
SUPPORT_SET_POSITION = 4
|
|
|
|
SUPPORT_STOP = 8
|
|
|
|
SUPPORT_OPEN_TILT = 16
|
|
|
|
SUPPORT_CLOSE_TILT = 32
|
|
|
|
SUPPORT_STOP_TILT = 64
|
|
|
|
SUPPORT_SET_TILT_POSITION = 128
|
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
ATTR_CURRENT_POSITION = 'current_position'
|
|
|
|
ATTR_CURRENT_TILT_POSITION = 'current_tilt_position'
|
|
|
|
ATTR_POSITION = 'position'
|
|
|
|
ATTR_TILT_POSITION = 'tilt_position'
|
|
|
|
|
2018-03-31 00:22:48 +00:00
|
|
|
INTENT_OPEN_COVER = 'HassOpenCover'
|
|
|
|
INTENT_CLOSE_COVER = 'HassCloseCover'
|
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
COVER_SERVICE_SCHEMA = vol.Schema({
|
2018-12-13 09:07:59 +00:00
|
|
|
vol.Optional(ATTR_ENTITY_ID): cv.comp_entity_ids,
|
2016-08-24 01:23:18 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
COVER_SET_COVER_POSITION_SCHEMA = COVER_SERVICE_SCHEMA.extend({
|
|
|
|
vol.Required(ATTR_POSITION):
|
|
|
|
vol.All(vol.Coerce(int), vol.Range(min=0, max=100)),
|
|
|
|
})
|
|
|
|
|
|
|
|
COVER_SET_COVER_TILT_POSITION_SCHEMA = COVER_SERVICE_SCHEMA.extend({
|
|
|
|
vol.Required(ATTR_TILT_POSITION):
|
|
|
|
vol.All(vol.Coerce(int), vol.Range(min=0, max=100)),
|
|
|
|
})
|
|
|
|
|
|
|
|
|
2017-07-16 17:14:46 +00:00
|
|
|
@bind_hass
|
2016-08-24 01:23:18 +00:00
|
|
|
def is_closed(hass, entity_id=None):
|
|
|
|
"""Return if the cover is closed based on the statemachine."""
|
|
|
|
entity_id = entity_id or ENTITY_ID_ALL_COVERS
|
|
|
|
return hass.states.is_state(entity_id, STATE_CLOSED)
|
|
|
|
|
|
|
|
|
2018-02-24 18:24:33 +00:00
|
|
|
async def async_setup(hass, config):
|
2016-08-24 01:23:18 +00:00
|
|
|
"""Track states and offer events for covers."""
|
2018-09-21 17:59:20 +00:00
|
|
|
component = hass.data[DOMAIN] = EntityComponent(
|
2016-08-24 01:23:18 +00:00
|
|
|
_LOGGER, DOMAIN, hass, SCAN_INTERVAL, GROUP_NAME_ALL_COVERS)
|
|
|
|
|
2018-02-24 18:24:33 +00:00
|
|
|
await component.async_setup(config)
|
2017-02-02 20:39:13 +00:00
|
|
|
|
2018-08-16 07:50:11 +00:00
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_OPEN_COVER, COVER_SERVICE_SCHEMA,
|
|
|
|
'async_open_cover'
|
|
|
|
)
|
|
|
|
|
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_CLOSE_COVER, COVER_SERVICE_SCHEMA,
|
|
|
|
'async_close_cover'
|
|
|
|
)
|
|
|
|
|
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_SET_COVER_POSITION, COVER_SET_COVER_POSITION_SCHEMA,
|
|
|
|
'async_set_cover_position'
|
|
|
|
)
|
|
|
|
|
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_STOP_COVER, COVER_SERVICE_SCHEMA,
|
|
|
|
'async_stop_cover'
|
|
|
|
)
|
|
|
|
|
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_OPEN_COVER_TILT, COVER_SERVICE_SCHEMA,
|
|
|
|
'async_open_cover_tilt'
|
|
|
|
)
|
|
|
|
|
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_CLOSE_COVER_TILT, COVER_SERVICE_SCHEMA,
|
|
|
|
'async_close_cover_tilt'
|
|
|
|
)
|
|
|
|
|
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_STOP_COVER_TILT, COVER_SERVICE_SCHEMA,
|
|
|
|
'async_stop_cover_tilt'
|
|
|
|
)
|
|
|
|
|
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_SET_COVER_TILT_POSITION, COVER_SET_COVER_TILT_POSITION_SCHEMA,
|
|
|
|
'async_set_cover_tilt_position'
|
|
|
|
)
|
|
|
|
|
2018-03-31 00:22:48 +00:00
|
|
|
hass.helpers.intent.async_register(intent.ServiceIntentHandler(
|
|
|
|
INTENT_OPEN_COVER, DOMAIN, SERVICE_OPEN_COVER,
|
|
|
|
"Opened {}"))
|
|
|
|
hass.helpers.intent.async_register(intent.ServiceIntentHandler(
|
|
|
|
INTENT_CLOSE_COVER, DOMAIN, SERVICE_CLOSE_COVER,
|
|
|
|
"Closed {}"))
|
2017-02-02 20:39:13 +00:00
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2018-09-21 17:59:20 +00:00
|
|
|
async def async_setup_entry(hass, entry):
|
|
|
|
"""Set up a config entry."""
|
|
|
|
return await hass.data[DOMAIN].async_setup_entry(entry)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_unload_entry(hass, entry):
|
|
|
|
"""Unload a config entry."""
|
|
|
|
return await hass.data[DOMAIN].async_unload_entry(entry)
|
|
|
|
|
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
class CoverDevice(Entity):
|
|
|
|
"""Representation a cover."""
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_cover_position(self):
|
|
|
|
"""Return current position of cover.
|
|
|
|
|
|
|
|
None is unknown, 0 is closed, 100 is fully open.
|
|
|
|
"""
|
2016-08-24 09:53:02 +00:00
|
|
|
pass
|
2016-08-24 01:23:18 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def current_cover_tilt_position(self):
|
|
|
|
"""Return current position of cover tilt.
|
|
|
|
|
|
|
|
None is unknown, 0 is closed, 100 is fully open.
|
|
|
|
"""
|
2016-08-24 09:53:02 +00:00
|
|
|
pass
|
2016-08-24 01:23:18 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the cover."""
|
2017-07-27 22:57:30 +00:00
|
|
|
if self.is_opening:
|
|
|
|
return STATE_OPENING
|
|
|
|
if self.is_closing:
|
|
|
|
return STATE_CLOSING
|
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
closed = self.is_closed
|
|
|
|
|
|
|
|
if closed is None:
|
2019-01-24 07:20:20 +00:00
|
|
|
return None
|
2016-08-24 01:23:18 +00:00
|
|
|
|
|
|
|
return STATE_CLOSED if closed else STATE_OPEN
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state_attributes(self):
|
|
|
|
"""Return the state attributes."""
|
2016-08-24 09:53:02 +00:00
|
|
|
data = {}
|
|
|
|
|
|
|
|
current = self.current_cover_position
|
|
|
|
if current is not None:
|
|
|
|
data[ATTR_CURRENT_POSITION] = self.current_cover_position
|
2016-08-24 01:23:18 +00:00
|
|
|
|
|
|
|
current_tilt = self.current_cover_tilt_position
|
|
|
|
if current_tilt is not None:
|
|
|
|
data[ATTR_CURRENT_TILT_POSITION] = self.current_cover_tilt_position
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
2017-02-19 02:11:03 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Flag supported features."""
|
|
|
|
supported_features = SUPPORT_OPEN | SUPPORT_CLOSE | SUPPORT_STOP
|
|
|
|
|
|
|
|
if self.current_cover_position is not None:
|
|
|
|
supported_features |= SUPPORT_SET_POSITION
|
|
|
|
|
|
|
|
if self.current_cover_tilt_position is not None:
|
|
|
|
supported_features |= (
|
|
|
|
SUPPORT_OPEN_TILT | SUPPORT_CLOSE_TILT | SUPPORT_STOP_TILT |
|
|
|
|
SUPPORT_SET_TILT_POSITION)
|
|
|
|
|
|
|
|
return supported_features
|
|
|
|
|
2017-07-27 22:57:30 +00:00
|
|
|
@property
|
|
|
|
def is_opening(self):
|
|
|
|
"""Return if the cover is opening or not."""
|
|
|
|
pass
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_closing(self):
|
|
|
|
"""Return if the cover is closing or not."""
|
|
|
|
pass
|
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
@property
|
|
|
|
def is_closed(self):
|
|
|
|
"""Return if the cover is closed or not."""
|
2016-08-24 18:36:43 +00:00
|
|
|
raise NotImplementedError()
|
2016-08-24 01:23:18 +00:00
|
|
|
|
|
|
|
def open_cover(self, **kwargs):
|
|
|
|
"""Open the cover."""
|
|
|
|
raise NotImplementedError()
|
|
|
|
|
2017-02-02 20:39:13 +00:00
|
|
|
def async_open_cover(self, **kwargs):
|
|
|
|
"""Open the cover.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(ft.partial(self.open_cover, **kwargs))
|
2017-02-02 20:39:13 +00:00
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
def close_cover(self, **kwargs):
|
|
|
|
"""Close cover."""
|
|
|
|
raise NotImplementedError()
|
|
|
|
|
2017-02-02 20:39:13 +00:00
|
|
|
def async_close_cover(self, **kwargs):
|
|
|
|
"""Close cover.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(ft.partial(self.close_cover, **kwargs))
|
2017-02-02 20:39:13 +00:00
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
def set_cover_position(self, **kwargs):
|
|
|
|
"""Move the cover to a specific position."""
|
|
|
|
pass
|
|
|
|
|
2017-02-02 20:39:13 +00:00
|
|
|
def async_set_cover_position(self, **kwargs):
|
|
|
|
"""Move the cover to a specific position.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(
|
|
|
|
ft.partial(self.set_cover_position, **kwargs))
|
2017-02-02 20:39:13 +00:00
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
def stop_cover(self, **kwargs):
|
|
|
|
"""Stop the cover."""
|
|
|
|
pass
|
|
|
|
|
2017-02-02 20:39:13 +00:00
|
|
|
def async_stop_cover(self, **kwargs):
|
|
|
|
"""Stop the cover.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(ft.partial(self.stop_cover, **kwargs))
|
2017-02-02 20:39:13 +00:00
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
def open_cover_tilt(self, **kwargs):
|
|
|
|
"""Open the cover tilt."""
|
|
|
|
pass
|
|
|
|
|
2017-02-02 20:39:13 +00:00
|
|
|
def async_open_cover_tilt(self, **kwargs):
|
|
|
|
"""Open the cover tilt.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(
|
|
|
|
ft.partial(self.open_cover_tilt, **kwargs))
|
2017-02-02 20:39:13 +00:00
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
def close_cover_tilt(self, **kwargs):
|
|
|
|
"""Close the cover tilt."""
|
|
|
|
pass
|
|
|
|
|
2017-02-02 20:39:13 +00:00
|
|
|
def async_close_cover_tilt(self, **kwargs):
|
|
|
|
"""Close the cover tilt.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(
|
|
|
|
ft.partial(self.close_cover_tilt, **kwargs))
|
2017-02-02 20:39:13 +00:00
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
def set_cover_tilt_position(self, **kwargs):
|
|
|
|
"""Move the cover tilt to a specific position."""
|
|
|
|
pass
|
|
|
|
|
2017-02-02 20:39:13 +00:00
|
|
|
def async_set_cover_tilt_position(self, **kwargs):
|
|
|
|
"""Move the cover tilt to a specific position.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(
|
|
|
|
ft.partial(self.set_cover_tilt_position, **kwargs))
|
2017-02-02 20:39:13 +00:00
|
|
|
|
2016-08-24 01:23:18 +00:00
|
|
|
def stop_cover_tilt(self, **kwargs):
|
|
|
|
"""Stop the cover."""
|
|
|
|
pass
|
2017-02-02 20:39:13 +00:00
|
|
|
|
|
|
|
def async_stop_cover_tilt(self, **kwargs):
|
|
|
|
"""Stop the cover.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(
|
|
|
|
ft.partial(self.stop_cover_tilt, **kwargs))
|