|
@ -35,81 +35,51 @@ extern game_descriptor_t _game_descriptors_end__[]; |
|
|
#define MENU_NEXTITEM(item) ((item + 1) % MENU_ITEM_MAX) |
|
|
#define MENU_NEXTITEM(item) ((item + 1) % MENU_ITEM_MAX) |
|
|
#define MENU_PREVITEM(item) ((item + MENU_ITEM_MAX - 1) % MENU_ITEM_MAX) |
|
|
#define MENU_PREVITEM(item) ((item + MENU_ITEM_MAX - 1) % MENU_ITEM_MAX) |
|
|
|
|
|
|
|
|
void menu() |
|
|
|
|
|
|
|
|
typedef enum menu_direction_t |
|
|
{ |
|
|
{ |
|
|
if (MENU_ITEM_MAX != 0) |
|
|
MENU_DIRECTION_LEFT, |
|
|
{ |
|
|
MENU_DIRECTION_RIGHT, |
|
|
// don't let WAIT() query fire button to prevent endless circular jumps
|
|
|
MENU_DIRECTION_STILL |
|
|
waitForFire = 0; |
|
|
} |
|
|
|
|
|
menu_direction_t; |
|
|
|
|
|
|
|
|
clear_screen(0); |
|
|
|
|
|
|
|
|
|
|
|
// wait as long as "fire" is pressed to prevent unwanted selections
|
|
|
static void menu_setpixel(uint8_t x, uint8_t y, uint8_t isSet) |
|
|
while (JOYISFIRE) |
|
|
{ |
|
|
{ |
|
|
uint8_t nColor; |
|
|
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; |
|
|
// 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; |
|
|
|
|
|
|
|
|
while (1) |
|
|
if (isSet != 0) |
|
|
{ |
|
|
|
|
|
// the user has made her/his choice
|
|
|
|
|
|
if (JOYISFIRE) |
|
|
|
|
|
{ |
|
|
{ |
|
|
// prevent unwanted selections
|
|
|
if ((x >= nMiddle - MENU_WIDTH_DELIMITER) && (x < (nMiddle |
|
|
while (JOYISFIRE) |
|
|
+ MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER))) |
|
|
{ |
|
|
{ |
|
|
wait(MENU_POLL_INTERVAL); |
|
|
nColor = 3; |
|
|
} |
|
|
|
|
|
// 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 ((x == (nMiddle - MENU_WIDTH_DELIMITER - 1)) || (x == (nMiddle |
|
|
else if (JOYISRIGHT) |
|
|
+ MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER))) |
|
|
{ |
|
|
{ |
|
|
menu_animate(miSelection, MENU_DIRECTION_LEFT); |
|
|
nColor = 2; |
|
|
miSelection = MENU_NEXTITEM(miSelection); |
|
|
|
|
|
nMenuIterations = MENU_TIMEOUT_ITERATIONS; |
|
|
|
|
|
} |
|
|
} |
|
|
else if (JOYISLEFT) |
|
|
else |
|
|
{ |
|
|
{ |
|
|
menu_animate(miSelection, MENU_DIRECTION_RIGHT); |
|
|
nColor = 1; |
|
|
miSelection = MENU_PREVITEM(miSelection); |
|
|
|
|
|
nMenuIterations = MENU_TIMEOUT_ITERATIONS; |
|
|
|
|
|
} |
|
|
} |
|
|
// exit menu
|
|
|
|
|
|
else if (JOYISUP) |
|
|
|
|
|
{ |
|
|
|
|
|
break; |
|
|
|
|
|
} |
|
|
} |
|
|
// return if timeout is reached
|
|
|
|
|
|
else |
|
|
else |
|
|
{ |
|
|
{ |
|
|
wait(MENU_POLL_INTERVAL); |
|
|
nColor = 0; |
|
|
if (--nMenuIterations == 0) |
|
|
|
|
|
break; |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
waitForFire = 1; |
|
|
setpixel((pixel){x, y}, nColor); |
|
|
} |
|
|
|
|
|
return; |
|
|
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t menu_getIconPixel(uint8_t item, int8_t x, int8_t y) |
|
|
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?
|
|
|
// is x within the icon or do we have reached the delimiter?
|
|
|
if (x < MENU_WIDTH_ICON) |
|
|
if (x < MENU_WIDTH_ICON) |
|
@ -125,16 +95,17 @@ uint8_t menu_getIconPixel(uint8_t item, int8_t x, int8_t y) |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void menu_animate(uint8_t miInitial, menu_direction_t direction) |
|
|
|
|
|
|
|
|
static void menu_animate(uint8_t miInitial, menu_direction_t direction) |
|
|
{ |
|
|
{ |
|
|
int16_t nWait= MENU_WAIT_INITIAL; |
|
|
int16_t nWait = MENU_WAIT_INITIAL; |
|
|
|
|
|
|
|
|
// space between left border and the icon in the middle
|
|
|
// space between left border and the icon in the middle
|
|
|
int8_t nWidthSide = (NUM_COLS - MENU_WIDTH_ICON) / 2; |
|
|
uint8_t nWidthSide = (NUM_COLS - MENU_WIDTH_ICON) / 2; |
|
|
|
|
|
|
|
|
// determine the icon at the leftmost position
|
|
|
// determine the icon at the leftmost position
|
|
|
uint8_t mi = miInitial + MENU_ITEM_MAX; |
|
|
uint8_t mi = miInitial + MENU_ITEM_MAX; |
|
|
int8_t nBack = nWidthSide / (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER); |
|
|
uint8_t nBack = nWidthSide / (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER); |
|
|
if ((nWidthSide % (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER)) != 0) |
|
|
if ((nWidthSide % (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER)) != 0) |
|
|
{ |
|
|
{ |
|
|
++nBack; |
|
|
++nBack; |
|
@ -142,7 +113,7 @@ void menu_animate(uint8_t miInitial, menu_direction_t direction) |
|
|
mi = (mi + MENU_ITEM_MAX - (nBack % MENU_ITEM_MAX)) % MENU_ITEM_MAX; |
|
|
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)
|
|
|
// start and stop offsets for the scrolling icons (both are 0 for stills)
|
|
|
int8_t nStart, nStop; |
|
|
uint8_t nStart, nStop; |
|
|
if (direction == MENU_DIRECTION_STILL) |
|
|
if (direction == MENU_DIRECTION_STILL) |
|
|
{ |
|
|
{ |
|
|
nStart = 0; |
|
|
nStart = 0; |
|
@ -155,41 +126,33 @@ void menu_animate(uint8_t miInitial, menu_direction_t direction) |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
// draw menu screen for each offset within the nStart/nStop range
|
|
|
// draw menu screen for each offset within the nStart/nStop range
|
|
|
int8_t i; |
|
|
uint8_t i; |
|
|
for (i = nStart; i <= nStop; ++i) |
|
|
for (i = nStart; i <= nStop; ++i) |
|
|
{ |
|
|
{ |
|
|
int8_t nOffset; |
|
|
|
|
|
if (direction == MENU_DIRECTION_LEFT) |
|
|
|
|
|
nOffset = i; |
|
|
|
|
|
else |
|
|
|
|
|
nOffset = -i; |
|
|
|
|
|
|
|
|
|
|
|
// offset of the left most icon if it is cut by the left border
|
|
|
// offset of the left most icon if it is cut by the left border
|
|
|
int8_t nInitialSideOffset = (((MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER) |
|
|
uint8_t nInitialSideOffset = (((MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER) |
|
|
- (nWidthSide % (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER))) |
|
|
- (nWidthSide % (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER))) |
|
|
+ nOffset + (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER)) |
|
|
+ (direction == MENU_DIRECTION_LEFT ? i : -i) |
|
|
|
|
|
+ (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER)) |
|
|
% (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER); |
|
|
% (MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER); |
|
|
|
|
|
|
|
|
// an initial side offset of 0 means the leftmost icon was changed
|
|
|
// 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 we are scrolling to the left, increment value for leftmost item
|
|
|
if (direction == MENU_DIRECTION_LEFT) |
|
|
if (direction == MENU_DIRECTION_LEFT && nInitialSideOffset == 0) |
|
|
{ |
|
|
|
|
|
if (nInitialSideOffset == 0) |
|
|
|
|
|
{ |
|
|
{ |
|
|
mi = MENU_NEXTITEM(mi); |
|
|
mi = MENU_NEXTITEM(mi); |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// draw the icons from the leftmost position (line by line)
|
|
|
// draw the icons from the leftmost position (line by line)
|
|
|
int8_t y; |
|
|
uint8_t y; |
|
|
for (y = 0; y < MENU_HEIGHT_ICON; ++y) |
|
|
for (y = 0; y < MENU_HEIGHT_ICON; ++y) |
|
|
{ |
|
|
{ |
|
|
uint8_t miCurrent = mi; |
|
|
uint8_t miCurrent = mi; |
|
|
int8_t nIconOffset = nInitialSideOffset; |
|
|
uint8_t nIconOffset = nInitialSideOffset; |
|
|
int8_t x; |
|
|
uint8_t x; |
|
|
for (x = 0; x < NUM_COLS; ++x) |
|
|
for (x = 0; x < NUM_COLS; ++x) |
|
|
{ |
|
|
{ |
|
|
int8_t nPixel = menu_getIconPixel(miCurrent, nIconOffset, y); |
|
|
uint8_t nPixel = menu_getIconPixel(miCurrent, nIconOffset, y); |
|
|
|
|
|
|
|
|
menu_setpixel(x, ((NUM_ROWS - MENU_HEIGHT_ICON) / 2) + y, |
|
|
menu_setpixel(x, ((NUM_ROWS - MENU_HEIGHT_ICON) / 2) + y, |
|
|
nPixel); |
|
|
nPixel); |
|
@ -203,13 +166,10 @@ void menu_animate(uint8_t miInitial, menu_direction_t direction) |
|
|
|
|
|
|
|
|
// an initial side offset of 0 means the leftmost icon was changed
|
|
|
// 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 we are scrolling to the right, decrement value for leftmost item
|
|
|
if (direction == MENU_DIRECTION_RIGHT) |
|
|
if (direction == MENU_DIRECTION_RIGHT && nInitialSideOffset == 0) |
|
|
{ |
|
|
|
|
|
if (nInitialSideOffset == 0) |
|
|
|
|
|
{ |
|
|
{ |
|
|
mi = MENU_PREVITEM(mi); |
|
|
mi = MENU_PREVITEM(mi); |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// wait between the frames so that the animation can be seen
|
|
|
// wait between the frames so that the animation can be seen
|
|
|
wait(nWait); |
|
|
wait(nWait); |
|
@ -218,36 +178,76 @@ void menu_animate(uint8_t miInitial, menu_direction_t direction) |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
void menu_setpixel(int8_t x, int8_t y, int8_t isSet) |
|
|
|
|
|
|
|
|
void menu() |
|
|
{ |
|
|
{ |
|
|
uint8_t nColor; |
|
|
if (MENU_ITEM_MAX != 0) |
|
|
|
|
|
{ |
|
|
|
|
|
// don't let WAIT() query fire button to prevent endless circular jumps
|
|
|
|
|
|
waitForFire = 0; |
|
|
|
|
|
|
|
|
// mirror mirror on the wall, what's the quirkiest API of them all...
|
|
|
clear_screen(0); |
|
|
x = NUM_COLS - 1 - x; |
|
|
|
|
|
uint8_t nMiddle = (NUM_COLS - MENU_WIDTH_ICON) / 2; |
|
|
|
|
|
|
|
|
|
|
|
if (isSet != 0) |
|
|
// wait as long as "fire" is pressed to prevent unwanted selections
|
|
|
|
|
|
while (JOYISFIRE) |
|
|
{ |
|
|
{ |
|
|
if ((x >= nMiddle - MENU_WIDTH_DELIMITER) && (x < (nMiddle |
|
|
wait(MENU_POLL_INTERVAL); |
|
|
+ MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER))) |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// 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) |
|
|
{ |
|
|
{ |
|
|
nColor = 3; |
|
|
// the user has made her/his choice
|
|
|
|
|
|
if (JOYISFIRE) |
|
|
|
|
|
{ |
|
|
|
|
|
// prevent unwanted selections
|
|
|
|
|
|
while (JOYISFIRE) |
|
|
|
|
|
{ |
|
|
|
|
|
wait(MENU_POLL_INTERVAL); |
|
|
} |
|
|
} |
|
|
else if ((x == (nMiddle - MENU_WIDTH_DELIMITER - 1)) || (x == (nMiddle |
|
|
// work against the chatter effects of dump joysticks
|
|
|
+ MENU_WIDTH_ICON + MENU_WIDTH_DELIMITER))) |
|
|
wait(MENU_WAIT_CHATTER); |
|
|
|
|
|
|
|
|
|
|
|
// call corresponding function
|
|
|
|
|
|
_game_descriptors_start__[miSelection].run(); |
|
|
|
|
|
|
|
|
|
|
|
break; |
|
|
|
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
// change selected item and do some scrolling
|
|
|
|
|
|
else if (JOYISRIGHT) |
|
|
{ |
|
|
{ |
|
|
nColor = 2; |
|
|
menu_animate(miSelection, MENU_DIRECTION_LEFT); |
|
|
|
|
|
miSelection = MENU_NEXTITEM(miSelection); |
|
|
|
|
|
nMenuIterations = MENU_TIMEOUT_ITERATIONS; |
|
|
} |
|
|
} |
|
|
else |
|
|
else if (JOYISLEFT) |
|
|
{ |
|
|
{ |
|
|
nColor = 1; |
|
|
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 |
|
|
else |
|
|
{ |
|
|
{ |
|
|
nColor = 0; |
|
|
wait(MENU_POLL_INTERVAL); |
|
|
|
|
|
if (--nMenuIterations == 0) |
|
|
|
|
|
break; |
|
|
|
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
setpixel((pixel){x, y}, nColor); |
|
|
waitForFire = 1; |
|
|
|
|
|
} |
|
|
|
|
|
return; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|