2021-11-30 16:04:24 +00:00
|
|
|
"""Support for Balboa Spa binary sensors."""
|
2022-01-05 14:04:17 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2023-02-05 18:10:35 +00:00
|
|
|
from collections.abc import Callable
|
|
|
|
from dataclasses import dataclass
|
|
|
|
|
|
|
|
from pybalboa import SpaClient
|
|
|
|
|
2021-11-30 16:04:24 +00:00
|
|
|
from homeassistant.components.binary_sensor import (
|
2021-12-08 21:11:49 +00:00
|
|
|
BinarySensorDeviceClass,
|
2021-11-30 16:04:24 +00:00
|
|
|
BinarySensorEntity,
|
2023-02-05 18:10:35 +00:00
|
|
|
BinarySensorEntityDescription,
|
2021-11-30 16:04:24 +00:00
|
|
|
)
|
2022-01-05 14:04:17 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2021-11-30 16:04:24 +00:00
|
|
|
|
2023-02-05 18:10:35 +00:00
|
|
|
from .const import DOMAIN
|
2021-11-30 16:04:24 +00:00
|
|
|
from .entity import BalboaEntity
|
|
|
|
|
|
|
|
|
2022-01-05 14:04:17 +00:00
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
|
|
|
) -> None:
|
2021-11-30 16:04:24 +00:00
|
|
|
"""Set up the spa's binary sensors."""
|
2023-02-05 18:10:35 +00:00
|
|
|
spa: SpaClient = hass.data[DOMAIN][entry.entry_id]
|
|
|
|
entities = [
|
|
|
|
BalboaBinarySensorEntity(spa, description)
|
|
|
|
for description in BINARY_SENSOR_DESCRIPTIONS
|
2022-01-05 14:04:17 +00:00
|
|
|
]
|
2023-02-05 18:10:35 +00:00
|
|
|
if spa.circulation_pump is not None:
|
|
|
|
entities.append(BalboaBinarySensorEntity(spa, CIRCULATION_PUMP_DESCRIPTION))
|
2021-11-30 16:04:24 +00:00
|
|
|
async_add_entities(entities)
|
|
|
|
|
|
|
|
|
2023-02-05 18:10:35 +00:00
|
|
|
@dataclass
|
|
|
|
class BalboaBinarySensorEntityDescriptionMixin:
|
|
|
|
"""Mixin for required keys."""
|
|
|
|
|
|
|
|
is_on_fn: Callable[[SpaClient], bool]
|
|
|
|
on_off_icons: tuple[str, str]
|
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
class BalboaBinarySensorEntityDescription(
|
|
|
|
BinarySensorEntityDescription, BalboaBinarySensorEntityDescriptionMixin
|
|
|
|
):
|
|
|
|
"""A class that describes Balboa binary sensor entities."""
|
|
|
|
|
|
|
|
|
|
|
|
FILTER_CYCLE_ICONS = ("mdi:sync", "mdi:sync-off")
|
|
|
|
BINARY_SENSOR_DESCRIPTIONS = (
|
|
|
|
BalboaBinarySensorEntityDescription(
|
|
|
|
key="filter_cycle_1",
|
|
|
|
name="Filter1",
|
|
|
|
device_class=BinarySensorDeviceClass.RUNNING,
|
|
|
|
is_on_fn=lambda spa: spa.filter_cycle_1_running,
|
|
|
|
on_off_icons=FILTER_CYCLE_ICONS,
|
|
|
|
),
|
|
|
|
BalboaBinarySensorEntityDescription(
|
|
|
|
key="filter_cycle_2",
|
|
|
|
name="Filter2",
|
|
|
|
device_class=BinarySensorDeviceClass.RUNNING,
|
|
|
|
is_on_fn=lambda spa: spa.filter_cycle_2_running,
|
|
|
|
on_off_icons=FILTER_CYCLE_ICONS,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
CIRCULATION_PUMP_DESCRIPTION = BalboaBinarySensorEntityDescription(
|
|
|
|
key="circulation_pump",
|
|
|
|
name="Circ Pump",
|
|
|
|
device_class=BinarySensorDeviceClass.RUNNING,
|
|
|
|
is_on_fn=lambda spa: (pump := spa.circulation_pump) is not None and pump.state > 0,
|
|
|
|
on_off_icons=("mdi:pump", "mdi:pump-off"),
|
|
|
|
)
|
2021-11-30 16:04:24 +00:00
|
|
|
|
|
|
|
|
2023-02-05 18:10:35 +00:00
|
|
|
class BalboaBinarySensorEntity(BalboaEntity, BinarySensorEntity):
|
|
|
|
"""Representation of a Balboa Spa binary sensor entity."""
|
2021-11-30 16:04:24 +00:00
|
|
|
|
2023-02-05 18:10:35 +00:00
|
|
|
entity_description: BalboaBinarySensorEntityDescription
|
2021-11-30 16:04:24 +00:00
|
|
|
|
2023-02-05 18:10:35 +00:00
|
|
|
def __init__(
|
|
|
|
self, spa: SpaClient, description: BalboaBinarySensorEntityDescription
|
|
|
|
) -> None:
|
|
|
|
"""Initialize a Balboa binary sensor entity."""
|
|
|
|
super().__init__(spa, description.name)
|
|
|
|
self.entity_description = description
|
2021-11-30 16:04:24 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self) -> bool:
|
2023-02-05 18:10:35 +00:00
|
|
|
"""Return true if the binary sensor is on."""
|
|
|
|
return self.entity_description.is_on_fn(self._client)
|
2021-11-30 16:04:24 +00:00
|
|
|
|
|
|
|
@property
|
2023-02-05 18:10:35 +00:00
|
|
|
def icon(self) -> str | None:
|
|
|
|
"""Return the icon to use in the frontend, if any."""
|
|
|
|
icons = self.entity_description.on_off_icons
|
|
|
|
return icons[0] if self.is_on else icons[1]
|