1 /***************************************************************************
3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___
4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
10 * Copyright (C) 2002 Kjell Ericson
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation; either version 2
15 * of the License, or (at your option) any later version.
17 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
18 * KIND, either express or implied.
20 ****************************************************************************/
25 /* variable button definitions */
26 #if CONFIG_KEYPAD == RECORDER_PAD
27 #define CHC_QUIT BUTTON_OFF
28 #define CHC_STARTSTOP BUTTON_PLAY
29 #define CHC_RESET BUTTON_LEFT
30 #define CHC_MENU BUTTON_F1
31 #define CHC_SETTINGS_INC BUTTON_UP
32 #define CHC_SETTINGS_DEC BUTTON_DOWN
33 #define CHC_SETTINGS_OK BUTTON_PLAY
34 #define CHC_SETTINGS_OK2 BUTTON_LEFT
35 #define CHC_SETTINGS_CANCEL BUTTON_OFF
37 #elif CONFIG_KEYPAD == ARCHOS_AV300_PAD
38 #define CHC_QUIT BUTTON_OFF
39 #define CHC_STARTSTOP BUTTON_SELECT
40 #define CHC_RESET BUTTON_LEFT
41 #define CHC_MENU BUTTON_F1
42 #define CHC_SETTINGS_INC BUTTON_UP
43 #define CHC_SETTINGS_DEC BUTTON_DOWN
44 #define CHC_SETTINGS_OK BUTTON_SELECT
45 #define CHC_SETTINGS_OK2 BUTTON_LEFT
46 #define CHC_SETTINGS_CANCEL BUTTON_OFF
48 #elif CONFIG_KEYPAD == ONDIO_PAD
49 #define CHC_QUIT BUTTON_OFF
50 #define CHC_STARTSTOP BUTTON_RIGHT
51 #define CHC_RESET BUTTON_LEFT
52 #define CHC_MENU BUTTON_MENU
53 #define CHC_SETTINGS_INC BUTTON_UP
54 #define CHC_SETTINGS_DEC BUTTON_DOWN
55 #define CHC_SETTINGS_OK BUTTON_RIGHT
56 #define CHC_SETTINGS_OK2 BUTTON_LEFT
57 #define CHC_SETTINGS_CANCEL BUTTON_MENU
59 #elif CONFIG_KEYPAD == PLAYER_PAD
60 #define CHC_QUIT BUTTON_ON
61 #define CHC_STARTSTOP BUTTON_PLAY
62 #define CHC_RESET BUTTON_STOP
63 #define CHC_MENU BUTTON_MENU
64 #define CHC_SETTINGS_INC BUTTON_RIGHT
65 #define CHC_SETTINGS_DEC BUTTON_LEFT
66 #define CHC_SETTINGS_OK BUTTON_PLAY
67 #define CHC_SETTINGS_CANCEL BUTTON_STOP
68 #define CHC_SETTINGS_CANCEL2 BUTTON_MENU
70 #elif (CONFIG_KEYPAD == IRIVER_H100_PAD) || \
71 (CONFIG_KEYPAD == IRIVER_H300_PAD)
72 #define CHC_QUIT BUTTON_SELECT
73 #define CHC_STARTSTOP BUTTON_ON
74 #define CHC_RESET BUTTON_OFF
75 #define CHC_MENU BUTTON_REC
76 #define CHC_SETTINGS_INC BUTTON_RIGHT
77 #define CHC_SETTINGS_DEC BUTTON_LEFT
78 #define CHC_SETTINGS_OK BUTTON_ON
79 #define CHC_SETTINGS_CANCEL BUTTON_OFF
80 #define CHC_SETTINGS_CANCEL2 BUTTON_REC
82 #elif (CONFIG_KEYPAD == IPOD_4G_PAD) || \
83 (CONFIG_KEYPAD == IPOD_3G_PAD) || \
84 (CONFIG_KEYPAD == IPOD_1G2G_PAD)
85 #define CHC_QUIT BUTTON_PLAY
86 #define CHC_STARTSTOP BUTTON_SELECT
87 #define CHC_RESET BUTTON_LEFT
88 #define CHC_MENU BUTTON_MENU
89 #define CHC_SETTINGS_INC BUTTON_SCROLL_FWD
90 #define CHC_SETTINGS_DEC BUTTON_SCROLL_BACK
91 #define CHC_SETTINGS_OK BUTTON_SELECT
92 #define CHC_SETTINGS_CANCEL BUTTON_MENU
94 #elif CONFIG_KEYPAD == IRIVER_IFP7XX_PAD
95 #define CHC_QUIT BUTTON_PLAY
96 #define CHC_STARTSTOP BUTTON_MODE
97 #define CHC_RESET BUTTON_EQ
98 #define CHC_MENU BUTTON_SELECT
99 #define CHC_SETTINGS_INC BUTTON_RIGHT
100 #define CHC_SETTINGS_DEC BUTTON_LEFT
101 #define CHC_SETTINGS_OK BUTTON_SELECT
102 #define CHC_SETTINGS_CANCEL BUTTON_PLAY
104 #elif CONFIG_KEYPAD == IAUDIO_X5M5_PAD
105 #define CHC_QUIT BUTTON_REC
106 #define CHC_STARTSTOP BUTTON_PLAY
107 #define CHC_RESET BUTTON_POWER
108 #define CHC_MENU BUTTON_SELECT
109 #define CHC_SETTINGS_INC BUTTON_RIGHT
110 #define CHC_SETTINGS_DEC BUTTON_LEFT
111 #define CHC_SETTINGS_OK BUTTON_SELECT
112 #define CHC_SETTINGS_CANCEL BUTTON_REC
114 #elif CONFIG_KEYPAD == GIGABEAT_PAD
115 #define CHC_QUIT BUTTON_POWER
116 #define CHC_STARTSTOP BUTTON_SELECT
117 #define CHC_RESET BUTTON_A
118 #define CHC_MENU BUTTON_MENU
119 #define CHC_SETTINGS_INC BUTTON_UP
120 #define CHC_SETTINGS_DEC BUTTON_DOWN
121 #define CHC_SETTINGS_OK BUTTON_SELECT
122 #define CHC_SETTINGS_CANCEL BUTTON_POWER
124 #elif (CONFIG_KEYPAD == SANSA_E200_PAD) || \
125 (CONFIG_KEYPAD == SANSA_C200_PAD)
126 #define CHC_QUIT BUTTON_POWER
127 #define CHC_STARTSTOP BUTTON_SELECT
128 #define CHC_RESET BUTTON_DOWN
129 #define CHC_MENU BUTTON_UP
130 #define CHC_SETTINGS_INC BUTTON_RIGHT
131 #define CHC_SETTINGS_DEC BUTTON_LEFT
132 #define CHC_SETTINGS_OK BUTTON_SELECT
133 #define CHC_SETTINGS_CANCEL BUTTON_POWER
135 #elif CONFIG_KEYPAD == IRIVER_H10_PAD
136 #define CHC_QUIT BUTTON_POWER
137 #define CHC_STARTSTOP BUTTON_PLAY
138 #define CHC_RESET BUTTON_FF
139 #define CHC_MENU BUTTON_REW
140 #define CHC_SETTINGS_INC BUTTON_RIGHT
141 #define CHC_SETTINGS_DEC BUTTON_LEFT
142 #define CHC_SETTINGS_OK BUTTON_PLAY
143 #define CHC_SETTINGS_CANCEL BUTTON_POWER
145 #elif CONFIG_KEYPAD == MROBE500_PAD
146 #define CHC_QUIT BUTTON_POWER
147 #define CHC_STARTSTOP BUTTON_RC_PLAY
148 #define CHC_RESET BUTTON_RC_HEART
149 #define CHC_MENU BUTTON_RC_MODE
150 #define CHC_SETTINGS_INC BUTTON_RC_VOL_UP
151 #define CHC_SETTINGS_DEC BUTTON_RC_VOL_DOWN
152 #define CHC_SETTINGS_OK BUTTON_RC_PLAY
153 #define CHC_SETTINGS_CANCEL BUTTON_POWER
155 #elif CONFIG_KEYPAD == GIGABEAT_S_PAD
156 #define CHC_QUIT BUTTON_BACK
157 #define CHC_STARTSTOP BUTTON_PLAY
158 #define CHC_RESET BUTTON_PREV
159 #define CHC_MENU BUTTON_MENU
160 #define CHC_SETTINGS_INC BUTTON_UP
161 #define CHC_SETTINGS_DEC BUTTON_DOWN
162 #define CHC_SETTINGS_OK BUTTON_SELECT
163 #define CHC_SETTINGS_CANCEL BUTTON_BACK
165 #elif CONFIG_KEYPAD == MROBE100_PAD
166 #define CHC_QUIT BUTTON_POWER
167 #define CHC_STARTSTOP BUTTON_SELECT
168 #define CHC_RESET BUTTON_DISPLAY
169 #define CHC_MENU BUTTON_MENU
170 #define CHC_SETTINGS_INC BUTTON_UP
171 #define CHC_SETTINGS_DEC BUTTON_DOWN
172 #define CHC_SETTINGS_OK BUTTON_SELECT
173 #define CHC_SETTINGS_CANCEL BUTTON_POWER
175 #elif CONFIG_KEYPAD == IAUDIO_M3_PAD
176 #define CHC_QUIT BUTTON_RC_REC
177 #define CHC_STARTSTOP BUTTON_RC_PLAY
178 #define CHC_RESET BUTTON_RC_REW
179 #define CHC_MENU BUTTON_RC_MENU
180 #define CHC_SETTINGS_INC BUTTON_RC_VOL_UP
181 #define CHC_SETTINGS_DEC BUTTON_RC_VOL_DOWN
182 #define CHC_SETTINGS_OK BUTTON_RC_PLAY
183 #define CHC_SETTINGS_CANCEL BUTTON_RC_REC
185 #elif CONFIG_KEYPAD == COWOND2_PAD
186 #define CHC_QUIT BUTTON_POWER
187 #define CHC_RESET (BUTTON_CENTER|BUTTON_MENU)
188 #define CHC_MENU BUTTON_MENU
189 #define CHC_SETTINGS_INC BUTTON_PLUS
190 #define CHC_SETTINGS_DEC BUTTON_MINUS
191 #define CHC_SETTINGS_CANCEL BUTTON_POWER
194 #error No keymap defined!
198 #ifndef CHC_SETTINGS_OK
199 #define CHC_SETTINGS_OK BUTTON_CENTER
201 #ifndef CHC_STARTSTOP
202 #define CHC_STARTSTOP BUTTON_CENTER
204 #ifndef CHC_SETTINGS_INC
205 #define CHC_SETTINGS_INC BUTTON_TOPMIDDLE
207 #ifndef CHC_SETTINGS_DEC
208 #define CHC_SETTINGS_DEC BUTTON_BOTTOMMIDDLE
211 #define CHC_RESET BUTTON_TOPLEFT
214 #define CHC_MENU BUTTON_TOPRIGHT
219 /* leave first line blank on bitmap display, for pause icon */
220 #ifdef HAVE_LCD_BITMAP
226 /* here is a global api struct pointer. while not strictly necessary,
227 it's nice not to have to pass the api pointer in all function calls
229 static const struct plugin_api
* rb
;
230 MEM_FUNCTION_WRAPPERS(rb
);
231 #define MAX_PLAYERS 10
243 } timer_holder
[MAX_PLAYERS
];
245 static int run_timer(int nr
);
246 static int chessclock_set_int(char* string
,
252 #define FLAGS_SET_INT_SECONDS 1
254 static char * show_time(int secs
);
255 static int simple_menu(int nr
, unsigned char **strarr
);
259 #define MAX_TIME 7200
261 /* this is the plugin entry point */
262 enum plugin_status
plugin_start(const struct plugin_api
* api
, const void* parameter
)
270 rb
->memset(&settings
, 0, sizeof(settings
));
272 /* now go ahead and have fun! */
273 rb
->splash(HZ
, "Chess Clock");
275 rb
->lcd_clear_display();
281 res
=chessclock_set_int("Number of players",
282 &settings
.nr_timers
, 1, 1,
286 res
=chessclock_set_int("Total time",
287 &settings
.total_time
, 10, 0, MAX_TIME
,
288 FLAGS_SET_INT_SECONDS
);
289 settings
.round_time
=settings
.total_time
;
292 res
=chessclock_set_int("Max round time", &settings
.round_time
,
293 10, 0, settings
.round_time
,
294 FLAGS_SET_INT_SECONDS
);
302 return PLUGIN_USB_CONNECTED
;
314 for (i
=0; i
<settings
.nr_timers
; i
++) {
315 timer_holder
[i
].total_time
=settings
.total_time
;
316 timer_holder
[i
].used_time
=0;
317 timer_holder
[i
].hidden
=false;
320 pause
=true; /* We start paused */
326 for (i
=0; done
&& i
<settings
.nr_timers
; i
++) {
327 if (!timer_holder
[i
].hidden
)
333 if (!timer_holder
[nr
].hidden
) {
342 return PLUGIN_USB_CONNECTED
;
345 if (nr
>=settings
.nr_timers
)
351 nr
=settings
.nr_timers
-1;
358 #ifdef HAVE_LCD_BITMAP
359 static void show_pause_mode(bool enabled
)
361 static const char pause_icon
[] = {0x00,0x7f,0x7f,0x00,0x7f,0x7f,0x00};
364 rb
->lcd_mono_bitmap((unsigned char *)pause_icon
, 52, 0, 7, 8);
367 rb
->lcd_set_drawmode(DRMODE_SOLID
|DRMODE_INVERSEVID
);
368 rb
->lcd_fillrect(52, 0, 7, 8);
369 rb
->lcd_set_drawmode(DRMODE_SOLID
);
380 static int run_timer(int nr
)
383 char player_info
[13];
387 long max_ticks
=timer_holder
[nr
].total_time
*HZ
-timer_holder
[nr
].used_time
;
389 bool round_time
=false;
391 #ifdef HAVE_LCD_CHARCELLS
392 rb
->lcd_icon(ICON_PAUSE
, pause
);
394 show_pause_mode(pause
);
397 if (settings
.round_time
*HZ
<max_ticks
) {
398 max_ticks
=settings
.round_time
*HZ
;
401 rb
->snprintf(player_info
, sizeof(player_info
), "Player %d", nr
+1);
402 rb
->lcd_puts(0, FIRST_LINE
, (unsigned char *)player_info
);
403 last_tick
=*rb
->current_tick
;
408 if (ticks
>max_ticks
) {
410 rb
->lcd_puts(0, FIRST_LINE
+1, (unsigned char *)"ROUND UP!");
412 rb
->lcd_puts(0, FIRST_LINE
+1, (unsigned char *)"TIME OUT!");
416 if (((int)(rb->current_tick - start_ticks)/HZ)&1) {
417 rb->lcd_puts(0, FIRST_LINE, player_info);
419 rb->lcd_puts(0, FIRST_LINE, player_info);
422 rb
->lcd_puts(0, FIRST_LINE
, (unsigned char *)player_info
);
423 now
=*rb
->current_tick
;
425 ticks
+=now
-last_tick
;
426 if ((max_ticks
-ticks
)/HZ
== 10) {
427 /* Backlight on if 10 seconds remain */
433 rb
->snprintf(buf
, sizeof(buf
), "%s/",
434 show_time((max_ticks
-ticks
+HZ
-1)/HZ
));
435 /* Append total time */
436 rb
->strcpy(&buf
[rb
->strlen(buf
)],
437 show_time((timer_holder
[nr
].total_time
*HZ
-
438 timer_holder
[nr
].used_time
-
440 rb
->lcd_puts(0, FIRST_LINE
+1, (unsigned char *)buf
);
442 rb
->lcd_puts(0, FIRST_LINE
+1,
443 (unsigned char *)show_time((max_ticks
-ticks
+HZ
-1)/HZ
));
448 button
= rb
->button_get(false);
450 /* OFF/ON key to exit */
452 return -1; /* Indicate exit */
454 /* PLAY = Stop/Start toggle */
457 #ifdef HAVE_LCD_CHARCELLS
458 rb
->lcd_icon(ICON_PAUSE
, pause
);
460 show_pause_mode(pause
);
464 /* LEFT = Reset timer */
473 char *menu
[]={"Delete player", "Restart round",
474 "Set round time", "Set total time"};
475 ret
=simple_menu(4, (unsigned char **)menu
);
479 } else if (ret
==-2) {
482 timer_holder
[nr
].hidden
=true;
493 int val
=(max_ticks
-ticks
)/HZ
;
494 res
=chessclock_set_int("Round time",
497 FLAGS_SET_INT_SECONDS
);
498 if (res
==-1) { /*usb*/
502 ticks
=max_ticks
-val
*HZ
;
507 int val
=timer_holder
[nr
].total_time
;
508 res
=chessclock_set_int("Total time",
511 FLAGS_SET_INT_SECONDS
);
512 if (res
==-1) { /*usb*/
516 timer_holder
[nr
].total_time
=val
;
522 /* UP (RIGHT/+) = Scroll Lap timer up */
523 case CHC_SETTINGS_INC
:
528 /* DOWN (LEFT/-) = Scroll Lap timer down */
529 case CHC_SETTINGS_DEC
:
535 if (rb
->default_event_handler(button
) == SYS_USB_CONNECTED
) {
536 retval
= 3; /* been in usb mode */
541 rb
->sleep(HZ
/4); /* Sleep 1/4 of a second */
544 timer_holder
[nr
].used_time
+=ticks
;
549 static int chessclock_set_int(char* string
,
559 rb
->lcd_clear_display();
560 rb
->lcd_puts_scroll(0, FIRST_LINE
, (unsigned char *)string
);
564 if (flags
& FLAGS_SET_INT_SECONDS
)
565 rb
->snprintf(str
, sizeof str
,"%s (m:s)", show_time(*variable
));
567 rb
->snprintf(str
, sizeof str
,"%d", *variable
);
568 rb
->lcd_puts(0, FIRST_LINE
+1, (unsigned char *)str
);
571 button
= rb
->button_get(true);
573 case CHC_SETTINGS_INC
:
574 case CHC_SETTINGS_INC
| BUTTON_REPEAT
:
578 case CHC_SETTINGS_DEC
:
579 case CHC_SETTINGS_DEC
| BUTTON_REPEAT
:
583 case CHC_SETTINGS_OK
:
584 #ifdef CHC_SETTINGS_OK2
585 case CHC_SETTINGS_OK2
:
590 case CHC_SETTINGS_CANCEL
:
591 #ifdef CHC_SETTINGS_CANCEL2
592 case CHC_SETTINGS_CANCEL2
:
594 return 0; /* cancel */
598 if (rb
->default_event_handler(button
) == SYS_USB_CONNECTED
)
599 return -1; /* been in usb mode */
610 rb
->lcd_stop_scroll();
615 static char * show_time(int seconds
)
617 static char buf
[]="00:00";
618 rb
->snprintf(buf
, sizeof(buf
), "%02d:%02d", seconds
/60, seconds
%60);
625 static int simple_menu(int nr
, unsigned char **strarr
)
629 rb
->lcd_clear_display();
636 rb
->lcd_puts_scroll(0, FIRST_LINE
, strarr
[show
]);
639 button
= rb
->button_get(true);
641 case CHC_SETTINGS_INC
:
642 case CHC_SETTINGS_INC
| BUTTON_REPEAT
:
646 case CHC_SETTINGS_DEC
:
647 case CHC_SETTINGS_DEC
| BUTTON_REPEAT
:
651 case CHC_SETTINGS_OK
:
652 #ifdef CHC_SETTINGS_OK2
653 case CHC_SETTINGS_OK2
:
658 case CHC_SETTINGS_CANCEL
:
659 #ifdef CHC_SETTINGS_CANCEL2
660 case CHC_SETTINGS_CANCEL2
:
662 return -2; /* cancel */
666 if (rb
->default_event_handler(button
) == SYS_USB_CONNECTED
)
667 return -1; /* been in usb mode */