Add pagination to game launcher - 4 games per page
- GameLauncher now displays only 4 games per page to keep menu in bounds - Added page navigation with page indicator (Page X/Y) - KEY0 navigates between pages and within page - KEY1 selects the highlighted game - Touch selection works on current page only - Helper methods: get_total_pages(), get_page_start_index(), get_page_end_index() - Updated both lib/ and emulator/ versions for consistency
This commit is contained in:
@@ -7,7 +7,7 @@
|
||||
extern Font font_5x5_obj;
|
||||
GameLauncher::GameLauncher(uint16_t width, uint16_t height, LowLevelRenderer* renderer, LowLevelGUI* gui, InputManager* input_manager)
|
||||
: width(width), height(height), renderer(renderer), gui(gui), input_manager(input_manager),
|
||||
selected_index(0), selected_game(nullptr) {}
|
||||
selected_index(0), selected_game(nullptr), current_page(0) {}
|
||||
void GameLauncher::register_game(const char* name, const char* description,
|
||||
std::function<Game*(uint16_t, uint16_t, LowLevelRenderer*, LowLevelGUI*, InputManager*)> factory) {
|
||||
GameEntry entry;
|
||||
@@ -20,18 +20,35 @@ void GameLauncher::register_game(const char* name, const char* description,
|
||||
void GameLauncher::draw() {
|
||||
LowLevelWindow* window = gui->draw_new_window(10, 10, width - 20, height - 20, "Game Launcher");
|
||||
renderer->set_font(&font_5x5_obj);
|
||||
renderer->draw_string_scaled(30, 40, "Select a Game:", 2);
|
||||
for (size_t i = 0; i < games.size(); i++) {
|
||||
int y = MENU_Y_START + (i * MENU_ITEM_HEIGHT);
|
||||
bool is_selected = ((int)i == selected_index);
|
||||
|
||||
int total_pages = get_total_pages();
|
||||
char title[64];
|
||||
snprintf(title, sizeof(title), "Select a Game: (Page %d/%d)", current_page + 1, total_pages);
|
||||
renderer->draw_string_scaled(30, 40, title, 2);
|
||||
|
||||
int page_start = get_page_start_index();
|
||||
int page_end = get_page_end_index();
|
||||
|
||||
for (int i = page_start; i < page_end && i < (int)games.size(); i++) {
|
||||
int item_index = i - page_start;
|
||||
int y = MENU_Y_START + (item_index * MENU_ITEM_HEIGHT);
|
||||
bool is_selected = (i == selected_index);
|
||||
gui->draw_button(window, 20, y, games[i].name, is_selected, true);
|
||||
renderer->set_font(&font_5x5_obj);
|
||||
renderer->set_text_color(true);
|
||||
renderer->draw_string_scaled(50, y + 36, games[i].description, 1);
|
||||
}
|
||||
|
||||
char nav_text[128];
|
||||
if (total_pages > 1) {
|
||||
snprintf(nav_text, sizeof(nav_text), "KEY0: Next Page | KEY1: Select");
|
||||
} else {
|
||||
snprintf(nav_text, sizeof(nav_text), "KEY0: Navigate | KEY1: Select");
|
||||
}
|
||||
|
||||
const char* instructions;
|
||||
if (input_manager->has_buttons()) {
|
||||
instructions = "Touch game or use KEY0/KEY1";
|
||||
instructions = nav_text;
|
||||
} else {
|
||||
instructions = "Touch game to play";
|
||||
}
|
||||
@@ -42,8 +59,12 @@ bool GameLauncher::update(const InputEvent& event) {
|
||||
bool needs_refresh = false;
|
||||
switch (event.type) {
|
||||
case INPUT_TOUCH_DOWN: {
|
||||
for (size_t i = 0; i < games.size(); i++) {
|
||||
int y = MENU_Y_START + (i * MENU_ITEM_HEIGHT);
|
||||
int page_start = get_page_start_index();
|
||||
int page_end = get_page_end_index();
|
||||
|
||||
for (int i = page_start; i < page_end && i < (int)games.size(); i++) {
|
||||
int item_index = i - page_start;
|
||||
int y = MENU_Y_START + (item_index * MENU_ITEM_HEIGHT);
|
||||
if (event.y >= y - 5 && event.y < y + MENU_ITEM_HEIGHT - 5) {
|
||||
selected_game = games[i].factory(width, height, renderer, gui, input_manager);
|
||||
if (selected_game) {
|
||||
@@ -56,10 +77,34 @@ bool GameLauncher::update(const InputEvent& event) {
|
||||
break;
|
||||
}
|
||||
case INPUT_BUTTON_0: {
|
||||
if (games.size() > 0) {
|
||||
selected_index = (selected_index + 1) % games.size();
|
||||
needs_refresh = true;
|
||||
int page_start = get_page_start_index();
|
||||
int page_end = get_page_end_index();
|
||||
int total_pages = get_total_pages();
|
||||
|
||||
if (page_end - page_start > 1) {
|
||||
int old_index = selected_index;
|
||||
selected_index++;
|
||||
if (selected_index >= page_end) {
|
||||
if (current_page + 1 < total_pages) {
|
||||
current_page++;
|
||||
selected_index = get_page_start_index();
|
||||
} else {
|
||||
current_page = 0;
|
||||
selected_index = 0;
|
||||
}
|
||||
} else if (selected_index < page_start) {
|
||||
selected_index = page_start;
|
||||
}
|
||||
} else {
|
||||
if (current_page + 1 < total_pages) {
|
||||
current_page++;
|
||||
selected_index = get_page_start_index();
|
||||
} else {
|
||||
current_page = 0;
|
||||
selected_index = 0;
|
||||
}
|
||||
}
|
||||
needs_refresh = true;
|
||||
break;
|
||||
}
|
||||
case INPUT_BUTTON_1: {
|
||||
@@ -84,4 +129,18 @@ void GameLauncher::reset() {
|
||||
selected_game = nullptr;
|
||||
}
|
||||
selected_index = 0;
|
||||
current_page = 0;
|
||||
}
|
||||
|
||||
int GameLauncher::get_total_pages() const {
|
||||
if (games.empty()) return 1;
|
||||
return (games.size() + GAMES_PER_PAGE - 1) / GAMES_PER_PAGE;
|
||||
}
|
||||
|
||||
int GameLauncher::get_page_start_index() const {
|
||||
return current_page * GAMES_PER_PAGE;
|
||||
}
|
||||
|
||||
int GameLauncher::get_page_end_index() const {
|
||||
return (current_page + 1) * GAMES_PER_PAGE;
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user