Home Assistant custom component integration to control a Nuvo Grand Concerto (NV-I8G)/Essentia G (NV-E6G) using a serial connection.
For the Concerto, Essentia D, and Simplese models see the nuvo_simple integration.
Auto discovers zone and source settings from a Nuvo Grand Concerto/Essentia G amplifier, create HA Entities for each settings, allowing control through the Home Assistant web interface.
Amplifier controls exposed:
- On/Off
- Mute
- Volume
- Source selection
- Bass control
- Treble control
- Speaker balance control
- Loudness compensation control
- Maximum volume level
- Initial: volume level when a zone is powered on for the first time
- Page: volume level when zone is in page mode
- Party: volume level when zone is in party mode
- Volume Reset: toggle to control whether a zone should always reset to Initial volume level at every power on
- Gain control
- Nuvonet/Non-nuvonet source toggle
- Party Mode
- Paging
- Zones AllOff
- Zones Mute All
Provide support for any Nuvo amplifier other than the Grand Concerto/Essentia G.
Interface or control any Nuvonet source component.
Provide a way to send metadata or menus to zone keypads.
Communication between the Nuvo and Home Assistant uses Local Push for state changes, meaning no polling of the Nuvo occurs, so the web interface is fast and responsive. Also as the underlying library nuvo-serial monitors the serial port for activity, any changes to the Nuvo made through zone keypads will be immediately reflected in Home Assistant.
Connection to the Nuvo amplifier is via the "Programming and Serial Control" RS-232 9-pin female connector, either cabled direct to the device running Home Assistant or remotely over a network using a serial-to-network device.
Direct connection to the device running Home Assistant, either through a serial-to-serial cable if the device has a serial port, or a USB-to-RS-232 9-pin male cable.
NOTE when using a usb to serial cable, the device name assigned e.g /dev/ttyUSB1 may change when the machine is rebooted and multiple devices on the USB bus are enumerated in a different order. Obviously if this happens Home Assistant will be using the old device name for the serial port and will not be able to contact the Nuvo. To prevent this it's a good idea to assign a fixed device name to the USB cable e.g. /dev/nuvo
Systems and syntax will vary depending on OS etc, but a good Linux-based overview is here.
It essentially comes to down to:
- Use lusb command to identify usb cable ID
- Create
/etc/udev/rules.d/99-usb-serial.rules
- Add line:
SUBSYSTEM=="tty", ATTRS{idVendor}=="067b", ATTRS{idProduct}=="2303", SYMLINK+="nuvo"
- Reboot, check for existence of
/dev/nuvo
- Use
/dev/nuvo
as Port url in the Integration configuration
Connecting to the Nuvo over a network is possible using a hardware serial-to-network adapter or software such as ser2net.
The port url should then be in the form: socket://host:port
e.g.
A possible ser2net configuration connecting TCP port 10003 to the nuvo device on /dev/ttyUSB1:
Old syntax:
10003:raw:0:/dev/ttyUSB1:57600 8DATABITS NONE 1STOPBIT
ser2net.yaml syntax:
connection: &nuvo
accepter: tcp,10003
enable: on
options:
kickolduser: true
connector: serialdev,
/dev/ttyUSB1,
57600n81,local
Port URL: socket://192.168.5.1:10003
Install using the Home Assistant Community Store HACS.
Once HACS is installed, search the available Integrations for Nuvo multi-zone amplifier (serial) and install to make the integration available for install to Home Assistant.
In Home Assistant home page go to Configuration->Integrations->Add Integration-> Nuvo multi-zone amplifier (serial)
Configure through the GUI:
Port: the url of the serial port e.g /dev/ttyUSB1
Model: Grand Concerto
or Essentia G
The existing physical zones and source names are automatically discovered and displayed for optional modification.
An Entity will now be created for each:
Entity | Integration Type |
---|---|
Zone | Media Player |
Zone balance | Number |
Zone bass | Number |
Zone treble | Number |
Zone loudness compensation | Switch |
Zone Volume Max | Number |
Zone Volume Initial | Number |
Zone Volume Page | Number |
Zone Volume Party | Number |
Zone Volume Reset | Switch |
Source Gain | Number |
Source nuvonet type toggle | Switch |
System AllOff | Button |
System Mute All | Switch |
Activate Paging feature for all zones. This uses the in-built Nuvo paging feature to switch all zones to use fixed source 6 at a pre-configured paging volume.
Deactivate Paging feature, restoring all zone to pre-page state.
Make the media player zone the party host.
Release the media player zone from being the party host.
Take a snapshot of the media player zone. Useful for implementing a custom paging feature using automations.
Restores the last snapshot of the media player zone.
The following services simulate pressing the media buttons on a zone's keypad. Useful for testing automations without having to physically press the keypad button.
Simulate pressing zone keypad Next button.
Simulate pressing zone keypad Prev button.
Simulate pressing zone keypad Play/Pause button.
In addition to the media player integration built-in triggers provided by Home Assistant, this integration also provides device triggers for zone keypad Prev, Next and Play/Pause buttons. A press of one of these buttons will cause a device automation trigger to fire. E.g. Triggers for the zone named "Kitchen" can be found in the Automations trigger interface named:
- Kitchen keypad prev button pressed
- Kitchen keypad next button pressed
- Kitchen keypad play/pause button pressed
NOTE: The Nuvo only emits keypad button event messages (and thus the Home Assistant keypad triggers will only fire) when the currently selected Source for a Zone is configured as a Non-Nuvonet source.
Multiple zones joined together in a group with one zone acting as the group controller.
- Pure HA implementation which does not use Nuvo groups under the hood - this gives a hybrid of Nuvo groups and Nuvo party mode.
- Group management via:
- HA services media_player{join, unjoin}
- Mini media player (MMP) speaker group management
- Multiple groups supported
- Uses the concept of a zone media_player being the group controller
- Group controller operations synced to group members:
- Volume level
- Mute
- Source selection
- Power on/off
- If the group controller leaves a group or is switched off, the group is disbanded and all group member zones are switched off
action: media_player.join
target:
entity_id: media_player.group_controller
data:
group_members:
- media_player.office
- media_player.kitchen
The MMP docs envisage a speaker_group object in each MMP card, allowing any zone to be chosen as a group controller. This makes sense if you have zone keypads and want to dynamically use a physical zone as a group controller.
Another approach when only GUI control is needed is to use an unused speakerless Nuvo zone as a "virtual" group controller. This allows zones to come and go as group members as required and avoids the situation of needing to create a new group if a physical zine group controller is switched off, but the other member zones wish to remain grouped. Note that the GC and Essentia G have 16 and 12 physical zones respectively and will be exposed to the integration at install time if the zones are not in a disabled state.
- MMP LEAVE and UNGROUP buttons do the same thing for the group controller
- MMP LEAVE for a group member removes the zone from the group and switches it off
- MMP speaker_group.supports_master option is supported and must be true (which is the default)
- MMP speaker_group.sync_volume option is not supported, but is implicitly true in the implementation
- MMP speaker_group.entities.volume_offset option is not supported
- MMP platform option: use media_player
Example MMP configuration:
type: entities
state_color: true
entities:
- type: custom:mini-media-player
entity: media_player.group_controller
icon: mdi:speaker-wireless
volume_stateless: false
hide:
controls: false
play_pause: true
icon: true
name: Group Controller
speaker_group:
platform: media_player
icon: mdi:hexagon-multiple
show_group_count: true
expanded: true
sync_volume: false
entities:
- entity_id: media_player.group_controller
name: Group Controller
- entity_id: media_player.kitchen
name: Kitchen
- entity_id: media_player.office
name: Office
Everything in this section is optional and shows a heavily opinionated method of configuring Lovelace to display the Nuvo entities. While it may not be to everyones taste, it should at least give some inspiration for configuration possibilites.
The core Media Player integration (and therefore any Lovelace media control card representing a media player entity) does not provide a way to control a media device's EQ settings. Each EQ setting is modeled using the Number integration. The advantage of this is the ability to use the native number ranges exposed by the Nuvo for each control rather than a card showing a generic 0-X scale.
While Home Assistant will auto-create Lovelace media control and number cards for each Nuvo entity, a more polished look can be achieved using third-party cards mini-media-player and lovelace-slider-entity-row, both cards are installable through HACS.
This example Lovelace configuration displays the EQ settings in a Conditional card which is only displayed when the zone is switched on and an input_boolean entity is True. This input_boolean is toggled by tapping the mini-media-player representing the zone. In order to achieve this, an additional input_boolean entity per-zone needs manually created (it's purely to control the frontend EQ Conditional card, it doesn't represent anything on the Nuvo itself).
The example uses two manually created entities:
input_boolean.eq_office
input_boolean.eq_kitchen
https://www.home-assistant.io/integrations/input_boolean/ As shown the yaml section below, the tap action on each mini-media-player will call the input_boolean.toggle service.
Example section in ui-lovelace.yaml:
views:
- title: MusicZones
cards:
- type: vertical-stack
cards:
- type: entities
entities:
- type: custom:mini-media-player
entity: media_player.office
group: true
hide:
controls: false
info: true
power_state: false
play_pause: true
prev: true
next: true
icon: mdi:speaker-wireless
volume_stateless: true
tap_action:
action: call-service
service: input_boolean.toggle
service_data:
entity_id: input_boolean.eq_office
- type: custom:slider-entity-row
entity: media_player.office
full_row: true
step: 1
hide_state: false
hide_when_off: true
- type: conditional
conditions:
- entity: media_player.office
state: "on"
- entity: input_boolean.eq_office
state: "on"
card:
type: entities
entities:
- type: custom:slider-entity-row
entity: number.office_bass
name: Bass
icon: mdi:music-clef-bass
hide_state: false
hide_when_off: true
full_row: false
- type: custom:slider-entity-row
entity: number.office_treble
full_row: false
name: Treble
icon: mdi:music-clef-treble
hide_state: false
hide_when_off: true
- type: custom:slider-entity-row
entity: number.office_balance
full_row: false
name: Balance
hide_state: false
hide_when_off: true
- entity: switch.office_loudcmp
name: LoudComp
show_state: true
- type: custom:slider-entity-row
entity: number.office_volume_max
full_row: false
name: Volume Max
hide_state: false
hide_when_off: true
icon: mdi:tune
- type: custom:slider-entity-row
entity: number.office_volume_initial
full_row: false
name: Volume Initial
hide_state: false
hide_when_off: true
icon: mdi:tune
- type: custom:slider-entity-row
entity: number.office_volume_page
full_row: false
name: Volume Page
hide_state: false
hide_when_off: true
icon: mdi:tune
- type: custom:slider-entity-row
entity: number.office_volume_party
full_row: false
name: Volume Party
hide_state: false
hide_when_off: true
icon: mdi:tune
- entity: switch.office_volume_reset
name: Volume Reset
show_state: true
- type: entities
entities:
- type: custom:mini-media-player
entity: media_player.kitchen
group: true
hide:
controls: false
info: true
power_state: false
play_pause: true
prev: true
next: true
icon: mdi:speaker-wireless
volume_stateless: true
tap_action:
action: call-service
service: input_boolean.toggle
service_data:
entity_id: input_boolean.eq_kitchen
- type: custom:slider-entity-row
entity: media_player.kitchen
full_row: true
step: 1
hide_state: false
hide_when_off: true
- type: conditional
conditions:
- entity: media_player.kitchen
state: "on"
- entity: input_boolean.eq_kitchen
state: "on"
card:
type: entities
entities:
- type: custom:slider-entity-row
entity: number.kitchen_bass
name: Bass
icon: mdi:music-clef-bass
hide_state: false
hide_when_off: true
full_row: false
- type: custom:slider-entity-row
entity: number.kitchen_treble
full_row: false
name: Treble
icon: mdi:music-clef-treble
hide_state: false
hide_when_off: true
- type: custom:slider-entity-row
entity: number.kitchen_balance
full_row: false
name: Balance
hide_state: false
hide_when_off: true
- entity: switch.kitchen_loudcmp
name: LoudComp
show_state: true
- type: custom:slider-entity-row
entity: number.kitchen_volume_max
full_row: false
name: Volume Max
hide_state: false
hide_when_off: true
icon: mdi:tune
- type: custom:slider-entity-row
entity: number.kitchen_volume_initial
full_row: false
name: Volume Initial
hide_state: false
hide_when_off: true
icon: mdi:tune
- type: custom:slider-entity-row
entity: number.kitchen_volume_page
full_row: false
name: Volume Page
hide_state: false
hide_when_off: true
icon: mdi:tune
- type: custom:slider-entity-row
entity: number.kitchen_volume_party
full_row: false
name: Volume Party
hide_state: false
hide_when_off: true
icon: mdi:tune
- entity: switch.kitchen_volume_reset
name: Volume Reset
show_state: true
This configuration will display the cards below, with the EQ settings card toggled by tapping on the media player, in any area not containing a control: