1694 lines
40 KiB
C
1694 lines
40 KiB
C
#include "e.h"
|
|
#include "evry_api.h"
|
|
|
|
/* Increment for Major Changes */
|
|
#define MOD_CONFIG_FILE_EPOCH 1
|
|
/* Increment for Minor Changes (ie: user doesn't need a new config) */
|
|
#define MOD_CONFIG_FILE_GENERATION 0
|
|
#define MOD_CONFIG_FILE_VERSION ((MOD_CONFIG_FILE_EPOCH * 1000000) + MOD_CONFIG_FILE_GENERATION)
|
|
|
|
// FIXME clear cache on .desktop chage event
|
|
|
|
/* #undef DBG
|
|
* #define DBG(...) ERR(__VA_ARGS__) */
|
|
|
|
#define MAX_ITEMS 200
|
|
#define MAX_EXE 50
|
|
#define DEFAULT_MATCH_PRIORITY 15
|
|
#define PREFIX_MATCH_PRIORITY 11
|
|
|
|
typedef struct _Plugin Plugin;
|
|
typedef struct _Module_Config Module_Config;
|
|
typedef struct _E_Exe E_Exe;
|
|
typedef struct _E_Exe_List E_Exe_List;
|
|
typedef struct _Item_Menu Item_Menu;
|
|
|
|
struct _Plugin
|
|
{
|
|
Evry_Plugin base;
|
|
Eina_Bool browse;
|
|
const char *input;
|
|
Eina_List *apps_mime;
|
|
Eina_List *apps_all;
|
|
Eina_List *apps_hist;
|
|
Eina_List *menu_items;
|
|
|
|
Eina_Hash *added;
|
|
Efreet_Menu *menu;
|
|
|
|
Evry_Item_App *command;
|
|
};
|
|
|
|
struct _E_Exe
|
|
{
|
|
unsigned int len;
|
|
const char *path;
|
|
};
|
|
|
|
struct _E_Exe_List
|
|
{
|
|
Eina_List *list;
|
|
};
|
|
|
|
struct _Module_Config
|
|
{
|
|
int version;
|
|
const char *cmd_terminal;
|
|
const char *cmd_sudo;
|
|
|
|
E_Config_Dialog *cfd;
|
|
E_Module *module;
|
|
};
|
|
|
|
struct _Item_Menu
|
|
{
|
|
Evry_Item base;
|
|
Efreet_Menu *menu;
|
|
};
|
|
|
|
static const Evry_API *evry = NULL;
|
|
static Evry_Module *evry_module = NULL;
|
|
static Eina_List *handlers = NULL;
|
|
|
|
static Module_Config *_conf;
|
|
static char _module_icon[] = "system-run";
|
|
static Eina_List *_plugins = NULL;
|
|
static Eina_List *_actions = NULL;
|
|
static Evry_Item *_act_open_with = NULL;
|
|
|
|
static Eina_List *exe_list = NULL;
|
|
static Eina_List *apps_cache = NULL;
|
|
|
|
static char _exebuf_cache_file[] = "evry_exebuf_cache";
|
|
static Eina_Bool update_path;
|
|
static char *current_path = NULL;
|
|
static Eina_List *dir_monitors = NULL;
|
|
static Eina_List *exe_path = NULL;
|
|
static Ecore_Idler *exe_scan_idler = NULL;
|
|
static E_Config_DD *exelist_exe_edd = NULL;
|
|
static E_Config_DD *exelist_edd = NULL;
|
|
static Eina_Iterator *exe_dir = NULL;
|
|
static Eina_List *exe_files = NULL;
|
|
|
|
static void _scan_executables();
|
|
|
|
#define GET_MENU(_m, _it) Item_Menu * _m = (Item_Menu *)_it
|
|
|
|
/***************************************************************************/
|
|
|
|
static void
|
|
_hash_free(void *data)
|
|
{
|
|
GET_APP(app, data);
|
|
EVRY_ITEM_FREE(app);
|
|
}
|
|
|
|
static int
|
|
_exec_open_file_action(Evry_Action *act)
|
|
{
|
|
return evry->util_exec_app(EVRY_ITEM(act), act->it1.item);
|
|
}
|
|
|
|
static Evry_Item_App *
|
|
_item_new(Plugin *p, const char *label, const char *id)
|
|
{
|
|
Evry_Item_App *app;
|
|
|
|
app = EVRY_ITEM_NEW(Evry_Item_App, p, label, NULL, evry_item_app_free);
|
|
EVRY_ACTN(app)->action = &_exec_open_file_action;
|
|
EVRY_ACTN(app)->it1.type = EVRY_TYPE_FILE;
|
|
EVRY_ITEM(app)->id = eina_stringshare_add(id);
|
|
|
|
eina_hash_add(p->added, id, app);
|
|
|
|
EVRY_ACTN(app)->remember_context = EINA_TRUE;
|
|
EVRY_ITEM(app)->subtype = EVRY_TYPE_ACTION;
|
|
|
|
return app;
|
|
}
|
|
|
|
static Item_Menu *
|
|
_item_menu_add(Plugin *p, Efreet_Menu *menu)
|
|
{
|
|
Item_Menu *m;
|
|
|
|
m = EVRY_ITEM_NEW(Item_Menu, p, NULL, NULL, NULL);
|
|
EVRY_ITEM(m)->type = EVRY_TYPE_NONE;
|
|
EVRY_ITEM(m)->browseable = EINA_TRUE;
|
|
EVRY_ITEM(m)->label = eina_stringshare_add(menu->name);
|
|
EVRY_ITEM(m)->icon = eina_stringshare_add(menu->icon);
|
|
EVRY_ITEM(m)->usage = -1;
|
|
|
|
m->menu = menu;
|
|
p->menu_items = eina_list_append(p->menu_items, m);
|
|
|
|
return m;
|
|
}
|
|
|
|
static int
|
|
_cb_sort(const void *data1, const void *data2)
|
|
{
|
|
const Evry_Item *it1 = data1;
|
|
const Evry_Item *it2 = data2;
|
|
|
|
if ((!eina_dbl_exact(it1->usage, 0)) && (!eina_dbl_exact(it2->usage, 0)))
|
|
return it1->usage > it2->usage ? -1 : 1;
|
|
if ((!eina_dbl_exact(it1->usage, 0)) && eina_dbl_exact(it2->usage, 0))
|
|
return -1;
|
|
if ((!eina_dbl_exact(it2->usage, 0)) && eina_dbl_exact(it1->usage, 0))
|
|
return 1;
|
|
|
|
if (it1->fuzzy_match || it2->fuzzy_match)
|
|
{
|
|
if (it1->fuzzy_match && !it2->fuzzy_match)
|
|
return -1;
|
|
|
|
if (!it1->fuzzy_match && it2->fuzzy_match)
|
|
return 1;
|
|
|
|
if (it1->fuzzy_match - it2->fuzzy_match)
|
|
return it1->fuzzy_match - it2->fuzzy_match;
|
|
}
|
|
|
|
return strcasecmp(it1->label, it2->label);
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
static Evry_Item_App *
|
|
_item_exe_add(Plugin *p, const char *exe, int match)
|
|
{
|
|
Evry_Item_App *app = NULL;
|
|
|
|
if ((app = eina_hash_find(p->added, exe)))
|
|
{
|
|
if (eina_list_data_find_list(p->base.items, app))
|
|
return app;
|
|
}
|
|
|
|
if (!app)
|
|
{
|
|
app = _item_new(p, ecore_file_file_get(exe), exe);
|
|
app->file = eina_stringshare_ref(EVRY_ITEM(app)->id);
|
|
}
|
|
|
|
EVRY_ITEM(app)->fuzzy_match = match;
|
|
EVRY_PLUGIN_ITEM_APPEND(p, app);
|
|
|
|
return app;
|
|
}
|
|
|
|
static Eina_Bool
|
|
_hist_exe_get_cb(const Eina_Hash *hash EINA_UNUSED, const void *key, void *data, void *fdata)
|
|
{
|
|
History_Entry *he = data;
|
|
History_Item *hi;
|
|
Plugin *p = fdata;
|
|
Eina_List *l;
|
|
Evry_Item_App *app;
|
|
int match;
|
|
const char *exe = key;
|
|
|
|
EINA_LIST_FOREACH (he->items, l, hi)
|
|
{
|
|
app = NULL;
|
|
|
|
if (strcmp(hi->plugin, EVRY_PLUGIN(p)->name))
|
|
continue;
|
|
|
|
if (!p->input)
|
|
{
|
|
app = _item_exe_add(p, exe, 0);
|
|
}
|
|
else if ((match = evry->fuzzy_match(exe, p->input)))
|
|
{
|
|
app = _item_exe_add(p, exe, match);
|
|
}
|
|
|
|
if (app)
|
|
{
|
|
EVRY_ITEM(app)->hi = hi;
|
|
evry->history_item_usage_set(EVRY_ITEM(app), p->input, NULL);
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
return EINA_TRUE;
|
|
}
|
|
|
|
static int
|
|
_fetch_exe(Evry_Plugin *plugin, const char *input)
|
|
{
|
|
GET_PLUGIN(p, plugin);
|
|
Eina_List *l;
|
|
Evry_Item *eit;
|
|
History_Types *ht;
|
|
unsigned int input_len = (input ? strlen(input) : 0);
|
|
double max = 0.0;
|
|
unsigned int cnt = 0, end = input_len;
|
|
Eina_Bool query = (input_len >= (unsigned int)plugin->config->min_query);
|
|
EVRY_PLUGIN_ITEMS_CLEAR(p);
|
|
|
|
p->input = input;
|
|
|
|
ht = evry->history_types_get(EVRY_TYPE_APP);
|
|
if (ht) eina_hash_foreach(ht->types, _hist_exe_get_cb, p);
|
|
|
|
if (input)
|
|
{
|
|
const char *tmp;
|
|
const E_Exe *ee, *match = NULL;
|
|
// begin of arguments (end of executable part)
|
|
if ((tmp = strchr(input, ' ')))
|
|
end = tmp - input;
|
|
|
|
if ((!exe_list) && (!exe_scan_idler))
|
|
_scan_executables();
|
|
EINA_LIST_FOREACH (exe_list, l, ee)
|
|
{
|
|
if ((end < input_len) && (ee->len > end))
|
|
continue;
|
|
if (!strncmp(input, ee->path, end))
|
|
{
|
|
if (query && (cnt++ < MAX_EXE) && (input_len != ee->len))
|
|
_item_exe_add(p, ee->path, DEFAULT_MATCH_PRIORITY);
|
|
|
|
if ((!match) || (ee->len < match->len))
|
|
match = ee;
|
|
|
|
if ((!query) && (ee->len == input_len))
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (match)
|
|
{
|
|
GET_ITEM(it, p->command);
|
|
const char *cmd;
|
|
|
|
if (match->len < input_len)
|
|
cmd = input;
|
|
else
|
|
cmd = match->path;
|
|
|
|
EVRY_ITEM_LABEL_SET(it, cmd);
|
|
|
|
IF_RELEASE(p->command->file);
|
|
p->command->file = eina_stringshare_ref(it->label);
|
|
it->fuzzy_match = PREFIX_MATCH_PRIORITY;
|
|
EVRY_PLUGIN_ITEM_APPEND(p, it);
|
|
evry->item_changed(it, 0, 0);
|
|
}
|
|
}
|
|
|
|
EINA_LIST_FOREACH (plugin->items, l, eit)
|
|
{
|
|
evry->history_item_usage_set(eit, input, NULL);
|
|
if (input && (eit->usage > max) && !strncmp(input, eit->label, input_len))
|
|
max = eit->usage;
|
|
}
|
|
EVRY_ITEM(p->command)->usage = (max * 2.0);
|
|
|
|
EVRY_PLUGIN_ITEMS_SORT(p, _cb_sort);
|
|
|
|
return EVRY_PLUGIN_HAS_ITEMS(p);
|
|
}
|
|
|
|
static Evry_Plugin *
|
|
_begin_exe(Evry_Plugin *plugin, const Evry_Item *item)
|
|
{
|
|
Plugin *p;
|
|
Evry_Item_App *app;
|
|
|
|
if (item && (item != _act_open_with))
|
|
return NULL;
|
|
|
|
EVRY_PLUGIN_INSTANCE(p, plugin)
|
|
|
|
p->added = eina_hash_string_small_new(_hash_free);
|
|
|
|
app = EVRY_ITEM_NEW(Evry_Item_App, p, NULL, NULL, evry_item_app_free);
|
|
EVRY_ACTN(app)->action = &_exec_open_file_action;
|
|
EVRY_ACTN(app)->remember_context = EINA_TRUE;
|
|
EVRY_ITEM(app)->subtype = EVRY_TYPE_ACTION;
|
|
p->command = app;
|
|
|
|
return EVRY_PLUGIN(p);
|
|
}
|
|
|
|
static void
|
|
_finish_exe(Evry_Plugin *plugin)
|
|
{
|
|
GET_PLUGIN(p, plugin);
|
|
char *str;
|
|
E_Exe *ee;
|
|
|
|
EVRY_PLUGIN_ITEMS_CLEAR(p);
|
|
EVRY_ITEM_FREE(p->command);
|
|
|
|
if (p->added)
|
|
eina_hash_free(p->added);
|
|
|
|
if (exe_dir)
|
|
{
|
|
eina_iterator_free(exe_dir);
|
|
exe_dir = NULL;
|
|
}
|
|
EINA_LIST_FREE (exe_path, str)
|
|
free(str);
|
|
|
|
if (exe_scan_idler)
|
|
{
|
|
ecore_idler_del(exe_scan_idler);
|
|
exe_scan_idler = NULL;
|
|
}
|
|
|
|
EINA_LIST_FREE (exe_list, ee)
|
|
{
|
|
eina_stringshare_del(ee->path);
|
|
free(ee);
|
|
}
|
|
|
|
EINA_LIST_FREE (exe_files, str)
|
|
eina_stringshare_del(str);
|
|
|
|
E_FREE(p);
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
static void
|
|
_item_desktop_add(Plugin *p, Efreet_Desktop *desktop, int match)
|
|
{
|
|
Evry_Item_App *app = NULL;
|
|
|
|
if (desktop->no_display) return;
|
|
if ((app = eina_hash_find(p->added, desktop->exec)))
|
|
{
|
|
if (eina_list_data_find_list(p->base.items, app))
|
|
return;
|
|
}
|
|
|
|
if (!app)
|
|
{
|
|
app = _item_new(p, desktop->name, desktop->exec);
|
|
efreet_desktop_ref(desktop);
|
|
app->desktop = desktop;
|
|
|
|
if (desktop->comment)
|
|
{
|
|
EVRY_ITEM_DETAIL_SET(app, desktop->comment);
|
|
}
|
|
else if (desktop->generic_name)
|
|
{
|
|
EVRY_ITEM_DETAIL_SET(app, desktop->generic_name);
|
|
}
|
|
}
|
|
|
|
EVRY_ITEM(app)->fuzzy_match = match;
|
|
EVRY_PLUGIN_ITEM_APPEND(p, app);
|
|
}
|
|
|
|
static void
|
|
_desktop_list_add(Plugin *p, Eina_List *apps, const char *input)
|
|
{
|
|
Efreet_Desktop *desktop;
|
|
Eina_List *l;
|
|
int m1, m2;
|
|
const char *exec, *end;
|
|
char buf[PATH_MAX];
|
|
|
|
EINA_LIST_FOREACH (apps, l, desktop)
|
|
{
|
|
if (eina_list_count(p->base.items) >= MAX_ITEMS) break;
|
|
|
|
m1 = m2 = 0;
|
|
|
|
if (input)
|
|
{
|
|
/* strip path and parameter */
|
|
exec = ecore_file_file_get(desktop->exec);
|
|
if ((exec) && (end = strchr(exec, '%')) &&
|
|
((end - exec) - 1 > 0))
|
|
{
|
|
strncpy(buf, exec, (end - exec) - 1);
|
|
buf[(end - exec) - 1] = '\0';
|
|
m1 = evry->fuzzy_match(buf, input);
|
|
}
|
|
else
|
|
{
|
|
m1 = evry->fuzzy_match(exec, input);
|
|
}
|
|
|
|
m2 = evry->fuzzy_match(desktop->name, input);
|
|
|
|
if (!m1 || (m2 && m2 < m1)) m1 = m2;
|
|
}
|
|
|
|
if (!input || m1) _item_desktop_add(p, desktop, m1);
|
|
}
|
|
}
|
|
|
|
static Eina_List *
|
|
_desktop_list_get(void)
|
|
{
|
|
Eina_List *apps = NULL;
|
|
Eina_List *cat_ss;
|
|
Eina_List *l, *ll;
|
|
Efreet_Desktop *d;
|
|
|
|
apps = efreet_util_desktop_name_glob_list("*");
|
|
|
|
/* remove screensaver */
|
|
cat_ss = efreet_util_desktop_category_list("Screensaver");
|
|
EINA_LIST_FREE(cat_ss, d)
|
|
{
|
|
if ((ll = eina_list_data_find_list(apps, d)))
|
|
{
|
|
efreet_desktop_free(d);
|
|
apps = eina_list_remove_list(apps, ll);
|
|
}
|
|
//printf("%d %s\n", d->ref, d->name);
|
|
|
|
efreet_desktop_free(d);
|
|
}
|
|
EINA_LIST_FOREACH_SAFE(apps, l, ll, d)
|
|
{
|
|
if (!d->no_display) continue;
|
|
apps = eina_list_remove_list(apps, l);
|
|
efreet_desktop_free(d);
|
|
}
|
|
|
|
return apps;
|
|
}
|
|
|
|
static Eina_Bool
|
|
_hist_items_get_cb(const Eina_Hash *hash EINA_UNUSED, const void *key, void *data, void *fdata)
|
|
{
|
|
History_Entry *he = data;
|
|
History_Item *hi;
|
|
Plugin *p = fdata;
|
|
Efreet_Desktop *d;
|
|
Eina_List *l, *ll;
|
|
const char *exec = key;
|
|
|
|
EINA_LIST_FOREACH (he->items, l, hi)
|
|
{
|
|
d = NULL;
|
|
|
|
if (hi->transient)
|
|
continue;
|
|
|
|
if (strcmp(hi->plugin, EVRY_PLUGIN(p)->name))
|
|
continue;
|
|
|
|
EINA_LIST_FOREACH (apps_cache, ll, d)
|
|
if (d->exec && !strcmp(d->exec, exec)) break;
|
|
|
|
if (!d)
|
|
{
|
|
if (!p->apps_all)
|
|
p->apps_all = _desktop_list_get();
|
|
|
|
EINA_LIST_FOREACH (p->apps_all, ll, d)
|
|
if (d->exec && !strcmp(d->exec, exec)) break;
|
|
|
|
if (d && (!d->no_display))
|
|
{
|
|
efreet_desktop_ref(d);
|
|
apps_cache = eina_list_append(apps_cache, d);
|
|
}
|
|
else
|
|
{
|
|
/* remove from history */
|
|
/* DBG("no desktop: %s", exec);
|
|
* hi->transient = 1; */
|
|
}
|
|
}
|
|
|
|
if (!d)
|
|
{
|
|
DBG("app not found %s", exec);
|
|
break;
|
|
}
|
|
|
|
p->apps_hist = eina_list_append(p->apps_hist, d);
|
|
break;
|
|
}
|
|
|
|
return EINA_TRUE;
|
|
}
|
|
|
|
static Evry_Plugin *
|
|
_begin(Evry_Plugin *plugin, const Evry_Item *item)
|
|
{
|
|
Plugin *p;
|
|
|
|
if (item && (item != _act_open_with))
|
|
return NULL;
|
|
|
|
EVRY_PLUGIN_INSTANCE(p, plugin);
|
|
p->added = eina_hash_string_small_new(_hash_free);
|
|
p->menu = efreet_menu_get();
|
|
|
|
return EVRY_PLUGIN(p);
|
|
}
|
|
|
|
static Evry_Plugin *
|
|
_browse(Evry_Plugin *plugin, const Evry_Item *item)
|
|
{
|
|
Plugin *p;
|
|
|
|
if (!item)
|
|
return NULL;
|
|
|
|
if (!CHECK_TYPE(item, EVRY_TYPE_NONE))
|
|
return NULL;
|
|
|
|
EVRY_PLUGIN_INSTANCE(p, plugin);
|
|
GET_MENU(m, item);
|
|
|
|
p->added = eina_hash_string_small_new(_hash_free);
|
|
p->menu = m->menu;
|
|
p->browse = EINA_TRUE;
|
|
|
|
return EVRY_PLUGIN(p);
|
|
}
|
|
|
|
static void
|
|
_finish(Evry_Plugin *plugin)
|
|
{
|
|
Efreet_Desktop *desktop;
|
|
Evry_Item *it;
|
|
|
|
GET_PLUGIN(p, plugin);
|
|
|
|
EVRY_PLUGIN_ITEMS_CLEAR(p);
|
|
|
|
/* TODO share with browse instances */
|
|
if (p->added)
|
|
eina_hash_free(p->added);
|
|
|
|
if ((!p->browse) && (p->menu))
|
|
efreet_menu_free(p->menu);
|
|
|
|
EINA_LIST_FREE (p->apps_all, desktop)
|
|
efreet_desktop_free(desktop);
|
|
|
|
EINA_LIST_FREE (p->apps_hist, desktop) ;
|
|
|
|
EINA_LIST_FREE (p->apps_mime, desktop)
|
|
efreet_desktop_free(desktop);
|
|
|
|
EINA_LIST_FREE (p->menu_items, it)
|
|
EVRY_ITEM_FREE(it);
|
|
|
|
E_FREE(p);
|
|
}
|
|
|
|
static int
|
|
_fetch(Evry_Plugin *plugin, const char *input)
|
|
{
|
|
GET_PLUGIN(p, plugin);
|
|
Eina_List *l;
|
|
Evry_Item *it;
|
|
History_Types *ht;
|
|
|
|
Efreet_Menu *entry;
|
|
int i = 1;
|
|
|
|
EVRY_PLUGIN_ITEMS_CLEAR(p);
|
|
|
|
if (!p->browse)
|
|
{
|
|
if (input)
|
|
{
|
|
if (!p->apps_all)
|
|
p->apps_all = _desktop_list_get();
|
|
|
|
_desktop_list_add(p, p->apps_all, input);
|
|
}
|
|
else
|
|
{
|
|
_desktop_list_add(p, p->apps_mime, input);
|
|
}
|
|
|
|
if ((!input) && (!plugin->items))
|
|
{
|
|
if (!p->apps_hist)
|
|
{
|
|
ht = evry->history_types_get(EVRY_TYPE_APP);
|
|
if (ht) eina_hash_foreach(ht->types, _hist_items_get_cb, p);
|
|
}
|
|
|
|
_desktop_list_add(p, p->apps_hist, NULL);
|
|
}
|
|
|
|
EINA_LIST_FOREACH (plugin->items, l, it)
|
|
evry->history_item_usage_set(it, input, NULL);
|
|
|
|
EVRY_PLUGIN_ITEMS_SORT(p, _cb_sort);
|
|
}
|
|
|
|
if ((p->menu) && (!p->menu_items))
|
|
{
|
|
EINA_LIST_FOREACH (p->menu->entries, l, entry)
|
|
{
|
|
if (entry->type == EFREET_MENU_ENTRY_DESKTOP)
|
|
{
|
|
_item_desktop_add(p, entry->desktop, i++);
|
|
}
|
|
else if (entry->type == EFREET_MENU_ENTRY_MENU)
|
|
{
|
|
_item_menu_add(p, entry);
|
|
}
|
|
/* else if (entry->type == EFREET_MENU_ENTRY_SEPARATOR)
|
|
* continue;
|
|
* else if (entry->type == EFREET_MENU_ENTRY_HEADER)
|
|
* continue; */
|
|
}
|
|
}
|
|
|
|
EINA_LIST_FOREACH (p->menu_items, l, it)
|
|
{
|
|
if (!input)
|
|
{
|
|
EVRY_PLUGIN_ITEM_APPEND(p, it);
|
|
continue;
|
|
}
|
|
|
|
if ((it->fuzzy_match = evry->fuzzy_match(it->label, input)))
|
|
EVRY_PLUGIN_ITEM_APPEND(p, it);
|
|
}
|
|
|
|
return EVRY_PLUGIN_HAS_ITEMS(p);
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
static Evry_Plugin *
|
|
_begin_mime(Evry_Plugin *plugin, const Evry_Item *item)
|
|
{
|
|
Plugin *p = NULL;
|
|
Efreet_Desktop *d;
|
|
const char *mime;
|
|
const char *path = NULL;
|
|
Eina_List *l;
|
|
|
|
if (CHECK_TYPE(item, EVRY_TYPE_ACTION))
|
|
{
|
|
GET_ACTION(act, item);
|
|
GET_FILE(file, act->it1.item);
|
|
|
|
if (!evry->file_path_get(file))
|
|
return NULL;
|
|
|
|
path = file->path;
|
|
mime = file->mime;
|
|
}
|
|
else if (CHECK_TYPE(item, EVRY_TYPE_FILE))
|
|
{
|
|
GET_FILE(file, item);
|
|
|
|
if (!evry->file_path_get(file))
|
|
return NULL;
|
|
|
|
path = file->path;
|
|
mime = file->mime;
|
|
}
|
|
else
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
if (!path || !mime || !(mime = efreet_mime_type_get(path)))
|
|
return NULL;
|
|
|
|
EVRY_PLUGIN_INSTANCE(p, plugin);
|
|
|
|
p->apps_mime = efreet_util_desktop_mime_list(mime);
|
|
|
|
if (strcmp(mime, "text/plain") && (!strncmp(mime, "text/", 5)))
|
|
{
|
|
l = efreet_util_desktop_mime_list("text/plain");
|
|
|
|
EINA_LIST_FREE (l, d)
|
|
{
|
|
if ((!d->no_display) && (!eina_list_data_find_list(p->apps_mime, d)))
|
|
p->apps_mime = eina_list_append(p->apps_mime, d);
|
|
else
|
|
efreet_desktop_free(d);
|
|
}
|
|
}
|
|
|
|
if (item->browseable && strcmp(mime, "x-directory/normal"))
|
|
{
|
|
l = efreet_util_desktop_mime_list("x-directory/normal");
|
|
|
|
EINA_LIST_FREE (l, d)
|
|
{
|
|
if ((!d->no_display) && (!eina_list_data_find_list(p->apps_mime, d)))
|
|
p->apps_mime = eina_list_append(p->apps_mime, d);
|
|
else
|
|
efreet_desktop_free(d);
|
|
}
|
|
}
|
|
|
|
if ((d = e_exehist_mime_desktop_get(mime)))
|
|
{
|
|
if ((l = eina_list_data_find_list(p->apps_mime, d)))
|
|
{
|
|
p->apps_mime = eina_list_promote_list(p->apps_mime, l);
|
|
efreet_desktop_free(d);
|
|
}
|
|
else
|
|
{
|
|
p->apps_mime = eina_list_prepend(p->apps_mime, d);
|
|
}
|
|
}
|
|
|
|
p->added = eina_hash_string_small_new(_hash_free);
|
|
|
|
return EVRY_PLUGIN(p);
|
|
}
|
|
|
|
static void
|
|
_finish_mime(Evry_Plugin *plugin)
|
|
{
|
|
GET_PLUGIN(p, plugin);
|
|
Efreet_Desktop *desktop;
|
|
|
|
EVRY_PLUGIN_ITEMS_CLEAR(p);
|
|
|
|
if (p->added)
|
|
eina_hash_free(p->added);
|
|
|
|
EINA_LIST_FREE (p->apps_mime, desktop)
|
|
efreet_desktop_free(desktop);
|
|
|
|
E_FREE(p);
|
|
}
|
|
|
|
static int
|
|
_fetch_mime(Evry_Plugin *plugin, const char *input)
|
|
{
|
|
GET_PLUGIN(p, plugin);
|
|
Eina_List *l;
|
|
Evry_Item *it;
|
|
|
|
EVRY_PLUGIN_ITEMS_CLEAR(p);
|
|
|
|
/* add apps for a given mimetype */
|
|
_desktop_list_add(p, p->apps_mime, input);
|
|
|
|
EINA_LIST_FOREACH (plugin->items, l, it)
|
|
evry->history_item_usage_set(it, input, NULL);
|
|
|
|
if (input)
|
|
EVRY_PLUGIN_ITEMS_SORT(p, _cb_sort);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_complete(Evry_Plugin *plugin EINA_UNUSED, const Evry_Item *it, char **input)
|
|
{
|
|
GET_APP(app, it);
|
|
|
|
char buf[128];
|
|
if (it->subtype != EVRY_TYPE_APP) return 0;
|
|
|
|
if ((app->desktop) && (app->desktop->exec))
|
|
{
|
|
char *space = strchr(app->desktop->exec, ' ');
|
|
|
|
snprintf(buf, sizeof(buf), "%s", app->desktop->exec);
|
|
if (space)
|
|
buf[1 + space - app->desktop->exec] = '\0';
|
|
}
|
|
else
|
|
snprintf(buf, sizeof(buf), "%s", app->file);
|
|
|
|
*input = strdup(buf);
|
|
|
|
return EVRY_COMPLETE_INPUT;
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
static int
|
|
_exec_app_check_item(Evry_Action *act EINA_UNUSED, const Evry_Item *it EINA_UNUSED)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_exec_app_action(Evry_Action *act)
|
|
{
|
|
return evry->util_exec_app(act->it1.item, act->it2.item);
|
|
}
|
|
|
|
static int
|
|
_exec_file_action(Evry_Action *act)
|
|
{
|
|
return evry->util_exec_app(act->it2.item, act->it1.item);
|
|
}
|
|
|
|
static int
|
|
_exec_term_action(Evry_Action *act)
|
|
{
|
|
GET_APP(app, act->it1.item);
|
|
Evry_Item_App *tmp;
|
|
char buf[1024];
|
|
int ret;
|
|
char *escaped = ecore_file_escape_name(app->file);
|
|
|
|
tmp = E_NEW(Evry_Item_App, 1);
|
|
snprintf(buf, sizeof(buf), "%s %s",
|
|
_conf->cmd_terminal,
|
|
(escaped ? escaped : app->file));
|
|
|
|
tmp->file = buf;
|
|
ret = evry->util_exec_app(EVRY_ITEM(tmp), NULL);
|
|
|
|
E_FREE(tmp);
|
|
E_FREE(escaped);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
_exec_term_check_item(Evry_Action *act EINA_UNUSED, const Evry_Item *it)
|
|
{
|
|
GET_APP(app, it);
|
|
|
|
if (app->file)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
_exec_sudo_action(Evry_Action *act)
|
|
{
|
|
GET_APP(app, act->it1.item);
|
|
Evry_Item_App *tmp;
|
|
char buf[1024];
|
|
int ret;
|
|
|
|
tmp = E_NEW(Evry_Item_App, 1);
|
|
snprintf(buf, sizeof(buf), "%s %s",
|
|
_conf->cmd_sudo,
|
|
(app->desktop ? app->desktop->exec : app->file));
|
|
|
|
tmp->file = buf;
|
|
ret = evry->util_exec_app(EVRY_ITEM(tmp), NULL);
|
|
|
|
E_FREE(tmp);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
_edit_app_check_item(Evry_Action *act EINA_UNUSED, const Evry_Item *it)
|
|
{
|
|
GET_APP(app, it);
|
|
|
|
if (app->desktop)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
_edit_app_action(Evry_Action *act)
|
|
{
|
|
Efreet_Desktop *desktop;
|
|
GET_APP(app, act->it1.item);
|
|
|
|
if (app->desktop)
|
|
desktop = app->desktop;
|
|
else
|
|
{
|
|
char buf[128];
|
|
snprintf(buf, 128, "%s/.local/share/applications/%s.desktop",
|
|
e_user_homedir_get(), app->file);
|
|
desktop = efreet_desktop_empty_new(eina_stringshare_add(buf));
|
|
/* XXX check if this is freed by efreet*/
|
|
desktop->exec = strdup(app->file);
|
|
}
|
|
|
|
e_desktop_edit(desktop);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_new_app_check_item(Evry_Action *act EINA_UNUSED, const Evry_Item *it)
|
|
{
|
|
GET_APP(app, it);
|
|
|
|
if (app->desktop)
|
|
return 1;
|
|
|
|
if (app->file && strlen(app->file) > 0)
|
|
return 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
_new_app_action(Evry_Action *act)
|
|
{
|
|
char *name;
|
|
char buf[4096];
|
|
char *end;
|
|
Efreet_Desktop *desktop;
|
|
int i;
|
|
|
|
GET_APP(app, act->it1.item);
|
|
|
|
if (app->desktop)
|
|
name = strdup(app->desktop->name);
|
|
else
|
|
/* TODO replace '/' and remove other special characters */
|
|
name = strdup(app->file);
|
|
|
|
if ((end = strchr(name, ' ')))
|
|
name[end - name] = '\0';
|
|
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
snprintf(buf, 4096, "%s/.local/share/applications/%s-%d.desktop",
|
|
e_user_homedir_get(), name, i);
|
|
if (ecore_file_exists(buf))
|
|
{
|
|
buf[0] = '\0';
|
|
continue;
|
|
}
|
|
else break;
|
|
}
|
|
free(name);
|
|
|
|
if (strlen(buf) == 0)
|
|
return 0;
|
|
|
|
if (!app->desktop)
|
|
{
|
|
desktop = efreet_desktop_empty_new(buf);
|
|
desktop->exec = (char *)eina_stringshare_add(app->file);
|
|
}
|
|
else
|
|
{
|
|
desktop = efreet_desktop_empty_new(buf);
|
|
if (app->desktop->name)
|
|
desktop->name = strdup(app->desktop->name);
|
|
if (app->desktop->comment)
|
|
desktop->comment = strdup(app->desktop->comment);
|
|
if (app->desktop->generic_name)
|
|
desktop->generic_name = strdup(app->desktop->generic_name);
|
|
if (app->desktop->exec)
|
|
desktop->exec = strdup(app->desktop->exec);
|
|
if (app->desktop->icon)
|
|
desktop->icon = strdup(app->desktop->icon);
|
|
if (app->desktop->mime_types)
|
|
desktop->mime_types = eina_list_clone(app->desktop->mime_types);
|
|
}
|
|
if (desktop)
|
|
e_desktop_edit(desktop);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
_open_term_action(Evry_Action *act)
|
|
{
|
|
GET_FILE(file, act->it1.item);
|
|
Evry_Item_App *tmp;
|
|
char cwd[4096];
|
|
char *dir;
|
|
int ret = 0;
|
|
|
|
if (!(evry->file_path_get(file)))
|
|
return 0;
|
|
|
|
if (IS_BROWSEABLE(file))
|
|
dir = strdup(file->path);
|
|
else
|
|
dir = ecore_file_dir_get(file->path);
|
|
|
|
if (dir)
|
|
{
|
|
if ((!getcwd(cwd, sizeof(cwd))) || (chdir(dir)))
|
|
{
|
|
free(dir);
|
|
return 0;
|
|
}
|
|
|
|
tmp = E_NEW(Evry_Item_App, 1);
|
|
tmp->file = _conf->cmd_terminal;
|
|
|
|
ret = evry->util_exec_app(EVRY_ITEM(tmp), NULL);
|
|
free(tmp);
|
|
free(dir);
|
|
if (chdir(cwd))
|
|
return 0;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
_check_executable(Evry_Action *act EINA_UNUSED, const Evry_Item *it)
|
|
{
|
|
GET_FILE(file, it);
|
|
|
|
return ecore_file_can_exec(evry->file_path_get(file));
|
|
}
|
|
|
|
static int
|
|
_run_executable(Evry_Action *act)
|
|
{
|
|
GET_FILE(file, act->it1.item);
|
|
|
|
e_exec(e_zone_current_get(), NULL, file->path, NULL, NULL);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/***************************************************************************/
|
|
static Eina_Bool
|
|
_desktop_cache_update(void *data EINA_UNUSED, int type EINA_UNUSED, void *event EINA_UNUSED)
|
|
{
|
|
Efreet_Desktop *d;
|
|
|
|
EINA_LIST_FREE (apps_cache, d)
|
|
efreet_desktop_unref(d);
|
|
|
|
return EINA_TRUE;
|
|
}
|
|
static void
|
|
_dir_watcher(void *data EINA_UNUSED, Ecore_File_Monitor *em, Ecore_File_Event event, const char *path EINA_UNUSED)
|
|
{
|
|
//printf("exebuf path changed\n");
|
|
switch (event)
|
|
{
|
|
case ECORE_FILE_EVENT_DELETED_SELF:
|
|
ecore_file_monitor_del(em);
|
|
dir_monitors = eina_list_remove(dir_monitors, em);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
update_path = EINA_TRUE;
|
|
}
|
|
|
|
static void
|
|
_dir_monitor_add()
|
|
{
|
|
Eina_List *l;
|
|
const char *dir;
|
|
|
|
EINA_LIST_FOREACH(exe_path, l, dir)
|
|
{
|
|
Ecore_File_Monitor *dir_mon;
|
|
dir_mon = ecore_file_monitor_add(dir, _dir_watcher, NULL);
|
|
if (dir_mon)
|
|
dir_monitors = eina_list_append(dir_monitors, dir_mon);
|
|
}
|
|
}
|
|
|
|
static void
|
|
_dir_monitor_free()
|
|
{
|
|
Ecore_File_Monitor *dir_mon;
|
|
|
|
EINA_LIST_FREE (dir_monitors, dir_mon)
|
|
ecore_file_monitor_del(dir_mon);
|
|
}
|
|
|
|
static int
|
|
_plugins_init(const Evry_API *api)
|
|
{
|
|
Evry_Plugin *p;
|
|
int prio = 0;
|
|
Eina_List *l;
|
|
Evry_Action *act;
|
|
const char *config_path;
|
|
|
|
evry = api;
|
|
|
|
if (!evry->api_version_check(EVRY_API_VERSION))
|
|
return EINA_FALSE;
|
|
|
|
config_path = eina_stringshare_add("launcher/everything-apps");
|
|
|
|
p = EVRY_PLUGIN_BASE(N_("Applications"), _module_icon, EVRY_TYPE_APP,
|
|
_begin, _finish, _fetch);
|
|
p->complete = &_complete;
|
|
p->browse = &_browse;
|
|
p->config_path = eina_stringshare_ref(config_path);
|
|
evry->plugin_register(p, EVRY_PLUGIN_SUBJECT, 1);
|
|
_plugins = eina_list_append(_plugins, p);
|
|
|
|
p = EVRY_PLUGIN_BASE(N_("Exebuf"), _module_icon, EVRY_TYPE_APP,
|
|
_begin_exe, _finish_exe, _fetch_exe);
|
|
p->complete = &_complete;
|
|
p->config_path = eina_stringshare_ref(config_path);
|
|
_plugins = eina_list_append(_plugins, p);
|
|
if (evry->plugin_register(p, EVRY_PLUGIN_SUBJECT, 3))
|
|
p->config->min_query = 3;
|
|
|
|
p = EVRY_PLUGIN_BASE(N_("Applications"), _module_icon, EVRY_TYPE_APP,
|
|
_begin_mime, _finish, _fetch);
|
|
p->complete = &_complete;
|
|
p->config_path = eina_stringshare_ref(config_path);
|
|
evry->plugin_register(p, EVRY_PLUGIN_OBJECT, 1);
|
|
_plugins = eina_list_append(_plugins, p);
|
|
|
|
p = EVRY_PLUGIN_BASE(N_("Open with..."), _module_icon, EVRY_TYPE_APP,
|
|
_begin_mime, _finish_mime, _fetch_mime);
|
|
p->config_path = eina_stringshare_ref(config_path);
|
|
evry->plugin_register(p, EVRY_PLUGIN_ACTION, 1);
|
|
_plugins = eina_list_append(_plugins, p);
|
|
|
|
act = EVRY_ACTION_NEW(N_("Launch"),
|
|
EVRY_TYPE_APP, 0,
|
|
"system-run",
|
|
_exec_app_action,
|
|
_exec_app_check_item);
|
|
_actions = eina_list_append(_actions, act);
|
|
|
|
act = EVRY_ACTION_NEW(N_("Open File..."),
|
|
EVRY_TYPE_APP, EVRY_TYPE_FILE,
|
|
"document-open",
|
|
_exec_app_action,
|
|
_exec_app_check_item);
|
|
_actions = eina_list_append(_actions, act);
|
|
|
|
act = EVRY_ACTION_NEW(N_("Run in Terminal"),
|
|
EVRY_TYPE_APP, 0,
|
|
"system-run",
|
|
_exec_term_action,
|
|
_exec_term_check_item);
|
|
_actions = eina_list_append(_actions, act);
|
|
|
|
act = EVRY_ACTION_NEW(N_("Edit Application Entry"),
|
|
EVRY_TYPE_APP, 0,
|
|
"everything-launch",
|
|
_edit_app_action,
|
|
_edit_app_check_item);
|
|
_actions = eina_list_append(_actions, act);
|
|
|
|
act = EVRY_ACTION_NEW(N_("New Application Entry"),
|
|
EVRY_TYPE_APP, 0,
|
|
"everything-launch",
|
|
_new_app_action,
|
|
_new_app_check_item);
|
|
_actions = eina_list_append(_actions, act);
|
|
|
|
act = EVRY_ACTION_NEW(N_("Run with Sudo"),
|
|
EVRY_TYPE_APP, 0,
|
|
"system-run",
|
|
_exec_sudo_action, NULL);
|
|
_actions = eina_list_append(_actions, act);
|
|
|
|
act = EVRY_ACTION_NEW(N_("Open with..."),
|
|
EVRY_TYPE_FILE, EVRY_TYPE_APP,
|
|
"everything-launch",
|
|
_exec_file_action, NULL);
|
|
_act_open_with = EVRY_ITEM(act);
|
|
_actions = eina_list_append(_actions, act);
|
|
|
|
act = EVRY_ACTION_NEW(N_("Open Terminal here"),
|
|
EVRY_TYPE_FILE, 0,
|
|
"system-run",
|
|
_open_term_action, NULL);
|
|
_actions = eina_list_append(_actions, act);
|
|
|
|
act = EVRY_ACTION_NEW(N_("Run Executable"),
|
|
EVRY_TYPE_FILE, 0,
|
|
"system-run",
|
|
_run_executable,
|
|
_check_executable);
|
|
_actions = eina_list_append(_actions, act);
|
|
|
|
EINA_LIST_FOREACH (_actions, l, act)
|
|
evry->action_register(act, prio++);
|
|
|
|
handlers = eina_list_append
|
|
(handlers, ecore_event_handler_add
|
|
(EFREET_EVENT_DESKTOP_CACHE_UPDATE, _desktop_cache_update, NULL));
|
|
|
|
eina_stringshare_del(config_path);
|
|
|
|
update_path = EINA_TRUE;
|
|
|
|
return EINA_TRUE;
|
|
}
|
|
|
|
static void
|
|
_plugins_shutdown(void)
|
|
{
|
|
Evry_Action *act;
|
|
Evry_Plugin *p;
|
|
Efreet_Desktop *d;
|
|
Ecore_Event_Handler *h;
|
|
|
|
EINA_LIST_FREE (apps_cache, d)
|
|
efreet_desktop_unref(d);
|
|
|
|
EINA_LIST_FREE (_plugins, p)
|
|
EVRY_PLUGIN_FREE(p);
|
|
|
|
EINA_LIST_FREE (_actions, act)
|
|
EVRY_ACTION_FREE(act);
|
|
|
|
EINA_LIST_FREE (handlers, h)
|
|
ecore_event_handler_del(h);
|
|
|
|
_dir_monitor_free();
|
|
|
|
E_FREE(current_path);
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
static E_Config_DD *conf_edd = NULL;
|
|
|
|
struct _E_Config_Dialog_Data
|
|
{
|
|
char *cmd_terminal;
|
|
char *cmd_sudo;
|
|
};
|
|
|
|
static void *_create_data(E_Config_Dialog *cfd);
|
|
static void _free_data(E_Config_Dialog *cfd, E_Config_Dialog_Data *cfdata);
|
|
static void _fill_data(E_Config_Dialog_Data *cfdata);
|
|
static Evas_Object *_basic_create(E_Config_Dialog *cfd, Evas *evas, E_Config_Dialog_Data *cfdata);
|
|
static int _basic_apply(E_Config_Dialog *cfd, E_Config_Dialog_Data *cfdata);
|
|
|
|
static E_Config_Dialog *
|
|
_conf_dialog(Evas_Object *parent EINA_UNUSED, const char *params EINA_UNUSED)
|
|
{
|
|
E_Config_Dialog *cfd = NULL;
|
|
E_Config_Dialog_View *v = NULL;
|
|
|
|
if (e_config_dialog_find("everything-apps", "launcher/everything-apps")) return NULL;
|
|
|
|
v = E_NEW(E_Config_Dialog_View, 1);
|
|
if (!v) return NULL;
|
|
|
|
v->create_cfdata = _create_data;
|
|
v->free_cfdata = _free_data;
|
|
v->basic.create_widgets = _basic_create;
|
|
v->basic.apply_cfdata = _basic_apply;
|
|
|
|
cfd = e_config_dialog_new(NULL, _("Everything Applications"), "everything-apps",
|
|
"launcher/everything-apps", _module_icon, 0, v, NULL);
|
|
|
|
_conf->cfd = cfd;
|
|
return cfd;
|
|
}
|
|
|
|
static void *
|
|
_create_data(E_Config_Dialog *cfd EINA_UNUSED)
|
|
{
|
|
E_Config_Dialog_Data *cfdata = NULL;
|
|
|
|
cfdata = E_NEW(E_Config_Dialog_Data, 1);
|
|
_fill_data(cfdata);
|
|
return cfdata;
|
|
}
|
|
|
|
static void
|
|
_free_data(E_Config_Dialog *cfd EINA_UNUSED, E_Config_Dialog_Data *cfdata)
|
|
{
|
|
_conf->cfd = NULL;
|
|
E_FREE(cfdata);
|
|
}
|
|
|
|
static Evas_Object *
|
|
_basic_create(E_Config_Dialog *cfd, Evas *e, E_Config_Dialog_Data *cfdata)
|
|
{
|
|
Evas_Object *o = NULL, *of = NULL, *ow = NULL;
|
|
|
|
o = e_widget_list_add(e, 0, 0);
|
|
|
|
of = e_widget_framelist_add(e, _("Commands"), 0);
|
|
ow = e_widget_label_add(e, _("Terminal Command"));
|
|
e_widget_framelist_object_append(of, ow);
|
|
ow = e_widget_entry_add(cfd->dia->win, &(cfdata->cmd_terminal), NULL, NULL, NULL);
|
|
e_widget_framelist_object_append(of, ow);
|
|
|
|
ow = e_widget_label_add(e, _("Sudo GUI"));
|
|
e_widget_framelist_object_append(of, ow);
|
|
ow = e_widget_entry_add(cfd->dia->win, &(cfdata->cmd_sudo), NULL, NULL, NULL);
|
|
e_widget_framelist_object_append(of, ow);
|
|
|
|
e_widget_list_object_append(o, of, 1, 1, 0.5);
|
|
|
|
return o;
|
|
}
|
|
|
|
static void
|
|
_fill_data(E_Config_Dialog_Data *cfdata)
|
|
{
|
|
#define CP(_name) cfdata->_name = strdup(_conf->_name);
|
|
#define C(_name) cfdata->_name = _conf->_name;
|
|
CP(cmd_terminal);
|
|
CP(cmd_sudo);
|
|
#undef CP
|
|
#undef C
|
|
}
|
|
|
|
static int
|
|
_basic_apply(E_Config_Dialog *cfd EINA_UNUSED, E_Config_Dialog_Data *cfdata)
|
|
{
|
|
#define CP(_name) \
|
|
eina_stringshare_replace(&(_conf->_name), cfdata->_name)
|
|
#define C(_name) _conf->_name = cfdata->_name;
|
|
CP(cmd_terminal);
|
|
CP(cmd_sudo);
|
|
#undef CP
|
|
#undef C
|
|
|
|
e_config_domain_save("module.everything-apps", conf_edd, _conf);
|
|
|
|
/* backwards compatible with exebuf */
|
|
eina_stringshare_replace(&(e_config->exebuf_term_cmd), _conf->cmd_terminal);
|
|
e_config_save_queue();
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
_conf_new(void)
|
|
{
|
|
if (!_conf)
|
|
{
|
|
_conf = E_NEW(Module_Config, 1);
|
|
/* setup defaults */
|
|
_conf->cmd_terminal = eina_stringshare_add("/usr/bin/xterm -hold -e");
|
|
_conf->cmd_sudo = eina_stringshare_add("/usr/bin/gksudo --preserve-env");
|
|
}
|
|
|
|
_conf->version = MOD_CONFIG_FILE_VERSION;
|
|
|
|
/* e_config_save_queue(); */
|
|
}
|
|
|
|
static void
|
|
_conf_free(void)
|
|
{
|
|
if (!_conf) return;
|
|
|
|
IF_RELEASE(_conf->cmd_sudo);
|
|
IF_RELEASE(_conf->cmd_terminal);
|
|
E_FREE(_conf);
|
|
}
|
|
|
|
static void
|
|
_conf_init(E_Module *m)
|
|
{
|
|
char title[4096];
|
|
|
|
snprintf(title, sizeof(title), "%s: %s", _("Everything Plugin"), _("Applications"));
|
|
|
|
e_configure_registry_item_add("launcher/everything-apps", 110, title,
|
|
NULL, _module_icon, _conf_dialog);
|
|
|
|
conf_edd = E_CONFIG_DD_NEW("Module_Config", Module_Config);
|
|
|
|
#undef T
|
|
#undef D
|
|
#define T Module_Config
|
|
#define D conf_edd
|
|
E_CONFIG_VAL(D, T, version, INT);
|
|
E_CONFIG_VAL(D, T, cmd_terminal, STR);
|
|
E_CONFIG_VAL(D, T, cmd_sudo, STR);
|
|
#undef T
|
|
#undef D
|
|
|
|
_conf = e_config_domain_load("module.everything-apps", conf_edd);
|
|
|
|
if (_conf && !e_util_module_config_check(_("Everything Applications"),
|
|
_conf->version,
|
|
MOD_CONFIG_FILE_VERSION))
|
|
_conf_free();
|
|
|
|
_conf_new();
|
|
_conf->module = m;
|
|
}
|
|
|
|
static void
|
|
_conf_shutdown(void)
|
|
{
|
|
e_configure_registry_item_del("launcher/everything-apps");
|
|
|
|
_conf_free();
|
|
|
|
E_CONFIG_DD_FREE(conf_edd);
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
Eina_Bool
|
|
evry_plug_apps_init(E_Module *m)
|
|
{
|
|
_conf_init(m);
|
|
|
|
EVRY_MODULE_NEW(evry_module, evry, _plugins_init, _plugins_shutdown);
|
|
|
|
exelist_exe_edd = E_CONFIG_DD_NEW("E_Exe", E_Exe);
|
|
#undef T
|
|
#undef D
|
|
#define T E_Exe
|
|
#define D exelist_exe_edd
|
|
E_CONFIG_VAL(D, T, path, STR);
|
|
E_CONFIG_VAL(D, T, len, UINT);
|
|
exelist_edd = E_CONFIG_DD_NEW("E_Exe_List", E_Exe_List);
|
|
#undef T
|
|
#undef D
|
|
#define T E_Exe_List
|
|
#define D exelist_edd
|
|
E_CONFIG_LIST(D, T, list, exelist_exe_edd);
|
|
|
|
return EINA_TRUE;
|
|
}
|
|
|
|
void
|
|
evry_plug_apps_shutdown(void)
|
|
{
|
|
EVRY_MODULE_FREE(evry_module);
|
|
|
|
_conf_shutdown();
|
|
|
|
E_CONFIG_DD_FREE(exelist_edd);
|
|
E_CONFIG_DD_FREE(exelist_exe_edd);
|
|
}
|
|
|
|
void
|
|
evry_plug_apps_save(void)
|
|
{
|
|
e_config_domain_save("module.everything-apps", conf_edd, _conf);
|
|
}
|
|
|
|
/***************************************************************************/
|
|
|
|
/* taken from e_exebuf.c */
|
|
static Eina_Bool
|
|
_scan_idler(void *data EINA_UNUSED)
|
|
{
|
|
char *dir;
|
|
|
|
/* no more path items left - stop scanning */
|
|
if (!exe_path)
|
|
{
|
|
Eina_Bool different = EINA_FALSE;
|
|
|
|
if (eina_list_count(exe_list) == eina_list_count(exe_files))
|
|
{
|
|
E_Exe *ee;
|
|
Eina_List *l, *l2 = exe_files;
|
|
|
|
EINA_LIST_FOREACH(exe_list, l ,ee)
|
|
{
|
|
if (ee->path != l2->data)
|
|
{
|
|
different = EINA_TRUE;
|
|
break;
|
|
}
|
|
l2 = l2->next;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
different = EINA_TRUE;
|
|
}
|
|
|
|
if (different)
|
|
{
|
|
E_Exe *ee;
|
|
|
|
EINA_LIST_FREE (exe_list, ee)
|
|
{
|
|
eina_stringshare_del(ee->path);
|
|
free(ee);
|
|
}
|
|
|
|
const char *s;
|
|
E_Exe_List *el;
|
|
|
|
el = calloc(1, sizeof(E_Exe_List));
|
|
if (!el) return ECORE_CALLBACK_CANCEL;
|
|
|
|
el->list = NULL;
|
|
|
|
EINA_LIST_FREE(exe_files, s)
|
|
{
|
|
ee = malloc(sizeof(E_Exe));
|
|
if (!ee) continue ;
|
|
|
|
ee->path = s;
|
|
ee->len = strlen(s);
|
|
el->list = eina_list_append(el->list, ee);
|
|
}
|
|
|
|
e_config_domain_save(_exebuf_cache_file, exelist_edd, el);
|
|
INF("plugin exebuf save: %s, %d", _exebuf_cache_file, eina_list_count(el->list));
|
|
|
|
exe_list = el->list;
|
|
free(el);
|
|
}
|
|
else
|
|
{
|
|
const char *str;
|
|
EINA_LIST_FREE(exe_files, str)
|
|
eina_stringshare_del(str);
|
|
}
|
|
|
|
exe_scan_idler = NULL;
|
|
return ECORE_CALLBACK_CANCEL;
|
|
}
|
|
/* no dir is open - open the first path item */
|
|
if (!exe_dir)
|
|
{
|
|
dir = exe_path->data;
|
|
exe_dir = eina_file_direct_ls(dir);
|
|
INF("scan dir: %s", dir);
|
|
|
|
}
|
|
/* if we have an opened dir - scan the next item */
|
|
if (exe_dir)
|
|
{
|
|
Eina_File_Direct_Info *info;
|
|
|
|
if (eina_iterator_next(exe_dir, (void**) &info))
|
|
{
|
|
Eina_Stat st;
|
|
|
|
if (!eina_file_statat(eina_iterator_container_get(exe_dir), info, &st) &&
|
|
(!S_ISDIR(st.mode)) &&
|
|
(!access(info->path, X_OK)))
|
|
exe_files = eina_list_append(exe_files,
|
|
eina_stringshare_add(info->path + info->name_start));
|
|
}
|
|
else
|
|
{
|
|
/* we reached the end of a dir - remove the dir at the head
|
|
* of the path list so we advance and next loop we will pick up
|
|
* the next item, or if null- abort
|
|
*/
|
|
dir = exe_path->data;
|
|
free(dir);
|
|
|
|
eina_iterator_free(exe_dir);
|
|
exe_dir = NULL;
|
|
|
|
exe_path = eina_list_remove_list(exe_path, exe_path);
|
|
}
|
|
}
|
|
/* obviously the dir open failed - so remove the first path item */
|
|
else
|
|
{
|
|
free(eina_list_data_get(exe_path));
|
|
exe_path = eina_list_remove_list(exe_path, exe_path);
|
|
}
|
|
/* we have more scannign to do */
|
|
return ECORE_CALLBACK_RENEW;
|
|
}
|
|
|
|
|
|
static Eina_Bool
|
|
_exe_path_list()
|
|
{
|
|
char *path, *pp, *last;
|
|
Eina_Bool changed;
|
|
|
|
path = getenv("PATH");
|
|
|
|
// nothing changed
|
|
if (!update_path && (current_path && path) && !strcmp(current_path, path))
|
|
return EINA_FALSE;
|
|
|
|
changed = (path && (!current_path || !strcmp(current_path, path)));
|
|
|
|
if (current_path)
|
|
{
|
|
free(current_path);
|
|
current_path = NULL;
|
|
}
|
|
|
|
if (path)
|
|
{
|
|
path = strdup(path);
|
|
current_path = strdup(path);
|
|
|
|
last = path;
|
|
for (pp = path; pp[0]; pp++)
|
|
{
|
|
if (pp[0] == ':') pp[0] = '\0';
|
|
if (pp[0] == 0)
|
|
{
|
|
exe_path = eina_list_append(exe_path, strdup(last));
|
|
last = pp + 1;
|
|
}
|
|
}
|
|
if (pp > last)
|
|
exe_path = eina_list_append(exe_path, strdup(last));
|
|
|
|
free(path);
|
|
}
|
|
|
|
if (changed)
|
|
{
|
|
_dir_monitor_free();
|
|
_dir_monitor_add();
|
|
}
|
|
|
|
return EINA_TRUE;
|
|
}
|
|
|
|
static void
|
|
_scan_executables()
|
|
{
|
|
E_Exe_List *el;
|
|
|
|
el = e_config_domain_load(_exebuf_cache_file, exelist_edd);
|
|
if (el)
|
|
{
|
|
exe_list = el->list;
|
|
INF("plugin exebuf load: %s, %d", _exebuf_cache_file, eina_list_count(el->list));
|
|
|
|
free(el);
|
|
}
|
|
|
|
if (_exe_path_list())
|
|
{
|
|
exe_scan_idler = ecore_idler_add(_scan_idler, NULL);
|
|
update_path = EINA_FALSE;
|
|
}
|
|
}
|