1 #include <Elementary.h>
3 #include "elm_widget_list.h"
5 #define ITEM_HIGHLIGHT_TIMER 0.1
7 EAPI const char ELM_LIST_SMART_NAME[] = "elm_list";
9 static const char SIG_ACTIVATED[] = "activated";
10 static const char SIG_CLICKED_DOUBLE[] = "clicked,double";
11 static const char SIG_SELECTED[] = "selected";
12 static const char SIG_UNSELECTED[] = "unselected";
13 static const char SIG_LONGPRESSED[] = "longpressed";
14 static const char SIG_EDGE_TOP[] = "edge,top";
15 static const char SIG_EDGE_BOTTOM[] = "edge,bottom";
16 static const char SIG_EDGE_LEFT[] = "edge,left";
17 static const char SIG_EDGE_RIGHT[] = "edge,right";
18 static const char SIG_LANG_CHANGED[] = "language,changed";
19 static const char SIG_SWIPE[] = "swipe";
20 static const char SIG_HIGHLIGHTED[] = "highlighted";
21 static const char SIG_UNHIGHLIGHTED[] = "unhighlighted";
22 static const char SIG_ACCESS_CHANGED[] = "access,changed";
23 static const Evas_Smart_Cb_Description _smart_callbacks[] = {
25 {SIG_CLICKED_DOUBLE, ""},
28 {SIG_LONGPRESSED, ""},
30 {SIG_EDGE_BOTTOM, ""},
33 {SIG_LANG_CHANGED, ""},
35 {SIG_HIGHLIGHTED, ""},
36 {SIG_UNHIGHLIGHTED, ""},
37 {SIG_ACCESS_CHANGED, ""},
49 static const Evas_Smart_Interface *_smart_interfaces[] =
51 (Evas_Smart_Interface *)&ELM_SCROLLABLE_IFACE, NULL
54 static void _size_hints_changed_cb(void *, Evas *, Evas_Object *, void *);
55 static void _mouse_up_cb(void *, Evas *, Evas_Object *, void *);
56 static void _mouse_down_cb(void *, Evas *, Evas_Object *, void *);
57 static void _mouse_move_cb(void *, Evas *, Evas_Object *, void *);
58 static void _items_fix(Evas_Object *);
59 static void _item_select(Elm_List_Item *it);
60 static void _item_unselect(Elm_List_Item *it);
61 static void _highlight_timer_disable(Elm_List_Item *it);
64 EVAS_SMART_SUBCLASS_IFACE_NEW
65 (ELM_LIST_SMART_NAME, _elm_list, Elm_List_Smart_Class,
66 Elm_Layout_Smart_Class, elm_layout_smart_class_get, _smart_callbacks,
70 _elm_list_item_free(Elm_List_Item *it)
72 evas_object_event_callback_del_full
73 (VIEW(it), EVAS_CALLBACK_MOUSE_DOWN, _mouse_down_cb, it);
74 evas_object_event_callback_del_full
75 (VIEW(it), EVAS_CALLBACK_MOUSE_UP, _mouse_up_cb, it);
76 evas_object_event_callback_del_full
77 (VIEW(it), EVAS_CALLBACK_MOUSE_MOVE, _mouse_move_cb, it);
79 _highlight_timer_disable(it);
80 if ((Elm_List_Item *)it->sd->focused == it) it->sd->focused = NULL;
82 edje_object_message_signal_process(VIEW(it));
85 evas_object_event_callback_del_full
86 (it->icon, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
87 _size_hints_changed_cb, WIDGET(it));
90 evas_object_event_callback_del_full
91 (it->end, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
92 _size_hints_changed_cb, WIDGET(it));
94 eina_stringshare_del(it->label);
96 if (it->swipe_timer) ecore_timer_del(it->swipe_timer);
97 it->swipe_timer = NULL;
98 if (it->long_timer) ecore_timer_del(it->long_timer);
99 it->long_timer = NULL;
100 if (it->icon) evas_object_del(it->icon);
101 if (it->end) evas_object_del(it->end);
105 _item_multi_select_up(Elm_List_Smart_Data *sd)
107 Elm_Object_Item *prev;
109 if (!sd->selected) return EINA_FALSE;
110 if (!sd->multi) return EINA_FALSE;
112 prev = elm_list_item_prev(sd->last_selected_item);
113 if (!prev) return EINA_TRUE;
115 if (elm_list_item_selected_get(prev))
117 elm_list_item_selected_set(sd->last_selected_item, EINA_FALSE);
118 sd->last_selected_item = prev;
119 elm_list_item_show(sd->last_selected_item);
123 elm_list_item_selected_set(prev, EINA_TRUE);
124 elm_list_item_show(prev);
130 _item_multi_select_down(Elm_List_Smart_Data *sd)
132 Elm_Object_Item *next;
134 if (!sd->selected) return EINA_FALSE;
135 if (!sd->multi) return EINA_FALSE;
137 next = elm_list_item_next(sd->last_selected_item);
138 if (!next) return EINA_TRUE;
140 if (elm_list_item_selected_get(next))
142 elm_list_item_selected_set(sd->last_selected_item, EINA_FALSE);
143 sd->last_selected_item = next;
144 elm_list_item_show(sd->last_selected_item);
148 elm_list_item_selected_set(next, EINA_TRUE);
149 elm_list_item_show(next);
155 _all_items_unselect(Elm_List_Smart_Data *sd)
157 if (!sd->selected) return EINA_FALSE;
160 elm_list_item_selected_set
161 ((Elm_Object_Item *)sd->selected->data, EINA_FALSE);
167 _item_single_select_up(Elm_List_Smart_Data *sd)
169 Elm_Object_Item *prev;
171 if (!sd->selected) prev = eina_list_data_get(eina_list_last(sd->items));
174 prev = elm_list_item_prev(sd->last_selected_item);
177 if (!elm_object_item_disabled_get(prev)) break;
178 prev = elm_list_item_prev(prev);
181 if (!prev) return EINA_FALSE;
183 _all_items_unselect(sd);
185 elm_list_item_selected_set(prev, EINA_TRUE);
186 elm_list_item_show(prev);
192 _item_single_select_down(Elm_List_Smart_Data *sd)
194 Elm_Object_Item *next;
196 if (!sd->selected) next = eina_list_data_get(sd->items);
199 next = elm_list_item_next(sd->last_selected_item);
202 if (!elm_object_item_disabled_get(next)) break;
203 next = elm_list_item_next(next);
206 if (!next) return EINA_FALSE;
208 _all_items_unselect(sd);
210 elm_list_item_selected_set(next, EINA_TRUE);
211 elm_list_item_show(next);
216 // FIXME: There are applications which do not use elm_win as top widget.
217 // This is workaround! Those could not use focus!
218 static Eina_Bool _focus_enabled(Evas_Object *obj)
220 if (!elm_widget_focus_get(obj)) return EINA_FALSE;
222 const Evas_Object *win = elm_widget_top_get(obj);
223 const char *type = evas_object_type_get(win);
225 if (type && !strcmp(type, "elm_win"))
227 return elm_win_focus_highlight_enabled_get(win);
232 static void _item_focused(Elm_List_Item *it)
235 Elm_List_Smart_Data *sd = it->sd;
236 Evas_Coord x, y, w, h, sx, sy, sw, sh;
238 evas_object_geometry_get(VIEW(it), &x, &y, &w, &h);
239 evas_object_geometry_get(ELM_WIDGET_DATA(sd)->obj, &sx, &sy, &sw, &sh);
240 if ((x < sx) || (y < sy) || ((x + w) > (sx + sw)) || ((y + h) > (sy + sh)))
242 elm_list_item_bring_in((Elm_Object_Item *)it);
245 if (_focus_enabled(ELM_WIDGET_DATA(sd)->obj))
246 edje_object_signal_emit
247 (VIEW(it), "elm,state,focused", "elm");
249 sd->focused = (Elm_Object_Item *)it;
252 static void _item_unfocused(Elm_List_Item *it)
255 Elm_List_Smart_Data *sd = it->sd;
256 if (!sd->focused) return;
258 if (sd->focused_content)
260 elm_object_focus_set(ELM_WIDGET_DATA(sd)->obj, EINA_FALSE);
261 elm_object_focus_set(ELM_WIDGET_DATA(sd)->obj, EINA_TRUE);
262 sd->focused_content = NULL;
264 edje_object_signal_emit
265 (VIEW(sd->focused), "elm,state,unfocused", "elm");
266 if (it == (Elm_List_Item *)sd->focused)
270 static Eina_Bool _item_focused_next(Elm_List_Smart_Data *sd, Focus_Dir dir)
272 Elm_List_Item *it = NULL;
273 Elm_List_Item *old_focused = (Elm_List_Item *)sd->focused;
275 if (dir == FOCUS_DIR_DOWN || dir == FOCUS_DIR_UP)
277 if (dir == FOCUS_DIR_DOWN)
281 Eina_List *l = eina_list_data_find_list(sd->items, sd->focused);
282 it = (Elm_List_Item *)eina_list_data_get(eina_list_next(l));
283 _item_unfocused((Elm_List_Item *)sd->focused);
285 else it = (Elm_List_Item *)eina_list_data_get(sd->items);
287 else if (dir == FOCUS_DIR_UP)
291 Eina_List *l = eina_list_data_find_list(sd->items, sd->focused);
292 it = (Elm_List_Item *)eina_list_data_get(eina_list_prev(l));
293 _item_unfocused((Elm_List_Item *)sd->focused);
295 else it = (Elm_List_Item *)eina_list_data_get(eina_list_last(sd->items));
300 _item_focused(old_focused);
305 else if (dir == FOCUS_DIR_LEFT || dir == FOCUS_DIR_RIGHT)
307 Evas_Object *obj = sd->focused_content;
309 if (!sd->focused) return EINA_FALSE;
310 it = (Elm_List_Item *)sd->focused;
311 if (sd->focused_content)
313 if ((sd->focused_content != it->icon) &&
314 (sd->focused_content != it->end))
316 if (dir == FOCUS_DIR_LEFT)
318 if (it->icon) obj = it->icon;
321 else if (dir == FOCUS_DIR_RIGHT)
323 if (it->end) obj = it->end;
326 else return EINA_FALSE;
330 if (dir == FOCUS_DIR_LEFT)
332 if ((it->end == sd->focused_content))
335 else if (dir == FOCUS_DIR_RIGHT)
337 if ((it->icon == sd->focused_content))
340 else return EINA_FALSE;
343 if (!obj && dir == FOCUS_DIR_LEFT)
345 if (it->end) obj = it->icon;
348 else if (!obj && dir == FOCUS_DIR_RIGHT)
350 if (it->end) obj = it->end;
356 sd->focused_content = obj;
357 elm_object_focus_set(obj, EINA_TRUE);
361 sd->focused_content = NULL;
365 else return EINA_FALSE;
371 _elm_list_smart_event(Evas_Object *obj,
372 Evas_Object *src __UNUSED__,
373 Evas_Callback_Type type,
380 Evas_Coord step_x = 0;
381 Evas_Coord step_y = 0;
382 Evas_Coord page_x = 0;
383 Evas_Coord page_y = 0;
384 Elm_List_Item *it = NULL;
385 Evas_Event_Key_Down *ev = event_info;
387 ELM_LIST_DATA_GET(obj, sd);
389 if (elm_widget_disabled_get(obj)) return EINA_FALSE;
390 if (type != EVAS_CALLBACK_KEY_DOWN) return EINA_FALSE;
391 if (!sd->items) return EINA_FALSE;
392 if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) return EINA_FALSE;
394 sd->s_iface->content_pos_get(obj, &x, &y);
395 sd->s_iface->step_size_get(obj, &step_x, &step_y);
396 sd->s_iface->page_size_get(obj, &page_x, &page_y);
397 sd->s_iface->content_viewport_size_get(obj, &v_w, &v_h);
399 /* TODO: fix logic for horizontal mode */
400 if ((!strcmp(ev->keyname, "Left")) ||
401 ((!strcmp(ev->keyname, "KP_Left")) && !ev->string))
403 if (sd->select_on_focus_enabled)
406 (((evas_key_modifier_is_set(ev->modifiers, "Shift")) &&
407 (_item_multi_select_up(sd)))
408 || (_item_single_select_up(sd))))
410 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
418 return _item_focused_next(sd, FOCUS_DIR_LEFT);
421 else if ((!strcmp(ev->keyname, "Right")) ||
422 ((!strcmp(ev->keyname, "KP_Right")) && !ev->string))
424 if (sd->select_on_focus_enabled)
427 (((evas_key_modifier_is_set(ev->modifiers, "Shift")) &&
428 (_item_multi_select_down(sd)))
429 || (_item_single_select_down(sd))))
431 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
439 return _item_focused_next(sd, FOCUS_DIR_RIGHT);
442 else if ((!strcmp(ev->keyname, "Up")) ||
443 ((!strcmp(ev->keyname, "KP_Up")) && !ev->string))
446 if (sd->select_on_focus_enabled)
449 (evas_key_modifier_is_set(ev->modifiers, "Shift")) &&
450 (_item_multi_select_up(sd)))
452 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
458 if (_item_focused_next(sd, FOCUS_DIR_UP))
460 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
467 else if ((!strcmp(ev->keyname, "Down")) ||
468 ((!strcmp(ev->keyname, "KP_Down")) && !ev->string))
470 if (sd->select_on_focus_enabled)
473 (evas_key_modifier_is_set(ev->modifiers, "Shift")) &&
474 (_item_multi_select_down(sd)))
476 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
482 if (_item_focused_next(sd, FOCUS_DIR_DOWN))
484 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
491 else if ((!strcmp(ev->keyname, "Home")) ||
492 ((!strcmp(ev->keyname, "KP_Home")) && !ev->string))
494 if (!sd->select_on_focus_enabled)
496 _item_unfocused((Elm_List_Item *)sd->focused);
497 _item_focused_next(sd, FOCUS_DIR_DOWN);
498 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
502 else if ((!strcmp(ev->keyname, "End")) ||
503 ((!strcmp(ev->keyname, "KP_End")) && !ev->string))
505 if (!sd->select_on_focus_enabled)
507 _item_unfocused((Elm_List_Item *)sd->focused);
508 _item_focused_next(sd, FOCUS_DIR_UP);
509 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
513 else if ((!strcmp(ev->keyname, "Prior")) ||
514 ((!strcmp(ev->keyname, "KP_Prior")) && !ev->string))
519 x -= -(page_x * v_w) / 100;
526 y -= -(page_y * v_h) / 100;
531 else if ((!strcmp(ev->keyname, "Next")) ||
532 ((!strcmp(ev->keyname, "KP_Next")) && !ev->string))
537 x += -(page_x * v_w) / 100;
544 y += -(page_y * v_h) / 100;
549 else if (((!strcmp(ev->keyname, "Return")) ||
550 (!strcmp(ev->keyname, "KP_Enter")) ||
551 (!strcmp(ev->keyname, "space"))) &&
556 it = (Elm_List_Item *)sd->focused;
558 evas_object_smart_callback_call(WIDGET(it), SIG_ACTIVATED, it);
559 it->selected = EINA_FALSE;
560 sd->selected = eina_list_remove(sd->selected, it);
563 else if (!strcmp(ev->keyname, "Escape"))
565 if (!_all_items_unselect(sd)) return EINA_FALSE;
566 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
569 else return EINA_FALSE;
571 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
572 sd->s_iface->content_pos_set(obj, x, y, EINA_TRUE);
578 _elm_list_smart_translate(Evas_Object *obj)
580 ELM_LIST_DATA_GET(obj, sd);
584 EINA_LIST_FOREACH(sd->items, l, it)
585 elm_widget_item_translate(it);
587 evas_object_smart_callback_call(obj, SIG_LANG_CHANGED, NULL);
593 _elm_list_deletions_process(Elm_List_Smart_Data *sd)
597 sd->walking++; // avoid nested deletion and also _sub_del() items_fix
599 EINA_LIST_FREE (sd->to_delete, it)
601 sd->items = eina_list_remove_list(sd->items, it->node);
603 /* issuing free because of "locking" item del pre hook */
604 _elm_list_item_free(it);
605 elm_widget_item_free(it);
612 _elm_list_smart_sizing_eval(Evas_Object *obj)
614 Evas_Coord vw = 0, vh = 0;
615 Evas_Coord minw, minh, maxw, maxh, w, h, vmw, vmh;
618 ELM_LIST_DATA_GET(obj, sd);
620 /* parent class' early call */
621 if (!sd->s_iface) return;
622 if (!sd->box) return;
624 evas_object_size_hint_min_get(sd->box, &minw, &minh);
625 evas_object_size_hint_max_get(sd->box, &maxw, &maxh);
626 evas_object_size_hint_weight_get(sd->box, &xw, &yw);
628 sd->s_iface->content_viewport_size_get(obj, &vw, &vh);
631 if ((minw > 0) && (vw < minw)) vw = minw;
632 else if ((maxw > 0) && (vw > maxw))
639 if ((minh > 0) && (vh < minh)) vh = minh;
640 else if ((maxh > 0) && (vh > maxh))
646 evas_object_resize(sd->box, vw, vh);
650 edje_object_size_min_calc
651 (ELM_WIDGET_DATA(sd)->resize_obj, &vmw, &vmh);
653 if (sd->scr_minw) w = vmw + minw;
654 if (sd->scr_minh) h = vmh + minh;
656 evas_object_size_hint_max_get(obj, &maxw, &maxh);
657 if ((maxw > 0) && (w > maxw))
659 if ((maxh > 0) && (h > maxh))
662 evas_object_size_hint_min_set(obj, w, h);
666 _elm_list_content_min_limit_cb(Evas_Object *obj,
670 ELM_LIST_DATA_GET(obj, sd);
675 elm_layout_sizing_eval(obj);
679 _elm_list_mode_set_internal(Elm_List_Smart_Data *sd)
681 if (sd->mode == ELM_LIST_LIMIT)
685 sd->scr_minw = EINA_TRUE;
686 sd->scr_minh = EINA_FALSE;
690 sd->scr_minw = EINA_FALSE;
691 sd->scr_minh = EINA_TRUE;
694 else if (sd->mode == ELM_LIST_EXPAND)
696 sd->scr_minw = EINA_TRUE;
697 sd->scr_minh = EINA_TRUE;
701 sd->scr_minw = EINA_FALSE;
702 sd->scr_minh = EINA_FALSE;
705 elm_layout_sizing_eval(ELM_WIDGET_DATA(sd)->obj);
709 _elm_list_walk(Elm_List_Smart_Data *sd)
713 ERR("ERROR: walking was negative. fixed!\n");
720 _elm_list_unwalk(Elm_List_Smart_Data *sd)
725 ERR("ERROR: walking became negative. fixed!\n");
733 _elm_list_deletions_process(sd);
737 sd->fix_pending = EINA_FALSE;
738 _items_fix(ELM_WIDGET_DATA(sd)->obj);
739 elm_layout_sizing_eval(ELM_WIDGET_DATA(sd)->obj);
744 _items_fix(Evas_Object *obj)
746 Evas_Coord minw[2] = { 0, 0 }, minh[2] = { 0, 0 };
754 const char *it_plain;
755 const char *it_compress;
756 const char *it_compress_odd;
758 ELM_LIST_DATA_GET(obj, sd);
760 style = elm_widget_style_get(obj);
761 it_plain = sd->h_mode ? "h_item" : "item";
762 it_odd = sd->h_mode ? "h_item_odd" : "item_odd";
763 it_compress = sd->h_mode ? "h_item_compress" : "item_compress";
764 it_compress_odd = sd->h_mode ? "h_item_compress_odd" : "item_compress_odd";
768 sd->fix_pending = EINA_TRUE;
772 evas_object_ref(obj);
773 _elm_list_walk(sd); // watch out "return" before unwalk!
775 EINA_LIST_FOREACH(sd->items, l, it)
777 if (it->deleted) continue;
780 evas_object_size_hint_min_get(it->icon, &mw, &mh);
781 if (mw > minw[0]) minw[0] = mw;
782 if (mh > minh[0]) minh[0] = mh;
786 evas_object_size_hint_min_get(it->end, &mw, &mh);
787 if (mw > minw[1]) minw[1] = mw;
788 if (mh > minh[1]) minh[1] = mh;
792 if ((minw[0] != sd->minw[0]) || (minw[1] != sd->minw[1]) ||
793 (minh[0] != sd->minh[0]) || (minh[1] != sd->minh[1]))
795 sd->minw[0] = minw[0];
796 sd->minw[1] = minw[1];
797 sd->minh[0] = minh[0];
798 sd->minh[1] = minh[1];
803 EINA_LIST_FOREACH(sd->items, l, it)
809 if ((it->even != it->is_even) || (!it->fixed) || (redo))
811 const char *stacking;
813 /* FIXME: separators' themes seem to be b0rked */
814 if (it->is_separator)
815 elm_widget_theme_object_set
816 (obj, VIEW(it), "separator", sd->h_mode ?
817 "horizontal" : "vertical", style);
818 else if (sd->mode == ELM_LIST_COMPRESS)
821 elm_widget_theme_object_set
822 (obj, VIEW(it), "list", it_compress, style);
824 elm_widget_theme_object_set
825 (obj, VIEW(it), "list", it_compress_odd, style);
830 elm_widget_theme_object_set
831 (obj, VIEW(it), "list", it_plain, style);
833 elm_widget_theme_object_set
834 (obj, VIEW(it), "list", it_odd, style);
836 stacking = edje_object_data_get(VIEW(it), "stacking");
839 if (!strcmp(stacking, "below"))
840 evas_object_lower(VIEW(it));
841 else if (!strcmp(stacking, "above"))
842 evas_object_raise(VIEW(it));
844 edje_object_part_text_escaped_set
845 (VIEW(it), "elm.text", it->label);
847 if ((!it->icon) && (minh[0] > 0))
849 it->icon = evas_object_rectangle_add
850 (evas_object_evas_get(VIEW(it)));
851 evas_object_color_set(it->icon, 0, 0, 0, 0);
852 it->dummy_icon = EINA_TRUE;
854 if ((!it->end) && (minh[1] > 0))
856 it->end = evas_object_rectangle_add
857 (evas_object_evas_get(VIEW(it)));
858 evas_object_color_set(it->end, 0, 0, 0, 0);
859 it->dummy_end = EINA_TRUE;
863 evas_object_size_hint_min_set(it->icon, minw[0], minh[0]);
864 evas_object_size_hint_max_set(it->icon, 99999, 99999);
865 edje_object_part_swallow
866 (VIEW(it), "elm.swallow.icon", it->icon);
870 evas_object_size_hint_min_set(it->end, minw[1], minh[1]);
871 evas_object_size_hint_max_set(it->end, 99999, 99999);
872 edje_object_part_swallow
873 (VIEW(it), "elm.swallow.end", it->end);
877 // this may call up user and it may modify the list item
878 // but we're safe as we're flagged as walking.
879 // just don't process further
880 edje_object_message_signal_process(VIEW(it));
884 elm_coords_finger_size_adjust(1, &mw, 1, &mh);
885 edje_object_size_min_restricted_calc
886 (VIEW(it), &mw, &mh, mw, mh);
887 elm_coords_finger_size_adjust(1, &mw, 1, &mh);
888 evas_object_size_hint_min_set(VIEW(it), mw, mh);
889 evas_object_show(VIEW(it));
891 if ((it->selected) || (it->highlighted))
893 const char *select_raise;
895 // this may call up user and it may modify the list item
896 // but we're safe as we're flagged as walking.
897 // just don't process further
898 edje_object_signal_emit
899 (VIEW(it), "elm,state,selected", "elm");
903 select_raise = edje_object_data_get(VIEW(it), "selectraise");
904 if ((select_raise) && (!strcmp(select_raise, "on")))
905 evas_object_raise(VIEW(it));
907 if (it->base.disabled)
908 edje_object_signal_emit(VIEW(it), "elm,state,disabled", "elm");
910 it->fixed = EINA_TRUE;
911 it->is_even = it->even;
916 _elm_list_mode_set_internal(sd);
917 _elm_list_unwalk(sd);
919 evas_object_unref(obj);
923 _size_hints_changed_cb(void *data,
925 Evas_Object *obj __UNUSED__,
926 void *event_info __UNUSED__)
928 ELM_LIST_DATA_GET(data, sd);
929 if (sd->delete_me) return;
932 elm_layout_sizing_eval(data);
935 /* FIXME: take off later. maybe this show region coords belong in the
936 * interface (new api functions, set/get)? */
938 _show_region_hook(void *data,
941 Evas_Coord x, y, w, h;
943 ELM_LIST_DATA_GET(data, sd);
945 elm_widget_show_region_get(obj, &x, &y, &w, &h);
946 sd->s_iface->content_region_set(obj, x, y, w, h);
950 _elm_list_smart_disable(Evas_Object *obj)
952 ELM_LIST_DATA_GET(obj, sd);
954 if (!ELM_WIDGET_CLASS(_elm_list_parent_sc)->disable(obj))
957 if (elm_widget_disabled_get(obj))
959 elm_widget_scroll_freeze_push(obj);
960 elm_widget_scroll_hold_push(obj);
961 /* FIXME: if we get to have a way to only un-highlight items
962 * in the future, keeping them selected... */
963 _all_items_unselect(sd);
967 elm_widget_scroll_freeze_pop(obj);
968 elm_widget_scroll_hold_pop(obj);
975 _mirrored_set(Evas_Object *obj,
981 ELM_LIST_DATA_GET(obj, sd);
983 sd->s_iface->mirrored_set(obj, rtl);
985 EINA_LIST_FOREACH(sd->items, n, it)
986 edje_object_mirrored_set(VIEW(it), rtl);
990 _elm_list_smart_theme(Evas_Object *obj)
995 ELM_LIST_DATA_GET(obj, sd);
997 if (!ELM_WIDGET_CLASS(_elm_list_parent_sc)->theme(obj)) return EINA_FALSE;
999 _mirrored_set(obj, elm_widget_mirrored_get(obj));
1001 EINA_LIST_FOREACH(sd->items, n, it)
1003 edje_object_scale_set
1004 (VIEW(it), elm_widget_scale_get(obj) * elm_config_scale_get());
1009 elm_layout_sizing_eval(obj);
1015 _elm_list_smart_on_focus(Evas_Object *obj)
1017 ELM_LIST_DATA_GET(obj, sd);
1019 if (!ELM_WIDGET_CLASS(_elm_list_parent_sc)->on_focus(obj))
1022 if (elm_widget_focus_get(obj) && sd->selected && !sd->last_selected_item)
1023 sd->last_selected_item = eina_list_data_get(sd->selected);
1025 if (sd->select_on_focus_enabled) return EINA_TRUE;
1026 if (elm_widget_focus_get(obj))
1028 if (_focus_enabled(obj))
1031 _item_focused((Elm_List_Item *)sd->focused);
1033 _item_focused_next(sd, FOCUS_DIR_DOWN);
1039 edje_object_signal_emit
1040 (VIEW(sd->focused), "elm,state,unfocused", "elm");
1046 _elm_list_smart_sub_object_del(Evas_Object *obj,
1052 ELM_LIST_DATA_GET(obj, sd);
1054 if (!ELM_WIDGET_CLASS(_elm_list_parent_sc)->sub_object_del(obj, sobj))
1057 if ((sobj == sd->box) || (sobj == obj)) goto end;
1059 EINA_LIST_FOREACH(sd->items, l, it)
1061 if ((sobj == it->icon) || (sobj == it->end))
1063 if (it->icon == sobj) it->icon = NULL;
1064 if (it->end == sobj) it->end = NULL;
1065 evas_object_event_callback_del_full
1066 (sobj, EVAS_CALLBACK_CHANGED_SIZE_HINTS, _size_hints_changed_cb,
1071 elm_layout_sizing_eval(obj);
1074 sd->fix_pending = EINA_TRUE;
1084 _item_highlight(Elm_List_Item *it)
1087 const char *select_raise;
1089 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
1091 ELM_LIST_DATA_GET(obj, sd);
1093 if ((it->highlighted) || (it->base.disabled) ||
1094 (sd->select_mode == ELM_OBJECT_SELECT_MODE_NONE)) return;
1096 evas_object_ref(obj);
1099 edje_object_signal_emit(VIEW(it), "elm,state,selected", "elm");
1100 evas_object_smart_callback_call(obj, SIG_HIGHLIGHTED, it);
1101 select_raise = edje_object_data_get(VIEW(it), "selectraise");
1102 if ((select_raise) && (!strcmp(select_raise, "on")))
1103 evas_object_raise(VIEW(it));
1104 it->highlighted = EINA_TRUE;
1106 _elm_list_unwalk(sd);
1107 evas_object_unref(obj);
1111 _item_select(Elm_List_Item *it)
1115 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
1117 ELM_LIST_DATA_GET(obj, sd);
1119 if (it->base.disabled || (sd->select_mode == ELM_OBJECT_SELECT_MODE_NONE))
1123 if (sd->select_mode == ELM_OBJECT_SELECT_MODE_ALWAYS) goto call;
1126 it->selected = EINA_TRUE;
1127 sd->selected = eina_list_append(sd->selected, it);
1130 evas_object_ref(obj);
1133 if (it->func) it->func((void *)it->base.data, WIDGET(it), it);
1134 evas_object_smart_callback_call(obj, SIG_SELECTED, it);
1135 it->sd->last_selected_item = (Elm_Object_Item *)it;
1137 _elm_list_unwalk(sd);
1138 evas_object_unref(obj);
1142 _item_unselect(Elm_List_Item *it)
1144 Evas_Object *obj = WIDGET(it);
1145 ELM_LIST_DATA_GET(obj, sd);
1146 const char *stacking, *select_raise;
1148 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
1149 if (!it->highlighted) return;
1151 evas_object_ref(obj);
1154 edje_object_signal_emit(VIEW(it), "elm,state,unselected", "elm");
1155 evas_object_smart_callback_call(obj, SIG_UNHIGHLIGHTED, it);
1156 stacking = edje_object_data_get(VIEW(it), "stacking");
1157 select_raise = edje_object_data_get(VIEW(it), "selectraise");
1158 if ((select_raise) && (!strcmp(select_raise, "on")))
1160 if ((stacking) && (!strcmp(stacking, "below")))
1161 evas_object_lower(VIEW(it));
1163 it->highlighted = EINA_FALSE;
1166 it->selected = EINA_FALSE;
1167 sd->selected = eina_list_remove(sd->selected, it);
1168 evas_object_smart_callback_call(WIDGET(it), SIG_UNSELECTED, it);
1171 _elm_list_unwalk(sd);
1172 evas_object_unref(obj);
1176 _swipe_cancel(void *data)
1178 Elm_List_Item *it = data;
1180 ELM_LIST_ITEM_CHECK_OR_RETURN(it, ECORE_CALLBACK_CANCEL);
1181 ELM_LIST_DATA_GET(WIDGET(it), sd);
1183 sd->swipe = EINA_FALSE;
1186 return ECORE_CALLBACK_RENEW;
1190 _edge_left_cb(Evas_Object *obj,
1191 void *data __UNUSED__)
1193 evas_object_smart_callback_call(obj, SIG_EDGE_LEFT, NULL);
1197 _edge_right_cb(Evas_Object *obj,
1198 void *data __UNUSED__)
1200 evas_object_smart_callback_call(obj, SIG_EDGE_RIGHT, NULL);
1204 _edge_top_cb(Evas_Object *obj,
1205 void *data __UNUSED__)
1207 evas_object_smart_callback_call(obj, SIG_EDGE_TOP, NULL);
1211 _edge_bottom_cb(Evas_Object *obj,
1212 void *data __UNUSED__)
1214 evas_object_smart_callback_call(obj, SIG_EDGE_BOTTOM, NULL);
1218 _long_press_cb(void *data)
1220 Elm_List_Item *it = data;
1223 ELM_LIST_ITEM_CHECK_OR_RETURN(it, ECORE_CALLBACK_CANCEL);
1225 ELM_LIST_DATA_GET(obj, sd);
1227 it->long_timer = NULL;
1228 if (it->base.disabled) goto end;
1230 sd->longpressed = EINA_TRUE;
1231 evas_object_smart_callback_call(WIDGET(it), SIG_LONGPRESSED, it);
1234 return ECORE_CALLBACK_CANCEL;
1238 _swipe_do(Elm_List_Item *it)
1242 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
1244 ELM_LIST_DATA_GET(WIDGET(it), sd);
1246 sd->swipe = EINA_FALSE;
1247 for (i = 0; i < sd->movements; i++)
1249 sum += sd->history[i].x;
1250 if (abs(sd->history[0].y - sd->history[i].y) > 10) return;
1253 sum /= sd->movements;
1254 if (abs(sum - sd->history[0].x) <= 10) return;
1256 evas_object_smart_callback_call(WIDGET(it), SIG_SWIPE, it);
1260 _highlight_timer(void *data)
1262 Elm_List_Item *it = data;
1263 it->highlight_timer = NULL;
1264 _item_highlight(it);
1270 _highlight_timer_disable(Elm_List_Item *it)
1272 if (it->highlight_timer)
1274 ecore_timer_del(it->highlight_timer);
1275 it->highlight_timer = NULL;
1280 _highlight_timer_enable(Elm_List_Item *it)
1282 if (it->highlight_timer) ecore_timer_del(it->highlight_timer);
1283 it->highlight_timer =
1284 ecore_timer_add(ITEM_HIGHLIGHT_TIMER, _highlight_timer, it);
1288 _unhighlight_timer(void *data)
1290 Elm_List_Item *it = data;
1291 it->highlight_timer = NULL;
1297 _item_highlight(it);
1300 else _item_unselect(it);
1306 while (it->sd->selected)
1307 _item_unselect(it->sd->selected->data);
1308 _item_highlight(it);
1313 const Eina_List *l, *l_next;
1316 EINA_LIST_FOREACH_SAFE(it->sd->selected, l, l_next, it2)
1317 if (it2 != it) _item_unselect(it2);
1318 _item_highlight(it);
1327 _unhighlight_timer_enable(Elm_List_Item *it)
1329 if (it->highlight_timer) ecore_timer_del(it->highlight_timer);
1330 it->highlight_timer =
1331 ecore_timer_add(ITEM_HIGHLIGHT_TIMER, _unhighlight_timer, it);
1335 _mouse_move_cb(void *data,
1336 Evas *evas __UNUSED__,
1337 Evas_Object *o __UNUSED__,
1341 Elm_List_Item *it = data;
1342 Evas_Event_Mouse_Move *ev = event_info;
1344 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
1346 ELM_LIST_DATA_GET(obj, sd);
1348 _item_unselect((Elm_List_Item *)it);
1349 // FIXME: It needs to do disable only when item down is called like genlist
1350 _highlight_timer_disable((Elm_List_Item *)it);
1352 evas_object_ref(obj);
1355 if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD)
1359 sd->on_hold = EINA_TRUE;
1362 ecore_timer_del(it->long_timer);
1363 it->long_timer = NULL;
1365 if (!sd->was_selected)
1368 if (sd->movements == ELM_LIST_SWIPE_MOVES) sd->swipe = EINA_TRUE;
1371 sd->history[sd->movements].x = ev->cur.canvas.x;
1372 sd->history[sd->movements].y = ev->cur.canvas.y;
1373 if (abs((sd->history[sd->movements].x - sd->history[0].x)) > 40)
1374 sd->swipe = EINA_TRUE;
1380 _elm_list_unwalk(sd);
1381 evas_object_unref(obj);
1385 _mouse_down_cb(void *data,
1386 Evas *evas __UNUSED__,
1387 Evas_Object *o __UNUSED__,
1390 Evas_Event_Mouse_Down *ev = event_info;
1391 Elm_List_Item *it = data;
1394 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
1396 ELM_LIST_DATA_GET(obj, sd);
1398 if (ev->button != 1) return;
1399 if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) sd->on_hold = EINA_TRUE;
1400 else sd->on_hold = EINA_FALSE;
1402 if (sd->on_hold) return;
1403 sd->was_selected = it->selected;
1405 evas_object_ref(obj);
1408 _highlight_timer_enable(it);
1410 sd->longpressed = EINA_FALSE;
1411 if (it->long_timer) ecore_timer_del(it->long_timer);
1412 it->long_timer = ecore_timer_add
1413 (_elm_config->longpress_timeout, _long_press_cb, it);
1414 if (it->swipe_timer) ecore_timer_del(it->swipe_timer);
1415 it->swipe_timer = ecore_timer_add(0.4, _swipe_cancel, it);
1417 /* Always call the callbacks last - the user may delete our context! */
1418 if (ev->flags & EVAS_BUTTON_DOUBLE_CLICK)
1420 evas_object_smart_callback_call(WIDGET(it), SIG_CLICKED_DOUBLE, it);
1421 evas_object_smart_callback_call(WIDGET(it), SIG_ACTIVATED, it);
1423 sd->swipe = EINA_FALSE;
1426 _elm_list_unwalk(sd);
1427 evas_object_unref(obj);
1431 _mouse_up_cb(void *data,
1432 Evas *evas __UNUSED__,
1433 Evas_Object *o __UNUSED__,
1437 Elm_List_Item *it = data;
1438 Evas_Event_Mouse_Up *ev = event_info;
1440 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
1442 ELM_LIST_DATA_GET(obj, sd);
1444 if (ev->button != 1) return;
1445 if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) sd->on_hold = EINA_TRUE;
1446 else sd->on_hold = EINA_FALSE;
1447 sd->longpressed = EINA_FALSE;
1449 _highlight_timer_disable((Elm_List_Item *)it);
1452 ecore_timer_del(it->long_timer);
1453 it->long_timer = NULL;
1455 if (it->swipe_timer)
1457 ecore_timer_del(it->swipe_timer);
1458 it->swipe_timer = NULL;
1462 if (sd->swipe) _swipe_do(data);
1463 sd->on_hold = EINA_FALSE;
1466 if (sd->longpressed)
1468 if (!sd->was_selected) _item_unselect(it);
1469 sd->was_selected = 0;
1473 if (it->base.disabled)
1475 if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) return;
1477 evas_object_ref(obj);
1480 _item_highlight((Elm_List_Item *)it);
1481 _unhighlight_timer_enable((Elm_List_Item *)it);
1483 _elm_list_unwalk(sd);
1484 evas_object_unref(obj);
1488 _item_disable_hook(Elm_Object_Item *it)
1490 Elm_List_Item *item = (Elm_List_Item *)it;
1492 if (it == item->sd->focused) _item_unfocused(item);
1494 if (item->base.disabled)
1495 edje_object_signal_emit(VIEW(item), "elm,state,disabled", "elm");
1497 edje_object_signal_emit(VIEW(item), "elm,state,enabled", "elm");
1501 _item_content_set_hook(Elm_Object_Item *it,
1503 Evas_Object *content)
1505 Elm_List_Item *item = (Elm_List_Item *)it;
1506 Evas_Object **icon_p = NULL;
1507 Eina_Bool dummy = EINA_FALSE;
1509 if ((!part) || (!strcmp(part, "start")))
1511 icon_p = &(item->icon);
1512 dummy = item->dummy_icon;
1513 if (!content) item->dummy_icon = EINA_TRUE;
1514 else item->dummy_icon = EINA_FALSE;
1516 else if (!strcmp(part, "end"))
1518 icon_p = &(item->end);
1519 dummy = item->dummy_end;
1520 if (!content) item->dummy_end = EINA_TRUE;
1521 else item->dummy_end = EINA_FALSE;
1526 if (content == *icon_p) return;
1527 if ((dummy) && (!content)) return;
1528 if (dummy) evas_object_del(*icon_p);
1532 evas_object_rectangle_add(evas_object_evas_get(WIDGET(item)));
1533 evas_object_color_set(content, 0, 0, 0, 0);
1537 evas_object_del(*icon_p);
1543 edje_object_part_swallow(VIEW(item), "elm.swallow.icon", content);
1546 static Evas_Object *
1547 _item_content_get_hook(const Elm_Object_Item *it,
1550 Elm_List_Item *item = (Elm_List_Item *)it;
1552 if ((!part) || (!strcmp(part, "start")))
1554 if (item->dummy_icon) return NULL;
1557 else if (!strcmp(part, "end"))
1559 if (item->dummy_end) return NULL;
1566 static Evas_Object *
1567 _item_content_unset_hook(const Elm_Object_Item *it,
1570 Elm_List_Item *item = (Elm_List_Item *)it;
1572 if ((!part) || (!strcmp(part, "start")))
1574 Evas_Object *obj = item->icon;
1575 _item_content_set_hook((Elm_Object_Item *)it, part, NULL);
1578 else if (!strcmp(part, "end"))
1580 Evas_Object *obj = item->end;
1581 _item_content_set_hook((Elm_Object_Item *)it, part, NULL);
1589 _item_text_set_hook(Elm_Object_Item *it,
1593 Elm_List_Item *list_it = (Elm_List_Item *)it;
1595 if (part && strcmp(part, "default")) return;
1596 if (!eina_stringshare_replace(&list_it->label, text)) return;
1598 edje_object_part_text_escaped_set(VIEW(list_it), "elm.text", text);
1602 _item_text_get_hook(const Elm_Object_Item *it,
1605 if (part && strcmp(part, "default")) return NULL;
1606 return ((Elm_List_Item *)it)->label;
1610 _item_del_pre_hook(Elm_Object_Item *it)
1612 Evas_Object *obj = WIDGET(it);
1613 Elm_List_Item *item = (Elm_List_Item *)it;
1615 ELM_LIST_DATA_GET(obj, sd);
1617 if (item->selected) _item_unselect(item);
1618 if (sd->focused == it) sd->focused = NULL;
1619 _highlight_timer_disable((Elm_List_Item *)it);
1621 if (sd->walking > 0)
1623 if (item->deleted) return EINA_FALSE;
1624 item->deleted = EINA_TRUE;
1625 sd->to_delete = eina_list_append(sd->to_delete, item);
1629 sd->items = eina_list_remove_list(sd->items, item->node);
1631 evas_object_ref(obj);
1634 _elm_list_item_free(item);
1636 _elm_list_unwalk(sd);
1637 evas_object_unref(obj);
1643 _item_signal_emit_hook(Elm_Object_Item *it,
1644 const char *emission,
1647 edje_object_signal_emit(VIEW(it), emission, source);
1651 _access_info_cb(void *data, Evas_Object *obj __UNUSED__)
1653 Elm_List_Item *it = (Elm_List_Item *)data;
1654 const char *txt = NULL;
1655 if (!it) return NULL;
1657 if (!txt) txt = it->label;
1658 if (txt) return strdup(txt);
1664 _access_state_cb(void *data, Evas_Object *obj __UNUSED__)
1666 Elm_List_Item *it = (Elm_List_Item *)data;
1667 if (!it) return NULL;
1669 if (it->base.disabled)
1670 return strdup(E_("State: Disabled"));
1676 _access_on_highlight_cb(void *data)
1678 Elm_Object_Item *it = (Elm_Object_Item *)data;
1681 elm_list_item_bring_in(it);
1685 _access_activate_cb(void *data __UNUSED__,
1686 Evas_Object *part_obj __UNUSED__,
1687 Elm_Object_Item *item)
1692 it = (Elm_List_Item *)item;
1693 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
1696 ELM_LIST_DATA_GET(obj, sd);
1698 evas_object_ref(obj);
1705 _item_highlight(it);
1708 else _item_unselect(it);
1714 while (sd->selected)
1715 _item_unselect(sd->selected->data);
1716 _item_highlight(it);
1721 const Eina_List *l, *l_next;
1724 EINA_LIST_FOREACH_SAFE(sd->selected, l, l_next, it2)
1725 if (it2 != it) _item_unselect(it2);
1726 _item_highlight(it);
1731 _elm_list_unwalk(sd);
1732 evas_object_unref(obj);
1736 _access_widget_item_register(Elm_List_Item *it, Eina_Bool is_access)
1738 Elm_Access_Info *ai;
1740 if (!is_access) _elm_access_widget_item_unregister((Elm_Widget_Item *)it);
1743 _elm_access_widget_item_register((Elm_Widget_Item *)it);
1745 ai = _elm_access_object_get(it->base.access_obj);
1747 _elm_access_callback_set(ai, ELM_ACCESS_INFO, _access_info_cb, it);
1748 _elm_access_callback_set(ai, ELM_ACCESS_STATE, _access_state_cb, it);
1749 _elm_access_on_highlight_hook_set(ai, _access_on_highlight_cb, it);
1750 _elm_access_activate_callback_set(ai, _access_activate_cb, it);
1754 static Elm_List_Item *
1755 _item_new(Evas_Object *obj,
1764 ELM_LIST_DATA_GET(obj, sd);
1766 it = elm_widget_item_new(obj, Elm_List_Item);
1767 if (!it) return NULL;
1770 it->label = eina_stringshare_add(label);
1774 it->base.data = data;
1776 VIEW(it) = edje_object_add(evas_object_evas_get(obj));
1779 if (_elm_config->access_mode == ELM_ACCESS_MODE_ON)
1780 _access_widget_item_register(it, EINA_TRUE);
1782 edje_object_mirrored_set(VIEW(it), elm_widget_mirrored_get(obj));
1783 evas_object_event_callback_add
1784 (VIEW(it), EVAS_CALLBACK_MOUSE_DOWN, _mouse_down_cb, it);
1785 evas_object_event_callback_add
1786 (VIEW(it), EVAS_CALLBACK_MOUSE_UP, _mouse_up_cb, it);
1787 evas_object_event_callback_add
1788 (VIEW(it), EVAS_CALLBACK_MOUSE_MOVE, _mouse_move_cb, it);
1789 evas_object_size_hint_weight_set
1790 (VIEW(it), EVAS_HINT_EXPAND, EVAS_HINT_EXPAND);
1791 evas_object_size_hint_align_set(VIEW(it), EVAS_HINT_FILL, EVAS_HINT_FILL);
1792 edje_object_mirrored_set(VIEW(it), elm_widget_mirrored_get(obj));
1796 elm_widget_sub_object_add(obj, it->icon);
1797 evas_object_event_callback_add
1798 (it->icon, EVAS_CALLBACK_CHANGED_SIZE_HINTS, _size_hints_changed_cb,
1803 elm_widget_sub_object_add(obj, it->end);
1804 evas_object_event_callback_add
1805 (it->end, EVAS_CALLBACK_CHANGED_SIZE_HINTS, _size_hints_changed_cb,
1809 elm_widget_item_disable_hook_set(it, _item_disable_hook);
1810 elm_widget_item_content_set_hook_set(it, _item_content_set_hook);
1811 elm_widget_item_content_get_hook_set(it, _item_content_get_hook);
1812 elm_widget_item_content_unset_hook_set(it, _item_content_unset_hook);
1813 elm_widget_item_text_set_hook_set(it, _item_text_set_hook);
1814 elm_widget_item_text_get_hook_set(it, _item_text_get_hook);
1815 elm_widget_item_del_pre_hook_set(it, _item_del_pre_hook);
1816 elm_widget_item_signal_emit_hook_set(it, _item_signal_emit_hook);
1822 _resize_cb(void *data,
1824 Evas_Object *obj __UNUSED__,
1825 void *event_info __UNUSED__)
1827 elm_layout_sizing_eval(data);
1831 _elm_list_smart_focus_next(const Evas_Object *obj,
1832 Elm_Focus_Direction dir,
1835 Eina_List *items = NULL;
1836 Eina_List *elist = NULL;
1839 ELM_LIST_CHECK(obj) EINA_FALSE;
1840 ELM_LIST_DATA_GET(obj, sd);
1842 if (_elm_config->access_mode != ELM_ACCESS_MODE_ON) return EINA_FALSE;
1844 EINA_LIST_FOREACH(sd->items, elist, it)
1846 items = eina_list_append(items, it->base.access_obj);
1847 if (it->icon) items = eina_list_append(items, it->icon);
1848 if (it->end) items = eina_list_append(items, it->end);
1851 return elm_widget_focus_list_next_get
1852 (obj, items, eina_list_data_get, dir, next);
1856 _elm_list_smart_add(Evas_Object *obj)
1858 Evas_Coord minw, minh;
1860 EVAS_SMART_DATA_ALLOC(obj, Elm_List_Smart_Data);
1862 ELM_WIDGET_CLASS(_elm_list_parent_sc)->base.add(obj);
1864 elm_widget_can_focus_set(obj, EINA_TRUE);
1866 priv->mode = ELM_LIST_SCROLL;
1868 elm_layout_theme_set(obj, "list", "base", elm_widget_style_get(obj));
1870 priv->hit_rect = evas_object_rectangle_add(evas_object_evas_get(obj));
1871 evas_object_data_set(priv->hit_rect, "_elm_leaveme", obj);
1872 evas_object_smart_member_add(priv->hit_rect, obj);
1873 elm_widget_sub_object_add(obj, priv->hit_rect);
1875 /* common scroller hit rectangle setup */
1876 evas_object_color_set(priv->hit_rect, 0, 0, 0, 0);
1877 evas_object_show(priv->hit_rect);
1878 evas_object_repeat_events_set(priv->hit_rect, EINA_TRUE);
1880 priv->s_iface = evas_object_smart_interface_get
1881 (obj, ELM_SCROLLABLE_IFACE_NAME);
1883 priv->s_iface->edge_left_cb_set(obj, _edge_left_cb);
1884 priv->s_iface->edge_right_cb_set(obj, _edge_right_cb);
1885 priv->s_iface->edge_top_cb_set(obj, _edge_top_cb);
1886 priv->s_iface->edge_bottom_cb_set(obj, _edge_bottom_cb);
1888 priv->s_iface->content_min_limit_cb_set
1889 (obj, _elm_list_content_min_limit_cb);
1891 priv->s_iface->objects_set
1892 (obj, ELM_WIDGET_DATA(priv)->resize_obj, priv->hit_rect);
1894 /* the scrollable interface may set this */
1895 evas_object_event_callback_add
1896 (ELM_WIDGET_DATA(priv)->resize_obj, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
1897 _size_hints_changed_cb, obj);
1899 edje_object_size_min_calc
1900 (ELM_WIDGET_DATA(priv)->resize_obj, &minw, &minh);
1901 evas_object_size_hint_min_set(obj, minw, minh);
1902 evas_object_event_callback_add(obj, EVAS_CALLBACK_RESIZE, _resize_cb, obj);
1904 priv->s_iface->bounce_allow_set
1905 (obj, EINA_FALSE, _elm_config->thumbscroll_bounce_enable);
1907 priv->box = elm_box_add(obj);
1908 elm_box_homogeneous_set(priv->box, EINA_TRUE);
1909 evas_object_size_hint_weight_set(priv->box, EVAS_HINT_EXPAND, 0.0);
1910 evas_object_size_hint_align_set(priv->box, EVAS_HINT_FILL, 0.0);
1912 /* FIXME: change this ugly code path later */
1913 elm_widget_on_show_region_hook_set(priv->box, _show_region_hook, obj);
1914 elm_widget_sub_object_add(obj, priv->box);
1916 priv->s_iface->content_set(obj, priv->box);
1917 evas_object_event_callback_add
1918 (priv->box, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
1919 _size_hints_changed_cb, obj);
1920 const char *str = edje_object_data_get(ELM_WIDGET_DATA(priv)->resize_obj,
1922 if ((str) && (!strcmp(str, "on")))
1923 elm_widget_highlight_in_theme_set(obj, EINA_TRUE);
1925 elm_widget_highlight_in_theme_set(obj, EINA_FALSE);
1926 priv->select_on_focus_enabled = EINA_FALSE;
1930 _elm_list_smart_del(Evas_Object *obj)
1935 ELM_LIST_DATA_GET(obj, sd);
1938 ERR("ERROR: list deleted while walking.\n");
1940 _item_unfocused((Elm_List_Item *)sd->focused);
1941 sd->delete_me = EINA_TRUE;
1942 EINA_LIST_FOREACH(sd->items, l, it)
1945 evas_object_event_callback_del
1946 (it->icon, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
1947 _size_hints_changed_cb);
1949 evas_object_event_callback_del
1950 (it->end, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
1951 _size_hints_changed_cb);
1954 evas_object_event_callback_del
1955 (obj, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
1956 _size_hints_changed_cb);
1957 evas_object_event_callback_del
1958 (sd->box, EVAS_CALLBACK_CHANGED_SIZE_HINTS, _size_hints_changed_cb);
1962 EINA_LIST_FREE (sd->items, it)
1964 /* issuing free because of "locking" item del pre hook */
1965 _elm_list_item_free(it);
1966 elm_widget_item_free(it);
1969 _elm_list_unwalk(sd);
1972 ERR("ERROR: leaking nodes!\n");
1974 eina_list_free(sd->selected);
1976 ELM_WIDGET_CLASS(_elm_list_parent_sc)->base.del(obj);
1980 _elm_list_smart_move(Evas_Object *obj,
1984 ELM_LIST_DATA_GET(obj, sd);
1986 ELM_WIDGET_CLASS(_elm_list_parent_sc)->base.move(obj, x, y);
1988 evas_object_move(sd->hit_rect, x, y);
1992 _elm_list_smart_resize(Evas_Object *obj,
1996 ELM_LIST_DATA_GET(obj, sd);
1998 ELM_WIDGET_CLASS(_elm_list_parent_sc)->base.resize(obj, w, h);
2000 evas_object_resize(sd->hit_rect, w, h);
2004 _elm_list_smart_member_add(Evas_Object *obj,
2005 Evas_Object *member)
2007 ELM_LIST_DATA_GET(obj, sd);
2009 ELM_WIDGET_CLASS(_elm_list_parent_sc)->base.member_add(obj, member);
2012 evas_object_raise(sd->hit_rect);
2016 _elm_list_smart_access(Evas_Object *obj, Eina_Bool is_access)
2018 Eina_List *elist = NULL;
2021 ELM_LIST_DATA_GET(obj, sd);
2024 ELM_WIDGET_CLASS(ELM_WIDGET_DATA(sd)->api)->focus_next =
2025 _elm_list_smart_focus_next;
2027 ELM_WIDGET_CLASS(ELM_WIDGET_DATA(sd)->api)->focus_next = NULL;
2029 EINA_LIST_FOREACH(sd->items, elist, it)
2030 _access_widget_item_register(it, is_access);
2032 evas_object_smart_callback_call(obj, SIG_ACCESS_CHANGED, NULL);
2036 _elm_list_smart_set_user(Elm_List_Smart_Class *sc)
2038 ELM_WIDGET_CLASS(sc)->base.add = _elm_list_smart_add;
2039 ELM_WIDGET_CLASS(sc)->base.del = _elm_list_smart_del;
2040 ELM_WIDGET_CLASS(sc)->base.move = _elm_list_smart_move;
2041 ELM_WIDGET_CLASS(sc)->base.resize = _elm_list_smart_resize;
2042 ELM_WIDGET_CLASS(sc)->base.member_add = _elm_list_smart_member_add;
2044 ELM_WIDGET_CLASS(sc)->sub_object_del = _elm_list_smart_sub_object_del;
2045 ELM_WIDGET_CLASS(sc)->on_focus = _elm_list_smart_on_focus;
2046 ELM_WIDGET_CLASS(sc)->focus_next = NULL;
2047 ELM_WIDGET_CLASS(sc)->focus_direction = NULL;
2048 ELM_WIDGET_CLASS(sc)->theme = _elm_list_smart_theme;
2049 ELM_WIDGET_CLASS(sc)->disable = _elm_list_smart_disable;
2050 ELM_WIDGET_CLASS(sc)->event = _elm_list_smart_event;
2051 ELM_WIDGET_CLASS(sc)->translate = _elm_list_smart_translate;
2052 ELM_WIDGET_CLASS(sc)->access = _elm_list_smart_access;
2054 ELM_LAYOUT_CLASS(sc)->sizing_eval = _elm_list_smart_sizing_eval;
2056 if (_elm_config->access_mode == ELM_ACCESS_MODE_ON)
2057 ELM_WIDGET_CLASS(sc)->focus_next = _elm_list_smart_focus_next;
2060 EAPI const Elm_List_Smart_Class *
2061 elm_list_smart_class_get(void)
2063 static Elm_List_Smart_Class _sc =
2064 ELM_LIST_SMART_CLASS_INIT_NAME_VERSION(ELM_LIST_SMART_NAME);
2065 static const Elm_List_Smart_Class *class = NULL;
2066 Evas_Smart_Class *esc = (Evas_Smart_Class *)&_sc;
2071 _elm_list_smart_set(&_sc);
2072 esc->callbacks = _smart_callbacks;
2079 elm_list_add(Evas_Object *parent)
2083 EINA_SAFETY_ON_NULL_RETURN_VAL(parent, NULL);
2085 obj = elm_widget_add(_elm_list_smart_class_new(), parent);
2086 if (!obj) return NULL;
2088 if (!elm_widget_sub_object_add(parent, obj))
2089 ERR("could not add %p as sub object of %p", obj, parent);
2095 elm_list_go(Evas_Object *obj)
2097 ELM_LIST_CHECK(obj);
2103 elm_list_multi_select_set(Evas_Object *obj,
2106 ELM_LIST_CHECK(obj);
2107 ELM_LIST_DATA_GET(obj, sd);
2113 elm_list_multi_select_get(const Evas_Object *obj)
2115 ELM_LIST_CHECK(obj) EINA_FALSE;
2116 ELM_LIST_DATA_GET(obj, sd);
2122 elm_list_mode_set(Evas_Object *obj,
2125 ELM_LIST_CHECK(obj);
2126 ELM_LIST_DATA_GET(obj, sd);
2128 if (sd->mode == mode)
2133 _elm_list_mode_set_internal(sd);
2137 elm_list_mode_get(const Evas_Object *obj)
2139 ELM_LIST_CHECK(obj) ELM_LIST_LAST;
2140 ELM_LIST_DATA_GET(obj, sd);
2146 elm_list_horizontal_set(Evas_Object *obj,
2147 Eina_Bool horizontal)
2149 Eina_Bool bounce = _elm_config->thumbscroll_bounce_enable;
2151 ELM_LIST_CHECK(obj);
2152 ELM_LIST_DATA_GET(obj, sd);
2154 horizontal = !!horizontal;
2156 if (sd->h_mode == horizontal)
2159 sd->h_mode = horizontal;
2160 elm_box_horizontal_set(sd->box, horizontal);
2164 evas_object_size_hint_weight_set(sd->box, 0.0, EVAS_HINT_EXPAND);
2165 evas_object_size_hint_align_set(sd->box, 0.0, EVAS_HINT_FILL);
2166 sd->s_iface->bounce_allow_set(obj, bounce, EINA_FALSE);
2170 evas_object_size_hint_weight_set(sd->box, EVAS_HINT_EXPAND, 0.0);
2171 evas_object_size_hint_align_set(sd->box, EVAS_HINT_FILL, 0.0);
2172 sd->s_iface->bounce_allow_set(obj, EINA_FALSE, bounce);
2175 _elm_list_mode_set_internal(sd);
2179 elm_list_horizontal_get(const Evas_Object *obj)
2181 ELM_LIST_CHECK(obj) EINA_FALSE;
2182 ELM_LIST_DATA_GET(obj, sd);
2188 elm_list_select_mode_set(Evas_Object *obj,
2189 Elm_Object_Select_Mode mode)
2191 ELM_LIST_CHECK(obj);
2192 ELM_LIST_DATA_GET(obj, sd);
2194 if (mode >= ELM_OBJECT_SELECT_MODE_MAX)
2197 if (sd->select_mode != mode)
2198 sd->select_mode = mode;
2201 EAPI Elm_Object_Select_Mode
2202 elm_list_select_mode_get(const Evas_Object *obj)
2204 ELM_LIST_CHECK(obj) ELM_OBJECT_SELECT_MODE_MAX;
2205 ELM_LIST_DATA_GET(obj, sd);
2207 return sd->select_mode;
2211 elm_list_bounce_set(Evas_Object *obj,
2215 ELM_LIST_CHECK(obj);
2216 ELM_LIST_DATA_GET(obj, sd);
2218 sd->s_iface->bounce_allow_set(obj, h_bounce, v_bounce);
2222 elm_list_bounce_get(const Evas_Object *obj,
2223 Eina_Bool *h_bounce,
2224 Eina_Bool *v_bounce)
2226 ELM_LIST_CHECK(obj);
2227 ELM_LIST_DATA_GET(obj, sd);
2229 sd->s_iface->bounce_allow_get(obj, h_bounce, v_bounce);
2233 elm_list_scroller_policy_set(Evas_Object *obj,
2234 Elm_Scroller_Policy policy_h,
2235 Elm_Scroller_Policy policy_v)
2237 ELM_LIST_CHECK(obj);
2238 ELM_LIST_DATA_GET(obj, sd);
2240 if ((policy_h >= ELM_SCROLLER_POLICY_LAST) ||
2241 (policy_v >= ELM_SCROLLER_POLICY_LAST))
2244 sd->s_iface->policy_set(obj, policy_h, policy_v);
2248 elm_list_scroller_policy_get(const Evas_Object *obj,
2249 Elm_Scroller_Policy *policy_h,
2250 Elm_Scroller_Policy *policy_v)
2252 ELM_LIST_CHECK(obj);
2253 ELM_LIST_DATA_GET(obj, sd);
2255 sd->s_iface->policy_get(obj, policy_h, policy_v);
2259 elm_list_clear(Evas_Object *obj)
2263 ELM_LIST_CHECK(obj);
2264 ELM_LIST_DATA_GET(obj, sd);
2266 if (!sd->items) return;
2268 eina_list_free(sd->selected);
2269 sd->selected = NULL;
2271 if (sd->walking > 0)
2275 EINA_LIST_FOREACH(sd->items, n, it)
2277 if (it->deleted) continue;
2278 it->deleted = EINA_TRUE;
2279 sd->to_delete = eina_list_append(sd->to_delete, it);
2284 evas_object_ref(obj);
2288 EINA_LIST_FREE (sd->items, it)
2290 /* issuing free because of "locking" item del pre hook */
2291 _elm_list_item_free(it);
2292 elm_widget_item_free(it);
2295 _elm_list_unwalk(sd);
2298 elm_layout_sizing_eval(obj);
2300 evas_object_unref(obj);
2303 EAPI const Eina_List *
2304 elm_list_items_get(const Evas_Object *obj)
2306 ELM_LIST_CHECK(obj) NULL;
2307 ELM_LIST_DATA_GET(obj, sd);
2312 EAPI Elm_Object_Item *
2313 elm_list_selected_item_get(const Evas_Object *obj)
2315 ELM_LIST_CHECK(obj) NULL;
2316 ELM_LIST_DATA_GET(obj, sd);
2318 if (sd->selected) return (Elm_Object_Item *)sd->selected->data;
2323 EAPI const Eina_List *
2324 elm_list_selected_items_get(const Evas_Object *obj)
2326 ELM_LIST_CHECK(obj) NULL;
2327 ELM_LIST_DATA_GET(obj, sd);
2329 return sd->selected;
2332 EAPI Elm_Object_Item *
2333 elm_list_item_append(Evas_Object *obj,
2342 ELM_LIST_CHECK(obj) NULL;
2343 ELM_LIST_DATA_GET(obj, sd);
2345 it = _item_new(obj, label, icon, end, func, data);
2347 sd->items = eina_list_append(sd->items, it);
2348 it->node = eina_list_last(sd->items);
2349 elm_box_pack_end(sd->box, VIEW(it));
2351 return (Elm_Object_Item *)it;
2354 EAPI Elm_Object_Item *
2355 elm_list_item_prepend(Evas_Object *obj,
2364 ELM_LIST_CHECK(obj) NULL;
2365 ELM_LIST_DATA_GET(obj, sd);
2367 it = _item_new(obj, label, icon, end, func, data);
2369 sd->items = eina_list_prepend(sd->items, it);
2370 it->node = sd->items;
2371 elm_box_pack_start(sd->box, VIEW(it));
2373 return (Elm_Object_Item *)it;
2376 EAPI Elm_Object_Item *
2377 elm_list_item_insert_before(Evas_Object *obj,
2378 Elm_Object_Item *before,
2385 Elm_List_Item *it, *before_it;
2387 ELM_LIST_CHECK(obj) NULL;
2388 ELM_LIST_ITEM_CHECK_OR_RETURN(before, NULL);
2389 ELM_LIST_DATA_GET(obj, sd);
2391 before_it = (Elm_List_Item *)before;
2392 if (!before_it->node) return NULL;
2394 it = _item_new(obj, label, icon, end, func, data);
2395 sd->items = eina_list_prepend_relative_list(sd->items, it, before_it->node);
2396 it->node = before_it->node->prev;
2397 elm_box_pack_before(sd->box, VIEW(it), VIEW(before_it));
2399 return (Elm_Object_Item *)it;
2402 EAPI Elm_Object_Item *
2403 elm_list_item_insert_after(Evas_Object *obj,
2404 Elm_Object_Item *after,
2411 Elm_List_Item *it, *after_it;
2413 ELM_LIST_CHECK(obj) NULL;
2414 ELM_LIST_ITEM_CHECK_OR_RETURN(after, NULL);
2415 ELM_LIST_DATA_GET(obj, sd);
2417 after_it = (Elm_List_Item *)after;
2418 if (!after_it->node) return NULL;
2420 it = _item_new(obj, label, icon, end, func, data);
2421 sd->items = eina_list_append_relative_list(sd->items, it, after_it->node);
2422 it->node = after_it->node->next;
2423 elm_box_pack_after(sd->box, VIEW(it), VIEW(after_it));
2425 return (Elm_Object_Item *)it;
2428 EAPI Elm_Object_Item *
2429 elm_list_item_sorted_insert(Evas_Object *obj,
2435 Eina_Compare_Cb cmp_func)
2440 ELM_LIST_CHECK(obj) NULL;
2441 ELM_LIST_DATA_GET(obj, sd);
2443 it = _item_new(obj, label, icon, end, func, data);
2445 sd->items = eina_list_sorted_insert(sd->items, cmp_func, it);
2446 l = eina_list_data_find_list(sd->items, it);
2447 l = eina_list_next(l);
2450 it->node = eina_list_last(sd->items);
2451 elm_box_pack_end(sd->box, VIEW(it));
2455 Elm_List_Item *before = eina_list_data_get(l);
2457 it->node = before->node->prev;
2458 elm_box_pack_before(sd->box, VIEW(it), VIEW(before));
2461 return (Elm_Object_Item *)it;
2465 elm_list_item_separator_set(Elm_Object_Item *it,
2468 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
2470 ((Elm_List_Item *)it)->is_separator = !!setting;
2474 elm_list_item_separator_get(const Elm_Object_Item *it)
2476 ELM_LIST_ITEM_CHECK_OR_RETURN(it, EINA_FALSE);
2478 return ((Elm_List_Item *)it)->is_separator;
2482 elm_list_item_selected_set(Elm_Object_Item *it,
2485 Elm_List_Item *item = (Elm_List_Item *)it;
2488 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
2490 ELM_LIST_DATA_GET(obj, sd);
2492 selected = !!selected;
2493 if (item->selected == selected) return;
2495 evas_object_ref(obj);
2502 while (sd->selected)
2503 _item_unselect(sd->selected->data);
2505 _item_highlight(item);
2509 _item_unselect(item);
2511 _elm_list_unwalk(sd);
2512 evas_object_unref(obj);
2516 elm_list_item_selected_get(const Elm_Object_Item *it)
2518 ELM_LIST_ITEM_CHECK_OR_RETURN(it, EINA_FALSE);
2520 return ((Elm_List_Item *)it)->selected;
2524 elm_list_item_show(Elm_Object_Item *it)
2526 Evas_Coord bx, by, bw, bh;
2527 Evas_Coord x, y, w, h;
2529 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
2530 ELM_LIST_DATA_GET(WIDGET(it), sd);
2532 evas_smart_objects_calculate(evas_object_evas_get(sd->box));
2533 evas_object_geometry_get(sd->box, &bx, &by, &bw, &bh);
2534 evas_object_geometry_get(VIEW(it), &x, &y, &w, &h);
2538 sd->s_iface->content_region_show(WIDGET(it), x, y, w, h);
2542 elm_list_item_bring_in(Elm_Object_Item *it)
2544 Evas_Coord bx, by, bw, bh;
2545 Evas_Coord x, y, w, h;
2547 ELM_LIST_ITEM_CHECK_OR_RETURN(it);
2548 ELM_LIST_DATA_GET(WIDGET(it), sd);
2550 evas_smart_objects_calculate(evas_object_evas_get(sd->box));
2551 evas_object_geometry_get(sd->box, &bx, &by, &bw, &bh);
2552 evas_object_geometry_get(VIEW(it), &x, &y, &w, &h);
2556 sd->s_iface->region_bring_in(WIDGET(it), x, y, w, h);
2560 elm_list_item_object_get(const Elm_Object_Item *it)
2562 ELM_LIST_ITEM_CHECK_OR_RETURN(it, NULL);
2567 EAPI Elm_Object_Item *
2568 elm_list_item_prev(const Elm_Object_Item *it)
2570 Elm_List_Item *item = (Elm_List_Item *)it;
2572 ELM_LIST_ITEM_CHECK_OR_RETURN(it, NULL);
2574 if (item->node->prev) return item->node->prev->data;
2578 EAPI Elm_Object_Item *
2579 elm_list_item_next(const Elm_Object_Item *it)
2581 Elm_List_Item *item = (Elm_List_Item *)it;
2583 ELM_LIST_ITEM_CHECK_OR_RETURN(it, NULL);
2585 if (item->node->next) return item->node->next->data;
2589 EAPI Elm_Object_Item *
2590 elm_list_first_item_get(const Evas_Object *obj)
2592 ELM_LIST_CHECK(obj) NULL;
2593 ELM_LIST_DATA_GET(obj, sd);
2595 if (!sd->items) return NULL;
2597 return eina_list_data_get(sd->items);
2600 EAPI Elm_Object_Item *
2601 elm_list_last_item_get(const Evas_Object *obj)
2603 ELM_LIST_CHECK(obj) NULL;
2604 ELM_LIST_DATA_GET(obj, sd);
2606 if (!sd->items) return NULL;
2608 return eina_list_data_get(eina_list_last(sd->items));
2611 EAPI Elm_Object_Item *
2612 elm_list_at_xy_item_get(const Evas_Object *obj,
2617 ELM_LIST_CHECK(obj) NULL;
2618 ELM_LIST_DATA_GET(obj, sd);
2622 evas_object_geometry_get(sd->hit_rect, &lasty, NULL, NULL, NULL);
2624 EINA_LIST_FOREACH(sd->items, l, it)
2626 Evas_Coord itx, ity;
2627 Evas_Object *vit = VIEW(it);
2628 Evas_Coord vx, vy, vw, vh;
2629 evas_object_geometry_get(vit, &vx, &vy, &vw, &vh);
2633 if (ELM_RECTS_INTERSECT
2634 (itx, ity, vw, vh, x, y, 1, 1))
2638 if (y <= (ity + (vh / 4))) *posret = -1;
2639 else if (y >= (ity + vh - (vh / 4)))
2644 return (Elm_Object_Item *) it;
2652 if (y > lasty) *posret = 1;