2019-02-13 20:21:14 +00:00
|
|
|
"""Support for Z-Wave climate devices."""
|
2016-08-19 07:17:28 +00:00
|
|
|
# Because we do not compile openzwave on CI
|
|
|
|
import logging
|
2018-10-16 12:58:25 +00:00
|
|
|
from homeassistant.core import callback
|
2019-02-14 19:34:43 +00:00
|
|
|
from homeassistant.components.climate import ClimateDevice
|
|
|
|
from homeassistant.components.climate.const import (
|
|
|
|
DOMAIN, STATE_AUTO, STATE_COOL, STATE_HEAT,
|
2018-06-29 14:09:46 +00:00
|
|
|
SUPPORT_TARGET_TEMPERATURE, SUPPORT_FAN_MODE,
|
2017-11-29 10:01:28 +00:00
|
|
|
SUPPORT_OPERATION_MODE, SUPPORT_SWING_MODE)
|
2018-10-16 12:58:25 +00:00
|
|
|
from homeassistant.components.zwave import ZWaveDeviceEntity
|
2016-09-09 17:06:53 +00:00
|
|
|
from homeassistant.const import (
|
2018-06-29 14:09:46 +00:00
|
|
|
STATE_OFF, TEMP_CELSIUS, TEMP_FAHRENHEIT, ATTR_TEMPERATURE)
|
2018-10-16 12:58:25 +00:00
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
CONF_NAME = 'name'
|
2016-11-18 22:05:03 +00:00
|
|
|
DEFAULT_NAME = 'Z-Wave Climate'
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
REMOTEC = 0x5254
|
|
|
|
REMOTEC_ZXT_120 = 0x8377
|
|
|
|
REMOTEC_ZXT_120_THERMOSTAT = (REMOTEC, REMOTEC_ZXT_120)
|
2016-11-18 20:42:30 +00:00
|
|
|
ATTR_OPERATING_STATE = 'operating_state'
|
|
|
|
ATTR_FAN_STATE = 'fan_state'
|
2016-09-07 09:22:51 +00:00
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
WORKAROUND_ZXT_120 = 'zxt_120'
|
|
|
|
|
|
|
|
DEVICE_MAPPINGS = {
|
2016-11-15 12:14:29 +00:00
|
|
|
REMOTEC_ZXT_120_THERMOSTAT: WORKAROUND_ZXT_120
|
2016-08-19 07:17:28 +00:00
|
|
|
}
|
|
|
|
|
2018-06-29 14:09:46 +00:00
|
|
|
STATE_MAPPINGS = {
|
|
|
|
'Off': STATE_OFF,
|
|
|
|
'Heat': STATE_HEAT,
|
|
|
|
'Heat Mode': STATE_HEAT,
|
|
|
|
'Heat (Default)': STATE_HEAT,
|
|
|
|
'Cool': STATE_COOL,
|
|
|
|
'Auto': STATE_AUTO,
|
|
|
|
}
|
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
|
2019-02-13 20:21:14 +00:00
|
|
|
async def async_setup_platform(
|
|
|
|
hass, config, async_add_entities, discovery_info=None):
|
2018-10-16 12:58:25 +00:00
|
|
|
"""Old method of setting up Z-Wave climate devices."""
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(hass, config_entry, async_add_entities):
|
|
|
|
"""Set up Z-Wave Climate device from Config Entry."""
|
|
|
|
@callback
|
|
|
|
def async_add_climate(climate):
|
|
|
|
"""Add Z-Wave Climate Device."""
|
|
|
|
async_add_entities([climate])
|
|
|
|
|
|
|
|
async_dispatcher_connect(hass, 'zwave_new_climate', async_add_climate)
|
|
|
|
|
|
|
|
|
2017-03-14 23:55:33 +00:00
|
|
|
def get_device(hass, values, **kwargs):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Create Z-Wave entity device."""
|
2016-08-31 19:50:03 +00:00
|
|
|
temp_unit = hass.config.units.temperature_unit
|
2017-03-14 23:55:33 +00:00
|
|
|
return ZWaveClimate(values, temp_unit)
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ZWaveClimate(ZWaveDeviceEntity, ClimateDevice):
|
2016-11-18 22:05:03 +00:00
|
|
|
"""Representation of a Z-Wave Climate device."""
|
2016-08-19 07:17:28 +00:00
|
|
|
|
2017-03-14 23:55:33 +00:00
|
|
|
def __init__(self, values, temp_unit):
|
2016-11-18 22:05:03 +00:00
|
|
|
"""Initialize the Z-Wave climate device."""
|
2017-03-14 23:55:33 +00:00
|
|
|
ZWaveDeviceEntity.__init__(self, values, DOMAIN)
|
2016-08-19 07:17:28 +00:00
|
|
|
self._target_temperature = None
|
|
|
|
self._current_temperature = None
|
|
|
|
self._current_operation = None
|
|
|
|
self._operation_list = None
|
2018-06-29 14:09:46 +00:00
|
|
|
self._operation_mapping = None
|
2016-10-28 05:25:17 +00:00
|
|
|
self._operating_state = None
|
2016-08-19 07:17:28 +00:00
|
|
|
self._current_fan_mode = None
|
|
|
|
self._fan_list = None
|
2016-11-15 12:14:29 +00:00
|
|
|
self._fan_state = None
|
2016-08-19 07:17:28 +00:00
|
|
|
self._current_swing_mode = None
|
|
|
|
self._swing_list = None
|
2016-08-31 19:50:03 +00:00
|
|
|
self._unit = temp_unit
|
|
|
|
_LOGGER.debug("temp_unit is %s", self._unit)
|
2016-08-19 07:17:28 +00:00
|
|
|
self._zxt_120 = None
|
|
|
|
# Make sure that we have values for the key before converting to int
|
2017-03-14 23:55:33 +00:00
|
|
|
if (self.node.manufacturer_id.strip() and
|
|
|
|
self.node.product_id.strip()):
|
|
|
|
specific_sensor_key = (
|
|
|
|
int(self.node.manufacturer_id, 16),
|
|
|
|
int(self.node.product_id, 16))
|
2016-08-19 07:17:28 +00:00
|
|
|
if specific_sensor_key in DEVICE_MAPPINGS:
|
|
|
|
if DEVICE_MAPPINGS[specific_sensor_key] == WORKAROUND_ZXT_120:
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER.debug(
|
|
|
|
"Remotec ZXT-120 Zwave Thermostat workaround")
|
2016-08-19 07:17:28 +00:00
|
|
|
self._zxt_120 = 1
|
2017-02-18 07:56:05 +00:00
|
|
|
self.update_properties()
|
2016-08-19 07:17:28 +00:00
|
|
|
|
2017-11-29 10:01:28 +00:00
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Return the list of supported features."""
|
|
|
|
support = SUPPORT_TARGET_TEMPERATURE
|
|
|
|
if self.values.fan_mode:
|
|
|
|
support |= SUPPORT_FAN_MODE
|
|
|
|
if self.values.mode:
|
|
|
|
support |= SUPPORT_OPERATION_MODE
|
|
|
|
if self._zxt_120 == 1 and self.values.zxt_120_swing_mode:
|
|
|
|
support |= SUPPORT_SWING_MODE
|
|
|
|
return support
|
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
def update_properties(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Handle the data changes for node values."""
|
2016-08-19 07:17:28 +00:00
|
|
|
# Operation Mode
|
2017-03-14 23:55:33 +00:00
|
|
|
if self.values.mode:
|
2018-06-29 14:09:46 +00:00
|
|
|
self._operation_list = []
|
|
|
|
self._operation_mapping = {}
|
2017-03-14 23:55:33 +00:00
|
|
|
operation_list = self.values.mode.data_items
|
|
|
|
if operation_list:
|
2018-06-29 14:09:46 +00:00
|
|
|
for mode in operation_list:
|
|
|
|
ha_mode = STATE_MAPPINGS.get(mode)
|
|
|
|
if ha_mode and ha_mode not in self._operation_mapping:
|
|
|
|
self._operation_mapping[ha_mode] = mode
|
|
|
|
self._operation_list.append(ha_mode)
|
|
|
|
continue
|
|
|
|
self._operation_list.append(mode)
|
|
|
|
current_mode = self.values.mode.data
|
|
|
|
self._current_operation = next(
|
|
|
|
(key for key, value in self._operation_mapping.items()
|
|
|
|
if value == current_mode), current_mode)
|
2017-02-09 12:40:35 +00:00
|
|
|
_LOGGER.debug("self._operation_list=%s", self._operation_list)
|
|
|
|
_LOGGER.debug("self._current_operation=%s", self._current_operation)
|
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
# Current Temp
|
2017-03-14 23:55:33 +00:00
|
|
|
if self.values.temperature:
|
|
|
|
self._current_temperature = self.values.temperature.data
|
|
|
|
device_unit = self.values.temperature.units
|
|
|
|
if device_unit is not None:
|
|
|
|
self._unit = device_unit
|
2017-02-09 12:40:35 +00:00
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
# Fan Mode
|
2017-03-14 23:55:33 +00:00
|
|
|
if self.values.fan_mode:
|
|
|
|
self._current_fan_mode = self.values.fan_mode.data
|
|
|
|
fan_list = self.values.fan_mode.data_items
|
|
|
|
if fan_list:
|
|
|
|
self._fan_list = list(fan_list)
|
2017-02-09 12:40:35 +00:00
|
|
|
_LOGGER.debug("self._fan_list=%s", self._fan_list)
|
|
|
|
_LOGGER.debug("self._current_fan_mode=%s",
|
|
|
|
self._current_fan_mode)
|
2016-08-19 07:17:28 +00:00
|
|
|
# Swing mode
|
|
|
|
if self._zxt_120 == 1:
|
2017-03-14 23:55:33 +00:00
|
|
|
if self.values.zxt_120_swing_mode:
|
|
|
|
self._current_swing_mode = self.values.zxt_120_swing_mode.data
|
|
|
|
swing_list = self.values.zxt_120_swing_mode.data_items
|
|
|
|
if swing_list:
|
|
|
|
self._swing_list = list(swing_list)
|
2017-02-09 12:40:35 +00:00
|
|
|
_LOGGER.debug("self._swing_list=%s", self._swing_list)
|
|
|
|
_LOGGER.debug("self._current_swing_mode=%s",
|
|
|
|
self._current_swing_mode)
|
2016-08-31 19:50:03 +00:00
|
|
|
# Set point
|
2017-03-14 23:55:33 +00:00
|
|
|
if self.values.primary.data == 0:
|
|
|
|
_LOGGER.debug("Setpoint is 0, setting default to "
|
|
|
|
"current_temperature=%s",
|
|
|
|
self._current_temperature)
|
2017-03-24 06:50:36 +00:00
|
|
|
if self._current_temperature is not None:
|
|
|
|
self._target_temperature = (
|
|
|
|
round((float(self._current_temperature)), 1))
|
2017-03-14 23:55:33 +00:00
|
|
|
else:
|
|
|
|
self._target_temperature = round(
|
|
|
|
(float(self.values.primary.data)), 1)
|
2017-02-09 12:40:35 +00:00
|
|
|
|
2016-10-28 05:25:17 +00:00
|
|
|
# Operating state
|
2017-03-14 23:55:33 +00:00
|
|
|
if self.values.operating_state:
|
|
|
|
self._operating_state = self.values.operating_state.data
|
2016-08-19 07:17:28 +00:00
|
|
|
|
2016-11-15 12:14:29 +00:00
|
|
|
# Fan operating state
|
2017-03-14 23:55:33 +00:00
|
|
|
if self.values.fan_state:
|
|
|
|
self._fan_state = self.values.fan_state.data
|
2016-11-15 12:14:29 +00:00
|
|
|
|
2016-08-19 07:17:28 +00:00
|
|
|
@property
|
|
|
|
def current_fan_mode(self):
|
|
|
|
"""Return the fan speed set."""
|
|
|
|
return self._current_fan_mode
|
|
|
|
|
|
|
|
@property
|
|
|
|
def fan_list(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Return a list of available fan modes."""
|
2016-08-19 07:17:28 +00:00
|
|
|
return self._fan_list
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_swing_mode(self):
|
|
|
|
"""Return the swing mode set."""
|
|
|
|
return self._current_swing_mode
|
|
|
|
|
|
|
|
@property
|
|
|
|
def swing_list(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Return a list of available swing modes."""
|
2016-08-19 07:17:28 +00:00
|
|
|
return self._swing_list
|
|
|
|
|
|
|
|
@property
|
2016-10-11 07:00:29 +00:00
|
|
|
def temperature_unit(self):
|
2016-08-19 07:17:28 +00:00
|
|
|
"""Return the unit of measurement."""
|
2016-09-02 05:17:41 +00:00
|
|
|
if self._unit == 'C':
|
|
|
|
return TEMP_CELSIUS
|
2018-07-23 08:16:05 +00:00
|
|
|
if self._unit == 'F':
|
2016-09-02 05:17:41 +00:00
|
|
|
return TEMP_FAHRENHEIT
|
2017-07-06 06:30:01 +00:00
|
|
|
return self._unit
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def current_temperature(self):
|
|
|
|
"""Return the current temperature."""
|
|
|
|
return self._current_temperature
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_operation(self):
|
|
|
|
"""Return the current operation mode."""
|
|
|
|
return self._current_operation
|
|
|
|
|
|
|
|
@property
|
|
|
|
def operation_list(self):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Return a list of available operation modes."""
|
2016-08-19 07:17:28 +00:00
|
|
|
return self._operation_list
|
|
|
|
|
|
|
|
@property
|
|
|
|
def target_temperature(self):
|
|
|
|
"""Return the temperature we try to reach."""
|
|
|
|
return self._target_temperature
|
|
|
|
|
2016-09-09 17:06:53 +00:00
|
|
|
def set_temperature(self, **kwargs):
|
2016-08-19 07:17:28 +00:00
|
|
|
"""Set new target temperature."""
|
2016-09-09 17:06:53 +00:00
|
|
|
if kwargs.get(ATTR_TEMPERATURE) is not None:
|
|
|
|
temperature = kwargs.get(ATTR_TEMPERATURE)
|
|
|
|
else:
|
|
|
|
return
|
|
|
|
|
2017-03-14 23:55:33 +00:00
|
|
|
self.values.primary.data = temperature
|
2016-08-19 07:17:28 +00:00
|
|
|
|
2018-02-11 17:20:28 +00:00
|
|
|
def set_fan_mode(self, fan_mode):
|
2016-08-19 07:17:28 +00:00
|
|
|
"""Set new target fan mode."""
|
2017-03-14 23:55:33 +00:00
|
|
|
if self.values.fan_mode:
|
2018-02-11 17:20:28 +00:00
|
|
|
self.values.fan_mode.data = fan_mode
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
def set_operation_mode(self, operation_mode):
|
|
|
|
"""Set new target operation mode."""
|
2017-03-14 23:55:33 +00:00
|
|
|
if self.values.mode:
|
2018-06-29 14:09:46 +00:00
|
|
|
self.values.mode.data = self._operation_mapping.get(
|
|
|
|
operation_mode, operation_mode)
|
2016-08-19 07:17:28 +00:00
|
|
|
|
|
|
|
def set_swing_mode(self, swing_mode):
|
|
|
|
"""Set new target swing mode."""
|
|
|
|
if self._zxt_120 == 1:
|
2017-03-14 23:55:33 +00:00
|
|
|
if self.values.zxt_120_swing_mode:
|
2017-04-25 00:59:51 +00:00
|
|
|
self.values.zxt_120_swing_mode.data = swing_mode
|
2016-10-28 05:25:17 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return the device specific state attributes."""
|
2016-11-18 20:42:30 +00:00
|
|
|
data = super().device_state_attributes
|
2016-10-28 05:25:17 +00:00
|
|
|
if self._operating_state:
|
2017-03-15 03:24:35 +00:00
|
|
|
data[ATTR_OPERATING_STATE] = self._operating_state
|
2016-11-15 12:14:29 +00:00
|
|
|
if self._fan_state:
|
2016-11-18 20:42:30 +00:00
|
|
|
data[ATTR_FAN_STATE] = self._fan_state
|
2016-11-15 12:14:29 +00:00
|
|
|
return data
|