2. Classroom Smart Air Quality Monitoring System
Let’s build an intelligent air quality monitoring system by a ENS160 sensor module and a 6812 RGB LED. When the air quality is poor, the RGB will issue an alarm, so we can create a healthy and comfortable learning environment for teachers and students!
2.1 6812 RGB LED Module
Our 6812 RGB LED module integrates four programmable RGB LEDs, supporting multiple colors and dynamic effects, so it can achieve rich lighting effects by simple signal control.

Parameters
Operating voltage: DC 3.3-5V
Maximum power: 1W
IC model: 4 pixels /SK6812
Gray scale level: 25
Emission Angle: 180°
Emission colors: Adjustable colors of white, red, yellow, blue, green, etc. through the controller
Operating temperature: -10°C to +50°C
Dimensions: 32 x 23 x 8mm
Positioning hole: Diameter of 4.8mm
Interface: 3-pin curved pin interface spacing 2.54mm
Principle
The 6812 RGB LED module adopts a single-line serial communication protocol. Each LED pixel is equipped with an internal driver IC, which transmits signals step by step through data lines, so their brightness can be independently controlled, creating dynamic light effects.
Each LED pixel contains three colors: red ®, green (G), and blue (B). Any color can be mixed by adjusting the brightness of the three.
Serial number of LED pixels:

Test Code
#include <Adafruit_NeoPixel.h>
#define LED_PIN 4 // control signal pin
#define LED_COUNT 4 // LED pixel number
// Create NeoPixel object
Adafruit_NeoPixel leds(LED_COUNT, LED_PIN, NEO_GRB + NEO_KHZ800);
void setup() {
leds.begin(); // initialize the LED strip
leds.show(); // turn off all led
leds.setBrightness(50); // set brightness(0~255)
}
void loop() {
// Set the first LED pixel to red
leds.setPixelColor(0, 255, 0, 0);
// Set the second LED pixel to green
leds.setPixelColor(1, 0, 255, 0);
// Set the third LED pixel to blue
leds.setPixelColor(2, 0, 0, 255);
// Set the forth LED pixel to white
leds.setPixelColor(3, 255, 255, 255);
// refresh
leds.show();
delay(1000);
// turn off all led
leds.clear();
leds.show();
delay(1000);
}
Code Explanation
1. Hardware and library
#include <Adafruit_NeoPixel.h>
Import Adafruit_NeoPixel library to control the SK6812 programmable RGB LED strip.
#define LED_PIN 4
Define the control signal pin as the IO4 pin of the board.
#define LED_COUNT 4
Define the number of LED pixels as 4.
Adafruit_NeoPixel leds(LED_COUNT, LED_PIN, NEO_GRB + NEO_KHZ800);
Create a NeoPixel object and name it
leds.Parameter description:
LED_COUNT: Number of LED pixels (4).LED_PIN: Control signal pin (IO4).NEO_GRB + NEO_KHZ800: Specify the color sequence (red - green - blue) and communication frequency (800kHz) of the LED pixels.
2. Initialization (setup)
leds.begin();
Initialize the LED light strip, set the pin mode and communication protocol.
leds.show();
Update the display status of the LED strip. Since no color is set at this time, all the LED pixels will be turned off.
leds.setBrightness(50);
Set the brightness of the LED pixels within the range of 0 (darkest) to 255 (brightest).
3. Main Loop (loop)
leds.setPixelColor(index, red, green, blue);
Set the color of the lamp pixels with the specified serial number.
Parameter description:
index: The serial number of the lamp pixels (starting from 0)red: Parameter of red (0~255)green: Parameter of green (0~255)blue: Parameter of blue (0~255)
leds.show();
Update the display status of the LED strip to light up the LED in the set color.
delay(1000);
Delay for 1 second to maintain the current state of the pixels.
leds.clear();
Turn off all the LED pixels (set the color of all LED pixels to 0).
leds.show();
Update the display status and turn off all the LED pixels.
delay(1000);
Delay for 1 second to keep the LED pixels off.
Test Result
After uploading the code, you can see a loop:
Four LEDs will light up in sequence as red, green, blue and white for one second. Due to its changing speed, human eyes may not be able to see the process of them lighting up one by one.
Then all the LED pixels turn off for 1 second.
==动图==
2.2 ENS160 Air Quality Sensor
ENS160 is a high-precision digital air quality sensor capable of real-time detection of volatile organic compounds (VOCs), carbon dioxide (CO₂), and air quality index (IAQ) in the air. It is widely used in environment monitoring, smart homes, and health devices.

