""" Support for deCONZ light. For more details about this component, please refer to the documentation at https://home-assistant.io/components/light.deconz/ """ from homeassistant.components.deconz.const import ( CONF_ALLOW_DECONZ_GROUPS, DOMAIN as DATA_DECONZ, DATA_DECONZ_ID, DATA_DECONZ_UNSUB, DECONZ_DOMAIN, SWITCH_TYPES) from homeassistant.components.light import ( ATTR_BRIGHTNESS, ATTR_COLOR_TEMP, ATTR_EFFECT, ATTR_FLASH, ATTR_HS_COLOR, ATTR_TRANSITION, EFFECT_COLORLOOP, FLASH_LONG, FLASH_SHORT, SUPPORT_BRIGHTNESS, SUPPORT_COLOR, SUPPORT_COLOR_TEMP, SUPPORT_EFFECT, SUPPORT_FLASH, SUPPORT_TRANSITION, Light) from homeassistant.core import callback from homeassistant.helpers.device_registry import CONNECTION_ZIGBEE from homeassistant.helpers.dispatcher import async_dispatcher_connect import homeassistant.util.color as color_util DEPENDENCIES = ['deconz'] async def async_setup_platform(hass, config, async_add_entities, discovery_info=None): """Old way of setting up deCONZ lights and group.""" pass async def async_setup_entry(hass, config_entry, async_add_entities): """Set up the deCONZ lights and groups from a config entry.""" @callback def async_add_light(lights): """Add light from deCONZ.""" entities = [] for light in lights: if light.type not in SWITCH_TYPES: entities.append(DeconzLight(light)) async_add_entities(entities, True) hass.data[DATA_DECONZ_UNSUB].append( async_dispatcher_connect(hass, 'deconz_new_light', async_add_light)) @callback def async_add_group(groups): """Add group from deCONZ.""" entities = [] allow_group = config_entry.data.get(CONF_ALLOW_DECONZ_GROUPS, True) for group in groups: if group.lights and allow_group: entities.append(DeconzLight(group)) async_add_entities(entities, True) hass.data[DATA_DECONZ_UNSUB].append( async_dispatcher_connect(hass, 'deconz_new_group', async_add_group)) async_add_light(hass.data[DATA_DECONZ].lights.values()) async_add_group(hass.data[DATA_DECONZ].groups.values()) class DeconzLight(Light): """Representation of a deCONZ light.""" def __init__(self, light): """Set up light and add update callback to get data from websocket.""" self._light = light self._features = SUPPORT_BRIGHTNESS self._features |= SUPPORT_FLASH self._features |= SUPPORT_TRANSITION if self._light.ct is not None: self._features |= SUPPORT_COLOR_TEMP if self._light.xy is not None: self._features |= SUPPORT_COLOR if self._light.effect is not None: self._features |= SUPPORT_EFFECT async def async_added_to_hass(self): """Subscribe to lights events.""" self._light.register_async_callback(self.async_update_callback) self.hass.data[DATA_DECONZ_ID][self.entity_id] = self._light.deconz_id @callback def async_update_callback(self, reason): """Update the light's state.""" self.async_schedule_update_ha_state() @property def brightness(self): """Return the brightness of this light between 0..255.""" return self._light.brightness @property def effect_list(self): """Return the list of supported effects.""" return [EFFECT_COLORLOOP] @property def color_temp(self): """Return the CT color value.""" if self._light.colormode != 'ct': return None return self._light.ct @property def hs_color(self): """Return the hs color value.""" if self._light.colormode in ('xy', 'hs') and self._light.xy: return color_util.color_xy_to_hs(*self._light.xy) return None @property def is_on(self): """Return true if light is on.""" return self._light.state @property def name(self): """Return the name of the light.""" return self._light.name @property def unique_id(self): """Return a unique identifier for this light.""" return self._light.uniqueid @property def supported_features(self): """Flag supported features.""" return self._features @property def available(self): """Return True if light is available.""" return self._light.reachable @property def should_poll(self): """No polling needed.""" return False async def async_turn_on(self, **kwargs): """Turn on light.""" data = {'on': True} if ATTR_COLOR_TEMP in kwargs: data['ct'] = kwargs[ATTR_COLOR_TEMP] if ATTR_HS_COLOR in kwargs: data['xy'] = color_util.color_hs_to_xy(*kwargs[ATTR_HS_COLOR]) if ATTR_BRIGHTNESS in kwargs: data['bri'] = kwargs[ATTR_BRIGHTNESS] if ATTR_TRANSITION in kwargs: data['transitiontime'] = int(kwargs[ATTR_TRANSITION] * 10) if ATTR_FLASH in kwargs: if kwargs[ATTR_FLASH] == FLASH_SHORT: data['alert'] = 'select' del data['on'] elif kwargs[ATTR_FLASH] == FLASH_LONG: data['alert'] = 'lselect' del data['on'] if ATTR_EFFECT in kwargs: if kwargs[ATTR_EFFECT] == EFFECT_COLORLOOP: data['effect'] = 'colorloop' else: data['effect'] = 'none' await self._light.async_set_state(data) async def async_turn_off(self, **kwargs): """Turn off light.""" data = {'on': False} if ATTR_TRANSITION in kwargs: data['bri'] = 0 data['transitiontime'] = int(kwargs[ATTR_TRANSITION] * 10) if ATTR_FLASH in kwargs: if kwargs[ATTR_FLASH] == FLASH_SHORT: data['alert'] = 'select' del data['on'] elif kwargs[ATTR_FLASH] == FLASH_LONG: data['alert'] = 'lselect' del data['on'] await self._light.async_set_state(data) @property def device_state_attributes(self): """Return the device state attributes.""" attributes = {} attributes['is_deconz_group'] = self._light.type == 'LightGroup' if self._light.type == 'LightGroup': attributes['all_on'] = self._light.all_on return attributes @property def device_info(self): """Return a device description for device registry.""" if (self._light.uniqueid is None or self._light.uniqueid.count(':') != 7): return None serial = self._light.uniqueid.split('-', 1)[0] return { 'connections': {(CONNECTION_ZIGBEE, serial)}, 'identifiers': {(DECONZ_DOMAIN, serial)}, 'manufacturer': self._light.manufacturer, 'model': self._light.modelid, 'name': self._light.name, 'sw_version': self._light.swversion, }