212 lines
6.7 KiB
Python
212 lines
6.7 KiB
Python
"""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
|