2019-03-25 07:50:47 +00:00
|
|
|
"""Support for Xiaomi Yeelight WiFi color bulb."""
|
2019-03-24 12:01:12 +00:00
|
|
|
|
|
|
|
import logging
|
|
|
|
from datetime import timedelta
|
|
|
|
|
|
|
|
import voluptuous as vol
|
2019-05-22 02:47:10 +00:00
|
|
|
from yeelight import Bulb, BulbException
|
2019-03-24 12:01:12 +00:00
|
|
|
from homeassistant.components.discovery import SERVICE_YEELIGHT
|
|
|
|
from homeassistant.const import CONF_DEVICES, CONF_NAME, CONF_SCAN_INTERVAL, \
|
2019-03-25 07:50:47 +00:00
|
|
|
CONF_HOST, ATTR_ENTITY_ID
|
2019-03-24 12:01:12 +00:00
|
|
|
from homeassistant.components.light import DOMAIN as LIGHT_DOMAIN
|
2019-03-25 18:06:43 +00:00
|
|
|
from homeassistant.components.binary_sensor import DOMAIN as \
|
|
|
|
BINARY_SENSOR_DOMAIN
|
2019-03-24 12:01:12 +00:00
|
|
|
from homeassistant.helpers import discovery
|
|
|
|
from homeassistant.helpers.discovery import load_platform
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2019-06-13 16:42:47 +00:00
|
|
|
from homeassistant.helpers.dispatcher import dispatcher_send, \
|
|
|
|
dispatcher_connect
|
2019-03-25 07:50:47 +00:00
|
|
|
from homeassistant.helpers.event import track_time_interval
|
2019-03-24 12:01:12 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
DOMAIN = "yeelight"
|
|
|
|
DATA_YEELIGHT = DOMAIN
|
2019-04-07 14:07:50 +00:00
|
|
|
DATA_UPDATED = 'yeelight_{}_data_updated'
|
2019-06-13 16:42:47 +00:00
|
|
|
DEVICE_INITIALIZED = '{}_device_initialized'.format(DOMAIN)
|
2019-03-24 12:01:12 +00:00
|
|
|
|
|
|
|
DEFAULT_NAME = 'Yeelight'
|
|
|
|
DEFAULT_TRANSITION = 350
|
|
|
|
|
|
|
|
CONF_MODEL = 'model'
|
|
|
|
CONF_TRANSITION = 'transition'
|
|
|
|
CONF_SAVE_ON_CHANGE = 'save_on_change'
|
|
|
|
CONF_MODE_MUSIC = 'use_music_mode'
|
|
|
|
CONF_FLOW_PARAMS = 'flow_params'
|
|
|
|
CONF_CUSTOM_EFFECTS = 'custom_effects'
|
|
|
|
|
|
|
|
ATTR_COUNT = 'count'
|
|
|
|
ATTR_ACTION = 'action'
|
|
|
|
ATTR_TRANSITIONS = 'transitions'
|
|
|
|
|
|
|
|
ACTION_RECOVER = 'recover'
|
|
|
|
ACTION_STAY = 'stay'
|
|
|
|
ACTION_OFF = 'off'
|
|
|
|
|
2019-06-13 16:42:47 +00:00
|
|
|
ACTIVE_MODE_NIGHTLIGHT = '1'
|
|
|
|
|
2019-03-24 12:01:12 +00:00
|
|
|
SCAN_INTERVAL = timedelta(seconds=30)
|
|
|
|
|
|
|
|
YEELIGHT_RGB_TRANSITION = 'RGBTransition'
|
|
|
|
YEELIGHT_HSV_TRANSACTION = 'HSVTransition'
|
|
|
|
YEELIGHT_TEMPERATURE_TRANSACTION = 'TemperatureTransition'
|
|
|
|
YEELIGHT_SLEEP_TRANSACTION = 'SleepTransition'
|
|
|
|
|
|
|
|
YEELIGHT_FLOW_TRANSITION_SCHEMA = {
|
|
|
|
vol.Optional(ATTR_COUNT, default=0): cv.positive_int,
|
|
|
|
vol.Optional(ATTR_ACTION, default=ACTION_RECOVER):
|
|
|
|
vol.Any(ACTION_RECOVER, ACTION_OFF, ACTION_STAY),
|
|
|
|
vol.Required(ATTR_TRANSITIONS): [{
|
|
|
|
vol.Exclusive(YEELIGHT_RGB_TRANSITION, CONF_TRANSITION):
|
|
|
|
vol.All(cv.ensure_list, [cv.positive_int]),
|
|
|
|
vol.Exclusive(YEELIGHT_HSV_TRANSACTION, CONF_TRANSITION):
|
|
|
|
vol.All(cv.ensure_list, [cv.positive_int]),
|
|
|
|
vol.Exclusive(YEELIGHT_TEMPERATURE_TRANSACTION, CONF_TRANSITION):
|
|
|
|
vol.All(cv.ensure_list, [cv.positive_int]),
|
|
|
|
vol.Exclusive(YEELIGHT_SLEEP_TRANSACTION, CONF_TRANSITION):
|
|
|
|
vol.All(cv.ensure_list, [cv.positive_int]),
|
|
|
|
}]
|
|
|
|
}
|
|
|
|
|
|
|
|
DEVICE_SCHEMA = vol.Schema({
|
|
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
|
|
vol.Optional(CONF_TRANSITION, default=DEFAULT_TRANSITION): cv.positive_int,
|
|
|
|
vol.Optional(CONF_MODE_MUSIC, default=False): cv.boolean,
|
|
|
|
vol.Optional(CONF_SAVE_ON_CHANGE, default=False): cv.boolean,
|
|
|
|
vol.Optional(CONF_MODEL): cv.string,
|
|
|
|
})
|
|
|
|
|
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
|
|
|
vol.Optional(CONF_DEVICES, default={}): {cv.string: DEVICE_SCHEMA},
|
|
|
|
vol.Optional(CONF_SCAN_INTERVAL, default=SCAN_INTERVAL):
|
|
|
|
cv.time_period,
|
|
|
|
vol.Optional(CONF_CUSTOM_EFFECTS): [{
|
|
|
|
vol.Required(CONF_NAME): cv.string,
|
|
|
|
vol.Required(CONF_FLOW_PARAMS): YEELIGHT_FLOW_TRANSITION_SCHEMA
|
|
|
|
}]
|
|
|
|
}),
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
YEELIGHT_SERVICE_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(ATTR_ENTITY_ID): cv.entity_ids,
|
|
|
|
})
|
|
|
|
|
|
|
|
UPDATE_REQUEST_PROPERTIES = [
|
|
|
|
"power",
|
2019-03-27 12:39:55 +00:00
|
|
|
"main_power",
|
2019-03-24 12:01:12 +00:00
|
|
|
"bright",
|
|
|
|
"ct",
|
|
|
|
"rgb",
|
|
|
|
"hue",
|
|
|
|
"sat",
|
|
|
|
"color_mode",
|
2019-03-25 18:06:43 +00:00
|
|
|
"bg_power",
|
2019-03-27 12:39:55 +00:00
|
|
|
"bg_lmode",
|
|
|
|
"bg_flowing",
|
|
|
|
"bg_ct",
|
|
|
|
"bg_bright",
|
|
|
|
"bg_hue",
|
|
|
|
"bg_sat",
|
|
|
|
"bg_rgb",
|
2019-03-24 12:01:12 +00:00
|
|
|
"nl_br",
|
|
|
|
"active_mode",
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
def setup(hass, config):
|
|
|
|
"""Set up the Yeelight bulbs."""
|
2019-03-28 02:51:22 +00:00
|
|
|
conf = config.get(DOMAIN, {})
|
2019-03-25 07:50:47 +00:00
|
|
|
yeelight_data = hass.data[DATA_YEELIGHT] = {}
|
2019-03-24 12:01:12 +00:00
|
|
|
|
2019-06-13 16:42:47 +00:00
|
|
|
def device_discovered(_, info):
|
2019-03-24 12:01:12 +00:00
|
|
|
_LOGGER.debug("Adding autodetected %s", info['hostname'])
|
|
|
|
|
|
|
|
device_type = info['device_type']
|
|
|
|
|
|
|
|
name = "yeelight_%s_%s" % (device_type,
|
|
|
|
info['properties']['mac'])
|
|
|
|
ipaddr = info[CONF_HOST]
|
|
|
|
device_config = DEVICE_SCHEMA({
|
|
|
|
CONF_NAME: name,
|
|
|
|
CONF_MODEL: device_type
|
|
|
|
})
|
|
|
|
|
|
|
|
_setup_device(hass, config, ipaddr, device_config)
|
|
|
|
|
|
|
|
discovery.listen(hass, SERVICE_YEELIGHT, device_discovered)
|
|
|
|
|
2019-06-13 16:42:47 +00:00
|
|
|
def update(_):
|
2019-04-07 20:05:38 +00:00
|
|
|
for device in list(yeelight_data.values()):
|
2019-03-24 12:01:12 +00:00
|
|
|
device.update()
|
|
|
|
|
2019-03-25 07:50:47 +00:00
|
|
|
track_time_interval(
|
2019-03-28 02:51:22 +00:00
|
|
|
hass, update, conf.get(CONF_SCAN_INTERVAL, SCAN_INTERVAL)
|
2019-03-24 12:01:12 +00:00
|
|
|
)
|
|
|
|
|
2019-06-13 16:42:47 +00:00
|
|
|
def load_platforms(ipaddr):
|
|
|
|
platform_config = hass.data[DATA_YEELIGHT][ipaddr].config.copy()
|
|
|
|
platform_config[CONF_HOST] = ipaddr
|
|
|
|
platform_config[CONF_CUSTOM_EFFECTS] = \
|
|
|
|
config.get(DOMAIN, {}).get(CONF_CUSTOM_EFFECTS, {})
|
|
|
|
load_platform(hass, LIGHT_DOMAIN, DOMAIN, platform_config, config)
|
|
|
|
load_platform(hass, BINARY_SENSOR_DOMAIN, DOMAIN, platform_config,
|
|
|
|
config)
|
|
|
|
|
|
|
|
dispatcher_connect(hass, DEVICE_INITIALIZED, load_platforms)
|
|
|
|
|
2019-03-28 02:51:22 +00:00
|
|
|
if DOMAIN in config:
|
|
|
|
for ipaddr, device_config in conf[CONF_DEVICES].items():
|
|
|
|
_LOGGER.debug("Adding configured %s", device_config[CONF_NAME])
|
|
|
|
_setup_device(hass, config, ipaddr, device_config)
|
2019-03-24 12:01:12 +00:00
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2019-06-13 16:42:47 +00:00
|
|
|
def _setup_device(hass, _, ipaddr, device_config):
|
2019-03-25 07:50:47 +00:00
|
|
|
devices = hass.data[DATA_YEELIGHT]
|
2019-03-24 12:01:12 +00:00
|
|
|
|
|
|
|
if ipaddr in devices:
|
|
|
|
return
|
|
|
|
|
|
|
|
device = YeelightDevice(hass, ipaddr, device_config)
|
|
|
|
|
|
|
|
devices[ipaddr] = device
|
2019-06-13 16:42:47 +00:00
|
|
|
hass.add_job(device.setup)
|
2019-03-24 12:01:12 +00:00
|
|
|
|
|
|
|
|
|
|
|
class YeelightDevice:
|
|
|
|
"""Represents single Yeelight device."""
|
|
|
|
|
|
|
|
def __init__(self, hass, ipaddr, config):
|
|
|
|
"""Initialize device."""
|
|
|
|
self._hass = hass
|
|
|
|
self._config = config
|
|
|
|
self._ipaddr = ipaddr
|
|
|
|
self._name = config.get(CONF_NAME)
|
|
|
|
self._model = config.get(CONF_MODEL)
|
2019-06-13 16:42:47 +00:00
|
|
|
self._bulb_device = Bulb(self.ipaddr, model=self._model)
|
|
|
|
self._device_type = None
|
2019-03-29 17:43:29 +00:00
|
|
|
self._available = False
|
2019-06-13 16:42:47 +00:00
|
|
|
self._initialized = False
|
2019-03-24 12:01:12 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def bulb(self):
|
|
|
|
"""Return bulb device."""
|
|
|
|
return self._bulb_device
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the device if any."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def config(self):
|
|
|
|
"""Return device config."""
|
|
|
|
return self._config
|
|
|
|
|
|
|
|
@property
|
|
|
|
def ipaddr(self):
|
|
|
|
"""Return ip address."""
|
|
|
|
return self._ipaddr
|
|
|
|
|
2019-03-29 17:43:29 +00:00
|
|
|
@property
|
|
|
|
def available(self):
|
|
|
|
"""Return true is device is available."""
|
|
|
|
return self._available
|
|
|
|
|
2019-06-13 16:42:47 +00:00
|
|
|
@property
|
|
|
|
def model(self):
|
|
|
|
"""Return configured device model."""
|
|
|
|
return self._model
|
|
|
|
|
2019-03-24 12:01:12 +00:00
|
|
|
@property
|
|
|
|
def is_nightlight_enabled(self) -> bool:
|
|
|
|
"""Return true / false if nightlight is currently enabled."""
|
2019-03-29 17:43:29 +00:00
|
|
|
if self.bulb is None:
|
2019-03-24 12:01:12 +00:00
|
|
|
return False
|
|
|
|
|
2019-06-13 16:42:47 +00:00
|
|
|
return self._active_mode == ACTIVE_MODE_NIGHTLIGHT
|
2019-03-24 12:01:12 +00:00
|
|
|
|
2019-03-25 18:06:43 +00:00
|
|
|
@property
|
|
|
|
def is_nightlight_supported(self) -> bool:
|
|
|
|
"""Return true / false if nightlight is supported."""
|
2019-06-13 16:42:47 +00:00
|
|
|
if self.model:
|
|
|
|
return self.bulb.get_model_specs().get('night_light', False)
|
|
|
|
|
|
|
|
return self._active_mode is not None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def _active_mode(self):
|
|
|
|
return self.bulb.last_properties.get('active_mode')
|
2019-03-25 18:06:43 +00:00
|
|
|
|
2019-03-27 12:39:55 +00:00
|
|
|
@property
|
2019-06-13 16:42:47 +00:00
|
|
|
def type(self):
|
|
|
|
"""Return bulb type."""
|
|
|
|
if not self._device_type:
|
|
|
|
self._device_type = self.bulb.bulb_type
|
|
|
|
|
|
|
|
return self._device_type
|
2019-03-27 12:39:55 +00:00
|
|
|
|
|
|
|
def turn_on(self, duration=DEFAULT_TRANSITION, light_type=None):
|
2019-03-24 12:01:12 +00:00
|
|
|
"""Turn on device."""
|
|
|
|
try:
|
2019-03-29 17:43:29 +00:00
|
|
|
self.bulb.turn_on(duration=duration, light_type=light_type)
|
2019-04-07 14:07:34 +00:00
|
|
|
except BulbException as ex:
|
2019-03-24 12:01:12 +00:00
|
|
|
_LOGGER.error("Unable to turn the bulb on: %s", ex)
|
|
|
|
|
2019-03-27 12:39:55 +00:00
|
|
|
def turn_off(self, duration=DEFAULT_TRANSITION, light_type=None):
|
2019-03-24 12:01:12 +00:00
|
|
|
"""Turn off device."""
|
|
|
|
try:
|
2019-03-29 17:43:29 +00:00
|
|
|
self.bulb.turn_off(duration=duration, light_type=light_type)
|
2019-04-07 14:07:34 +00:00
|
|
|
except BulbException as ex:
|
2019-06-13 16:42:47 +00:00
|
|
|
_LOGGER.error("Unable to turn the bulb off: %s, %s: %s",
|
|
|
|
self.ipaddr, self.name, ex)
|
2019-03-24 12:01:12 +00:00
|
|
|
|
2019-06-13 16:42:47 +00:00
|
|
|
def _update_properties(self):
|
2019-03-24 12:01:12 +00:00
|
|
|
"""Read new properties from the device."""
|
|
|
|
if not self.bulb:
|
|
|
|
return
|
|
|
|
|
2019-03-29 17:43:29 +00:00
|
|
|
try:
|
|
|
|
self.bulb.get_properties(UPDATE_REQUEST_PROPERTIES)
|
|
|
|
self._available = True
|
2019-06-13 16:42:47 +00:00
|
|
|
if not self._initialized:
|
|
|
|
self._initialize_device()
|
2019-04-07 14:07:34 +00:00
|
|
|
except BulbException as ex:
|
2019-03-29 17:43:29 +00:00
|
|
|
if self._available: # just inform once
|
2019-06-13 16:42:47 +00:00
|
|
|
_LOGGER.error("Unable to update device %s, %s: %s",
|
|
|
|
self.ipaddr, self.name, ex)
|
2019-03-29 17:43:29 +00:00
|
|
|
self._available = False
|
|
|
|
|
2019-06-13 16:42:47 +00:00
|
|
|
return self._available
|
|
|
|
|
|
|
|
def _initialize_device(self):
|
|
|
|
self._initialized = True
|
|
|
|
dispatcher_send(self._hass, DEVICE_INITIALIZED, self.ipaddr)
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Update device properties and send data updated signal."""
|
|
|
|
self._update_properties()
|
2019-04-07 14:07:50 +00:00
|
|
|
dispatcher_send(self._hass, DATA_UPDATED.format(self._ipaddr))
|
2019-06-13 16:42:47 +00:00
|
|
|
|
|
|
|
def setup(self):
|
|
|
|
"""Fetch initial device properties."""
|
|
|
|
initial_update = self._update_properties()
|
|
|
|
|
|
|
|
# We can build correct class anyway.
|
|
|
|
if not initial_update and self.model:
|
|
|
|
self._initialize_device()
|