2020-08-28 15:33:34 +00:00
|
|
|
"""Light for Shelly."""
|
2021-03-18 13:31:38 +00:00
|
|
|
from __future__ import annotations
|
2020-09-07 15:29:51 +00:00
|
|
|
|
2021-05-04 16:10:28 +00:00
|
|
|
import asyncio
|
|
|
|
import logging
|
2021-07-21 17:11:44 +00:00
|
|
|
from typing import Any, Final, cast
|
2021-05-04 16:10:28 +00:00
|
|
|
|
2020-08-28 15:33:34 +00:00
|
|
|
from aioshelly import Block
|
2021-05-04 16:10:28 +00:00
|
|
|
import async_timeout
|
2020-08-28 15:33:34 +00:00
|
|
|
|
2020-09-07 15:29:51 +00:00
|
|
|
from homeassistant.components.light import (
|
|
|
|
ATTR_BRIGHTNESS,
|
|
|
|
ATTR_COLOR_TEMP,
|
2021-05-15 19:43:06 +00:00
|
|
|
ATTR_EFFECT,
|
2021-04-29 16:13:58 +00:00
|
|
|
ATTR_RGB_COLOR,
|
|
|
|
ATTR_RGBW_COLOR,
|
2021-08-09 17:55:14 +00:00
|
|
|
ATTR_TRANSITION,
|
2021-04-29 16:13:58 +00:00
|
|
|
COLOR_MODE_BRIGHTNESS,
|
|
|
|
COLOR_MODE_COLOR_TEMP,
|
|
|
|
COLOR_MODE_ONOFF,
|
|
|
|
COLOR_MODE_RGB,
|
|
|
|
COLOR_MODE_RGBW,
|
2021-05-15 19:43:06 +00:00
|
|
|
SUPPORT_EFFECT,
|
2021-08-09 17:55:14 +00:00
|
|
|
SUPPORT_TRANSITION,
|
2020-09-07 15:29:51 +00:00
|
|
|
LightEntity,
|
2021-04-29 16:13:58 +00:00
|
|
|
brightness_supported,
|
2020-09-07 15:29:51 +00:00
|
|
|
)
|
2021-07-21 17:11:44 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2020-09-07 15:29:51 +00:00
|
|
|
from homeassistant.util.color import (
|
|
|
|
color_temperature_kelvin_to_mired,
|
|
|
|
color_temperature_mired_to_kelvin,
|
|
|
|
)
|
2020-08-28 15:33:34 +00:00
|
|
|
|
2020-09-07 12:13:20 +00:00
|
|
|
from . import ShellyDeviceWrapper
|
2021-01-11 15:47:49 +00:00
|
|
|
from .const import (
|
2021-05-04 16:10:28 +00:00
|
|
|
AIOSHELLY_DEVICE_TIMEOUT_SEC,
|
2021-01-11 15:47:49 +00:00
|
|
|
COAP,
|
|
|
|
DATA_CONFIG_ENTRY,
|
|
|
|
DOMAIN,
|
2021-08-09 17:55:14 +00:00
|
|
|
FIRMWARE_PATTERN,
|
2021-01-11 15:47:49 +00:00
|
|
|
KELVIN_MAX_VALUE,
|
2021-02-26 22:28:16 +00:00
|
|
|
KELVIN_MIN_VALUE_COLOR,
|
|
|
|
KELVIN_MIN_VALUE_WHITE,
|
2021-08-09 17:55:14 +00:00
|
|
|
LIGHT_TRANSITION_MIN_FIRMWARE_DATE,
|
|
|
|
MAX_TRANSITION_TIME,
|
|
|
|
MODELS_SUPPORTING_LIGHT_TRANSITION,
|
2021-05-15 19:43:06 +00:00
|
|
|
SHBLB_1_RGB_EFFECTS,
|
|
|
|
STANDARD_RGB_EFFECTS,
|
2021-01-11 15:47:49 +00:00
|
|
|
)
|
2020-09-07 12:13:20 +00:00
|
|
|
from .entity import ShellyBlockEntity
|
2020-11-19 10:42:24 +00:00
|
|
|
from .utils import async_remove_shelly_entity
|
2020-08-28 15:33:34 +00:00
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
_LOGGER: Final = logging.getLogger(__name__)
|
2021-05-04 16:10:28 +00:00
|
|
|
|
2020-08-28 15:33:34 +00:00
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config_entry: ConfigEntry,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
) -> None:
|
2020-08-28 15:33:34 +00:00
|
|
|
"""Set up lights for device."""
|
2020-11-11 19:13:14 +00:00
|
|
|
wrapper = hass.data[DOMAIN][DATA_CONFIG_ENTRY][config_entry.entry_id][COAP]
|
2020-11-05 17:38:53 +00:00
|
|
|
|
|
|
|
blocks = []
|
|
|
|
for block in wrapper.device.blocks:
|
|
|
|
if block.type == "light":
|
|
|
|
blocks.append(block)
|
2020-11-12 09:12:56 +00:00
|
|
|
elif block.type == "relay":
|
|
|
|
appliance_type = wrapper.device.settings["relays"][int(block.channel)].get(
|
2020-11-05 17:38:53 +00:00
|
|
|
"appliance_type"
|
|
|
|
)
|
2020-11-12 09:12:56 +00:00
|
|
|
if appliance_type and appliance_type.lower() == "light":
|
|
|
|
blocks.append(block)
|
|
|
|
unique_id = (
|
|
|
|
f'{wrapper.device.shelly["mac"]}-{block.type}_{block.channel}'
|
|
|
|
)
|
2020-11-19 10:42:24 +00:00
|
|
|
await async_remove_shelly_entity(hass, "switch", unique_id)
|
2020-08-28 15:33:34 +00:00
|
|
|
|
|
|
|
if not blocks:
|
|
|
|
return
|
|
|
|
|
|
|
|
async_add_entities(ShellyLight(wrapper, block) for block in blocks)
|
|
|
|
|
|
|
|
|
|
|
|
class ShellyLight(ShellyBlockEntity, LightEntity):
|
|
|
|
"""Switch that controls a relay block on Shelly devices."""
|
|
|
|
|
|
|
|
def __init__(self, wrapper: ShellyDeviceWrapper, block: Block) -> None:
|
|
|
|
"""Initialize light."""
|
|
|
|
super().__init__(wrapper, block)
|
2021-07-21 17:11:44 +00:00
|
|
|
self.control_result: dict[str, Any] | None = None
|
|
|
|
self.mode_result: dict[str, Any] | None = None
|
|
|
|
self._supported_color_modes: set[str] = set()
|
|
|
|
self._supported_features: int = 0
|
|
|
|
self._min_kelvin: int = KELVIN_MIN_VALUE_WHITE
|
|
|
|
self._max_kelvin: int = KELVIN_MAX_VALUE
|
2021-01-11 15:47:49 +00:00
|
|
|
|
|
|
|
if hasattr(block, "red") and hasattr(block, "green") and hasattr(block, "blue"):
|
2021-02-26 22:28:16 +00:00
|
|
|
self._min_kelvin = KELVIN_MIN_VALUE_COLOR
|
2021-04-29 16:13:58 +00:00
|
|
|
if hasattr(block, "white"):
|
|
|
|
self._supported_color_modes.add(COLOR_MODE_RGBW)
|
2021-05-01 14:43:03 +00:00
|
|
|
else:
|
|
|
|
self._supported_color_modes.add(COLOR_MODE_RGB)
|
2020-09-07 15:29:51 +00:00
|
|
|
|
2021-04-29 16:13:58 +00:00
|
|
|
if hasattr(block, "colorTemp"):
|
|
|
|
self._supported_color_modes.add(COLOR_MODE_COLOR_TEMP)
|
|
|
|
|
|
|
|
if not self._supported_color_modes:
|
|
|
|
if hasattr(block, "brightness") or hasattr(block, "gain"):
|
|
|
|
self._supported_color_modes.add(COLOR_MODE_BRIGHTNESS)
|
|
|
|
else:
|
|
|
|
self._supported_color_modes.add(COLOR_MODE_ONOFF)
|
2020-08-28 15:33:34 +00:00
|
|
|
|
2021-05-15 19:43:06 +00:00
|
|
|
if hasattr(block, "effect"):
|
|
|
|
self._supported_features |= SUPPORT_EFFECT
|
|
|
|
|
2021-08-09 17:55:14 +00:00
|
|
|
if wrapper.model in MODELS_SUPPORTING_LIGHT_TRANSITION:
|
|
|
|
match = FIRMWARE_PATTERN.search(wrapper.device.settings.get("fw"))
|
|
|
|
if (
|
|
|
|
match is not None
|
|
|
|
and int(match[0]) >= LIGHT_TRANSITION_MIN_FIRMWARE_DATE
|
|
|
|
):
|
|
|
|
self._supported_features |= SUPPORT_TRANSITION
|
|
|
|
|
2021-05-15 19:43:06 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self) -> int:
|
|
|
|
"""Supported features."""
|
|
|
|
return self._supported_features
|
|
|
|
|
2020-08-28 15:33:34 +00:00
|
|
|
@property
|
|
|
|
def is_on(self) -> bool:
|
|
|
|
"""If light is on."""
|
|
|
|
if self.control_result:
|
2021-07-21 17:11:44 +00:00
|
|
|
return cast(bool, self.control_result["ison"])
|
2020-08-28 15:33:34 +00:00
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
return bool(self.block.output)
|
2020-08-28 15:33:34 +00:00
|
|
|
|
2021-01-11 15:47:49 +00:00
|
|
|
@property
|
2021-07-21 17:11:44 +00:00
|
|
|
def mode(self) -> str:
|
2021-01-11 15:47:49 +00:00
|
|
|
"""Return the color mode of the light."""
|
|
|
|
if self.mode_result:
|
2021-07-21 17:11:44 +00:00
|
|
|
return cast(str, self.mode_result["mode"])
|
2021-01-11 15:47:49 +00:00
|
|
|
|
|
|
|
if hasattr(self.block, "mode"):
|
2021-07-21 17:11:44 +00:00
|
|
|
return cast(str, self.block.mode)
|
2021-01-11 15:47:49 +00:00
|
|
|
|
|
|
|
if (
|
|
|
|
hasattr(self.block, "red")
|
|
|
|
and hasattr(self.block, "green")
|
|
|
|
and hasattr(self.block, "blue")
|
|
|
|
):
|
|
|
|
return "color"
|
|
|
|
|
|
|
|
return "white"
|
|
|
|
|
2020-08-28 15:33:34 +00:00
|
|
|
@property
|
2021-07-21 17:11:44 +00:00
|
|
|
def brightness(self) -> int:
|
2021-04-29 16:13:58 +00:00
|
|
|
"""Return the brightness of this light between 0..255."""
|
2021-01-11 15:47:49 +00:00
|
|
|
if self.mode == "color":
|
|
|
|
if self.control_result:
|
2021-04-12 22:52:51 +00:00
|
|
|
brightness_pct = self.control_result["gain"]
|
2021-01-11 15:47:49 +00:00
|
|
|
else:
|
2021-04-12 22:52:51 +00:00
|
|
|
brightness_pct = self.block.gain
|
2020-08-28 15:33:34 +00:00
|
|
|
else:
|
2021-01-11 15:47:49 +00:00
|
|
|
if self.control_result:
|
2021-04-12 22:52:51 +00:00
|
|
|
brightness_pct = self.control_result["brightness"]
|
2021-01-11 15:47:49 +00:00
|
|
|
else:
|
2021-04-12 22:52:51 +00:00
|
|
|
brightness_pct = self.block.brightness
|
|
|
|
|
|
|
|
return round(255 * brightness_pct / 100)
|
2020-08-28 15:33:34 +00:00
|
|
|
|
2021-01-11 15:47:49 +00:00
|
|
|
@property
|
2021-07-21 17:11:44 +00:00
|
|
|
def color_mode(self) -> str:
|
2021-04-29 16:13:58 +00:00
|
|
|
"""Return the color mode of the light."""
|
|
|
|
if self.mode == "color":
|
|
|
|
if hasattr(self.block, "white"):
|
|
|
|
return COLOR_MODE_RGBW
|
|
|
|
return COLOR_MODE_RGB
|
2021-01-11 15:47:49 +00:00
|
|
|
|
2021-04-29 16:13:58 +00:00
|
|
|
if hasattr(self.block, "colorTemp"):
|
|
|
|
return COLOR_MODE_COLOR_TEMP
|
2021-01-11 15:47:49 +00:00
|
|
|
|
2021-04-29 16:13:58 +00:00
|
|
|
if hasattr(self.block, "brightness") or hasattr(self.block, "gain"):
|
|
|
|
return COLOR_MODE_BRIGHTNESS
|
|
|
|
|
|
|
|
return COLOR_MODE_ONOFF
|
|
|
|
|
|
|
|
@property
|
2021-05-01 14:43:03 +00:00
|
|
|
def rgb_color(self) -> tuple[int, int, int]:
|
2021-04-29 16:13:58 +00:00
|
|
|
"""Return the rgb color value [int, int, int]."""
|
2021-01-11 15:47:49 +00:00
|
|
|
if self.control_result:
|
|
|
|
red = self.control_result["red"]
|
|
|
|
green = self.control_result["green"]
|
|
|
|
blue = self.control_result["blue"]
|
|
|
|
else:
|
|
|
|
red = self.block.red
|
|
|
|
green = self.block.green
|
|
|
|
blue = self.block.blue
|
2021-05-01 14:43:03 +00:00
|
|
|
return (red, green, blue)
|
2021-04-29 16:13:58 +00:00
|
|
|
|
|
|
|
@property
|
2021-05-01 14:43:03 +00:00
|
|
|
def rgbw_color(self) -> tuple[int, int, int, int]:
|
2021-04-29 16:13:58 +00:00
|
|
|
"""Return the rgbw color value [int, int, int, int]."""
|
|
|
|
if self.control_result:
|
|
|
|
white = self.control_result["white"]
|
|
|
|
else:
|
|
|
|
white = self.block.white
|
|
|
|
|
2021-05-01 14:43:03 +00:00
|
|
|
return (*self.rgb_color, white)
|
2021-01-11 15:47:49 +00:00
|
|
|
|
2020-08-28 15:33:34 +00:00
|
|
|
@property
|
2021-07-21 17:11:44 +00:00
|
|
|
def color_temp(self) -> int:
|
2020-09-07 15:29:51 +00:00
|
|
|
"""Return the CT color value in mireds."""
|
|
|
|
if self.control_result:
|
|
|
|
color_temp = self.control_result["temp"]
|
|
|
|
else:
|
|
|
|
color_temp = self.block.colorTemp
|
|
|
|
|
2021-02-26 22:28:16 +00:00
|
|
|
color_temp = min(self._max_kelvin, max(self._min_kelvin, color_temp))
|
2020-09-07 15:29:51 +00:00
|
|
|
|
|
|
|
return int(color_temperature_kelvin_to_mired(color_temp))
|
|
|
|
|
|
|
|
@property
|
2021-02-23 21:42:33 +00:00
|
|
|
def min_mireds(self) -> int:
|
2020-09-07 15:29:51 +00:00
|
|
|
"""Return the coldest color_temp that this light supports."""
|
2021-02-26 22:28:16 +00:00
|
|
|
return int(color_temperature_kelvin_to_mired(self._max_kelvin))
|
2020-09-07 15:29:51 +00:00
|
|
|
|
|
|
|
@property
|
2021-02-23 21:42:33 +00:00
|
|
|
def max_mireds(self) -> int:
|
2020-09-07 15:29:51 +00:00
|
|
|
"""Return the warmest color_temp that this light supports."""
|
2021-02-26 22:28:16 +00:00
|
|
|
return int(color_temperature_kelvin_to_mired(self._min_kelvin))
|
2020-08-28 15:33:34 +00:00
|
|
|
|
2021-04-29 16:13:58 +00:00
|
|
|
@property
|
|
|
|
def supported_color_modes(self) -> set | None:
|
|
|
|
"""Flag supported color modes."""
|
|
|
|
return self._supported_color_modes
|
|
|
|
|
2021-05-15 19:43:06 +00:00
|
|
|
@property
|
|
|
|
def effect_list(self) -> list[str] | None:
|
|
|
|
"""Return the list of supported effects."""
|
|
|
|
if not self.supported_features & SUPPORT_EFFECT:
|
|
|
|
return None
|
|
|
|
|
|
|
|
if self.wrapper.model == "SHBLB-1":
|
|
|
|
return list(SHBLB_1_RGB_EFFECTS.values())
|
|
|
|
|
|
|
|
return list(STANDARD_RGB_EFFECTS.values())
|
|
|
|
|
|
|
|
@property
|
|
|
|
def effect(self) -> str | None:
|
|
|
|
"""Return the current effect."""
|
|
|
|
if not self.supported_features & SUPPORT_EFFECT:
|
|
|
|
return None
|
|
|
|
|
|
|
|
if self.control_result:
|
|
|
|
effect_index = self.control_result["effect"]
|
|
|
|
else:
|
|
|
|
effect_index = self.block.effect
|
|
|
|
|
|
|
|
if self.wrapper.model == "SHBLB-1":
|
|
|
|
return SHBLB_1_RGB_EFFECTS[effect_index]
|
|
|
|
|
|
|
|
return STANDARD_RGB_EFFECTS[effect_index]
|
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
async def async_turn_on(self, **kwargs: Any) -> None:
|
2020-08-28 15:33:34 +00:00
|
|
|
"""Turn on light."""
|
2021-02-11 09:40:03 +00:00
|
|
|
if self.block.type == "relay":
|
2021-05-04 16:10:28 +00:00
|
|
|
self.control_result = await self.set_state(turn="on")
|
2021-02-11 09:40:03 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
return
|
|
|
|
|
2021-02-26 22:28:16 +00:00
|
|
|
set_mode = None
|
2021-04-29 16:13:58 +00:00
|
|
|
supported_color_modes = self._supported_color_modes
|
2021-05-09 17:46:53 +00:00
|
|
|
params: dict[str, Any] = {"turn": "on"}
|
2021-04-29 16:13:58 +00:00
|
|
|
|
2021-08-09 17:55:14 +00:00
|
|
|
if ATTR_TRANSITION in kwargs:
|
|
|
|
params["transition"] = min(
|
|
|
|
int(kwargs[ATTR_TRANSITION] * 1000), MAX_TRANSITION_TIME
|
|
|
|
)
|
|
|
|
|
2021-04-29 16:13:58 +00:00
|
|
|
if ATTR_BRIGHTNESS in kwargs and brightness_supported(supported_color_modes):
|
2021-04-12 22:52:51 +00:00
|
|
|
brightness_pct = int(100 * (kwargs[ATTR_BRIGHTNESS] + 1) / 255)
|
2021-01-11 15:47:49 +00:00
|
|
|
if hasattr(self.block, "gain"):
|
2021-04-12 22:52:51 +00:00
|
|
|
params["gain"] = brightness_pct
|
2021-01-11 15:47:49 +00:00
|
|
|
if hasattr(self.block, "brightness"):
|
2021-04-12 22:52:51 +00:00
|
|
|
params["brightness"] = brightness_pct
|
2021-04-29 16:13:58 +00:00
|
|
|
|
|
|
|
if ATTR_COLOR_TEMP in kwargs and COLOR_MODE_COLOR_TEMP in supported_color_modes:
|
2020-09-07 15:29:51 +00:00
|
|
|
color_temp = color_temperature_mired_to_kelvin(kwargs[ATTR_COLOR_TEMP])
|
2021-02-26 22:28:16 +00:00
|
|
|
color_temp = min(self._max_kelvin, max(self._min_kelvin, color_temp))
|
2021-01-11 15:47:49 +00:00
|
|
|
# Color temperature change - used only in white mode, switch device mode to white
|
2021-02-26 22:28:16 +00:00
|
|
|
set_mode = "white"
|
2020-09-07 15:29:51 +00:00
|
|
|
params["temp"] = int(color_temp)
|
2021-04-29 16:13:58 +00:00
|
|
|
|
|
|
|
if ATTR_RGB_COLOR in kwargs and COLOR_MODE_RGB in supported_color_modes:
|
2021-01-11 15:47:49 +00:00
|
|
|
# Color channels change - used only in color mode, switch device mode to color
|
2021-02-26 22:28:16 +00:00
|
|
|
set_mode = "color"
|
2021-04-29 16:13:58 +00:00
|
|
|
(params["red"], params["green"], params["blue"]) = kwargs[ATTR_RGB_COLOR]
|
|
|
|
|
|
|
|
if ATTR_RGBW_COLOR in kwargs and COLOR_MODE_RGBW in supported_color_modes:
|
|
|
|
# Color channels change - used only in color mode, switch device mode to color
|
2021-02-26 22:28:16 +00:00
|
|
|
set_mode = "color"
|
2021-04-29 16:13:58 +00:00
|
|
|
(params["red"], params["green"], params["blue"], params["white"]) = kwargs[
|
|
|
|
ATTR_RGBW_COLOR
|
|
|
|
]
|
2021-02-26 22:28:16 +00:00
|
|
|
|
2021-05-15 19:43:06 +00:00
|
|
|
if ATTR_EFFECT in kwargs:
|
|
|
|
# Color effect change - used only in color mode, switch device mode to color
|
|
|
|
set_mode = "color"
|
|
|
|
if self.wrapper.model == "SHBLB-1":
|
|
|
|
effect_dict = SHBLB_1_RGB_EFFECTS
|
|
|
|
else:
|
|
|
|
effect_dict = STANDARD_RGB_EFFECTS
|
|
|
|
if kwargs[ATTR_EFFECT] in effect_dict.values():
|
|
|
|
params["effect"] = [
|
|
|
|
k for k, v in effect_dict.items() if v == kwargs[ATTR_EFFECT]
|
|
|
|
][0]
|
|
|
|
else:
|
|
|
|
_LOGGER.error(
|
|
|
|
"Effect '%s' not supported by device %s",
|
|
|
|
kwargs[ATTR_EFFECT],
|
|
|
|
self.wrapper.model,
|
|
|
|
)
|
|
|
|
|
2021-05-04 16:10:28 +00:00
|
|
|
if await self.set_light_mode(set_mode):
|
|
|
|
self.control_result = await self.set_state(**params)
|
2021-02-26 22:28:16 +00:00
|
|
|
|
2020-08-28 15:33:34 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
async def async_turn_off(self, **kwargs: Any) -> None:
|
2020-08-28 15:33:34 +00:00
|
|
|
"""Turn off light."""
|
2021-08-09 17:55:14 +00:00
|
|
|
params: dict[str, Any] = {"turn": "off"}
|
|
|
|
|
|
|
|
if ATTR_TRANSITION in kwargs:
|
|
|
|
params["transition"] = min(
|
|
|
|
int(kwargs[ATTR_TRANSITION] * 1000), MAX_TRANSITION_TIME
|
|
|
|
)
|
|
|
|
|
|
|
|
self.control_result = await self.set_state(**params)
|
|
|
|
|
2020-08-28 15:33:34 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
|
2021-07-21 17:11:44 +00:00
|
|
|
async def set_light_mode(self, set_mode: str | None) -> bool:
|
2021-05-04 16:10:28 +00:00
|
|
|
"""Change device mode color/white if mode has changed."""
|
|
|
|
if set_mode is None or self.mode == set_mode:
|
|
|
|
return True
|
|
|
|
|
|
|
|
_LOGGER.debug("Setting light mode for entity %s, mode: %s", self.name, set_mode)
|
|
|
|
try:
|
|
|
|
async with async_timeout.timeout(AIOSHELLY_DEVICE_TIMEOUT_SEC):
|
|
|
|
self.mode_result = await self.wrapper.device.switch_light_mode(set_mode)
|
|
|
|
except (asyncio.TimeoutError, OSError) as err:
|
|
|
|
_LOGGER.error(
|
|
|
|
"Setting light mode for entity %s failed, state: %s, error: %s",
|
|
|
|
self.name,
|
|
|
|
set_mode,
|
|
|
|
repr(err),
|
|
|
|
)
|
|
|
|
self.wrapper.last_update_success = False
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
2020-08-28 15:33:34 +00:00
|
|
|
@callback
|
2021-07-21 17:11:44 +00:00
|
|
|
def _update_callback(self) -> None:
|
2021-01-11 15:47:49 +00:00
|
|
|
"""When device updates, clear control & mode result that overrides state."""
|
2020-08-28 15:33:34 +00:00
|
|
|
self.control_result = None
|
2021-01-11 15:47:49 +00:00
|
|
|
self.mode_result = None
|
2020-08-28 15:33:34 +00:00
|
|
|
super()._update_callback()
|