Parameters
Operating voltage: DC 3.3-5V
Operating current: 29 mA
Preheating time: <3 minutes
Communication interface: I2C
I2C address: 0x53 by default, can be modified to 0x52
Operating temperature: -10°C to +50°C
Operating humidity: 5% RH to 95% RH
eCO₂ measurement range: 400 ppm~65000 ppm
TVOC measurement range: 0 ppb~65000 ppb
Dimensions: 32 x 23 x 8mm
Positioning hole: Diameter of 3.2mm
Interface: 4-pin curved pin interface spacing 2.54mm
Principle
Working Principle of ENS160
The ENS160 detects the gas composition in the air through its built-in metal oxide semiconductor sensor. Its working principle is as follows:
VOCs detection:
Detection principle
The metal oxide on the sensor surface undergoes a chemical reaction with VOCs in the air, causing a change in its resistance. The concentration of VOCs can be detected by measuring the resistance change.
However, it cannot distinguish exactly which type of VOCs it is. Instead, it converts all detectable VOCs mixed signals into total volatile organic compounds(TVOC) concentrations (in ppb) for output.
Therefore, ENS160 directly provides TVOC data rather than individual VOCs components, which is suitable for quickly assessing the overall air pollution level and cannot precisely analyze specific gas types.
TVOC detection range
Output unit: ppb(parts per billion)
Detection range: 0~65,000 ppb (The actual effective accuracy is usually 0~10,000 ppb)
TVOC level reference
TVOC concentration (ppb)
Level
Health impacts and suggestions
0-250
Great
The air quality is excellent and has no impact on the human body, and is suitable for a long stay
250-500
Good
The air quality is good, but sensitive people may experience mild discomfort. It is recommended to ventilate moderately
500-1000
Mild pollution
The air quality is acceptable. Long-term exposure may cause headaches and fatigue, so ventilation needs to be enhanced
1000-3000
Middle level pollution
Poor air quality may cause eye and throat irritation and drowsiness. It is recommended to reduce exposure and investigate pollution sources
>3000
Severe pollution
Severe pollution may cause damage to the nervous system and liver and kidney toxicity. Immediate ventilation or evacuation is necessary
CO₂ estimation:
Estimation principle
ENS160 cannot directly measure CO₂. Instead, it estimates eCO₂(equivalent carbon dioxide) by detecting the concentration of VOCs(volatile organic compounds) in the air with a built-in algorithm. eCO₂ is an indirect approximation of CO₂. It is suitable for trend monitoring, yet its accuracy is not as good as that of professional CO₂ sensors.
eCO₂ output range
Unit: ppm(parts per million)
Detection range: 400~65,000 ppm (The actual effective accuracy is usually 400~5,000 ppm)
eCO₂ level reference
eCO₂
Level
Suggestions
400-600
Great
The air quality is excellent with good ventilation.
It is suitable for a long stay600-800
Good
The air quality is acceptable.
It is recommended to ventilate moderately800-1000
Mild pollution
Air quality has declined and ventilation needs to be enhanced
1000-1500
Severe pollution
There is a distinct sense of oppression and turbidity.
It is recommended to reduce the density of people or enforce ventilation>1500
Dangerous
It is a severely oxygen-deficient environment.
Ventilation or evacuation must be carried out immediately
IAQ calculation:
Calculation principle
Based on the concentrations of TVOC and eCO₂, calculate the indoor air quality Index (IAQ) by the built-in algorithm.
AQI level reference
AQI
AQI level
Health impact
1
Great
The air is fresh and there are no health risks
2
Good
It’s basically safe here. Sensitive people may feel mild discomfort
3
Middle level
An unpleasant smell can be detected, which may cause headaches and fatigue
4
Bad
It obviously irritates the eyes and throat, and long-term exposure is harmful to health
5
Extremely bad
It causes intense discomfort and has a high risk of acute health problems
ENS160 control principle
The control principle of ENS160 is based on the I²C communication protocol, which controls the sensor and reads the measurement results by sending and receiving specific commands and data.
I²C communication address
I²C communication address is 0x53 by default.
If you need to change it to 0x52, solder a 0603 package 0Ω resistor at 0X52 empty pad of the module, and modify the device address in the code at the same time.

