2021-01-10 23:08:25 +00:00
|
|
|
"""Support for Z-Wave lights."""
|
|
|
|
import logging
|
2021-01-13 15:28:51 +00:00
|
|
|
from typing import Any, Callable, Optional, Tuple
|
2021-01-10 23:08:25 +00:00
|
|
|
|
|
|
|
from zwave_js_server.client import Client as ZwaveClient
|
|
|
|
from zwave_js_server.const import CommandClass
|
|
|
|
|
|
|
|
from homeassistant.components.light import (
|
|
|
|
ATTR_BRIGHTNESS,
|
|
|
|
ATTR_COLOR_TEMP,
|
|
|
|
ATTR_HS_COLOR,
|
|
|
|
ATTR_TRANSITION,
|
|
|
|
ATTR_WHITE_VALUE,
|
|
|
|
DOMAIN as LIGHT_DOMAIN,
|
|
|
|
SUPPORT_BRIGHTNESS,
|
|
|
|
SUPPORT_COLOR,
|
|
|
|
SUPPORT_COLOR_TEMP,
|
|
|
|
SUPPORT_TRANSITION,
|
|
|
|
SUPPORT_WHITE_VALUE,
|
|
|
|
LightEntity,
|
|
|
|
)
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
|
|
|
import homeassistant.util.color as color_util
|
|
|
|
|
|
|
|
from .const import DATA_CLIENT, DATA_UNSUBSCRIBE, DOMAIN
|
|
|
|
from .discovery import ZwaveDiscoveryInfo
|
|
|
|
from .entity import ZWaveBaseEntity
|
|
|
|
|
|
|
|
LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant, config_entry: ConfigEntry, async_add_entities: Callable
|
|
|
|
) -> None:
|
|
|
|
"""Set up Z-Wave Light from Config Entry."""
|
|
|
|
client: ZwaveClient = hass.data[DOMAIN][config_entry.entry_id][DATA_CLIENT]
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_add_light(info: ZwaveDiscoveryInfo) -> None:
|
|
|
|
"""Add Z-Wave Light."""
|
|
|
|
|
2021-01-19 15:40:01 +00:00
|
|
|
light = ZwaveLight(config_entry, client, info)
|
2021-01-10 23:08:25 +00:00
|
|
|
async_add_entities([light])
|
|
|
|
|
|
|
|
hass.data[DOMAIN][config_entry.entry_id][DATA_UNSUBSCRIBE].append(
|
2021-01-19 15:40:01 +00:00
|
|
|
async_dispatcher_connect(
|
|
|
|
hass,
|
|
|
|
f"{DOMAIN}_{config_entry.entry_id}_add_{LIGHT_DOMAIN}",
|
|
|
|
async_add_light,
|
|
|
|
)
|
2021-01-10 23:08:25 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def byte_to_zwave_brightness(value: int) -> int:
|
|
|
|
"""Convert brightness in 0-255 scale to 0-99 scale.
|
|
|
|
|
|
|
|
`value` -- (int) Brightness byte value from 0-255.
|
|
|
|
"""
|
|
|
|
if value > 0:
|
|
|
|
return max(1, round((value / 255) * 99))
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
|
|
|
class ZwaveLight(ZWaveBaseEntity, LightEntity):
|
|
|
|
"""Representation of a Z-Wave light."""
|
|
|
|
|
2021-01-19 15:40:01 +00:00
|
|
|
def __init__(
|
|
|
|
self, config_entry: ConfigEntry, client: ZwaveClient, info: ZwaveDiscoveryInfo
|
|
|
|
) -> None:
|
2021-01-10 23:08:25 +00:00
|
|
|
"""Initialize the light."""
|
2021-01-19 15:40:01 +00:00
|
|
|
super().__init__(config_entry, client, info)
|
2021-01-10 23:08:25 +00:00
|
|
|
self._supports_color = False
|
|
|
|
self._supports_white_value = False
|
|
|
|
self._supports_color_temp = False
|
2021-01-13 15:28:51 +00:00
|
|
|
self._hs_color: Optional[Tuple[float, float]] = None
|
2021-01-10 23:08:25 +00:00
|
|
|
self._white_value: Optional[int] = None
|
|
|
|
self._color_temp: Optional[int] = None
|
|
|
|
self._min_mireds = 153 # 6500K as a safe default
|
|
|
|
self._max_mireds = 370 # 2700K as a safe default
|
|
|
|
self._supported_features = SUPPORT_BRIGHTNESS
|
|
|
|
|
|
|
|
# get additional (optional) values and set features
|
|
|
|
self._target_value = self.get_zwave_value("targetValue")
|
|
|
|
self._dimming_duration = self.get_zwave_value("duration")
|
|
|
|
if self._dimming_duration is not None:
|
|
|
|
self._supported_features |= SUPPORT_TRANSITION
|
|
|
|
self._calculate_color_values()
|
|
|
|
if self._supports_color:
|
|
|
|
self._supported_features |= SUPPORT_COLOR
|
|
|
|
if self._supports_color_temp:
|
|
|
|
self._supported_features |= SUPPORT_COLOR_TEMP
|
|
|
|
if self._supports_white_value:
|
|
|
|
self._supported_features |= SUPPORT_WHITE_VALUE
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def on_value_update(self) -> None:
|
|
|
|
"""Call when a watched value is added or updated."""
|
|
|
|
self._calculate_color_values()
|
|
|
|
|
|
|
|
@property
|
|
|
|
def brightness(self) -> int:
|
|
|
|
"""Return the brightness of this light between 0..255.
|
|
|
|
|
|
|
|
Z-Wave multilevel switches use a range of [0, 99] to control brightness.
|
|
|
|
"""
|
|
|
|
if self._target_value is not None and self._target_value.value is not None:
|
|
|
|
return round((self._target_value.value / 99) * 255)
|
|
|
|
if self.info.primary_value.value is not None:
|
|
|
|
return round((self.info.primary_value.value / 99) * 255)
|
|
|
|
return 0
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self) -> bool:
|
|
|
|
"""Return true if device is on (brightness above 0)."""
|
|
|
|
return self.brightness > 0
|
|
|
|
|
|
|
|
@property
|
2021-01-13 15:28:51 +00:00
|
|
|
def hs_color(self) -> Optional[Tuple[float, float]]:
|
2021-01-10 23:08:25 +00:00
|
|
|
"""Return the hs color."""
|
|
|
|
return self._hs_color
|
|
|
|
|
|
|
|
@property
|
|
|
|
def white_value(self) -> Optional[int]:
|
|
|
|
"""Return the white value of this light between 0..255."""
|
|
|
|
return self._white_value
|
|
|
|
|
|
|
|
@property
|
|
|
|
def color_temp(self) -> Optional[int]:
|
|
|
|
"""Return the color temperature."""
|
|
|
|
return self._color_temp
|
|
|
|
|
|
|
|
@property
|
|
|
|
def min_mireds(self) -> int:
|
|
|
|
"""Return the coldest color_temp that this light supports."""
|
|
|
|
return self._min_mireds
|
|
|
|
|
|
|
|
@property
|
|
|
|
def max_mireds(self) -> int:
|
|
|
|
"""Return the warmest color_temp that this light supports."""
|
|
|
|
return self._max_mireds
|
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_features(self) -> Optional[int]:
|
|
|
|
"""Flag supported features."""
|
|
|
|
return self._supported_features
|
|
|
|
|
|
|
|
async def async_turn_on(self, **kwargs: Any) -> None:
|
|
|
|
"""Turn the device on."""
|
|
|
|
# RGB/HS color
|
|
|
|
hs_color = kwargs.get(ATTR_HS_COLOR)
|
|
|
|
if hs_color is not None and self._supports_color:
|
|
|
|
# set white levels to 0 when setting rgb
|
|
|
|
await self._async_set_color("Warm White", 0)
|
|
|
|
await self._async_set_color("Cold White", 0)
|
|
|
|
red, green, blue = color_util.color_hs_to_RGB(*hs_color)
|
|
|
|
await self._async_set_color("Red", red)
|
|
|
|
await self._async_set_color("Green", green)
|
|
|
|
await self._async_set_color("Blue", blue)
|
|
|
|
else:
|
|
|
|
# turn off rgb when setting white values
|
|
|
|
await self._async_set_color("Red", 0)
|
|
|
|
await self._async_set_color("Green", 0)
|
|
|
|
await self._async_set_color("Blue", 0)
|
|
|
|
|
|
|
|
# Color temperature
|
|
|
|
color_temp = kwargs.get(ATTR_COLOR_TEMP)
|
|
|
|
if color_temp is not None and self._supports_color_temp:
|
|
|
|
# Limit color temp to min/max values
|
|
|
|
cold = max(
|
|
|
|
0,
|
|
|
|
min(
|
|
|
|
255,
|
|
|
|
round(
|
|
|
|
(self._max_mireds - color_temp)
|
|
|
|
/ (self._max_mireds - self._min_mireds)
|
|
|
|
* 255
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
warm = 255 - cold
|
|
|
|
await self._async_set_color("Warm White", warm)
|
|
|
|
await self._async_set_color("Cold White", cold)
|
|
|
|
|
|
|
|
# White value
|
|
|
|
white_value = kwargs.get(ATTR_WHITE_VALUE)
|
|
|
|
if white_value is not None and self._supports_white_value:
|
|
|
|
await self._async_set_color("Warm White", white_value)
|
|
|
|
|
|
|
|
# set brightness
|
|
|
|
await self._async_set_brightness(
|
|
|
|
kwargs.get(ATTR_BRIGHTNESS), kwargs.get(ATTR_TRANSITION)
|
|
|
|
)
|
|
|
|
|
|
|
|
async def async_turn_off(self, **kwargs: Any) -> None:
|
|
|
|
"""Turn the light off."""
|
|
|
|
await self._async_set_brightness(0, kwargs.get(ATTR_TRANSITION))
|
|
|
|
|
|
|
|
async def _async_set_color(self, color_name: str, new_value: int) -> None:
|
|
|
|
"""Set defined color to given value."""
|
|
|
|
cur_zwave_value = self.get_zwave_value(
|
|
|
|
"currentColor",
|
|
|
|
CommandClass.SWITCH_COLOR,
|
|
|
|
value_property_key_name=color_name,
|
|
|
|
)
|
|
|
|
# guard for unsupported command
|
|
|
|
if cur_zwave_value is None:
|
|
|
|
return
|
|
|
|
# no need to send same value
|
|
|
|
if cur_zwave_value.value == new_value:
|
|
|
|
return
|
|
|
|
# actually set the new color value
|
|
|
|
target_zwave_value = self.get_zwave_value(
|
|
|
|
"targetColor",
|
|
|
|
CommandClass.SWITCH_COLOR,
|
|
|
|
value_property_key_name=color_name,
|
|
|
|
)
|
|
|
|
if target_zwave_value is None:
|
|
|
|
return
|
|
|
|
await self.info.node.async_set_value(target_zwave_value, new_value)
|
|
|
|
|
|
|
|
async def _async_set_brightness(
|
|
|
|
self, brightness: Optional[int], transition: Optional[int] = None
|
|
|
|
) -> None:
|
|
|
|
"""Set new brightness to light."""
|
|
|
|
if brightness is None and self.info.primary_value.value:
|
|
|
|
# there is no point in setting default brightness when light is already on
|
|
|
|
return
|
|
|
|
if brightness is None:
|
|
|
|
# Level 255 means to set it to previous value.
|
2021-01-13 15:28:51 +00:00
|
|
|
zwave_brightness = 255
|
2021-01-10 23:08:25 +00:00
|
|
|
else:
|
|
|
|
# Zwave multilevel switches use a range of [0, 99] to control brightness.
|
2021-01-13 15:28:51 +00:00
|
|
|
zwave_brightness = byte_to_zwave_brightness(brightness)
|
|
|
|
|
|
|
|
if self.info.primary_value.value == zwave_brightness:
|
|
|
|
# no point in setting same brightness
|
|
|
|
return
|
2021-01-10 23:08:25 +00:00
|
|
|
# set transition value before seinding new brightness
|
|
|
|
await self._async_set_transition_duration(transition)
|
|
|
|
# setting a value requires setting targetValue
|
2021-01-13 15:28:51 +00:00
|
|
|
await self.info.node.async_set_value(self._target_value, zwave_brightness)
|
2021-01-10 23:08:25 +00:00
|
|
|
|
|
|
|
async def _async_set_transition_duration(
|
|
|
|
self, duration: Optional[int] = None
|
|
|
|
) -> None:
|
|
|
|
"""Set the transition time for the brightness value."""
|
|
|
|
if self._dimming_duration is None:
|
|
|
|
return
|
|
|
|
# pylint: disable=fixme,unreachable
|
|
|
|
# TODO: setting duration needs to be fixed upstream
|
|
|
|
# https://github.com/zwave-js/node-zwave-js/issues/1321
|
|
|
|
return
|
|
|
|
|
|
|
|
if duration is None: # type: ignore
|
|
|
|
# no transition specified by user, use defaults
|
|
|
|
duration = 7621 # anything over 7620 uses the factory default
|
2021-01-13 15:28:51 +00:00
|
|
|
else: # pragma: no cover
|
2021-01-10 23:08:25 +00:00
|
|
|
# transition specified by user
|
|
|
|
transition = duration
|
|
|
|
if transition <= 127:
|
|
|
|
duration = transition
|
|
|
|
else:
|
|
|
|
minutes = round(transition / 60)
|
|
|
|
LOGGER.debug(
|
|
|
|
"Transition rounded to %d minutes for %s",
|
|
|
|
minutes,
|
|
|
|
self.entity_id,
|
|
|
|
)
|
|
|
|
duration = minutes + 128
|
|
|
|
|
|
|
|
# only send value if it differs from current
|
|
|
|
# this prevents sending a command for nothing
|
2021-01-13 15:28:51 +00:00
|
|
|
if self._dimming_duration.value != duration: # pragma: no cover
|
2021-01-10 23:08:25 +00:00
|
|
|
await self.info.node.async_set_value(self._dimming_duration, duration)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _calculate_color_values(self) -> None:
|
|
|
|
"""Calculate light colors."""
|
|
|
|
|
|
|
|
# RGB support
|
|
|
|
red_val = self.get_zwave_value(
|
|
|
|
"currentColor", CommandClass.SWITCH_COLOR, value_property_key_name="Red"
|
|
|
|
)
|
|
|
|
green_val = self.get_zwave_value(
|
|
|
|
"currentColor", CommandClass.SWITCH_COLOR, value_property_key_name="Green"
|
|
|
|
)
|
|
|
|
blue_val = self.get_zwave_value(
|
|
|
|
"currentColor", CommandClass.SWITCH_COLOR, value_property_key_name="Blue"
|
|
|
|
)
|
|
|
|
if red_val and green_val and blue_val:
|
|
|
|
self._supports_color = True
|
|
|
|
# convert to HS
|
|
|
|
if (
|
|
|
|
red_val.value is not None
|
|
|
|
and green_val.value is not None
|
|
|
|
and blue_val.value is not None
|
|
|
|
):
|
2021-01-13 15:28:51 +00:00
|
|
|
self._hs_color = color_util.color_RGB_to_hs(
|
2021-01-10 23:08:25 +00:00
|
|
|
red_val.value, green_val.value, blue_val.value
|
|
|
|
)
|
|
|
|
|
|
|
|
# White colors
|
|
|
|
ww_val = self.get_zwave_value(
|
|
|
|
"currentColor",
|
|
|
|
CommandClass.SWITCH_COLOR,
|
|
|
|
value_property_key_name="Warm White",
|
|
|
|
)
|
|
|
|
cw_val = self.get_zwave_value(
|
|
|
|
"currentColor",
|
|
|
|
CommandClass.SWITCH_COLOR,
|
|
|
|
value_property_key_name="Cold White",
|
|
|
|
)
|
|
|
|
if ww_val and cw_val:
|
|
|
|
# Color temperature (CW + WW) Support
|
|
|
|
self._supports_color_temp = True
|
|
|
|
# Calculate color temps based on whites
|
|
|
|
cold_level = cw_val.value or 0
|
|
|
|
if cold_level or ww_val.value is not None:
|
|
|
|
self._color_temp = round(
|
|
|
|
self._max_mireds
|
|
|
|
- ((cold_level / 255) * (self._max_mireds - self._min_mireds))
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
self._color_temp = None
|
|
|
|
elif ww_val or cw_val:
|
|
|
|
# only one white channel
|
|
|
|
self._supports_white_value = True
|
2021-01-13 15:28:51 +00:00
|
|
|
# FIXME: Update self._white_value
|