summaryrefslogtreecommitdiff
path: root/apps/plugins/mpegplayer/mpeg_settings.c
diff options
context:
space:
mode:
authorRobert Kukla <roolku@rockbox.org>2007-10-09 20:42:20 +0000
committerRobert Kukla <roolku@rockbox.org>2007-10-09 20:42:20 +0000
commitfd3fe45bc14a0a540f2525102551c92a64a73b76 (patch)
tree1ef8103bbfa5b33f684a94bddc5ecb4685ec5e88 /apps/plugins/mpegplayer/mpeg_settings.c
parentce135909b9393d9824b3f69a70659400480cc069 (diff)
downloadrockbox-fd3fe45bc14a0a540f2525102551c92a64a73b76.zip
rockbox-fd3fe45bc14a0a540f2525102551c92a64a73b76.tar.gz
rockbox-fd3fe45bc14a0a540f2525102551c92a64a73b76.tar.bz2
rockbox-fd3fe45bc14a0a540f2525102551c92a64a73b76.tar.xz
FS#7487 - mpegplayer - video start time seek with resume
by John S. Gwynne & Brian J. Morey This should stop the patch from breaking again and give them opportunity to improve it further. git-svn-id: svn://svn.rockbox.org/rockbox/trunk@15052 a1c6a512-1295-4272-9138-f99709370657
Diffstat (limited to 'apps/plugins/mpegplayer/mpeg_settings.c')
-rw-r--r--apps/plugins/mpegplayer/mpeg_settings.c410
1 files changed, 384 insertions, 26 deletions
diff --git a/apps/plugins/mpegplayer/mpeg_settings.c b/apps/plugins/mpegplayer/mpeg_settings.c
index 28062f4..197fa09 100644
--- a/apps/plugins/mpegplayer/mpeg_settings.c
+++ b/apps/plugins/mpegplayer/mpeg_settings.c
@@ -7,20 +7,99 @@
extern struct plugin_api* rb;
struct mpeg_settings settings;
-static struct mpeg_settings old_settings;
+
+ssize_t seek_PTS(int in_file, int startTime, int accept_button);
+void display_thumb(int in_file);
+
+#ifndef HAVE_LCD_COLOR
+void gray_show(bool enable);
+#endif
#define SETTINGS_VERSION 2
#define SETTINGS_MIN_VERSION 1
#define SETTINGS_FILENAME "mpegplayer.cfg"
+enum slider_state_t {state0, state1, state2,
+ state3, state4, state5} slider_state;
+
+volatile long thumbDelayTimer;
+
+/* button definitions */
+#if (CONFIG_KEYPAD == IRIVER_H100_PAD) || \
+ (CONFIG_KEYPAD == IRIVER_H300_PAD)
+#define MPEG_SELECT BUTTON_ON
+#define MPEG_RIGHT BUTTON_RIGHT
+#define MPEG_LEFT BUTTON_LEFT
+#define MPEG_SCROLL_DOWN BUTTON_UP
+#define MPEG_SCROLL_UP BUTTON_DOWN
+#define MPEG_EXIT BUTTON_OFF
+
+#elif (CONFIG_KEYPAD == IAUDIO_X5M5_PAD)
+#define MPEG_SELECT BUTTON_PLAY
+#define MPEG_RIGHT BUTTON_RIGHT
+#define MPEG_LEFT BUTTON_LEFT
+#define MPEG_SCROLL_DOWN BUTTON_UP
+#define MPEG_SCROLL_UP BUTTON_DOWN
+#define MPEG_EXIT BUTTON_POWER
+
+#elif (CONFIG_KEYPAD == IPOD_4G_PAD) || \
+ (CONFIG_KEYPAD == IPOD_3G_PAD) || \
+ (CONFIG_KEYPAD == IPOD_1G2G_PAD)
+#define MPEG_SELECT BUTTON_SELECT
+#define MPEG_RIGHT BUTTON_RIGHT
+#define MPEG_LEFT BUTTON_LEFT
+#define MPEG_SCROLL_DOWN BUTTON_SCROLL_FWD
+#define MPEG_SCROLL_UP BUTTON_SCROLL_BACK
+#define MPEG_EXIT BUTTON_MENU
+
+#elif CONFIG_KEYPAD == GIGABEAT_PAD
+#define MPEG_SELECT BUTTON_SELECT
+#define MPEG_LEFT BUTTON_LEFT
+#define MPEG_RIGHT BUTTON_RIGHT
+#define MPEG_UP BUTTON_UP
+#define MPEG_DOWN BUTTON_DOWN
+#define MPEG_SCROLL_DOWN BUTTON_VOL_DOWN
+#define MPEG_SCROLL_UP BUTTON_VOL_UP
+#define MPEG_EXIT BUTTON_POWER
+
+#elif CONFIG_KEYPAD == IRIVER_H10_PAD
+#define MPEG_SELECT BUTTON_PLAY
+#define MPEG_SCROLL_UP BUTTON_SCROLL_UP
+#define MPEG_SCROLL_DOWN BUTTON_SCROLL_DOWN
+#define MPEG_LEFT BUTTON_LEFT
+#define MPEG_RIGHT BUTTON_RIGHT
+#define MPEG_EXIT BUTTON_POWER
+
+#elif (CONFIG_KEYPAD == SANSA_E200_PAD)
+#define MPEG_SELECT BUTTON_SELECT
+#define MPEG_SCROLL_UP BUTTON_SCROLL_UP
+#define MPEG_SCROLL_DOWN BUTTON_SCROLL_DOWN
+#define MPEG_LEFT BUTTON_LEFT
+#define MPEG_RIGHT BUTTON_RIGHT
+#define MPEG_UP BUTTON_UP
+#define MPEG_DOWN BUTTON_DOWN
+#define MPEG_EXIT BUTTON_POWER
+
+#elif (CONFIG_KEYPAD == SANSA_C200_PAD)
+#define MPEG_SELECT BUTTON_SELECT
+#define MPEG_SCROLL_UP BUTTON_VOL_UP
+#define MPEG_SCROLL_DOWN BUTTON_VOL_DOWN
+#define MPEG_LEFT BUTTON_LEFT
+#define MPEG_RIGHT BUTTON_RIGHT
+#define MPEG_UP BUTTON_UP
+#define MPEG_DOWN BUTTON_DOWN
+#define MPEG_EXIT BUTTON_POWER
+
+#else
+#error MPEGPLAYER: Unsupported keypad
+#endif
+
static struct configdata config[] =
{
- {TYPE_ENUM, 0, 2, &settings.showfps, "Show FPS",
- (char *[]){ "No", "Yes" }, NULL},
- {TYPE_ENUM, 0, 2, &settings.limitfps, "Limit FPS",
- (char *[]){ "No", "Yes" }, NULL},
- {TYPE_ENUM, 0, 2, &settings.skipframes, "Skip frames",
- (char *[]){ "No", "Yes" }, NULL},
+ {TYPE_INT, 0, 2, &settings.showfps, "Show FPS", NULL, NULL},
+ {TYPE_INT, 0, 2, &settings.limitfps, "Limit FPS", NULL, NULL},
+ {TYPE_INT, 0, 2, &settings.skipframes, "Skip frames", NULL, NULL},
+
#if defined(TOSHIBA_GIGABEAT_F) || defined(SANSA_E200)
{TYPE_INT, 0, INT_MAX, &settings.displayoptions, "Display options",
NULL, NULL},
@@ -36,6 +115,7 @@ enum mpeg_menu_ids
MPEG_OPTION_DISPLAY_FPS,
MPEG_OPTION_LIMIT_FPS,
MPEG_OPTION_SKIP_FRAMES,
+ MPEG_OPTION_CLEAR_RESUMES,
MPEG_OPTION_QUIT,
};
@@ -68,13 +148,250 @@ static void display_options(void)
}
#endif /* #ifdef TOSHIBA_GIGABEAT_F */
+void draw_slider(int slider_ypos, int max_val, int current_val)
+{
+ int slider_margin = LCD_WIDTH*12/100; /* 12% */
+ int slider_width = LCD_WIDTH-(slider_margin*2);
+ char resume_str[32];
+
+ /* max_val and current_val are in half minutes
+ determine value .0 or .5 to display */
+ int max_hol = max_val/2;
+ int max_rem = (max_val-(max_hol*2))*5;
+ int current_hol = current_val/2;
+ int current_rem = (current_val-(current_hol*2))*5;
+
+ rb->snprintf(resume_str, sizeof(resume_str), "0.0");
+ rb->lcd_putsxy(slider_margin, slider_ypos, resume_str);
+
+ rb->snprintf(resume_str, sizeof(resume_str), "%u.%u", max_hol, max_rem);
+ rb->lcd_putsxy(LCD_WIDTH-slider_margin-25, slider_ypos, resume_str);
+
+ rb->lcd_drawrect(slider_margin, slider_ypos+17, slider_width, 8);
+ rb->lcd_fillrect(slider_margin, slider_ypos+17,
+ current_val*slider_width/max_val, 8);
+
+ rb->snprintf(resume_str, sizeof(resume_str), "%u.%u", current_hol,
+ current_rem);
+ rb->lcd_putsxy(slider_margin+(current_val*slider_width/max_val)-16,
+ slider_ypos+29, resume_str);
+
+ rb->lcd_update_rect(0, slider_ypos, LCD_WIDTH, LCD_HEIGHT-slider_ypos);
+}
+
+int get_start_time(int play_time, int in_file)
+{
+ int quit = 0;
+ int button = 0;
+ int resume_time = settings.resume_time;
+ int slider_ypos = LCD_HEIGHT-45;
+ int seek_rtn;
+
+ slider_state = state0;
+ thumbDelayTimer = *(rb->current_tick);
+ rb->lcd_clear_display();
+ rb->lcd_update();
+
+ while(quit == 0)
+ {
+ button = rb->button_get(false);
+ switch (button)
+ {
+#if (CONFIG_KEYPAD == GIGABEAT_PAD) || \
+ (CONFIG_KEYPAD == SANSA_E200_PAD) || \
+ (CONFIG_KEYPAD == SANSA_C200_PAD)
+ case MPEG_DOWN:
+ case MPEG_DOWN | BUTTON_REPEAT:
+ if ((resume_time -= 20) < 0)
+ resume_time = 0;
+ slider_state = state0;
+ thumbDelayTimer = *(rb->current_tick);
+ break;
+ case MPEG_UP:
+ case MPEG_UP | BUTTON_REPEAT:
+ if ((resume_time += 20) > play_time)
+ resume_time = play_time;
+ slider_state = state0;
+ thumbDelayTimer = *(rb->current_tick);
+ break;
+#endif
+ case MPEG_LEFT:
+ case MPEG_LEFT | BUTTON_REPEAT:
+ case MPEG_SCROLL_UP:
+ case MPEG_SCROLL_UP | BUTTON_REPEAT:
+ if (--resume_time < 0)
+ resume_time = 0;
+ slider_state = state0;
+ thumbDelayTimer = *(rb->current_tick);
+ break;
+ case MPEG_RIGHT:
+ case MPEG_RIGHT | BUTTON_REPEAT:
+ case MPEG_SCROLL_DOWN:
+ case MPEG_SCROLL_DOWN | BUTTON_REPEAT:
+ if (++resume_time > play_time)
+ resume_time = play_time;
+ slider_state = state0;
+ thumbDelayTimer = *(rb->current_tick);
+ break;
+ case MPEG_SELECT:
+ quit = 1;
+ break;
+ case MPEG_EXIT:
+ resume_time = -1;
+ quit = 1;
+ break;
+ default:
+ if (rb->default_event_handler(button) == SYS_USB_CONNECTED)
+ {
+ resume_time = -1;
+ quit = 1;
+ }
+ break;
+ }
+
+ rb->yield();
+
+ switch(slider_state)
+ {
+ case state0:
+ rb->lcd_clear_display();
+ rb->lcd_update();
+#ifdef HAVE_LCD_COLOR
+ if (resume_time > 0)
+ rb->splash(0, "loading ...");
+#endif
+ slider_state = state1;
+ break;
+ case state1:
+ if (*(rb->current_tick) - thumbDelayTimer > 75)
+ slider_state = state2;
+ if (resume_time == 0)
+ {
+ seek_rtn = 0;
+ slider_state = state5;
+ }
+ draw_slider(slider_ypos, play_time, resume_time);
+ break;
+ case state2:
+ if ( (seek_rtn = seek_PTS(in_file, resume_time, 1)) >= 0)
+ slider_state = state3;
+ else if (seek_rtn == -101)
+ {
+ slider_state = state0;
+ thumbDelayTimer = *(rb->current_tick);
+ }
+ else
+ slider_state = state4;
+ break;
+ case state3:
+ display_thumb(in_file);
+ draw_slider(slider_ypos, play_time, resume_time);
+ slider_state = state4;
+ break;
+ case state4:
+ draw_slider(slider_ypos, play_time, resume_time);
+ slider_state = state5;
+ break;
+ case state5:
+ break;
+ }
+ }
+
+ return resume_time;
+}
+
+int mpeg_start_menu(int play_time, int in_file)
+{
+ int m;
+ int result = 0;
+ int menu_quit = 0;
+
+ /* add the resume time to the menu display */
+ char resume_str[32];
+ int time_hol = (int)(settings.resume_time/2);
+ int time_rem = ((settings.resume_time%2)==0) ? 0 : 5;
+ rb->snprintf(resume_str, sizeof(resume_str),
+ "Resume time (min): %d.%d", time_hol, time_rem);
+
+ struct menu_item items[] = {
+ { "Play from beginning", NULL },
+ { resume_str, NULL },
+ { "Set start time (min)", NULL },
+ { "Quit mpegplayer", NULL },
+ };
+
+ m = menu_init(rb, items, sizeof(items) / sizeof(*items),
+ NULL, NULL, NULL, NULL);
+
+ rb->button_clear_queue();
+
+ while(menu_quit == 0)
+ {
+ result = menu_show(m);
+
+ switch (result)
+ {
+ case 0:
+ menu_quit = 1;
+ result = 0;
+ break;
+ case 1:
+ menu_quit = 1;
+ result = settings.resume_time;
+ break;
+ case 2:
+#ifndef HAVE_LCD_COLOR
+ gray_show(true);
+#endif
+ if ((result = get_start_time(play_time, in_file)) >= 0)
+ menu_quit = 1;
+#ifndef HAVE_LCD_COLOR
+ gray_show(false);
+#endif
+ break;
+ case 3:
+ menu_quit = 1;
+ result = -1;
+ break;
+ default:
+ if (result == MENU_ATTACHED_USB)
+ {
+ menu_quit = 1;
+ result = -1;
+ }
+ break;
+ }
+ }
+ menu_exit(m);
+
+ settings.resume_time = result;
+ return (int)result;
+}
+
+void clear_resume_count(void)
+{
+ configfile_save(SETTINGS_FILENAME, config,
+ sizeof(config)/sizeof(*config),
+ SETTINGS_VERSION);
+
+ settings.resume_count = 0;
+
+ /* add this place holder so the count is above resume entries */
+ configfile_update_entry(SETTINGS_FILENAME, "Resume count", 0);
+}
+
bool mpeg_menu(void)
{
int m;
int result;
int menu_quit=0;
- static const struct menu_item items[] = {
+ /* add the clear resume option to the menu display */
+ char clear_str[32];
+ rb->snprintf(clear_str, sizeof(clear_str),
+ "Clear all resumes: %u", settings.resume_count);
+
+ struct menu_item items[] = {
#if defined(TOSHIBA_GIGABEAT_F) || defined(SANSA_E200)
[MPEG_OPTION_DISPLAY_SETTINGS] =
{ "Display Options", NULL },
@@ -85,6 +402,8 @@ bool mpeg_menu(void)
{ "Limit FPS", NULL },
[MPEG_OPTION_SKIP_FRAMES] =
{ "Skip frames", NULL },
+ [MPEG_OPTION_CLEAR_RESUMES] =
+ { clear_str, NULL },
[MPEG_OPTION_QUIT] =
{ "Quit mpegplayer", NULL },
};
@@ -115,6 +434,11 @@ bool mpeg_menu(void)
rb->set_option("Skip frames",&settings.skipframes,INT,
noyes, 2, NULL);
break;
+ case MPEG_OPTION_CLEAR_RESUMES:
+ clear_resume_count();
+ rb->snprintf(clear_str, sizeof(clear_str),
+ "Clear all resumes: %u", 0);
+ break;
case MPEG_OPTION_QUIT:
default:
menu_quit=1;
@@ -132,48 +456,82 @@ bool mpeg_menu(void)
return (result==MPEG_OPTION_QUIT);
}
-
-void init_settings(void)
+void init_settings(const char* filename)
{
/* Set the default settings */
settings.showfps = 0; /* Do not show FPS */
settings.limitfps = 1; /* Limit FPS */
settings.skipframes = 1; /* Skip frames */
+ settings.resume_count = -1;
#if defined(TOSHIBA_GIGABEAT_F) || defined(SANSA_E200)
settings.displayoptions = 0; /* No visual effects */
#endif
configfile_init(rb);
- if (configfile_load(SETTINGS_FILENAME, config,
- sizeof(config)/sizeof(*config),
- SETTINGS_MIN_VERSION
- ) < 0)
+ /* If the config file don't contain resume count
+ or the load fails, then rebuild the config file.
+ This eliminates the worry for older config files
+ having unused data. */
+ if (((settings.resume_count = configfile_get_value
+ (SETTINGS_FILENAME, "Resume count")) < 0) ||
+ (configfile_load(SETTINGS_FILENAME, config,
+ sizeof(config)/sizeof(*config),
+ SETTINGS_MIN_VERSION) < 0))
{
- /* If the loading failed, save a new config file (as the disk is
- already spinning) */
+ /* Generate a new config file with default values */
configfile_save(SETTINGS_FILENAME, config,
sizeof(config)/sizeof(*config),
SETTINGS_VERSION);
}
- /* Keep a copy of the saved version of the settings - so we can check if
- the settings have changed when we quit */
- old_settings = settings;
#if defined(TOSHIBA_GIGABEAT_F) || defined(SANSA_E200)
+ if ((settings.displayoptions =
+ configfile_get_value(SETTINGS_FILENAME, "Display options")) < 0)
+ {
+ configfile_update_entry(SETTINGS_FILENAME, "Display options",
+ (settings.displayoptions=0));
+ }
rb->lcd_yuv_set_options(settings.displayoptions);
#endif
+
+ if (settings.resume_count < 0)
+ {
+ settings.resume_count = 0;
+
+ /* add this place holder so the count is above resume entries */
+ configfile_update_entry(SETTINGS_FILENAME, "Resume count", 0);
+ }
+
+ rb->strcpy(settings.resume_filename, filename);
+
+ /* get the resume time for the current mpeg if it exist */
+ if ((settings.resume_time = configfile_get_value
+ (SETTINGS_FILENAME, filename)) < 0)
+ {
+ settings.resume_time = 0;
+ }
}
void save_settings(void)
{
- /* Save the user settings if they have changed */
- if (rb->memcmp(&settings,&old_settings,sizeof(settings))!=0) {
- configfile_save(SETTINGS_FILENAME, config,
- sizeof(config)/sizeof(*config),
- SETTINGS_VERSION);
+ configfile_update_entry(SETTINGS_FILENAME, "Show FPS",
+ settings.showfps);
+ configfile_update_entry(SETTINGS_FILENAME, "Limit FPS",
+ settings.limitfps);
+ configfile_update_entry(SETTINGS_FILENAME, "Skip frames",
+ settings.skipframes);
- /* Store the settings in old_settings - to check for future changes */
- old_settings = settings;
+ /* If this was a new resume entry then update the total resume count */
+ if (configfile_update_entry(SETTINGS_FILENAME, settings.resume_filename,
+ settings.resume_time) == 0)
+ {
+ configfile_update_entry(SETTINGS_FILENAME, "Resume count",
+ ++settings.resume_count);
}
+
+#if defined(TOSHIBA_GIGABEAT_F) || defined(SANSA_E200)
+ configfile_update_entry(SETTINGS_FILENAME, "Display options",
+ settings.displayoptions);
+#endif
}