Test Code
#include <DFRobot_ENS160.h>
#include <Wire.h>
// For I2C communication, the default address is 0x53
DFRobot_ENS160_I2C ENS160(&Wire, 0x53);
void setup(void)
{
Serial.begin(115200);
// Initialize the sensor
while(NO_ERR != ENS160.begin()) {
Serial.println("Communication with device failed, please check connection");
delay(3000);
}
Serial.println("Begin ok!");
// Set the power mode to standard gas sensing mode
ENS160.setPWRMode(ENS160_STANDARD_MODE);
// Set the ambient temperature and humidity for calibration (sample: 25°C, 50%RH)
ENS160.setTempAndHum(25.0, 50.0);
}
void loop()
{
// Obtain the working status of the sensor
uint8_t Status = ENS160.getENS160Status();
Serial.print("Sensor operating status: ");
Serial.println(Status);
// Obtain the air quality index(1-5)
uint8_t AQI = ENS160.getAQI();
Serial.print("Air quality index: ");
Serial.println(AQI);
// Obtain the TVOC concentration(0-65000 ppb)
uint16_t TVOC = ENS160.getTVOC();
Serial.print("TVOC concentration: ");
Serial.print(TVOC);
Serial.println(" ppb");
// Obtain the equivalent CO2 concentration(400-65000 ppm)
uint16_t ECO2 = ENS160.getECO2();
Serial.print("eCO2 concentration: ");
Serial.print(ECO2);
Serial.println(" ppm");
Serial.println();
delay(1000);
}
Code Explanation
1. Header file import and library initialization
#include <DFRobot_ENS160.h>
#include <Wire.h>
DFRobot_ENS160_I2C ENS160(&Wire, 0x53);
DFRobot_ENS160_I2C ENS160(&Wire, 0x53): Create an ENS160 sensor objectUse I2C communication (&Wire)
The sensor address is 0x53 (default address)
2. Initialization (setup)
void setup(void) {
Serial.begin(115200);
// Initialize the sensor
while(NO_ERR != ENS160.begin()) {
Serial.println("Communication with device failed, please check connection");
delay(3000);
}
Serial.println("Begin ok!");
ENS160.setPWRMode(ENS160_STANDARD_MODE);
ENS160.setTempAndHum(25.0, 50.0);
}
Initialization of sensor:
ENS160.begin(): Try to initialize the sensorKeep trying using the while loop until the initialization is successful
Wait for 3 seconds after each failure (
delay(3000))
Working mode:
setPWRMode(ENS160_STANDARD_MODE): Set the sensor to the standard working mode
Environment parameters:
setTempAndHum(25.0, 50.0): Provide ambient temperature and humidity for sensor calibration and compensation
3. Main Loop (loop)
void loop() {
// Obtain the working status of the sensor
uint8_t Status = ENS160.getENS160Status();
Serial.print("Sensor operating status: ");
Serial.println(Status);
// Obtain the air quality index(1-5)
uint8_t AQI = ENS160.getAQI();
Serial.print("Air quality index: ");
Serial.println(AQI);
// Obtain the TVOC concentration(0-65000 ppb)
uint16_t TVOC = ENS160.getTVOC();
Serial.print("TVOC concentration: ");
Serial.print(TVOC);
Serial.println(" ppb");
// Obtain the equivalent CO2 concentration(400-65000 ppm)
uint16_t ECO2 = ENS160.getECO2();
Serial.print("eCO2 concentration: ");
Serial.print(ECO2);
Serial.println(" ppm");
Serial.println();
delay(1000);
}
First, obtain the working status of the sensor by
getENS160Status()to ensure the normal operation of the deviceThen read the three key parameters in sequence:
getAQI()obtains the comprehensive IAQ of levels 1 to 5getTVOC()detects the concentration of VOCsgetECO2()measures the eCO2 valueFinally, all the data are displayed on the serial monitor and the frequency of data collection per second is controlled by
delay(1000), forming a complete air quality monitoring cycle.
Test Result
After uploading the code, open the serial monitor and set the baud rate to 115200.
The sensor needs to be preheated for 1 ~2 minutes after being powered on for the first time for stabilization. After that, the values are printed on the serial monitor.

