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
|
2021-02-19 07:05:09 +00:00
|
|
|
import math
|
2021-01-27 23:44:36 +00:00
|
|
|
from typing import List, 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
|
2021-01-27 23:44:36 +00:00
|
|
|
from homeassistant.util.percentage import (
|
|
|
|
ordered_list_item_to_percentage,
|
|
|
|
percentage_to_ordered_list_item,
|
2021-02-19 07:05:09 +00:00
|
|
|
percentage_to_ranged_value,
|
|
|
|
ranged_value_to_percentage,
|
2021-01-27 23:44:36 +00:00
|
|
|
)
|
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
|
2021-01-27 23:44:36 +00:00
|
|
|
SUPPORT_PRESET_MODE = 8
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SERVICE_SET_SPEED = "set_speed"
|
2021-02-19 07:05:09 +00:00
|
|
|
SERVICE_INCREASE_SPEED = "increase_speed"
|
|
|
|
SERVICE_DECREASE_SPEED = "decrease_speed"
|
2019-07-31 19:25:30 +00:00
|
|
|
SERVICE_OSCILLATE = "oscillate"
|
|
|
|
SERVICE_SET_DIRECTION = "set_direction"
|
2021-01-27 23:44:36 +00:00
|
|
|
SERVICE_SET_PERCENTAGE = "set_percentage"
|
|
|
|
SERVICE_SET_PRESET_MODE = "set_preset_mode"
|
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"
|
2021-01-27 23:44:36 +00:00
|
|
|
ATTR_PERCENTAGE = "percentage"
|
2021-02-19 07:05:09 +00:00
|
|
|
ATTR_PERCENTAGE_STEP = "percentage_step"
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_SPEED_LIST = "speed_list"
|
|
|
|
ATTR_OSCILLATING = "oscillating"
|
|
|
|
ATTR_DIRECTION = "direction"
|
2021-01-27 23:44:36 +00:00
|
|
|
ATTR_PRESET_MODE = "preset_mode"
|
|
|
|
ATTR_PRESET_MODES = "preset_modes"
|
|
|
|
|
|
|
|
# Invalid speeds do not conform to the entity model, but have crept
|
|
|
|
# into core integrations at some point so we are temporarily
|
|
|
|
# accommodating them in the transition to percentages.
|
|
|
|
_NOT_SPEED_OFF = "off"
|
2021-02-06 11:48:18 +00:00
|
|
|
_NOT_SPEED_ON = "on"
|
2021-01-27 23:44:36 +00:00
|
|
|
_NOT_SPEED_AUTO = "auto"
|
|
|
|
_NOT_SPEED_SMART = "smart"
|
|
|
|
_NOT_SPEED_INTERVAL = "interval"
|
|
|
|
_NOT_SPEED_IDLE = "idle"
|
|
|
|
_NOT_SPEED_FAVORITE = "favorite"
|
2021-02-06 11:48:18 +00:00
|
|
|
_NOT_SPEED_SLEEP = "sleep"
|
2021-03-04 21:02:50 +00:00
|
|
|
_NOT_SPEED_SILENT = "silent"
|
2021-01-27 23:44:36 +00:00
|
|
|
|
|
|
|
_NOT_SPEEDS_FILTER = {
|
|
|
|
_NOT_SPEED_OFF,
|
2021-02-06 11:48:18 +00:00
|
|
|
_NOT_SPEED_ON,
|
2021-01-27 23:44:36 +00:00
|
|
|
_NOT_SPEED_AUTO,
|
|
|
|
_NOT_SPEED_SMART,
|
|
|
|
_NOT_SPEED_INTERVAL,
|
|
|
|
_NOT_SPEED_IDLE,
|
2021-03-04 21:02:50 +00:00
|
|
|
_NOT_SPEED_SILENT,
|
2021-02-06 11:48:18 +00:00
|
|
|
_NOT_SPEED_SLEEP,
|
2021-01-27 23:44:36 +00:00
|
|
|
_NOT_SPEED_FAVORITE,
|
|
|
|
}
|
|
|
|
|
|
|
|
_FAN_NATIVE = "_fan_native"
|
|
|
|
|
|
|
|
OFF_SPEED_VALUES = [SPEED_OFF, None]
|
|
|
|
|
2021-02-06 11:48:18 +00:00
|
|
|
LEGACY_SPEED_LIST = [SPEED_LOW, SPEED_MEDIUM, SPEED_HIGH]
|
|
|
|
|
2021-01-27 23:44:36 +00:00
|
|
|
|
|
|
|
class NoValidSpeedsError(ValueError):
|
|
|
|
"""Exception class when there are no valid speeds."""
|
|
|
|
|
|
|
|
|
|
|
|
class NotValidSpeedError(ValueError):
|
|
|
|
"""Exception class when the speed in not in the speed list."""
|
|
|
|
|
|
|
|
|
|
|
|
class NotValidPresetModeError(ValueError):
|
|
|
|
"""Exception class when the preset_mode in not in the preset_modes list."""
|
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:
|
2021-01-27 23:44:36 +00:00
|
|
|
return state.attributes[ATTR_SPEED] not in OFF_SPEED_VALUES
|
2020-08-21 14:20:28 +00:00
|
|
|
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
|
|
|
|
2021-01-27 23:44:36 +00:00
|
|
|
# After the transition to percentage and preset_modes concludes,
|
|
|
|
# switch this back to async_turn_on and remove async_turn_on_compat
|
2018-08-16 12:28:59 +00:00
|
|
|
component.async_register_entity_service(
|
2021-01-27 23:44:36 +00:00
|
|
|
SERVICE_TURN_ON,
|
|
|
|
{
|
|
|
|
vol.Optional(ATTR_SPEED): cv.string,
|
|
|
|
vol.Optional(ATTR_PERCENTAGE): vol.All(
|
|
|
|
vol.Coerce(int), vol.Range(min=0, max=100)
|
|
|
|
),
|
|
|
|
vol.Optional(ATTR_PRESET_MODE): cv.string,
|
|
|
|
},
|
|
|
|
"async_turn_on_compat",
|
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")
|
2021-01-27 23:44:36 +00:00
|
|
|
# After the transition to percentage and preset_modes concludes,
|
|
|
|
# remove this service
|
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},
|
2021-01-27 23:44:36 +00:00
|
|
|
"async_set_speed_deprecated",
|
2020-01-06 16:10:51 +00:00
|
|
|
[SUPPORT_SET_SPEED],
|
2018-08-16 12:28:59 +00:00
|
|
|
)
|
2021-02-19 07:05:09 +00:00
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_INCREASE_SPEED,
|
|
|
|
{
|
|
|
|
vol.Optional(ATTR_PERCENTAGE_STEP): vol.All(
|
|
|
|
vol.Coerce(int), vol.Range(min=0, max=100)
|
|
|
|
)
|
|
|
|
},
|
|
|
|
"async_increase_speed",
|
|
|
|
[SUPPORT_SET_SPEED],
|
|
|
|
)
|
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_DECREASE_SPEED,
|
|
|
|
{
|
|
|
|
vol.Optional(ATTR_PERCENTAGE_STEP): vol.All(
|
|
|
|
vol.Coerce(int), vol.Range(min=0, max=100)
|
|
|
|
)
|
|
|
|
},
|
|
|
|
"async_decrease_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
|
|
|
)
|
2021-01-27 23:44:36 +00:00
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_SET_PERCENTAGE,
|
|
|
|
{
|
|
|
|
vol.Required(ATTR_PERCENTAGE): vol.All(
|
|
|
|
vol.Coerce(int), vol.Range(min=0, max=100)
|
|
|
|
)
|
|
|
|
},
|
|
|
|
"async_set_percentage",
|
|
|
|
[SUPPORT_SET_SPEED],
|
|
|
|
)
|
|
|
|
component.async_register_entity_service(
|
|
|
|
SERVICE_SET_PRESET_MODE,
|
|
|
|
{vol.Required(ATTR_PRESET_MODE): cv.string},
|
|
|
|
"async_set_preset_mode",
|
|
|
|
[SUPPORT_SET_SPEED, SUPPORT_PRESET_MODE],
|
|
|
|
)
|
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)
|
|
|
|
|
|
|
|
|
2021-01-27 23:44:36 +00:00
|
|
|
def _fan_native(method):
|
|
|
|
"""Native fan method not overridden."""
|
|
|
|
setattr(method, _FAN_NATIVE, True)
|
|
|
|
return method
|
|
|
|
|
|
|
|
|
2016-08-27 20:53:12 +00:00
|
|
|
class FanEntity(ToggleEntity):
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Representation of a fan."""
|
|
|
|
|
2021-01-27 23:44:36 +00:00
|
|
|
@_fan_native
|
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()
|
|
|
|
|
2021-01-27 23:44:36 +00:00
|
|
|
async def async_set_speed_deprecated(self, speed: str):
|
|
|
|
"""Set the speed of the fan."""
|
|
|
|
_LOGGER.warning(
|
|
|
|
"fan.set_speed is deprecated, use fan.set_percentage or fan.set_preset_mode instead."
|
|
|
|
)
|
|
|
|
await self.async_set_speed(speed)
|
|
|
|
|
|
|
|
@_fan_native
|
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()
|
2021-01-27 23:44:36 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
if speed in self.preset_modes:
|
|
|
|
if not hasattr(self.async_set_preset_mode, _FAN_NATIVE):
|
|
|
|
await self.async_set_preset_mode(speed)
|
|
|
|
return
|
|
|
|
if not hasattr(self.set_preset_mode, _FAN_NATIVE):
|
|
|
|
await self.hass.async_add_executor_job(self.set_preset_mode, speed)
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
if not hasattr(self.async_set_percentage, _FAN_NATIVE):
|
|
|
|
await self.async_set_percentage(self.speed_to_percentage(speed))
|
|
|
|
return
|
|
|
|
if not hasattr(self.set_percentage, _FAN_NATIVE):
|
|
|
|
await self.hass.async_add_executor_job(
|
|
|
|
self.set_percentage, self.speed_to_percentage(speed)
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
await self.hass.async_add_executor_job(self.set_speed, speed)
|
|
|
|
|
|
|
|
@_fan_native
|
|
|
|
def set_percentage(self, percentage: int) -> None:
|
|
|
|
"""Set the speed of the fan, as a percentage."""
|
|
|
|
raise NotImplementedError()
|
|
|
|
|
|
|
|
@_fan_native
|
|
|
|
async def async_set_percentage(self, percentage: int) -> None:
|
|
|
|
"""Set the speed of the fan, as a percentage."""
|
|
|
|
if percentage == 0:
|
|
|
|
await self.async_turn_off()
|
|
|
|
elif not hasattr(self.set_percentage, _FAN_NATIVE):
|
|
|
|
await self.hass.async_add_executor_job(self.set_percentage, percentage)
|
2020-01-29 21:59:45 +00:00
|
|
|
else:
|
2021-01-27 23:44:36 +00:00
|
|
|
await self.async_set_speed(self.percentage_to_speed(percentage))
|
|
|
|
|
2021-02-20 05:57:21 +00:00
|
|
|
async def async_increase_speed(self, percentage_step: Optional[int] = None) -> None:
|
2021-02-19 07:05:09 +00:00
|
|
|
"""Increase the speed of the fan."""
|
|
|
|
await self._async_adjust_speed(1, percentage_step)
|
|
|
|
|
2021-02-20 05:57:21 +00:00
|
|
|
async def async_decrease_speed(self, percentage_step: Optional[int] = None) -> None:
|
2021-02-19 07:05:09 +00:00
|
|
|
"""Decrease the speed of the fan."""
|
|
|
|
await self._async_adjust_speed(-1, percentage_step)
|
|
|
|
|
2021-02-20 05:57:21 +00:00
|
|
|
async def _async_adjust_speed(
|
|
|
|
self, modifier: int, percentage_step: Optional[int]
|
|
|
|
) -> None:
|
2021-02-19 07:05:09 +00:00
|
|
|
"""Increase or decrease the speed of the fan."""
|
|
|
|
current_percentage = self.percentage or 0
|
|
|
|
|
|
|
|
if percentage_step is not None:
|
|
|
|
new_percentage = current_percentage + (percentage_step * modifier)
|
|
|
|
else:
|
|
|
|
speed_range = (1, self.speed_count)
|
|
|
|
speed_index = math.ceil(
|
|
|
|
percentage_to_ranged_value(speed_range, current_percentage)
|
|
|
|
)
|
|
|
|
new_percentage = ranged_value_to_percentage(
|
|
|
|
speed_range, speed_index + modifier
|
|
|
|
)
|
|
|
|
|
|
|
|
new_percentage = max(0, min(100, new_percentage))
|
|
|
|
|
|
|
|
await self.async_set_percentage(new_percentage)
|
|
|
|
|
2021-01-27 23:44:36 +00:00
|
|
|
@_fan_native
|
|
|
|
def set_preset_mode(self, preset_mode: str) -> None:
|
|
|
|
"""Set new preset mode."""
|
|
|
|
self._valid_preset_mode_or_raise(preset_mode)
|
|
|
|
self.set_speed(preset_mode)
|
|
|
|
|
|
|
|
@_fan_native
|
|
|
|
async def async_set_preset_mode(self, preset_mode: str) -> None:
|
|
|
|
"""Set new preset mode."""
|
|
|
|
if not hasattr(self.set_preset_mode, _FAN_NATIVE):
|
|
|
|
await self.hass.async_add_executor_job(self.set_preset_mode, preset_mode)
|
|
|
|
return
|
|
|
|
|
|
|
|
self._valid_preset_mode_or_raise(preset_mode)
|
|
|
|
await self.async_set_speed(preset_mode)
|
|
|
|
|
|
|
|
def _valid_preset_mode_or_raise(self, preset_mode):
|
|
|
|
"""Raise NotValidPresetModeError on invalid preset_mode."""
|
|
|
|
preset_modes = self.preset_modes
|
|
|
|
if preset_mode not in preset_modes:
|
|
|
|
raise NotValidPresetModeError(
|
|
|
|
f"The preset_mode {preset_mode} is not a valid preset_mode: {preset_modes}"
|
|
|
|
)
|
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."""
|
2020-10-08 07:21:29 +00:00
|
|
|
await self.hass.async_add_executor_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
|
2021-01-27 23:44:36 +00:00
|
|
|
def turn_on(
|
|
|
|
self,
|
|
|
|
speed: Optional[str] = None,
|
|
|
|
percentage: Optional[int] = None,
|
|
|
|
preset_mode: 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
|
|
|
|
2021-01-27 23:44:36 +00:00
|
|
|
async def async_turn_on_compat(
|
|
|
|
self,
|
|
|
|
speed: Optional[str] = None,
|
|
|
|
percentage: Optional[int] = None,
|
|
|
|
preset_mode: Optional[str] = None,
|
|
|
|
**kwargs,
|
|
|
|
) -> None:
|
|
|
|
"""Turn on the fan.
|
|
|
|
|
|
|
|
This _compat version wraps async_turn_on with
|
|
|
|
backwards and forward compatibility.
|
|
|
|
|
|
|
|
After the transition to percentage and preset_modes concludes, it
|
|
|
|
should be removed.
|
|
|
|
"""
|
|
|
|
if preset_mode is not None:
|
|
|
|
self._valid_preset_mode_or_raise(preset_mode)
|
|
|
|
speed = preset_mode
|
|
|
|
percentage = None
|
|
|
|
elif speed is not None:
|
|
|
|
_LOGGER.warning(
|
|
|
|
"Calling fan.turn_on with the speed argument is deprecated, use percentage or preset_mode instead."
|
|
|
|
)
|
|
|
|
if speed in self.preset_modes:
|
|
|
|
preset_mode = speed
|
|
|
|
percentage = None
|
|
|
|
else:
|
|
|
|
percentage = self.speed_to_percentage(speed)
|
|
|
|
elif percentage is not None:
|
|
|
|
speed = self.percentage_to_speed(percentage)
|
|
|
|
|
|
|
|
await self.async_turn_on(
|
|
|
|
speed=speed,
|
|
|
|
percentage=percentage,
|
|
|
|
preset_mode=preset_mode,
|
|
|
|
**kwargs,
|
|
|
|
)
|
|
|
|
|
|
|
|
# pylint: disable=arguments-differ
|
|
|
|
async def async_turn_on(
|
|
|
|
self,
|
|
|
|
speed: Optional[str] = None,
|
|
|
|
percentage: Optional[int] = None,
|
|
|
|
preset_mode: Optional[str] = None,
|
|
|
|
**kwargs,
|
|
|
|
) -> None:
|
2020-01-29 21:59:45 +00:00
|
|
|
"""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:
|
2020-10-08 07:21:29 +00:00
|
|
|
await self.hass.async_add_executor_job(
|
2021-01-27 23:44:36 +00:00
|
|
|
ft.partial(
|
|
|
|
self.turn_on,
|
|
|
|
speed=speed,
|
|
|
|
percentage=percentage,
|
|
|
|
preset_mode=preset_mode,
|
|
|
|
**kwargs,
|
|
|
|
)
|
2020-10-08 07:21:29 +00:00
|
|
|
)
|
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-11-19 11:05:52 +00:00
|
|
|
raise NotImplementedError()
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2020-01-29 21:59:45 +00:00
|
|
|
async def async_oscillate(self, oscillating: bool):
|
|
|
|
"""Oscillate the fan."""
|
2020-10-08 07:21:29 +00:00
|
|
|
await self.hass.async_add_executor_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
|
|
|
|
2021-01-27 23:44:36 +00:00
|
|
|
@property
|
|
|
|
def _implemented_percentage(self):
|
|
|
|
"""Return true if percentage has been implemented."""
|
|
|
|
return not hasattr(self.set_percentage, _FAN_NATIVE) or not hasattr(
|
|
|
|
self.async_set_percentage, _FAN_NATIVE
|
|
|
|
)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def _implemented_preset_mode(self):
|
|
|
|
"""Return true if preset_mode has been implemented."""
|
|
|
|
return not hasattr(self.set_preset_mode, _FAN_NATIVE) or not hasattr(
|
|
|
|
self.async_set_preset_mode, _FAN_NATIVE
|
|
|
|
)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def _implemented_speed(self):
|
|
|
|
"""Return true if speed has been implemented."""
|
|
|
|
return not hasattr(self.set_speed, _FAN_NATIVE) or not hasattr(
|
|
|
|
self.async_set_speed, _FAN_NATIVE
|
|
|
|
)
|
|
|
|
|
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."""
|
2021-01-27 23:44:36 +00:00
|
|
|
if self._implemented_preset_mode:
|
|
|
|
preset_mode = self.preset_mode
|
|
|
|
if preset_mode:
|
|
|
|
return preset_mode
|
|
|
|
if self._implemented_percentage:
|
2021-02-06 11:48:18 +00:00
|
|
|
percentage = self.percentage
|
|
|
|
if percentage is None:
|
|
|
|
return None
|
|
|
|
return self.percentage_to_speed(percentage)
|
2017-01-14 06:08:13 +00:00
|
|
|
return None
|
2016-08-27 20:53:12 +00:00
|
|
|
|
2021-01-27 23:44:36 +00:00
|
|
|
@property
|
|
|
|
def percentage(self) -> Optional[int]:
|
|
|
|
"""Return the current speed as a percentage."""
|
|
|
|
if not self._implemented_preset_mode:
|
|
|
|
if self.speed in self.preset_modes:
|
|
|
|
return None
|
|
|
|
if not self._implemented_percentage:
|
|
|
|
return self.speed_to_percentage(self.speed)
|
|
|
|
return 0
|
|
|
|
|
2021-02-19 07:05:09 +00:00
|
|
|
@property
|
2021-02-20 05:57:21 +00:00
|
|
|
def speed_count(self) -> int:
|
2021-02-19 07:05:09 +00:00
|
|
|
"""Return the number of speeds the fan supports."""
|
|
|
|
speed_list = speed_list_without_preset_modes(self.speed_list)
|
|
|
|
if speed_list:
|
|
|
|
return len(speed_list)
|
|
|
|
return 100
|
|
|
|
|
|
|
|
@property
|
2021-02-20 05:57:21 +00:00
|
|
|
def percentage_step(self) -> float:
|
2021-02-19 07:05:09 +00:00
|
|
|
"""Return the step size for percentage."""
|
|
|
|
return 100 / self.speed_count
|
|
|
|
|
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."""
|
2021-01-27 23:44:36 +00:00
|
|
|
speeds = []
|
|
|
|
if self._implemented_percentage:
|
2021-02-06 11:48:18 +00:00
|
|
|
speeds += [SPEED_OFF, *LEGACY_SPEED_LIST]
|
2021-01-27 23:44:36 +00:00
|
|
|
if self._implemented_preset_mode:
|
|
|
|
speeds += self.preset_modes
|
|
|
|
return speeds
|
2016-08-25 12:47:07 +00:00
|
|
|
|
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."""
|
2021-01-27 23:44:36 +00:00
|
|
|
attrs = {}
|
2020-05-06 07:58:07 +00:00
|
|
|
if self.supported_features & SUPPORT_SET_SPEED:
|
2021-01-27 23:44:36 +00:00
|
|
|
attrs[ATTR_SPEED_LIST] = self.speed_list
|
|
|
|
|
|
|
|
if (
|
|
|
|
self.supported_features & SUPPORT_SET_SPEED
|
|
|
|
or self.supported_features & SUPPORT_PRESET_MODE
|
|
|
|
):
|
|
|
|
attrs[ATTR_PRESET_MODES] = self.preset_modes
|
|
|
|
|
|
|
|
return attrs
|
|
|
|
|
2021-02-06 11:48:18 +00:00
|
|
|
@property
|
|
|
|
def _speed_list_without_preset_modes(self) -> list:
|
|
|
|
"""Return the speed list without preset modes.
|
|
|
|
|
|
|
|
This property provides forward and backwards
|
|
|
|
compatibility for conversion to percentage speeds.
|
|
|
|
"""
|
|
|
|
if not self._implemented_speed:
|
|
|
|
return LEGACY_SPEED_LIST
|
|
|
|
return speed_list_without_preset_modes(self.speed_list)
|
|
|
|
|
2021-01-27 23:44:36 +00:00
|
|
|
def speed_to_percentage(self, speed: str) -> int:
|
|
|
|
"""
|
|
|
|
Map a speed to a percentage.
|
|
|
|
|
|
|
|
Officially this should only have to deal with the 4 pre-defined speeds:
|
|
|
|
|
|
|
|
return {
|
|
|
|
SPEED_OFF: 0,
|
|
|
|
SPEED_LOW: 33,
|
|
|
|
SPEED_MEDIUM: 66,
|
|
|
|
SPEED_HIGH: 100,
|
|
|
|
}[speed]
|
|
|
|
|
|
|
|
Unfortunately lots of fans make up their own speeds. So the default
|
|
|
|
mapping is more dynamic.
|
|
|
|
"""
|
|
|
|
if speed in OFF_SPEED_VALUES:
|
|
|
|
return 0
|
|
|
|
|
2021-02-06 11:48:18 +00:00
|
|
|
speed_list = self._speed_list_without_preset_modes
|
2021-01-27 23:44:36 +00:00
|
|
|
|
|
|
|
if speed_list and speed not in speed_list:
|
|
|
|
raise NotValidSpeedError(f"The speed {speed} is not a valid speed.")
|
|
|
|
|
|
|
|
try:
|
|
|
|
return ordered_list_item_to_percentage(speed_list, speed)
|
|
|
|
except ValueError as ex:
|
|
|
|
raise NoValidSpeedsError(
|
|
|
|
f"The speed_list {speed_list} does not contain any valid speeds."
|
|
|
|
) from ex
|
|
|
|
|
|
|
|
def percentage_to_speed(self, percentage: int) -> str:
|
|
|
|
"""
|
|
|
|
Map a percentage onto self.speed_list.
|
|
|
|
|
|
|
|
Officially, this should only have to deal with 4 pre-defined speeds.
|
|
|
|
|
|
|
|
if value == 0:
|
|
|
|
return SPEED_OFF
|
|
|
|
elif value <= 33:
|
|
|
|
return SPEED_LOW
|
|
|
|
elif value <= 66:
|
|
|
|
return SPEED_MEDIUM
|
|
|
|
else:
|
|
|
|
return SPEED_HIGH
|
|
|
|
|
|
|
|
Unfortunately there is currently a high degree of non-conformancy.
|
|
|
|
Until fans have been corrected a more complicated and dynamic
|
|
|
|
mapping is used.
|
|
|
|
"""
|
|
|
|
if percentage == 0:
|
|
|
|
return SPEED_OFF
|
|
|
|
|
2021-02-06 11:48:18 +00:00
|
|
|
speed_list = self._speed_list_without_preset_modes
|
2021-01-27 23:44:36 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
return percentage_to_ordered_list_item(speed_list, percentage)
|
|
|
|
except ValueError as ex:
|
|
|
|
raise NoValidSpeedsError(
|
|
|
|
f"The speed_list {speed_list} does not contain any valid speeds."
|
|
|
|
) from ex
|
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
|
2021-01-27 23:44:36 +00:00
|
|
|
data[ATTR_PERCENTAGE] = self.percentage
|
2021-02-19 07:05:09 +00:00
|
|
|
data[ATTR_PERCENTAGE_STEP] = self.percentage_step
|
2021-01-27 23:44:36 +00:00
|
|
|
|
|
|
|
if (
|
|
|
|
supported_features & SUPPORT_PRESET_MODE
|
|
|
|
or supported_features & SUPPORT_SET_SPEED
|
|
|
|
):
|
|
|
|
data[ATTR_PRESET_MODE] = self.preset_mode
|
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
|
2021-01-27 23:44:36 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def preset_mode(self) -> Optional[str]:
|
|
|
|
"""Return the current preset mode, e.g., auto, smart, interval, favorite.
|
|
|
|
|
|
|
|
Requires SUPPORT_SET_SPEED.
|
|
|
|
"""
|
|
|
|
speed = self.speed
|
|
|
|
if speed in self.preset_modes:
|
|
|
|
return speed
|
|
|
|
return None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def preset_modes(self) -> Optional[List[str]]:
|
|
|
|
"""Return a list of available preset modes.
|
|
|
|
|
|
|
|
Requires SUPPORT_SET_SPEED.
|
|
|
|
"""
|
|
|
|
return preset_modes_from_speed_list(self.speed_list)
|
|
|
|
|
|
|
|
|
|
|
|
def speed_list_without_preset_modes(speed_list: List):
|
|
|
|
"""Filter out non-speeds from the speed list.
|
|
|
|
|
|
|
|
The goal is to get the speeds in a list from lowest to
|
|
|
|
highest by removing speeds that are not valid or out of order
|
|
|
|
so we can map them to percentages.
|
|
|
|
|
|
|
|
Examples:
|
|
|
|
input: ["off", "low", "low-medium", "medium", "medium-high", "high", "auto"]
|
|
|
|
output: ["low", "low-medium", "medium", "medium-high", "high"]
|
|
|
|
|
|
|
|
input: ["off", "auto", "low", "medium", "high"]
|
|
|
|
output: ["low", "medium", "high"]
|
|
|
|
|
|
|
|
input: ["off", "1", "2", "3", "4", "5", "6", "7", "smart"]
|
|
|
|
output: ["1", "2", "3", "4", "5", "6", "7"]
|
|
|
|
|
|
|
|
input: ["Auto", "Silent", "Favorite", "Idle", "Medium", "High", "Strong"]
|
2021-03-04 21:02:50 +00:00
|
|
|
output: ["Medium", "High", "Strong"]
|
2021-01-27 23:44:36 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
return [speed for speed in speed_list if speed.lower() not in _NOT_SPEEDS_FILTER]
|
|
|
|
|
|
|
|
|
|
|
|
def preset_modes_from_speed_list(speed_list: List):
|
|
|
|
"""Filter out non-preset modes from the speed list.
|
|
|
|
|
|
|
|
The goal is to return only preset modes.
|
|
|
|
|
|
|
|
Examples:
|
|
|
|
input: ["off", "low", "low-medium", "medium", "medium-high", "high", "auto"]
|
|
|
|
output: ["auto"]
|
|
|
|
|
|
|
|
input: ["off", "auto", "low", "medium", "high"]
|
|
|
|
output: ["auto"]
|
|
|
|
|
|
|
|
input: ["off", "1", "2", "3", "4", "5", "6", "7", "smart"]
|
|
|
|
output: ["smart"]
|
|
|
|
|
|
|
|
input: ["Auto", "Silent", "Favorite", "Idle", "Medium", "High", "Strong"]
|
2021-03-04 21:02:50 +00:00
|
|
|
output: ["Auto", "Silent", "Favorite", "Idle"]
|
2021-01-27 23:44:36 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
return [
|
|
|
|
speed
|
|
|
|
for speed in speed_list
|
|
|
|
if speed.lower() in _NOT_SPEEDS_FILTER and speed.lower() != SPEED_OFF
|
|
|
|
]
|