2023-09-10 12:08:27 +08:00
|
|
|
|
from homeassistant.components.climate import (
|
|
|
|
|
ClimateEntity,
|
|
|
|
|
ClimateEntityFeature,
|
|
|
|
|
HVACMode,
|
|
|
|
|
ATTR_HVAC_MODE,
|
|
|
|
|
)
|
2023-09-09 00:14:41 +08:00
|
|
|
|
from homeassistant.const import (
|
|
|
|
|
Platform,
|
2025-09-12 00:15:14 +08:00
|
|
|
|
ATTR_TEMPERATURE,
|
2023-09-09 00:14:41 +08:00
|
|
|
|
)
|
2025-09-09 23:52:48 +08:00
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2023-09-10 12:08:27 +08:00
|
|
|
|
|
2025-09-12 00:15:14 +08:00
|
|
|
|
from .const import DOMAIN
|
2025-09-17 22:46:38 +08:00
|
|
|
|
from .core.logger import MideaLogger
|
2025-09-09 23:52:48 +08:00
|
|
|
|
from .midea_entity import MideaEntity
|
2025-09-12 00:15:14 +08:00
|
|
|
|
from . import load_device_config
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
|
|
|
|
|
2025-09-09 23:52:48 +08:00
|
|
|
|
async def async_setup_entry(
|
|
|
|
|
hass: HomeAssistant,
|
|
|
|
|
config_entry: ConfigEntry,
|
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
|
) -> None:
|
|
|
|
|
"""Set up climate entities for Midea devices."""
|
2025-09-12 00:15:14 +08:00
|
|
|
|
# 账号型 entry:从 __init__ 写入的 accounts 桶加载设备和协调器
|
|
|
|
|
account_bucket = hass.data.get(DOMAIN, {}).get("accounts", {}).get(config_entry.entry_id)
|
|
|
|
|
if not account_bucket:
|
|
|
|
|
async_add_entities([])
|
|
|
|
|
return
|
|
|
|
|
device_list = account_bucket.get("device_list", {})
|
|
|
|
|
coordinator_map = account_bucket.get("coordinator_map", {})
|
|
|
|
|
|
2023-09-09 00:14:41 +08:00
|
|
|
|
devs = []
|
2025-09-12 00:15:14 +08:00
|
|
|
|
for device_id, info in device_list.items():
|
|
|
|
|
device_type = info.get("type")
|
|
|
|
|
sn8 = info.get("sn8")
|
2025-09-17 23:15:15 +08:00
|
|
|
|
config = await load_device_config(hass, device_type, sn8) or {}
|
2025-09-12 00:15:14 +08:00
|
|
|
|
entities_cfg = (config.get("entities") or {}).get(Platform.CLIMATE, {})
|
|
|
|
|
manufacturer = config.get("manufacturer")
|
|
|
|
|
rationale = config.get("rationale")
|
|
|
|
|
coordinator = coordinator_map.get(device_id)
|
|
|
|
|
device = coordinator.device if coordinator else None
|
2025-09-17 22:46:38 +08:00
|
|
|
|
|
2025-09-12 00:15:14 +08:00
|
|
|
|
for entity_key, ecfg in entities_cfg.items():
|
2025-09-09 23:52:48 +08:00
|
|
|
|
devs.append(MideaClimateEntity(
|
2025-09-12 00:15:14 +08:00
|
|
|
|
coordinator, device, manufacturer, rationale, entity_key, ecfg
|
2025-09-09 23:52:48 +08:00
|
|
|
|
))
|
2023-09-09 00:14:41 +08:00
|
|
|
|
async_add_entities(devs)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class MideaClimateEntity(MideaEntity, ClimateEntity):
|
2025-09-09 23:52:48 +08:00
|
|
|
|
def __init__(self, coordinator, device, manufacturer, rationale, entity_key, config):
|
|
|
|
|
super().__init__(
|
|
|
|
|
coordinator,
|
|
|
|
|
device.device_id,
|
|
|
|
|
device.device_name,
|
|
|
|
|
f"T0x{device.device_type:02X}",
|
|
|
|
|
device.sn,
|
|
|
|
|
device.sn8,
|
|
|
|
|
device.model,
|
2025-09-24 19:57:11 +08:00
|
|
|
|
entity_key,
|
|
|
|
|
device=device,
|
|
|
|
|
manufacturer=manufacturer,
|
|
|
|
|
rationale=rationale,
|
|
|
|
|
config=config,
|
2025-09-09 23:52:48 +08:00
|
|
|
|
)
|
|
|
|
|
self._device = device
|
|
|
|
|
self._manufacturer = manufacturer
|
|
|
|
|
self._rationale = rationale
|
|
|
|
|
self._config = config
|
2023-09-09 00:14:41 +08:00
|
|
|
|
self._key_power = self._config.get("power")
|
|
|
|
|
self._key_hvac_modes = self._config.get("hvac_modes")
|
|
|
|
|
self._key_preset_modes = self._config.get("preset_modes")
|
|
|
|
|
self._key_aux_heat = self._config.get("aux_heat")
|
|
|
|
|
self._key_swing_modes = self._config.get("swing_modes")
|
|
|
|
|
self._key_fan_modes = self._config.get("fan_modes")
|
|
|
|
|
self._key_min_temp = self._config.get("min_temp")
|
|
|
|
|
self._key_max_temp = self._config.get("max_temp")
|
2023-09-10 12:08:27 +08:00
|
|
|
|
self._key_current_temperature = self._config.get("current_temperature")
|
2023-09-09 00:14:41 +08:00
|
|
|
|
self._key_target_temperature = self._config.get("target_temperature")
|
|
|
|
|
self._attr_temperature_unit = self._config.get("temperature_unit")
|
|
|
|
|
self._attr_precision = self._config.get("precision")
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def supported_features(self):
|
|
|
|
|
features = 0
|
|
|
|
|
if self._key_target_temperature is not None:
|
|
|
|
|
features |= ClimateEntityFeature.TARGET_TEMPERATURE
|
|
|
|
|
if self._key_preset_modes is not None:
|
|
|
|
|
features |= ClimateEntityFeature.PRESET_MODE
|
2025-09-12 00:15:14 +08:00
|
|
|
|
# if self._key_aux_heat is not None:
|
|
|
|
|
# features |= ClimateEntityFeature.AUX_HEAT
|
2023-09-09 00:14:41 +08:00
|
|
|
|
if self._key_swing_modes is not None:
|
|
|
|
|
features |= ClimateEntityFeature.SWING_MODE
|
|
|
|
|
if self._key_fan_modes is not None:
|
|
|
|
|
features |= ClimateEntityFeature.FAN_MODE
|
|
|
|
|
return features
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def current_temperature(self):
|
2025-09-25 10:50:39 +08:00
|
|
|
|
temp = self.device_attributes.get(self._key_current_temperature)
|
|
|
|
|
if temp is not None:
|
|
|
|
|
try:
|
|
|
|
|
return float(temp)
|
|
|
|
|
except (ValueError, TypeError):
|
|
|
|
|
return None
|
|
|
|
|
return None
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def target_temperature(self):
|
|
|
|
|
if isinstance(self._key_target_temperature, list):
|
2025-09-09 23:52:48 +08:00
|
|
|
|
temp_int = self.device_attributes.get(self._key_target_temperature[0])
|
|
|
|
|
tem_dec = self.device_attributes.get(self._key_target_temperature[1])
|
2023-09-09 00:14:41 +08:00
|
|
|
|
if temp_int is not None and tem_dec is not None:
|
2025-09-25 10:50:39 +08:00
|
|
|
|
try:
|
|
|
|
|
return float(temp_int) + float(tem_dec)
|
|
|
|
|
except (ValueError, TypeError):
|
|
|
|
|
return None
|
2023-09-09 00:14:41 +08:00
|
|
|
|
return None
|
|
|
|
|
else:
|
2025-09-25 10:50:39 +08:00
|
|
|
|
temp = self.device_attributes.get(self._key_target_temperature)
|
|
|
|
|
if temp is not None:
|
|
|
|
|
try:
|
|
|
|
|
return float(temp)
|
|
|
|
|
except (ValueError, TypeError):
|
|
|
|
|
return None
|
|
|
|
|
return None
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def min_temp(self):
|
|
|
|
|
if isinstance(self._key_min_temp, str):
|
2025-09-09 23:52:48 +08:00
|
|
|
|
return float(self.device_attributes.get(self._key_min_temp, 16))
|
2023-09-09 00:14:41 +08:00
|
|
|
|
else:
|
|
|
|
|
return float(self._key_min_temp)
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def max_temp(self):
|
|
|
|
|
if isinstance(self._key_max_temp, str):
|
2025-09-09 23:52:48 +08:00
|
|
|
|
return float(self.device_attributes.get(self._key_max_temp, 30))
|
2023-09-09 00:14:41 +08:00
|
|
|
|
else:
|
|
|
|
|
return float(self._key_max_temp)
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def target_temperature_low(self):
|
|
|
|
|
return self.min_temp
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def target_temperature_high(self):
|
|
|
|
|
return self.max_temp
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def preset_modes(self):
|
|
|
|
|
return list(self._key_preset_modes.keys())
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def preset_mode(self):
|
2023-09-10 12:08:27 +08:00
|
|
|
|
return self._dict_get_selected(self._key_preset_modes)
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def fan_modes(self):
|
|
|
|
|
return list(self._key_fan_modes.keys())
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def fan_mode(self):
|
2025-09-24 19:57:11 +08:00
|
|
|
|
return self._dict_get_selected(self._key_fan_modes, "EQUALLY")
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def swing_modes(self):
|
|
|
|
|
return list(self._key_swing_modes.keys())
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def swing_mode(self):
|
2025-09-24 19:57:11 +08:00
|
|
|
|
return self._dict_get_selected(self._key_swing_modes, "EQUALLY")
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def is_on(self) -> bool:
|
|
|
|
|
return self.hvac_mode != HVACMode.OFF
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def hvac_mode(self):
|
2025-09-24 19:57:11 +08:00
|
|
|
|
return self._dict_get_selected(self._key_hvac_modes, "EQUALLY")
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def hvac_modes(self):
|
|
|
|
|
return list(self._key_hvac_modes.keys())
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def is_aux_heat(self):
|
2023-09-10 12:08:27 +08:00
|
|
|
|
return self._get_status_on_off(self._key_aux_heat)
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
2025-09-09 23:52:48 +08:00
|
|
|
|
async def async_turn_on(self):
|
|
|
|
|
await self._async_set_status_on_off(self._key_power, True)
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
2025-09-09 23:52:48 +08:00
|
|
|
|
async def async_turn_off(self):
|
|
|
|
|
await self._async_set_status_on_off(self._key_power, False)
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
2025-09-09 23:52:48 +08:00
|
|
|
|
async def async_set_temperature(self, **kwargs):
|
2023-09-09 00:14:41 +08:00
|
|
|
|
if ATTR_TEMPERATURE not in kwargs:
|
|
|
|
|
return
|
|
|
|
|
temperature = kwargs.get(ATTR_TEMPERATURE)
|
|
|
|
|
temp_int, temp_dec = divmod(temperature, 1)
|
|
|
|
|
temp_int = int(temp_int)
|
|
|
|
|
hvac_mode = kwargs.get(ATTR_HVAC_MODE)
|
|
|
|
|
if hvac_mode is not None:
|
|
|
|
|
new_status = self._key_hvac_modes.get(hvac_mode)
|
|
|
|
|
else:
|
|
|
|
|
new_status = {}
|
|
|
|
|
if isinstance(self._key_target_temperature, list):
|
|
|
|
|
new_status[self._key_target_temperature[0]] = temp_int
|
|
|
|
|
new_status[self._key_target_temperature[1]] = temp_dec
|
|
|
|
|
else:
|
|
|
|
|
new_status[self._key_target_temperature] = temperature
|
2025-09-09 23:52:48 +08:00
|
|
|
|
await self.async_set_attributes(new_status)
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
2025-09-09 23:52:48 +08:00
|
|
|
|
async def async_set_fan_mode(self, fan_mode: str):
|
2023-09-10 12:08:27 +08:00
|
|
|
|
new_status = self._key_fan_modes.get(fan_mode)
|
2025-09-09 23:52:48 +08:00
|
|
|
|
await self.async_set_attributes(new_status)
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
2025-09-09 23:52:48 +08:00
|
|
|
|
async def async_set_preset_mode(self, preset_mode: str):
|
2023-09-10 12:08:27 +08:00
|
|
|
|
new_status = self._key_preset_modes.get(preset_mode)
|
2025-09-09 23:52:48 +08:00
|
|
|
|
await self.async_set_attributes(new_status)
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
2025-09-09 23:52:48 +08:00
|
|
|
|
async def async_set_hvac_mode(self, hvac_mode: str):
|
2023-09-09 00:14:41 +08:00
|
|
|
|
new_status = self._key_hvac_modes.get(hvac_mode)
|
2025-09-09 23:52:48 +08:00
|
|
|
|
await self.async_set_attributes(new_status)
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
2025-09-09 23:52:48 +08:00
|
|
|
|
async def async_set_swing_mode(self, swing_mode: str):
|
2023-09-09 00:14:41 +08:00
|
|
|
|
new_status = self._key_swing_modes.get(swing_mode)
|
2025-09-09 23:52:48 +08:00
|
|
|
|
await self.async_set_attributes(new_status)
|
|
|
|
|
|
|
|
|
|
async def async_turn_aux_heat_on(self) -> None:
|
|
|
|
|
await self._async_set_status_on_off(self._key_aux_heat, True)
|
|
|
|
|
|
|
|
|
|
async def async_turn_aux_heat_off(self) -> None:
|
|
|
|
|
await self._async_set_status_on_off(self._key_aux_heat, False)
|
|
|
|
|
|
|
|
|
|
def _get_status_on_off(self, key):
|
|
|
|
|
"""Get on/off status from device attributes."""
|
|
|
|
|
if key is None:
|
|
|
|
|
return False
|
|
|
|
|
value = self.device_attributes.get(key)
|
|
|
|
|
if isinstance(value, bool):
|
|
|
|
|
return value
|
|
|
|
|
return value == 1 or value == "on" or value == "true"
|
|
|
|
|
|
|
|
|
|
async def _async_set_status_on_off(self, key, value):
|
|
|
|
|
"""Set on/off status for device attribute."""
|
|
|
|
|
if key is None:
|
|
|
|
|
return
|
|
|
|
|
await self.async_set_attribute(key, value)
|
2023-09-09 00:14:41 +08:00
|
|
|
|
|
2025-09-24 19:57:11 +08:00
|
|
|
|
def _dict_get_selected(self, dict_config, rationale="EQUALLY"):
|
2025-09-09 23:52:48 +08:00
|
|
|
|
"""Get selected value from dictionary configuration."""
|
|
|
|
|
if dict_config is None:
|
|
|
|
|
return None
|
2025-09-17 22:46:38 +08:00
|
|
|
|
|
|
|
|
|
MideaLogger.debug(f"dict_config={dict_config}, rationale={rationale}, self.device_attributes={self.device_attributes} ")
|
2025-09-09 23:52:48 +08:00
|
|
|
|
for key, config in dict_config.items():
|
|
|
|
|
if isinstance(config, dict):
|
|
|
|
|
# Check if all conditions match
|
|
|
|
|
match = True
|
|
|
|
|
for attr_key, attr_value in config.items():
|
|
|
|
|
device_value = self.device_attributes.get(attr_key)
|
2025-09-17 22:46:38 +08:00
|
|
|
|
if device_value is None:
|
|
|
|
|
match = False
|
|
|
|
|
break
|
2025-09-24 19:57:11 +08:00
|
|
|
|
if rationale == "EQUALLY":
|
2025-09-09 23:52:48 +08:00
|
|
|
|
if device_value != attr_value:
|
|
|
|
|
match = False
|
|
|
|
|
break
|
2025-09-24 19:57:11 +08:00
|
|
|
|
elif rationale == "LESS":
|
2025-09-09 23:52:48 +08:00
|
|
|
|
if device_value >= attr_value:
|
|
|
|
|
match = False
|
|
|
|
|
break
|
2025-09-24 19:57:11 +08:00
|
|
|
|
elif rationale == "GREATER":
|
2025-09-09 23:52:48 +08:00
|
|
|
|
if device_value <= attr_value:
|
|
|
|
|
match = False
|
|
|
|
|
break
|
|
|
|
|
if match:
|
|
|
|
|
return key
|
|
|
|
|
return None
|