mirror of https://github.com/home-assistant/core
294 lines
9.8 KiB
Python
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()
|