|
本帖最后由 vany5921 于 2019-9-9 10:57 编辑
本项目将利用ESP32(M5StickC)制作一个局域网的Emoji显示屏
1. 准备工作
ESP32 开发板
我们需要一个ESP32开发板,这里我选用了M5Stack,当然NodeMCU或者其他相同类型的开发板也是可以的。
NeoPixels项目中用到了Adafruit_NeoPixel 的库,因此各种可编程的LED都支持比如WS2812, WS2812B, SK6812, SK6812mini...,项目中使用了SK6812mini来制作8x8的点阵面板,当然你也可以买现成的LED点阵屏
底座
项目中使用了一个柔性可弯曲的USB连接线
2. 3D打印和装配(可选)
如果用的是8X8点阵屏可以直接跳过此步
1.3D打印外壳
files.zip
(54.03 KB, 下载次数: 2)
2.将SK6812旋转一定方向放入外壳中
3.焊接电路
4.将开发板用热熔胶粘到外壳内部(使用M5Stack可以省略)
5.固定USB供电接口,(使用M5Stack可以省略)
6.安装螺丝
由于项目需要显示8X8的emoji,从这里找到相关素材,修改图片的像素尺寸并且进行base64编码,以便嵌入到HTML网页中。
3.HTML界面制作
写一个HTML脚本来选择emoji并且传送到ESP32进行显示,在这里用了Web Socket协议传送二进制数据
[mw_shl_code=html,true]<!DOCTYPE html>
<html>
<head>
<meta name='viewport' content='user-scalable=no,initial-scale=2.0' />
<style>
body {
background: #000;
color: #fff;
text-align: center;
}
div {
margin: 10px;
}
</style>
</head>
<body>
<div>
<h3>Pick An Emoji</h3>
<img id='emojis' onclick='clickEmoji(event);'
src=''>
</div>
<div>
<p id='out'></p>
<canvas id='emoji' width='8' height='8'></canvas>
</div>
<script>
function nw() { return new WebSocket('ws://' + location.hostname + ':81/', ['arduino']); }
var ws = nw();
function cc(c) {
return String.fromCharCode(c / 64);
}
function clickEmoji(e) {
var xo = 3;
var yo = 3;
var xs = 13;
var ys = 12;
var x = e.offsetX;
var y = e.offsetY;
var c = Math.round((x - xo - 4) / xs);
var r = Math.round((y - yo - 4) / ys);
document.getElementById('out').innerText = 'Selected [' + c + ', ' + r + ']';
var img = document.getElementById('emojis');
var ce = document.getElementById('emoji');
var ctx = ce.getContext('2d');
// ctx.drawImage(img, -(xo + (xs * c)), -(yo + (ys * r)), 131, 122);
ctx.drawImage(img, -(xo + (xs * c)), -(yo + (ys * r)), 131, 122);
var d = ctx.getImageData(0, 0, 8, 8).data;
var t = '';
for (var i = 0; i < 64; i++) {
t += cc(d[(i * 4)]);
t += cc(d[(i * 4) + 1]);
t += cc(d[(i * 4) + 2]);
}
ws.send(t, { binary: true });
}
</script>
</body>
</html>[/mw_shl_code]
使用html-minifier将上述文件转换为单行代码,嵌入到arduino程序中。以下代码仅在M5StickC中进行测试验证,ESP32可参考
[mw_shl_code=arduino,true]#include <WiFi.h>
#include <WiFiClient.h>
#include <WebServer.h>
#include <WebSocketsServer.h>
#include <ESPmDNS.h>
#include <Adafruit_NeoPixel.h>
#include <M5StickC.h>
#define NEOPIXEL_PIN 32
#define WIDTH 8
#define HEIGHT 8
#define ZIGZAG_MATRIX
const char *ssid = "M5STACK";
const char *password = "M5STACK";
WebServer server(80);
WebSocketsServer webSocket = WebSocketsServer(81);
Adafruit_NeoPixel pixels = Adafruit_NeoPixel(WIDTH * HEIGHT, NEOPIXEL_PIN, NEO_RGB + NEO_KHZ400);
void webSocketEvent(uint8_t num, WStype_t type, uint8_t * payload, size_t length) {
switch (type) {
case WStype_DISCONNECTED:
Serial.printf("[%u] Disconnected!\n", num);
break;
case WStype_CONNECTED: {
IPAddress ip = webSocket.remoteIP(num);
Serial.printf("[%u] Connected from %d.%d.%d.%d url: %s\n", num, ip[0], ip[1], ip[2], ip[3], payload);
// send message to client
webSocket.sendTXT(num, "Connected");
}
break;
case WStype_TEXT:
Serial.printf("[%u] get Text[%u].\n", num, length);
Serial.println(*payload);
uint8_t i = 0;
for (uint8_t y = 0; y < HEIGHT; y++) {
for (uint8_t x = 0; x < WIDTH; x++) {
uint8_t r = payload[i++];
uint8_t g = payload[i++];
uint8_t b = payload[i++];
#ifdef ZIGZAG_MATRIX
uint16_t p = (y % 2) ? (((y + 1) * 8) - x - 1) : ((y * 8) + x);
#else
uint16_t p = (y * 8) + x;
#endif
pixels.setPixelColor(p, pixels.Color(r, g, b));
Serial.printf("(%u, %u) [%u] color: %u, %u, %u\n", x, y, p, r, g, b);
}
}
pixels.show(); // This sends the updated pixel color to the hardware.
break;
}
}
void handleNotFound() {
String message = "File Not Found\n\n";
message += "URI: ";
message += server.uri();
message += "\nMethod: ";
message += (server.method() == HTTP_GET) ? "GET" : "POST";
message += "\nArguments: ";
message += server.args();
message += "\n";
for (uint8_t i = 0; i < server.args(); i++) {
message += " " + server.argName(i) + ": " + server.arg(i) + "\n";
}
server.send(404, "text/plain", message);
}
void setup(void) {
M5.begin();
pixels.begin();
pixels.fill(pixels.Color(0, 0, 0));
pixels.show();
WiFi.mode(WIFI_STA);
WiFi.begin(ssid, password);
Serial.println("");
// Wait for connection
while (WiFi.status() != WL_CONNECTED) {
delay(500);
Serial.print(".");
}
Serial.println("");
Serial.print("Connected to ");
Serial.println(ssid);
Serial.print("IP address: ");
Serial.println(WiFi.localIP());
server.on("/", []() {
server.send(200, "text/html", "<!DOCTYPE html><html><head><meta name='viewport' content='user-scalable=no,initial-scale=2.0' /><style>body{background:#000;color:#fff;text-align:center}div{margin:10px}</style></head><body><div><h3>Pick An Emoji</h3> <img id='emojis' onclick='clickEmoji(event);' src=''></div><div><p id='out'></p> <canvas id='emoji' width='8' height='8'></canvas></div> <script>function nw(){return new WebSocket('ws://'+location.hostname+':81/',['arduino']);} var ws=nw();function cc(c){return String.fromCharCode(c/64);} function clickEmoji(e){var xo=3;var yo=3;var xs=13;var ys=12;var x=e.offsetX;var y=e.offsetY;var c=Math.round((x-xo-4)/xs);var r=Math.round((y-yo-4)/ys);document.getElementById('out').innerText='Selected ['+c+', '+r+']';var img=document.getElementById('emojis');var ce=document.getElementById('emoji');var ctx=ce.getContext('2d');ctx.drawImage(img,-(xo+(xs*c)),-(yo+(ys*r)),131,122);var d=ctx.getImageData(0,0,8,8).data;var t='';for(var i=0;i<64;i++){t+=cc(d[(i*4)]);t+=cc(d[(i*4)+1]);t+=cc(d[(i*4)+2]);} ws.send(t,{binary:true});}</script> </body></html>");
});
server.onNotFound(handleNotFound);
server.begin();
Serial.println("HTTP server started");
webSocket.begin();
webSocket.onEvent(webSocketEvent);
MDNS.addService("http", "tcp", 80);
MDNS.addService("ws", "tcp", 81);
}
unsigned long last_10sec = 0;
unsigned int counter = 0;
void loop(void) {
unsigned long t = millis();
webSocket.loop();
server.handleClient();
if((t - last_10sec) > 10 *1000) {
counter++;
bool ping = (counter %2);
int i = webSocket.connectedClients(ping);
Serial.printf("d% Connected websocket clients ping: %d\n", i, ping);
last_10sec = millis();
}
}[/mw_shl_code]
|
|