core/homeassistant/components/braviatv/media_player.py

294 lines
9.8 KiB
Python

"""Media player support for Bravia TV integration."""
from __future__ import annotations
from datetime import datetime
from typing import Any
from homeassistant.components.media_player import (
BrowseError,
BrowseMedia,
MediaClass,
MediaPlayerDeviceClass,
MediaPlayerEntity,
MediaPlayerEntityFeature,
MediaPlayerState,
MediaType,
)
from homeassistant.core import HomeAssistant
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from . import BraviaTVConfigEntry
from .const import SourceType
from .entity import BraviaTVEntity
async def async_setup_entry(
hass: HomeAssistant,
config_entry: BraviaTVConfigEntry,
async_add_entities: AddEntitiesCallback,
) -> None:
"""Set up Bravia TV Media Player from a config_entry."""
coordinator = config_entry.runtime_data
unique_id = config_entry.unique_id
assert unique_id is not None
async_add_entities(
[BraviaTVMediaPlayer(coordinator, unique_id, config_entry.title)]
)
class BraviaTVMediaPlayer(BraviaTVEntity, MediaPlayerEntity):
"""Representation of a Bravia TV Media Player."""
_attr_name = None
_attr_assumed_state = True
_attr_device_class = MediaPlayerDeviceClass.TV
_attr_supported_features = (
MediaPlayerEntityFeature.PAUSE
| MediaPlayerEntityFeature.VOLUME_STEP
| MediaPlayerEntityFeature.VOLUME_MUTE
| MediaPlayerEntityFeature.VOLUME_SET
| MediaPlayerEntityFeature.PREVIOUS_TRACK
| MediaPlayerEntityFeature.NEXT_TRACK
| MediaPlayerEntityFeature.TURN_ON
| MediaPlayerEntityFeature.TURN_OFF
| MediaPlayerEntityFeature.SELECT_SOURCE
| MediaPlayerEntityFeature.PLAY
| MediaPlayerEntityFeature.STOP
| MediaPlayerEntityFeature.PLAY_MEDIA
| MediaPlayerEntityFeature.BROWSE_MEDIA
)
@property
def state(self) -> MediaPlayerState:
"""Return the state of the device."""
if self.coordinator.is_on:
return MediaPlayerState.ON
return MediaPlayerState.OFF
@property
def source(self) -> str | None:
"""Return the current input source."""
return self.coordinator.source
@property
def source_list(self) -> list[str]:
"""List of available input sources."""
return self.coordinator.source_list
@property
def volume_level(self) -> float | None:
"""Volume level of the media player (0..1)."""
return self.coordinator.volume_level
@property
def is_volume_muted(self) -> bool:
"""Boolean if volume is currently muted."""
return self.coordinator.volume_muted
@property
def media_title(self) -> str | None:
"""Title of current playing media."""
return self.coordinator.media_title
@property
def media_channel(self) -> str | None:
"""Channel currently playing."""
return self.coordinator.media_channel
@property
def media_content_id(self) -> str | None:
"""Content ID of current playing media."""
return self.coordinator.media_content_id
@property
def media_content_type(self) -> MediaType | None:
"""Content type of current playing media."""
return self.coordinator.media_content_type
@property
def media_duration(self) -> int | None:
"""Duration of current playing media in seconds."""
return self.coordinator.media_duration
@property
def media_position(self) -> int | None:
"""Position of current playing media in seconds."""
return self.coordinator.media_position
@property
def media_position_updated_at(self) -> datetime | None:
"""When was the position of the current playing media valid."""
return self.coordinator.media_position_updated_at
async def async_turn_on(self) -> None:
"""Turn the device on."""
await self.coordinator.async_turn_on()
async def async_turn_off(self) -> None:
"""Turn the device off."""
await self.coordinator.async_turn_off()
async def async_set_volume_level(self, volume: float) -> None:
"""Set volume level, range 0..1."""
await self.coordinator.async_set_volume_level(volume)
async def async_volume_up(self) -> None:
"""Send volume up command."""
await self.coordinator.async_volume_up()
async def async_volume_down(self) -> None:
"""Send volume down command."""
await self.coordinator.async_volume_down()
async def async_mute_volume(self, mute: bool) -> None:
"""Send mute command."""
await self.coordinator.async_volume_mute(mute)
async def async_browse_media(
self,
media_content_type: MediaType | str | None = None,
media_content_id: str | None = None,
) -> BrowseMedia:
"""Browse apps and channels."""
if not media_content_id:
await self.coordinator.async_update_sources()
return await self.async_browse_media_root()
path = media_content_id.partition("/")
if path[0] == "apps":
return await self.async_browse_media_apps(True)
if path[0] == "channels":
return await self.async_browse_media_channels(True)
raise BrowseError(f"Media not found: {media_content_type} / {media_content_id}")
async def async_browse_media_root(self) -> BrowseMedia:
"""Return root media objects."""
return BrowseMedia(
title="Sony TV",
media_class=MediaClass.DIRECTORY,
media_content_id="",
media_content_type="",
can_play=False,
can_expand=True,
children=[
await self.async_browse_media_apps(),
await self.async_browse_media_channels(),
],
)
async def async_browse_media_apps(self, expanded: bool = False) -> BrowseMedia:
"""Return apps media objects."""
if expanded:
children = [
BrowseMedia(
title=item["title"],
media_class=MediaClass.APP,
media_content_id=uri,
media_content_type=MediaType.APP,
can_play=False,
can_expand=False,
thumbnail=self.get_browse_image_url(
MediaType.APP, uri, media_image_id=None
),
)
for uri, item in self.coordinator.source_map.items()
if item["type"] == SourceType.APP
]
else:
children = None
return BrowseMedia(
title="Applications",
media_class=MediaClass.DIRECTORY,
media_content_id="apps",
media_content_type=MediaType.APPS,
children_media_class=MediaClass.APP,
can_play=False,
can_expand=True,
children=children,
)
async def async_browse_media_channels(self, expanded: bool = False) -> BrowseMedia:
"""Return channels media objects."""
if expanded:
children = [
BrowseMedia(
title=item["title"],
media_class=MediaClass.CHANNEL,
media_content_id=uri,
media_content_type=MediaType.CHANNEL,
can_play=False,
can_expand=False,
)
for uri, item in self.coordinator.source_map.items()
if item["type"] == SourceType.CHANNEL
]
else:
children = None
return BrowseMedia(
title="Channels",
media_class=MediaClass.DIRECTORY,
media_content_id="channels",
media_content_type=MediaType.CHANNELS,
children_media_class=MediaClass.CHANNEL,
can_play=False,
can_expand=True,
children=children,
)
async def async_get_browse_image(
self,
media_content_type: MediaType | str,
media_content_id: str,
media_image_id: str | None = None,
) -> tuple[bytes | None, str | None]:
"""Serve album art. Returns (content, content_type)."""
if media_content_type == MediaType.APP and media_content_id:
if icon := self.coordinator.source_map[media_content_id].get("icon"):
(content, content_type) = await self._async_fetch_image(icon)
if content_type:
# Fix invalid Content-Type header returned by Bravia
content_type = content_type.replace("Content-Type: ", "")
return (content, content_type)
return None, None
async def async_play_media(
self, media_type: MediaType | str, media_id: str, **kwargs: Any
) -> None:
"""Play a piece of media."""
await self.coordinator.async_play_media(media_type, media_id, **kwargs)
async def async_select_source(self, source: str) -> None:
"""Set the input source."""
await self.coordinator.async_select_source(source)
async def async_media_play(self) -> None:
"""Send play command."""
await self.coordinator.async_media_play()
async def async_media_pause(self) -> None:
"""Send pause command."""
await self.coordinator.async_media_pause()
async def async_media_play_pause(self) -> None:
"""Send pause command that toggle play/pause."""
await self.coordinator.async_media_pause()
async def async_media_stop(self) -> None:
"""Send media stop command to media player."""
await self.coordinator.async_media_stop()
async def async_media_next_track(self) -> None:
"""Send next track command."""
await self.coordinator.async_media_next_track()
async def async_media_previous_track(self) -> None:
"""Send previous track command."""
await self.coordinator.async_media_previous_track()