forked from btclock/btclock_v3
More efficient handling of multiple fonts, restore data bugfix
This commit is contained in:
parent
90d91ba216
commit
13c8e67b4c
9 changed files with 96 additions and 94 deletions
2
data
2
data
|
@ -1 +1 @@
|
|||
Subproject commit 48e585d4ec12bbc441499936d7cbf53d4307b9ec
|
||||
Subproject commit 0041ec3d9a174955383836bba02caf79f3961072
|
|
@ -15,9 +15,35 @@ struct FontData {
|
|||
const uint8_t yAdvance;
|
||||
};
|
||||
|
||||
// Font name constants
|
||||
namespace FontNames {
|
||||
static const String ANTONIO = "antonio";
|
||||
static const String OSWALD = "oswald";
|
||||
|
||||
static const std::array<String, 2> AVAILABLE_FONTS = {
|
||||
ANTONIO,
|
||||
OSWALD
|
||||
};
|
||||
|
||||
static const std::array<String, 2>& getAvailableFonts() {
|
||||
return AVAILABLE_FONTS;
|
||||
}
|
||||
}
|
||||
|
||||
class FontLoader {
|
||||
public:
|
||||
static GFXfont* loadCompressedFont(const FontData& fontData) {
|
||||
return loadCompressedFont(
|
||||
fontData.compressedData,
|
||||
fontData.glyphs,
|
||||
fontData.compressedSize,
|
||||
fontData.originalSize,
|
||||
fontData.first,
|
||||
fontData.last,
|
||||
fontData.yAdvance
|
||||
);
|
||||
}
|
||||
|
||||
static GFXfont* loadCompressedFont(
|
||||
const uint8_t* compressedData,
|
||||
const GFXglyph* glyphs,
|
||||
|
|
114
src/lib/epd.cpp
114
src/lib/epd.cpp
|
@ -161,73 +161,21 @@ void forceFullRefresh()
|
|||
|
||||
GFXfont font90;
|
||||
|
||||
void setupDisplays()
|
||||
{
|
||||
String fontName = preferences.getString("fontName", DEFAULT_FONT_NAME); // Default to antonio
|
||||
|
||||
if (fontName == "antonio")
|
||||
{
|
||||
void loadFonts(const String& fontName) {
|
||||
if (fontName == FontNames::ANTONIO) {
|
||||
// Load Antonio fonts
|
||||
antonioFonts.big = FontLoader::loadCompressedFont(
|
||||
Antonio_SemiBold90pt7b_Properties.compressedData,
|
||||
Antonio_SemiBold90pt7b_Properties.glyphs,
|
||||
Antonio_SemiBold90pt7b_Properties.compressedSize,
|
||||
Antonio_SemiBold90pt7b_Properties.originalSize,
|
||||
Antonio_SemiBold90pt7b_Properties.first,
|
||||
Antonio_SemiBold90pt7b_Properties.last,
|
||||
Antonio_SemiBold90pt7b_Properties.yAdvance);
|
||||
|
||||
antonioFonts.medium = FontLoader::loadCompressedFont(
|
||||
Antonio_SemiBold40pt7b_Properties.compressedData,
|
||||
Antonio_SemiBold40pt7b_Properties.glyphs,
|
||||
Antonio_SemiBold40pt7b_Properties.compressedSize,
|
||||
Antonio_SemiBold40pt7b_Properties.originalSize,
|
||||
Antonio_SemiBold40pt7b_Properties.first,
|
||||
Antonio_SemiBold40pt7b_Properties.last,
|
||||
Antonio_SemiBold40pt7b_Properties.yAdvance);
|
||||
|
||||
antonioFonts.small = FontLoader::loadCompressedFont(
|
||||
Antonio_SemiBold20pt7b_Properties.compressedData,
|
||||
Antonio_SemiBold20pt7b_Properties.glyphs,
|
||||
Antonio_SemiBold20pt7b_Properties.compressedSize,
|
||||
Antonio_SemiBold20pt7b_Properties.originalSize,
|
||||
Antonio_SemiBold20pt7b_Properties.first,
|
||||
Antonio_SemiBold20pt7b_Properties.last,
|
||||
Antonio_SemiBold20pt7b_Properties.yAdvance);
|
||||
antonioFonts.big = FontLoader::loadCompressedFont(Antonio_SemiBold90pt7b_Properties);
|
||||
antonioFonts.medium = FontLoader::loadCompressedFont(Antonio_SemiBold40pt7b_Properties);
|
||||
antonioFonts.small = FontLoader::loadCompressedFont(Antonio_SemiBold20pt7b_Properties);
|
||||
|
||||
FONT_BIG = antonioFonts.big;
|
||||
FONT_MEDIUM = antonioFonts.medium;
|
||||
FONT_SMALL = antonioFonts.small;
|
||||
}
|
||||
else if (fontName == "oswald")
|
||||
{
|
||||
} else if (fontName == FontNames::OSWALD) {
|
||||
// Load Oswald fonts
|
||||
oswaldFonts.big = FontLoader::loadCompressedFont(
|
||||
Oswald_Medium80pt7b_Properties.compressedData,
|
||||
Oswald_Medium80pt7b_Properties.glyphs,
|
||||
Oswald_Medium80pt7b_Properties.compressedSize,
|
||||
Oswald_Medium80pt7b_Properties.originalSize,
|
||||
Oswald_Medium80pt7b_Properties.first,
|
||||
Oswald_Medium80pt7b_Properties.last,
|
||||
Oswald_Medium80pt7b_Properties.yAdvance);
|
||||
|
||||
oswaldFonts.medium = FontLoader::loadCompressedFont(
|
||||
Oswald_Medium30pt7b_Properties.compressedData,
|
||||
Oswald_Medium30pt7b_Properties.glyphs,
|
||||
Oswald_Medium30pt7b_Properties.compressedSize,
|
||||
Oswald_Medium30pt7b_Properties.originalSize,
|
||||
Oswald_Medium30pt7b_Properties.first,
|
||||
Oswald_Medium30pt7b_Properties.last,
|
||||
Oswald_Medium30pt7b_Properties.yAdvance);
|
||||
|
||||
oswaldFonts.small = FontLoader::loadCompressedFont(
|
||||
Oswald_Medium20pt7b_Properties.compressedData,
|
||||
Oswald_Medium20pt7b_Properties.glyphs,
|
||||
Oswald_Medium20pt7b_Properties.compressedSize,
|
||||
Oswald_Medium20pt7b_Properties.originalSize,
|
||||
Oswald_Medium20pt7b_Properties.first,
|
||||
Oswald_Medium20pt7b_Properties.last,
|
||||
Oswald_Medium20pt7b_Properties.yAdvance);
|
||||
oswaldFonts.big = FontLoader::loadCompressedFont(Oswald_Medium80pt7b_Properties);
|
||||
oswaldFonts.medium = FontLoader::loadCompressedFont(Oswald_Medium30pt7b_Properties);
|
||||
oswaldFonts.small = FontLoader::loadCompressedFont(Oswald_Medium20pt7b_Properties);
|
||||
|
||||
FONT_BIG = oswaldFonts.big;
|
||||
FONT_MEDIUM = oswaldFonts.medium;
|
||||
|
@ -235,53 +183,49 @@ void setupDisplays()
|
|||
}
|
||||
|
||||
FONT_SATSYMBOL = &Satoshi_Symbol90pt7b;
|
||||
}
|
||||
|
||||
void setupDisplays() {
|
||||
// Load fonts based on preference
|
||||
String fontName = preferences.getString("fontName", DEFAULT_FONT_NAME);
|
||||
loadFonts(fontName);
|
||||
|
||||
// Initialize displays
|
||||
std::lock_guard<std::mutex> lockMcp(mcpMutex);
|
||||
|
||||
for (uint i = 0; i < NUM_SCREENS; i++)
|
||||
{
|
||||
for (uint i = 0; i < NUM_SCREENS; i++) {
|
||||
displays[i].init(0, true, 30);
|
||||
}
|
||||
|
||||
// Create update queue and task
|
||||
updateQueue = xQueueCreate(UPDATE_QUEUE_SIZE, sizeof(UpdateDisplayTaskItem));
|
||||
|
||||
xTaskCreate(prepareDisplayUpdateTask, "PrepareUpd", EPD_TASK_STACK_SIZE * 2, NULL, 11, NULL);
|
||||
|
||||
for (uint i = 0; i < NUM_SCREENS; i++)
|
||||
{
|
||||
// Create display update tasks
|
||||
for (uint i = 0; i < NUM_SCREENS; i++) {
|
||||
int *taskParam = new int;
|
||||
*taskParam = i;
|
||||
|
||||
xTaskCreate(updateDisplay, ("EpdUpd" + String(i)).c_str(), EPD_TASK_STACK_SIZE, taskParam,
|
||||
11, &tasks[i]); // create task
|
||||
xTaskCreate(updateDisplay, ("EpdUpd" + String(i)).c_str(), EPD_TASK_STACK_SIZE, taskParam, 11, &tasks[i]);
|
||||
}
|
||||
|
||||
// Hold lower button to enable "storage mode" (prevents burn-in of ePaper displays)
|
||||
if (mcp1.read1(0) == LOW)
|
||||
{
|
||||
// Check for storage mode (prevents burn-in)
|
||||
if (mcp1.read1(0) == LOW) {
|
||||
setFgColor(GxEPD_BLACK);
|
||||
setBgColor(GxEPD_WHITE);
|
||||
|
||||
epdContent.fill("");
|
||||
}
|
||||
else
|
||||
{
|
||||
// Get custom text from preferences or use default "BTCLOCK"
|
||||
} else {
|
||||
// Initialize with custom text or default
|
||||
String customText = preferences.getString("displayText", DEFAULT_BOOT_TEXT);
|
||||
|
||||
// Initialize array with spaces
|
||||
std::array<String, NUM_SCREENS> newContent;
|
||||
newContent.fill(" ");
|
||||
|
||||
// Fill in the custom text, truncating if longer than NUM_SCREENS
|
||||
for (size_t i = 0; i < std::min(customText.length(), (size_t)NUM_SCREENS); i++) {
|
||||
newContent[i] = String(customText[i]);
|
||||
newContent[i] = String(customText[i]);
|
||||
}
|
||||
|
||||
epdContent = newContent;
|
||||
}
|
||||
|
||||
setEpdContent(epdContent);
|
||||
setEpdContent(epdContent);
|
||||
}
|
||||
|
||||
void setEpdContent(std::array<String, NUM_SCREENS> newEpdContent)
|
||||
|
@ -527,8 +471,8 @@ void showDigit(const uint dispNum, char chr, bool partial, const GFXfont *font)
|
|||
|
||||
if (chr == '.')
|
||||
{
|
||||
displays[dispNum].fillRect(x, y, displays[dispNum].width(),
|
||||
round(displays[dispNum].height() * 0.9), getBgColor());
|
||||
displays[dispNum].fillRect(0, 0, displays[dispNum].width(),
|
||||
round(displays[dispNum].height() * 0.67), getBgColor());
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -31,7 +31,6 @@
|
|||
#ifdef USE_QR
|
||||
#include "qrcodegen.h"
|
||||
#endif
|
||||
// extern TaskHandle_t epdTaskHandle;
|
||||
|
||||
typedef struct {
|
||||
char dispNum;
|
||||
|
@ -39,6 +38,7 @@ typedef struct {
|
|||
|
||||
void forceFullRefresh();
|
||||
void setupDisplays();
|
||||
void loadFonts(const String& fontName);
|
||||
|
||||
void splitText(const uint dispNum, const String &top, const String &bottom,
|
||||
bool partial);
|
||||
|
|
|
@ -110,15 +110,17 @@ void processNewPrice(uint newPrice, char currency)
|
|||
if (lastUpdateMap.find(currency) == lastUpdateMap.end() ||
|
||||
(currentTime - lastUpdateMap[currency]) > minSecPriceUpd)
|
||||
{
|
||||
// const unsigned long oldPrice = currentPrice;
|
||||
currencyMap[currency] = newPrice;
|
||||
if (currency == CURRENCY_USD && ( lastUpdateMap[currency] == 0 ||
|
||||
(currentTime - lastUpdateMap[currency]) > 120))
|
||||
|
||||
// Store price in preferences if enough time has passed
|
||||
if (lastUpdateMap[currency] == 0 || (currentTime - lastUpdateMap[currency]) > 120)
|
||||
{
|
||||
preferences.putUInt("lastPrice", currentPrice);
|
||||
String prefKey = String("lastPrice_") + getCurrencyCode(currency).c_str();
|
||||
preferences.putUInt(prefKey.c_str(), newPrice);
|
||||
}
|
||||
|
||||
lastUpdateMap[currency] = currentTime;
|
||||
// if (abs((int)(oldPrice-currentPrice)) > round(0.0015*oldPrice)) {
|
||||
|
||||
if (workQueue != nullptr && (ScreenHandler::getCurrentScreen() == SCREEN_BTC_TICKER ||
|
||||
ScreenHandler::getCurrentScreen() == SCREEN_SATS_PER_CURRENCY ||
|
||||
ScreenHandler::getCurrentScreen() == SCREEN_MARKET_CAP))
|
||||
|
@ -126,7 +128,24 @@ void processNewPrice(uint newPrice, char currency)
|
|||
WorkItem priceUpdate = {TASK_PRICE_UPDATE, currency};
|
||||
xQueueSend(workQueue, &priceUpdate, portMAX_DELAY);
|
||||
}
|
||||
//}
|
||||
}
|
||||
}
|
||||
|
||||
void loadStoredPrices()
|
||||
{
|
||||
// Load prices for all supported currencies
|
||||
std::vector<std::string> currencies = getAvailableCurrencies();
|
||||
|
||||
for (const std::string ¤cy : currencies) {
|
||||
// Get first character as the currency identifier
|
||||
String prefKey = String("lastPrice_") + currency.c_str();
|
||||
uint storedPrice = preferences.getUInt(prefKey.c_str(), 0);
|
||||
|
||||
if (storedPrice > 0) {
|
||||
currencyMap[getCurrencyChar(currency)] = storedPrice;
|
||||
// Initialize lastUpdateMap to 0 so next update will store immediately
|
||||
lastUpdateMap[getCurrencyChar(currency)] = 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -28,3 +28,4 @@ void restartPriceNotify();
|
|||
|
||||
bool getPriceNotifyInit();
|
||||
uint getLastPriceUpdate(char currency);
|
||||
void loadStoredPrices();
|
|
@ -321,6 +321,7 @@ void taskScreenRotate(void *pvParameters) {
|
|||
|
||||
void setupTasks() {
|
||||
workQueue = xQueueCreate(WORK_QUEUE_SIZE, sizeof(WorkItem));
|
||||
loadStoredPrices();
|
||||
|
||||
xTaskCreate(workerTask, "workerTask", 4096, NULL, tskIDLE_PRIORITY,
|
||||
&workerTaskHandle);
|
||||
|
|
|
@ -97,6 +97,16 @@ namespace ArduinoJson {
|
|||
array.add(item);
|
||||
}
|
||||
};
|
||||
|
||||
template <size_t N>
|
||||
struct Converter<std::array<String, N>> {
|
||||
static void toJson(const std::array<String, N>& src, JsonVariant dst) {
|
||||
JsonArray array = dst.to<JsonArray>();
|
||||
for (const String& item : src) {
|
||||
array.add(item);
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
class HttpHelper {
|
||||
|
|
|
@ -676,6 +676,7 @@ void onApiSettingsGet(AsyncWebServerRequest *request)
|
|||
root["nostrZapPubkey"] = preferences.getString("nostrZapPubkey", DEFAULT_ZAP_NOTIFY_PUBKEY);
|
||||
root["ledFlashOnZap"] = preferences.getBool("ledFlashOnZap", DEFAULT_LED_FLASH_ON_ZAP);
|
||||
root["fontName"] = preferences.getString("fontName", DEFAULT_FONT_NAME);
|
||||
root["availableFonts"] = FontNames::getAvailableFonts();
|
||||
// Custom endpoint settings (only used for CUSTOM_SOURCE)
|
||||
root["customEndpoint"] = preferences.getString("customEndpoint", DEFAULT_CUSTOM_ENDPOINT);
|
||||
root["customEndpointDisableSSL"] = preferences.getBool("customEndpointDisableSSL", DEFAULT_CUSTOM_ENDPOINT_DISABLE_SSL);
|
||||
|
|
Loading…
Reference in a new issue