1 /***************************************************************************
3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___
4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
10 * Copyright (C) 2003 Pierre Delore
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 ****************************************************************************/
22 #include "lib/configfile.h"
24 #ifdef HAVE_LCD_CHARCELLS
26 /* Euro converter for the player */
34 ON+PLAY : Swap Euro<>Home
35 MENU : Display the Menu
36 Currency -> Allows to choose the currency
37 Exit-> Exit the plugin
41 I use signed long long (64 bits).
42 A value have 5 digits after the . (123.45 = 12345000)
45 - The Irish currency needs 6 digits after the . to have sufficient precision on big number
50 /* Name and path of the config file*/
51 static const char cfg_filename
[] = "euroconverter.cfg";
52 #define CFGFILE_VERSION 0 /* Current config file version */
53 #define CFGFILE_MINVERSION 0 /* Minimum config file version to accept */
55 /* typedef for simplifying usage of long long type */
56 typedef long long int longlong_t
;
58 /*Pattern for the converter*/
59 static unsigned char pattern_euro
[]={0x07, 0x08, 0x1E, 0x10, 0x1E, 0x08, 0x07}; /* € */
60 static unsigned char pattern_home
[]={0x04, 0x0A, 0x11, 0x1F, 0x11, 0x11, 0x1F}; /* Home icon*/
62 /* 1 euro = ... (remenber 5 digits after the .)*/
63 static int currency
[12]={
64 655957, /*FRF France*/
65 195583, /*DEM Germany*/
66 1376030, /*ATS Austria*/
67 4033990, /*BEF Belgium*/
68 16638600, /*ESP Spain*/
69 594573, /*FIM Finland*/
70 78756, /*IEP Ireland*/
71 193627000, /*ITL Italy*/
72 4033990, /*LUF Luxemburg*/
73 220371, /*NLG Netherlands*/
74 20048200, /*PTE Portugal*/
75 34075100, /*GRD Greece*/
78 /* Number of digit of the currency (for the display) */
79 static int nb_digit
[12]={
89 2, /*NLG Netherlands*/
94 /* max euro to have home currency */
95 static longlong_t max_euro
[12]={
96 99999999000LL, /*FRF France 999 999.99 */
97 99999999000LL, /*DEM Germany 999 999.99 */
98 99999999000LL, /*ATS Austria 999 999.99 */
99 99999999000LL, /*BEF Belgium 999 999.99 */
100 99999999000LL, /*ESP Spain 99 999 999 */
101 99999999000LL, /*FIM Finland 999 999.99 */
102 99999999000LL, /*IEP Ireland 999 999.99 */
103 51645690000LL, /*ITL Italy 999 999 999 */
104 99999999000LL, /*LUF Luxemburg 999 999.99 */
105 99999999000LL, /*NLG Netherlands 999 999.99 */
106 99999999000LL, /*PTE Portugal 99 999 999 */
107 29347028000LL /*GRD Greece 99 999 999 */
110 /* max home to have euro currency */
111 /* 92233720300000 Limitation due to the max capacity of long long (2^63)*/
112 static longlong_t max_curr
[12]={
113 99999999000LL, /*FRF France 152449.02 */
114 99999999000LL, /*DEM Germany 511291.88 */
115 99999999000LL, /*ATS Austria 72672.83 */
116 99999999000LL, /*BEF Belgium 24789.35 */
117 92233720300000LL,/*ESP Spain 5543358.23 */
118 99999999000LL, /*FIM Finland 168187.92 */
119 9999999900LL, /*IEP Ireland 1269744.51 exact value=1269738.07 */
120 92233720300000LL,/*ITL Italy 476347.41 */
121 99999999000LL, /*LUF Luxemburg 24789.35 */
122 99999999000LL, /*NLG Netherlands 453780.21 */
123 92233720300000LL,/*PTE Portugal 4600598.57 */
124 92233720300000LL /*GRD Greece 2706777.69 */
127 static unsigned char *abbrev_str
[12] = {
128 "...FRF...", /*France*/
129 "...DEM...", /*Germany*/
130 "...ATS...", /*Austria*/
131 "...BEF...", /*Belgium*/
132 "...ESP...", /*Spain*/
133 "...FIM...", /*Finland*/
134 "...IEP...", /*Ireland*/
135 "...ITL...", /*Italy*/
136 "...LUF...", /*Luxemburg*/
137 "...NLG...", /*Netherlands*/
138 "...PTE...", /*Portugal*/
139 "...GRD..." /*Greece*/
143 static unsigned long heuro
,hhome
; /*Handles for the new patterns*/
145 static char *currency_str
[12] = {
161 static int country
; /*Country selected*/
162 static int cur_pos
; /*Cursor position*/
163 static longlong_t inc
;
165 /* Persistent settings */
166 static struct configdata config
[] = {
167 { TYPE_ENUM
, 0, 12, { .int_p
= &country
}, "country", currency_str
}
171 /* 64bits*64 bits with 5 digits after the . */
172 static longlong_t
mymul(longlong_t a
, longlong_t b
)
174 return((a
*b
)/100000LL);
178 /* 64bits/64 bits with 5 digits after the . */
179 static longlong_t
mydiv(longlong_t a
, longlong_t b
)
181 return((a
*100000LL)/b
);
185 /* 123.45=12345000 split => i=123 f=45000*/
186 static void split(longlong_t v
, longlong_t
* i
, longlong_t
* f
)
192 (*f
)=(v
-(t
*100000LL));
197 static longlong_t
pow10(int n
)
209 /* round the i.f at n digit after the . */
210 static void round(longlong_t
* i
, longlong_t
* f
, int n
)
221 (*f
)=((*f
)/(int)pow10(5-n
))+add
;
236 /* Display the imput and the result
237 pos: false : first line
240 static void display(longlong_t euro
, longlong_t home
, bool pos
)
243 unsigned char str
[20];
244 unsigned char s1
[20];
245 unsigned char s2
[20];
248 { /*Edit the second line*/
249 rb
->strcpy(s1
," %6d.%02d");
250 if (nb_digit
[country
]==2)
251 rb
->strcpy(s2
,"\xee\x84\x90%06d.%02d");
253 rb
->strcpy(s2
,"\xee\x84\x90%09d");
257 rb
->strcpy(s1
,"\xee\x84\x90%06d.%02d");
258 if (nb_digit
[country
]==2)
259 rb
->strcpy(s2
," %6d.%02d");
261 rb
->strcpy(s2
," %9d");
264 rb
->lcd_remove_cursor();
266 rb
->lcd_putc(0,0,heuro
);
270 rb
->snprintf(str
,sizeof(str
),s1
,(int)i
,(int)f
);
274 rb
->lcd_puts(1,0,str
);
275 rb
->lcd_put_cursor(10-cur_pos
,0,0x5F);
278 rb
->lcd_puts_scroll(1,0,str
);
281 rb
->lcd_putc(0,1,hhome
);
284 round(&i
,&f
,nb_digit
[country
]);
285 rb
->snprintf(str
,sizeof(str
),s2
,(int)i
,(int)f
);
288 rb
->lcd_puts(1,1,str
);
289 rb
->lcd_put_cursor(10-cur_pos
,1,0x5F);
292 rb
->lcd_puts_scroll(1,1,str
);
298 /* Show country Abbreviation */
299 static void show_abbrev(void)
301 rb
->splash(HZ
*3/4,abbrev_str
[country
]);
305 /* Save the config to disk */
306 static void save_config(void)
308 configfile_save(cfg_filename
, config
, 1, CFGFILE_VERSION
);
312 /* Load the config from disk */
313 static void load_config(void)
315 configfile_load(cfg_filename
, config
, 1, CFGFILE_MINVERSION
);
320 static void currency_menu(void)
324 rb
->lcd_clear_display();
327 rb
->lcd_puts(0,0,"Currency:");
328 rb
->lcd_puts(0,1,currency_str
[c
]);
330 switch (rb
->button_get(true))
332 case BUTTON_RIGHT
|BUTTON_REL
:
337 case BUTTON_LEFT
|BUTTON_REL
:
342 case BUTTON_PLAY
|BUTTON_REL
:
347 case BUTTON_STOP
|BUTTON_REL
:
354 /* Display the choice menu. */
355 static int euro_menu(void)
362 rb
->lcd_clear_display();
363 rb
->lcd_puts(0,0," Currency");
364 rb
->lcd_puts(0,1," Exit");
365 rb
->lcd_putc(0,c
,0xe110);
368 switch (rb
->button_get(true))
370 case BUTTON_RIGHT
|BUTTON_REL
:
373 case BUTTON_LEFT
|BUTTON_REL
:
376 case BUTTON_PLAY
|BUTTON_REL
:
382 case BUTTON_STOP
|BUTTON_REL
:
389 /* Call when the program end */
390 static void euro_exit(void *parameter
)
394 //Restore the old pattern (i don't find another way to do this. An idea?)
395 rb
->lcd_unlock_pattern(heuro
);
396 rb
->lcd_unlock_pattern(hhome
);
399 rb
->lcd_clear_display();
404 /* this is the plugin entry point */
405 enum plugin_status
plugin_start(const void* parameter
)
408 longlong_t e
,h
,old_e
,old_h
;
411 /* if you don't use the parameter, you can do like
412 this to avoid the compiler warning about it */
415 /*Get the pattern handle*/
416 heuro
=rb
->lcd_get_locked_pattern();
417 hhome
=rb
->lcd_get_locked_pattern();
418 rb
->lcd_define_pattern(heuro
, pattern_euro
);
419 rb
->lcd_define_pattern(hhome
, pattern_home
);
431 /*Empty the event queue*/
432 rb
->button_clear_queue();
441 button
= rb
->button_get(true);
444 case BUTTON_MENU
|BUTTON_REL
:
453 if (e
>max_euro
[country
])
459 if (h
>max_curr
[country
])
461 if (nb_digit
[country
]==2)
470 case BUTTON_ON
| BUTTON_PLAY
:
473 case BUTTON_ON
| BUTTON_REL
:
484 if (nb_digit
[country
]==2)
492 case BUTTON_STOP
|BUTTON_REL
:
501 inc
=pow10(3+cur_pos
-1);
503 inc
=pow10(3+cur_pos
);
509 if (nb_digit
[country
]==2)
514 inc
=pow10(3+cur_pos
-1);
516 inc
=pow10(3+cur_pos
);
519 inc
=pow10(5+cur_pos
);
524 case BUTTON_PLAY
|BUTTON_REL
:
533 inc
=pow10(3+cur_pos
-1);
535 inc
=pow10(3+cur_pos
);
541 if (nb_digit
[country
]==2)
546 inc
=pow10(3+cur_pos
-1);
548 inc
=pow10(3+cur_pos
);
551 inc
=pow10(5+cur_pos
);
555 case BUTTON_LEFT
|BUTTON_REL
:
556 case BUTTON_LEFT
|BUTTON_REPEAT
:
571 case BUTTON_RIGHT
|BUTTON_REL
:
572 case BUTTON_RIGHT
|BUTTON_REPEAT
:
578 if (e
>max_euro
[country
])
584 if (h
>max_curr
[country
])
590 if (rb
->default_event_handler_ex(button
, euro_exit
, NULL
)
591 == SYS_USB_CONNECTED
)
592 return PLUGIN_USB_CONNECTED
;
596 if (!pos
) /*Euro>home*/
597 h
=mymul(e
,currency
[country
]);
599 e
=mydiv(h
,currency
[country
]);