An object orientated library for Arduino to control ETCs EOS Family Consoles with OSC over USB or Ethernet UDP. The goal of the library is to have a smart toolbox to create your own boxes which covers your needing by endless combinations of hardware elements.
The library depends on:
This library and examples are a replacement for the #lighthack project, it uses Ethernet instead of USB so the library does not depend on the board type. The in the examples used Ethernet library only supports the WIZnet 5500 chip, used on Ethernet Shield 2 or the popular USR-ES1 module which you can buy for a small pice at aliexpress.com
Following libraries must downloaded for use with Ethernet !!! Beware, the Ethernet libraries have different init procedures !!!
WIZNet w5500 boards like Ethernet Shield 2
Teensy 4.1 with buildin Ethernet
!!! This doesn't work in the moment because of a bug with the Arduino String library in conjunction with the NativeEthernet libray
STM32duino (https://github.com/stm32duino)
!!! This works not as ecpected, there is a bug in the LWIP libray which is causing a massive lost of packages
The library support hardware elements like encoders, faders, buttons with some helper functions. The library allows you to use hardware elements as an object and with the use of the helper functions, code becomes much easier to write and read and to understand.
Buttons
You can use every momentary push button on the market, e.g. MX Keys, the keys are available with different push characters and have therefore different color markers. One pin must connect to a Digital Pin Dn the other to ground.
! A 100nF capacitor is recommended between the button pins !
Faders
Recommended are linear faders with 10k Ohm from Bourns or ALPS which are available in different lengths and qualities.
Beware that ARM boards like STM32-Nucleo or Teensy 4.x use 3.3V, classic AVR boards like Arduino UNO use 5V. The leveler must connect to the Analog Pin An. The other pins must connect to ground and 3.3V or 5V.
! A 10nF capacitor is recommended between leveler and ground !
Rotary Encoders
You can use encoders from ALPS or equivalent.
The middle pin of the encoders must connect to ground, the both other pins A/B must connect to Digital Pins Dn.
! Two 100nF capacitors are recommended between the button pin A/B and ground !
The most problems comes from bad grounding and cables that are to long, on PCB's the shielding design is very important.
Arduino UNO, MEGA with WIZnet 5500 Use AREF Pin instead +5V to the top (single pin) of the fader (100%). Use GND next to AREF and connect to the center button pin (2 pins, the outer pin is normally for the leveler) of the fader (0%)
STM32-Nucleo use IOREF Pin instead +3.3V to the top (single pin) of the fader (100%). GND to the center button pin (2 pins, the outer pin is normally for the leveler) of the fader (0%).
TEENSY 3.x with WIZnet 5500 +3.3V to the top (single pin) of the fader (100%) use ANALOG GND instead the normal GND to the center button pin (2 pins, the outer pin is normally for the leveler) of the fader (0%).
The library support hardware elements like encoders, fader, buttons with some helper functions. The library allows you to use hardware elements as an object and with the use of the helper functions, code becomes much easier to write and read and to understand. Please refer to the EOS manual for more information about OSC.
For use with PlatformIO https://platformio.org, as a recommended IDE with MS VSCode, there is an extra start example folder called #lighthack and #lighthack_ETH.
If you have wishes for other functions or classes make an issue. If you find bugs also, nobody is perfect.
Before using USB one things that must be extra done. You must add a constructor for init the library. This must done before setup(), don't change the name of the constructor!
EOS eos;
The Ethernet functionality is now independent from the hardware port (e.g. WIFI or other Ethernet hardware than WizNet W5500) and libraries. Behind the scenes it uses the virtual Arduino UDP class.
Before using Ethernet there a some things that must be done. It can be different between the diverse libraries.
#include "Ethernet3.h"
#include "EthernetUdp3.h"
// configuration example, must done before setup()
uint8_t mac[] = {0x90, 0xA2, 0xDA, 0x10, 0x14, 0x48};
IPAddress localIP(10, 101, 1, 201);
IPAddress subnet(255, 255, 0, 0);
uint16_t localPort = 8001; // on this port Arduino listen for data
// in EOS Setup > System > Showcontrol > OSC > OSC UDP TX Port
IPAddress eosIP(10, 101, 1, 100);
uint16_t eosPort = 8000; // on this port EOS listen for data
// in EOS Setup > System > Showcontrol > OSC > OSC UDP RX Port
EthernetUDP udp;
EOS eos(udp, eosIP, eosPort);
Ethernet.begin(mac, localIP, subnet);
while(!udp.begin(localPort)); // wait for a connection
It is based on the original box1, no extras
It is based on the original box1, no extras. On an Arduino UNO you can't use the Pins 10 to 13 because they are needed for communication with the Ethernet Shield / Module.
Is a modifcation of the original box1 and uses the former Next and Last buttons to step thru a parameter list. Because of the limited RAM of the Arduino UNO, the parameter list is limited and depends on the size of the parameter names. On an Arduino UNO the maximum is 14 parameter names. On an Arduino MEGA more than 14 parameters can be used, also on other boards like Teensy 3.x or 4. For Teensy you need to fake the PID/VID of the USB connection to work with a console, have a look to the forum. Added a keyword „none“ for gaps in the parameter list, former titled as „empty“.
This box uses 6 buttons for Next, Last, SelectLast, Shift and Parameter Up/Down. It also uses the buttons of the encoder for posting the Home position. So, all Pins of an Arduino UNO are used.
This box is an example for Arduino MEGA with Ethernet Shield 2 and a large 20x4 LCD. It uses 6 buttons for Go, Back, SelectLast, Shift and Parameter Up/Down for stepping to a parameter list. Additional you have a Go/Back button and the LCD displays also cue informations. It also uses the buttons of the encoder for posting the Home position. It is a mixture between #lighthack box2B and cuebox for USB. But you can use more parameters and you have also endless pins on the Arduino MEGA
Is a small faderwing with 6 faders and 12 buttons for Stop / Go
Is a box with 12 buttons to fire Macros, actual 101 thru 112 but this can easily changed.
Is a simple box which shows you the active and pending cue and have buttons for GO and BACK/STOP. There is a parser for the cue text who gives you access to the different parts of the message.
Is a project folder for use with PlatformIO and includes the box1 code as a starting point.
Is a project folder for use with PlatformIO and includes the box1 code as a starting point with the box X code.
void filter(String pattern);
With a Filter you can get messages from the console which you can use for proceeding informations.
void subscribe(String parameter);
void unSubscribe(String parameter); // unsubscribe a parameter
With subscriptions you can get special informations about dedicated parameters.
void ping(); // send a ping without a message
void ping(String message); // send a ping with a message
With a ping you can get a reaction from the console which helps you to identify your box and if is alive. You should send a ping regularly with message to identify your box on the console.
void command(String cmd); // send a command
void newCommand(String newCmd); // clears cmd line before applying
You can send a string to the command line.
void user(int16_t userID);
This function allows you to change the user ID e.g.
void shiftButton(uint8_t pin);
This function allows you to assign a hardware button as a Shift button. Shift set the encoder and wheel messages to the Fine mode or for opposite the acceleration of the Intens parameter. It can replaced by using the optional encoder buttons as a Shift function.
void initFaders(uint8_t page = 1, uint8_t faders = 10, uint8_t bank = 1);
The initFaders() function is basic configuration and must use before you can use your Fader objects.
initFaders(); without a parameter gives you a configuration for use your faders on page 1 of your console
Eos is the main class and must init before you can use this library. The constructor name is fixed! You can use any UDP communication you want including WIFI. Beware of the different methods of initializing the interfaces.
!!! The class name eos is fixed and can’t changed in the moment !!!
EOS(UDP &udp, IPAddress ip, uint16_t port, interface_t interface = EOSUDP); // used for Ethernet
EOS(interface_t interface = EOSUSB); // used for USB
Example for Ethernet UDP, this should done before the setup()
EthernetUDP udp;
EOS eos(udp, eosIP, eosPort);
Example for USB
EOS eos;
void EOS::sendOSC(OSCMessage& msg);
void EOS::sendOSC(OSCMessage& msg, IPAddress ip, uint16_t port);
It is a part of the EOS main class and send an OSC Message, depending on the chosen interface. Example, this function can called inside setup() or loop() after init the interface.
eos.sendOSC(message);
eos.sendOSC(message, ip, port);
With this new class you can create generic buttons which allows you to control other OSC compatible software in the network like QLab. The class initializer is overloaded to allow sending different OSC data types: Integer 32 bit, Float, Strings or no data.
For USB
OscButton(uint8_t pin, String pattern, int32_t integer32);
OscButton(uint8_t pin, String pattern, float float32);
OscButton(uint8_t pin, String pattern, String message);
OscButton(uint8_t pin, String pattern);
For Ethernet UDP
OscButton(uint8_t pin, String pattern, int32_t integer32, IPAddress ip, uint16_t port);
OscButton(uint8_t pin, String pattern, float float32, IPAddress ip, uint16_t port);
OscButton(uint8_t pin, String pattern, String message, IPAddress ip, uint16_t port);
OscButton(uint8_t pin, String pattern, IPAddress ip, uint16_t port);
Example for Ethernet UDP, this should done before the setup()
OscButton pingButton(8 , "/eos/ping", "hello EOS");
Example for Ethernet UDP, this should done before the setup()
IPAddress qlabIP(10, 101, 1, 101); // IP of QLab
uint16_t qlabPort = 53000; // QLab receive port
OscButton qlabGo(8 , "/go", qlabIP, qlabPort);
To get the actual button state you must call inside the loop():
void update();
Example, this must happen in the loop()
qlabGo.update();
The Encoder class creates an encoder object which allows to control parameters:
Encoder(uint8_t pinA, uint8_t pinB, uint8_t direction = FORWARD);
Example, this should done before the setup()
Encoder encoder1(A0, A1, REVERSE);
If the Encoder have an extra button build in, you can add it with following class member:
void button(uint8_t buttonPin, ButtonMode buttonMode = HOME);
buttonMode is the function you can assign, following functions are available
Example, this must happen in the setup() before assigning a Parameter
encoder1.button(A3, HOME);
Before using you must assign a Parameter:
void parameter(String param); // set a parameter
String parameter(); // get the parameter name
Example, this must happen in the setup() before assigning an encoder button
encoder1.parameter("Pan");
To get the actual encoder state you must call inside the loop():
void update();
Example, this must happen in the loop()
encoder1.update();
This class is similar to the Encoder class. It uses the wheel index instead a concrete Parameter. The Wheel index ist send by EOS with implicit OSC output “/eos/out/active/wheel/wheelIndex". You need a helper function to get the Wheel index and other information like parameter name and value. I will do a helper function later. With Wheel it is possible to handle dynamic parameter lists
Wheel(uint8_t pinA, uint8_t pinB, uint8_t direction = FORWARD);
Example, this should done before the setup()
Wheel wheel1(A0, A1, REVERSE);
If the Encoder have an extro button build in, you can add it with following class member:
void button(uint8_t buttonPin, ButtonMode buttonMode = HOME);
buttonMode is the function you can assign, following functions are available
Example, this must happen in the setup() before assigning a index
wheel1.button(A3, HOME);
wheel1.button(A3); // HOME is the standard mode
Before using you must assign a Wheel index:
void index(uint8_t idx); // set a Wheel index
uint8_t idx(); // get the actual index
Example, this must happen in the setup() before assigning an encoder button
wheel.index(1); // most time index 1 is the Intens parameter
To get the actual encoder state you must call inside the loop():
void update();
Example, this must happen in the loop()
wheel1.update();
With this class you can create Key objects which can be triggered with a button.
Key(uint8_t pin, String key);
Example, this should done before the setup()
Key next(8, "NEXT"); // "Next" button on pin 8
To get the actual button state you must call inside the loop():
void update();
Example, this must happen in the loop()
next.update();
With this class you can create Macro objects which can be triggered with a button.
Macro(uint8_t pin, uint16_t macro);
Example, this should done before the setup()
Makro macro1(11, 101); // the button on pin 11 fires the macro
To get the actual button state you must call inside the loop():
void update();
Example, this must happen in the loop()
macro1.update();
This class allows you to control a submaster with a hardware (slide) potentiometer as a fader.
The fader is a linear 10kOhm, from Bourns or ALPS and can be 45/60/100mm long. Put a 10nF ceramic capacitor between ground and fader leveler to prevent analog noise.
Additional Advices
Arduino UNO, MEGA use IOREF instead +5V to the top (single pin) of the fader (100%) GND to the center button pin (2 pins, the outer pin is normaly for the leveler) of the fader (0%)
TEENSY +3.3V to the top (single pin) of the fader (100%) use ANALOG GND instead the normal GND to the center button pin (2 pins, the outer pin is normally for the leveler) of the fader (0%)
Submaster(uint8_t analogPin, uint8_t firePin, uint8_t sub);
Example, this should done before the setup()
Submaster submaster1(A5, 0, 1); // leveler is Analog Pin 5, no bump button, submaster 1 controlled
To get the actual button state you must call inside the loop():
void update();
Example, this must happen in the loop()
submaster1.update();
This class allows you to control a fader containing two control buttons, all functions configured in EOS Tab 36, with a hardware (slide) potentiometer as a fader and buttons. Before using Faders you must call initFaders(page, faders, bank); For fader hardware see also Submaster.
Fader(uint8_t analogPin, uint8_t firePin, uint8_t stopPin, uint8_t fader, uint8_t bank = 1);
Example, this should done before the setup()
Fader fader1(A4, 12, 13, 3); // leveler is Analog Pin 4, fire button at Pin 12, stop button at Pin 13, fader number 3 is controlled
To get the actual button state you must call inside the loop():
void update();
Example, this must happen in the loop()
fader1.update();