/Sonos-Kids-Controller

Software for self made touchscreen jukeboxes for kids. Supports Spotify streaming and uses Sonos for audio output.

Primary LanguageJavaScript

Sonos-Kids-Controller

Content

About Sonos-Kids-Controller
Dependencies
Usage
Configuration
Adding Content
Autostart
Update
Hardware Player
Raspberry Pi optimization
Alternative Installation using Docker






About Sonos-Kids-Controller

This software can be used to create a touch-based Sonos controller for your kids.

It can also be used with Spotify Connect compatible devices instead of Sonos if you use this software instead of the node-sonos-http-api discribed later in this document.

Sonos-Kids-Controller uses TTS (text to speech) when you click on texts in the UI, so your kids don't have to aks you anymore about the name of a specific episode.

The recommended use case is in combination with Spotify Premium, as it's web API allows you to add albums using artist and album name instead of cryptic album-IDs. It's also possible to add multiple albums with a single search query (e.g. all albums from a sepcific artist).

But you can also add albums from the local Sonos library (in case an album isn't available in your favorite streaming service), from Apple Music or from Amazon Music Unlimited by specifiying the corresponding album IDs. See the music services section about how to retrieve these IDs.

The software consists of 2 parts:

  • The server component, running in an node express environment. Handles the album library and serves the client component to the browser
  • The client component, developed in Ionic/Angular, which can be opened in a browser

Dependencies

This software uses node-sonos-http-api to control your Sonos hardware. So you need to have it running somewhere, for example on the same system as this software.
This doesn't have to be the Pi itself, but should be possible too (if it can handle everything performance-wise without any lags).

Usage

Ensure that you have Node.js and npm installed. Also install node-sonos-http-api as described in the readme of the software. If you plan on using Spotify, follow the instructions here.

Then install this software from Github:

sudo npm install -g @ionic/cli

wget https://github.com/Thyraz/Sonos-Kids-Controller/archive/master.zip

unzip master.zip

rm master.zip 

cd Sonos-Kids-Controller-master

npm install

ionic build --prod

Create the configuration file by making a copy of the included example:

cd server/config

cp config-example.json config.json

Edit the config file as discribed in the chapter configuration

Then start the software like this:

npm start

After that open a browser window and navigate to

http://ip.of.the.server:8200

Now the user interface should appear

Configuration

{
    "node-sonos-http-api": {
        "server": "127.0.0.1",
        "port": "5005",
        "rooms": [
            "Livingroom",
            "Kitchen"
        ]
    },
    "spotify": {
        "clientId": "your_id",
        "clientSecret": "your_secret"
    }
}

Point the node-sonos-http-api section to the adress and the port where the service is running. The rooms are the Sonos room names that you want to be allowed as target.

Room selection isn't implemented yet, so only the first room will be used at the moment.

The spotify section is only needed when you want to use Spotify Premium as source. The id and the secret are the same values as entered in the node-sonos-http-api configuration as described here.

Adding Content

There's a hidden button in the root view on the right side of the top navigation bar. If you click there, you should see an overlay lighting up. Click this button quickly 10 times to open the library editor.

Then click the "+" button on the top right to add a new entry.

Local Sonos Library:

  • Enter artist name and album name exactly as it's displayed in the Sonos app.
  • Enter an artwork link for an artwork image (remember, you can open the UI on any browser, so you can use a desktop pc or an mobile phone to add items to the library and use copy and paste for artwork links.) A good source for album artworks is the iTunes Artwork Finder: https://bendodson.com/projects/itunes-artwork-finder/

Spotify Premium:

  • Enter artist and album name to add a single album
  • Add a query instead, to search for multiple albums
  • Album artwork will be automatically retreived from Spotify

Examples for query strings:

artist:Max Kruse album:Urmel

artist:Grüffelo

artist:Benjamin Blümchen album:folge NOT gute-nacht

artist:"Super Wings"

More details on Spotify web API search querys: https://developer.spotify.com/documentation/web-api/reference/search/search/#writing-a-query---guidelines

Apple Music or Amazon Music Unlimited:

Pro Tip: As Amazon and Apple don't provide a full public API to search for content like Spotify does, adding AlbumIDs and artwork links through the UI might be time consuming and complicated. you can also edit the library by editing server/config/data.json (created after you added the first content through the UI). The structure should be self-explaining. Just be sure to shutdown Sonos-Kids-Controller before editing the file, as the software might otherwise overwrite your changes with an in-memory copy of the data. Also a backup of the file might be a good idea, as the software might overwrite the file with an empty library on startup, when you have some syntax errors in your edits, preventing the data from loading.

Autostart

I use pm2 as process manager for node.js projects. So both services (node-sonos-http-api and Sonos-Kids-Controller) are startet on boot time in this case

sudo npm install pm2 -g

Then build a startup script for pm2 (don't run with sudo):

pm2 startup

after that pm2 should show you a command that has to be run as sudo to finish install the startup scripts. Copy and paste it into the terminal and execute it.

then in the directory of node-sonos-http-api:

pm2 start server.js
pm2 save

again in the directory of Sonos-Kids-Controller:

pm2 start server.js
pm2 save

After a reboot, enter pm2 list in the terminal and you should see that the 2 services are running.

Update

Updating to a newer Sonos-Kids-Controller version works similar to the initial installation. Execute these commands, starting from the parent directory of you current Sonos-Kids-Controller-master installation:

wget https://github.com/Thyraz/Sonos-Kids-Controller/archive/master.zip

unzip master.zip

rm master.zip 

cd Sonos-Kids-Controller-master

npm install

ionic build --prod

If you run into out of memory erros during the build process, try to set a memory limit manually:

export NODE_OPTIONS=--max-old-space-size=3072

ionic build --prod

Hardware Player

While you can simply run this software on any server supported by node.js and open it in the browser of your choice (as long as it isn't IE or Edge), the typical use case will be a small box powered by an Raspberry Pi and a capacitive touch screen.

I recommend a 5" touch screen with a resolution of 800x480, as you otherwise might have to edit the layout of the software.

Part List:

Here's a list of what I bought for my player:

Front Cover Cutout:

As first step, choose the position of the cutout for the touchscreen in the wooden front cover. Pay attention how far the HDMI and USB connectors stick out of the touch screen. Because of that you won't be able to center the touchscreen vertically in the front cover.

After that I drilled 4 holes in each edge of the cutout. Then I used a fredsaw to remove the part between the holes. Keep the cutout a little bit smaller than the touchscreen, as working with a fredsaw might not be that precise.

Now the hard part begins:
Tweak the shape of the cutout with a rasp until the touchscreen fits in tightly. This might take a while and I recommend to hold the touchscreen and the cover in front of each other toward a light source from time to time, to see where you need to remove more of the wood.

When the touchscreen fits into the cutout, use sand paper to smooth the surface. I started with grain size 80 and ended with 300. If you want an uniform look, use the finer grained sandpaper also on the front surface of the wood and apply some wood oil everywhere for a consistant looking finish.

Assembly:

Cut off the connector of the power supply, and drill a hole for the cable in the back of the case. Stick the cable through the hole and solder the connector back on.

Assemble the Pi in the bottom part of the raspberry case and use the Powerstrips to attach it inside the box.

If the touchscreen isn't sitting tight enough in the cutout, use short screws to attach it to the front cover. Maybe predrill the holes, so the wood won't crack.

Connect the cables, insert the microSD card and you're done. The wooden front cover should stick firmly if you press it onto the backside. I didn't need any screws to hold it in place. If you need to open it again, push a knive between the front cover and the backside and use it as lever.

Kiosk Mode Installation

Use raspbian light (currently Buster is the latest stable version as I write this readme) as we don't want to install the default desktop environment.

a) because we don't need it
b) because our Ionic/Angular app will stress the Pi enough, so we want to avoid too much running services and RAM usage.

edit the config.txt in the boot partition and add the following lines at the end to configure the display:

max_usb_current = 1 
Hdmi_group = 2 
Hdmi_mode = 87 
Hdmi_cvt 800 480 60 6 0 0 0 

After the initial boot process, start raspi-config and configure:

  • Localisation options and keyboard layout
  • Configure Wifi
  • Enable SSH if you wish to be able to log into the box remotely
  • Disable HDMI overscan in the advanced option
  • Change user password for pi
  • In Boot Options Select 'Desktop/CLI' and then 'Console Autologin' for automatic login of the user pi

Now reboot the Pi. If everything worked, you should be logged into the terminal session without having to enter you password at the end the boot process.

Update all preinstalled packages:

sudo apt-get update
sudo apt-get upgrade

Now we install Openbox as a lightweight window manager:

sudo apt-get install --no-install-recommends xserver-xorg x11-xserver-utils xinit openbox

And Chromium as a browser:

sudo apt-get install --no-install-recommends chromium-browser

You can now install node-sonos-http-api and Sonos-Kids-Controller on the Pi (as described in the previous chapters) if you don't want to run it on a different server. Depending on where the services are running edit the automatic startup of Openbox and Chromium in /etc/xdg/openbox/autostart

# Disable screen saver / power management
xset s off
xset s noblank
xset -dpms

# Start Chromium
sed -i 's/"exited_cleanly":false/"exited_cleanly":true/' ~/.config/chromium/'Local State'
sed -i 's/"exited_cleanly":false/"exited_cleanly":true/; s/"exit_type":"[^"]\+"/"exit_type":"Normal"/' ~/.config/chromium/Default/Preferences
chromium-browser --disable-infobars --kiosk 'http://url-to-sonos-kids-controller:8200'

Now Chromium should display our web app when Openbox is started. The last thing to do is to start the X server automatically when the Pi is powered on. As we already have automatic login in the terminal session, we can use .bash_profile for starting X. Append the following line to the file:

[[ -z $DISPLAY && $XDG_VTNR -eq 1 ]] && startx -- -nocursor

This starts X when the user logged into the first system terminal (which is the one autologin uses).

Restart the pi to see if everything works.

If you see a bubble in Chromium after some time, about Chromium not beeing up to date, use this workaround from StackOverflow and execute the following command:

sudo touch /etc/chromium-browser/customizations/01-disable-update-check;echo CHROMIUM_FLAGS=\"\$\{CHROMIUM_FLAGS\} --check-for-update-interval=31536000\" | sudo tee /etc/chromium-browser/customizations/01-disable-update-check

Raspberry Pi optimization

You may perform the following changes to Rasperry Pi OS AT YOUR OWN RISK.

Disable boot up messages

According to the guide on the site https://florianmuller.com/polish-your-raspberry-pi-clean-boot-splash-screen-video-noconsole-zram you can disable the boot up messages by adding the following text to the file /boot/cmdline.txt (no linefeed, no double space):

 loglevel=0 plymouth.enable=0 vt.global_cursor_default=0 plymouth.ignore-serial-consoles splash fastboot noatime nodiratime noram

Log to ramdisk

You may add the folling lines to the file /etc/fstab to write temporary files and log files to ram. Attention: they get lost with every reboot.

tmpfs    /tmp    tmpfs    defaults,noatime,nosuid,size=100m    0 0
tmpfs    /var/tmp    tmpfs    defaults,noatime,nosuid,size=25m    0 0
tmpfs    /var/log    tmpfs    defaults,noatime,nosuid,mode=0755,size=25m    0 0

This will speed up the boot up process and helps to improve the lifetime of the SD card.

Fixed IP-Adress

a fixed IP Adress will speed up the boot process by about 3-5 seconds. Add your details to /etc/dhcpcd.conf. For example:

interface wlan0
static ip_address=192.168.0.4/24    
static routers=192.168.0.254
static domain_name_servers=192.168.0.254 8.8.8.8

Overclocking CPU

You may consider overclock your hardware by add the following lines to __/boot/config.txt. THIS MAY HARM YOUR HARDWARE!

Example for Raspberry 3B:

# Overclock CPU
arm_freq=1200
over_voltage=4
temp_limit=75
core_freq=500

# Overclock GPU
h264_freq=333
avoid_pwm_pll=1
gpu_mem=320
v3d_freq=500

# Overclock RAM
sdram_freq=588
sdram_schmoo=0x02000020
over_voltage_sdram_p=6
over_voltage_sdram_i=4
over_voltage_sdram_c=4

Overclocking sd card

If you have a suitable SD Card you may consider overclock the SD card reader by add the following lines to __/boot/config.txt. THIS MAY HARM YOUR HARDWARE!

dtparam=sd_overclock=100

Docker

There is now also an easy way to setup this software using Docker. This avoids the compilation on small hardware.

The image is maintained by stepman0
Get it here.