Add ESP8226 client
This commit is contained in:
parent
e57966523e
commit
695f05e2ef
1
client/.gitignore
vendored
Normal file
1
client/.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
wifi_credentials.h
|
BIN
client/animations/spinny.gif
Normal file
BIN
client/animations/spinny.gif
Normal file
Binary file not shown.
After Width: | Height: | Size: 9.6 KiB |
83
client/bitmaps.h
Normal file
83
client/bitmaps.h
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
static const uint16_t PROGMEM spinny_dims[] = {8, 8, 8};
|
||||||
|
static const uint8_t PROGMEM spinny[][8] = {
|
||||||
|
{
|
||||||
|
0x38,
|
||||||
|
0x04,
|
||||||
|
0x92,
|
||||||
|
0x92,
|
||||||
|
0x92,
|
||||||
|
0x40,
|
||||||
|
0x38,
|
||||||
|
0x00,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
0x18,
|
||||||
|
0x44,
|
||||||
|
0x92,
|
||||||
|
0x92,
|
||||||
|
0x92,
|
||||||
|
0x44,
|
||||||
|
0x30,
|
||||||
|
0x00,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
0x28,
|
||||||
|
0x44,
|
||||||
|
0x8a,
|
||||||
|
0x92,
|
||||||
|
0xa2,
|
||||||
|
0x44,
|
||||||
|
0x28,
|
||||||
|
0x00,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
0x30,
|
||||||
|
0x44,
|
||||||
|
0x8a,
|
||||||
|
0x92,
|
||||||
|
0xa2,
|
||||||
|
0x44,
|
||||||
|
0x18,
|
||||||
|
0x00,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
0x38,
|
||||||
|
0x40,
|
||||||
|
0x82,
|
||||||
|
0xba,
|
||||||
|
0x82,
|
||||||
|
0x04,
|
||||||
|
0x38,
|
||||||
|
0x00,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
0x38,
|
||||||
|
0x44,
|
||||||
|
0x80,
|
||||||
|
0xba,
|
||||||
|
0x02,
|
||||||
|
0x44,
|
||||||
|
0x38,
|
||||||
|
0x00,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
0x38,
|
||||||
|
0x44,
|
||||||
|
0xa2,
|
||||||
|
0x10,
|
||||||
|
0x8a,
|
||||||
|
0x44,
|
||||||
|
0x38,
|
||||||
|
0x00,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
0x38,
|
||||||
|
0x44,
|
||||||
|
0x22,
|
||||||
|
0x92,
|
||||||
|
0x88,
|
||||||
|
0x44,
|
||||||
|
0x38,
|
||||||
|
0x00,
|
||||||
|
},
|
||||||
|
};
|
137
client/client.ino
Normal file
137
client/client.ino
Normal file
@ -0,0 +1,137 @@
|
|||||||
|
// Screen libraries etc.
|
||||||
|
#include <ESP8266WiFi.h>
|
||||||
|
#include <SPI.h>
|
||||||
|
#include <Wire.h>
|
||||||
|
#include <Adafruit_GFX.h>
|
||||||
|
#include <Adafruit_SSD1306.h>
|
||||||
|
#include <ESP8266HTTPClient.h>
|
||||||
|
#include <WiFiClientSecureBearSSL.h>
|
||||||
|
|
||||||
|
#include "bitmaps.h"
|
||||||
|
#include "wifi_credentials.h"
|
||||||
|
|
||||||
|
// Screen stuff
|
||||||
|
#define SCREEN_WIDTH 128 // OLED display width, in pixels
|
||||||
|
#define SCREEN_HEIGHT 64 // OLED display height, in pixels
|
||||||
|
#define OLED_RESET -1 // Reset pin # (or -1 if sharing Arduino reset pin)
|
||||||
|
#define SCREEN_ADDRESS 0x3C
|
||||||
|
|
||||||
|
// Button stuff
|
||||||
|
#define BUTTON_PIN D6
|
||||||
|
|
||||||
|
#define FRAME_DELAY 80
|
||||||
|
|
||||||
|
// Peripherals
|
||||||
|
|
||||||
|
Adafruit_SSD1306 display(SCREEN_WIDTH, SCREEN_HEIGHT, &Wire, OLED_RESET);
|
||||||
|
|
||||||
|
// App state
|
||||||
|
|
||||||
|
enum {
|
||||||
|
CONNECTING,
|
||||||
|
SHOWING_DEPARTURES
|
||||||
|
} state = CONNECTING;
|
||||||
|
int frame = 0;
|
||||||
|
int buttonPushed = false;
|
||||||
|
|
||||||
|
void ICACHE_RAM_ATTR onButtonFalling() {
|
||||||
|
buttonPushed = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setup() {
|
||||||
|
Serial.begin(9600);
|
||||||
|
|
||||||
|
// SSD1306_SWITCHCAPVCC = generate display voltage from 3.3V internally
|
||||||
|
if(!display.begin(SSD1306_SWITCHCAPVCC, SCREEN_ADDRESS)) {
|
||||||
|
Serial.println("SSD1306 allocation failed");
|
||||||
|
for(;;); // Don't proceed, loop forever
|
||||||
|
}
|
||||||
|
Serial.println("SSD1306 initialized");
|
||||||
|
display.clearDisplay();
|
||||||
|
|
||||||
|
WiFi.begin(WIFI_SSID, WIFI_PASSWORD);
|
||||||
|
|
||||||
|
pinMode(BUTTON_PIN, INPUT_PULLUP);
|
||||||
|
attachInterrupt(BUTTON_PIN, onButtonFalling, FALLING);
|
||||||
|
|
||||||
|
display.setTextColor(SSD1306_WHITE);
|
||||||
|
display.setTextSize(1);
|
||||||
|
display.setCursor(0, 0);
|
||||||
|
display.printf("Connecting to\n%s\n", WIFI_SSID);
|
||||||
|
display.display();
|
||||||
|
/*
|
||||||
|
display.print(" 5 Mannheim sofort\n"
|
||||||
|
"26 Kirchheim 9 min\n"
|
||||||
|
" 5 Weinheim 20:13");
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
|
void drawSpinner (int frame) {
|
||||||
|
display.fillRect(120, 56, spinny_dims[0], spinny_dims[1], SSD1306_BLACK);
|
||||||
|
display.drawBitmap(120, 56, spinny[frame % spinny_dims[2]], spinny_dims[0], spinny_dims[1], SSD1306_WHITE);
|
||||||
|
}
|
||||||
|
|
||||||
|
void loop() {
|
||||||
|
switch (state) {
|
||||||
|
case CONNECTING: {
|
||||||
|
if (WiFi.status() != WL_CONNECTED) {
|
||||||
|
drawSpinner(frame++);
|
||||||
|
display.display();
|
||||||
|
delay(FRAME_DELAY);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
display.clearDisplay();
|
||||||
|
display.setCursor(0, 0);
|
||||||
|
display.print("Connected!\nRequesting departures");
|
||||||
|
display.display();
|
||||||
|
|
||||||
|
std::unique_ptr<BearSSL::WiFiClientSecure> https(new BearSSL::WiFiClientSecure);
|
||||||
|
https->setInsecure();
|
||||||
|
HTTPClient client;
|
||||||
|
if (!client.begin(*https, "vrnp.beany.club", 443, "/departures?stop_id=de:08221:1225&platform=A")) {
|
||||||
|
display.clearDisplay();
|
||||||
|
display.setCursor(0, 0);
|
||||||
|
display.print("begin failed");
|
||||||
|
display.display();
|
||||||
|
for (;;);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
client.addHeader("Authorization", AUTH_TOKEN);
|
||||||
|
int statusCode = client.GET();
|
||||||
|
if (statusCode != 200) {
|
||||||
|
display.clearDisplay();
|
||||||
|
display.setCursor(0, 0);
|
||||||
|
display.printf("http non-ok: %d\n", statusCode);
|
||||||
|
display.display();
|
||||||
|
for (;;);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
display.clearDisplay();
|
||||||
|
display.setCursor(0, 0);
|
||||||
|
display.print(client.getString());
|
||||||
|
display.display();
|
||||||
|
|
||||||
|
state = SHOWING_DEPARTURES;
|
||||||
|
delay(FRAME_DELAY);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
case SHOWING_DEPARTURES:
|
||||||
|
if (WiFi.status() != WL_CONNECTED) {
|
||||||
|
state = CONNECTING;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (buttonPushed) {
|
||||||
|
buttonPushed = false;
|
||||||
|
delay(FRAME_DELAY);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
delay(FRAME_DELAY);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
45
client/scripts/extract_bitmaps.py
Executable file
45
client/scripts/extract_bitmaps.py
Executable file
@ -0,0 +1,45 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
|
||||||
|
import sys
|
||||||
|
|
||||||
|
from PIL import Image
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
def get_bit(im, x, y):
|
||||||
|
px = im.getpixel((x, y))
|
||||||
|
|
||||||
|
if px == 0 or px == (0, 0, 0, 255):
|
||||||
|
return 0
|
||||||
|
if px == 1 or px == (255, 255, 255, 255):
|
||||||
|
return 1
|
||||||
|
print(x, y)
|
||||||
|
print(px)
|
||||||
|
raise
|
||||||
|
|
||||||
|
def generate_bitmap_from_gif(file: Path, fd):
|
||||||
|
variable_name = file.with_suffix("").name
|
||||||
|
|
||||||
|
im = Image.open(file)
|
||||||
|
(w, h) = im.size
|
||||||
|
|
||||||
|
assert w % 8 == 0
|
||||||
|
|
||||||
|
fd.write(f"static const uint16_t PROGMEM {variable_name}_dims[] = {{{w}, {h}, {im.n_frames}}};\n")
|
||||||
|
fd.write(f"static const uint8_t PROGMEM {variable_name}[][{w * h // 8}] = {{\n")
|
||||||
|
for i in range(im.n_frames):
|
||||||
|
im.seek(i)
|
||||||
|
fd.write(" {\n")
|
||||||
|
for y in range(h):
|
||||||
|
line = []
|
||||||
|
for x in range(0, w, 8):
|
||||||
|
byte_str = "".join(str(get_bit(im, x + i, y)) for i in range(8))
|
||||||
|
byte = int(byte_str, 2)
|
||||||
|
line.append(f"0x{byte:02x}")
|
||||||
|
line_str = ", ".join(line)
|
||||||
|
fd.write(f" {line_str},\n")
|
||||||
|
fd.write(" },\n")
|
||||||
|
fd.write("};\n")
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
generate_bitmap_from_gif(Path("./animations/spinny.gif"), sys.stdout)
|
Loading…
x
Reference in New Issue
Block a user