core/homeassistant/components/roku/select.py

174 lines
4.7 KiB
Python

"""Support for Roku selects."""
from __future__ import annotations
from collections.abc import Awaitable, Callable
from dataclasses import dataclass
from rokuecp import Roku
from rokuecp.models import Device as RokuDevice
from homeassistant.components.select import SelectEntity, SelectEntityDescription
from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from .const import DOMAIN
from .coordinator import RokuDataUpdateCoordinator
from .entity import RokuEntity
from .helpers import format_channel_name, roku_exception_handler
@dataclass
class RokuSelectEntityDescriptionMixin:
"""Mixin for required keys."""
options_fn: Callable[[RokuDevice], list[str]]
value_fn: Callable[[RokuDevice], str | None]
set_fn: Callable[[RokuDevice, Roku, str], Awaitable[None]]
def _get_application_name(device: RokuDevice) -> str | None:
if device.app is None or device.app.name is None:
return None
if device.app.name == "Roku":
return "Home"
return device.app.name
def _get_applications(device: RokuDevice) -> list[str]:
return ["Home"] + sorted(app.name for app in device.apps if app.name is not None)
def _get_channel_name(device: RokuDevice) -> str | None:
if device.channel is None:
return None
return format_channel_name(device.channel.number, device.channel.name)
def _get_channels(device: RokuDevice) -> list[str]:
return sorted(
format_channel_name(channel.number, channel.name) for channel in device.channels
)
async def _launch_application(device: RokuDevice, roku: Roku, value: str) -> None:
if value == "Home":
await roku.remote("home")
appl = next(
(app for app in device.apps if value == app.name),
None,
)
if appl is not None and appl.app_id is not None:
await roku.launch(appl.app_id)
async def _tune_channel(device: RokuDevice, roku: Roku, value: str) -> None:
_channel = next(
(
channel
for channel in device.channels
if (
channel.name is not None
and value == format_channel_name(channel.number, channel.name)
)
or value == channel.number
),
None,
)
if _channel is not None:
await roku.tune(_channel.number)
@dataclass
class RokuSelectEntityDescription(
SelectEntityDescription, RokuSelectEntityDescriptionMixin
):
"""Describes Roku select entity."""
ENTITIES: tuple[RokuSelectEntityDescription, ...] = (
RokuSelectEntityDescription(
key="application",
name="Application",
icon="mdi:application",
set_fn=_launch_application,
value_fn=_get_application_name,
options_fn=_get_applications,
entity_registry_enabled_default=False,
),
)
CHANNEL_ENTITY = RokuSelectEntityDescription(
key="channel",
name="Channel",
icon="mdi:television",
set_fn=_tune_channel,
value_fn=_get_channel_name,
options_fn=_get_channels,
)
async def async_setup_entry(
hass: HomeAssistant,
entry: ConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up Roku select based on a config entry."""
coordinator: RokuDataUpdateCoordinator = hass.data[DOMAIN][entry.entry_id]
device: RokuDevice = coordinator.data
unique_id = device.info.serial_number
entities: list[RokuSelectEntity] = []
for description in ENTITIES:
entities.append(
RokuSelectEntity(
device_id=unique_id,
coordinator=coordinator,
description=description,
)
)
if len(device.channels) > 0:
entities.append(
RokuSelectEntity(
device_id=unique_id,
coordinator=coordinator,
description=CHANNEL_ENTITY,
)
)
async_add_entities(entities)
class RokuSelectEntity(RokuEntity, SelectEntity):
"""Defines a Roku select entity."""
entity_description: RokuSelectEntityDescription
@property
def current_option(self) -> str | None:
"""Return the current value."""
return self.entity_description.value_fn(self.coordinator.data)
@property
def options(self) -> list[str]:
"""Return a set of selectable options."""
return self.entity_description.options_fn(self.coordinator.data)
@roku_exception_handler()
async def async_select_option(self, option: str) -> None:
"""Set the option."""
await self.entity_description.set_fn(
self.coordinator.data,
self.coordinator.roku,
option,
)
await self.coordinator.async_request_refresh()