enlightenment/src/modules/everything/evry_plug_files.c

1633 lines
38 KiB
C
Raw Normal View History

/***************************************************
TODO option for maximum items to cache
TODO keep common list for recent file instances
FIXME
*/
#include "e.h"
#include "evry_api.h"
#include <Efreet_Trash.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)
#define MAX_ITEMS 10
#define MAX_SHOWN 300
#define TERM_ACTION_DIR "%s"
#define CMD_NONE 0
#define CMD_SHOW_ROOT 1
#define CMD_SHOW_HIDDEN 2
#define CMD_SHOW_PARENT 3
#define ACT_TRASH 1
#define ACT_DELETE 2
#define ACT_COPY 3
#define ACT_MOVE 4
#define ACT_SORT_DATE 5
#define ACT_SORT_NAME 6
#define ONE_DAY 86400.0
#define SIX_DAYS_AGO (ecore_time_unix_get() - ONE_DAY * 6)
#define MIN_USAGE 0.0
/* #undef DBG
* #define DBG(...) ERR(__VA_ARGS__) */
typedef struct _Plugin Plugin;
typedef struct _Data Data;
typedef struct _Module_Config Module_Config;
struct _Plugin
{
Evry_Plugin base;
Eina_List *files;
const char *directory;
const char *input;
unsigned int command;
unsigned int min_query;
Eina_Bool parent;
Eina_Bool show_hidden;
Eina_Bool dirs_only;
Eina_Bool show_recent;
Eina_Bool sort_by_date;
Ecore_Thread *thread;
Ecore_File_Monitor *dir_mon;
int waiting_to_finish;
};
struct _Data
{
Plugin *plugin;
char *directory;
long id;
int level;
int cnt;
Eina_List *files;
DIR *dirp;
int run_cnt;
};
struct _Module_Config
{
int version;
unsigned char show_homedir;
unsigned char show_recent;
unsigned char search_recent;
unsigned char cache_dirs;
unsigned char search_cache;
// TODO
int sort_by;
Eina_List *search_dirs;
E_Config_Dialog *cfd;
E_Module *module;
};
static const Evry_API *evry = NULL;
static Evry_Module *evry_module = NULL;
static Module_Config *_conf;
2010-04-30 05:55:01 -07:00
static char _module_icon[] = "system-file-manager";
static Eina_List *_plugins = NULL;
static Eina_List *_actions = NULL;
2010-04-30 05:55:01 -07:00
static const char *_mime_dir;
static const char *_mime_mount;
static const char *_mime_unknown;
static Eina_Bool clear_cache = EINA_FALSE;
/***************************************************************************/
static void
_item_fill(Evry_Item_File *file)
{
if (!file->mime)
{
const char *mime = efreet_mime_type_get(file->path);
if (mime)
file->mime = eina_stringshare_ref(mime);
else
file->mime = eina_stringshare_add("unknown");
}
if ((file->mime == _mime_dir) ||
(file->mime == _mime_mount))
EVRY_ITEM(file)->browseable = EINA_TRUE;
EVRY_ITEM(file)->context = eina_stringshare_ref(file->mime);
if (!EVRY_ITEM(file)->detail)
evry->util_file_detail_set(file);
evry->util_file_detail_set(file);
}
static int
_cb_sort(const void *data1, const void *data2)
{
const Evry_Item *it1 = data1;
const Evry_Item *it2 = data2;
if (it1->browseable && !it2->browseable)
return -1;
if (!it1->browseable && it2->browseable)
return 1;
if (it1->fuzzy_match && it2->fuzzy_match)
if (it1->fuzzy_match - it2->fuzzy_match)
return it1->fuzzy_match - it2->fuzzy_match;
return strcasecmp(it1->label, it2->label);
}
static int
_cb_sort_date(const void *data1, const void *data2)
{
const Evry_Item_File *it1 = data1;
const Evry_Item_File *it2 = data2;
return it2->modified - it1->modified;
}
static void
_scan_func(void *data, Ecore_Thread *thread)
{
Data *d = data;
Eina_Iterator *ls;
Eina_File_Direct_Info *info;
Evry_Item_File *file;
if (!(ls = eina_file_stat_ls(d->directory)))
return;
EINA_ITERATOR_FOREACH (ls, info)
{
if ((d->plugin->show_hidden) != (*(info->path + info->name_start) == '.'))
continue;
file = EVRY_ITEM_NEW(Evry_Item_File, d->plugin, NULL, NULL, evry_item_file_free);
file->path = strdup(info->path);
EVRY_ITEM(file)->label = strdup(info->path + info->name_start);
EVRY_ITEM(file)->browseable = (info->type == EINA_FILE_DIR);
d->files = eina_list_append(d->files, file);
/* TODO dont append files in thread, run mime scan
* simultaneously. ecore_thread_feedback(thread, file); */
if (ecore_thread_check(thread))
break;
}
eina_iterator_free(ls);
}
static void
_scan_mime_func(void *data, Ecore_Thread *thread)
{
Data *d = data;
Evry_Item_File *file;
Eina_List *l;
int cnt = 0;
EINA_LIST_FOREACH (d->files, l, file)
{
if (ecore_thread_check(thread))
break;
if ((file->mime = efreet_mime_type_get(file->path)))
{
if (!strncmp(file->mime, "inode/", 6) &&
ecore_file_is_dir(file->path))
EVRY_ITEM(file)->browseable = EINA_TRUE;
}
else
file->mime = _mime_unknown;
if (cnt++ > MAX_ITEMS * d->run_cnt) break;
}
}
static int
_files_filter(Plugin *p)
{
int match;
int cnt = 0;
Evry_Item *it;
Eina_List *l;
unsigned int len = p->input ? strlen(p->input) : 0;
EVRY_PLUGIN_ITEMS_CLEAR(p);
if (!p->command && p->min_query)
{
if (!p->input)
return 0;
if (len < p->min_query)
return 0;
}
EINA_LIST_FOREACH (p->files, l, it)
{
if (cnt >= MAX_SHOWN) break;
if (p->dirs_only && !it->browseable)
continue;
if (len && (match = evry->fuzzy_match(it->label, p->input)))
{
it->fuzzy_match = match;
if (!it->browseable)
it->priority = 1;
EVRY_PLUGIN_ITEM_APPEND(p, it);
cnt++;
}
else if (len == 0)
{
if (!it->browseable)
it->priority = 1;
EVRY_PLUGIN_ITEM_APPEND(p, it);
cnt++;
}
}
return cnt;
}
static void
_scan_cancel_func(void *data, Ecore_Thread *thread __UNUSED__)
{
Data *d = data;
Plugin *p = d->plugin;
Evry_Item_File *file;
/* EINA_LIST_FREE(p->files, file)
* EVRY_ITEM_FREE(file); */
EINA_LIST_FREE (d->files, file)
{
if (file->base.label) free((char *)(file->base.label));
if (file->path) free((char *)file->path);
E_FREE(file);
}
p->thread = NULL;
if (p->waiting_to_finish)
E_FREE(p);
free(d->directory);
E_FREE(d);
}
static void
_cache_mime_get(History_Types *ht, Evry_Item_File *file)
{
History_Entry *he;
History_Item *hi;
Eina_List *l;
if ((he = eina_hash_find(ht->types, file->path)))
{
EINA_LIST_FOREACH (he->items, l, hi)
{
if (!hi->data)
continue;
if (!file->mime)
file->mime = hi->data;
hi->transient = 0;
EVRY_ITEM(file)->hi = hi;
break;
}
}
}
static void
_cache_dir_add(Eina_List *files)
{
Eina_List *l;
Evry_Item *item;
History_Item *hi;
int cnt = 0;
EINA_LIST_REVERSE_FOREACH (files, l, item)
{
GET_FILE(file, item);
if (!(item->hi) &&
(hi = evry->history_item_add(item, NULL, NULL)))
{
hi->last_used = SIX_DAYS_AGO;
hi->usage = MIN_USAGE * (double)cnt++;
hi->data = eina_stringshare_ref(file->mime);
item->hi = hi;
}
else if (item->hi && (item->hi->count == 1) &&
(item->hi->last_used < SIX_DAYS_AGO))
{
item->hi->last_used = SIX_DAYS_AGO;
item->hi->usage = MIN_USAGE * (double)cnt++;
}
}
}
static void
_file_add(Plugin *p, Evry_Item *item)
{
GET_FILE(file, item);
char *filename = (char *)item->label;
char *path = (char *)file->path;
file->path = eina_stringshare_add(path);
file->mime = eina_stringshare_ref(file->mime);
item->label = eina_stringshare_add(filename);
item->id = eina_stringshare_ref(file->path);
item->context = eina_stringshare_ref(file->mime);
evry->util_file_detail_set(file);
p->files = eina_list_append(p->files, file);
E_FREE(filename);
E_FREE(path);
}
static void
_scan_end_func(void *data, Ecore_Thread *thread __UNUSED__)
{
Data *d = data;
Plugin *p = d->plugin;
Evry_Item *item;
Evry_Item_File *file;
Eina_List *l, *ll;
History_Types *ht = NULL;
if (_conf->cache_dirs)
ht = evry->history_types_get(EVRY_TYPE_FILE);
if (!d->run_cnt) /* _scan_func finished */
{
EINA_LIST_FOREACH_SAFE (d->files, l, ll, item)
{
GET_FILE(browse, item);
if (item->browseable)
browse->mime = _mime_dir;
else if (ht)
_cache_mime_get(ht, browse);
if (browse->mime)
{
d->files = eina_list_remove_list(d->files, l);
_file_add(p, item);
}
}
/* sort files by name for mimetypes scan */
if (d->files)
d->files = eina_list_sort(d->files, -1, _cb_sort);
}
else /* _scan_mime_func finished */
{
EINA_LIST_FREE (d->files, file)
{
if (!file->mime) break;
_file_add(p, (Evry_Item *)file);
}
}
if (d->files) /* scan mimetypes */
{
d->run_cnt++;
p->thread = ecore_thread_run(_scan_mime_func,
_scan_end_func,
_scan_cancel_func, d);
/* wait for first mime scan to finish */
if (d->run_cnt == 1) return;
}
else /* finished all file/mime scan */
{
free(d->directory);
E_FREE(d);
p->thread = NULL;
if (_conf->cache_dirs && !(p->command == CMD_SHOW_HIDDEN))
_cache_dir_add(p->files);
}
p->files = eina_list_sort(p->files, -1, _cb_sort);
_files_filter(p);
EVRY_PLUGIN_UPDATE(p, EVRY_UPDATE_ADD);
}
static void
_dir_watcher(void *data, Ecore_File_Monitor *em __UNUSED__, Ecore_File_Event event, const char *path)
{
Plugin *p = data;
Evry_Item_File *file;
const char *label;
Eina_List *ll, *l;
switch (event)
{
case ECORE_FILE_EVENT_DELETED_SELF:
EINA_LIST_FREE (p->files, file)
evry->item_free(EVRY_ITEM(file));
break;
case ECORE_FILE_EVENT_CREATED_DIRECTORY:
case ECORE_FILE_EVENT_CREATED_FILE:
label = ecore_file_file_get(path);
file = EVRY_ITEM_NEW(Evry_Item_File, p, label, NULL, evry_item_file_free);
file->path = eina_stringshare_add(path);
if (event == ECORE_FILE_EVENT_CREATED_DIRECTORY)
file->mime = eina_stringshare_ref(_mime_dir);
_item_fill(file);
p->files = eina_list_append(p->files, file);
break;
case ECORE_FILE_EVENT_DELETED_FILE:
case ECORE_FILE_EVENT_DELETED_DIRECTORY:
label = eina_stringshare_add(path);
EINA_LIST_FOREACH_SAFE (p->files, l, ll, file)
{
if (file->path != label) continue;
p->files = eina_list_remove_list(p->files, l);
EVRY_ITEM_FREE(file);
break;
}
eina_stringshare_del(label);
break;
default:
return;
}
_files_filter(p);
EVRY_PLUGIN_UPDATE(p, EVRY_UPDATE_ADD);
}
static void
_read_directory(Plugin *p)
{
Data *d = E_NEW(Data, 1);
d->plugin = p;
d->directory = strdup(p->directory);
d->run_cnt = 0;
p->thread = ecore_thread_run(_scan_func, _scan_end_func, _scan_cancel_func, d);
if (p->dir_mon)
ecore_file_monitor_del(p->dir_mon);
p->dir_mon = ecore_file_monitor_add(p->directory, _dir_watcher, p);
}
static Evry_Plugin *
_browse(Evry_Plugin *plugin, const Evry_Item *it)
{
Plugin *p = NULL;
if (!it || !(CHECK_TYPE(it, EVRY_TYPE_FILE)))
return NULL;
GET_FILE(file, it);
if (!evry->file_path_get(file) ||
!ecore_file_is_dir(file->path))
return NULL;
EVRY_PLUGIN_INSTANCE(p, plugin);
p->directory = eina_stringshare_add(file->path);
p->parent = EINA_TRUE;
_read_directory(p);
return EVRY_PLUGIN(p);
}
static Evry_Plugin *
_begin(Evry_Plugin *plugin, const Evry_Item *it)
{
Plugin *p = NULL;
if (it)
{
const char *dir = NULL;
if ((CHECK_TYPE(it, EVRY_TYPE_FILE)) ||
(CHECK_SUBTYPE(it, EVRY_TYPE_FILE)))
{
/* browse */
GET_FILE(file, it);
if (!evry->file_path_get(file))
return NULL;
if (!ecore_file_is_dir(file->path))
{
char *tmp = ecore_file_dir_get(file->path);
dir = eina_stringshare_add(tmp);
E_FREE(tmp);
}
else
{
dir = eina_stringshare_add(file->path);
}
}
else
{
/* provide object */
if (!CHECK_TYPE(it, EVRY_TYPE_ACTION))
return NULL;
}
if (!dir)
dir = eina_stringshare_add(e_user_homedir_get());
EVRY_PLUGIN_INSTANCE(p, plugin);
p->directory = dir;
p->parent = EINA_FALSE;
p->min_query = 0;
_read_directory(p);
return EVRY_PLUGIN(p);
}
else
{
/* provide subject */
EVRY_PLUGIN_INSTANCE(p, plugin);
p->parent = EINA_FALSE;
p->directory = eina_stringshare_add(e_user_homedir_get());
p->min_query = plugin->config->min_query;
_read_directory(p);
return EVRY_PLUGIN(p);
}
return NULL;
}
static void
_folder_item_add(Plugin *p, const char *path, int prio)
{
Evry_Item_File *file;
file = EVRY_ITEM_NEW(Evry_Item_File, p, path, NULL, evry_item_file_free);
file->path = eina_stringshare_add(path);
file->mime = eina_stringshare_ref(_mime_dir);
EVRY_ITEM(file)->browseable = EINA_TRUE;
EVRY_ITEM(file)->priority = prio;
EVRY_ITEM(file)->usage = -1;
p->files = eina_list_append(p->files, file);
EVRY_PLUGIN_ITEM_APPEND(p, file);
}
static void
_free_files(Plugin *p)
{
Evry_Item_File *file;
2010-05-14 21:22:12 -07:00
EVRY_PLUGIN_ITEMS_CLEAR(p);
if (p->thread)
ecore_thread_cancel(p->thread);
p->thread = NULL;
EINA_LIST_FREE (p->files, file)
EVRY_ITEM_FREE(file);
if (p->dir_mon)
ecore_file_monitor_del(p->dir_mon);
p->dir_mon = NULL;
}
static void
_finish(Evry_Plugin *plugin)
{
GET_PLUGIN(p, plugin);
IF_RELEASE(p->input);
IF_RELEASE(p->directory);
if (p->thread)
p->waiting_to_finish = 1;
_free_files(p);
if (!p->waiting_to_finish)
E_FREE(p);
}
static int
_fetch(Evry_Plugin *plugin, const char *input)
{
GET_PLUGIN(p, plugin);
unsigned int len = (input ? strlen(input) : 0);
if (!p->command)
EVRY_PLUGIN_ITEMS_CLEAR(p);
IF_RELEASE(p->input);
if (!p->parent && input && (input[0] == '/'))
{
if (p->command != CMD_SHOW_ROOT)
{
_free_files(p);
eina_stringshare_replace(&p->directory, input);
_read_directory(p);
p->command = CMD_SHOW_ROOT;
return 0;
}
int lendir = p->directory ? strlen(p->directory) : 0;
lendir = (lendir <= 1) ? lendir : lendir + 1;
p->input = eina_stringshare_add(input + lendir);
}
else if (p->directory && input && !strncmp(input, "..", 2))
{
if (p->command != CMD_SHOW_PARENT)
{
char *dir;
char buf[PATH_MAX+1];
int prio = 0;
if (strncmp(p->directory, "/", 1))
return 0;
_free_files(p);
2012-10-04 16:30:35 -07:00
strncpy(buf, p->directory, sizeof(buf) - 1);
_folder_item_add(p, p->directory, prio++);
2012-10-04 16:30:35 -07:00
buf[sizeof(buf) - 1] = 0;
while (strlen(buf) > 1)
{
dir = dirname(buf);
_folder_item_add(p, dir, prio++);
strcpy(buf, dir);
}
p->command = CMD_SHOW_PARENT;
}
return 1;
}
else if (p->directory && input && !strncmp(input, ".", 1))
{
if (p->command != CMD_SHOW_HIDDEN)
{
_free_files(p);
p->show_hidden = EINA_TRUE;
_read_directory(p);
p->command = CMD_SHOW_HIDDEN;
return 0;
}
p->input = eina_stringshare_add(input);
}
else if (p->command)
{
/* clear command items */
_free_files(p);
if (p->command == CMD_SHOW_ROOT)
{
IF_RELEASE(p->directory);
p->directory = eina_stringshare_add(e_user_homedir_get());
}
p->command = CMD_NONE;
p->show_hidden = EINA_FALSE;
_read_directory(p);
}
if (input && !p->command)
p->input = eina_stringshare_add(input);
if ((p->command) || (!p->min_query) || (len >= p->min_query))
_files_filter(p);
return EVRY_PLUGIN_HAS_ITEMS(p);
}
/***************************************************************************/
/* recent files */
static int
_cb_sort_recent(const void *data1, const void *data2)
{
const Evry_Item *it1 = data1;
const Evry_Item *it2 = data2;
if (it1->browseable && !it2->browseable)
return -1;
if (!it1->browseable && it2->browseable)
return 1;
if (it1->hi && it2->hi)
return it1->hi->last_used > it2->hi->last_used ? -1 : 1;
if (it1->fuzzy_match && it2->fuzzy_match)
if (it1->fuzzy_match - it2->fuzzy_match)
return it1->fuzzy_match - it2->fuzzy_match;
return strcasecmp(it1->label, it2->label);
}
static int
_recentf_files_filter(Plugin *p)
{
int match;
int cnt = 0;
Evry_Item *it;
Eina_List *l, *new = NULL;
EVRY_PLUGIN_ITEMS_CLEAR(p);
EINA_LIST_FOREACH (p->files, l, it)
{
if (p->dirs_only && !it->browseable)
continue;
if (!p->input)
{
if (!it->browseable)
it->priority = 1;
new = eina_list_append(new, it);
continue;
}
if (it->fuzzy_match <= 0)
{
if ((match = evry->fuzzy_match(it->label, p->input)) ||
(match = evry->fuzzy_match(EVRY_FILE(it)->path, p->input)))
it->fuzzy_match = match;
else
it->fuzzy_match = 0;
DBG("check match %d %s", it->fuzzy_match, it->label);
}
if (it->fuzzy_match)
{
if (!it->browseable)
it->priority = 1;
new = eina_list_append(new, it);
}
}
new = eina_list_sort(new, -1, _cb_sort_recent);
EINA_LIST_FREE (new, it)
{
if (cnt++ < MAX_SHOWN)
EVRY_PLUGIN_ITEM_APPEND(p, it);
}
return cnt;
}
#if 0
/* use thread only to not block ui for ecore_file_exists ... */
static void
_recentf_func(void *data)
{
Data *d = data;
Eina_List *l;
Evry_Item_File *file;
EINA_LIST_FOREACH (d->files, l, file)
{
if ((!evry->file_path_get(file)) ||
(!ecore_file_exists(file->path)))
{
EVRY_ITEM(file)->hi->last_used -= ONE_DAY;
EVRY_ITEM(file)->hi = NULL;
}
}
}
static void
_recentf_cancel_func(void *data)
{
Data *d = data;
Plugin *p = d->plugin;
Evry_Item_File *file;
EINA_LIST_FREE (d->files, file)
EVRY_ITEM_FREE(file);
E_FREE(d);
if (p->waiting_to_finish)
E_FREE(p);
}
static void
_recentf_end_func(void *data)
{
Data *d = data;
Plugin *p = d->plugin;
Evry_Item *it;
EINA_LIST_FREE (d->files, it)
{
GET_FILE(file, it);
if (!it->hi)
{
evry->item_free(it);
continue;
}
_item_fill(file);
if (!it->hi->data)
it->hi->data = eina_stringshare_ref(file->mime);
p->files = eina_list_append(p->files, it);
}
_recentf_files_filter(p);
EVRY_PLUGIN_UPDATE(p, EVRY_UPDATE_ADD);
p->thread = NULL;
E_FREE(d);
}
#endif
static Eina_Bool
_recentf_items_add_cb(const Eina_Hash *hash __UNUSED__, const void *key, void *data, void *fdata)
{
History_Entry *he = data;
History_Item *hi = NULL, *hi2;
Eina_List *l, *ll;
Evry_Item_File *file;
double last_used = 0.0;
Data *d = fdata;
Plugin *p = d->plugin;
const char *label;
const char *path;
int match = 0;
EINA_LIST_FOREACH (he->items, l, hi2)
if (hi2->last_used > last_used)
{
last_used = hi2->last_used;
hi = hi2;
}
if (!hi)
return EINA_TRUE;
if (clear_cache)
{
DBG("clear %s", (char *)key);
/* transient marks them for deletion */
if (hi->count == 1)
{
hi->usage = 0;
hi->count = 0;
hi->transient = 1;
}
return EINA_TRUE;
}
if (hi->transient)
return EINA_TRUE;
if (!_conf->search_cache)
{
if ((hi->count == 1) && (hi->last_used < SIX_DAYS_AGO))
return EINA_TRUE;
}
path = (const char *)key;
if (!(label = ecore_file_file_get(path)))
return EINA_TRUE;
path = eina_stringshare_add(path);
EINA_LIST_FOREACH (p->files, ll, file)
{
if (file->path == path)
{
eina_stringshare_del(path);
EVRY_ITEM(file)->fuzzy_match = -1;
return EINA_TRUE;
}
}
/* searching subdirs */
if (p->directory)
{
/* dont show recent files from same dir */
int len = strlen(p->directory);
char *end = strrchr(path, '/');
if (strncmp(path, p->directory, len) ||
(end - path) <= len)
{
/* DBG("not in dir %s", path); */
eina_stringshare_del(path);
return EINA_TRUE;
}
}
if (!(match = evry->fuzzy_match(label, p->input)) &&
!(match = evry->fuzzy_match(path, p->input)))
{
eina_stringshare_del(path);
return EINA_TRUE;
}
file = EVRY_ITEM_NEW(Evry_Item_File, p, label, NULL, evry_item_file_free);
file->path = path;
if (hi->data)
file->mime = eina_stringshare_add(hi->data);
EVRY_ITEM(file)->hi = hi;
EVRY_ITEM(file)->fuzzy_match = match;
EVRY_ITEM(file)->id = eina_stringshare_ref(file->path);
_item_fill(file);
if (!hi->data)
hi->data = eina_stringshare_ref(file->mime);
d->files = eina_list_append(d->files, file);
if (eina_list_count(d->files) > 100)
return EINA_FALSE;
return EINA_TRUE;
}
static Evry_Plugin *
_recentf_browse(Evry_Plugin *plugin, const Evry_Item *it)
{
Plugin *p = NULL;
if (!it || !CHECK_TYPE(it, EVRY_TYPE_FILE))
return NULL;
GET_FILE(file, it);
if (!evry->file_path_get(file) ||
!ecore_file_is_dir(file->path))
return NULL;
EVRY_PLUGIN_INSTANCE(p, plugin);
p->directory = eina_stringshare_add(file->path);
p->parent = EINA_TRUE;
return EVRY_PLUGIN(p);
}
static Evry_Plugin *
_recentf_begin(Evry_Plugin *plugin, const Evry_Item *it)
{
Plugin *p;
if (it && !CHECK_TYPE(it, EVRY_TYPE_ACTION))
return NULL;
EVRY_PLUGIN_INSTANCE(p, plugin);
p->parent = EINA_FALSE;
if (it)
{
/* provide object */
}
else
{
/* provide subject */
p->min_query = plugin->config->min_query;
if (clear_cache)
{
History_Types *ht = evry->history_types_get(EVRY_TYPE_FILE);
if (ht)
eina_hash_foreach(ht->types, _recentf_items_add_cb, p);
clear_cache = EINA_FALSE;
}
}
return EVRY_PLUGIN(p);
}
static int
_recentf_fetch(Evry_Plugin *plugin, const char *input)
{
GET_PLUGIN(p, plugin);
Evry_Item_File *file;
History_Types *ht;
int len = (input ? strlen(input) : 0);
IF_RELEASE(p->input);
/* if (p->thread)
* ecore_thread_cancel(p->thread);
* p->thread = NULL; */
if (input && isspace(input[len - 1]))
return EVRY_PLUGIN_HAS_ITEMS(p);
if (len >= plugin->config->min_query)
{
if (input)
p->input = eina_stringshare_add(input);
if ((ht = evry->history_types_get(EVRY_TYPE_FILE)))
{
Data *d = E_NEW(Data, 1);
d->plugin = p;
eina_hash_foreach(ht->types, _recentf_items_add_cb, d);
EINA_LIST_FREE (d->files, file)
p->files = eina_list_append(p->files, file);
E_FREE(d);
_recentf_files_filter(p);
/* _recentf_end_func(d);
* p->thread = NULL; */
/* p->thread = ecore_thread_run(_recentf_func, _recentf_end_func,
* _recentf_cancel_func, d); */
}
return EVRY_PLUGIN_HAS_ITEMS(p);
}
EVRY_PLUGIN_ITEMS_CLEAR(p);
return 0;
}
/***************************************************************************/
/* actions */
static int
_open_folder_check(Evry_Action *act __UNUSED__, const Evry_Item *it)
{
return it->browseable && e_action_find("fileman");
}
static int
_open_folder_action(Evry_Action *act)
{
E_Action *action;
Eina_List *m;
char *dir;
if (!(action = e_action_find("fileman")))
return 0;
GET_FILE(file, act->it1.item);
if (!(evry->file_path_get(file)))
return 0;
m = e_manager_list();
if (!IS_BROWSEABLE(file))
{
dir = ecore_file_dir_get(file->path);
if (!dir) return 0;
action->func.go(E_OBJECT(m->data), dir);
free(dir);
}
else
{
action->func.go(E_OBJECT(m->data), file->path);
}
return 1;
}
static int
_file_trash_action(Evry_Action *act)
{
Efreet_Uri *euri;
int ok = 0;
int force = (EVRY_ITEM_DATA_INT_GET(act) == ACT_DELETE);
GET_FILE(file, act->it1.item);
if (!(evry->file_url_get(file)))
return 0;
euri = efreet_uri_decode(file->url);
if (euri)
{
ok = efreet_trash_delete_uri(euri, force);
efreet_uri_free(euri);
}
return ok > 0;
}
static int
_file_copy_action(Evry_Action *act)
{
GET_FILE(src, act->it1.item);
GET_FILE(dst, act->it2.item);
char buf[PATH_MAX];
char *ddst;
if (!(evry->file_path_get(src)))
return 0;
if (!(evry->file_path_get(dst)))
return 0;
if (!ecore_file_is_dir(dst->path))
ddst = ecore_file_dir_get(dst->path);
else
ddst = strdup(dst->path);
if (!ddst)
return 0;
snprintf(buf, sizeof(buf), "%s/%s", ddst, ecore_file_file_get(src->path));
free(ddst);
DBG(" %s -> %s\n", src->path, buf);
if (EVRY_ITEM_DATA_INT_GET(act) == ACT_COPY)
{
return ecore_file_cp(src->path, buf);
}
else if (EVRY_ITEM_DATA_INT_GET(act) == ACT_MOVE)
{
return ecore_file_mv(src->path, buf);
}
return 0;
}
static void
_sort_by_date(Plugin *p)
{
Eina_List *l;
Evry_Item_File *file;
struct stat s;
EINA_LIST_FOREACH (p->files, l, file)
{
if (file->modified)
continue;
if (lstat(file->path, &s) == 0)
file->modified = s.st_mtime;
EVRY_ITEM(file)->usage = -1;
}
p->files = eina_list_sort(p->files, -1, _cb_sort_date);
_files_filter(p);
EVRY_PLUGIN_UPDATE(p, EVRY_UPDATE_ADD);
}
static void
_sort_by_name(Plugin *p)
{
Eina_List *l;
Evry_Item *it;
EINA_LIST_FOREACH (p->files, l, it)
it->usage = 0;
p->files = eina_list_sort(p->files, -1, _cb_sort);
_files_filter(p);
EVRY_PLUGIN_UPDATE(p, EVRY_UPDATE_ADD);
}
static int
_file_sort_action(Evry_Action *act)
{
GET_PLUGIN(p, act->it1.item);
if (!p) return 0;
if (EVRY_ITEM_DATA_INT_GET(act) == ACT_SORT_DATE)
{
_sort_by_date(p);
}
else
{
_sort_by_name(p);
}
return 0;
}
static int
_cb_key_down(Evry_Plugin *plugin, const Ecore_Event_Key *ev)
{
GET_PLUGIN(p, plugin);
if (!strcmp(ev->key, "F1"))
{
_sort_by_name(p);
return 1;
}
if (!strcmp(ev->key, "F2"))
{
_sort_by_date(p);
return 1;
}
return 0;
}
static int
_plugins_init(const Evry_API *api)
{
Evry_Action *act, *act_sort_date, *act_sort_name;
Evry_Plugin *p;
int prio = 0;
const char *config_path;
evry = api;
if (!evry->api_version_check(EVRY_API_VERSION))
return EINA_FALSE;
config_path = eina_stringshare_add("launcher/everything-files");
_mime_dir = eina_stringshare_add("inode/directory");
_mime_mount = eina_stringshare_add("inode/mountpoint");
_mime_unknown = eina_stringshare_add("unknown");
#define ACTION_NEW(_name, _type2, _icon, _act, _check, _register) \
act = EVRY_ACTION_NEW(_name, EVRY_TYPE_FILE, _type2, _icon, _act, _check); \
if (_register) evry->action_register(act, prio++); \
_actions = eina_list_append(_actions, act); \
ACTION_NEW(N_("Copy To ..."), EVRY_TYPE_FILE, "go-next",
_file_copy_action, NULL, 1);
act->it2.subtype = EVRY_TYPE_DIR;
EVRY_ITEM_DATA_INT_SET(act, ACT_COPY);
ACTION_NEW(N_("Move To ..."), EVRY_TYPE_FILE, "go-next",
_file_copy_action, NULL, 1);
act->it2.subtype = EVRY_TYPE_DIR;
EVRY_ITEM_DATA_INT_SET(act, ACT_MOVE);
ACTION_NEW(N_("Move to Trash"), 0, "user-trash",
_file_trash_action, NULL, 1);
EVRY_ITEM_DATA_INT_SET(act, ACT_TRASH);
ACTION_NEW(N_("Open Directory"), 0, "folder-open",
_open_folder_action, _open_folder_check, 1);
act->remember_context = EINA_TRUE;
ACTION_NEW(N_("Sort by Date"), 0, "go-up",
_file_sort_action, NULL, 0);
EVRY_ITEM_DATA_INT_SET(act, ACT_SORT_DATE);
act_sort_date = act;
ACTION_NEW(N_("Sort by Name"), 0, "go-up",
_file_sort_action, NULL, 0);
EVRY_ITEM_DATA_INT_SET(act, ACT_SORT_NAME);
act_sort_name = act;
#undef ACTION_NEW
p = EVRY_PLUGIN_BASE(N_("Files"), _module_icon, EVRY_TYPE_FILE,
_begin, _finish, _fetch);
p->input_type = EVRY_TYPE_FILE;
p->cb_key_down = &_cb_key_down;
p->browse = &_browse;
p->config_path = eina_stringshare_ref(config_path);
p->actions = eina_list_append(p->actions, act_sort_date);
p->actions = eina_list_append(p->actions, act_sort_name);
_plugins = eina_list_append(_plugins, p);
if (evry->plugin_register(p, EVRY_PLUGIN_SUBJECT, 2))
p->config->min_query = 1;
p = EVRY_PLUGIN_BASE(N_("Files"), _module_icon, EVRY_TYPE_FILE,
_begin, _finish, _fetch);
p->cb_key_down = &_cb_key_down;
p->browse = &_browse;
p->config_path = eina_stringshare_ref(config_path);
p->actions = eina_list_append(p->actions, act_sort_date);
p->actions = eina_list_append(p->actions, act_sort_name);
_plugins = eina_list_append(_plugins, p);
evry->plugin_register(p, EVRY_PLUGIN_OBJECT, 2);
if (!_conf->show_recent && !_conf->search_recent)
return EINA_TRUE;
p = EVRY_PLUGIN_BASE(N_("Recent Files"), _module_icon, EVRY_TYPE_FILE,
_recentf_begin, _finish, _recentf_fetch);
p->browse = &_recentf_browse;
p->config_path = eina_stringshare_ref(config_path);
if (evry->plugin_register(p, EVRY_PLUGIN_SUBJECT, 3))
{
p->config->top_level = EINA_FALSE;
p->config->min_query = 3;
}
_plugins = eina_list_append(_plugins, p);
p = EVRY_PLUGIN_BASE(N_("Recent Files"), _module_icon, EVRY_TYPE_FILE,
_recentf_begin, _finish, _recentf_fetch);
p->browse = &_recentf_browse;
p->config_path = eina_stringshare_ref(config_path);
if (evry->plugin_register(p, EVRY_PLUGIN_OBJECT, 3))
{
p->config->top_level = EINA_FALSE;
p->config->min_query = 3;
}
_plugins = eina_list_append(_plugins, p);
eina_stringshare_del(config_path);
return EINA_TRUE;
}
static void
_plugins_shutdown(void)
{
Evry_Action *act;
Evry_Plugin *p;
eina_stringshare_del(_mime_dir);
eina_stringshare_del(_mime_mount);
eina_stringshare_del(_mime_unknown);
EINA_LIST_FREE (_plugins, p)
{
if (p->actions)
eina_list_free(p->actions);
EVRY_PLUGIN_FREE(p);
}
EINA_LIST_FREE (_actions, act)
evry->action_free(act);
}
/***************************************************************************/
static E_Config_DD *conf_edd = NULL;
struct _E_Config_Dialog_Data
{
int show_homedir;
int show_recent;
int search_recent;
int search_cache;
int cache_dirs;
};
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 *
compositor rewrite / charlie-foxtrot situation huge fustercluck commit because there wasn't really a way to separate out the changes. better to just rip it all out at once. * compositor and window management completely rewritten. this was the goal for E19, but it pretty much required everything existing to be scrapped since it wasn't optimized, streamlined, or sensible. now instead of having the compositor strapped to the window manager like an outboard motor, it's housed more like an automobile engine. ** various comp structs have been merged into other places (eg. E_Comp_Zone is now just part of E_Zone where applicable), leading to a large deduplication of attributes ** awful E_Comp_Win is totally dead, having been replaced with e_comp_object smart objects which work just like normal canvas objects ** protocol-specific window management and compositor functionality is now kept exclusively in backend files ** e_pixmap api provides generic client finding and rendering api ** screen/xinerama screens are now provided directly by compositor on startup and re-set on change ** e_comp_render_update finally replaced with eina_tiler ** wayland compositor no longer creates X windows ** compositor e_layout removed entirely * e_container is gone. this was made unnecessary in E18, but I kept it to avoid having too much code churn in one release. its sole purpose was to catch some events and handle window stacking, both of which are now just done by the compositor infra * e_manager is just for screensaver and keybind stuff now, possibly remove later? * e_border is gone along with a lot of its api. e_client has replaced it, and e_client has been rewritten completely; some parts may be similar, but the design now relies upon having a functional compositor ** window configuration/focus functions are all removed. all windows are now managed solely with evas_object_X functions on the "frame" member of a client, just as any other canvas object can be managed. *** do NOT set interceptors on a client's comp_object. seriously. * startup order rewritten: compositor now starts much earlier, other things just use attrs and members of the compositor * ecore_x_pointer_xy_get usage replaced with ecore_evas_pointer_xy_get * e_popup is totally gone, existing usage replaced by e_comp_object_util_add where applicable, otherwise just placed normally on the canvas * deskmirror is (more) broken for now * illume is totally fucked * Ecore_X_Window replaced with Ecore_Window in most cases * edge binding XWindows replaced with regular canvas objects * some E_Win functionality has changed such that delete callbacks are now correctly called in ALL cases. various dialogs have been updated to not crash as a result comp files and descriptions: e_comp.c - overall compositor functions, rendering/update loop, shape cutting e_comp_x.c - X window management and compositor functionality e_comp_wl.c - Wayland surface management and compositor functionality e_comp_canvas.c - general compositor canvas functions and utilities e_comp_object.c - E_Client->frame member for managing clients as Evas_Objects, utility functions for adding objects to the compositor rendering systems additional authors: ivan.briano@intel.com feature: new compositor removal: e_border, e_container, e_popup
2014-01-14 17:19:12 -08:00
_conf_dialog(E_Comp *comp, const char *params __UNUSED__)
{
E_Config_Dialog *cfd = NULL;
E_Config_Dialog_View *v = NULL;
if (e_config_dialog_find("everything-files", "extensions/everything-files")) 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;
compositor rewrite / charlie-foxtrot situation huge fustercluck commit because there wasn't really a way to separate out the changes. better to just rip it all out at once. * compositor and window management completely rewritten. this was the goal for E19, but it pretty much required everything existing to be scrapped since it wasn't optimized, streamlined, or sensible. now instead of having the compositor strapped to the window manager like an outboard motor, it's housed more like an automobile engine. ** various comp structs have been merged into other places (eg. E_Comp_Zone is now just part of E_Zone where applicable), leading to a large deduplication of attributes ** awful E_Comp_Win is totally dead, having been replaced with e_comp_object smart objects which work just like normal canvas objects ** protocol-specific window management and compositor functionality is now kept exclusively in backend files ** e_pixmap api provides generic client finding and rendering api ** screen/xinerama screens are now provided directly by compositor on startup and re-set on change ** e_comp_render_update finally replaced with eina_tiler ** wayland compositor no longer creates X windows ** compositor e_layout removed entirely * e_container is gone. this was made unnecessary in E18, but I kept it to avoid having too much code churn in one release. its sole purpose was to catch some events and handle window stacking, both of which are now just done by the compositor infra * e_manager is just for screensaver and keybind stuff now, possibly remove later? * e_border is gone along with a lot of its api. e_client has replaced it, and e_client has been rewritten completely; some parts may be similar, but the design now relies upon having a functional compositor ** window configuration/focus functions are all removed. all windows are now managed solely with evas_object_X functions on the "frame" member of a client, just as any other canvas object can be managed. *** do NOT set interceptors on a client's comp_object. seriously. * startup order rewritten: compositor now starts much earlier, other things just use attrs and members of the compositor * ecore_x_pointer_xy_get usage replaced with ecore_evas_pointer_xy_get * e_popup is totally gone, existing usage replaced by e_comp_object_util_add where applicable, otherwise just placed normally on the canvas * deskmirror is (more) broken for now * illume is totally fucked * Ecore_X_Window replaced with Ecore_Window in most cases * edge binding XWindows replaced with regular canvas objects * some E_Win functionality has changed such that delete callbacks are now correctly called in ALL cases. various dialogs have been updated to not crash as a result comp files and descriptions: e_comp.c - overall compositor functions, rendering/update loop, shape cutting e_comp_x.c - X window management and compositor functionality e_comp_wl.c - Wayland surface management and compositor functionality e_comp_canvas.c - general compositor canvas functions and utilities e_comp_object.c - E_Client->frame member for managing clients as Evas_Objects, utility functions for adding objects to the compositor rendering systems additional authors: ivan.briano@intel.com feature: new compositor removal: e_border, e_container, e_popup
2014-01-14 17:19:12 -08:00
cfd = e_config_dialog_new(comp, _("Everything Files"), "everything-files",
"extensions/everything-files", _module_icon, 0, v, NULL);
_conf->cfd = cfd;
return cfd;
}
static void
_clear_cache_cb(void *data __UNUSED__, void *data2 __UNUSED__)
{
clear_cache = EINA_TRUE;
}
static Evas_Object *
_basic_create(E_Config_Dialog *cfd __UNUSED__, Evas *evas, E_Config_Dialog_Data *cfdata)
{
Evas_Object *o = NULL, *of = NULL, *ow = NULL;
o = e_widget_list_add(evas, 0, 0);
of = e_widget_framelist_add(evas, _("General"), 0);
e_widget_framelist_content_align_set(of, 0.0, 0.0);
/* ow = e_widget_check_add(evas, _("Show home directory"),
* &(cfdata->show_homedir));
* e_widget_framelist_object_append(of, ow); */
ow = e_widget_check_add(evas, _("Show recent files"),
&(cfdata->show_recent));
e_widget_framelist_object_append(of, ow);
ow = e_widget_check_add(evas, _("Search recent files"),
&(cfdata->search_recent));
e_widget_framelist_object_append(of, ow);
ow = e_widget_check_add(evas, _("Search cached files"),
&(cfdata->search_cache));
e_widget_framelist_object_append(of, ow);
ow = e_widget_check_add(evas, _("Cache visited directories"),
&(cfdata->cache_dirs));
e_widget_framelist_object_append(of, ow);
ow = e_widget_button_add(evas, _("Clear cache"), NULL,
_clear_cache_cb,
NULL, NULL);
e_widget_framelist_object_append(of, ow);
e_widget_list_object_append(o, of, 1, 1, 0.5);
return o;
}
static void *
_create_data(E_Config_Dialog *cfd __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 __UNUSED__, E_Config_Dialog_Data *cfdata)
{
_conf->cfd = NULL;
E_FREE(cfdata);
}
static void
_fill_data(E_Config_Dialog_Data *cfdata)
{
#define C(_name) cfdata->_name = _conf->_name;
C(show_homedir);
C(show_recent);
C(search_recent);
C(search_cache);
C(cache_dirs);
#undef C
}
static int
_basic_apply(E_Config_Dialog *cfd __UNUSED__, E_Config_Dialog_Data *cfdata)
{
#define C(_name) _conf->_name = cfdata->_name;
C(show_homedir);
C(show_recent);
C(search_recent);
C(search_cache);
C(cache_dirs);
#undef C
e_config_domain_save("module.everything-files", conf_edd, _conf);
e_config_save_queue();
return 1;
}
static void
_conf_new(void)
{
_conf = E_NEW(Module_Config, 1);
_conf->show_recent = 0;
_conf->show_homedir = 1;
_conf->search_recent = 1;
_conf->cache_dirs = 0;
_conf->search_cache = 0;
_conf->version = MOD_CONFIG_FILE_VERSION;
}
static void
_conf_free(void)
{
E_FREE(_conf);
}
static void
_conf_init(E_Module *m)
{
char title[4096];
snprintf(title, sizeof(title), "%s: %s", _("Everything Plugin"), _("Files"));
e_configure_registry_item_add("launcher/everything-files", 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, show_homedir, UCHAR);
E_CONFIG_VAL(D, T, show_recent, UCHAR);
E_CONFIG_VAL(D, T, search_recent, UCHAR);
E_CONFIG_VAL(D, T, search_cache, UCHAR);
E_CONFIG_VAL(D, T, cache_dirs, UCHAR);
#undef T
#undef D
_conf = e_config_domain_load("module.everything-files", conf_edd);
if (_conf && !e_util_module_config_check(_("Everything Files"),
_conf->version,
MOD_CONFIG_FILE_VERSION))
_conf_free();
if (!_conf) _conf_new();
_conf->module = m;
}
static void
_conf_shutdown(void)
{
e_configure_registry_item_del("launcher/everything-files");
E_FREE(_conf);
E_CONFIG_DD_FREE(conf_edd);
}
/***************************************************************************/
Eina_Bool
evry_plug_files_init(E_Module *m)
{
_conf_init(m);
EVRY_MODULE_NEW(evry_module, evry, _plugins_init, _plugins_shutdown);
return EINA_TRUE;
}
void
evry_plug_files_shutdown(void)
{
EVRY_MODULE_FREE(evry_module);
_conf_shutdown();
}
void
evry_plug_files_save(void)
{
e_config_domain_save("module.everything-files", conf_edd, _conf);
}