mirror of
https://github.com/daveallie/crosspoint-reader.git
synced 2026-02-06 15:47:39 +03:00
Merge branch 'master' into feature/exfat-support
This commit is contained in:
commit
fd6373c128
@ -87,6 +87,21 @@ std::string Xtc::getTitle() const {
|
|||||||
return filepath.substr(lastSlash, lastDot - lastSlash);
|
return filepath.substr(lastSlash, lastDot - lastSlash);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool Xtc::hasChapters() const {
|
||||||
|
if (!loaded || !parser) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return parser->hasChapters();
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<xtc::ChapterInfo>& Xtc::getChapters() const {
|
||||||
|
static const std::vector<xtc::ChapterInfo> kEmpty;
|
||||||
|
if (!loaded || !parser) {
|
||||||
|
return kEmpty;
|
||||||
|
}
|
||||||
|
return parser->getChapters();
|
||||||
|
}
|
||||||
|
|
||||||
std::string Xtc::getCoverBmpPath() const { return cachePath + "/cover.bmp"; }
|
std::string Xtc::getCoverBmpPath() const { return cachePath + "/cover.bmp"; }
|
||||||
|
|
||||||
bool Xtc::generateCoverBmp() const {
|
bool Xtc::generateCoverBmp() const {
|
||||||
|
|||||||
@ -9,6 +9,7 @@
|
|||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
#include "Xtc/XtcParser.h"
|
#include "Xtc/XtcParser.h"
|
||||||
#include "Xtc/XtcTypes.h"
|
#include "Xtc/XtcTypes.h"
|
||||||
@ -55,6 +56,8 @@ class Xtc {
|
|||||||
|
|
||||||
// Metadata
|
// Metadata
|
||||||
std::string getTitle() const;
|
std::string getTitle() const;
|
||||||
|
bool hasChapters() const;
|
||||||
|
const std::vector<xtc::ChapterInfo>& getChapters() const;
|
||||||
|
|
||||||
// Cover image support (for sleep screen)
|
// Cover image support (for sleep screen)
|
||||||
std::string getCoverBmpPath() const;
|
std::string getCoverBmpPath() const;
|
||||||
|
|||||||
@ -20,6 +20,7 @@ XtcParser::XtcParser()
|
|||||||
m_defaultWidth(DISPLAY_WIDTH),
|
m_defaultWidth(DISPLAY_WIDTH),
|
||||||
m_defaultHeight(DISPLAY_HEIGHT),
|
m_defaultHeight(DISPLAY_HEIGHT),
|
||||||
m_bitDepth(1),
|
m_bitDepth(1),
|
||||||
|
m_hasChapters(false),
|
||||||
m_lastError(XtcError::OK) {
|
m_lastError(XtcError::OK) {
|
||||||
memset(&m_header, 0, sizeof(m_header));
|
memset(&m_header, 0, sizeof(m_header));
|
||||||
}
|
}
|
||||||
@ -57,6 +58,14 @@ XtcError XtcParser::open(const char* filepath) {
|
|||||||
return m_lastError;
|
return m_lastError;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Read chapters if present
|
||||||
|
m_lastError = readChapters();
|
||||||
|
if (m_lastError != XtcError::OK) {
|
||||||
|
Serial.printf("[%lu] [XTC] Failed to read chapters: %s\n", millis(), errorToString(m_lastError));
|
||||||
|
m_file.close();
|
||||||
|
return m_lastError;
|
||||||
|
}
|
||||||
|
|
||||||
m_isOpen = true;
|
m_isOpen = true;
|
||||||
Serial.printf("[%lu] [XTC] Opened file: %s (%u pages, %dx%d)\n", millis(), filepath, m_header.pageCount,
|
Serial.printf("[%lu] [XTC] Opened file: %s (%u pages, %dx%d)\n", millis(), filepath, m_header.pageCount,
|
||||||
m_defaultWidth, m_defaultHeight);
|
m_defaultWidth, m_defaultHeight);
|
||||||
@ -69,7 +78,9 @@ void XtcParser::close() {
|
|||||||
m_isOpen = false;
|
m_isOpen = false;
|
||||||
}
|
}
|
||||||
m_pageTable.clear();
|
m_pageTable.clear();
|
||||||
|
m_chapters.clear();
|
||||||
m_title.clear();
|
m_title.clear();
|
||||||
|
m_hasChapters = false;
|
||||||
memset(&m_header, 0, sizeof(m_header));
|
memset(&m_header, 0, sizeof(m_header));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -91,7 +102,9 @@ XtcError XtcParser::readHeader() {
|
|||||||
m_bitDepth = (m_header.magic == XTCH_MAGIC) ? 2 : 1;
|
m_bitDepth = (m_header.magic == XTCH_MAGIC) ? 2 : 1;
|
||||||
|
|
||||||
// Check version
|
// Check version
|
||||||
if (m_header.version > 1) {
|
// Currently, version 1 is the only valid version, however some generators are using big endian for the version code
|
||||||
|
// so we also accept version 256 (0x0100)
|
||||||
|
if (m_header.version != 1 && m_header.version != 256) {
|
||||||
Serial.printf("[%lu] [XTC] Unsupported version: %d\n", millis(), m_header.version);
|
Serial.printf("[%lu] [XTC] Unsupported version: %d\n", millis(), m_header.version);
|
||||||
return XtcError::INVALID_VERSION;
|
return XtcError::INVALID_VERSION;
|
||||||
}
|
}
|
||||||
@ -166,6 +179,112 @@ XtcError XtcParser::readPageTable() {
|
|||||||
return XtcError::OK;
|
return XtcError::OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
XtcError XtcParser::readChapters() {
|
||||||
|
m_hasChapters = false;
|
||||||
|
m_chapters.clear();
|
||||||
|
|
||||||
|
uint8_t hasChaptersFlag = 0;
|
||||||
|
if (!m_file.seek(0x0B)) {
|
||||||
|
return XtcError::READ_ERROR;
|
||||||
|
}
|
||||||
|
if (m_file.read(&hasChaptersFlag, sizeof(hasChaptersFlag)) != sizeof(hasChaptersFlag)) {
|
||||||
|
return XtcError::READ_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hasChaptersFlag != 1) {
|
||||||
|
return XtcError::OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t chapterOffset = 0;
|
||||||
|
if (!m_file.seek(0x30)) {
|
||||||
|
return XtcError::READ_ERROR;
|
||||||
|
}
|
||||||
|
if (m_file.read(reinterpret_cast<uint8_t*>(&chapterOffset), sizeof(chapterOffset)) != sizeof(chapterOffset)) {
|
||||||
|
return XtcError::READ_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (chapterOffset == 0) {
|
||||||
|
return XtcError::OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
const uint64_t fileSize = m_file.size();
|
||||||
|
if (chapterOffset < sizeof(XtcHeader) || chapterOffset >= fileSize || chapterOffset + 96 > fileSize) {
|
||||||
|
return XtcError::OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t maxOffset = 0;
|
||||||
|
if (m_header.pageTableOffset > chapterOffset) {
|
||||||
|
maxOffset = m_header.pageTableOffset;
|
||||||
|
} else if (m_header.dataOffset > chapterOffset) {
|
||||||
|
maxOffset = m_header.dataOffset;
|
||||||
|
} else {
|
||||||
|
maxOffset = fileSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (maxOffset <= chapterOffset) {
|
||||||
|
return XtcError::OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr size_t chapterSize = 96;
|
||||||
|
const uint64_t available = maxOffset - chapterOffset;
|
||||||
|
const size_t chapterCount = static_cast<size_t>(available / chapterSize);
|
||||||
|
if (chapterCount == 0) {
|
||||||
|
return XtcError::OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!m_file.seek(chapterOffset)) {
|
||||||
|
return XtcError::READ_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<uint8_t> chapterBuf(chapterSize);
|
||||||
|
for (size_t i = 0; i < chapterCount; i++) {
|
||||||
|
if (m_file.read(chapterBuf.data(), chapterSize) != chapterSize) {
|
||||||
|
return XtcError::READ_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
char nameBuf[81];
|
||||||
|
memcpy(nameBuf, chapterBuf.data(), 80);
|
||||||
|
nameBuf[80] = '\0';
|
||||||
|
const size_t nameLen = strnlen(nameBuf, 80);
|
||||||
|
std::string name(nameBuf, nameLen);
|
||||||
|
|
||||||
|
uint16_t startPage = 0;
|
||||||
|
uint16_t endPage = 0;
|
||||||
|
memcpy(&startPage, chapterBuf.data() + 0x50, sizeof(startPage));
|
||||||
|
memcpy(&endPage, chapterBuf.data() + 0x52, sizeof(endPage));
|
||||||
|
|
||||||
|
if (name.empty() && startPage == 0 && endPage == 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (startPage > 0) {
|
||||||
|
startPage--;
|
||||||
|
}
|
||||||
|
if (endPage > 0) {
|
||||||
|
endPage--;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (startPage >= m_header.pageCount) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (endPage >= m_header.pageCount) {
|
||||||
|
endPage = m_header.pageCount - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (startPage > endPage) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
ChapterInfo chapter{std::move(name), startPage, endPage};
|
||||||
|
m_chapters.push_back(std::move(chapter));
|
||||||
|
}
|
||||||
|
|
||||||
|
m_hasChapters = !m_chapters.empty();
|
||||||
|
Serial.printf("[%lu] [XTC] Chapters: %u\n", millis(), static_cast<unsigned int>(m_chapters.size()));
|
||||||
|
return XtcError::OK;
|
||||||
|
}
|
||||||
|
|
||||||
bool XtcParser::getPageInfo(uint32_t pageIndex, PageInfo& info) const {
|
bool XtcParser::getPageInfo(uint32_t pageIndex, PageInfo& info) const {
|
||||||
if (pageIndex >= m_pageTable.size()) {
|
if (pageIndex >= m_pageTable.size()) {
|
||||||
return false;
|
return false;
|
||||||
|
|||||||
@ -70,6 +70,9 @@ class XtcParser {
|
|||||||
// Get title from metadata
|
// Get title from metadata
|
||||||
std::string getTitle() const { return m_title; }
|
std::string getTitle() const { return m_title; }
|
||||||
|
|
||||||
|
bool hasChapters() const { return m_hasChapters; }
|
||||||
|
const std::vector<ChapterInfo>& getChapters() const { return m_chapters; }
|
||||||
|
|
||||||
// Validation
|
// Validation
|
||||||
static bool isValidXtcFile(const char* filepath);
|
static bool isValidXtcFile(const char* filepath);
|
||||||
|
|
||||||
@ -81,16 +84,19 @@ class XtcParser {
|
|||||||
bool m_isOpen;
|
bool m_isOpen;
|
||||||
XtcHeader m_header;
|
XtcHeader m_header;
|
||||||
std::vector<PageInfo> m_pageTable;
|
std::vector<PageInfo> m_pageTable;
|
||||||
|
std::vector<ChapterInfo> m_chapters;
|
||||||
std::string m_title;
|
std::string m_title;
|
||||||
uint16_t m_defaultWidth;
|
uint16_t m_defaultWidth;
|
||||||
uint16_t m_defaultHeight;
|
uint16_t m_defaultHeight;
|
||||||
uint8_t m_bitDepth; // 1 = XTC/XTG (1-bit), 2 = XTCH/XTH (2-bit)
|
uint8_t m_bitDepth; // 1 = XTC/XTG (1-bit), 2 = XTCH/XTH (2-bit)
|
||||||
|
bool m_hasChapters;
|
||||||
XtcError m_lastError;
|
XtcError m_lastError;
|
||||||
|
|
||||||
// Internal helper functions
|
// Internal helper functions
|
||||||
XtcError readHeader();
|
XtcError readHeader();
|
||||||
XtcError readPageTable();
|
XtcError readPageTable();
|
||||||
XtcError readTitle();
|
XtcError readTitle();
|
||||||
|
XtcError readChapters();
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace xtc
|
} // namespace xtc
|
||||||
|
|||||||
@ -13,6 +13,7 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
namespace xtc {
|
namespace xtc {
|
||||||
|
|
||||||
@ -92,6 +93,12 @@ struct PageInfo {
|
|||||||
uint8_t padding; // Alignment padding
|
uint8_t padding; // Alignment padding
|
||||||
}; // 16 bytes total
|
}; // 16 bytes total
|
||||||
|
|
||||||
|
struct ChapterInfo {
|
||||||
|
std::string name;
|
||||||
|
uint16_t startPage;
|
||||||
|
uint16_t endPage;
|
||||||
|
};
|
||||||
|
|
||||||
// Error codes
|
// Error codes
|
||||||
enum class XtcError {
|
enum class XtcError {
|
||||||
OK = 0,
|
OK = 0,
|
||||||
|
|||||||
@ -59,7 +59,7 @@ class CrossPointSettings {
|
|||||||
// Get singleton instance
|
// Get singleton instance
|
||||||
static CrossPointSettings& getInstance() { return instance; }
|
static CrossPointSettings& getInstance() { return instance; }
|
||||||
|
|
||||||
uint16_t getPowerButtonDuration() const { return shortPwrBtn ? 10 : 500; }
|
uint16_t getPowerButtonDuration() const { return shortPwrBtn ? 10 : 400; }
|
||||||
|
|
||||||
bool saveToFile() const;
|
bool saveToFile() const;
|
||||||
bool loadFromFile();
|
bool loadFromFile();
|
||||||
|
|||||||
@ -1,5 +1,7 @@
|
|||||||
#include "MappedInputManager.h"
|
#include "MappedInputManager.h"
|
||||||
|
|
||||||
|
#include "CrossPointSettings.h"
|
||||||
|
|
||||||
decltype(InputManager::BTN_BACK) MappedInputManager::mapButton(const Button button) const {
|
decltype(InputManager::BTN_BACK) MappedInputManager::mapButton(const Button button) const {
|
||||||
const auto frontLayout = static_cast<CrossPointSettings::FRONT_BUTTON_LAYOUT>(SETTINGS.frontButtonLayout);
|
const auto frontLayout = static_cast<CrossPointSettings::FRONT_BUTTON_LAYOUT>(SETTINGS.frontButtonLayout);
|
||||||
const auto sideLayout = static_cast<CrossPointSettings::SIDE_BUTTON_LAYOUT>(SETTINGS.sideButtonLayout);
|
const auto sideLayout = static_cast<CrossPointSettings::SIDE_BUTTON_LAYOUT>(SETTINGS.sideButtonLayout);
|
||||||
|
|||||||
@ -2,8 +2,6 @@
|
|||||||
|
|
||||||
#include <InputManager.h>
|
#include <InputManager.h>
|
||||||
|
|
||||||
#include "CrossPointSettings.h"
|
|
||||||
|
|
||||||
class MappedInputManager {
|
class MappedInputManager {
|
||||||
public:
|
public:
|
||||||
enum class Button { Back, Confirm, Left, Right, Up, Down, Power, PageBack, PageForward };
|
enum class Button { Back, Confirm, Left, Right, Up, Down, Power, PageBack, PageForward };
|
||||||
|
|||||||
@ -122,12 +122,16 @@ void HomeActivity::render() const {
|
|||||||
if (bookName.length() > 5 && bookName.substr(bookName.length() - 5) == ".epub") {
|
if (bookName.length() > 5 && bookName.substr(bookName.length() - 5) == ".epub") {
|
||||||
bookName.resize(bookName.length() - 5);
|
bookName.resize(bookName.length() - 5);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Truncate if too long
|
// Truncate if too long
|
||||||
if (bookName.length() > 25) {
|
|
||||||
bookName.resize(22);
|
|
||||||
bookName += "...";
|
|
||||||
}
|
|
||||||
std::string continueLabel = "Continue: " + bookName;
|
std::string continueLabel = "Continue: " + bookName;
|
||||||
|
int itemWidth = renderer.getTextWidth(UI_FONT_ID, continueLabel.c_str());
|
||||||
|
while (itemWidth > renderer.getScreenWidth() - 40 && continueLabel.length() > 8) {
|
||||||
|
continueLabel.replace(continueLabel.length() - 5, 5, "...");
|
||||||
|
itemWidth = renderer.getTextWidth(UI_FONT_ID, continueLabel.c_str());
|
||||||
|
Serial.printf("[%lu] [HOM] width: %lu, pageWidth: %lu\n", millis(), itemWidth, pageWidth);
|
||||||
|
}
|
||||||
|
|
||||||
renderer.drawText(UI_FONT_ID, 20, menuY, continueLabel.c_str(), selectorIndex != menuIndex);
|
renderer.drawText(UI_FONT_ID, 20, menuY, continueLabel.c_str(), selectorIndex != menuIndex);
|
||||||
menuY += 30;
|
menuY += 30;
|
||||||
menuIndex++;
|
menuIndex++;
|
||||||
|
|||||||
@ -13,6 +13,7 @@
|
|||||||
|
|
||||||
#include "CrossPointState.h"
|
#include "CrossPointState.h"
|
||||||
#include "MappedInputManager.h"
|
#include "MappedInputManager.h"
|
||||||
|
#include "XtcReaderChapterSelectionActivity.h"
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
@ -27,7 +28,7 @@ void XtcReaderActivity::taskTrampoline(void* param) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void XtcReaderActivity::onEnter() {
|
void XtcReaderActivity::onEnter() {
|
||||||
Activity::onEnter();
|
ActivityWithSubactivity::onEnter();
|
||||||
|
|
||||||
if (!xtc) {
|
if (!xtc) {
|
||||||
return;
|
return;
|
||||||
@ -56,7 +57,7 @@ void XtcReaderActivity::onEnter() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void XtcReaderActivity::onExit() {
|
void XtcReaderActivity::onExit() {
|
||||||
Activity::onExit();
|
ActivityWithSubactivity::onExit();
|
||||||
|
|
||||||
// Wait until not rendering to delete task
|
// Wait until not rendering to delete task
|
||||||
xSemaphoreTake(renderingMutex, portMAX_DELAY);
|
xSemaphoreTake(renderingMutex, portMAX_DELAY);
|
||||||
@ -70,6 +71,32 @@ void XtcReaderActivity::onExit() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void XtcReaderActivity::loop() {
|
void XtcReaderActivity::loop() {
|
||||||
|
// Pass input responsibility to sub activity if exists
|
||||||
|
if (subActivity) {
|
||||||
|
subActivity->loop();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enter chapter selection activity
|
||||||
|
if (mappedInput.wasReleased(MappedInputManager::Button::Confirm)) {
|
||||||
|
if (xtc && xtc->hasChapters() && !xtc->getChapters().empty()) {
|
||||||
|
xSemaphoreTake(renderingMutex, portMAX_DELAY);
|
||||||
|
exitActivity();
|
||||||
|
enterNewActivity(new XtcReaderChapterSelectionActivity(
|
||||||
|
this->renderer, this->mappedInput, xtc, currentPage,
|
||||||
|
[this] {
|
||||||
|
exitActivity();
|
||||||
|
updateRequired = true;
|
||||||
|
},
|
||||||
|
[this](const uint32_t newPage) {
|
||||||
|
currentPage = newPage;
|
||||||
|
exitActivity();
|
||||||
|
updateRequired = true;
|
||||||
|
}));
|
||||||
|
xSemaphoreGive(renderingMutex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Long press BACK (1s+) goes directly to home
|
// Long press BACK (1s+) goes directly to home
|
||||||
if (mappedInput.isPressed(MappedInputManager::Button::Back) && mappedInput.getHeldTime() >= goHomeMs) {
|
if (mappedInput.isPressed(MappedInputManager::Button::Back) && mappedInput.getHeldTime() >= goHomeMs) {
|
||||||
onGoHome();
|
onGoHome();
|
||||||
|
|||||||
@ -12,9 +12,9 @@
|
|||||||
#include <freertos/semphr.h>
|
#include <freertos/semphr.h>
|
||||||
#include <freertos/task.h>
|
#include <freertos/task.h>
|
||||||
|
|
||||||
#include "activities/Activity.h"
|
#include "activities/ActivityWithSubactivity.h"
|
||||||
|
|
||||||
class XtcReaderActivity final : public Activity {
|
class XtcReaderActivity final : public ActivityWithSubactivity {
|
||||||
std::shared_ptr<Xtc> xtc;
|
std::shared_ptr<Xtc> xtc;
|
||||||
TaskHandle_t displayTaskHandle = nullptr;
|
TaskHandle_t displayTaskHandle = nullptr;
|
||||||
SemaphoreHandle_t renderingMutex = nullptr;
|
SemaphoreHandle_t renderingMutex = nullptr;
|
||||||
@ -34,7 +34,10 @@ class XtcReaderActivity final : public Activity {
|
|||||||
public:
|
public:
|
||||||
explicit XtcReaderActivity(GfxRenderer& renderer, MappedInputManager& mappedInput, std::unique_ptr<Xtc> xtc,
|
explicit XtcReaderActivity(GfxRenderer& renderer, MappedInputManager& mappedInput, std::unique_ptr<Xtc> xtc,
|
||||||
const std::function<void()>& onGoBack, const std::function<void()>& onGoHome)
|
const std::function<void()>& onGoBack, const std::function<void()>& onGoHome)
|
||||||
: Activity("XtcReader", renderer, mappedInput), xtc(std::move(xtc)), onGoBack(onGoBack), onGoHome(onGoHome) {}
|
: ActivityWithSubactivity("XtcReader", renderer, mappedInput),
|
||||||
|
xtc(std::move(xtc)),
|
||||||
|
onGoBack(onGoBack),
|
||||||
|
onGoHome(onGoHome) {}
|
||||||
void onEnter() override;
|
void onEnter() override;
|
||||||
void onExit() override;
|
void onExit() override;
|
||||||
void loop() override;
|
void loop() override;
|
||||||
|
|||||||
151
src/activities/reader/XtcReaderChapterSelectionActivity.cpp
Normal file
151
src/activities/reader/XtcReaderChapterSelectionActivity.cpp
Normal file
@ -0,0 +1,151 @@
|
|||||||
|
#include "XtcReaderChapterSelectionActivity.h"
|
||||||
|
|
||||||
|
#include <GfxRenderer.h>
|
||||||
|
|
||||||
|
#include "MappedInputManager.h"
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
constexpr int SKIP_PAGE_MS = 700;
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
int XtcReaderChapterSelectionActivity::getPageItems() const {
|
||||||
|
constexpr int startY = 60;
|
||||||
|
constexpr int lineHeight = 30;
|
||||||
|
|
||||||
|
const int screenHeight = renderer.getScreenHeight();
|
||||||
|
const int availableHeight = screenHeight - startY;
|
||||||
|
int items = availableHeight / lineHeight;
|
||||||
|
if (items < 1) {
|
||||||
|
items = 1;
|
||||||
|
}
|
||||||
|
return items;
|
||||||
|
}
|
||||||
|
|
||||||
|
int XtcReaderChapterSelectionActivity::findChapterIndexForPage(uint32_t page) const {
|
||||||
|
if (!xtc) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto& chapters = xtc->getChapters();
|
||||||
|
for (size_t i = 0; i < chapters.size(); i++) {
|
||||||
|
if (page >= chapters[i].startPage && page <= chapters[i].endPage) {
|
||||||
|
return static_cast<int>(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void XtcReaderChapterSelectionActivity::taskTrampoline(void* param) {
|
||||||
|
auto* self = static_cast<XtcReaderChapterSelectionActivity*>(param);
|
||||||
|
self->displayTaskLoop();
|
||||||
|
}
|
||||||
|
|
||||||
|
void XtcReaderChapterSelectionActivity::onEnter() {
|
||||||
|
Activity::onEnter();
|
||||||
|
|
||||||
|
if (!xtc) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
renderingMutex = xSemaphoreCreateMutex();
|
||||||
|
selectorIndex = findChapterIndexForPage(currentPage);
|
||||||
|
|
||||||
|
updateRequired = true;
|
||||||
|
xTaskCreate(&XtcReaderChapterSelectionActivity::taskTrampoline, "XtcReaderChapterSelectionActivityTask",
|
||||||
|
4096, // Stack size
|
||||||
|
this, // Parameters
|
||||||
|
1, // Priority
|
||||||
|
&displayTaskHandle // Task handle
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
void XtcReaderChapterSelectionActivity::onExit() {
|
||||||
|
Activity::onExit();
|
||||||
|
|
||||||
|
xSemaphoreTake(renderingMutex, portMAX_DELAY);
|
||||||
|
if (displayTaskHandle) {
|
||||||
|
vTaskDelete(displayTaskHandle);
|
||||||
|
displayTaskHandle = nullptr;
|
||||||
|
}
|
||||||
|
vSemaphoreDelete(renderingMutex);
|
||||||
|
renderingMutex = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void XtcReaderChapterSelectionActivity::loop() {
|
||||||
|
const bool prevReleased = mappedInput.wasReleased(MappedInputManager::Button::Up) ||
|
||||||
|
mappedInput.wasReleased(MappedInputManager::Button::Left);
|
||||||
|
const bool nextReleased = mappedInput.wasReleased(MappedInputManager::Button::Down) ||
|
||||||
|
mappedInput.wasReleased(MappedInputManager::Button::Right);
|
||||||
|
|
||||||
|
const bool skipPage = mappedInput.getHeldTime() > SKIP_PAGE_MS;
|
||||||
|
const int pageItems = getPageItems();
|
||||||
|
|
||||||
|
if (mappedInput.wasReleased(MappedInputManager::Button::Confirm)) {
|
||||||
|
const auto& chapters = xtc->getChapters();
|
||||||
|
if (!chapters.empty() && selectorIndex >= 0 && selectorIndex < static_cast<int>(chapters.size())) {
|
||||||
|
onSelectPage(chapters[selectorIndex].startPage);
|
||||||
|
}
|
||||||
|
} else if (mappedInput.wasReleased(MappedInputManager::Button::Back)) {
|
||||||
|
onGoBack();
|
||||||
|
} else if (prevReleased) {
|
||||||
|
const int total = static_cast<int>(xtc->getChapters().size());
|
||||||
|
if (total == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (skipPage) {
|
||||||
|
selectorIndex = ((selectorIndex / pageItems - 1) * pageItems + total) % total;
|
||||||
|
} else {
|
||||||
|
selectorIndex = (selectorIndex + total - 1) % total;
|
||||||
|
}
|
||||||
|
updateRequired = true;
|
||||||
|
} else if (nextReleased) {
|
||||||
|
const int total = static_cast<int>(xtc->getChapters().size());
|
||||||
|
if (total == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (skipPage) {
|
||||||
|
selectorIndex = ((selectorIndex / pageItems + 1) * pageItems) % total;
|
||||||
|
} else {
|
||||||
|
selectorIndex = (selectorIndex + 1) % total;
|
||||||
|
}
|
||||||
|
updateRequired = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void XtcReaderChapterSelectionActivity::displayTaskLoop() {
|
||||||
|
while (true) {
|
||||||
|
if (updateRequired) {
|
||||||
|
updateRequired = false;
|
||||||
|
xSemaphoreTake(renderingMutex, portMAX_DELAY);
|
||||||
|
renderScreen();
|
||||||
|
xSemaphoreGive(renderingMutex);
|
||||||
|
}
|
||||||
|
vTaskDelay(10 / portTICK_PERIOD_MS);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void XtcReaderChapterSelectionActivity::renderScreen() {
|
||||||
|
renderer.clearScreen();
|
||||||
|
|
||||||
|
const auto pageWidth = renderer.getScreenWidth();
|
||||||
|
const int pageItems = getPageItems();
|
||||||
|
renderer.drawCenteredText(READER_FONT_ID, 10, "Select Chapter", true, BOLD);
|
||||||
|
|
||||||
|
const auto& chapters = xtc->getChapters();
|
||||||
|
if (chapters.empty()) {
|
||||||
|
renderer.drawCenteredText(UI_FONT_ID, 120, "No chapters", true, REGULAR);
|
||||||
|
renderer.displayBuffer();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto pageStartIndex = selectorIndex / pageItems * pageItems;
|
||||||
|
renderer.fillRect(0, 60 + (selectorIndex % pageItems) * 30 - 2, pageWidth - 1, 30);
|
||||||
|
for (int i = pageStartIndex; i < static_cast<int>(chapters.size()) && i < pageStartIndex + pageItems; i++) {
|
||||||
|
const auto& chapter = chapters[i];
|
||||||
|
const char* title = chapter.name.empty() ? "Unnamed" : chapter.name.c_str();
|
||||||
|
renderer.drawText(UI_FONT_ID, 20, 60 + (i % pageItems) * 30, title, i != selectorIndex);
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer.displayBuffer();
|
||||||
|
}
|
||||||
41
src/activities/reader/XtcReaderChapterSelectionActivity.h
Normal file
41
src/activities/reader/XtcReaderChapterSelectionActivity.h
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
#pragma once
|
||||||
|
#include <Xtc.h>
|
||||||
|
#include <freertos/FreeRTOS.h>
|
||||||
|
#include <freertos/semphr.h>
|
||||||
|
#include <freertos/task.h>
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
#include "../Activity.h"
|
||||||
|
|
||||||
|
class XtcReaderChapterSelectionActivity final : public Activity {
|
||||||
|
std::shared_ptr<Xtc> xtc;
|
||||||
|
TaskHandle_t displayTaskHandle = nullptr;
|
||||||
|
SemaphoreHandle_t renderingMutex = nullptr;
|
||||||
|
uint32_t currentPage = 0;
|
||||||
|
int selectorIndex = 0;
|
||||||
|
bool updateRequired = false;
|
||||||
|
const std::function<void()> onGoBack;
|
||||||
|
const std::function<void(uint32_t newPage)> onSelectPage;
|
||||||
|
|
||||||
|
int getPageItems() const;
|
||||||
|
int findChapterIndexForPage(uint32_t page) const;
|
||||||
|
|
||||||
|
static void taskTrampoline(void* param);
|
||||||
|
[[noreturn]] void displayTaskLoop();
|
||||||
|
void renderScreen();
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit XtcReaderChapterSelectionActivity(GfxRenderer& renderer, MappedInputManager& mappedInput,
|
||||||
|
const std::shared_ptr<Xtc>& xtc, uint32_t currentPage,
|
||||||
|
const std::function<void()>& onGoBack,
|
||||||
|
const std::function<void(uint32_t newPage)>& onSelectPage)
|
||||||
|
: Activity("XtcReaderChapterSelection", renderer, mappedInput),
|
||||||
|
xtc(xtc),
|
||||||
|
currentPage(currentPage),
|
||||||
|
onGoBack(onGoBack),
|
||||||
|
onSelectPage(onSelectPage) {}
|
||||||
|
void onEnter() override;
|
||||||
|
void onExit() override;
|
||||||
|
void loop() override;
|
||||||
|
};
|
||||||
@ -159,6 +159,9 @@ void SettingsActivity::render() const {
|
|||||||
// Draw header
|
// Draw header
|
||||||
renderer.drawCenteredText(READER_FONT_ID, 10, "Settings", true, BOLD);
|
renderer.drawCenteredText(READER_FONT_ID, 10, "Settings", true, BOLD);
|
||||||
|
|
||||||
|
// Draw selection
|
||||||
|
renderer.fillRect(0, 60 + selectedSettingIndex * 30 - 2, pageWidth - 1, 30);
|
||||||
|
|
||||||
// Draw all settings
|
// Draw all settings
|
||||||
for (int i = 0; i < settingsCount; i++) {
|
for (int i = 0; i < settingsCount; i++) {
|
||||||
const int settingY = 60 + i * 30; // 30 pixels between settings
|
const int settingY = 60 + i * 30; // 30 pixels between settings
|
||||||
@ -169,18 +172,19 @@ void SettingsActivity::render() const {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Draw setting name
|
// Draw setting name
|
||||||
renderer.drawText(UI_FONT_ID, 20, settingY, settingsList[i].name);
|
renderer.drawText(UI_FONT_ID, 20, settingY, settingsList[i].name, i != selectedSettingIndex);
|
||||||
|
|
||||||
// Draw value based on setting type
|
// Draw value based on setting type
|
||||||
|
std::string valueText = "";
|
||||||
if (settingsList[i].type == SettingType::TOGGLE && settingsList[i].valuePtr != nullptr) {
|
if (settingsList[i].type == SettingType::TOGGLE && settingsList[i].valuePtr != nullptr) {
|
||||||
const bool value = SETTINGS.*(settingsList[i].valuePtr);
|
const bool value = SETTINGS.*(settingsList[i].valuePtr);
|
||||||
renderer.drawText(UI_FONT_ID, pageWidth - 80, settingY, value ? "ON" : "OFF");
|
valueText = value ? "ON" : "OFF";
|
||||||
} else if (settingsList[i].type == SettingType::ENUM && settingsList[i].valuePtr != nullptr) {
|
} else if (settingsList[i].type == SettingType::ENUM && settingsList[i].valuePtr != nullptr) {
|
||||||
const uint8_t value = SETTINGS.*(settingsList[i].valuePtr);
|
const uint8_t value = SETTINGS.*(settingsList[i].valuePtr);
|
||||||
auto valueText = settingsList[i].enumValues[value];
|
valueText = settingsList[i].enumValues[value];
|
||||||
const auto width = renderer.getTextWidth(UI_FONT_ID, valueText.c_str());
|
|
||||||
renderer.drawText(UI_FONT_ID, pageWidth - 50 - width, settingY, valueText.c_str());
|
|
||||||
}
|
}
|
||||||
|
const auto width = renderer.getTextWidth(UI_FONT_ID, valueText.c_str());
|
||||||
|
renderer.drawText(UI_FONT_ID, pageWidth - 20 - width, settingY, valueText.c_str(), i != selectedSettingIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Draw version text above button hints
|
// Draw version text above button hints
|
||||||
|
|||||||
@ -84,7 +84,7 @@ void verifyWakeupLongPress() {
|
|||||||
const auto start = millis();
|
const auto start = millis();
|
||||||
bool abort = false;
|
bool abort = false;
|
||||||
// It takes us some time to wake up from deep sleep, so we need to subtract that from the duration
|
// It takes us some time to wake up from deep sleep, so we need to subtract that from the duration
|
||||||
uint16_t calibration = 25;
|
uint16_t calibration = 29;
|
||||||
uint16_t calibratedPressDuration =
|
uint16_t calibratedPressDuration =
|
||||||
(calibration < SETTINGS.getPowerButtonDuration()) ? SETTINGS.getPowerButtonDuration() - calibration : 1;
|
(calibration < SETTINGS.getPowerButtonDuration()) ? SETTINGS.getPowerButtonDuration() - calibration : 1;
|
||||||
|
|
||||||
@ -179,8 +179,6 @@ void setup() {
|
|||||||
Serial.begin(115200);
|
Serial.begin(115200);
|
||||||
}
|
}
|
||||||
|
|
||||||
Serial.printf("[%lu] [ ] Starting CrossPoint version " CROSSPOINT_VERSION "\n", millis());
|
|
||||||
|
|
||||||
inputManager.begin();
|
inputManager.begin();
|
||||||
// Initialize pins
|
// Initialize pins
|
||||||
pinMode(BAT_GPIO0, INPUT);
|
pinMode(BAT_GPIO0, INPUT);
|
||||||
@ -203,6 +201,9 @@ void setup() {
|
|||||||
// verify power button press duration after we've read settings.
|
// verify power button press duration after we've read settings.
|
||||||
verifyWakeupLongPress();
|
verifyWakeupLongPress();
|
||||||
|
|
||||||
|
// First serial output only here to avoid timing inconsistencies for power button press duration verification
|
||||||
|
Serial.printf("[%lu] [ ] Starting CrossPoint version " CROSSPOINT_VERSION "\n", millis());
|
||||||
|
|
||||||
setupDisplayAndFonts();
|
setupDisplayAndFonts();
|
||||||
|
|
||||||
exitActivity();
|
exitActivity();
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user