2.3 Smart Air Quality Monitoring System
In the previous sections, we have already mastered the color control principle of the 6812 RGB module and the data acquisition method of the ENS160 air quality sensor. Now, let’s make an smart air quality monitoring system for our classroom!
Here we will implement an intelligent device capable of monitoring the eCO₂ concentration in the classroom in real time. When the air quality deteriorates (eCO₂ concentration is too high), the system will trigger a red alert on the RGB to remind that teachers and students need to exchange air in time to ensure a healthy learning environment.
Let’s build this practical and technological monitoring system together!
Code Flow
graph TD
A[START] --> B[Initialize ENS160 and LED strip]
B --> C[Detect eCO2 concentration]
C --> D{Determine the concentration}
D -->|<=800ppm| E[Green always on]
D -->|800-1000ppm| F[Blue always on]
D -->|1000-1500ppm| G[Yellow always on]
D -->|>1500ppm| H[Red always on]
E --> I[Delay 2s]
F --> I
G --> I
H --> I
I --> C
Test Code
#include <DFRobot_ENS160.h>
#include <Adafruit_NeoPixel.h>
// Hardware configuration
#define LED_PIN 4 // RGB control pin
#define LED_COUNT 4 // LED pixel number
#define I2C_ADDR 0x53 // ENS160 default I2C address
// eCO2 concentration level threshold (ppm)
#define ECO2_EXCELLENT 400
#define ECO2_GOOD 800
#define ECO2_MODERATE 1000
#define ECO2_POOR 1500
// Create an object
DFRobot_ENS160_I2C ENS160(&Wire, I2C_ADDR);
Adafruit_NeoPixel leds(LED_COUNT, LED_PIN, NEO_GRB + NEO_KHZ800);
void setup() {
Serial.begin(115200);
// Initialize ENS160
while (NO_ERR != ENS160.begin()) {
Serial.println("Communication with device failed, please check connection");
delay(3000);
}
Serial.println("Begin ok!");
// Set the sensor mode
ENS160.setPWRMode(ENS160_STANDARD_MODE);
ENS160.setTempAndHum(25.0, 50.0); // Set calibration temperature and humidity
// Initialize the RGB strip
leds.begin();
leds.show(); // Initially clear the light strip
leds.setBrightness(100); // Medium brightness
}
void loop() {
// Obtain the eCO2 concentration
uint16_t eco2 = ENS160.getECO2();
Serial.print("eCO2 concentration: ");
Serial.print(eco2);
Serial.println(" ppm");
// Control leds based on air quality
updateAirQualityLED(eco2);
delay(2000); // Updated every two seconds
}
// Air quality visualization function
void updateAirQualityLED(uint16_t eco2) {
if (eco2 <= ECO2_GOOD) {
// Great/Good (Green always on)
setAllLEDs(0, 255, 0);
}
else if (eco2 <= ECO2_MODERATE) {
// Normal (blue always on)
setAllLEDs(0, 0, 255);
}
else if (eco2 <= ECO2_POOR) {
// bad(yellow always on)
setAllLEDs(255, 255, 0);
}
else {
// Severe(red always on)
setAllLEDs(255, 0, 0);
}
}
// Set all LED colors
void setAllLEDs(uint8_t r, uint8_t g, uint8_t b) {
for (int i = 0; i < LED_COUNT; i++) {
leds.setPixelColor(i, leds.Color(r, g, b));
}
leds.show();
}
Code Explanation
1. Hardware configuration and object initialization
Hardware and parameters
// Hardware configuration
#define LED_PIN 4 // RGB control pin
#define LED_COUNT 4 // LED pixel number
#define I2C_ADDR 0x53 // ENS160 default I2C address
// eCO2 concentration level threshold (ppm)
#define ECO2_EXCELLENT 400
#define ECO2_GOOD 800
#define ECO2_MODERATE 1000
#define ECO2_POOR 1500
Define the hardware connection method and key parameters
Set four eCO2 concentration threshold levels
Object initialization
DFRobot_ENS160_I2C ENS160(&Wire, I2C_ADDR);
Adafruit_NeoPixel leds(LED_COUNT, LED_PIN, NEO_GRB + NEO_KHZ800);
Create an ENS160 sensor object and initialize the NeoPixel LED strip
2. Initialization (setup)
void setup() {
Serial.begin(115200);
// Initialize ENS160
while (NO_ERR != ENS160.begin()) {
Serial.println("Communication with device failed, please check connection");
delay(3000);
}
Serial.println("Begin ok!");
// Set the sensor mode
ENS160.setPWRMode(ENS160_STANDARD_MODE);
ENS160.setTempAndHum(25.0, 50.0); // Set calibration temperature and humidity
// Initialize the RGB strip
leds.begin();
leds.show(); // Initially clear the light strip
leds.setBrightness(100); // Medium brightness
}
Complete all hardware initialization
Set the working mode and calibration parameters of the sensor
Configure the RGB LED module
3. Main Loop (loop)
eCO₂ concentration reading
uint16_t eco2 = ENS160.getECO2();
Read the eCO₂ concentration value from the ENS160 sensor
Output monitoring data on serial port
Serial.print("eCO2 concentration: ");
Serial.print(eco2);
Serial.println(" ppm");
Output the concentration value through the serial port
LED visualization control
updateAirQualityLED(eco2);
Call the LED display function to update the status and control the LED display status based on the eCO₂ concentration value
Loop delay control
delay(2000); // Updated every two seconds
Avoid frequent reading of the sensor to prevent unstable data and ensure that the LED effect has sufficient time to display
4. Core function
LED visualization function
void updateAirQualityLED(uint16_t eco2) {
if (eco2 <= ECO2_GOOD) {
// Great/Good (Green always on)
setAllLEDs(0, 255, 0);
}
else if (eco2 <= ECO2_MODERATE) {
// Normal (blue always on)
setAllLEDs(0, 0, 255);
}
else if (eco2 <= ECO2_POOR) {
// bad(yellow always on)
setAllLEDs(255, 255, 0);
}
else {
// Severe(red always on)
setAllLEDs(255, 0, 0);
}
}
Select different display effects based on eCO₂ concentration. Four display modes correspond to four air quality levels:
Excellent/Good (Green always on)
Normal (blue always on)
Bad (yellow always on)
Extremely bad (red always on)
Set all LED color functions
void setAllLEDs(uint8_t r, uint8_t g, uint8_t b) {
for (int i = 0; i < LED_COUNT; i++) {
leds.setPixelColor(i, leds.Color(r, g, b));
}
leds.show();
}
Test Result
After uploading the code, the ENS160 sensor reads the eCO₂ concentration value every two seconds.
According to the eCO₂ concentration value, it is divided into four levels, and the current air quality can be visually displayed through RGB LED:
Excellent/Good(≤800ppm): Green always on
Normal(≤1000ppm): blue always on
Bad(≤1500ppm): yellow always on
Extremely bad(>1500ppm): red always on
You can simulate changes in air quality by taking a deep breath and then exhaling slowly onto the ENS160 sensor. At this time, the eCO₂ concentration value will rise, and you can observe the color of the RGB LED changing accordingly.

==动图==