2002-11-15 00:47:11 +01:00
|
|
|
|
2006-03-25 21:49:36 +01:00
|
|
|
#include "config.h"
|
|
|
|
|
2002-11-15 00:47:11 +01:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <dirent.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
|
2006-03-25 21:49:36 +01:00
|
|
|
#include "mp_msg.h"
|
|
|
|
#include "help_mp.h"
|
2002-11-15 00:47:11 +01:00
|
|
|
|
2005-10-26 02:07:43 +02:00
|
|
|
#include "m_struct.h"
|
|
|
|
#include "m_option.h"
|
2006-03-25 21:49:36 +01:00
|
|
|
#include "m_property.h"
|
2005-10-26 02:07:43 +02:00
|
|
|
#include "asxparser.h"
|
2002-11-15 00:47:11 +01:00
|
|
|
|
2006-08-18 15:58:03 +02:00
|
|
|
#include "libmpcodecs/img_format.h"
|
|
|
|
#include "libmpcodecs/mp_image.h"
|
2002-11-15 00:47:11 +01:00
|
|
|
|
|
|
|
#include "menu.h"
|
|
|
|
#include "menu_list.h"
|
2005-10-26 02:07:43 +02:00
|
|
|
#include "input/input.h"
|
|
|
|
#include "osdep/keycodes.h"
|
2002-11-15 00:47:11 +01:00
|
|
|
|
2006-08-25 21:22:51 +02:00
|
|
|
#include "codec-cfg.h"
|
|
|
|
#include "stream/stream.h"
|
|
|
|
#include "libmpdemux/demuxer.h"
|
|
|
|
#include "libmpdemux/stheader.h"
|
|
|
|
|
|
|
|
extern demuxer_t *get_current_demuxer (void);
|
|
|
|
|
2002-11-15 00:47:11 +01:00
|
|
|
struct list_entry_s {
|
|
|
|
struct list_entry p;
|
2006-03-25 21:49:36 +01:00
|
|
|
char* name;
|
2002-11-15 00:47:11 +01:00
|
|
|
m_option_t* opt;
|
2006-03-25 21:49:36 +01:00
|
|
|
char* menu;
|
2002-11-15 00:47:11 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct menu_priv_s {
|
|
|
|
menu_list_priv_t p;
|
2006-03-25 21:49:36 +01:00
|
|
|
char* ptr;
|
|
|
|
int edit;
|
2002-11-15 00:47:11 +01:00
|
|
|
/// Cfg fields
|
2006-03-25 21:49:36 +01:00
|
|
|
char* na;
|
|
|
|
int hide_na;
|
2002-11-15 00:47:11 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct menu_priv_s cfg_dflt = {
|
|
|
|
MENU_LIST_PRIV_DFLT,
|
|
|
|
NULL,
|
2006-03-25 21:49:36 +01:00
|
|
|
0,
|
|
|
|
"N/A",
|
|
|
|
1
|
2002-11-15 00:47:11 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
static m_option_t cfg_fields[] = {
|
|
|
|
MENU_LIST_PRIV_FIELDS,
|
|
|
|
{ "title", M_ST_OFF(menu_list_priv_t,title), CONF_TYPE_STRING, 0, 0, 0, NULL },
|
2006-03-25 21:49:36 +01:00
|
|
|
{ "na", M_ST_OFF(struct menu_priv_s,na), CONF_TYPE_STRING, 0, 0, 0, NULL },
|
|
|
|
{ "hide-na", M_ST_OFF(struct menu_priv_s,hide_na), CONF_TYPE_FLAG, CONF_RANGE, 0, 1, NULL },
|
2002-11-15 00:47:11 +01:00
|
|
|
{ NULL, NULL, NULL, 0,0,0,NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
#define mpriv (menu->priv)
|
|
|
|
|
2006-08-25 21:22:51 +02:00
|
|
|
#define OPT_NAME "name"
|
|
|
|
#define OPT_VCODEC "vcodec"
|
|
|
|
#define OPT_VBITRATE "vbitrate"
|
|
|
|
#define OPT_RESOLUTION "resolution"
|
|
|
|
#define OPT_ACODEC "acodec"
|
|
|
|
#define OPT_ABITRATE "abitrate"
|
|
|
|
#define OPT_SAMPLES "asamples"
|
|
|
|
#define OPT_INFO_TITLE "title"
|
|
|
|
#define OPT_INFO_ARTIST "artist"
|
|
|
|
#define OPT_INFO_ALBUM "album"
|
|
|
|
#define OPT_INFO_YEAR "year"
|
|
|
|
#define OPT_INFO_COMMENT "comment"
|
|
|
|
#define OPT_INFO_TRACK "track"
|
|
|
|
#define OPT_INFO_GENRE "genre"
|
|
|
|
|
|
|
|
#define mp_basename(s) (strrchr(s,'/')==NULL?(char*)s:(strrchr(s,'/')+1))
|
|
|
|
|
2006-07-13 15:04:26 +02:00
|
|
|
m_option_t* mp_property_find(const char* name);
|
2006-03-25 21:49:36 +01:00
|
|
|
|
|
|
|
static void entry_set_text(menu_t* menu, list_entry_t* e) {
|
|
|
|
char* val = m_property_print(e->opt);
|
|
|
|
int l,edit = (mpriv->edit && e == mpriv->p.current);
|
|
|
|
if(!val) {
|
|
|
|
if(mpriv->hide_na) {
|
|
|
|
e->p.hide = 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
val = strdup(mpriv->na);
|
|
|
|
} else if(mpriv->hide_na)
|
|
|
|
e->p.hide = 0;
|
|
|
|
l = strlen(e->name) + 2 + strlen(val) + (edit ? 4 : 0) + 1;
|
|
|
|
if(e->p.txt) free(e->p.txt);
|
|
|
|
e->p.txt = malloc(l);
|
|
|
|
sprintf(e->p.txt,"%s: %s%s%s",e->name,edit ? "> " : "",val,edit ? " <" : "");
|
|
|
|
free(val);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void update_entries(menu_t* menu) {
|
|
|
|
list_entry_t* e;
|
|
|
|
for(e = mpriv->p.menu ; e ; e = e->p.next)
|
|
|
|
if(e->opt) entry_set_text(menu,e);
|
|
|
|
}
|
2002-11-15 00:47:11 +01:00
|
|
|
|
2006-08-25 21:22:51 +02:00
|
|
|
static int is_valid_video_property(char *prop) {
|
|
|
|
demuxer_t *demuxer = get_current_demuxer ();
|
|
|
|
sh_video_t *video = (sh_video_t *) demuxer->video->sh;
|
|
|
|
|
|
|
|
if (!prop || !video)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (strcmp (prop, OPT_VCODEC) != 0 &&
|
|
|
|
strcmp (prop, OPT_VBITRATE) != 0 &&
|
|
|
|
strcmp (prop, OPT_RESOLUTION) != 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int is_valid_audio_property(char *prop) {
|
|
|
|
demuxer_t *demuxer = get_current_demuxer ();
|
|
|
|
sh_audio_t *audio = (sh_audio_t *) demuxer->audio->sh;
|
|
|
|
|
|
|
|
if (!prop || !audio)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (strcmp (prop, OPT_ACODEC) != 0 &&
|
|
|
|
strcmp (prop, OPT_ABITRATE) != 0 &&
|
|
|
|
strcmp (prop, OPT_SAMPLES) != 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int is_valid_info_property(char *prop) {
|
|
|
|
demuxer_t *demuxer = get_current_demuxer ();
|
|
|
|
|
|
|
|
if (!prop || !demuxer)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (strcmp (prop, OPT_INFO_TITLE) != 0 &&
|
|
|
|
strcmp (prop, OPT_INFO_ARTIST) != 0 &&
|
|
|
|
strcmp (prop, OPT_INFO_ALBUM) != 0 &&
|
|
|
|
strcmp (prop, OPT_INFO_YEAR) != 0 &&
|
|
|
|
strcmp (prop, OPT_INFO_COMMENT) != 0 &&
|
|
|
|
strcmp (prop, OPT_INFO_TRACK) != 0 &&
|
|
|
|
strcmp (prop, OPT_INFO_GENRE) != 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *grab_demuxer_info(char *tag) {
|
|
|
|
demuxer_t *demuxer = get_current_demuxer ();
|
|
|
|
char **info = demuxer->info;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
if (!info || !tag)
|
|
|
|
return strdup ("");
|
|
|
|
|
|
|
|
for (n = 0; info[2*n] != NULL ; n++)
|
|
|
|
if (!strcmp (info[2*n], tag))
|
|
|
|
break;
|
|
|
|
|
|
|
|
return info[2*n+1] ? strdup (info[2*n+1]) : strdup ("");
|
|
|
|
}
|
|
|
|
|
2002-11-15 00:47:11 +01:00
|
|
|
static int parse_args(menu_t* menu,char* args) {
|
2006-08-25 21:22:51 +02:00
|
|
|
char *element,*body, **attribs, *name, *meta, *val;
|
2002-11-15 00:47:11 +01:00
|
|
|
list_entry_t* m = NULL;
|
|
|
|
int r;
|
|
|
|
m_option_t* opt;
|
|
|
|
ASX_Parser_t* parser = asx_parser_new();
|
|
|
|
|
|
|
|
|
|
|
|
while(1) {
|
|
|
|
r = asx_get_element(parser,&args,&element,&body,&attribs);
|
|
|
|
if(r < 0) {
|
2006-03-30 08:40:58 +02:00
|
|
|
mp_msg(MSGT_OSD_MENU,MSGL_ERR,MSGTR_LIBMENU_SyntaxErrorAtLine,parser->line);
|
2002-11-15 00:47:11 +01:00
|
|
|
asx_parser_free(parser);
|
|
|
|
return -1;
|
|
|
|
} else if(r == 0) {
|
|
|
|
asx_parser_free(parser);
|
|
|
|
if(!m)
|
2006-03-30 08:40:58 +02:00
|
|
|
mp_msg(MSGT_OSD_MENU,MSGL_WARN,MSGTR_LIBMENU_NoEntryFoundInTheMenuDefinition);
|
2006-03-25 21:49:36 +01:00
|
|
|
m = calloc(1,sizeof(struct list_entry_s));
|
|
|
|
m->p.txt = strdup("Back");
|
|
|
|
menu_list_add_entry(menu,m);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if(!strcmp(element,"menu")) {
|
|
|
|
name = asx_get_attrib("menu",attribs);
|
|
|
|
if(!name) {
|
2006-03-30 08:40:58 +02:00
|
|
|
mp_msg(MSGT_OSD_MENU,MSGL_WARN,MSGTR_LIBMENU_SubmenuDefinitionNeedAMenuAttribut);
|
2006-03-25 21:49:36 +01:00
|
|
|
goto next_element;
|
|
|
|
}
|
|
|
|
m = calloc(1,sizeof(struct list_entry_s));
|
|
|
|
m->menu = name;
|
|
|
|
name = NULL; // we want to keep it
|
|
|
|
m->p.txt = asx_get_attrib("name",attribs);
|
|
|
|
if(!m->p.txt) m->p.txt = strdup(m->menu);
|
|
|
|
menu_list_add_entry(menu,m);
|
|
|
|
goto next_element;
|
2002-11-15 00:47:11 +01:00
|
|
|
}
|
2006-03-25 21:49:36 +01:00
|
|
|
|
2006-08-25 21:22:51 +02:00
|
|
|
meta = asx_get_attrib("meta",attribs);
|
|
|
|
val = NULL;
|
|
|
|
if(meta) {
|
|
|
|
demuxer_t *demuxer = get_current_demuxer ();
|
|
|
|
sh_video_t *video = (sh_video_t *) demuxer->video->sh;
|
|
|
|
sh_audio_t *audio = (sh_audio_t *) demuxer->audio->sh;
|
|
|
|
if (!strcmp (meta, OPT_NAME)) {
|
|
|
|
extern char *filename;
|
|
|
|
val = strdup (mp_basename (filename));
|
|
|
|
} else if(!strcmp (meta, OPT_VCODEC) && is_valid_video_property(meta)) {
|
|
|
|
val = (char *) malloc (8);
|
|
|
|
if (video->format == 0x10000001)
|
|
|
|
sprintf (val, "mpeg1");
|
|
|
|
else if (video->format == 0x10000002)
|
|
|
|
sprintf (val, "mpeg2");
|
|
|
|
else if (video->format == 0x10000004)
|
|
|
|
sprintf (val, "mpeg4");
|
|
|
|
else if (video->format == 0x10000005)
|
|
|
|
sprintf (val, "h264");
|
|
|
|
else if (video->format >= 0x20202020)
|
|
|
|
sprintf (val, "%.4s", (char *) &video->format);
|
|
|
|
else
|
|
|
|
sprintf (val, "0x%08X", video->format);
|
|
|
|
} else if (!strcmp(meta, OPT_VBITRATE)&& is_valid_video_property(meta)){
|
|
|
|
val = (char *) malloc (16);
|
|
|
|
sprintf (val, "%d kbps", (int)(video->i_bps * 8 / 1024));
|
|
|
|
} else if(!strcmp(meta, OPT_RESOLUTION)
|
|
|
|
&& is_valid_video_property(meta)) {
|
|
|
|
val = (char *) malloc (16);
|
|
|
|
sprintf(val, "%d x %d", video->disp_w, video->disp_h);
|
|
|
|
} else if (!strcmp(meta, OPT_ACODEC) && is_valid_audio_property(meta)) {
|
|
|
|
val = strdup (audio->codec->name);
|
|
|
|
} else if(!strcmp(meta, OPT_ABITRATE) && is_valid_audio_property(meta)){
|
|
|
|
val = (char *) malloc (16);
|
|
|
|
sprintf (val, "%d kbps", (int) (audio->i_bps * 8/1000));
|
|
|
|
} else if(!strcmp(meta, OPT_SAMPLES) && is_valid_audio_property(meta)) {
|
|
|
|
val = (char *) malloc (16);
|
|
|
|
sprintf (val, "%d Hz, %d ch.", audio->samplerate, audio->channels);
|
|
|
|
} else if ((!strcmp (meta, OPT_INFO_TITLE) ||
|
|
|
|
!strcmp (meta, OPT_INFO_ARTIST) ||
|
|
|
|
!strcmp (meta, OPT_INFO_ALBUM) ||
|
|
|
|
!strcmp (meta, OPT_INFO_YEAR) ||
|
|
|
|
!strcmp (meta, OPT_INFO_COMMENT) ||
|
|
|
|
!strcmp (meta, OPT_INFO_TRACK) ||
|
|
|
|
!strcmp (meta, OPT_INFO_GENRE)) &&
|
|
|
|
is_valid_info_property(meta) &&
|
|
|
|
strcmp(grab_demuxer_info(meta), "") ) {
|
|
|
|
val = grab_demuxer_info (meta);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (val) {
|
|
|
|
char *item = asx_get_attrib("name",attribs);
|
|
|
|
int l;
|
|
|
|
|
|
|
|
if (!item)
|
|
|
|
item = strdup (meta);
|
|
|
|
l = strlen(item) + 2 + strlen(val) + 1;
|
|
|
|
m = calloc(1,sizeof(struct list_entry_s));
|
|
|
|
m->p.txt = malloc(l);
|
|
|
|
sprintf(m->p.txt,"%s: %s",item,val);
|
|
|
|
free(val);
|
|
|
|
free(item);
|
|
|
|
menu_list_add_entry(menu,m);
|
|
|
|
goto next_element;
|
|
|
|
}
|
|
|
|
if (meta)
|
|
|
|
goto next_element;
|
|
|
|
|
2006-03-25 21:49:36 +01:00
|
|
|
name = asx_get_attrib("property",attribs);
|
|
|
|
opt = name ? mp_property_find(name) : NULL;
|
2002-11-15 00:47:11 +01:00
|
|
|
if(!opt) {
|
2006-03-30 08:40:58 +02:00
|
|
|
mp_msg(MSGT_OSD_MENU,MSGL_WARN,MSGTR_LIBMENU_PrefMenuEntryDefinitionsNeed,parser->line);
|
2006-03-25 21:49:36 +01:00
|
|
|
goto next_element;
|
2002-11-15 00:47:11 +01:00
|
|
|
}
|
|
|
|
m = calloc(1,sizeof(struct list_entry_s));
|
|
|
|
m->opt = opt;
|
2006-03-25 21:49:36 +01:00
|
|
|
m->name = asx_get_attrib("name",attribs);
|
|
|
|
if(!m->name) m->name = strdup(opt->name);
|
|
|
|
entry_set_text(menu,m);
|
2002-11-15 00:47:11 +01:00
|
|
|
menu_list_add_entry(menu,m);
|
|
|
|
|
2006-03-25 21:49:36 +01:00
|
|
|
next_element:
|
2002-11-15 00:47:11 +01:00
|
|
|
free(element);
|
|
|
|
if(body) free(body);
|
2006-03-25 21:49:36 +01:00
|
|
|
if(name) free(name);
|
2002-11-15 00:47:11 +01:00
|
|
|
asx_free_attribs(attribs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void read_key(menu_t* menu,int c) {
|
|
|
|
menu_list_read_key(menu,c,0);
|
|
|
|
}
|
|
|
|
|
2006-03-25 21:49:36 +01:00
|
|
|
static void read_cmd(menu_t* menu,int cmd) {
|
|
|
|
list_entry_t* e = mpriv->p.current;
|
|
|
|
|
|
|
|
if(e->opt) {
|
|
|
|
switch(cmd) {
|
|
|
|
case MENU_CMD_UP:
|
|
|
|
if(!mpriv->edit) break;
|
|
|
|
case MENU_CMD_RIGHT:
|
|
|
|
if(m_property_do(e->opt,M_PROPERTY_STEP_UP,NULL) > 0)
|
|
|
|
update_entries(menu);
|
|
|
|
return;
|
|
|
|
case MENU_CMD_DOWN:
|
|
|
|
if(!mpriv->edit) break;
|
|
|
|
case MENU_CMD_LEFT:
|
|
|
|
if(m_property_do(e->opt,M_PROPERTY_STEP_DOWN,NULL) > 0)
|
|
|
|
update_entries(menu);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case MENU_CMD_OK:
|
|
|
|
// check that the property is writable
|
|
|
|
if(m_property_do(e->opt,M_PROPERTY_SET,NULL) < 0) return;
|
|
|
|
// shortcut for flags
|
|
|
|
if(e->opt->type == CONF_TYPE_FLAG) {
|
|
|
|
if(m_property_do(e->opt,M_PROPERTY_STEP_UP,NULL) > 0)
|
|
|
|
update_entries(menu);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// switch
|
|
|
|
mpriv->edit = !mpriv->edit;
|
|
|
|
// update the menu
|
|
|
|
update_entries(menu);
|
|
|
|
// switch the pointer
|
|
|
|
if(mpriv->edit) {
|
|
|
|
mpriv->ptr = mpriv->p.ptr;
|
|
|
|
mpriv->p.ptr = NULL;
|
|
|
|
} else
|
|
|
|
mpriv->p.ptr = mpriv->ptr;
|
|
|
|
return;
|
|
|
|
case MENU_CMD_CANCEL:
|
|
|
|
if(!mpriv->edit) break;
|
|
|
|
mpriv->edit = 0;
|
|
|
|
update_entries(menu);
|
|
|
|
mpriv->p.ptr = mpriv->ptr;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else if(e->menu) {
|
|
|
|
switch(cmd) {
|
|
|
|
case MENU_CMD_RIGHT:
|
|
|
|
case MENU_CMD_OK: {
|
|
|
|
mp_cmd_t* c;
|
|
|
|
char* txt = malloc(10 + strlen(e->menu) + 1);
|
|
|
|
sprintf(txt,"set_menu %s",e->menu);
|
|
|
|
c = mp_input_parse_cmd(txt);
|
|
|
|
if(c) mp_input_queue_cmd(c);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
switch(cmd) {
|
|
|
|
case MENU_CMD_RIGHT:
|
|
|
|
case MENU_CMD_OK:
|
|
|
|
menu->show = 0;
|
|
|
|
menu->cl = 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
menu_list_read_cmd(menu,cmd);
|
|
|
|
}
|
|
|
|
|
2002-11-15 00:47:11 +01:00
|
|
|
static void free_entry(list_entry_t* entry) {
|
|
|
|
free(entry->p.txt);
|
2006-03-25 21:49:36 +01:00
|
|
|
if(entry->name) free(entry->name);
|
|
|
|
if(entry->menu) free(entry->menu);
|
2002-11-15 00:47:11 +01:00
|
|
|
free(entry);
|
|
|
|
}
|
|
|
|
|
2003-09-13 20:02:20 +02:00
|
|
|
static void closeMenu(menu_t* menu) {
|
2002-11-15 00:47:11 +01:00
|
|
|
menu_list_uninit(menu,free_entry);
|
|
|
|
}
|
|
|
|
|
2003-09-13 20:02:20 +02:00
|
|
|
static int openMenu(menu_t* menu, char* args) {
|
2002-11-15 00:47:11 +01:00
|
|
|
|
|
|
|
menu->draw = menu_list_draw;
|
2006-03-25 21:49:36 +01:00
|
|
|
menu->read_cmd = read_cmd;
|
2002-11-15 00:47:11 +01:00
|
|
|
menu->read_key = read_key;
|
2003-09-13 20:02:20 +02:00
|
|
|
menu->close = closeMenu;
|
2002-11-15 00:47:11 +01:00
|
|
|
|
|
|
|
|
|
|
|
if(!args) {
|
2006-03-31 07:49:05 +02:00
|
|
|
mp_msg(MSGT_OSD_MENU,MSGL_ERR,MSGTR_LIBMENU_PrefMenuNeedsAnArgument);
|
2002-11-15 00:47:11 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
menu_list_init(menu);
|
2006-03-25 21:49:36 +01:00
|
|
|
return parse_args(menu,args);
|
2002-11-15 00:47:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const menu_info_t menu_info_pref = {
|
|
|
|
"Preferences menu",
|
|
|
|
"pref",
|
|
|
|
"Albeu",
|
|
|
|
"",
|
|
|
|
{
|
|
|
|
"pref_cfg",
|
|
|
|
sizeof(struct menu_priv_s),
|
|
|
|
&cfg_dflt,
|
|
|
|
cfg_fields
|
|
|
|
},
|
2003-09-13 20:02:20 +02:00
|
|
|
openMenu
|
2002-11-15 00:47:11 +01:00
|
|
|
};
|