forked from github/ebk_co2ampel
Compare commits
1 commit
Author | SHA1 | Date | |
---|---|---|---|
|
0abcf7d246 |
7 changed files with 191 additions and 10 deletions
3
NOTES.md
Normal file
3
NOTES.md
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
http://tomeko.net/online_tools/cpp_text_escape.php?lang=en
|
||||||
|
https://www.willpeavy.com/tools/minifier/
|
||||||
|
|
2
config.h.example
Normal file
2
config.h.example
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
#define SSID "co2ampel";
|
||||||
|
#define PASSWORD "<change this!>";
|
|
@ -2,6 +2,10 @@
|
||||||
#include "MHZ19.h"
|
#include "MHZ19.h"
|
||||||
#include "SSD1306Wire.h"
|
#include "SSD1306Wire.h"
|
||||||
#include <Adafruit_NeoPixel.h>
|
#include <Adafruit_NeoPixel.h>
|
||||||
|
#include <WiFi.h>
|
||||||
|
#include "config.h"
|
||||||
|
#include "html.h"
|
||||||
|
#include "javascript.h"
|
||||||
|
|
||||||
// Maximum CO² levels for green and yellow, everything above is considered red.
|
// Maximum CO² levels for green and yellow, everything above is considered red.
|
||||||
#define GREEN_CO2 800
|
#define GREEN_CO2 800
|
||||||
|
@ -21,14 +25,26 @@
|
||||||
// Pin for LED
|
// Pin for LED
|
||||||
#define LED_PIN 4
|
#define LED_PIN 4
|
||||||
|
|
||||||
|
const char* ssid = SSID
|
||||||
|
const char* password = PASSWORD
|
||||||
|
|
||||||
|
const char* html = HTML;
|
||||||
|
const char* chartjs = CHARTJS;
|
||||||
|
|
||||||
|
WiFiServer server(80);
|
||||||
MHZ19 myMHZ19;
|
MHZ19 myMHZ19;
|
||||||
HardwareSerial mySerial(1);
|
HardwareSerial mySerial(1);
|
||||||
SSD1306Wire display(0x3c, SDA_PIN, SCL_PIN);
|
SSD1306Wire display(0x3c, SDA_PIN, SCL_PIN);
|
||||||
Adafruit_NeoPixel pixels = Adafruit_NeoPixel(1, LED_PIN, NEO_RGB + NEO_KHZ400);
|
Adafruit_NeoPixel pixels = Adafruit_NeoPixel(1, LED_PIN, NEO_RGB + NEO_KHZ400);
|
||||||
|
String header;
|
||||||
|
String response_body;
|
||||||
|
WiFiClient client;
|
||||||
|
|
||||||
unsigned long getDataTimer = 0;
|
unsigned long getDataTimer = 0;
|
||||||
int lastvals[120];
|
int lastvals[12000];
|
||||||
int dheight;
|
int dheight;
|
||||||
|
int CO2;
|
||||||
|
IPAddress IP;
|
||||||
|
|
||||||
void setup() {
|
void setup() {
|
||||||
Serial.begin(9600);
|
Serial.begin(9600);
|
||||||
|
@ -39,14 +55,19 @@ void setup() {
|
||||||
delay(1000);
|
delay(1000);
|
||||||
display.clear();
|
display.clear();
|
||||||
dheight = display.getHeight();
|
dheight = display.getHeight();
|
||||||
myMHZ19.autoCalibration();
|
myMHZ19.autoCalibration(false);
|
||||||
// Fill array of last measurements with -1
|
// Fill array of last measurements with -1
|
||||||
for (int x = 0; x <= 119; x = x + 1) {
|
for (int x = 0; x <= 11999; x = x + 1) {
|
||||||
lastvals[x] = -1;
|
lastvals[x] = -1;
|
||||||
}
|
}
|
||||||
pixels.begin();
|
pixels.begin();
|
||||||
pixels.setPixelColor(0, 30,0,0);
|
pixels.setPixelColor(0, 30,0,0);
|
||||||
pixels.show();
|
pixels.show();
|
||||||
|
WiFi.softAP(ssid, password);
|
||||||
|
IP = WiFi.softAPIP();
|
||||||
|
Serial.print("AP IP address: ");
|
||||||
|
Serial.println(IP);
|
||||||
|
server.begin();
|
||||||
}
|
}
|
||||||
|
|
||||||
int calc_vpos_for_co2(int co2val, int display_height) {
|
int calc_vpos_for_co2(int co2val, int display_height) {
|
||||||
|
@ -67,24 +88,56 @@ void set_led_color(int co2) {
|
||||||
pixels.show();
|
pixels.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void json_header() {
|
||||||
|
client.println("HTTP/1.1 200 OK");
|
||||||
|
client.println("Content-type:application/json");
|
||||||
|
client.println("Connection: close");
|
||||||
|
client.println();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void html_header() {
|
||||||
|
client.println("HTTP/1.1 200 OK");
|
||||||
|
client.println("Content-type:text/html");
|
||||||
|
client.println("Connection: close");
|
||||||
|
client.println();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void js_header() {
|
||||||
|
client.println("HTTP/1.1 200 OK");
|
||||||
|
client.println("Content-type:text/javascript");
|
||||||
|
client.println("Connection: close");
|
||||||
|
client.println();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void not_found() {
|
||||||
|
client.println("HTTP/1.1 404 NOT FOUND");
|
||||||
|
client.println("Content-type:text/plain");
|
||||||
|
client.println("Connection: close");
|
||||||
|
client.println();
|
||||||
|
}
|
||||||
|
|
||||||
void loop() {
|
void loop() {
|
||||||
if (millis() - getDataTimer >= INTERVAL) {
|
if (millis() - getDataTimer >= INTERVAL) {
|
||||||
// Get new CO² value.
|
// Get new CO² value.
|
||||||
int CO2 = myMHZ19.getCO2();
|
CO2 = myMHZ19.getCO2();
|
||||||
// Shift entries in array back one position.
|
// Shift entries in array back one position.
|
||||||
for (int x = 1; x <= 119; x = x + 1) {
|
for (int x = 1; x <= 11999; x = x + 1) {
|
||||||
lastvals[x - 1] = lastvals[x];
|
lastvals[x - 1] = lastvals[x];
|
||||||
}
|
}
|
||||||
// Add new measurement at the end.
|
// Add new measurement at the end.
|
||||||
lastvals[119] = CO2;
|
lastvals[11999] = CO2;
|
||||||
// Clear display and redraw whole graph.
|
// Clear display and redraw graph of last 120 values.
|
||||||
display.clear();
|
display.clear();
|
||||||
for (int h = 1; h < 120; h = h + 1) {
|
for (int hs = 1; hs < 120; hs = hs + 1) {
|
||||||
|
int h = 1200-120+hs;
|
||||||
int curval = lastvals[h];
|
int curval = lastvals[h];
|
||||||
if (curval > 0) {
|
if (curval > 0) {
|
||||||
int vpos = calc_vpos_for_co2(lastvals[h], dheight);
|
int vpos = calc_vpos_for_co2(lastvals[h], dheight);
|
||||||
int vpos_last = calc_vpos_for_co2(lastvals[h - 1], dheight);
|
int vpos_last = calc_vpos_for_co2(lastvals[h - 1], dheight);
|
||||||
display.drawLine(h - 1, vpos_last, h, vpos);
|
display.drawLine(h, vpos - 1, h, vpos + 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Set LED color and print value on display
|
// Set LED color and print value on display
|
||||||
|
@ -96,6 +149,77 @@ void loop() {
|
||||||
// Debug output
|
// Debug output
|
||||||
Serial.print("CO2 (ppm): ");
|
Serial.print("CO2 (ppm): ");
|
||||||
Serial.println(CO2);
|
Serial.println(CO2);
|
||||||
|
Serial.println(IP);
|
||||||
getDataTimer = millis();
|
getDataTimer = millis();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
WiFiClient client = server.available(); // Listen for incoming clients
|
||||||
|
|
||||||
|
if (client) { // If a new client connects,
|
||||||
|
Serial.println("New Client."); // print a message out in the serial port
|
||||||
|
String currentLine = ""; // make a String to hold incoming data from the client
|
||||||
|
while (client.connected()) { // loop while the client's connected
|
||||||
|
if (client.available()) { // if there's bytes to read from the client,
|
||||||
|
char c = client.read(); // read a byte, then
|
||||||
|
Serial.write(c); // print it out the serial monitor
|
||||||
|
header += c;
|
||||||
|
if (c == '\n') { // if the byte is a newline character
|
||||||
|
// if the current line is blank, you got two newline characters in a row.
|
||||||
|
// that's the end of the client HTTP request.
|
||||||
|
// handle requests here
|
||||||
|
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:
|
||||||
|
|
||||||
|
|
||||||
|
if (header.indexOf("GET /data.json") >= 0) {
|
||||||
|
json_header();
|
||||||
|
client.print("{ data: [");
|
||||||
|
for (int x = 0; x <= 11999; x = x + 1) {
|
||||||
|
if (x % 60 == 0) {
|
||||||
|
client.print("{ minutes:");
|
||||||
|
client.print(x/60);
|
||||||
|
client.print(", ppm:");
|
||||||
|
client.print(lastvals[x]);
|
||||||
|
client.println("}, ");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
client.print("] }");
|
||||||
|
} else if (header.indexOf("GET /chart.js") >= 0) {
|
||||||
|
js_header();
|
||||||
|
client.print(chartjs);
|
||||||
|
} else if (header.indexOf("GET /favicon.ico") >= 0) {
|
||||||
|
not_found();
|
||||||
|
client.println("nope");
|
||||||
|
} else if (header.indexOf("GET /") >= 0) {
|
||||||
|
html_header();
|
||||||
|
client.println(html);
|
||||||
|
// client.println(CO2);
|
||||||
|
// client.println(co2_html_2);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// The HTTP response ends with another blank line
|
||||||
|
client.println();
|
||||||
|
// Break out of the while loop
|
||||||
|
break;
|
||||||
|
} else { // if you got a newline, then clear currentLine
|
||||||
|
currentLine = "";
|
||||||
|
}
|
||||||
|
} else if (c != '\r') { // if you got anything else but a carriage return character,
|
||||||
|
currentLine += c; // add it to the end of the currentLine
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
header = "";
|
||||||
|
response_body = "";
|
||||||
|
// Close the connection
|
||||||
|
client.stop();
|
||||||
|
Serial.println("Client disconnected.");
|
||||||
|
Serial.println("");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
1
src/html.h
Normal file
1
src/html.h
Normal file
|
@ -0,0 +1 @@
|
||||||
|
#define HTML "<html> <head> <script src=\"/chart.js\"></script> <title>CO²-Ampel</title></head> <body> <canvas id=\"myChart\" width=\"100%\" height=\"100%\"></canvas> <script> var ctx = document.getElementById('myChart');var myChart = new Chart(ctx, { type: 'bar', data: { labels: ['Red', 'Blue', 'Yellow', 'Green', 'Purple', 'Orange'], datasets: [{ label: '# of Votes', data: [12, 19, 3, 5, 2, 3], backgroundColor: [ 'rgba(255, 99, 132, 0.2)', 'rgba(54, 162, 235, 0.2)', 'rgba(255, 206, 86, 0.2)', 'rgba(75, 192, 192, 0.2)', 'rgba(153, 102, 255, 0.2)', 'rgba(255, 159, 64, 0.2)' ], borderColor: [ 'rgba(255, 99, 132, 1)', 'rgba(54, 162, 235, 1)', 'rgba(255, 206, 86, 1)', 'rgba(75, 192, 192, 1)', 'rgba(153, 102, 255, 1)', 'rgba(255, 159, 64, 1)' ], borderWidth: 1 }] }, options: { scales: { yAxes: [{ ticks: { beginAtZero: true } }] } }}); </script> </body> </html>";
|
1
src/javascript.h
Normal file
1
src/javascript.h
Normal file
File diff suppressed because one or more lines are too long
1
static/Chart.bundle.min.js
vendored
Executable file
1
static/Chart.bundle.min.js
vendored
Executable file
File diff suppressed because one or more lines are too long
49
static/index.html
Normal file
49
static/index.html
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<script src="/chart.js"></script>
|
||||||
|
<title>CO²-Ampel</title></head>
|
||||||
|
<body>
|
||||||
|
<canvas id="myChart" width="100%" height="100%"></canvas>
|
||||||
|
<script>
|
||||||
|
var ctx = document.getElementById('myChart');
|
||||||
|
var myChart = new Chart(ctx, {
|
||||||
|
type: 'bar',
|
||||||
|
data: {
|
||||||
|
labels: ['Red', 'Blue', 'Yellow', 'Green', 'Purple', 'Orange'],
|
||||||
|
datasets: [{
|
||||||
|
label: '# of Votes',
|
||||||
|
data: [12, 19, 3, 5, 2, 3],
|
||||||
|
backgroundColor: [
|
||||||
|
'rgba(255, 99, 132, 0.2)',
|
||||||
|
'rgba(54, 162, 235, 0.2)',
|
||||||
|
'rgba(255, 206, 86, 0.2)',
|
||||||
|
'rgba(75, 192, 192, 0.2)',
|
||||||
|
'rgba(153, 102, 255, 0.2)',
|
||||||
|
'rgba(255, 159, 64, 0.2)'
|
||||||
|
],
|
||||||
|
borderColor: [
|
||||||
|
'rgba(255, 99, 132, 1)',
|
||||||
|
'rgba(54, 162, 235, 1)',
|
||||||
|
'rgba(255, 206, 86, 1)',
|
||||||
|
'rgba(75, 192, 192, 1)',
|
||||||
|
'rgba(153, 102, 255, 1)',
|
||||||
|
'rgba(255, 159, 64, 1)'
|
||||||
|
],
|
||||||
|
borderWidth: 1
|
||||||
|
}]
|
||||||
|
},
|
||||||
|
options: {
|
||||||
|
scales: {
|
||||||
|
yAxes: [{
|
||||||
|
ticks: {
|
||||||
|
beginAtZero: true
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
Loading…
Reference in a new issue