Compare commits
6 commits
Author | SHA1 | Date | |
---|---|---|---|
|
c0c542a350 | ||
|
483622f430 | ||
|
7d0d07e928 | ||
|
ff5754f2d1 | ||
|
879aeb6c3c | ||
|
415aa3a606 |
29 changed files with 237 additions and 1683 deletions
|
@ -63,8 +63,8 @@
|
|||
<div class="progress-bar progress-bar-striped" style="width: {{ memUsage }}%">{{ memUsage }}%</div>
|
||||
</div>
|
||||
<div class="d-flex justify-content-between">
|
||||
<div>Memory usage</div>
|
||||
<div>{{ memFree }} / {{ memTotal }} kB</div>
|
||||
<div>Memory free</div>
|
||||
<div>{{ memFree }} / {{ memTotal }} KiB</div>
|
||||
</div>
|
||||
</div>
|
||||
<hr>
|
||||
|
@ -191,6 +191,13 @@
|
|||
<input type="text" name="mempoolInstance" id="mempoolInstance" class="form-control">
|
||||
</div>
|
||||
</div>
|
||||
<div class="row">
|
||||
<label for="hostnamePrefix" class="col-sm-6 col-form-label col-form-label-sm">Hostname prefix</label>
|
||||
<div class="col-sm-6">
|
||||
<input type="text" name="hostnamePrefix" id="hostnamePrefix" class="form-control">
|
||||
<div class="form-text">A restart is required to apply new hostname prefix.</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="row">
|
||||
<div class=" col-sm-6">
|
||||
<div class="form-check form-switch">
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
import './helpers.js';
|
||||
|
||||
var screens = ["Block Height", "Moscow Time", "Ticker", "Time", "Halving countdown"];
|
||||
var screens = ["Block Height", "Moscow Time", "Ticker", "Time", "Halving countdown", "Market Cap"];
|
||||
|
||||
toTime = (secs) => {
|
||||
var hours = Math.floor(secs / (60 * 60));
|
||||
|
@ -28,7 +28,7 @@ getBcStatus = () => {
|
|||
var source = document.getElementById("entry-template").innerHTML;
|
||||
var template = Handlebars.compile(source);
|
||||
|
||||
var context = { timerRunning: jsonData.timerRunning, memUsage: Math.round(jsonData.espFreeHeap / jsonData.espHeapSize * 100), memFree: (jsonData.espFreeHeap / 1000), memTotal: (jsonData.espHeapSize / 1000), uptime: toTime(jsonData.espUptime), currentScreen: jsonData.currentScreen, rendered: jsonData.rendered, data: jsonData.data, screens: screens, ledStatus: jsonData.ledStatus ? jsonData.ledStatus.map((t) => (t).toString(16)) : [] };
|
||||
var context = { timerRunning: jsonData.timerRunning, memUsage: Math.round(jsonData.espFreeHeap / jsonData.espHeapSize * 100), memFree: (jsonData.espFreeHeap / 1024), memTotal: (jsonData.espHeapSize / 1024), uptime: toTime(jsonData.espUptime), currentScreen: jsonData.currentScreen, rendered: jsonData.rendered, data: jsonData.data, screens: screens, ledStatus: jsonData.ledStatus ? jsonData.ledStatus.map((t) => (t).toString(16)) : [] };
|
||||
|
||||
|
||||
document.getElementById('output').innerHTML = template(context);
|
||||
|
@ -76,6 +76,7 @@ fetch('/api/settings', {
|
|||
document.getElementById('fullRefreshMin').value = jsonData.fullRefreshMin;
|
||||
document.getElementById('wpTimeout').value = jsonData.wpTimeout;
|
||||
document.getElementById('mempoolInstance').value = jsonData.mempoolInstance;
|
||||
document.getElementById('hostnamePrefix').value = jsonData.hostnamePrefix;
|
||||
|
||||
if (jsonData.gitRev)
|
||||
document.getElementById('gitRev').innerHTML = "Version: " + jsonData.gitRev;
|
||||
|
|
BIN
doc/Rev.B/IMG_3310.jpeg
Normal file
BIN
doc/Rev.B/IMG_3310.jpeg
Normal file
Binary file not shown.
After Width: | Height: | Size: 1.3 MiB |
BIN
doc/Rev.B/IMG_3313.jpeg
Normal file
BIN
doc/Rev.B/IMG_3313.jpeg
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.2 MiB |
BIN
doc/Rev.B/IMG_3314.jpeg
Normal file
BIN
doc/Rev.B/IMG_3314.jpeg
Normal file
Binary file not shown.
After Width: | Height: | Size: 2 MiB |
BIN
doc/Rev.B/IMG_3317.jpeg
Normal file
BIN
doc/Rev.B/IMG_3317.jpeg
Normal file
Binary file not shown.
After Width: | Height: | Size: 1.9 MiB |
|
@ -13,7 +13,7 @@ data_dir = data/build
|
|||
|
||||
[env]
|
||||
platform = espressif32
|
||||
framework = arduino, espidf
|
||||
framework = arduino
|
||||
monitor_speed = 115200
|
||||
upload_speed = 921600
|
||||
monitor_filters = esp32_exception_decoder, colorize
|
||||
|
@ -22,7 +22,7 @@ build_flags = !python scripts/git_rev.py
|
|||
|
||||
[esp32wemos-s3-mini_BW_base]
|
||||
platform = espressif32
|
||||
framework = arduino, espidf
|
||||
framework = arduino
|
||||
board = lolin_s3_mini
|
||||
board_build.partitions = partition.csv
|
||||
lib_deps =
|
||||
|
@ -37,7 +37,7 @@ build_flags =
|
|||
-DLAST_BUILD_TIME=$UNIX_TIME
|
||||
-D IS_S3
|
||||
-D IS_BW
|
||||
-D CONFIG_FREERTOS_USE_TRACE_FACILITY
|
||||
# -D CONFIG_FREERTOS_USE_TRACE_FACILITY
|
||||
-D WITH_RGB_LED
|
||||
-D NEOPIXEL_COUNT=4
|
||||
-DASYNCWEBSERVER_REGEX
|
||||
|
@ -47,9 +47,7 @@ build_flags =
|
|||
-mfix-esp32-psram-cache-issue
|
||||
-fexceptions
|
||||
-DPIO_FRAMEWORK_ARDUINO_ENABLE_EXCEPTIONS
|
||||
build_unflags =
|
||||
-fno-exceptions
|
||||
-Werror=all
|
||||
build_unflags = -fno-exceptions
|
||||
|
||||
zinggjm/GxEPD2@^1.5.2
|
||||
|
||||
|
|
1525
sdkconfig.default
1525
sdkconfig.default
File diff suppressed because it is too large
Load diff
|
@ -1,21 +0,0 @@
|
|||
CONFIG_AUTOSTART_ARDUINO=y
|
||||
# CONFIG_WS2812_LED_ENABLE is not set
|
||||
CONFIG_FREERTOS_HZ=1000
|
||||
CONFIG_FREERTOS_USE_TRACE_FACILITY=y
|
||||
CONFIG_FREERTOS_USE_STATS_FORMATTING_FUNCTIONS=y
|
||||
CONFIG_FREERTOS_GENERATE_RUN_TIME_STATS=y
|
||||
|
||||
CONFIG_MBEDTLS_PSK_MODES=y
|
||||
CONFIG_MBEDTLS_KEY_EXCHANGE_PSK=y
|
||||
|
||||
CONFIG_BOOTLOADER_LOG_LEVEL=0
|
||||
CONFIG_LOG_BOOTLOADER_LEVEL_NONE=y
|
||||
CONFIG_BOOTLOADER_LOG_LEVEL_NONE=y
|
||||
CONFIG_ESP32S3_SPIRAM_SUPPORT=y
|
||||
CONFIG_LOG_DEFAULT_LEVEL_NONE=y
|
||||
CONFIG_LOG_DEFAULT_LEVEL=0
|
||||
CONFIG_LOG_MAXIMUM_LEVEL=0
|
||||
CONFIG_LOG_BOOTLOADER_LEVEL_NONE=y
|
||||
CONFIG_LOG_BOOTLOADER_LEVEL=0
|
||||
|
||||
CONFIG_HEAP_CORRUPTION_DETECTION=CONFIG_HEAP_POISONING_LIGHT
|
|
@ -26,7 +26,7 @@ void setFgColor(int color)
|
|||
|
||||
void showSetupQr(const String &ssid, const String &password)
|
||||
{
|
||||
char displayIndex = 6;
|
||||
uint displayIndex = 6;
|
||||
|
||||
const String text = "WIFI:S:" + ssid + ";T:WPA;P:" + password + ";;";
|
||||
|
||||
|
|
|
@ -36,8 +36,8 @@ void setupSoftAP()
|
|||
{
|
||||
byte mac[6];
|
||||
WiFi.macAddress(mac);
|
||||
softAP_SSID = String("BTClock" + String(mac[5], 16) + String(mac[1], 16));
|
||||
WiFi.setHostname(softAP_SSID.c_str());
|
||||
softAP_SSID = getMyHostname().c_str();
|
||||
WiFi.setHostname(getMyHostname().c_str());
|
||||
softAP_password = base64::encode(String(mac[2], 16) + String(mac[4], 16) + String(mac[5], 16) + String(mac[1], 16)).substring(2, 10);
|
||||
}
|
||||
|
||||
|
@ -60,25 +60,8 @@ void setupComponents()
|
|||
pixels.show();
|
||||
#endif
|
||||
|
||||
// delay(6000);
|
||||
// Serial.println(F("I2C Master"));
|
||||
|
||||
// int i2c_master_port = 0;
|
||||
// i2c_config_t conf = {
|
||||
// .mode = I2C_MODE_MASTER,
|
||||
// .sda_io_num = 35, // select SDA GPIO specific to your project
|
||||
// .scl_io_num = 36, // select SCL GPIO specific to your project
|
||||
// .sda_pullup_en = GPIO_PULLUP_ENABLE,
|
||||
// .scl_pullup_en = GPIO_PULLUP_ENABLE,
|
||||
// .master = {
|
||||
// .clk_speed = 400000,
|
||||
// }, // select frequency specific to your project
|
||||
// .clk_flags = 0, // optional; you can use I2C_SCLK_SRC_FLAG_* flags to choose i2c source clock here
|
||||
// };
|
||||
|
||||
// i2c_param_config(i2c_master_port, &conf);
|
||||
// ESP_ERROR_CHECK(i2c_driver_install(i2c_master_port, conf.mode, 0, 0, 0));
|
||||
Wire.begin(35, 36);
|
||||
// delay(3000);
|
||||
// Serial.println(F("Leds should be on"));
|
||||
|
||||
#ifndef NO_MCP
|
||||
if (!mcp.begin_I2C(0x20))
|
||||
|
@ -89,8 +72,8 @@ Wire.begin(35, 36);
|
|||
pixels.setPixelColor(2, pixels.Color(255, 0, 0));
|
||||
pixels.setPixelColor(3, pixels.Color(255, 0, 0));
|
||||
pixels.show();
|
||||
// while (1)
|
||||
// ;
|
||||
while (1)
|
||||
;
|
||||
}
|
||||
else
|
||||
{
|
||||
|
@ -197,7 +180,9 @@ void setupPreferences()
|
|||
{SCREEN_MSCW_TIME, "Sats per dollar"},
|
||||
{SCREEN_BTC_TICKER, "Ticker"},
|
||||
{SCREEN_TIME, "Time"},
|
||||
{SCREEN_HALVING_COUNTDOWN, "Halving countdown"}};
|
||||
{SCREEN_HALVING_COUNTDOWN, "Halving countdown"},
|
||||
{SCREEN_MARKET_CAP, "Market Cap"}
|
||||
};
|
||||
|
||||
#ifdef WITH_RGB_LED
|
||||
pixels.setBrightness(preferences.getUInt("ledBrightness", 128));
|
||||
|
@ -239,28 +224,24 @@ void handleScreenTasks(uint screen)
|
|||
vTaskSuspend(blockNotifyTaskHandle);
|
||||
if (getPriceTaskHandle)
|
||||
vTaskSuspend(getPriceTaskHandle);
|
||||
// if (minuteTaskHandle)
|
||||
// vTaskSuspend(minuteTaskHandle);
|
||||
switch (currentScreen)
|
||||
{
|
||||
case SCREEN_BLOCK_HEIGHT:
|
||||
if (blockNotifyTaskHandle)
|
||||
{
|
||||
vTaskResume(blockNotifyTaskHandle);
|
||||
}
|
||||
break;
|
||||
case SCREEN_HALVING_COUNTDOWN:
|
||||
if (blockNotifyTaskHandle)
|
||||
vTaskResume(blockNotifyTaskHandle);
|
||||
break;
|
||||
case SCREEN_BTC_TICKER:
|
||||
if (getPriceTaskHandle)
|
||||
vTaskResume(getPriceTaskHandle);
|
||||
break;
|
||||
case SCREEN_MSCW_TIME:
|
||||
if (getPriceTaskHandle)
|
||||
vTaskResume(getPriceTaskHandle);
|
||||
break;
|
||||
case SCREEN_MARKET_CAP:
|
||||
if (getPriceTaskHandle)
|
||||
vTaskResume(getPriceTaskHandle);
|
||||
if (blockNotifyTaskHandle)
|
||||
vTaskResume(blockNotifyTaskHandle);
|
||||
break;
|
||||
case SCREEN_TIME:
|
||||
if (minuteTaskHandle)
|
||||
{
|
||||
|
@ -407,9 +388,7 @@ void showNetworkSettings()
|
|||
}
|
||||
epdContent[NUM_SCREENS-2] = "RAM/Status";
|
||||
|
||||
// char buf[32];
|
||||
// snprintf(buf, sizeof(buf), "%s/%s", round(ESP.getFreeHeap()/1000), );
|
||||
epdContent[NUM_SCREENS-1] = String((int)round(ESP.getFreeHeap()/1000)) + "/" + (int)round(ESP.getHeapSize()/1000);
|
||||
epdContent[NUM_SCREENS-1] = String((int)round(ESP.getFreeHeap()/1024)) + "/" + (int)round(ESP.getHeapSize()/1024);
|
||||
|
||||
CustomTextScreen::setText(epdContent);
|
||||
|
||||
|
|
|
@ -5,6 +5,7 @@
|
|||
//#include <WiFiMulti.h>
|
||||
|
||||
#include "config.h"
|
||||
#include "utils.hpp"
|
||||
#include "shared.hpp"
|
||||
#include "Adafruit_GFX.h"
|
||||
#include "lib/epd.hpp"
|
||||
|
|
54
src/lib/utils.cpp
Normal file
54
src/lib/utils.cpp
Normal file
|
@ -0,0 +1,54 @@
|
|||
#include "utils.hpp";
|
||||
|
||||
double getSupplyAtBlock(uint blockNr) {
|
||||
if (blockNr >= 33 * 210000) {
|
||||
return 20999999.9769;
|
||||
}
|
||||
|
||||
const int initialBlockReward = 50; // Initial block reward
|
||||
const int halvingInterval = 210000; // Number of blocks before halving
|
||||
|
||||
int halvingCount = blockNr / halvingInterval;
|
||||
double totalBitcoinInCirculation = 0;
|
||||
|
||||
for (int i = 0; i < halvingCount; ++i) {
|
||||
totalBitcoinInCirculation += halvingInterval * initialBlockReward * std::pow(0.5, i);
|
||||
}
|
||||
|
||||
totalBitcoinInCirculation += (blockNr % halvingInterval) * initialBlockReward * std::pow(0.5, halvingCount);
|
||||
|
||||
return totalBitcoinInCirculation;
|
||||
}
|
||||
|
||||
std::string formatNumberWithSuffix(int64_t num) {
|
||||
const long long quadrillion = 1000000000000000LL;
|
||||
const long long trillion = 1000000000000LL;
|
||||
const long long billion = 1000000000;
|
||||
const long long million = 1000000;
|
||||
const long long thousand = 1000;
|
||||
|
||||
if (num >= quadrillion) {
|
||||
return std::to_string(num / quadrillion) + "Q";
|
||||
} else if (num >= trillion) {
|
||||
return std::to_string(num / trillion) + "T";
|
||||
} else if (num >= billion) {
|
||||
return std::to_string(num / billion) + "B";
|
||||
} else if (num >= million) {
|
||||
return std::to_string(num / million) + "M";
|
||||
} else if (num >= thousand) {
|
||||
return std::to_string(num / thousand) + "K";
|
||||
} else {
|
||||
return std::to_string(num);
|
||||
}
|
||||
}
|
||||
|
||||
String getMyHostname() {
|
||||
uint8_t mac[6];
|
||||
//WiFi.macAddress(mac);
|
||||
esp_efuse_mac_get_default(mac);
|
||||
char hostname[15];
|
||||
String hostnamePrefix = preferences.getString("hostnamePrefix", "btclock");
|
||||
snprintf(hostname, sizeof(hostname), "%s-%02x%02x%02x",
|
||||
hostnamePrefix, mac[3], mac[4], mac[5]);
|
||||
return hostname;
|
||||
}
|
7
src/lib/utils.hpp
Normal file
7
src/lib/utils.hpp
Normal file
|
@ -0,0 +1,7 @@
|
|||
#pragma once
|
||||
|
||||
#include "shared.hpp"
|
||||
|
||||
double getSupplyAtBlock(uint blockNr);
|
||||
std::string formatNumberWithSuffix(int64_t num);
|
||||
String getMyHostname();
|
|
@ -52,7 +52,7 @@ void setupWebserver()
|
|||
|
||||
// Start server
|
||||
server.begin();
|
||||
if (!MDNS.begin(HOSTNAME))
|
||||
if (!MDNS.begin(getMyHostname()))
|
||||
{
|
||||
Serial.println(F("Error setting up MDNS responder!"));
|
||||
while (1)
|
||||
|
@ -61,6 +61,9 @@ void setupWebserver()
|
|||
}
|
||||
}
|
||||
MDNS.addService("http", "tcp", 80);
|
||||
MDNS.addServiceTxt("http", "tcp", "model", "BTClock");
|
||||
MDNS.addServiceTxt("http", "tcp", "version", "2.0");
|
||||
MDNS.addServiceTxt("http", "tcp", "rev", GIT_REV);
|
||||
Serial.println(F("Webserver should be running"));
|
||||
}
|
||||
|
||||
|
@ -170,7 +173,7 @@ void onApiActionUpdate(AsyncWebServerRequest *request)
|
|||
*/
|
||||
void onApiSettingsGet(AsyncWebServerRequest *request)
|
||||
{
|
||||
StaticJsonDocument<768> root;
|
||||
StaticJsonDocument<1536> root;
|
||||
root["numScreens"] = NUM_SCREENS;
|
||||
root["fgColor"] = getFgColor();
|
||||
root["bgColor"] = getBgColor();
|
||||
|
@ -184,6 +187,8 @@ void onApiSettingsGet(AsyncWebServerRequest *request)
|
|||
root["rpcUser"] = preferences.getString("rpcUser", BITCOIND_RPC_USER);
|
||||
root["rpcHost"] = preferences.getString("rpcHost", BITCOIND_HOST);
|
||||
root["mempoolInstance"] = preferences.getString("mempoolInstance", DEFAULT_MEMPOOL_INSTANCE);
|
||||
root["hostnamePrefix"] = preferences.getString("hostnamePrefix", "btclock");
|
||||
root["hostname"] = getMyHostname();
|
||||
|
||||
#ifdef IS_BW
|
||||
root["epdColors"] = 2;
|
||||
|
@ -290,6 +295,14 @@ void onApiSettingsPost(AsyncWebServerRequest *request)
|
|||
settingsChanged = true;
|
||||
}
|
||||
|
||||
if (request->hasParam("hostnamePrefix", true))
|
||||
{
|
||||
AsyncWebParameter *hostnamePrefix = request->getParam("hostnamePrefix", true);
|
||||
|
||||
preferences.putString("hostnamePrefix", hostnamePrefix->value().c_str());
|
||||
settingsChanged = true;
|
||||
}
|
||||
|
||||
if (request->hasParam("ledBrightness", true))
|
||||
{
|
||||
AsyncWebParameter *ledBrightness = request->getParam("ledBrightness", true);
|
||||
|
|
|
@ -18,6 +18,7 @@
|
|||
#include "screens/ticker.hpp"
|
||||
#include "screens/time.hpp"
|
||||
#include "screens/halvingcountdown.hpp"
|
||||
#include "screens/market_cap.hpp"
|
||||
|
||||
#include "tasks/ha.hpp"
|
||||
#include "tasks/epd.hpp"
|
||||
|
@ -62,6 +63,7 @@ void setup()
|
|||
BlockHeightScreen::init();
|
||||
HalvingCountdownScreen::init();
|
||||
TickerScreen::init();
|
||||
MarketCapScreen::init();
|
||||
|
||||
#ifdef WITH_BUTTONS
|
||||
setupButtonTask();
|
||||
|
@ -87,5 +89,4 @@ void setup()
|
|||
|
||||
void loop()
|
||||
{
|
||||
vTaskDelay(pdMS_TO_TICKS(5000));
|
||||
}
|
||||
|
|
|
@ -58,3 +58,7 @@ std::array<String, NUM_SCREENS> BlockHeightScreen::getEpdContent()
|
|||
|
||||
return ret;
|
||||
}
|
||||
|
||||
uint BlockHeightScreen::getBlockNr() {
|
||||
return blockNr;
|
||||
}
|
|
@ -17,4 +17,5 @@ public:
|
|||
static void showScreen();
|
||||
static void onNewBlock(uint blockNr);
|
||||
static std::array<String, NUM_SCREENS> getEpdContent();
|
||||
static uint getBlockNr();
|
||||
};
|
30
src/screens/market_cap.cpp
Normal file
30
src/screens/market_cap.cpp
Normal file
|
@ -0,0 +1,30 @@
|
|||
#include "market_cap.hpp"
|
||||
|
||||
uint MarketCapScreen::satsPerDollar = 0;
|
||||
std::array<String, NUM_SCREENS> MarketCapScreen::epdContent = {"", "", "", "", "", "", ""};
|
||||
|
||||
void MarketCapScreen::init()
|
||||
{
|
||||
// Dependent on price and blocks
|
||||
MarketCapScreen::showScreen();
|
||||
}
|
||||
|
||||
void MarketCapScreen::showScreen()
|
||||
{
|
||||
double supply = getSupplyAtBlock(BlockHeightScreen::getBlockNr());
|
||||
int64_t marketCap = static_cast<std::int64_t>(supply * double(TickerScreen::getPrice()));
|
||||
|
||||
std::string priceString = "$" + formatNumberWithSuffix(marketCap);
|
||||
priceString.insert(priceString.begin(), NUM_SCREENS - priceString.length(), ' ');
|
||||
|
||||
epdContent[0] = "USD/MCAP";
|
||||
for (uint i = 1; i < NUM_SCREENS; i++)
|
||||
{
|
||||
MarketCapScreen::epdContent[i] = priceString[i];
|
||||
}
|
||||
}
|
||||
|
||||
std::array<String, NUM_SCREENS> MarketCapScreen::getEpdContent()
|
||||
{
|
||||
return MarketCapScreen::epdContent;
|
||||
}
|
20
src/screens/market_cap.hpp
Normal file
20
src/screens/market_cap.hpp
Normal file
|
@ -0,0 +1,20 @@
|
|||
#pragma once
|
||||
|
||||
#include "base.hpp"
|
||||
#include "config.h"
|
||||
#include "shared.hpp"
|
||||
#include "lib/utils.hpp"
|
||||
#include "blockheight.hpp";
|
||||
#include "ticker.hpp";
|
||||
#include "tasks/epd.hpp"
|
||||
|
||||
class MarketCapScreen {
|
||||
protected:
|
||||
static uint satsPerDollar;
|
||||
static std::array<String, NUM_SCREENS> epdContent;
|
||||
public:
|
||||
static void init();
|
||||
static void showScreen();
|
||||
static void onPriceUpdate(uint price);
|
||||
static std::array<String, NUM_SCREENS> getEpdContent();
|
||||
};
|
|
@ -1,30 +0,0 @@
|
|||
// #include "sats_per_dollar.hpp"
|
||||
|
||||
// uint SatsPerDollarScreen::satsPerDollar = 0;
|
||||
// std::array<String, NUM_SCREENS> SatsPerDollarScreen::epdContent = { "", "", "", "", "", "", "" };
|
||||
|
||||
// void SatsPerDollarScreen::init() {
|
||||
// SatsPerDollarScreen::satsPerDollar = int(round(1 / preferences.getFloat("btcPrice", 12345) * 10e7));
|
||||
// setupGetPriceTask();
|
||||
// SatsPerDollarScreen::showScreen();
|
||||
// }
|
||||
|
||||
// void SatsPerDollarScreen::showScreen() {
|
||||
// std::string satsPerDollarString = String(SatsPerDollarScreen::satsPerDollar).c_str();
|
||||
// satsPerDollarString.insert(satsPerDollarString.begin(), 7 - satsPerDollarString.length(), ' ');
|
||||
// epdContent[0] = "MSCW/TIME";
|
||||
// for (uint i = 1; i < NUM_SCREENS; i++)
|
||||
// {
|
||||
// SatsPerDollarScreen::epdContent[i] = satsPerDollarString[i];
|
||||
// }
|
||||
// }
|
||||
|
||||
// void SatsPerDollarScreen::onPriceUpdate(uint price) {
|
||||
// SatsPerDollarScreen::satsPerDollar = int(round(1 / float(price) * 10e7));
|
||||
|
||||
// SatsPerDollarScreen::showScreen();
|
||||
// }
|
||||
|
||||
// std::array<String, NUM_SCREENS> SatsPerDollarScreen::getEpdContent() {
|
||||
// return SatsPerDollarScreen::epdContent;
|
||||
// }
|
|
@ -1,17 +0,0 @@
|
|||
// #pragma once
|
||||
|
||||
// #include "base.hpp"
|
||||
// #include "config.h"
|
||||
// #include "shared.hpp"
|
||||
// #include "tasks/epd.hpp"
|
||||
|
||||
// class SatsPerDollarScreen {
|
||||
// protected:
|
||||
// static uint satsPerDollar;
|
||||
// static std::array<String, NUM_SCREENS> epdContent;
|
||||
// public:
|
||||
// static void init();
|
||||
// static void showScreen();
|
||||
// static void onPriceUpdate(uint price);
|
||||
// static std::array<String, NUM_SCREENS> getEpdContent();
|
||||
// };
|
|
@ -49,3 +49,7 @@ std::array<String, NUM_SCREENS> TickerScreen::getEpdContentSats() {
|
|||
|
||||
return epdContentSats;
|
||||
}
|
||||
|
||||
uint TickerScreen::getPrice() {
|
||||
return price;
|
||||
}
|
|
@ -18,4 +18,5 @@ public:
|
|||
static void onPriceUpdate(uint price);
|
||||
static std::array<String, NUM_SCREENS> getEpdContent();
|
||||
static std::array<String, NUM_SCREENS> getEpdContentSats();
|
||||
static uint getPrice();
|
||||
};
|
|
@ -55,9 +55,11 @@ const PROGMEM int SCREEN_MSCW_TIME = 1;
|
|||
const PROGMEM int SCREEN_BTC_TICKER = 2;
|
||||
const PROGMEM int SCREEN_TIME = 3;
|
||||
const PROGMEM int SCREEN_HALVING_COUNTDOWN = 4;
|
||||
const PROGMEM int SCREEN_MARKET_CAP = 5;
|
||||
|
||||
const PROGMEM int SCREEN_COUNTDOWN = 98;
|
||||
const PROGMEM int SCREEN_CUSTOM = 99;
|
||||
const PROGMEM int screens[5] = { SCREEN_BLOCK_HEIGHT, SCREEN_MSCW_TIME, SCREEN_BTC_TICKER, SCREEN_TIME, SCREEN_HALVING_COUNTDOWN };
|
||||
const PROGMEM int screens[6] = { SCREEN_BLOCK_HEIGHT, SCREEN_MSCW_TIME, SCREEN_BTC_TICKER, SCREEN_TIME, SCREEN_HALVING_COUNTDOWN, SCREEN_MARKET_CAP };
|
||||
|
||||
const uint screenCount = sizeof(screens) / sizeof(int);
|
||||
|
||||
|
|
|
@ -192,6 +192,9 @@ void taskEpd(void *pvParameters)
|
|||
case SCREEN_HALVING_COUNTDOWN:
|
||||
epdContent = HalvingCountdownScreen::getEpdContent();
|
||||
break;
|
||||
case SCREEN_MARKET_CAP:
|
||||
epdContent = MarketCapScreen::getEpdContent();
|
||||
break;
|
||||
case SCREEN_COUNTDOWN:
|
||||
epdContent = CountdownScreen::getEpdContent();
|
||||
break;
|
||||
|
@ -325,7 +328,7 @@ extern "C" void updateDisplay(void *pvParameters) noexcept
|
|||
|
||||
if (epdContent[epdIndex].compareTo(currentEpdContent[epdIndex]) != 0)
|
||||
{
|
||||
displays[epdIndex].init(0, false, 20); // Little longer reset duration because of MCP
|
||||
displays[epdIndex].init(0, false); // Little longer reset duration because of MCP
|
||||
#ifndef USE_UNIVERSAL_PIN
|
||||
resetSingleDisplay(epdIndex);
|
||||
#endif
|
||||
|
|
|
@ -11,10 +11,12 @@
|
|||
#include "screens/blockheight.hpp"
|
||||
#include "screens/ticker.hpp"
|
||||
#include "screens/time.hpp"
|
||||
#include "screens/sats_per_dollar.hpp"
|
||||
#include "screens/market_cap.hpp"
|
||||
#include "screens/countdown.hpp"
|
||||
#include "screens/custom_text.hpp"
|
||||
#include "screens/halvingcountdown.hpp"
|
||||
|
||||
|
||||
#ifdef USE_UNIVERSAL_PIN
|
||||
#include <native_pin.hpp>
|
||||
#include <mcp23x17_pin.hpp>
|
||||
|
|
|
@ -4,32 +4,47 @@ TaskHandle_t minuteTaskHandle = NULL;
|
|||
// Define a type for the event callback
|
||||
std::vector<EventCallback> minuteEventCallbacks; // Define a vector to hold multiple event callbacks
|
||||
bool eventTriggered = false; // Initialize the event triggered flag to false
|
||||
const int usPerMinute = 60 * 1000000;
|
||||
|
||||
void minuteTask(void * parameter) {
|
||||
while(1) {
|
||||
#ifdef IS_3C // wait 5 minutes in case of a 3 color screen otherwise it keeps refreshing
|
||||
if(rtc.getMinute() % 5 == 0 && !eventTriggered) {
|
||||
eventTriggered = true;
|
||||
for(auto &callback : minuteEventCallbacks) { // Loop through all the event callbacks and call them
|
||||
void minuteTask(void *parameter)
|
||||
{
|
||||
|
||||
esp_timer_handle_t minuteTimer;
|
||||
const esp_timer_create_args_t minuteTimerConfig = {
|
||||
.callback = &minuteTimerISR,
|
||||
.name = "minute_timer"};
|
||||
|
||||
esp_timer_create(&minuteTimerConfig, &minuteTimer);
|
||||
|
||||
time_t currentTime;
|
||||
struct tm timeinfo;
|
||||
time(¤tTime);
|
||||
localtime_r(¤tTime, &timeinfo);
|
||||
uint32_t secondsUntilNextMinute = 60 - timeinfo.tm_sec;
|
||||
|
||||
if (secondsUntilNextMinute > 0)
|
||||
vTaskDelay(pdMS_TO_TICKS((secondsUntilNextMinute * 1000)));
|
||||
|
||||
esp_timer_start_periodic(minuteTimer, usPerMinute);
|
||||
|
||||
while (1)
|
||||
{
|
||||
for (auto &callback : minuteEventCallbacks)
|
||||
{
|
||||
callback();
|
||||
}
|
||||
}
|
||||
if(rtc.getMinute() % 5 != 0 && eventTriggered) { // Reset the event triggered flag if the second is not 0
|
||||
eventTriggered = false;
|
||||
}
|
||||
vTaskDelay(pdMS_TO_TICKS(1000)); // Sleep for 1000 milliseconds to avoid busy waiting
|
||||
#else
|
||||
if(rtc.getSecond() == 0 && !eventTriggered) {
|
||||
eventTriggered = true;
|
||||
for(auto &callback : minuteEventCallbacks) { // Loop through all the event callbacks and call them
|
||||
callback();
|
||||
|
||||
ulTaskNotifyTake(pdTRUE, portMAX_DELAY);
|
||||
}
|
||||
}
|
||||
if(rtc.getSecond() != 0) { // Reset the event triggered flag if the second is not 0
|
||||
eventTriggered = false;
|
||||
}
|
||||
vTaskDelay(pdMS_TO_TICKS(1000)); // Sleep for 1000 milliseconds to avoid busy waiting
|
||||
#endif
|
||||
|
||||
void IRAM_ATTR minuteTimerISR(void *arg)
|
||||
{
|
||||
BaseType_t xHigherPriorityTaskWoken = pdFALSE;
|
||||
vTaskNotifyGiveFromISR(minuteTaskHandle, &xHigherPriorityTaskWoken);
|
||||
if (xHigherPriorityTaskWoken == pdTRUE)
|
||||
{
|
||||
portYIELD_FROM_ISR();
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -6,9 +6,13 @@
|
|||
#include <freertos/task.h>
|
||||
#include <vector>
|
||||
#include "shared.hpp"
|
||||
#include <esp_timer.h>
|
||||
#include <time.h>
|
||||
|
||||
extern TaskHandle_t minuteTaskHandle;
|
||||
|
||||
void minuteTask(void *pvParameters);
|
||||
void setupMinuteEvent();
|
||||
void IRAM_ATTR minuteTimerISR(void* arg);
|
||||
|
||||
void registerNewMinuteCallback(const EventCallback cb);
|
Loading…
Reference in a new issue