230 lines
6.8 KiB
Python
230 lines
6.8 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
|
|
from homeassistant.components.media_player.const import (
|
|
SUPPORT_NEXT_TRACK,
|
|
SUPPORT_PAUSE,
|
|
SUPPORT_PLAY,
|
|
SUPPORT_PREVIOUS_TRACK,
|
|
SUPPORT_SHUFFLE_SET,
|
|
SUPPORT_TURN_OFF,
|
|
SUPPORT_TURN_ON,
|
|
SUPPORT_VOLUME_MUTE,
|
|
SUPPORT_VOLUME_SET,
|
|
)
|
|
from homeassistant.const import (
|
|
CONF_HOST,
|
|
STATE_IDLE,
|
|
STATE_OFF,
|
|
STATE_PAUSED,
|
|
STATE_PLAYING,
|
|
)
|
|
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"
|
|
|
|
SUPPORTED_FEATURES = (
|
|
SUPPORT_VOLUME_MUTE
|
|
| SUPPORT_VOLUME_SET
|
|
| SUPPORT_TURN_OFF
|
|
| SUPPORT_TURN_ON
|
|
| SUPPORT_PAUSE
|
|
| SUPPORT_SHUFFLE_SET
|
|
| SUPPORT_PREVIOUS_TRACK
|
|
| SUPPORT_NEXT_TRACK
|
|
| SUPPORT_PLAY
|
|
)
|
|
|
|
|
|
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."""
|
|
|
|
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):
|
|
"""Add listeners after this object has been initialized."""
|
|
self._table.add_listener(self.async_write_ha_state)
|
|
|
|
async def async_update(self):
|
|
"""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):
|
|
"""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):
|
|
"""Return the current state of the table; sleeping maps to off."""
|
|
if self._table.state in ["homing", "playing"]:
|
|
return STATE_PLAYING
|
|
if self._table.state == "paused":
|
|
if self._table.is_sleeping:
|
|
return STATE_OFF
|
|
|
|
return STATE_PAUSED
|
|
if self._table.state == "waiting":
|
|
return STATE_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):
|
|
"""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 supported_features(self):
|
|
"""Return the features supported by this table."""
|
|
return SUPPORTED_FEATURES
|
|
|
|
@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):
|
|
"""Wake up a sleeping table."""
|
|
await self._table.wakeup()
|
|
|
|
async def async_turn_off(self):
|
|
"""Put the table to sleep."""
|
|
await self._table.sleep()
|
|
|
|
async def async_volume_down(self):
|
|
"""Slow down playback."""
|
|
await self._table.set_speed(max(0, self._table.speed - 0.1))
|
|
|
|
async def async_volume_up(self):
|
|
"""Speed up playback."""
|
|
await self._table.set_speed(min(1.0, self._table.speed + 0.1))
|
|
|
|
async def async_set_volume_level(self, volume):
|
|
"""Set playback speed (0..1)."""
|
|
await self._table.set_speed(volume)
|
|
|
|
async def async_media_play(self):
|
|
"""Start playing."""
|
|
await self._table.play()
|
|
|
|
async def async_media_pause(self):
|
|
"""Pause."""
|
|
await self._table.pause()
|
|
|
|
async def async_media_next_track(self):
|
|
"""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):
|
|
"""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
|