You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
587 lines
16 KiB
587 lines
16 KiB
#include "ephoto.h" |
|
|
|
#define ZOOM_MAX 512 |
|
#define ZOOM_MIN 128 |
|
#define ZOOM_STEP 32 |
|
|
|
#define PARENT_DIR "Up" |
|
|
|
typedef struct _Ephoto_Thumb_Browser Ephoto_Thumb_Browser; |
|
|
|
struct _Ephoto_Thumb_Browser |
|
{ |
|
Ephoto *ephoto; |
|
Evas_Object *layout; |
|
Evas_Object *edje; |
|
Evas_Object *grid; |
|
Evas_Object *toolbar; |
|
Eio_File *ls; |
|
struct { |
|
Elm_Toolbar_Item *zoom_in; |
|
Elm_Toolbar_Item *zoom_out; |
|
Elm_Toolbar_Item *view_flow; |
|
Elm_Toolbar_Item *slideshow; |
|
} action; |
|
struct { |
|
const char *path; |
|
void (*cb)(void *data, Ephoto_Entry *entry); |
|
const void *data; |
|
} pending; |
|
struct { |
|
Ecore_Job *change_dir; |
|
} job; |
|
}; |
|
|
|
static Ephoto_Entry * |
|
_first_file_entry_find(Ephoto_Thumb_Browser *tb) |
|
{ |
|
const Eina_List *l; |
|
Ephoto_Entry *entry; |
|
EINA_LIST_FOREACH(tb->ephoto->entries, l, entry) |
|
if (!entry->is_dir) return entry; |
|
return NULL; |
|
} |
|
|
|
static char * |
|
_ephoto_thumb_item_label_get(void *data, Evas_Object *obj __UNUSED__, const char *part __UNUSED__) |
|
{ |
|
Ephoto_Entry *e = data; |
|
return strdup(e->label); |
|
} |
|
|
|
static Evas_Object * |
|
_ephoto_thumb_dir_icon_get(void *data, Evas_Object *obj, const char *part) |
|
{ |
|
Ephoto_Entry *e = data; |
|
const char *f; |
|
int n; |
|
|
|
if (strncmp(part, "elm.swallow.icon.", sizeof("elm.swallow.icon.") - 1) != 0) |
|
return NULL; |
|
|
|
n = atoi(part + sizeof("elm.swallow.icon.") - 1); |
|
if (n < 1) |
|
return NULL; |
|
n--; |
|
|
|
f = eina_list_nth(e->dir_files, n); |
|
if (f) |
|
return ephoto_thumb_add(e->ephoto, obj, f); |
|
|
|
if (e->dir_files_checked) |
|
return NULL; |
|
|
|
return ephoto_directory_thumb_add(obj, e); |
|
} |
|
|
|
static Eina_Bool |
|
_ephoto_thumb_dir_state_get(void *data, Evas_Object *obj __UNUSED__, const char *part) |
|
{ |
|
Ephoto_Entry *e = data; |
|
int n; |
|
|
|
if (strcmp(part, "have_files") == 0) |
|
return !!e->dir_files; |
|
|
|
if (strncmp(part, "have_file.", sizeof("have_file.") - 1) != 0) |
|
return EINA_FALSE; |
|
|
|
n = atoi(part + sizeof("have_file.") - 1); |
|
if (n < 1) |
|
return EINA_FALSE; |
|
return n <= (int)eina_list_count(e->dir_files); |
|
} |
|
|
|
static Evas_Object * |
|
_ephoto_thumb_file_icon_get(void *data, Evas_Object *obj, const char *part __UNUSED__) |
|
{ |
|
Ephoto_Entry *e = data; |
|
return ephoto_thumb_add(e->ephoto, obj, e->path); |
|
} |
|
|
|
static void |
|
_ephoto_thumb_item_del(void *data, Evas_Object *obj __UNUSED__) |
|
{ |
|
Ephoto_Entry *e = data; |
|
e->item = NULL; |
|
} |
|
|
|
static const Elm_Gengrid_Item_Class _ephoto_thumb_dir_class = { |
|
"dir", |
|
{ |
|
_ephoto_thumb_item_label_get, |
|
_ephoto_thumb_dir_icon_get, |
|
_ephoto_thumb_dir_state_get, |
|
_ephoto_thumb_item_del |
|
} |
|
}; |
|
|
|
static const Elm_Gengrid_Item_Class _ephoto_thumb_up_class = { |
|
"up", |
|
{ |
|
_ephoto_thumb_item_label_get, |
|
NULL, |
|
NULL, |
|
_ephoto_thumb_item_del |
|
} |
|
}; |
|
|
|
static const Elm_Gengrid_Item_Class _ephoto_thumb_file_class = { |
|
"file", |
|
{ |
|
_ephoto_thumb_item_label_get, |
|
_ephoto_thumb_file_icon_get, |
|
NULL, |
|
_ephoto_thumb_item_del |
|
} |
|
}; |
|
|
|
static int |
|
_entry_cmp(const void *pa, const void *pb) |
|
{ |
|
const Ephoto_Entry *a = pa, *b = pb; |
|
if (a->is_dir == b->is_dir) |
|
return strcoll(a->basename, b->basename); |
|
else if (a->is_dir) |
|
return -1; |
|
else |
|
return 1; |
|
} |
|
|
|
static void |
|
_ephoto_populate_main(void *data, const Eina_File_Direct_Info *info) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
const Elm_Gengrid_Item_Class *ic; |
|
Edje_Message_Int msg; |
|
Ephoto_Entry *e; |
|
|
|
e = ephoto_entry_new(tb->ephoto, info->path, info->path + info->name_start); |
|
if (info->dirent->d_type == DT_DIR) e->is_dir = EINA_TRUE; |
|
else if (info->dirent->d_type == DT_REG) e->is_dir = EINA_FALSE; |
|
else e->is_dir = !_ephoto_eina_file_direct_info_image_useful(info); |
|
|
|
if (e->is_dir) ic = &_ephoto_thumb_dir_class; |
|
else ic = &_ephoto_thumb_file_class; |
|
|
|
if (!tb->ephoto->entries) |
|
{ |
|
e->item = elm_gengrid_item_append(tb->grid, ic, e, NULL, NULL); |
|
tb->ephoto->entries = eina_list_append(tb->ephoto->entries, e); |
|
} |
|
else |
|
{ |
|
int near_cmp; |
|
Ephoto_Entry *near_entry; |
|
Elm_Gengrid_Item *near_item; |
|
Eina_List *near_node = eina_list_search_sorted_near_list |
|
(tb->ephoto->entries, _entry_cmp, e, &near_cmp); |
|
|
|
near_entry = near_node->data; |
|
near_item = near_entry->item; |
|
if (near_cmp < 0) |
|
{ |
|
e->item = elm_gengrid_item_insert_after |
|
(tb->grid, ic, e, near_item, NULL, NULL); |
|
tb->ephoto->entries = eina_list_append_relative_list |
|
(tb->ephoto->entries, e, near_node); |
|
} |
|
else |
|
{ |
|
e->item = elm_gengrid_item_insert_before |
|
(tb->grid, ic, e, near_item, NULL, NULL); |
|
tb->ephoto->entries = eina_list_prepend_relative_list |
|
(tb->ephoto->entries, e, near_node); |
|
} |
|
} |
|
|
|
if (!e->item) |
|
{ |
|
ERR("could not add item to grid: path '%s'", info->path); |
|
ephoto_entry_free(e); |
|
return; |
|
} |
|
|
|
msg.val = eina_list_count(tb->ephoto->entries); |
|
edje_object_message_send(tb->edje, EDJE_MESSAGE_INT, 1, &msg); |
|
DBG("populate add '%s'", e->path); |
|
|
|
if (tb->pending.path == e->path) |
|
{ |
|
tb->pending.cb((void*)tb->pending.data, e); |
|
tb->pending.cb = NULL; |
|
tb->pending.data = NULL; |
|
eina_stringshare_replace(&tb->pending.path, NULL); |
|
} |
|
} |
|
|
|
static Eina_Bool |
|
_ephoto_populate_filter(void *data __UNUSED__, const Eina_File_Direct_Info *info) |
|
{ |
|
const char *bname = info->path + info->name_start; |
|
|
|
if (bname[0] == '.') return EINA_FALSE; |
|
if (info->dirent->d_type == DT_DIR) return EINA_TRUE; |
|
if (info->dirent->d_type == DT_UNKNOWN) |
|
{ |
|
struct stat st; |
|
if ((stat(info->path, &st) == 0) && (S_ISDIR(st.st_mode))) |
|
return EINA_TRUE; |
|
} |
|
|
|
return _ephoto_eina_file_direct_info_image_useful(info); |
|
} |
|
|
|
static void |
|
_ephoto_populate_end(void *data) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
tb->ls = NULL; |
|
if (tb->pending.cb) |
|
{ |
|
tb->pending.cb((void*)tb->pending.data, NULL); |
|
tb->pending.cb = NULL; |
|
} |
|
tb->pending.data = NULL; |
|
eina_stringshare_replace(&tb->pending.path, NULL); |
|
edje_object_signal_emit(tb->edje, "populate,stop", "ephoto"); |
|
} |
|
|
|
static void |
|
_ephoto_populate_error(int error, void *data) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
if (error) ERR("could not populate: %s", strerror(error)); |
|
edje_object_signal_emit(tb->edje, "populate,error", "ephoto"); |
|
_ephoto_populate_end(tb); |
|
} |
|
|
|
static void |
|
_ephoto_populate_entries(Ephoto_Thumb_Browser *tb) |
|
{ |
|
Edje_External_Param param; |
|
char *parent_dir; |
|
DBG("populate from '%s'", tb->ephoto->config->directory); |
|
|
|
elm_gengrid_clear(tb->grid); |
|
ephoto_entries_free(tb->ephoto); |
|
|
|
parent_dir = ecore_file_dir_get(tb->ephoto->config->directory); |
|
if (parent_dir) |
|
{ |
|
Ephoto_Entry *e = ephoto_entry_new(tb->ephoto, parent_dir, PARENT_DIR); |
|
free(parent_dir); |
|
EINA_SAFETY_ON_NULL_RETURN(e); |
|
e->is_up = EINA_TRUE; |
|
e->is_dir = EINA_TRUE; |
|
e->item = elm_gengrid_item_append |
|
(tb->grid, &_ephoto_thumb_up_class, e, NULL, NULL); |
|
/* does not go into entries as it is always the first - no sort! */ |
|
} |
|
|
|
param.name = "text"; |
|
param.type = EDJE_EXTERNAL_PARAM_TYPE_STRING; |
|
param.s = tb->ephoto->config->directory; |
|
edje_object_part_external_param_set(tb->edje, "ephoto.location", ¶m); |
|
edje_object_signal_emit(tb->edje, "location,set", "ephoto"); |
|
|
|
edje_object_signal_emit(tb->edje, "populate,start", "ephoto"); |
|
tb->ls = eio_file_direct_ls(tb->ephoto->config->directory, |
|
_ephoto_populate_filter, |
|
_ephoto_populate_main, |
|
_ephoto_populate_end, |
|
_ephoto_populate_error, |
|
tb); |
|
} |
|
|
|
static void |
|
_ephoto_thumb_change_dir(void *data) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
tb->job.change_dir = NULL; |
|
_ephoto_populate_entries(tb); |
|
} |
|
|
|
static void |
|
_ephoto_thumb_selected(void *data, Evas_Object *o __UNUSED__, void *event_info) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
Elm_Gengrid_Item *it = event_info; |
|
Ephoto_Entry *e = elm_gengrid_item_data_get(it); |
|
|
|
elm_gengrid_item_selected_set(it, EINA_FALSE); |
|
|
|
if (e->is_dir) |
|
ephoto_thumb_browser_directory_set(tb->layout, e->path); |
|
else |
|
evas_object_smart_callback_call(tb->layout, "view", e); |
|
} |
|
|
|
/* TODO: elm_fileselector_entry and use the activated/file,chosen here */ |
|
/* |
|
static void |
|
_changed_dir(void *data, Evas_Object *o __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
Edje_External_Param p; |
|
|
|
p.name = "text"; |
|
p.type = EDJE_EXTERNAL_PARAM_TYPE_STRING; |
|
p.s = NULL; |
|
if (!edje_object_part_external_param_get(tb->edje, "ephoto.location", &p)) |
|
return; |
|
|
|
ephoto_thumb_browser_directory_set(tb->layout, p.s); |
|
} |
|
*/ |
|
|
|
static void |
|
_zoom_set(Ephoto_Thumb_Browser *tb, int zoom) |
|
{ |
|
if (zoom > ZOOM_MAX) zoom = ZOOM_MAX; |
|
else if (zoom < ZOOM_MIN) zoom = ZOOM_MIN; |
|
|
|
ephoto_thumb_size_set(tb->ephoto, zoom); |
|
elm_gengrid_item_size_set(tb->grid, zoom, zoom); |
|
|
|
elm_toolbar_item_disabled_set(tb->action.zoom_out, zoom == ZOOM_MIN); |
|
elm_toolbar_item_disabled_set(tb->action.zoom_in, zoom == ZOOM_MAX); |
|
} |
|
|
|
static void |
|
_zoom_in(void *data, Evas_Object *o __UNUSED__, void *event_info __UNUSED__) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
elm_toolbar_item_unselect(tb->action.zoom_in); |
|
_zoom_set(tb, tb->ephoto->config->thumb_size + ZOOM_STEP); |
|
} |
|
|
|
static void |
|
_zoom_out(void *data, Evas_Object *o __UNUSED__, void *event_info __UNUSED__) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
elm_toolbar_item_unselect(tb->action.zoom_out); |
|
_zoom_set(tb, tb->ephoto->config->thumb_size - ZOOM_STEP); |
|
} |
|
|
|
static void |
|
_view_flow(void *data, Evas_Object *o __UNUSED__, void *event_info __UNUSED__) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
Elm_Gengrid_Item *it = elm_gengrid_selected_item_get(tb->grid); |
|
Ephoto_Entry *entry; |
|
|
|
elm_toolbar_item_unselect(tb->action.view_flow); |
|
|
|
if (it) entry = elm_gengrid_item_data_get(it); |
|
else entry = _first_file_entry_find(tb); |
|
|
|
if (!entry) return; |
|
if (entry->is_dir) |
|
ephoto_thumb_browser_directory_set(tb->layout, entry->path); |
|
else |
|
evas_object_smart_callback_call(tb->layout, "view", entry); |
|
} |
|
|
|
static void |
|
_slideshow(void *data, Evas_Object *o __UNUSED__, void *event_info __UNUSED__) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
Elm_Gengrid_Item *it = elm_gengrid_selected_item_get(tb->grid); |
|
Ephoto_Entry *entry; |
|
|
|
elm_toolbar_item_unselect(tb->action.slideshow); |
|
|
|
if (it) entry = elm_gengrid_item_data_get(it); |
|
else entry = _first_file_entry_find(tb); |
|
|
|
if (!entry) return; |
|
evas_object_smart_callback_call(tb->layout, "slideshow", entry); |
|
} |
|
|
|
static void |
|
_key_down(void *data, Evas *e __UNUSED__, Evas_Object *o __UNUSED__, void *event_info) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
Evas_Event_Key_Down *ev = event_info; |
|
Eina_Bool alt = evas_key_modifier_is_set(ev->modifiers, "Alt"); |
|
const char *k = ev->keyname; |
|
|
|
if (alt) |
|
{ |
|
if (!strcmp(k, "Up")) |
|
{ |
|
char *parent = ecore_file_dir_get(tb->ephoto->config->directory); |
|
if (parent) |
|
ephoto_thumb_browser_directory_set(tb->layout, parent); |
|
free(parent); |
|
} |
|
|
|
return; |
|
} |
|
|
|
if (!strcmp(k, "F5")) |
|
{ |
|
Elm_Gengrid_Item *it = elm_gengrid_selected_item_get(tb->grid); |
|
Ephoto_Entry *entry; |
|
if (it) entry = elm_gengrid_item_data_get(it); |
|
else entry = _first_file_entry_find(tb); |
|
|
|
if (entry) |
|
evas_object_smart_callback_call(tb->layout, "slideshow", entry); |
|
} |
|
} |
|
|
|
|
|
static void |
|
_layout_del(void *data, Evas *e __UNUSED__, Evas_Object *o __UNUSED__, void *event_info __UNUSED__) |
|
{ |
|
Ephoto_Thumb_Browser *tb = data; |
|
if (tb->pending.cb) tb->pending.cb((void*)tb->pending.data, NULL); |
|
eina_stringshare_del(tb->pending.path); |
|
if (tb->ls) eio_file_cancel(tb->ls); |
|
if (tb->job.change_dir) ecore_job_del(tb->job.change_dir); |
|
free(tb); |
|
} |
|
|
|
/* FIXME: this should go in elm_icon itself! */ |
|
static Eina_Bool |
|
_icon_load(Evas_Object *ic, const char *name, int size) |
|
{ |
|
char *path; |
|
|
|
if (elm_icon_standard_set(ic, name)) return EINA_TRUE; |
|
|
|
path = efreet_icon_path_find(getenv("E_ICON_THEME"), name, size); |
|
if (!path) |
|
{ |
|
const char **itr, *themes[] = { |
|
"default", "highcolor", "gnome", "Human", "oxygen", NULL |
|
}; |
|
for (itr = themes; *itr; itr++) |
|
{ |
|
path = efreet_icon_path_find(*itr, name, size); |
|
if (path) break; |
|
} |
|
} |
|
if (!path) return EINA_FALSE; |
|
elm_icon_file_set(ic, path, NULL); |
|
free(path); |
|
return EINA_TRUE; |
|
} |
|
|
|
static Elm_Toolbar_Item * |
|
_toolbar_item_add(Ephoto_Thumb_Browser *tb, const char *icon, const char *label, Evas_Smart_Cb cb) |
|
{ |
|
/* FIXME TODO: toolbar should get a string as icon, get it from theme |
|
* for toolbar or menu, and fallback to freedesktop.org icon if not found |
|
*/ |
|
Evas_Object *ic = elm_icon_add(tb->toolbar); |
|
if (!_icon_load(ic, icon, elm_toolbar_icon_size_get(tb->toolbar))) |
|
{ |
|
evas_object_del(ic); |
|
ic = NULL; |
|
} |
|
return elm_toolbar_item_add(tb->toolbar, ic, label, cb, tb); |
|
} |
|
|
|
Evas_Object * |
|
ephoto_thumb_browser_add(Ephoto *ephoto, Evas_Object *parent) |
|
{ |
|
Evas_Object *layout = elm_layout_add(parent); |
|
Ephoto_Thumb_Browser *tb; |
|
|
|
EINA_SAFETY_ON_NULL_RETURN_VAL(layout, NULL); |
|
|
|
tb = calloc(1, sizeof(Ephoto_Thumb_Browser)); |
|
EINA_SAFETY_ON_NULL_GOTO(tb, error); |
|
tb->ephoto = ephoto; |
|
tb->layout = layout; |
|
tb->edje = elm_layout_edje_get(layout); |
|
evas_object_event_callback_add(layout, EVAS_CALLBACK_DEL, _layout_del, tb); |
|
evas_object_event_callback_add |
|
(layout, EVAS_CALLBACK_KEY_DOWN, _key_down, tb); |
|
evas_object_data_set(layout, "thumb_browser", tb); |
|
|
|
if (!elm_layout_theme_set |
|
(layout, "layout", "application", "toolbar-content")) |
|
{ |
|
ERR("could not load style 'toolbar-content' from theme"); |
|
goto error; |
|
} |
|
|
|
tb->toolbar = edje_object_part_external_object_get |
|
(tb->edje, "elm.external.toolbar"); |
|
if (!tb->toolbar) |
|
{ |
|
ERR("no toolbar in layout!"); |
|
goto error; |
|
} |
|
elm_toolbar_homogenous_set(tb->toolbar, EINA_FALSE); |
|
|
|
tb->action.slideshow = _toolbar_item_add |
|
(tb, "media-playback-start", "Slideshow", _slideshow); |
|
tb->action.zoom_in = _toolbar_item_add |
|
(tb, "zoom-in", "Zoom In", _zoom_in); |
|
tb->action.zoom_out = _toolbar_item_add |
|
(tb, "zoom-out", "Zoom Out", _zoom_out); |
|
tb->action.view_flow = _toolbar_item_add |
|
(tb, "image", "Larger", _view_flow); |
|
|
|
/* TODO: elm_fileselector_entry and use it with a toolbar-vbox |
|
* |
|
* TODO: elm_layout_box_*, similar to elm_layout_content_set(), it will |
|
* register the sub-object and re-add it during theme changes. |
|
*/ |
|
|
|
tb->grid = elm_gengrid_add(layout); |
|
EINA_SAFETY_ON_NULL_GOTO(tb->grid, error); |
|
elm_object_style_set(tb->grid, "ephoto"); |
|
|
|
elm_gengrid_align_set(tb->grid, 0.5, 0.5); |
|
elm_gengrid_bounce_set(tb->grid, EINA_FALSE, EINA_TRUE); |
|
evas_object_size_hint_align_set |
|
(tb->grid, EVAS_HINT_FILL, EVAS_HINT_FILL); |
|
evas_object_size_hint_weight_set |
|
(tb->grid, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND); |
|
|
|
evas_object_smart_callback_add |
|
(tb->grid, "selected", _ephoto_thumb_selected, tb); |
|
|
|
_zoom_set(tb, tb->ephoto->config->thumb_size); |
|
|
|
elm_layout_content_set(tb->layout, "elm.swallow.content", tb->grid); |
|
|
|
return layout; |
|
|
|
error: |
|
evas_object_del(layout); |
|
return NULL; |
|
} |
|
|
|
void |
|
ephoto_thumb_browser_directory_set(Evas_Object *obj, const char *path) |
|
{ |
|
Ephoto_Thumb_Browser *tb = evas_object_data_get(obj, "thumb_browser"); |
|
EINA_SAFETY_ON_NULL_RETURN(tb); |
|
|
|
eina_stringshare_replace(&tb->pending.path, NULL); |
|
tb->pending.cb = NULL; |
|
tb->pending.data = NULL; |
|
|
|
ephoto_title_set(tb->ephoto, path); |
|
|
|
eina_stringshare_replace(&tb->ephoto->config->directory, path); |
|
if (tb->job.change_dir) ecore_job_del(tb->job.change_dir); |
|
tb->job.change_dir = ecore_job_add(_ephoto_thumb_change_dir, tb); |
|
} |
|
|
|
void |
|
ephoto_thumb_browser_path_pending_set(Evas_Object *obj, const char *path, void (*cb)(void *data, Ephoto_Entry *entry), const void *data) |
|
{ |
|
Ephoto_Thumb_Browser *tb = evas_object_data_get(obj, "thumb_browser"); |
|
EINA_SAFETY_ON_NULL_RETURN(tb); |
|
eina_stringshare_replace(&tb->pending.path, path); |
|
tb->pending.cb = cb; |
|
tb->pending.data = data; |
|
}
|
|
|