/* * TODO: * - child elements focusing support * - user defined icon/label cb * - show/hide/add buttons ??? * - Pattern Filter support */ #ifdef HAVE_CONFIG_H # include "elementary_config.h" #endif #include #include "elm_priv.h" #include "elm_interface_fileselector.h" #include "elm_widget_fileselector.h" EAPI Eo_Op ELM_OBJ_FILESELECTOR_BASE_ID = EO_NOOP; #define MY_CLASS ELM_OBJ_FILESELECTOR_CLASS #define MY_CLASS_NAME "Elm_Fileselector" #define MY_CLASS_NAME_LEGACY "elm_fileselector" /* FIXME: need a way to find a gap between the size of item and thumbnail */ #define GENGRID_PADDING 16 static Elm_Genlist_Item_Class *list_itc[ELM_FILE_LAST]; static Elm_Gengrid_Item_Class *grid_itc[ELM_FILE_LAST]; #define ELM_PRIV_FILESELECTOR_SIGNALS(cmd) \ cmd(SIG_ACTIVATED, "activated", "s") \ cmd(SIG_DIRECTORY_OPEN, "directory,open", "s") \ cmd(SIG_DONE, "done", "s") \ cmd(SIG_SELECTED, "selected", "s") \ cmd(SIG_SELECTED_INVALID, "selected,invalid", "s") ELM_PRIV_FILESELECTOR_SIGNALS(ELM_PRIV_STATIC_VARIABLE_DECLARE); static const Evas_Smart_Cb_Description _smart_callbacks[] = { ELM_PRIV_FILESELECTOR_SIGNALS(ELM_PRIV_SMART_CALLBACKS_DESC) {NULL, NULL} }; #undef ELM_PRIV_FILESELECTOR_SIGNALS /* final routine on deletion */ static void _elm_fileselector_smart_del_do(Elm_Fileselector_Smart_Data *sd) { eina_stringshare_del(sd->path); eina_stringshare_del(sd->selection); free(ecore_idler_del(sd->populate_idler)); eo_do_super(sd->obj, MY_CLASS, evas_obj_smart_del()); } static void _elm_fileselector_smart_sizing_eval(Eo *obj, void *_pd EINA_UNUSED, va_list *list EINA_UNUSED) { Evas_Coord minw = -1, minh = -1; ELM_WIDGET_DATA_GET_OR_RETURN(obj, wd); elm_coords_finger_size_adjust(1, &minw, 1, &minh); edje_object_size_min_restricted_calc (wd->resize_obj, &minw, &minh, minw, minh); evas_object_size_hint_min_set(obj, minw, minh); } static void _mirrored_set(Evas_Object *obj, Eina_Bool rtl) { ELM_FILESELECTOR_DATA_GET(obj, sd); elm_widget_mirrored_set(sd->cancel_button, rtl); elm_widget_mirrored_set(sd->ok_button, rtl); elm_widget_mirrored_set(sd->files_view, rtl); elm_widget_mirrored_set(sd->up_button, rtl); elm_widget_mirrored_set(sd->home_button, rtl); } static void _elm_fileselector_smart_theme(Eo *obj, void *_pd, va_list *list) { const char *style; const char *data; char buf[1024]; Elm_Fileselector_Smart_Data *sd = _pd; ELM_WIDGET_DATA_GET_OR_RETURN(obj, wd); Eina_Bool *ret = va_arg(*list, Eina_Bool *); if (ret) *ret = EINA_FALSE; Eina_Bool int_ret; eo_do_super(obj, MY_CLASS, elm_wdg_theme_apply(&int_ret)); if (!int_ret) return; style = elm_widget_style_get(obj); _mirrored_set(obj, elm_widget_mirrored_get(obj)); data = edje_object_data_get (wd->resize_obj, "path_separator"); if (data) sd->path_separator = data; else sd->path_separator = "/"; snprintf(buf, sizeof(buf), "fileselector/%s", style); #define SWALLOW(part_name, object_ptn) \ if (object_ptn) \ { \ elm_widget_style_set(object_ptn, buf); \ if (!elm_layout_content_set(obj, part_name, object_ptn)) \ evas_object_hide(object_ptn); \ } SWALLOW("elm.swallow.up", sd->up_button); SWALLOW("elm.swallow.home", sd->home_button); SWALLOW("elm.swallow.spinner", sd->spinner); elm_object_style_set(sd->spinner, "wheel"); if (!elm_layout_content_set(obj, "elm.swallow.files", sd->files_view)) evas_object_hide(sd->files_view); SWALLOW("elm.swallow.path", sd->path_entry); SWALLOW("elm.swallow.filename", sd->name_entry); snprintf(buf, sizeof(buf), "fileselector/actions/%s", style); SWALLOW("elm.swallow.filters", sd->filter_hoversel); SWALLOW("elm.swallow.cancel", sd->cancel_button); SWALLOW("elm.swallow.ok", sd->ok_button); #undef SWALLOW edje_object_message_signal_process(wd->resize_obj); elm_layout_sizing_eval(obj); if (ret) *ret = EINA_TRUE; } /*** GENLIST/GENGRID "MODEL" ***/ static char * _itc_text_get(void *data, Evas_Object *obj EINA_UNUSED, const char *source EINA_UNUSED) { return elm_entry_utf8_to_markup (ecore_file_file_get(data)); /* NOTE this will be free()'d by * the caller */ } static Evas_Object * _itc_icon_folder_get(void *data EINA_UNUSED, Evas_Object *obj, const char *source) { Evas_Object *ic, *grid, *f; if (strcmp(source, "elm.swallow.icon")) return NULL; f = evas_object_data_get(obj, "parent"); ELM_FILESELECTOR_DATA_GET(f, sd); ic = elm_icon_add(obj); elm_icon_standard_set(ic, "folder"); evas_object_size_hint_aspect_set(ic, EVAS_ASPECT_CONTROL_VERTICAL, 1, 1); evas_object_show(ic); grid = elm_grid_add(obj); elm_grid_size_set(grid, 1, 1); elm_grid_pack(grid, ic, 0, 0, 1, 1); evas_object_size_hint_min_set(grid, sd->thumbnail_size.w, sd->thumbnail_size.h); evas_object_show(grid); return grid; } static Evas_Object * _itc_icon_image_get(void *data, Evas_Object *obj, const char *source) { const char *filename = data; Evas_Object *ic, *grid, *f; if (strcmp(source, "elm.swallow.icon")) return NULL; f = evas_object_data_get(obj, "parent"); ELM_FILESELECTOR_DATA_GET(f, sd); ic = elm_icon_add(obj); elm_icon_standard_set(ic, "image"); elm_icon_thumb_set(ic, filename, NULL); evas_object_size_hint_aspect_set(ic, EVAS_ASPECT_CONTROL_VERTICAL, 1, 1); evas_object_show(ic); grid = elm_grid_add(obj); elm_grid_size_set(grid, 1, 1); elm_grid_pack(grid, ic, 0, 0, 1, 1); evas_object_size_hint_min_set(grid, sd->thumbnail_size.w, sd->thumbnail_size.h); evas_object_show(grid); return grid; } static Evas_Object * _itc_icon_file_get(void *data EINA_UNUSED, Evas_Object *obj, const char *source) { Evas_Object *ic, *grid, *f; if (strcmp(source, "elm.swallow.icon")) return NULL; f = evas_object_data_get(obj, "parent"); ELM_FILESELECTOR_DATA_GET(f, sd); ic = elm_icon_add(obj); elm_icon_standard_set(ic, "file"); evas_object_size_hint_aspect_set(ic, EVAS_ASPECT_CONTROL_VERTICAL, 1, 1); evas_object_show(ic); grid = elm_grid_add(obj); elm_grid_size_set(grid, 1, 1); elm_grid_pack(grid, ic, 0, 0, 1, 1); evas_object_size_hint_min_set(grid, sd->thumbnail_size.w, sd->thumbnail_size.h); evas_object_show(grid); return grid; } static Eina_Bool _itc_state_get(void *data EINA_UNUSED, Evas_Object *obj EINA_UNUSED, const char *source EINA_UNUSED) { return EINA_FALSE; } static void _itc_del(void *data, Evas_Object *obj EINA_UNUSED) { eina_stringshare_del(data); } static void _anchors_do(Evas_Object *obj, const char *path) { char **tok, buf[PATH_MAX * 3], *s; int i, j; ELM_FILESELECTOR_DATA_GET(obj, sd); s = elm_entry_utf8_to_markup(path); if (!s) return; buf[0] = '\0'; tok = eina_str_split(s, "/", 0); free(s); eina_strlcat(buf, "root", sizeof(buf)); for (i = 0; tok[i]; i++) { if ((!tok[i]) || (!tok[i][0])) continue; eina_strlcat(buf, sd->path_separator, sizeof(buf)); eina_strlcat(buf, "", sizeof(buf)); eina_strlcat(buf, tok[i], sizeof(buf)); eina_strlcat(buf, "", sizeof(buf)); } free(tok[0]); free(tok); elm_object_text_set(sd->path_entry, buf); } static Eina_Bool _mime_type_matched(const char *mime_filter, const char *mime_type) { int i = 0; while (mime_filter[i] != '\0') { if (mime_filter[i] != mime_type[i]) { if (mime_filter[i] == '*' && mime_filter[i + 1] == '\0') return EINA_TRUE; return EINA_FALSE; } i++; } if (mime_type[i] != '\0') return EINA_FALSE; return EINA_TRUE; } static Eina_Bool _check_mime_type_filter(const Elm_Fileselector_Filter *filter, const char *file_name) { const char *mime_type = NULL; int i; mime_type = efreet_mime_type_get(file_name); if (!mime_type) return EINA_FALSE; for (i = 0; filter->filter.mime_types[i]; ++i) { if (_mime_type_matched(filter->filter.mime_types[i], mime_type)) return EINA_TRUE; } return EINA_FALSE; } static Eina_Bool _ls_filter_cb(void *data, Eio_File *handler EINA_UNUSED, const Eina_File_Direct_Info *info) { Listing_Request *lreq = data; Elm_Fileselector_Filter *cf; Eina_Bool dir = EINA_FALSE; if (!lreq->sd->hidden_visible && info->path[info->name_start] == '.') return EINA_FALSE; if (info->type == EINA_FILE_DIR) dir = EINA_TRUE; if (lreq->sd->only_folder && !dir) return EINA_FALSE; cf = lreq->sd->current_filter; if (!cf) return EINA_TRUE; switch (cf->filter_type) { case ELM_FILESELECTOR_MIME_FILTER: return dir || _check_mime_type_filter(cf, info->path); case ELM_FILESELECTOR_CUSTOM_FILTER: return cf->filter.custom->func(info->path, dir, cf->filter.custom->data); default: return EINA_FALSE; } return EINA_FALSE; } static const char * _file_type(const char *a) { char *p = strrchr(a, '.'); if (!p) return ""; return p; } static int _strcoll_rev(const char *a, const char *b) { return strcoll(b, a); } static int _strcoll_type(const char *a, const char *b) { return strcoll(_file_type(a), _file_type(b)); } static int _strcoll_type_rev(const char *a, const char *b) { return _strcoll_type(b, a); } static int _size_cmp(const char *a, const char *b) { return ecore_file_size(a) - ecore_file_size(b); } static int _size_cmp_rev(const char *a, const char *b) { return _size_cmp(b, a); } static int _modified_cmp(const char *a, const char *b) { if (ecore_file_mod_time(a) > ecore_file_mod_time(b)) return 1; return -1; } static int _modified_cmp_rev(const char *a, const char *b) { return _modified_cmp(b, a); } static int _file_grid_cmp(const void *a, const void *b) { Evas_Object *f; const Elm_Object_Item *ga = a; const Elm_Object_Item *gb = b; const Elm_Gengrid_Item_Class *ca = elm_gengrid_item_item_class_get(ga); const Elm_Gengrid_Item_Class *cb = elm_gengrid_item_item_class_get(gb); if (ca == grid_itc[ELM_DIRECTORY]) { if (cb != grid_itc[ELM_DIRECTORY]) return -1; } else if (cb == grid_itc[ELM_DIRECTORY]) { return 1; } f = evas_object_data_get(elm_object_item_widget_get(ga), "parent"); ELM_FILESELECTOR_DATA_GET(f, sd); return sd->sort_method(elm_object_item_data_get(ga), elm_object_item_data_get(gb)); } static int _file_list_cmp(const void *a, const void *b) { Evas_Object *f; const Elm_Object_Item *la = a; const Elm_Object_Item *lb = b; const Elm_Genlist_Item_Class *ca = elm_genlist_item_item_class_get(la); const Elm_Genlist_Item_Class *cb = elm_genlist_item_item_class_get(lb); if (ca == list_itc[ELM_DIRECTORY]) { if (cb != list_itc[ELM_DIRECTORY]) return -1; } else if (cb == list_itc[ELM_DIRECTORY]) { return 1; } f = evas_object_data_get(elm_object_item_widget_get(la), "parent"); ELM_FILESELECTOR_DATA_GET(f, sd); return sd->sort_method(elm_object_item_data_get(la), elm_object_item_data_get(lb)); } static void _signal_first(Listing_Request *lreq) { if (!lreq->first) return; if (lreq->sd->multi) { char *path; EINA_LIST_FREE(lreq->sd->paths, path) free(path); } evas_object_smart_callback_call (lreq->obj, SIG_DIRECTORY_OPEN, (void *)lreq->path); if (!lreq->parent_it) { if (lreq->sd->mode == ELM_FILESELECTOR_LIST) elm_genlist_clear(lreq->sd->files_view); else elm_gengrid_clear(lreq->sd->files_view); eina_stringshare_replace(&lreq->sd->path, lreq->path); _anchors_do(lreq->obj, lreq->path); elm_object_text_set(lreq->sd->name_entry, ""); } lreq->first = EINA_FALSE; } static void _ls_main_cb(void *data, Eio_File *handler, const Eina_File_Direct_Info *info) { Listing_Request *lreq = data; Elm_Object_Item *item; int itcn = ELM_FILE_UNKNOW; if (eio_file_check(handler)) return; if (!lreq->sd->files_view || lreq->sd->current != handler) { eio_file_cancel(handler); return; } _signal_first(lreq); if (info->type == EINA_FILE_DIR) itcn = ELM_DIRECTORY; else { if (evas_object_image_extension_can_load_get (info->path + info->name_start)) itcn = ELM_FILE_IMAGE; } if (lreq->sd->mode == ELM_FILESELECTOR_LIST) { item = elm_genlist_item_sorted_insert(lreq->sd->files_view, list_itc[itcn], eina_stringshare_add(info->path), lreq->parent_it, ((lreq->sd->expand) && (itcn == ELM_DIRECTORY)) ? ELM_GENLIST_ITEM_TREE : ELM_GENLIST_ITEM_NONE, _file_list_cmp, NULL, NULL); if (lreq->selected && !strcmp(info->path, lreq->selected)) { elm_genlist_item_selected_set(item, EINA_TRUE); elm_object_text_set(lreq->sd->name_entry, ecore_file_file_get(info->path)); } } else if (lreq->sd->mode == ELM_FILESELECTOR_GRID) { item = elm_gengrid_item_sorted_insert(lreq->sd->files_view, grid_itc[itcn], eina_stringshare_add(info->path), _file_grid_cmp, NULL, NULL); if (lreq->selected && !strcmp(info->path, lreq->selected)) { elm_gengrid_item_selected_set(item, EINA_TRUE); elm_object_text_set(lreq->sd->name_entry, ecore_file_file_get(info->path)); } } } static void _listing_request_cleanup(Listing_Request *lreq) { EINA_REFCOUNT_UNREF(lreq->sd) _elm_fileselector_smart_del_do(lreq->sd); eina_stringshare_del(lreq->path); eina_stringshare_del(lreq->selected); free(lreq); } static void _ls_done_cb(void *data, Eio_File *handler EINA_UNUSED) { Listing_Request *lreq = data; _signal_first(lreq); elm_progressbar_pulse(lreq->sd->spinner, EINA_FALSE); elm_layout_signal_emit(lreq->obj, "elm,action,spinner,hide", "elm"); lreq->sd->current = NULL; _listing_request_cleanup(lreq); } static void _ls_error_cb(void *data, Eio_File *handler, int error EINA_UNUSED) { Listing_Request *lreq = data; elm_progressbar_pulse(lreq->sd->spinner, EINA_FALSE); elm_layout_signal_emit(lreq->obj, "elm,action,spinner,hide", "elm"); if (lreq->sd->current == handler) lreq->sd->current = NULL; _listing_request_cleanup(lreq); } static void _populate(Evas_Object *obj, const char *path, Elm_Object_Item *parent_it, const char *selected) { ELM_FILESELECTOR_DATA_GET(obj, sd); if (!path) return; Listing_Request *lreq; if (sd->expand && sd->current) return; if (sd->monitor) eio_monitor_del(sd->monitor); if (sd->current) eio_file_cancel(sd->current); lreq = malloc(sizeof (Listing_Request)); if (!lreq) return; lreq->sd = sd; EINA_REFCOUNT_REF(lreq->sd); lreq->parent_it = parent_it; /* FIXME: should we refcount the parent_it ? */ lreq->obj = obj; lreq->path = eina_stringshare_add(path); lreq->first = EINA_TRUE; if (selected) lreq->selected = eina_stringshare_add(selected); else lreq->selected = NULL; /* TODO: sub directory should be monitored for expand mode */ sd->monitor = eio_monitor_add(path); sd->current = eio_file_stat_ls(path, _ls_filter_cb, _ls_main_cb, _ls_done_cb, _ls_error_cb, lreq); elm_progressbar_pulse(sd->spinner, EINA_TRUE); elm_layout_signal_emit(lreq->obj, "elm,action,spinner,show", "elm"); } static void _on_list_expanded(void *data, Evas_Object *obj EINA_UNUSED, void *event_info) { Elm_Object_Item *it = event_info; const char *path = elm_object_item_data_get(it); _populate(data, path, it, NULL); } static void _on_list_contracted(void *data EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info) { Elm_Object_Item *it = event_info; elm_genlist_item_subitems_clear(it); } static void _on_list_expand_req(void *data EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info) { Elm_Object_Item *it = event_info; elm_genlist_item_expanded_set(it, EINA_TRUE); } static void _on_list_contract_req(void *data EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info) { Elm_Object_Item *it = event_info; elm_genlist_item_expanded_set(it, EINA_FALSE); } static Eina_Bool _populate_do(void *data) { struct sel_data *sdata = data; const char *p; ELM_FILESELECTOR_DATA_GET(sdata->fs, sd); /* keep a ref to path 'couse it will be destroyed by _populate */ p = eina_stringshare_add(sdata->path); _populate(sdata->fs, p, NULL, NULL); eina_stringshare_del(p); sd->populate_idler = NULL; free(sdata); return ECORE_CALLBACK_CANCEL; } static void _on_item_activated(void *data, Evas_Object *obj EINA_UNUSED, void *event_info) { //This event_info could be a list or gengrid item Elm_Object_Item *it = event_info; struct sel_data *sdata; void *old_sdata; const char *path; Eina_Bool is_dir; ELM_FILESELECTOR_DATA_GET(data, sd); path = elm_object_item_data_get(it); if (!path) return; is_dir = ecore_file_is_dir(path); if (!is_dir) { evas_object_smart_callback_call(data, SIG_ACTIVATED, (void *)path); return; } if (!sd->double_tap_navigation) return; sdata = malloc(sizeof(*sdata)); if (!sdata) return; sdata->fs = data; sdata->path = path; if (sd->populate_idler) { old_sdata = ecore_idler_del(sd->populate_idler); free(old_sdata); } sd->populate_idler = ecore_idler_add(_populate_do, sdata); } static void _clear_selections(Elm_Fileselector_Smart_Data *sd, Elm_Object_Item *last_selected) { Eina_List *items; Elm_Object_Item *sel; if (sd->mode == ELM_FILESELECTOR_LIST) { items = eina_list_clone(elm_genlist_selected_items_get(sd->files_view)); EINA_LIST_FREE(items, sel) { if (sel == last_selected) continue; elm_genlist_item_selected_set(sel, EINA_FALSE); } } else if (sd->mode == ELM_FILESELECTOR_GRID) { items = eina_list_clone(elm_gengrid_selected_items_get(sd->files_view)); EINA_LIST_FREE(items, sel) { if (sel == last_selected) continue; elm_gengrid_item_selected_set(sel, EINA_FALSE); } } } static void _on_item_selected(void *data, Evas_Object *obj EINA_UNUSED, void *event_info) { //This event_info could be a list or gengrid item Elm_Object_Item *it = event_info; struct sel_data *sdata; void *old_sdata; const char *path; char *parent_path; Eina_Bool is_dir; ELM_FILESELECTOR_DATA_GET(data, sd); path = elm_object_item_data_get(it); if (!path) return; is_dir = ecore_file_is_dir(path); /* We need to send callback when: * - path is dir and mode is ONLY FOLDER * - path is file and mode is NOT ONLY FOLDER */ if (is_dir == sd->only_folder) { if (sd->multi) { Eina_List *li; const char *p; Eina_Strbuf *buf; if (sd->dir_selected) { _clear_selections(sd, it); sd->dir_selected = EINA_FALSE; } buf = eina_strbuf_new(); EINA_LIST_FOREACH(sd->paths, li, p) { eina_strbuf_append(buf, ecore_file_file_get(p)); eina_strbuf_append_length(buf, ", ", 2); } sd->paths = eina_list_append(sd->paths, strdup(path)); eina_strbuf_append(buf, ecore_file_file_get(path)); elm_object_text_set(sd->name_entry, eina_strbuf_string_get(buf)); eina_strbuf_free(buf); } else elm_object_text_set(sd->name_entry, ecore_file_file_get(path)); evas_object_smart_callback_call(data, SIG_SELECTED, (void *)path); } else { if (sd->multi && is_dir && sd->double_tap_navigation) { _clear_selections(sd, it); sd->dir_selected = EINA_TRUE; } elm_object_text_set(sd->name_entry, ""); } /* We need to populate, if path is directory and: * - mode is GRID; * - mode is LIST and 'not expand mode'; * in other cases update anchors. */ if (!is_dir) return; if (sd->expand && sd->mode == ELM_FILESELECTOR_LIST) { if (sd->only_folder) { parent_path = ecore_file_dir_get(path); eina_stringshare_replace(&sd->path, parent_path); _anchors_do(data, parent_path); free(parent_path); } else { eina_stringshare_replace(&sd->path, path); _anchors_do(data, path); } return; } if (sd->double_tap_navigation) return; sdata = malloc(sizeof(*sdata)); if (!sdata) return; sdata->fs = data; sdata->path = path; if (sd->populate_idler) { old_sdata = ecore_idler_del(sd->populate_idler); free(old_sdata); } sd->populate_idler = ecore_idler_add(_populate_do, sdata); } static void _on_item_unselected(void *data, Evas_Object *obj EINA_UNUSED, void *event_info) { Eina_List *li, *l; char *path; const char *unselected_path; Eina_Strbuf *buf; Elm_Object_Item *it = event_info; Eina_Bool first = EINA_TRUE; ELM_FILESELECTOR_DATA_GET(data, sd); if (!sd->multi) return; unselected_path = elm_object_item_data_get(it); if (!unselected_path) return; buf = eina_strbuf_new(); EINA_LIST_FOREACH_SAFE(sd->paths, li, l, path) { if (!strcmp(path, unselected_path)) { sd->paths = eina_list_remove_list(sd->paths, li); free(path); } else { if (!first) eina_strbuf_append_length(buf, ", ", 2); else first = EINA_FALSE; eina_strbuf_append(buf, ecore_file_file_get(path)); } } elm_object_text_set(sd->name_entry, eina_strbuf_string_get(buf)); eina_strbuf_free(buf); } static void _on_dir_up(void *data, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED) { Evas_Object *fs = data; char *parent; ELM_FILESELECTOR_DATA_GET(fs, sd); parent = ecore_file_dir_get(sd->path); _populate(fs, parent, NULL, NULL); free(parent); } static void _home(void *data, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED) { Evas_Object *fs = data; _populate(fs, getenv("HOME"), NULL, NULL); } static void _current_filter_changed(void *data, Evas_Object *obj, void *event_info EINA_UNUSED) { Elm_Fileselector_Filter *filter = data; if (filter->sd->current_filter == filter) return; elm_object_text_set(obj, filter->filter_name); filter->sd->current_filter = filter; _populate(filter->sd->obj, filter->sd->path, NULL, NULL); } static void _ok(void *data, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED) { const char *name; const char *selection = NULL; Evas_Object *fs = data; ELM_FILESELECTOR_DATA_GET(fs, sd); if (!sd->path) { evas_object_smart_callback_call(fs, SIG_DONE, NULL); return; } name = elm_object_text_get(sd->name_entry); if (name && name[0] != '\0') selection = eina_stringshare_printf("%s/%s", sd->path, name); else selection = eina_stringshare_add(elm_fileselector_selected_get(fs)); evas_object_smart_callback_call(fs, SIG_DONE, (void *)selection); eina_stringshare_del(selection); } static void _canc(void *data, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED) { Evas_Object *fs = data; evas_object_smart_callback_call(fs, SIG_DONE, NULL); } static void _on_text_clicked(void *data EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED) { ELM_FILESELECTOR_DATA_GET(data, sd); /* FIXME: When anchor is clicked, current callback is also called. * But when it is "anchor,clicked" entry should be unfocused, so we remove * focus in achor_clicked. * * Check if entry is focused. * It will be so if empty place (not anchor) was clicked. */ if (!elm_object_focus_get(obj)) return; elm_entry_entry_set(obj, sd->path); elm_entry_cursor_pos_set(obj, eina_stringshare_strlen(sd->path)); } static void _on_text_activated(void *data, Evas_Object *obj, void *event_info EINA_UNUSED) { Evas_Object *fs = data; const char *p, *path; char *dir; ELM_FILESELECTOR_DATA_GET(fs, sd); path = elm_object_text_get(obj); if (!ecore_file_exists(path)) { evas_object_smart_callback_call(fs, SIG_SELECTED, (void *)path); evas_object_smart_callback_call(fs, SIG_SELECTED_INVALID, (void *)path); goto end; } if (ecore_file_is_dir(path)) { // keep a ref to path 'couse it will be destroyed by _populate p = eina_stringshare_add(path); _populate(fs, p, NULL, NULL); eina_stringshare_del(p); if (sd->only_folder) evas_object_smart_callback_call(fs, SIG_SELECTED, (void *)path); goto end; } dir = ecore_file_dir_get(path); if (!dir) goto end; if (strcmp(dir, sd->path)) { _populate(fs, dir, NULL, path); if (sd->only_folder) evas_object_smart_callback_call(fs, SIG_SELECTED, (void *)path); } else { if (sd->mode == ELM_FILESELECTOR_LIST) { Elm_Object_Item *item = elm_genlist_first_item_get(sd->files_view); while (item) { const char *item_path = elm_object_item_data_get(item); if (!strcmp(item_path, path)) { elm_genlist_item_selected_set(item, EINA_TRUE); elm_object_text_set(sd->name_entry, ecore_file_file_get(path)); break; } item = elm_genlist_item_next_get(item); } } else { Elm_Object_Item *item = elm_gengrid_first_item_get(sd->files_view); while (item) { const char *item_path = elm_object_item_data_get(item); if (!strcmp(item_path, path)) { elm_gengrid_item_selected_set(item, EINA_TRUE); elm_object_text_set(sd->name_entry, ecore_file_file_get(path)); break; } item = elm_gengrid_item_next_get(item); } } } free(dir); end: elm_object_focus_set(obj, EINA_FALSE); } static void _on_text_unfocused(void *data, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED) { ELM_FILESELECTOR_DATA_GET(data, sd); _anchors_do(data, sd->path); } static void _anchor_clicked(void *data, Evas_Object *obj EINA_UNUSED, void *event_info) { Elm_Entry_Anchor_Info *info = event_info; Evas_Object *fs = data; const char *p; // keep a ref to path 'couse it will be destroyed by _populate p = eina_stringshare_add(info->name); _populate(fs, p, NULL, NULL); eina_stringshare_del(p); /* After anchor was clicked, entry will be focused, and will be editable. * It's wrong. So remove focus. */ elm_object_focus_set(obj, EINA_FALSE); } static Evas_Object * _files_list_add(Evas_Object *obj) { Evas_Object *li; li = elm_genlist_add(obj); evas_object_data_set(li, "parent", obj); elm_widget_mirrored_automatic_set(li, EINA_FALSE); evas_object_size_hint_align_set(li, EVAS_HINT_FILL, EVAS_HINT_FILL); evas_object_size_hint_weight_set(li, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND); evas_object_size_hint_min_set(li, 100, 100); evas_object_smart_callback_add(li, "selected", _on_item_selected, obj); evas_object_smart_callback_add(li, "unselected", _on_item_unselected, obj); evas_object_smart_callback_add(li, "activated", _on_item_activated, obj); evas_object_smart_callback_add (li, "expand,request", _on_list_expand_req, obj); evas_object_smart_callback_add (li, "contract,request", _on_list_contract_req, obj); evas_object_smart_callback_add(li, "expanded", _on_list_expanded, obj); evas_object_smart_callback_add(li, "contracted", _on_list_contracted, obj); elm_widget_sub_object_add(obj, li); return li; } static Evas_Object * _files_grid_add(Evas_Object *obj) { Evas_Object *grid; ELM_FILESELECTOR_DATA_GET(obj, sd); grid = elm_gengrid_add(obj); evas_object_data_set(grid, "parent", obj); elm_widget_mirrored_automatic_set(grid, EINA_FALSE); evas_object_size_hint_align_set(grid, EVAS_HINT_FILL, EVAS_HINT_FILL); evas_object_size_hint_weight_set(grid, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND); elm_gengrid_item_size_set(grid, sd->thumbnail_size.w + GENGRID_PADDING, sd->thumbnail_size.h + GENGRID_PADDING); elm_gengrid_align_set(grid, 0.0, 0.0); evas_object_smart_callback_add(grid, "selected", _on_item_selected, obj); evas_object_smart_callback_add(grid, "unselected", _on_item_unselected, obj); evas_object_smart_callback_add(grid, "activated", _on_item_activated, obj); elm_widget_sub_object_add(obj, grid); return grid; } static Eina_Bool _resource_created(void *data, int type, void *ev) { Evas_Object *obj = data; Eio_Monitor_Event *event = ev; int itcn = ELM_FILE_UNKNOW; Eina_Bool dir = EINA_FALSE; ELM_FILESELECTOR_DATA_GET(obj, sd); if (type == EIO_MONITOR_DIRECTORY_CREATED) dir = EINA_TRUE; Elm_Fileselector_Filter *cf = sd->current_filter; if (cf) { switch (cf->filter_type) { case ELM_FILESELECTOR_MIME_FILTER: if (!dir && !_check_mime_type_filter(cf, event->filename)) return ECORE_CALLBACK_PASS_ON; break; case ELM_FILESELECTOR_CUSTOM_FILTER: if (!cf->filter.custom->func(event->filename, dir, cf->filter.custom->data)) return ECORE_CALLBACK_PASS_ON; break; default: break; } } if (dir) itcn = ELM_DIRECTORY; else { if (evas_object_image_extension_can_load_get(event->filename)) itcn = ELM_FILE_IMAGE; } if (sd->mode == ELM_FILESELECTOR_LIST) elm_genlist_item_sorted_insert(sd->files_view, list_itc[itcn], eina_stringshare_add(event->filename), NULL, (sd->expand && itcn == ELM_DIRECTORY) ? ELM_GENLIST_ITEM_TREE : ELM_GENLIST_ITEM_NONE, _file_list_cmp, NULL, NULL); else elm_gengrid_item_sorted_insert(sd->files_view, grid_itc[itcn], eina_stringshare_add(event->filename), _file_grid_cmp, NULL, NULL); return ECORE_CALLBACK_PASS_ON; } static Eina_Bool _resource_deleted(void *data, int type EINA_UNUSED, void *ev) { Evas_Object *obj = data; Eio_Monitor_Event *event = ev; Elm_Object_Item *it = NULL; Eina_Bool selected = EINA_FALSE; ELM_FILESELECTOR_DATA_GET(obj, sd); if (sd->mode == ELM_FILESELECTOR_LIST) { it = elm_genlist_first_item_get(sd->files_view); while(it) { if (!strcmp(elm_object_item_data_get(it), event->filename)) { selected = elm_genlist_item_selected_get(it); break; } it = elm_genlist_item_next_get(it); } } else { it = elm_gengrid_first_item_get(sd->files_view); while(it) { if (!strcmp(elm_object_item_data_get(it), event->filename)) { selected = elm_genlist_item_selected_get(it); break; } it = elm_gengrid_item_next_get(it); } } if (it) elm_object_item_del(it); if (selected) { if (sd->multi) { Eina_List *li, *l; char *path; Eina_Strbuf *buf; Eina_Bool first = EINA_TRUE; buf = eina_strbuf_new(); EINA_LIST_FOREACH_SAFE(sd->paths, li, l, path) { if (!strcmp(path, event->filename)) { sd->paths = eina_list_remove_list(sd->paths, li); free(path); } else { if (!first) eina_strbuf_append_length(buf, ", ", 2); else first = EINA_FALSE; eina_strbuf_append(buf, ecore_file_file_get(path)); } } elm_object_text_set(sd->name_entry, eina_strbuf_string_get(buf)); eina_strbuf_free(buf); } else elm_object_text_set(sd->name_entry, ""); } return ECORE_CALLBACK_PASS_ON; } static void _name_entry_activated_cb(void *data, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED) { _ok(data, NULL, NULL); } static void _elm_fileselector_smart_add(Eo *obj, void *_pd, va_list *list EINA_UNUSED) { Evas_Object *ic, *bt, *en, *pb; Elm_Fileselector_Smart_Data *priv = _pd; EINA_REFCOUNT_INIT(priv); eo_do_super(obj, MY_CLASS, evas_obj_smart_add()); elm_widget_sub_object_parent_add(obj); elm_widget_can_focus_set(obj, EINA_FALSE); priv->expand = !!_elm_config->fileselector_expand_enable; priv->double_tap_navigation = !!_elm_config->fileselector_double_tap_navigation_enable; if (!elm_layout_theme_set (obj, "fileselector", "base", elm_widget_style_get(obj))) CRI("Failed to set layout!"); // up btn ic = elm_icon_add(obj); elm_icon_standard_set(ic, "arrow_up"); evas_object_size_hint_aspect_set(ic, EVAS_ASPECT_CONTROL_VERTICAL, 1, 1); bt = elm_button_add(obj); elm_widget_mirrored_automatic_set(bt, EINA_FALSE); elm_object_part_content_set(bt, "icon", ic); elm_object_domain_translatable_text_set(bt, PACKAGE, N_("Up")); evas_object_size_hint_align_set(bt, 0.0, 0.0); evas_object_smart_callback_add(bt, "clicked", _on_dir_up, obj); elm_widget_sub_object_add(obj, bt); priv->up_button = bt; // home btn ic = elm_icon_add(obj); elm_icon_standard_set(ic, "home"); evas_object_size_hint_aspect_set(ic, EVAS_ASPECT_CONTROL_VERTICAL, 1, 1); bt = elm_button_add(obj); elm_widget_mirrored_automatic_set(bt, EINA_FALSE); elm_object_part_content_set(bt, "icon", ic); elm_object_domain_translatable_text_set(bt, PACKAGE, N_("Home")); evas_object_size_hint_align_set(bt, 0.0, 0.0); evas_object_smart_callback_add(bt, "clicked", _home, obj); elm_widget_sub_object_add(obj, bt); priv->home_button = bt; // spinner pb = elm_progressbar_add(obj); elm_widget_sub_object_add(obj, pb); priv->spinner = pb; // XXX: will fail for dynamic finger size changing priv->thumbnail_size.w = elm_config_finger_size_get() * 2 - GENGRID_PADDING; priv->thumbnail_size.h = priv->thumbnail_size.w; priv->sort_type = ELM_FILESELECTOR_SORT_BY_FILENAME_ASC; priv->sort_method = strcoll; // files_view priv->files_view = _files_list_add(obj); // path entry en = elm_entry_add(obj); elm_entry_scrollable_set(en, EINA_TRUE); elm_widget_mirrored_automatic_set(en, EINA_FALSE); elm_entry_single_line_set(en, EINA_TRUE); elm_entry_line_wrap_set(en, ELM_WRAP_CHAR); evas_object_size_hint_weight_set(en, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND); evas_object_size_hint_align_set(en, EVAS_HINT_FILL, EVAS_HINT_FILL); evas_object_smart_callback_add(en, "anchor,clicked", _anchor_clicked, obj); evas_object_smart_callback_add(en, "clicked", _on_text_clicked, obj); evas_object_smart_callback_add(en, "unfocused", _on_text_unfocused, obj); evas_object_smart_callback_add(en, "activated", _on_text_activated, obj); elm_widget_sub_object_add(obj, en); priv->path_entry = en; // name entry en = elm_entry_add(obj); elm_entry_scrollable_set(en, EINA_TRUE); elm_widget_mirrored_automatic_set(en, EINA_FALSE); elm_entry_editable_set(en, EINA_TRUE); elm_entry_single_line_set(en, EINA_TRUE); elm_entry_line_wrap_set(en, ELM_WRAP_CHAR); evas_object_size_hint_weight_set(en, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND); evas_object_size_hint_align_set(en, EVAS_HINT_FILL, EVAS_HINT_FILL); evas_object_smart_callback_add(en, "activated", _name_entry_activated_cb, obj); elm_widget_sub_object_add(obj, en); priv->name_entry = en; elm_fileselector_buttons_ok_cancel_set(obj, EINA_TRUE); elm_fileselector_is_save_set(obj, EINA_FALSE); #define HANDLER_ADD(e, fn) \ priv->handlers = eina_list_append(priv->handlers, \ ecore_event_handler_add(e, fn, obj)); HANDLER_ADD(EIO_MONITOR_FILE_CREATED, _resource_created); HANDLER_ADD(EIO_MONITOR_DIRECTORY_CREATED, _resource_created); HANDLER_ADD(EIO_MONITOR_FILE_DELETED, _resource_deleted); HANDLER_ADD(EIO_MONITOR_DIRECTORY_DELETED, _resource_deleted); #undef HANDLER_ADD eo_do(obj, elm_wdg_theme_apply(NULL)); } static void _elm_fileselector_smart_del(Eo *obj EINA_UNUSED, void *_pd, va_list *list EINA_UNUSED) { Elm_Fileselector_Smart_Data *sd = _pd; Elm_Fileselector_Filter *filter; char *path; Ecore_Event_Handler *h; if (sd->monitor) eio_monitor_del(sd->monitor); if (sd->current) eio_file_cancel(sd->current); EINA_LIST_FREE(sd->handlers, h) { ecore_event_handler_del(h); } EINA_LIST_FREE(sd->filter_list, filter) { eina_stringshare_del(filter->filter_name); if (filter->filter_type == ELM_FILESELECTOR_MIME_FILTER) { free(filter->filter.mime_types[0]); free(filter->filter.mime_types); } else free(filter->filter.custom); free(filter); } EINA_LIST_FREE(sd->paths, path) free(path); sd->files_view = NULL; /* this one matching EINA_REFCOUNT_INIT() */ EINA_REFCOUNT_UNREF(sd) _elm_fileselector_smart_del_do(sd); } EAPI Evas_Object * elm_fileselector_add(Evas_Object *parent) { EINA_SAFETY_ON_NULL_RETURN_VAL(parent, NULL); Evas_Object *obj = eo_add(MY_CLASS, parent); eo_unref(obj); return obj; } static void _constructor(Eo *obj, void *_pd EINA_UNUSED, va_list *list EINA_UNUSED) { Elm_Fileselector_Smart_Data *sd = _pd; sd->obj = obj; eo_do_super(obj, MY_CLASS, eo_constructor()); eo_do(obj, evas_obj_type_set(MY_CLASS_NAME_LEGACY), evas_obj_smart_callbacks_descriptions_set(_smart_callbacks, NULL)); } EAPI void elm_fileselector_is_save_set(Evas_Object *obj, Eina_Bool is_save) { ELM_FILESELECTOR_INTERFACE_CHECK(obj); eo_do(obj, elm_fileselector_interface_is_save_set(is_save)); } static void _is_save_set(Eo *obj, void *_pd, va_list *list) { Eina_Bool is_save = va_arg(*list, int); Elm_Fileselector_Smart_Data *sd = _pd; elm_object_disabled_set(sd->name_entry, !is_save); if (is_save) elm_layout_signal_emit(obj, "elm,state,save,on", "elm"); else elm_layout_signal_emit(obj, "elm,state,save,off", "elm"); } EAPI Eina_Bool elm_fileselector_is_save_get(const Evas_Object *obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, EINA_FALSE); Eina_Bool ret = EINA_FALSE; eo_do((Eo *) obj, elm_fileselector_interface_is_save_get(&ret)); return ret; } static void _is_save_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); Elm_Fileselector_Smart_Data *sd = _pd; if (ret) *ret = !elm_object_disabled_get(sd->name_entry); } EAPI void elm_fileselector_folder_only_set(Evas_Object *obj, Eina_Bool only) { ELM_FILESELECTOR_INTERFACE_CHECK(obj); eo_do(obj, elm_fileselector_interface_folder_only_set(only)); } static void _folder_only_set(Eo *obj, void *_pd, va_list *list) { Eina_Bool only = va_arg(*list, int); Elm_Fileselector_Smart_Data *sd = _pd; if (sd->only_folder == only) return; sd->only_folder = !!only; if (sd->path) _populate(obj, sd->path, NULL, NULL); } EAPI Eina_Bool elm_fileselector_folder_only_get(const Evas_Object *obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, EINA_FALSE); Eina_Bool ret = EINA_FALSE; eo_do((Eo *) obj, elm_fileselector_interface_folder_only_get(&ret)); return ret; } static void _folder_only_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); Elm_Fileselector_Smart_Data *sd = _pd; if (ret) *ret = sd->only_folder; } EAPI void elm_fileselector_buttons_ok_cancel_set(Evas_Object *obj, Eina_Bool visible) { ELM_FILESELECTOR_CHECK(obj); eo_do(obj, elm_obj_fileselector_buttons_ok_cancel_set(visible)); } static void _buttons_ok_cancel_set(Eo *obj, void *_pd, va_list *list) { Eina_Bool visible = va_arg(*list, int); Evas_Object *bt; Elm_Fileselector_Smart_Data *sd = _pd; if (visible && (!sd->ok_button)) { // cancel btn bt = elm_button_add(obj); elm_widget_mirrored_automatic_set(bt, EINA_FALSE); elm_object_domain_translatable_text_set(bt, PACKAGE, N_("Cancel")); evas_object_smart_callback_add(bt, "clicked", _canc, obj); sd->cancel_button = bt; // ok btn bt = elm_button_add(obj); elm_widget_mirrored_automatic_set(bt, EINA_FALSE); elm_object_domain_translatable_text_set(bt, PACKAGE, N_("OK")); evas_object_smart_callback_add(bt, "clicked", _ok, obj); sd->ok_button = bt; eo_do(obj, elm_wdg_theme_apply(NULL)); } else if (!visible) { evas_object_del(sd->cancel_button); sd->cancel_button = NULL; evas_object_del(sd->ok_button); sd->ok_button = NULL; } } EAPI Eina_Bool elm_fileselector_buttons_ok_cancel_get(const Evas_Object *obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, EINA_FALSE); Eina_Bool ret = EINA_FALSE; eo_do((Eo *) obj, elm_obj_fileselector_buttons_ok_cancel_get(&ret)); return ret; } static void _buttons_ok_cancel_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); Elm_Fileselector_Smart_Data *sd = _pd; if (ret) *ret = sd->ok_button ? EINA_TRUE : EINA_FALSE; } EAPI void elm_fileselector_expandable_set(Evas_Object *obj, Eina_Bool expand) { ELM_FILESELECTOR_INTERFACE_CHECK(obj); eo_do(obj, elm_fileselector_interface_expandable_set(expand)); } static void _expandable_set(Eo *obj, void *_pd, va_list *list) { Eina_Bool expand = va_arg(*list, int); Elm_Fileselector_Smart_Data *sd = _pd; sd->expand = !!expand; if (sd->path) _populate(obj, sd->path, NULL, NULL); } EAPI Eina_Bool elm_fileselector_expandable_get(const Evas_Object *obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, EINA_FALSE); Eina_Bool ret = EINA_FALSE; eo_do((Eo *) obj, elm_fileselector_interface_expandable_get(&ret)); return ret; } static void _expandable_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); Elm_Fileselector_Smart_Data *sd = _pd; if (ret) *ret = sd->expand; } EAPI void elm_fileselector_path_set(Evas_Object *obj, const char *_path) { ELM_FILESELECTOR_INTERFACE_CHECK(obj); eo_do(obj, elm_fileselector_interface_path_set(_path)); } static void _path_set(Eo *obj, void *_pd EINA_UNUSED, va_list *list) { const char *_path = va_arg(*list, const char *); char *path; path = ecore_file_realpath(_path); _populate(obj, path, NULL, NULL); free(path); } EAPI const char * elm_fileselector_path_get(const Evas_Object *obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, NULL); const char *ret = NULL; eo_do((Eo *) obj, elm_fileselector_interface_path_get(&ret)); return ret; } static void _path_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { const char **ret = va_arg(*list, const char **); Elm_Fileselector_Smart_Data *sd = _pd; if (ret) *ret = sd->path; } EAPI void elm_fileselector_mode_set(Evas_Object *obj, Elm_Fileselector_Mode mode) { ELM_FILESELECTOR_INTERFACE_CHECK(obj); eo_do(obj, elm_fileselector_interface_mode_set(mode)); } static void _mode_set(Eo *obj, void *_pd, va_list *list) { Evas_Object *old; Elm_Fileselector_Mode mode = va_arg(*list, Elm_Fileselector_Mode); Elm_Fileselector_Smart_Data *sd = _pd; if (mode == sd->mode) return; old = elm_layout_content_unset(obj, "elm.swallow.files"); if (mode == ELM_FILESELECTOR_LIST) { sd->files_view = _files_list_add(obj); if (sd->multi) elm_genlist_multi_select_set(sd->files_view, EINA_TRUE); } else { sd->files_view = _files_grid_add(obj); if (sd->multi) elm_gengrid_multi_select_set(sd->files_view, EINA_TRUE); } elm_layout_content_set(obj, "elm.swallow.files", sd->files_view); evas_object_del(old); sd->mode = mode; _populate(obj, sd->path, NULL, NULL); } EAPI Elm_Fileselector_Mode elm_fileselector_mode_get(const Evas_Object *obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, ELM_FILESELECTOR_LAST); Elm_Fileselector_Mode ret = ELM_FILESELECTOR_LAST; eo_do((Eo *) obj, elm_fileselector_interface_mode_get(&ret)); return ret; } static void _mode_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Elm_Fileselector_Mode *ret = va_arg(*list, Elm_Fileselector_Mode *); Elm_Fileselector_Smart_Data *sd = _pd; if (ret) *ret = sd->mode; } EAPI void elm_fileselector_multi_select_set(Evas_Object *obj, Eina_Bool multi) { ELM_FILESELECTOR_INTERFACE_CHECK(obj); eo_do(obj, elm_fileselector_interface_multi_select_set(multi)); } static void _multi_select_set(Eo *obj EINA_UNUSED, void *_pd, va_list *list EINA_UNUSED) { Eina_Bool multi = va_arg(*list, int); Elm_Fileselector_Smart_Data *sd = _pd; char *path; multi = !!multi; if (sd->multi == multi) return; sd->multi = multi; if (sd->mode == ELM_FILESELECTOR_LIST) elm_genlist_multi_select_set(sd->files_view, multi); else elm_gengrid_multi_select_set(sd->files_view, multi); if (!sd->multi) { _clear_selections(sd, NULL); EINA_LIST_FREE(sd->paths, path) free(path); } else { const Eina_List *selected_items, *li; const Elm_Object_Item *it; if (sd->mode == ELM_FILESELECTOR_LIST) selected_items = elm_genlist_selected_items_get(sd->files_view); else selected_items = elm_gengrid_selected_items_get(sd->files_view); EINA_LIST_FOREACH(selected_items, li, it) { path = elm_object_item_data_get(it); sd->paths = eina_list_append(sd->paths, strdup(path)); } } } EAPI Eina_Bool elm_fileselector_multi_select_get(const Evas_Object *obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, EINA_FALSE); Eina_Bool ret = EINA_FALSE; eo_do((Eo *) obj, elm_fileselector_interface_multi_select_get(&ret)); return ret; } static void _multi_select_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); Elm_Fileselector_Smart_Data *sd = _pd; if (ret) *ret = sd->multi; } EAPI const char * elm_fileselector_selected_get(const Evas_Object *obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, NULL); const char *ret = NULL; eo_do((Eo *) obj, elm_fileselector_interface_selected_get(&ret)); return ret; } static void _selected_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { const char *fp; const char **ret = va_arg(*list, const char **); Elm_Fileselector_Smart_Data *sd = _pd; if (!ret) return; *ret = NULL; if (!sd->path) return; fp = elm_object_text_get(sd->path_entry); if (ecore_file_exists(fp)) { eina_stringshare_replace(&sd->selection, fp); *ret = sd->selection; return; } *ret = sd->path; if (sd->mode == ELM_FILESELECTOR_LIST) { Elm_Object_Item *gl_it = elm_genlist_selected_item_get(sd->files_view); if (gl_it) *ret = elm_object_item_data_get(gl_it); } else { Elm_Object_Item *gg_it = elm_gengrid_selected_item_get(sd->files_view); if (gg_it) *ret = elm_object_item_data_get(gg_it); } } EAPI Eina_Bool elm_fileselector_selected_set(Evas_Object *obj, const char *_path) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, EINA_FALSE); Eina_Bool ret = EINA_FALSE; eo_do(obj, elm_fileselector_interface_selected_set(_path, &ret)); return ret; } static void _selected_set(Eo *obj, void *_pd, va_list *list) { const char *_path = va_arg(*list, const char *); Eina_Bool *ret = va_arg(*list, Eina_Bool *); char *selected; if (ret) *ret = EINA_TRUE; char *path; Elm_Fileselector_Smart_Data *sd = _pd; path = ecore_file_realpath(_path); if (ecore_file_is_dir(path)) _populate(obj, path, NULL, NULL); else { if (!ecore_file_exists(path)) { if (ret) *ret = EINA_FALSE; goto clean_up; } selected = ecore_file_dir_get(path); _populate(obj, selected, NULL, NULL); eina_stringshare_replace(&sd->selection, path); free(selected); } clean_up: free(path); } EAPI const Eina_List * elm_fileselector_selected_paths_get(const Evas_Object* obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, NULL); const Eina_List *ret = NULL; eo_do((Eo *) obj, elm_fileselector_interface_selected_paths_get(&ret)); return ret; } static void _selected_paths_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { const Eina_List **ret = va_arg(*list, const Eina_List**); Elm_Fileselector_Smart_Data *sd = _pd; if (!ret) return; if (sd->multi) *ret = sd->paths; else *ret = NULL; } static Elm_Fileselector_Filter * _filter_add(Elm_Fileselector_Smart_Data *sd, const char *filter_name) { Elm_Fileselector_Filter *ff; ff = malloc(sizeof(Elm_Fileselector_Filter)); ff->filter_name = eina_stringshare_add(filter_name); ff->sd = sd; return ff; } EAPI Eina_Bool elm_fileselector_mime_types_filter_append(Evas_Object *obj, const char *mime_type, const char *filter_name) { ELM_FILESELECTOR_CHECK(obj) EINA_FALSE; Eina_Bool ret = EINA_FALSE; eo_do(obj, elm_fileselector_interface_mime_types_filter_append(mime_type, filter_name, &ret)); return ret; } static void _mime_types_filter_append(Eo *obj, void *_pd, va_list *list) { const char *mime_types = va_arg(*list, const char *); const char *filter_name = va_arg(*list, const char *); Eina_Bool *ret = va_arg(*list, Eina_Bool *); Elm_Fileselector_Smart_Data *sd; Elm_Fileselector_Filter *ff; Eina_Bool int_ret = EINA_FALSE; Eina_Bool need_theme = EINA_FALSE; if (!mime_types) goto end; sd = _pd; ff = _filter_add(sd, filter_name ? filter_name : mime_types); ff->filter_type = ELM_FILESELECTOR_MIME_FILTER; ff->filter.mime_types = eina_str_split(mime_types, ",", 0); if (!sd->filter_list) { sd->current_filter = ff; sd->filter_hoversel = elm_hoversel_add(obj); elm_object_text_set(sd->filter_hoversel, ff->filter_name); need_theme = EINA_TRUE; } elm_hoversel_item_add(sd->filter_hoversel, ff->filter_name, NULL, ELM_ICON_NONE, _current_filter_changed, ff); sd->filter_list = eina_list_append(sd->filter_list, ff); _populate(obj, sd->path, NULL, NULL); if (need_theme) eo_do(obj, elm_wdg_theme_apply(NULL)); int_ret = EINA_TRUE; end: if (ret) *ret = int_ret; } EAPI Eina_Bool elm_fileselector_custom_filter_append(Evas_Object *obj, Elm_Fileselector_Filter_Func func, void *data, const char *filter_name) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, EINA_FALSE); Eina_Bool ret = EINA_FALSE; eo_do(obj, elm_fileselector_interface_custom_filter_append(func, data, filter_name, &ret)); return ret; } static void _custom_filter_append(Eo *obj, void *_pd, va_list *list) { Elm_Fileselector_Filter_Func func = va_arg(*list, Elm_Fileselector_Filter_Func); void *data = va_arg(*list, void *); const char *filter_name = va_arg(*list, const char *); Eina_Bool *ret = va_arg(*list, Eina_Bool *); Elm_Fileselector_Smart_Data *sd; Elm_Fileselector_Filter *ff; Eina_Bool int_ret = EINA_FALSE; Eina_Bool need_theme = EINA_FALSE; if (!func) goto end; sd = _pd; ff = _filter_add(sd, filter_name ? filter_name : "custom"); ff->filter_type = ELM_FILESELECTOR_CUSTOM_FILTER; ff->filter.custom = malloc(sizeof(Elm_Fileselector_Custom_Filter)); ff->filter.custom->func = func; ff->filter.custom->data = data; if (!sd->filter_list) { sd->current_filter = ff; sd->filter_hoversel = elm_hoversel_add(obj); elm_object_text_set(sd->filter_hoversel, ff->filter_name); need_theme = EINA_TRUE; } elm_hoversel_item_add(sd->filter_hoversel, ff->filter_name, NULL, ELM_ICON_NONE, _current_filter_changed, ff); sd->filter_list = eina_list_append(sd->filter_list, ff); _populate(obj, sd->path, NULL, NULL); if (need_theme) eo_do(obj, elm_wdg_theme_apply(NULL)); int_ret = EINA_TRUE; end: if (ret) *ret = int_ret; } EAPI void elm_fileselector_filters_clear(Evas_Object *obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj); eo_do(obj, elm_fileselector_interface_filters_clear()); } static void _filters_clear(Eo *obj, void *_pd, va_list *list EINA_UNUSED) { Elm_Fileselector_Smart_Data *sd = _pd; Elm_Fileselector_Filter *filter; EINA_LIST_FREE(sd->filter_list, filter) { eina_stringshare_del(filter->filter_name); if (filter->filter_type == ELM_FILESELECTOR_MIME_FILTER) { free(filter->filter.mime_types[0]); free(filter->filter.mime_types); } else free(filter->filter.custom); free(filter); } ELM_SAFE_FREE(sd->filter_hoversel, evas_object_del); _populate(obj, sd->path, NULL, NULL); } EAPI void elm_fileselector_hidden_visible_set(Evas_Object *obj, Eina_Bool visible) { ELM_FILESELECTOR_INTERFACE_CHECK(obj); eo_do(obj, elm_fileselector_interface_hidden_visible_set(visible)); } static void _hidden_visible_set(Eo *obj EINA_UNUSED, void *_pd, va_list *list EINA_UNUSED) { Eina_Bool visible = va_arg(*list, int); Elm_Fileselector_Smart_Data *sd = _pd; visible = !!visible; if (sd->hidden_visible == visible) return; sd->hidden_visible = visible; _clear_selections(sd, NULL); _populate(obj, sd->path, NULL, NULL); } EAPI Eina_Bool elm_fileselector_hidden_visible_get(const Evas_Object *obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, EINA_FALSE); Eina_Bool ret = EINA_FALSE; eo_do((Eo *) obj, elm_fileselector_interface_hidden_visible_get(&ret)); return ret; } static void _hidden_visible_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list EINA_UNUSED) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); Elm_Fileselector_Smart_Data *sd = _pd; if (ret) *ret = sd->hidden_visible; } EAPI void elm_fileselector_thumbnail_size_set(Evas_Object *obj, Evas_Coord w, Evas_Coord h) { ELM_FILESELECTOR_INTERFACE_CHECK(obj); eo_do(obj, elm_fileselector_interface_thumbnail_size_set(w, h)); } static void _thumbnail_size_set(Eo *obj EINA_UNUSED, void *_pd, va_list *list EINA_UNUSED) { Evas_Coord w = va_arg(*list, Evas_Coord); Evas_Coord h = va_arg(*list, Evas_Coord); Elm_Fileselector_Smart_Data *sd = _pd; if (sd->thumbnail_size.w == w && sd->thumbnail_size.h == h) return; if (!w || !h) w = h = elm_config_finger_size_get() * 2 - GENGRID_PADDING; sd->thumbnail_size.w = w; sd->thumbnail_size.h = h; if (sd->mode == ELM_FILESELECTOR_GRID) elm_gengrid_item_size_set(sd->files_view, w + GENGRID_PADDING, h + GENGRID_PADDING); _populate(obj, sd->path, NULL, NULL); } EAPI void elm_fileselector_thumbnail_size_get(const Evas_Object *obj, Evas_Coord *w, Evas_Coord *h) { ELM_FILESELECTOR_INTERFACE_CHECK(obj); eo_do((Eo *) obj, elm_fileselector_interface_thumbnail_size_get(w, h)); } static void _thumbnail_size_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list EINA_UNUSED) { Evas_Coord *w = va_arg(*list, Evas_Coord *); Evas_Coord *h = va_arg(*list, Evas_Coord *); Elm_Fileselector_Smart_Data *sd = _pd; if (w) *w = sd->thumbnail_size.w; if (h) *h = sd->thumbnail_size.h; } EAPI void elm_fileselector_sort_method_set(Evas_Object *obj, Elm_Fileselector_Sort sort) { ELM_FILESELECTOR_INTERFACE_CHECK(obj); eo_do(obj, elm_fileselector_interface_sort_method_set(sort)); } static void _sort_method_set(Eo *obj EINA_UNUSED, void *_pd, va_list *list EINA_UNUSED) { Elm_Fileselector_Sort sort = va_arg(*list, Elm_Fileselector_Sort); Elm_Fileselector_Smart_Data *sd = _pd; if (sd->sort_type == sort) return; sd->sort_type = sort; switch (sd->sort_type) { case ELM_FILESELECTOR_SORT_BY_FILENAME_ASC: sd->sort_method = strcoll; break; case ELM_FILESELECTOR_SORT_BY_FILENAME_DESC: sd->sort_method = _strcoll_rev; break; case ELM_FILESELECTOR_SORT_BY_TYPE_ASC: sd->sort_method = _strcoll_type; break; case ELM_FILESELECTOR_SORT_BY_TYPE_DESC: sd->sort_method = _strcoll_type_rev; break; case ELM_FILESELECTOR_SORT_BY_SIZE_ASC: sd->sort_method = _size_cmp; break; case ELM_FILESELECTOR_SORT_BY_SIZE_DESC: sd->sort_method = _size_cmp_rev; break; case ELM_FILESELECTOR_SORT_BY_MODIFIED_ASC: sd->sort_method = _modified_cmp; break; case ELM_FILESELECTOR_SORT_BY_MODIFIED_DESC: sd->sort_method = _modified_cmp_rev; break; case ELM_FILESELECTOR_SORT_LAST: default: sd->sort_method = strcoll; } _populate(obj, sd->path, NULL, NULL); } EAPI Elm_Fileselector_Sort elm_fileselector_sort_method_get(const Evas_Object *obj) { ELM_FILESELECTOR_INTERFACE_CHECK(obj, ELM_FILESELECTOR_SORT_LAST); Elm_Fileselector_Sort ret = ELM_FILESELECTOR_SORT_LAST; eo_do((Eo *) obj, elm_fileselector_interface_sort_method_get(&ret)); return ret; } static void _sort_method_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list EINA_UNUSED) { Elm_Fileselector_Sort *ret = va_arg(*list, Elm_Fileselector_Sort *); Elm_Fileselector_Smart_Data *sd = _pd; if (ret) *ret = sd->sort_type; } static void _elm_fileselector_smart_focus_next_manager_is(Eo *obj EINA_UNUSED, void *_pd EINA_UNUSED, va_list *list) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); if (ret) *ret = EINA_FALSE; } static void _elm_fileselector_smart_focus_direction_manager_is(Eo *obj EINA_UNUSED, void *_pd EINA_UNUSED, va_list *list) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); if (ret) *ret = EINA_FALSE; } static void _elm_fileselector_smart_text_set(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Elm_Fileselector_Smart_Data *sd = _pd; const char *part = va_arg(*list, const char *); const char *label = va_arg(*list, const char *); Eina_Bool *ret = va_arg(*list, Eina_Bool *); if (ret) *ret = EINA_FALSE; Eina_Bool int_ret = EINA_FALSE; if (!part) return; if (sd->ok_button && !strcmp(part, "ok")) { elm_object_text_set(sd->ok_button, label); int_ret = EINA_TRUE; } else if (sd->cancel_button && !strcmp(part, "cancel")) { elm_object_text_set(sd->cancel_button, label); int_ret = EINA_TRUE; } else eo_do_super(obj, MY_CLASS, elm_obj_layout_text_set(part, label, &int_ret)); if (ret) *ret = int_ret; } static void _class_constructor(Eo_Class *klass) { unsigned int i; const Eo_Op_Func_Description func_desc[] = { EO_OP_FUNC(EO_BASE_ID(EO_BASE_SUB_ID_CONSTRUCTOR), _constructor), EO_OP_FUNC(EVAS_OBJ_SMART_ID(EVAS_OBJ_SMART_SUB_ID_ADD), _elm_fileselector_smart_add), EO_OP_FUNC(EVAS_OBJ_SMART_ID(EVAS_OBJ_SMART_SUB_ID_DEL), _elm_fileselector_smart_del), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_THEME_APPLY), _elm_fileselector_smart_theme), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_FOCUS_NEXT_MANAGER_IS), _elm_fileselector_smart_focus_next_manager_is), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_FOCUS_DIRECTION_MANAGER_IS), _elm_fileselector_smart_focus_direction_manager_is), EO_OP_FUNC(ELM_OBJ_LAYOUT_ID(ELM_OBJ_LAYOUT_SUB_ID_SIZING_EVAL), _elm_fileselector_smart_sizing_eval), EO_OP_FUNC(ELM_OBJ_LAYOUT_ID(ELM_OBJ_LAYOUT_SUB_ID_TEXT_SET), _elm_fileselector_smart_text_set), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_FOLDER_ONLY_SET), _folder_only_set), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_FOLDER_ONLY_GET), _folder_only_get), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_IS_SAVE_SET), _is_save_set), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_IS_SAVE_GET), _is_save_get), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_EXPANDABLE_SET), _expandable_set), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_EXPANDABLE_GET), _expandable_get), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_PATH_SET), _path_set), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_PATH_GET), _path_get), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_MODE_SET), _mode_set), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_MODE_GET), _mode_get), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_MULTI_SELECT_SET), _multi_select_set), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_MULTI_SELECT_GET), _multi_select_get), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_SELECTED_GET), _selected_get), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_SELECTED_SET), _selected_set), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_SELECTED_PATHS_GET), _selected_paths_get), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_MIME_TYPES_FILTER_APPEND), _mime_types_filter_append), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_CUSTOM_FILTER_APPEND), _custom_filter_append), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_FILTERS_CLEAR), _filters_clear), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_HIDDEN_VISIBLE_SET), _hidden_visible_set), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_HIDDEN_VISIBLE_GET), _hidden_visible_get), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_THUMBNAIL_SIZE_SET), _thumbnail_size_set), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_THUMBNAIL_SIZE_GET), _thumbnail_size_get), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_SORT_METHOD_SET), _sort_method_set), EO_OP_FUNC(ELM_FILESELECTOR_INTERFACE_ID(ELM_FILESELECTOR_INTERFACE_SUB_ID_SORT_METHOD_GET), _sort_method_get), EO_OP_FUNC(ELM_OBJ_FILESELECTOR_ID(ELM_OBJ_FILESELECTOR_SUB_ID_BUTTONS_OK_CANCEL_SET), _buttons_ok_cancel_set), EO_OP_FUNC(ELM_OBJ_FILESELECTOR_ID(ELM_OBJ_FILESELECTOR_SUB_ID_BUTTONS_OK_CANCEL_GET), _buttons_ok_cancel_get), EO_OP_FUNC_SENTINEL }; eo_class_funcs_set(klass, func_desc); evas_smart_legacy_type_register(MY_CLASS_NAME_LEGACY, klass); for (i = 0; i < ELM_FILE_LAST; ++i) { list_itc[i] = elm_genlist_item_class_new(); grid_itc[i] = elm_gengrid_item_class_new(); list_itc[i]->item_style = "default"; list_itc[i]->func.text_get = grid_itc[i]->func.text_get = _itc_text_get; list_itc[i]->func.state_get = grid_itc[i]->func.state_get = _itc_state_get; list_itc[i]->func.del = grid_itc[i]->func.del = _itc_del; } list_itc[ELM_DIRECTORY]->func.content_get = grid_itc[ELM_DIRECTORY]->func.content_get = _itc_icon_folder_get; list_itc[ELM_FILE_IMAGE]->func.content_get = grid_itc[ELM_FILE_IMAGE]->func.content_get = _itc_icon_image_get; list_itc[ELM_FILE_UNKNOW]->func.content_get = grid_itc[ELM_FILE_UNKNOW]->func.content_get = _itc_icon_file_get; } static void _class_destructor(Eo_Class *klass EINA_UNUSED) { unsigned int i; for (i = 0; i < ELM_FILE_LAST; ++i) { elm_genlist_item_class_free(list_itc[i]); elm_gengrid_item_class_free(grid_itc[i]); } } static const Eo_Op_Description op_desc[] = { EO_OP_DESCRIPTION(ELM_OBJ_FILESELECTOR_SUB_ID_BUTTONS_OK_CANCEL_SET, "Enable/disable the 'ok' and 'cancel' buttons on a given file selector widget."), EO_OP_DESCRIPTION(ELM_OBJ_FILESELECTOR_SUB_ID_BUTTONS_OK_CANCEL_GET, "Get whether the 'ok' and 'cancel' buttons on a given file selector widget are being shown."), EO_OP_DESCRIPTION_SENTINEL }; static const Eo_Class_Description class_desc = { EO_VERSION, MY_CLASS_NAME, EO_CLASS_TYPE_REGULAR, EO_CLASS_DESCRIPTION_OPS(&ELM_OBJ_FILESELECTOR_BASE_ID, op_desc, ELM_OBJ_FILESELECTOR_SUB_ID_LAST), NULL, sizeof(Elm_Fileselector_Smart_Data), _class_constructor, _class_destructor }; EO_DEFINE_CLASS(elm_obj_fileselector_class_get, &class_desc, ELM_OBJ_LAYOUT_CLASS, ELM_FILESELECTOR_INTERFACE, NULL);