X-Git-Url: http://git.openpandora.org/cgi-bin/gitweb.cgi?p=pandora-libraries.git;a=blobdiff_plain;f=minimenu%2Fmmenu.c;h=02fe2a1312cc34935f25baa9e983771f0c3a34c2;hp=d0f12808890826000ab69b0c7e8bc7e95e6bff70;hb=dfdfc979e18d84804e0fff5d87bc6c2c1a875e66;hpb=66767ebbbf5f79c3d36d53889e93f3bbe95ce878;ds=sidebyside diff --git a/minimenu/mmenu.c b/minimenu/mmenu.c index d0f1280..02fe2a1 100644 --- a/minimenu/mmenu.c +++ b/minimenu/mmenu.c @@ -35,6 +35,8 @@ #include #include #include +#include +#include // for sigaction #include "pnd_logger.h" #include "pnd_pxml.h" @@ -45,6 +47,10 @@ #include "pnd_locate.h" #include "pnd_device.h" #include "pnd_pndfiles.h" +#include "../lib/pnd_pathiter.h" +#include "pnd_notify.h" +#include "pnd_dbusnotify.h" +#include "pnd_apps.h" #include "mmenu.h" #include "mmwrapcmd.h" @@ -52,6 +58,7 @@ #include "mmcache.h" #include "mmcat.h" #include "mmui.h" +#include "mmconf.h" pnd_box_handle g_active_apps = NULL; unsigned int g_active_appcount = 0; @@ -63,11 +70,18 @@ char *pnd_run_script = NULL; unsigned char g_x11_present = 1; // >0 if X is present unsigned char g_catmap = 0; // if 1, we're doing category mapping unsigned char g_pvwcache = 0; // if 1, we're trying to do preview caching +unsigned char g_autorescan = 1; // default to auto rescan + +pnd_dbusnotify_handle dbh = 0; // if >0, dbusnotify is active +pnd_notify_handle nh = 0; // if >0, inotify is active char g_skin_selected [ 100 ] = "default"; char *g_skinpath = NULL; // where 'skin_selected' is located .. the fullpath including skin-dir-name pnd_conf_handle g_skinconf = NULL; +void sigquit_handler ( int n ); +unsigned char app_is_visible ( pnd_conf_handle h, char *uniqueid ); + int main ( int argc, char *argv[] ) { int logall = -1; // -1 means normal logging rules; >=0 means log all! int i; @@ -155,8 +169,10 @@ int main ( int argc, char *argv[] ) { } // spin pnd_log ( pndn_rem, "Looks like user '%s' is in, continue.\n", g_username ); - /* conf file + /* conf files */ + + // mmenu conf g_conf = pnd_conf_fetch_by_name ( MMENU_CONF, MMENU_CONF_SEARCHPATH ); if ( ! g_conf ) { @@ -164,6 +180,11 @@ int main ( int argc, char *argv[] ) { emit_and_quit ( MM_QUIT ); } + // override mmenu conf via user preference conf + conf_merge_into ( conf_determine_location ( g_conf ), g_conf ); + conf_setup_missing ( g_conf ); + + // desktop conf for app finding preferences g_desktopconf = pnd_conf_fetch_by_id ( pnd_conf_desktop, PND_CONF_SEARCHPATH ); if ( ! g_desktopconf ) { @@ -171,6 +192,36 @@ int main ( int argc, char *argv[] ) { emit_and_quit ( MM_QUIT ); } + /* set up quit signal handler + */ + sigset_t ss; + sigemptyset ( &ss ); + + struct sigaction siggy; + siggy.sa_handler = sigquit_handler; + siggy.sa_mask = ss; /* implicitly blocks the origin signal */ + siggy.sa_flags = SA_RESTART; /* don't need anything */ + sigaction ( SIGQUIT, &siggy, NULL ); + + /* category conf file + */ + { + char *locater = pnd_locate_filename ( pnd_conf_get_as_char ( g_conf, "categories.catmap_searchpath" ), + pnd_conf_get_as_char ( g_conf, "categories.catmap_confname" ) ); + + if ( locater ) { + pnd_log ( pndn_rem, "Found category conf at '%s'\n", locater ); + pnd_conf_handle h = pnd_conf_fetch_by_path ( locater ); + if ( h ) { + // lets just merge the skin conf onto the regular conf, so it just magicly works + pnd_box_append ( g_conf, h ); + } + } else { + pnd_log ( pndn_debug, "No additional category conf file found.\n" ); + } + + } // cat conf + // redo log filter pnd_log_set_filter ( pnd_conf_get_as_int_d ( g_conf, "minimenu.loglevel", pndn_error ) ); @@ -210,6 +261,12 @@ int main ( int argc, char *argv[] ) { pnd_log ( pndn_rem, "Found pnd_run.sh at '%s'\n", pnd_run_script ); + // auto rescan? + if ( pnd_conf_get_as_int ( g_conf, "minimenu.auto_rescan" ) != PND_CONF_BADNUM ) { + g_autorescan = pnd_conf_get_as_int ( g_conf, "minimenu.auto_rescan" ); + } + pnd_log ( pndn_debug, "application rescan is set to: %s\n", g_autorescan ? "auto" : "manual" ); + // figure out what skin is selected (or default) FILE *f; char *s = strdup ( pnd_conf_get_as_char ( g_conf, "minimenu.skin_selected" ) ); @@ -266,6 +323,17 @@ int main ( int argc, char *argv[] ) { // lets just merge the skin conf onto the regular conf, so it just magicly works pnd_box_append ( g_conf, g_skinconf ); + // did user override the splash image? + char *splash = pnd_conf_get_as_char ( g_conf, "minimenu.force_wallpaper" ); + if ( splash ) { + // we've got a filename, presumably; lets see if it exists + struct stat statbuf; + if ( stat ( splash, &statbuf ) == 0 ) { + // file seems to exist, lets set our override to that.. + pnd_conf_set_char ( g_conf, "graphics.IMG_BACKGROUND_800480", splash ); + } + } + // attempt to set up UI if ( ! ui_setup() ) { pnd_log ( pndn_error, "ERROR: Couldn't set up the UI!\n" ); @@ -286,9 +354,12 @@ int main ( int argc, char *argv[] ) { emit_and_quit ( MM_QUIT ); } + // init categories + category_init(); + // create all cat if ( pnd_conf_get_as_int_d ( g_conf, "categories.do_all_cat", 1 ) ) { - category_push ( g_x11_present ? CATEGORY_ALL " (X11)" : CATEGORY_ALL " (No X11)", NULL, 0 ); + category_push ( g_x11_present ? CATEGORY_ALL " (X11)" : CATEGORY_ALL " (No X11)", NULL /* parent cat */, NULL /*app*/, 0, NULL /* fspath */, 1 /* visible */ ); } // set up category mappings @@ -302,7 +373,33 @@ int main ( int argc, char *argv[] ) { /* actual work now */ + unsigned char block = 1; + + if ( g_autorescan ) { + block = 0; + + // set up notifications + dbh = pnd_dbusnotify_init(); + pnd_log ( pndn_debug, "Setting up dbusnotify\n" ); + //setup_notifications(); + + } // set up rescan + + /* set speed to minimenu run-speed, now that we're all set up + */ +#if 0 /* something crashes at high speed image caching.. */ + int use_mm_speed = pnd_conf_get_as_int_d ( g_conf, "minimenu.use_mm_speed", 0 ); + if ( use_mm_speed > 0 ) { + int mm_speed = pnd_conf_get_as_int_d ( g_conf, "minimenu.mm_speed", -1 ); + if ( mm_speed > 50 && mm_speed < 800 ) { + char buffer [ 512 ]; + snprintf ( buffer, 500, "sudo /usr/pandora/scripts/op_cpuspeed.sh %d", mm_speed ); + system ( buffer ); + } + } // do speed change? +#endif + // do it! while ( 1 ) { // forever! // show the menu, or changes thereof @@ -310,10 +407,31 @@ int main ( int argc, char *argv[] ) { // wait for input or time-based events (like animations) // deal with inputs - ui_process_input ( 1 /* block */ ); + ui_process_input ( block /* block */ ); + + // did a rescan event trigger? + if ( g_autorescan ) { + unsigned char watch_dbus = 0; + unsigned char watch_inotify = 0; + + if ( dbh ) { + watch_dbus = pnd_dbusnotify_rediscover_p ( dbh ); + } + + if ( nh ) { + watch_inotify = pnd_notify_rediscover_p ( nh ); + } + + if ( watch_dbus || watch_inotify ) { + pnd_log ( pndn_debug, "dbusnotify detected SD event\n" ); + applications_free(); + applications_scan(); + } + + } // rescan? // sleep? block? - usleep ( 5000 ); + usleep ( 100000 /*5000*/ ); } // while @@ -322,9 +440,50 @@ int main ( int argc, char *argv[] ) { void emit_and_quit ( char *s ) { printf ( "%s\n", s ); + // shutdown notifications + if ( g_autorescan ) { + + if ( dbh ) { + pnd_dbusnotify_shutdown ( dbh ); + } + if ( nh ) { + pnd_notify_shutdown ( nh ); + } + + } + exit ( 0 ); } +static unsigned int is_dir_empty ( char *fullpath ) { + DIR *d = opendir ( fullpath ); + + if ( ! d ) { + return ( 0 ); // not empty, since we don't know + } + + struct dirent *de = readdir ( d ); + + while ( de ) { + + if ( strcmp ( de -> d_name, "." ) == 0 ) { + // irrelevent + } else if ( strcmp ( de -> d_name, ".." ) == 0 ) { + // irrelevent + } else { + // something else came in, so dir must not be empty + closedir ( d ); + return ( 0 ); + } + + de = readdir ( d ); + } + + closedir ( d ); + + return ( 1 ); // dir is empty +} + void applications_free ( void ) { // free up all our category apprefs, but keep the preview and icon cache's.. @@ -347,6 +506,11 @@ void applications_free ( void ) { void applications_scan ( void ) { + // has user disabled pnd scanning, by chance? + if ( ! pnd_conf_get_as_int_d ( g_conf, "filesystem.do_pnd_disco", 1 ) ) { + goto dirbrowser_scan; // skip pnd's + } + // show disco screen ui_discoverscreen ( 1 /* clear screen */ ); @@ -373,6 +537,23 @@ void applications_scan ( void ) { // merge lists if ( merge_apps ) { if ( g_active_apps ) { + // the key from pnd_disco_search() is the _path_, so easy to look for duplicates + // this is pretty inefficient, being linked lists; perhaps should switch to hash tables when + // we expect thousands of apps.. or at least an index or something. + void *a = pnd_box_get_head ( merge_apps ); + void *nexta = NULL; + while ( a ) { + nexta = pnd_box_get_next ( a ); + + // if the key for the node is also found in active apps, toss out the merging one + if ( pnd_box_find_by_key ( g_active_apps, pnd_box_get_key ( a ) ) ) { + //fprintf ( stderr, "Merging app id '%s' is duplicate; discarding it.\n", pnd_box_get_key ( a ) ); + pnd_box_delete_node ( merge_apps, a ); + } + + a = nexta; + } + // got menu apps, and got desktop apps, merge pnd_box_append ( g_active_apps, merge_apps ); } else { @@ -393,6 +574,28 @@ void applications_scan ( void ) { // merge aux apps if ( merge_apps ) { if ( g_active_apps ) { + + // LAME: snipped from above; should just catenate the 3 sets of searchpaths into a + // master searchpath, possibly removing duplicate paths _then_, and keep all this much + // more efficient + + // the key from pnd_disco_search() is the _path_, so easy to look for duplicates + // this is pretty inefficient, being linked lists; perhaps should switch to hash tables when + // we expect thousands of apps.. or at least an index or something. + void *a = pnd_box_get_head ( merge_apps ); + void *nexta = NULL; + while ( a ) { + nexta = pnd_box_get_next ( a ); + + // if the key for the node is also found in active apps, toss out the merging one + if ( pnd_box_find_by_key ( g_active_apps, pnd_box_get_key ( a ) ) ) { + fprintf ( stderr, "Merging app id '%s' is duplicate; discarding it.\n", pnd_box_get_key ( a ) ); + pnd_box_delete_node ( merge_apps, a ); + } + + a = nexta; + } + pnd_box_append ( g_active_apps, merge_apps ); } else { g_active_apps = merge_apps; @@ -466,52 +669,32 @@ void applications_scan ( void ) { ( pnd_pxml_get_x11 ( iter -> option_no_x11 ) == pnd_pxml_x11_required && g_x11_present == 1 ) ) { + + if ( iter -> title_en == NULL || iter -> title_en [ 0 ] == '\0' ) { + // null title; just skip it. + } else { - // push to All category - // we do this first, so first category is always All - if ( pnd_conf_get_as_int_d ( g_conf, "categories.do_all_cat", 1 ) ) { - if ( ! category_push ( g_x11_present ? CATEGORY_ALL " (X11)" : CATEGORY_ALL " (No X11)", iter, ovrh ) ) { - pnd_log ( pndn_warning, " Couldn't categorize to All: '%s'\n", IFNULL(iter -> title_en, "No Name") ); - } - } // all? + // push to All category + // we do this first, so first category is always All + if ( pnd_conf_get_as_int_d ( g_conf, "categories.do_all_cat", 1 ) ) { + category_push ( g_x11_present ? CATEGORY_ALL " (X11)" : CATEGORY_ALL " (No X11)", NULL /* parent cat */, iter, ovrh, NULL /* fspath */, 1 /* visible */ ); + } // all? - // main categories - if ( iter -> main_category && pnd_conf_get_as_int_d ( g_conf, "tabs.top_maincat", 1 ) ) { - if ( ! category_meta_push ( iter -> main_category, iter, ovrh ) ) { - pnd_log ( pndn_warning, " Couldn't categorize to %s: '%s'\n", iter -> main_category, IFNULL(iter -> title_en, "No Name") ); - } - } + // is this app suppressed? if not, show it in whatever categories the user is allowing + if ( iter -> unique_id && app_is_visible ( g_conf, iter -> unique_id ) ) { - if ( iter -> main_category1 && pnd_conf_get_as_int_d ( g_conf, "tabs.top_maincat1", 0 ) ) { - if ( ! category_meta_push ( iter -> main_category1, iter, ovrh ) ) { - pnd_log ( pndn_warning, " Couldn't categorize to %s: '%s'\n", iter -> main_category1, IFNULL(iter -> title_en, "No Name") ); - } - } + // main categories + category_meta_push ( iter -> main_category, NULL /* no parent cat */, iter, ovrh, cat_is_visible ( g_conf, iter -> main_category ) ); //pnd_conf_get_as_int_d ( g_conf, "tabs.top_maincat", 1 ) ); + category_meta_push ( iter -> main_category1, iter -> main_category, iter, ovrh, cat_is_visible ( g_conf, iter -> main_category1 ) ); //pnd_conf_get_as_int_d ( g_conf, "tabs.top_maincat1", 0 ) ); + category_meta_push ( iter -> main_category2, iter -> main_category, iter, ovrh, cat_is_visible ( g_conf, iter -> main_category2 ) ); //pnd_conf_get_as_int_d ( g_conf, "tabs.top_maincat2", 0 ) ); + // alt categories + category_meta_push ( iter -> alt_category, NULL /* no parent cat */, iter, ovrh, cat_is_visible ( g_conf, iter -> alt_category ) ); //pnd_conf_get_as_int_d ( g_conf, "tabs.top_altcat", 0 ) ); + category_meta_push ( iter -> alt_category1, iter -> alt_category, iter, ovrh, cat_is_visible ( g_conf, iter -> alt_category1 ) ); //pnd_conf_get_as_int_d ( g_conf, "tabs.top_altcat1", 0 ) ); + category_meta_push ( iter -> alt_category2, iter -> alt_category, iter, ovrh, cat_is_visible ( g_conf, iter -> alt_category2 ) ); //pnd_conf_get_as_int_d ( g_conf, "tabs.top_altcat2", 0 ) ); - if ( iter -> main_category2 && pnd_conf_get_as_int_d ( g_conf, "tabs.top_maincat2", 0 ) ) { - if ( ! category_meta_push ( iter -> main_category2, iter, ovrh ) ) { - pnd_log ( pndn_warning, " Couldn't categorize to %s: '%s'\n", iter -> main_category2, IFNULL(iter -> title_en, "No Name") ); - } - } + } // app is visible? - // alt categories - if ( iter -> alt_category && pnd_conf_get_as_int_d ( g_conf, "tabs.top_altcat", 0 ) ) { - if ( ! category_meta_push ( iter -> alt_category, iter, ovrh ) ) { - pnd_log ( pndn_warning, " Couldn't categorize to %s: '%s'\n", iter -> alt_category, IFNULL(iter -> title_en, "No Name") ); - } - } - - if ( iter -> alt_category1 && pnd_conf_get_as_int_d ( g_conf, "tabs.top_altcat1", 0 ) ) { - if ( ! category_meta_push ( iter -> alt_category1, iter, ovrh ) ) { - pnd_log ( pndn_warning, " Couldn't categorize to %s: '%s'\n", iter -> alt_category1, IFNULL(iter -> title_en, "No Name") ); - } - } - - if ( iter -> alt_category2 && pnd_conf_get_as_int_d ( g_conf, "tabs.top_altcat2", 0 ) ) { - if ( ! category_meta_push ( iter -> alt_category2, iter, ovrh ) ) { - pnd_log ( pndn_warning, " Couldn't categorize to %s: '%s'\n", iter -> alt_category2, IFNULL(iter -> title_en, "No Name") ); - } - } + } // has title? } // register with categories or filter out @@ -520,11 +703,165 @@ void applications_scan ( void ) { itercount++; } // while + dirbrowser_scan: + + // set up filesystem browser tabs + if ( pnd_conf_get_as_int_d ( g_conf, "filesystem.do_browser", 0 ) ) { + char *searchpath = pnd_conf_get_as_char ( g_conf, "filesystem.tab_searchpaths" ); + + SEARCHPATH_PRE + { + char *c, *tabname; + c = strrchr ( buffer, '/' ); + if ( c && (*(c+1)!='\0') ) { + tabname = c; + } else { + tabname = buffer; + } + + // check if dir is empty; if so, skip it. + if ( ! is_dir_empty ( buffer ) ) { + category_push ( tabname /* tab name */, NULL /* parent cat */, NULL /* app */, 0 /* override */, buffer /* fspath */, 1 /* visible */ ); + } + + } + SEARCHPATH_POST + + } // set up fs browser tabs + // dump categories //category_dump(); + // publish desired categories + category_publish ( CFNORMAL, NULL ); + // let deferred icon cache go now ui_post_scan(); return; } + +static char _vbuf [ 512 ]; +unsigned char cat_is_visible ( pnd_conf_handle h, char *catname ) { + snprintf ( _vbuf, 500, "tabshow.%s", catname ); + return ( pnd_conf_get_as_int_d ( g_conf, _vbuf, 1 ) ); // default to 'show' when unknown +} + +unsigned char app_is_visible ( pnd_conf_handle h, char *uniqueid ) { + snprintf ( _vbuf, 500, "appshow.%s", uniqueid ); + return ( pnd_conf_get_as_int_d ( g_conf, _vbuf, 1 ) ); // default to 'show' when unknown +} + +void sigquit_handler ( int n ) { + pnd_log ( pndn_rem, "SIGQUIT received; graceful exit.\n" ); + emit_and_quit ( MM_QUIT ); +} + +void setup_notifications ( void ) { + + // figure out notify path + char *configpath; + char *notifypath = NULL; + + configpath = pnd_conf_query_searchpath(); + + pnd_conf_handle apph; + + apph = pnd_conf_fetch_by_id ( pnd_conf_apps, configpath ); + + if ( apph ) { + + notifypath = pnd_conf_get_as_char ( apph, PND_APPS_NOTIFY_KEY ); + + if ( ! notifypath ) { + notifypath = PND_APPS_NOTIFYPATH; + } + + } + + // given notify path.. ripped from pndnotifyd :( + char *searchpath = notifypath; + + // if this is first time through, we can just set it up; for subsequent times + // through, we need to close existing fd and re-open it, since we're too lame + // to store the list of watches and 'rm' them +#if 1 + if ( nh ) { + pnd_notify_shutdown ( nh ); + nh = 0; + } +#endif + + // set up a new set of notifies + if ( ! nh ) { + nh = pnd_notify_init(); + } + + if ( ! nh ) { + pnd_log ( pndn_rem, "INOTIFY failed to init.\n" ); + exit ( -1 ); + } + +#if 0 + pnd_log ( pndn_rem, "INOTIFY is up.\n" ); +#endif + + SEARCHPATH_PRE + { + + pnd_log ( pndn_rem, "Watching path '%s' and its descendents.\n", buffer ); + pnd_notify_watch_path ( nh, buffer, PND_NOTIFY_RECURSE ); + + } + SEARCHPATH_POST + + return; +} + +// for Pleng +// Goal: normally menu will quit when an app is invoked, but there are cases when some folks +// may configure their system and want mmenu to live instead (save on restart time, don't care +// about RAM, using a multitasking tray/window manager setup...), so instead of 'exit and emit' +// here we just run the app directly and cross fingers! +void emit_and_run ( char *buffer ) { + + // run the bloody thing + int f; + + if ( ( f = fork() ) < 0 ) { + // error forking + } else if ( f > 0 ) { + // parent + } else { + // child, do it + execl ( "/bin/sh", "/bin/sh", "-c", buffer + strlen(MM_RUN) + 1, (char*) NULL ); + } + + return; +} + +// this code was swiped from pnd_utility pnd_exec_no_wait_1 as it became a little too minimenu-specific to remain there +void exec_raw_binary ( char *fullpath ) { + int i; + + if ( ( i = fork() ) < 0 ) { + printf ( "ERROR: Couldn't fork()\n" ); + return; + } + + if ( i ) { + return; // parent process, don't care + } + + // child process, do something + execl ( "/bin/sh", "/bin/sh", "-c", fullpath, (char*) NULL ); + //execl ( fullpath, fullpath, (char*) NULL ); + + // error invoking something, and we're the child process, so just die before all hell breaks lose with us thinking we're the (second!) parent on return! + exit ( -1 ); + + // getting here is an error + //printf ( "Error attempting to run %s\n", fullpath ); + + return; +}