#include "../../options.h" #if DSP_MODEL==DSP_SSD1306 || DSP_MODEL==DSP_SSD1306x32 #include "displaySSD1306.h" #include #include "../../player.h" #include "../../config.h" #include "../../network.h" #ifndef SCREEN_ADDRESS #define SCREEN_ADDRESS 0x3C ///< See datasheet for Address; 0x3D for 128x64, 0x3C for 128x32 or scan it https://create.arduino.cc/projecthub/abdularbi17/how-to-scan-i2c-address-in-arduino-eaadda #endif #if DSP_MODEL==DSP_SSD1306 #define LOGO_WIDTH 21 #define LOGO_HEIGHT 32 const unsigned char logo [] PROGMEM= { 0x06, 0x03, 0x00, 0x0f, 0x07, 0x80, 0x1f, 0x8f, 0xc0, 0x1f, 0x8f, 0xc0, 0x0f, 0x07, 0x80, 0x06, 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xf8, 0x00, 0x03, 0xff, 0x00, 0x0f, 0xff, 0x80, 0x1f, 0xff, 0xc0, 0x1f, 0xff, 0xc0, 0x3f, 0x8f, 0xe0, 0x7e, 0x03, 0xf0, 0x7c, 0x01, 0xf0, 0x7c, 0x01, 0xf0, 0x7f, 0xff, 0xf0, 0xff, 0xff, 0xf8, 0xff, 0xff, 0xf8, 0xff, 0xff, 0xf8, 0x7c, 0x00, 0x00, 0x7c, 0x00, 0x00, 0x7e, 0x00, 0x00, 0x7f, 0x00, 0x00, 0x3f, 0xc0, 0xe0, 0x3f, 0xff, 0xe0, 0x1f, 0xff, 0xe0, 0x0f, 0xff, 0xe0, 0x03, 0xff, 0xc0, 0x00, 0xfe, 0x00 }; #endif #ifndef I2CFREQ_HZ #define I2CFREQ_HZ 4000000UL #endif TwoWire I2CSSD1306 = TwoWire(0); DspCore::DspCore(): Adafruit_SSD1306(128, ((DSP_MODEL==DSP_SSD1306)?64:32), &I2CSSD1306, I2C_RST, I2CFREQ_HZ) { } char* DspCore::utf8Rus(const char* str, bool uppercase) { int index = 0; static char strn[BUFLEN]; bool E = false; strlcpy(strn, str, BUFLEN); if (uppercase) { bool next = false; for (char *iter = strn; *iter != '\0'; ++iter) { if (E) { E = false; continue; } byte rus = (byte) * iter; if (rus == 208 && (byte) * (iter + 1) == 129) { // ёКостыли *iter = (char)209; *(iter + 1) = (char)145; E = true; continue; } if (rus == 209 && (byte) * (iter + 1) == 145) { *iter = (char)209; *(iter + 1) = (char)145; E = true; continue; } if (next) { if (rus >= 128 && rus <= 143) *iter = (char)(rus + 32); if (rus >= 176 && rus <= 191) *iter = (char)(rus - 32); next = false; } if (rus == 208) next = true; if (rus == 209) { *iter = (char)208; next = true; } *iter = toupper(*iter); } } while (strn[index]) { if (strn[index] >= 0xBF) { switch (strn[index]) { case 0xD0: { if (strn[index + 1] == 0x81) { strn[index] = 0xA8; break; } if (strn[index + 1] >= 0x90 && strn[index + 1] <= 0xBF) strn[index] = strn[index + 1] + 0x30; break; } case 0xD1: { if (strn[index + 1] == 0x91) { //strn[index] = 0xB7; strn[index] = 0xB8; break; } if (strn[index + 1] >= 0x80 && strn[index + 1] <= 0x8F) strn[index] = strn[index + 1] + 0x70; break; } } int sind = index + 2; while (strn[sind]) { strn[sind - 1] = strn[sind]; sind++; } strn[sind - 1] = 0; } index++; } return strn; } void DspCore::apScreen() { setTextSize(1); setTextColor(TFT_FG, TFT_BG); setCursor(TFT_FRAMEWDT, TFT_FRAMEWDT + ((DSP_MODEL==DSP_SSD1306)?2:1) * TFT_LINEHGHT); print("AP NAME: "); print(apSsid); setCursor(TFT_FRAMEWDT, TFT_FRAMEWDT + ((DSP_MODEL==DSP_SSD1306)?3:2) * TFT_LINEHGHT); print("PASSWORD: "); print(apPassword); setTextColor(SILVER, TFT_BG); #if DSP_MODEL==DSP_SSD1306 setCursor(TFT_FRAMEWDT, sheight - TFT_LINEHGHT * 2); print("SETTINGS PAGE ON: "); #endif setCursor(TFT_FRAMEWDT, sheight - TFT_LINEHGHT); print("http://"); print(WiFi.softAPIP().toString().c_str()); print("/"); } void DspCore::initD(uint16_t &screenwidth, uint16_t &screenheight) { I2CSSD1306.begin(I2C_SDA, I2C_SCL); if (!begin(SSD1306_SWITCHCAPVCC, SCREEN_ADDRESS)) { Serial.println(F("SSD1306 allocation failed")); for (;;); // Don't proceed, loop forever } maxcontrast = DSP_MODEL==DSP_SSD1306?0xCF:0x8F; config.theme.background = TFT_BG; config.theme.meta = TFT_FG; config.theme.title1 = TFT_FG; config.theme.title2 = TFT_FG; config.theme.rssi = TFT_FG; for(byte i=0;i<5;i++) config.theme.playlist[i] = TFT_FG; cp437(true); fillScreen(TFT_BG); flip(); invert(); setTextWrap(false); screenwidth = width(); screenheight = height(); swidth = screenwidth; sheight = screenheight; fillSpaces = true; } void DspCore::drawLogo() { clearDisplay(); #if DSP_MODEL==DSP_SSD1306 drawBitmap( (width() - LOGO_WIDTH ) / 2, 8, logo, LOGO_WIDTH, LOGO_HEIGHT, 1); #else setTextSize(2); centerText(utf8Rus("ёRadio", false), 0, TFT_FG, TFT_BG); setTextSize(1); #endif display(); } void DspCore::drawPlaylist(uint16_t currentItem, char* currentItemText) { for (byte i = 0; i < PLMITEMS; i++) { plMenu[i][0] = '\0'; } config.fillPlMenu(plMenu, currentItem - 2, PLMITEMS); setTextSize((DSP_MODEL==DSP_SSD1306)?2:1); int yStart = (sheight / 2 - PLMITEMHEIGHT / 2) - PLMITEMHEIGHT * (PLMITEMS - 1) / 2 + 3; fillRect(0, (sheight / 2 - PLMITEMHEIGHT / 2) + 1, swidth, PLMITEMHEIGHT, TFT_LOGO); setTextColor(TFT_FG, TFT_BG); for (byte i = 0; i < PLMITEMS; i++) { if (i == 2) { strlcpy(currentItemText, plMenu[i], PLMITEMLENGHT - 1); } else { setCursor(TFT_FRAMEWDT, yStart + i * PLMITEMHEIGHT); fillRect(0, yStart + i * PLMITEMHEIGHT, swidth, PLMITEMHEIGHT - 1, TFT_BG); print(utf8Rus(plMenu[i], true)); } } } void DspCore::clearDsp() { fillScreen(TFT_BG); } void DspCore::drawScrollFrame(uint16_t texttop, uint16_t textheight, uint16_t bg) { if (TFT_FRAMEWDT == 0) return; fillRect(0, texttop, TFT_FRAMEWDT, textheight, bg); fillRect(swidth - TFT_FRAMEWDT, texttop, TFT_FRAMEWDT, textheight, bg); } void DspCore::getScrolBbounds(const char* text, const char* separator, byte textsize, uint16_t &tWidth, uint16_t &tHeight, uint16_t &sWidth) { int16_t x1, y1; uint16_t w, h; setTextSize(textsize); getTextBounds(text, 0, 0, &x1, &y1, &w, &h); tWidth = w; tHeight = h; getTextBounds(separator, 0, 0, &x1, &y1, &w, &h); sWidth = w; } void DspCore::clearScroll(uint16_t texttop, uint16_t textheight, uint16_t bg) { fillRect(0, texttop, swidth, textheight, bg); } void DspCore::centerText(const char* text, byte y, uint16_t fg, uint16_t bg) { int16_t x1, y1; uint16_t w, h; const char* txt = text; getTextBounds(txt, 0, 0, &x1, &y1, &w, &h); setTextColor(fg); setCursor((swidth - w) / 2, y); fillRect(0, y, swidth, h, bg); print(txt); } void DspCore::rightText(const char* text, byte y, uint16_t fg, uint16_t bg) { int16_t x1, y1; uint16_t w, h; getTextBounds(text, 0, 0, &x1, &y1, &w, &h); setTextColor(fg); setCursor(swidth - w - TFT_FRAMEWDT, y); fillRect(swidth - w - TFT_FRAMEWDT-((DSP_MODEL==DSP_SSD1306)?0:2), y, w+((DSP_MODEL==DSP_SSD1306)?0:2), h, bg); print(text); } void DspCore::displayHeapForDebug() { } void DspCore::printClock(const char* timestr) { #if DSP_MODEL==DSP_SSD1306 setTextSize(2); centerText(timestr, 34, TFT_FG, TFT_BG); setTextSize(1); #else setTextSize(1); rightText(timestr, 0, TFT_FG, TFT_BG); #endif } void DspCore::printClock(struct tm timeinfo, bool dots, bool redraw) { #if DSP_MODEL==DSP_SSD1306x32 strftime(insideClc, sizeof(insideClc), dots?" %H %M":" %H:%M", &timeinfo); #endif } void DspCore::drawVolumeBar(bool withNumber) { int16_t vTop = sheight - 4; (void)vTop; int16_t vWidth = swidth-TFT_FRAMEWDT*2; #if DSP_MODEL==DSP_SSD1306 uint8_t ww = map(config.store.volume, 0, 254, 0, vWidth - 2); fillRect(TFT_FRAMEWDT, vTop, vWidth, 3, TFT_BG); drawRect(TFT_FRAMEWDT, vTop, vWidth, 3, TFT_LOGO); fillRect(TFT_FRAMEWDT + 1, vTop + 1, ww, 1, TFT_LOGO); #else uint8_t ww = map(config.store.volume, 0, 254, 0, vWidth); if(fillSpaces) { drawFastHLine(TFT_FRAMEWDT, sheight-1, swidth-TFT_FRAMEWDT*2, TFT_BG); drawFastHLine(TFT_FRAMEWDT, sheight-1, ww, TFT_LOGO); } #endif if (withNumber) { setTextSize(2); setTextColor(TFT_FG); char volstr[4]; uint16_t wv, hv; int16_t x1, y1; sprintf(volstr, "%d", config.store.volume); getTextBounds(volstr, 0, 0, &x1, &y1, &wv, &hv); fillRect(TFT_FRAMEWDT, VOL_TOP, swidth - TFT_FRAMEWDT / 2, hv + 3, TFT_BG); setCursor((swidth - wv) / 2, VOL_TOP); print(volstr); } } void DspCore::drawNextStationNum(uint16_t num) { setTextSize(2); setTextColor(TFT_FG); char numstr[7]; uint16_t wv, hv; int16_t x1, y1; sprintf(numstr, "%d", num); getTextBounds(numstr, 0, 0, &x1, &y1, &wv, &hv); fillRect(TFT_FRAMEWDT, VOL_TOP, swidth - TFT_FRAMEWDT / 2, hv + 3, TFT_BG); setCursor((swidth - wv) / 2, VOL_TOP); print(numstr); } void DspCore::frameTitle(const char* str) { setTextSize((DSP_MODEL==DSP_SSD1306?2:1)); centerText(str, TFT_FRAMEWDT, TFT_LOGO, TFT_BG); } void DspCore::rssi(const char* str) { if(!fillSpaces && DSP_MODEL==DSP_SSD1306x32) return; char buf[4]; strlcpy(buf, str, strlen(str)-2); int16_t vTop = sheight - TFT_LINEHGHT - ((DSP_MODEL==DSP_SSD1306)?4:2); setTextSize(1); rightText(buf, vTop, SILVER, TFT_BG); } void DspCore::ip(const char* str) { if(!fillSpaces && DSP_MODEL==DSP_SSD1306x32) return; int16_t vTop = sheight - TFT_LINEHGHT - ((DSP_MODEL==DSP_SSD1306)?4:2); setTextSize(1); setTextColor(SILVER, TFT_BG); setCursor(0, vTop); print(str); } void DspCore::set_TextSize(uint8_t s) { setTextSize(s); } void DspCore::set_TextColor(uint16_t fg, uint16_t bg) { setTextColor(fg, bg); } void DspCore::set_Cursor(int16_t x, int16_t y) { setCursor(x, y); } void DspCore::printText(const char* txt) { print(txt); } void DspCore::loop(bool force) { if (checkdelay(SCROLLTIME, loopdelay) || force) { #if DSP_MODEL==DSP_SSD1306x32 if(fillSpaces) printClock(insideClc); #endif display(); } yield(); } boolean DspCore::checkdelay(int m, unsigned long &tstamp) { if (millis() - tstamp > m) { tstamp = millis(); return true; } else { return false; } } void DspCore::flip(){ setRotation(config.store.flipscreen?2:0); } void DspCore::invert(){ invertDisplay(config.store.invertdisplay); } void DspCore::sleep(void) { ssd1306_command(SSD1306_DISPLAYOFF); } void DspCore::wake(void) { ssd1306_command(SSD1306_DISPLAYON); } #endif