2019-02-13 20:21:14 +00:00
|
|
|
"""Support for Eufy lights."""
|
2022-01-03 14:02:21 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2022-07-31 18:46:13 +00:00
|
|
|
from typing import Any
|
|
|
|
|
2019-10-10 18:16:30 +00:00
|
|
|
import lakeside
|
2018-04-11 01:38:23 +00:00
|
|
|
|
|
|
|
from homeassistant.components.light import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_BRIGHTNESS,
|
|
|
|
ATTR_COLOR_TEMP,
|
|
|
|
ATTR_HS_COLOR,
|
2022-04-23 19:18:47 +00:00
|
|
|
ColorMode,
|
2020-04-26 16:49:41 +00:00
|
|
|
LightEntity,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2022-01-03 14:02:21 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
2018-04-11 01:38:23 +00:00
|
|
|
import homeassistant.util.color as color_util
|
|
|
|
from homeassistant.util.color import (
|
2019-07-31 19:25:30 +00:00
|
|
|
color_temperature_kelvin_to_mired as kelvin_to_mired,
|
2019-12-08 22:34:18 +00:00
|
|
|
color_temperature_mired_to_kelvin as mired_to_kelvin,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-04-11 01:38:23 +00:00
|
|
|
|
|
|
|
EUFY_MAX_KELVIN = 6500
|
|
|
|
EUFY_MIN_KELVIN = 2700
|
|
|
|
|
|
|
|
|
2022-01-03 14:02:21 +00:00
|
|
|
def setup_platform(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config: ConfigType,
|
|
|
|
add_entities: AddEntitiesCallback,
|
|
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
|
|
) -> None:
|
2018-04-11 01:38:23 +00:00
|
|
|
"""Set up Eufy bulbs."""
|
|
|
|
if discovery_info is None:
|
|
|
|
return
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities([EufyLight(discovery_info)], True)
|
2018-04-11 01:38:23 +00:00
|
|
|
|
|
|
|
|
2020-04-26 16:49:41 +00:00
|
|
|
class EufyLight(LightEntity):
|
2018-04-11 01:38:23 +00:00
|
|
|
"""Representation of a Eufy light."""
|
|
|
|
|
|
|
|
def __init__(self, device):
|
|
|
|
"""Initialize the light."""
|
|
|
|
|
|
|
|
self._temp = None
|
|
|
|
self._brightness = None
|
|
|
|
self._hs = None
|
|
|
|
self._state = None
|
2019-07-31 19:25:30 +00:00
|
|
|
self._name = device["name"]
|
|
|
|
self._address = device["address"]
|
|
|
|
self._code = device["code"]
|
|
|
|
self._type = device["type"]
|
2018-04-11 01:38:23 +00:00
|
|
|
self._bulb = lakeside.bulb(self._address, self._code, self._type)
|
2018-04-15 07:54:02 +00:00
|
|
|
self._colormode = False
|
2018-04-11 01:38:23 +00:00
|
|
|
if self._type == "T1011":
|
2022-04-23 19:18:47 +00:00
|
|
|
self._attr_supported_color_modes = {ColorMode.BRIGHTNESS}
|
2018-04-11 01:38:23 +00:00
|
|
|
elif self._type == "T1012":
|
2022-04-23 19:18:47 +00:00
|
|
|
self._attr_supported_color_modes = {ColorMode.COLOR_TEMP}
|
2022-04-03 10:45:58 +00:00
|
|
|
else: # T1013
|
2022-04-23 19:18:47 +00:00
|
|
|
self._attr_supported_color_modes = {ColorMode.COLOR_TEMP, ColorMode.HS}
|
2018-04-11 01:38:23 +00:00
|
|
|
self._bulb.connect()
|
|
|
|
|
2022-07-31 18:46:13 +00:00
|
|
|
def update(self) -> None:
|
2018-04-11 01:38:23 +00:00
|
|
|
"""Synchronise state from the bulb."""
|
|
|
|
self._bulb.update()
|
2018-04-27 20:39:07 +00:00
|
|
|
if self._bulb.power:
|
|
|
|
self._brightness = self._bulb.brightness
|
|
|
|
self._temp = self._bulb.temperature
|
|
|
|
if self._bulb.colors:
|
|
|
|
self._colormode = True
|
|
|
|
self._hs = color_util.color_RGB_to_hs(*self._bulb.colors)
|
|
|
|
else:
|
|
|
|
self._colormode = False
|
2018-04-11 01:38:23 +00:00
|
|
|
self._state = self._bulb.power
|
|
|
|
|
|
|
|
@property
|
|
|
|
def unique_id(self):
|
|
|
|
"""Return the ID of this light."""
|
|
|
|
return self._address
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the device if any."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
|
|
|
"""Return true if device is on."""
|
|
|
|
return self._state
|
|
|
|
|
|
|
|
@property
|
|
|
|
def brightness(self):
|
|
|
|
"""Return the brightness of this light between 0..255."""
|
|
|
|
return int(self._brightness * 255 / 100)
|
|
|
|
|
|
|
|
@property
|
2022-07-31 18:46:13 +00:00
|
|
|
def min_mireds(self) -> int:
|
2018-04-11 01:38:23 +00:00
|
|
|
"""Return minimum supported color temperature."""
|
|
|
|
return kelvin_to_mired(EUFY_MAX_KELVIN)
|
|
|
|
|
|
|
|
@property
|
2022-07-31 18:46:13 +00:00
|
|
|
def max_mireds(self) -> int:
|
2022-01-17 14:29:10 +00:00
|
|
|
"""Return maximum supported color temperature."""
|
2018-04-11 01:38:23 +00:00
|
|
|
return kelvin_to_mired(EUFY_MIN_KELVIN)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def color_temp(self):
|
|
|
|
"""Return the color temperature of this light."""
|
2019-07-31 19:25:30 +00:00
|
|
|
temp_in_k = int(
|
|
|
|
EUFY_MIN_KELVIN + (self._temp * (EUFY_MAX_KELVIN - EUFY_MIN_KELVIN) / 100)
|
|
|
|
)
|
2018-04-11 01:38:23 +00:00
|
|
|
return kelvin_to_mired(temp_in_k)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def hs_color(self):
|
|
|
|
"""Return the color of this light."""
|
|
|
|
return self._hs
|
|
|
|
|
|
|
|
@property
|
2022-07-31 18:46:13 +00:00
|
|
|
def color_mode(self) -> ColorMode:
|
2022-04-03 10:45:58 +00:00
|
|
|
"""Return the color mode of the light."""
|
|
|
|
if self._type == "T1011":
|
2022-04-23 19:18:47 +00:00
|
|
|
return ColorMode.BRIGHTNESS
|
2022-04-03 10:45:58 +00:00
|
|
|
if self._type == "T1012":
|
2022-04-23 19:18:47 +00:00
|
|
|
return ColorMode.COLOR_TEMP
|
2022-04-03 10:45:58 +00:00
|
|
|
# T1013
|
|
|
|
if not self._colormode:
|
2022-04-23 19:18:47 +00:00
|
|
|
return ColorMode.COLOR_TEMP
|
|
|
|
return ColorMode.HS
|
2018-04-11 01:38:23 +00:00
|
|
|
|
2022-07-31 18:46:13 +00:00
|
|
|
def turn_on(self, **kwargs: Any) -> None:
|
2018-04-11 01:38:23 +00:00
|
|
|
"""Turn the specified light on."""
|
|
|
|
brightness = kwargs.get(ATTR_BRIGHTNESS)
|
|
|
|
colortemp = kwargs.get(ATTR_COLOR_TEMP)
|
|
|
|
# pylint: disable=invalid-name
|
|
|
|
hs = kwargs.get(ATTR_HS_COLOR)
|
|
|
|
|
|
|
|
if brightness is not None:
|
|
|
|
brightness = int(brightness * 100 / 255)
|
|
|
|
else:
|
2018-04-27 20:39:07 +00:00
|
|
|
if self._brightness is None:
|
|
|
|
self._brightness = 100
|
|
|
|
brightness = self._brightness
|
2018-04-11 01:38:23 +00:00
|
|
|
|
|
|
|
if colortemp is not None:
|
2018-04-15 07:54:02 +00:00
|
|
|
self._colormode = False
|
2018-04-11 01:38:23 +00:00
|
|
|
temp_in_k = mired_to_kelvin(colortemp)
|
|
|
|
relative_temp = temp_in_k - EUFY_MIN_KELVIN
|
2019-07-31 19:25:30 +00:00
|
|
|
temp = int(relative_temp * 100 / (EUFY_MAX_KELVIN - EUFY_MIN_KELVIN))
|
2018-04-11 01:38:23 +00:00
|
|
|
else:
|
|
|
|
temp = None
|
|
|
|
|
|
|
|
if hs is not None:
|
2019-07-31 19:25:30 +00:00
|
|
|
rgb = color_util.color_hsv_to_RGB(hs[0], hs[1], brightness / 255 * 100)
|
2018-04-15 07:54:02 +00:00
|
|
|
self._colormode = True
|
|
|
|
elif self._colormode:
|
|
|
|
rgb = color_util.color_hsv_to_RGB(
|
2019-07-31 19:25:30 +00:00
|
|
|
self._hs[0], self._hs[1], brightness / 255 * 100
|
|
|
|
)
|
2018-04-11 01:38:23 +00:00
|
|
|
else:
|
|
|
|
rgb = None
|
|
|
|
|
|
|
|
try:
|
2019-07-31 19:25:30 +00:00
|
|
|
self._bulb.set_state(
|
|
|
|
power=True, brightness=brightness, temperature=temp, colors=rgb
|
|
|
|
)
|
2018-04-11 01:38:23 +00:00
|
|
|
except BrokenPipeError:
|
|
|
|
self._bulb.connect()
|
2019-07-31 19:25:30 +00:00
|
|
|
self._bulb.set_state(
|
|
|
|
power=True, brightness=brightness, temperature=temp, colors=rgb
|
|
|
|
)
|
2018-04-11 01:38:23 +00:00
|
|
|
|
2022-07-31 18:46:13 +00:00
|
|
|
def turn_off(self, **kwargs: Any) -> None:
|
2018-04-11 01:38:23 +00:00
|
|
|
"""Turn the specified light off."""
|
|
|
|
try:
|
|
|
|
self._bulb.set_state(power=False)
|
|
|
|
except BrokenPipeError:
|
|
|
|
self._bulb.connect()
|
|
|
|
self._bulb.set_state(power=False)
|