2019-04-03 15:40:03 +00:00
|
|
|
"""Provides functionality to interact with fans."""
|
2017-01-05 23:16:12 +00:00
|
|
|
from datetime import timedelta
|
2017-02-02 20:07:00 +00:00
|
|
|
import functools as ft
|
2016-08-25 12:47:07 +00:00
|
|
|
import logging
|
2019-08-12 03:38:18 +00:00
|
|
|
from typing import Optional
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
2020-08-21 14:20:28 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
SERVICE_TOGGLE,
|
|
|
|
SERVICE_TURN_OFF,
|
|
|
|
SERVICE_TURN_ON,
|
|
|
|
STATE_ON,
|
|
|
|
)
|
2019-12-08 15:58:47 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2019-11-16 09:22:07 +00:00
|
|
|
from homeassistant.helpers.config_validation import ( # noqa: F401
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA,
|
|
|
|
PLATFORM_SCHEMA_BASE,
|
|
|
|
)
|
2019-12-08 15:58:47 +00:00
|
|
|
from homeassistant.helpers.entity import ToggleEntity
|
|
|
|
from homeassistant.helpers.entity_component import EntityComponent
|
|
|
|
from homeassistant.loader import bind_hass
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN = "fan"
|
2017-04-30 05:04:49 +00:00
|
|
|
SCAN_INTERVAL = timedelta(seconds=30)
|
2017-02-02 20:07:00 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ENTITY_ID_FORMAT = DOMAIN + ".{}"
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
# Bitfield of features supported by the fan entity
|
|
|
|
SUPPORT_SET_SPEED = 1
|
|
|
|
SUPPORT_OSCILLATE = 2
|
2017-01-14 06:08:13 +00:00
|
|
|
SUPPORT_DIRECTION = 4
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SERVICE_SET_SPEED = "set_speed"
|
|
|
|
SERVICE_OSCILLATE = "oscillate"
|
|
|
|
SERVICE_SET_DIRECTION = "set_direction"
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SPEED_OFF = "off"
|
|
|
|
SPEED_LOW = "low"
|
|
|
|
SPEED_MEDIUM = "medium"
|
|
|
|
SPEED_HIGH = "high"
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DIRECTION_FORWARD = "forward"
|
|
|
|
DIRECTION_REVERSE = "reverse"
|
2017-01-14 06:08:13 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_SPEED = "speed"
|
|
|
|
ATTR_SPEED_LIST = "speed_list"
|
|
|
|
ATTR_OSCILLATING = "oscillating"
|
|
|
|
ATTR_DIRECTION = "direction"
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
|
2017-07-16 17:14:46 +00:00
|
|
|
@bind_hass
|
2020-01-07 16:30:53 +00:00
|
|
|
def is_on(hass, entity_id: str) -> bool:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Return if the fans are on based on the statemachine."""
|
2016-08-27 20:53:12 +00:00
|
|
|
state = hass.states.get(entity_id)
|
2020-08-21 14:20:28 +00:00
|
|
|
if ATTR_SPEED in state.attributes:
|
|
|
|
return state.attributes[ATTR_SPEED] not in [SPEED_OFF, None]
|
|
|
|
return state.state == STATE_ON
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
|
2018-09-29 18:53:48 +00:00
|
|
|
async def async_setup(hass, config: dict):
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Expose fan control via statemachine and services."""
|
2018-09-29 18:53:48 +00:00
|
|
|
component = hass.data[DOMAIN] = EntityComponent(
|
2020-01-07 16:30:53 +00:00
|
|
|
_LOGGER, DOMAIN, hass, SCAN_INTERVAL
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2018-09-29 18:53:48 +00:00
|
|
|
await component.async_setup(config)
|
2017-02-02 20:07:00 +00:00
|
|
|
|
2018-08-16 12:28:59 +00:00
|
|
|
component.async_register_entity_service(
|
2019-12-03 00:23:12 +00:00
|
|
|
SERVICE_TURN_ON, {vol.Optional(ATTR_SPEED): cv.string}, "async_turn_on"
|
2018-08-16 12:28:59 +00:00
|
|
|
)
|
2019-12-03 00:23:12 +00:00
|
|
|
component.async_register_entity_service(SERVICE_TURN_OFF, {}, "async_turn_off")
|
|
|
|
component.async_register_entity_service(SERVICE_TOGGLE, {}, "async_toggle")
|
2018-08-16 12:28:59 +00:00
|
|
|
component.async_register_entity_service(
|
2020-01-06 16:10:51 +00:00
|
|
|
SERVICE_SET_SPEED,
|
|
|
|
{vol.Required(ATTR_SPEED): cv.string},
|
|
|
|
"async_set_speed",
|
|
|
|
[SUPPORT_SET_SPEED],
|
2018-08-16 12:28:59 +00:00
|
|
|
)
|
|
|
|
component.async_register_entity_service(
|
2019-12-03 00:23:12 +00:00
|
|
|
SERVICE_OSCILLATE,
|
|
|
|
{vol.Required(ATTR_OSCILLATING): cv.boolean},
|
|
|
|
"async_oscillate",
|
2020-01-06 16:10:51 +00:00
|
|
|
[SUPPORT_OSCILLATE],
|
2018-08-16 12:28:59 +00:00
|
|
|
)
|
|
|
|
component.async_register_entity_service(
|
2019-12-03 00:23:12 +00:00
|
|
|
SERVICE_SET_DIRECTION,
|
|
|
|
{vol.Optional(ATTR_DIRECTION): cv.string},
|
|
|
|
"async_set_direction",
|
2020-01-06 16:10:51 +00:00
|
|
|
[SUPPORT_DIRECTION],
|
2018-08-16 12:28:59 +00:00
|
|
|
)
|
2017-01-14 06:08:13 +00:00
|
|
|
|
2016-08-25 12:47:07 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2018-09-29 18:53:48 +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-27 20:53:12 +00:00
|
|
|
class FanEntity(ToggleEntity):
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Representation of a fan."""
|
|
|
|
|
2018-07-30 11:44:31 +00:00
|
|
|
def set_speed(self, speed: str) -> None:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Set the speed of the fan."""
|
2017-01-14 06:08:13 +00:00
|
|
|
raise NotImplementedError()
|
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_set_speed(self, speed: str):
|
|
|
|
"""Set the speed of the fan."""
|
2020-07-11 00:45:12 +00:00
|
|
|
if speed == SPEED_OFF:
|
2020-01-29 21:59:45 +00:00
|
|
|
await self.async_turn_off()
|
|
|
|
else:
|
|
|
|
await self.hass.async_add_job(self.set_speed, speed)
|
2017-02-02 20:07:00 +00:00
|
|
|
|
2018-07-30 11:44:31 +00:00
|
|
|
def set_direction(self, direction: str) -> None:
|
2017-01-14 06:08:13 +00:00
|
|
|
"""Set the direction of the fan."""
|
|
|
|
raise NotImplementedError()
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_set_direction(self, direction: str):
|
|
|
|
"""Set the direction of the fan."""
|
|
|
|
await self.hass.async_add_job(self.set_direction, direction)
|
2017-02-02 20:07:00 +00:00
|
|
|
|
2018-02-11 17:20:28 +00:00
|
|
|
# pylint: disable=arguments-differ
|
2019-08-12 03:38:18 +00:00
|
|
|
def turn_on(self, speed: Optional[str] = None, **kwargs) -> None:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Turn on the fan."""
|
2016-08-27 20:53:12 +00:00
|
|
|
raise NotImplementedError()
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2018-02-11 17:20:28 +00:00
|
|
|
# pylint: disable=arguments-differ
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_turn_on(self, speed: Optional[str] = None, **kwargs):
|
|
|
|
"""Turn on the fan."""
|
2020-07-11 00:45:12 +00:00
|
|
|
if speed == SPEED_OFF:
|
2020-01-29 21:59:45 +00:00
|
|
|
await self.async_turn_off()
|
|
|
|
else:
|
|
|
|
await self.hass.async_add_job(ft.partial(self.turn_on, speed, **kwargs))
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2018-07-30 11:44:31 +00:00
|
|
|
def oscillate(self, oscillating: bool) -> None:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Oscillate the fan."""
|
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_oscillate(self, oscillating: bool):
|
|
|
|
"""Oscillate the fan."""
|
|
|
|
await self.hass.async_add_job(self.oscillate, oscillating)
|
2017-02-02 20:07:00 +00:00
|
|
|
|
2016-08-25 12:47:07 +00:00
|
|
|
@property
|
2016-08-27 20:53:12 +00:00
|
|
|
def is_on(self):
|
|
|
|
"""Return true if the entity is on."""
|
2019-01-24 07:20:20 +00:00
|
|
|
return self.speed not in [SPEED_OFF, None]
|
2017-01-14 06:08:13 +00:00
|
|
|
|
|
|
|
@property
|
2019-08-12 03:38:18 +00:00
|
|
|
def speed(self) -> Optional[str]:
|
2017-01-14 06:08:13 +00:00
|
|
|
"""Return the current speed."""
|
|
|
|
return None
|
2016-08-27 20:53:12 +00:00
|
|
|
|
|
|
|
@property
|
2018-07-30 11:44:31 +00:00
|
|
|
def speed_list(self) -> list:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Get the list of available speeds."""
|
|
|
|
return []
|
|
|
|
|
2017-01-14 06:08:13 +00:00
|
|
|
@property
|
2019-08-12 03:38:18 +00:00
|
|
|
def current_direction(self) -> Optional[str]:
|
2017-01-14 06:08:13 +00:00
|
|
|
"""Return the current direction of the fan."""
|
|
|
|
return None
|
|
|
|
|
2020-05-06 07:58:07 +00:00
|
|
|
@property
|
|
|
|
def oscillating(self):
|
|
|
|
"""Return whether or not the fan is currently oscillating."""
|
|
|
|
return None
|
|
|
|
|
2020-01-08 20:22:56 +00:00
|
|
|
@property
|
|
|
|
def capability_attributes(self):
|
2020-01-31 16:33:00 +00:00
|
|
|
"""Return capability attributes."""
|
2020-05-06 07:58:07 +00:00
|
|
|
if self.supported_features & SUPPORT_SET_SPEED:
|
|
|
|
return {ATTR_SPEED_LIST: self.speed_list}
|
|
|
|
return {}
|
2020-01-08 20:22:56 +00:00
|
|
|
|
2016-08-25 12:47:07 +00:00
|
|
|
@property
|
2018-07-30 11:44:31 +00:00
|
|
|
def state_attributes(self) -> dict:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Return optional state attributes."""
|
2019-09-07 06:48:58 +00:00
|
|
|
data = {}
|
2020-05-06 07:58:07 +00:00
|
|
|
supported_features = self.supported_features
|
|
|
|
|
|
|
|
if supported_features & SUPPORT_DIRECTION:
|
|
|
|
data[ATTR_DIRECTION] = self.current_direction
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2020-05-06 07:58:07 +00:00
|
|
|
if supported_features & SUPPORT_OSCILLATE:
|
|
|
|
data[ATTR_OSCILLATING] = self.oscillating
|
2016-08-27 20:53:12 +00:00
|
|
|
|
2020-05-06 07:58:07 +00:00
|
|
|
if supported_features & SUPPORT_SET_SPEED:
|
|
|
|
data[ATTR_SPEED] = self.speed
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
return data
|
|
|
|
|
|
|
|
@property
|
2018-07-30 11:44:31 +00:00
|
|
|
def supported_features(self) -> int:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Flag supported features."""
|
|
|
|
return 0
|