2025-01-16 22:09:00 +01:00
|
|
|
"""Support for Smart Home RGB lights."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
import logging
|
|
|
|
import async_timeout
|
|
|
|
import aiohttp
|
|
|
|
from homeassistant.components.light import (
|
|
|
|
ATTR_BRIGHTNESS,
|
|
|
|
ATTR_EFFECT,
|
|
|
|
ATTR_RGB_COLOR,
|
|
|
|
ColorMode,
|
|
|
|
LightEntity,
|
|
|
|
LightEntityFeature,
|
|
|
|
)
|
|
|
|
from homeassistant.config_entries import ConfigEntry
|
|
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from homeassistant.exceptions import HomeAssistantError
|
|
|
|
|
|
|
|
from .const import DOMAIN, DATA_COORDINATOR
|
|
|
|
from .coordinator import SmartHomeDataUpdateCoordinator
|
|
|
|
from .entity import SmartHomeEntity
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
async def async_fetch_effects(api_url: str) -> dict:
|
|
|
|
"""Fetch available effects from API."""
|
|
|
|
try:
|
|
|
|
async with aiohttp.ClientSession() as session:
|
|
|
|
async with async_timeout.timeout(10):
|
|
|
|
async with session.get(f"{api_url}/api/effects") as response:
|
|
|
|
if response.status == 200:
|
|
|
|
return await response.json()
|
|
|
|
else:
|
|
|
|
raise HomeAssistantError(f"Failed to fetch effects: {response.status}")
|
|
|
|
except Exception as e:
|
|
|
|
raise HomeAssistantError(f"Error fetching effects: {e}")
|
|
|
|
return {}
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
entry: ConfigEntry,
|
|
|
|
async_add_entities: AddEntitiesCallback,
|
|
|
|
) -> None:
|
|
|
|
"""Set up Smart Home RGB lights."""
|
|
|
|
coordinator: SmartHomeDataUpdateCoordinator = hass.data[DOMAIN][entry.entry_id][DATA_COORDINATOR]
|
|
|
|
|
|
|
|
try:
|
|
|
|
# Fetch available effects from API
|
|
|
|
effect_map = await async_fetch_effects(coordinator.api_url)
|
|
|
|
_LOGGER.debug("Fetched effects: %s", effect_map)
|
|
|
|
|
|
|
|
entities = []
|
|
|
|
for device_id, device in coordinator._devices.items():
|
|
|
|
if device["device_type"] == "rgb_led":
|
|
|
|
entities.append(SmartHomeLight(coordinator, device_id, effect_map))
|
|
|
|
|
|
|
|
async_add_entities(entities)
|
|
|
|
except HomeAssistantError as e:
|
|
|
|
_LOGGER.error("Failed to set up RGB lights: %s", e)
|
|
|
|
|
|
|
|
|
|
|
|
class SmartHomeLight(SmartHomeEntity, LightEntity):
|
|
|
|
"""Representation of a Smart Home RGB light."""
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
coordinator: SmartHomeDataUpdateCoordinator,
|
|
|
|
device_id: str,
|
|
|
|
effect_map: dict[str, str],
|
|
|
|
) -> None:
|
|
|
|
"""Initialize the light."""
|
|
|
|
super().__init__(coordinator, device_id)
|
|
|
|
self._effect_map = effect_map
|
|
|
|
self._attr_supported_features |= LightEntityFeature.EFFECT
|
|
|
|
|
|
|
|
# Set up supported features
|
|
|
|
self._attr_supported_color_modes = {ColorMode.RGB}
|
|
|
|
self._attr_color_mode = ColorMode.RGB
|
|
|
|
|
|
|
|
# No color temperature support
|
|
|
|
self._attr_min_mireds = 0
|
|
|
|
self._attr_max_mireds = 0
|
|
|
|
|
|
|
|
_LOGGER.debug("Initializing RGB light: %s id %s", device_id, self.unique_id)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self) -> bool | None:
|
|
|
|
"""Return true if light is on."""
|
|
|
|
if not self.device_state:
|
|
|
|
return None
|
|
|
|
return self.device_state.get("state", False)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def brightness(self) -> int | None:
|
|
|
|
"""Return the brightness of this light between 0..255."""
|
|
|
|
if not self.device_state:
|
|
|
|
return None
|
|
|
|
brightness = self.device_state.get("brightness", 0)
|
|
|
|
return int(brightness)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def rgb_color(self) -> tuple[int, int, int] | None:
|
|
|
|
"""Return the rgb color value [int, int, int]."""
|
|
|
|
if not self.device_state or "colors" not in self.device_state:
|
|
|
|
return None
|
|
|
|
# Use first color from the array
|
|
|
|
if not self.device_state["colors"]:
|
|
|
|
return None
|
|
|
|
color_hex = self.device_state["colors"][0]
|
|
|
|
# Convert hex string to RGB tuple, stripping any leading '0x'
|
|
|
|
color_hex = color_hex.replace('0x', '')
|
|
|
|
return (
|
|
|
|
int(color_hex[0:2], 16),
|
|
|
|
int(color_hex[2:4], 16),
|
|
|
|
int(color_hex[4:6], 16),
|
|
|
|
)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def effect_list(self) -> list[str] | None:
|
|
|
|
"""Return the list of supported effects."""
|
|
|
|
return list(self._effect_map.values())
|
|
|
|
|
|
|
|
@property
|
|
|
|
def effect(self) -> str | None:
|
|
|
|
"""Return the current effect."""
|
|
|
|
if not self.device_state:
|
|
|
|
return None
|
|
|
|
animation = self.device_state.get("animation")
|
|
|
|
# Handle mapping from API animation name to display name
|
|
|
|
if animation in self._effect_map:
|
|
|
|
return self._effect_map[animation]
|
|
|
|
_LOGGER.debug("Unknown animation mode: %s", animation)
|
|
|
|
return None
|
|
|
|
|
2025-01-16 22:57:55 +01:00
|
|
|
async def set_animation_speed(self, speed: int) -> None:
|
|
|
|
"""Set the animation speed."""
|
|
|
|
data = {
|
|
|
|
"animation_speed": speed
|
|
|
|
}
|
|
|
|
_LOGGER.debug("Setting animation speed to %s", speed)
|
|
|
|
async with aiohttp.ClientSession() as session:
|
|
|
|
url = f"{self.coordinator.api_url}/api/devices/{self._device_id}/state"
|
|
|
|
async with session.put(url, json=data) as response:
|
|
|
|
if response.status != 200:
|
|
|
|
_LOGGER.error("Failed to update LED configuration: %s", response.status)
|
|
|
|
return
|
|
|
|
|
2025-01-16 22:09:00 +01:00
|
|
|
async def async_turn_on(self, **kwargs: Any) -> None:
|
|
|
|
"""Turn the light on."""
|
|
|
|
data = {
|
|
|
|
"state": True
|
|
|
|
}
|
|
|
|
_LOGGER.debug("Data given in async_turn_on: %s", kwargs)
|
|
|
|
if ATTR_BRIGHTNESS in kwargs:
|
|
|
|
brightness = kwargs[ATTR_BRIGHTNESS]
|
|
|
|
data["brightness"] = int(brightness)
|
|
|
|
|
|
|
|
if ATTR_RGB_COLOR in kwargs:
|
|
|
|
r, g, b = kwargs[ATTR_RGB_COLOR]
|
|
|
|
# Convert RGB values to hex string
|
|
|
|
color_hex = f"{r:02x}{g:02x}{b:02x}"
|
|
|
|
# Always maintain 5 colors array, set first color and pad with black
|
|
|
|
data["colors"] = [color_hex] + ['000000'] * 4
|
|
|
|
|
|
|
|
if ATTR_EFFECT in kwargs:
|
|
|
|
# Convert HA effect name back to API animation mode
|
|
|
|
effect_name = kwargs[ATTR_EFFECT]
|
2025-01-17 16:02:38 +01:00
|
|
|
effect_key = list(self._effect_map.keys())[list(self._effect_map.values()).index(effect_name)]
|
|
|
|
if effect_key is not None:
|
|
|
|
# get effect key from value
|
|
|
|
data["animation"] = effect_key
|
2025-01-16 22:09:00 +01:00
|
|
|
else:
|
2025-01-17 16:02:38 +01:00
|
|
|
_LOGGER.warning("Unknown effect name: %s (%s). Valid effects are: %s", effect_name, effect_key, list(self._effect_map.keys()))
|
2025-01-16 22:09:00 +01:00
|
|
|
|
|
|
|
_LOGGER.debug("Sending data to API: %s", data)
|
|
|
|
async with aiohttp.ClientSession() as session:
|
|
|
|
url = f"{self.coordinator.api_url}/api/devices/{self._device_id}/state"
|
|
|
|
async with session.put(url, json=data) as response:
|
|
|
|
if response.status != 200:
|
|
|
|
_LOGGER.error("Failed to turn on light: %s", response.status)
|
|
|
|
|
|
|
|
async def async_turn_off(self, **kwargs: Any) -> None:
|
|
|
|
"""Turn the light off."""
|
|
|
|
async with aiohttp.ClientSession() as session:
|
|
|
|
url = f"{self.coordinator.api_url}/api/devices/{self._device_id}/state"
|
|
|
|
async with session.put(url, json={"state": False}) as response:
|
|
|
|
if response.status != 200:
|
|
|
|
_LOGGER.error("Failed to turn off light: %s", response.status)
|