2 #include "lib/helper.h"
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
127 #elif CONFIG_KEYPAD == IAUDIO67_PAD
128 #define MPEG_START_TIME_SELECT BUTTON_MENU
129 #define MPEG_START_TIME_LEFT BUTTON_LEFT
130 #define MPEG_START_TIME_RIGHT BUTTON_RIGHT
131 #define MPEG_START_TIME_UP BUTTON_STOP
132 #define MPEG_START_TIME_DOWN BUTTON_PLAY
133 #define MPEG_START_TIME_EXIT BUTTON_POWER
136 #error No keymap defined!
139 #ifdef HAVE_TOUCHSCREEN
140 #ifndef MPEG_START_TIME_SELECT
141 #define MPEG_START_TIME_SELECT BUTTON_CENTER
143 #ifndef MPEG_START_TIME_SCROLL_UP
144 #define MPEG_START_TIME_SCROLL_UP BUTTON_TOPRIGHT
146 #ifndef MPEG_START_TIME_SCROLL_DOWN
147 #define MPEG_START_TIME_SCROLL_DOWN BUTTON_TOPLEFT
149 #ifndef MPEG_START_TIME_LEFT
150 #define MPEG_START_TIME_LEFT BUTTON_MIDLEFT
152 #ifndef MPEG_START_TIME_RIGHT
153 #define MPEG_START_TIME_RIGHT BUTTON_MIDRIGHT
155 #ifndef MPEG_START_TIME_UP
156 #define MPEG_START_TIME_UP BUTTON_TOPMIDDLE
158 #ifndef MPEG_START_TIME_DOWN
159 #define MPEG_START_TIME_DOWN BUTTON_BOTTOMMIDDLE
161 #ifndef MPEG_START_TIME_EXIT
162 #define MPEG_START_TIME_EXIT BUTTON_TOPLEFT
166 static struct configdata config
[] =
168 {TYPE_INT
, 0, 2, &settings
.showfps
, "Show FPS", NULL
, NULL
},
169 {TYPE_INT
, 0, 2, &settings
.limitfps
, "Limit FPS", NULL
, NULL
},
170 {TYPE_INT
, 0, 2, &settings
.skipframes
, "Skip frames", NULL
, NULL
},
171 {TYPE_INT
, 0, INT_MAX
, &settings
.resume_count
, "Resume count",
173 {TYPE_INT
, 0, MPEG_RESUME_NUM_OPTIONS
, &settings
.resume_options
,
174 "Resume options", NULL
, NULL
},
175 #if defined(TOSHIBA_GIGABEAT_F) || defined(SANSA_E200) || defined(SANSA_C200)
176 {TYPE_INT
, 0, INT_MAX
, &settings
.displayoptions
, "Display options",
179 {TYPE_INT
, 0, 2, &settings
.tone_controls
, "Tone controls", NULL
, NULL
},
180 {TYPE_INT
, 0, 2, &settings
.channel_modes
, "Channel modes", NULL
, NULL
},
181 {TYPE_INT
, 0, 2, &settings
.crossfeed
, "Crossfeed", NULL
, NULL
},
182 {TYPE_INT
, 0, 2, &settings
.equalizer
, "Equalizer", NULL
, NULL
},
183 {TYPE_INT
, 0, 2, &settings
.dithering
, "Dithering", NULL
, NULL
},
184 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
185 {TYPE_INT
, -1, INT_MAX
, &settings
.backlight_brightness
,
186 "Backlight brightness", NULL
, NULL
},
190 static const struct opt_items noyes
[2] = {
195 static const struct opt_items enabledisable
[2] = {
200 static const struct opt_items globaloff
[2] = {
202 { "Use sound setting", -1 },
205 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
206 #define BACKLIGHT_OPTION_DEFAULT "Use setting"
209 static long mpeg_menu_sysevent_id
;
211 void mpeg_menu_sysevent_clear(void)
213 mpeg_menu_sysevent_id
= 0;
216 int mpeg_menu_sysevent_callback(int btn
, int menu
)
220 case SYS_USB_CONNECTED
:
222 mpeg_menu_sysevent_id
= btn
;
223 return ACTION_STD_CANCEL
;
230 long mpeg_menu_sysevent(void)
232 return mpeg_menu_sysevent_id
;
235 void mpeg_menu_sysevent_handle(void)
237 long id
= mpeg_menu_sysevent();
239 rb
->default_event_handler(id
);
242 static void format_menu_item(struct menu_item
*item
, int bufsize
,
243 const char *fmt
, ...)
248 rb
->vsnprintf(item
->desc
, bufsize
, fmt
, ap
);
253 static bool mpeg_set_option(const char* string
,
255 enum optiontype type
,
256 const struct opt_items
* options
,
258 void (*function
)(int))
260 mpeg_menu_sysevent_clear();
262 /* This eats SYS_POWEROFF - :\ */
263 bool usb
= rb
->set_option(string
, variable
, type
, options
, numoptions
,
267 mpeg_menu_sysevent_id
= ACTION_STD_CANCEL
;
272 #ifdef HAVE_BACKLIGHT_BRIGHTNESS /* Only used for this atm */
273 static bool mpeg_set_int(const char *string
, const char *unit
,
274 int voice_unit
, const int *variable
,
275 void (*function
)(int), int step
,
278 void (*formatter
)(char*, size_t, int, const char*))
280 mpeg_menu_sysevent_clear();
282 bool usb
= rb
->set_int(string
, unit
, voice_unit
, variable
, function
,
283 step
, min
, max
, formatter
);
286 mpeg_menu_sysevent_id
= ACTION_STD_CANCEL
;
290 #endif /* HAVE_BACKLIGHT_BRIGHTNESS */
292 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
293 void mpeg_backlight_update_brightness(int value
)
297 value
+= MIN_BRIGHTNESS_SETTING
;
298 backlight_brightness_set(rb
, value
);
302 backlight_brightness_use_setting(rb
);
306 static void backlight_brightness_function(int value
)
308 mpeg_backlight_update_brightness(value
);
311 static void backlight_brightness_formatter(char *buf
, size_t length
,
312 int value
, const char *input
)
315 rb
->strncpy(buf
, BACKLIGHT_OPTION_DEFAULT
, length
);
317 rb
->snprintf(buf
, length
, "%d", value
+ MIN_BRIGHTNESS_SETTING
);
321 #endif /* HAVE_BACKLIGHT_BRIGHTNESS */
323 /* Sync a particular audio setting to global or mpegplayer forced off */
324 static void sync_audio_setting(int setting
, bool global
)
330 case MPEG_AUDIO_TONE_CONTROLS
:
331 if (global
|| settings
.tone_controls
)
333 val0
= rb
->global_settings
->bass
;
334 val1
= rb
->global_settings
->treble
;
338 val0
= rb
->sound_default(SOUND_BASS
);
339 val1
= rb
->sound_default(SOUND_TREBLE
);
341 rb
->sound_set(SOUND_BASS
, val0
);
342 rb
->sound_set(SOUND_TREBLE
, val1
);
345 case MPEG_AUDIO_CHANNEL_MODES
:
346 val0
= (global
|| settings
.channel_modes
) ?
347 rb
->global_settings
->channel_config
:
349 rb
->sound_set(SOUND_CHANNELS
, val0
);
352 case MPEG_AUDIO_CROSSFEED
:
353 rb
->dsp_set_crossfeed((global
|| settings
.crossfeed
) ?
354 rb
->global_settings
->crossfeed
: false);
357 case MPEG_AUDIO_EQUALIZER
:
358 rb
->dsp_set_eq((global
|| settings
.equalizer
) ?
359 rb
->global_settings
->eq_enabled
: false);
362 case MPEG_AUDIO_DITHERING
:
363 rb
->dsp_dither_enable((global
|| settings
.dithering
) ?
364 rb
->global_settings
->dithering_enabled
: false);
369 /* Sync all audio settings to global or mpegplayer forced off */
370 static void sync_audio_settings(bool global
)
372 static const int setting_index
[] =
374 MPEG_AUDIO_TONE_CONTROLS
,
375 MPEG_AUDIO_CHANNEL_MODES
,
376 MPEG_AUDIO_CROSSFEED
,
377 MPEG_AUDIO_EQUALIZER
,
378 MPEG_AUDIO_DITHERING
,
382 for (i
= 0; i
< ARRAYLEN(setting_index
); i
++)
384 sync_audio_setting(setting_index
[i
], global
);
388 #ifndef HAVE_LCD_COLOR
389 /* Cheapo splash implementation for the grey surface */
390 static void grey_splash(int ticks
, const unsigned char *fmt
, ...)
392 unsigned char buffer
[256];
399 rb
->vsnprintf(buffer
, sizeof (buffer
), fmt
, ap
);
403 grey_getstringsize(buffer
, &w
, &h
);
405 oldfg
= grey_get_foreground();
406 oldmode
= grey_get_drawmode();
408 grey_set_drawmode(DRMODE_FG
);
409 grey_set_foreground(GREY_LIGHTGRAY
);
411 x
= (LCD_WIDTH
- w
) / 2;
412 y
= (LCD_HEIGHT
- h
) / 2;
414 grey_fillrect(x
- 1, y
- 1, w
+ 2, h
+ 2);
416 grey_set_foreground(GREY_BLACK
);
418 grey_putsxy(x
, y
, buffer
);
419 grey_drawrect(x
- 2, y
- 2, w
+ 4, h
+ 4);
421 grey_set_foreground(oldfg
);
422 grey_set_drawmode(oldmode
);
429 #endif /* !HAVE_LCD_COLOR */
431 static void show_loading(struct vo_rect
*rc
)
433 int oldmode
= lcd_(get_drawmode
)();
434 lcd_(set_drawmode
)(DRMODE_SOLID
| DRMODE_INVERSEVID
);
435 lcd_(fillrect
)(rc
->l
-1, rc
->t
-1, rc
->r
- rc
->l
+ 2, rc
->b
- rc
->t
+ 2);
436 lcd_(set_drawmode
)(oldmode
);
437 lcd_(splash
)(0, "Loading...");
440 static void draw_slider(uint32_t range
, uint32_t pos
, struct vo_rect
*rc
)
442 #define SLIDER_WIDTH (LCD_WIDTH-SLIDER_LMARGIN-SLIDER_RMARGIN)
443 #define SLIDER_X SLIDER_LMARGIN
444 #define SLIDER_Y (LCD_HEIGHT-SLIDER_HEIGHT-SLIDER_BMARGIN)
445 #define SLIDER_HEIGHT 8
446 #define SLIDER_TEXTMARGIN 1
447 #define SLIDER_LMARGIN 1
448 #define SLIDER_RMARGIN 1
449 #define SLIDER_TMARGIN 1
450 #define SLIDER_BMARGIN 1
451 #define SCREEN_MARGIN 1
455 int text_w
, text_h
, text_y
;
457 /* Put positition on left */
458 ts_to_hms(pos
, &hms
);
459 hms_format(str
, sizeof(str
), &hms
);
460 lcd_(getstringsize
)(str
, NULL
, &text_h
);
461 text_y
= SLIDER_Y
- SLIDER_TEXTMARGIN
- text_h
;
465 int oldmode
= lcd_(get_drawmode
)();
466 lcd_(set_drawmode
)(DRMODE_BG
| DRMODE_INVERSEVID
);
467 lcd_(fillrect
)(SLIDER_X
, text_y
, SLIDER_WIDTH
,
468 LCD_HEIGHT
- SLIDER_BMARGIN
- text_y
470 lcd_(set_drawmode
)(oldmode
);
472 lcd_(putsxy
)(SLIDER_X
, text_y
, str
);
474 /* Put duration on right */
475 ts_to_hms(range
, &hms
);
476 hms_format(str
, sizeof(str
), &hms
);
477 lcd_(getstringsize
)(str
, &text_w
, NULL
);
479 lcd_(putsxy
)(SLIDER_X
+ SLIDER_WIDTH
- text_w
, text_y
, str
);
482 lcd_(drawrect
)(SLIDER_X
, SLIDER_Y
, SLIDER_WIDTH
, SLIDER_HEIGHT
);
483 lcd_(fillrect
)(SLIDER_X
, SLIDER_Y
,
484 muldiv_uint32(pos
, SLIDER_WIDTH
, range
),
488 lcd_(update_rect
)(SLIDER_X
, text_y
- SLIDER_TMARGIN
, SLIDER_WIDTH
,
489 LCD_HEIGHT
- SLIDER_BMARGIN
- text_y
+ SLIDER_TEXTMARGIN
);
493 /* Just return slider rectangle */
495 rc
->t
= text_y
- SLIDER_TMARGIN
;
496 rc
->r
= rc
->l
+ SLIDER_WIDTH
;
497 rc
->b
= rc
->t
+ LCD_HEIGHT
- SLIDER_BMARGIN
- text_y
;
501 static bool display_thumb_image(const struct vo_rect
*rc
)
503 if (!stream_display_thumb(rc
))
505 lcd_(splash
)(0, "Frame not available");
509 /* Draw a raised border around the frame */
510 int oldcolor
= lcd_(get_foreground
)();
511 lcd_(set_foreground
)(DRAW_LIGHTGRAY
);
513 lcd_(hline
)(rc
->l
-1, rc
->r
-1, rc
->t
-1);
514 lcd_(vline
)(rc
->l
-1, rc
->t
, rc
->b
-1);
516 lcd_(set_foreground
)(DRAW_DARKGRAY
);
518 lcd_(hline
)(rc
->l
-1, rc
->r
, rc
->b
);
519 lcd_(vline
)(rc
->r
, rc
->t
-1, rc
->b
);
521 lcd_(set_foreground
)(oldcolor
);
523 lcd_(update_rect
)(rc
->l
-1, rc
->t
-1, rc
->r
- rc
->l
+ 2, 1);
524 lcd_(update_rect
)(rc
->l
-1, rc
->t
, 1, rc
->b
- rc
->t
);
525 lcd_(update_rect
)(rc
->l
-1, rc
->b
, rc
->r
- rc
->l
+ 2, 1);
526 lcd_(update_rect
)(rc
->r
, rc
->t
, 1, rc
->b
- rc
->t
);
531 /* Add an amount to the specified time - with saturation */
532 static uint32_t increment_time(uint32_t val
, int32_t amount
, uint32_t range
)
536 uint32_t off
= -amount
;
537 if (range
> off
&& val
>= off
)
544 uint32_t off
= amount
;
545 if (range
> off
&& val
<= range
- off
)
554 #ifdef HAVE_LCD_ENABLE
555 static void get_start_time_lcd_enable_hook(void)
557 rb
->queue_post(rb
->button_queue
, LCD_ENABLE_EVENT_0
, 0);
559 #endif /* HAVE_LCD_ENABLE */
561 static int get_start_time(uint32_t duration
)
564 int tmo
= TIMEOUT_NOBLOCK
;
565 uint32_t resume_time
= settings
.resume_time
;
566 struct vo_rect rc_vid
, rc_bound
;
567 uint32_t aspect_vid
, aspect_bound
;
569 enum state_enum slider_state
= state0
;
571 lcd_(clear_display
)();
574 #if defined(HAVE_LCD_ENABLE) && defined(HAVE_LCD_COLOR)
575 rb
->lcd_set_enable_hook(get_start_time_lcd_enable_hook
);
578 draw_slider(0, 100, &rc_bound
);
579 rc_bound
.b
= rc_bound
.t
- SLIDER_TMARGIN
;
580 rc_bound
.t
= SCREEN_MARGIN
;
582 DEBUGF("rc_bound: %d, %d, %d, %d\n", rc_bound
.l
, rc_bound
.t
,
583 rc_bound
.r
, rc_bound
.b
);
585 rc_vid
.l
= rc_vid
.t
= 0;
586 if (!stream_vo_get_size((struct vo_ext
*)&rc_vid
.r
))
588 /* Can't get size - fill whole thing */
589 rc_vid
.r
= rc_bound
.r
- rc_bound
.l
;
590 rc_vid
.b
= rc_bound
.b
- rc_bound
.t
;
593 /* Get aspect ratio of bounding rectangle and video in u16.16 */
594 aspect_bound
= ((rc_bound
.r
- rc_bound
.l
) << 16) /
595 (rc_bound
.b
- rc_bound
.t
);
597 DEBUGF("aspect_bound: %u.%02u\n", (unsigned)(aspect_bound
>> 16),
598 (unsigned)(100*(aspect_bound
& 0xffff) >> 16));
600 aspect_vid
= (rc_vid
.r
<< 16) / rc_vid
.b
;
602 DEBUGF("aspect_vid: %u.%02u\n", (unsigned)(aspect_vid
>> 16),
603 (unsigned)(100*(aspect_vid
& 0xffff) >> 16));
605 if (aspect_vid
>= aspect_bound
)
607 /* Video proportionally wider than or same as bounding rectangle */
608 if (rc_vid
.r
> rc_bound
.r
- rc_bound
.l
)
610 rc_vid
.r
= rc_bound
.r
- rc_bound
.l
;
611 rc_vid
.b
= (rc_vid
.r
<< 16) / aspect_vid
;
613 /* else already fits */
617 /* Video proportionally narrower than bounding rectangle */
618 if (rc_vid
.b
> rc_bound
.b
- rc_bound
.t
)
620 rc_vid
.b
= rc_bound
.b
- rc_bound
.t
;
621 rc_vid
.r
= (aspect_vid
* rc_vid
.b
) >> 16;
623 /* else already fits */
626 /* Even width and height >= 2 */
627 rc_vid
.r
= (rc_vid
.r
< 2) ? 2 : (rc_vid
.r
& ~1);
628 rc_vid
.b
= (rc_vid
.b
< 2) ? 2 : (rc_vid
.b
& ~1);
630 /* Center display in bounding rectangle */
631 rc_vid
.l
= ((rc_bound
.l
+ rc_bound
.r
) - rc_vid
.r
) / 2;
632 rc_vid
.r
+= rc_vid
.l
;
634 rc_vid
.t
= ((rc_bound
.t
+ rc_bound
.b
) - rc_vid
.b
) / 2;
635 rc_vid
.b
+= rc_vid
.t
;
637 DEBUGF("rc_vid: %d, %d, %d, %d\n", rc_vid
.l
, rc_vid
.t
,
640 #ifndef HAVE_LCD_COLOR
641 stream_gray_show(true);
644 while (slider_state
< state9
)
646 mpeg_menu_sysevent_clear();
647 button
= tmo
== TIMEOUT_BLOCK
?
648 rb
->button_get(true) : rb
->button_get_w_tmo(tmo
);
650 button
= mpeg_menu_sysevent_callback(button
, -1);
657 /* Coarse (1 minute) control */
658 case MPEG_START_TIME_DOWN
:
659 case MPEG_START_TIME_DOWN
| BUTTON_REPEAT
:
660 #ifdef MPEG_START_TIME_RC_DOWN
661 case MPEG_START_TIME_RC_DOWN
:
662 case MPEG_START_TIME_RC_DOWN
| BUTTON_REPEAT
:
664 resume_time
= increment_time(resume_time
, -60*TS_SECOND
, duration
);
665 slider_state
= state0
;
668 case MPEG_START_TIME_UP
:
669 case MPEG_START_TIME_UP
| BUTTON_REPEAT
:
670 #ifdef MPEG_START_TIME_RC_UP
671 case MPEG_START_TIME_RC_UP
:
672 case MPEG_START_TIME_RC_UP
| BUTTON_REPEAT
:
674 resume_time
= increment_time(resume_time
, 60*TS_SECOND
, duration
);
675 slider_state
= state0
;
678 /* Fine (1 second) control */
679 case MPEG_START_TIME_LEFT
:
680 case MPEG_START_TIME_LEFT
| BUTTON_REPEAT
:
681 #ifdef MPEG_START_TIME_RC_LEFT
682 case MPEG_START_TIME_RC_LEFT
:
683 case MPEG_START_TIME_RC_LEFT
| BUTTON_REPEAT
:
685 #ifdef MPEG_START_TIME_SCROLL_UP
686 case MPEG_START_TIME_SCROLL_UP
:
687 case MPEG_START_TIME_SCROLL_UP
| BUTTON_REPEAT
:
689 resume_time
= increment_time(resume_time
, -TS_SECOND
, duration
);
690 slider_state
= state0
;
693 case MPEG_START_TIME_RIGHT
:
694 case MPEG_START_TIME_RIGHT
| BUTTON_REPEAT
:
695 #ifdef MPEG_START_TIME_RC_RIGHT
696 case MPEG_START_TIME_RC_RIGHT
:
697 case MPEG_START_TIME_RC_RIGHT
| BUTTON_REPEAT
:
699 #ifdef MPEG_START_TIME_SCROLL_DOWN
700 case MPEG_START_TIME_SCROLL_DOWN
:
701 case MPEG_START_TIME_SCROLL_DOWN
| BUTTON_REPEAT
:
703 resume_time
= increment_time(resume_time
, TS_SECOND
, duration
);
704 slider_state
= state0
;
707 case MPEG_START_TIME_SELECT
:
708 #ifdef MPEG_START_TIME_RC_SELECT
709 case MPEG_START_TIME_RC_SELECT
:
711 settings
.resume_time
= resume_time
;
712 button
= MPEG_START_SEEK
;
713 slider_state
= state9
;
716 case MPEG_START_TIME_EXIT
:
717 #ifdef MPEG_START_TIME_RC_EXIT
718 case MPEG_START_TIME_RC_EXIT
:
720 button
= MPEG_START_EXIT
;
721 slider_state
= state9
;
724 case ACTION_STD_CANCEL
:
725 button
= MPEG_START_QUIT
;
726 slider_state
= state9
;
729 #ifdef HAVE_LCD_ENABLE
730 case LCD_ENABLE_EVENT_0
:
731 if (slider_state
== state2
)
732 display_thumb_image(&rc_vid
);
737 rb
->default_event_handler(button
);
742 switch (slider_state
)
746 stream_seek(resume_time
, SEEK_SET
);
747 show_loading(&rc_bound
);
748 draw_slider(duration
, resume_time
, NULL
);
749 slider_state
= state1
;
753 display_thumb_image(&rc_vid
);
754 slider_state
= state2
;
765 #ifdef HAVE_LCD_COLOR
766 #ifdef HAVE_LCD_ENABLE
767 rb
->lcd_set_enable_hook(NULL
);
770 stream_gray_show(false);
771 grey_clear_display();
780 static int show_start_menu(uint32_t duration
)
784 bool menu_quit
= false;
786 /* add the resume time to the menu display */
791 struct menu_item items
[] =
793 [MPEG_START_RESTART
] =
794 { "Play from beginning", NULL
},
795 [MPEG_START_RESUME
] =
796 { resume_str
, NULL
},
798 { "Set start time", NULL
},
799 [MPEG_START_SETTINGS
] =
800 { "Settings", NULL
},
802 { "Quit mpegplayer", NULL
},
805 ts_to_hms(settings
.resume_time
, &hms
);
806 hms_format(hms_str
, sizeof(hms_str
), &hms
);
807 format_menu_item(&items
[MPEG_START_RESUME
], sizeof (resume_str
),
808 "Resume at: %s", hms_str
);
810 menu_id
= menu_init(rb
, items
, ARRAYLEN(items
),
811 mpeg_menu_sysevent_callback
, NULL
, NULL
, NULL
);
813 rb
->button_clear_queue();
817 mpeg_menu_sysevent_clear();
818 result
= menu_show(menu_id
);
822 case MPEG_START_RESTART
:
823 settings
.resume_time
= 0;
827 case MPEG_START_RESUME
:
831 case MPEG_START_SEEK
:
832 if (!stream_can_seek())
834 rb
->splash(HZ
, "Unavailable");
838 result
= get_start_time(duration
);
840 if (result
!= MPEG_START_EXIT
)
844 case MPEG_START_SETTINGS
:
845 if (mpeg_menu(MPEG_MENU_HIDE_QUIT_ITEM
) != MPEG_MENU_QUIT
)
849 result
= MPEG_START_QUIT
;
854 if (mpeg_menu_sysevent() != 0)
856 result
= MPEG_START_QUIT
;
863 rb
->lcd_clear_display();
869 /* Return the desired resume action */
870 int mpeg_start_menu(uint32_t duration
)
872 mpeg_menu_sysevent_clear();
874 switch (settings
.resume_options
)
876 case MPEG_RESUME_MENU_IF_INCOMPLETE
:
877 if (!stream_can_seek() || settings
.resume_time
== 0)
879 case MPEG_RESUME_RESTART
:
880 settings
.resume_time
= 0;
881 return MPEG_START_RESTART
;
884 case MPEG_RESUME_MENU_ALWAYS
:
885 return show_start_menu(duration
);
886 case MPEG_RESUME_ALWAYS
:
887 return MPEG_START_SEEK
;
892 static void display_options(void)
896 bool menu_quit
= false;
898 static const struct menu_item items
[] = {
899 #if MPEG_OPTION_DITHERING_ENABLED
900 [MPEG_OPTION_DITHERING
] =
901 { "Dithering", NULL
},
903 [MPEG_OPTION_DISPLAY_FPS
] =
904 { "Display FPS", NULL
},
905 [MPEG_OPTION_LIMIT_FPS
] =
906 { "Limit FPS", NULL
},
907 [MPEG_OPTION_SKIP_FRAMES
] =
908 { "Skip frames", NULL
},
909 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
910 [MPEG_OPTION_BACKLIGHT_BRIGHTNESS
] =
911 { "Backlight brightness", NULL
},
915 menu_id
= menu_init(rb
, items
, ARRAYLEN(items
),
916 mpeg_menu_sysevent_callback
, NULL
, NULL
, NULL
);
918 rb
->button_clear_queue();
922 mpeg_menu_sysevent_clear();
923 result
= menu_show(menu_id
);
927 #if MPEG_OPTION_DITHERING_ENABLED
928 case MPEG_OPTION_DITHERING
:
929 result
= (settings
.displayoptions
& LCD_YUV_DITHER
) ? 1 : 0;
930 mpeg_set_option("Dithering", &result
, INT
, noyes
, 2, NULL
);
931 settings
.displayoptions
=
932 (settings
.displayoptions
& ~LCD_YUV_DITHER
)
933 | ((result
!= 0) ? LCD_YUV_DITHER
: 0);
934 rb
->lcd_yuv_set_options(settings
.displayoptions
);
936 #endif /* MPEG_OPTION_DITHERING_ENABLED */
938 case MPEG_OPTION_DISPLAY_FPS
:
939 mpeg_set_option("Display FPS", &settings
.showfps
, INT
,
943 case MPEG_OPTION_LIMIT_FPS
:
944 mpeg_set_option("Limit FPS", &settings
.limitfps
, INT
,
948 case MPEG_OPTION_SKIP_FRAMES
:
949 mpeg_set_option("Skip frames", &settings
.skipframes
, INT
,
953 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
954 case MPEG_OPTION_BACKLIGHT_BRIGHTNESS
:
955 result
= settings
.backlight_brightness
;
956 mpeg_backlight_update_brightness(result
);
957 mpeg_set_int("Backlight brightness", NULL
, -1, &result
,
958 backlight_brightness_function
, 1, -1,
959 MAX_BRIGHTNESS_SETTING
- MIN_BRIGHTNESS_SETTING
,
960 backlight_brightness_formatter
);
961 settings
.backlight_brightness
= result
;
962 mpeg_backlight_update_brightness(-1);
964 #endif /* HAVE_BACKLIGHT_BRIGHTNESS */
971 if (mpeg_menu_sysevent() != 0)
978 static void audio_options(void)
982 bool menu_quit
= false;
984 static const struct menu_item items
[] = {
985 [MPEG_AUDIO_TONE_CONTROLS
] =
986 { "Tone Controls", NULL
},
987 [MPEG_AUDIO_CHANNEL_MODES
] =
988 { "Channel Modes", NULL
},
989 [MPEG_AUDIO_CROSSFEED
] =
990 { "Crossfeed", NULL
},
991 [MPEG_AUDIO_EQUALIZER
] =
992 { "Equalizer", NULL
},
993 [MPEG_AUDIO_DITHERING
] =
994 { "Dithering", NULL
},
997 menu_id
= menu_init(rb
, items
, ARRAYLEN(items
),
998 mpeg_menu_sysevent_callback
, NULL
, NULL
, NULL
);
1000 rb
->button_clear_queue();
1004 mpeg_menu_sysevent_clear();
1005 result
= menu_show(menu_id
);
1009 case MPEG_AUDIO_TONE_CONTROLS
:
1010 mpeg_set_option("Tone Controls", &settings
.tone_controls
, INT
,
1011 globaloff
, 2, NULL
);
1012 sync_audio_setting(result
, false);
1015 case MPEG_AUDIO_CHANNEL_MODES
:
1016 mpeg_set_option("Channel Modes", &settings
.channel_modes
,
1017 INT
, globaloff
, 2, NULL
);
1018 sync_audio_setting(result
, false);
1021 case MPEG_AUDIO_CROSSFEED
:
1022 mpeg_set_option("Crossfeed", &settings
.crossfeed
, INT
,
1023 globaloff
, 2, NULL
);
1024 sync_audio_setting(result
, false);
1027 case MPEG_AUDIO_EQUALIZER
:
1028 mpeg_set_option("Equalizer", &settings
.equalizer
, INT
,
1029 globaloff
, 2, NULL
);
1030 sync_audio_setting(result
, false);
1033 case MPEG_AUDIO_DITHERING
:
1034 mpeg_set_option("Dithering", &settings
.dithering
, INT
,
1035 globaloff
, 2, NULL
);
1036 sync_audio_setting(result
, false);
1044 if (mpeg_menu_sysevent() != 0)
1051 static void resume_options(void)
1053 static const struct opt_items items
[MPEG_RESUME_NUM_OPTIONS
] = {
1054 [MPEG_RESUME_MENU_ALWAYS
] =
1055 { "Start menu", -1 },
1056 [MPEG_RESUME_MENU_IF_INCOMPLETE
] =
1057 { "Start menu if not completed", -1 },
1058 [MPEG_RESUME_ALWAYS
] =
1059 { "Resume automatically", -1 },
1060 [MPEG_RESUME_RESTART
] =
1061 { "Play from beginning", -1 },
1064 mpeg_set_option("Resume Options", &settings
.resume_options
,
1065 INT
, items
, MPEG_RESUME_NUM_OPTIONS
, NULL
);
1068 static void clear_resume_count(void)
1070 configfile_save(SETTINGS_FILENAME
, config
, ARRAYLEN(config
),
1073 settings
.resume_count
= 0;
1075 /* add this place holder so the count is above resume entries */
1076 configfile_update_entry(SETTINGS_FILENAME
, "Resume count", 0);
1079 int mpeg_menu(unsigned flags
)
1083 bool menu_quit
= false;
1087 struct menu_item items
[] = {
1088 [MPEG_MENU_DISPLAY_SETTINGS
] =
1089 { "Display Options", NULL
},
1090 [MPEG_MENU_AUDIO_SETTINGS
] =
1091 { "Audio Options", NULL
},
1092 [MPEG_MENU_ENABLE_START_MENU
] =
1093 { "Resume Options", NULL
},
1094 [MPEG_MENU_CLEAR_RESUMES
] =
1095 { clear_str
, NULL
},
1097 { "Quit mpegplayer", NULL
},
1100 item_count
= ARRAYLEN(items
);
1102 if (flags
& MPEG_MENU_HIDE_QUIT_ITEM
)
1105 menu_id
= menu_init(rb
, items
, item_count
,
1106 mpeg_menu_sysevent_callback
, NULL
, NULL
, NULL
);
1108 rb
->button_clear_queue();
1112 mpeg_menu_sysevent_clear();
1114 /* Format and add resume option to the menu display */
1115 format_menu_item(&items
[MPEG_MENU_CLEAR_RESUMES
], sizeof(clear_str
),
1116 "Clear all resumes: %u", settings
.resume_count
);
1118 result
= menu_show(menu_id
);
1122 case MPEG_MENU_DISPLAY_SETTINGS
:
1126 case MPEG_MENU_AUDIO_SETTINGS
:
1130 case MPEG_MENU_ENABLE_START_MENU
:
1134 case MPEG_MENU_CLEAR_RESUMES
:
1135 clear_resume_count();
1138 case MPEG_MENU_QUIT
:
1144 if (mpeg_menu_sysevent() != 0)
1146 result
= MPEG_MENU_QUIT
;
1153 rb
->lcd_clear_display();
1159 void init_settings(const char* filename
)
1161 /* Set the default settings */
1162 settings
.showfps
= 0; /* Do not show FPS */
1163 settings
.limitfps
= 1; /* Limit FPS */
1164 settings
.skipframes
= 1; /* Skip frames */
1165 settings
.resume_options
= MPEG_RESUME_MENU_ALWAYS
; /* Enable start menu */
1166 settings
.resume_count
= -1;
1167 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
1168 settings
.backlight_brightness
= -1; /* Use default setting */
1170 #if MPEG_OPTION_DITHERING_ENABLED
1171 settings
.displayoptions
= 0; /* No visual effects */
1173 settings
.tone_controls
= false;
1174 settings
.channel_modes
= false;
1175 settings
.crossfeed
= false;
1176 settings
.equalizer
= false;
1177 settings
.dithering
= false;
1179 configfile_init(rb
);
1181 if (configfile_load(SETTINGS_FILENAME
, config
,
1182 sizeof(config
)/sizeof(*config
),
1183 SETTINGS_MIN_VERSION
) < 0)
1185 /* Generate a new config file with default values */
1186 configfile_save(SETTINGS_FILENAME
, config
,
1187 sizeof(config
)/sizeof(*config
),
1191 #if MPEG_OPTION_DITHERING_ENABLED
1192 if ((settings
.displayoptions
=
1193 configfile_get_value(SETTINGS_FILENAME
, "Display options")) < 0)
1195 configfile_update_entry(SETTINGS_FILENAME
, "Display options",
1196 (settings
.displayoptions
=0));
1198 rb
->lcd_yuv_set_options(settings
.displayoptions
);
1201 if (settings
.resume_count
< 0)
1203 settings
.resume_count
= 0;
1204 configfile_update_entry(SETTINGS_FILENAME
, "Resume count", 0);
1207 rb
->snprintf(settings
.resume_filename
, MAX_PATH
, "%s", filename
);
1209 /* get the resume time for the current mpeg if it exist */
1210 if ((settings
.resume_time
= configfile_get_value
1211 (SETTINGS_FILENAME
, filename
)) < 0)
1213 settings
.resume_time
= 0;
1216 /* Set our audio options */
1217 sync_audio_settings(false);
1220 void save_settings(void)
1222 configfile_update_entry(SETTINGS_FILENAME
, "Show FPS",
1224 configfile_update_entry(SETTINGS_FILENAME
, "Limit FPS",
1226 configfile_update_entry(SETTINGS_FILENAME
, "Skip frames",
1227 settings
.skipframes
);
1228 configfile_update_entry(SETTINGS_FILENAME
, "Resume options",
1229 settings
.resume_options
);
1231 /* If this was a new resume entry then update the total resume count */
1232 if (configfile_update_entry(SETTINGS_FILENAME
, settings
.resume_filename
,
1233 settings
.resume_time
) == 0)
1235 configfile_update_entry(SETTINGS_FILENAME
, "Resume count",
1236 ++settings
.resume_count
);
1239 #ifdef HAVE_BACKLIGHT_BRIGHTNESS
1240 configfile_update_entry(SETTINGS_FILENAME
, "Backlight brightness",
1241 settings
.backlight_brightness
);
1244 #if MPEG_OPTION_DITHERING_ENABLED
1245 configfile_update_entry(SETTINGS_FILENAME
, "Display options",
1246 settings
.displayoptions
);
1248 configfile_update_entry(SETTINGS_FILENAME
, "Tone controls",
1249 settings
.tone_controls
);
1250 configfile_update_entry(SETTINGS_FILENAME
, "Channel modes",
1251 settings
.channel_modes
);
1252 configfile_update_entry(SETTINGS_FILENAME
, "Crossfeed",
1253 settings
.crossfeed
);
1254 configfile_update_entry(SETTINGS_FILENAME
, "Equalizer",
1255 settings
.equalizer
);
1256 configfile_update_entry(SETTINGS_FILENAME
, "Dithering",
1257 settings
.dithering
);
1259 /* Restore audio options */
1260 sync_audio_settings(true);