core/homeassistant/components/homematicip_cloud/light.py

262 lines
8.4 KiB
Python
Raw Normal View History

"""Support for HomematicIP Cloud lights."""
import logging
from homematicip.aio.device import (
2019-07-31 19:25:30 +00:00
AsyncBrandDimmer,
AsyncBrandSwitchMeasuring,
AsyncBrandSwitchNotificationLight,
AsyncDimmer,
AsyncFullFlushDimmer,
AsyncPluggableDimmer,
)
2019-04-25 22:13:07 +00:00
from homematicip.aio.home import AsyncHome
from homematicip.base.enums import RGBColorState
2019-04-25 22:13:07 +00:00
from homematicip.base.functionalChannels import NotificationLightChannel
2018-08-21 19:25:16 +00:00
from homeassistant.components.light import (
2019-07-31 19:25:30 +00:00
ATTR_BRIGHTNESS,
ATTR_COLOR_NAME,
ATTR_HS_COLOR,
SUPPORT_BRIGHTNESS,
SUPPORT_COLOR,
Light,
)
2019-04-25 22:13:07 +00:00
from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant
from . import DOMAIN as HMIPC_DOMAIN, HMIPC_HAPID, HomematicipGenericDevice
_LOGGER = logging.getLogger(__name__)
2019-07-31 19:25:30 +00:00
ATTR_ENERGY_COUNTER = "energy_counter_kwh"
ATTR_POWER_CONSUMPTION = "power_consumption"
2019-07-31 19:25:30 +00:00
async def async_setup_platform(hass, config, async_add_entities, discovery_info=None):
2018-08-21 19:25:16 +00:00
"""Old way of setting up HomematicIP Cloud lights."""
Add HomematicIP Cloud Config Flow and Entries loading (#14861) * Add HomematicIP Cloud to config flow * Inititial trial for config_flow * Integrations text files * Load and write config_flow and init homematicip_cloud * Split into dedicated files * Ceanup of text messages * Working config_flow * Move imports inside a function * Enable laoding even no accesspoints are defined * Revert unnecassary changes in CONFIG_SCHEMA * Better error handling * fix flask8 * Migration to async for token generation * A few fixes * Simplify config_flow * Bump version to 9.6 with renamed package * Requirements file * First fixes after review * Implement async_step_import * Cleanup for Config Flow * First tests for homematicip_cloud setup * Remove config_flow tests * Really remove all things * Fix comment * Update picture * Add support for async_setup_entry to switch and climate platform * Update path of the config_flow picture * Refactoring for better tesability * Further tests implemented * Move 3th party lib inside function * Fix lint * Update requirments_test_all.txt file * UPdate of requirments_test_all.txt did not work * Furder cleanup in websocket connection * Remove a test for the hap * Revert "Remove a test for the hap" This reverts commit 968d58cba108e0f371022c7ab540374aa2ab13f4. * First tests implemented for config_flow * Fix lint * Rework of client registration process * Implemented tests for config_flow 100% coverage * Cleanup * Cleanup comments and code * Try to fix import problem * Add homematicip to the test env requirements
2018-07-06 21:05:34 +00:00
pass
2019-07-31 19:25:30 +00:00
async def async_setup_entry(
hass: HomeAssistant, config_entry: ConfigEntry, async_add_entities
) -> None:
2018-08-21 19:25:16 +00:00
"""Set up the HomematicIP Cloud lights from a config entry."""
Add HomematicIP Cloud Config Flow and Entries loading (#14861) * Add HomematicIP Cloud to config flow * Inititial trial for config_flow * Integrations text files * Load and write config_flow and init homematicip_cloud * Split into dedicated files * Ceanup of text messages * Working config_flow * Move imports inside a function * Enable laoding even no accesspoints are defined * Revert unnecassary changes in CONFIG_SCHEMA * Better error handling * fix flask8 * Migration to async for token generation * A few fixes * Simplify config_flow * Bump version to 9.6 with renamed package * Requirements file * First fixes after review * Implement async_step_import * Cleanup for Config Flow * First tests for homematicip_cloud setup * Remove config_flow tests * Really remove all things * Fix comment * Update picture * Add support for async_setup_entry to switch and climate platform * Update path of the config_flow picture * Refactoring for better tesability * Further tests implemented * Move 3th party lib inside function * Fix lint * Update requirments_test_all.txt file * UPdate of requirments_test_all.txt did not work * Furder cleanup in websocket connection * Remove a test for the hap * Revert "Remove a test for the hap" This reverts commit 968d58cba108e0f371022c7ab540374aa2ab13f4. * First tests implemented for config_flow * Fix lint * Rework of client registration process * Implemented tests for config_flow 100% coverage * Cleanup * Cleanup comments and code * Try to fix import problem * Add homematicip to the test env requirements
2018-07-06 21:05:34 +00:00
home = hass.data[HMIPC_DOMAIN][config_entry.data[HMIPC_HAPID]].home
devices = []
for device in home.devices:
if isinstance(device, AsyncBrandSwitchMeasuring):
devices.append(HomematicipLightMeasuring(home, device))
elif isinstance(device, AsyncBrandSwitchNotificationLight):
devices.append(HomematicipLight(home, device))
2019-07-31 19:25:30 +00:00
devices.append(
HomematicipNotificationLight(home, device, device.topLightChannelIndex)
)
devices.append(
HomematicipNotificationLight(
home, device, device.bottomLightChannelIndex
)
)
elif isinstance(
device,
(AsyncDimmer, AsyncPluggableDimmer, AsyncBrandDimmer, AsyncFullFlushDimmer),
):
devices.append(HomematicipDimmer(home, device))
if devices:
async_add_entities(devices)
class HomematicipLight(HomematicipGenericDevice, Light):
2018-08-21 19:25:16 +00:00
"""Representation of a HomematicIP Cloud light device."""
2019-04-25 22:13:07 +00:00
def __init__(self, home: AsyncHome, device) -> None:
"""Initialize the light device."""
super().__init__(home, device)
@property
2019-04-25 22:13:07 +00:00
def is_on(self) -> bool:
"""Return true if device is on."""
return self._device.on
async def async_turn_on(self, **kwargs):
"""Turn the device on."""
await self._device.turn_on()
async def async_turn_off(self, **kwargs):
"""Turn the device off."""
await self._device.turn_off()
class HomematicipLightMeasuring(HomematicipLight):
2018-08-21 19:25:16 +00:00
"""Representation of a HomematicIP Cloud measuring light device."""
@property
def device_state_attributes(self):
"""Return the state attributes of the generic device."""
attr = super().device_state_attributes
if self._device.currentPowerConsumption > 0.05:
2019-07-31 19:25:30 +00:00
attr[ATTR_POWER_CONSUMPTION] = round(
self._device.currentPowerConsumption, 2
)
attr[ATTR_ENERGY_COUNTER] = round(self._device.energyCounter, 2)
return attr
class HomematicipDimmer(HomematicipGenericDevice, Light):
2018-08-21 19:25:16 +00:00
"""Representation of HomematicIP Cloud dimmer light device."""
2019-04-25 22:13:07 +00:00
def __init__(self, home: AsyncHome, device) -> None:
"""Initialize the dimmer light device."""
super().__init__(home, device)
@property
2019-04-25 22:13:07 +00:00
def is_on(self) -> bool:
"""Return true if device is on."""
2019-07-31 19:25:30 +00:00
return self._device.dimLevel is not None and self._device.dimLevel > 0.0
@property
2019-04-25 22:13:07 +00:00
def brightness(self) -> int:
"""Return the brightness of this light between 0..255."""
if self._device.dimLevel:
2019-07-31 19:25:30 +00:00
return int(self._device.dimLevel * 255)
return 0
@property
2019-04-25 22:13:07 +00:00
def supported_features(self) -> int:
"""Flag supported features."""
return SUPPORT_BRIGHTNESS
async def async_turn_on(self, **kwargs):
"""Turn the light on."""
if ATTR_BRIGHTNESS in kwargs:
2019-07-31 19:25:30 +00:00
await self._device.set_dim_level(kwargs[ATTR_BRIGHTNESS] / 255.0)
else:
await self._device.set_dim_level(1)
async def async_turn_off(self, **kwargs):
"""Turn the light off."""
await self._device.set_dim_level(0)
class HomematicipNotificationLight(HomematicipGenericDevice, Light):
"""Representation of HomematicIP Cloud dimmer light device."""
2019-04-25 22:13:07 +00:00
def __init__(self, home: AsyncHome, device, channel: int) -> None:
"""Initialize the dimmer light device."""
self.channel = channel
if self.channel == 2:
2019-07-31 19:25:30 +00:00
super().__init__(home, device, "Top")
else:
2019-07-31 19:25:30 +00:00
super().__init__(home, device, "Bottom")
self._color_switcher = {
RGBColorState.WHITE: [0.0, 0.0],
RGBColorState.RED: [0.0, 100.0],
RGBColorState.YELLOW: [60.0, 100.0],
RGBColorState.GREEN: [120.0, 100.0],
RGBColorState.TURQUOISE: [180.0, 100.0],
RGBColorState.BLUE: [240.0, 100.0],
2019-07-31 19:25:30 +00:00
RGBColorState.PURPLE: [300.0, 100.0],
}
@property
2019-04-25 22:13:07 +00:00
def _func_channel(self) -> NotificationLightChannel:
return self._device.functionalChannels[self.channel]
@property
2019-04-25 22:13:07 +00:00
def is_on(self) -> bool:
"""Return true if device is on."""
2019-07-31 19:25:30 +00:00
return (
self._func_channel.dimLevel is not None
and self._func_channel.dimLevel > 0.0
)
@property
2019-04-25 22:13:07 +00:00
def brightness(self) -> int:
"""Return the brightness of this light between 0..255."""
if self._func_channel.dimLevel:
return int(self._func_channel.dimLevel * 255)
return 0
@property
2019-04-25 22:13:07 +00:00
def hs_color(self) -> tuple:
"""Return the hue and saturation color value [float, float]."""
simple_rgb_color = self._func_channel.simpleRGBColorState
return self._color_switcher.get(simple_rgb_color, [0.0, 0.0])
@property
def device_state_attributes(self):
"""Return the state attributes of the generic device."""
attr = super().device_state_attributes
if self.is_on:
attr[ATTR_COLOR_NAME] = self._func_channel.simpleRGBColorState
return attr
@property
2019-04-25 22:13:07 +00:00
def name(self) -> str:
"""Return the name of the generic device."""
2019-07-31 19:25:30 +00:00
return "{} {}".format(super().name, "Notification")
@property
2019-04-25 22:13:07 +00:00
def supported_features(self) -> int:
"""Flag supported features."""
return SUPPORT_BRIGHTNESS | SUPPORT_COLOR
@property
def unique_id(self) -> str:
"""Return a unique ID."""
return f"{self.__class__.__name__}_{self.post}_{self._device.id}"
async def async_turn_on(self, **kwargs):
"""Turn the light on."""
# Use hs_color from kwargs,
# if not applicable use current hs_color.
hs_color = kwargs.get(ATTR_HS_COLOR, self.hs_color)
simple_rgb_color = _convert_color(hs_color)
# Use brightness from kwargs,
# if not applicable use current brightness.
brightness = kwargs.get(ATTR_BRIGHTNESS, self.brightness)
# If no kwargs, use default value.
if not kwargs:
brightness = 255
# Minimum brightness is 10, otherwise the led is disabled
brightness = max(10, brightness)
dim_level = brightness / 255.0
2019-07-31 19:25:30 +00:00
await self._device.set_rgb_dim_level(self.channel, simple_rgb_color, dim_level)
async def async_turn_off(self, **kwargs):
"""Turn the light off."""
simple_rgb_color = self._func_channel.simpleRGBColorState
2019-07-31 19:25:30 +00:00
await self._device.set_rgb_dim_level(self.channel, simple_rgb_color, 0.0)
def _convert_color(color) -> RGBColorState:
"""
Convert the given color to the reduced RGBColorState color.
RGBColorStat contains only 8 colors including white and black,
so a conversion is required.
"""
if color is None:
return RGBColorState.WHITE
hue = int(color[0])
saturation = int(color[1])
if saturation < 5:
return RGBColorState.WHITE
if 30 < hue <= 90:
return RGBColorState.YELLOW
if 90 < hue <= 160:
return RGBColorState.GREEN
if 150 < hue <= 210:
return RGBColorState.TURQUOISE
if 210 < hue <= 270:
return RGBColorState.BLUE
if 270 < hue <= 330:
return RGBColorState.PURPLE
return RGBColorState.RED