1 /***************************************************************************
3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___
4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
10 * Copyright (C) 2005 Eli Sherer
12 * Heavily modified for embedded use by Björn Stenberg (bjorn@haxx.se)
14 * This program is free software; you can redistribute it and/or
15 * modify it under the terms of the GNU General Public License
16 * as published by the Free Software Foundation; either version 2
17 * of the License, or (at your option) any later version.
19 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
20 * KIND, either express or implied.
22 ****************************************************************************/
24 #include "lib/display_text.h"
25 #include "lib/helper.h"
26 #include "lib/highscore.h"
27 #include "lib/playback_control.h"
28 #include "lib/playergfx.h"
29 #include "lib/mylcd.h"
33 #if (CONFIG_KEYPAD == IPOD_4G_PAD) || \
34 (CONFIG_KEYPAD == IPOD_3G_PAD) || \
35 (CONFIG_KEYPAD == IPOD_1G2G_PAD)
37 #define ROCKBLOX_OFF (BUTTON_MENU | BUTTON_SELECT)
38 #define ROCKBLOX_ROTATE_CCW BUTTON_SCROLL_BACK
39 #define ROCKBLOX_ROTATE_CCW2 (BUTTON_MENU | BUTTON_REL)
40 #define ROCKBLOX_ROTATE_CW BUTTON_SCROLL_FWD
41 #define ROCKBLOX_LEFT BUTTON_LEFT
42 #define ROCKBLOX_RIGHT BUTTON_RIGHT
43 #define ROCKBLOX_DOWN BUTTON_PLAY
44 #define ROCKBLOX_RESTART (BUTTON_SELECT | BUTTON_PLAY)
45 #define ROCKBLOX_DROP (BUTTON_SELECT | BUTTON_REL)
47 #elif (CONFIG_KEYPAD == IRIVER_H100_PAD) || \
48 (CONFIG_KEYPAD == IRIVER_H300_PAD)
50 #define ROCKBLOX_OFF BUTTON_OFF
51 #define ROCKBLOX_ROTATE_CCW BUTTON_UP
52 #define ROCKBLOX_ROTATE_CW BUTTON_SELECT
53 #define ROCKBLOX_DOWN BUTTON_DOWN
54 #define ROCKBLOX_LEFT BUTTON_LEFT
55 #define ROCKBLOX_RIGHT BUTTON_RIGHT
56 #define ROCKBLOX_DROP BUTTON_MODE
57 #define ROCKBLOX_RESTART BUTTON_ON
59 #define ROCKBLOX_RC_OFF BUTTON_RC_STOP
61 #elif CONFIG_KEYPAD == RECORDER_PAD
63 #define ROCKBLOX_OFF BUTTON_OFF
64 #define ROCKBLOX_ROTATE_CCW BUTTON_UP
65 #define ROCKBLOX_ROTATE_CW BUTTON_PLAY
66 #define ROCKBLOX_DOWN BUTTON_DOWN
67 #define ROCKBLOX_LEFT BUTTON_LEFT
68 #define ROCKBLOX_RIGHT BUTTON_RIGHT
69 #define ROCKBLOX_DROP BUTTON_ON
70 #define ROCKBLOX_RESTART BUTTON_F1
72 #elif CONFIG_KEYPAD == ARCHOS_AV300_PAD
74 #define ROCKBLOX_OFF BUTTON_OFF
75 #define ROCKBLOX_ROTATE_CCW BUTTON_UP
76 #define ROCKBLOX_ROTATE_CW BUTTON_SELECT
77 #define ROCKBLOX_DOWN BUTTON_DOWN
78 #define ROCKBLOX_LEFT BUTTON_LEFT
79 #define ROCKBLOX_RIGHT BUTTON_RIGHT
80 #define ROCKBLOX_DROP BUTTON_ON
81 #define ROCKBLOX_RESTART BUTTON_F1
83 #elif CONFIG_KEYPAD == PLAYER_PAD
85 #define ROCKBLOX_OFF_PRE BUTTON_STOP
86 #define ROCKBLOX_OFF (BUTTON_STOP|BUTTON_REL)
87 #define ROCKBLOX_ROTATE_CCW BUTTON_PLAY
88 #define ROCKBLOX_ROTATE_CW (BUTTON_ON|BUTTON_PLAY)
89 #define ROCKBLOX_DOWN BUTTON_MENU
90 #define ROCKBLOX_LEFT BUTTON_LEFT
91 #define ROCKBLOX_RIGHT BUTTON_RIGHT
92 #define ROCKBLOX_DROP_PRE BUTTON_ON
93 #define ROCKBLOX_DROP (BUTTON_ON|BUTTON_REL)
94 #define ROCKBLOX_RESTART (BUTTON_STOP|BUTTON_MENU)
96 #elif CONFIG_KEYPAD == ONDIO_PAD
98 #define ROCKBLOX_OFF_PRE BUTTON_OFF
99 #define ROCKBLOX_OFF (BUTTON_OFF|BUTTON_REL)
100 #define ROCKBLOX_ROTATE_CCW BUTTON_UP
101 #define ROCKBLOX_ROTATE_CW (BUTTON_MENU|BUTTON_UP)
102 #define ROCKBLOX_DOWN BUTTON_DOWN
103 #define ROCKBLOX_LEFT BUTTON_LEFT
104 #define ROCKBLOX_RIGHT BUTTON_RIGHT
105 #define ROCKBLOX_DROP_PRE BUTTON_MENU
106 #define ROCKBLOX_DROP (BUTTON_MENU|BUTTON_REL)
107 #define ROCKBLOX_RESTART (BUTTON_OFF|BUTTON_MENU)
109 #elif CONFIG_KEYPAD == IAUDIO_X5M5_PAD
111 #define ROCKBLOX_OFF BUTTON_POWER
112 #define ROCKBLOX_ROTATE_CCW BUTTON_UP
113 #define ROCKBLOX_ROTATE_CW BUTTON_SELECT
114 #define ROCKBLOX_DOWN BUTTON_DOWN
115 #define ROCKBLOX_LEFT BUTTON_LEFT
116 #define ROCKBLOX_RIGHT BUTTON_RIGHT
117 #define ROCKBLOX_DROP BUTTON_REC
118 #define ROCKBLOX_RESTART BUTTON_PLAY
120 #elif CONFIG_KEYPAD == SANSA_E200_PAD
122 #define ROCKBLOX_OFF BUTTON_POWER
123 #define ROCKBLOX_ROTATE_CCW BUTTON_SCROLL_BACK
124 #define ROCKBLOX_ROTATE_CCW2 BUTTON_UP
125 #define ROCKBLOX_ROTATE_CW BUTTON_SCROLL_FWD
126 #define ROCKBLOX_DOWN BUTTON_DOWN
127 #define ROCKBLOX_LEFT BUTTON_LEFT
128 #define ROCKBLOX_RIGHT BUTTON_RIGHT
129 #define ROCKBLOX_DROP BUTTON_SELECT
130 #define ROCKBLOX_RESTART BUTTON_REC
132 #elif CONFIG_KEYPAD == SANSA_FUZE_PAD
134 #define ROCKBLOX_OFF (BUTTON_HOME|BUTTON_REPEAT)
135 #define ROCKBLOX_ROTATE_CCW BUTTON_SCROLL_BACK
136 #define ROCKBLOX_ROTATE_CCW2 BUTTON_UP
137 #define ROCKBLOX_ROTATE_CW BUTTON_SCROLL_FWD
138 #define ROCKBLOX_DOWN BUTTON_DOWN
139 #define ROCKBLOX_LEFT BUTTON_LEFT
140 #define ROCKBLOX_RIGHT BUTTON_RIGHT
141 #define ROCKBLOX_DROP (BUTTON_SELECT | BUTTON_REL)
142 #define ROCKBLOX_RESTART (BUTTON_SELECT | BUTTON_UP)
145 #elif CONFIG_KEYPAD == SANSA_C200_PAD
147 #define ROCKBLOX_OFF BUTTON_POWER
148 #define ROCKBLOX_ROTATE_CCW BUTTON_UP
149 #define ROCKBLOX_ROTATE_CCW2 BUTTON_VOL_DOWN
150 #define ROCKBLOX_ROTATE_CW BUTTON_VOL_UP
151 #define ROCKBLOX_DOWN BUTTON_DOWN
152 #define ROCKBLOX_LEFT BUTTON_LEFT
153 #define ROCKBLOX_RIGHT BUTTON_RIGHT
154 #define ROCKBLOX_DROP BUTTON_SELECT
155 #define ROCKBLOX_RESTART BUTTON_REC
157 #elif CONFIG_KEYPAD == SANSA_CLIP_PAD
159 #define ROCKBLOX_OFF BUTTON_POWER
160 #define ROCKBLOX_ROTATE_CCW BUTTON_UP
161 #define ROCKBLOX_ROTATE_CCW2 BUTTON_VOL_DOWN
162 #define ROCKBLOX_ROTATE_CW BUTTON_VOL_UP
163 #define ROCKBLOX_DOWN BUTTON_DOWN
164 #define ROCKBLOX_LEFT BUTTON_LEFT
165 #define ROCKBLOX_RIGHT BUTTON_RIGHT
166 #define ROCKBLOX_DROP BUTTON_SELECT
167 #define ROCKBLOX_RESTART BUTTON_HOME
169 #elif CONFIG_KEYPAD == SANSA_M200_PAD
171 #define ROCKBLOX_OFF BUTTON_POWER
172 #define ROCKBLOX_ROTATE_CCW BUTTON_UP
173 #define ROCKBLOX_ROTATE_CCW2 BUTTON_VOL_DOWN
174 #define ROCKBLOX_ROTATE_CW BUTTON_VOL_UP
175 #define ROCKBLOX_DOWN BUTTON_DOWN
176 #define ROCKBLOX_LEFT BUTTON_LEFT
177 #define ROCKBLOX_RIGHT BUTTON_RIGHT
178 #define ROCKBLOX_RESTART (BUTTON_SELECT | BUTTON_UP)
179 #define ROCKBLOX_DROP (BUTTON_SELECT | BUTTON_REL)
181 #elif CONFIG_KEYPAD == IRIVER_H10_PAD
183 #define ROCKBLOX_OFF BUTTON_POWER
184 #define ROCKBLOX_ROTATE_CCW BUTTON_SCROLL_UP
185 #define ROCKBLOX_ROTATE_CW BUTTON_REW
186 #define ROCKBLOX_DOWN BUTTON_SCROLL_DOWN
187 #define ROCKBLOX_LEFT BUTTON_LEFT
188 #define ROCKBLOX_RIGHT BUTTON_RIGHT
189 #define ROCKBLOX_DROP BUTTON_FF
190 #define ROCKBLOX_RESTART BUTTON_PLAY
192 #elif CONFIG_KEYPAD == GIGABEAT_PAD
194 #define ROCKBLOX_OFF BUTTON_POWER
195 #define ROCKBLOX_ROTATE_CCW BUTTON_VOL_DOWN
196 #define ROCKBLOX_ROTATE_CW BUTTON_VOL_UP
197 #define ROCKBLOX_ROTATE BUTTON_UP
198 #define ROCKBLOX_DOWN BUTTON_DOWN
199 #define ROCKBLOX_LEFT BUTTON_LEFT
200 #define ROCKBLOX_RIGHT BUTTON_RIGHT
201 #define ROCKBLOX_DROP BUTTON_SELECT
202 #define ROCKBLOX_RESTART BUTTON_A
204 #elif CONFIG_KEYPAD == IRIVER_IFP7XX_PAD
206 #define ROCKBLOX_OFF BUTTON_PLAY
207 #define ROCKBLOX_ROTATE_CCW BUTTON_UP
208 #define ROCKBLOX_ROTATE_CW BUTTON_SELECT
209 #define ROCKBLOX_DOWN BUTTON_DOWN
210 #define ROCKBLOX_LEFT BUTTON_LEFT
211 #define ROCKBLOX_RIGHT BUTTON_RIGHT
212 #define ROCKBLOX_DROP BUTTON_MODE
213 #define ROCKBLOX_RESTART BUTTON_EQ
215 #elif CONFIG_KEYPAD == MROBE500_PAD
216 #define ROCKBLOX_OFF BUTTON_POWER
218 #elif CONFIG_KEYPAD == GIGABEAT_S_PAD
219 #define ROCKBLOX_OFF BUTTON_BACK
220 #define ROCKBLOX_ROTATE_CCW BUTTON_VOL_DOWN
221 #define ROCKBLOX_ROTATE_CW BUTTON_VOL_UP
222 #define ROCKBLOX_ROTATE BUTTON_UP
223 #define ROCKBLOX_DOWN BUTTON_DOWN
224 #define ROCKBLOX_LEFT BUTTON_LEFT
225 #define ROCKBLOX_RIGHT BUTTON_RIGHT
226 #define ROCKBLOX_DROP BUTTON_SELECT
227 #define ROCKBLOX_RESTART BUTTON_PLAY
229 #elif CONFIG_KEYPAD == MROBE100_PAD
231 #define ROCKBLOX_OFF BUTTON_POWER
232 #define ROCKBLOX_ROTATE_CCW BUTTON_MENU
233 #define ROCKBLOX_ROTATE_CW BUTTON_PLAY
234 #define ROCKBLOX_ROTATE BUTTON_UP
235 #define ROCKBLOX_DOWN BUTTON_DOWN
236 #define ROCKBLOX_LEFT BUTTON_LEFT
237 #define ROCKBLOX_RIGHT BUTTON_RIGHT
238 #define ROCKBLOX_DROP BUTTON_SELECT
239 #define ROCKBLOX_RESTART BUTTON_DISPLAY
241 #elif CONFIG_KEYPAD == IAUDIO_M3_PAD
243 #define ROCKBLOX_OFF BUTTON_RC_REC
244 #define ROCKBLOX_ROTATE_CCW BUTTON_RC_VOL_DOWN
245 #define ROCKBLOX_ROTATE_CW BUTTON_RC_VOL_UP
246 #define ROCKBLOX_DOWN BUTTON_RC_MENU
247 #define ROCKBLOX_LEFT BUTTON_RC_REW
248 #define ROCKBLOX_RIGHT BUTTON_RC_FF
249 #define ROCKBLOX_DROP BUTTON_RC_PLAY
250 #define ROCKBLOX_RESTART BUTTON_RC_MODE
252 #elif CONFIG_KEYPAD == COWON_D2_PAD
254 #define ROCKBLOX_OFF BUTTON_POWER
255 #define ROCKBLOX_RESTART BUTTON_MENU
257 #elif CONFIG_KEYPAD == IAUDIO67_PAD
259 #define ROCKBLOX_OFF BUTTON_POWER
260 #define ROCKBLOX_ROTATE_CCW BUTTON_VOLDOWN
261 #define ROCKBLOX_ROTATE_CW BUTTON_VOLUP
262 #define ROCKBLOX_DOWN BUTTON_STOP
263 #define ROCKBLOX_LEFT BUTTON_LEFT
264 #define ROCKBLOX_RIGHT BUTTON_RIGHT
265 #define ROCKBLOX_DROP BUTTON_PLAY
266 #define ROCKBLOX_RESTART BUTTON_MENU
268 #elif CONFIG_KEYPAD == CREATIVEZVM_PAD
269 #define ROCKBLOX_OFF BUTTON_BACK
270 #define ROCKBLOX_ROTATE_CCW BUTTON_UP
271 #define ROCKBLOX_ROTATE_CW BUTTON_PLAY
272 #define ROCKBLOX_DOWN BUTTON_DOWN
273 #define ROCKBLOX_LEFT BUTTON_LEFT
274 #define ROCKBLOX_RIGHT BUTTON_RIGHT
275 #define ROCKBLOX_DROP BUTTON_SELECT
276 #define ROCKBLOX_RESTART BUTTON_CUSTOM
278 #elif CONFIG_KEYPAD == PHILIPS_HDD1630_PAD
280 #define ROCKBLOX_OFF BUTTON_POWER
281 #define ROCKBLOX_ROTATE_CCW BUTTON_VOL_DOWN
282 #define ROCKBLOX_ROTATE_CW BUTTON_VOL_UP
283 #define ROCKBLOX_ROTATE BUTTON_UP
284 #define ROCKBLOX_DOWN BUTTON_DOWN
285 #define ROCKBLOX_LEFT BUTTON_LEFT
286 #define ROCKBLOX_RIGHT BUTTON_RIGHT
287 #define ROCKBLOX_DROP BUTTON_SELECT
288 #define ROCKBLOX_RESTART BUTTON_MENU
290 # elif CONFIG_KEYPAD == PHILIPS_SA9200_PAD
292 #define ROCKBLOX_OFF BUTTON_POWER
293 #define ROCKBLOX_ROTATE_CCW BUTTON_VOL_DOWN
294 #define ROCKBLOX_ROTATE_CW BUTTON_VOL_UP
295 #define ROCKBLOX_ROTATE BUTTON_UP
296 #define ROCKBLOX_DOWN BUTTON_DOWN
297 #define ROCKBLOX_LEFT BUTTON_PREV
298 #define ROCKBLOX_RIGHT BUTTON_NEXT
299 #define ROCKBLOX_DROP BUTTON_PLAY
300 #define ROCKBLOX_RESTART BUTTON_MENU
302 #elif CONFIG_KEYPAD == ONDAVX747_PAD
303 #define ROCKBLOX_OFF BUTTON_POWER
304 #define ROCKBLOX_RESTART BUTTON_MENU
305 #elif CONFIG_KEYPAD == ONDAVX777_PAD
306 #define ROCKBLOX_OFF BUTTON_POWER
308 #elif CONFIG_KEYPAD == SAMSUNG_YH_PAD
310 #define ROCKBLOX_OFF (BUTTON_REC|BUTTON_PLAY)
311 #define ROCKBLOX_ROTATE_CCW BUTTON_UP
312 #define ROCKBLOX_ROTATE_CW BUTTON_DOWN
313 #define ROCKBLOX_DOWN BUTTON_REW
314 #define ROCKBLOX_LEFT BUTTON_LEFT
315 #define ROCKBLOX_RIGHT BUTTON_RIGHT
316 #define ROCKBLOX_DROP BUTTON_FFWD
317 #define ROCKBLOX_RESTART (BUTTON_REC|BUTTON_REW)
319 #elif CONFIG_KEYPAD == PBELL_VIBE500_PAD
321 #define ROCKBLOX_OFF BUTTON_REC
322 #define ROCKBLOX_ROTATE_CCW BUTTON_PLAY
323 #define ROCKBLOX_ROTATE_CW BUTTON_MENU
324 #define ROCKBLOX_DOWN BUTTON_DOWN
325 #define ROCKBLOX_LEFT BUTTON_PREV
326 #define ROCKBLOX_RIGHT BUTTON_NEXT
327 #define ROCKBLOX_DROP BUTTON_OK
328 #define ROCKBLOX_RESTART BUTTON_CANCEL
330 #elif CONFIG_KEYPAD == MPIO_HD200_PAD
331 #define ROCKBLOX_OFF (BUTTON_REC|BUTTON_PLAY)
332 #define ROCKBLOX_ROTATE_CCW BUTTON_PREV
333 #define ROCKBLOX_ROTATE_CW BUTTON_NEXT
334 #define ROCKBLOX_DOWN BUTTON_SELECT
335 #define ROCKBLOX_LEFT BUTTON_VOL_DOWN
336 #define ROCKBLOX_RIGHT BUTTON_VOL_UP
337 #define ROCKBLOX_DROP BUTTON_PLAY
338 #define ROCKBLOX_RESTART BUTTON_REC
341 #error No keymap defined!
344 #ifdef HAVE_TOUCHSCREEN
346 #define ROCKBLOX_OFF BUTTON_TOPLEFT
348 #ifdef ROCKBLOX_ROTATE_CCW
349 #define ROCKBLOX_ROTATE_CCW2 BUTTON_BOTTOMLEFT
351 #define ROCKBLOX_ROTATE_CCW BUTTON_BOTTOMLEFT
353 #ifdef ROCKBLOX_ROTATE_CW
354 #define ROCKBLOX_ROTATE_CW2 BUTTON_BOTTOMRIGHT
356 #define ROCKBLOX_ROTATE_CW BUTTON_BOTTOMRIGHT
357 #define ROCKBLOX_ROTATE_CW2 BUTTON_TOPMIDDLE
359 #ifndef ROCKBLOX_DOWN
360 #define ROCKBLOX_DOWN BUTTON_BOTTOMMIDDLE
362 #ifndef ROCKBLOX_LEFT
363 #define ROCKBLOX_LEFT BUTTON_MIDLEFT
365 #ifndef ROCKBLOX_RIGHT
366 #define ROCKBLOX_RIGHT BUTTON_MIDRIGHT
368 #ifndef ROCKBLOX_DROP
369 #define ROCKBLOX_DROP BUTTON_CENTER
371 #ifndef ROCKBLOX_RESTART
372 #define ROCKBLOX_RESTART BUTTON_TOPRIGHT
377 #define EMPTY_BLOCK 7
379 #define BOARD_WIDTH 10
381 #ifdef HAVE_LCD_BITMAP
383 #define BOARD_HEIGHT 20
385 #if (LCD_WIDTH == 640) && (LCD_HEIGHT == 480)
387 #define BLOCK_WIDTH 24
388 #define BLOCK_HEIGHT 24
398 #elif (LCD_WIDTH == 480) && (LCD_HEIGHT == 640)
400 #define BLOCK_WIDTH 30
401 #define BLOCK_HEIGHT 30
404 #define PREVIEW_X 342
405 #define PREVIEW_Y 482
411 #elif (LCD_WIDTH == 320) && (LCD_HEIGHT == 240)
413 #define BLOCK_WIDTH 12
414 #define BLOCK_HEIGHT 12
424 #elif (LCD_WIDTH == 240) && ((LCD_HEIGHT == 320) || (LCD_HEIGHT == 400))
426 #define BLOCK_WIDTH 15
427 #define BLOCK_HEIGHT 15
430 #define PREVIEW_X 171
431 #define PREVIEW_Y 241
436 #define HIGH_LABEL_X 172
437 #define HIGH_SCORE_Y 163
438 #define HIGH_LEVEL_Y 172
440 #elif (LCD_WIDTH == 220) && (LCD_HEIGHT == 176)
442 #define BLOCK_WIDTH 8
443 #define BLOCK_HEIGHT 8
446 #define PREVIEW_X 158
447 #define PREVIEW_Y 130
453 #elif (LCD_WIDTH == 176) && (LCD_HEIGHT == 132)
455 #define BLOCK_WIDTH 6
456 #define BLOCK_HEIGHT 6
459 #define PREVIEW_X 126
460 #define PREVIEW_Y 102
466 #elif (LCD_WIDTH == 176) && (LCD_HEIGHT == 220)
468 /* no room for the space in the highscore list */
471 #define BLOCK_WIDTH 10
472 #define BLOCK_HEIGHT 10
475 #define PREVIEW_X 124
476 #define PREVIEW_Y 174
481 #define HIGH_SCORE_Y 119
482 #define HIGH_LEVEL_Y 126
483 #define HIGH_LABEL_X 114
485 #elif (LCD_WIDTH == 160) && (LCD_HEIGHT == 128)
488 #define BLOCK_WIDTH 6
489 #define BLOCK_HEIGHT 6
492 #define PREVIEW_X 114
493 #define PREVIEW_Y 100
499 #elif (LCD_WIDTH == 138) && (LCD_HEIGHT == 110)
501 #define BLOCK_WIDTH 5
502 #define BLOCK_HEIGHT 5
512 #elif (LCD_WIDTH == 132) && (LCD_HEIGHT == 80)
514 #define BLOCK_WIDTH 4
515 #define BLOCK_HEIGHT 4
525 #elif (LCD_WIDTH == 128) && (LCD_HEIGHT == 128)
527 #define BLOCK_WIDTH 6
528 #define BLOCK_HEIGHT 6
532 #define PREVIEW_Y 100
538 /* NOTE: This is for the GoGear SA9200 and is only
539 temporary until I can get better coordinates! */
540 #elif (LCD_WIDTH == 128) && (LCD_HEIGHT == 160)
542 #define BLOCK_WIDTH 6
543 #define BLOCK_HEIGHT 6
547 #define PREVIEW_Y 100
553 #elif (LCD_WIDTH == 128) && (LCD_HEIGHT == 96)
555 #define BLOCK_WIDTH 4
556 #define BLOCK_HEIGHT 4
566 #elif (LCD_WIDTH == 128) && (LCD_HEIGHT == 64)
568 #define BLOCK_WIDTH 3
569 #define BLOCK_HEIGHT 3
579 #elif (LCD_WIDTH == 112) && (LCD_HEIGHT == 64)
581 #define BLOCK_WIDTH 4
582 #define BLOCK_HEIGHT 3
596 #define LEVEL_X LABEL_X
600 #define LINES_X LABEL_X
603 extern const fb_data rockblox_background
[];
605 #else /* HAVE_LCD_CHARCELLS */
607 #define BOARD_HEIGHT 14
609 #define BLOCK_WIDTH 1
610 #define BLOCK_HEIGHT 1
621 /* <<Explanation on Rockblox shapes>>
624 %% - O has 1 orientation
627 %% %% - Z has 2 orientations
631 %% %% - S has 2 orientations
635 % %%%% - I has 2 orientations
639 % % % %%% - L has 4 orientations
643 % % % %%% - J has 4 orientations
647 %% % %% % - T has 4 orientations
651 /* c=current f=figure o=orientation n=next */
652 static struct _rockblox_status
663 short board
[BOARD_HEIGHT
][BOARD_WIDTH
]; /* 20 rows of 10 blocks */
667 static void draw_next_block(void);
668 static void new_block(void);
670 #ifdef HAVE_SCROLLWHEEL
671 int wheel_events
= 0, last_wheel_event
= 0;
672 bool wheel_enabled
= false;
675 static const short scoring
[4] = { /* scoring for each number of lines */
676 #if BOARD_HEIGHT == 20
677 40 /* single */ , 100 /* double */ , 300 /* triple */ , 1200 /* rockblox */
678 #elif BOARD_HEIGHT == 14 /* Player special values */
679 60 /* single */ , 150 /* double */ , 500 /* triple */ , 2000 /* rockblox */
686 unsigned short color
[3]; /* color of figure (light,middle,shadow) */
688 unsigned short max_or
; /* max orientations */
689 signed short shapeX
[4], shapeY
[4]; /* implementation of figures */
692 /* array of figures */
693 figures
[BLOCKS_NUM
] = {
697 {LCD_RGBPACK (153, 255, 255), LCD_RGBPACK(0, 255, 255),
698 LCD_RGBPACK(0,153,153)},
700 {LCD_WHITE
, LCD_LIGHTGRAY
, LCD_DARKGRAY
},
709 {LCD_RGBPACK (255, 153, 128), LCD_RGBPACK (255, 0, 0),
710 LCD_RGBPACK (153, 0, 0)},
712 {LCD_WHITE
, LCD_LIGHTGRAY
, LCD_DARKGRAY
},
721 {LCD_RGBPACK (153, 255, 153), LCD_RGBPACK (0, 255, 0),
722 LCD_RGBPACK (0, 153, 0)},
724 {LCD_WHITE
, LCD_LIGHTGRAY
, LCD_DARKGRAY
},
733 {LCD_RGBPACK (153, 153, 255), LCD_RGBPACK (0, 0, 255),
734 LCD_RGBPACK (0, 0, 153)},
736 {LCD_WHITE
, LCD_LIGHTGRAY
, LCD_DARKGRAY
},
745 {LCD_RGBPACK (255, 255, 153), LCD_RGBPACK (255, 255, 0),
746 LCD_RGBPACK (153, 153, 0)},
748 {LCD_WHITE
, LCD_LIGHTGRAY
, LCD_DARKGRAY
},
757 {LCD_RGBPACK (255, 153, 255), LCD_RGBPACK (255, 0, 255),
758 LCD_RGBPACK (153, 0, 153)},
760 {LCD_WHITE
, LCD_LIGHTGRAY
, LCD_DARKGRAY
},
769 {LCD_RGBPACK (204, 204, 204), LCD_RGBPACK (153, 153, 153),
770 LCD_RGBPACK (85, 85, 85)},
772 {LCD_WHITE
, LCD_LIGHTGRAY
, LCD_DARKGRAY
},
780 bool resume_file
= false;
782 /* Rockbox File System only supports full filenames inc dir */
783 #define SCORE_FILE PLUGIN_GAMES_DIR "/rockblox.score"
784 #define RESUME_FILE PLUGIN_GAMES_DIR "/rockblox.resume"
787 /* Default High Scores... */
788 struct highscore highscores
[NUM_SCORES
];
790 /* get random number from (0) to (range-1) */
791 static int t_rand (int range
)
793 return rb
->rand () % range
;
796 static inline void show_game_over (void)
798 rb
->splash(HZ
,"Game over!");
801 /* init the board array to have no blocks */
802 static void init_board (void)
805 for (i
= 0; i
< BOARD_WIDTH
; i
++)
806 for (j
= 0; j
< BOARD_HEIGHT
; j
++)
807 rockblox_status
.board
[j
][i
] = EMPTY_BLOCK
;
810 /* show the score, level and lines */
811 static void show_details (void)
813 char str
[25]; /* for strings */
815 #ifdef HAVE_LCD_BITMAP
817 rb
->lcd_set_foreground (LCD_BLACK
);
818 rb
->lcd_set_background (LCD_WHITE
);
820 rb
->snprintf (str
, sizeof (str
), "%d", rockblox_status
.score
);
821 rb
->lcd_putsxy (LABEL_X
, SCORE_Y
, str
);
822 rb
->snprintf (str
, sizeof (str
), "%d", rockblox_status
.level
);
823 rb
->lcd_putsxy (LEVEL_X
, LEVEL_Y
, str
);
824 rb
->snprintf (str
, sizeof (str
), "%d", rockblox_status
.lines
);
825 rb
->lcd_putsxy (LINES_X
, LINES_Y
, str
);
826 #else /* HAVE_LCD_CHARCELLS */
827 rb
->snprintf (str
, sizeof (str
), "L%d/%d", rockblox_status
.level
,
828 rockblox_status
.lines
);
829 rb
->lcd_puts (5, 0, str
);
830 rb
->snprintf (str
, sizeof (str
), "S%d", rockblox_status
.score
);
831 rb
->lcd_puts (5, 1, str
);
836 static void show_highscores (void)
839 char str
[25]; /* for strings */
841 for (i
= 0; i
<NUM_SCORES
; i
++)
843 rb
->snprintf (str
, sizeof (str
), "%06d" _SPACE
"L%1d",
844 highscores
[i
].score
, highscores
[i
].level
);
845 rb
->lcd_putsxy (HIGH_LABEL_X
, HIGH_SCORE_Y
+ (10 * i
), str
);
850 static void load_game(void)
856 fd
= rb
->open(RESUME_FILE
, O_RDONLY
);
859 if (rb
->read(fd
, &rockblox_status
, sizeof(struct _rockblox_status
))
860 < (ssize_t
)sizeof(struct _rockblox_status
))
862 rb
->splash(HZ
/2, "Loading Rockblox resume info failed");
873 static void dump_resume(void)
877 fd
= rb
->open(RESUME_FILE
, O_WRONLY
|O_CREAT
, 0666);
881 if (rb
->write(fd
, &rockblox_status
, sizeof(struct _rockblox_status
))
891 rb
->splash(HZ
/2, "Writing Rockblox resume info failed");
895 static void init_rockblox (bool resume
)
901 rb
->snprintf(score_name
, sizeof(score_name
), "%04d%02d%02d %02d%02d%02d",
902 tm
->tm_year
+ 1900, tm
->tm_mon
+ 1, tm
->tm_mday
,
903 tm
->tm_hour
, tm
->tm_min
, tm
->tm_sec
);
905 #ifdef HAVE_LCD_BITMAP
906 rb
->lcd_bitmap (rockblox_background
, 0, 0, LCD_WIDTH
, LCD_HEIGHT
);
907 #else /* HAVE_LCD_CHARCELLS */
909 pgfx_display_block (3, 0, 3, 1);
910 pgfx_display_block (4, 0, 3, 0);
911 rb
->lcd_puts(4, 1, " ");
912 pgfx_clear_display();
913 pgfx_fillrect (3, 0, 2, 14);
914 pgfx_fillrect (15, 7, 2, 7);
919 rockblox_status
.level
= 1;
920 rockblox_status
.lines
= 0;
921 rockblox_status
.score
= 0;
922 rockblox_status
.nf
= t_rand(BLOCKS_NUM
);
934 static inline int level_speed(int level
)
936 #if BOARD_HEIGHT == 20
937 return (5*HZ
) / (level
+ 9);
938 #elif BOARD_HEIGHT == 14
939 return (7*HZ
) / (level
+ 9);
943 static int getRelativeX (int figure
, int square
, int orientation
)
945 switch (orientation
) {
947 return figures
[figure
].shapeX
[square
];
949 return figures
[figure
].shapeY
[square
];
951 return -figures
[figure
].shapeX
[square
];
953 return -figures
[figure
].shapeY
[square
];
959 static int getRelativeY (int figure
, int square
, int orientation
)
961 switch (orientation
) {
963 return figures
[figure
].shapeY
[square
];
965 return -figures
[figure
].shapeX
[square
];
967 return -figures
[figure
].shapeY
[square
];
969 return figures
[figure
].shapeX
[square
];
975 /* redraw the while board on the screen */
976 static void refresh_board (void)
978 int i
, j
, x
, y
, block
;
981 rb
->lcd_set_foreground (LCD_BLACK
);
983 mylcd_set_drawmode (DRMODE_SOLID
| DRMODE_INVERSEVID
);
986 mylcd_fillrect (BOARD_X
, BOARD_Y
, BOARD_WIDTH
* BLOCK_WIDTH
,
987 BOARD_HEIGHT
* BLOCK_HEIGHT
);
990 mylcd_set_drawmode (DRMODE_SOLID
);
993 for (i
= 0; i
< BOARD_WIDTH
; i
++)
994 for (j
= 0; j
< BOARD_HEIGHT
; j
++) {
995 block
= rockblox_status
.board
[j
][i
];
996 if (block
!= EMPTY_BLOCK
) {
997 #ifdef HAVE_LCD_BITMAP
1000 rb
->lcd_set_foreground (figures
[block
].color
[1]);
1002 rb
->lcd_fillrect (BOARD_X
+ i
* BLOCK_WIDTH
,
1003 BOARD_Y
+ j
* BLOCK_HEIGHT
,
1004 BLOCK_WIDTH
, BLOCK_HEIGHT
);
1007 rb
->lcd_set_foreground (figures
[block
].color
[0]);
1009 rb
->lcd_vline (BOARD_X
+ i
* BLOCK_WIDTH
,
1010 BOARD_Y
+ j
* BLOCK_HEIGHT
,
1011 BOARD_Y
+ (j
+ 1) * BLOCK_HEIGHT
- 2);
1012 rb
->lcd_hline (BOARD_X
+ i
* BLOCK_WIDTH
,
1013 BOARD_X
+ (i
+ 1) * BLOCK_WIDTH
- 2,
1014 BOARD_Y
+ j
* BLOCK_HEIGHT
);
1016 /* shadow drawing */
1017 rb
->lcd_set_foreground (figures
[block
].color
[2]);
1019 rb
->lcd_vline (BOARD_X
+ (i
+ 1) * BLOCK_WIDTH
- 1,
1020 BOARD_Y
+ j
* BLOCK_HEIGHT
+ 1,
1021 BOARD_Y
+ (j
+ 1) * BLOCK_HEIGHT
- 1);
1022 rb
->lcd_hline (BOARD_X
+ i
* BLOCK_WIDTH
+ 1,
1023 BOARD_X
+ (i
+ 1) * BLOCK_WIDTH
- 1,
1024 BOARD_Y
+ (j
+ 1) * BLOCK_HEIGHT
- 1);
1025 #else /* HAVE_LCD_CHARCELLS */
1026 pgfx_drawpixel (BOARD_X
+ i
, BOARD_Y
+ j
);
1031 for (i
= 0; i
< 4; i
++) {
1032 x
= getRelativeX (rockblox_status
.cf
, i
, rockblox_status
.co
)
1033 + rockblox_status
.cx
;
1034 y
= getRelativeY (rockblox_status
.cf
, i
, rockblox_status
.co
)
1035 + rockblox_status
.cy
;
1036 #ifdef HAVE_LCD_BITMAP
1038 /* middle drawing */
1039 rb
->lcd_set_foreground (figures
[rockblox_status
.cf
].color
[1]);
1041 rb
->lcd_fillrect (BOARD_X
+ x
* BLOCK_WIDTH
,
1042 BOARD_Y
+ y
* BLOCK_HEIGHT
,
1043 BLOCK_WIDTH
, BLOCK_HEIGHT
);
1046 rb
->lcd_set_foreground (figures
[rockblox_status
.cf
].color
[0]);
1048 rb
->lcd_vline (BOARD_X
+ x
* BLOCK_WIDTH
, BOARD_Y
+ y
* BLOCK_HEIGHT
,
1049 BOARD_Y
+ (y
+ 1) * BLOCK_HEIGHT
- 2);
1050 rb
->lcd_hline (BOARD_X
+ x
* BLOCK_WIDTH
,
1051 BOARD_X
+ (x
+ 1) * BLOCK_WIDTH
- 2,
1052 BOARD_Y
+ y
* BLOCK_HEIGHT
);
1054 /* shadow drawing */
1055 rb
->lcd_set_foreground (figures
[rockblox_status
.cf
].color
[2]);
1057 rb
->lcd_vline (BOARD_X
+ (x
+ 1) * BLOCK_WIDTH
- 1,
1058 BOARD_Y
+ y
* BLOCK_HEIGHT
+ 1,
1059 BOARD_Y
+ (y
+ 1) * BLOCK_HEIGHT
- 1);
1060 rb
->lcd_hline (BOARD_X
+ x
* BLOCK_WIDTH
+ 1,
1061 BOARD_X
+ (x
+ 1) * BLOCK_WIDTH
- 1,
1062 BOARD_Y
+ (y
+ 1) * BLOCK_HEIGHT
- 1);
1063 #else /* HAVE_LCD_CHARCELLS */
1064 pgfx_drawpixel (BOARD_X
+ x
, BOARD_Y
+ y
);
1070 static bool canMoveTo (int x
, int y
, int newOrientation
)
1073 for (i
= 0; i
< 4; i
++) {
1074 ry
= getRelativeY (rockblox_status
.cf
, i
, newOrientation
) + y
;
1075 rx
= getRelativeX (rockblox_status
.cf
, i
, newOrientation
) + x
;
1076 if ((rx
< 0 || rx
>= BOARD_WIDTH
) ||
1077 (ry
< 0 || ry
>= BOARD_HEIGHT
) ||
1078 (rockblox_status
.board
[ry
][rx
] != EMPTY_BLOCK
))
1084 /* draws the preview of next block in the preview window */
1085 static void draw_next_block (void)
1088 /* clear preview window first */
1090 rb
->lcd_set_foreground (LCD_BLACK
);
1091 #elif LCD_DEPTH == 1
1092 mylcd_set_drawmode (DRMODE_SOLID
| DRMODE_INVERSEVID
);
1096 mylcd_fillrect (PREVIEW_X
, PREVIEW_Y
, BLOCK_WIDTH
* 4, BLOCK_HEIGHT
* 4);
1099 mylcd_set_drawmode (DRMODE_SOLID
);
1102 /* draw the lightgray rectangles */
1104 rb
->lcd_set_foreground (LCD_RGBPACK (40, 40, 40));
1105 #elif LCD_DEPTH == 2
1106 rb
->lcd_set_foreground (LCD_DARKGRAY
);
1110 for (rx
= 0; rx
< 4; rx
++)
1111 for (ry
= 0; ry
< 4; ry
++)
1112 rb
->lcd_drawrect (PREVIEW_X
+ rx
* BLOCK_WIDTH
,
1113 PREVIEW_Y
+ ry
* BLOCK_HEIGHT
, BLOCK_WIDTH
,
1117 /* draw the figure */
1118 for (i
= 0; i
< 4; i
++) {
1119 rx
= getRelativeX (rockblox_status
.nf
, i
, 0) + 2;
1120 ry
= getRelativeY (rockblox_status
.nf
, i
, 0) + 2;
1121 #ifdef HAVE_LCD_BITMAP
1123 rb
->lcd_set_foreground (figures
[rockblox_status
.nf
].color
[1]); /* middle drawing */
1125 rb
->lcd_fillrect (PREVIEW_X
+ rx
* BLOCK_WIDTH
,
1126 PREVIEW_Y
+ ry
* BLOCK_HEIGHT
,
1127 BLOCK_WIDTH
, BLOCK_HEIGHT
);
1129 rb
->lcd_set_foreground (figures
[rockblox_status
.nf
].color
[0]); /* light drawing */
1131 rb
->lcd_vline (PREVIEW_X
+ rx
* BLOCK_WIDTH
,
1132 PREVIEW_Y
+ ry
* BLOCK_HEIGHT
,
1133 PREVIEW_Y
+ (ry
+ 1) * BLOCK_HEIGHT
- 2);
1134 rb
->lcd_hline (PREVIEW_X
+ rx
* BLOCK_WIDTH
,
1135 PREVIEW_X
+ (rx
+ 1) * BLOCK_WIDTH
- 2,
1136 PREVIEW_Y
+ ry
* BLOCK_HEIGHT
);
1138 rb
->lcd_set_foreground (figures
[rockblox_status
.nf
].color
[2]); /* shadow drawing */
1140 rb
->lcd_vline (PREVIEW_X
+ (rx
+ 1) * BLOCK_WIDTH
- 1,
1141 PREVIEW_Y
+ ry
* BLOCK_HEIGHT
+ 1,
1142 PREVIEW_Y
+ (ry
+ 1) * BLOCK_HEIGHT
- 1);
1143 rb
->lcd_hline (PREVIEW_X
+ rx
* BLOCK_WIDTH
+ 1,
1144 PREVIEW_X
+ (rx
+ 1) * BLOCK_WIDTH
- 1,
1145 PREVIEW_Y
+ (ry
+ 1) * BLOCK_HEIGHT
- 1);
1146 #else /* HAVE_LCD_CHARCELLS */
1147 pgfx_drawpixel (PREVIEW_X
+ rx
, PREVIEW_Y
+ ry
);
1153 /* move the block to a relative location */
1154 static void move_block (int x
, int y
, int o
)
1156 if (canMoveTo (rockblox_status
.cx
+ x
, rockblox_status
.cy
+ y
, o
)) {
1157 rockblox_status
.cy
+= y
;
1158 rockblox_status
.cx
+= x
;
1159 rockblox_status
.co
= o
;
1163 /* try to add a new block to play with (return true if gameover) */
1164 static void new_block (void)
1166 rockblox_status
.cy
= 1;
1167 rockblox_status
.cx
= 5;
1168 rockblox_status
.cf
= rockblox_status
.nf
;
1169 rockblox_status
.co
= 0; /* start at the same orientation all time */
1170 rockblox_status
.nf
= t_rand (BLOCKS_NUM
);
1171 rockblox_status
.gameover
= !canMoveTo (rockblox_status
.cx
,
1172 rockblox_status
.cy
, rockblox_status
.co
);
1178 /* check for filled rockblox_status.lines and do what necessary */
1179 static int check_lines (void)
1185 for (j
= 0; j
< BOARD_HEIGHT
; j
++) {
1186 for (i
= 0; ((i
< BOARD_WIDTH
) &&
1187 (rockblox_status
.board
[j
][i
] != EMPTY_BLOCK
)); i
++);
1188 if (i
== BOARD_WIDTH
) { /* woo hoo, we have a line */
1190 for (y
= j
; y
> 0; y
--)
1192 for (i
= 0; i
< BOARD_WIDTH
; i
++)
1194 rockblox_status
.board
[y
][i
] = rockblox_status
.board
[y
- 1][i
];
1203 /* moves down the figure and returns true if gameover */
1204 static void move_down (void)
1208 if (!canMoveTo (rockblox_status
.cx
, rockblox_status
.cy
+ 1, rockblox_status
.co
)) {
1209 /* save figure to board */
1210 for (i
= 0; i
< 4; i
++) {
1211 rx
= getRelativeX (rockblox_status
.cf
, i
, rockblox_status
.co
) + rockblox_status
.cx
;
1212 ry
= getRelativeY (rockblox_status
.cf
, i
, rockblox_status
.co
) + rockblox_status
.cy
;
1213 rockblox_status
.board
[ry
][rx
] = rockblox_status
.cf
;
1215 /* check if formed some lines */
1218 /* the original scoring from "http://en.wikipedia.org/wiki/Rockblox" */
1219 rockblox_status
.score
+= scoring
[l
- 1] * rockblox_status
.level
;
1220 rockblox_status
.lines
+= l
;
1221 rockblox_status
.level
= (int) rockblox_status
.lines
/ 10 + 1;
1227 /* generate a new figure */
1230 move_block (0, 1, rockblox_status
.co
);
1233 static bool rockblox_help(void)
1235 static char *help_text
[] = {
1236 "Rockblox", "", "Aim", "",
1237 "Make", "the", "falling", "blocks", "of",
1238 "different", "shapes", "form", "full", "rows.",
1239 "Whenever", "a", "row", "is", "completed,",
1240 "it", "will", "be", "cleared", "away",
1241 "and", "you", "gain", "points."
1243 static struct style_text formation
[]={
1244 { 0, TEXT_CENTER
|TEXT_UNDERLINE
},
1249 #ifdef HAVE_LCD_BITMAP
1250 rb
->lcd_setfont(FONT_UI
);
1252 #ifdef HAVE_LCD_COLOR
1253 rb
->lcd_set_background(LCD_BLACK
);
1254 rb
->lcd_set_foreground(LCD_WHITE
);
1256 if (display_text(ARRAYLEN(help_text
), help_text
, formation
, NULL
, true))
1258 #ifdef HAVE_LCD_BITMAP
1259 rb
->lcd_setfont(FONT_SYSFIXED
);
1265 static int rockblox_menu_cb(int action
, const struct menu_item_ex
*this_item
)
1267 int i
= ((intptr_t)this_item
);
1268 if(action
== ACTION_REQUEST_MENUITEM
1269 && !resume
&& (i
==0 || i
==5))
1270 return ACTION_EXIT_MENUITEM
;
1274 static int rockblox_menu(void)
1278 MENUITEM_STRINGLIST(main_menu
, "Rockblox Menu", rockblox_menu_cb
,
1279 "Resume Game", "Start New Game",
1280 "Help", "High Scores", "Playback Control",
1281 "Quit without Saving", "Quit");
1283 rb
->button_clear_queue();
1285 switch (rb
->do_menu(&main_menu
, &selected
, NULL
, false)) {
1288 rb
->remove(RESUME_FILE
);
1289 init_rockblox(true);
1292 init_rockblox(false);
1295 if (rockblox_help())
1299 highscore_show(-1, highscores
, NUM_SCORES
, true);
1302 if (playback_control(NULL
))
1309 rb
->splash(HZ
*1, "Saving game ...");
1313 case MENU_ATTACHED_USB
:
1323 static int rockblox_loop (void)
1326 int lastbutton
= BUTTON_NONE
;
1327 long next_down_tick
= *rb
->current_tick
+ level_speed(rockblox_status
.level
);
1329 if (rockblox_menu()) {
1333 resume_file
= false;
1336 #ifdef HAS_BUTTON_HOLD
1337 if (rb
->button_hold ()) {
1338 /* Turn on backlight timeout (revert to settings) */
1339 backlight_use_settings(); /* backlight control in lib/helper.c */
1340 rb
->splash(0, "Paused");
1341 while (rb
->button_hold ())
1344 /* Turn off backlight timeout */
1345 backlight_force_on(); /* backlight control in lib/helper.c */
1347 /* get rid of the splash text */
1348 rb
->lcd_bitmap (rockblox_background
, 0, 0, LCD_WIDTH
, LCD_HEIGHT
);
1358 button
= rb
->button_get_w_tmo (MAX(next_down_tick
- *rb
->current_tick
, 1));
1360 #ifdef ROCKBLOX_RC_OFF
1361 case ROCKBLOX_RC_OFF
:
1364 #ifdef ROCKBLOX_OFF_PRE
1365 if (lastbutton
!= ROCKBLOX_OFF_PRE
)
1372 #if defined(ROCKBLOX_ROTATE)
1373 case ROCKBLOX_ROTATE
:
1375 case ROCKBLOX_ROTATE_CCW
:
1376 case ROCKBLOX_ROTATE_CCW
| BUTTON_REPEAT
:
1377 #ifdef HAVE_SCROLLWHEEL
1378 /* if the wheel is disabled, add an event to the stack. */
1379 if(wheel_enabled
== false)
1382 /* if it's enabled, go ahead and rotate.. */
1385 #ifdef ROCKBLOX_ROTATE_CCW2
1387 case ROCKBLOX_ROTATE_CCW2
:
1389 move_block (0, 0, (rockblox_status
.co
+ 1) % figures
[rockblox_status
.cf
].max_or
);
1392 case ROCKBLOX_ROTATE_CW
:
1393 case ROCKBLOX_ROTATE_CW
| BUTTON_REPEAT
:
1394 #ifdef HAVE_SCROLLWHEEL
1395 if(wheel_enabled
== false)
1400 #ifdef ROCKBLOX_ROTATE_CW2
1402 case ROCKBLOX_ROTATE_CW2
:
1405 (rockblox_status
.co
+ figures
[rockblox_status
.cf
].max_or
-
1406 1) % figures
[rockblox_status
.cf
].max_or
);
1410 case ROCKBLOX_DOWN
| BUTTON_REPEAT
:
1411 move_block (0, 1, rockblox_status
.co
);
1414 case ROCKBLOX_RIGHT
:
1415 case ROCKBLOX_RIGHT
| BUTTON_REPEAT
:
1416 move_block (1, 0, rockblox_status
.co
);
1420 case ROCKBLOX_LEFT
| BUTTON_REPEAT
:
1421 move_block (-1, 0, rockblox_status
.co
);
1425 #ifdef ROCKBLOX_DROP_PRE
1426 if (lastbutton
!= ROCKBLOX_DROP_PRE
)
1429 while (canMoveTo (rockblox_status
.cx
, rockblox_status
.cy
+ 1, rockblox_status
.co
))
1430 move_block (0, 1, rockblox_status
.co
);
1432 #ifdef ROCKBLOX_RESTART
1433 case ROCKBLOX_RESTART
:
1434 rb
->splash (HZ
* 1, "Restarting...");
1435 init_rockblox (false);
1440 if (rb
->default_event_handler (button
) == SYS_USB_CONNECTED
)
1441 return PLUGIN_USB_CONNECTED
;
1444 if (button
!= BUTTON_NONE
)
1445 lastbutton
= button
;
1447 #ifdef HAVE_SCROLLWHEEL
1448 /* check if we should enable the scroll wheel, if events
1449 * begin to stack up... */
1450 if(wheel_enabled
== false)
1452 /* stopped rotating the wheel, reset the count */
1453 if(wheel_events
== last_wheel_event
)
1455 last_wheel_event
= 0;
1458 /* rotated the wheel a while constantly, enable it. */
1459 else if(wheel_events
> 3)
1461 wheel_enabled
= true;
1464 /* this evens out the last event and the "current" event.
1465 * if we get an event next time through button reading, it will
1466 * remain ahead of last_event. if we don't, they'll end up equaling
1467 * each other.. thus, the scroll count will be reset. */
1468 if(wheel_enabled
== false && wheel_events
> last_wheel_event
)
1473 if (TIME_AFTER(*rb
->current_tick
, next_down_tick
)) {
1475 next_down_tick
+= level_speed(rockblox_status
.level
);
1476 if (TIME_AFTER(*rb
->current_tick
, next_down_tick
))
1477 /* restart time "raster" when we had to wait longer than usual
1478 * (pause, game restart etc) */
1479 next_down_tick
= *rb
->current_tick
+ level_speed(rockblox_status
.level
);
1482 if (rockblox_status
.gameover
) {
1484 rb
->lcd_set_foreground (LCD_BLACK
);
1497 enum plugin_status
plugin_start (const void *parameter
)
1502 rb
->srand (*rb
->current_tick
);
1504 /* Load HighScore if any */
1505 highscore_load(SCORE_FILE
, highscores
, NUM_SCORES
);
1508 rb
->lcd_set_backdrop(NULL
);
1511 #ifdef HAVE_LCD_BITMAP
1512 rb
->lcd_setfont (FONT_SYSFIXED
);
1514 if (!pgfx_init(4, 2))
1516 rb
->splash(HZ
*2, "Old LCD :(");
1520 /* Turn off backlight timeout */
1521 backlight_force_on(); /* backlight control in lib/helper.c */
1523 resume_file
= resume
;
1524 while(!rockblox_loop()) {
1526 int position
= highscore_update(rockblox_status
.score
,
1527 rockblox_status
.level
, "",
1528 highscores
, NUM_SCORES
);
1529 if (position
!= -1) {
1531 rb
->splash(HZ
*2, "New High Score");
1532 highscore_show(position
, highscores
, NUM_SCORES
, true);
1537 #ifndef HAVE_LCD_BITMAP
1540 /* Save user's HighScore */
1541 highscore_save(SCORE_FILE
, highscores
, NUM_SCORES
);
1542 backlight_use_settings(); /* backlight control in lib/helper.c */