2016-03-23 22:29:21 +00:00
|
|
|
"""
|
|
|
|
Support for Yamaha Receivers.
|
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/media_player.yamaha/
|
|
|
|
"""
|
|
|
|
import logging
|
|
|
|
|
2016-09-14 04:39:03 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2016-03-23 22:29:21 +00:00
|
|
|
from homeassistant.components.media_player import (
|
|
|
|
SUPPORT_TURN_OFF, SUPPORT_TURN_ON, SUPPORT_VOLUME_MUTE, SUPPORT_VOLUME_SET,
|
2016-10-19 01:04:15 +00:00
|
|
|
SUPPORT_SELECT_SOURCE, SUPPORT_PLAY_MEDIA,
|
|
|
|
MEDIA_TYPE_MUSIC,
|
|
|
|
MediaPlayerDevice, PLATFORM_SCHEMA)
|
2016-09-21 05:26:43 +00:00
|
|
|
from homeassistant.const import (CONF_NAME, CONF_HOST, STATE_OFF, STATE_ON)
|
2016-09-14 04:39:03 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-04-14 04:13:12 +00:00
|
|
|
|
2016-10-19 01:04:15 +00:00
|
|
|
REQUIREMENTS = ['rxv==0.2.0']
|
2016-04-14 04:13:12 +00:00
|
|
|
|
2016-03-23 22:29:21 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
SUPPORT_YAMAHA = SUPPORT_VOLUME_SET | SUPPORT_VOLUME_MUTE | \
|
2016-10-19 01:04:15 +00:00
|
|
|
SUPPORT_TURN_ON | SUPPORT_TURN_OFF | SUPPORT_SELECT_SOURCE | \
|
|
|
|
SUPPORT_PLAY_MEDIA
|
2016-04-28 10:03:24 +00:00
|
|
|
|
|
|
|
CONF_SOURCE_NAMES = 'source_names'
|
|
|
|
CONF_SOURCE_IGNORE = 'source_ignore'
|
2016-03-23 22:29:21 +00:00
|
|
|
|
2016-09-14 04:39:03 +00:00
|
|
|
DEFAULT_NAME = 'Yamaha Receiver'
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
2016-09-21 05:26:43 +00:00
|
|
|
vol.Optional(CONF_HOST): cv.string,
|
2016-09-14 04:39:03 +00:00
|
|
|
vol.Optional(CONF_SOURCE_IGNORE, default=[]):
|
|
|
|
vol.All(cv.ensure_list, [cv.string]),
|
|
|
|
vol.Optional(CONF_SOURCE_NAMES, default={}): {cv.string: cv.string},
|
|
|
|
})
|
|
|
|
|
2016-03-23 22:29:21 +00:00
|
|
|
|
|
|
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
|
|
|
"""Setup the Yamaha platform."""
|
|
|
|
import rxv
|
2016-04-28 10:03:24 +00:00
|
|
|
|
2016-09-14 04:39:03 +00:00
|
|
|
name = config.get(CONF_NAME)
|
2016-09-21 05:26:43 +00:00
|
|
|
host = config.get(CONF_HOST)
|
2016-09-14 04:39:03 +00:00
|
|
|
source_ignore = config.get(CONF_SOURCE_IGNORE)
|
|
|
|
source_names = config.get(CONF_SOURCE_NAMES)
|
2016-04-28 10:03:24 +00:00
|
|
|
|
2016-10-27 06:46:44 +00:00
|
|
|
if discovery_info is not None:
|
|
|
|
name = discovery_info[0]
|
|
|
|
model = discovery_info[1]
|
|
|
|
ctrl_url = discovery_info[2]
|
|
|
|
desc_url = discovery_info[3]
|
|
|
|
receivers = rxv.RXV(
|
|
|
|
ctrl_url,
|
|
|
|
model_name=model,
|
|
|
|
friendly_name=name,
|
|
|
|
unit_desc_url=desc_url).zone_controllers()
|
|
|
|
_LOGGER.info("Receivers: %s", receivers)
|
|
|
|
elif host is None:
|
2016-10-19 01:04:15 +00:00
|
|
|
receivers = []
|
|
|
|
for recv in rxv.find():
|
|
|
|
receivers.extend(recv.zone_controllers())
|
2016-09-21 05:26:43 +00:00
|
|
|
else:
|
2016-10-19 01:04:15 +00:00
|
|
|
ctrl_url = "http://{}:80/YamahaRemoteControl/ctrl".format(host)
|
|
|
|
receivers = rxv.RXV(ctrl_url, name).zone_controllers()
|
2016-09-21 05:26:43 +00:00
|
|
|
|
2016-04-28 10:03:24 +00:00
|
|
|
add_devices(
|
2016-09-14 04:39:03 +00:00
|
|
|
YamahaDevice(name, receiver, source_ignore, source_names)
|
2016-09-21 05:26:43 +00:00
|
|
|
for receiver in receivers)
|
2016-03-23 22:29:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
class YamahaDevice(MediaPlayerDevice):
|
|
|
|
"""Representation of a Yamaha device."""
|
|
|
|
|
|
|
|
# pylint: disable=too-many-public-methods, abstract-method
|
2016-04-28 10:03:24 +00:00
|
|
|
# pylint: disable=too-many-instance-attributes
|
|
|
|
def __init__(self, name, receiver, source_ignore, source_names):
|
2016-03-23 22:29:21 +00:00
|
|
|
"""Initialize the Yamaha Receiver."""
|
|
|
|
self._receiver = receiver
|
|
|
|
self._muted = False
|
|
|
|
self._volume = 0
|
|
|
|
self._pwstate = STATE_OFF
|
2016-04-14 04:13:12 +00:00
|
|
|
self._current_source = None
|
|
|
|
self._source_list = None
|
2016-10-27 06:46:44 +00:00
|
|
|
self._source_ignore = source_ignore or []
|
|
|
|
self._source_names = source_names or {}
|
2016-04-28 10:03:24 +00:00
|
|
|
self._reverse_mapping = None
|
2016-03-23 22:29:21 +00:00
|
|
|
self.update()
|
|
|
|
self._name = name
|
2016-10-19 01:04:15 +00:00
|
|
|
self._zone = receiver.zone
|
2016-03-23 22:29:21 +00:00
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Get the latest details from the device."""
|
|
|
|
if self._receiver.on:
|
|
|
|
self._pwstate = STATE_ON
|
|
|
|
else:
|
|
|
|
self._pwstate = STATE_OFF
|
|
|
|
self._muted = self._receiver.mute
|
2016-04-28 10:03:24 +00:00
|
|
|
self._volume = (self._receiver.volume / 100) + 1
|
|
|
|
|
|
|
|
if self.source_list is None:
|
|
|
|
self.build_source_list()
|
|
|
|
|
|
|
|
current_source = self._receiver.input
|
2016-09-14 04:39:03 +00:00
|
|
|
self._current_source = self._source_names.get(
|
|
|
|
current_source, current_source)
|
2016-04-28 10:03:24 +00:00
|
|
|
|
|
|
|
def build_source_list(self):
|
|
|
|
"""Build the source list."""
|
|
|
|
self._reverse_mapping = {alias: source for source, alias in
|
|
|
|
self._source_names.items()}
|
|
|
|
|
|
|
|
self._source_list = sorted(
|
|
|
|
self._source_names.get(source, source) for source in
|
|
|
|
self._receiver.inputs()
|
|
|
|
if source not in self._source_ignore)
|
2016-03-23 22:29:21 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the device."""
|
2016-10-19 01:04:15 +00:00
|
|
|
name = self._name
|
|
|
|
if self._zone != "Main_Zone":
|
|
|
|
# Zone will be one of Main_Zone, Zone_2, Zone_3
|
|
|
|
name += " " + self._zone.replace('_', ' ')
|
|
|
|
return name
|
2016-03-23 22:29:21 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the device."""
|
|
|
|
return self._pwstate
|
|
|
|
|
|
|
|
@property
|
|
|
|
def volume_level(self):
|
|
|
|
"""Volume level of the media player (0..1)."""
|
|
|
|
return self._volume
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_volume_muted(self):
|
|
|
|
"""Boolean if volume is currently muted."""
|
|
|
|
return self._muted
|
|
|
|
|
2016-04-14 04:13:12 +00:00
|
|
|
@property
|
|
|
|
def source(self):
|
|
|
|
"""Return the current input source."""
|
|
|
|
return self._current_source
|
|
|
|
|
|
|
|
@property
|
|
|
|
def source_list(self):
|
|
|
|
"""List of available input sources."""
|
|
|
|
return self._source_list
|
|
|
|
|
2016-03-23 22:29:21 +00:00
|
|
|
@property
|
|
|
|
def supported_media_commands(self):
|
|
|
|
"""Flag of media commands that are supported."""
|
|
|
|
return SUPPORT_YAMAHA
|
|
|
|
|
|
|
|
def turn_off(self):
|
|
|
|
"""Turn off media player."""
|
|
|
|
self._receiver.on = False
|
|
|
|
|
|
|
|
def set_volume_level(self, volume):
|
|
|
|
"""Set volume level, range 0..1."""
|
2016-04-28 10:03:24 +00:00
|
|
|
receiver_vol = 100 - (volume * 100)
|
2016-03-23 22:29:21 +00:00
|
|
|
negative_receiver_vol = -receiver_vol
|
|
|
|
self._receiver.volume = negative_receiver_vol
|
|
|
|
|
|
|
|
def mute_volume(self, mute):
|
|
|
|
"""Mute (true) or unmute (false) media player."""
|
|
|
|
self._receiver.mute = mute
|
|
|
|
|
|
|
|
def turn_on(self):
|
|
|
|
"""Turn the media player on."""
|
|
|
|
self._receiver.on = True
|
2016-04-28 10:03:24 +00:00
|
|
|
self._volume = (self._receiver.volume / 100) + 1
|
2016-04-14 04:13:12 +00:00
|
|
|
|
|
|
|
def select_source(self, source):
|
|
|
|
"""Select input source."""
|
2016-09-14 04:39:03 +00:00
|
|
|
self._receiver.input = self._reverse_mapping.get(source, source)
|
2016-10-19 01:04:15 +00:00
|
|
|
|
|
|
|
def play_media(self, media_type, media_id, **kwargs):
|
|
|
|
"""Play media from an ID.
|
|
|
|
|
|
|
|
This exposes a pass through for various input sources in the
|
|
|
|
Yamaha to direct play certain kinds of media. media_type is
|
|
|
|
treated as the input type that we are setting, and media id is
|
|
|
|
specific to it.
|
|
|
|
"""
|
|
|
|
if media_type == "NET RADIO":
|
|
|
|
self._receiver.net_radio(media_id)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_content_type(self):
|
|
|
|
"""Return the media content type."""
|
|
|
|
if self.source == "NET RADIO":
|
|
|
|
return MEDIA_TYPE_MUSIC
|
|
|
|
|
|
|
|
@property
|
|
|
|
def media_title(self):
|
|
|
|
"""Return the media title.
|
|
|
|
|
|
|
|
This will vary by input source, as they provide different
|
|
|
|
information in metadata.
|
|
|
|
|
|
|
|
"""
|
|
|
|
if self.source == "NET RADIO":
|
|
|
|
info = self._receiver.play_status()
|
|
|
|
if info.song:
|
|
|
|
return "%s: %s" % (info.station, info.song)
|
|
|
|
else:
|
|
|
|
return info.station
|