"""Support for track controls on the Sisyphus Kinetic Art Table.""" from __future__ import annotations import aiohttp from sisyphus_control import Track from homeassistant.components.media_player import ( MediaPlayerEntity, MediaPlayerEntityFeature, MediaPlayerState, ) from homeassistant.const import CONF_HOST from homeassistant.core import HomeAssistant from homeassistant.exceptions import PlatformNotReady from homeassistant.helpers.entity_platform import AddEntitiesCallback from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType from . import DATA_SISYPHUS MEDIA_TYPE_TRACK = "sisyphus_track" async def async_setup_platform( hass: HomeAssistant, config: ConfigType, add_entities: AddEntitiesCallback, discovery_info: DiscoveryInfoType | None = None, ) -> None: """Set up a media player entity for a Sisyphus table.""" if not discovery_info: return host = discovery_info[CONF_HOST] try: table_holder = hass.data[DATA_SISYPHUS][host] table = await table_holder.get_table() except aiohttp.ClientError as err: raise PlatformNotReady() from err add_entities([SisyphusPlayer(table_holder.name, host, table)], True) class SisyphusPlayer(MediaPlayerEntity): """Representation of a Sisyphus table as a media player device.""" _attr_supported_features = ( MediaPlayerEntityFeature.VOLUME_MUTE | MediaPlayerEntityFeature.VOLUME_SET | MediaPlayerEntityFeature.TURN_OFF | MediaPlayerEntityFeature.TURN_ON | MediaPlayerEntityFeature.PAUSE | MediaPlayerEntityFeature.SHUFFLE_SET | MediaPlayerEntityFeature.PREVIOUS_TRACK | MediaPlayerEntityFeature.NEXT_TRACK | MediaPlayerEntityFeature.PLAY ) def __init__(self, name, host, table): """Initialize the Sisyphus media device.""" self._name = name self._host = host self._table = table async def async_added_to_hass(self) -> None: """Add listeners after this object has been initialized.""" self._table.add_listener(self.async_write_ha_state) async def async_update(self) -> None: """Force update table state.""" await self._table.refresh() @property def unique_id(self): """Return the UUID of the table.""" return self._table.id @property def available(self) -> bool: """Return true if the table is responding to heartbeats.""" return self._table.is_connected @property def name(self): """Return the name of the table.""" return self._name @property def state(self) -> MediaPlayerState | None: """Return the current state of the table; sleeping maps to off.""" if self._table.state in ["homing", "playing"]: return MediaPlayerState.PLAYING if self._table.state == "paused": if self._table.is_sleeping: return MediaPlayerState.OFF return MediaPlayerState.PAUSED if self._table.state == "waiting": return MediaPlayerState.IDLE return None @property def volume_level(self): """Return the current playback speed (0..1).""" return self._table.speed @property def shuffle(self): """Return True if the current playlist is in shuffle mode.""" return self._table.is_shuffle async def async_set_shuffle(self, shuffle: bool) -> None: """Change the shuffle mode of the current playlist.""" await self._table.set_shuffle(shuffle) @property def media_playlist(self): """Return the name of the current playlist.""" return self._table.active_playlist.name if self._table.active_playlist else None @property def media_title(self): """Return the title of the current track.""" return self._table.active_track.name if self._table.active_track else None @property def media_content_type(self): """Return the content type currently playing; i.e. a Sisyphus track.""" return MEDIA_TYPE_TRACK @property def media_content_id(self): """Return the track ID of the current track.""" return self._table.active_track.id if self._table.active_track else None @property def media_duration(self): """Return the total time it will take to run this track at the current speed.""" return self._table.active_track_total_time.total_seconds() @property def media_position(self): """Return the current position within the track.""" return ( self._table.active_track_total_time - self._table.active_track_remaining_time ).total_seconds() @property def media_position_updated_at(self): """Return the last time we got a position update.""" return self._table.active_track_remaining_time_as_of @property def media_image_url(self): """Return the URL for a thumbnail image of the current track.""" if self._table.active_track: return self._table.active_track.get_thumbnail_url(Track.ThumbnailSize.LARGE) return super().media_image_url async def async_turn_on(self) -> None: """Wake up a sleeping table.""" await self._table.wakeup() async def async_turn_off(self) -> None: """Put the table to sleep.""" await self._table.sleep() async def async_volume_down(self) -> None: """Slow down playback.""" await self._table.set_speed(max(0, self._table.speed - 0.1)) async def async_volume_up(self) -> None: """Speed up playback.""" await self._table.set_speed(min(1.0, self._table.speed + 0.1)) async def async_set_volume_level(self, volume: float) -> None: """Set playback speed (0..1).""" await self._table.set_speed(volume) async def async_media_play(self) -> None: """Start playing.""" await self._table.play() async def async_media_pause(self) -> None: """Pause.""" await self._table.pause() async def async_media_next_track(self) -> None: """Skip to next track.""" cur_track_index = self._get_current_track_index() await self._table.active_playlist.play( self._table.active_playlist.tracks[cur_track_index + 1] ) async def async_media_previous_track(self) -> None: """Skip to previous track.""" cur_track_index = self._get_current_track_index() await self._table.active_playlist.play( self._table.active_playlist.tracks[cur_track_index - 1] ) def _get_current_track_index(self): for index, track in enumerate(self._table.active_playlist.tracks): if track.id == self._table.active_track.id: return index return -1