3 #include "lib/configfile.h"
4 #include "lib/oldmenuapi.h"
6 #include "mpegplayer.h"
7 #include "mpeg_settings.h"
9 struct mpeg_settings settings
;
11 #define THUMB_DELAY (75*HZ/100)
13 /* button definitions */
14 #if (CONFIG_KEYPAD == IRIVER_H100_PAD) || \
15 (CONFIG_KEYPAD == IRIVER_H300_PAD)
16 #define MPEG_START_TIME_SELECT BUTTON_ON
17 #define MPEG_START_TIME_RIGHT BUTTON_RIGHT
18 #define MPEG_START_TIME_LEFT BUTTON_LEFT
19 #define MPEG_START_TIME_UP BUTTON_UP
20 #define MPEG_START_TIME_DOWN BUTTON_DOWN
21 #define MPEG_START_TIME_EXIT BUTTON_OFF
23 #elif (CONFIG_KEYPAD == IAUDIO_X5M5_PAD)
24 #define MPEG_START_TIME_SELECT BUTTON_PLAY
25 #define MPEG_START_TIME_RIGHT BUTTON_RIGHT
26 #define MPEG_START_TIME_LEFT BUTTON_LEFT
27 #define MPEG_START_TIME_UP BUTTON_UP
28 #define MPEG_START_TIME_DOWN BUTTON_DOWN
29 #define MPEG_START_TIME_EXIT BUTTON_POWER
31 #elif (CONFIG_KEYPAD == IPOD_4G_PAD) || \
32 (CONFIG_KEYPAD == IPOD_3G_PAD) || \
33 (CONFIG_KEYPAD == IPOD_1G2G_PAD)
34 #define MPEG_START_TIME_SELECT BUTTON_SELECT
35 #define MPEG_START_TIME_RIGHT BUTTON_RIGHT
36 #define MPEG_START_TIME_LEFT BUTTON_LEFT
37 #define MPEG_START_TIME_UP BUTTON_SCROLL_FWD
38 #define MPEG_START_TIME_DOWN BUTTON_SCROLL_BACK
39 #define MPEG_START_TIME_EXIT BUTTON_MENU
41 #elif CONFIG_KEYPAD == GIGABEAT_PAD
42 #define MPEG_START_TIME_SELECT BUTTON_SELECT
43 #define MPEG_START_TIME_LEFT BUTTON_LEFT
44 #define MPEG_START_TIME_RIGHT BUTTON_RIGHT
45 #define MPEG_START_TIME_UP BUTTON_UP
46 #define MPEG_START_TIME_DOWN BUTTON_DOWN
47 #define MPEG_START_TIME_SCROLL_DOWN BUTTON_VOL_DOWN
48 #define MPEG_START_TIME_SCROLL_UP BUTTON_VOL_UP
49 #define MPEG_START_TIME_EXIT BUTTON_POWER
51 #define MPEG_START_TIME_RC_SELECT (BUTTON_RC_PLAY | BUTTON_REL)
52 #define MPEG_START_TIME_RC_LEFT BUTTON_RC_REW
53 #define MPEG_START_TIME_RC_RIGHT BUTTON_RC_FF
54 #define MPEG_START_TIME_RC_UP BUTTON_RC_VOL_UP
55 #define MPEG_START_TIME_RC_DOWN BUTTON_RC_VOL_DOWN
56 #define MPEG_START_TIME_RC_EXIT (BUTTON_RC_PLAY | BUTTON_REPEAT)
58 #elif CONFIG_KEYPAD == GIGABEAT_S_PAD
59 #define MPEG_START_TIME_SELECT BUTTON_SELECT
60 #define MPEG_START_TIME_LEFT BUTTON_LEFT
61 #define MPEG_START_TIME_RIGHT BUTTON_RIGHT
62 #define MPEG_START_TIME_UP BUTTON_UP
63 #define MPEG_START_TIME_DOWN BUTTON_DOWN
64 #define MPEG_START_TIME_SCROLL_DOWN BUTTON_VOL_DOWN
65 #define MPEG_START_TIME_SCROLL_UP BUTTON_VOL_UP
66 #define MPEG_START_TIME_EXIT BUTTON_POWER
68 #elif CONFIG_KEYPAD == IRIVER_H10_PAD
69 #define MPEG_START_TIME_SELECT BUTTON_PLAY
70 #define MPEG_START_TIME_LEFT BUTTON_LEFT
71 #define MPEG_START_TIME_RIGHT BUTTON_RIGHT
72 #define MPEG_START_TIME_UP BUTTON_SCROLL_UP
73 #define MPEG_START_TIME_DOWN BUTTON_SCROLL_DOWN
74 #define MPEG_START_TIME_EXIT BUTTON_POWER
76 #elif (CONFIG_KEYPAD == SANSA_E200_PAD)
77 #define MPEG_START_TIME_SELECT BUTTON_SELECT
78 #define MPEG_START_TIME_SCROLL_UP BUTTON_SCROLL_BACK
79 #define MPEG_START_TIME_SCROLL_DOWN BUTTON_SCROLL_FWD
80 #define MPEG_START_TIME_LEFT BUTTON_LEFT
81 #define MPEG_START_TIME_RIGHT BUTTON_RIGHT
82 #define MPEG_START_TIME_UP BUTTON_UP
83 #define MPEG_START_TIME_DOWN BUTTON_DOWN
84 #define MPEG_START_TIME_EXIT BUTTON_POWER
86 #elif (CONFIG_KEYPAD == SANSA_C200_PAD)
87 #define MPEG_START_TIME_SELECT BUTTON_SELECT
88 #define MPEG_START_TIME_SCROLL_UP BUTTON_VOL_UP
89 #define MPEG_START_TIME_SCROLL_DOWN BUTTON_VOL_DOWN
90 #define MPEG_START_TIME_LEFT BUTTON_LEFT
91 #define MPEG_START_TIME_RIGHT BUTTON_RIGHT
92 #define MPEG_START_TIME_UP BUTTON_UP
93 #define MPEG_START_TIME_DOWN BUTTON_DOWN
94 #define MPEG_START_TIME_EXIT BUTTON_POWER
96 #elif CONFIG_KEYPAD == MROBE500_PAD
97 #define MPEG_START_TIME_SELECT BUTTON_RC_HEART
98 #define MPEG_START_TIME_SCROLL_UP BUTTON_RC_VOL_UP
99 #define MPEG_START_TIME_SCROLL_DOWN BUTTON_RC_VOL_DOWN
100 #define MPEG_START_TIME_LEFT BUTTON_LEFT
101 #define MPEG_START_TIME_RIGHT BUTTON_RIGHT
102 #define MPEG_START_TIME_UP BUTTON_RC_PLAY
103 #define MPEG_START_TIME_DOWN BUTTON_RC_DOWN
104 #define MPEG_START_TIME_EXIT BUTTON_POWER
106 #elif CONFIG_KEYPAD == MROBE100_PAD
107 #define MPEG_START_TIME_SELECT BUTTON_SELECT
108 #define MPEG_START_TIME_LEFT BUTTON_LEFT
109 #define MPEG_START_TIME_RIGHT BUTTON_RIGHT
110 #define MPEG_START_TIME_UP BUTTON_UP
111 #define MPEG_START_TIME_DOWN BUTTON_DOWN
112 #define MPEG_START_TIME_SCROLL_DOWN BUTTON_MENU
113 #define MPEG_START_TIME_SCROLL_UP BUTTON_PLAY
114 #define MPEG_START_TIME_EXIT BUTTON_POWER
116 #elif CONFIG_KEYPAD == IAUDIO_M3_PAD
117 #define MPEG_START_TIME_SELECT BUTTON_RC_PLAY
118 #define MPEG_START_TIME_LEFT BUTTON_RC_REW
119 #define MPEG_START_TIME_RIGHT BUTTON_RC_FF
120 #define MPEG_START_TIME_UP BUTTON_RC_VOL_UP
121 #define MPEG_START_TIME_DOWN BUTTON_RC_VOL_DOWN
122 #define MPEG_START_TIME_EXIT BUTTON_RC_REC
124 #elif CONFIG_KEYPAD == COWOND2_PAD
125 #define MPEG_START_TIME_EXIT BUTTON_POWER
128 #error No keymap defined!
132 #ifndef MPEG_START_TIME_SELECT
133 #define MPEG_START_TIME_SELECT BUTTON_CENTER
135 #ifndef MPEG_START_TIME_SCROLL_UP
136 #define MPEG_START_TIME_SCROLL_UP BUTTON_TOPRIGHT
138 #ifndef MPEG_START_TIME_SCROLL_DOWN
139 #define MPEG_START_TIME_SCROLL_DOWN BUTTON_TOPLEFT
141 #ifndef MPEG_START_TIME_LEFT
142 #define MPEG_START_TIME_LEFT BUTTON_MIDLEFT
144 #ifndef MPEG_START_TIME_RIGHT
145 #define MPEG_START_TIME_RIGHT BUTTON_MIDRIGHT
147 #ifndef MPEG_START_TIME_UP
148 #define MPEG_START_TIME_UP BUTTON_TOPMIDDLE
150 #ifndef MPEG_START_TIME_DOWN
151 #define MPEG_START_TIME_DOWN BUTTON_BOTTOMMIDDLE
153 #ifndef MPEG_START_TIME_EXIT
154 #define MPEG_START_TIME_EXIT BUTTON_TOPLEFT
158 static struct configdata config
[] =
160 {TYPE_INT
, 0, 2, &settings
.showfps
, "Show FPS", NULL
, NULL
},
161 {TYPE_INT
, 0, 2, &settings
.limitfps
, "Limit FPS", NULL
, NULL
},
162 {TYPE_INT
, 0, 2, &settings
.skipframes
, "Skip frames", NULL
, NULL
},
163 {TYPE_INT
, 0, INT_MAX
, &settings
.resume_count
, "Resume count",
165 {TYPE_INT
, 0, MPEG_RESUME_NUM_OPTIONS
, &settings
.resume_options
,
166 "Resume options", NULL
, NULL
},
167 #if defined(TOSHIBA_GIGABEAT_F) || defined(SANSA_E200) || defined(SANSA_C200)
168 {TYPE_INT
, 0, INT_MAX
, &settings
.displayoptions
, "Display options",
171 {TYPE_INT
, 0, 2, &settings
.tone_controls
, "Tone controls", NULL
, NULL
},
172 {TYPE_INT
, 0, 2, &settings
.channel_modes
, "Channel modes", NULL
, NULL
},
173 {TYPE_INT
, 0, 2, &settings
.crossfeed
, "Crossfeed", NULL
, NULL
},
174 {TYPE_INT
, 0, 2, &settings
.equalizer
, "Equalizer", NULL
, NULL
},
175 {TYPE_INT
, 0, 2, &settings
.dithering
, "Dithering", NULL
, NULL
},
176 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
177 {TYPE_INT
, -1, INT_MAX
, &settings
.backlight_brightness
,
178 "Backlight brightness", NULL
, NULL
},
182 static const struct opt_items noyes
[2] = {
187 static const struct opt_items enabledisable
[2] = {
192 static const struct opt_items globaloff
[2] = {
194 { "Use sound setting", -1 },
197 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
198 #define BACKLIGHT_OPTION_DEFAULT "Use setting"
201 static long mpeg_menu_sysevent_id
;
203 void mpeg_menu_sysevent_clear(void)
205 mpeg_menu_sysevent_id
= 0;
208 int mpeg_menu_sysevent_callback(int btn
, int menu
)
212 case SYS_USB_CONNECTED
:
214 mpeg_menu_sysevent_id
= btn
;
215 return ACTION_STD_CANCEL
;
222 long mpeg_menu_sysevent(void)
224 return mpeg_menu_sysevent_id
;
227 void mpeg_menu_sysevent_handle(void)
229 long id
= mpeg_menu_sysevent();
231 rb
->default_event_handler(id
);
234 static void format_menu_item(struct menu_item
*item
, int bufsize
,
235 const char *fmt
, ...)
240 rb
->vsnprintf(item
->desc
, bufsize
, fmt
, ap
);
245 static bool mpeg_set_option(const char* string
,
247 enum optiontype type
,
248 const struct opt_items
* options
,
250 void (*function
)(int))
252 mpeg_menu_sysevent_clear();
254 /* This eats SYS_POWEROFF - :\ */
255 bool usb
= rb
->set_option(string
, variable
, type
, options
, numoptions
,
259 mpeg_menu_sysevent_id
= ACTION_STD_CANCEL
;
264 #ifdef HAVE_BACKLIGHT_BRIGHTNESS /* Only used for this atm */
265 static bool mpeg_set_int(const char *string
, const char *unit
,
266 int voice_unit
, const int *variable
,
267 void (*function
)(int), int step
,
270 void (*formatter
)(char*, size_t, int, const char*))
272 mpeg_menu_sysevent_clear();
274 bool usb
= rb
->set_int(string
, unit
, voice_unit
, variable
, function
,
275 step
, min
, max
, formatter
);
278 mpeg_menu_sysevent_id
= ACTION_STD_CANCEL
;
282 #endif /* HAVE_BACKLIGHT_BRIGHTNESS */
284 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
285 void mpeg_backlight_update_brightness(int value
)
289 value
+= MIN_BRIGHTNESS_SETTING
;
290 backlight_brightness_set(rb
, value
);
294 backlight_brightness_use_setting(rb
);
298 static void backlight_brightness_function(int value
)
300 mpeg_backlight_update_brightness(value
);
303 static void backlight_brightness_formatter(char *buf
, size_t length
,
304 int value
, const char *input
)
307 rb
->strncpy(buf
, BACKLIGHT_OPTION_DEFAULT
, length
);
309 rb
->snprintf(buf
, length
, "%d", value
+ MIN_BRIGHTNESS_SETTING
);
313 #endif /* HAVE_BACKLIGHT_BRIGHTNESS */
315 /* Sync a particular audio setting to global or mpegplayer forced off */
316 static void sync_audio_setting(int setting
, bool global
)
322 case MPEG_AUDIO_TONE_CONTROLS
:
323 if (global
|| settings
.tone_controls
)
325 val0
= rb
->global_settings
->bass
;
326 val1
= rb
->global_settings
->treble
;
330 val0
= rb
->sound_default(SOUND_BASS
);
331 val1
= rb
->sound_default(SOUND_TREBLE
);
333 rb
->sound_set(SOUND_BASS
, val0
);
334 rb
->sound_set(SOUND_TREBLE
, val1
);
337 case MPEG_AUDIO_CHANNEL_MODES
:
338 val0
= (global
|| settings
.channel_modes
) ?
339 rb
->global_settings
->channel_config
:
341 rb
->sound_set(SOUND_CHANNELS
, val0
);
344 case MPEG_AUDIO_CROSSFEED
:
345 rb
->dsp_set_crossfeed((global
|| settings
.crossfeed
) ?
346 rb
->global_settings
->crossfeed
: false);
349 case MPEG_AUDIO_EQUALIZER
:
350 rb
->dsp_set_eq((global
|| settings
.equalizer
) ?
351 rb
->global_settings
->eq_enabled
: false);
354 case MPEG_AUDIO_DITHERING
:
355 rb
->dsp_dither_enable((global
|| settings
.dithering
) ?
356 rb
->global_settings
->dithering_enabled
: false);
361 /* Sync all audio settings to global or mpegplayer forced off */
362 static void sync_audio_settings(bool global
)
364 static const int setting_index
[] =
366 MPEG_AUDIO_TONE_CONTROLS
,
367 MPEG_AUDIO_CHANNEL_MODES
,
368 MPEG_AUDIO_CROSSFEED
,
369 MPEG_AUDIO_EQUALIZER
,
370 MPEG_AUDIO_DITHERING
,
374 for (i
= 0; i
< ARRAYLEN(setting_index
); i
++)
376 sync_audio_setting(setting_index
[i
], global
);
380 #ifndef HAVE_LCD_COLOR
381 /* Cheapo splash implementation for the grey surface */
382 static void grey_splash(int ticks
, const unsigned char *fmt
, ...)
384 unsigned char buffer
[256];
391 rb
->vsnprintf(buffer
, sizeof (buffer
), fmt
, ap
);
395 grey_getstringsize(buffer
, &w
, &h
);
397 oldfg
= grey_get_foreground();
398 oldmode
= grey_get_drawmode();
400 grey_set_drawmode(DRMODE_FG
);
401 grey_set_foreground(GREY_LIGHTGRAY
);
403 x
= (LCD_WIDTH
- w
) / 2;
404 y
= (LCD_HEIGHT
- h
) / 2;
406 grey_fillrect(x
- 1, y
- 1, w
+ 2, h
+ 2);
408 grey_set_foreground(GREY_BLACK
);
410 grey_putsxy(x
, y
, buffer
);
411 grey_drawrect(x
- 2, y
- 2, w
+ 4, h
+ 4);
413 grey_set_foreground(oldfg
);
414 grey_set_drawmode(oldmode
);
421 #endif /* !HAVE_LCD_COLOR */
423 static void show_loading(struct vo_rect
*rc
)
425 int oldmode
= lcd_(get_drawmode
)();
426 lcd_(set_drawmode
)(DRMODE_SOLID
| DRMODE_INVERSEVID
);
427 lcd_(fillrect
)(rc
->l
-1, rc
->t
-1, rc
->r
- rc
->l
+ 2, rc
->b
- rc
->t
+ 2);
428 lcd_(set_drawmode
)(oldmode
);
429 lcd_(splash
)(0, "Loading...");
432 static void draw_slider(uint32_t range
, uint32_t pos
, struct vo_rect
*rc
)
434 #define SLIDER_WIDTH (LCD_WIDTH-SLIDER_LMARGIN-SLIDER_RMARGIN)
435 #define SLIDER_X SLIDER_LMARGIN
436 #define SLIDER_Y (LCD_HEIGHT-SLIDER_HEIGHT-SLIDER_BMARGIN)
437 #define SLIDER_HEIGHT 8
438 #define SLIDER_TEXTMARGIN 1
439 #define SLIDER_LMARGIN 1
440 #define SLIDER_RMARGIN 1
441 #define SLIDER_TMARGIN 1
442 #define SLIDER_BMARGIN 1
443 #define SCREEN_MARGIN 1
447 int text_w
, text_h
, text_y
;
449 /* Put positition on left */
450 ts_to_hms(pos
, &hms
);
451 hms_format(str
, sizeof(str
), &hms
);
452 lcd_(getstringsize
)(str
, NULL
, &text_h
);
453 text_y
= SLIDER_Y
- SLIDER_TEXTMARGIN
- text_h
;
457 int oldmode
= lcd_(get_drawmode
)();
458 lcd_(set_drawmode
)(DRMODE_BG
| DRMODE_INVERSEVID
);
459 lcd_(fillrect
)(SLIDER_X
, text_y
, SLIDER_WIDTH
,
460 LCD_HEIGHT
- SLIDER_BMARGIN
- text_y
462 lcd_(set_drawmode
)(oldmode
);
464 lcd_(putsxy
)(SLIDER_X
, text_y
, str
);
466 /* Put duration on right */
467 ts_to_hms(range
, &hms
);
468 hms_format(str
, sizeof(str
), &hms
);
469 lcd_(getstringsize
)(str
, &text_w
, NULL
);
471 lcd_(putsxy
)(SLIDER_X
+ SLIDER_WIDTH
- text_w
, text_y
, str
);
474 lcd_(drawrect
)(SLIDER_X
, SLIDER_Y
, SLIDER_WIDTH
, SLIDER_HEIGHT
);
475 lcd_(fillrect
)(SLIDER_X
, SLIDER_Y
,
476 muldiv_uint32(pos
, SLIDER_WIDTH
, range
),
480 lcd_(update_rect
)(SLIDER_X
, text_y
- SLIDER_TMARGIN
, SLIDER_WIDTH
,
481 LCD_HEIGHT
- SLIDER_BMARGIN
- text_y
+ SLIDER_TEXTMARGIN
);
485 /* Just return slider rectangle */
487 rc
->t
= text_y
- SLIDER_TMARGIN
;
488 rc
->r
= rc
->l
+ SLIDER_WIDTH
;
489 rc
->b
= rc
->t
+ LCD_HEIGHT
- SLIDER_BMARGIN
- text_y
;
493 static bool display_thumb_image(const struct vo_rect
*rc
)
495 if (!stream_display_thumb(rc
))
497 lcd_(splash
)(0, "Frame not available");
501 /* Draw a raised border around the frame */
502 int oldcolor
= lcd_(get_foreground
)();
503 lcd_(set_foreground
)(DRAW_LIGHTGRAY
);
505 lcd_(hline
)(rc
->l
-1, rc
->r
-1, rc
->t
-1);
506 lcd_(vline
)(rc
->l
-1, rc
->t
, rc
->b
-1);
508 lcd_(set_foreground
)(DRAW_DARKGRAY
);
510 lcd_(hline
)(rc
->l
-1, rc
->r
, rc
->b
);
511 lcd_(vline
)(rc
->r
, rc
->t
-1, rc
->b
);
513 lcd_(set_foreground
)(oldcolor
);
515 lcd_(update_rect
)(rc
->l
-1, rc
->t
-1, rc
->r
- rc
->l
+ 2, 1);
516 lcd_(update_rect
)(rc
->l
-1, rc
->t
, 1, rc
->b
- rc
->t
);
517 lcd_(update_rect
)(rc
->l
-1, rc
->b
, rc
->r
- rc
->l
+ 2, 1);
518 lcd_(update_rect
)(rc
->r
, rc
->t
, 1, rc
->b
- rc
->t
);
523 /* Add an amount to the specified time - with saturation */
524 static uint32_t increment_time(uint32_t val
, int32_t amount
, uint32_t range
)
528 uint32_t off
= -amount
;
529 if (range
> off
&& val
>= off
)
536 uint32_t off
= amount
;
537 if (range
> off
&& val
<= range
- off
)
546 #ifdef HAVE_LCD_ENABLE
547 static void get_start_time_lcd_enable_hook(void)
549 rb
->queue_post(rb
->button_queue
, LCD_ENABLE_EVENT_0
, 0);
551 #endif /* HAVE_LCD_ENABLE */
553 static int get_start_time(uint32_t duration
)
556 int tmo
= TIMEOUT_NOBLOCK
;
557 uint32_t resume_time
= settings
.resume_time
;
558 struct vo_rect rc_vid
, rc_bound
;
559 uint32_t aspect_vid
, aspect_bound
;
561 enum state_enum slider_state
= state0
;
563 lcd_(clear_display
)();
566 #ifdef HAVE_LCD_ENABLE
567 rb
->lcd_set_enable_hook(get_start_time_lcd_enable_hook
);
570 draw_slider(0, 100, &rc_bound
);
571 rc_bound
.b
= rc_bound
.t
- SLIDER_TMARGIN
;
572 rc_bound
.t
= SCREEN_MARGIN
;
574 DEBUGF("rc_bound: %d, %d, %d, %d\n", rc_bound
.l
, rc_bound
.t
,
575 rc_bound
.r
, rc_bound
.b
);
577 rc_vid
.l
= rc_vid
.t
= 0;
578 if (!stream_vo_get_size((struct vo_ext
*)&rc_vid
.r
))
580 /* Can't get size - fill whole thing */
581 rc_vid
.r
= rc_bound
.r
- rc_bound
.l
;
582 rc_vid
.b
= rc_bound
.b
- rc_bound
.t
;
585 /* Get aspect ratio of bounding rectangle and video in u16.16 */
586 aspect_bound
= ((rc_bound
.r
- rc_bound
.l
) << 16) /
587 (rc_bound
.b
- rc_bound
.t
);
589 DEBUGF("aspect_bound: %u.%02u\n", (unsigned)(aspect_bound
>> 16),
590 (unsigned)(100*(aspect_bound
& 0xffff) >> 16));
592 aspect_vid
= (rc_vid
.r
<< 16) / rc_vid
.b
;
594 DEBUGF("aspect_vid: %u.%02u\n", (unsigned)(aspect_vid
>> 16),
595 (unsigned)(100*(aspect_vid
& 0xffff) >> 16));
597 if (aspect_vid
>= aspect_bound
)
599 /* Video proportionally wider than or same as bounding rectangle */
600 if (rc_vid
.r
> rc_bound
.r
- rc_bound
.l
)
602 rc_vid
.r
= rc_bound
.r
- rc_bound
.l
;
603 rc_vid
.b
= (rc_vid
.r
<< 16) / aspect_vid
;
605 /* else already fits */
609 /* Video proportionally narrower than bounding rectangle */
610 if (rc_vid
.b
> rc_bound
.b
- rc_bound
.t
)
612 rc_vid
.b
= rc_bound
.b
- rc_bound
.t
;
613 rc_vid
.r
= (aspect_vid
* rc_vid
.b
) >> 16;
615 /* else already fits */
618 /* Even width and height >= 2 */
619 rc_vid
.r
= (rc_vid
.r
< 2) ? 2 : (rc_vid
.r
& ~1);
620 rc_vid
.b
= (rc_vid
.b
< 2) ? 2 : (rc_vid
.b
& ~1);
622 /* Center display in bounding rectangle */
623 rc_vid
.l
= ((rc_bound
.l
+ rc_bound
.r
) - rc_vid
.r
) / 2;
624 rc_vid
.r
+= rc_vid
.l
;
626 rc_vid
.t
= ((rc_bound
.t
+ rc_bound
.b
) - rc_vid
.b
) / 2;
627 rc_vid
.b
+= rc_vid
.t
;
629 DEBUGF("rc_vid: %d, %d, %d, %d\n", rc_vid
.l
, rc_vid
.t
,
632 #ifndef HAVE_LCD_COLOR
633 stream_gray_show(true);
636 while (slider_state
< state9
)
638 mpeg_menu_sysevent_clear();
639 button
= tmo
== TIMEOUT_BLOCK
?
640 rb
->button_get(true) : rb
->button_get_w_tmo(tmo
);
642 button
= mpeg_menu_sysevent_callback(button
, -1);
649 /* Coarse (1 minute) control */
650 case MPEG_START_TIME_DOWN
:
651 case MPEG_START_TIME_DOWN
| BUTTON_REPEAT
:
652 #ifdef MPEG_START_TIME_RC_DOWN
653 case MPEG_START_TIME_RC_DOWN
:
654 case MPEG_START_TIME_RC_DOWN
| BUTTON_REPEAT
:
656 resume_time
= increment_time(resume_time
, -60*TS_SECOND
, duration
);
657 slider_state
= state0
;
660 case MPEG_START_TIME_UP
:
661 case MPEG_START_TIME_UP
| BUTTON_REPEAT
:
662 #ifdef MPEG_START_TIME_RC_UP
663 case MPEG_START_TIME_RC_UP
:
664 case MPEG_START_TIME_RC_UP
| BUTTON_REPEAT
:
666 resume_time
= increment_time(resume_time
, 60*TS_SECOND
, duration
);
667 slider_state
= state0
;
670 /* Fine (1 second) control */
671 case MPEG_START_TIME_LEFT
:
672 case MPEG_START_TIME_LEFT
| BUTTON_REPEAT
:
673 #ifdef MPEG_START_TIME_RC_LEFT
674 case MPEG_START_TIME_RC_LEFT
:
675 case MPEG_START_TIME_RC_LEFT
| BUTTON_REPEAT
:
677 #ifdef MPEG_START_TIME_SCROLL_UP
678 case MPEG_START_TIME_SCROLL_UP
:
679 case MPEG_START_TIME_SCROLL_UP
| BUTTON_REPEAT
:
681 resume_time
= increment_time(resume_time
, -TS_SECOND
, duration
);
682 slider_state
= state0
;
685 case MPEG_START_TIME_RIGHT
:
686 case MPEG_START_TIME_RIGHT
| BUTTON_REPEAT
:
687 #ifdef MPEG_START_TIME_RC_RIGHT
688 case MPEG_START_TIME_RC_RIGHT
:
689 case MPEG_START_TIME_RC_RIGHT
| BUTTON_REPEAT
:
691 #ifdef MPEG_START_TIME_SCROLL_DOWN
692 case MPEG_START_TIME_SCROLL_DOWN
:
693 case MPEG_START_TIME_SCROLL_DOWN
| BUTTON_REPEAT
:
695 resume_time
= increment_time(resume_time
, TS_SECOND
, duration
);
696 slider_state
= state0
;
699 case MPEG_START_TIME_SELECT
:
700 #ifdef MPEG_START_TIME_RC_SELECT
701 case MPEG_START_TIME_RC_SELECT
:
703 settings
.resume_time
= resume_time
;
704 button
= MPEG_START_SEEK
;
705 slider_state
= state9
;
708 case MPEG_START_TIME_EXIT
:
709 #ifdef MPEG_START_TIME_RC_EXIT
710 case MPEG_START_TIME_RC_EXIT
:
712 button
= MPEG_START_EXIT
;
713 slider_state
= state9
;
716 case ACTION_STD_CANCEL
:
717 button
= MPEG_START_QUIT
;
718 slider_state
= state9
;
721 #ifdef HAVE_LCD_ENABLE
722 case LCD_ENABLE_EVENT_0
:
723 if (slider_state
== state2
)
724 display_thumb_image(&rc_vid
);
729 rb
->default_event_handler(button
);
734 switch (slider_state
)
738 stream_seek(resume_time
, SEEK_SET
);
739 show_loading(&rc_bound
);
740 draw_slider(duration
, resume_time
, NULL
);
741 slider_state
= state1
;
745 display_thumb_image(&rc_vid
);
746 slider_state
= state2
;
757 #ifdef HAVE_LCD_ENABLE
758 rb
->lcd_set_enable_hook(NULL
);
761 #ifndef HAVE_LCD_COLOR
762 stream_gray_show(false);
763 grey_clear_display();
772 static int show_start_menu(uint32_t duration
)
776 bool menu_quit
= false;
778 /* add the resume time to the menu display */
783 struct menu_item items
[] =
785 [MPEG_START_RESTART
] =
786 { "Play from beginning", NULL
},
787 [MPEG_START_RESUME
] =
788 { resume_str
, NULL
},
790 { "Set start time", NULL
},
791 [MPEG_START_SETTINGS
] =
792 { "Settings", NULL
},
794 { "Quit mpegplayer", NULL
},
797 ts_to_hms(settings
.resume_time
, &hms
);
798 hms_format(hms_str
, sizeof(hms_str
), &hms
);
799 format_menu_item(&items
[MPEG_START_RESUME
], sizeof (resume_str
),
800 "Resume at: %s", hms_str
);
802 menu_id
= menu_init(rb
, items
, ARRAYLEN(items
),
803 mpeg_menu_sysevent_callback
, NULL
, NULL
, NULL
);
805 rb
->button_clear_queue();
809 mpeg_menu_sysevent_clear();
810 result
= menu_show(menu_id
);
814 case MPEG_START_RESTART
:
815 settings
.resume_time
= 0;
819 case MPEG_START_RESUME
:
823 case MPEG_START_SEEK
:
824 if (!stream_can_seek())
826 rb
->splash(HZ
, "Unavailable");
830 result
= get_start_time(duration
);
832 if (result
!= MPEG_START_EXIT
)
836 case MPEG_START_SETTINGS
:
837 if (mpeg_menu(MPEG_MENU_HIDE_QUIT_ITEM
) != MPEG_MENU_QUIT
)
841 result
= MPEG_START_QUIT
;
846 if (mpeg_menu_sysevent() != 0)
848 result
= MPEG_START_QUIT
;
855 rb
->lcd_clear_display();
861 /* Return the desired resume action */
862 int mpeg_start_menu(uint32_t duration
)
864 mpeg_menu_sysevent_clear();
866 switch (settings
.resume_options
)
868 case MPEG_RESUME_MENU_IF_INCOMPLETE
:
869 if (!stream_can_seek() || settings
.resume_time
== 0)
871 case MPEG_RESUME_RESTART
:
872 settings
.resume_time
= 0;
873 return MPEG_START_RESTART
;
876 case MPEG_RESUME_MENU_ALWAYS
:
877 return show_start_menu(duration
);
878 case MPEG_RESUME_ALWAYS
:
879 return MPEG_START_SEEK
;
884 static void display_options(void)
888 bool menu_quit
= false;
890 static const struct menu_item items
[] = {
891 #if MPEG_OPTION_DITHERING_ENABLED
892 [MPEG_OPTION_DITHERING
] =
893 { "Dithering", NULL
},
895 [MPEG_OPTION_DISPLAY_FPS
] =
896 { "Display FPS", NULL
},
897 [MPEG_OPTION_LIMIT_FPS
] =
898 { "Limit FPS", NULL
},
899 [MPEG_OPTION_SKIP_FRAMES
] =
900 { "Skip frames", NULL
},
901 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
902 [MPEG_OPTION_BACKLIGHT_BRIGHTNESS
] =
903 { "Backlight brightness", NULL
},
907 menu_id
= menu_init(rb
, items
, ARRAYLEN(items
),
908 mpeg_menu_sysevent_callback
, NULL
, NULL
, NULL
);
910 rb
->button_clear_queue();
914 mpeg_menu_sysevent_clear();
915 result
= menu_show(menu_id
);
919 #if MPEG_OPTION_DITHERING_ENABLED
920 case MPEG_OPTION_DITHERING
:
921 result
= (settings
.displayoptions
& LCD_YUV_DITHER
) ? 1 : 0;
922 mpeg_set_option("Dithering", &result
, INT
, noyes
, 2, NULL
);
923 settings
.displayoptions
=
924 (settings
.displayoptions
& ~LCD_YUV_DITHER
)
925 | ((result
!= 0) ? LCD_YUV_DITHER
: 0);
926 rb
->lcd_yuv_set_options(settings
.displayoptions
);
928 #endif /* MPEG_OPTION_DITHERING_ENABLED */
930 case MPEG_OPTION_DISPLAY_FPS
:
931 mpeg_set_option("Display FPS", &settings
.showfps
, INT
,
935 case MPEG_OPTION_LIMIT_FPS
:
936 mpeg_set_option("Limit FPS", &settings
.limitfps
, INT
,
940 case MPEG_OPTION_SKIP_FRAMES
:
941 mpeg_set_option("Skip frames", &settings
.skipframes
, INT
,
945 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
946 case MPEG_OPTION_BACKLIGHT_BRIGHTNESS
:
947 result
= settings
.backlight_brightness
;
948 mpeg_backlight_update_brightness(result
);
949 mpeg_set_int("Backlight brightness", NULL
, -1, &result
,
950 backlight_brightness_function
, 1, -1,
951 MAX_BRIGHTNESS_SETTING
- MIN_BRIGHTNESS_SETTING
,
952 backlight_brightness_formatter
);
953 settings
.backlight_brightness
= result
;
954 mpeg_backlight_update_brightness(-1);
956 #endif /* HAVE_BACKLIGHT_BRIGHTNESS */
963 if (mpeg_menu_sysevent() != 0)
970 static void audio_options(void)
974 bool menu_quit
= false;
976 static const struct menu_item items
[] = {
977 [MPEG_AUDIO_TONE_CONTROLS
] =
978 { "Tone Controls", NULL
},
979 [MPEG_AUDIO_CHANNEL_MODES
] =
980 { "Channel Modes", NULL
},
981 [MPEG_AUDIO_CROSSFEED
] =
982 { "Crossfeed", NULL
},
983 [MPEG_AUDIO_EQUALIZER
] =
984 { "Equalizer", NULL
},
985 [MPEG_AUDIO_DITHERING
] =
986 { "Dithering", NULL
},
989 menu_id
= menu_init(rb
, items
, ARRAYLEN(items
),
990 mpeg_menu_sysevent_callback
, NULL
, NULL
, NULL
);
992 rb
->button_clear_queue();
996 mpeg_menu_sysevent_clear();
997 result
= menu_show(menu_id
);
1001 case MPEG_AUDIO_TONE_CONTROLS
:
1002 mpeg_set_option("Tone Controls", &settings
.tone_controls
, INT
,
1003 globaloff
, 2, NULL
);
1004 sync_audio_setting(result
, false);
1007 case MPEG_AUDIO_CHANNEL_MODES
:
1008 mpeg_set_option("Channel Modes", &settings
.channel_modes
,
1009 INT
, globaloff
, 2, NULL
);
1010 sync_audio_setting(result
, false);
1013 case MPEG_AUDIO_CROSSFEED
:
1014 mpeg_set_option("Crossfeed", &settings
.crossfeed
, INT
,
1015 globaloff
, 2, NULL
);
1016 sync_audio_setting(result
, false);
1019 case MPEG_AUDIO_EQUALIZER
:
1020 mpeg_set_option("Equalizer", &settings
.equalizer
, INT
,
1021 globaloff
, 2, NULL
);
1022 sync_audio_setting(result
, false);
1025 case MPEG_AUDIO_DITHERING
:
1026 mpeg_set_option("Dithering", &settings
.dithering
, INT
,
1027 globaloff
, 2, NULL
);
1028 sync_audio_setting(result
, false);
1036 if (mpeg_menu_sysevent() != 0)
1043 static void resume_options(void)
1045 static const struct opt_items items
[MPEG_RESUME_NUM_OPTIONS
] = {
1046 [MPEG_RESUME_MENU_ALWAYS
] =
1047 { "Start menu", -1 },
1048 [MPEG_RESUME_MENU_IF_INCOMPLETE
] =
1049 { "Start menu if not completed", -1 },
1050 [MPEG_RESUME_ALWAYS
] =
1051 { "Resume automatically", -1 },
1052 [MPEG_RESUME_RESTART
] =
1053 { "Play from beginning", -1 },
1056 mpeg_set_option("Resume Options", &settings
.resume_options
,
1057 INT
, items
, MPEG_RESUME_NUM_OPTIONS
, NULL
);
1060 static void clear_resume_count(void)
1062 configfile_save(SETTINGS_FILENAME
, config
, ARRAYLEN(config
),
1065 settings
.resume_count
= 0;
1067 /* add this place holder so the count is above resume entries */
1068 configfile_update_entry(SETTINGS_FILENAME
, "Resume count", 0);
1071 int mpeg_menu(unsigned flags
)
1075 bool menu_quit
= false;
1079 struct menu_item items
[] = {
1080 [MPEG_MENU_DISPLAY_SETTINGS
] =
1081 { "Display Options", NULL
},
1082 [MPEG_MENU_AUDIO_SETTINGS
] =
1083 { "Audio Options", NULL
},
1084 [MPEG_MENU_ENABLE_START_MENU
] =
1085 { "Resume Options", NULL
},
1086 [MPEG_MENU_CLEAR_RESUMES
] =
1087 { clear_str
, NULL
},
1089 { "Quit mpegplayer", NULL
},
1092 item_count
= ARRAYLEN(items
);
1094 if (flags
& MPEG_MENU_HIDE_QUIT_ITEM
)
1097 menu_id
= menu_init(rb
, items
, item_count
,
1098 mpeg_menu_sysevent_callback
, NULL
, NULL
, NULL
);
1100 rb
->button_clear_queue();
1104 mpeg_menu_sysevent_clear();
1106 /* Format and add resume option to the menu display */
1107 format_menu_item(&items
[MPEG_MENU_CLEAR_RESUMES
], sizeof(clear_str
),
1108 "Clear all resumes: %u", settings
.resume_count
);
1110 result
= menu_show(menu_id
);
1114 case MPEG_MENU_DISPLAY_SETTINGS
:
1118 case MPEG_MENU_AUDIO_SETTINGS
:
1122 case MPEG_MENU_ENABLE_START_MENU
:
1126 case MPEG_MENU_CLEAR_RESUMES
:
1127 clear_resume_count();
1130 case MPEG_MENU_QUIT
:
1136 if (mpeg_menu_sysevent() != 0)
1138 result
= MPEG_MENU_QUIT
;
1145 rb
->lcd_clear_display();
1151 void init_settings(const char* filename
)
1153 /* Set the default settings */
1154 settings
.showfps
= 0; /* Do not show FPS */
1155 settings
.limitfps
= 1; /* Limit FPS */
1156 settings
.skipframes
= 1; /* Skip frames */
1157 settings
.resume_options
= MPEG_RESUME_MENU_ALWAYS
; /* Enable start menu */
1158 settings
.resume_count
= -1;
1159 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
1160 settings
.backlight_brightness
= -1; /* Use default setting */
1162 #if MPEG_OPTION_DITHERING_ENABLED
1163 settings
.displayoptions
= 0; /* No visual effects */
1165 settings
.tone_controls
= false;
1166 settings
.channel_modes
= false;
1167 settings
.crossfeed
= false;
1168 settings
.equalizer
= false;
1169 settings
.dithering
= false;
1171 configfile_init(rb
);
1173 if (configfile_load(SETTINGS_FILENAME
, config
,
1174 sizeof(config
)/sizeof(*config
),
1175 SETTINGS_MIN_VERSION
) < 0)
1177 /* Generate a new config file with default values */
1178 configfile_save(SETTINGS_FILENAME
, config
,
1179 sizeof(config
)/sizeof(*config
),
1183 #if MPEG_OPTION_DITHERING_ENABLED
1184 if ((settings
.displayoptions
=
1185 configfile_get_value(SETTINGS_FILENAME
, "Display options")) < 0)
1187 configfile_update_entry(SETTINGS_FILENAME
, "Display options",
1188 (settings
.displayoptions
=0));
1190 rb
->lcd_yuv_set_options(settings
.displayoptions
);
1193 if (settings
.resume_count
< 0)
1195 settings
.resume_count
= 0;
1196 configfile_update_entry(SETTINGS_FILENAME
, "Resume count", 0);
1199 rb
->snprintf(settings
.resume_filename
, MAX_PATH
, "%s", filename
);
1201 /* get the resume time for the current mpeg if it exist */
1202 if ((settings
.resume_time
= configfile_get_value
1203 (SETTINGS_FILENAME
, filename
)) < 0)
1205 settings
.resume_time
= 0;
1208 /* Set our audio options */
1209 sync_audio_settings(false);
1212 void save_settings(void)
1214 configfile_update_entry(SETTINGS_FILENAME
, "Show FPS",
1216 configfile_update_entry(SETTINGS_FILENAME
, "Limit FPS",
1218 configfile_update_entry(SETTINGS_FILENAME
, "Skip frames",
1219 settings
.skipframes
);
1220 configfile_update_entry(SETTINGS_FILENAME
, "Resume options",
1221 settings
.resume_options
);
1223 /* If this was a new resume entry then update the total resume count */
1224 if (configfile_update_entry(SETTINGS_FILENAME
, settings
.resume_filename
,
1225 settings
.resume_time
) == 0)
1227 configfile_update_entry(SETTINGS_FILENAME
, "Resume count",
1228 ++settings
.resume_count
);
1231 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
1232 configfile_update_entry(SETTINGS_FILENAME
, "Backlight brightness",
1233 settings
.backlight_brightness
);
1236 #if MPEG_OPTION_DITHERING_ENABLED
1237 configfile_update_entry(SETTINGS_FILENAME
, "Display options",
1238 settings
.displayoptions
);
1240 configfile_update_entry(SETTINGS_FILENAME
, "Tone controls",
1241 settings
.tone_controls
);
1242 configfile_update_entry(SETTINGS_FILENAME
, "Channel modes",
1243 settings
.channel_modes
);
1244 configfile_update_entry(SETTINGS_FILENAME
, "Crossfeed",
1245 settings
.crossfeed
);
1246 configfile_update_entry(SETTINGS_FILENAME
, "Equalizer",
1247 settings
.equalizer
);
1248 configfile_update_entry(SETTINGS_FILENAME
, "Dithering",
1249 settings
.dithering
);
1251 /* Restore audio options */
1252 sync_audio_settings(true);