/* A game chooser for borgs * by: Christian Kroll * date: Thursday, 2008/03/16 */ #include <stdlib.h> #include <assert.h> #include <inttypes.h> // architecture dependent stuff #include "../compat/pgmspace.h" #include "menu.h" #include "../config.h" #include "../util.h" #include "../pixel.h" #include "../joystick/joystick.h" extern game_descriptor_t _game_descriptors_start__[]; extern game_descriptor_t _game_descriptors_end__[]; // defines #define MENU_ITEM_MAX ((uint8_t)(((size_t)_game_descriptors_end__ - (size_t)_game_descriptors_start__) / sizeof(game_descriptor_t))) #define MENU_WIDTH_ICON 8 #define MENU_HEIGHT_ICON 8 #define MENU_WIDTH_DELIMITER 2 #define MENU_POLL_INTERVAL 10 #define MENU_TIMEOUT_ITERATIONS 2000 #define MENU_WAIT_CHATTER 60 #define MENU_WAIT_INITIAL 40 #define MENU_WAIT_INCREMENT 0 #define MENU_NEXTITEM(item) ((item + 1) % MENU_ITEM_MAX) #define MENU_PREVITEM(item) ((item + MENU_ITEM_MAX - 1) % MENU_ITEM_MAX) enum menu_direction { MENU_DIRECTION_LEFT, MENU_DIRECTION_RIGHT, MENU_DIRECTION_STILL }; #ifdef NDEBUG typedef uint8_t menu_direction_t; #else typedef enum menu_direction menu_direction_t; #endif static void menu_setpixel(uint8_t x, uint8_t y, uint8_t isSet) { uint8_t nColor; // mirror mirror on the wall, what's the quirkiest API of them all... x = NUM_COLS - 1 - x; uint8_t nMiddle = (NUM_COLS - MENU_WIDTH_ICON) / 2; if (isSet != 0) { if ((x >= nMiddle - MENU_WIDTH_DELIMITER) && (x < (nMiddle + MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER))) { nColor = 3; } else if ((x == (nMiddle - MENU_WIDTH_DELIMITER - 1)) || (x == (nMiddle + MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER))) { nColor = 2; } else { nColor = 1; } } else { nColor = 0; } setpixel((pixel){x, y}, nColor); } static uint8_t menu_getIconPixel(uint8_t item, uint8_t x, uint8_t y) { // is x within the icon or do we have reached the delimiter? if (x < MENU_WIDTH_ICON) { // return pixel return (0x80 >> x) & pgm_read_byte(&_game_descriptors_start__[item].icon[y]); } else { // delimiter return 0; } } static void menu_animate(uint8_t miInitial, menu_direction_t direction) { int16_t nWait = MENU_WAIT_INITIAL; // space between left border and the icon in the middle uint8_t nWidthSide = (NUM_COLS - MENU_WIDTH_ICON) / 2; // determine the icon at the leftmost position uint8_t mi = miInitial + MENU_ITEM_MAX; uint8_t nBack = nWidthSide / (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER); if ((nWidthSide % (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER)) != 0) { ++nBack; } mi = (mi + MENU_ITEM_MAX - (nBack % MENU_ITEM_MAX)) % MENU_ITEM_MAX; // start and stop offsets for the scrolling icons (both are 0 for stills) uint8_t nStart, nStop; if (direction == MENU_DIRECTION_STILL) { nStart = 0; nStop = 0; } else { nStart = 1; nStop = MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER; } // draw menu screen for each offset within the nStart/nStop range uint8_t i; for (i = nStart; i <= nStop; ++i) { // offset of the left most icon if it is cut by the left border uint8_t nInitialSideOffset = (((MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER) - (nWidthSide % (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER))) + (direction == MENU_DIRECTION_LEFT ? i : -i) + (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER)) % (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER); // an initial side offset of 0 means the leftmost icon was changed // if we are scrolling to the left, increment value for leftmost item if (direction == MENU_DIRECTION_LEFT && nInitialSideOffset == 0) { mi = MENU_NEXTITEM(mi); } // draw the icons from the leftmost position (line by line) uint8_t y; for (y = 0; y < MENU_HEIGHT_ICON; ++y) { uint8_t miCurrent = mi; uint8_t nIconOffset = nInitialSideOffset; uint8_t x; for (x = 0; x < NUM_COLS; ++x) { uint8_t nPixel = menu_getIconPixel(miCurrent, nIconOffset, y); menu_setpixel(x, ((NUM_ROWS - MENU_HEIGHT_ICON) / 2) + y, nPixel); if (++nIconOffset >= (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER)) { nIconOffset = 0; miCurrent = MENU_NEXTITEM(miCurrent); } } } // an initial side offset of 0 means the leftmost icon was changed // if we are scrolling to the right, decrement value for leftmost item if (direction == MENU_DIRECTION_RIGHT && nInitialSideOffset == 0) { mi = MENU_PREVITEM(mi); } // wait between the frames so that the animation can be seen wait(nWait); // animation speed can be throttled nWait += MENU_WAIT_INCREMENT; } } void menu() { if (MENU_ITEM_MAX != 0) { // don't let WAIT() query fire button to prevent endless circular jumps waitForFire = 0; clear_screen(0); // wait as long as "fire" is pressed to prevent unwanted selections while (JOYISFIRE) { wait(MENU_POLL_INTERVAL); } // set initial menu item static uint8_t miSelection = 0; // scroll in currently selected menu item menu_animate(MENU_PREVITEM(miSelection), MENU_DIRECTION_LEFT); uint16_t nMenuIterations= MENU_TIMEOUT_ITERATIONS; while (1) { // the user has made her/his choice if (JOYISFIRE) { // prevent unwanted selections while (JOYISFIRE) { wait(MENU_POLL_INTERVAL); } // work against the chatter effects of dump joysticks wait(MENU_WAIT_CHATTER); // call corresponding function _game_descriptors_start__[miSelection].run(); break; } // change selected item and do some scrolling else if (JOYISRIGHT) { menu_animate(miSelection, MENU_DIRECTION_LEFT); miSelection = MENU_NEXTITEM(miSelection); nMenuIterations = MENU_TIMEOUT_ITERATIONS; } else if (JOYISLEFT) { menu_animate(miSelection, MENU_DIRECTION_RIGHT); miSelection = MENU_PREVITEM(miSelection); nMenuIterations = MENU_TIMEOUT_ITERATIONS; } // exit menu else if (JOYISUP) { break; } // return if timeout is reached else { wait(MENU_POLL_INTERVAL); if (--nMenuIterations == 0) break; } } waitForFire = 1; } return; }