Add LCN cover platform (#20288)
* Add LCN cover platform * Removed unused default value * Moved cover component to lcn platform directory. Small changes due to change request * Closed state is set before updating
This commit is contained in:
parent
2aa7bdb1d5
commit
8f70c16863
2 changed files with 115 additions and 12 deletions
|
@ -5,8 +5,8 @@ import re
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
|
||||||
from homeassistant.const import (
|
from homeassistant.const import (
|
||||||
CONF_ADDRESS, CONF_HOST, CONF_LIGHTS, CONF_NAME, CONF_PASSWORD, CONF_PORT,
|
CONF_ADDRESS, CONF_COVERS, CONF_HOST, CONF_LIGHTS, CONF_NAME,
|
||||||
CONF_SWITCHES, CONF_USERNAME)
|
CONF_PASSWORD, CONF_PORT, CONF_SWITCHES, CONF_USERNAME)
|
||||||
import homeassistant.helpers.config_validation as cv
|
import homeassistant.helpers.config_validation as cv
|
||||||
from homeassistant.helpers.discovery import async_load_platform
|
from homeassistant.helpers.discovery import async_load_platform
|
||||||
from homeassistant.helpers.entity import Entity
|
from homeassistant.helpers.entity import Entity
|
||||||
|
@ -25,11 +25,15 @@ CONF_OUTPUT = 'output'
|
||||||
CONF_TRANSITION = 'transition'
|
CONF_TRANSITION = 'transition'
|
||||||
CONF_DIMMABLE = 'dimmable'
|
CONF_DIMMABLE = 'dimmable'
|
||||||
CONF_CONNECTIONS = 'connections'
|
CONF_CONNECTIONS = 'connections'
|
||||||
|
CONF_MOTOR = 'motor'
|
||||||
|
|
||||||
DIM_MODES = ['STEPS50', 'STEPS200']
|
DIM_MODES = ['STEPS50', 'STEPS200']
|
||||||
OUTPUT_PORTS = ['OUTPUT1', 'OUTPUT2', 'OUTPUT3', 'OUTPUT4']
|
OUTPUT_PORTS = ['OUTPUT1', 'OUTPUT2', 'OUTPUT3', 'OUTPUT4']
|
||||||
RELAY_PORTS = ['RELAY1', 'RELAY2', 'RELAY3', 'RELAY4',
|
RELAY_PORTS = ['RELAY1', 'RELAY2', 'RELAY3', 'RELAY4',
|
||||||
'RELAY5', 'RELAY6', 'RELAY7', 'RELAY8']
|
'RELAY5', 'RELAY6', 'RELAY7', 'RELAY8',
|
||||||
|
'MOTORONOFF1', 'MOTORUPDOWN1', 'MOTORONOFF2', 'MOTORUPDOWN2',
|
||||||
|
'MOTORONOFF3', 'MOTORUPDOWN3', 'MOTORONOFF4', 'MOTORUPDOWN4']
|
||||||
|
MOTOR_PORTS = ['MOTOR1', 'MOTOR2', 'MOTOR3', 'MOTOR4']
|
||||||
|
|
||||||
# Regex for address validation
|
# Regex for address validation
|
||||||
PATTERN_ADDRESS = re.compile('^((?P<conn_id>\\w+)\\.)?s?(?P<seg_id>\\d+)'
|
PATTERN_ADDRESS = re.compile('^((?P<conn_id>\\w+)\\.)?s?(?P<seg_id>\\d+)'
|
||||||
|
@ -78,6 +82,12 @@ def is_address(value):
|
||||||
raise vol.error.Invalid('Not a valid address string.')
|
raise vol.error.Invalid('Not a valid address string.')
|
||||||
|
|
||||||
|
|
||||||
|
COVERS_SCHEMA = vol.Schema({
|
||||||
|
vol.Required(CONF_NAME): cv.string,
|
||||||
|
vol.Required(CONF_ADDRESS): is_address,
|
||||||
|
vol.Required(CONF_MOTOR): vol.All(vol.Upper, vol.In(MOTOR_PORTS))
|
||||||
|
})
|
||||||
|
|
||||||
LIGHTS_SCHEMA = vol.Schema({
|
LIGHTS_SCHEMA = vol.Schema({
|
||||||
vol.Required(CONF_NAME): cv.string,
|
vol.Required(CONF_NAME): cv.string,
|
||||||
vol.Required(CONF_ADDRESS): is_address,
|
vol.Required(CONF_ADDRESS): is_address,
|
||||||
|
@ -111,8 +121,12 @@ CONFIG_SCHEMA = vol.Schema({
|
||||||
DOMAIN: vol.Schema({
|
DOMAIN: vol.Schema({
|
||||||
vol.Required(CONF_CONNECTIONS): vol.All(
|
vol.Required(CONF_CONNECTIONS): vol.All(
|
||||||
cv.ensure_list, has_unique_connection_names, [CONNECTION_SCHEMA]),
|
cv.ensure_list, has_unique_connection_names, [CONNECTION_SCHEMA]),
|
||||||
vol.Optional(CONF_LIGHTS): vol.All(cv.ensure_list, [LIGHTS_SCHEMA]),
|
vol.Optional(CONF_COVERS): vol.All(
|
||||||
vol.Optional(CONF_SWITCHES): vol.All(cv.ensure_list, [SWITCHES_SCHEMA])
|
cv.ensure_list, [COVERS_SCHEMA]),
|
||||||
|
vol.Optional(CONF_LIGHTS): vol.All(
|
||||||
|
cv.ensure_list, [LIGHTS_SCHEMA]),
|
||||||
|
vol.Optional(CONF_SWITCHES): vol.All(
|
||||||
|
cv.ensure_list, [SWITCHES_SCHEMA])
|
||||||
})
|
})
|
||||||
}, extra=vol.ALLOW_EXTRA)
|
}, extra=vol.ALLOW_EXTRA)
|
||||||
|
|
||||||
|
@ -166,13 +180,13 @@ async def async_setup(hass, config):
|
||||||
|
|
||||||
hass.data[DATA_LCN][CONF_CONNECTIONS] = connections
|
hass.data[DATA_LCN][CONF_CONNECTIONS] = connections
|
||||||
|
|
||||||
hass.async_create_task(
|
# load platforms
|
||||||
async_load_platform(hass, 'light', DOMAIN,
|
for component, conf_key in (('cover', CONF_COVERS),
|
||||||
config[DOMAIN][CONF_LIGHTS], config))
|
('light', CONF_LIGHTS),
|
||||||
|
('switch', CONF_SWITCHES)):
|
||||||
hass.async_create_task(
|
hass.async_create_task(
|
||||||
async_load_platform(hass, 'switch', DOMAIN,
|
async_load_platform(hass, component, DOMAIN,
|
||||||
config[DOMAIN][CONF_SWITCHES], config))
|
config[DOMAIN][conf_key], config))
|
||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
89
homeassistant/components/lcn/cover.py
Executable file
89
homeassistant/components/lcn/cover.py
Executable file
|
@ -0,0 +1,89 @@
|
||||||
|
"""Support for LCN covers."""
|
||||||
|
from homeassistant.components.cover import CoverDevice
|
||||||
|
from homeassistant.components.lcn import (
|
||||||
|
CONF_CONNECTIONS, CONF_MOTOR, DATA_LCN, LcnDevice, get_connection)
|
||||||
|
from homeassistant.const import CONF_ADDRESS
|
||||||
|
|
||||||
|
DEPENDENCIES = ['lcn']
|
||||||
|
|
||||||
|
|
||||||
|
async def async_setup_platform(hass, hass_config, async_add_entities,
|
||||||
|
discovery_info=None):
|
||||||
|
"""Setups the LCN cover platform."""
|
||||||
|
if discovery_info is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
import pypck
|
||||||
|
|
||||||
|
devices = []
|
||||||
|
for config in discovery_info:
|
||||||
|
address, connection_id = config[CONF_ADDRESS]
|
||||||
|
addr = pypck.lcn_addr.LcnAddr(*address)
|
||||||
|
connections = hass.data[DATA_LCN][CONF_CONNECTIONS]
|
||||||
|
connection = get_connection(connections, connection_id)
|
||||||
|
address_connection = connection.get_address_conn(addr)
|
||||||
|
|
||||||
|
devices.append(LcnCover(config, address_connection))
|
||||||
|
|
||||||
|
async_add_entities(devices)
|
||||||
|
|
||||||
|
|
||||||
|
class LcnCover(LcnDevice, CoverDevice):
|
||||||
|
"""Representation of a LCN cover."""
|
||||||
|
|
||||||
|
def __init__(self, config, address_connection):
|
||||||
|
"""Initialize the LCN cover."""
|
||||||
|
super().__init__(config, address_connection)
|
||||||
|
|
||||||
|
self.motor = self.pypck.lcn_defs.MotorPort[config[CONF_MOTOR]]
|
||||||
|
self.motor_port_onoff = self.motor.value * 2
|
||||||
|
self.motor_port_updown = self.motor_port_onoff + 1
|
||||||
|
|
||||||
|
self._closed = None
|
||||||
|
|
||||||
|
async def async_added_to_hass(self):
|
||||||
|
"""Run when entity about to be added to hass."""
|
||||||
|
await super().async_added_to_hass()
|
||||||
|
self.hass.async_create_task(
|
||||||
|
self.address_connection.activate_status_request_handler(
|
||||||
|
self.motor))
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_closed(self):
|
||||||
|
"""Return if the cover is closed."""
|
||||||
|
return self._closed
|
||||||
|
|
||||||
|
async def async_close_cover(self, **kwargs):
|
||||||
|
"""Close the cover."""
|
||||||
|
self._closed = True
|
||||||
|
states = [self.pypck.lcn_defs.MotorStateModifier.NOCHANGE] * 4
|
||||||
|
states[self.motor.value] = self.pypck.lcn_defs.MotorStateModifier.DOWN
|
||||||
|
self.address_connection.control_motors(states)
|
||||||
|
await self.async_update_ha_state()
|
||||||
|
|
||||||
|
async def async_open_cover(self, **kwargs):
|
||||||
|
"""Open the cover."""
|
||||||
|
self._closed = False
|
||||||
|
states = [self.pypck.lcn_defs.MotorStateModifier.NOCHANGE] * 4
|
||||||
|
states[self.motor.value] = self.pypck.lcn_defs.MotorStateModifier.UP
|
||||||
|
self.address_connection.control_motors(states)
|
||||||
|
await self.async_update_ha_state()
|
||||||
|
|
||||||
|
async def async_stop_cover(self, **kwargs):
|
||||||
|
"""Stop the cover."""
|
||||||
|
self._closed = None
|
||||||
|
states = [self.pypck.lcn_defs.MotorStateModifier.NOCHANGE] * 4
|
||||||
|
states[self.motor.value] = self.pypck.lcn_defs.MotorStateModifier.STOP
|
||||||
|
self.address_connection.control_motors(states)
|
||||||
|
await self.async_update_ha_state()
|
||||||
|
|
||||||
|
def input_received(self, input_obj):
|
||||||
|
"""Set cover states when LCN input object (command) is received."""
|
||||||
|
if not isinstance(input_obj, self.pypck.inputs.ModStatusRelays):
|
||||||
|
return
|
||||||
|
|
||||||
|
states = input_obj.states # list of boolean values (relay on/off)
|
||||||
|
if states[self.motor_port_onoff]: # motor is on
|
||||||
|
self._closed = states[self.motor_port_updown] # set direction
|
||||||
|
|
||||||
|
self.async_schedule_update_ha_state()
|
Loading…
Add table
Add a link
Reference in a new issue