Add nad telnet (#19704)
* fix cla-bot * fix bug introduced after linter complaint * merge two components into one support telnet port configuration * remove obsolete nadtcp component. nad component must be used instead. * back to correct nad_receiver versionpull/19979/head
parent
96d20a64d5
commit
47f64b472d
|
@ -8,53 +8,87 @@ import logging
|
|||
|
||||
import voluptuous as vol
|
||||
|
||||
import homeassistant.helpers.config_validation as cv
|
||||
from homeassistant.components.media_player import (
|
||||
PLATFORM_SCHEMA, SUPPORT_SELECT_SOURCE, SUPPORT_TURN_OFF, SUPPORT_TURN_ON,
|
||||
SUPPORT_VOLUME_MUTE, SUPPORT_VOLUME_SET, SUPPORT_VOLUME_STEP,
|
||||
MediaPlayerDevice)
|
||||
from homeassistant.const import CONF_NAME, STATE_OFF, STATE_ON
|
||||
import homeassistant.helpers.config_validation as cv
|
||||
|
||||
REQUIREMENTS = ['nad_receiver==0.0.9']
|
||||
REQUIREMENTS = ['nad_receiver==0.0.11']
|
||||
|
||||
_LOGGER = logging.getLogger(__name__)
|
||||
|
||||
DEFAULT_TYPE = 'RS232'
|
||||
DEFAULT_SERIAL_PORT = '/dev/ttyUSB0'
|
||||
DEFAULT_PORT = 53
|
||||
DEFAULT_NAME = 'NAD Receiver'
|
||||
DEFAULT_MIN_VOLUME = -92
|
||||
DEFAULT_MAX_VOLUME = -20
|
||||
DEFAULT_VOLUME_STEP = 4
|
||||
|
||||
SUPPORT_NAD = SUPPORT_VOLUME_SET | SUPPORT_VOLUME_MUTE | \
|
||||
SUPPORT_TURN_ON | SUPPORT_TURN_OFF | SUPPORT_VOLUME_STEP | \
|
||||
SUPPORT_SELECT_SOURCE
|
||||
SUPPORT_TURN_ON | SUPPORT_TURN_OFF | SUPPORT_VOLUME_STEP | \
|
||||
SUPPORT_SELECT_SOURCE
|
||||
|
||||
CONF_SERIAL_PORT = 'serial_port'
|
||||
CONF_TYPE = 'type'
|
||||
CONF_SERIAL_PORT = 'serial_port' # for NADReceiver
|
||||
CONF_HOST = 'host' # for NADReceiverTelnet
|
||||
CONF_PORT = 'port' # for NADReceiverTelnet
|
||||
CONF_MIN_VOLUME = 'min_volume'
|
||||
CONF_MAX_VOLUME = 'max_volume'
|
||||
CONF_SOURCE_DICT = 'sources'
|
||||
CONF_VOLUME_STEP = 'volume_step' # for NADReceiverTCP
|
||||
CONF_SOURCE_DICT = 'sources' # for NADReceiver
|
||||
|
||||
SOURCE_DICT_SCHEMA = vol.Schema({
|
||||
vol.Range(min=1, max=10): cv.string
|
||||
})
|
||||
|
||||
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
||||
vol.Required(CONF_SERIAL_PORT): cv.string,
|
||||
vol.Optional(CONF_TYPE, default=DEFAULT_TYPE):
|
||||
vol.In(['RS232', 'Telnet', 'TCP']),
|
||||
vol.Optional(CONF_SERIAL_PORT, default=DEFAULT_SERIAL_PORT):
|
||||
cv.string,
|
||||
vol.Optional(CONF_HOST): cv.string,
|
||||
vol.Optional(CONF_PORT, default=DEFAULT_PORT): int,
|
||||
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
||||
vol.Optional(CONF_MIN_VOLUME, default=DEFAULT_MIN_VOLUME): int,
|
||||
vol.Optional(CONF_MAX_VOLUME, default=DEFAULT_MAX_VOLUME): int,
|
||||
vol.Optional(CONF_SOURCE_DICT, default={}): SOURCE_DICT_SCHEMA,
|
||||
vol.Optional(CONF_VOLUME_STEP, default=DEFAULT_VOLUME_STEP): int,
|
||||
})
|
||||
|
||||
|
||||
def setup_platform(hass, config, add_entities, discovery_info=None):
|
||||
"""Set up the NAD platform."""
|
||||
from nad_receiver import NADReceiver
|
||||
add_entities([NAD(
|
||||
config.get(CONF_NAME),
|
||||
NADReceiver(config.get(CONF_SERIAL_PORT)),
|
||||
config.get(CONF_MIN_VOLUME),
|
||||
config.get(CONF_MAX_VOLUME),
|
||||
config.get(CONF_SOURCE_DICT)
|
||||
)], True)
|
||||
if config.get(CONF_TYPE) == 'RS232':
|
||||
from nad_receiver import NADReceiver
|
||||
add_entities([NAD(
|
||||
config.get(CONF_NAME),
|
||||
NADReceiver(config.get(CONF_SERIAL_PORT)),
|
||||
config.get(CONF_MIN_VOLUME),
|
||||
config.get(CONF_MAX_VOLUME),
|
||||
config.get(CONF_SOURCE_DICT)
|
||||
)], True)
|
||||
elif config.get(CONF_TYPE) == 'Telnet':
|
||||
from nad_receiver import NADReceiverTelnet
|
||||
add_entities([NAD(
|
||||
config.get(CONF_NAME),
|
||||
NADReceiverTelnet(config.get(CONF_HOST),
|
||||
config.get(CONF_PORT)),
|
||||
config.get(CONF_MIN_VOLUME),
|
||||
config.get(CONF_MAX_VOLUME),
|
||||
config.get(CONF_SOURCE_DICT)
|
||||
)], True)
|
||||
else:
|
||||
from nad_receiver import NADReceiverTCP
|
||||
add_entities([NADtcp(
|
||||
config.get(CONF_NAME),
|
||||
NADReceiverTCP(config.get(CONF_HOST)),
|
||||
config.get(CONF_MIN_VOLUME),
|
||||
config.get(CONF_MAX_VOLUME),
|
||||
config.get(CONF_VOLUME_STEP),
|
||||
)], True)
|
||||
|
||||
|
||||
class NAD(MediaPlayerDevice):
|
||||
|
@ -73,24 +107,6 @@ class NAD(MediaPlayerDevice):
|
|||
|
||||
self._volume = self._state = self._mute = self._source = None
|
||||
|
||||
def calc_volume(self, decibel):
|
||||
"""
|
||||
Calculate the volume given the decibel.
|
||||
|
||||
Return the volume (0..1).
|
||||
"""
|
||||
return abs(self._min_volume - decibel) / abs(
|
||||
self._min_volume - self._max_volume)
|
||||
|
||||
def calc_db(self, volume):
|
||||
"""
|
||||
Calculate the decibel given the volume.
|
||||
|
||||
Return the dB.
|
||||
"""
|
||||
return self._min_volume + round(
|
||||
abs(self._min_volume - self._max_volume) * volume)
|
||||
|
||||
@property
|
||||
def name(self):
|
||||
"""Return the name of the device."""
|
||||
|
@ -101,22 +117,6 @@ class NAD(MediaPlayerDevice):
|
|||
"""Return the state of the device."""
|
||||
return self._state
|
||||
|
||||
def update(self):
|
||||
"""Retrieve latest state."""
|
||||
if self._nad_receiver.main_power('?') == 'Off':
|
||||
self._state = STATE_OFF
|
||||
else:
|
||||
self._state = STATE_ON
|
||||
|
||||
if self._nad_receiver.main_mute('?') == 'Off':
|
||||
self._mute = False
|
||||
else:
|
||||
self._mute = True
|
||||
|
||||
self._volume = self.calc_volume(self._nad_receiver.main_volume('?'))
|
||||
self._source = self._source_dict.get(
|
||||
self._nad_receiver.main_source('?'))
|
||||
|
||||
@property
|
||||
def volume_level(self):
|
||||
"""Volume level of the media player (0..1)."""
|
||||
|
@ -152,6 +152,13 @@ class NAD(MediaPlayerDevice):
|
|||
"""Set volume level, range 0..1."""
|
||||
self._nad_receiver.main_volume('=', self.calc_db(volume))
|
||||
|
||||
def mute_volume(self, mute):
|
||||
"""Mute (true) or unmute (false) media player."""
|
||||
if mute:
|
||||
self._nad_receiver.main_mute('=', 'On')
|
||||
else:
|
||||
self._nad_receiver.main_mute('=', 'Off')
|
||||
|
||||
def select_source(self, source):
|
||||
"""Select input source."""
|
||||
self._nad_receiver.main_source('=', self._reverse_mapping.get(source))
|
||||
|
@ -166,9 +173,162 @@ class NAD(MediaPlayerDevice):
|
|||
"""List of available input sources."""
|
||||
return sorted(list(self._reverse_mapping.keys()))
|
||||
|
||||
def update(self):
|
||||
"""Retrieve latest state."""
|
||||
if self._nad_receiver.main_power('?') == 'Off':
|
||||
self._state = STATE_OFF
|
||||
else:
|
||||
self._state = STATE_ON
|
||||
|
||||
if self._nad_receiver.main_mute('?') == 'Off':
|
||||
self._mute = False
|
||||
else:
|
||||
self._mute = True
|
||||
|
||||
self._volume = self.calc_volume(self._nad_receiver.main_volume('?'))
|
||||
self._source = self._source_dict.get(
|
||||
self._nad_receiver.main_source('?'))
|
||||
|
||||
def calc_volume(self, decibel):
|
||||
"""
|
||||
Calculate the volume given the decibel.
|
||||
|
||||
Return the volume (0..1).
|
||||
"""
|
||||
return abs(self._min_volume - decibel) / abs(
|
||||
self._min_volume - self._max_volume)
|
||||
|
||||
def calc_db(self, volume):
|
||||
"""
|
||||
Calculate the decibel given the volume.
|
||||
|
||||
Return the dB.
|
||||
"""
|
||||
return self._min_volume + round(
|
||||
abs(self._min_volume - self._max_volume) * volume)
|
||||
|
||||
|
||||
class NADtcp(MediaPlayerDevice):
|
||||
"""Representation of a NAD Digital amplifier."""
|
||||
|
||||
def __init__(self, name, nad_device, min_volume, max_volume, volume_step):
|
||||
"""Initialize the amplifier."""
|
||||
self._name = name
|
||||
self._nad_receiver = nad_device
|
||||
self._min_vol = (min_volume + 90) * 2 # from dB to nad vol (0-200)
|
||||
self._max_vol = (max_volume + 90) * 2 # from dB to nad vol (0-200)
|
||||
self._volume_step = volume_step
|
||||
self._state = None
|
||||
self._mute = None
|
||||
self._nad_volume = None
|
||||
self._volume = None
|
||||
self._source = None
|
||||
self._source_list = self._nad_receiver.available_sources()
|
||||
|
||||
@property
|
||||
def name(self):
|
||||
"""Return the name of the device."""
|
||||
return self._name
|
||||
|
||||
@property
|
||||
def state(self):
|
||||
"""Return the state of the device."""
|
||||
return self._state
|
||||
|
||||
@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._mute
|
||||
|
||||
@property
|
||||
def supported_features(self):
|
||||
"""Flag media player features that are supported."""
|
||||
return SUPPORT_NAD
|
||||
|
||||
def turn_off(self):
|
||||
"""Turn the media player off."""
|
||||
self._nad_receiver.power_off()
|
||||
|
||||
def turn_on(self):
|
||||
"""Turn the media player on."""
|
||||
self._nad_receiver.power_on()
|
||||
|
||||
def volume_up(self):
|
||||
"""Step volume up in the configured increments."""
|
||||
self._nad_receiver.set_volume(self._nad_volume + 2 * self._volume_step)
|
||||
|
||||
def volume_down(self):
|
||||
"""Step volume down in the configured increments."""
|
||||
self._nad_receiver.set_volume(self._nad_volume - 2 * self._volume_step)
|
||||
|
||||
def set_volume_level(self, volume):
|
||||
"""Set volume level, range 0..1."""
|
||||
nad_volume_to_set = \
|
||||
int(round(volume * (self._max_vol - self._min_vol) +
|
||||
self._min_vol))
|
||||
self._nad_receiver.set_volume(nad_volume_to_set)
|
||||
|
||||
def mute_volume(self, mute):
|
||||
"""Mute (true) or unmute (false) media player."""
|
||||
if mute:
|
||||
self._nad_receiver.main_mute('=', 'On')
|
||||
self._nad_receiver.mute()
|
||||
else:
|
||||
self._nad_receiver.main_mute('=', 'Off')
|
||||
self._nad_receiver.unmute()
|
||||
|
||||
def select_source(self, source):
|
||||
"""Select input source."""
|
||||
self._nad_receiver.select_source(source)
|
||||
|
||||
@property
|
||||
def source(self):
|
||||
"""Name of the current input source."""
|
||||
return self._source
|
||||
|
||||
@property
|
||||
def source_list(self):
|
||||
"""List of available input sources."""
|
||||
return self._nad_receiver.available_sources()
|
||||
|
||||
def update(self):
|
||||
"""Get the latest details from the device."""
|
||||
try:
|
||||
nad_status = self._nad_receiver.status()
|
||||
except OSError:
|
||||
return
|
||||
if nad_status is None:
|
||||
return
|
||||
|
||||
# Update on/off state
|
||||
if nad_status['power']:
|
||||
self._state = STATE_ON
|
||||
else:
|
||||
self._state = STATE_OFF
|
||||
|
||||
# Update current volume
|
||||
self._volume = self.nad_vol_to_internal_vol(nad_status['volume'])
|
||||
self._nad_volume = nad_status['volume']
|
||||
|
||||
# Update muted state
|
||||
self._mute = nad_status['muted']
|
||||
|
||||
# Update current source
|
||||
self._source = nad_status['source']
|
||||
|
||||
def nad_vol_to_internal_vol(self, nad_volume):
|
||||
"""Convert nad volume range (0-200) to internal volume range.
|
||||
|
||||
Takes into account configured min and max volume.
|
||||
"""
|
||||
if nad_volume < self._min_vol:
|
||||
volume_internal = 0.0
|
||||
elif nad_volume > self._max_vol:
|
||||
volume_internal = 1.0
|
||||
else:
|
||||
volume_internal = (nad_volume - self._min_vol) / \
|
||||
(self._max_vol - self._min_vol)
|
||||
return volume_internal
|
||||
|
|
|
@ -1,178 +0,0 @@
|
|||
"""
|
||||
Support for NAD digital amplifiers which can be remote controlled via tcp/ip.
|
||||
|
||||
For more details about this platform, please refer to the documentation at
|
||||
https://home-assistant.io/components/media_player.nadtcp/
|
||||
"""
|
||||
import logging
|
||||
|
||||
import voluptuous as vol
|
||||
|
||||
from homeassistant.components.media_player import (
|
||||
PLATFORM_SCHEMA, SUPPORT_SELECT_SOURCE, SUPPORT_TURN_OFF, SUPPORT_TURN_ON,
|
||||
SUPPORT_VOLUME_MUTE, SUPPORT_VOLUME_SET, SUPPORT_VOLUME_STEP,
|
||||
MediaPlayerDevice)
|
||||
from homeassistant.const import CONF_HOST, CONF_NAME, STATE_OFF, STATE_ON
|
||||
import homeassistant.helpers.config_validation as cv
|
||||
|
||||
REQUIREMENTS = ['nad_receiver==0.0.9']
|
||||
|
||||
_LOGGER = logging.getLogger(__name__)
|
||||
|
||||
DEFAULT_NAME = 'NAD amplifier'
|
||||
DEFAULT_MIN_VOLUME = -60
|
||||
DEFAULT_MAX_VOLUME = -10
|
||||
DEFAULT_VOLUME_STEP = 4
|
||||
|
||||
SUPPORT_NAD = SUPPORT_VOLUME_SET | SUPPORT_VOLUME_MUTE | SUPPORT_TURN_ON | \
|
||||
SUPPORT_TURN_OFF | SUPPORT_VOLUME_STEP | SUPPORT_SELECT_SOURCE
|
||||
|
||||
CONF_MIN_VOLUME = 'min_volume'
|
||||
CONF_MAX_VOLUME = 'max_volume'
|
||||
CONF_VOLUME_STEP = 'volume_step'
|
||||
|
||||
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
||||
vol.Required(CONF_HOST): cv.string,
|
||||
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
||||
vol.Optional(CONF_MIN_VOLUME, default=DEFAULT_MIN_VOLUME): int,
|
||||
vol.Optional(CONF_MAX_VOLUME, default=DEFAULT_MAX_VOLUME): int,
|
||||
vol.Optional(CONF_VOLUME_STEP, default=DEFAULT_VOLUME_STEP): int,
|
||||
})
|
||||
|
||||
|
||||
def setup_platform(hass, config, add_entities, discovery_info=None):
|
||||
"""Set up the NAD platform."""
|
||||
from nad_receiver import NADReceiverTCP
|
||||
add_entities([NADtcp(
|
||||
NADReceiverTCP(config.get(CONF_HOST)),
|
||||
config.get(CONF_NAME),
|
||||
config.get(CONF_MIN_VOLUME),
|
||||
config.get(CONF_MAX_VOLUME),
|
||||
config.get(CONF_VOLUME_STEP),
|
||||
)], True)
|
||||
|
||||
|
||||
class NADtcp(MediaPlayerDevice):
|
||||
"""Representation of a NAD Digital amplifier."""
|
||||
|
||||
def __init__(self, nad_device, name, min_volume, max_volume, volume_step):
|
||||
"""Initialize the amplifier."""
|
||||
self._name = name
|
||||
self.nad_device = nad_device
|
||||
self._min_vol = (min_volume + 90) * 2 # from dB to nad vol (0-200)
|
||||
self._max_vol = (max_volume + 90) * 2 # from dB to nad vol (0-200)
|
||||
self._volume_step = volume_step
|
||||
self._state = None
|
||||
self._mute = None
|
||||
self._nad_volume = None
|
||||
self._volume = None
|
||||
self._source = None
|
||||
self._source_list = self.nad_device.available_sources()
|
||||
|
||||
@property
|
||||
def name(self):
|
||||
"""Return the name of the device."""
|
||||
return self._name
|
||||
|
||||
@property
|
||||
def state(self):
|
||||
"""Return the state of the device."""
|
||||
return self._state
|
||||
|
||||
def update(self):
|
||||
"""Get the latest details from the device."""
|
||||
try:
|
||||
nad_status = self.nad_device.status()
|
||||
except OSError:
|
||||
return
|
||||
if nad_status is None:
|
||||
return
|
||||
|
||||
# Update on/off state
|
||||
if nad_status['power']:
|
||||
self._state = STATE_ON
|
||||
else:
|
||||
self._state = STATE_OFF
|
||||
|
||||
# Update current volume
|
||||
self._volume = self.nad_vol_to_internal_vol(nad_status['volume'])
|
||||
self._nad_volume = nad_status['volume']
|
||||
|
||||
# Update muted state
|
||||
self._mute = nad_status['muted']
|
||||
|
||||
# Update current source
|
||||
self._source = nad_status['source']
|
||||
|
||||
def nad_vol_to_internal_vol(self, nad_volume):
|
||||
"""Convert nad volume range (0-200) to internal volume range.
|
||||
|
||||
Takes into account configured min and max volume.
|
||||
"""
|
||||
if nad_volume < self._min_vol:
|
||||
volume_internal = 0.0
|
||||
if nad_volume > self._max_vol:
|
||||
volume_internal = 1.0
|
||||
else:
|
||||
volume_internal = (nad_volume - self._min_vol) / \
|
||||
(self._max_vol - self._min_vol)
|
||||
return volume_internal
|
||||
|
||||
@property
|
||||
def supported_features(self):
|
||||
"""Flag media player features that are supported."""
|
||||
return SUPPORT_NAD
|
||||
|
||||
def turn_off(self):
|
||||
"""Turn the media player off."""
|
||||
self.nad_device.power_off()
|
||||
|
||||
def turn_on(self):
|
||||
"""Turn the media player on."""
|
||||
self.nad_device.power_on()
|
||||
|
||||
def volume_up(self):
|
||||
"""Step volume up in the configured increments."""
|
||||
self.nad_device.set_volume(self._nad_volume + 2 * self._volume_step)
|
||||
|
||||
def volume_down(self):
|
||||
"""Step volume down in the configured increments."""
|
||||
self.nad_device.set_volume(self._nad_volume - 2 * self._volume_step)
|
||||
|
||||
def set_volume_level(self, volume):
|
||||
"""Set volume level, range 0..1."""
|
||||
nad_volume_to_set = \
|
||||
int(round(volume * (self._max_vol - self._min_vol) +
|
||||
self._min_vol))
|
||||
self.nad_device.set_volume(nad_volume_to_set)
|
||||
|
||||
def mute_volume(self, mute):
|
||||
"""Mute (true) or unmute (false) media player."""
|
||||
if mute:
|
||||
self.nad_device.mute()
|
||||
else:
|
||||
self.nad_device.unmute()
|
||||
|
||||
def select_source(self, source):
|
||||
"""Select input source."""
|
||||
self.nad_device.select_source(source)
|
||||
|
||||
@property
|
||||
def source(self):
|
||||
"""Name of the current input source."""
|
||||
return self._source
|
||||
|
||||
@property
|
||||
def source_list(self):
|
||||
"""List of available input sources."""
|
||||
return self.nad_device.available_sources()
|
||||
|
||||
@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._mute
|
|
@ -689,8 +689,7 @@ mycroftapi==2.0
|
|||
myusps==1.3.2
|
||||
|
||||
# homeassistant.components.media_player.nad
|
||||
# homeassistant.components.media_player.nadtcp
|
||||
nad_receiver==0.0.9
|
||||
nad_receiver==0.0.11
|
||||
|
||||
# homeassistant.components.light.nanoleaf_aurora
|
||||
nanoleaf==0.4.1
|
||||
|
|
Loading…
Reference in New Issue