mirror of https://github.com/home-assistant/core
396 lines
12 KiB
Python
396 lines
12 KiB
Python
"""Support for deCONZ lights."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from typing import Any, TypedDict, cast
|
|
|
|
from pydeconz.interfaces.groups import GroupHandler
|
|
from pydeconz.interfaces.lights import LightHandler
|
|
from pydeconz.models.event import EventType
|
|
from pydeconz.models.group import Group, TypedGroupAction
|
|
from pydeconz.models.light.light import Light, LightAlert, LightColorMode, LightEffect
|
|
|
|
from homeassistant.components.light import (
|
|
ATTR_BRIGHTNESS,
|
|
ATTR_COLOR_TEMP,
|
|
ATTR_EFFECT,
|
|
ATTR_FLASH,
|
|
ATTR_HS_COLOR,
|
|
ATTR_TRANSITION,
|
|
ATTR_XY_COLOR,
|
|
DOMAIN as LIGHT_DOMAIN,
|
|
EFFECT_COLORLOOP,
|
|
FLASH_LONG,
|
|
FLASH_SHORT,
|
|
ColorMode,
|
|
LightEntity,
|
|
LightEntityFeature,
|
|
)
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.core import HomeAssistant, callback
|
|
from homeassistant.helpers.device_registry import DeviceInfo
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.util.color import color_hs_to_xy
|
|
|
|
from .const import DOMAIN as DECONZ_DOMAIN, POWER_PLUGS
|
|
from .entity import DeconzDevice
|
|
from .hub import DeconzHub
|
|
|
|
DECONZ_GROUP = "is_deconz_group"
|
|
EFFECT_TO_DECONZ = {
|
|
EFFECT_COLORLOOP: LightEffect.COLOR_LOOP,
|
|
"none": LightEffect.NONE,
|
|
# Specific to Philips Hue
|
|
"candle": LightEffect.CANDLE,
|
|
"cosmos": LightEffect.COSMOS,
|
|
"enchant": LightEffect.ENCHANT,
|
|
"fire": LightEffect.FIRE,
|
|
"fireplace": LightEffect.FIREPLACE,
|
|
"glisten": LightEffect.GLISTEN,
|
|
"loop": LightEffect.LOOP,
|
|
"opal": LightEffect.OPAL,
|
|
"prism": LightEffect.PRISM,
|
|
"sparkle": LightEffect.SPARKLE,
|
|
"sunbeam": LightEffect.SUNBEAM,
|
|
"sunrise": LightEffect.SUNRISE,
|
|
"sunset": LightEffect.SUNSET,
|
|
"underwater": LightEffect.UNDERWATER,
|
|
# Specific to Lidl christmas light
|
|
"carnival": LightEffect.CARNIVAL,
|
|
"collide": LightEffect.COLLIDE,
|
|
"fading": LightEffect.FADING,
|
|
"fireworks": LightEffect.FIREWORKS,
|
|
"flag": LightEffect.FLAG,
|
|
"glow": LightEffect.GLOW,
|
|
"rainbow": LightEffect.RAINBOW,
|
|
"snake": LightEffect.SNAKE,
|
|
"snow": LightEffect.SNOW,
|
|
"sparkles": LightEffect.SPARKLES,
|
|
"steady": LightEffect.STEADY,
|
|
"strobe": LightEffect.STROBE,
|
|
"twinkle": LightEffect.TWINKLE,
|
|
"updown": LightEffect.UPDOWN,
|
|
"vintage": LightEffect.VINTAGE,
|
|
"waves": LightEffect.WAVES,
|
|
}
|
|
FLASH_TO_DECONZ = {FLASH_SHORT: LightAlert.SHORT, FLASH_LONG: LightAlert.LONG}
|
|
|
|
DECONZ_TO_COLOR_MODE = {
|
|
LightColorMode.CT: ColorMode.COLOR_TEMP,
|
|
LightColorMode.GRADIENT: ColorMode.XY,
|
|
LightColorMode.HS: ColorMode.HS,
|
|
LightColorMode.XY: ColorMode.XY,
|
|
}
|
|
|
|
XMAS_LIGHT_EFFECTS = [
|
|
"carnival",
|
|
"collide",
|
|
"fading",
|
|
"fireworks",
|
|
"flag",
|
|
"glow",
|
|
"rainbow",
|
|
"snake",
|
|
"snow",
|
|
"sparkles",
|
|
"steady",
|
|
"strobe",
|
|
"twinkle",
|
|
"updown",
|
|
"vintage",
|
|
"waves",
|
|
]
|
|
|
|
|
|
class SetStateAttributes(TypedDict, total=False):
|
|
"""Attributes available with set state call."""
|
|
|
|
alert: LightAlert
|
|
brightness: int
|
|
color_temperature: int
|
|
effect: LightEffect
|
|
hue: int
|
|
on: bool
|
|
saturation: int
|
|
transition_time: int
|
|
xy: tuple[float, float]
|
|
|
|
|
|
def update_color_state(
|
|
group: Group, lights: list[Light], override: bool = False
|
|
) -> None:
|
|
"""Sync group color state with light."""
|
|
data = {
|
|
attribute: light_attribute
|
|
for light in lights
|
|
for attribute in ("bri", "ct", "hue", "sat", "xy", "colormode", "effect")
|
|
if (light_attribute := light.raw["state"].get(attribute)) is not None
|
|
}
|
|
|
|
if override:
|
|
group.raw["action"] = cast(TypedGroupAction, data)
|
|
else:
|
|
group.update(cast(dict[str, dict[str, Any]], {"action": data}))
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant,
|
|
config_entry: ConfigEntry,
|
|
async_add_entities: AddEntitiesCallback,
|
|
) -> None:
|
|
"""Set up the deCONZ lights and groups from a config entry."""
|
|
hub = DeconzHub.get_hub(hass, config_entry)
|
|
hub.entities[LIGHT_DOMAIN] = set()
|
|
|
|
@callback
|
|
def async_add_light(_: EventType, light_id: str) -> None:
|
|
"""Add light from deCONZ."""
|
|
light = hub.api.lights.lights[light_id]
|
|
if light.type in POWER_PLUGS:
|
|
return
|
|
|
|
async_add_entities([DeconzLight(light, hub)])
|
|
|
|
hub.register_platform_add_device_callback(
|
|
async_add_light,
|
|
hub.api.lights.lights,
|
|
)
|
|
|
|
@callback
|
|
def async_add_group(_: EventType, group_id: str) -> None:
|
|
"""Add group from deCONZ.
|
|
|
|
Update group states based on its sum of related lights.
|
|
"""
|
|
if (group := hub.api.groups[group_id]) and not group.lights:
|
|
return
|
|
|
|
lights = [
|
|
light
|
|
for light_id in group.lights
|
|
if (light := hub.api.lights.lights.get(light_id)) and light.reachable
|
|
]
|
|
update_color_state(group, lights, True)
|
|
|
|
async_add_entities([DeconzGroup(group, hub)])
|
|
|
|
hub.register_platform_add_device_callback(
|
|
async_add_group,
|
|
hub.api.groups,
|
|
)
|
|
|
|
|
|
class DeconzBaseLight[_LightDeviceT: Group | Light](
|
|
DeconzDevice[_LightDeviceT], LightEntity
|
|
):
|
|
"""Representation of a deCONZ light."""
|
|
|
|
TYPE = LIGHT_DOMAIN
|
|
_attr_color_mode = ColorMode.UNKNOWN
|
|
|
|
def __init__(self, device: _LightDeviceT, hub: DeconzHub) -> None:
|
|
"""Set up light."""
|
|
super().__init__(device, hub)
|
|
|
|
self.api: GroupHandler | LightHandler
|
|
if isinstance(self._device, Light):
|
|
self.api = self.hub.api.lights.lights
|
|
elif isinstance(self._device, Group):
|
|
self.api = self.hub.api.groups
|
|
|
|
self._attr_supported_color_modes: set[ColorMode] = set()
|
|
|
|
if device.color_temp is not None:
|
|
self._attr_supported_color_modes.add(ColorMode.COLOR_TEMP)
|
|
|
|
if device.hue is not None and device.saturation is not None:
|
|
self._attr_supported_color_modes.add(ColorMode.HS)
|
|
|
|
if device.xy is not None:
|
|
self._attr_supported_color_modes.add(ColorMode.XY)
|
|
|
|
if not self._attr_supported_color_modes and device.brightness is not None:
|
|
self._attr_supported_color_modes.add(ColorMode.BRIGHTNESS)
|
|
|
|
if not self._attr_supported_color_modes:
|
|
self._attr_supported_color_modes.add(ColorMode.ONOFF)
|
|
|
|
if device.brightness is not None:
|
|
self._attr_supported_features |= (
|
|
LightEntityFeature.FLASH | LightEntityFeature.TRANSITION
|
|
)
|
|
|
|
if device.effect is not None:
|
|
self._attr_supported_features |= LightEntityFeature.EFFECT
|
|
self._attr_effect_list = [EFFECT_COLORLOOP]
|
|
|
|
# For lights that report supported effects.
|
|
if isinstance(device, Light):
|
|
if device.supported_effects is not None:
|
|
self._attr_effect_list = [
|
|
EFFECT_TO_DECONZ[el]
|
|
for el in device.supported_effects
|
|
if el in EFFECT_TO_DECONZ
|
|
]
|
|
if device.model_id in ("HG06467", "TS0601"):
|
|
self._attr_effect_list = XMAS_LIGHT_EFFECTS
|
|
|
|
@property
|
|
def color_mode(self) -> str | None:
|
|
"""Return the color mode of the light."""
|
|
if self._device.color_mode in DECONZ_TO_COLOR_MODE:
|
|
color_mode = DECONZ_TO_COLOR_MODE[self._device.color_mode]
|
|
elif self._device.brightness is not None:
|
|
color_mode = ColorMode.BRIGHTNESS
|
|
else:
|
|
color_mode = ColorMode.ONOFF
|
|
if color_mode not in self._attr_supported_color_modes:
|
|
# Some lights controlled by ZigBee scenes can get unsupported color mode
|
|
return self._attr_color_mode
|
|
self._attr_color_mode = color_mode
|
|
return color_mode
|
|
|
|
@property
|
|
def brightness(self) -> int | None:
|
|
"""Return the brightness of this light between 0..255."""
|
|
return self._device.brightness
|
|
|
|
@property
|
|
def color_temp(self) -> int | None:
|
|
"""Return the CT color value."""
|
|
return self._device.color_temp
|
|
|
|
@property
|
|
def hs_color(self) -> tuple[float, float] | None:
|
|
"""Return the hs color value."""
|
|
if (hue := self._device.hue) and (sat := self._device.saturation):
|
|
return (hue / 65535 * 360, sat / 255 * 100)
|
|
return None
|
|
|
|
@property
|
|
def xy_color(self) -> tuple[float, float] | None:
|
|
"""Return the XY color value."""
|
|
return self._device.xy
|
|
|
|
@property
|
|
def is_on(self) -> bool | None:
|
|
"""Return true if light is on."""
|
|
return self._device.state
|
|
|
|
async def async_turn_on(self, **kwargs: Any) -> None:
|
|
"""Turn on light."""
|
|
data: SetStateAttributes = {"on": True}
|
|
|
|
if ATTR_BRIGHTNESS in kwargs:
|
|
data["brightness"] = kwargs[ATTR_BRIGHTNESS]
|
|
|
|
if ATTR_COLOR_TEMP in kwargs:
|
|
data["color_temperature"] = kwargs[ATTR_COLOR_TEMP]
|
|
|
|
if ATTR_HS_COLOR in kwargs:
|
|
if ColorMode.XY in self._attr_supported_color_modes:
|
|
data["xy"] = color_hs_to_xy(*kwargs[ATTR_HS_COLOR])
|
|
else:
|
|
data["hue"] = int(kwargs[ATTR_HS_COLOR][0] / 360 * 65535)
|
|
data["saturation"] = int(kwargs[ATTR_HS_COLOR][1] / 100 * 255)
|
|
|
|
if ATTR_XY_COLOR in kwargs:
|
|
data["xy"] = kwargs[ATTR_XY_COLOR]
|
|
|
|
if ATTR_TRANSITION in kwargs:
|
|
data["transition_time"] = int(kwargs[ATTR_TRANSITION] * 10)
|
|
elif "IKEA" in self._device.manufacturer:
|
|
data["transition_time"] = 0
|
|
|
|
if ATTR_FLASH in kwargs and kwargs[ATTR_FLASH] in FLASH_TO_DECONZ:
|
|
data["alert"] = FLASH_TO_DECONZ[kwargs[ATTR_FLASH]]
|
|
del data["on"]
|
|
|
|
if ATTR_EFFECT in kwargs and kwargs[ATTR_EFFECT] in EFFECT_TO_DECONZ:
|
|
data["effect"] = EFFECT_TO_DECONZ[kwargs[ATTR_EFFECT]]
|
|
|
|
await self.api.set_state(id=self._device.resource_id, **data)
|
|
|
|
async def async_turn_off(self, **kwargs: Any) -> None:
|
|
"""Turn off light."""
|
|
if not self._device.state:
|
|
return
|
|
|
|
data: SetStateAttributes = {"on": False}
|
|
|
|
if ATTR_TRANSITION in kwargs:
|
|
data["brightness"] = 0
|
|
data["transition_time"] = int(kwargs[ATTR_TRANSITION] * 10)
|
|
|
|
if ATTR_FLASH in kwargs and kwargs[ATTR_FLASH] in FLASH_TO_DECONZ:
|
|
data["alert"] = FLASH_TO_DECONZ[kwargs[ATTR_FLASH]]
|
|
del data["on"]
|
|
|
|
await self.api.set_state(id=self._device.resource_id, **data)
|
|
|
|
@property
|
|
def extra_state_attributes(self) -> dict[str, bool]:
|
|
"""Return the device state attributes."""
|
|
return {DECONZ_GROUP: isinstance(self._device, Group)}
|
|
|
|
|
|
class DeconzLight(DeconzBaseLight[Light]):
|
|
"""Representation of a deCONZ light."""
|
|
|
|
@property
|
|
def max_mireds(self) -> int:
|
|
"""Return the warmest color_temp that this light supports."""
|
|
return self._device.max_color_temp or super().max_mireds
|
|
|
|
@property
|
|
def min_mireds(self) -> int:
|
|
"""Return the coldest color_temp that this light supports."""
|
|
return self._device.min_color_temp or super().min_mireds
|
|
|
|
@callback
|
|
def async_update_callback(self) -> None:
|
|
"""Light state will also reflect in relevant groups."""
|
|
super().async_update_callback()
|
|
|
|
if self._device.reachable and "attr" not in self._device.changed_keys:
|
|
for group in self.hub.api.groups.values():
|
|
if self._device.resource_id in group.lights:
|
|
update_color_state(group, [self._device])
|
|
|
|
|
|
class DeconzGroup(DeconzBaseLight[Group]):
|
|
"""Representation of a deCONZ group."""
|
|
|
|
_attr_has_entity_name = True
|
|
|
|
def __init__(self, device: Group, hub: DeconzHub) -> None:
|
|
"""Set up group and create an unique id."""
|
|
self._unique_id = f"{hub.bridgeid}-{device.deconz_id}"
|
|
super().__init__(device, hub)
|
|
|
|
self._attr_name = None
|
|
|
|
@property
|
|
def unique_id(self) -> str:
|
|
"""Return a unique identifier for this device."""
|
|
return self._unique_id
|
|
|
|
@property
|
|
def device_info(self) -> DeviceInfo:
|
|
"""Return a device description for device registry."""
|
|
return DeviceInfo(
|
|
identifiers={(DECONZ_DOMAIN, self.unique_id)},
|
|
manufacturer="Dresden Elektronik",
|
|
model="deCONZ group",
|
|
name=self._device.name,
|
|
via_device=(DECONZ_DOMAIN, self.hub.api.config.bridge_id),
|
|
)
|
|
|
|
@property
|
|
def extra_state_attributes(self) -> dict[str, bool]:
|
|
"""Return the device state attributes."""
|
|
attributes = dict(super().extra_state_attributes)
|
|
attributes["all_on"] = self._device.all_on
|
|
|
|
return attributes
|