2019-04-03 15:40:03 +00:00
|
|
|
"""Support for interfacing with Monoprice Blackbird 4k 8x8 HDBaseT Matrix."""
|
2022-01-03 15:04:15 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2018-04-19 09:35:38 +00:00
|
|
|
import logging
|
2018-05-02 13:21:50 +00:00
|
|
|
import socket
|
2018-04-19 09:35:38 +00:00
|
|
|
|
2019-10-18 13:12:36 +00:00
|
|
|
from pyblackbird import get_blackbird
|
|
|
|
from serial import SerialException
|
2018-04-19 09:35:38 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2022-04-05 21:53:45 +00:00
|
|
|
from homeassistant.components.media_player import (
|
|
|
|
PLATFORM_SCHEMA,
|
|
|
|
MediaPlayerEntity,
|
|
|
|
MediaPlayerEntityFeature,
|
2022-09-08 09:03:10 +00:00
|
|
|
MediaPlayerState,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-04-19 09:35:38 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_ENTITY_ID,
|
|
|
|
CONF_HOST,
|
|
|
|
CONF_NAME,
|
|
|
|
CONF_PORT,
|
|
|
|
CONF_TYPE,
|
|
|
|
)
|
2022-01-03 15:04:15 +00:00
|
|
|
from homeassistant.core import HomeAssistant, ServiceCall
|
2018-04-19 09:35:38 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2022-01-03 15:04:15 +00:00
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
2019-12-09 12:52:18 +00:00
|
|
|
|
2019-11-26 19:10:13 +00:00
|
|
|
from .const import DOMAIN, SERVICE_SETALLZONES
|
2018-04-19 09:35:38 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
MEDIA_PLAYER_SCHEMA = vol.Schema({ATTR_ENTITY_ID: cv.comp_entity_ids})
|
2019-07-24 00:54:59 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ZONE_SCHEMA = vol.Schema({vol.Required(CONF_NAME): cv.string})
|
2018-04-19 09:35:38 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SOURCE_SCHEMA = vol.Schema({vol.Required(CONF_NAME): cv.string})
|
2018-04-19 09:35:38 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_ZONES = "zones"
|
|
|
|
CONF_SOURCES = "sources"
|
2018-04-19 09:35:38 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DATA_BLACKBIRD = "blackbird"
|
2018-04-19 09:35:38 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ATTR_SOURCE = "source"
|
2018-04-19 09:35:38 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
BLACKBIRD_SETALLZONES_SCHEMA = MEDIA_PLAYER_SCHEMA.extend(
|
|
|
|
{vol.Required(ATTR_SOURCE): cv.string}
|
|
|
|
)
|
2018-04-19 09:35:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
# Valid zone ids: 1-8
|
|
|
|
ZONE_IDS = vol.All(vol.Coerce(int), vol.Range(min=1, max=8))
|
|
|
|
|
|
|
|
# Valid source ids: 1-8
|
|
|
|
SOURCE_IDS = vol.All(vol.Coerce(int), vol.Range(min=1, max=8))
|
|
|
|
|
2018-05-02 13:21:50 +00:00
|
|
|
PLATFORM_SCHEMA = vol.All(
|
|
|
|
cv.has_at_least_one_key(CONF_PORT, CONF_HOST),
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Exclusive(CONF_PORT, CONF_TYPE): cv.string,
|
|
|
|
vol.Exclusive(CONF_HOST, CONF_TYPE): cv.string,
|
|
|
|
vol.Required(CONF_ZONES): vol.Schema({ZONE_IDS: ZONE_SCHEMA}),
|
|
|
|
vol.Required(CONF_SOURCES): vol.Schema({SOURCE_IDS: SOURCE_SCHEMA}),
|
|
|
|
}
|
|
|
|
),
|
|
|
|
)
|
2018-04-19 09:35:38 +00:00
|
|
|
|
|
|
|
|
2022-01-03 15:04:15 +00:00
|
|
|
def setup_platform(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config: ConfigType,
|
|
|
|
add_entities: AddEntitiesCallback,
|
|
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
|
|
) -> None:
|
2018-04-19 09:35:38 +00:00
|
|
|
"""Set up the Monoprice Blackbird 4k 8x8 HDBaseT Matrix platform."""
|
2018-05-02 13:21:50 +00:00
|
|
|
if DATA_BLACKBIRD not in hass.data:
|
|
|
|
hass.data[DATA_BLACKBIRD] = {}
|
|
|
|
|
2018-04-19 09:35:38 +00:00
|
|
|
port = config.get(CONF_PORT)
|
|
|
|
host = config.get(CONF_HOST)
|
|
|
|
|
2018-05-02 13:21:50 +00:00
|
|
|
connection = None
|
|
|
|
if port is not None:
|
2018-04-19 09:35:38 +00:00
|
|
|
try:
|
|
|
|
blackbird = get_blackbird(port)
|
2018-05-02 13:21:50 +00:00
|
|
|
connection = port
|
2018-04-19 09:35:38 +00:00
|
|
|
except SerialException:
|
|
|
|
_LOGGER.error("Error connecting to the Blackbird controller")
|
|
|
|
return
|
|
|
|
|
2018-05-02 13:21:50 +00:00
|
|
|
if host is not None:
|
2018-04-19 09:35:38 +00:00
|
|
|
try:
|
|
|
|
blackbird = get_blackbird(host, False)
|
2018-05-02 13:21:50 +00:00
|
|
|
connection = host
|
2018-04-19 09:35:38 +00:00
|
|
|
except socket.timeout:
|
|
|
|
_LOGGER.error("Error connecting to the Blackbird controller")
|
|
|
|
return
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
sources = {
|
|
|
|
source_id: extra[CONF_NAME] for source_id, extra in config[CONF_SOURCES].items()
|
|
|
|
}
|
2018-04-19 09:35:38 +00:00
|
|
|
|
2018-05-02 13:21:50 +00:00
|
|
|
devices = []
|
2018-04-19 09:35:38 +00:00
|
|
|
for zone_id, extra in config[CONF_ZONES].items():
|
|
|
|
_LOGGER.info("Adding zone %d - %s", zone_id, extra[CONF_NAME])
|
2019-09-03 15:09:59 +00:00
|
|
|
unique_id = f"{connection}-{zone_id}"
|
2018-05-02 13:21:50 +00:00
|
|
|
device = BlackbirdZone(blackbird, sources, zone_id, extra[CONF_NAME])
|
|
|
|
hass.data[DATA_BLACKBIRD][unique_id] = device
|
|
|
|
devices.append(device)
|
2018-04-19 09:35:38 +00:00
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities(devices, True)
|
2018-04-19 09:35:38 +00:00
|
|
|
|
2021-12-30 17:50:51 +00:00
|
|
|
def service_handle(service: ServiceCall) -> None:
|
2018-04-19 09:35:38 +00:00
|
|
|
"""Handle for services."""
|
|
|
|
entity_ids = service.data.get(ATTR_ENTITY_ID)
|
|
|
|
source = service.data.get(ATTR_SOURCE)
|
|
|
|
if entity_ids:
|
2019-07-31 19:25:30 +00:00
|
|
|
devices = [
|
|
|
|
device
|
|
|
|
for device in hass.data[DATA_BLACKBIRD].values()
|
|
|
|
if device.entity_id in entity_ids
|
|
|
|
]
|
2018-04-19 09:35:38 +00:00
|
|
|
|
|
|
|
else:
|
2018-05-02 13:21:50 +00:00
|
|
|
devices = hass.data[DATA_BLACKBIRD].values()
|
2018-04-19 09:35:38 +00:00
|
|
|
|
|
|
|
for device in devices:
|
|
|
|
if service.service == SERVICE_SETALLZONES:
|
|
|
|
device.set_all_zones(source)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
hass.services.register(
|
|
|
|
DOMAIN, SERVICE_SETALLZONES, service_handle, schema=BLACKBIRD_SETALLZONES_SCHEMA
|
|
|
|
)
|
2018-04-19 09:35:38 +00:00
|
|
|
|
|
|
|
|
2020-04-25 16:00:57 +00:00
|
|
|
class BlackbirdZone(MediaPlayerEntity):
|
2018-04-19 09:35:38 +00:00
|
|
|
"""Representation of a Blackbird matrix zone."""
|
|
|
|
|
2022-04-05 21:53:45 +00:00
|
|
|
_attr_supported_features = (
|
|
|
|
MediaPlayerEntityFeature.TURN_ON
|
|
|
|
| MediaPlayerEntityFeature.TURN_OFF
|
|
|
|
| MediaPlayerEntityFeature.SELECT_SOURCE
|
|
|
|
)
|
2021-07-13 18:08:22 +00:00
|
|
|
|
2018-04-19 09:35:38 +00:00
|
|
|
def __init__(self, blackbird, sources, zone_id, zone_name):
|
|
|
|
"""Initialize new zone."""
|
|
|
|
self._blackbird = blackbird
|
|
|
|
# dict source_id -> source name
|
|
|
|
self._source_id_name = sources
|
|
|
|
# dict source name -> source_id
|
|
|
|
self._source_name_id = {v: k for k, v in sources.items()}
|
|
|
|
# ordered list of all source names
|
2021-07-13 18:08:22 +00:00
|
|
|
self._attr_source_list = sorted(
|
2019-07-31 19:25:30 +00:00
|
|
|
self._source_name_id.keys(), key=lambda v: self._source_name_id[v]
|
|
|
|
)
|
2018-04-19 09:35:38 +00:00
|
|
|
self._zone_id = zone_id
|
2021-07-13 18:08:22 +00:00
|
|
|
self._attr_name = zone_name
|
2018-04-19 09:35:38 +00:00
|
|
|
|
2022-08-19 07:54:13 +00:00
|
|
|
def update(self) -> None:
|
2018-04-19 09:35:38 +00:00
|
|
|
"""Retrieve latest state."""
|
|
|
|
state = self._blackbird.zone_status(self._zone_id)
|
|
|
|
if not state:
|
2018-05-02 13:21:50 +00:00
|
|
|
return
|
2022-09-08 09:03:10 +00:00
|
|
|
self._attr_state = MediaPlayerState.ON if state.power else MediaPlayerState.OFF
|
2018-04-19 09:35:38 +00:00
|
|
|
idx = state.av
|
|
|
|
if idx in self._source_id_name:
|
2021-07-13 18:08:22 +00:00
|
|
|
self._attr_source = self._source_id_name[idx]
|
2018-04-19 09:35:38 +00:00
|
|
|
else:
|
2021-07-13 18:08:22 +00:00
|
|
|
self._attr_source = None
|
2018-04-19 09:35:38 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def media_title(self):
|
|
|
|
"""Return the current source as media title."""
|
2021-07-13 18:08:22 +00:00
|
|
|
return self.source
|
2018-04-19 09:35:38 +00:00
|
|
|
|
|
|
|
def set_all_zones(self, source):
|
|
|
|
"""Set all zones to one source."""
|
|
|
|
if source not in self._source_name_id:
|
|
|
|
return
|
|
|
|
idx = self._source_name_id[source]
|
|
|
|
_LOGGER.debug("Setting all zones source to %s", idx)
|
|
|
|
self._blackbird.set_all_zone_source(idx)
|
|
|
|
|
2022-08-19 07:54:13 +00:00
|
|
|
def select_source(self, source: str) -> None:
|
2018-04-19 09:35:38 +00:00
|
|
|
"""Set input source."""
|
|
|
|
if source not in self._source_name_id:
|
|
|
|
return
|
|
|
|
idx = self._source_name_id[source]
|
|
|
|
_LOGGER.debug("Setting zone %d source to %s", self._zone_id, idx)
|
|
|
|
self._blackbird.set_zone_source(self._zone_id, idx)
|
|
|
|
|
2022-08-19 07:54:13 +00:00
|
|
|
def turn_on(self) -> None:
|
2018-04-19 09:35:38 +00:00
|
|
|
"""Turn the media player on."""
|
|
|
|
_LOGGER.debug("Turning zone %d on", self._zone_id)
|
|
|
|
self._blackbird.set_zone_power(self._zone_id, True)
|
|
|
|
|
2022-08-19 07:54:13 +00:00
|
|
|
def turn_off(self) -> None:
|
2018-04-19 09:35:38 +00:00
|
|
|
"""Turn the media player off."""
|
|
|
|
_LOGGER.debug("Turning zone %d off", self._zone_id)
|
|
|
|
self._blackbird.set_zone_power(self._zone_id, False)
|