/* * TODO * userdefined icon/label cb * show/hide/add buttons ??? * need a background ??? * save mode (you can type a name somewhere) * show/Hide hidden files * double click to choose a file * multiselection * write docs */ #include #include #include "elm_priv.h" typedef struct _Widget_Data Widget_Data; struct _Widget_Data { Evas_Object *vbox, *entry, *list; const char *path; Eina_Bool expand; }; Elm_Genlist_Item_Class itc; static void _populate(Evas_Object *obj, const char *path, Elm_Genlist_Item *parent); /*** ELEMENTARY WIDGET ***/ static void _del_hook(Evas_Object *obj) { Widget_Data *wd = elm_widget_data_get(obj); if (wd->path) eina_stringshare_del(wd->path); free(wd); } static void _sizing_eval(Evas_Object *obj) { Widget_Data *wd = elm_widget_data_get(obj); Evas_Coord minw = -1, minh = -1; evas_object_size_hint_min_get(wd->vbox, &minw, &minh); evas_object_size_hint_min_set(obj, minw, minh); // printf("***** SIZING EVAL [min %d %d] *************\n", minw, minh); } /*** GENLIST "MODEL" ***/ static char* _itc_label_get(const void *data, Evas_Object *obj, const char *source) { //~ printf("LABEL_GET: %s\n", (char*) data); return strdup(ecore_file_file_get(data)); // NOTE this will be free() by the caller } static Evas_Object* _itc_icon_get(const void *data, Evas_Object *obj, const char *source) { Evas_Object *ic; //~ printf("ICON GET for %s (source: %s)\n", (char*)data, source); if (!strcmp(source, "elm.swallow.icon")) { ic = elm_icon_add(obj); if (ecore_file_is_dir((char*)data)) elm_icon_standard_set(ic, "folder"); // TODO we need at least a standard 'folder' and 'file' icon else elm_icon_standard_set(ic, "file"); evas_object_size_hint_aspect_set(ic, EVAS_ASPECT_CONTROL_VERTICAL, 1, 1); evas_object_show(ic); return ic; } return NULL; } static Eina_Bool _itc_state_get(const void *data, Evas_Object *obj, const char *source) { return EINA_FALSE; } static void _itc_del(const void *data, Evas_Object *obj) { //~ printf("DEL DATA [%s]\n", (char*)data); eina_stringshare_del(data); } static void _expand_done(void *data, Evas_Object *obj, void *event_info) { Elm_Genlist_Item *it = event_info; const char *path = elm_genlist_item_data_get(it); // printf("EXPAND %s\n", path); _populate(data, path, it); } static void _contract_done(void *data, Evas_Object *obj, void *event_info) { Elm_Genlist_Item *it = event_info; const char *path = elm_genlist_item_data_get(it); // printf("CONTRACT %s\n", path); elm_genlist_item_subitems_clear(it); } static void _expand_req(void *data, Evas_Object *obj, void *event_info) { Elm_Genlist_Item *it = event_info; elm_genlist_item_expanded_set(it, 1); } static void _contract_req(void *data, Evas_Object *obj, void *event_info) { Elm_Genlist_Item *it = event_info; elm_genlist_item_expanded_set(it, 0); } /*** PRIVATES ***/ static void _sel(void *data, Evas_Object *obj, void *event_info) { const char *path = elm_genlist_item_data_get(event_info); const char *p; if (ecore_file_is_dir(path)) { // printf("SELECTED DIR: %s\n", path); // keep a ref to path 'couse it will be destroyed by _populate p = eina_stringshare_add(path); _populate(data, p, NULL); eina_stringshare_del(p); return; } evas_object_smart_callback_call(data, "selected", (void*)path); } static void _up(void *data, Evas_Object *obj, void *event_info) { Widget_Data *wd = elm_widget_data_get(data); char *parent = ecore_file_dir_get(wd->path); _populate(data, parent, NULL); free(parent); } static void _home(void *data, Evas_Object *obj, void *event_info) { _populate(data, getenv("HOME"), NULL); } static void _ok(void *data, Evas_Object *obj, void *event_info) { evas_object_smart_callback_call(data, "done", (void*)elm_fileselector_selected_get(data)); } static void _canc(void *data, Evas_Object *obj, void *event_info) { evas_object_smart_callback_call(data, "done", NULL); } static void _anchor_clicked(void *data, Evas_Object *obj, void *event_info) { Widget_Data *wd = elm_widget_data_get(data); Elm_Entry_Anchor_Info *info = event_info; const char *p; // printf("ANCHOR CLICKED %s\n", info->name); // keep a ref to path 'couse it will be destroyed by _populate p = eina_stringshare_add(info->name); _populate(data, p, NULL); eina_stringshare_del(p); //~ evas_object_smart_callback_call(data, "selected", (void*)path); } _do_anchors(Evas_Object *obj, const char *path) { Widget_Data *wd = elm_widget_data_get(obj); char **tok, buf[PATH_MAX*3]; int i, j; //~ printf("ANCHORIZE...\n"); buf[0] = '\0'; tok = ecore_str_split(path, "/", 0); for (i = 0; tok[i]; i++) { if (strlen(tok[i]) < 1) continue; //~ printf("TOK: %s\n", tok[i]); ecore_strlcat(buf, "/", sizeof(buf)); ecore_strlcat(buf, tok[i], sizeof(buf)); ecore_strlcat(buf, "", sizeof(buf)); } free(tok[0]); free(tok); //~ printf("ANCHOR: %s\n", buf); elm_entry_entry_set(wd->entry, buf); } static void _populate(Evas_Object *obj, const char *path, Elm_Genlist_Item *parent) { Widget_Data *wd = elm_widget_data_get(obj); DIR *dir; struct dirent *dp; char buf[PATH_MAX]; char *real; Eina_List *files = NULL, *dirs = NULL, *l; if (!wd || !ecore_file_is_dir(path)) return; dir = opendir(path); if (!dir) return; if (!parent) { elm_genlist_clear(wd->list); if (wd->path) eina_stringshare_del(wd->path); wd->path = eina_stringshare_add(path); _do_anchors(obj, path); } while ((dp = readdir(dir)) != NULL) { if (dp->d_name[0] == '.') continue; // TODO make this configurable snprintf(buf, sizeof(buf), "%s/%s", path, dp->d_name); real = ecore_file_realpath(buf); //TODO this will resolv symlinks...I dont like it if (ecore_file_is_dir(real)) dirs = eina_list_append(dirs, real); else files = eina_list_append(files, real); } closedir(dir); files = eina_list_sort(files, ECORE_SORT_MIN, ECORE_COMPARE_CB(strcoll)); dirs = eina_list_sort(dirs, ECORE_SORT_MIN, ECORE_COMPARE_CB(strcoll)); EINA_LIST_FOREACH(dirs, l, real) { // printf("DIR: %s\n", real); elm_genlist_item_append(wd->list, &itc, eina_stringshare_add(real), /* item data */ parent, wd->expand ? ELM_GENLIST_ITEM_SUBITEMS : ELM_GENLIST_ITEM_NONE, NULL, NULL); free(real); } eina_list_free(dirs); EINA_LIST_FOREACH(files, l, real) { // printf("FILE: %s\n", real); elm_genlist_item_append(wd->list, &itc, eina_stringshare_add(real), /* item data */ parent, ELM_GENLIST_ITEM_NONE, NULL, NULL); free(real); } eina_list_free(files); } /*** API ***/ EAPI Evas_Object * elm_fileselector_add(Evas_Object *parent) { Evas_Object *obj, *ic, *bt, *box; Evas *e; Widget_Data *wd; // Elementary Widget wd = ELM_NEW(Widget_Data); wd->path = NULL; wd->expand = EINA_FALSE; e = evas_object_evas_get(parent); obj = elm_widget_add(e); elm_widget_type_set(obj, "fileselector"); elm_widget_sub_object_add(parent, obj); elm_widget_data_set(obj, wd); elm_widget_del_hook_set(obj, _del_hook); // TODO Do we need a bg object? a frame? // vbox wd->vbox = elm_box_add(parent); evas_object_size_hint_weight_set(wd->vbox, 1.0, 1.0); elm_widget_resize_object_set(obj, wd->vbox); evas_object_show(wd->vbox); // buttons box box = elm_box_add(parent); elm_box_horizontal_set(box, 1); elm_widget_sub_object_add(obj, box); elm_box_pack_end(wd->vbox, box); evas_object_size_hint_align_set(box, 0.0, 0.0); evas_object_show(box); // up btn ic = elm_icon_add(parent); elm_icon_standard_set(ic, "arrow_up"); evas_object_size_hint_aspect_set(ic, EVAS_ASPECT_CONTROL_VERTICAL, 1, 1); bt = elm_button_add(parent); elm_button_icon_set(bt, ic); elm_button_label_set(bt, "Up"); evas_object_size_hint_align_set(bt, 0.0, 0.0); elm_widget_sub_object_add(obj, bt); elm_box_pack_end(box, bt); evas_object_smart_callback_add(bt, "clicked", _up, obj); evas_object_show(bt); // home btn ic = elm_icon_add(parent); elm_icon_standard_set(ic, "home"); evas_object_size_hint_aspect_set(ic, EVAS_ASPECT_CONTROL_VERTICAL, 1, 1); bt = elm_button_add(parent); elm_button_icon_set(bt, ic); elm_button_label_set(bt, "Home"); evas_object_size_hint_align_set(bt, 0.0, 0.0); elm_widget_sub_object_add(obj, bt); elm_box_pack_end(box, bt); evas_object_smart_callback_add(bt, "clicked", _home, obj); evas_object_show(bt); // entry wd->entry = elm_entry_add(parent); elm_widget_sub_object_add(obj, wd->entry); elm_entry_entry_set(wd->entry, "TODO"); elm_entry_editable_set(wd->entry, 0); elm_entry_single_line_set(wd->entry, EINA_TRUE); evas_object_size_hint_weight_set(wd->entry, 1.0, 0.0); evas_object_size_hint_align_set(wd->entry, 0.0, 0.0); elm_box_pack_end(wd->vbox, wd->entry); evas_object_show(wd->entry); evas_object_smart_callback_add(wd->entry, "anchor,clicked", _anchor_clicked, obj); //TODO try this //~ evas_object_event_callback_add(wd->entry, EVAS_CALLBACK_CHANGED_SIZE_HINTS, //~ _changed_size_hints, obj); // genlist itc.item_style = "default"; itc.func.label_get = _itc_label_get; itc.func.icon_get = _itc_icon_get; itc.func.state_get = _itc_state_get; itc.func.del = _itc_del; wd->list = elm_genlist_add(parent); evas_object_size_hint_align_set(wd->list, -1.0, -1.0); evas_object_size_hint_weight_set(wd->list, 1.0, 1.0); evas_object_size_hint_min_set(wd->list, 100, 100); elm_widget_sub_object_add(obj, wd->list); elm_box_pack_end(wd->vbox, wd->list); evas_object_show(wd->list); evas_object_smart_callback_add(wd->list, "selected", _sel, obj); evas_object_smart_callback_add(wd->list, "expand,request", _expand_req, obj); evas_object_smart_callback_add(wd->list, "contract,request", _contract_req, obj); evas_object_smart_callback_add(wd->list, "expanded", _expand_done, obj); evas_object_smart_callback_add(wd->list, "contracted", _contract_done, obj); // buttons box box = elm_box_add(parent); elm_box_horizontal_set(box, 1); elm_widget_sub_object_add(obj, box); elm_box_pack_end(wd->vbox, box); evas_object_show(box); // cancel btn bt = elm_button_add(parent); elm_button_label_set(bt, "Cancel"); elm_widget_sub_object_add(obj, bt); elm_box_pack_end(box, bt); evas_object_smart_callback_add(bt, "clicked", _canc, obj); evas_object_show(bt); // ok btn bt = elm_button_add(parent); elm_button_label_set(bt, "Ok"); elm_widget_sub_object_add(obj, bt); elm_box_pack_end(box, bt); evas_object_smart_callback_add(bt, "clicked", _ok, obj); evas_object_show(bt); // Is this the right way to show sub-objs ?? or use the show/hide cbs ?? //~ evas_object_event_callback_add(obj, EVAS_CALLBACK_SHOW, _show, obj); //~ evas_object_event_callback_add(obj, EVAS_CALLBACK_CHANGED_SIZE_HINTS, //~ _changed_size_hints, obj); _sizing_eval(obj); return obj; } EAPI void elm_fileselector_expandable_set(Evas_Object *obj, Eina_Bool expand) { Widget_Data *wd = elm_widget_data_get(obj); wd->expand = expand; } EAPI const char* elm_fileselector_selected_get(Evas_Object *obj) { Widget_Data *wd = elm_widget_data_get(obj); Elm_Genlist_Item *it; it = elm_genlist_selected_item_get(wd->list); if (!it) return NULL; return elm_genlist_item_data_get(it); } EAPI void elm_fileselector_path_set(Evas_Object *obj, const char *path) { _populate(obj, path, NULL); }