Home Assistant Unofficial Reference 2024.12.1
media_player.py
Go to the documentation of this file.
1 """Support for interfacing with NAD receivers through RS-232."""
2 
3 from __future__ import annotations
4 
5 from nad_receiver import NADReceiver, NADReceiverTCP, NADReceiverTelnet
6 import voluptuous as vol
7 
9  PLATFORM_SCHEMA as MEDIA_PLAYER_PLATFORM_SCHEMA,
10  MediaPlayerEntity,
11  MediaPlayerEntityFeature,
12  MediaPlayerState,
13 )
14 from homeassistant.const import CONF_HOST, CONF_NAME, CONF_PORT, CONF_TYPE
15 from homeassistant.core import HomeAssistant
17 from homeassistant.helpers.entity_platform import AddEntitiesCallback
18 from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
19 
20 DEFAULT_TYPE = "RS232"
21 DEFAULT_SERIAL_PORT = "/dev/ttyUSB0"
22 DEFAULT_PORT = 53
23 DEFAULT_NAME = "NAD Receiver"
24 DEFAULT_MIN_VOLUME = -92
25 DEFAULT_MAX_VOLUME = -20
26 DEFAULT_VOLUME_STEP = 4
27 
28 SUPPORT_NAD = (
29  MediaPlayerEntityFeature.VOLUME_SET
30  | MediaPlayerEntityFeature.VOLUME_MUTE
31  | MediaPlayerEntityFeature.TURN_ON
32  | MediaPlayerEntityFeature.TURN_OFF
33  | MediaPlayerEntityFeature.VOLUME_STEP
34  | MediaPlayerEntityFeature.SELECT_SOURCE
35 )
36 
37 CONF_SERIAL_PORT = "serial_port" # for NADReceiver
38 CONF_MIN_VOLUME = "min_volume"
39 CONF_MAX_VOLUME = "max_volume"
40 CONF_VOLUME_STEP = "volume_step" # for NADReceiverTCP
41 CONF_SOURCE_DICT = "sources" # for NADReceiver
42 
43 # Max value based on a C658 with an MDC HDM-2 card installed
44 SOURCE_DICT_SCHEMA = vol.Schema({vol.Range(min=1, max=12): cv.string})
45 
46 PLATFORM_SCHEMA = MEDIA_PLAYER_PLATFORM_SCHEMA.extend(
47  {
48  vol.Optional(CONF_TYPE, default=DEFAULT_TYPE): vol.In(
49  ["RS232", "Telnet", "TCP"]
50  ),
51  vol.Optional(CONF_SERIAL_PORT, default=DEFAULT_SERIAL_PORT): cv.string,
52  vol.Optional(CONF_HOST): cv.string,
53  vol.Optional(CONF_PORT, default=DEFAULT_PORT): int,
54  vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
55  vol.Optional(CONF_MIN_VOLUME, default=DEFAULT_MIN_VOLUME): int,
56  vol.Optional(CONF_MAX_VOLUME, default=DEFAULT_MAX_VOLUME): int,
57  vol.Optional(CONF_SOURCE_DICT, default={}): SOURCE_DICT_SCHEMA,
58  vol.Optional(CONF_VOLUME_STEP, default=DEFAULT_VOLUME_STEP): int,
59  }
60 )
61 
62 
64  hass: HomeAssistant,
65  config: ConfigType,
66  add_entities: AddEntitiesCallback,
67  discovery_info: DiscoveryInfoType | None = None,
68 ) -> None:
69  """Set up the NAD platform."""
70  if config.get(CONF_TYPE) in ("RS232", "Telnet"):
72  [NAD(config)],
73  True,
74  )
75  else:
77  [NADtcp(config)],
78  True,
79  )
80 
81 
83  """Representation of a NAD Receiver."""
84 
85  _attr_icon = "mdi:speaker-multiple"
86  _attr_supported_features = SUPPORT_NAD
87 
88  def __init__(self, config):
89  """Initialize the NAD Receiver device."""
90  self.configconfig = config
91  self._instantiate_nad_receiver_instantiate_nad_receiver()
92  self._attr_name_attr_name = self.configconfig[CONF_NAME]
93  self._min_volume_min_volume = config[CONF_MIN_VOLUME]
94  self._max_volume_max_volume = config[CONF_MAX_VOLUME]
95  self._source_dict_source_dict = config[CONF_SOURCE_DICT]
96  self._reverse_mapping_reverse_mapping = {value: key for key, value in self._source_dict_source_dict.items()}
97 
98  def _instantiate_nad_receiver(self) -> NADReceiver:
99  if self.configconfig[CONF_TYPE] == "RS232":
100  self._nad_receiver_nad_receiver = NADReceiver(self.configconfig[CONF_SERIAL_PORT])
101  else:
102  host = self.configconfig.get(CONF_HOST)
103  port = self.configconfig[CONF_PORT]
104  self._nad_receiver_nad_receiver = NADReceiverTelnet(host, port)
105 
106  def turn_off(self) -> None:
107  """Turn the media player off."""
108  self._nad_receiver_nad_receiver.main_power("=", "Off")
109 
110  def turn_on(self) -> None:
111  """Turn the media player on."""
112  self._nad_receiver_nad_receiver.main_power("=", "On")
113 
114  def volume_up(self) -> None:
115  """Volume up the media player."""
116  self._nad_receiver_nad_receiver.main_volume("+")
117 
118  def volume_down(self) -> None:
119  """Volume down the media player."""
120  self._nad_receiver_nad_receiver.main_volume("-")
121 
122  def set_volume_level(self, volume: float) -> None:
123  """Set volume level, range 0..1."""
124  self._nad_receiver_nad_receiver.main_volume("=", self.calc_dbcalc_db(volume))
125 
126  def mute_volume(self, mute: bool) -> None:
127  """Mute (true) or unmute (false) media player."""
128  if mute:
129  self._nad_receiver_nad_receiver.main_mute("=", "On")
130  else:
131  self._nad_receiver_nad_receiver.main_mute("=", "Off")
132 
133  def select_source(self, source: str) -> None:
134  """Select input source."""
135  self._nad_receiver_nad_receiver.main_source("=", self._reverse_mapping_reverse_mapping.get(source))
136 
137  @property
138  def source_list(self):
139  """List of available input sources."""
140  return sorted(self._reverse_mapping_reverse_mapping)
141 
142  @property
143  def available(self) -> bool:
144  """Return if device is available."""
145  return self.statestatestatestatestate is not None
146 
147  def update(self) -> None:
148  """Retrieve latest state."""
149  power_state = self._nad_receiver_nad_receiver.main_power("?")
150  if not power_state:
151  self._attr_state_attr_state = None
152  return
153  self._attr_state_attr_state = (
154  MediaPlayerState.ON
155  if self._nad_receiver_nad_receiver.main_power("?") == "On"
156  else MediaPlayerState.OFF
157  )
158 
159  if self.statestatestatestatestate == MediaPlayerState.ON:
160  self._attr_is_volume_muted_attr_is_volume_muted = self._nad_receiver_nad_receiver.main_mute("?") == "On"
161  volume = self._nad_receiver_nad_receiver.main_volume("?")
162  # Some receivers cannot report the volume, e.g. C 356BEE,
163  # instead they only support stepping the volume up or down
164  self._attr_volume_level_attr_volume_level = (
165  self.calc_volumecalc_volume(volume) if volume is not None else None
166  )
167  self._attr_source_attr_source = self._source_dict_source_dict.get(
168  self._nad_receiver_nad_receiver.main_source("?")
169  )
170 
171  def calc_volume(self, decibel):
172  """Calculate the volume given the decibel.
173 
174  Return the volume (0..1).
175  """
176  return abs(self._min_volume_min_volume - decibel) / abs(
177  self._min_volume_min_volume - self._max_volume_max_volume
178  )
179 
180  def calc_db(self, volume):
181  """Calculate the decibel given the volume.
182 
183  Return the dB.
184  """
185  return self._min_volume_min_volume + round(
186  abs(self._min_volume_min_volume - self._max_volume_max_volume) * volume
187  )
188 
189 
191  """Representation of a NAD Digital amplifier."""
192 
193  _attr_supported_features = SUPPORT_NAD
194 
195  def __init__(self, config):
196  """Initialize the amplifier."""
197  self._attr_name_attr_name = config[CONF_NAME]
198  self._nad_receiver_nad_receiver = NADReceiverTCP(config.get(CONF_HOST))
199  self._min_vol_min_vol = (config[CONF_MIN_VOLUME] + 90) * 2 # from dB to nad vol (0-200)
200  self._max_vol_max_vol = (config[CONF_MAX_VOLUME] + 90) * 2 # from dB to nad vol (0-200)
201  self._volume_step_volume_step = config[CONF_VOLUME_STEP]
202  self._nad_volume_nad_volume = None
203  self._source_list_source_list = self._nad_receiver_nad_receiver.available_sources()
204 
205  def turn_off(self) -> None:
206  """Turn the media player off."""
207  self._nad_receiver_nad_receiver.power_off()
208 
209  def turn_on(self) -> None:
210  """Turn the media player on."""
211  self._nad_receiver_nad_receiver.power_on()
212 
213  def volume_up(self) -> None:
214  """Step volume up in the configured increments."""
215  self._nad_receiver_nad_receiver.set_volume(self._nad_volume_nad_volume + 2 * self._volume_step_volume_step)
216 
217  def volume_down(self) -> None:
218  """Step volume down in the configured increments."""
219  self._nad_receiver_nad_receiver.set_volume(self._nad_volume_nad_volume - 2 * self._volume_step_volume_step)
220 
221  def set_volume_level(self, volume: float) -> None:
222  """Set volume level, range 0..1."""
223  nad_volume_to_set = int(
224  round(volume * (self._max_vol_max_vol - self._min_vol_min_vol) + self._min_vol_min_vol)
225  )
226  self._nad_receiver_nad_receiver.set_volume(nad_volume_to_set)
227 
228  def mute_volume(self, mute: bool) -> None:
229  """Mute (true) or unmute (false) media player."""
230  if mute:
231  self._nad_receiver_nad_receiver.mute()
232  else:
233  self._nad_receiver_nad_receiver.unmute()
234 
235  def select_source(self, source: str) -> None:
236  """Select input source."""
237  self._nad_receiver_nad_receiver.select_source(source)
238 
239  @property
240  def source_list(self):
241  """List of available input sources."""
242  return self._nad_receiver_nad_receiver.available_sources()
243 
244  def update(self) -> None:
245  """Get the latest details from the device."""
246  try:
247  nad_status = self._nad_receiver_nad_receiver.status()
248  except OSError:
249  return
250  if nad_status is None:
251  return
252 
253  # Update on/off state
254  if nad_status["power"]:
255  self._attr_state_attr_state = MediaPlayerState.ON
256  else:
257  self._attr_state_attr_state = MediaPlayerState.OFF
258 
259  # Update current volume
260  self._attr_volume_level_attr_volume_level = self.nad_vol_to_internal_volnad_vol_to_internal_vol(nad_status["volume"])
261  self._nad_volume_nad_volume = nad_status["volume"]
262 
263  # Update muted state
264  self._attr_is_volume_muted_attr_is_volume_muted = nad_status["muted"]
265 
266  # Update current source
267  self._attr_source_attr_source = nad_status["source"]
268 
269  def nad_vol_to_internal_vol(self, nad_volume):
270  """Convert nad volume range (0-200) to internal volume range.
271 
272  Takes into account configured min and max volume.
273  """
274  if nad_volume < self._min_vol_min_vol:
275  volume_internal = 0.0
276  elif nad_volume > self._max_vol_max_vol:
277  volume_internal = 1.0
278  else:
279  volume_internal = (nad_volume - self._min_vol_min_vol) / (
280  self._max_vol_max_vol - self._min_vol_min_vol
281  )
282  return volume_internal
None add_entities(AsusWrtRouter router, AddEntitiesCallback async_add_entities, set[str] tracked)
web.Response get(self, web.Request request, str config_key)
Definition: view.py:88
None setup_platform(HomeAssistant hass, ConfigType config, AddEntitiesCallback add_entities, DiscoveryInfoType|None discovery_info=None)
Definition: media_player.py:68