2019-02-14 15:01:46 +00:00
|
|
|
"""Support for Homekit covers."""
|
2019-01-13 18:09:47 +00:00
|
|
|
import logging
|
|
|
|
|
2020-02-24 09:55:33 +00:00
|
|
|
from aiohomekit.model.characteristics import CharacteristicsTypes
|
2019-08-14 16:14:15 +00:00
|
|
|
|
2019-01-13 18:09:47 +00:00
|
|
|
from homeassistant.components.cover import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_POSITION,
|
|
|
|
ATTR_TILT_POSITION,
|
|
|
|
SUPPORT_CLOSE,
|
|
|
|
SUPPORT_CLOSE_TILT,
|
|
|
|
SUPPORT_OPEN,
|
|
|
|
SUPPORT_OPEN_TILT,
|
|
|
|
SUPPORT_SET_POSITION,
|
|
|
|
SUPPORT_SET_TILT_POSITION,
|
2019-12-08 16:50:57 +00:00
|
|
|
SUPPORT_STOP,
|
2020-04-25 16:07:15 +00:00
|
|
|
CoverEntity,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
from homeassistant.const import STATE_CLOSED, STATE_CLOSING, STATE_OPEN, STATE_OPENING
|
2020-01-29 21:59:45 +00:00
|
|
|
from homeassistant.core import callback
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2019-03-26 06:49:51 +00:00
|
|
|
from . import KNOWN_DEVICES, HomeKitEntity
|
2019-03-21 05:56:46 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
STATE_STOPPED = "stopped"
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
CURRENT_GARAGE_STATE_MAP = {
|
|
|
|
0: STATE_OPEN,
|
|
|
|
1: STATE_CLOSED,
|
|
|
|
2: STATE_OPENING,
|
|
|
|
3: STATE_CLOSING,
|
2019-07-31 19:25:30 +00:00
|
|
|
4: STATE_STOPPED,
|
2019-01-13 18:09:47 +00:00
|
|
|
}
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
TARGET_GARAGE_STATE_MAP = {STATE_OPEN: 0, STATE_CLOSED: 1, STATE_STOPPED: 2}
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2019-09-25 04:16:08 +00:00
|
|
|
CURRENT_WINDOW_STATE_MAP = {0: STATE_CLOSING, 1: STATE_OPENING, 2: STATE_STOPPED}
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
|
2019-05-13 06:56:05 +00:00
|
|
|
async def async_setup_entry(hass, config_entry, async_add_entities):
|
|
|
|
"""Set up Homekit covers."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hkid = config_entry.data["AccessoryPairingID"]
|
2019-05-13 06:56:05 +00:00
|
|
|
conn = hass.data[KNOWN_DEVICES][hkid]
|
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
@callback
|
2019-05-13 06:56:05 +00:00
|
|
|
def async_add_service(aid, service):
|
2019-07-31 19:25:30 +00:00
|
|
|
info = {"aid": aid, "iid": service["iid"]}
|
|
|
|
if service["stype"] == "garage-door-opener":
|
2019-05-13 06:56:05 +00:00
|
|
|
async_add_entities([HomeKitGarageDoorCover(conn, info)], True)
|
|
|
|
return True
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if service["stype"] in ("window-covering", "window"):
|
2019-05-13 06:56:05 +00:00
|
|
|
async_add_entities([HomeKitWindowCover(conn, info)], True)
|
|
|
|
return True
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
conn.add_listener(async_add_service)
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
|
2020-04-25 16:07:15 +00:00
|
|
|
class HomeKitGarageDoorCover(HomeKitEntity, CoverEntity):
|
2019-01-13 18:09:47 +00:00
|
|
|
"""Representation of a HomeKit Garage Door."""
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_class(self):
|
|
|
|
"""Define this cover as a garage door."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "garage"
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2019-01-28 16:21:20 +00:00
|
|
|
def get_characteristic_types(self):
|
|
|
|
"""Define the homekit characteristics the entity cares about."""
|
|
|
|
return [
|
|
|
|
CharacteristicsTypes.DOOR_STATE_CURRENT,
|
|
|
|
CharacteristicsTypes.DOOR_STATE_TARGET,
|
|
|
|
CharacteristicsTypes.OBSTRUCTION_DETECTED,
|
|
|
|
]
|
|
|
|
|
2019-01-13 18:09:47 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Flag supported features."""
|
|
|
|
return SUPPORT_OPEN | SUPPORT_CLOSE
|
|
|
|
|
2020-03-11 11:40:47 +00:00
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the current state of the garage door."""
|
|
|
|
value = self.service.value(CharacteristicsTypes.DOOR_STATE_CURRENT)
|
|
|
|
return CURRENT_GARAGE_STATE_MAP[value]
|
|
|
|
|
2019-01-13 18:09:47 +00:00
|
|
|
@property
|
|
|
|
def is_closed(self):
|
|
|
|
"""Return true if cover is closed, else False."""
|
2020-03-11 11:40:47 +00:00
|
|
|
return self.state == STATE_CLOSED
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_closing(self):
|
|
|
|
"""Return if the cover is closing or not."""
|
2020-03-11 11:40:47 +00:00
|
|
|
return self.state == STATE_CLOSING
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_opening(self):
|
|
|
|
"""Return if the cover is opening or not."""
|
2020-03-11 11:40:47 +00:00
|
|
|
return self.state == STATE_OPENING
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2019-03-11 18:59:41 +00:00
|
|
|
async def async_open_cover(self, **kwargs):
|
2019-01-13 18:09:47 +00:00
|
|
|
"""Send open command."""
|
2019-03-11 18:59:41 +00:00
|
|
|
await self.set_door_state(STATE_OPEN)
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2019-03-11 18:59:41 +00:00
|
|
|
async def async_close_cover(self, **kwargs):
|
2019-01-13 18:09:47 +00:00
|
|
|
"""Send close command."""
|
2019-03-11 18:59:41 +00:00
|
|
|
await self.set_door_state(STATE_CLOSED)
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2019-03-11 18:59:41 +00:00
|
|
|
async def set_door_state(self, state):
|
2019-01-13 18:09:47 +00:00
|
|
|
"""Send state command."""
|
2020-03-11 16:27:20 +00:00
|
|
|
await self.async_put_characteristics(
|
|
|
|
{CharacteristicsTypes.DOOR_STATE_TARGET: TARGET_GARAGE_STATE_MAP[state]}
|
|
|
|
)
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the optional state attributes."""
|
2020-03-11 11:40:47 +00:00
|
|
|
attributes = {}
|
|
|
|
|
|
|
|
obstruction_detected = self.service.value(
|
|
|
|
CharacteristicsTypes.OBSTRUCTION_DETECTED
|
|
|
|
)
|
|
|
|
if obstruction_detected:
|
|
|
|
attributes["obstruction-detected"] = obstruction_detected
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2020-03-11 11:40:47 +00:00
|
|
|
return attributes
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
|
2020-04-25 16:07:15 +00:00
|
|
|
class HomeKitWindowCover(HomeKitEntity, CoverEntity):
|
2019-01-13 18:09:47 +00:00
|
|
|
"""Representation of a HomeKit Window or Window Covering."""
|
|
|
|
|
2019-01-28 16:21:20 +00:00
|
|
|
def get_characteristic_types(self):
|
|
|
|
"""Define the homekit characteristics the entity cares about."""
|
|
|
|
return [
|
|
|
|
CharacteristicsTypes.POSITION_STATE,
|
|
|
|
CharacteristicsTypes.POSITION_CURRENT,
|
|
|
|
CharacteristicsTypes.POSITION_TARGET,
|
|
|
|
CharacteristicsTypes.POSITION_HOLD,
|
|
|
|
CharacteristicsTypes.VERTICAL_TILT_CURRENT,
|
|
|
|
CharacteristicsTypes.VERTICAL_TILT_TARGET,
|
|
|
|
CharacteristicsTypes.HORIZONTAL_TILT_CURRENT,
|
|
|
|
CharacteristicsTypes.HORIZONTAL_TILT_TARGET,
|
|
|
|
CharacteristicsTypes.OBSTRUCTION_DETECTED,
|
|
|
|
]
|
|
|
|
|
2020-03-18 15:12:55 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Flag supported features."""
|
|
|
|
features = SUPPORT_OPEN | SUPPORT_CLOSE | SUPPORT_SET_POSITION
|
|
|
|
|
|
|
|
if self.service.has(CharacteristicsTypes.POSITION_HOLD):
|
|
|
|
features |= SUPPORT_STOP
|
2019-04-17 17:02:04 +00:00
|
|
|
|
2020-03-18 15:12:55 +00:00
|
|
|
supports_tilt = any(
|
|
|
|
(
|
|
|
|
self.service.has(CharacteristicsTypes.VERTICAL_TILT_CURRENT),
|
|
|
|
self.service.has(CharacteristicsTypes.HORIZONTAL_TILT_CURRENT),
|
|
|
|
)
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-04-17 17:02:04 +00:00
|
|
|
|
2020-03-18 15:12:55 +00:00
|
|
|
if supports_tilt:
|
|
|
|
features |= (
|
|
|
|
SUPPORT_OPEN_TILT | SUPPORT_CLOSE_TILT | SUPPORT_SET_TILT_POSITION
|
|
|
|
)
|
2019-04-17 17:02:04 +00:00
|
|
|
|
2020-03-18 15:12:55 +00:00
|
|
|
return features
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def current_cover_position(self):
|
|
|
|
"""Return the current position of cover."""
|
2020-03-11 11:40:47 +00:00
|
|
|
return self.service.value(CharacteristicsTypes.POSITION_CURRENT)
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_closed(self):
|
|
|
|
"""Return true if cover is closed, else False."""
|
2020-03-11 11:40:47 +00:00
|
|
|
return self.current_cover_position == 0
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_closing(self):
|
|
|
|
"""Return if the cover is closing or not."""
|
2020-03-11 11:40:47 +00:00
|
|
|
value = self.service.value(CharacteristicsTypes.POSITION_STATE)
|
|
|
|
state = CURRENT_WINDOW_STATE_MAP[value]
|
|
|
|
return state == STATE_CLOSING
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_opening(self):
|
|
|
|
"""Return if the cover is opening or not."""
|
2020-03-11 11:40:47 +00:00
|
|
|
value = self.service.value(CharacteristicsTypes.POSITION_STATE)
|
|
|
|
state = CURRENT_WINDOW_STATE_MAP[value]
|
|
|
|
return state == STATE_OPENING
|
|
|
|
|
2020-03-11 16:27:20 +00:00
|
|
|
@property
|
|
|
|
def is_horizontal_tilt(self):
|
|
|
|
"""Return True if the service has a horizontal tilt characteristic."""
|
|
|
|
return (
|
|
|
|
self.service.value(CharacteristicsTypes.HORIZONTAL_TILT_CURRENT) is not None
|
|
|
|
)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_vertical_tilt(self):
|
|
|
|
"""Return True if the service has a vertical tilt characteristic."""
|
|
|
|
return (
|
|
|
|
self.service.value(CharacteristicsTypes.VERTICAL_TILT_CURRENT) is not None
|
|
|
|
)
|
|
|
|
|
2020-03-11 11:40:47 +00:00
|
|
|
@property
|
|
|
|
def current_cover_tilt_position(self):
|
|
|
|
"""Return current position of cover tilt."""
|
|
|
|
tilt_position = self.service.value(CharacteristicsTypes.VERTICAL_TILT_CURRENT)
|
|
|
|
if not tilt_position:
|
|
|
|
tilt_position = self.service.value(
|
|
|
|
CharacteristicsTypes.HORIZONTAL_TILT_CURRENT
|
|
|
|
)
|
|
|
|
return tilt_position
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2019-04-17 17:02:04 +00:00
|
|
|
async def async_stop_cover(self, **kwargs):
|
|
|
|
"""Send hold command."""
|
2020-03-11 16:27:20 +00:00
|
|
|
await self.async_put_characteristics({CharacteristicsTypes.POSITION_HOLD: 1})
|
2019-04-17 17:02:04 +00:00
|
|
|
|
2019-03-11 18:59:41 +00:00
|
|
|
async def async_open_cover(self, **kwargs):
|
2019-01-13 18:09:47 +00:00
|
|
|
"""Send open command."""
|
2019-03-11 18:59:41 +00:00
|
|
|
await self.async_set_cover_position(position=100)
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2019-03-13 01:37:33 +00:00
|
|
|
async def async_close_cover(self, **kwargs):
|
2019-01-13 18:09:47 +00:00
|
|
|
"""Send close command."""
|
2019-03-11 18:59:41 +00:00
|
|
|
await self.async_set_cover_position(position=0)
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2019-03-11 18:59:41 +00:00
|
|
|
async def async_set_cover_position(self, **kwargs):
|
2019-01-13 18:09:47 +00:00
|
|
|
"""Send position command."""
|
|
|
|
position = kwargs[ATTR_POSITION]
|
2020-03-11 16:27:20 +00:00
|
|
|
await self.async_put_characteristics(
|
|
|
|
{CharacteristicsTypes.POSITION_TARGET: position}
|
|
|
|
)
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2019-03-11 18:59:41 +00:00
|
|
|
async def async_set_cover_tilt_position(self, **kwargs):
|
2019-01-13 18:09:47 +00:00
|
|
|
"""Move the cover tilt to a specific position."""
|
|
|
|
tilt_position = kwargs[ATTR_TILT_POSITION]
|
2020-03-11 16:27:20 +00:00
|
|
|
if self.is_vertical_tilt:
|
|
|
|
await self.async_put_characteristics(
|
|
|
|
{CharacteristicsTypes.VERTICAL_TILT_TARGET: tilt_position}
|
|
|
|
)
|
|
|
|
elif self.is_horizontal_tilt:
|
|
|
|
await self.async_put_characteristics(
|
|
|
|
{CharacteristicsTypes.HORIZONTAL_TILT_TARGET: tilt_position}
|
|
|
|
)
|
2019-01-13 18:09:47 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the optional state attributes."""
|
2020-03-11 11:40:47 +00:00
|
|
|
attributes = {}
|
|
|
|
|
|
|
|
obstruction_detected = self.service.value(
|
|
|
|
CharacteristicsTypes.OBSTRUCTION_DETECTED
|
|
|
|
)
|
|
|
|
if obstruction_detected:
|
|
|
|
attributes["obstruction-detected"] = obstruction_detected
|
2019-01-13 18:09:47 +00:00
|
|
|
|
2020-03-11 11:40:47 +00:00
|
|
|
return attributes
|