#include "M5Display.h" #ifdef M5Stack_M5Core2 #include #endif /* M5Stack_M5Core2 */ #define BLK_PWM_CHANNEL 7 // LEDC_CHANNEL_7 // So we can use this instance without including all of M5Core2 / M5Stack M5Display* M5Display::instance; M5Display::M5Display() : TFT_eSPI() { if (!instance) instance = this; } void M5Display::begin() { TFT_eSPI::begin(); setRotation(1); fillScreen(0); // Init the back-light LED PWM ledcSetup(BLK_PWM_CHANNEL, 44100, 8); ledcAttachPin(TFT_BL, BLK_PWM_CHANNEL); ledcWrite(BLK_PWM_CHANNEL, 80); } void M5Display::sleep() { startWrite(); writecommand(ILI9341_SLPIN); // Software reset endWrite(); } void M5Display::wakeup() { startWrite(); writecommand(ILI9341_SLPOUT); endWrite(); } void M5Display::setBrightness(uint8_t brightness) { ledcWrite(BLK_PWM_CHANNEL, brightness); } void M5Display::drawBitmap(int16_t x0, int16_t y0, int16_t w, int16_t h, const uint16_t *data) { bool swap = getSwapBytes(); setSwapBytes(true); pushImage((int32_t)x0, (int32_t)y0, (uint32_t)w, (uint32_t)h, data); setSwapBytes(swap); } void M5Display::drawBitmap(int16_t x0, int16_t y0, int16_t w, int16_t h, uint16_t *data) { bool swap = getSwapBytes(); setSwapBytes(true); pushImage((int32_t)x0, (int32_t)y0, (uint32_t)w, (uint32_t)h, data); setSwapBytes(swap); } void M5Display::drawBitmap(int16_t x0, int16_t y0, int16_t w, int16_t h, const uint16_t *data, uint16_t transparent) { bool swap = getSwapBytes(); setSwapBytes(true); pushImage((int32_t)x0, (int32_t)y0, (uint32_t)w, (uint32_t)h, data, transparent); setSwapBytes(swap); } void M5Display::drawBitmap(int16_t x0, int16_t y0, int16_t w, int16_t h, const uint8_t *data) { bool swap = getSwapBytes(); setSwapBytes(true); pushImage((int32_t)x0, (int32_t)y0, (uint32_t)w, (uint32_t)h, (const uint16_t*)data); setSwapBytes(swap); } void M5Display::drawBitmap(int16_t x0, int16_t y0, int16_t w, int16_t h, uint8_t *data) { bool swap = getSwapBytes(); setSwapBytes(true); pushImage((int32_t)x0, (int32_t)y0, (uint32_t)w, (uint32_t)h, (uint16_t*)data); setSwapBytes(swap); } void M5Display::progressBar(int x, int y, int w, int h, uint8_t val) { drawRect(x, y, w, h, 0x09F1); fillRect(x + 1, y + 1, w * (((float)val) / 100.0), h - 1, 0x09F1); } #include "utility/qrcode.h" void M5Display::qrcode(const char *string, uint16_t x, uint16_t y, uint8_t width, uint8_t version) { // Create the QR code QRCode qrcode; uint8_t qrcodeData[qrcode_getBufferSize(version)]; qrcode_initText(&qrcode, qrcodeData, version, 0, string); // Top quiet zone uint8_t thickness = width / qrcode.size; uint16_t lineLength = qrcode.size * thickness; uint8_t xOffset = x + (width-lineLength)/2; uint8_t yOffset = y + (width-lineLength)/2; fillRect(x, y, width, width, TFT_WHITE); for (uint8_t y = 0; y < qrcode.size; y++) { for (uint8_t x = 0; x < qrcode.size; x++) { uint8_t q = qrcode_getModule(&qrcode, x, y); if (q) fillRect(x * thickness + xOffset, y * thickness + yOffset, thickness, thickness, TFT_BLACK); } } } void M5Display::qrcode(const String &string, uint16_t x, uint16_t y, uint8_t width, uint8_t version) { int16_t len = string.length() + 2; char buffer[len]; string.toCharArray(buffer, len); qrcode(buffer, x, y, width, version); } // These read 16- and 32-bit types from the SD card file. // BMP data is stored little-endian, Arduino is little-endian too. // May need to reverse subscript order if porting elsewhere. uint16_t read16(fs::File &f) { uint16_t result; ((uint8_t *)&result)[0] = f.read(); // LSB ((uint8_t *)&result)[1] = f.read(); // MSB return result; } uint32_t read32(fs::File &f) { uint32_t result; ((uint8_t *)&result)[0] = f.read(); // LSB ((uint8_t *)&result)[1] = f.read(); ((uint8_t *)&result)[2] = f.read(); ((uint8_t *)&result)[3] = f.read(); // MSB return result; } // Bodmers BMP image rendering function void M5Display::drawBmpFile(fs::FS &fs, const char *path, uint16_t x, uint16_t y) { if ((x >= width()) || (y >= height())) return; // Open requested file on SD card File bmpFS = fs.open(path, "r"); if (!bmpFS) { Serial.print("File not found"); return; } uint32_t seekOffset; uint16_t w, h, row, col; uint8_t r, g, b; uint32_t startTime = millis(); if (read16(bmpFS) == 0x4D42) { read32(bmpFS); read32(bmpFS); seekOffset = read32(bmpFS); read32(bmpFS); w = read32(bmpFS); h = read32(bmpFS); if ((read16(bmpFS) == 1) && (read16(bmpFS) == 24) && (read32(bmpFS) == 0)) { y += h - 1; setSwapBytes(true); bmpFS.seek(seekOffset); uint16_t padding = (4 - ((w * 3) & 3)) & 3; uint8_t lineBuffer[w * 3 + padding]; for (row = 0; row < h; row++) { bmpFS.read(lineBuffer, sizeof(lineBuffer)); uint8_t* bptr = lineBuffer; uint16_t* tptr = (uint16_t*)lineBuffer; // Convert 24 to 16 bit colours for (col = 0; col < w; col++) { b = *bptr++; g = *bptr++; r = *bptr++; *tptr++ = ((r & 0xF8) << 8) | ((g & 0xFC) << 3) | (b >> 3); } // Push the pixel row to screen, pushImage will crop the line if needed // y is decremented as the BMP image is drawn bottom up pushImage(x, y--, w, 1, (uint16_t*)lineBuffer); } Serial.print("Loaded in "); Serial.print(millis() - startTime); Serial.println(" ms"); } else Serial.println("BMP format not recognized."); } bmpFS.close(); } // void M5Display::drawBmp(fs::FS &fs, const char *path, uint16_t x, uint16_t y) { // drawBmpFile(fs, path, x, y); // } /*************************************************** This library is written to be compatible with Adafruit's ILI9341 library and automatically detects the display type on ESP_WROVER_KITs Earlier WROVERs had ILI9341, while newer releases have ST7789V MIT license, all text above must be included in any redistribution ****************************************************/ /* * JPEG */ #include "rom/tjpgd.h" #define jpgColor(c) \ (((uint16_t)(((uint8_t *)(c))[0] & 0xF8) << 8) | \ ((uint16_t)(((uint8_t *)(c))[1] & 0xFC) << 3) | \ ((((uint8_t *)(c))[2] & 0xF8) >> 3)) #if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_ERROR const char *jd_errors[] = {"Succeeded", "Interrupted by output function", "Device error or wrong termination of input stream", "Insufficient memory pool for the image", "Insufficient stream input buffer", "Parameter error", "Data format error", "Right format but not supported", "Not supported JPEG standard"}; #endif typedef struct { uint16_t x; uint16_t y; uint16_t maxWidth; uint16_t maxHeight; uint16_t offX; uint16_t offY; jpeg_div_t scale; const void *src; size_t len; size_t index; M5Display *tft; uint16_t outWidth; uint16_t outHeight; } jpg_file_decoder_t; static uint32_t jpgReadFile(JDEC *decoder, uint8_t *buf, uint32_t len) { jpg_file_decoder_t *jpeg = (jpg_file_decoder_t *)decoder->device; File *file = (File *)jpeg->src; if (buf) { return file->read(buf, len); } else { file->seek(len, SeekCur); } return len; } static uint32_t jpgRead(JDEC *decoder, uint8_t *buf, uint32_t len) { jpg_file_decoder_t *jpeg = (jpg_file_decoder_t *)decoder->device; if (buf) { memcpy(buf, (const uint8_t *)jpeg->src + jpeg->index, len); } jpeg->index += len; return len; } static uint32_t jpgWrite(JDEC *decoder, void *bitmap, JRECT *rect) { jpg_file_decoder_t *jpeg = (jpg_file_decoder_t *)decoder->device; uint16_t x = rect->left; uint16_t y = rect->top; uint16_t w = rect->right + 1 - x; uint16_t h = rect->bottom + 1 - y; uint16_t oL = 0, oR = 0; uint8_t *data = (uint8_t *)bitmap; if (rect->right < jpeg->offX) { return 1; } if (rect->left >= (jpeg->offX + jpeg->outWidth)) { return 1; } if (rect->bottom < jpeg->offY) { return 1; } if (rect->top >= (jpeg->offY + jpeg->outHeight)) { return 1; } if (rect->top < jpeg->offY) { uint16_t linesToSkip = jpeg->offY - rect->top; data += linesToSkip * w * 3; h -= linesToSkip; y += linesToSkip; } if (rect->bottom >= (jpeg->offY + jpeg->outHeight)) { uint16_t linesToSkip = (rect->bottom + 1) - (jpeg->offY + jpeg->outHeight); h -= linesToSkip; } if (rect->left < jpeg->offX) { oL = jpeg->offX - rect->left; } if (rect->right >= (jpeg->offX + jpeg->outWidth)) { oR = (rect->right + 1) - (jpeg->offX + jpeg->outWidth); } uint16_t pixBuf[32]; uint8_t pixIndex = 0; uint16_t line; jpeg->tft->startWrite(); // jpeg->tft->setAddrWindow(x - jpeg->offX + jpeg->x + oL, y - jpeg->offY + // jpeg->y, w - (oL + oR), h); jpeg->tft->setWindow(x - jpeg->offX + jpeg->x + oL, y - jpeg->offY + jpeg->y, x - jpeg->offX + jpeg->x + oL + w - (oL + oR) - 1, y - jpeg->offY + jpeg->y + h - 1); while (h--) { data += 3 * oL; line = w - (oL + oR); while (line--) { pixBuf[pixIndex++] = jpgColor(data); data += 3; if (pixIndex == 32) { jpeg->tft->writePixels(pixBuf, 32); // SPI.writePixels((uint8_t *)pixBuf, 64); pixIndex = 0; } } data += 3 * oR; } if (pixIndex) { jpeg->tft->writePixels(pixBuf, pixIndex); // SPI.writePixels((uint8_t *)pixBuf, pixIndex * 2); } jpeg->tft->endWrite(); return 1; } static bool jpgDecode(jpg_file_decoder_t *jpeg, uint32_t (*reader)(JDEC *, uint8_t *, uint32_t)) { static uint8_t work[3100]; JDEC decoder; JRESULT jres = jd_prepare(&decoder, reader, work, 3100, jpeg); if (jres != JDR_OK) { log_e("jd_prepare failed! %s", jd_errors[jres]); return false; } uint16_t jpgWidth = decoder.width / (1 << (uint8_t)(jpeg->scale)); uint16_t jpgHeight = decoder.height / (1 << (uint8_t)(jpeg->scale)); if (jpeg->offX >= jpgWidth || jpeg->offY >= jpgHeight) { log_e("Offset Outside of JPEG size"); return false; } size_t jpgMaxWidth = jpgWidth - jpeg->offX; size_t jpgMaxHeight = jpgHeight - jpeg->offY; jpeg->outWidth = (jpgMaxWidth > jpeg->maxWidth) ? jpeg->maxWidth : jpgMaxWidth; jpeg->outHeight = (jpgMaxHeight > jpeg->maxHeight) ? jpeg->maxHeight : jpgMaxHeight; jres = jd_decomp(&decoder, jpgWrite, (uint8_t)jpeg->scale); if (jres != JDR_OK) { log_e("jd_decomp failed! %s", jd_errors[jres]); return false; } return true; } void M5Display::drawJpg(const uint8_t *jpg_data, size_t jpg_len, uint16_t x, uint16_t y, uint16_t maxWidth, uint16_t maxHeight, uint16_t offX, uint16_t offY, jpeg_div_t scale) { if ((x + maxWidth) > width() || (y + maxHeight) > height()) { log_e("Bad dimensions given"); return; } jpg_file_decoder_t jpeg; if (!maxWidth) { maxWidth = width() - x; } if (!maxHeight) { maxHeight = height() - y; } jpeg.src = jpg_data; jpeg.len = jpg_len; jpeg.index = 0; jpeg.x = x; jpeg.y = y; jpeg.maxWidth = maxWidth; jpeg.maxHeight = maxHeight; jpeg.offX = offX; jpeg.offY = offY; jpeg.scale = scale; jpeg.tft = this; jpgDecode(&jpeg, jpgRead); } void M5Display::drawJpgFile(fs::FS &fs, const char *path, uint16_t x, uint16_t y, uint16_t maxWidth, uint16_t maxHeight, uint16_t offX, uint16_t offY, jpeg_div_t scale) { if ((x + maxWidth) > width() || (y + maxHeight) > height()) { log_e("Bad dimensions given"); return; } File file = fs.open(path); if (!file) { log_e("Failed to open file for reading"); return; } jpg_file_decoder_t jpeg; if (!maxWidth) { maxWidth = width() - x; } if (!maxHeight) { maxHeight = height() - y; } jpeg.src = &file; jpeg.len = file.size(); jpeg.index = 0; jpeg.x = x; jpeg.y = y; jpeg.maxWidth = maxWidth; jpeg.maxHeight = maxHeight; jpeg.offX = offX; jpeg.offY = offY; jpeg.scale = scale; jpeg.tft = this; jpgDecode(&jpeg, jpgReadFile); file.close(); } /* * PNG */ #include "utility/pngle.h" #include typedef struct _png_draw_params { uint16_t x; uint16_t y; uint16_t maxWidth; uint16_t maxHeight; uint16_t offX; uint16_t offY; double scale; uint8_t alphaThreshold; M5Display *tft; } png_file_decoder_t; static void pngle_draw_callback(pngle_t *pngle, uint32_t x, uint32_t y, uint32_t w, uint32_t h, uint8_t rgba[4]) { png_file_decoder_t *p = (png_file_decoder_t *)pngle_get_user_data(pngle); uint16_t color = jpgColor(rgba); // XXX: It's PNG ;) if (x < p->offX || y < p->offY) return ; x -= p->offX; y -= p->offY; // An interlaced file with alpha channel causes disaster, so use 1 here for simplicity w = 1; h = 1; if (p->scale != 1.0) { x = (uint32_t)ceil(x * p->scale); y = (uint32_t)ceil(y * p->scale); w = (uint32_t)ceil(w * p->scale); h = (uint32_t)ceil(h * p->scale); } if (x >= p->maxWidth || y >= p->maxHeight) return ; if (x + w >= p->maxWidth) w = p->maxWidth - x; if (y + h >= p->maxHeight) h = p->maxHeight - y; x += p->x; y += p->y; if (rgba[3] >= p->alphaThreshold) { p->tft->fillRect(x, y, w, h, color); } } void M5Display::drawPngFile(fs::FS &fs, const char *path, uint16_t x, uint16_t y, uint16_t maxWidth, uint16_t maxHeight, uint16_t offX, uint16_t offY, double scale, uint8_t alphaThreshold) { File file = fs.open(path); if (!file) { log_e("Failed to open file for reading"); return ; } pngle_t *pngle = pngle_new(); png_file_decoder_t png; if (!maxWidth) { maxWidth = width() - x; } if (!maxHeight) { maxHeight = height() - y; } png.x = x; png.y = y; png.maxWidth = maxWidth; png.maxHeight = maxHeight; png.offX = offX; png.offY = offY; png.scale = scale; png.alphaThreshold = alphaThreshold; png.tft = this; pngle_set_user_data(pngle, &png); pngle_set_draw_callback(pngle, pngle_draw_callback); // Feed data to pngle uint8_t buf[1024]; int remain = 0; int len; while ((len = file.read(buf + remain, sizeof(buf) - remain)) > 0) { int fed = pngle_feed(pngle, buf, remain + len); if (fed < 0) { log_e("[pngle error] %s", pngle_error(pngle)); break; } remain = remain + len - fed; if (remain > 0) memmove(buf, buf + fed, remain); } pngle_destroy(pngle); file.close(); } void M5Display::drawPngUrl(const char *url, uint16_t x, uint16_t y, uint16_t maxWidth, uint16_t maxHeight, uint16_t offX, uint16_t offY, double scale, uint8_t alphaThreshold) { HTTPClient http; if (WiFi.status() != WL_CONNECTED) { log_e("Not connected"); return ; } http.begin(url); int httpCode = http.GET(); if (httpCode != HTTP_CODE_OK) { log_e("HTTP ERROR: %d\n", httpCode); http.end(); return ; } WiFiClient *stream = http.getStreamPtr(); pngle_t *pngle = pngle_new(); png_file_decoder_t png; if (!maxWidth) { maxWidth = width() - x; } if (!maxHeight) { maxHeight = height() - y; } png.x = x; png.y = y; png.maxWidth = maxWidth; png.maxHeight = maxHeight; png.offX = offX; png.offY = offY; png.scale = scale; png.alphaThreshold = alphaThreshold; png.tft = this; pngle_set_user_data(pngle, &png); pngle_set_draw_callback(pngle, pngle_draw_callback); // Feed data to pngle uint8_t buf[1024]; int remain = 0; int len; while (http.connected()) { size_t size = stream->available(); if (!size) { delay(1); continue; } if (size > sizeof(buf) - remain) size = sizeof(buf) - remain; if ((len = stream->readBytes(buf + remain, size)) > 0) { int fed = pngle_feed(pngle, buf, remain + len); if (fed < 0) { log_e("[pngle error] %s", pngle_error(pngle)); break; } remain = remain + len - fed; if (remain > 0) memmove(buf, buf + fed, remain); } } pngle_destroy(pngle); http.end(); } // Saves and restores font properties, datum, cursor, colors void M5Display::pushState() { DisplayState s; s.textfont = textfont; s.textsize = textsize; s.textcolor = textcolor; s.textbgcolor = textbgcolor; s.cursor_x = cursor_x; s.cursor_y = cursor_y; s.padX = padX; s.gfxFont = gfxFont; _displayStateStack.push_back(s); } void M5Display::popState() { if (_displayStateStack.empty()) return; DisplayState s = _displayStateStack.back(); _displayStateStack.pop_back(); textfont = s.textfont; textsize = s.textsize; textcolor = s.textcolor; textbgcolor = s.textbgcolor; cursor_x = s.cursor_x; cursor_y = s.cursor_y; padX = s.padX; if (s.gfxFont && s.gfxFont != gfxFont) setFreeFont(s.gfxFont); } #ifdef M5Stack_M5Core2 #ifdef TFT_eSPI_TOUCH_EMULATION // Emulates the native (resistive) TFT_eSPI touch interface using M5.Touch uint8_t M5Display::getTouchRaw(uint16_t *x, uint16_t *y) { return getTouch(x, y); } uint16_t M5Display::getTouchRawZ(void) { return (TOUCH->ispressed()) ? 1000 : 0; } void M5Display::convertRawXY(uint16_t *x, uint16_t *y) { return; } uint8_t M5Display::getTouch(uint16_t *x, uint16_t *y, uint16_t threshold /* = 600 */) { TOUCH->read(); if (TOUCH->points) { *x = TOUCH->point[0].x; *y = TOUCH->point[0].y; return true; } return false; } void M5Display::calibrateTouch(uint16_t *data, uint32_t color_fg, uint32_t color_bg, uint8_t size) { return; } void M5Display::setTouch(uint16_t *data) { return; } #endif /* TFT_eSPI_TOUCH_EMULATION */ #endif /* M5Stack_M5Core2 */