制造商零件编号 C008
ATOM LITE ESP32 DEV KIT
M5Stack Technology Co., Ltd.
所订产品一般在 7-10个工作日 内送达中国,具体时间取决于收货地点。
最低订购金额为人民币 300 元,顺丰快递免运费配送。
当用人民币下单时,按照国际贸易条款 DDP(DigiKey 支付关税、海关费用和当地税款)方式结算。
电汇预付
更多来自全授权合作伙伴的产品
下单后,从合作伙伴发货平均需要时间 1-3 天,也可能产生额外运费。可能另外收取运费。 实际发货时间请留意产品详情页、购物车和结账页面上的说明。
国际贸易结算方式:CPT(交货时支付关税、海关费用和适用 VAT/应付税金)
有关详情,请访问帮助和支持
License: General Public License Environmental Wifi Arduino ESP32 M5Stack
* Thanks for the source code and project information provided by @Hans-Günther Nusseck
Things used in this project
M5Stack ATOM Lite ESP32 Development Kit×1
M5Stack ENV III Unit×1
Software apps and online services
Web-based Data Monitor?
The ATOM Lite ESP32 IoT Development Kit from M5Stack is a very small development board that can be used out of the box and can be powered via the USB-C interface. External sensors can be easily connected via the Grove connector. Sounds perfect, but one thing is missing: A display! For example, if I connect an environmental sensor to measure the temperature, air pressure and humidity, then I want to see the values. Otherwise, it wouldn't make sense, would it?
And let's also say that I want to use the sensor in different places in my house. And I want to read the values when I sit at my PC. That sounds exactly like a job for a wireless web-based solution.
Sensor
If you want to display measured values of a sensor, then of course you need a sensor. The ENV III unit from M5Stack has two sensors in one unit:
The sensor unit can be connected directly to the ATOM device via the Grove connector. The values can then be read via I2C with the help of the corresponding libraries.
The M5ATOM lite device can be powered with a power bank via the USB-C port. But beware: Many power banks turn themselves off when not enough power is drawn. And this setup consumes very little power!
Simple and Clean Approach
To keep things as simple as possible, I don't want to use an external IOT platform to send the data to, I want to pull the data directly from the device via a web browser. This is not as difficult as it sounds, because there is a good web server library available for the ESP32. The basic framework for the program looks like this:
// Start TCP/IP-Server
server.begin();
}
void loop() {
// check for incoming clients
WiFiClient client = server.available();
if (client) {
// loop while the client's connected
while (client.connected()) {
// if there's bytes to read from the client,
if (client.available()) {
// MAGIC HAPPENs HERE
// The client sends a request
// an empty line is indicating the end
// of the client HTTP request
// I should then send the content...
}
}
}
}
The WiFi Server is configured to port 80, which is the port for HTTP. After the connection to my home WiFi is established in the setup() function, the server is started. Now, the server is listening on port 80 for Incoming requests. And this is where the fun begins, because when I connect myself to the IP address of the ATOM device via a web browser, an HTTP request is received:
New Client.
GET / HTTP/1.1
Host: 192.168.178.49
Connection: keep-alive
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.45 Safari/537.36
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9
Accept-Encoding: gzip, deflate
Accept-Language: de-DE,de;q=0.9,en-US;q=0.8,en;q=0.7
That's more information than I need. In fact, I only need the line with the "GET", because it says what the client wants to retrieve from the server. Therefore, the code first consists of going through the client's request line by line and searching for the line that starts with a "GET".
HTTP-Requests
If you look at the requests in the serial monitor, you can see that the client is not only requesting a request for the home page "/", but also wants to load the favicon "/favicon.ico":
The first GET request signals that the client, in this case my web browser, would like to see the web server's home page. I requested this with: http://192.168.178.49/
The second GET request is something the web browser always does automatically: it asks for the favicon. This is an icon that is displayed in the web browser tab for the web page. If that is not there, then the default icon is used. With my own favicon the tab of the web browser then looks like this:
A favicon can be created online from an image file. For example, here. You will get a file with the name "favicon.ico". This file must then be sent to the client when it asks for it. So, we have to put this file on the ATOM device to be able to send it to the client. One method is to store the file as a char array in the PROGMEM. The PROGMEM is the Flash memory (program space), where the compiled program itself is stored. The only thing you have to do is to convert the binary image file into a char array header file.
Binary Files to char Arrays conversion
Fortunately, there's an app for that. For Windows, there is a simple command line tool. And James Swineson has written a Python script that can be used to convert binary data into char arrays. Both are well documented, so it should not be a problem to generate the arrays. Simply create a header file with the definition for the variable:
#include <pgmspace.h> // PROGMEM support header
// Image is stored in this array
PROGMEM const char electric_favicon[] = {
... paste converted binary data here ...
};
Then all you need to do is to paste the binary file converted as hex values in between:
#include <pgmspace.h> // PROGMEM support header
// Image is stored in this array
PROGMEM const char electric_favicon[] = {
0x00, 0x00, 0x01, 0x00, 0x03, 0x00, 0x30, 0x30, 0x00, 0x00, 0x01, 0x00, 0x08,
0x00, 0xa8, 0x0e, 0x00, 0x00, 0x36, 0x00, 0x00, 0x00, 0x20, 0x20, 0x00, 0x00,
.....
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
};
Now this new header file only needs to be included and the favicon is available as a char array. If the client then requests the favicon, it can be sent to the client via this command:
case GET_favicon: {
client.println("HTTP/1.1 200 OK");
client.println("Content-type:image/x-icon");
client.println();
client.write_P(electric_favicon, sizeof(electric_favicon));
break;
}
HTTP response headers always start with a response code (e.g. HTTP/1.1 200 OK) and a content-type so the client knows what's coming, then a blank line, followed by the content. At the end, the HTTP response ends with another blank line. How easy is that?
Client Connection Timeout
This method for a web server follows the classic Arduino example. It is simple and slim but has a few disadvantages. One drawback is that the "while (client.connected())"-loop loops until one of the two (the client or the server) closes the connection or gives up. If something doesn't go according to plan in the transfer, then the program hangs at that point. That's why I added a timeout that hits after 1 second:
// force a disconnect after 1 second
unsigned long timeout_millis = millis()+1000;
...
while (client.connected()) {
// if the client is still connected after 1 second,
// something is wrong. So kill the connection
if(millis() > timeout_millis){
Serial.println("Force Client stop!");
client.stop();
}
On a call where nothing goes wrong, the client terminates the request with an empty line (two newline characters in a row). Having received that, I send the content back and signal the end with an empty line as well. This is super fast, so I can assume that if the loop is active for more than a second, that something is wrong.
Dynamic Web Page
I created a simple web page for the presentation of the data from the environmental sensor. The page should look nice and therefore I also included a logo. The logo must be included as a char array, as described above for the favicon, so that it can be sent to the client. It will ask for it with a separate GET request. The website is programmed in plain HTML. Since it is just text, you could send it to the client directly from the code, but then the code becomes cluttered. One way to include the HTML file as a char array is to convert it just like the images with the tool. Then the code stays clean and readable.
But on the web page the current values of the sensor should be displayed. To realize this without having to change the source code of the web page (index.html), I have outsourced this to a JavaScript. Each column of the table, in which one of the values should be placed, has its own ID:
<table style="float: center;" cellspacing="10">
<tbody id="DataFont">
<tr>
<td style="text-align: right;">Temperature:</td>
<td id="temperatureOutput" style="letter-spacing: 0px;"></td>
</tr>
<tr>
<td style="text-align: right;">Humidity:</td>
<td id="humidityOutput"></td>
</tr>
<tr>
<td style="text-align: right;">Air Pressure:</td>
<td id="pressureOutput"></td>
</tr>
</tbody>
</table>
In the HTML code there is nothing in the field yet. This is done by a JavaScript after the pages is loaded:
<script>
window.onload = function(){
document.getElementById('temperatureOutput').innerHTML = temperatureValue+"°C";
document.getElementById('humidityOutput').innerHTML = humidityValue+"%";
document.getElementById('pressureOutput').innerHTML = pressureValue+"hPa";
};
</script>
The only missing things are the values that should be inserted. In detail: The three variables whose values the JavaScript wants to insert. I have outsourced them to an external JavaScript:
<script type="text/javascript" src="data.js"></script>
And you may already guess: the client will request this with its own GET request. So, in the program code I only need to return the JavaScript code that contains the variable definitions:
case GET_script: {
client.println("HTTP/1.1 200 OK");
client.println("Content-type:application/javascript");
client.println();
client.printf("var temperatureValue = %3.2f;\n", sht30_Temperature);
client.printf("var humidityValue = %3.2f;", sht30_Humidity);
client.printf("var pressureValue = %3.2f;", qmp_Pressure/100.0F);
break;
}
And that's it. A web-based data monitor without bells and whistles.
I hope you like this short example and this code can prove to be useful to some of you.
Enjoy!
Code
main.cpp C/C++
main code for the M5ATOM Lite
谢谢!
敬请关注收件箱中的 DigiKey 新闻与更新!
请输入电子邮件地址
/******************************************************************************
* M5ATOM ENV web-monitor
* A simple web server to display the environment sensor data as a web page.
*
* Hague Nusseck @ electricidea
* v1.0 | 26.November.2021
* https://github.com/electricidea/M5ATOM/tree/master/ATOM-Web-Monitor
*
*
* used Resources:
* https://github.com/espressif/arduino-esp32/blob/master/libraries/WiFi/examples/SimpleWiFiServer/SimpleWiFiServer.ino
* https://realfavicongenerator.net/
* https://github.com/Jamesits/bin2array
* https://www.deadnode.org/sw/bin2h/
* https://html-css-js.com/css/generator/font/
*
* Distributed as-is; no warranty is given.
******************************************************************************/
#include <Arduino.h>
#include "M5Atom.h"
// install the M5ATOM library:
// pio lib install "M5Atom"
// You also need the FastLED library
// https://platformio.org/lib/show/126/FastLED
// FastLED is a library for programming addressable rgb led strips
// (APA102/Dotstar, WS2812/Neopixel, LPD8806, and a dozen others)
// acting both as a driver and as a library for color management and fast math.
// install the library:
// pio lib install "fastled/FastLED"
#define LED_ERROR 0x110000
#define LED_OK 0x001100
#define LED_NETWORK 0x000011
#define LED_MEASURE 0x111111
unsigned long next_millis;
#include "UNIT_ENV.h"
// ENVIII:
// SHT30: temperature and humidity sensor I2C: 0x44
// QMP6988: absolute air pressure sensor I2C: 0x70
SHT3X sht30;
QMP6988 qmp6988;
float qmp_Pressure = 0.0;
float sht30_Temperature = 0.0;
float sht30_Humidity = 0.0;
int n_average = 1;
// WIFI and https client librarys:
#include "WiFi.h"
#include <WiFiClientSecure.h>
// WiFi network configuration:
char wifi_ssid[33];
char wifi_key[65];
const char* ssid = "YourWiFi";
const char* password = "YourPassword";
WiFiClient myclient;
WiFiServer server(80);
// GET
#define GET_unknown 0
#define GET_index_page 1
#define GET_favicon 2
#define GET_logo 3
#define GET_script 4
int html_get_request;
#include "index.h"
#include "electric_logo.h"
#include "favicon.h"
// forward declarations:
void I2Cscan();
boolean connect_Wifi();
void setup() {
// start the ATOM device with Serial and Display (one LED)
// begin(SerialEnable, I2CEnable, DisplayEnable)
M5.begin(true, false, true);
// Wire.begin() must be called after M5.begin()
// Atom Matrix I2C GPIO Pin is 26 and 32
Wire.begin(26, 32);
delay(50);
// set LED to red
M5.dis.fillpix(LED_ERROR);
Serial.println("M5ATOM ENV montitor");
Serial.println("v1.0 | 26.11.2021");
// scan for I2C devices
I2Cscan();
// Set WiFi to station mode and disconnect
// from an AP if it was previously connected
WiFi.mode(WIFI_STA);
WiFi.disconnect();
delay(1000);
// connect to the configured AP
connect_Wifi();
// Start TCP/IP-Server
server.begin();
if(qmp6988.init()==1){
Serial.println("[OK] QMP6988 ready");
} else {
Serial.println("[ERR] QMP6988 not ready");
}
// high precission mode
qmp6988.setFilter(QMP6988_FILTERCOEFF_32);
qmp6988.setOversamplingP(QMP6988_OVERSAMPLING_32X);
qmp6988.setOversamplingT(QMP6988_OVERSAMPLING_4X);
next_millis = millis() + 1000;
}
void loop() {
// get actual time in miliseconds
unsigned long current_millis = millis();
// check if next measure interval is reached
if(current_millis > next_millis){
Serial.println("Measure");
next_millis = current_millis + 3000;
M5.dis.fillpix(LED_MEASURE);
if (sht30.get() != 0) {
return;
}
Serial.println(qmp6988.calcPressure());
Serial.println(sht30.cTemp);
Serial.println(sht30.humidity);
// calculate running average
qmp_Pressure = ((qmp_Pressure*(n_average-1)) + qmp6988.calcPressure())/n_average;
sht30_Temperature = ((sht30_Temperature*(n_average-1)) + sht30.cTemp)/n_average;
sht30_Humidity = ((sht30_Humidity*(n_average-1)) + sht30.humidity)/n_average;
if(n_average < 10)
n_average++;
}
// check if WIFI is still connected
// if the WIFI is not connected (anymore)
// a reconnect is triggert
wl_status_t wifi_Status = WiFi.status();
if(wifi_Status != WL_CONNECTED){
// set LED to red
M5.dis.fillpix(LED_ERROR);
// reconnect if the connection get lost
Serial.println("[ERR] Lost WiFi connection, reconnecting...");
if(connect_Wifi()){
Serial.println("[OK] WiFi reconnected");
} else {
Serial.println("[ERR] unable to reconnect");
}
}
// check if WIFI is connected
// needed because of the above mentioned reconnection attempt
wifi_Status = WiFi.status();
if(wifi_Status == WL_CONNECTED){
// set LED to green
M5.dis.fillpix(LED_OK);
}
// check for incoming clients
WiFiClient client = server.available();
if (client) {
// force a disconnect after 1 second
unsigned long timeout_millis = millis()+1000;
// set LED to blue
M5.dis.fillpix(LED_NETWORK);
Serial.println("New Client.");
// a String to hold incoming data from the client line by line
String currentLine = "";
// loop while the client's connected
while (client.connected()) {
// if the client is still connected after 1 second,
// something is wrong. So kill the connection
if(millis() > timeout_millis){
Serial.println("Force Client stop!");
client.stop();
}
// if there's bytes to read from the client,
if (client.available()) {
char c = client.read();
Serial.write(c);
// if the byte is a newline character
if (c == '\n') {
// two newline characters in a row (empty line) are indicating
// the end of the client HTTP request, so send a response:
if (currentLine.length() == 0) {
// HTTP headers always start with a response code (e.g. HTTP/1.1 200 OK)
// and a content-type so the client knows what's coming, then a blank line,
// followed by the content:
switch (html_get_request)
{
case GET_index_page: {
client.println("HTTP/1.1 200 OK");
client.println("Content-type:text/html");
client.println();
client.write_P(index_html, sizeof(index_html));
break;
}
case GET_favicon: {
client.println("HTTP/1.1 200 OK");
client.println("Content-type:image/x-icon");
client.println();
client.write_P(electric_favicon, sizeof(electric_favicon));
break;
}
case GET_logo: {
client.println("HTTP/1.1 200 OK");
client.println("Content-type:image/jpeg");
client.println();
client.write_P(electric_logo, sizeof(electric_logo));
break;
}
case GET_script: {
client.println("HTTP/1.1 200 OK");
client.println("Content-type:application/javascript");
client.println();
client.printf("var temperatureValue = %3.2f;\n", sht30_Temperature);
client.printf("var humidityValue = %3.2f;", sht30_Humidity);
client.printf("var pressureValue = %3.2f;", qmp_Pressure/100.0F);
break;
}
default:
client.println("HTTP/1.1 404 Not Found");
client.println("Content-type:text/html");
client.println();
client.print("404 Page not found.<br>");
break;
}
// The HTTP response ends with another blank line:
client.println();
// break out of the while loop:
break;
} else { // if a newline is found
// Analyze the currentLine:
// detect the specific GET requests:
if(currentLine.startsWith("GET /")){
html_get_request = GET_unknown;
// if no specific target is requested
if(currentLine.startsWith("GET / ")){
html_get_request = GET_index_page;
}
// if the logo image is requested
if(currentLine.startsWith("GET /electric-idea_100x100.jpg")){
html_get_request = GET_logo;
}
// if the favicon icon is requested
if(currentLine.startsWith("GET /favicon.ico")){
html_get_request = GET_favicon;
}
// if the dynamic script part is requested
if(currentLine.startsWith("GET /data.js")){
html_get_request = GET_script;
}
}
currentLine = "";
}
} else if (c != '\r') {
// add anything else than a carriage return
// character to the currentLine
currentLine += c;
}
}
}
// close the connection:
client.stop();
Serial.println("Client Disconnected.");
}
}
//==============================================================
void I2Cscan(){
// scan for i2c devices
byte error, address;
int nDevices;
Serial.println("Scanning I2C bus...\n");
nDevices = 0;
for(address = 1; address < 127; address++ ){
// The i2c_scanner uses the return value of
// the Write.endTransmisstion to see if
// a device did acknowledge to the address.
Wire.beginTransmission(address);
error = Wire.endTransmission();
// Errors:
// 0 : Success
// 1 : Data too long
// 2 : NACK on transmit of address
// 3 : NACK on transmit of data
// 4 : Other error
if (error == 0){
nDevices++;
Serial.printf("[OK] %i 0x%.2X\n", nDevices, address);
} else{
if(error == 4)
Serial.printf("[ERR] %i 0x%.2X\n", nDevices, address);
}
}
Serial.printf("\n%i devices found\n\n", nDevices);
}
// =============================================================
// connect_Wifi()
// connect to configured Wifi Access point
// returns true if the connection was successful otherwise false
// =============================================================
boolean connect_Wifi(){
// Establish connection to the specified network until success.
// Important to disconnect in case that there is a valid connection
WiFi.disconnect();
Serial.print("Connecting to ");
Serial.println(ssid);
delay(1500);
//Start connecting (done by the ESP in the background)
WiFi.begin(ssid, password);
// read wifi Status
wl_status_t wifi_Status = WiFi.status();
int n_trials = 0;
// loop while Wifi is not connected
// run only for 20 trials.
while (wifi_Status != WL_CONNECTED && n_trials < 20) {
// Check periodicaly the connection status using WiFi.status()
// Keep checking until ESP has successfuly connected
wifi_Status = WiFi.status();
n_trials++;
switch(wifi_Status){
case WL_NO_SSID_AVAIL:
Serial.println("[ERR] WIFI SSID not available");
break;
case WL_CONNECT_FAILED:
Serial.println("[ERR] WIFI Connection failed");
break;
case WL_CONNECTION_LOST:
Serial.println("[ERR] WIFI Connection lost");
break;
case WL_DISCONNECTED:
Serial.println("[STATE] WiFi disconnected");
break;
case WL_IDLE_STATUS:
Serial.println("[STATE] WiFi idle status");
break;
case WL_SCAN_COMPLETED:
Serial.println("[OK] WiFi scan completed");
break;
case WL_CONNECTED:
Serial.println("[OK] WiFi connected");
break;
default:
Serial.println("[ERR] WIFI unknown Status");
break;
}
delay(500);
}
if(wifi_Status == WL_CONNECTED){
// if connected
Serial.print("IP address: ");
Serial.println(WiFi.localIP());
return true;
} else {
// if not connected
Serial.println("[ERR] unable to connect Wifi");
return false;
}
}
<!doctype html>
<html>
<head>
<meta charset="utf-8">
<title>M5ATOM ENV Monitor</title>
<style>
body {
background: #FFFFFF;
margin: 0;
}
#HeadFont {
font-family: Impact, Charcoal, sans-serif;
font-size: 25px;
letter-spacing: 2px;
word-spacing: 2px;
color: #FFFFFF;
font-weight: normal;
}
#DataFont {
font-family: "Lucida Console", Monaco, monospace;
font-size: 18px;
letter-spacing: 2px;
word-spacing: 2px;
color: #FFFFFF;
font-weight: 400;
}
</style>
<script>
window.onload = function(){
document.getElementById('temperatureOutput').innerHTML = temperatureValue+"°C";
document.getElementById('humidityOutput').innerHTML = humidityValue+"%";
document.getElementById('pressureOutput').innerHTML = pressureValue+"hPa";
};
</script>
<script type="text/javascript" src="data.js"></script>
</head>
<body>
<table style="background-color: #7f7f7f; border-color: #000000; margin-left: auto; margin-right: auto; cellspacing=10">
<tbody>
<tr>
<td id="HeadFont" style="text-align: center;">M5ATOM ENV monitor</td>
<td><img alt="" src="electric-idea_100x100.jpg"/></td>
</tr>
<tr>
<td>
<table style="float: center;" cellspacing="10">
<tbody id="DataFont">
<tr>
<td style="text-align: right;">Temperature:</td>
<td id="temperatureOutput" style="letter-spacing: 0px;"></td>
</tr>
<tr>
<td style="text-align: right;">Humidity:</td>
<td id="humidityOutput"></td>
</tr>
<tr>
<td style="text-align: right;">Air Pressure:</td>
<td id="pressureOutput"></td>
</tr>
</tbody>
</table>
</td>
</tr>
</tbody>
</table>
</body>
</html>