core/homeassistant/components/nexia/switch.py

167 lines
5.9 KiB
Python

"""Support for Nexia switches."""
from __future__ import annotations
from collections.abc import Iterable
import functools as ft
from typing import Any
from nexia.const import OPERATION_MODE_OFF
from nexia.roomiq import NexiaRoomIQHarmonizer
from nexia.sensor import NexiaSensor
from nexia.thermostat import NexiaThermostat
from nexia.zone import NexiaThermostatZone
from homeassistant.components.switch import SwitchEntity
from homeassistant.const import EVENT_HOMEASSISTANT_STOP
from homeassistant.core import Event, HomeAssistant
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
from .coordinator import NexiaDataUpdateCoordinator
from .entity import NexiaThermostatEntity, NexiaThermostatZoneEntity
from .types import NexiaConfigEntry
async def _stop_harmonizers(
_: Event, harmonizers: Iterable[NexiaRoomIQHarmonizer]
) -> None:
"""Run the shutdown methods when preparing to stop."""
for harmonizer in harmonizers:
await harmonizer.async_shutdown() # Never suspends
async def async_setup_entry(
hass: HomeAssistant,
config_entry: NexiaConfigEntry,
async_add_entities: AddConfigEntryEntitiesCallback,
) -> None:
"""Set up switches for a Nexia device."""
coordinator = config_entry.runtime_data
nexia_home = coordinator.nexia_home
entities: list[SwitchEntity] = []
room_iq_zones: dict[int, NexiaRoomIQHarmonizer] = {}
for thermostat_id in nexia_home.get_thermostat_ids():
thermostat: NexiaThermostat = nexia_home.get_thermostat_by_id(thermostat_id)
if thermostat.has_emergency_heat():
entities.append(NexiaEmergencyHeatSwitch(coordinator, thermostat))
for zone_id in thermostat.get_zone_ids():
zone: NexiaThermostatZone = thermostat.get_zone_by_id(zone_id)
entities.append(NexiaHoldSwitch(coordinator, zone))
if len(zone_sensors := zone.get_sensors()) > 1:
entities.extend(
NexiaRoomIQSwitch(coordinator, zone, sensor, room_iq_zones)
for sensor in zone_sensors
)
async_add_entities(entities)
if room_iq_zones:
listener = ft.partial(_stop_harmonizers, harmonizers=room_iq_zones.values())
config_entry.async_on_unload(
hass.bus.async_listen(EVENT_HOMEASSISTANT_STOP, listener)
)
class NexiaHoldSwitch(NexiaThermostatZoneEntity, SwitchEntity):
"""Provides Nexia hold switch support."""
_attr_translation_key = "hold"
def __init__(
self, coordinator: NexiaDataUpdateCoordinator, zone: NexiaThermostatZone
) -> None:
"""Initialize the hold mode switch."""
zone_id = zone.zone_id
super().__init__(coordinator, zone, zone_id)
@property
def is_on(self) -> bool:
"""Return if the zone is in hold mode."""
return self._zone.is_in_permanent_hold()
async def async_turn_on(self, **kwargs: Any) -> None:
"""Enable permanent hold."""
if self._zone.get_current_mode() == OPERATION_MODE_OFF:
await self._zone.call_permanent_off()
else:
await self._zone.set_permanent_hold()
self._signal_zone_update()
async def async_turn_off(self, **kwargs: Any) -> None:
"""Disable permanent hold."""
await self._zone.call_return_to_schedule()
self._signal_zone_update()
class NexiaRoomIQSwitch(NexiaThermostatZoneEntity, SwitchEntity):
"""Provides Nexia RoomIQ sensor switch support."""
_attr_translation_key = "room_iq_sensor"
def __init__(
self,
coordinator: NexiaDataUpdateCoordinator,
zone: NexiaThermostatZone,
sensor: NexiaSensor,
room_iq_zones: dict[int, NexiaRoomIQHarmonizer],
) -> None:
"""Initialize the RoomIQ sensor switch."""
super().__init__(coordinator, zone, f"{sensor.id}_room_iq_sensor")
self._attr_translation_placeholders = {"sensor_name": sensor.name}
self._sensor_id = sensor.id
if zone.zone_id in room_iq_zones:
self._harmonizer = room_iq_zones[zone.zone_id]
else:
self._harmonizer = NexiaRoomIQHarmonizer(
zone, coordinator.async_refresh, self._signal_zone_update
)
room_iq_zones[zone.zone_id] = self._harmonizer
@property
def is_on(self) -> bool:
"""Return if the sensor is part of the zone average temperature."""
if self._harmonizer.request_pending():
return self._sensor_id in self._harmonizer.selected_sensor_ids
return self._zone.get_sensor_by_id(self._sensor_id).weight > 0.0
async def async_turn_on(self, **kwargs: Any) -> None:
"""Include this sensor."""
self._harmonizer.trigger_add_sensor(self._sensor_id)
self._signal_zone_update()
async def async_turn_off(self, **kwargs: Any) -> None:
"""Remove this sensor."""
self._harmonizer.trigger_remove_sensor(self._sensor_id)
self._signal_zone_update()
class NexiaEmergencyHeatSwitch(NexiaThermostatEntity, SwitchEntity):
"""Provides Nexia emergency heat switch support."""
_attr_translation_key = "emergency_heat"
def __init__(
self, coordinator: NexiaDataUpdateCoordinator, thermostat: NexiaThermostat
) -> None:
"""Initialize the emergency heat mode switch."""
super().__init__(
coordinator,
thermostat,
unique_id=f"{thermostat.thermostat_id}_emergency_heat",
)
@property
def is_on(self) -> bool:
"""Return if the zone is in hold mode."""
return self._thermostat.is_emergency_heat_active()
async def async_turn_on(self, **kwargs: Any) -> None:
"""Enable permanent hold."""
await self._thermostat.set_emergency_heat(True)
self._signal_thermostat_update()
async def async_turn_off(self, **kwargs: Any) -> None:
"""Disable permanent hold."""
await self._thermostat.set_emergency_heat(False)
self._signal_thermostat_update()