Blame | Last modification | View Log | RSS feed
#ifndef LED_BUILTIN_LED#define BUILTIN_LED 15#endif#define BOOT_BUTTON 0#if defined(ESP8266)#include <ESP8266mDNS.h>#elif defined(ESP32)#include <ESPmDNS.h>#endif#include <FS.h>#include <LittleFS.h>#include <AsyncFsWebServer.h> // https://github.com/cotestatnt/async-esp-fs-webserver#define FILESYSTEM LittleFSchar const* hostname = "fsbrowser";AsyncFsWebServer server(FILESYSTEM, 80, hostname);long unsigned int timer = 0; // used to determine when to send websocket packets// type declarations for websocket packet// data typetypedef struct trig_t {float sin;float cos;float tan;uint16_t angle; // storing angle (x1000) as an integer, just to be different!uint16_t X; // column on chart};// union of data type and a byte array for websocketstypedef union trig_u {trig_t data;uint8_t byteArray[sizeof(trig_t)];};trig_u trig; // union to store trig data to be transmitteduint16_t X=0; // chart column// HTML of homepage:static const char homepage[] PROGMEM = R"EOF(<!DOCTYPE html><html><body onload="onBodyLoad()"><canvas id="chart" width="500" height="202"></canvas><span>Angle (radians): </span><span id="angle"></span><script type="text/javascript">var ws = null; // reference to websocketvar v16; // 16-bit view of incoming datavar v32; // 32-bit view of incoming datavar ctxChart = null; // reference to chart canvas 2d contextvar angle; // angle (radians)var sin = 0; // sin of anglevar cos = 1; // cos of anglevar tan = 0; // tan of angle// Handles incoming messagesfunction rxMessage(msg) {// console.log(msg);v16 = new Uint16Array(msg);v32 = new Float32Array(msg);angle = v16[6] / 1000; // (1000x) angle is stored as an int in the 7th pair of bytes in the websocket packetvar X = v16[7]; // current column is stored as an int in the 8th pair of bytes in the websocket packetdocument.getElementById("angle").innerHTML = angle;// Draw a line from previous sin point to current sin pointctxChart.strokeStyle = "red";ctxChart.beginPath();ctxChart.moveTo(X-1, 100 - 100*sin);sin = v32[0]; // sin is stored as a float in the first four bytes of the websocket packetctxChart.lineTo(X, 100 - 100*sin);ctxChart.stroke();// Draw a line from previous cos point to current sin pointctxChart.strokeStyle = "green";ctxChart.beginPath();ctxChart.moveTo(X-1, 100 - 100*cos);cos = v32[1];ctxChart.lineTo(X, 100 - 100*cos);ctxChart.stroke();// Draw a line from previous tan point to current tan pointctxChart.strokeStyle = "blue";ctxChart.beginPath();ctxChart.moveTo(X-1, 100 - 100*tan);tan = v32[2];ctxChart.lineTo(X, 100 - 100*tan);ctxChart.stroke();// increment X and wrap if necessaryif (++X >= 500) {X = 0;ctxChart.clearRect(0,0,500,202);}}// Configure and start WebSocket clientfunction startSocket() {ws = new WebSocket('ws://' + document.location.host + '/ws', ['arduino']);ws.binaryType = "arraybuffer";ws.onopen = function (e) {console.log("WebSocket client connected to " + 'ws://' + document.location.host + '/ws');};ws.onclose = function (e) {addMessage("WebSocket client disconnected");};ws.onerror = function (e) {console.log("ws error", e);addMessage("Error");};ws.onmessage = function (e) {rxMessage(e.data)};}// When page is fully loaded start connectionfunction onBodyLoad() {startSocket();ctxChart = document.getElementById("chart").getContext("2d");}</script></body></html>)EOF";// In this example a custom websocket event handler is used instead defaultvoid onWsEvent(AsyncWebSocket* server, AsyncWebSocketClient* client, AwsEventType type, void* arg, uint8_t* data, size_t len) {switch (type) {case WS_EVT_CONNECT:client->printf("{\"Websocket connected\": true, \"clients\": %u}", client->id());Serial.printf("Websocket client %u connected\n", client->id());break;case WS_EVT_DISCONNECT:Serial.printf("Websocket client %u connected\n", client->id());break;case WS_EVT_DATA:{AwsFrameInfo* info = (AwsFrameInfo*)arg;if (info->opcode == WS_TEXT) {char msg[len+1];msg[len] = '\0';memcpy(msg, data, len);Serial.printf("Received message \"%s\"\n", msg);}}break;default:break;}}//////////////////////////////// Filesystem /////////////////////////////////////////bool startFilesystem() {if (LittleFS.begin()){server.printFileList(LittleFS, "/", 1);Serial.println();return true;}else {Serial.println("ERROR on mounting filesystem. It will be reformatted!");LittleFS.format();ESP.restart();}return false;}void setup() {pinMode(BUILTIN_LED, OUTPUT);digitalWrite(BUILTIN_LED, HIGH);Serial.begin(115200);while (!Serial) {if (millis() > 3000) break; // 3 second timeout to start Serial}digitalWrite(BUILTIN_LED, LOW);if (Serial) { // flash LED if Serial is availabledelay(200);digitalWrite(BUILTIN_LED, HIGH);delay(200);digitalWrite(BUILTIN_LED, LOW);}// FILESYSTEM INITstartFilesystem();// Try to connect to WiFi (will start AP if not connected after timeout)if (!server.startWiFi(10000)) {Serial.println("\nWiFi not connected! Starting AP mode...");char ssid[21];#ifdef defined(ESP8266)snprintf(ssid, sizeof(ssid), "ESP-%dX", ESP.getChipId());#elif defined(ESP32)snprintf(ssid, sizeof(ssid), "ESP-%llX", ESP.getEfuseMac());#endifserver.startCaptivePortal(ssid, "123456789", "/setup");}// Add custom page handlersserver.on("/", HTTP_GET, [](AsyncWebServerRequest *request){request->send(200, "text/html", homepage);});// Enable ACE FS file web editor and add FS info callback functionserver.enableFsCodeEditor();// Init with custom WebSocket event handler and start serverserver.init(onWsEvent);Serial.print(F("ESP Web Server started on IP Address: "));Serial.println(server.getServerIP());Serial.println(F("This is \"BinaryWebSocket.ino\" example.\n""Open /setup page to configure optional parameters.\n""Open /edit page to view, edit or upload example or your custom webserver source files."));// Set hostname#ifdef ESP8266WiFi.hostname(hostname);#elif defined(ESP32)WiFi.setHostname(hostname);#endif// Start MDSN responderif (WiFi.status() == WL_CONNECTED) {if (MDNS.begin(hostname)) {Serial.println(F("MDNS responder started."));Serial.printf("You should be able to connect with address http://%s.local/\n", hostname);// Add service to MDNS-SDMDNS.addService("http", "tcp", 80);}}}void loop() {// send websocket every 50msif (millis() > 50 * timer) {timer++;trig.data.angle = 1000.0 * 2 * PI * X/500; // angle (x1000) is stored as a 16-bit integer, just to be different!trig.data.sin = sin(trig.data.angle/1000.0);trig.data.cos = cos(trig.data.angle/1000.0);trig.data.tan = tan(trig.data.angle/1000.0);trig.data.X = X;if (++X >= 500) X = 0;server.wsBroadcastBinary(trig.byteArray, sizeof(trig.byteArray));}}