2019-02-13 20:21:14 +00:00
|
|
|
"""Provide functionality to interact with Cast devices on the network."""
|
2018-06-25 19:59:05 +00:00
|
|
|
import asyncio
|
2020-05-11 17:22:26 +00:00
|
|
|
import json
|
2015-03-04 07:50:54 +00:00
|
|
|
import logging
|
2019-09-10 20:05:46 +00:00
|
|
|
from typing import Optional
|
2015-03-04 07:50:54 +00:00
|
|
|
|
2019-09-10 20:05:46 +00:00
|
|
|
import pychromecast
|
2019-12-08 16:30:57 +00:00
|
|
|
from pychromecast.controllers.homeassistant import HomeAssistantController
|
|
|
|
from pychromecast.controllers.multizone import MultizoneManager
|
2020-05-11 17:22:26 +00:00
|
|
|
from pychromecast.quick_play import quick_play
|
2019-09-10 20:05:46 +00:00
|
|
|
from pychromecast.socket_client import (
|
|
|
|
CONNECTION_STATUS_CONNECTED,
|
|
|
|
CONNECTION_STATUS_DISCONNECTED,
|
|
|
|
)
|
2018-09-09 12:26:06 +00:00
|
|
|
import voluptuous as vol
|
2016-09-04 02:14:28 +00:00
|
|
|
|
2020-05-15 16:09:21 +00:00
|
|
|
from homeassistant.components import zeroconf
|
2020-04-25 16:00:57 +00:00
|
|
|
from homeassistant.components.media_player import PLATFORM_SCHEMA, MediaPlayerEntity
|
2019-02-08 22:18:18 +00:00
|
|
|
from homeassistant.components.media_player.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
MEDIA_TYPE_MOVIE,
|
|
|
|
MEDIA_TYPE_MUSIC,
|
|
|
|
MEDIA_TYPE_TVSHOW,
|
|
|
|
SUPPORT_NEXT_TRACK,
|
|
|
|
SUPPORT_PAUSE,
|
|
|
|
SUPPORT_PLAY,
|
|
|
|
SUPPORT_PLAY_MEDIA,
|
|
|
|
SUPPORT_PREVIOUS_TRACK,
|
|
|
|
SUPPORT_SEEK,
|
|
|
|
SUPPORT_STOP,
|
|
|
|
SUPPORT_TURN_OFF,
|
|
|
|
SUPPORT_TURN_ON,
|
|
|
|
SUPPORT_VOLUME_MUTE,
|
|
|
|
SUPPORT_VOLUME_SET,
|
|
|
|
)
|
2016-02-19 05:27:50 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_HOST,
|
|
|
|
EVENT_HOMEASSISTANT_STOP,
|
|
|
|
STATE_IDLE,
|
|
|
|
STATE_OFF,
|
|
|
|
STATE_PAUSED,
|
|
|
|
STATE_PLAYING,
|
|
|
|
)
|
2018-09-09 12:26:06 +00:00
|
|
|
from homeassistant.core import callback
|
|
|
|
from homeassistant.exceptions import PlatformNotReady
|
2016-09-04 02:14:28 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2019-09-10 20:05:46 +00:00
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
2018-09-09 12:26:06 +00:00
|
|
|
from homeassistant.helpers.typing import ConfigType, HomeAssistantType
|
2016-12-04 23:30:55 +00:00
|
|
|
import homeassistant.util.dt as dt_util
|
2019-04-29 16:53:22 +00:00
|
|
|
from homeassistant.util.logging import async_create_catching_coro
|
2015-03-04 07:50:54 +00:00
|
|
|
|
2019-09-10 20:05:46 +00:00
|
|
|
from .const import (
|
|
|
|
ADDED_CAST_DEVICES_KEY,
|
|
|
|
CAST_MULTIZONE_MANAGER_KEY,
|
|
|
|
DEFAULT_PORT,
|
2019-12-08 16:30:57 +00:00
|
|
|
DOMAIN as CAST_DOMAIN,
|
|
|
|
KNOWN_CHROMECAST_INFO_KEY,
|
|
|
|
SIGNAL_CAST_DISCOVERED,
|
2019-09-11 18:34:10 +00:00
|
|
|
SIGNAL_HASS_CAST_SHOW_VIEW,
|
2019-09-10 20:05:46 +00:00
|
|
|
)
|
2020-04-10 15:19:44 +00:00
|
|
|
from .discovery import setup_internal_discovery
|
2020-04-10 17:36:57 +00:00
|
|
|
from .helpers import CastStatusListener, ChromecastInfo, ChromeCastZeroconf
|
2016-09-04 02:14:28 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_IGNORE_CEC = "ignore_cec"
|
2020-06-30 14:35:10 +00:00
|
|
|
CONF_UUID = "uuid"
|
2020-02-22 00:10:02 +00:00
|
|
|
CAST_SPLASH = "https://www.home-assistant.io/images/cast/splash.png"
|
2016-09-04 02:14:28 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_CAST = (
|
|
|
|
SUPPORT_PAUSE
|
|
|
|
| SUPPORT_PLAY
|
|
|
|
| SUPPORT_PLAY_MEDIA
|
|
|
|
| SUPPORT_STOP
|
|
|
|
| SUPPORT_TURN_OFF
|
|
|
|
| SUPPORT_TURN_ON
|
|
|
|
| SUPPORT_VOLUME_MUTE
|
|
|
|
| SUPPORT_VOLUME_SET
|
|
|
|
)
|
2016-09-04 02:14:28 +00:00
|
|
|
|
2019-02-12 23:00:54 +00:00
|
|
|
|
2020-06-30 14:35:10 +00:00
|
|
|
ENTITY_SCHEMA = vol.All(
|
|
|
|
cv.deprecated(CONF_HOST, invalidation_version="0.116"),
|
|
|
|
vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Exclusive(CONF_HOST, "device_identifier"): cv.string,
|
|
|
|
vol.Exclusive(CONF_UUID, "device_identifier"): cv.string,
|
|
|
|
vol.Optional(CONF_IGNORE_CEC): vol.All(cv.ensure_list, [cv.string]),
|
|
|
|
}
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = vol.All(
|
|
|
|
cv.deprecated(CONF_HOST, invalidation_version="0.116"),
|
|
|
|
PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Exclusive(CONF_HOST, "device_identifier"): cv.string,
|
|
|
|
vol.Exclusive(CONF_UUID, "device_identifier"): cv.string,
|
|
|
|
vol.Optional(CONF_IGNORE_CEC): vol.All(cv.ensure_list, [cv.string]),
|
|
|
|
}
|
|
|
|
),
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2015-08-18 05:37:01 +00:00
|
|
|
|
2015-03-04 07:50:54 +00:00
|
|
|
|
2018-02-23 17:31:22 +00:00
|
|
|
@callback
|
2019-07-31 19:25:30 +00:00
|
|
|
def _async_create_cast_device(hass: HomeAssistantType, info: ChromecastInfo):
|
2018-02-23 17:31:22 +00:00
|
|
|
"""Create a CastDevice Entity from the chromecast object.
|
|
|
|
|
2018-03-23 21:02:52 +00:00
|
|
|
Returns None if the cast device has already been added.
|
2018-02-23 17:31:22 +00:00
|
|
|
"""
|
2019-03-30 17:19:18 +00:00
|
|
|
_LOGGER.debug("_async_create_cast_device: %s", info)
|
2018-03-23 21:02:52 +00:00
|
|
|
if info.uuid is None:
|
2020-04-12 22:02:28 +00:00
|
|
|
_LOGGER.error("_async_create_cast_device uuid none: %s", info)
|
|
|
|
return None
|
2018-02-23 17:31:22 +00:00
|
|
|
|
|
|
|
# Found a cast with UUID
|
|
|
|
added_casts = hass.data[ADDED_CAST_DEVICES_KEY]
|
2018-03-23 21:02:52 +00:00
|
|
|
if info.uuid in added_casts:
|
|
|
|
# Already added this one, the entity will take care of moved hosts
|
|
|
|
# itself
|
2018-02-23 17:31:22 +00:00
|
|
|
return None
|
2018-03-23 21:02:52 +00:00
|
|
|
# -> New cast device
|
|
|
|
added_casts.add(info.uuid)
|
|
|
|
return CastDevice(info)
|
2015-03-04 07:50:54 +00:00
|
|
|
|
2017-08-06 16:15:01 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
async def async_setup_platform(
|
|
|
|
hass: HomeAssistantType, config: ConfigType, async_add_entities, discovery_info=None
|
|
|
|
):
|
2020-06-30 14:35:10 +00:00
|
|
|
"""Set up the Cast platform.
|
2018-06-14 19:17:54 +00:00
|
|
|
|
|
|
|
Deprecated.
|
|
|
|
"""
|
|
|
|
_LOGGER.warning(
|
2019-07-31 19:25:30 +00:00
|
|
|
"Setting configuration for Cast via platform is deprecated. "
|
|
|
|
"Configure via Cast integration instead."
|
2020-07-05 21:04:19 +00:00
|
|
|
"This option will become invalid in version 0.116"
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
await _async_setup_platform(hass, config, async_add_entities, discovery_info)
|
2018-06-14 19:17:54 +00:00
|
|
|
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
async def async_setup_entry(hass, config_entry, async_add_entities):
|
2018-06-14 19:17:54 +00:00
|
|
|
"""Set up Cast from a config entry."""
|
2019-07-31 19:25:30 +00:00
|
|
|
config = hass.data[CAST_DOMAIN].get("media_player", {})
|
2018-06-25 19:59:05 +00:00
|
|
|
if not isinstance(config, list):
|
|
|
|
config = [config]
|
|
|
|
|
2018-09-23 21:35:07 +00:00
|
|
|
# no pending task
|
2019-07-31 19:25:30 +00:00
|
|
|
done, _ = await asyncio.wait(
|
2020-06-30 14:35:10 +00:00
|
|
|
[
|
|
|
|
_async_setup_platform(hass, ENTITY_SCHEMA(cfg), async_add_entities, None)
|
|
|
|
for cfg in config
|
|
|
|
]
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-09-23 21:35:07 +00:00
|
|
|
if any([task.exception() for task in done]):
|
2019-03-11 02:57:30 +00:00
|
|
|
exceptions = [task.exception() for task in done]
|
|
|
|
for exception in exceptions:
|
|
|
|
_LOGGER.debug("Failed to setup chromecast", exc_info=exception)
|
2018-09-23 21:35:07 +00:00
|
|
|
raise PlatformNotReady
|
2018-06-14 19:17:54 +00:00
|
|
|
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
async def _async_setup_platform(
|
|
|
|
hass: HomeAssistantType, config: ConfigType, async_add_entities, discovery_info
|
|
|
|
):
|
2018-02-23 17:31:22 +00:00
|
|
|
"""Set up the cast platform."""
|
|
|
|
# Import CEC IGNORE attributes
|
|
|
|
pychromecast.IGNORE_CEC += config.get(CONF_IGNORE_CEC, [])
|
2018-03-23 21:02:52 +00:00
|
|
|
hass.data.setdefault(ADDED_CAST_DEVICES_KEY, set())
|
2020-06-30 14:35:10 +00:00
|
|
|
hass.data.setdefault(KNOWN_CHROMECAST_INFO_KEY, dict())
|
2015-08-18 05:37:01 +00:00
|
|
|
|
2018-03-23 21:02:52 +00:00
|
|
|
info = None
|
|
|
|
if discovery_info is not None:
|
2020-06-30 14:35:10 +00:00
|
|
|
info = ChromecastInfo(
|
|
|
|
host=discovery_info["host"], port=discovery_info["port"], services=None
|
|
|
|
)
|
|
|
|
elif CONF_UUID in config:
|
|
|
|
info = ChromecastInfo(uuid=config[CONF_UUID], services=None)
|
2018-02-23 17:31:22 +00:00
|
|
|
elif CONF_HOST in config:
|
2020-06-30 14:35:10 +00:00
|
|
|
info = ChromecastInfo(host=config[CONF_HOST], port=DEFAULT_PORT, services=None)
|
2018-02-23 17:31:22 +00:00
|
|
|
|
2018-03-23 21:02:52 +00:00
|
|
|
@callback
|
|
|
|
def async_cast_discovered(discover: ChromecastInfo) -> None:
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Handle discovery of a new chromecast."""
|
2020-06-30 14:35:10 +00:00
|
|
|
if info is not None and (
|
|
|
|
(info.uuid is not None and info.uuid != discover.uuid)
|
|
|
|
or (info.host is not None and info.host_port != discover.host_port)
|
|
|
|
):
|
2020-04-12 22:02:28 +00:00
|
|
|
# Waiting for a specific cast device, this is not it.
|
2018-03-23 21:02:52 +00:00
|
|
|
return
|
2018-02-23 17:31:22 +00:00
|
|
|
|
2018-03-23 21:02:52 +00:00
|
|
|
cast_device = _async_create_cast_device(hass, discover)
|
|
|
|
if cast_device is not None:
|
2018-08-24 14:37:30 +00:00
|
|
|
async_add_entities([cast_device])
|
2018-03-23 21:02:52 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
async_dispatcher_connect(hass, SIGNAL_CAST_DISCOVERED, async_cast_discovered)
|
2018-03-23 21:02:52 +00:00
|
|
|
# Re-play the callback for all past chromecasts, store the objects in
|
|
|
|
# a list to avoid concurrent modification resulting in exception.
|
2020-06-30 14:35:10 +00:00
|
|
|
for chromecast in hass.data[KNOWN_CHROMECAST_INFO_KEY].values():
|
2018-03-23 21:02:52 +00:00
|
|
|
async_cast_discovered(chromecast)
|
2018-02-23 17:31:22 +00:00
|
|
|
|
2020-05-15 16:09:21 +00:00
|
|
|
ChromeCastZeroconf.set_zeroconf(await zeroconf.async_get_instance(hass))
|
2020-04-10 15:19:44 +00:00
|
|
|
hass.async_add_executor_job(setup_internal_discovery, hass)
|
2015-03-04 07:50:54 +00:00
|
|
|
|
|
|
|
|
2020-04-25 16:00:57 +00:00
|
|
|
class CastDevice(MediaPlayerEntity):
|
2018-03-23 21:02:52 +00:00
|
|
|
"""Representation of a Cast device on the network.
|
|
|
|
|
|
|
|
This class is the holder of the pychromecast.Chromecast object and its
|
|
|
|
socket client. It therefore handles all reconnects and audio group changing
|
|
|
|
"elected leader" itself.
|
|
|
|
"""
|
|
|
|
|
2019-09-07 06:48:58 +00:00
|
|
|
def __init__(self, cast_info: ChromecastInfo):
|
2018-03-23 21:02:52 +00:00
|
|
|
"""Initialize the cast device."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-09-07 06:48:58 +00:00
|
|
|
self._cast_info = cast_info
|
2020-06-30 14:35:10 +00:00
|
|
|
self.services = cast_info.services
|
2019-09-07 06:48:58 +00:00
|
|
|
self._chromecast: Optional[pychromecast.Chromecast] = None
|
2018-03-23 21:02:52 +00:00
|
|
|
self.cast_status = None
|
|
|
|
self.media_status = None
|
2018-04-22 20:32:15 +00:00
|
|
|
self.media_status_received = None
|
2019-03-30 17:19:18 +00:00
|
|
|
self.mz_media_status = {}
|
|
|
|
self.mz_media_status_received = {}
|
2019-03-31 03:07:01 +00:00
|
|
|
self.mz_mgr = None
|
2019-09-07 06:48:58 +00:00
|
|
|
self._available = False
|
|
|
|
self._status_listener: Optional[CastStatusListener] = None
|
2019-09-11 18:34:10 +00:00
|
|
|
self._hass_cast_controller: Optional[HomeAssistantController] = None
|
2019-09-07 06:48:58 +00:00
|
|
|
|
2019-02-12 23:00:54 +00:00
|
|
|
self._add_remove_handler = None
|
2019-09-11 18:34:10 +00:00
|
|
|
self._cast_view_remove_handler = None
|
2018-03-23 21:02:52 +00:00
|
|
|
|
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""Create chromecast object when added to hass."""
|
2019-02-12 23:00:54 +00:00
|
|
|
self._add_remove_handler = async_dispatcher_connect(
|
2019-09-10 20:05:46 +00:00
|
|
|
self.hass, SIGNAL_CAST_DISCOVERED, self._async_cast_discovered
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-09-10 20:05:46 +00:00
|
|
|
self.hass.bus.async_listen_once(EVENT_HOMEASSISTANT_STOP, self._async_stop)
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.async_create_task(
|
|
|
|
async_create_catching_coro(self.async_set_cast_info(self._cast_info))
|
|
|
|
)
|
2015-03-04 07:50:54 +00:00
|
|
|
|
2019-09-11 18:34:10 +00:00
|
|
|
self._cast_view_remove_handler = async_dispatcher_connect(
|
|
|
|
self.hass, SIGNAL_HASS_CAST_SHOW_VIEW, self._handle_signal_show_view
|
|
|
|
)
|
|
|
|
|
2018-03-23 21:02:52 +00:00
|
|
|
async def async_will_remove_from_hass(self) -> None:
|
|
|
|
"""Disconnect Chromecast object when removed."""
|
2018-04-30 12:46:44 +00:00
|
|
|
await self._async_disconnect()
|
2018-03-23 21:02:52 +00:00
|
|
|
if self._cast_info.uuid is not None:
|
|
|
|
# Remove the entity from the added casts so that it can dynamically
|
|
|
|
# be re-added again.
|
|
|
|
self.hass.data[ADDED_CAST_DEVICES_KEY].remove(self._cast_info.uuid)
|
2019-02-12 23:00:54 +00:00
|
|
|
if self._add_remove_handler:
|
|
|
|
self._add_remove_handler()
|
2019-09-11 18:34:10 +00:00
|
|
|
self._add_remove_handler = None
|
|
|
|
if self._cast_view_remove_handler:
|
|
|
|
self._cast_view_remove_handler()
|
|
|
|
self._cast_view_remove_handler = None
|
2018-03-23 21:02:52 +00:00
|
|
|
|
|
|
|
async def async_set_cast_info(self, cast_info):
|
|
|
|
"""Set the cast information and set up the chromecast object."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2018-03-23 21:02:52 +00:00
|
|
|
self._cast_info = cast_info
|
|
|
|
|
|
|
|
if self._chromecast is not None:
|
2019-02-12 23:00:54 +00:00
|
|
|
# Only setup the chromecast once, added elements to services
|
|
|
|
# will automatically be picked up.
|
|
|
|
return
|
2018-03-23 21:02:52 +00:00
|
|
|
|
2020-04-12 22:02:28 +00:00
|
|
|
_LOGGER.debug(
|
2020-06-30 14:35:10 +00:00
|
|
|
"[%s %s] Connecting to cast device by service %s",
|
2020-04-12 22:02:28 +00:00
|
|
|
self.entity_id,
|
|
|
|
self._cast_info.friendly_name,
|
|
|
|
self.services,
|
|
|
|
)
|
|
|
|
chromecast = await self.hass.async_add_executor_job(
|
2020-04-21 05:57:39 +00:00
|
|
|
pychromecast.get_chromecast_from_service,
|
2020-04-12 22:02:28 +00:00
|
|
|
(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.services,
|
2020-04-12 22:02:28 +00:00
|
|
|
cast_info.uuid,
|
|
|
|
cast_info.model_name,
|
|
|
|
cast_info.friendly_name,
|
2020-07-19 10:36:59 +00:00
|
|
|
None,
|
|
|
|
None,
|
2020-04-12 22:02:28 +00:00
|
|
|
),
|
2020-06-30 14:35:10 +00:00
|
|
|
ChromeCastZeroconf.get_zeroconf(),
|
2020-04-12 22:02:28 +00:00
|
|
|
)
|
2018-03-23 21:02:52 +00:00
|
|
|
self._chromecast = chromecast
|
2019-03-30 17:19:18 +00:00
|
|
|
|
|
|
|
if CAST_MULTIZONE_MANAGER_KEY not in self.hass.data:
|
|
|
|
self.hass.data[CAST_MULTIZONE_MANAGER_KEY] = MultizoneManager()
|
2019-09-10 20:05:46 +00:00
|
|
|
|
2019-03-31 03:07:01 +00:00
|
|
|
self.mz_mgr = self.hass.data[CAST_MULTIZONE_MANAGER_KEY]
|
2019-03-30 17:19:18 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
self._status_listener = CastStatusListener(self, chromecast, self.mz_mgr)
|
2019-03-11 02:57:30 +00:00
|
|
|
self._available = False
|
2018-03-23 21:02:52 +00:00
|
|
|
self.cast_status = chromecast.status
|
|
|
|
self.media_status = chromecast.media_controller.status
|
2019-03-11 02:57:30 +00:00
|
|
|
self._chromecast.start()
|
2020-04-01 21:19:51 +00:00
|
|
|
self.async_write_ha_state()
|
2018-03-23 21:02:52 +00:00
|
|
|
|
2018-04-30 12:46:44 +00:00
|
|
|
async def _async_disconnect(self):
|
2018-03-23 21:02:52 +00:00
|
|
|
"""Disconnect Chromecast object if it is set."""
|
|
|
|
if self._chromecast is None:
|
|
|
|
# Can't disconnect if not connected.
|
|
|
|
return
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.debug(
|
2020-07-05 21:04:19 +00:00
|
|
|
"[%s %s] Disconnecting from chromecast socket",
|
2019-07-31 19:25:30 +00:00
|
|
|
self.entity_id,
|
|
|
|
self._cast_info.friendly_name,
|
|
|
|
)
|
2018-03-23 21:02:52 +00:00
|
|
|
self._available = False
|
2020-04-01 21:19:51 +00:00
|
|
|
self.async_write_ha_state()
|
2018-04-30 12:46:44 +00:00
|
|
|
|
2019-03-30 17:19:18 +00:00
|
|
|
await self.hass.async_add_executor_job(self._chromecast.disconnect)
|
2018-04-30 12:46:44 +00:00
|
|
|
|
2018-09-20 08:48:45 +00:00
|
|
|
self._invalidate()
|
|
|
|
|
2020-04-01 21:19:51 +00:00
|
|
|
self.async_write_ha_state()
|
2018-09-20 08:48:45 +00:00
|
|
|
|
|
|
|
def _invalidate(self):
|
|
|
|
"""Invalidate some attributes."""
|
2018-03-23 21:02:52 +00:00
|
|
|
self._chromecast = None
|
2018-02-23 17:31:22 +00:00
|
|
|
self.cast_status = None
|
|
|
|
self.media_status = None
|
2018-04-22 20:32:15 +00:00
|
|
|
self.media_status_received = None
|
2019-03-30 17:19:18 +00:00
|
|
|
self.mz_media_status = {}
|
|
|
|
self.mz_media_status_received = {}
|
2019-03-31 03:07:01 +00:00
|
|
|
self.mz_mgr = None
|
2019-09-11 18:34:10 +00:00
|
|
|
self._hass_cast_controller = None
|
2018-04-30 12:46:44 +00:00
|
|
|
if self._status_listener is not None:
|
|
|
|
self._status_listener.invalidate()
|
|
|
|
self._status_listener = None
|
|
|
|
|
2018-03-23 21:02:52 +00:00
|
|
|
# ========== Callbacks ==========
|
|
|
|
def new_cast_status(self, cast_status):
|
|
|
|
"""Handle updates of the cast status."""
|
|
|
|
self.cast_status = cast_status
|
|
|
|
self.schedule_update_ha_state()
|
|
|
|
|
|
|
|
def new_media_status(self, media_status):
|
|
|
|
"""Handle updates of the media status."""
|
|
|
|
self.media_status = media_status
|
2018-04-22 20:32:15 +00:00
|
|
|
self.media_status_received = dt_util.utcnow()
|
2018-03-23 21:02:52 +00:00
|
|
|
self.schedule_update_ha_state()
|
|
|
|
|
|
|
|
def new_connection_status(self, connection_status):
|
|
|
|
"""Handle updates of connection status."""
|
2019-02-12 23:00:54 +00:00
|
|
|
_LOGGER.debug(
|
2020-06-30 14:35:10 +00:00
|
|
|
"[%s %s] Received cast device connection status: %s",
|
2019-07-31 19:25:30 +00:00
|
|
|
self.entity_id,
|
|
|
|
self._cast_info.friendly_name,
|
|
|
|
connection_status.status,
|
|
|
|
)
|
2018-09-20 08:48:45 +00:00
|
|
|
if connection_status.status == CONNECTION_STATUS_DISCONNECTED:
|
|
|
|
self._available = False
|
|
|
|
self._invalidate()
|
|
|
|
self.schedule_update_ha_state()
|
|
|
|
return
|
2018-03-23 21:02:52 +00:00
|
|
|
|
|
|
|
new_available = connection_status.status == CONNECTION_STATUS_CONNECTED
|
|
|
|
if new_available != self._available:
|
|
|
|
# Connection status callbacks happen often when disconnected.
|
|
|
|
# Only update state when availability changed to put less pressure
|
|
|
|
# on state machine.
|
2019-02-12 23:00:54 +00:00
|
|
|
_LOGGER.debug(
|
2020-06-30 14:35:10 +00:00
|
|
|
"[%s %s] Cast device availability changed: %s",
|
2019-07-31 19:25:30 +00:00
|
|
|
self.entity_id,
|
|
|
|
self._cast_info.friendly_name,
|
|
|
|
connection_status.status,
|
|
|
|
)
|
2018-03-23 21:02:52 +00:00
|
|
|
self._available = new_available
|
|
|
|
self.schedule_update_ha_state()
|
|
|
|
|
2019-03-30 17:19:18 +00:00
|
|
|
def multizone_new_media_status(self, group_uuid, media_status):
|
|
|
|
"""Handle updates of audio group media status."""
|
|
|
|
_LOGGER.debug(
|
2020-06-30 14:35:10 +00:00
|
|
|
"[%s %s] Multizone %s media status: %s",
|
2019-07-31 19:25:30 +00:00
|
|
|
self.entity_id,
|
|
|
|
self._cast_info.friendly_name,
|
|
|
|
group_uuid,
|
|
|
|
media_status,
|
|
|
|
)
|
2019-03-30 17:19:18 +00:00
|
|
|
self.mz_media_status[group_uuid] = media_status
|
|
|
|
self.mz_media_status_received[group_uuid] = dt_util.utcnow()
|
|
|
|
self.schedule_update_ha_state()
|
|
|
|
|
2018-03-23 21:02:52 +00:00
|
|
|
# ========== Service Calls ==========
|
2019-03-31 03:07:01 +00:00
|
|
|
def _media_controller(self):
|
|
|
|
"""
|
|
|
|
Return media status.
|
|
|
|
|
2020-04-10 17:36:57 +00:00
|
|
|
First try from our own cast, then groups which our cast is a member in.
|
2019-03-31 03:07:01 +00:00
|
|
|
"""
|
|
|
|
media_status = self.media_status
|
|
|
|
media_controller = self._chromecast.media_controller
|
|
|
|
|
|
|
|
if media_status is None or media_status.player_state == "UNKNOWN":
|
|
|
|
groups = self.mz_media_status
|
|
|
|
for k, val in groups.items():
|
|
|
|
if val and val.player_state != "UNKNOWN":
|
2019-07-31 19:25:30 +00:00
|
|
|
media_controller = self.mz_mgr.get_multizone_mediacontroller(k)
|
2019-03-31 03:07:01 +00:00
|
|
|
break
|
|
|
|
|
|
|
|
return media_controller
|
|
|
|
|
2018-03-23 21:02:52 +00:00
|
|
|
def turn_on(self):
|
|
|
|
"""Turn on the cast device."""
|
|
|
|
|
|
|
|
if not self._chromecast.is_idle:
|
|
|
|
# Already turned on
|
|
|
|
return
|
|
|
|
|
|
|
|
if self._chromecast.app_id is not None:
|
|
|
|
# Quit the previous app before starting splash screen
|
|
|
|
self._chromecast.quit_app()
|
|
|
|
|
|
|
|
# The only way we can turn the Chromecast is on is by launching an app
|
2019-07-31 19:25:30 +00:00
|
|
|
self._chromecast.play_media(CAST_SPLASH, pychromecast.STREAM_TYPE_BUFFERED)
|
2018-03-23 21:02:52 +00:00
|
|
|
|
|
|
|
def turn_off(self):
|
|
|
|
"""Turn off the cast device."""
|
|
|
|
self._chromecast.quit_app()
|
2015-05-30 19:35:35 +00:00
|
|
|
|
2018-03-23 21:02:52 +00:00
|
|
|
def mute_volume(self, mute):
|
|
|
|
"""Mute the volume."""
|
|
|
|
self._chromecast.set_volume_muted(mute)
|
|
|
|
|
|
|
|
def set_volume_level(self, volume):
|
|
|
|
"""Set volume level, range 0..1."""
|
|
|
|
self._chromecast.set_volume(volume)
|
|
|
|
|
|
|
|
def media_play(self):
|
|
|
|
"""Send play command."""
|
2019-03-31 03:07:01 +00:00
|
|
|
media_controller = self._media_controller()
|
|
|
|
media_controller.play()
|
2018-03-23 21:02:52 +00:00
|
|
|
|
|
|
|
def media_pause(self):
|
|
|
|
"""Send pause command."""
|
2019-03-31 03:07:01 +00:00
|
|
|
media_controller = self._media_controller()
|
|
|
|
media_controller.pause()
|
2018-03-23 21:02:52 +00:00
|
|
|
|
|
|
|
def media_stop(self):
|
|
|
|
"""Send stop command."""
|
2019-03-31 03:07:01 +00:00
|
|
|
media_controller = self._media_controller()
|
|
|
|
media_controller.stop()
|
2018-03-23 21:02:52 +00:00
|
|
|
|
|
|
|
def media_previous_track(self):
|
|
|
|
"""Send previous track command."""
|
2019-03-31 03:07:01 +00:00
|
|
|
media_controller = self._media_controller()
|
2019-04-03 02:58:02 +00:00
|
|
|
media_controller.queue_prev()
|
2018-03-23 21:02:52 +00:00
|
|
|
|
|
|
|
def media_next_track(self):
|
|
|
|
"""Send next track command."""
|
2019-03-31 03:07:01 +00:00
|
|
|
media_controller = self._media_controller()
|
2019-04-03 02:58:02 +00:00
|
|
|
media_controller.queue_next()
|
2018-03-23 21:02:52 +00:00
|
|
|
|
|
|
|
def media_seek(self, position):
|
|
|
|
"""Seek the media to a specific location."""
|
2019-03-31 03:07:01 +00:00
|
|
|
media_controller = self._media_controller()
|
|
|
|
media_controller.seek(position)
|
2018-03-23 21:02:52 +00:00
|
|
|
|
|
|
|
def play_media(self, media_type, media_id, **kwargs):
|
|
|
|
"""Play media from a URL."""
|
2020-04-10 17:36:57 +00:00
|
|
|
# We do not want this to be forwarded to a group
|
2020-05-11 17:22:26 +00:00
|
|
|
if media_type == CAST_DOMAIN:
|
|
|
|
try:
|
|
|
|
app_data = json.loads(media_id)
|
|
|
|
except json.JSONDecodeError:
|
|
|
|
_LOGGER.error("Invalid JSON in media_content_id")
|
|
|
|
raise
|
|
|
|
|
|
|
|
# Special handling for passed `app_id` parameter. This will only launch
|
|
|
|
# an arbitrary cast app, generally for UX.
|
|
|
|
if "app_id" in app_data:
|
|
|
|
app_id = app_data.pop("app_id")
|
|
|
|
_LOGGER.info("Starting Cast app by ID %s", app_id)
|
|
|
|
self._chromecast.start_app(app_id)
|
|
|
|
if app_data:
|
|
|
|
_LOGGER.warning(
|
2020-07-05 21:04:19 +00:00
|
|
|
"Extra keys %s were ignored. Please use app_name to cast media",
|
2020-05-11 17:22:26 +00:00
|
|
|
app_data.keys(),
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
app_name = app_data.pop("app_name")
|
|
|
|
try:
|
|
|
|
quick_play(self._chromecast, app_name, app_data)
|
|
|
|
except NotImplementedError:
|
|
|
|
_LOGGER.error("App %s not supported", app_name)
|
|
|
|
else:
|
|
|
|
self._chromecast.media_controller.play_media(media_id, media_type)
|
2018-02-23 17:31:22 +00:00
|
|
|
|
2018-03-23 21:02:52 +00:00
|
|
|
# ========== Properties ==========
|
2015-05-30 19:35:35 +00:00
|
|
|
@property
|
|
|
|
def should_poll(self):
|
2018-04-22 20:32:15 +00:00
|
|
|
"""No polling needed."""
|
|
|
|
return False
|
2015-05-30 19:35:35 +00:00
|
|
|
|
2015-03-04 07:50:54 +00:00
|
|
|
@property
|
|
|
|
def name(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Return the name of the device."""
|
2018-03-23 21:02:52 +00:00
|
|
|
return self._cast_info.friendly_name
|
2015-03-04 07:50:54 +00:00
|
|
|
|
2018-08-29 13:46:09 +00:00
|
|
|
@property
|
|
|
|
def device_info(self):
|
|
|
|
"""Return information about the device."""
|
|
|
|
cast_info = self._cast_info
|
|
|
|
|
|
|
|
if cast_info.model_name == "Google Cast Group":
|
|
|
|
return None
|
|
|
|
|
|
|
|
return {
|
2019-07-31 19:25:30 +00:00
|
|
|
"name": cast_info.friendly_name,
|
|
|
|
"identifiers": {(CAST_DOMAIN, cast_info.uuid.replace("-", ""))},
|
|
|
|
"model": cast_info.model_name,
|
|
|
|
"manufacturer": cast_info.manufacturer,
|
2018-08-29 13:46:09 +00:00
|
|
|
}
|
|
|
|
|
2019-03-30 17:19:18 +00:00
|
|
|
def _media_status(self):
|
|
|
|
"""
|
|
|
|
Return media status.
|
|
|
|
|
2020-04-10 17:36:57 +00:00
|
|
|
First try from our own cast, then groups which our cast is a member in.
|
2019-03-30 17:19:18 +00:00
|
|
|
"""
|
|
|
|
media_status = self.media_status
|
|
|
|
media_status_received = self.media_status_received
|
|
|
|
|
|
|
|
if media_status is None or media_status.player_state == "UNKNOWN":
|
|
|
|
groups = self.mz_media_status
|
|
|
|
for k, val in groups.items():
|
|
|
|
if val and val.player_state != "UNKNOWN":
|
|
|
|
media_status = val
|
|
|
|
media_status_received = self.mz_media_status_received[k]
|
|
|
|
break
|
|
|
|
|
|
|
|
return (media_status, media_status_received)
|
|
|
|
|
2015-05-30 05:36:40 +00:00
|
|
|
@property
|
2015-06-03 22:17:03 +00:00
|
|
|
def state(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Return the state of the player."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
|
|
|
|
if media_status is None:
|
2018-03-23 21:02:52 +00:00
|
|
|
return None
|
2019-03-30 17:19:18 +00:00
|
|
|
if media_status.player_is_playing:
|
2015-06-03 22:17:03 +00:00
|
|
|
return STATE_PLAYING
|
2019-03-30 17:19:18 +00:00
|
|
|
if media_status.player_is_paused:
|
2015-06-03 22:17:03 +00:00
|
|
|
return STATE_PAUSED
|
2019-03-30 17:19:18 +00:00
|
|
|
if media_status.player_is_idle:
|
2015-06-03 22:17:03 +00:00
|
|
|
return STATE_IDLE
|
2018-07-23 08:16:05 +00:00
|
|
|
if self._chromecast is not None and self._chromecast.is_idle:
|
2015-06-03 22:17:03 +00:00
|
|
|
return STATE_OFF
|
2018-03-23 21:02:52 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def available(self):
|
|
|
|
"""Return True if the cast device is connected."""
|
|
|
|
return self._available
|
2015-03-04 07:50:54 +00:00
|
|
|
|
|
|
|
@property
|
2015-06-03 22:17:03 +00:00
|
|
|
def volume_level(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Volume level of the media player (0..1)."""
|
2015-06-09 05:49:32 +00:00
|
|
|
return self.cast_status.volume_level if self.cast_status else None
|
2015-06-03 22:17:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_volume_muted(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Boolean if volume is currently muted."""
|
2015-06-09 05:49:32 +00:00
|
|
|
return self.cast_status.volume_muted if self.cast_status else None
|
2015-06-03 22:17:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_content_id(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Content ID of current playing media."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
return media_status.content_id if media_status else None
|
2015-05-29 20:19:42 +00:00
|
|
|
|
2015-06-03 22:17:03 +00:00
|
|
|
@property
|
|
|
|
def media_content_type(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Content type of current playing media."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
if media_status is None:
|
2015-06-09 05:49:32 +00:00
|
|
|
return None
|
2019-03-30 17:19:18 +00:00
|
|
|
if media_status.media_is_tvshow:
|
2015-06-09 05:49:32 +00:00
|
|
|
return MEDIA_TYPE_TVSHOW
|
2019-03-30 17:19:18 +00:00
|
|
|
if media_status.media_is_movie:
|
2018-04-05 16:44:38 +00:00
|
|
|
return MEDIA_TYPE_MOVIE
|
2019-03-30 17:19:18 +00:00
|
|
|
if media_status.media_is_musictrack:
|
2015-06-09 05:49:32 +00:00
|
|
|
return MEDIA_TYPE_MUSIC
|
2015-06-03 22:17:03 +00:00
|
|
|
return None
|
2015-03-04 07:50:54 +00:00
|
|
|
|
2015-06-03 22:17:03 +00:00
|
|
|
@property
|
|
|
|
def media_duration(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Duration of current playing media in seconds."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
return media_status.duration if media_status else None
|
2015-05-31 20:27:59 +00:00
|
|
|
|
2015-06-03 22:17:03 +00:00
|
|
|
@property
|
|
|
|
def media_image_url(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Image url of current playing media."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
if media_status is None:
|
2015-06-09 05:49:32 +00:00
|
|
|
return None
|
|
|
|
|
2019-03-30 17:19:18 +00:00
|
|
|
images = media_status.images
|
2015-06-09 05:49:32 +00:00
|
|
|
|
2018-01-12 04:14:37 +00:00
|
|
|
return images[0].url if images and images[0].url else None
|
2015-05-30 07:52:33 +00:00
|
|
|
|
2019-04-25 05:37:29 +00:00
|
|
|
@property
|
|
|
|
def media_image_remotely_accessible(self) -> bool:
|
|
|
|
"""If the image url is remotely accessible."""
|
|
|
|
return True
|
|
|
|
|
2015-06-03 22:17:03 +00:00
|
|
|
@property
|
|
|
|
def media_title(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Title of current playing media."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
return media_status.title if media_status else None
|
2015-05-30 07:52:33 +00:00
|
|
|
|
2015-06-03 22:17:03 +00:00
|
|
|
@property
|
|
|
|
def media_artist(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Artist of current playing media (Music track only)."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
return media_status.artist if media_status else None
|
2015-03-04 07:50:54 +00:00
|
|
|
|
2015-06-03 22:17:03 +00:00
|
|
|
@property
|
2019-01-25 17:49:50 +00:00
|
|
|
def media_album_name(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Album of current playing media (Music track only)."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
return media_status.album_name if media_status else None
|
2015-06-09 05:49:32 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_album_artist(self):
|
2018-01-29 22:37:19 +00:00
|
|
|
"""Album artist of current playing media (Music track only)."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
return media_status.album_artist if media_status else None
|
2015-03-04 07:50:54 +00:00
|
|
|
|
2015-06-03 22:17:03 +00:00
|
|
|
@property
|
|
|
|
def media_track(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Track number of current playing media (Music track only)."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
return media_status.track if media_status else None
|
2015-06-03 22:17:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_series_title(self):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Return the title of the series of current playing media."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
return media_status.series_title if media_status else None
|
2015-06-03 22:17:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_season(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Season of current playing media (TV Show only)."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
return media_status.season if media_status else None
|
2015-06-03 22:17:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_episode(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Episode of current playing media (TV Show only)."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
|
|
|
return media_status.episode if media_status else None
|
2015-06-03 22:17:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def app_id(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Return the ID of the current running app."""
|
2018-03-23 21:02:52 +00:00
|
|
|
return self._chromecast.app_id if self._chromecast else None
|
2015-06-03 22:17:03 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def app_name(self):
|
2016-03-08 09:34:33 +00:00
|
|
|
"""Name of the current running app."""
|
2018-03-23 21:02:52 +00:00
|
|
|
return self._chromecast.app_display_name if self._chromecast else None
|
2015-06-03 22:17:03 +00:00
|
|
|
|
|
|
|
@property
|
2017-02-08 04:42:45 +00:00
|
|
|
def supported_features(self):
|
|
|
|
"""Flag media player features that are supported."""
|
2019-04-03 02:58:02 +00:00
|
|
|
support = SUPPORT_CAST
|
|
|
|
media_status, _ = self._media_status()
|
|
|
|
|
|
|
|
if media_status:
|
|
|
|
if media_status.supports_queue_next:
|
|
|
|
support |= SUPPORT_PREVIOUS_TRACK
|
|
|
|
if media_status.supports_queue_next:
|
|
|
|
support |= SUPPORT_NEXT_TRACK
|
|
|
|
if media_status.supports_seek:
|
|
|
|
support |= SUPPORT_SEEK
|
|
|
|
|
|
|
|
return support
|
2015-03-04 07:50:54 +00:00
|
|
|
|
2016-12-04 23:30:55 +00:00
|
|
|
@property
|
|
|
|
def media_position(self):
|
|
|
|
"""Position of current playing media in seconds."""
|
2019-03-30 17:19:18 +00:00
|
|
|
media_status, _ = self._media_status()
|
2019-07-31 19:25:30 +00:00
|
|
|
if media_status is None or not (
|
|
|
|
media_status.player_is_playing
|
|
|
|
or media_status.player_is_paused
|
|
|
|
or media_status.player_is_idle
|
|
|
|
):
|
2018-04-22 20:32:15 +00:00
|
|
|
return None
|
2019-03-30 17:19:18 +00:00
|
|
|
return media_status.current_time
|
2016-12-04 23:30:55 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_position_updated_at(self):
|
|
|
|
"""When was the position of the current playing media valid.
|
|
|
|
|
|
|
|
Returns value from homeassistant.util.dt.utcnow().
|
|
|
|
"""
|
2019-03-30 17:19:18 +00:00
|
|
|
_, media_status_recevied = self._media_status()
|
|
|
|
return media_status_recevied
|
2016-12-04 23:30:55 +00:00
|
|
|
|
2018-02-23 17:31:22 +00:00
|
|
|
@property
|
2018-03-23 21:02:52 +00:00
|
|
|
def unique_id(self) -> Optional[str]:
|
2018-03-03 18:23:55 +00:00
|
|
|
"""Return a unique ID."""
|
2018-03-23 21:02:52 +00:00
|
|
|
return self._cast_info.uuid
|
2019-09-10 20:05:46 +00:00
|
|
|
|
|
|
|
async def _async_cast_discovered(self, discover: ChromecastInfo):
|
|
|
|
"""Handle discovery of new Chromecast."""
|
|
|
|
if self._cast_info.uuid is None:
|
|
|
|
# We can't handle empty UUIDs
|
|
|
|
return
|
|
|
|
|
|
|
|
if self._cast_info.uuid != discover.uuid:
|
|
|
|
# Discovered is not our device.
|
|
|
|
return
|
|
|
|
|
|
|
|
_LOGGER.debug("Discovered chromecast with same UUID: %s", discover)
|
|
|
|
await self.async_set_cast_info(discover)
|
|
|
|
|
|
|
|
async def _async_stop(self, event):
|
|
|
|
"""Disconnect socket on Home Assistant stop."""
|
|
|
|
await self._async_disconnect()
|
2019-09-11 18:34:10 +00:00
|
|
|
|
|
|
|
def _handle_signal_show_view(
|
2020-03-26 23:23:46 +00:00
|
|
|
self,
|
|
|
|
controller: HomeAssistantController,
|
|
|
|
entity_id: str,
|
|
|
|
view_path: str,
|
|
|
|
url_path: Optional[str],
|
2019-09-11 18:34:10 +00:00
|
|
|
):
|
|
|
|
"""Handle a show view signal."""
|
|
|
|
if entity_id != self.entity_id:
|
|
|
|
return
|
|
|
|
|
|
|
|
if self._hass_cast_controller is None:
|
|
|
|
self._hass_cast_controller = controller
|
|
|
|
self._chromecast.register_handler(controller)
|
|
|
|
|
2020-03-26 23:23:46 +00:00
|
|
|
self._hass_cast_controller.show_lovelace_view(view_path, url_path)
|