mirror of
https://github.com/home-assistant/core.git
synced 2025-07-23 21:27:38 +00:00
Add LiteJet (a lighting control system) component (#4125)
* Initial submission of LiteJet integration. * Add LiteJet switch pressed automation trigger. (State changes are too slow to catch a press-release.) Add LiteJet scene, replacing commented out code that treated these as lights. Include LiteJet numbers in the device state so that it is easy to lookup entity -> number. * Fix missing global. * Allow light's brightness to be set explicitly. * Support optional 'ignore' key to ignore prefixes of loads, switches, and scenes that weren't configured for use in the LiteJet system. * Fix lint errors and warnings. * Cleanup header comments. Default to not creating LiteJet switches as these are generally not useful. * Lint fixes. * Fixes from pull request feedback. * Use hass.data instead of globals for data storage. * Fix lint warnings.
This commit is contained in:
parent
2a7b7ebd6a
commit
ba13951fff
@ -37,6 +37,9 @@ omit =
|
|||||||
homeassistant/components/isy994.py
|
homeassistant/components/isy994.py
|
||||||
homeassistant/components/*/isy994.py
|
homeassistant/components/*/isy994.py
|
||||||
|
|
||||||
|
homeassistant/components/litejet.py
|
||||||
|
homeassistant/components/*/litejet.py
|
||||||
|
|
||||||
homeassistant/components/modbus.py
|
homeassistant/components/modbus.py
|
||||||
homeassistant/components/*/modbus.py
|
homeassistant/components/*/modbus.py
|
||||||
|
|
||||||
|
41
homeassistant/components/automation/litejet.py
Normal file
41
homeassistant/components/automation/litejet.py
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
"""
|
||||||
|
Trigger an automation when a LiteJet switch is released.
|
||||||
|
|
||||||
|
For more details about this platform, please refer to the documentation at
|
||||||
|
https://home-assistant.io/components/automation.litejet/
|
||||||
|
"""
|
||||||
|
import logging
|
||||||
|
|
||||||
|
import voluptuous as vol
|
||||||
|
|
||||||
|
from homeassistant.core import callback
|
||||||
|
from homeassistant.const import CONF_PLATFORM
|
||||||
|
import homeassistant.helpers.config_validation as cv
|
||||||
|
|
||||||
|
DEPENDENCIES = ['litejet']
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
CONF_NUMBER = 'number'
|
||||||
|
|
||||||
|
TRIGGER_SCHEMA = vol.Schema({
|
||||||
|
vol.Required(CONF_PLATFORM): 'litejet',
|
||||||
|
vol.Required(CONF_NUMBER): cv.positive_int
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
def async_trigger(hass, config, action):
|
||||||
|
"""Listen for events based on configuration."""
|
||||||
|
number = config.get(CONF_NUMBER)
|
||||||
|
|
||||||
|
@callback
|
||||||
|
def call_action():
|
||||||
|
"""Call action with right context."""
|
||||||
|
hass.async_run_job(action, {
|
||||||
|
'trigger': {
|
||||||
|
CONF_PLATFORM: 'litejet',
|
||||||
|
CONF_NUMBER: number
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
hass.data['litejet_system'].on_switch_released(number, call_action)
|
94
homeassistant/components/light/litejet.py
Normal file
94
homeassistant/components/light/litejet.py
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
"""
|
||||||
|
Support for LiteJet lights.
|
||||||
|
|
||||||
|
For more details about this platform, please refer to the documentation at
|
||||||
|
https://home-assistant.io/components/light.litejet/
|
||||||
|
"""
|
||||||
|
|
||||||
|
import logging
|
||||||
|
|
||||||
|
import homeassistant.components.litejet as litejet
|
||||||
|
from homeassistant.components.light import ATTR_BRIGHTNESS, Light
|
||||||
|
|
||||||
|
DEPENDENCIES = ['litejet']
|
||||||
|
|
||||||
|
ATTR_NUMBER = 'number'
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
||||||
|
"""Setup lights for the LiteJet platform."""
|
||||||
|
litejet_ = hass.data['litejet_system']
|
||||||
|
|
||||||
|
devices = []
|
||||||
|
for i in litejet_.loads():
|
||||||
|
name = litejet_.get_load_name(i)
|
||||||
|
if not litejet.is_ignored(hass, name):
|
||||||
|
devices.append(LiteJetLight(hass, litejet_, i, name))
|
||||||
|
add_devices(devices)
|
||||||
|
|
||||||
|
|
||||||
|
class LiteJetLight(Light):
|
||||||
|
"""Represents a single LiteJet light."""
|
||||||
|
|
||||||
|
def __init__(self, hass, lj, i, name):
|
||||||
|
"""Initialize a LiteJet light."""
|
||||||
|
self._hass = hass
|
||||||
|
self._lj = lj
|
||||||
|
self._index = i
|
||||||
|
self._brightness = 0
|
||||||
|
self._name = name
|
||||||
|
|
||||||
|
lj.on_load_activated(i, self._on_load_changed)
|
||||||
|
lj.on_load_deactivated(i, self._on_load_changed)
|
||||||
|
|
||||||
|
self.update()
|
||||||
|
|
||||||
|
def _on_load_changed(self):
|
||||||
|
"""Called on a LiteJet thread when a load's state changes."""
|
||||||
|
_LOGGER.debug("Updating due to notification for %s", self._name)
|
||||||
|
self._hass.loop.create_task(self.async_update_ha_state(True))
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
"""The light's name."""
|
||||||
|
return self._name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def brightness(self):
|
||||||
|
"""Return the light's brightness."""
|
||||||
|
return self._brightness
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_on(self):
|
||||||
|
"""Return if the light is on."""
|
||||||
|
return self._brightness != 0
|
||||||
|
|
||||||
|
@property
|
||||||
|
def should_poll(self):
|
||||||
|
"""Return that lights do not require polling."""
|
||||||
|
return False
|
||||||
|
|
||||||
|
@property
|
||||||
|
def device_state_attributes(self):
|
||||||
|
"""Return the device state attributes."""
|
||||||
|
return {
|
||||||
|
ATTR_NUMBER: self._index
|
||||||
|
}
|
||||||
|
|
||||||
|
def turn_on(self, **kwargs):
|
||||||
|
"""Turn on the light."""
|
||||||
|
if ATTR_BRIGHTNESS in kwargs:
|
||||||
|
brightness = int(kwargs[ATTR_BRIGHTNESS] / 255 * 99)
|
||||||
|
self._lj.activate_load_at(self._index, brightness, 0)
|
||||||
|
else:
|
||||||
|
self._lj.activate_load(self._index)
|
||||||
|
|
||||||
|
def turn_off(self, **kwargs):
|
||||||
|
"""Turn off the light."""
|
||||||
|
self._lj.deactivate_load(self._index)
|
||||||
|
|
||||||
|
def update(self):
|
||||||
|
"""Retrieve the light's brightness from the LiteJet system."""
|
||||||
|
self._brightness = self._lj.get_load_level(self._index) / 99 * 255
|
53
homeassistant/components/litejet.py
Normal file
53
homeassistant/components/litejet.py
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
"""Allows the LiteJet lighting system to be controlled by Home Assistant.
|
||||||
|
|
||||||
|
For more details about this component, please refer to the documentation at
|
||||||
|
https://home-assistant.io/components/litejet/
|
||||||
|
"""
|
||||||
|
import logging
|
||||||
|
import voluptuous as vol
|
||||||
|
|
||||||
|
from homeassistant.helpers import discovery
|
||||||
|
from homeassistant.const import CONF_URL
|
||||||
|
import homeassistant.helpers.config_validation as cv
|
||||||
|
|
||||||
|
DOMAIN = 'litejet'
|
||||||
|
|
||||||
|
REQUIREMENTS = ['pylitejet==0.1']
|
||||||
|
|
||||||
|
CONF_EXCLUDE_NAMES = 'exclude_names'
|
||||||
|
CONF_INCLUDE_SWITCHES = 'include_switches'
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
CONFIG_SCHEMA = vol.Schema({
|
||||||
|
DOMAIN: vol.Schema({
|
||||||
|
vol.Required(CONF_URL): cv.string,
|
||||||
|
vol.Optional(CONF_EXCLUDE_NAMES): vol.All(cv.ensure_list, [cv.string]),
|
||||||
|
vol.Optional(CONF_INCLUDE_SWITCHES, default=False): cv.boolean
|
||||||
|
})
|
||||||
|
}, extra=vol.ALLOW_EXTRA)
|
||||||
|
|
||||||
|
|
||||||
|
def setup(hass, config):
|
||||||
|
"""Initialize the LiteJet component."""
|
||||||
|
from pylitejet import LiteJet
|
||||||
|
|
||||||
|
url = config[DOMAIN].get(CONF_URL)
|
||||||
|
|
||||||
|
hass.data['litejet_system'] = LiteJet(url)
|
||||||
|
hass.data['litejet_config'] = config[DOMAIN]
|
||||||
|
|
||||||
|
discovery.load_platform(hass, 'light', DOMAIN, {}, config)
|
||||||
|
if config[DOMAIN].get(CONF_INCLUDE_SWITCHES):
|
||||||
|
discovery.load_platform(hass, 'switch', DOMAIN, {}, config)
|
||||||
|
discovery.load_platform(hass, 'scene', DOMAIN, {}, config)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
def is_ignored(hass, name):
|
||||||
|
"""Determine if a load, switch, or scene should be ignored."""
|
||||||
|
for prefix in hass.data['litejet_config'].get(CONF_EXCLUDE_NAMES, []):
|
||||||
|
if name.startswith(prefix):
|
||||||
|
return True
|
||||||
|
return False
|
58
homeassistant/components/scene/litejet.py
Normal file
58
homeassistant/components/scene/litejet.py
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
"""
|
||||||
|
Support for LiteJet scenes.
|
||||||
|
|
||||||
|
For more details about this platform, please refer to the documentation at
|
||||||
|
https://home-assistant.io/components/scene.litejet/
|
||||||
|
"""
|
||||||
|
import logging
|
||||||
|
import homeassistant.components.litejet as litejet
|
||||||
|
from homeassistant.components.scene import Scene
|
||||||
|
|
||||||
|
DEPENDENCIES = ['litejet']
|
||||||
|
|
||||||
|
ATTR_NUMBER = 'number'
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
||||||
|
"""Setup scenes for the LiteJet platform."""
|
||||||
|
litejet_ = hass.data['litejet_system']
|
||||||
|
|
||||||
|
devices = []
|
||||||
|
for i in litejet_.scenes():
|
||||||
|
name = litejet_.get_scene_name(i)
|
||||||
|
if not litejet.is_ignored(hass, name):
|
||||||
|
devices.append(LiteJetScene(litejet_, i, name))
|
||||||
|
add_devices(devices)
|
||||||
|
|
||||||
|
|
||||||
|
class LiteJetScene(Scene):
|
||||||
|
"""Represents a single LiteJet scene."""
|
||||||
|
|
||||||
|
def __init__(self, lj, i, name):
|
||||||
|
"""Initialize the scene."""
|
||||||
|
self._lj = lj
|
||||||
|
self._index = i
|
||||||
|
self._name = name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
"""Return the name of the scene."""
|
||||||
|
return self._name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def should_poll(self):
|
||||||
|
"""Return that polling is not necessary."""
|
||||||
|
return False
|
||||||
|
|
||||||
|
@property
|
||||||
|
def device_state_attributes(self):
|
||||||
|
"""Return the device-specific state attributes."""
|
||||||
|
return {
|
||||||
|
ATTR_NUMBER: self._index
|
||||||
|
}
|
||||||
|
|
||||||
|
def activate(self, **kwargs):
|
||||||
|
"""Activate the scene."""
|
||||||
|
self._lj.activate_scene(self._index)
|
84
homeassistant/components/switch/litejet.py
Normal file
84
homeassistant/components/switch/litejet.py
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
"""
|
||||||
|
Support for LiteJet switch.
|
||||||
|
|
||||||
|
For more details about this platform, please refer to the documentation at
|
||||||
|
https://home-assistant.io/components/switch.litejet/
|
||||||
|
"""
|
||||||
|
import logging
|
||||||
|
import homeassistant.components.litejet as litejet
|
||||||
|
from homeassistant.components.switch import SwitchDevice
|
||||||
|
|
||||||
|
DEPENDENCIES = ['litejet']
|
||||||
|
|
||||||
|
ATTR_NUMBER = 'number'
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
||||||
|
"""Setup the LiteJet switch platform."""
|
||||||
|
litejet_ = hass.data['litejet_system']
|
||||||
|
|
||||||
|
devices = []
|
||||||
|
for i in litejet_.button_switches():
|
||||||
|
name = litejet_.get_switch_name(i)
|
||||||
|
if not litejet.is_ignored(hass, name):
|
||||||
|
devices.append(LiteJetSwitch(hass, litejet_, i, name))
|
||||||
|
add_devices(devices)
|
||||||
|
|
||||||
|
|
||||||
|
class LiteJetSwitch(SwitchDevice):
|
||||||
|
"""Represents a single LiteJet switch."""
|
||||||
|
|
||||||
|
def __init__(self, hass, lj, i, name):
|
||||||
|
"""Initialize a LiteJet switch."""
|
||||||
|
self._hass = hass
|
||||||
|
self._lj = lj
|
||||||
|
self._index = i
|
||||||
|
self._state = False
|
||||||
|
self._name = name
|
||||||
|
|
||||||
|
lj.on_switch_pressed(i, self._on_switch_pressed)
|
||||||
|
lj.on_switch_released(i, self._on_switch_released)
|
||||||
|
|
||||||
|
self.update()
|
||||||
|
|
||||||
|
def _on_switch_pressed(self):
|
||||||
|
_LOGGER.debug("Updating pressed for %s", self._name)
|
||||||
|
self._state = True
|
||||||
|
self._hass.loop.create_task(self.async_update_ha_state())
|
||||||
|
|
||||||
|
def _on_switch_released(self):
|
||||||
|
_LOGGER.debug("Updating released for %s", self._name)
|
||||||
|
self._state = False
|
||||||
|
self._hass.loop.create_task(self.async_update_ha_state())
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
"""Return the name of the switch."""
|
||||||
|
return self._name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_on(self):
|
||||||
|
"""Return if the switch is pressed."""
|
||||||
|
return self._state
|
||||||
|
|
||||||
|
@property
|
||||||
|
def should_poll(self):
|
||||||
|
"""Return that polling is not necessary."""
|
||||||
|
return False
|
||||||
|
|
||||||
|
@property
|
||||||
|
def device_state_attributes(self):
|
||||||
|
"""Return the device-specific state attributes."""
|
||||||
|
return {
|
||||||
|
ATTR_NUMBER: self._index
|
||||||
|
}
|
||||||
|
|
||||||
|
def turn_on(self, **kwargs):
|
||||||
|
"""Press the switch."""
|
||||||
|
self._lj.press_switch(self._index)
|
||||||
|
|
||||||
|
def turn_off(self, **kwargs):
|
||||||
|
"""Release the switch."""
|
||||||
|
self._lj.release_switch(self._index)
|
@ -367,6 +367,9 @@ pyicloud==0.9.1
|
|||||||
# homeassistant.components.sensor.lastfm
|
# homeassistant.components.sensor.lastfm
|
||||||
pylast==1.6.0
|
pylast==1.6.0
|
||||||
|
|
||||||
|
# homeassistant.components.litejet
|
||||||
|
pylitejet==0.1
|
||||||
|
|
||||||
# homeassistant.components.sensor.loopenergy
|
# homeassistant.components.sensor.loopenergy
|
||||||
pyloopenergy==0.0.15
|
pyloopenergy==0.0.15
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user