176 lines
5.7 KiB
Python
176 lines
5.7 KiB
Python
"""Support for controlling projector via the PJLink protocol."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from pypjlink import MUTE_AUDIO, Projector
|
|
from pypjlink.projector import ProjectorError
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components.media_player import (
|
|
PLATFORM_SCHEMA as MEDIA_PLAYER_PLATFORM_SCHEMA,
|
|
MediaPlayerEntity,
|
|
MediaPlayerEntityFeature,
|
|
MediaPlayerState,
|
|
)
|
|
from homeassistant.const import CONF_HOST, CONF_NAME, CONF_PASSWORD, CONF_PORT
|
|
from homeassistant.core import HomeAssistant
|
|
import homeassistant.helpers.config_validation as cv
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
|
|
|
from .const import CONF_ENCODING, DEFAULT_ENCODING, DEFAULT_PORT, DOMAIN
|
|
|
|
ERR_PROJECTOR_UNAVAILABLE = "projector unavailable"
|
|
|
|
PLATFORM_SCHEMA = MEDIA_PLAYER_PLATFORM_SCHEMA.extend(
|
|
{
|
|
vol.Required(CONF_HOST): cv.string,
|
|
vol.Optional(CONF_PORT, default=DEFAULT_PORT): cv.port,
|
|
vol.Optional(CONF_NAME): cv.string,
|
|
vol.Optional(CONF_ENCODING, default=DEFAULT_ENCODING): cv.string,
|
|
vol.Optional(CONF_PASSWORD): cv.string,
|
|
}
|
|
)
|
|
|
|
|
|
def setup_platform(
|
|
hass: HomeAssistant,
|
|
config: ConfigType,
|
|
add_entities: AddEntitiesCallback,
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
) -> None:
|
|
"""Set up the PJLink platform."""
|
|
host = config.get(CONF_HOST)
|
|
port = config.get(CONF_PORT)
|
|
name = config.get(CONF_NAME)
|
|
encoding = config.get(CONF_ENCODING)
|
|
password = config.get(CONF_PASSWORD)
|
|
|
|
if DOMAIN not in hass.data:
|
|
hass.data[DOMAIN] = {}
|
|
hass_data = hass.data[DOMAIN]
|
|
|
|
device_label = f"{host}:{port}"
|
|
if device_label in hass_data:
|
|
return
|
|
|
|
device = PjLinkDevice(host, port, name, encoding, password)
|
|
hass_data[device_label] = device
|
|
add_entities([device], True)
|
|
|
|
|
|
def format_input_source(input_source_name, input_source_number):
|
|
"""Format input source for display in UI."""
|
|
return f"{input_source_name} {input_source_number}"
|
|
|
|
|
|
class PjLinkDevice(MediaPlayerEntity):
|
|
"""Representation of a PJLink device."""
|
|
|
|
_attr_supported_features = (
|
|
MediaPlayerEntityFeature.VOLUME_MUTE
|
|
| MediaPlayerEntityFeature.TURN_ON
|
|
| MediaPlayerEntityFeature.TURN_OFF
|
|
| MediaPlayerEntityFeature.SELECT_SOURCE
|
|
)
|
|
|
|
def __init__(self, host, port, name, encoding, password):
|
|
"""Iinitialize the PJLink device."""
|
|
self._host = host
|
|
self._port = port
|
|
self._password = password
|
|
self._encoding = encoding
|
|
self._source_name_mapping = {}
|
|
|
|
self._attr_name = name
|
|
self._attr_is_volume_muted = False
|
|
self._attr_state = MediaPlayerState.OFF
|
|
self._attr_source = None
|
|
self._attr_source_list = []
|
|
self._attr_available = False
|
|
|
|
def _force_off(self):
|
|
self._attr_state = MediaPlayerState.OFF
|
|
self._attr_is_volume_muted = False
|
|
self._attr_source = None
|
|
|
|
def _setup_projector(self):
|
|
try:
|
|
with self.projector() as projector:
|
|
if not self._attr_name:
|
|
self._attr_name = projector.get_name()
|
|
inputs = projector.get_inputs()
|
|
except ProjectorError as err:
|
|
if str(err) == ERR_PROJECTOR_UNAVAILABLE:
|
|
return False
|
|
raise
|
|
|
|
self._source_name_mapping = {format_input_source(*x): x for x in inputs}
|
|
self._attr_source_list = sorted(self._source_name_mapping)
|
|
return True
|
|
|
|
def projector(self):
|
|
"""Create PJLink Projector instance."""
|
|
|
|
try:
|
|
projector = Projector.from_address(self._host, self._port)
|
|
projector.authenticate(self._password)
|
|
except (TimeoutError, OSError) as err:
|
|
self._attr_available = False
|
|
raise ProjectorError(ERR_PROJECTOR_UNAVAILABLE) from err
|
|
|
|
return projector
|
|
|
|
def update(self) -> None:
|
|
"""Get the latest state from the device."""
|
|
|
|
if not self._attr_available:
|
|
self._attr_available = self._setup_projector()
|
|
|
|
if not self._attr_available:
|
|
self._force_off()
|
|
return
|
|
|
|
try:
|
|
with self.projector() as projector:
|
|
pwstate = projector.get_power()
|
|
if pwstate in ("on", "warm-up"):
|
|
self._attr_state = MediaPlayerState.ON
|
|
self._attr_is_volume_muted = projector.get_mute()[1]
|
|
self._attr_source = format_input_source(*projector.get_input())
|
|
else:
|
|
self._force_off()
|
|
except KeyError as err:
|
|
if str(err) == "'OK'":
|
|
self._force_off()
|
|
else:
|
|
raise
|
|
except ProjectorError as err:
|
|
if str(err) == "unavailable time":
|
|
self._force_off()
|
|
elif str(err) == ERR_PROJECTOR_UNAVAILABLE:
|
|
self._attr_available = False
|
|
else:
|
|
raise
|
|
|
|
def turn_off(self) -> None:
|
|
"""Turn projector off."""
|
|
with self.projector() as projector:
|
|
projector.set_power("off")
|
|
|
|
def turn_on(self) -> None:
|
|
"""Turn projector on."""
|
|
with self.projector() as projector:
|
|
projector.set_power("on")
|
|
|
|
def mute_volume(self, mute: bool) -> None:
|
|
"""Mute (true) of unmute (false) media player."""
|
|
with self.projector() as projector:
|
|
projector.set_mute(MUTE_AUDIO, mute)
|
|
|
|
def select_source(self, source: str) -> None:
|
|
"""Set the input source."""
|
|
source = self._source_name_mapping[source]
|
|
with self.projector() as projector:
|
|
projector.set_input(*source)
|