Authors: Parth Agrawal([email protected]), Zhaoliang Zheng([email protected])
Previously, we introduced about how to control multiple WIFI-based control board through Blynk app. Though it's also convenient, it has two main drawbacks:
- The local Blynk server requires devices to have different authorization token every time when people reboot the Blynk server, which means that we have to upload new "Auth token" to our devices.
- Blynk app is basically an app, everything on that app is all virtual, including buttons, sliders, joysticks. So, it doesn't even have physical feedback, which will make control through hand really hard and force people to watch the interface.
The overall process flowchart:
The flowchart shows the step-by-step process, from the every beginning to the final control.
To use the code I introduce below, please first clone this git to your local computers:
git clone https://github.com/zhz03/Tutorial-Controlling-multiple-control-boards-through-game-controllers
There are 5 main programs that people should care about:
- Controller mapping program: controller_test.py
- controller_test.py is to test the mapping relationship between game controller and their corresponding relationship
- MAC address programs: MAC_esp32.ino and MAC_esp8266.ino
- MAC_esp32.ino and MAC_esp8266.ino is to get the mac address from ESP-32 based control board and ESP-8266 based control board
- Master control board programs: Esp8266_master.ino and Esp32_master.ino
- These programs should be uploaded to the master control board that controls different slave control boards.
- Slave control board programs: Esp8266_slave.ino and Esp32_slave.ino
- These programs should be uploaded to the slave control boards that is controlled by master board
- Joystick program: Joystick.py
- Joystick.py is to interpret input data from game controllers and send these data to the serial port. It should be running while we're using game controllers as control inputs.
- A laptop that runs Windows 10 or Ubuntu 20 (haven't tested MACbook yet)
- A XBox game controller
- A NodeMCU-Esp8266 as the master board
- Several NodeMCU-Esp8266 as slave boards
- One USB to Micro-USB cable that connects the master board with your laptop
-
python 3
-
If you don't have python 3 on your laptop, I will strongly recommend to install Anaconda, it will make your life easier, especially for beginners.
-
pygame library
-
Use one of the following commands:
pip install pygame pip3 install pygame
-
-
pyserial library
-
Use one of the following commands:
pip install pyserial
-
The wiring of this whole system is shown as follows:
There are four main steps:
-
Initialization:
- if you use a new XBox game controller, it's recommended that your initialize it for the first time and check if your game controller match the one that I used.
- If you use a game controller that is not Xbox, then you must initialize it for the first time.
- If you have run this step before and know the mapping already, then you can just skip this step.
-
Get MAC address:
- MAC address is unique for each edge device, if you don't know the MAC address of your edge devices, then you need to run this step
- If you already know the MAC address of your edge devices, then you can just skip this step
-
Program upload to edge devices
- Master program has to be uploaded to the master board
- Slave program has to be uploaded to the slave board
- If you already finish this step and you don't want to change anything (add more devices to be controlled by master board), then you can just skip this step
-
Operation and execution
- This step is to control your slave edge devices through game controller
-
Plug in game controller
-
You need to install pygame lib first before you run this program
-
Run controller_test.py program
-
You'll see a window pup up like this:
-
Now, press each buttons and joystick on your game controller at one time and notice the number change on the above windows.
-
The normal XBox game controller should look like this:
-
Now, try to match the buttons and its physical buttons using the following tables: (The values are the example of my controller)
Button_name Value Left_stick (left,right) Axis 0 (-1,1) Left_stick (up,down) Axis 1 (-1,1) Left_trigger Axis 2 (0.996) Right_trigger Axis 2 (-0.996) Right_stick (up,down) Axis 3 (-1,1) Right_stick (left,right) Axis 4 (-1,1) A button 0 (0,1) B button 1 (0,1) X button 2 (0,1) Y button 3 (0,1) Left_bumper button 4 (0,1) Right_bumper button 5 (0,1) Back button 6 (0,1) Start button 7 (0,1) Left_stick_button button 8 (0,1) Right_stick_button button 9 (0,1) D-pad (Left,right,up,down) Hat 0 (-1,0) (1,0) (0,1) (0,-1) -
Write down the mapping value on this table, we will need this later.
-
-
Plug in the edge devices that you want to know its MAC address
-
If your edge devices are NodeMCU-esp8266, please check the setup process first and make sure the setup is correct.\
-
Upload MAC_esp8266.ino to your NodeMCU-esp8266
-
After upload is done, open the serial monitor and you'll see its MAC address as follows:
-
Write it down, we will need it later
-
Modify Esp8266_master.ino code to suit your needs:
-
Specify how the MAC address of the slave control board that the master board would control:
uint8_t broadcastAddress1[] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF};
Your MAC address should be formatted like "FF:FF:FF:FF:FF:FF", put that into the above code where FF is.
-
If you need to control multiple slave boards, add more MAC address:
uint8_t broadcastAddress1[] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF}; uint8_t broadcastAddress2[] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF}; uint8_t broadcastAddress3[] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF}; // ... more if you need uint8_t broadcastAddressn[] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF};
-
If you want to talk to only one slave board, then in the setup function of arduino program, use the following code:
esp_now_add_peer(broadcastAddress1, ESP_NOW_ROLE_SLAVE, 1, NULL, 0);
In the loop function of arduino program, use the following code:
esp_now_send(broadcastAddress1, (uint8_t *) &myData, sizeof(myData));
The example code is in the Esp8266_master_single folder.
-
If you need to talk to multiple slave boards, then in the setup function, use the following code:
esp_now_add_peer(broadcastAddress1, ESP_NOW_ROLE_SLAVE, 1, NULL, 0); esp_now_add_peer(broadcastAddress2, ESP_NOW_ROLE_SLAVE, 1, NULL, 0);
In the loop function, use the following code to switch between different devices:
state = Serial.read() - '0'; if (state == 0) { esp_now_send(0, (uint8_t *) &myData, sizeof(myData)); } else if (state == 1) { esp_now_send(broadcastAddress1, (uint8_t *) &myData, sizeof(myData)); } else if (state == 2) { esp_now_send(broadcastAddress2, (uint8_t *) &myData, sizeof(myData)); }
The example code can be found in the Esp8266_master_multiple folder
-
- Modify the OnDataRecv function in the arduino program and control the slave board to let it do the corresponding action as we send data.
-
Modify Joystick.py to feed our needs:
-
Change the serial port name to the one that current NodeMCU is using:
serial_port = serial.Serial("COM9",115200)
For example, "COM9" to "COM20". It depends on your laptop. In the ubuntu system, normally it should be something like "/dev/ttyUSB1"
To check the port name, go to Arduino IDE>Tools>Port: "COM20"
-
-
Plug in game controller
-
Plug in your master board
-
Run Joystick.py in the terminal