"""Support for AVM FRITZ!SmartHome lightbulbs.""" from __future__ import annotations from typing import Any from requests.exceptions import HTTPError from homeassistant.components.light import ( ATTR_BRIGHTNESS, ATTR_COLOR_TEMP, ATTR_HS_COLOR, ColorMode, LightEntity, ) from homeassistant.config_entries import ConfigEntry from homeassistant.core import HomeAssistant from homeassistant.helpers.entity_platform import AddEntitiesCallback from homeassistant.util import color from . import FritzBoxEntity from .const import ( COLOR_MODE, COLOR_TEMP_MODE, CONF_COORDINATOR, DOMAIN as FRITZBOX_DOMAIN, LOGGER, ) from .coordinator import FritzboxDataUpdateCoordinator SUPPORTED_COLOR_MODES = {ColorMode.COLOR_TEMP, ColorMode.HS} async def async_setup_entry( hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback ) -> None: """Set up the FRITZ!SmartHome light from ConfigEntry.""" entities: list[FritzboxLight] = [] coordinator = hass.data[FRITZBOX_DOMAIN][entry.entry_id][CONF_COORDINATOR] for ain, device in coordinator.data.items(): if not device.has_lightbulb: continue supported_color_temps = await hass.async_add_executor_job( device.get_color_temps ) supported_colors = await hass.async_add_executor_job(device.get_colors) entities.append( FritzboxLight( coordinator, ain, supported_colors, supported_color_temps, ) ) async_add_entities(entities) class FritzboxLight(FritzBoxEntity, LightEntity): """The light class for FRITZ!SmartHome lightbulbs.""" def __init__( self, coordinator: FritzboxDataUpdateCoordinator, ain: str, supported_colors: dict, supported_color_temps: list[str], ) -> None: """Initialize the FritzboxLight entity.""" super().__init__(coordinator, ain, None) max_kelvin = int(max(supported_color_temps)) min_kelvin = int(min(supported_color_temps)) # max kelvin is min mireds and min kelvin is max mireds self._attr_min_mireds = color.color_temperature_kelvin_to_mired(max_kelvin) self._attr_max_mireds = color.color_temperature_kelvin_to_mired(min_kelvin) # Fritz!DECT 500 only supports 12 values for hue, with 3 saturations each. # Map supported colors to dict {hue: [sat1, sat2, sat3]} for easier lookup self._supported_hs = {} for values in supported_colors.values(): hue = int(values[0][0]) self._supported_hs[hue] = [ int(values[0][1]), int(values[1][1]), int(values[2][1]), ] @property def is_on(self) -> bool: """If the light is currently on or off.""" return self.device.state # type: ignore [no-any-return] @property def brightness(self) -> int: """Return the current Brightness.""" return self.device.level # type: ignore [no-any-return] @property def hs_color(self) -> tuple[float, float] | None: """Return the hs color value.""" if self.device.color_mode != COLOR_MODE: return None hue = self.device.hue saturation = self.device.saturation return (hue, float(saturation) * 100.0 / 255.0) @property def color_temp(self) -> int | None: """Return the CT color value.""" if self.device.color_mode != COLOR_TEMP_MODE: return None kelvin = self.device.color_temp return color.color_temperature_kelvin_to_mired(kelvin) @property def color_mode(self) -> ColorMode: """Return the color mode of the light.""" if self.device.color_mode == COLOR_MODE: return ColorMode.HS return ColorMode.COLOR_TEMP @property def supported_color_modes(self) -> set[ColorMode]: """Flag supported color modes.""" return SUPPORTED_COLOR_MODES async def async_turn_on(self, **kwargs: Any) -> None: """Turn the light on.""" if kwargs.get(ATTR_BRIGHTNESS) is not None: level = kwargs[ATTR_BRIGHTNESS] await self.hass.async_add_executor_job(self.device.set_level, level) if kwargs.get(ATTR_HS_COLOR) is not None: # Try setunmappedcolor first. This allows free color selection, # but we don't know if its supported by all devices. try: # HA gives 0..360 for hue, fritz light only supports 0..359 unmapped_hue = int(kwargs[ATTR_HS_COLOR][0] % 360) unmapped_saturation = round(kwargs[ATTR_HS_COLOR][1] * 255.0 / 100.0) await self.hass.async_add_executor_job( self.device.set_unmapped_color, (unmapped_hue, unmapped_saturation) ) # This will raise 400 BAD REQUEST if the setunmappedcolor is not available except HTTPError as err: if err.response.status_code != 400: raise LOGGER.debug( "fritzbox does not support method 'setunmappedcolor', fallback to 'setcolor'" ) # find supported hs values closest to what user selected hue = min( self._supported_hs.keys(), key=lambda x: abs(x - unmapped_hue) ) saturation = min( self._supported_hs[hue], key=lambda x: abs(x - unmapped_saturation), ) await self.hass.async_add_executor_job( self.device.set_color, (hue, saturation) ) if kwargs.get(ATTR_COLOR_TEMP) is not None: kelvin = color.color_temperature_kelvin_to_mired(kwargs[ATTR_COLOR_TEMP]) await self.hass.async_add_executor_job(self.device.set_color_temp, kelvin) await self.hass.async_add_executor_job(self.device.set_state_on) await self.coordinator.async_refresh() async def async_turn_off(self, **kwargs: Any) -> None: """Turn the light off.""" await self.hass.async_add_executor_job(self.device.set_state_off) await self.coordinator.async_refresh()