2020-05-17 23:54:32 +00:00
|
|
|
"""BleBox light entities implementation."""
|
2022-06-29 09:57:55 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
from datetime import timedelta
|
2020-05-17 23:54:32 +00:00
|
|
|
import logging
|
2022-08-19 07:54:13 +00:00
|
|
|
from typing import Any
|
2020-05-17 23:54:32 +00:00
|
|
|
|
2022-10-19 15:49:40 +00:00
|
|
|
from blebox_uniapi.box import Box
|
2022-06-29 09:57:55 +00:00
|
|
|
import blebox_uniapi.light
|
|
|
|
from blebox_uniapi.light import BleboxColorMode
|
2020-05-17 23:54:32 +00:00
|
|
|
|
|
|
|
from homeassistant.components.light import (
|
|
|
|
ATTR_BRIGHTNESS,
|
2022-06-29 09:57:55 +00:00
|
|
|
ATTR_COLOR_TEMP,
|
|
|
|
ATTR_EFFECT,
|
|
|
|
ATTR_RGB_COLOR,
|
2021-05-24 10:24:07 +00:00
|
|
|
ATTR_RGBW_COLOR,
|
2022-06-29 09:57:55 +00:00
|
|
|
ATTR_RGBWW_COLOR,
|
2022-04-23 05:49:39 +00:00
|
|
|
ColorMode,
|
2020-05-17 23:54:32 +00:00
|
|
|
LightEntity,
|
2022-06-29 09:57:55 +00:00
|
|
|
LightEntityFeature,
|
2020-05-17 23:54:32 +00:00
|
|
|
)
|
2022-01-03 14:02:21 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2020-05-17 23:54:32 +00:00
|
|
|
|
2022-10-19 15:49:40 +00:00
|
|
|
from . import BleBoxEntity
|
|
|
|
from .const import DOMAIN, PRODUCT
|
2020-05-17 23:54:32 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2022-06-29 09:57:55 +00:00
|
|
|
SCAN_INTERVAL = timedelta(seconds=5)
|
|
|
|
|
2020-05-17 23:54:32 +00:00
|
|
|
|
2022-01-03 14:02:21 +00:00
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config_entry: ConfigEntry,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
) -> None:
|
2020-05-17 23:54:32 +00:00
|
|
|
"""Set up a BleBox entry."""
|
2022-10-19 15:49:40 +00:00
|
|
|
product: Box = hass.data[DOMAIN][config_entry.entry_id][PRODUCT]
|
|
|
|
entities = [
|
|
|
|
BleBoxLightEntity(feature) for feature in product.features.get("lights", [])
|
|
|
|
]
|
|
|
|
async_add_entities(entities, True)
|
2020-05-17 23:54:32 +00:00
|
|
|
|
|
|
|
|
2022-06-29 09:57:55 +00:00
|
|
|
COLOR_MODE_MAP = {
|
|
|
|
BleboxColorMode.RGBW: ColorMode.RGBW,
|
|
|
|
BleboxColorMode.RGB: ColorMode.RGB,
|
|
|
|
BleboxColorMode.MONO: ColorMode.BRIGHTNESS,
|
|
|
|
BleboxColorMode.RGBorW: ColorMode.RGBW, # white hex is prioritised over RGB channel
|
|
|
|
BleboxColorMode.CT: ColorMode.COLOR_TEMP,
|
|
|
|
BleboxColorMode.CTx2: ColorMode.COLOR_TEMP, # two instances
|
|
|
|
BleboxColorMode.RGBWW: ColorMode.RGBWW,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-10-19 15:49:40 +00:00
|
|
|
class BleBoxLightEntity(BleBoxEntity[blebox_uniapi.light.Light], LightEntity):
|
2020-05-17 23:54:32 +00:00
|
|
|
"""Representation of BleBox lights."""
|
|
|
|
|
2022-08-19 08:58:51 +00:00
|
|
|
def __init__(self, feature: blebox_uniapi.light.Light) -> None:
|
2021-07-12 20:52:38 +00:00
|
|
|
"""Initialize a BleBox light."""
|
|
|
|
super().__init__(feature)
|
|
|
|
self._attr_supported_color_modes = {self.color_mode}
|
2022-08-19 08:58:51 +00:00
|
|
|
if feature.effect_list:
|
|
|
|
self._attr_supported_features = LightEntityFeature.EFFECT
|
2020-05-17 23:54:32 +00:00
|
|
|
|
|
|
|
@property
|
2021-07-12 20:52:38 +00:00
|
|
|
def is_on(self) -> bool:
|
2020-05-17 23:54:32 +00:00
|
|
|
"""Return if light is on."""
|
|
|
|
return self._feature.is_on
|
|
|
|
|
|
|
|
@property
|
|
|
|
def brightness(self):
|
|
|
|
"""Return the name."""
|
|
|
|
return self._feature.brightness
|
|
|
|
|
2022-06-29 09:57:55 +00:00
|
|
|
@property
|
|
|
|
def color_temp(self):
|
|
|
|
"""Return color temperature."""
|
|
|
|
return self._feature.color_temp
|
|
|
|
|
2020-05-17 23:54:32 +00:00
|
|
|
@property
|
2021-05-24 10:24:07 +00:00
|
|
|
def color_mode(self):
|
2022-06-29 09:57:55 +00:00
|
|
|
"""Return the color mode.
|
|
|
|
|
|
|
|
Set values to _attr_ibutes if needed.
|
|
|
|
"""
|
|
|
|
color_mode_tmp = COLOR_MODE_MAP.get(self._feature.color_mode, ColorMode.ONOFF)
|
|
|
|
if color_mode_tmp == ColorMode.COLOR_TEMP:
|
|
|
|
self._attr_min_mireds = 1
|
|
|
|
self._attr_max_mireds = 255
|
|
|
|
|
|
|
|
return color_mode_tmp
|
|
|
|
|
|
|
|
@property
|
2022-08-19 08:58:51 +00:00
|
|
|
def effect_list(self) -> list[str]:
|
2022-06-29 09:57:55 +00:00
|
|
|
"""Return the list of supported effects."""
|
|
|
|
return self._feature.effect_list
|
|
|
|
|
|
|
|
@property
|
|
|
|
def effect(self) -> str | None:
|
|
|
|
"""Return the current effect."""
|
|
|
|
return self._feature.effect
|
|
|
|
|
|
|
|
@property
|
|
|
|
def rgb_color(self):
|
|
|
|
"""Return value for rgb."""
|
|
|
|
if (rgb_hex := self._feature.rgb_hex) is None:
|
|
|
|
return None
|
|
|
|
return tuple(
|
|
|
|
blebox_uniapi.light.Light.normalise_elements_of_rgb(
|
|
|
|
blebox_uniapi.light.Light.rgb_hex_to_rgb_list(rgb_hex)[0:3]
|
|
|
|
)
|
|
|
|
)
|
2020-05-17 23:54:32 +00:00
|
|
|
|
|
|
|
@property
|
2021-05-24 10:24:07 +00:00
|
|
|
def rgbw_color(self):
|
2020-05-17 23:54:32 +00:00
|
|
|
"""Return the hue and saturation."""
|
2021-10-17 17:56:00 +00:00
|
|
|
if (rgbw_hex := self._feature.rgbw_hex) is None:
|
2020-05-17 23:54:32 +00:00
|
|
|
return None
|
2022-06-29 09:57:55 +00:00
|
|
|
return tuple(blebox_uniapi.light.Light.rgb_hex_to_rgb_list(rgbw_hex)[0:4])
|
2020-05-17 23:54:32 +00:00
|
|
|
|
2022-06-29 09:57:55 +00:00
|
|
|
@property
|
|
|
|
def rgbww_color(self):
|
|
|
|
"""Return value for rgbww."""
|
|
|
|
if (rgbww_hex := self._feature.rgbww_hex) is None:
|
|
|
|
return None
|
|
|
|
return tuple(blebox_uniapi.light.Light.rgb_hex_to_rgb_list(rgbww_hex))
|
2020-05-17 23:54:32 +00:00
|
|
|
|
2022-08-19 08:58:51 +00:00
|
|
|
async def async_turn_on(self, **kwargs: Any) -> None:
|
2020-05-17 23:54:32 +00:00
|
|
|
"""Turn the light on."""
|
|
|
|
|
2021-05-24 10:24:07 +00:00
|
|
|
rgbw = kwargs.get(ATTR_RGBW_COLOR)
|
2020-05-18 20:30:15 +00:00
|
|
|
brightness = kwargs.get(ATTR_BRIGHTNESS)
|
2022-06-29 09:57:55 +00:00
|
|
|
effect = kwargs.get(ATTR_EFFECT)
|
|
|
|
color_temp = kwargs.get(ATTR_COLOR_TEMP)
|
|
|
|
rgbww = kwargs.get(ATTR_RGBWW_COLOR)
|
2020-05-17 23:54:32 +00:00
|
|
|
feature = self._feature
|
|
|
|
value = feature.sensible_on_value
|
2022-06-29 09:57:55 +00:00
|
|
|
rgb = kwargs.get(ATTR_RGB_COLOR)
|
2020-05-17 23:54:32 +00:00
|
|
|
|
2021-05-24 10:24:07 +00:00
|
|
|
if rgbw is not None:
|
2022-06-29 09:57:55 +00:00
|
|
|
value = list(rgbw)
|
|
|
|
if color_temp is not None:
|
|
|
|
value = feature.return_color_temp_with_brightness(
|
|
|
|
int(color_temp), self.brightness
|
2020-05-17 23:54:32 +00:00
|
|
|
)
|
|
|
|
|
2022-06-29 09:57:55 +00:00
|
|
|
if rgbww is not None:
|
|
|
|
value = list(rgbww)
|
|
|
|
|
|
|
|
if rgb is not None:
|
|
|
|
if self.color_mode == ColorMode.RGB and brightness is None:
|
|
|
|
brightness = self.brightness
|
|
|
|
value = list(rgb)
|
|
|
|
|
|
|
|
if brightness is not None:
|
|
|
|
if self.color_mode == ATTR_COLOR_TEMP:
|
|
|
|
value = feature.return_color_temp_with_brightness(
|
|
|
|
self.color_temp, brightness
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
value = feature.apply_brightness(value, brightness)
|
|
|
|
|
2022-07-14 09:21:01 +00:00
|
|
|
try:
|
|
|
|
await self._feature.async_on(value)
|
|
|
|
except ValueError as exc:
|
|
|
|
raise ValueError(
|
|
|
|
f"Turning on '{self.name}' failed: Bad value {value}"
|
|
|
|
) from exc
|
|
|
|
|
2022-06-29 09:57:55 +00:00
|
|
|
if effect is not None:
|
|
|
|
try:
|
2022-07-14 09:21:01 +00:00
|
|
|
effect_value = self.effect_list.index(effect)
|
|
|
|
await self._feature.async_api_command("effect", effect_value)
|
2022-07-03 15:06:38 +00:00
|
|
|
except ValueError as exc:
|
|
|
|
raise ValueError(
|
2022-12-22 09:12:50 +00:00
|
|
|
f"Turning on with effect '{self.name}' failed: {effect} not in"
|
|
|
|
" effect list."
|
2022-07-03 15:06:38 +00:00
|
|
|
) from exc
|
2022-06-29 09:57:55 +00:00
|
|
|
|
2022-08-19 07:54:13 +00:00
|
|
|
async def async_turn_off(self, **kwargs: Any) -> None:
|
2020-05-17 23:54:32 +00:00
|
|
|
"""Turn the light off."""
|
|
|
|
await self._feature.async_off()
|