#include "HomeActivity.h" #include #include #include #include #include #include #include #include #include #include #include #include #include "BookManageMenuActivity.h" #include "CrossPointSettings.h" #include "CrossPointState.h" #include "MappedInputManager.h" #include "OpdsServerStore.h" #include "RecentBooksStore.h" #include "activities/ActivityResult.h" #include "components/UITheme.h" #include "fontIds.h" #include "util/BookManager.h" #include "util/StringUtils.h" int HomeActivity::getMenuItemCount() const { int count = 4; // File Browser, Recents, File transfer, Settings if (!recentBooks.empty()) { count += recentBooks.size(); } if (hasOpdsServers) { count++; } return count; } void HomeActivity::loadRecentBooks(int maxBooks) { recentBooks.clear(); const auto& books = RECENT_BOOKS.getBooks(); recentBooks.reserve(std::min(static_cast(books.size()), maxBooks)); for (const RecentBook& book : books) { // Limit to maximum number of recent books if (recentBooks.size() >= maxBooks) { break; } // Skip if file no longer exists if (!Storage.exists(book.path.c_str())) { continue; } recentBooks.push_back(book); } } void HomeActivity::loadRecentCovers(int coverHeight) { recentsLoading = true; bool showingLoading = false; Rect popupRect; int progress = 0; for (RecentBook& book : recentBooks) { if (!book.coverBmpPath.empty()) { std::string coverPath = UITheme::getCoverThumbPath(book.coverBmpPath, coverHeight); if (!Epub::isValidThumbnailBmp(coverPath)) { if (!showingLoading) { showingLoading = true; popupRect = GUI.drawPopup(renderer, tr(STR_LOADING)); } GUI.fillPopupProgress(renderer, popupRect, 10 + progress * (90 / recentBooks.size())); bool success = false; if (StringUtils::checkFileExtension(book.path, ".epub")) { Epub epub(book.path, "/.crosspoint"); if (!epub.load(false, true)) { epub.load(true, true); } success = epub.generateThumbBmp(coverHeight); if (success) { const std::string thumbPath = epub.getThumbBmpPath(coverHeight); RECENT_BOOKS.updateBook(book.path, book.title, book.author, thumbPath); book.coverBmpPath = thumbPath; } else { const int thumbWidth = static_cast(coverHeight * 0.6); success = PlaceholderCoverGenerator::generate(coverPath, book.title, book.author, thumbWidth, coverHeight); if (!success) { epub.generateInvalidFormatThumbBmp(coverHeight); } } } else if (StringUtils::checkFileExtension(book.path, ".xtch") || StringUtils::checkFileExtension(book.path, ".xtc")) { Xtc xtc(book.path, "/.crosspoint"); if (xtc.load()) { success = xtc.generateThumbBmp(coverHeight); if (success) { const std::string thumbPath = xtc.getThumbBmpPath(coverHeight); RECENT_BOOKS.updateBook(book.path, book.title, book.author, thumbPath); book.coverBmpPath = thumbPath; } } if (!success) { const int thumbWidth = static_cast(coverHeight * 0.6); PlaceholderCoverGenerator::generate(coverPath, book.title, book.author, thumbWidth, coverHeight); } } else { const int thumbWidth = static_cast(coverHeight * 0.6); PlaceholderCoverGenerator::generate(coverPath, book.title, book.author, thumbWidth, coverHeight); } coverRendered = false; requestUpdate(); } } progress++; } recentsLoaded = true; recentsLoading = false; } void HomeActivity::onEnter() { Activity::onEnter(); hasOpdsServers = OPDS_STORE.hasServers(); selectorIndex = 0; const auto& metrics = UITheme::getInstance().getMetrics(); loadRecentBooks(metrics.homeRecentBooksCount); // Trigger first update requestUpdate(); } void HomeActivity::onExit() { Activity::onExit(); // Free the stored cover buffer if any freeCoverBuffer(); } bool HomeActivity::storeCoverBuffer() { uint8_t* frameBuffer = renderer.getFrameBuffer(); if (!frameBuffer) { return false; } // Free any existing buffer first freeCoverBuffer(); const size_t bufferSize = GfxRenderer::getBufferSize(); coverBuffer = static_cast(malloc(bufferSize)); if (!coverBuffer) { return false; } memcpy(coverBuffer, frameBuffer, bufferSize); return true; } bool HomeActivity::restoreCoverBuffer() { if (!coverBuffer) { return false; } uint8_t* frameBuffer = renderer.getFrameBuffer(); if (!frameBuffer) { return false; } const size_t bufferSize = GfxRenderer::getBufferSize(); memcpy(frameBuffer, coverBuffer, bufferSize); return true; } void HomeActivity::freeCoverBuffer() { if (coverBuffer) { free(coverBuffer); coverBuffer = nullptr; } coverBufferStored = false; } void HomeActivity::loop() { const int menuCount = getMenuItemCount(); buttonNavigator.onNext([this, menuCount] { selectorIndex = ButtonNavigator::nextIndex(selectorIndex, menuCount); requestUpdate(); }); buttonNavigator.onPrevious([this, menuCount] { selectorIndex = ButtonNavigator::previousIndex(selectorIndex, menuCount); requestUpdate(); }); // Long-press Confirm: manage menu for recent books, or browse archive for Browse Files if (mappedInput.isPressed(MappedInputManager::Button::Confirm) && mappedInput.getHeldTime() >= LONG_PRESS_MS && !ignoreNextConfirmRelease) { if (selectorIndex < static_cast(recentBooks.size())) { ignoreNextConfirmRelease = true; openManageMenu(recentBooks[selectorIndex].path); return; } const int menuSelectedIndex = selectorIndex - static_cast(recentBooks.size()); if (menuSelectedIndex == 0) { ignoreNextConfirmRelease = true; activityManager.goToFileBrowser("/.archive"); return; } } if (mappedInput.wasReleased(MappedInputManager::Button::Confirm)) { if (ignoreNextConfirmRelease) { ignoreNextConfirmRelease = false; return; } int idx = 0; int menuSelectedIndex = selectorIndex - static_cast(recentBooks.size()); const int fileBrowserIdx = idx++; const int recentsIdx = idx++; const int opdsLibraryIdx = hasOpdsServers ? idx++ : -1; const int fileTransferIdx = idx++; const int settingsIdx = idx; if (selectorIndex < static_cast(recentBooks.size())) { onSelectBook(recentBooks[selectorIndex].path); } else if (menuSelectedIndex == fileBrowserIdx) { onFileBrowserOpen(); } else if (menuSelectedIndex == recentsIdx) { onRecentsOpen(); } else if (menuSelectedIndex == opdsLibraryIdx) { onOpdsBrowserOpen(); } else if (menuSelectedIndex == fileTransferIdx) { onFileTransferOpen(); } else if (menuSelectedIndex == settingsIdx) { onSettingsOpen(); } } } void HomeActivity::render(RenderLock&&) { const auto& metrics = UITheme::getInstance().getMetrics(); const auto pageWidth = renderer.getScreenWidth(); const auto pageHeight = renderer.getScreenHeight(); renderer.clearScreen(); bool bufferRestored = coverBufferStored && restoreCoverBuffer(); GUI.drawHeader(renderer, Rect{0, metrics.topPadding, pageWidth, metrics.homeTopPadding}, nullptr); GUI.drawRecentBookCover(renderer, Rect{0, metrics.homeTopPadding, pageWidth, metrics.homeCoverTileHeight}, recentBooks, selectorIndex, coverRendered, coverBufferStored, bufferRestored, std::bind(&HomeActivity::storeCoverBuffer, this)); // Build menu items dynamically std::vector menuItems = {tr(STR_BROWSE_FILES), tr(STR_MENU_RECENT_BOOKS), tr(STR_FILE_TRANSFER), tr(STR_SETTINGS_TITLE)}; std::vector menuIcons = {Folder, Recent, Transfer, Settings}; if (hasOpdsServers) { // Insert OPDS Browser after File Browser menuItems.insert(menuItems.begin() + 2, tr(STR_OPDS_BROWSER)); menuIcons.insert(menuIcons.begin() + 2, Library); } GUI.drawButtonMenu( renderer, Rect{0, metrics.homeTopPadding + metrics.homeCoverTileHeight + metrics.verticalSpacing, pageWidth, pageHeight - (metrics.headerHeight + metrics.homeTopPadding + metrics.verticalSpacing * 2 + metrics.buttonHintsHeight)}, static_cast(menuItems.size()), selectorIndex - recentBooks.size(), [&menuItems](int index) { return std::string(menuItems[index]); }, [&menuIcons](int index) { return menuIcons[index]; }); const auto labels = mappedInput.mapLabels("", tr(STR_SELECT), tr(STR_DIR_UP), tr(STR_DIR_DOWN)); GUI.drawButtonHints(renderer, labels.btn1, labels.btn2, labels.btn3, labels.btn4); renderer.displayBuffer(); if (!firstRenderDone) { firstRenderDone = true; requestUpdate(); } else if (!recentsLoaded && !recentsLoading) { recentsLoading = true; loadRecentCovers(metrics.homeCoverHeight); } } void HomeActivity::onSelectBook(const std::string& path) { activityManager.goToReader(path); } void HomeActivity::onFileBrowserOpen() { activityManager.goToFileBrowser(); } void HomeActivity::onRecentsOpen() { activityManager.goToRecentBooks(); } void HomeActivity::onSettingsOpen() { activityManager.goToSettings(); } void HomeActivity::onFileTransferOpen() { activityManager.goToFileTransfer(); } void HomeActivity::onOpdsBrowserOpen() { activityManager.goToBrowser(); } void HomeActivity::openManageMenu(const std::string& bookPath) { const bool isArchived = BookManager::isArchived(bookPath); const std::string capturedPath = bookPath; startActivityForResult( std::make_unique(renderer, mappedInput, capturedPath, isArchived, true), [this, capturedPath](const ActivityResult& result) { if (result.isCancelled) { requestUpdate(); return; } const auto& menuResult = std::get(result.data); auto action = static_cast(menuResult.action); bool success = false; switch (action) { case BookManageMenuActivity::Action::ARCHIVE: success = BookManager::archiveBook(capturedPath); break; case BookManageMenuActivity::Action::UNARCHIVE: success = BookManager::unarchiveBook(capturedPath); break; case BookManageMenuActivity::Action::DELETE: success = BookManager::deleteBook(capturedPath); break; case BookManageMenuActivity::Action::DELETE_CACHE: success = BookManager::deleteBookCache(capturedPath); break; case BookManageMenuActivity::Action::REINDEX: success = BookManager::reindexBook(capturedPath, false); break; case BookManageMenuActivity::Action::REINDEX_FULL: success = BookManager::reindexBook(capturedPath, true); break; } { RenderLock lock(*this); GUI.drawPopup(renderer, success ? tr(STR_DONE) : tr(STR_ACTION_FAILED)); } requestUpdateAndWait(); recentBooks.clear(); recentsLoaded = false; recentsLoading = false; coverRendered = false; freeCoverBuffer(); selectorIndex = 0; firstRenderDone = false; requestUpdate(); }); }