Home Assistant Unofficial Reference 2024.12.1
climate.py
Go to the documentation of this file.
1 """YoLink Thermostat."""
2 
3 from __future__ import annotations
4 
5 from typing import Any
6 
7 from yolink.const import ATTR_DEVICE_THERMOSTAT
8 from yolink.thermostat_request_builder import ThermostatRequestBuilder, ThermostatState
9 
11  ATTR_TARGET_TEMP_HIGH,
12  ATTR_TARGET_TEMP_LOW,
13  FAN_AUTO,
14  FAN_ON,
15  PRESET_ECO,
16  PRESET_NONE,
17  ClimateEntity,
18  ClimateEntityFeature,
19  HVACAction,
20  HVACMode,
21 )
22 from homeassistant.config_entries import ConfigEntry
23 from homeassistant.const import UnitOfTemperature
24 from homeassistant.core import HomeAssistant, callback
25 from homeassistant.helpers.entity_platform import AddEntitiesCallback
26 
27 from .const import DOMAIN
28 from .coordinator import YoLinkCoordinator
29 from .entity import YoLinkEntity
30 
31 YOLINK_MODEL_2_HA = {
32  "cool": HVACMode.COOL,
33  "heat": HVACMode.HEAT,
34  "auto": HVACMode.AUTO,
35  "off": HVACMode.OFF,
36 }
37 
38 HA_MODEL_2_YOLINK = {v: k for k, v in YOLINK_MODEL_2_HA.items()}
39 
40 YOLINK_ACTION_2_HA = {
41  "cool": HVACAction.COOLING,
42  "heat": HVACAction.HEATING,
43  "idle": HVACAction.IDLE,
44 }
45 
46 
48  hass: HomeAssistant,
49  config_entry: ConfigEntry,
50  async_add_entities: AddEntitiesCallback,
51 ) -> None:
52  """Set up YoLink Thermostat from a config entry."""
53  device_coordinators = hass.data[DOMAIN][config_entry.entry_id].device_coordinators
54  entities = [
55  YoLinkClimateEntity(config_entry, device_coordinator)
56  for device_coordinator in device_coordinators.values()
57  if device_coordinator.device.device_type == ATTR_DEVICE_THERMOSTAT
58  ]
59  async_add_entities(entities)
60 
61 
63  """YoLink Climate Entity."""
64 
65  _attr_name = None
66  _enable_turn_on_off_backwards_compatibility = False
67 
68  def __init__(
69  self,
70  config_entry: ConfigEntry,
71  coordinator: YoLinkCoordinator,
72  ) -> None:
73  """Init YoLink Thermostat."""
74  super().__init__(config_entry, coordinator)
75  self._attr_unique_id_attr_unique_id = f"{coordinator.device.device_id}_climate"
76  self._attr_temperature_unit_attr_temperature_unit = UnitOfTemperature.CELSIUS
77  self._attr_fan_modes_attr_fan_modes = [FAN_ON, FAN_AUTO]
78  self._attr_min_temp_attr_min_temp = -10
79  self._attr_max_temp_attr_max_temp = 50
80  self._attr_hvac_mode_attr_hvac_mode = None
81  self._attr_hvac_modes_attr_hvac_modes = [
82  HVACMode.COOL,
83  HVACMode.HEAT,
84  HVACMode.AUTO,
85  HVACMode.OFF,
86  ]
87  self._attr_preset_modes_attr_preset_modes = [PRESET_NONE, PRESET_ECO]
88  self._attr_supported_features_attr_supported_features = (
89  ClimateEntityFeature.FAN_MODE
90  | ClimateEntityFeature.PRESET_MODE
91  | ClimateEntityFeature.TARGET_TEMPERATURE_RANGE
92  | ClimateEntityFeature.TURN_OFF
93  | ClimateEntityFeature.TURN_ON
94  )
95 
96  @callback
97  def update_entity_state(self, state: dict[str, Any]) -> None:
98  """Update HA Entity State."""
99  normal_state = state.get("state")
100  if normal_state is not None:
101  self._attr_current_temperature_attr_current_temperature = normal_state.get("temperature")
102  self._attr_current_humidity_attr_current_humidity = normal_state.get("humidity")
103  self._attr_target_temperature_low_attr_target_temperature_low = normal_state.get("lowTemp")
104  self._attr_target_temperature_high_attr_target_temperature_high = normal_state.get("highTemp")
105  self._attr_fan_mode_attr_fan_mode = normal_state.get("fan")
106  self._attr_hvac_mode_attr_hvac_mode = YOLINK_MODEL_2_HA.get(normal_state.get("mode"))
107  self._attr_hvac_action_attr_hvac_action = YOLINK_ACTION_2_HA.get(normal_state.get("running"))
108  eco_setting = state.get("eco")
109  if eco_setting is not None:
110  self._attr_preset_mode_attr_preset_mode = (
111  PRESET_NONE if eco_setting.get("mode") == "on" else PRESET_ECO
112  )
113  self.async_write_ha_stateasync_write_ha_state()
114 
115  async def async_set_hvac_mode(self, hvac_mode: HVACMode) -> None:
116  """Set new target hvac mode."""
117  if (hvac_mode_id := HA_MODEL_2_YOLINK.get(hvac_mode)) is None:
118  raise ValueError(f"Received an invalid hvac mode: {hvac_mode}")
119  await self.call_devicecall_device(
120  ThermostatRequestBuilder.set_state_request(
121  ThermostatState(mode=hvac_mode_id)
122  )
123  )
124  await self.coordinator.async_refresh()
125 
126  async def async_set_fan_mode(self, fan_mode: str) -> None:
127  """Set fan mode."""
128  await self.call_devicecall_device(
129  ThermostatRequestBuilder.set_state_request(ThermostatState(fan=fan_mode))
130  )
131  self._attr_fan_mode_attr_fan_mode = fan_mode
132  self.async_write_ha_stateasync_write_ha_state()
133 
134  async def async_set_temperature(self, **kwargs: Any) -> None:
135  """Set temperature."""
136  target_temp_low = kwargs.get(ATTR_TARGET_TEMP_LOW)
137  target_temp_high = kwargs.get(ATTR_TARGET_TEMP_HIGH)
138  if target_temp_low is not None:
139  await self.call_devicecall_device(
140  ThermostatRequestBuilder.set_state_request(
141  ThermostatState(lowTemp=target_temp_low)
142  )
143  )
144  self._attr_target_temperature_low_attr_target_temperature_low = target_temp_low
145  if target_temp_high is not None:
146  await self.call_devicecall_device(
147  ThermostatRequestBuilder.set_state_request(
148  ThermostatState(highTemp=target_temp_high)
149  )
150  )
151  self._attr_target_temperature_high_attr_target_temperature_high = target_temp_high
152  await self.coordinator.async_refresh()
153 
154  async def async_set_preset_mode(self, preset_mode: str) -> None:
155  """Set preset mode."""
156  eco_params = "on" if preset_mode == PRESET_ECO else "off"
157  await self.call_devicecall_device(ThermostatRequestBuilder.set_eco_request(eco_params))
158  self._attr_preset_mode_attr_preset_mode = PRESET_ECO if eco_params == "on" else PRESET_NONE
159  self.async_write_ha_stateasync_write_ha_state()