2021-09-14 19:46:52 +00:00
|
|
|
"""Support for Crownstone devices."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
from functools import partial
|
|
|
|
from typing import TYPE_CHECKING, Any
|
|
|
|
|
|
|
|
from crownstone_cloud.cloud_models.crownstones import Crownstone
|
2021-09-23 07:23:45 +00:00
|
|
|
from crownstone_cloud.const import DIMMING_ABILITY
|
2021-09-14 19:46:52 +00:00
|
|
|
from crownstone_cloud.exceptions import CrownstoneAbilityError
|
|
|
|
from crownstone_uart import CrownstoneUart
|
|
|
|
|
2022-04-22 14:25:20 +00:00
|
|
|
from homeassistant.components.light import ATTR_BRIGHTNESS, ColorMode, LightEntity
|
2021-09-14 19:46:52 +00:00
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant
|
2021-09-23 07:23:45 +00:00
|
|
|
from homeassistant.exceptions import HomeAssistantError
|
2021-09-14 19:46:52 +00:00
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
|
|
|
|
from .const import (
|
|
|
|
CROWNSTONE_INCLUDE_TYPES,
|
|
|
|
CROWNSTONE_SUFFIX,
|
|
|
|
DOMAIN,
|
|
|
|
SIG_CROWNSTONE_STATE_UPDATE,
|
|
|
|
SIG_UART_STATE_CHANGE,
|
|
|
|
)
|
2021-09-23 07:23:45 +00:00
|
|
|
from .devices import CrownstoneBaseEntity
|
2021-09-14 19:46:52 +00:00
|
|
|
from .helpers import map_from_to
|
|
|
|
|
|
|
|
if TYPE_CHECKING:
|
|
|
|
from .entry_manager import CrownstoneEntryManager
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config_entry: ConfigEntry,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
) -> None:
|
|
|
|
"""Set up crownstones from a config entry."""
|
|
|
|
manager: CrownstoneEntryManager = hass.data[DOMAIN][config_entry.entry_id]
|
|
|
|
|
|
|
|
entities: list[CrownstoneEntity] = []
|
|
|
|
|
|
|
|
# Add Crownstone entities that support switching/dimming
|
|
|
|
for sphere in manager.cloud.cloud_data:
|
|
|
|
for crownstone in sphere.crownstones:
|
|
|
|
if crownstone.type in CROWNSTONE_INCLUDE_TYPES:
|
|
|
|
# Crownstone can communicate with Crownstone USB
|
|
|
|
if manager.uart and sphere.cloud_id == manager.usb_sphere_id:
|
|
|
|
entities.append(CrownstoneEntity(crownstone, manager.uart))
|
|
|
|
# Crownstone can't communicate with Crownstone USB
|
|
|
|
else:
|
|
|
|
entities.append(CrownstoneEntity(crownstone))
|
|
|
|
|
|
|
|
async_add_entities(entities)
|
|
|
|
|
|
|
|
|
|
|
|
def crownstone_state_to_hass(value: int) -> int:
|
|
|
|
"""Crownstone 0..100 to hass 0..255."""
|
|
|
|
return map_from_to(value, 0, 100, 0, 255)
|
|
|
|
|
|
|
|
|
|
|
|
def hass_to_crownstone_state(value: int) -> int:
|
|
|
|
"""Hass 0..255 to Crownstone 0..100."""
|
|
|
|
return map_from_to(value, 0, 255, 0, 100)
|
|
|
|
|
|
|
|
|
2021-09-23 07:23:45 +00:00
|
|
|
class CrownstoneEntity(CrownstoneBaseEntity, LightEntity):
|
2023-02-03 22:08:48 +00:00
|
|
|
"""Representation of a crownstone.
|
2021-09-14 19:46:52 +00:00
|
|
|
|
|
|
|
Light platform is used to support dimming.
|
|
|
|
"""
|
|
|
|
|
|
|
|
_attr_icon = "mdi:power-socket-de"
|
|
|
|
|
2021-09-23 07:23:45 +00:00
|
|
|
def __init__(
|
|
|
|
self, crownstone_data: Crownstone, usb: CrownstoneUart | None = None
|
|
|
|
) -> None:
|
2021-09-14 19:46:52 +00:00
|
|
|
"""Initialize the crownstone."""
|
|
|
|
super().__init__(crownstone_data)
|
|
|
|
self.usb = usb
|
|
|
|
# Entity class attributes
|
|
|
|
self._attr_name = str(self.device.name)
|
|
|
|
self._attr_unique_id = f"{self.cloud_id}-{CROWNSTONE_SUFFIX}"
|
|
|
|
|
|
|
|
@property
|
|
|
|
def brightness(self) -> int | None:
|
|
|
|
"""Return the brightness if dimming enabled."""
|
|
|
|
return crownstone_state_to_hass(self.device.state)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self) -> bool:
|
|
|
|
"""Return if the device is on."""
|
|
|
|
return crownstone_state_to_hass(self.device.state) > 0
|
|
|
|
|
|
|
|
@property
|
2022-04-01 16:18:13 +00:00
|
|
|
def color_mode(self) -> str:
|
|
|
|
"""Return the color mode of the light."""
|
2021-09-14 19:46:52 +00:00
|
|
|
if self.device.abilities.get(DIMMING_ABILITY).is_enabled:
|
2022-04-22 14:25:20 +00:00
|
|
|
return ColorMode.BRIGHTNESS
|
|
|
|
return ColorMode.ONOFF
|
2022-04-01 16:18:13 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_color_modes(self) -> set[str] | None:
|
|
|
|
"""Flag supported color modes."""
|
|
|
|
return {self.color_mode}
|
2021-09-14 19:46:52 +00:00
|
|
|
|
|
|
|
async def async_added_to_hass(self) -> None:
|
|
|
|
"""Set up a listener when this entity is added to HA."""
|
|
|
|
# new state received
|
|
|
|
self.async_on_remove(
|
|
|
|
async_dispatcher_connect(
|
|
|
|
self.hass, SIG_CROWNSTONE_STATE_UPDATE, self.async_write_ha_state
|
|
|
|
)
|
|
|
|
)
|
|
|
|
# updates state attributes when usb connects/disconnects
|
|
|
|
self.async_on_remove(
|
|
|
|
async_dispatcher_connect(
|
|
|
|
self.hass, SIG_UART_STATE_CHANGE, self.async_write_ha_state
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
async def async_turn_on(self, **kwargs: Any) -> None:
|
|
|
|
"""Turn on this light via dongle or cloud."""
|
|
|
|
if ATTR_BRIGHTNESS in kwargs:
|
2021-09-23 07:23:45 +00:00
|
|
|
if self.usb is not None and self.usb.is_ready():
|
2021-09-14 19:46:52 +00:00
|
|
|
await self.hass.async_add_executor_job(
|
|
|
|
partial(
|
|
|
|
self.usb.dim_crownstone,
|
|
|
|
self.device.unique_id,
|
|
|
|
hass_to_crownstone_state(kwargs[ATTR_BRIGHTNESS]),
|
|
|
|
)
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
await self.device.async_set_brightness(
|
|
|
|
hass_to_crownstone_state(kwargs[ATTR_BRIGHTNESS])
|
|
|
|
)
|
|
|
|
except CrownstoneAbilityError as ability_error:
|
2021-09-23 07:23:45 +00:00
|
|
|
raise HomeAssistantError(ability_error) from ability_error
|
2021-09-14 19:46:52 +00:00
|
|
|
|
|
|
|
# assume brightness is set on device
|
|
|
|
self.device.state = hass_to_crownstone_state(kwargs[ATTR_BRIGHTNESS])
|
|
|
|
self.async_write_ha_state()
|
|
|
|
|
2021-09-23 07:23:45 +00:00
|
|
|
elif self.usb is not None and self.usb.is_ready():
|
2021-09-14 19:46:52 +00:00
|
|
|
await self.hass.async_add_executor_job(
|
|
|
|
partial(self.usb.switch_crownstone, self.device.unique_id, on=True)
|
|
|
|
)
|
|
|
|
self.device.state = 100
|
|
|
|
self.async_write_ha_state()
|
|
|
|
|
|
|
|
else:
|
|
|
|
await self.device.async_turn_on()
|
|
|
|
self.device.state = 100
|
|
|
|
self.async_write_ha_state()
|
|
|
|
|
|
|
|
async def async_turn_off(self, **kwargs: Any) -> None:
|
|
|
|
"""Turn off this device via dongle or cloud."""
|
2021-09-23 07:23:45 +00:00
|
|
|
if self.usb is not None and self.usb.is_ready():
|
2021-09-14 19:46:52 +00:00
|
|
|
await self.hass.async_add_executor_job(
|
|
|
|
partial(self.usb.switch_crownstone, self.device.unique_id, on=False)
|
|
|
|
)
|
|
|
|
|
|
|
|
else:
|
|
|
|
await self.device.async_turn_off()
|
|
|
|
|
|
|
|
self.device.state = 0
|
|
|
|
self.async_write_ha_state()
|