rockbox/apps/plugins/flipit.c

500 lines
14 KiB
C
Raw Normal View History

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id: flipit.c,v 1.0 2003/01/18 23:51:47
*
* Copyright (C) 2002 Vicentini Martin
*
* All files in this archive are subject to the GNU General Public License.
* See the file COPYING in the source tree root for full license agreement.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
#include "plugin.h"
#ifdef HAVE_LCD_BITMAP
PLUGIN_HEADER
/* variable button definitions */
#if CONFIG_KEYPAD == RECORDER_PAD
#define FLIPIT_UP BUTTON_UP
#define FLIPIT_DOWN BUTTON_DOWN
#define FLIPIT_QUIT BUTTON_OFF
#define FLIPIT_SHUFFLE BUTTON_F1
#define FLIPIT_SOLVE BUTTON_F2
#define FLIPIT_STEP_BY_STEP BUTTON_F3
#define FLIPIT_TOGGLE BUTTON_PLAY
#elif CONFIG_KEYPAD == ONDIO_PAD
#define FLIPIT_UP BUTTON_UP
#define FLIPIT_DOWN BUTTON_DOWN
#define FLIPIT_QUIT BUTTON_OFF
#define FLIPIT_SHUFFLE (BUTTON_MENU | BUTTON_LEFT)
#define FLIPIT_SOLVE (BUTTON_MENU | BUTTON_UP)
#define FLIPIT_STEP_BY_STEP (BUTTON_MENU | BUTTON_RIGHT)
#define FLIPIT_TOGGLE_PRE BUTTON_MENU
#define FLIPIT_TOGGLE (BUTTON_MENU | BUTTON_REL)
#elif (CONFIG_KEYPAD == IRIVER_H100_PAD) || \
(CONFIG_KEYPAD == IRIVER_H300_PAD)
#define FLIPIT_UP BUTTON_UP
#define FLIPIT_DOWN BUTTON_DOWN
#define FLIPIT_QUIT BUTTON_OFF
#define FLIPIT_SHUFFLE BUTTON_MODE
#define FLIPIT_SOLVE BUTTON_ON
#define FLIPIT_STEP_BY_STEP BUTTON_REC
#define FLIPIT_TOGGLE_PRE BUTTON_SELECT
#define FLIPIT_TOGGLE (BUTTON_SELECT | BUTTON_REL)
#define FLIPIT_RC_QUIT BUTTON_RC_STOP
#elif (CONFIG_KEYPAD == IPOD_4G_PAD) || \
(CONFIG_KEYPAD == IPOD_3G_PAD)
#define FLIPIT_UP BUTTON_MENU
#define FLIPIT_DOWN BUTTON_PLAY
#define FLIPIT_QUIT (BUTTON_SELECT | BUTTON_MENU)
#define FLIPIT_SHUFFLE (BUTTON_SELECT | BUTTON_LEFT)
#define FLIPIT_SOLVE (BUTTON_SELECT | BUTTON_PLAY)
#define FLIPIT_STEP_BY_STEP (BUTTON_SELECT | BUTTON_RIGHT)
#define FLIPIT_TOGGLE_PRE BUTTON_SELECT
#define FLIPIT_TOGGLE (BUTTON_SELECT | BUTTON_REL)
#elif (CONFIG_KEYPAD == IAUDIO_X5_PAD)
#define FLIPIT_UP BUTTON_UP
#define FLIPIT_DOWN BUTTON_DOWN
#define FLIPIT_QUIT BUTTON_POWER
#define FLIPIT_SHUFFLE (BUTTON_PLAY | BUTTON_LEFT)
#define FLIPIT_SOLVE (BUTTON_PLAY | BUTTON_RIGHT)
#define FLIPIT_STEP_BY_STEP (BUTTON_PLAY | BUTTON_UP)
#define FLIPIT_TOGGLE_PRE BUTTON_SELECT
#define FLIPIT_TOGGLE (BUTTON_SELECT | BUTTON_REL)
#elif (CONFIG_KEYPAD == GIGABEAT_PAD)
#define FLIPIT_UP BUTTON_UP
#define FLIPIT_DOWN BUTTON_DOWN
#define FLIPIT_QUIT BUTTON_A
#define FLIPIT_SHUFFLE (BUTTON_POWER | BUTTON_LEFT)
#define FLIPIT_SOLVE (BUTTON_POWER | BUTTON_RIGHT)
#define FLIPIT_STEP_BY_STEP (BUTTON_POWER | BUTTON_UP)
#define FLIPIT_TOGGLE_PRE BUTTON_MENU
#define FLIPIT_TOGGLE (BUTTON_MENU | BUTTON_REL)
#elif (CONFIG_KEYPAD == IRIVER_H10_PAD)
#define FLIPIT_UP BUTTON_SCROLL_UP
#define FLIPIT_DOWN BUTTON_SCROLL_DOWN
#define FLIPIT_QUIT BUTTON_POWER
#define FLIPIT_SHUFFLE (BUTTON_PLAY | BUTTON_LEFT)
#define FLIPIT_SOLVE (BUTTON_PLAY | BUTTON_RIGHT)
#define FLIPIT_STEP_BY_STEP (BUTTON_PLAY | BUTTON_SCROLL_UP)
#define FLIPIT_TOGGLE_PRE BUTTON_REW
#define FLIPIT_TOGGLE (BUTTON_REW | BUTTON_REL)
#endif
static struct plugin_api* rb;
static int spots[20];
static int toggle[20];
static int cursor_pos, moves;
static char s[5];
#if LCD_DEPTH >= 2
#ifdef HAVE_LCD_COLOR
#if LCD_HEIGHT >= 200
#define tksize 50
#elif LCD_HEIGHT >=160
#define tksize 40
#else
#define tksize 30
#endif
#else
#if LCD_WIDTH >= 150
#define tksize 30
#else
#define tksize 20
#endif
#endif
extern const fb_data flipit_tokens[];
#else
static char *ptr;
#define tksize 16
static unsigned char spot_pic[2][28] = {
{ 0xe0, 0xf8, 0xfc, 0xfe, 0xfe, 0xff, 0xff,
0xff, 0xff, 0xfe, 0xfe, 0xfc, 0xf8, 0xe0,
0x01, 0x07, 0x0f, 0x1f, 0x1f, 0x3f, 0x3f,
0x3f, 0x3f, 0x1f, 0x1f, 0x0f, 0x07, 0x01 },
{ 0xe0, 0x18, 0x04, 0x02, 0x02, 0x01, 0x01,
0x01, 0x01, 0x02, 0x02, 0x04, 0x18, 0xe0,
0x01, 0x06, 0x08, 0x10, 0x10, 0x20, 0x20,
0x20, 0x20, 0x10, 0x10, 0x08, 0x06, 0x01 }
};
static unsigned char cursor_pic[32] = {
0x55, 0x00, 0x01, 0x00, 0x01, 0x00, 0x01, 0x00,
0x01, 0x00, 0x01, 0x00, 0x01, 0x00, 0x01, 0xaa,
0x55, 0x80, 0x00, 0x80, 0x00, 0x80, 0x00, 0x80,
0x00, 0x80, 0x00, 0x80, 0x00, 0x80, 0x00, 0xaa };
#endif
#define INFO_WIDTH 32
#if LCD_WIDTH - 5*tksize - INFO_WIDTH < 0
# define INFO_UNDER
# undef INFO_WIDTH
# define INFO_WIDTH 60
# define GRID_LEFT ((LCD_WIDTH-5*tksize)/2)
# define GRID_TOP 0
#else
# define GRID_LEFT 0
# define GRID_TOP ((LCD_HEIGHT-4*tksize)/2)
#endif
/* draw a spot at the coordinates (x,y), range of p is 0-19 */
static void draw_spot(int p) {
#if LCD_DEPTH >= 2
rb->lcd_bitmap_part( flipit_tokens, 0, spots[p]*tksize, tksize,
(p%5)*tksize+GRID_LEFT, (p/5)*tksize+GRID_TOP,
tksize, tksize );
#else
ptr = spot_pic[spots[p]];
rb->lcd_mono_bitmap (ptr, (p%5)*16+1, (p/5)*16+1, 14, 8);
ptr += 14;
rb->lcd_mono_bitmap (ptr, (p%5)*16+1, (p/5)*16+9, 14, 6);
#endif
}
/* draw the cursor at the current cursor position */
static void draw_cursor(void) {
#ifdef HAVE_LCD_COLOR
rb->lcd_bitmap_transparent_part( flipit_tokens, 0, 2*tksize, tksize,
(cursor_pos%5)*tksize+GRID_LEFT,
(cursor_pos/5)*tksize+GRID_TOP,
tksize, tksize );
#elif LCD_DEPTH >= 2
/* grayscale doesn't have transparent bitmap ... */
int i,j;
i = ( cursor_pos%5 )*tksize;
j = ( cursor_pos/5 )*tksize;
rb->lcd_set_drawmode( DRMODE_SOLID );
rb->lcd_drawline( i+GRID_LEFT, j+GRID_TOP,
i+tksize-1+GRID_LEFT, j+GRID_TOP );
rb->lcd_drawline( i+GRID_LEFT, j+tksize-1+GRID_TOP,
i+tksize-1+GRID_LEFT, j+tksize-1+GRID_TOP );
rb->lcd_drawline( i+GRID_LEFT, j+GRID_TOP,
i+GRID_LEFT, j+tksize-1+GRID_TOP );
rb->lcd_drawline( i+tksize-1+GRID_LEFT, j+GRID_TOP,
i+tksize-1+GRID_LEFT, j+tksize-1+GRID_TOP );
#else
int i,j;
i = (cursor_pos%5)*16;
j = (cursor_pos/5)*16;
rb->lcd_set_drawmode(DRMODE_FG);
ptr = cursor_pic;
rb->lcd_mono_bitmap (ptr, i, j, 16, 8);
ptr += 16;
rb->lcd_mono_bitmap (ptr, i, j+8, 16, 8);
rb->lcd_set_drawmode(DRMODE_SOLID);
#endif
}
/* clear the cursor where it is */
static void clear_cursor(void) {
#if LCD_DEPTH >= 2
draw_spot( cursor_pos );
#else
int i,j;
i = (cursor_pos%5)*16;
j = (cursor_pos/5)*16;
rb->lcd_set_drawmode(DRMODE_SOLID|DRMODE_INVERSEVID);
rb->lcd_drawline(i, j, i+15, j);
rb->lcd_drawline(i, j+15, i+15, j+15);
rb->lcd_drawline(i, j, i, j+15);
rb->lcd_drawline(i+15, j, i+15, j+15);
rb->lcd_set_drawmode(DRMODE_SOLID);
#endif
}
/* draw the info panel ... duh */
static void draw_info_panel(void)
{
#ifdef INFO_UNDER
# define HEIGHT ( LCD_HEIGHT - 4*tksize )
# define LEFT ( ( LCD_WIDTH - INFO_WIDTH ) / 2 )
# define TOP ( LCD_HEIGHT - HEIGHT )
#else
# define HEIGHT 64
# define LEFT ( LCD_WIDTH - (LCD_WIDTH- 5*tksize-INFO_WIDTH)/2 - INFO_WIDTH )
# define TOP ( ( LCD_HEIGHT - HEIGHT ) / 2 )
#endif
rb->lcd_set_drawmode( DRMODE_SOLID|DRMODE_INVERSEVID );
rb->lcd_fillrect( LEFT, TOP, INFO_WIDTH, HEIGHT );
rb->lcd_set_drawmode( DRMODE_SOLID );
rb->lcd_drawrect( LEFT, TOP, INFO_WIDTH, HEIGHT );
#ifdef INFO_UNDER
rb->lcd_putsxy( LEFT+1, TOP+1, "Flips" );
#else
rb->lcd_putsxy( LEFT+1, TOP+10, "Flips" );
#endif
rb->snprintf( s, sizeof(s), "%d", moves );
#ifdef INFO_UNDER
rb->lcd_putsxy( LEFT+32, TOP+1, s );
#else
rb->lcd_putsxy( LEFT+1, TOP+20, s );
#endif
# undef HEIGHT
# undef LEFT
# undef TOP
}
/* check if the puzzle is finished */
static bool flipit_finished(void) {
int i;
for (i=0; i<20; i++)
if (!spots[i])
return false;
clear_cursor();
return true;
}
/* draws the toggled spots */
static void flipit_toggle(void) {
spots[cursor_pos] = 1-spots[cursor_pos];
toggle[cursor_pos] = 1-toggle[cursor_pos];
draw_spot(cursor_pos);
if (cursor_pos%5 > 0) {
spots[cursor_pos-1] = 1-spots[cursor_pos-1];
draw_spot(cursor_pos-1);
}
if (cursor_pos%5 < 4) {
spots[cursor_pos+1] = 1-spots[cursor_pos+1];
draw_spot(cursor_pos+1);
}
if (cursor_pos/5 > 0) {
spots[cursor_pos-5] = 1-spots[cursor_pos-5];
draw_spot(cursor_pos-5);
}
if (cursor_pos/5 < 3) {
spots[cursor_pos+5] = 1-spots[cursor_pos+5];
draw_spot(cursor_pos+5);
}
moves++;
draw_info_panel();
if (flipit_finished())
clear_cursor();
}
/* move the cursor in any direction */
static void move_cursor(int x, int y) {
if (!(flipit_finished())) {
clear_cursor();
cursor_pos = ( x + 5 + cursor_pos%5 )%5
+ ( ( y + 4 + cursor_pos/5 )%4 )*5;
draw_cursor();
}
rb->lcd_update();
}
/* initialize the board */
static void flipit_init(void) {
int i;
rb->lcd_clear_display();
moves = 0;
for (i=0; i<20; i++) {
spots[i]=1;
toggle[i]=1;
draw_spot(i);
}
rb->lcd_update();
for (i=0; i<20; i++) {
cursor_pos = (rb->rand() % 20);
flipit_toggle();
}
cursor_pos = 0;
draw_cursor();
moves = 0;
draw_info_panel();
rb->lcd_update();
}
/* the main game loop */
static bool flipit_loop(void) {
int i;
int button;
int lastbutton = BUTTON_NONE;
flipit_init();
while(true) {
button = rb->button_get(true);
switch (button) {
#ifdef FLIPIT_RC_QUIT
case FLIPIT_RC_QUIT:
#endif
case FLIPIT_QUIT:
/* get out of here */
return PLUGIN_OK;
case FLIPIT_SHUFFLE:
/* mix up the pieces */
flipit_init();
break;
case FLIPIT_SOLVE:
/* solve the puzzle */
if (!flipit_finished()) {
for (i=0; i<20; i++)
if (!toggle[i]) {
clear_cursor();
cursor_pos = i;
flipit_toggle();
draw_cursor();
rb->lcd_update();
rb->sleep(HZ*2/3);
}
}
break;
case FLIPIT_STEP_BY_STEP:
if (!flipit_finished()) {
for (i=0; i<20; i++)
if (!toggle[i]) {
clear_cursor();
cursor_pos = i;
flipit_toggle();
draw_cursor();
rb->lcd_update();
break;
}
}
break;
case FLIPIT_TOGGLE:
#ifdef FLIPIT_TOGGLE_PRE
if (lastbutton != FLIPIT_TOGGLE_PRE)
break;
#endif
/* toggle the pieces */
if (!flipit_finished()) {
flipit_toggle();
#if LCD_DEPTH >= 2
draw_cursor();
#endif
rb->lcd_update();
}
break;
case BUTTON_LEFT:
move_cursor(-1, 0);
break;
case BUTTON_RIGHT:
move_cursor(1, 0);
break;
case FLIPIT_UP:
move_cursor(0, -1);
break;
case FLIPIT_DOWN:
move_cursor(0, 1);
break;
default:
if (rb->default_event_handler(button) == SYS_USB_CONNECTED)
return PLUGIN_USB_CONNECTED;
break;
}
if (button != BUTTON_NONE)
lastbutton = button;
}
}
/* called function from outside */
enum plugin_status plugin_start(struct plugin_api* api, void* parameter)
{
int w, h, i;
(void)parameter;
rb = api;
#ifdef HAVE_LCD_COLOR
rb->lcd_set_background(LCD_WHITE);
#endif
/* print title */
rb->lcd_getstringsize("FlipIt!", &w, &h);
w = (w+1)/2;
h = (h+1)/2;
rb->lcd_clear_display();
rb->lcd_putsxy(LCD_WIDTH/2-w, (LCD_HEIGHT/2)-h, "FlipIt!");
rb->lcd_update();
rb->sleep(HZ);
/* print instructions */
rb->lcd_clear_display();
rb->lcd_setfont(FONT_SYSFIXED);
#if CONFIG_KEYPAD == RECORDER_PAD
rb->lcd_putsxy(2, 8, "[OFF] to stop");
rb->lcd_putsxy(2, 18, "[PLAY] toggle");
rb->lcd_putsxy(2, 28, "[F1] shuffle");
rb->lcd_putsxy(2, 38, "[F2] solution");
rb->lcd_putsxy(2, 48, "[F3] step by step");
#elif CONFIG_KEYPAD == ONDIO_PAD
rb->lcd_putsxy(2, 8, "[OFF] to stop");
rb->lcd_putsxy(2, 18, "[MODE] toggle");
rb->lcd_putsxy(2, 28, "[M-LEFT] shuffle");
rb->lcd_putsxy(2, 38, "[M-UP] solution");
rb->lcd_putsxy(2, 48, "[M-RIGHT] step by step");
#elif CONFIG_KEYPAD == IRIVER_H100_PAD
rb->lcd_putsxy(2, 8, "[STOP] to stop");
rb->lcd_putsxy(2, 18, "[SELECT] toggle");
rb->lcd_putsxy(2, 28, "[MODE] shuffle");
rb->lcd_putsxy(2, 38, "[PLAY] solution");
rb->lcd_putsxy(2, 48, "[REC] step by step");
#elif (CONFIG_KEYPAD == IPOD_3G_PAD) || \
(CONFIG_KEYPAD == IPOD_4G_PAD)
rb->lcd_putsxy(2, 8, "[S-MENU] to stop");
rb->lcd_putsxy(2, 18, "[SELECT] toggle");
rb->lcd_putsxy(2, 28, "[S-LEFT] shuffle");
rb->lcd_putsxy(2, 38, "[S-PLAY] solution");
rb->lcd_putsxy(2, 48, "[S-RIGHT] step by step");
#endif
rb->lcd_update();
rb->button_get_w_tmo(HZ*3);
rb->lcd_clear_display();
draw_info_panel();
for (i=0; i<20; i++) {
spots[i]=1;
draw_spot(i);
}
rb->lcd_update();
rb->sleep(HZ*3/2);
rb->srand(*rb->current_tick);
return flipit_loop();
}
#endif