529 lines
18 KiB
Python
529 lines
18 KiB
Python
"""Vizio SmartCast Device support."""
|
|
from datetime import timedelta
|
|
import logging
|
|
from typing import Any, Callable, Dict, List, Optional, Union
|
|
|
|
from pyvizio import VizioAsync
|
|
from pyvizio.api.apps import find_app_name
|
|
from pyvizio.const import APP_HOME, INPUT_APPS, NO_APP_RUNNING, UNKNOWN_APP
|
|
|
|
from homeassistant.components.media_player import (
|
|
DEVICE_CLASS_SPEAKER,
|
|
DEVICE_CLASS_TV,
|
|
SUPPORT_SELECT_SOUND_MODE,
|
|
MediaPlayerEntity,
|
|
)
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
CONF_ACCESS_TOKEN,
|
|
CONF_DEVICE_CLASS,
|
|
CONF_EXCLUDE,
|
|
CONF_HOST,
|
|
CONF_INCLUDE,
|
|
CONF_NAME,
|
|
STATE_OFF,
|
|
STATE_ON,
|
|
)
|
|
from homeassistant.core import callback
|
|
from homeassistant.exceptions import PlatformNotReady
|
|
from homeassistant.helpers import entity_platform
|
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
|
from homeassistant.helpers.dispatcher import (
|
|
async_dispatcher_connect,
|
|
async_dispatcher_send,
|
|
)
|
|
from homeassistant.helpers.entity import Entity
|
|
from homeassistant.helpers.typing import HomeAssistantType
|
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
|
|
|
from .const import (
|
|
CONF_ADDITIONAL_CONFIGS,
|
|
CONF_APPS,
|
|
CONF_VOLUME_STEP,
|
|
DEFAULT_TIMEOUT,
|
|
DEFAULT_VOLUME_STEP,
|
|
DEVICE_ID,
|
|
DOMAIN,
|
|
ICON,
|
|
SERVICE_UPDATE_SETTING,
|
|
SUPPORTED_COMMANDS,
|
|
UPDATE_SETTING_SCHEMA,
|
|
VIZIO_AUDIO_SETTINGS,
|
|
VIZIO_DEVICE_CLASSES,
|
|
VIZIO_MUTE,
|
|
VIZIO_MUTE_ON,
|
|
VIZIO_SOUND_MODE,
|
|
VIZIO_VOLUME,
|
|
)
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
SCAN_INTERVAL = timedelta(seconds=30)
|
|
PARALLEL_UPDATES = 0
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistantType,
|
|
config_entry: ConfigEntry,
|
|
async_add_entities: Callable[[List[Entity], bool], None],
|
|
) -> None:
|
|
"""Set up a Vizio media player entry."""
|
|
host = config_entry.data[CONF_HOST]
|
|
token = config_entry.data.get(CONF_ACCESS_TOKEN)
|
|
name = config_entry.data[CONF_NAME]
|
|
device_class = config_entry.data[CONF_DEVICE_CLASS]
|
|
|
|
# If config entry options not set up, set them up, otherwise assign values managed in options
|
|
volume_step = config_entry.options.get(
|
|
CONF_VOLUME_STEP, config_entry.data.get(CONF_VOLUME_STEP, DEFAULT_VOLUME_STEP)
|
|
)
|
|
|
|
params = {}
|
|
if not config_entry.options:
|
|
params["options"] = {CONF_VOLUME_STEP: volume_step}
|
|
|
|
include_or_exclude_key = next(
|
|
(
|
|
key
|
|
for key in config_entry.data.get(CONF_APPS, {})
|
|
if key in [CONF_INCLUDE, CONF_EXCLUDE]
|
|
),
|
|
None,
|
|
)
|
|
if include_or_exclude_key:
|
|
params["options"][CONF_APPS] = {
|
|
include_or_exclude_key: config_entry.data[CONF_APPS][
|
|
include_or_exclude_key
|
|
].copy()
|
|
}
|
|
|
|
if not config_entry.data.get(CONF_VOLUME_STEP):
|
|
new_data = config_entry.data.copy()
|
|
new_data.update({CONF_VOLUME_STEP: volume_step})
|
|
params["data"] = new_data
|
|
|
|
if params:
|
|
hass.config_entries.async_update_entry(config_entry, **params)
|
|
|
|
device = VizioAsync(
|
|
DEVICE_ID,
|
|
host,
|
|
name,
|
|
auth_token=token,
|
|
device_type=VIZIO_DEVICE_CLASSES[device_class],
|
|
session=async_get_clientsession(hass, False),
|
|
timeout=DEFAULT_TIMEOUT,
|
|
)
|
|
|
|
if not await device.can_connect_with_auth_check():
|
|
_LOGGER.warning("Failed to connect to %s", host)
|
|
raise PlatformNotReady
|
|
|
|
apps_coordinator = hass.data[DOMAIN].get(CONF_APPS)
|
|
|
|
entity = VizioDevice(config_entry, device, name, device_class, apps_coordinator)
|
|
|
|
async_add_entities([entity], update_before_add=True)
|
|
platform = entity_platform.current_platform.get()
|
|
platform.async_register_entity_service(
|
|
SERVICE_UPDATE_SETTING, UPDATE_SETTING_SCHEMA, "async_update_setting"
|
|
)
|
|
|
|
|
|
class VizioDevice(MediaPlayerEntity):
|
|
"""Media Player implementation which performs REST requests to device."""
|
|
|
|
def __init__(
|
|
self,
|
|
config_entry: ConfigEntry,
|
|
device: VizioAsync,
|
|
name: str,
|
|
device_class: str,
|
|
apps_coordinator: DataUpdateCoordinator,
|
|
) -> None:
|
|
"""Initialize Vizio device."""
|
|
self._config_entry = config_entry
|
|
self._apps_coordinator = apps_coordinator
|
|
|
|
self._name = name
|
|
self._state = None
|
|
self._volume_level = None
|
|
self._volume_step = config_entry.options[CONF_VOLUME_STEP]
|
|
self._is_volume_muted = None
|
|
self._current_input = None
|
|
self._current_app = None
|
|
self._current_app_config = None
|
|
self._current_sound_mode = None
|
|
self._available_sound_modes = []
|
|
self._available_inputs = []
|
|
self._available_apps = []
|
|
self._all_apps = apps_coordinator.data if apps_coordinator else None
|
|
self._conf_apps = config_entry.options.get(CONF_APPS, {})
|
|
self._additional_app_configs = config_entry.data.get(CONF_APPS, {}).get(
|
|
CONF_ADDITIONAL_CONFIGS, []
|
|
)
|
|
self._device_class = device_class
|
|
self._supported_commands = SUPPORTED_COMMANDS[device_class]
|
|
self._device = device
|
|
self._max_volume = float(self._device.get_max_volume())
|
|
self._icon = ICON[device_class]
|
|
self._available = True
|
|
self._model = None
|
|
self._sw_version = None
|
|
|
|
def _apps_list(self, apps: List[str]) -> List[str]:
|
|
"""Return process apps list based on configured filters."""
|
|
if self._conf_apps.get(CONF_INCLUDE):
|
|
return [app for app in apps if app in self._conf_apps[CONF_INCLUDE]]
|
|
|
|
if self._conf_apps.get(CONF_EXCLUDE):
|
|
return [app for app in apps if app not in self._conf_apps[CONF_EXCLUDE]]
|
|
|
|
return apps
|
|
|
|
async def async_update(self) -> None:
|
|
"""Retrieve latest state of the device."""
|
|
if not self._model:
|
|
self._model = await self._device.get_model_name()
|
|
|
|
if not self._sw_version:
|
|
self._sw_version = await self._device.get_version()
|
|
|
|
is_on = await self._device.get_power_state(log_api_exception=False)
|
|
|
|
if is_on is None:
|
|
if self._available:
|
|
_LOGGER.warning(
|
|
"Lost connection to %s", self._config_entry.data[CONF_HOST]
|
|
)
|
|
self._available = False
|
|
return
|
|
|
|
if not self._available:
|
|
_LOGGER.info(
|
|
"Restored connection to %s", self._config_entry.data[CONF_HOST]
|
|
)
|
|
self._available = True
|
|
|
|
if not is_on:
|
|
self._state = STATE_OFF
|
|
self._volume_level = None
|
|
self._is_volume_muted = None
|
|
self._current_input = None
|
|
self._current_app = None
|
|
self._current_app_config = None
|
|
self._current_sound_mode = None
|
|
return
|
|
|
|
self._state = STATE_ON
|
|
|
|
audio_settings = await self._device.get_all_settings(
|
|
VIZIO_AUDIO_SETTINGS, log_api_exception=False
|
|
)
|
|
|
|
if audio_settings:
|
|
self._volume_level = float(audio_settings[VIZIO_VOLUME]) / self._max_volume
|
|
if VIZIO_MUTE in audio_settings:
|
|
self._is_volume_muted = (
|
|
audio_settings[VIZIO_MUTE].lower() == VIZIO_MUTE_ON
|
|
)
|
|
else:
|
|
self._is_volume_muted = None
|
|
|
|
if VIZIO_SOUND_MODE in audio_settings:
|
|
self._supported_commands |= SUPPORT_SELECT_SOUND_MODE
|
|
self._current_sound_mode = audio_settings[VIZIO_SOUND_MODE]
|
|
if not self._available_sound_modes:
|
|
self._available_sound_modes = (
|
|
await self._device.get_setting_options(
|
|
VIZIO_AUDIO_SETTINGS, VIZIO_SOUND_MODE
|
|
)
|
|
)
|
|
else:
|
|
# Explicitly remove SUPPORT_SELECT_SOUND_MODE from supported features
|
|
self._supported_commands &= ~SUPPORT_SELECT_SOUND_MODE
|
|
|
|
input_ = await self._device.get_current_input(log_api_exception=False)
|
|
if input_:
|
|
self._current_input = input_
|
|
|
|
inputs = await self._device.get_inputs_list(log_api_exception=False)
|
|
|
|
# If no inputs returned, end update
|
|
if not inputs:
|
|
return
|
|
|
|
self._available_inputs = [input_.name for input_ in inputs]
|
|
|
|
# Return before setting app variables if INPUT_APPS isn't in available inputs
|
|
if self._device_class == DEVICE_CLASS_SPEAKER or not any(
|
|
app for app in INPUT_APPS if app in self._available_inputs
|
|
):
|
|
return
|
|
|
|
# Create list of available known apps from known app list after
|
|
# filtering by CONF_INCLUDE/CONF_EXCLUDE
|
|
self._available_apps = self._apps_list([app["name"] for app in self._all_apps])
|
|
|
|
self._current_app_config = await self._device.get_current_app_config(
|
|
log_api_exception=False
|
|
)
|
|
|
|
self._current_app = find_app_name(
|
|
self._current_app_config,
|
|
[APP_HOME, *self._all_apps, *self._additional_app_configs],
|
|
)
|
|
|
|
if self._current_app == NO_APP_RUNNING:
|
|
self._current_app = None
|
|
|
|
def _get_additional_app_names(self) -> List[Dict[str, Any]]:
|
|
"""Return list of additional apps that were included in configuration.yaml."""
|
|
return [
|
|
additional_app["name"] for additional_app in self._additional_app_configs
|
|
]
|
|
|
|
@staticmethod
|
|
async def _async_send_update_options_signal(
|
|
hass: HomeAssistantType, config_entry: ConfigEntry
|
|
) -> None:
|
|
"""Send update event when Vizio config entry is updated."""
|
|
# Move this method to component level if another entity ever gets added for a single config entry.
|
|
# See here: https://github.com/home-assistant/core/pull/30653#discussion_r366426121
|
|
async_dispatcher_send(hass, config_entry.entry_id, config_entry)
|
|
|
|
async def _async_update_options(self, config_entry: ConfigEntry) -> None:
|
|
"""Update options if the update signal comes from this entity."""
|
|
self._volume_step = config_entry.options[CONF_VOLUME_STEP]
|
|
# Update so that CONF_ADDITIONAL_CONFIGS gets retained for imports
|
|
self._conf_apps.update(config_entry.options.get(CONF_APPS, {}))
|
|
|
|
async def async_update_setting(
|
|
self, setting_type: str, setting_name: str, new_value: Union[int, str]
|
|
) -> None:
|
|
"""Update a setting when update_setting service is called."""
|
|
await self._device.set_setting(
|
|
setting_type,
|
|
setting_name,
|
|
new_value,
|
|
)
|
|
|
|
async def async_added_to_hass(self) -> None:
|
|
"""Register callbacks when entity is added."""
|
|
# Register callback for when config entry is updated.
|
|
self.async_on_remove(
|
|
self._config_entry.add_update_listener(
|
|
self._async_send_update_options_signal
|
|
)
|
|
)
|
|
|
|
# Register callback for update event
|
|
self.async_on_remove(
|
|
async_dispatcher_connect(
|
|
self.hass, self._config_entry.entry_id, self._async_update_options
|
|
)
|
|
)
|
|
|
|
# Register callback for app list updates if device is a TV
|
|
@callback
|
|
def apps_list_update():
|
|
"""Update list of all apps."""
|
|
self._all_apps = self._apps_coordinator.data
|
|
self.async_write_ha_state()
|
|
|
|
if self._device_class == DEVICE_CLASS_TV:
|
|
self.async_on_remove(
|
|
self._apps_coordinator.async_add_listener(apps_list_update)
|
|
)
|
|
|
|
@property
|
|
def available(self) -> bool:
|
|
"""Return the availabiliity of the device."""
|
|
return self._available
|
|
|
|
@property
|
|
def state(self) -> Optional[str]:
|
|
"""Return the state of the device."""
|
|
return self._state
|
|
|
|
@property
|
|
def name(self) -> str:
|
|
"""Return the name of the device."""
|
|
return self._name
|
|
|
|
@property
|
|
def icon(self) -> str:
|
|
"""Return the icon of the device."""
|
|
return self._icon
|
|
|
|
@property
|
|
def volume_level(self) -> Optional[float]:
|
|
"""Return the volume level of the device."""
|
|
return self._volume_level
|
|
|
|
@property
|
|
def is_volume_muted(self):
|
|
"""Boolean if volume is currently muted."""
|
|
return self._is_volume_muted
|
|
|
|
@property
|
|
def source(self) -> Optional[str]:
|
|
"""Return current input of the device."""
|
|
if self._current_app is not None and self._current_input in INPUT_APPS:
|
|
return self._current_app
|
|
|
|
return self._current_input
|
|
|
|
@property
|
|
def source_list(self) -> List[str]:
|
|
"""Return list of available inputs of the device."""
|
|
# If Smartcast app is in input list, and the app list has been retrieved,
|
|
# show the combination with , otherwise just return inputs
|
|
if self._available_apps:
|
|
return [
|
|
*[
|
|
_input
|
|
for _input in self._available_inputs
|
|
if _input not in INPUT_APPS
|
|
],
|
|
*self._available_apps,
|
|
*[
|
|
app
|
|
for app in self._get_additional_app_names()
|
|
if app not in self._available_apps
|
|
],
|
|
]
|
|
|
|
return self._available_inputs
|
|
|
|
@property
|
|
def app_id(self) -> Optional[str]:
|
|
"""Return the ID of the current app if it is unknown by pyvizio."""
|
|
if self._current_app_config and self.app_name == UNKNOWN_APP:
|
|
return {
|
|
"APP_ID": self._current_app_config.APP_ID,
|
|
"NAME_SPACE": self._current_app_config.NAME_SPACE,
|
|
"MESSAGE": self._current_app_config.MESSAGE,
|
|
}
|
|
|
|
return None
|
|
|
|
@property
|
|
def app_name(self) -> Optional[str]:
|
|
"""Return the friendly name of the current app."""
|
|
return self._current_app
|
|
|
|
@property
|
|
def supported_features(self) -> int:
|
|
"""Flag device features that are supported."""
|
|
return self._supported_commands
|
|
|
|
@property
|
|
def unique_id(self) -> str:
|
|
"""Return the unique id of the device."""
|
|
return self._config_entry.unique_id
|
|
|
|
@property
|
|
def device_info(self) -> Dict[str, Any]:
|
|
"""Return device registry information."""
|
|
return {
|
|
"identifiers": {(DOMAIN, self._config_entry.unique_id)},
|
|
"name": self.name,
|
|
"manufacturer": "VIZIO",
|
|
"model": self._model,
|
|
"sw_version": self._sw_version,
|
|
}
|
|
|
|
@property
|
|
def device_class(self) -> str:
|
|
"""Return device class for entity."""
|
|
return self._device_class
|
|
|
|
@property
|
|
def sound_mode(self) -> Optional[str]:
|
|
"""Name of the current sound mode."""
|
|
return self._current_sound_mode
|
|
|
|
@property
|
|
def sound_mode_list(self) -> Optional[List[str]]:
|
|
"""List of available sound modes."""
|
|
return self._available_sound_modes
|
|
|
|
async def async_select_sound_mode(self, sound_mode):
|
|
"""Select sound mode."""
|
|
if sound_mode in self._available_sound_modes:
|
|
await self._device.set_setting(
|
|
VIZIO_AUDIO_SETTINGS, VIZIO_SOUND_MODE, sound_mode
|
|
)
|
|
|
|
async def async_turn_on(self) -> None:
|
|
"""Turn the device on."""
|
|
await self._device.pow_on()
|
|
|
|
async def async_turn_off(self) -> None:
|
|
"""Turn the device off."""
|
|
await self._device.pow_off()
|
|
|
|
async def async_mute_volume(self, mute: bool) -> None:
|
|
"""Mute the volume."""
|
|
if mute:
|
|
await self._device.mute_on()
|
|
self._is_volume_muted = True
|
|
else:
|
|
await self._device.mute_off()
|
|
self._is_volume_muted = False
|
|
|
|
async def async_media_previous_track(self) -> None:
|
|
"""Send previous channel command."""
|
|
await self._device.ch_down()
|
|
|
|
async def async_media_next_track(self) -> None:
|
|
"""Send next channel command."""
|
|
await self._device.ch_up()
|
|
|
|
async def async_select_source(self, source: str) -> None:
|
|
"""Select input source."""
|
|
if source in self._available_inputs:
|
|
await self._device.set_input(source)
|
|
elif source in self._get_additional_app_names():
|
|
await self._device.launch_app_config(
|
|
**next(
|
|
app["config"]
|
|
for app in self._additional_app_configs
|
|
if app["name"] == source
|
|
)
|
|
)
|
|
elif source in self._available_apps:
|
|
await self._device.launch_app(source, self._all_apps)
|
|
|
|
async def async_volume_up(self) -> None:
|
|
"""Increase volume of the device."""
|
|
await self._device.vol_up(num=self._volume_step)
|
|
|
|
if self._volume_level is not None:
|
|
self._volume_level = min(
|
|
1.0, self._volume_level + self._volume_step / self._max_volume
|
|
)
|
|
|
|
async def async_volume_down(self) -> None:
|
|
"""Decrease volume of the device."""
|
|
await self._device.vol_down(num=self._volume_step)
|
|
|
|
if self._volume_level is not None:
|
|
self._volume_level = max(
|
|
0.0, self._volume_level - self._volume_step / self._max_volume
|
|
)
|
|
|
|
async def async_set_volume_level(self, volume: float) -> None:
|
|
"""Set volume level."""
|
|
if self._volume_level is not None:
|
|
if volume > self._volume_level:
|
|
num = int(self._max_volume * (volume - self._volume_level))
|
|
await self._device.vol_up(num=num)
|
|
self._volume_level = volume
|
|
|
|
elif volume < self._volume_level:
|
|
num = int(self._max_volume * (self._volume_level - volume))
|
|
await self._device.vol_down(num=num)
|
|
self._volume_level = volume
|