IOT-emoji 显示装置-Arduino中文社区 - Powered by Discuz! Archiver

vany5921 发表于 2019-9-9 10:55

IOT-emoji 显示装置

本帖最后由 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打印外壳
2.将SK6812旋转一定方向放入外壳中
3.焊接电路
4.将开发板用热熔胶粘到外壳内部(使用M5Stack可以省略)
5.固定USB供电接口,(使用M5Stack可以省略)
6.安装螺丝
由于项目需要显示8X8的emoji,从这里找到相关素材,修改图片的像素尺寸并且进行base64编码,以便嵌入到HTML网页中。

3.HTML界面制作

写一个HTML脚本来选择emoji并且传送到ESP32进行显示,在这里用了Web Socket协议传送二进制数据
<!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>


使用html-minifier将上述文件转换为单行代码,嵌入到arduino程序中。以下代码仅在M5StickC中进行测试验证,ESP32可参考

#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, ip, ip, ip, 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;
      uint8_t g = payload;
      uint8_t b = payload;
#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();
}
}


aspendove 发表于 2020-4-10 00:32

你好,请问led矩阵的电路是用什么方式连接的?
我也想做一个密集的led矩阵,但是不知道用PCB还是用3d打印来固定灯珠。

vany5921 发表于 2020-4-10 09:49

aspendove 发表于 2020-4-10 00:32
你好,请问led矩阵的电路是用什么方式连接的?
我也想做一个密集的led矩阵,但是不知道用PCB还是用3d打印来 ...

PCB工作量小一些

aspendove 发表于 2020-4-10 11:23

vany5921 发表于 2020-4-10 09:49
PCB工作量小一些

请问您的这块led板是怎么连接电路的?
感觉您的这种设计方式对我来说就比较理想了。
页: [1]
查看完整版本: IOT-emoji 显示装置