In our day-to-day life, we use various gadgets such as television, set-top boxes, air conditioners, home theatre, DVD players, and many other remote-operated devices. Different gadgets means maintaining different remote controls which are not only clumsy but also difficult to manage. Universal remote control simplifies our life because it helps us control any IR devices in our daily life. So in this project we are going to build a single IR-based universal remote which can control all available IR-remote controlled electronic gadgets. Also thanks to the sponsors of this project pcbway.com we were able to build this Smart Remote on a sleek shiny black PCB with touch capacitive buttons making it look more cool.
In this project, we are going to build a universal remote which will be versatile and can quickly learn new devices and new IR protocols. This is not just going to be a universal remote but a Smart Universal remote with smart features like IR Learning, Control over WiFi, and Google Assistant Voice command support.
Components Required to Build Smart Universal Remote
The components required to build this project are very few. The components required are the following:
- ESP32-WROOM-32 Module x1
- TP5100 2S Li-Ion Charger IC x1
- CH340C USB -TTL Converter IC x1
- MT3608 Boost Converter IC x1
- HY2120CB 2S Battery Protection IC x1
- 8205A SMD N-Channel MOSFET TSSOP-8 x1
- AO3401 P-Channel MOSFET SOT-23-3 x1
- S8050 SMD transistor SOT-23-3 x3
- VS1838B IR Sensor x1
- RGB LED 1615 Common Cathode x1
- RGB LED 1615 Common Anode x1
- 5mm IR LED x3
- SS34 Diode DO-214AC x2
- 22uH SMD Inductor 7x6.8mm x2
- 0.1uF Capacitor 0603 x6
- 1nF Capacitor 0603 x1
- 10uF Capacitor 0603 x2
- 10uF Capacitor 1206 x3
- 22uF Capacitor 1206 x2
- 0.5Ω Resistor 1206 x2
- 330Ω Resistor 0603 x2
- 470Ω Resistor 0603 x2
- 680Ω Resistor 0603 x1
- 1KΩ Resistor 0603 x5
- 1.2KΩ Resistor 0603 x1
- 2KΩ Resistor 0603 x1
- 4.7K Ω Resistor 0603 x2
- 10KΩ Resistor 0603 x6
- 22KΩ Resistor 0603 x1
- 47KΩ Resistor 0603 x1
- 100KΩ Resistor 0603 x1
- SPDT SMD Switch x3
- Micro USB Connector x1
- 2S 18650 Holder SMD x1
- Li-ion Battery 18650 x2
- PCB
- SMD Rework station
- Other Necessary tools and consumables
Smart Universal Remote Schematics
The complete schematics diagram for the Smart Universal Remote control is given below.
Here is the full assembled smart remote.
Now let’s discuss each of the sections. The 5V input from the USB connector is connected to a Boost converter circuit built around the popular MT3608 IC from XI'AN Aerosemi Tech. The MT3608 is a constant frequency, 6-pin SOT23 current mode step-up converter intended for small, low-power applications. The MT3608 switches at 1.2MHz and this allows the use of tiny, low-cost capacitors and inductors. Internal soft-start results in a small inrush current and extends battery life. The MT3608 features automatic shifting to pulse frequency modulation mode at light loads. The MT3608 includes under-voltage lockout, current limiting, and thermal overload protection to prevent damage in the event of an output overload. Since we are using a 2S battery configuration, we will need at least 8.4V to fully charge our batteries. That’s why we are using this boost converter circuit to boost the 5V input to 8.4V. You can change the output voltage of the boost converter by changing the values of resistors R1 and R2.
The charger circuit is built around the TP5100 charging IC from TOPPOWER. TP5100 is a step-down switching double 8.4V / 4.2V single lithium battery charge management chip in an ultra-small QFN16 package. It requires a very minimal external circuit which makes it ideal for portable devices. Meanwhile, TP5100 has built-in input overcurrent, under-voltage protection, over-temperature protection, short circuit protection, battery temperature monitoring, and reverse battery protection. TP5100 can handle a wide input voltage of 5V-18V. The charge current can be adjusted through an external resistor. The maximum charging current is 2A. TP5100 uses a switching frequency of 400KHz.
A RGB LED in the 1615 package is used as a charging indicator. The charging current is set below 500mA to make sure the MT3608 is not overdriven, which would result in overheating. The charging current can be adjusted by changing the value of Resistors R17 and R26.
The battery protection circuit is based on the HY2120CB from HYCONTEK. This chip along with 8205A dual N-Channel MOSFET will protect the battery from over-discharge, overcharge, and accidental short circuit situations. This is the most common configuration you can find on most commercially available 2S BMS solutions.
The following section contains the 3.3V voltage regulator and the power switch. The output from the BMS is connected to a P-Channel MOSFET and the MOSFET gate is controlled to power the rest of the circuit. This approach is used since the small SMD slide switch can’t handle a current of more than 300mA. When we connect the gate of the MOSFET to a 0V level, the MOSFET will turn on and the current will pass through. A pullup resistor is used to ensure the MOSFET is fully turned OFF. AO3401 MOSFET from ALPHA & OMEGA Semiconductors is used since they are available in a small SOT-23 package and can handle a current up to 4A with very low RDS(On) of 85mΩ at 2.5V Vgs. The 3.3V for the SoC is made using the popular AMS1117-3.3 LDO. Sufficient bypass capacitors are used to eliminate any possible noise.
The programming circuit is made around the CH340C USB – TTL IC from WCH. The C variant doesn’t require an external crystal, and this will help save precious board space and will give better performance. The ESP32 auto-reset circuit is made with two NPN transistors. One thing to keep in mind is that since the GPIO0 is one of the touch inputs, the auto-reset circuit will interfere with the touch readings. We can avoid this by not installing the resistor R37. If you don’t install this resistor, you must short the resistor pads during the programming using a tweezer and remove that after the programming starts. Another workaround is to install the resistor and remove it after the initial programming. The future updates can be done over OTA.
The heart of this project is the ESP32-WROOM-32 module. The basic connections for the ESP32 are given below. An RGB LED is used here for notifications and mode indications.
The IR transmitter section consists of three IR LEDs along with the current limit resistors and a driver transistor. The transistor is driven with the PWM signal from GPIO16. The receiver is simple a VS1838 IR sensor and is connected to the GPIO17 pin.
Battery voltage monitoring is done using a voltage divider, which will translate the high battery voltage to a safe voltage level below 3.3V. The Smart Remote has three operational modes and can be selected using the two slide switches. More details are provided in the operation instruction below.
Finally, the 10 touch pads are used to detect each corresponding touch. The pads are Arranged neatly for better visual appeal and ease of use.
DIY Smart Universal Remote PCB Design
The PCB for the Smart IR Remote has a dimension of 145mm x 50mm. The image below shows the top side of the two-layer PCB. The top side has only two indicator LEDs. One on the bottom, near the USB port, will show the charging status and the other one on the top will show the mode status and other indications.
And here is the bottom side. The PCB is designed in a way that most of the components are on the bottom side. This will not only make the assembly process easier and cheaper, but the top side will look neat and clean.
You can download the Universal Smart Remote PCB Layout Gerber File from here.
Ordering the Smart Universal IR Remote PCB from PCBWay
Now, after finalizing the design, you can proceed with ordering the PCB:
Step 1: Get into https://www.pcbway.com/, and sign up if this is your first time. Then, in the PCB Prototype tab, enter the dimensions of your PCB, the number of layers, and the number of PCBs you require.
Step 2: Proceed by clicking on the ‘Quote Now’ button. You will be taken to a page to set a few additional parameters like the Board type, Layers, Material for PCB, Thickness, and more. Most of them are selected by default, if you are opting for any specific parameters, you can select it here.
Step 3: The final step is to upload the Gerber file and proceed with the payment. To make sure the process is smooth, PCBWAY verifies if your Gerber file is valid before proceeding with the payment. This way, you can be sure that your PCB is fabrication friendly and will reach you as committed.
Once you upload the Gerber file and make the payment, your job is done and you will get a confirmation email with all your details in your email address.
Assembling the Smart Universal Remote Control PCB
After the board was ordered, it reached me after some days through our courier service in a neatly labeled well-packed box. The PCB quality was good as always. The top layer and the bottom layer of the board are shown below:
After making sure the tracks and footprints were correct. I proceeded with the assembling of the PCB. The completely soldered board looks like the image below:
Now, we can move on to the next step that is configuring Adafruit IO and IFTTT
Setting Up Adafruit IO and IFTTT for Google Assistant Integration
As we have already mentioned, this Smart Remote supports Google Assistant Voice Command. For the Google Assistant integration, we are going to use the Adafruit IO and the IFTTT. So to start, go to Adafruit IO and create an account. If you already have an account login into it. Once logged in, go to the Dashboards page and create a new dashboard. Once created, open that dashboard and create a block by selecting the Create New Block option from the setting menu.
In the Create New Block wizard choose text block and in the connect a feed windows create a new feed called Smart Remote as shown below.
Click on Next Step and in the next window enter the block title and click Create Block to finish. This will add a text block to your dashboard. Once it’s done, click on the My Key link on the top of the dashboard and copy your USERNAME and KEY. Add this to the Arduino code.
Now to set up IFTTT, go to the IFTTT website and create a new account or login into the existing account. Once Logged in, click on Create button on the top. In the Create Applet widget, select Google Assistant for If This and connect the same google account you’re using on your phone when asked. In the next window, set everything as shown below and click save.
And for Then That select Send data to Adafruit IO template and link your Adafruit IO account when asked. And in the setting select the Smart remote feed you have previously created and add a text field ingredient as shown below.
Then click on save and that’s all.
Arduino Code for Smart Remote
Download the entire code from the Circuit Digest GitHub repo link given at the bottom of this article. Extract it to a folder and make sure not to change any file or folder structure. All the necessary libraries and files are included in the sketch folder. Open the sketch in the Arduino IDE and select ESP32 Dev Module as the board. Also, set the partition scheme to Minimal SPIFFS (1.9MB APP with OTA/109KB SPIFFS).
Add your Adafruit IO username and key to the sketch. Then compile the code and upload it.
And while programming, if you haven’t installed the R37 resistor, you must short the resistor pads during the programming using a tweezer and remove that after the programming starts. And if you have installed it, remove it as soon as the programming is done.
Smart Remote Operation Instruction
The Smart Remote has three operation modes namely Learning Mode, Smart Mode, and Power Saving Mode. You can select the mode using the two slide switches on board. Please check the image below to know how to select each mode. Keep in mind that you should change to Smart / Power saving once the learning or IR training is done.
Now let’s see how each mode works and how to operate in each mode.
Learning Mode
In this mode, we can store code for each touch input. To enter learning mode, change the Learning mode switch to its down position, and the status led will show an RGB pattern to indicate that the remote is in the learning mode. To train or store the IR code, touch on the corresponding touchpad, and press the switch on the other remote you want to learn, pointing to the IR receiver as shown below. Once valid code is stored the RGB led will flash an RGB pattern indicating the code is saved. Repeat this process for all other touchpads and change the Learning to the upward position once done.
Smart Mode
In the smart mode, you can operate the remote in three ways. By directly touching the corresponding touch pad, by using a smartphone or using the Google Assistant Voice command. To use the Smart mode, make sure the learning mode is off and put the other mode selection switch to the downward position. The RGB LED will blink in red color while booting to indicate that the remote is in Smart mode. And the LED will stay on in red color.
During the initial set-up or if the remote Wi-Fi settings are reset, the Remote will change to configuration mode. During this, the touch pads won’t work. To configure the remote, use a smartphone to connect to the Wi-Fi hotspot named Smart Remote. The password for the hotspot is circuitdigest.
Once connected the following page will open automatically. Click on Configure WiFi.
In a few seconds, a new page will open showing all the available networks. You can either select one or enter the SSID manually.
Once the password is entered, click on Save to save the WiFi configuration.
Once done, after a few seconds turn off and turn on the remote. The Smart remote will connect to the WiFi you have configured. If the WiFi connection is failed the remote will go to the configuration mode and you can set the WiFi settings again. This setup is only necessary at once. The remote will connect to the WiFi network automatically after this.
Once the initial setup is complete, you can access the remote through any web browser. If you are using an Apple device or a PC, you can access the remote by going to the following address http://smartremote.local/. If you’re using an Android device, you can access the remote by entering the device’s IP address. You can find this IP address either from your router or by connecting it to the PC and opening the Serial monitor and then rebooting. You can also use any network scanning app to find the IP.
You can operate the remote by pressing any button on the resulting webpage. The touch pads will also work so that you can use the remote simultaneously. You can operate the remote with Google Assistant voice command too. For this, make sure your WiFi has internet access. To use the Voice command, make sure you’re using the same Google account on your phone, which you have linked to the IFTTT applet. To operate, simply say “Ok Google, Send button next” or “Hey Google, Send button next”. Change the last phrase to the corresponding keyword. The current keywords are power, ok, Up, Down, Left, Right, Plus, Minus, Next, and Previous. You can change or customize these phrases in the Arduino code if you like.
Power Saving Mode
In this mode, the remote will work as a normal remote. The WiFi and other smart functions will be disabled. You can operate the remote with the touchpad. The remote will go to deep sleep mode if it’s not used for 20 seconds to save power and will wake automatically when you touch touchpads.
Resetting the Smart Remote
You can reset the remote if you want to reset the WiFi configurations. To do this, make sure the remote is in Smart mode. Turn off the device. Turn it back on while touching and holding power, minus, and previous buttons. Keep the touch until you see the LED flash in red and blue color alternatively. Once the LED starts to flash, release the touch and touch the Ok button to confirm reset. The Remote will clear all the WiFi configurations and will go into configuration mode. Now you can set the WiFi configurations as mentioned earlier.
Updating the Smart Remote
To make the Smart Remote, we have included the OTA firmware update. You can update your Smart Remote’s firmware from any browser, and it is pretty easy too. To update the firmware, first make sure the Smart remote is in the smart mode and the smartphone or computer you’re using to update the firmware is connected to the same network as the Smart Remote. Then go to the update URL http://ip_address/update. Replace the ip_address with the IP address of the Smart Remote, for eg example the IP address of my smart remote is 192.168.29.246 so the OTA update URL of my Smart Remote will be http://192.168.29.246/update.
If you are using an Apple device or a computer, you can also access the OTA update page by going to the MDNS address http://smartremote.local/update. Keep in mind that this address won’t work with all Android phones since the older Android Versions don’t support MDNS. On the update page, make sure the Firmware radio option is selected. Then click on Choose File button and select your firmware bin file. Once a valid firmware file is selected, the Smart Remote will be automatically updated to it.
Once the update is complete and successful, the Smart Remote will automatically reboot.
Charging the Smart Remote
You can charge the remote by simply connecting the USB cable from any 5V charger to the micro-USB port at the bottom. The charging indicator will glow in red, indicating that the device is being charged. Once fully charged the LED will change to blue.
You can download all the necessary files from the Circuit Digest GitHub repo, from the following link.
/***************************************************************************************************** * Poject : Smart Universal Remote * Description : Smart Universal Remote with Learning function and Google Assistant Support * Author : Jobit Joseph * For : CircuitDigest * Date : 10-06-2022 * Copy Writes : © Jobit Joseph – Circuit Digest (Semicon Media Pvt Ltd) *****************************************************************************************************/ /* Connection pinout Pwr -- Touch4 -- IO13 Ok -- touch -- 6 io14 up -- touch9 -- io32 down -- touch8 -- i033 left -- touch5 -- io12 right -- touch7 -- io27 next -- touch2 -- io2 prev -- touch3 -- io15 plus -- touch1 -- io0 minus -- touch0 -- io4 Mode -- switch -- IO23 Learn -- Switch -- IO22 */ #include <Arduino.h> #include <WiFi.h> #include <ESPmDNS.h> #include "SPIFFS.h" #include "PinDefinitionsAndMore.h" //Define macros for input and output pin etc. #include "src/WiFiManager.h" #include "src/AsyncElegantOTA.h" #include "src/IRremote.hpp" #include "src/AsyncTCP.h" #include "src/ESPAsyncWebServer.h" #define mode_switch 23 #define learn_switch 22 #include "src/Adafruit_MQTT.h" #include "src/Adafruit_MQTT_Client.h" /************************* Adafruit.io Setup *********************************/ #define AIO_SERVER "io.adafruit.com" #define AIO_SERVERPORT 1883 // use 8883 for SSL #define AIO_USERNAME "jobitxxxxxxxx" // Replace it with your Adafruit IO username #define AIO_KEY "aio_qOXrxxxxxxxxxxxxxxxxxxM4yyyi4" // Replace with your Adafruit IO Project Auth Key WiFiClient client; // Setup the MQTT client class by passing in the WiFi client and MQTT server and login details. Adafruit_MQTT_Client mqtt(&client, AIO_SERVER, AIO_SERVERPORT, AIO_USERNAME, AIO_KEY); /****************************** Feeds ***************************************/ // Setup a feed called 'onoff' for subscribing to changes. Adafruit_MQTT_Subscribe SRemote = Adafruit_MQTT_Subscribe(&mqtt, AIO_USERNAME"/feeds/smart-remote"); // FeedName // Create AsyncWebServer object on port 80 AsyncWebServer server(80); AsyncWebSocket ws("/ws"); int STATUS_PIN = 19; int DELAY_BETWEEN_REPEAT = 50; int DEFAULT_NUMBER_OF_REPEATS_TO_SEND = 3; /* Html page for the websever*/ const char index_html[] PROGMEM = R"rawliteral( <!DOCTYPE html> <html> <head> <link rel="icon" type="image/png" href=""> <meta name="viewport" content="width=device-width, initial-scale=1"> <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css"> <style type="text/css"> body{ margin:0; padding: 0; color:#1c1c1c; font-family: roboto; position: relative; text-align:center; } p{margin:0;padding:0;} .clr{ clear:both; } #bg_mob{ background-image: linear-gradient(#28313B, #485461); //background-image:url(""); background-repeat:no-repeat; background-size:cover; max-width: 540px; margin:0 auto; position: relative; height:100vh; } .mob_inner_con{ padding: 35px; } .btn_click{ background: none; border: none; } .btn_click:active { transform: translateY(2px); } #content-wrap { padding-bottom: 2.5rem; } #footer { position: absolute; bottom: 2%; width: 100%; } </style> <script> var gateway = `ws://${window.location.hostname}/ws`; var websocket; window.addEventListener('load', onLoad); function initWebSocket() { console.log('Trying to open a WebSocket connection...'); websocket = new WebSocket(gateway); websocket.onopen = onOpen; websocket.onclose = onClose; } function onOpen(event) { console.log('Connection opened'); } function onClose(event) { console.log('Connection closed'); setTimeout(initWebSocket, 2000); } function onLoad(event) { initWebSocket(); initButton(); } function initButton() { document.getElementById('pwr_button').addEventListener('click', pwr_function); document.getElementById('up_button').addEventListener('click', up_function); document.getElementById('down_button').addEventListener('click', down_function); document.getElementById('left_button').addEventListener('click', left_function); document.getElementById('ok_button').addEventListener('click', ok_function); document.getElementById('right_button').addEventListener('click', right_function); document.getElementById('plus_button').addEventListener('click', plus_function); document.getElementById('minus_button').addEventListener('click', minus_function); document.getElementById('next_button').addEventListener('click', next_function); document.getElementById('prev_button').addEventListener('click', prev_function); } function pwr_function(){ websocket.send('pwrbtn'); } function up_function(){ websocket.send('upbtn'); } function down_function(){ websocket.send('downbtn'); } function left_function(){ websocket.send('leftbtn'); } function ok_function(){ websocket.send('okbtn'); } function right_function(){ websocket.send('rightbtn'); } function plus_function(){ websocket.send('plusbtn'); } function minus_function(){ websocket.send('minusbtn'); } function next_function(){ websocket.send('nextbtn'); } function prev_function(){ websocket.send('prevbtn'); } </script> </head> <body> <div id="bg_mob"> <div class="mob_inner_con" id="content-wrap"> <div class="innercontainer" style="margin-top: 5%;margin-bottom: 6%;"> <button type="button" id="pwr_button" class="btn_click" onClick="" ><img src="" alt="Red dot" style="width: 38px;border: 4px solid white;border-radius: 50%;padding: 6px 8px;"></button> </div> <div> <div style="margin-top: 4%;"><button type="button" id="up_button" class="btn_click" onClick="" ><img src="" alt="Red dot" style="width: 46px;border: 4px solid white;border-radius: 50%;padding: 2px 2px;"></button></div> <div> <button type="button" id="left_button" class="btn_click" onClick="" ><img src="" alt="Red dot" style="width: 46px;border: 4px solid white;border-radius: 50%;padding: 2px 2px;margin:15px 5px"></button> <button type="button" id="ok_button" class="btn_click" onClick="" ><img src="" alt="Red dot" style="width: 46px;border: 4px solid white;border-radius: 50%;padding: 2px 2px;margin:15px 5px"></button> <button type="button" id="right_button" class="btn_click" onClick="" ><img src="" alt="Red dot" style="width: 46px;border: 4px solid white;border-radius: 50%;padding: 2px 2px;margin:15px 5px"></button> </div> <div style="margin-bottom: 4%;"><button type="button" id="down_button" class="btn_click" onClick="" ><img src="" alt="Red dot" style="width: 46px;border: 4px solid white;border-radius: 50%;padding: 2px 2px;"></button></div> </div> <div> <div class="main_bottom_div" style="max-width: 320px;margin: 0 auto;"> <div class="left_bottom_div" style="float: left;width: 50%;"> <button type="button" id="plus_button" class="btn_click" onClick="" > <img src="" alt="Red dot" style="width: 46px;border: 4px solid white;border-radius: 50%;padding: 2px 2px;margin:15px 5px"> </button> <button type="button" id="minus_button" class="btn_click" onClick="" > <img src="" alt="Red dot" style="width: 46px;border: 4px solid white;border-radius: 50%;padding: 2px 2px;margin:15px 5px"> </button> </div> <div class="left_bottom_div" style="float: left;width: 50%;"> <button type="button" id="next_button" class="btn_click" onClick="" > <img src="" alt="Red dot" style="width: 46px;border: 4px solid white;border-radius: 50%;padding: 2px 2px;margin:15px 5px"> </button><br> <button type="button" id="prev_button" class="btn_click" onClick="" > <img src="" alt="Red dot" style="width: 46px;border: 4px solid white;border-radius: 50%;padding: 2px 2px;margin:15px 5px"> </button> </div> <div class="clr"></div> </div> </div> </div> <footer id="footer"><div style="margin-top: 2%;"><img src=""></div></footer> </div> </body> </html> )rawliteral"; /*IR data structure*/ struct storedIRDataStruct { IRData receivedIRData; // extensions for sendRaw uint8_t rawCode[RAW_BUFFER_LENGTH]; // The durations if raw uint8_t rawCodeLength; // The length of the code } sStoredIRData; void storeCode(IRData *aIRReceivedData); void sendCode(storedIRDataStruct *aIRDataToSend); int mode_sw, learn_sw; int SRmode = 0; int t0size, t1size, t2size, t3size, t4size, t5size, t6size, t7size, t8size, t9size; uint8_t array0[750], array1[750], array2[750], array3[750], array4[750], array5[750], array6[750], array7[750], array8[750], array9[750]; const long touchDelay = 350; //ms int value = 0; int threshold = 40; int thresholdT1 = 40; bool touch_pwr = false; bool touch_ok = false; bool touch_up = false; bool touch_down = false; bool touch_left = false; bool touch_right = false; bool touch_next = false; bool touch_prev = false; bool touch_plus = false; bool touch_minus = false; bool boot = true; unsigned long lasttouch = 0; unsigned long lastwificheck = 0; unsigned long lastmqttcheck = 0; volatile unsigned long sinceLastTouchT0 = 0; volatile unsigned long sinceLastTouchT1 = 0; volatile unsigned long sinceLastTouchT2 = 0; volatile unsigned long sinceLastTouchT3 = 0; volatile unsigned long sinceLastTouchT4 = 0; volatile unsigned long sinceLastTouchT5 = 0; volatile unsigned long sinceLastTouchT6 = 0; volatile unsigned long sinceLastTouchT7 = 0; volatile unsigned long sinceLastTouchT8 = 0; volatile unsigned long sinceLastTouchT9 = 0; /*Touch interrupt routines*/ void IRAM_ATTR gotTouch_minus() { touch_minus = true; } void IRAM_ATTR gotTouch_plus() { touch_plus = true; } void IRAM_ATTR gotTouch_next() { touch_next = true; } void IRAM_ATTR gotTouch_prev() { touch_prev = true; } void IRAM_ATTR gotTouch_pwr() { touch_pwr = true; } void IRAM_ATTR gotTouch_left() { touch_left = true; } void IRAM_ATTR gotTouch_ok() { touch_ok = true; } void IRAM_ATTR gotTouch_right() { touch_right = true; } void IRAM_ATTR gotTouch_down() { touch_down = true; } void IRAM_ATTR gotTouch_up() { touch_up = true; } /* Function for touch debouncing*/ bool touchDelayComp(unsigned long lastTouch) { if (millis() - lastTouch < touchDelay) return false; return true; } /*Fucntion to reboot ESP32 incase of mode change*/ void modereset() { if (digitalRead(mode_switch) != mode_sw || digitalRead(learn_switch) != learn_sw) { ESP.restart(); } } void MQTT_connect() { int8_t ret; // Stop if already connected. if (mqtt.connected()) { return; } if(millis() - lastmqttcheck > 10000 || boot == true) { if(boot == true) { boot = false; Serial.println("Boot flag detected "); } Serial.print("Connecting to MQTT... "); int8_t retry = 5; while ((ret = mqtt.connect()) != 0) { // connect will return 0 for connected Serial.println(mqtt.connectErrorString(ret)); Serial.println("Retrying MQTT connection "); mqtt.disconnect(); retry--; Serial.print("."); if (retry == 0) { lastmqttcheck = millis(); Serial.println("Retry failed . will try after 10Sec "); return; } } Serial.println("MQTT Connected!"); } } /*Function to interpret and identify touch*/ void touch_detect() { if (touch_pwr) { touch_pwr = false; if (touchDelayComp(sinceLastTouchT4)) { value = value - 1; Serial.println("PWR button"); IrSender.sendRaw(array4, t4size, 38); sinceLastTouchT4 = millis(); lasttouch = millis(); } } if (touch_ok) { touch_ok = false; if (touchDelayComp(sinceLastTouchT6)) { value = value - 1; Serial.println("Ok button"); IrSender.sendRaw(array6, t6size, 38); sinceLastTouchT6 = millis(); lasttouch = millis(); } } if (touch_up) { touch_up = false; if (touchDelayComp(sinceLastTouchT9)) { value = value - 1; Serial.println("Up button"); IrSender.sendRaw(array9, t9size, 38); sinceLastTouchT9 = millis(); lasttouch = millis(); } } if (touch_down) { touch_down = false; if (touchDelayComp(sinceLastTouchT8)) { value = value - 1; Serial.println("Down button"); IrSender.sendRaw(array8, t8size, 38); sinceLastTouchT8 = millis(); lasttouch = millis(); } } if (touch_left) { touch_left = false; if (touchDelayComp(sinceLastTouchT7)) { value = value - 1; Serial.println("Left button"); IrSender.sendRaw(array7, t7size, 38); sinceLastTouchT7 = millis(); lasttouch = millis(); } } if (touch_right) { touch_right = false; if (touchDelayComp(sinceLastTouchT5)) { value = value - 1; Serial.println("Right button"); IrSender.sendRaw(array5, t5size, 38); sinceLastTouchT5 = millis(); lasttouch = millis(); } } if (touch_next) { touch_next = false; if (touchDelayComp(sinceLastTouchT2)) { value = value - 1; Serial.println("Next button"); IrSender.sendRaw(array2, t2size, 38); sinceLastTouchT2 = millis(); lasttouch = millis(); } } if (touch_prev) { touch_prev = false; if (touchDelayComp(sinceLastTouchT3)) { value = value - 1; Serial.println("Prev button"); IrSender.sendRaw(array3, t3size, 38); sinceLastTouchT3 = millis(); lasttouch = millis(); } } if (touch_plus) { touch_plus = false; if (touchDelayComp(sinceLastTouchT1)) { value = value - 1; Serial.println("Plus button"); IrSender.sendRaw(array1, t1size, 38); sinceLastTouchT1 = millis(); lasttouch = millis(); } } if (touch_minus) { touch_minus = false; if (touchDelayComp(sinceLastTouchT0)) { value = value - 1; Serial.println("Minus button"); IrSender.sendRaw(array0, t0size, 38); sinceLastTouchT0 = millis(); lasttouch = millis(); } } } /*RGB LED flashing when a signal data is saved*/ void rgbeffect() { for (int i = 0; i < 3; i++) { digitalWrite(19, LOW); digitalWrite(5, HIGH); delay(100); digitalWrite(5, LOW); digitalWrite(18, HIGH); delay(100); digitalWrite(18, LOW); digitalWrite(19, HIGH); delay(100); } digitalWrite(19, HIGH); } /*Function to save the IR sginal data in the learing mode*/ void storeCode(IRData *aIRReceivedData) { if (aIRReceivedData->flags & IRDATA_FLAGS_IS_REPEAT) { Serial.println(F("Ignore repeat")); return; } if (aIRReceivedData->flags & IRDATA_FLAGS_IS_AUTO_REPEAT) { Serial.println(F("Ignore autorepeat")); return; } if (aIRReceivedData->flags & IRDATA_FLAGS_PARITY_FAILED) { Serial.println(F("Ignore parity error")); return; } /* Copy decoded data */ sStoredIRData.receivedIRData = *aIRReceivedData; sStoredIRData.rawCodeLength = IrReceiver.decodedIRData.rawDataPtr->rawlen - 1; IrReceiver.compensateAndStoreIRResultInArray(sStoredIRData.rawCode); if (sStoredIRData.receivedIRData.protocol == UNKNOWN) { Serial.print(F("Received unknown code and store ")); Serial.print(IrReceiver.decodedIRData.rawDataPtr->rawlen - 1); Serial.println(F(" timing entries as raw ")); IrReceiver.printIRResultRawFormatted(&Serial, true); // Output the results in RAW format sStoredIRData.rawCodeLength = IrReceiver.decodedIRData.rawDataPtr->rawlen - 1; /* Store the current raw data in a dedicated array for later usage */ IrReceiver.compensateAndStoreIRResultInArray(sStoredIRData.rawCode); } else { IrReceiver.printIRResultShort(&Serial); sStoredIRData.receivedIRData.flags = 0; // clear flags -esp. repeat- for later sending Serial.println(); } Serial.println(" timing entries as raw "); IrReceiver.printIRResultRawFormatted(&Serial, true); // Output the results in RAW format Serial.println(""); for (uint8_t i = 0; i < sStoredIRData.rawCodeLength; i++) { Serial.print(sStoredIRData.rawCode[i]); Serial.print(" , "); } Serial.println(); uint8_t size1 = sizeof(sStoredIRData.rawCode); //uint8_t array2[sStoredIRData.rawCodeLength]; //memcpy( array2, sStoredIRData.rawCode, sStoredIRData.rawCodeLength ); if (touchRead(T0) < threshold) { Serial.println("Saving - key"); File file_t0 = SPIFFS.open("/0.txt", FILE_WRITE); for (int n = 0; n < sStoredIRData.rawCodeLength; n++) { file_t0.println(sStoredIRData.rawCode[n]); } file_t0.close(); rgbeffect(); } else if (touchRead(T1) < thresholdT1) { Serial.println("Saving + key"); File file_t1 = SPIFFS.open("/1.txt", FILE_WRITE); for (int n = 0; n < sStoredIRData.rawCodeLength; n++) { file_t1.println(sStoredIRData.rawCode[n]); } file_t1.close(); rgbeffect(); } else if (touchRead(T2) < threshold) { Serial.println("Saving next key"); File file_t2 = SPIFFS.open("/2.txt", FILE_WRITE); for (int n = 0; n < sStoredIRData.rawCodeLength; n++) { file_t2.println(sStoredIRData.rawCode[n]); } file_t2.close(); rgbeffect(); } else if (touchRead(T3) < threshold) { Serial.println("Saving prev key"); File file_t3 = SPIFFS.open("/3.txt", FILE_WRITE); for (int n = 0; n < sStoredIRData.rawCodeLength; n++) { file_t3.println(sStoredIRData.rawCode[n]); } file_t3.close(); rgbeffect(); } else if (touchRead(T4) < threshold) { Serial.println("Saving pwr key"); File file_t4 = SPIFFS.open("/4.txt", FILE_WRITE); for (int n = 0; n < sStoredIRData.rawCodeLength; n++) { file_t4.println(sStoredIRData.rawCode[n]); } file_t4.close(); rgbeffect(); } else if (touchRead(T5) < threshold) { Serial.println("Saving > key"); File file_t5 = SPIFFS.open("/5.txt", FILE_WRITE); for (int n = 0; n < sStoredIRData.rawCodeLength; n++) { file_t5.println(sStoredIRData.rawCode[n]); } file_t5.close(); rgbeffect(); } else if (touchRead(T6) < threshold) { Serial.println("Saving ok key"); File file_t6 = SPIFFS.open("/6.txt", FILE_WRITE); for (int n = 0; n < sStoredIRData.rawCodeLength; n++) { file_t6.println(sStoredIRData.rawCode[n]); } file_t6.close(); rgbeffect(); } else if (touchRead(T7) < threshold) { Serial.println("Saving < key"); File file_t7 = SPIFFS.open("/7.txt", FILE_WRITE); for (int n = 0; n < sStoredIRData.rawCodeLength; n++) { file_t7.println(sStoredIRData.rawCode[n]); } file_t7.close(); rgbeffect(); } else if (touchRead(T8) < thresholdT1) { Serial.println("Saving Down key"); File file_t8 = SPIFFS.open("/8.txt", FILE_WRITE); for (int n = 0; n < sStoredIRData.rawCodeLength; n++) { file_t8.println(sStoredIRData.rawCode[n]); } file_t8.close(); rgbeffect(); } else if (touchRead(T9) < threshold) { Serial.println("Saving Up key"); File file_t9 = SPIFFS.open("/9.txt", FILE_WRITE); for (int n = 0; n < sStoredIRData.rawCodeLength; n++) { file_t9.println(sStoredIRData.rawCode[n]); } file_t9.close(); rgbeffect(); } else { Serial.println("No key pressed"); return; } Serial.println("Key Code Saved"); } /*Function to handle WebSocket Messages*/ void handleWebSocketMessage(void *arg, uint8_t *data, size_t len) { AwsFrameInfo *info = (AwsFrameInfo*)arg; if (info->final && info->index == 0 && info->len == len && info->opcode == WS_TEXT) { data[len] = 0; if (strcmp((char*)data, "pwrbtn") == 0) { Serial.println("PWR button"); IrSender.sendRaw(array4, t4size, 38); } else if (strcmp((char*)data, "upbtn") == 0) { Serial.println("Up button"); IrSender.sendRaw(array9, t9size, 38); } else if (strcmp((char*)data, "downbtn") == 0) { Serial.println("Down button"); IrSender.sendRaw(array8, t8size, 38); } else if (strcmp((char*)data, "leftbtn") == 0) { Serial.println("Left button"); IrSender.sendRaw(array7, t7size, 38); } else if (strcmp((char*)data, "okbtn") == 0) { Serial.println("Ok button"); IrSender.sendRaw(array6, t6size, 38); } else if (strcmp((char*)data, "rightbtn") == 0) { Serial.println("Right button"); IrSender.sendRaw(array5, t5size, 38); } else if (strcmp((char*)data, "plusbtn") == 0) { Serial.println("Plus button"); IrSender.sendRaw(array1, t1size, 38); } else if (strcmp((char*)data, "minusbtn") == 0) { Serial.println("Minus button"); IrSender.sendRaw(array0, t0size, 38); } else if (strcmp((char*)data, "nextbtn") == 0) { Serial.println("Next button"); IrSender.sendRaw(array2, t2size, 38); } else if (strcmp((char*)data, "prevbtn") == 0) { Serial.println("Prev button"); IrSender.sendRaw(array3, t3size, 38); } } } /*Event handler Function*/ void onEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventType type, void *arg, uint8_t *data, size_t len) { switch (type) { case WS_EVT_CONNECT: Serial.printf("WebSocket client #%u connected from %s\n", client->id(), client->remoteIP().toString().c_str()); break; case WS_EVT_DISCONNECT: Serial.printf("WebSocket client #%u disconnected\n", client->id()); break; case WS_EVT_DATA: handleWebSocketMessage(arg, data, len); break; case WS_EVT_PONG: case WS_EVT_ERROR: break; } } /*Function to initialise swebsocket*/ void initWebSocket() { ws.onEvent(onEvent); server.addHandler(&ws); } void MQTT_loop() { if (WiFi.status() != WL_CONNECTED && millis()-lastwificheck > 10000) { Serial.print("Connecting to Wifi"); uint8_t errcount = 25; WiFi.reconnect(); while (WiFi.status() != WL_CONNECTED) { delay(50); Serial.print("."); errcount--; if (errcount == 0) { Serial.println("Wifi connection failed"); lastwificheck = millis(); break; } if (WiFi.status() == WL_CONNECTED) { Serial.println("WiFi connected"); Serial.println("IP address: "); Serial.println(WiFi.localIP()); } } } else if (WiFi.status() == WL_CONNECTED) { MQTT_connect(); Adafruit_MQTT_Subscribe *subscription; while ((subscription = mqtt.readSubscription(50))) { if (subscription == &SRemote) { Serial.print(F("Got: ")); Serial.println((char *)SRemote.lastread); if (String((char *)SRemote.lastread) == String("power")||String((char *)SRemote.lastread) == String("Power")) // hear we are checking for the string Big FM { Serial.println("Sending power button code "); IrSender.sendRaw(array4, t4size, 38); } else if (String((char *)SRemote.lastread) == String("ok") ||String((char *)SRemote.lastread) == String("Ok")) // hear we are checking for the string Big FM { Serial.println("Sending Ok button code "); IrSender.sendRaw(array6, t6size, 38); } else if (String((char *)SRemote.lastread) == String("left") ||String((char *)SRemote.lastread) == String("Left")) // hear we are checking for the string Big FM { Serial.println("Sending Left button code "); IrSender.sendRaw(array7, t7size, 38); } else if (String((char *)SRemote.lastread) == String("right")||String((char *)SRemote.lastread) == String("Right")) // hear we are checking for the string Big FM { Serial.println("Sending Right button code "); IrSender.sendRaw(array5, t5size, 38); } else if (String((char *)SRemote.lastread) == String("up")||String((char *)SRemote.lastread) == String("Up")) // hear we are checking for the string Big FM { Serial.println("Sending Up button code "); IrSender.sendRaw(array9, t9size, 38); } else if (String((char *)SRemote.lastread) == String("down")||String((char *)SRemote.lastread) == String("Down")) // hear we are checking for the string Big FM { Serial.println("Sending Down button code "); IrSender.sendRaw(array8, t8size, 38); } else if (String((char *)SRemote.lastread) == String("plus")||String((char *)SRemote.lastread) == String("Plus")) // hear we are checking for the string Big FM { Serial.println("Sending Plus button code "); IrSender.sendRaw(array1, t1size, 38); } else if (String((char *)SRemote.lastread) == String("minus")||String((char *)SRemote.lastread) == String("Minus")) // hear we are checking for the string Big FM { Serial.println("Sending Minus button code "); IrSender.sendRaw(array0, t0size, 38); } else if (String((char *)SRemote.lastread) == String("next")||String((char *)SRemote.lastread) == String("Next")) // hear we are checking for the string Big FM { Serial.println("Sending Next button code "); IrSender.sendRaw(array2, t2size, 38); } else if (String((char *)SRemote.lastread) == String("previous")||String((char *)SRemote.lastread) == String("Previous")) // hear we are checking for the string Big FM { Serial.println("Sending Previous button code "); IrSender.sendRaw(array3, t3size, 38); } } } } } /*Setup Function*/ void setup() { pinMode(mode_switch, INPUT); pinMode(learn_switch, INPUT); pinMode(5, OUTPUT); pinMode(18, OUTPUT); pinMode(19, OUTPUT); digitalWrite(5, LOW); digitalWrite(18, LOW); digitalWrite(19, LOW); mode_sw = digitalRead(mode_switch); learn_sw = digitalRead(learn_switch); Serial.begin(115200); delay(1000); // give me time to bring up serial monitor if (!SPIFFS.begin(true)) { Serial.println("An Error has occurred while mounting SPIFFS"); return; } Serial.println("ESP32 Touch Interrupt Test"); if (learn_sw) { SRmode = 0; for (int i = 0; i < 3; i++) { digitalWrite(19, LOW); digitalWrite(5, HIGH); delay(100); digitalWrite(5, LOW); digitalWrite(18, HIGH); delay(100); digitalWrite(18, LOW); digitalWrite(19, HIGH); delay(100); } digitalWrite(19, HIGH); // Start the receiver and if not 3. parameter specified, take LED_BUILTIN pin from the internal boards definition as default feedback LED IrReceiver.begin(IR_RECEIVE_PIN, ENABLE_LED_FEEDBACK); Serial.print(F("Ready to receive IR signals of protocols: ")); printActiveIRProtocols(&Serial); Serial.println(F("at pin " STR(IR_RECEIVE_PIN))); } else { File file0 = SPIFFS.open("/0.txt", FILE_READ); File file1 = SPIFFS.open("/1.txt", FILE_READ); File file2 = SPIFFS.open("/2.txt", FILE_READ); File file3 = SPIFFS.open("/3.txt", FILE_READ); File file4 = SPIFFS.open("/4.txt", FILE_READ); File file5 = SPIFFS.open("/5.txt", FILE_READ); File file6 = SPIFFS.open("/6.txt", FILE_READ); File file7 = SPIFFS.open("/7.txt", FILE_READ); File file8 = SPIFFS.open("/8.txt", FILE_READ); File file9 = SPIFFS.open("/9.txt", FILE_READ); String string_temp; int counter = 0; while (file0.available()) { string_temp = file0.readStringUntil('\r'); array0[counter] = string_temp.toInt(); counter++; } t0size = counter; file0.close(); string_temp = ""; counter = 0; while (file1.available()) { string_temp = file1.readStringUntil('\r'); array1[counter] = string_temp.toInt(); counter++; } t1size = counter; file1.close(); string_temp = ""; counter = 0; while (file2.available()) { string_temp = file2.readStringUntil('\r'); array2[counter] = string_temp.toInt(); counter++; } t2size = counter; file2.close(); string_temp = ""; counter = 0; while (file3.available()) { string_temp = file3.readStringUntil('\r'); array3[counter] = string_temp.toInt(); counter++; } t3size = counter; file3.close(); string_temp = ""; counter = 0; while (file4.available()) { string_temp = file4.readStringUntil('\r'); array4[counter] = string_temp.toInt(); counter++; } t4size = counter; file4.close(); string_temp = ""; counter = 0; while (file5.available()) { string_temp = file5.readStringUntil('\r'); array5[counter] = string_temp.toInt(); counter++; } t5size = counter; file5.close(); string_temp = ""; counter = 0; while (file6.available()) { string_temp = file6.readStringUntil('\r'); array6[counter] = string_temp.toInt(); counter++; } t6size = counter; file6.close(); string_temp = ""; counter = 0; while (file7.available()) { string_temp = file7.readStringUntil('\r'); array7[counter] = string_temp.toInt(); counter++; } t7size = counter; file7.close(); string_temp = ""; counter = 0; while (file8.available()) { string_temp = file8.readStringUntil('\r'); array8[counter] = string_temp.toInt(); counter++; } t8size = counter; file8.close(); string_temp = ""; counter = 0; while (file9.available()) { string_temp = file9.readStringUntil('\r'); array9[counter] = string_temp.toInt(); counter++; } t9size = counter; file9.close(); string_temp = ""; counter = 0; Serial.println("Key read from files"); touchAttachInterrupt(T0, gotTouch_minus, threshold); touchAttachInterrupt(T1, gotTouch_plus, thresholdT1); touchAttachInterrupt(T2, gotTouch_next, threshold); touchAttachInterrupt(T3, gotTouch_prev, threshold); touchAttachInterrupt(T4, gotTouch_pwr, threshold); touchAttachInterrupt(T5, gotTouch_right, threshold); touchAttachInterrupt(T6, gotTouch_ok, threshold); touchAttachInterrupt(T7, gotTouch_left, threshold); touchAttachInterrupt(T8, gotTouch_down, threshold); touchAttachInterrupt(T9, gotTouch_up, threshold); IrSender.begin(IR_SEND_PIN, ENABLE_LED_FEEDBACK); // Specify send pin and enable feedback LED at default feedback LED pin if (mode_sw) { //wifi mode Serial.println("Wifi Mode");// Start access point WiFi.mode(WIFI_STA); WiFiManager wm; if(touchRead(T0) < 30 && touchRead(T3) < 30 && touchRead(T4) < 30 ) { while(touchRead(T6) > 30) { digitalWrite(18, LOW); digitalWrite(5, HIGH); delay(100); digitalWrite(5, LOW); digitalWrite(18, HIGH); delay(100); } wm.resetSettings(); } bool res; res = wm.autoConnect("SmartRemote","circuitdigest"); // password protected ap if(!res) { Serial.println("Failed to connect"); lastwificheck = millis(); // ESP.restart(); } else { //if you get here you have connected to the WiFi Serial.println("WiFi connected..."); lastwificheck = millis(); } Serial.println(); if (WiFi.status() == WL_CONNECTED) { Serial.println("WiFi connected"); Serial.println("IP address: "); Serial.println(WiFi.localIP()); } if (!MDNS.begin("smartremote")) { Serial.println("Error setting up MDNS responder!"); } MDNS.addService("http", "tcp", 80); Serial.println("Connect using http://smartremote.local"); // Setup MQTT subscription for onoff feed. mqtt.subscribe(&SRemote); initWebSocket(); // Route for root / web page server.on("/", HTTP_GET, [](AsyncWebServerRequest *request){ request->send_P(200, "text/html", index_html); }); AsyncElegantOTA.begin(&server, "admin", "circuitdigest"); // Start server server.begin(); SRmode = 1; for (int i = 0; i < 3; i++) { digitalWrite(5, HIGH); delay(100); digitalWrite(5, LOW); delay(100); } digitalWrite(5, HIGH); } else { SRmode = 2; Serial.println("Normal Mode"); for (int i = 0; i < 3; i++) { digitalWrite(18, HIGH); delay(100); digitalWrite(18, LOW); delay(100); } digitalWrite(18, HIGH); } } lasttouch = millis(); } /*Loop Function*/ void loop() { modereset(); if (SRmode == 0) { if (IrReceiver.available()) { storeCode(IrReceiver.read()); IrReceiver.resume(); // resume receiver } } else { touch_detect(); if (SRmode == 1) { MQTT_loop(); /*Web client handling*/ ws.cleanupClients(); } else { /*In normal mode the remote will goto sleep after 20 Seconds. Adjust the sleep time by modifiying the value below*/ if(millis()- lasttouch > 20000) { esp_sleep_enable_touchpad_wakeup(); Serial.println("Going to sleep now"); esp_deep_sleep_start(); } } } }