9b780038e257f3b3edf71323eef155bcd9603597
[framework/uifw/elementary.git] / src / lib / elm_genlist.c
1 #include <assert.h>
2 #include <fnmatch.h>
3 #include <Elementary.h>
4 #include <Elementary_Cursor.h>
5 #include "elm_priv.h"
6 #include "els_scroller.h"
7 //#include "elm_gen_common.h" //upstream
8
9 #define SWIPE_MOVES         12
10 #define MAX_ITEMS_PER_BLOCK 32
11 #define REORDER_EFFECT_TIME 0.5
12
13 typedef struct _Widget_Data Widget_Data;
14 typedef struct _Item_Block  Item_Block;
15 typedef struct _Pan         Pan;
16 typedef struct _Item_Cache  Item_Cache;
17 typedef enum _Elm_Genlist_Item_Move_effect_Mode
18 {
19    ELM_GENLIST_ITEM_MOVE_EFFECT_NONE         = 0,
20    ELM_GENLIST_ITEM_MOVE_EFFECT_EXPAND       = (1 << 0),
21    ELM_GENLIST_ITEM_MOVE_EFFECT_CONTRACT     = (1 << 1),
22    ELM_GENLIST_ITEM_MOVE_EFFECT_DELETE       = (1 << 2),
23 } Elm_Genlist_Item_Move_effect_Mode;
24
25 typedef enum _Elm_Genlist_Item_Scrollto_Type
26 {
27    ELM_GENLIST_ITEM_SCROLLTO_NONE = 0, /**< no scrollto */
28    ELM_GENLIST_ITEM_SCROLLTO_IN = (1 << 0), /**< show, bring in */
29    ELM_GENLIST_ITEM_SCROLLTO_TOP = (1 << 1), /**< top show, top bring in */
30    ELM_GENLIST_ITEM_SCROLLTO_MIDDLE = (1 << 2) /**< middle show, middle bring in */
31 } Elm_Genlist_Item_Scrollto_Type;
32
33 struct _Widget_Data
34 {
35    Evas_Object      *obj, *scr, *pan_smart;
36    Eina_Inlist      *items, *blocks;
37    Eina_List        *group_items;
38    Pan              *pan;
39    Evas_Coord        pan_x, pan_y, old_pan_y, w, h, minw, minh, realminw, prev_viewport_w;
40    Ecore_Job        *calc_job, *update_job;
41    Ecore_Idle_Enterer *queue_idle_enterer;
42    Ecore_Idler        *must_recalc_idler;
43    Eina_List        *queue, *selected;
44    Elm_Object_Item  *last_selected_item;
45    Elm_Genlist_Item *show_item, *anchor_item, *mode_item, *reorder_it, *reorder_rel, *expanded_item;
46    Eina_Inlist      *item_cache;
47    Evas_Coord        anchor_y, reorder_start_y;
48    Elm_List_Mode     mode;
49    Ecore_Timer      *multi_timer, *scr_hold_timer;
50    Ecore_Animator   *reorder_move_animator;
51    const char       *mode_type;
52    unsigned int      start_time;
53    Evas_Coord        prev_x, prev_y, prev_mx, prev_my;
54    Evas_Coord        cur_x, cur_y, cur_mx, cur_my;
55    Eina_Bool         mouse_down : 1;
56    Eina_Bool         multi_down : 1;
57    Eina_Bool         multi_timeout : 1;
58    Eina_Bool         multitouched : 1;
59    Eina_Bool         on_hold : 1;
60    Eina_Bool         multi : 1;
61    Eina_Bool         always_select : 1;
62    Eina_Bool         longpressed : 1;
63    Eina_Bool         wasselected : 1;
64    Eina_Bool         no_select : 1;
65    Eina_Bool         bring_in : 1; /* a flag to describe the scroll animation. (show, bring in) */
66    Eina_Bool         compress : 1;
67    Eina_Bool         height_for_width : 1;
68    Eina_Bool         pan_resize : 1;
69    Eina_Bool         homogeneous : 1;
70    Eina_Bool         clear_me : 1;
71    Eina_Bool         swipe : 1;
72    Eina_Bool         reorder_mode : 1;
73    Eina_Bool         reorder_pan_move : 1;
74    struct
75    {
76       Evas_Coord x, y;
77    } history[SWIPE_MOVES];
78    int               multi_device;
79    int               item_cache_count;
80    int               item_cache_max;
81    int               movements;
82    int               walking;
83    int               item_width;
84    int               item_height;
85    int               group_item_width;
86    int               group_item_height;
87    int               max_items_per_block;
88    double            longpress_timeout;
89
90    // TODO : refactoring
91    Eina_Bool         reorder_deleted : 1;
92    Eina_Bool         effect_mode : 1;
93    Eina_Bool         auto_scroll_enabled : 1;
94    Eina_Bool         pan_changed : 1;
95    Eina_Bool         drag_started : 1;
96    Eina_Bool         check_scroll : 1; /* this flag means genlist is supposed to be scrolled. if this flag is set to EINA_TRUE, genlist checks whether it's ok to scroll genlist now or not. */
97    Eina_Bool         expanded_move_effect_set : 1;
98    int               edit_mode;
99    Elm_Genlist_Item_Scrollto_Type scrollto_type; /* a scrollto type which remembers where to scroll ex) in, top, middle */
100    Ecore_Animator   *item_moving_effect_timer;
101    Evas_Object      *alpha_bg;
102    Evas_Coord        expand_item_end;
103    Evas_Coord        expand_item_gap;
104    int               move_effect_mode;
105    Ecore_Job        *changed_job;
106    Elm_Genlist_Item *rename_it, *expanded_next_item;
107 };
108
109 struct _Elm_Genlist_Item
110 {
111    ELM_WIDGET_ITEM;
112    EINA_INLIST;
113    Widget_Data                  *wd;
114    Item_Block                   *block;
115    Eina_List                    *items;
116    Evas_Coord                    x, y, w, h, minw, minh;
117    const Elm_Genlist_Item_Class *itc;
118    Elm_Genlist_Item             *parent;
119    Elm_Genlist_Item             *group_item;
120    Elm_Genlist_Item_Flags        flags;
121    struct
122    {
123       Evas_Smart_Cb func;
124       const void   *data;
125    } func;
126
127    Evas_Object                  *spacer;
128    Eina_List                    *texts, *contents, *states, *content_objs;
129    Eina_List                    *mode_texts, *mode_contents, *mode_states, *mode_content_objs;
130    Ecore_Timer                  *long_timer;
131    Ecore_Timer                  *swipe_timer;
132    Evas_Coord                    dx, dy;
133    Evas_Coord                    scrl_x, scrl_y;
134
135    Elm_Genlist_Item             *rel;
136    Evas_Object                  *mode_view;
137
138    struct
139    {
140       const void                 *data;
141       Elm_Tooltip_Item_Content_Cb content_cb;
142       Evas_Smart_Cb               del_cb;
143       const char                 *style;
144    } tooltip;
145
146    const char                   *mouse_cursor;
147
148    int                           relcount;
149    int                           walking;
150    int                           expanded_depth;
151    int                           order_num_in;
152    int                           position;
153
154    Eina_Bool                     position_update : 1;
155    Eina_Bool                     before : 1;
156
157    Eina_Bool                     want_unrealize : 1;
158    Eina_Bool                     realized : 1;
159    Eina_Bool                     selected : 1;
160    Eina_Bool                     highlighted : 1;
161    Eina_Bool                     expanded : 1;
162    Eina_Bool                     display_only : 1;
163    Eina_Bool                     mincalcd : 1;
164    Eina_Bool                     queued : 1;
165    Eina_Bool                     showme : 1;
166    Eina_Bool                     delete_me : 1;
167    Eina_Bool                     down : 1;
168    Eina_Bool                     dragging : 1;
169    Eina_Bool                     updateme : 1;
170    Eina_Bool                     nocache : 1;
171    Eina_Bool                     stacking_even : 1;
172    Eina_Bool                     nostacking : 1;
173    Eina_Bool                     move_effect_enabled : 1;
174    Eina_Bool                     defer_unrealize : 1;
175    Eina_Bool                     can_focus : 1;
176    Eina_Bool                     no_select : 1;
177
178
179    // TODO: refactoring
180    Eina_Bool   effect_done : 1;
181    Eina_Bool   reordering : 1;
182    Eina_Bool   renamed : 1;
183    Eina_Bool   effect_item_realized : 1;
184    Eina_List  *edit_content_objs;
185    Evas_Object *edit_obj;
186    int         num;
187    Ecore_Animator *item_moving_effect_timer;
188    Evas_Coord  old_scrl_y;
189 };
190
191 struct _Item_Block
192 {
193    EINA_INLIST;
194    int          count;
195    int          num;
196    int          reorder_offset;
197    Widget_Data *wd;
198    Eina_List   *items;
199    Evas_Coord   x, y, w, h, minw, minh;
200    int          position;
201    int          item_position_stamp;
202
203    Eina_Bool    position_update : 1;
204    Eina_Bool    want_unrealize : 1;
205    Eina_Bool    realized : 1;
206    Eina_Bool    changed : 1;
207    Eina_Bool    updateme : 1;
208    Eina_Bool    changeme : 1;
209    Eina_Bool    showme : 1;
210    Eina_Bool    must_recalc : 1;
211 };
212
213 struct _Item_Cache
214 {
215    EINA_INLIST;
216
217    Evas_Object *base_view, *spacer;
218
219    const char  *item_style; // it->itc->item_style
220    Eina_Bool    tree : 1; // it->flags & ELM_GENLIST_ITEM_SUBITEMS
221    Eina_Bool    compress : 1; // it->wd->compress
222
223    Eina_Bool    selected : 1; // it->selected
224    Eina_Bool    disabled : 1; // it->disabled
225    Eina_Bool    expanded : 1; // it->expanded
226 };
227
228 #define ELM_GENLIST_ITEM_FROM_INLIST(item) \
229   ((item) ? EINA_INLIST_CONTAINER_GET(item, Elm_Genlist_Item) : NULL)
230
231 struct _Pan
232 {
233    Evas_Object_Smart_Clipped_Data __clipped_data;
234    Widget_Data                   *wd;
235    Ecore_Job                     *resize_job;
236 };
237
238 static const char *widtype = NULL;
239 static void      _item_cache_zero(Widget_Data *wd);
240 static void      _del_hook(Evas_Object *obj);
241 static void      _mirrored_set(Evas_Object *obj,
242                                Eina_Bool    rtl);
243 static void      _theme_hook(Evas_Object *obj);
244 static void      _show_region_hook(void        *data,
245                                    Evas_Object *obj);
246 static void      _sizing_eval(Evas_Object *obj);
247 static void      _item_realize(Elm_Genlist_Item *it,
248                                int               in,
249                                Eina_Bool         calc);
250 static void      _item_unrealize(Elm_Genlist_Item *it,
251                                  Eina_Bool         calc);
252 static void      _item_block_unrealize(Item_Block *itb);
253 static void      _calc_job(void *data);
254 static void      _on_focus_hook(void        *data,
255                                 Evas_Object *obj);
256 static void      _changed_size_hints(void *data, Evas *e, Evas_Object *obj, void *event_info);
257 static void      _changed_job(void *data);
258 static Eina_Bool _item_multi_select_up(Widget_Data *wd);
259 static Eina_Bool _item_multi_select_down(Widget_Data *wd);
260 static Eina_Bool _item_single_select_up(Widget_Data *wd);
261 static Eina_Bool _item_single_select_down(Widget_Data *wd);
262 static Eina_Bool _event_hook(Evas_Object       *obj,
263                              Evas_Object       *src,
264                              Evas_Callback_Type type,
265                              void              *event_info);
266 static void      _signal_emit_hook(Evas_Object *obj,
267                                    const char *emission,
268                                    const char *source);
269 static Eina_Bool _deselect_all_items(Widget_Data *wd);
270 static void      _pan_calculate(Evas_Object *obj);
271 static void      _pan_max_get(Evas_Object *obj,
272                               Evas_Coord  *x,
273                               Evas_Coord  *y);
274 static void      _item_position(Elm_Genlist_Item *it,
275                                 Evas_Object      *obj,
276                                 Evas_Coord        it_x,
277                                 Evas_Coord        it_y);
278 static void      _mode_item_realize(Elm_Genlist_Item *it);
279 static void      _mode_item_unrealize(Elm_Genlist_Item *it);
280 static void      _item_mode_set(Elm_Genlist_Item *it);
281 static void      _item_mode_unset(Widget_Data *wd);
282 static void      _item_move_after(Elm_Genlist_Item *it,
283                                   Elm_Genlist_Item *after);
284 static void      _item_move_before(Elm_Genlist_Item *it,
285                                    Elm_Genlist_Item *before);
286
287 // TODO : refactoring
288 static Evas_Object* _create_tray_alpha_bg(const Evas_Object *obj);
289 static unsigned int current_time_get();
290 static Eina_Bool _item_moving_effect_timer_cb(void *data);
291 static int _item_flip_effect_show(Elm_Genlist_Item *it);
292 static void _effect_item_controls(Elm_Genlist_Item *it, int itx, int ity);
293 static void _effect_item_realize(Elm_Genlist_Item *it, Eina_Bool effect_on);
294 static void _effect_item_unrealize(Elm_Genlist_Item *it);
295 static void      _item_auto_scroll(Widget_Data *wd);
296 static void      _pan_child_size_get(Evas_Object *obj,
297                                      Evas_Coord  *w,
298                                      Evas_Coord  *h);
299 static void      _elm_genlist_clear(Evas_Object *obj,
300                                     Eina_Bool    standby);
301 static void      _elm_genlist_item_del_serious(Elm_Genlist_Item *it);
302 static void      _elm_genlist_item_del_notserious(Elm_Genlist_Item *it);
303
304 static Evas_Smart_Class _pan_sc = EVAS_SMART_CLASS_INIT_VERSION;
305
306 static const char SIG_ACTIVATED[] = "activated";
307 static const char SIG_CLICKED_DOUBLE[] = "clicked,double";
308 static const char SIG_SELECTED[] = "selected";
309 static const char SIG_UNSELECTED[] = "unselected";
310 static const char SIG_EXPANDED[] = "expanded";
311 static const char SIG_CONTRACTED[] = "contracted";
312 static const char SIG_EXPAND_REQUEST[] = "expand,request";
313 static const char SIG_CONTRACT_REQUEST[] = "contract,request";
314 static const char SIG_REALIZED[] = "realized";
315 static const char SIG_UNREALIZED[] = "unrealized";
316 static const char SIG_DRAG_START_UP[] = "drag,start,up";
317 static const char SIG_DRAG_START_DOWN[] = "drag,start,down";
318 static const char SIG_DRAG_START_LEFT[] = "drag,start,left";
319 static const char SIG_DRAG_START_RIGHT[] = "drag,start,right";
320 static const char SIG_DRAG_STOP[] = "drag,stop";
321 static const char SIG_DRAG[] = "drag";
322 static const char SIG_LONGPRESSED[] = "longpressed";
323 static const char SIG_SCROLL_ANIM_START[] = "scroll,anim,start";
324 static const char SIG_SCROLL_ANIM_STOP[] = "scroll,anim,stop";
325 static const char SIG_SCROLL_DRAG_START[] = "scroll,drag,start";
326 static const char SIG_SCROLL_DRAG_STOP[] = "scroll,drag,stop";
327 static const char SIG_EDGE_TOP[] = "edge,top";
328 static const char SIG_EDGE_BOTTOM[] = "edge,bottom";
329 static const char SIG_EDGE_LEFT[] = "edge,left";
330 static const char SIG_EDGE_RIGHT[] = "edge,right";
331 static const char SIG_MULTI_SWIPE_LEFT[] = "multi,swipe,left";
332 static const char SIG_MULTI_SWIPE_RIGHT[] = "multi,swipe,right";
333 static const char SIG_MULTI_SWIPE_UP[] = "multi,swipe,up";
334 static const char SIG_MULTI_SWIPE_DOWN[] = "multi,swipe,down";
335 static const char SIG_MULTI_PINCH_OUT[] = "multi,pinch,out";
336 static const char SIG_MULTI_PINCH_IN[] = "multi,pinch,in";
337 static const char SIG_SWIPE[] = "swipe";
338 static const char SIG_INDEX_UPDATE[] = "index,update";
339
340 static const Evas_Smart_Cb_Description _signals[] = {
341    {SIG_CLICKED_DOUBLE, ""},
342    {SIG_ACTIVATED, ""},
343    {SIG_SELECTED, ""},
344    {SIG_UNSELECTED, ""},
345    {SIG_EXPANDED, ""},
346    {SIG_CONTRACTED, ""},
347    {SIG_EXPAND_REQUEST, ""},
348    {SIG_CONTRACT_REQUEST, ""},
349    {SIG_REALIZED, ""},
350    {SIG_UNREALIZED, ""},
351    {SIG_DRAG_START_UP, ""},
352    {SIG_DRAG_START_DOWN, ""},
353    {SIG_DRAG_START_LEFT, ""},
354    {SIG_DRAG_START_RIGHT, ""},
355    {SIG_DRAG_STOP, ""},
356    {SIG_DRAG, ""},
357    {SIG_LONGPRESSED, ""},
358    {SIG_SCROLL_ANIM_START, ""},
359    {SIG_SCROLL_ANIM_STOP, ""},
360    {SIG_SCROLL_DRAG_START, ""},
361    {SIG_SCROLL_DRAG_STOP, ""},
362    {SIG_EDGE_TOP, ""},
363    {SIG_EDGE_BOTTOM, ""},
364    {SIG_EDGE_LEFT, ""},
365    {SIG_EDGE_RIGHT, ""},
366    {SIG_MULTI_SWIPE_LEFT, ""},
367    {SIG_MULTI_SWIPE_RIGHT, ""},
368    {SIG_MULTI_SWIPE_UP, ""},
369    {SIG_MULTI_SWIPE_DOWN, ""},
370    {SIG_MULTI_PINCH_OUT, ""},
371    {SIG_MULTI_PINCH_IN, ""},
372    {SIG_SWIPE, ""},
373    {NULL, NULL}
374 };
375
376 static Eina_Bool
377 _event_hook(Evas_Object       *obj,
378             Evas_Object       *src __UNUSED__,
379             Evas_Callback_Type type,
380             void              *event_info)
381 {
382    if (type != EVAS_CALLBACK_KEY_DOWN) return EINA_FALSE;
383    Evas_Event_Key_Down *ev = event_info;
384    Widget_Data *wd = elm_widget_data_get(obj);
385    Evas_Coord pan_max_x = 0, pan_max_y = 0;
386    if (!wd) return EINA_FALSE;
387    if (!wd->items) return EINA_FALSE;
388    if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) return EINA_FALSE;
389    if (elm_widget_disabled_get(obj)) return EINA_FALSE;
390
391    Elm_Object_Item *it = NULL;
392    Evas_Coord x = 0;
393    Evas_Coord y = 0;
394    Evas_Coord step_x = 0;
395    Evas_Coord step_y = 0;
396    Evas_Coord v_w = 0;
397    Evas_Coord v_h = 0;
398    Evas_Coord page_x = 0;
399    Evas_Coord page_y = 0;
400
401    elm_smart_scroller_child_pos_get(wd->scr, &x, &y);
402    elm_smart_scroller_step_size_get(wd->scr, &step_x, &step_y);
403    elm_smart_scroller_page_size_get(wd->scr, &page_x, &page_y);
404    elm_smart_scroller_child_viewport_size_get(wd->scr, &v_w, &v_h);
405
406    if ((!strcmp(ev->keyname, "Left")) || (!strcmp(ev->keyname, "KP_Left")))
407      {
408         x -= step_x;
409      }
410    else if ((!strcmp(ev->keyname, "Right")) || (!strcmp(ev->keyname, "KP_Right")))
411      {
412         x += step_x;
413      }
414    else if ((!strcmp(ev->keyname, "Up")) || (!strcmp(ev->keyname, "KP_Up")))
415      {
416         if (((evas_key_modifier_is_set(ev->modifiers, "Shift")) &&
417              (_item_multi_select_up(wd)))
418             || (_item_single_select_up(wd)))
419           {
420              ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
421              return EINA_TRUE;
422           }
423         else
424           y -= step_y;
425      }
426    else if ((!strcmp(ev->keyname, "Down")) || (!strcmp(ev->keyname, "KP_Down")))
427      {
428         if (((evas_key_modifier_is_set(ev->modifiers, "Shift")) &&
429              (_item_multi_select_down(wd)))
430             || (_item_single_select_down(wd)))
431           {
432              ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
433              return EINA_TRUE;
434           }
435         else
436           y += step_y;
437      }
438    else if ((!strcmp(ev->keyname, "Home")) || (!strcmp(ev->keyname, "KP_Home")))
439      {
440         it = elm_genlist_first_item_get(obj);
441         elm_genlist_item_bring_in(it);
442         ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
443         return EINA_TRUE;
444      }
445    else if ((!strcmp(ev->keyname, "End")) || (!strcmp(ev->keyname, "KP_End")))
446      {
447         it = elm_genlist_last_item_get(obj);
448         elm_genlist_item_bring_in(it);
449         ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
450         return EINA_TRUE;
451      }
452    else if ((!strcmp(ev->keyname, "Prior")) || (!strcmp(ev->keyname, "KP_Prior")))
453      {
454         if (page_y < 0)
455           y -= -(page_y * v_h) / 100;
456         else
457           y -= page_y;
458      }
459    else if ((!strcmp(ev->keyname, "Next")) || (!strcmp(ev->keyname, "KP_Next")))
460      {
461         if (page_y < 0)
462           y += -(page_y * v_h) / 100;
463         else
464           y += page_y;
465      }
466    else if (!strcmp(ev->keyname, "Escape"))
467      {
468         if (!_deselect_all_items(wd)) return EINA_FALSE;
469         ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
470         return EINA_TRUE;
471      }
472    else if (((!strcmp(ev->keyname, "Return")) ||
473              (!strcmp(ev->keyname, "KP_Enter")) ||
474              (!strcmp(ev->keyname, "space")))
475             && (!wd->multi) && (wd->selected))
476      {
477         it = elm_genlist_selected_item_get(obj);
478         elm_genlist_item_expanded_set(it,
479                                       !elm_genlist_item_expanded_get(it));
480         evas_object_smart_callback_call(WIDGET(it), SIG_ACTIVATED, it);
481      }
482    else return EINA_FALSE;
483
484    ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
485    _pan_max_get(wd->pan_smart, &pan_max_x, &pan_max_y);
486    if (x < 0) x = 0;
487    if (x > pan_max_x) x = pan_max_x;
488    if (y < 0) y = 0;
489    if (y > pan_max_y) y = pan_max_y;
490    elm_smart_scroller_child_pos_set(wd->scr, x, y);
491    return EINA_TRUE;
492 }
493
494 static Eina_Bool
495 _deselect_all_items(Widget_Data *wd)
496 {
497    if (!wd->selected) return EINA_FALSE;
498    while (wd->selected)
499      elm_genlist_item_selected_set(wd->selected->data, EINA_FALSE);
500
501    return EINA_TRUE;
502 }
503
504 static Eina_Bool
505 _item_multi_select_up(Widget_Data *wd)
506 {
507    if (!wd->selected) return EINA_FALSE;
508    if (!wd->multi) return EINA_FALSE;
509
510    Elm_Object_Item *prev = elm_genlist_item_prev_get(wd->last_selected_item);
511    if (!prev) return EINA_TRUE;
512
513    if (elm_genlist_item_selected_get(prev))
514      {
515         elm_genlist_item_selected_set(wd->last_selected_item, EINA_FALSE);
516         wd->last_selected_item = prev;
517         elm_genlist_item_show(wd->last_selected_item);
518      }
519    else
520      {
521         elm_genlist_item_selected_set(prev, EINA_TRUE);
522         elm_genlist_item_show(prev);
523      }
524    return EINA_TRUE;
525 }
526
527 static Eina_Bool
528 _item_multi_select_down(Widget_Data *wd)
529 {
530    if (!wd->selected) return EINA_FALSE;
531    if (!wd->multi) return EINA_FALSE;
532
533    Elm_Object_Item *next;
534    next = elm_genlist_item_next_get(wd->last_selected_item);
535    if (!next) return EINA_TRUE;
536
537    if (elm_genlist_item_selected_get(next))
538      {
539         elm_genlist_item_selected_set(wd->last_selected_item, EINA_FALSE);
540         wd->last_selected_item = next;
541         elm_genlist_item_show(wd->last_selected_item);
542      }
543    else
544      {
545         elm_genlist_item_selected_set(next, EINA_TRUE);
546         elm_genlist_item_show(next);
547      }
548    return EINA_TRUE;
549 }
550
551 static Eina_Bool
552 _item_single_select_up(Widget_Data *wd)
553 {
554    Elm_Genlist_Item *prev;
555    if (!wd->selected)
556      {
557         prev = ELM_GENLIST_ITEM_FROM_INLIST(wd->items->last);
558         while ((prev) && (prev->delete_me))
559           prev = ELM_GENLIST_ITEM_FROM_INLIST(EINA_INLIST_GET(prev)->prev);
560      }
561    else prev = (Elm_Genlist_Item *)elm_genlist_item_prev_get(wd->last_selected_item);
562
563    if (!prev) return EINA_FALSE;
564
565    _deselect_all_items(wd);
566
567    elm_genlist_item_selected_set((Elm_Object_Item *)prev, EINA_TRUE);
568    elm_genlist_item_show((Elm_Object_Item *)prev);
569    return EINA_TRUE;
570 }
571
572 static Eina_Bool
573 _item_single_select_down(Widget_Data *wd)
574 {
575    Elm_Genlist_Item *next;
576    if (!wd->selected)
577      {
578         next = ELM_GENLIST_ITEM_FROM_INLIST(wd->items);
579         while ((next) && (next->delete_me))
580           next = ELM_GENLIST_ITEM_FROM_INLIST(EINA_INLIST_GET(next)->next);
581      }
582    else next = (Elm_Genlist_Item *) elm_genlist_item_next_get(wd->last_selected_item);
583
584    if (!next) return EINA_FALSE;
585
586    _deselect_all_items(wd);
587
588    elm_genlist_item_selected_set((Elm_Object_Item *) next, EINA_TRUE);
589    elm_genlist_item_show((Elm_Object_Item *) next);
590    return EINA_TRUE;
591 }
592
593 static void
594 _on_focus_hook(void        *data __UNUSED__,
595                Evas_Object *obj)
596 {
597    Widget_Data *wd = elm_widget_data_get(obj);
598    if (!wd) return;
599    if (elm_widget_focus_get(obj))
600      {
601         elm_object_signal_emit(wd->obj, "elm,action,focus", "elm");
602         evas_object_focus_set(wd->obj, EINA_TRUE);
603         if ((wd->items) && (wd->selected) && (!wd->last_selected_item))
604           wd->last_selected_item = eina_list_data_get(wd->selected);
605      }
606    else
607      {
608         elm_object_signal_emit(wd->obj, "elm,action,unfocus", "elm");
609         evas_object_focus_set(wd->obj, EINA_FALSE);
610      }
611 }
612
613 static void
614 _del_hook(Evas_Object *obj)
615 {
616    Widget_Data *wd = elm_widget_data_get(obj);
617    if (!wd) return;
618    _item_cache_zero(wd);
619    if (wd->calc_job) ecore_job_del(wd->calc_job);
620    if (wd->update_job) ecore_job_del(wd->update_job);
621    if (wd->queue_idle_enterer) ecore_idle_enterer_del(wd->queue_idle_enterer);
622    if (wd->changed_job) ecore_job_del(wd->changed_job);
623    if (wd->must_recalc_idler) ecore_idler_del(wd->must_recalc_idler);
624    if (wd->multi_timer) ecore_timer_del(wd->multi_timer);
625    if (wd->mode_type) eina_stringshare_del(wd->mode_type);
626    if (wd->scr_hold_timer) ecore_timer_del(wd->scr_hold_timer);
627    free(wd);
628 }
629
630 static void
631 _del_pre_hook(Evas_Object *obj)
632 {
633    Widget_Data *wd = elm_widget_data_get(obj);
634    if (!wd) return;
635    if (wd->edit_mode) elm_genlist_edit_mode_set(wd->obj, EINA_FALSE);
636    elm_genlist_clear(obj);
637    evas_object_del(wd->pan_smart);
638    wd->pan_smart = NULL;
639 }
640
641 static void
642 _mirrored_set(Evas_Object *obj,
643               Eina_Bool    rtl)
644 {
645    Widget_Data *wd = elm_widget_data_get(obj);
646    if (!wd) return;
647    _item_cache_zero(wd);
648    elm_smart_scroller_mirrored_set(wd->scr, rtl);
649 }
650
651 static void
652 _theme_hook(Evas_Object *obj)
653 {
654    Widget_Data *wd = elm_widget_data_get(obj);
655    Item_Block *itb;
656    if (!wd) return;
657    evas_event_freeze(evas_object_evas_get(wd->obj));
658    _item_cache_zero(wd);
659    _elm_widget_mirrored_reload(obj);
660    _mirrored_set(obj, elm_widget_mirrored_get(obj));
661    elm_smart_scroller_object_theme_set(obj, wd->scr, "genlist", "base",
662                                        elm_widget_style_get(obj));
663    //   edje_object_scale_set(wd->scr, elm_widget_scale_get(obj) * _elm_config->scale);
664    wd->item_width = wd->item_height = 0;
665    wd->group_item_width = wd->group_item_height = 0;
666    wd->minw = wd->minh = wd->realminw = 0;
667    EINA_INLIST_FOREACH(wd->blocks, itb)
668      {
669         Eina_List *l;
670         Elm_Genlist_Item *it;
671
672         if (itb->realized) _item_block_unrealize(itb);
673         EINA_LIST_FOREACH(itb->items, l, it)
674           it->mincalcd = EINA_FALSE;
675
676         itb->changed = EINA_TRUE;
677      }
678    if (wd->calc_job) ecore_job_del(wd->calc_job);
679    wd->calc_job = ecore_job_add(_calc_job, wd);
680    _sizing_eval(obj);
681    evas_event_thaw(evas_object_evas_get(wd->obj));
682    evas_event_thaw_eval(evas_object_evas_get(wd->obj));
683 }
684
685 static void
686 _show_region_hook(void        *data,
687                   Evas_Object *obj)
688 {
689    Widget_Data *wd = elm_widget_data_get(data);
690    Evas_Coord x, y, w, h;
691    if (!wd) return;
692    elm_widget_show_region_get(obj, &x, &y, &w, &h);
693    //x & y are screen coordinates, Add with pan coordinates
694    x += wd->pan_x;
695    y += wd->pan_y;
696    elm_smart_scroller_child_region_show(wd->scr, x, y, w, h);
697 }
698
699 static void
700 _translate_hook(Evas_Object *obj)
701 {
702    evas_object_smart_callback_call(obj, "language,changed", NULL);
703 }
704
705 static void
706 _sizing_eval(Evas_Object *obj)
707 {
708    Widget_Data *wd = elm_widget_data_get(obj);
709    Evas_Coord minw = -1, minh = -1, maxw = -1, maxh = -1;
710    if (!wd) return;
711    evas_object_size_hint_min_get(wd->scr, &minw, &minh);
712    evas_object_size_hint_max_get(wd->scr, &maxw, &maxh);
713    minh = -1;
714    if (wd->height_for_width)
715      {
716         Evas_Coord vw, vh;
717
718         elm_smart_scroller_child_viewport_size_get(wd->scr, &vw, &vh);
719         if ((vw != 0) && (vw != wd->prev_viewport_w))
720           {
721              Item_Block *itb;
722
723              wd->prev_viewport_w = vw;
724              EINA_INLIST_FOREACH(wd->blocks, itb)
725                {
726                   itb->must_recalc = EINA_TRUE;
727                }
728              if (wd->calc_job) ecore_job_del(wd->calc_job);
729              wd->calc_job = ecore_job_add(_calc_job, wd);
730           }
731      }
732    if (wd->mode == ELM_LIST_LIMIT)
733      {
734         Evas_Coord vmw, vmh;
735
736         minw = wd->realminw;
737         maxw = -1;
738         elm_smart_scroller_child_viewport_size_get(wd->scr, &vmw, &vmh);
739         edje_object_size_min_calc
740           (elm_smart_scroller_edje_object_get(wd->scr), &vmw, &vmh);
741         minw = vmw + minw;
742      }
743    else
744      {
745         Evas_Coord vmw, vmh;
746
747         edje_object_size_min_calc
748           (elm_smart_scroller_edje_object_get(wd->scr), &vmw, &vmh);
749         minw = vmw;
750         minh = vmh;
751      }
752    evas_object_size_hint_min_set(obj, minw, minh);
753    evas_object_size_hint_max_set(obj, maxw, maxh);
754 }
755
756 static void
757 _signal_emit_hook(Evas_Object *obj,
758                   const char  *emission,
759                   const char  *source)
760 {
761    Widget_Data *wd = elm_widget_data_get(obj);
762    edje_object_signal_emit(elm_smart_scroller_edje_object_get(wd->scr),
763                            emission, source);
764 }
765
766 static void
767 _item_highlight(Elm_Genlist_Item *it)
768 {
769    const char *selectraise;
770    if ((it->wd->no_select) || (it->no_select) || (it->delete_me) || (it->highlighted) ||
771        elm_widget_item_disabled_get(it) || (it->display_only))
772      return;
773    edje_object_signal_emit(VIEW(it), "elm,state,selected", "elm");
774    if (it->edit_obj) edje_object_signal_emit(it->edit_obj, "elm,state,selected", "elm");
775    selectraise = edje_object_data_get(VIEW(it), "selectraise");
776    if ((selectraise) && (!strcmp(selectraise, "on")))
777      {
778         if (it->edit_obj) evas_object_raise(it->edit_obj);
779         else evas_object_raise(VIEW(it));
780         if ((it->group_item) && (it->group_item->realized))
781           evas_object_raise(it->VIEW(group_item));
782      }
783    it->highlighted = EINA_TRUE;
784 }
785
786 static void
787 _item_unhighlight(Elm_Genlist_Item *it)
788 {
789    if ((it->delete_me) || (!it->highlighted)) return;
790    edje_object_signal_emit(VIEW(it), "elm,state,unselected", "elm");
791    if (it->edit_obj) edje_object_signal_emit(it->edit_obj, "elm,state,unselected", "elm");
792    if (!it->nostacking)
793      {
794        if ((it->order_num_in & 0x1) ^ it->stacking_even) evas_object_lower(VIEW(it));
795        else evas_object_raise(VIEW(it));
796      }
797    it->highlighted = EINA_FALSE;
798 }
799
800 static void
801 _item_block_position_update(Eina_Inlist *list, int idx)
802 {
803    Item_Block *tmp;
804
805    EINA_INLIST_FOREACH(list, tmp)
806      {
807         tmp->position = idx++;
808         tmp->position_update = EINA_TRUE;
809      }
810 }
811
812 static void
813 _item_position_update(Eina_List *list, int idx)
814 {
815    Elm_Genlist_Item *it;
816    Eina_List *l;
817
818    EINA_LIST_FOREACH(list, l, it)
819      {
820         it->position = idx++;
821         it->position_update = EINA_TRUE;
822      }
823 }
824
825 static void
826 _item_block_del(Elm_Genlist_Item *it)
827 {
828    Eina_Inlist *il;
829    Item_Block *itb = it->block;
830    Eina_Bool block_changed;
831
832    itb->items = eina_list_remove(itb->items, it);
833    itb->count--;
834    itb->changed = EINA_TRUE;
835    if (!it->wd->reorder_deleted)
836      {
837         if (it->wd->calc_job) ecore_job_del(it->wd->calc_job);
838         it->wd->calc_job = ecore_job_add(_calc_job, it->wd);
839      }
840    if (itb->count < 1)
841      {
842         il = EINA_INLIST_GET(itb);
843         Item_Block *itbn = (Item_Block *)(il->next);
844         if (it->parent)
845           it->parent->items = eina_list_remove(it->parent->items, it);
846         else
847           {
848              _item_block_position_update(il->next, itb->position);
849              it->wd->blocks = eina_inlist_remove(it->wd->blocks, il);
850           }
851         free(itb);
852         if (itbn) itbn->changed = EINA_TRUE;
853      }
854    else
855      {
856         if (itb->count < itb->wd->max_items_per_block/2)
857           {
858              il = EINA_INLIST_GET(itb);
859              Item_Block *itbp = (Item_Block *)(il->prev);
860              Item_Block *itbn = (Item_Block *)(il->next);
861              if ((itbp) && ((itbp->count + itb->count) < itb->wd->max_items_per_block + itb->wd->max_items_per_block/2))
862                {
863                   Elm_Genlist_Item *it2;
864
865                   EINA_LIST_FREE(itb->items, it2)
866                     {
867                        it2->block = itbp;
868                        itbp->items = eina_list_append(itbp->items, it2);
869                        itbp->count++;
870                        itbp->changed = EINA_TRUE;
871                     }
872                   _item_block_position_update(EINA_INLIST_GET(itb)->next,
873                                               itb->position);
874                   it->wd->blocks = eina_inlist_remove(it->wd->blocks,
875                                                       EINA_INLIST_GET(itb));
876                   free(itb);
877                   block_changed = EINA_TRUE;
878                }
879              else if ((itbn) && ((itbn->count + itb->count) < itb->wd->max_items_per_block + itb->wd->max_items_per_block/2))
880                {
881                   while (itb->items)
882                     {
883                        Eina_List *last = eina_list_last(itb->items);
884                        Elm_Genlist_Item *it2 = last->data;
885
886                        it2->block = itbn;
887                        itb->items = eina_list_remove_list(itb->items, last);
888                        itbn->items = eina_list_prepend(itbn->items, it2);
889                        itbn->count++;
890                        itbn->changed = EINA_TRUE;
891                     }
892                   _item_block_position_update(EINA_INLIST_GET(itb)->next,
893                                               itb->position);
894                   it->wd->blocks =
895                     eina_inlist_remove(it->wd->blocks, EINA_INLIST_GET(itb));
896                   free(itb);
897                   block_changed = EINA_TRUE;
898                }
899           }
900      }
901    if (block_changed)
902      {
903         it->wd->pan_changed = EINA_TRUE;
904         evas_object_smart_changed(it->wd->pan_smart);
905         if (it->wd->calc_job) ecore_job_del(it->wd->calc_job);
906         it->wd->calc_job = NULL;
907      }
908 }
909
910 static void
911 _item_subitems_clear(Elm_Genlist_Item *it)
912 {
913    if (!it) return;
914    Eina_List *tl = NULL, *l;
915    Elm_Object_Item *it2;
916
917    EINA_LIST_FOREACH(it->items, l, it2)
918       tl = eina_list_append(tl, it2);
919
920    EINA_LIST_FREE(tl, it2)
921      elm_object_item_del(it2);
922 }
923
924 static void
925 _item_del(Elm_Genlist_Item *it)
926 {
927    Evas_Object *obj = WIDGET(it);
928
929    evas_event_freeze(evas_object_evas_get(obj));
930    elm_genlist_item_subitems_clear((Elm_Object_Item *) it);
931    if (it->wd->show_item == it) it->wd->show_item = NULL;
932    if (it->realized) _item_unrealize(it, EINA_FALSE);
933    if (it->block) _item_block_del(it);
934    if (it->effect_item_realized) _effect_item_unrealize(it);
935    if (it->queued)
936      it->wd->queue = eina_list_remove(it->wd->queue, it);
937 #ifdef ANCHOR_ITEM
938    if (it->wd->anchor_item == it)
939      {
940         it->wd->anchor_item = ELM_GENLIST_ITEM_FROM_INLIST(EINA_INLIST_GET(it)->next);
941         if (!it->wd->anchor_item)
942           it->wd->anchor_item = ELM_GENLIST_ITEM_FROM_INLIST(EINA_INLIST_GET(it)->prev);
943      }
944 #endif
945    if (it->parent)
946      it->parent->items = eina_list_remove(it->parent->items, it);
947    if (it->swipe_timer) ecore_timer_del(it->swipe_timer);
948    _elm_genlist_item_del_serious(it);
949    evas_event_thaw(evas_object_evas_get(obj));
950    evas_event_thaw_eval(evas_object_evas_get(obj));
951 }
952
953 static void
954 _item_select(Elm_Genlist_Item *it)
955 {
956    Eina_List *l;
957    Evas_Object *obj;
958
959    if ((it->wd->no_select) || (it->no_select) || (it->delete_me)) return;
960    if (!it->selected)
961      {
962         it->selected = EINA_TRUE;
963         it->wd->selected = eina_list_append(it->wd->selected, it);
964      }
965    else if (!it->wd->always_select) return;
966
967    evas_object_ref(WIDGET(it));
968    it->walking++;
969    it->wd->walking++;
970    if (it->wd->last_selected_item &&
971        (it != (Elm_Genlist_Item *) it->wd->last_selected_item))
972      {
973         EINA_LIST_FOREACH(((Elm_Genlist_Item *)it->wd->last_selected_item)->content_objs, l, obj)
974           {
975              elm_widget_focused_object_clear(obj);
976              elm_widget_tree_unfocusable_set(obj, EINA_TRUE);
977           }
978         ((Elm_Genlist_Item *)it->wd->last_selected_item)->can_focus = EINA_FALSE;
979      }
980    if (it->func.func) it->func.func((void *)it->func.data, WIDGET(it), it);
981    if (!it->delete_me)
982      evas_object_smart_callback_call(WIDGET(it), SIG_SELECTED, it);
983    it->walking--;
984    it->wd->walking--;
985    evas_object_unref(WIDGET(it));
986    if ((it->wd->clear_me) && (!it->wd->walking))
987      _elm_genlist_clear(WIDGET(it), EINA_TRUE);
988    else
989      {
990         if ((!it->walking) && (it->delete_me))
991           {
992              if (!it->relcount)
993                {
994                   _item_del(it);
995                   elm_widget_item_free(it);
996                }
997           }
998         else
999           it->wd->last_selected_item = (Elm_Object_Item *) it;
1000      }
1001 }
1002
1003 static void
1004 _item_unselect(Elm_Genlist_Item *it)
1005 {
1006    if ((it->delete_me) || (!it->selected)) return;
1007    it->selected = EINA_FALSE;
1008    it->wd->selected = eina_list_remove(it->wd->selected, it);
1009    evas_object_smart_callback_call(WIDGET(it), SIG_UNSELECTED, it);
1010 }
1011
1012 static void
1013 _mouse_move(void        *data,
1014             Evas        *evas __UNUSED__,
1015             Evas_Object *obj,
1016             void        *event_info)
1017 {
1018    Elm_Genlist_Item *it = data;
1019    Evas_Event_Mouse_Move *ev = event_info;
1020    Evas_Coord minw = 0, minh = 0, x, y, dx, dy, adx, ady;
1021    Evas_Coord ox, oy, ow, oh, it_scrl_y, y_pos;
1022
1023    if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD)
1024      {
1025         if (!it->wd->on_hold)
1026           {
1027              it->wd->on_hold = EINA_TRUE;
1028              if ((!it->wd->wasselected) && (!it->renamed))
1029                {
1030                   _item_unhighlight(it);
1031                   _item_unselect(it);
1032                }
1033           }
1034      }
1035    if (it->wd->multitouched)
1036      {
1037         it->wd->cur_x = ev->cur.canvas.x;
1038         it->wd->cur_y = ev->cur.canvas.y;
1039         return;
1040      }
1041    if ((it->dragging) && (it->down))
1042      {
1043         if (it->wd->movements == SWIPE_MOVES) it->wd->swipe = EINA_TRUE;
1044         else
1045           {
1046              it->wd->history[it->wd->movements].x = ev->cur.canvas.x;
1047              it->wd->history[it->wd->movements].y = ev->cur.canvas.y;
1048              if (abs((it->wd->history[it->wd->movements].x -
1049                       it->wd->history[0].x)) > 40)
1050                it->wd->swipe = EINA_TRUE;
1051              else
1052                it->wd->movements++;
1053           }
1054         if (it->long_timer)
1055           {
1056              ecore_timer_del(it->long_timer);
1057              it->long_timer = NULL;
1058           }
1059         evas_object_smart_callback_call(WIDGET(it), SIG_DRAG, it);
1060         return;
1061      }
1062    if ((!it->down) /* || (it->wd->on_hold)*/ || (it->wd->longpressed))
1063      {
1064         if (it->long_timer)
1065           {
1066              ecore_timer_del(it->long_timer);
1067              it->long_timer = NULL;
1068           }
1069         if ((it->wd->reorder_mode) && (it->wd->reorder_it))
1070           {
1071              evas_object_geometry_get(it->wd->pan_smart, &ox, &oy, &ow, &oh);
1072              it_scrl_y = ev->cur.canvas.y - it->wd->reorder_it->dy;
1073
1074              if (!it->wd->reorder_start_y)
1075                it->wd->reorder_start_y = it->block->y + it->y;
1076
1077              if (it_scrl_y < oy)
1078                y_pos = oy;
1079              else if (it_scrl_y + it->wd->reorder_it->h > oy + oh)
1080                y_pos = oy + oh - it->wd->reorder_it->h;
1081              else
1082                y_pos = it_scrl_y;
1083
1084              if (it->edit_obj)
1085                _item_position(it, it->edit_obj, it->scrl_x, y_pos);
1086              else
1087                _item_position(it, VIEW(it), it->scrl_x, y_pos);
1088
1089              if (it->wd->calc_job) ecore_job_del(it->wd->calc_job);
1090              it->wd->calc_job = ecore_job_add(_calc_job, it->wd);
1091           }
1092         return;
1093      }
1094    if (!it->display_only)
1095      elm_coords_finger_size_adjust(1, &minw, 1, &minh);
1096    evas_object_geometry_get(obj, &x, &y, NULL, NULL);
1097    x = ev->cur.canvas.x - x;
1098    y = ev->cur.canvas.y - y;
1099    dx = x - it->dx;
1100    adx = dx;
1101    if (adx < 0) adx = -dx;
1102    dy = y - it->dy;
1103    ady = dy;
1104    if (ady < 0) ady = -dy;
1105    minw /= 2;
1106    minh /= 2;
1107    if ((adx > minw) || (ady > minh))
1108      {
1109         it->dragging = EINA_TRUE;
1110         if (it->long_timer)
1111           {
1112              ecore_timer_del(it->long_timer);
1113              it->long_timer = NULL;
1114           }
1115         if ((!it->wd->wasselected) && (!it->renamed))
1116           {
1117              _item_unhighlight(it);
1118              _item_unselect(it);
1119           }
1120         if (dy < 0)
1121           {
1122              if ((ady > adx) && (!it->wd->drag_started))
1123                evas_object_smart_callback_call(WIDGET(it),
1124                                                SIG_DRAG_START_UP, it);
1125              else
1126                {
1127                   if (dx < 0)
1128                     evas_object_smart_callback_call(WIDGET(it),
1129                                                     SIG_DRAG_START_LEFT, it);
1130                   else
1131                     evas_object_smart_callback_call(WIDGET(it),
1132                                                     SIG_DRAG_START_RIGHT, it);
1133                }
1134           }
1135         else
1136           {
1137              if ((ady > adx) && (!it->wd->drag_started))
1138                evas_object_smart_callback_call(WIDGET(it),
1139                                                SIG_DRAG_START_DOWN, it);
1140              else
1141                {
1142                   if (dx < 0)
1143                     evas_object_smart_callback_call(WIDGET(it),
1144                                                     SIG_DRAG_START_LEFT, it);
1145                   else
1146                     evas_object_smart_callback_call(WIDGET(it),
1147                                                     SIG_DRAG_START_RIGHT, it);
1148                }
1149           }
1150      }
1151 }
1152
1153 static Eina_Bool
1154 _long_press(void *data)
1155 {
1156    Elm_Genlist_Item *it = data, *it_tmp;
1157    Eina_List *list, *l;
1158
1159    it->long_timer = NULL;
1160    if (elm_widget_item_disabled_get(it) || (it->dragging) || (it->display_only) || (it->wd->rename_it))
1161      return ECORE_CALLBACK_CANCEL;
1162    it->wd->longpressed = EINA_TRUE;
1163    evas_object_smart_callback_call(WIDGET(it), SIG_LONGPRESSED, it);
1164    if ((it->wd->reorder_mode) && (it->flags != ELM_GENLIST_ITEM_GROUP))
1165      {
1166         it->wd->reorder_it = it;
1167         it->wd->reorder_start_y = 0;
1168         if (it->edit_obj)
1169           evas_object_raise(it->edit_obj);
1170         else
1171           evas_object_raise(VIEW(it));
1172
1173         elm_smart_scroller_hold_set(it->wd->scr, EINA_TRUE);
1174         elm_smart_scroller_bounce_allow_set(it->wd->scr, EINA_FALSE, EINA_FALSE);
1175
1176         list = elm_genlist_realized_items_get(it->wd->obj);
1177         EINA_LIST_FOREACH(list, l, it_tmp)
1178           {
1179              if (it != it_tmp) _item_unselect(it_tmp);
1180           }
1181         if (it->items)
1182           {
1183              EINA_LIST_FOREACH(it->items, l, it_tmp)
1184                {
1185                   if (elm_genlist_item_expanded_get((Elm_Object_Item *) it_tmp))
1186                     {
1187                        elm_genlist_item_expanded_set((Elm_Object_Item *)it_tmp,
1188                                                      EINA_FALSE);
1189                        return ECORE_CALLBACK_RENEW;
1190                     }
1191                }
1192           }
1193         if (elm_genlist_item_expanded_get((Elm_Object_Item *) it))
1194           {
1195              elm_genlist_item_expanded_set((Elm_Object_Item *) it, EINA_FALSE);
1196              return ECORE_CALLBACK_RENEW;
1197           }
1198         if (it->edit_obj)
1199           edje_object_signal_emit(it->edit_obj, "elm,action,item,reorder_start", "elm");
1200         else
1201           edje_object_signal_emit(VIEW(it), "elm,state,reorder,enabled", "elm");
1202      }
1203    return ECORE_CALLBACK_CANCEL;
1204 }
1205
1206 static void
1207 _swipe(Elm_Genlist_Item *it)
1208 {
1209    int i, sum = 0;
1210
1211    if (!it) return;
1212    if ((it->display_only) || elm_widget_item_disabled_get(it)) return;
1213    it->wd->swipe = EINA_FALSE;
1214    for (i = 0; i < it->wd->movements; i++)
1215      {
1216         sum += it->wd->history[i].x;
1217         if (abs(it->wd->history[0].y - it->wd->history[i].y) > 10) return;
1218      }
1219
1220    sum /= it->wd->movements;
1221    if (abs(sum - it->wd->history[0].x) <= 10) return;
1222    evas_object_smart_callback_call(WIDGET(it), SIG_SWIPE, it);
1223 }
1224
1225 static Eina_Bool
1226 _swipe_cancel(void *data)
1227 {
1228    Elm_Genlist_Item *it = data;
1229
1230    if (!it) return ECORE_CALLBACK_CANCEL;
1231    it->wd->swipe = EINA_FALSE;
1232    it->wd->movements = 0;
1233    return ECORE_CALLBACK_RENEW;
1234 }
1235
1236 static Eina_Bool
1237 _multi_cancel(void *data)
1238 {
1239    Widget_Data *wd = data;
1240
1241    if (!wd) return ECORE_CALLBACK_CANCEL;
1242    wd->multi_timeout = EINA_TRUE;
1243    return ECORE_CALLBACK_RENEW;
1244 }
1245
1246 static void
1247 _multi_touch_gesture_eval(void *data)
1248 {
1249    Elm_Genlist_Item *it = data;
1250
1251    it->wd->multitouched = EINA_FALSE;
1252    if (it->wd->multi_timer)
1253      {
1254         ecore_timer_del(it->wd->multi_timer);
1255         it->wd->multi_timer = NULL;
1256      }
1257    if (it->wd->multi_timeout)
1258      {
1259         it->wd->multi_timeout = EINA_FALSE;
1260         return;
1261      }
1262
1263    Evas_Coord minw = 0, minh = 0;
1264    Evas_Coord off_x, off_y, off_mx, off_my;
1265
1266    elm_coords_finger_size_adjust(1, &minw, 1, &minh);
1267    off_x = abs(it->wd->cur_x - it->wd->prev_x);
1268    off_y = abs(it->wd->cur_y - it->wd->prev_y);
1269    off_mx = abs(it->wd->cur_mx - it->wd->prev_mx);
1270    off_my = abs(it->wd->cur_my - it->wd->prev_my);
1271
1272    if (((off_x > minw) || (off_y > minh)) && ((off_mx > minw) || (off_my > minh)))
1273      {
1274         if ((off_x + off_mx) > (off_y + off_my))
1275           {
1276              if ((it->wd->cur_x > it->wd->prev_x) && (it->wd->cur_mx > it->wd->prev_mx))
1277                evas_object_smart_callback_call(WIDGET(it),
1278                                                SIG_MULTI_SWIPE_RIGHT, it);
1279              else if ((it->wd->cur_x < it->wd->prev_x) && (it->wd->cur_mx < it->wd->prev_mx))
1280                evas_object_smart_callback_call(WIDGET(it),
1281                                                SIG_MULTI_SWIPE_LEFT, it);
1282              else if (abs(it->wd->cur_x - it->wd->cur_mx) > abs(it->wd->prev_x - it->wd->prev_mx))
1283                evas_object_smart_callback_call(WIDGET(it),
1284                                                SIG_MULTI_PINCH_OUT, it);
1285              else
1286                evas_object_smart_callback_call(WIDGET(it),
1287                                                SIG_MULTI_PINCH_IN, it);
1288           }
1289         else
1290           {
1291              if ((it->wd->cur_y > it->wd->prev_y) && (it->wd->cur_my > it->wd->prev_my))
1292                evas_object_smart_callback_call(WIDGET(it),
1293                                                SIG_MULTI_SWIPE_DOWN, it);
1294              else if ((it->wd->cur_y < it->wd->prev_y) && (it->wd->cur_my < it->wd->prev_my))
1295                evas_object_smart_callback_call(WIDGET(it),
1296                                                SIG_MULTI_SWIPE_UP, it);
1297              else if (abs(it->wd->cur_y - it->wd->cur_my) > abs(it->wd->prev_y - it->wd->prev_my))
1298                evas_object_smart_callback_call(WIDGET(it),
1299                                                SIG_MULTI_PINCH_OUT, it);
1300              else
1301                evas_object_smart_callback_call(WIDGET(it),
1302                                                SIG_MULTI_PINCH_IN, it);
1303           }
1304      }
1305    it->wd->multi_timeout = EINA_FALSE;
1306 }
1307
1308 static void
1309 _multi_down(void        *data,
1310             Evas        *evas __UNUSED__,
1311             Evas_Object *obj __UNUSED__,
1312             void        *event_info)
1313 {
1314    Elm_Genlist_Item *it = data;
1315    Evas_Event_Multi_Down *ev = event_info;
1316
1317    if ((it->wd->multi_device != 0) || (it->wd->multitouched) || (it->wd->multi_timeout)) return;
1318    it->wd->multi_device = ev->device;
1319    it->wd->multi_down = EINA_TRUE;
1320    it->wd->multitouched = EINA_TRUE;
1321    it->wd->prev_mx = ev->canvas.x;
1322    it->wd->prev_my = ev->canvas.y;
1323    if (!it->wd->wasselected)
1324      {
1325         _item_unhighlight(it);
1326         _item_unselect(it);
1327      }
1328    it->wd->wasselected = EINA_FALSE;
1329    it->wd->longpressed = EINA_FALSE;
1330    if (it->long_timer)
1331      {
1332         ecore_timer_del(it->long_timer);
1333         it->long_timer = NULL;
1334      }
1335    if (it->dragging)
1336      {
1337         it->dragging = EINA_FALSE;
1338         evas_object_smart_callback_call(WIDGET(it), SIG_DRAG_STOP, it);
1339      }
1340    if (it->swipe_timer)
1341      {
1342         ecore_timer_del(it->swipe_timer);
1343         it->swipe_timer = NULL;
1344      }
1345    if (it->wd->on_hold)
1346      {
1347         it->wd->swipe = EINA_FALSE;
1348         it->wd->movements = 0;
1349         it->wd->on_hold = EINA_FALSE;
1350      }
1351 }
1352
1353 static void
1354 _multi_up(void        *data,
1355           Evas        *evas __UNUSED__,
1356           Evas_Object *obj __UNUSED__,
1357           void        *event_info)
1358 {
1359    Elm_Genlist_Item *it = data;
1360    Evas_Event_Multi_Up *ev = event_info;
1361
1362    if (it->wd->multi_device != ev->device) return;
1363    it->wd->multi_device = 0;
1364    it->wd->multi_down = EINA_FALSE;
1365    if (it->wd->mouse_down) return;
1366    _multi_touch_gesture_eval(data);
1367 }
1368
1369 static void
1370 _multi_move(void        *data,
1371             Evas        *evas __UNUSED__,
1372             Evas_Object *obj __UNUSED__,
1373             void        *event_info)
1374 {
1375    Elm_Genlist_Item *it = data;
1376    Evas_Event_Multi_Move *ev = event_info;
1377
1378    if (it->wd->multi_device != ev->device) return;
1379    it->wd->cur_mx = ev->cur.canvas.x;
1380    it->wd->cur_my = ev->cur.canvas.y;
1381 }
1382
1383 static void
1384 _mouse_down(void        *data,
1385             Evas        *evas __UNUSED__,
1386             Evas_Object *obj,
1387             void        *event_info)
1388 {
1389    Elm_Genlist_Item *it = data;
1390    Evas_Event_Mouse_Down *ev = event_info;
1391    Evas_Coord x, y;
1392    Eina_List *l;
1393    Evas_Object *iobj;
1394
1395    if (ev->button != 1) return;
1396    if (!it->can_focus)
1397      {
1398         EINA_LIST_FOREACH(it->content_objs, l, iobj)
1399           {
1400              elm_widget_tree_unfocusable_set(iobj, EINA_FALSE);
1401           }
1402         it->can_focus = EINA_TRUE;
1403      }
1404    if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD)
1405      {
1406         it->wd->on_hold = EINA_TRUE;
1407      }
1408
1409    it->down = EINA_TRUE;
1410    it->dragging = EINA_FALSE;
1411    evas_object_geometry_get(obj, &x, &y, NULL, NULL);
1412    it->dx = ev->canvas.x - x;
1413    it->dy = ev->canvas.y - y;
1414    it->wd->mouse_down = EINA_TRUE;
1415    if (!it->wd->multitouched)
1416      {
1417         it->wd->prev_x = ev->canvas.x;
1418         it->wd->prev_y = ev->canvas.y;
1419         it->wd->multi_timeout = EINA_FALSE;
1420         if (it->wd->multi_timer) ecore_timer_del(it->wd->multi_timer);
1421         it->wd->multi_timer = ecore_timer_add(1, _multi_cancel, it->wd);
1422      }
1423    it->wd->longpressed = EINA_FALSE;
1424    if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) it->wd->on_hold = EINA_TRUE;
1425    else it->wd->on_hold = EINA_FALSE;
1426    if (it->wd->on_hold) return;
1427    it->wd->wasselected = it->selected;
1428    _item_highlight(it);
1429    if (ev->flags & EVAS_BUTTON_DOUBLE_CLICK)
1430      if ((!elm_widget_item_disabled_get(it)) && (!it->display_only))
1431        {
1432           evas_object_smart_callback_call(WIDGET(it), SIG_CLICKED_DOUBLE, it);
1433           evas_object_smart_callback_call(WIDGET(it), SIG_ACTIVATED, it);
1434        }
1435    if (it->long_timer) ecore_timer_del(it->long_timer);
1436    if (it->swipe_timer) ecore_timer_del(it->swipe_timer);
1437    it->swipe_timer = ecore_timer_add(0.4, _swipe_cancel, it);
1438    if (it->realized)
1439      it->long_timer = ecore_timer_add(it->wd->longpress_timeout, _long_press,
1440                                       it);
1441    else
1442      it->long_timer = NULL;
1443    it->wd->swipe = EINA_FALSE;
1444    it->wd->movements = 0;
1445 }
1446
1447 static void
1448 _mouse_up(void        *data,
1449           Evas        *evas __UNUSED__,
1450           Evas_Object *obj __UNUSED__,
1451           void        *event_info)
1452 {
1453    Elm_Genlist_Item *it = data;
1454    Evas_Event_Mouse_Up *ev = event_info;
1455    Eina_Bool dragged = EINA_FALSE;
1456
1457    if (ev->button != 1) return;
1458    it->down = EINA_FALSE;
1459    it->wd->mouse_down = EINA_FALSE;
1460    if (it->wd->multitouched)
1461      {
1462         // in single selection modes, some multi touching can make multi highlighted items.
1463         // if a item not selected and highlighted, it should be unhighlighted.
1464         if ((!it->wd->multi) && (!it->selected) && (it->highlighted)) _item_unhighlight(it);
1465         if (it->wd->multi_down) return;
1466         _multi_touch_gesture_eval(data);
1467         return;
1468      }
1469    if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) it->wd->on_hold = EINA_TRUE;
1470    else it->wd->on_hold = EINA_FALSE;
1471    if (it->long_timer)
1472      {
1473         ecore_timer_del(it->long_timer);
1474         it->long_timer = NULL;
1475      }
1476    if (it->dragging)
1477      {
1478         it->dragging = EINA_FALSE;
1479         evas_object_smart_callback_call(WIDGET(it), SIG_DRAG_STOP, it);
1480         dragged = 1;
1481      }
1482    if (it->swipe_timer)
1483      {
1484         ecore_timer_del(it->swipe_timer);
1485         it->swipe_timer = NULL;
1486      }
1487    if (it->wd->multi_timer)
1488      {
1489         ecore_timer_del(it->wd->multi_timer);
1490         it->wd->multi_timer = NULL;
1491         it->wd->multi_timeout = EINA_FALSE;
1492      }
1493    if (it->wd->on_hold)
1494      {
1495         if (it->wd->swipe) _swipe(data);
1496         it->wd->longpressed = EINA_FALSE;
1497         it->wd->on_hold = EINA_FALSE;
1498         return;
1499      }
1500    if ((it->wd->reorder_mode) && (it->wd->reorder_it))
1501      {
1502         Evas_Coord it_scrl_y = ev->canvas.y - it->wd->reorder_it->dy;
1503
1504         if (it->wd->reorder_rel && (it->wd->reorder_it->parent == it->wd->reorder_rel->parent))
1505           {
1506              if (it_scrl_y <= it->wd->reorder_rel->scrl_y)
1507                _item_move_before(it->wd->reorder_it, it->wd->reorder_rel);
1508              else
1509                _item_move_after(it->wd->reorder_it, it->wd->reorder_rel);
1510           }
1511         else
1512           {
1513              if (it->wd->calc_job) ecore_job_del(it->wd->calc_job);
1514              it->wd->calc_job = ecore_job_add(_calc_job, it->wd);
1515           }
1516         it->wd->reorder_deleted = EINA_FALSE;
1517         edje_object_signal_emit(it->edit_obj, "elm,action,item,reorder_end", "elm");
1518         it->wd->reorder_it = it->wd->reorder_rel = NULL;
1519         elm_smart_scroller_hold_set(it->wd->scr, EINA_FALSE);
1520         elm_smart_scroller_bounce_allow_set(it->wd->scr, EINA_FALSE, EINA_TRUE);
1521
1522      }
1523    if (it->wd->longpressed)
1524      {
1525         it->wd->longpressed = EINA_FALSE;
1526         if ((!it->wd->wasselected) && (!it->renamed))
1527           {
1528              _item_unhighlight(it);
1529              _item_unselect(it);
1530           }
1531         it->wd->wasselected = EINA_FALSE;
1532         return;
1533      }
1534    if (dragged)
1535      {
1536         if (it->want_unrealize)
1537           {
1538              _item_unrealize(it, EINA_FALSE);
1539              if (it->block->want_unrealize)
1540                _item_block_unrealize(it->block);
1541           }
1542      }
1543    if (elm_widget_item_disabled_get(it) || (dragged) || (it->display_only))
1544      return;
1545    if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) return;
1546    if (it->wd->multi)
1547      {
1548         if (!it->selected)
1549           {
1550              _item_highlight(it);
1551              _item_select(it);
1552           }
1553         else
1554           {
1555              _item_unhighlight(it);
1556              _item_unselect(it);
1557           }
1558      }
1559    else
1560      {
1561         if (!it->selected)
1562           {
1563              Widget_Data *wd = it->wd;
1564              if (wd)
1565                {
1566                   while (wd->selected)
1567                     {
1568                        _item_unhighlight(wd->selected->data);
1569                        _item_unselect(wd->selected->data);
1570                     }
1571                }
1572           }
1573         else
1574           {
1575              const Eina_List *l, *l_next;
1576              Elm_Genlist_Item *it2;
1577
1578              EINA_LIST_FOREACH_SAFE(it->wd->selected, l, l_next, it2)
1579                if (it2 != it)
1580                   {
1581                      _item_unhighlight(it2);
1582                      _item_unselect(it2);
1583                   }
1584              //_item_highlight(it);
1585              //_item_select(it);
1586           }
1587         _item_highlight(it);
1588         _item_select(it);
1589      }
1590 }
1591
1592 static void
1593 _mouse_down_scroller(void        *data,
1594                      Evas        *evas __UNUSED__,
1595                      Evas_Object *obj __UNUSED__,
1596                      void        *event_info __UNUSED__)
1597 {
1598    Widget_Data *wd = elm_widget_data_get(data);
1599
1600    if (!wd) return;
1601    wd->drag_started = EINA_FALSE;
1602 }
1603
1604 static void
1605 _mouse_up_scroller(void        *data,
1606                    Evas        *evas __UNUSED__,
1607                    Evas_Object *obj __UNUSED__,
1608                    void        *event_info __UNUSED__)
1609 {
1610    Widget_Data *wd = elm_widget_data_get(data);
1611
1612    if (!wd) return;
1613    wd->drag_started = EINA_FALSE;
1614 }
1615
1616 static void
1617 _mouse_move_scroller(void        *data,
1618                      Evas        *evas __UNUSED__,
1619                      Evas_Object *obj __UNUSED__,
1620                      void        *event_info)
1621 {
1622    Widget_Data *wd = elm_widget_data_get(data);
1623    Evas_Event_Mouse_Move *ev = event_info;
1624    Evas_Coord minw = 0, minh = 0, dx, dy, adx, ady;
1625
1626    if (!wd) return;
1627    if (wd->drag_started) return;
1628
1629    elm_coords_finger_size_adjust(1, &minw, 1, &minh);
1630    dx = ev->cur.canvas.x - ev->prev.canvas.x;
1631    dy = ev->cur.canvas.y - ev->prev.canvas.y;
1632    adx = dx;
1633    ady = dy;
1634    if (adx < 0) adx = -dx;
1635    if (ady < 0) ady = -dy;
1636    if (((ady < minh) && (ady > minh / 2)) && (ady > adx))
1637      {
1638         if (dy < 0)
1639           {
1640              evas_object_smart_callback_call(data, SIG_DRAG_START_UP, NULL);
1641              wd->drag_started = EINA_TRUE;
1642           }
1643         else
1644           {
1645              evas_object_smart_callback_call(data, SIG_DRAG_START_DOWN, NULL);
1646              wd->drag_started = EINA_TRUE;
1647           }
1648      }
1649 }
1650
1651 static void
1652 _signal_expand_toggle(void        *data,
1653                       Evas_Object *obj __UNUSED__,
1654                       const char  *emission __UNUSED__,
1655                       const char  *source __UNUSED__)
1656 {
1657    Elm_Genlist_Item *it = data;
1658
1659    if (it->expanded)
1660      evas_object_smart_callback_call(WIDGET(it), SIG_CONTRACT_REQUEST, it);
1661    else
1662      evas_object_smart_callback_call(WIDGET(it), SIG_EXPAND_REQUEST, it);
1663 }
1664
1665 static void
1666 _signal_expand(void        *data,
1667                Evas_Object *obj __UNUSED__,
1668                const char  *emission __UNUSED__,
1669                const char  *source __UNUSED__)
1670 {
1671    Elm_Genlist_Item *it = data;
1672
1673    if (!it->expanded)
1674      evas_object_smart_callback_call(WIDGET(it), SIG_EXPAND_REQUEST, it);
1675 }
1676
1677 static void
1678 _signal_contract(void        *data,
1679                  Evas_Object *obj __UNUSED__,
1680                  const char  *emission __UNUSED__,
1681                  const char  *source __UNUSED__)
1682 {
1683    Elm_Genlist_Item *it = data;
1684
1685    if (it->expanded)
1686      evas_object_smart_callback_call(WIDGET(it), SIG_CONTRACT_REQUEST, it);
1687 }
1688
1689 static Eina_Bool
1690 _scr_hold_timer_cb(void *data)
1691 {
1692    if (!data) return ECORE_CALLBACK_CANCEL;
1693    Widget_Data *wd = data;
1694    elm_smart_scroller_hold_set(wd->scr, EINA_FALSE);
1695    wd->scr_hold_timer = NULL;
1696    return ECORE_CALLBACK_CANCEL;
1697 }
1698
1699 static void
1700 _mode_finished_signal_cb(void        *data,
1701                          Evas_Object *obj,
1702                          const char  *emission __UNUSED__,
1703                          const char  *source __UNUSED__)
1704 {
1705    if (!data) return;
1706    if (!obj) return;
1707    Elm_Genlist_Item *it = data;
1708    if ((it->delete_me) || (!it->realized) || (!it->mode_view)) return;
1709    char buf[1024];
1710    Evas *te = evas_object_evas_get(obj);
1711
1712    evas_event_freeze(te);
1713    it->nocache = EINA_FALSE;
1714    _mode_item_unrealize(it);
1715    snprintf(buf, sizeof(buf), "elm,state,%s,passive,finished", it->wd->mode_type);
1716    edje_object_signal_callback_del_full(obj, buf, "elm", _mode_finished_signal_cb, it);
1717    evas_event_thaw(te);
1718    evas_event_thaw_eval(te);
1719 }
1720
1721 static void
1722 _item_cache_clean(Widget_Data *wd)
1723 {
1724    evas_event_freeze(evas_object_evas_get(wd->obj));
1725    while ((wd->item_cache) && (wd->item_cache_count > wd->item_cache_max))
1726      {
1727         Item_Cache *itc;
1728
1729         itc = EINA_INLIST_CONTAINER_GET(wd->item_cache->last, Item_Cache);
1730         wd->item_cache = eina_inlist_remove(wd->item_cache,
1731                                             wd->item_cache->last);
1732         wd->item_cache_count--;
1733         if (itc->spacer) evas_object_del(itc->spacer);
1734         if (itc->base_view) evas_object_del(itc->base_view);
1735         if (itc->item_style) eina_stringshare_del(itc->item_style);
1736         free(itc);
1737      }
1738    evas_event_thaw(evas_object_evas_get(wd->obj));
1739    evas_event_thaw_eval(evas_object_evas_get(wd->obj));
1740 }
1741
1742 static void
1743 _item_cache_zero(Widget_Data *wd)
1744 {
1745    int pmax = wd->item_cache_max;
1746    wd->item_cache_max = 0;
1747    _item_cache_clean(wd);
1748    wd->item_cache_max = pmax;
1749 }
1750
1751 static void
1752 _item_cache_add(Elm_Genlist_Item *it)
1753 {
1754    Item_Cache *itc;
1755
1756    evas_event_freeze(evas_object_evas_get(it->wd->obj));
1757    if (it->wd->item_cache_max <= 0)
1758      {
1759         evas_object_del(VIEW(it));
1760         VIEW(it) = NULL;
1761         evas_object_del(it->spacer);
1762         it->spacer = NULL;
1763         evas_event_thaw(evas_object_evas_get(it->wd->obj));
1764         evas_event_thaw_eval(evas_object_evas_get(it->wd->obj));
1765         return;
1766      }
1767
1768    it->wd->item_cache_count++;
1769    itc = calloc(1, sizeof(Item_Cache));
1770    if (!itc)
1771      {
1772         evas_event_thaw(evas_object_evas_get(it->wd->obj));
1773         evas_event_thaw_eval(evas_object_evas_get(it->wd->obj));
1774         return;
1775      }
1776    it->wd->item_cache = eina_inlist_prepend(it->wd->item_cache,
1777                                             EINA_INLIST_GET(itc));
1778    itc->spacer = it->spacer;
1779    it->spacer = NULL;
1780    itc->base_view = VIEW(it);
1781    VIEW(it) = NULL;
1782    evas_object_hide(itc->base_view);
1783    evas_object_move(itc->base_view, -9999, -9999);
1784    itc->item_style = eina_stringshare_add(it->itc->item_style);
1785    if (it->flags & ELM_GENLIST_ITEM_SUBITEMS) itc->tree = 1;
1786    itc->compress = (it->wd->compress);
1787    itc->selected = it->selected;
1788    itc->disabled = elm_widget_item_disabled_get(it);
1789    itc->expanded = it->expanded;
1790    if (it->long_timer)
1791      {
1792         ecore_timer_del(it->long_timer);
1793         it->long_timer = NULL;
1794      }
1795    if (it->swipe_timer)
1796      {
1797         ecore_timer_del(it->swipe_timer);
1798         it->swipe_timer = NULL;
1799      }
1800    // FIXME: other callbacks?
1801    edje_object_signal_callback_del_full(itc->base_view,
1802                                         "elm,action,expand,toggle",
1803                                         "elm", _signal_expand_toggle, it);
1804    edje_object_signal_callback_del_full(itc->base_view, "elm,action,expand",
1805                                         "elm",
1806                                         _signal_expand, it);
1807    edje_object_signal_callback_del_full(itc->base_view, "elm,action,contract",
1808                                         "elm", _signal_contract, it);
1809    evas_object_event_callback_del_full(itc->base_view, EVAS_CALLBACK_MOUSE_DOWN,
1810                                        _mouse_down, it);
1811    evas_object_event_callback_del_full(itc->base_view, EVAS_CALLBACK_MOUSE_UP,
1812                                        _mouse_up, it);
1813    evas_object_event_callback_del_full(itc->base_view, EVAS_CALLBACK_MOUSE_MOVE,
1814                                        _mouse_move, it);
1815    evas_object_event_callback_del_full(itc->base_view, EVAS_CALLBACK_MULTI_DOWN,
1816                                        _multi_down, it);
1817    evas_object_event_callback_del_full(itc->base_view, EVAS_CALLBACK_MULTI_UP,
1818                                        _multi_up, it);
1819    evas_object_event_callback_del_full(itc->base_view, EVAS_CALLBACK_MULTI_MOVE,
1820                                        _multi_move, it);
1821    _item_cache_clean(it->wd);
1822    evas_event_thaw(evas_object_evas_get(it->wd->obj));
1823    evas_event_thaw_eval(evas_object_evas_get(it->wd->obj));
1824 }
1825
1826 static Item_Cache *
1827 _item_cache_find(Elm_Genlist_Item *it)
1828 {
1829    Item_Cache *itc;
1830    Eina_Bool tree = 0;
1831
1832    if (it->flags & ELM_GENLIST_ITEM_SUBITEMS) tree = 1;
1833    EINA_INLIST_FOREACH(it->wd->item_cache, itc)
1834      {
1835         if ((itc->selected) || (itc->disabled) || (itc->expanded))
1836           continue;
1837         if ((itc->tree == tree) &&
1838             (itc->compress == it->wd->compress) &&
1839             (!strcmp(it->itc->item_style, itc->item_style)))
1840           {
1841              it->wd->item_cache = eina_inlist_remove(it->wd->item_cache,
1842                                                      EINA_INLIST_GET(itc));
1843              it->wd->item_cache_count--;
1844              return itc;
1845           }
1846      }
1847    return NULL;
1848 }
1849
1850 static void
1851 _elm_genlist_item_index_update(Elm_Genlist_Item *it)
1852 {
1853    if (it->position_update || it->block->position_update)
1854      {
1855         evas_object_smart_callback_call(WIDGET(it), SIG_INDEX_UPDATE, it);
1856         it->position_update = EINA_FALSE;
1857      }
1858 }
1859
1860 static void
1861 _elm_genlist_item_odd_even_update(Elm_Genlist_Item *it)
1862 {
1863    if (!it->nostacking)
1864      {
1865         if ((it->order_num_in & 0x1) ^ it->stacking_even)
1866           evas_object_lower(VIEW(it));
1867         else
1868           evas_object_raise(VIEW(it));
1869      }
1870
1871    if (it->order_num_in & 0x1)
1872      edje_object_signal_emit(VIEW(it), "elm,state,odd", "elm");
1873    else
1874      edje_object_signal_emit(VIEW(it), "elm,state,even", "elm");
1875 }
1876
1877 static void
1878 _elm_genlist_item_state_update(Elm_Genlist_Item *it, Item_Cache *itc)
1879 {
1880    if (itc)
1881      {
1882         if (it->selected != itc->selected)
1883           {
1884              if (it->selected)
1885                {
1886                   edje_object_signal_emit(VIEW(it),
1887                                           "elm,state,selected", "elm");
1888                   if (it->edit_obj)
1889                     edje_object_signal_emit(it->edit_obj,
1890                                             "elm,state,selected", "elm");
1891                }
1892           }
1893         if (elm_widget_item_disabled_get(it) != itc->disabled)
1894           {
1895              if (elm_widget_item_disabled_get(it))
1896                {
1897                   edje_object_signal_emit(VIEW(it),
1898                                           "elm,state,disabled", "elm");
1899                   if (it->edit_obj)
1900                     edje_object_signal_emit(it->edit_obj,
1901                                             "elm,state,disabled", "elm");
1902                }
1903           }
1904         if (it->expanded != itc->expanded)
1905           {
1906              if (it->expanded)
1907                {
1908                   edje_object_signal_emit(VIEW(it),
1909                                           "elm,state,expanded", "elm");
1910                   if (it->edit_obj)
1911                     edje_object_signal_emit(it->edit_obj,
1912                                             "elm,state,expanded", "elm");
1913                }
1914           }
1915      }
1916    else
1917      {
1918         if (it->selected)
1919           {
1920              edje_object_signal_emit(VIEW(it),
1921                                      "elm,state,selected", "elm");
1922              if (it->edit_obj)
1923                edje_object_signal_emit(it->edit_obj,
1924                                        "elm,state,selected", "elm");
1925           }
1926         if (elm_widget_item_disabled_get(it))
1927           {
1928              edje_object_signal_emit(VIEW(it),
1929                                      "elm,state,disabled", "elm");
1930              if (it->edit_obj)
1931                edje_object_signal_emit(it->edit_obj,
1932                                        "elm,state,disabled", "elm");
1933           }
1934         if (it->expanded)
1935           {
1936              edje_object_signal_emit(VIEW(it),
1937                                      "elm,state,expanded", "elm");
1938              if (it->edit_obj)
1939                edje_object_signal_emit(it->edit_obj,
1940                                        "elm,state,expanded", "elm");
1941           }
1942      }
1943 }
1944
1945 static void
1946 _content_focused(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1947 {
1948    Elm_Genlist_Item *it = data;
1949    if (it) it->defer_unrealize = EINA_TRUE;
1950 }
1951
1952 static void
1953 _content_unfocused(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
1954 {
1955    Elm_Genlist_Item *it = data;
1956    if (it) it->defer_unrealize = EINA_FALSE;
1957 }
1958
1959 static Eina_List *
1960 _item_mode_content_unrealize(Elm_Genlist_Item *it,
1961                           Evas_Object *target,
1962                           Eina_List **source,
1963                           const char *parts,
1964                           Eina_List **contents_list)
1965 {
1966    Eina_List *res = *contents_list;
1967
1968    if (it->itc->func.content_get)
1969      {
1970         const Eina_List *l;
1971         const char *key;
1972         Evas_Object *ic = NULL;
1973
1974         EINA_LIST_FOREACH(*source, l, key)
1975           {
1976              if (parts && fnmatch(parts, key, FNM_PERIOD))
1977                continue;
1978
1979              ic = edje_object_part_swallow_get(target, key);
1980              if (ic)
1981                {
1982                   res = eina_list_remove(res, ic);
1983                   edje_object_part_unswallow(target, ic);
1984                   evas_object_del(ic);
1985                }
1986           }
1987      }
1988
1989    return res;
1990 }
1991
1992 static Eina_List *
1993 _item_mode_content_realize(Elm_Genlist_Item *it,
1994                         Evas_Object *target,
1995                         Eina_List **source,
1996                         const char *parts,
1997                         Eina_List **contents_list,
1998                         const char *contents_part)
1999
2000 {
2001    Eina_List *res = NULL;
2002
2003    if (it->itc->func.content_get)
2004      {
2005         const Eina_List *l;
2006         const char *key;
2007
2008         *source = elm_widget_stringlist_get(edje_object_data_get(target, contents_part));
2009         if (parts && (eina_list_count(*source) != eina_list_count(*contents_list)))
2010              res = *contents_list;
2011         EINA_LIST_FOREACH(*source, l, key)
2012           {
2013              if (parts && fnmatch(parts, key, FNM_PERIOD))
2014                continue;
2015
2016              Evas_Object *ic = it->itc->func.content_get
2017                 ((void *)it->base.data, WIDGET(it), l->data);
2018
2019              if (ic)
2020                {
2021                   res = eina_list_append(res, ic);
2022                   edje_object_part_swallow(target, key, ic);
2023                   evas_object_show(ic);
2024                   elm_widget_sub_object_add(WIDGET(it), ic);
2025                   if (it->mode_view || it->wd->edit_mode)
2026                     {
2027                        if (elm_widget_item_disabled_get(it))
2028                          elm_widget_disabled_set(ic, EINA_TRUE);
2029                     }
2030                   else if (it->renamed)
2031                     {
2032                        // FIXME: if entry calcuates its cursor position correctly and conformant works,
2033                        //        genlist does not need to handle this focus thing.
2034                        evas_object_smart_callback_add(ic, "focused", _content_focused, it);
2035                        evas_object_smart_callback_add(ic, "unfocused", _content_unfocused, it);
2036                     }
2037                }
2038           }
2039      }
2040
2041    return res;
2042 }
2043
2044 static void
2045 _item_cache_free(Item_Cache *itc)
2046 {
2047    if (itc->spacer) evas_object_del(itc->spacer);
2048    if (itc->base_view) evas_object_del(itc->base_view);
2049    if (itc->item_style) eina_stringshare_del(itc->item_style);
2050    free(itc);
2051 }
2052
2053 static void
2054 _item_text_realize(Elm_Genlist_Item *it,
2055                     Evas_Object *target,
2056                     Eina_List **source,
2057                     const char *parts)
2058 {
2059    if (it->itc->func.text_get)
2060      {
2061         const Eina_List *l;
2062         const char *key;
2063
2064         *source = elm_widget_stringlist_get(edje_object_data_get(target, "texts"));
2065         EINA_LIST_FOREACH(*source, l, key)
2066           {
2067              if (parts && fnmatch(parts, key, FNM_PERIOD))
2068                continue;
2069
2070              char *s = it->itc->func.text_get
2071                 ((void *)it->base.data, WIDGET(it), key);
2072
2073              if (s)
2074                {
2075                   edje_object_part_text_set(target, key, s);
2076                   free(s);
2077                }
2078              else
2079                {
2080                   edje_object_part_text_set(target, key, "");
2081                }
2082           }
2083      }
2084 }
2085
2086 static Eina_List *
2087 _item_content_unrealize(Elm_Genlist_Item *it,
2088                      Evas_Object *target,
2089                      Eina_List **source,
2090                      const char *parts)
2091 {
2092    Eina_List *res = it->content_objs;
2093
2094    if (it->itc->func.content_get)
2095      {
2096         const Eina_List *l;
2097         const char *key;
2098         Evas_Object *ic = NULL;
2099
2100         EINA_LIST_FOREACH(*source, l, key)
2101           {
2102              if (parts && fnmatch(parts, key, FNM_PERIOD))
2103                continue;
2104
2105              ic = edje_object_part_swallow_get(target, key);
2106              if (ic)
2107                {
2108                   res = eina_list_remove(res, ic);
2109                   edje_object_part_unswallow(target, ic);
2110                   evas_object_del(ic);
2111                }
2112           }
2113      }
2114
2115    return res;
2116 }
2117
2118 static Eina_List *
2119 _item_content_realize(Elm_Genlist_Item *it,
2120                    Evas_Object *target,
2121                    Eina_List **source,
2122                    const char *parts)
2123 {
2124    Eina_List *res = NULL;
2125
2126    if (it->itc->func.content_get)
2127      {
2128         const Eina_List *l;
2129         const char *key;
2130         Evas_Object *ic = NULL;
2131
2132         *source = elm_widget_stringlist_get(edje_object_data_get(target, "contents"));
2133
2134         if (parts && (eina_list_count(*source) != eina_list_count(it->content_objs)))
2135              res = it->content_objs;
2136
2137         EINA_LIST_FOREACH(*source, l, key)
2138           {
2139              if (parts && fnmatch(parts, key, FNM_PERIOD))
2140                continue;
2141
2142              if (it->itc->func.content_get)
2143                ic = it->itc->func.content_get
2144                   ((void *)it->base.data, WIDGET(it), key);
2145              if (ic)
2146                {
2147                   res = eina_list_append(res, ic);
2148                   edje_object_part_swallow(target, key, ic);
2149                   evas_object_show(ic);
2150                   elm_widget_sub_object_add(WIDGET(it), ic);
2151                   evas_object_event_callback_add(ic, EVAS_CALLBACK_CHANGED_SIZE_HINTS, _changed_size_hints, it);
2152                   if (elm_widget_item_disabled_get(it))
2153                     elm_widget_disabled_set(ic, EINA_TRUE);
2154                }
2155           }
2156      }
2157
2158    return res;
2159 }
2160
2161 static void
2162 _item_state_realize(Elm_Genlist_Item *it,
2163                     Evas_Object *target,
2164                     Eina_List **source,
2165                     const char *parts)
2166 {
2167    if (it->itc->func.state_get)
2168      {
2169         const Eina_List *l;
2170         const char *key;
2171         char buf[4096];
2172
2173         *source = elm_widget_stringlist_get(edje_object_data_get(target, "states"));
2174         EINA_LIST_FOREACH(*source, l, key)
2175           {
2176              if (parts && fnmatch(parts, key, FNM_PERIOD))
2177                continue;
2178
2179              Eina_Bool on = it->itc->func.state_get
2180                 ((void *)it->base.data, WIDGET(it), key);
2181
2182              if (on)
2183                {
2184                   snprintf(buf, sizeof(buf), "elm,state,%s,active", key);
2185                   edje_object_signal_emit(target, buf, "elm");
2186                }
2187              else
2188                {
2189                   snprintf(buf, sizeof(buf), "elm,state,%s,passive", key);
2190                   edje_object_signal_emit(target, buf, "elm");
2191                }
2192           }
2193      }
2194 }
2195
2196 static void
2197 _item_realize(Elm_Genlist_Item *it,
2198               int               in,
2199               Eina_Bool         calc)
2200 {
2201    const char *treesize;
2202    char buf[1024];
2203    int tsize = 20;
2204    Item_Cache *itc = NULL;
2205
2206    if ((it->realized) || (it->delete_me)) return;
2207    evas_event_freeze(evas_object_evas_get(it->wd->obj));
2208    if (it->realized)
2209      {
2210         if (it->order_num_in != in)
2211           {
2212              it->order_num_in = in;
2213              _elm_genlist_item_odd_even_update(it);
2214              _elm_genlist_item_state_update(it, NULL);
2215              _elm_genlist_item_index_update(it);
2216           }
2217         evas_event_thaw(evas_object_evas_get(it->wd->obj));
2218         evas_event_thaw_eval(evas_object_evas_get(it->wd->obj));
2219         return;
2220      }
2221    it->order_num_in = in;
2222    if (it->wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_DELETE) calc = EINA_FALSE;
2223    if ((it->nocache) && (!it->renamed))
2224      it->nocache = EINA_FALSE;
2225    else
2226      itc = _item_cache_find(it);
2227    if (itc && (!it->wd->effect_mode))
2228      {
2229         VIEW(it) = itc->base_view;
2230         itc->base_view = NULL;
2231         it->spacer = itc->spacer;
2232         itc->spacer = NULL;
2233      }
2234    else
2235      {
2236         const char *stacking_even;
2237         const char *stacking;
2238
2239         VIEW(it) = edje_object_add(evas_object_evas_get(WIDGET(it)));
2240         edje_object_scale_set(VIEW(it),
2241                               elm_widget_scale_get(WIDGET(it)) *
2242                               _elm_config->scale);
2243         evas_object_smart_member_add(VIEW(it), it->wd->pan_smart);
2244         elm_widget_sub_object_add(WIDGET(it), VIEW(it));
2245
2246         if (it->flags & ELM_GENLIST_ITEM_SUBITEMS)
2247           strncpy(buf, "tree", sizeof(buf));
2248         else strncpy(buf, "item", sizeof(buf));
2249         if (it->wd->compress)
2250           strncat(buf, "_compress", sizeof(buf) - strlen(buf));
2251
2252         strncat(buf, "/", sizeof(buf) - strlen(buf));
2253         strncat(buf, it->itc->item_style, sizeof(buf) - strlen(buf));
2254
2255         _elm_theme_object_set(WIDGET(it), VIEW(it), "genlist", buf,
2256                               elm_widget_style_get(WIDGET(it)));
2257
2258         stacking_even = edje_object_data_get(VIEW(it), "stacking_even");
2259         if (!stacking_even) stacking_even = "above";
2260         it->stacking_even = !!strcmp("above", stacking_even);
2261
2262         stacking = edje_object_data_get(VIEW(it), "stacking");
2263         if (!stacking) stacking = "yes";
2264         it->nostacking = !!strcmp("yes", stacking);
2265
2266         edje_object_mirrored_set(VIEW(it),
2267                                  elm_widget_mirrored_get(WIDGET(it)));
2268         it->spacer =
2269           evas_object_rectangle_add(evas_object_evas_get(WIDGET(it)));
2270         evas_object_color_set(it->spacer, 0, 0, 0, 0);
2271         elm_widget_sub_object_add(WIDGET(it), it->spacer);
2272      }
2273
2274    _elm_genlist_item_odd_even_update(it);
2275
2276    treesize = edje_object_data_get(VIEW(it), "treesize");
2277    if (treesize) tsize = atoi(treesize);
2278    evas_object_size_hint_min_set(it->spacer,
2279                                  (it->expanded_depth * tsize) * _elm_config->scale, 1);
2280    edje_object_part_swallow(VIEW(it), "elm.swallow.pad", it->spacer);
2281    if (!calc)
2282      {
2283         edje_object_signal_callback_add(VIEW(it),
2284                                         "elm,action,expand,toggle",
2285                                         "elm", _signal_expand_toggle, it);
2286         edje_object_signal_callback_add(VIEW(it), "elm,action,expand",
2287                                         "elm", _signal_expand, it);
2288         edje_object_signal_callback_add(VIEW(it), "elm,action,contract",
2289                                         "elm", _signal_contract, it);
2290         evas_object_event_callback_add(VIEW(it), EVAS_CALLBACK_MOUSE_DOWN,
2291                                        _mouse_down, it);
2292         evas_object_event_callback_add(VIEW(it), EVAS_CALLBACK_MOUSE_UP,
2293                                        _mouse_up, it);
2294         evas_object_event_callback_add(VIEW(it), EVAS_CALLBACK_MOUSE_MOVE,
2295                                        _mouse_move, it);
2296         evas_object_event_callback_add(VIEW(it), EVAS_CALLBACK_MULTI_DOWN,
2297                                        _multi_down, it);
2298         evas_object_event_callback_add(VIEW(it), EVAS_CALLBACK_MULTI_UP,
2299                                        _multi_up, it);
2300         evas_object_event_callback_add(VIEW(it), EVAS_CALLBACK_MULTI_MOVE,
2301                                        _multi_move, it);
2302
2303         if ((!it->edit_obj) &&(it->wd->edit_mode) &&
2304             (it->flags != ELM_GENLIST_ITEM_GROUP) && (it->itc->edit_item_style))
2305           _effect_item_realize(it, EINA_FALSE);
2306
2307         _elm_genlist_item_state_update(it, itc);
2308         _elm_genlist_item_index_update(it);
2309      }
2310
2311    if ((calc) && (it->wd->homogeneous) &&
2312        ((it->wd->item_width) ||
2313         ((it->wd->item_width) && (it->wd->group_item_width))))
2314      {
2315         /* homogenous genlist shortcut */
2316         if (!it->mincalcd)
2317           {
2318              if (it->flags & ELM_GENLIST_ITEM_GROUP)
2319                {
2320                   it->w = it->minw = it->wd->group_item_width;
2321                   it->h = it->minh = it->wd->group_item_height;
2322                }
2323              else
2324                {
2325                   it->w = it->minw = it->wd->item_width;
2326                   it->h = it->minh = it->wd->item_height;
2327                }
2328              it->mincalcd = EINA_TRUE;
2329           }
2330      }
2331    else
2332      {
2333         _item_text_realize(it, VIEW(it), &it->texts, NULL);
2334         it->content_objs = _item_content_realize(it, VIEW(it), &it->contents, NULL);
2335         _item_state_realize(it, VIEW(it), &it->states, NULL);
2336         if (it->itc->func.content_get) // for rename mode
2337           {
2338              const Eina_List *l;
2339              const char *key;
2340              if (it->renamed)
2341                {
2342                   edje_object_signal_emit(VIEW(it), "elm,state,rename,enabled", "elm");
2343                   it->contents =
2344                      elm_widget_stringlist_get(edje_object_data_get(VIEW(it), "renames"));
2345                   EINA_LIST_FOREACH(it->contents, l, key)
2346                     {
2347                        Evas_Object *ic = it->itc->func.content_get
2348                           ((void *)it->base.data, WIDGET(it), l->data);
2349                        if (ic)
2350                          {
2351                             it->content_objs = eina_list_append(it->content_objs, ic);
2352                             edje_object_part_swallow(VIEW(it), key, ic);
2353                             evas_object_show(ic);
2354                             elm_widget_sub_object_add(WIDGET(it), ic);
2355                             // FIXME: if entry calcuates its cursor position correctly and conformant works,
2356                             //        genlist does not need to handle this focus thing.
2357                             evas_object_smart_callback_add(ic, "focused", _content_focused, it);
2358                             evas_object_smart_callback_add(ic, "unfocused", _content_unfocused, it);
2359                          }
2360                     }
2361                }
2362           }
2363         if (!it->mincalcd || it->wd->pan_resize || ((it->wd->height_for_width) && (it->w != it->minw)))
2364           {
2365              Evas_Coord mw = -1, mh = -1;
2366
2367              if (it->wd->height_for_width) mw = it->wd->w; // TODO: Removed this. This was removed from upstream.
2368
2369              if (!it->display_only)
2370                elm_coords_finger_size_adjust(1, &mw, 1, &mh);
2371              if (it->wd->height_for_width) mw = it->wd->prev_viewport_w;
2372              edje_object_size_min_restricted_calc(VIEW(it), &mw, &mh, mw,
2373                                                   mh);
2374              if (!it->display_only)
2375                elm_coords_finger_size_adjust(1, &mw, 1, &mh);
2376              it->w = it->minw = mw;
2377              it->h = it->minh = mh;
2378              it->mincalcd = EINA_TRUE;
2379
2380              if ((!it->wd->group_item_width) && (it->flags == ELM_GENLIST_ITEM_GROUP))
2381                {
2382                   it->wd->group_item_width = mw;
2383                   it->wd->group_item_height = mh;
2384                }
2385              else if ((!it->wd->item_width) && (it->flags == ELM_GENLIST_ITEM_NONE))
2386                {
2387                   it->wd->item_width = mw;
2388                   it->wd->item_height = mh;
2389                }
2390           }
2391         if (!calc) evas_object_show(VIEW(it));
2392      }
2393
2394    if (it->tooltip.content_cb)
2395      {
2396         elm_widget_item_tooltip_content_cb_set(it,
2397                                                it->tooltip.content_cb,
2398                                                it->tooltip.data, NULL);
2399         elm_widget_item_tooltip_style_set(it, it->tooltip.style);
2400         //elm_widget_item_tooltip_window_mode_set(it, it->tooltip.free_size); //upstream
2401      }
2402
2403    if (it->mouse_cursor)
2404      elm_widget_item_cursor_set(it, it->mouse_cursor);
2405
2406    it->realized = EINA_TRUE;
2407    it->want_unrealize = EINA_FALSE;
2408
2409    if (itc) _item_cache_free(itc);
2410    evas_event_thaw(evas_object_evas_get(it->wd->obj));
2411    evas_event_thaw_eval(evas_object_evas_get(it->wd->obj));
2412    if (!calc)
2413      evas_object_smart_callback_call(WIDGET(it), SIG_REALIZED, it);
2414    if ((!calc) && (it->wd->edit_mode) && (it->flags != ELM_GENLIST_ITEM_GROUP))
2415      {
2416         if (it->itc->edit_item_style)
2417           {
2418              if (!it->edit_obj) _effect_item_realize(it, EINA_FALSE);
2419              edje_object_message_signal_process(it->edit_obj);
2420           }
2421      }
2422    edje_object_message_signal_process(VIEW(it));
2423 }
2424
2425 static void
2426 _item_unrealize(Elm_Genlist_Item *it,
2427                 Eina_Bool         calc)
2428 {
2429    Evas_Object *content;
2430
2431    if (!it->realized) return;
2432    if (it->wd->reorder_it == it) return;
2433    if (it->defer_unrealize && !it->updateme) return;
2434    evas_event_freeze(evas_object_evas_get(it->wd->obj));
2435    if (!calc)
2436      evas_object_smart_callback_call(WIDGET(it), SIG_UNREALIZED, it);
2437    if (it->long_timer)
2438      {
2439         ecore_timer_del(it->long_timer);
2440         it->long_timer = NULL;
2441      }
2442    if (it->nocache)
2443      {
2444         evas_object_del(VIEW(it));
2445         VIEW(it) = NULL;
2446         evas_object_del(it->spacer);
2447         it->spacer = NULL;
2448      }
2449    else
2450      {
2451         edje_object_mirrored_set(VIEW(it),
2452                                  elm_widget_mirrored_get(WIDGET(it)));
2453         edje_object_scale_set(VIEW(it),
2454                               elm_widget_scale_get(WIDGET(it))
2455                               * _elm_config->scale);
2456         _item_cache_add(it);
2457      }
2458    elm_widget_stringlist_free(it->texts);
2459    it->texts = NULL;
2460    elm_widget_stringlist_free(it->contents);
2461    it->contents = NULL;
2462    elm_widget_stringlist_free(it->states);
2463
2464    EINA_LIST_FREE(it->content_objs, content)
2465      evas_object_del(content);
2466
2467    _mode_item_unrealize(it);
2468    it->states = NULL;
2469    it->realized = EINA_FALSE;
2470    it->want_unrealize = EINA_FALSE;
2471    evas_event_thaw(evas_object_evas_get(it->wd->obj));
2472    evas_event_thaw_eval(evas_object_evas_get(it->wd->obj));
2473    if (it->wd->edit_mode) _effect_item_unrealize(it);
2474 }
2475
2476 static Eina_Bool
2477 _item_block_recalc(Item_Block *itb,
2478                    int         in,
2479                    Eina_Bool   qadd)
2480 {
2481    const Eina_List *l;
2482    Elm_Genlist_Item *it;
2483    Evas_Coord minw = 0, minh = 0;
2484    Eina_Bool showme = EINA_FALSE, changed = EINA_FALSE;
2485    Evas_Coord y = 0;
2486
2487    evas_event_freeze(evas_object_evas_get(itb->wd->obj));
2488    itb->num = in;
2489    EINA_LIST_FOREACH(itb->items, l, it)
2490      {
2491         if (it->delete_me) continue;
2492         showme |= it->showme;
2493         if (!itb->realized)
2494           {
2495              if (qadd)
2496                {
2497                   if (!it->mincalcd) changed = EINA_TRUE;
2498                   if (changed)
2499                     {
2500                        _item_realize(it, in, EINA_TRUE);
2501                        _item_unrealize(it, EINA_TRUE);
2502                     }
2503                }
2504              else
2505                {
2506                   _item_realize(it, in, EINA_TRUE);
2507                   _item_unrealize(it, EINA_TRUE);
2508                }
2509           }
2510         else
2511           _item_realize(it, in, EINA_FALSE);
2512         minh += it->minh;
2513         if (minw < it->minw) minw = it->minw;
2514         in++;
2515         it->x = 0;
2516         it->y = y;
2517         y += it->h;
2518      }
2519    itb->minw = minw;
2520    itb->minh = minh;
2521    itb->changed = EINA_FALSE;
2522    itb->position_update = EINA_FALSE;
2523    evas_event_thaw(evas_object_evas_get(itb->wd->obj));
2524    evas_event_thaw_eval(evas_object_evas_get(itb->wd->obj));
2525    return showme;
2526 }
2527
2528 static void
2529 _item_block_realize(Item_Block *itb)
2530 {
2531    if (itb->realized) return;
2532    itb->realized = EINA_TRUE;
2533    itb->want_unrealize = EINA_FALSE;
2534 }
2535
2536 static void
2537 _item_block_unrealize(Item_Block *itb)
2538 {
2539    const Eina_List *l;
2540    Elm_Genlist_Item *it;
2541    Eina_Bool dragging = EINA_FALSE;
2542
2543    if (!itb->realized) return;
2544    evas_event_freeze(evas_object_evas_get(itb->wd->obj));
2545    EINA_LIST_FOREACH(itb->items, l, it)
2546      {
2547         if (itb->must_recalc || it->flags != ELM_GENLIST_ITEM_GROUP)
2548           {
2549              if (it->dragging)
2550                {
2551                   dragging = EINA_TRUE;
2552                   it->want_unrealize = EINA_TRUE;
2553                }
2554              else
2555                _item_unrealize(it, EINA_FALSE);
2556           }
2557      }
2558    if (!dragging)
2559      {
2560         itb->realized = EINA_FALSE;
2561         itb->want_unrealize = EINA_TRUE;
2562      }
2563    else
2564      itb->want_unrealize = EINA_FALSE;
2565    evas_event_thaw(evas_object_evas_get(itb->wd->obj));
2566    evas_event_thaw_eval(evas_object_evas_get(itb->wd->obj));
2567 }
2568
2569 static int
2570 _get_space_for_reorder_item(Elm_Genlist_Item *it)
2571 {
2572    Evas_Coord rox, roy, row, roh, oy, oh;
2573    Eina_Bool top = EINA_FALSE;
2574    Elm_Genlist_Item *reorder_it = it->wd->reorder_it;
2575    if (!reorder_it) return 0;
2576
2577    evas_object_geometry_get(it->wd->pan_smart, NULL, &oy, NULL, &oh);
2578    evas_object_geometry_get(it->wd->VIEW(reorder_it), &rox, &roy, &row, &roh);
2579
2580    if ((it->wd->reorder_start_y < it->block->y) &&
2581        (roy - oy + (roh / 2) >= it->block->y - it->wd->pan_y))
2582      {
2583         it->block->reorder_offset = it->wd->reorder_it->h * -1;
2584         if (it->block->count == 1)
2585           it->wd->reorder_rel = it;
2586      }
2587    else if ((it->wd->reorder_start_y >= it->block->y) &&
2588             (roy - oy + (roh / 2) <= it->block->y - it->wd->pan_y))
2589      {
2590         it->block->reorder_offset = it->wd->reorder_it->h;
2591      }
2592    else
2593      it->block->reorder_offset = 0;
2594
2595    it->scrl_y += it->block->reorder_offset;
2596
2597    top = (ELM_RECTS_INTERSECT(it->scrl_x, it->scrl_y, it->w, it->h,
2598                               rox, roy + (roh / 2), row, 1));
2599    if (top)
2600      {
2601         it->wd->reorder_rel = it;
2602         it->scrl_y += it->wd->reorder_it->h;
2603         return it->wd->reorder_it->h;
2604      }
2605    else
2606      return 0;
2607 }
2608
2609 static Eina_Bool
2610 _reorder_move_animator_cb(void *data)
2611 {
2612    Elm_Genlist_Item *it = data;
2613    Eina_Bool down = EINA_FALSE;
2614    double t;
2615    int y, dy = it->h / 10 * _elm_config->scale, diff;
2616
2617    t = ((0.0 > (t = ecore_loop_time_get()-it->wd->start_time)) ? 0.0 : t);
2618
2619    if (t <= REORDER_EFFECT_TIME) y = (1 * sin((t / REORDER_EFFECT_TIME) * (M_PI / 2)) * dy);
2620    else y = dy;
2621
2622    diff = abs(it->old_scrl_y - it->scrl_y);
2623    if (diff > it->h) y = diff / 2;
2624    if (it->old_scrl_y < it->scrl_y)
2625      {
2626         it->old_scrl_y += y;
2627         down = EINA_TRUE;
2628      }
2629    else if (it->old_scrl_y > it->scrl_y)
2630      {
2631         it->old_scrl_y -= y;
2632         down = EINA_FALSE;
2633      }
2634
2635    if (it->edit_obj)
2636      _item_position(it, it->edit_obj, it->scrl_x, it->old_scrl_y);
2637    else
2638      _item_position(it, VIEW(it), it->scrl_x, it->old_scrl_y);
2639    if ((it->wd->reorder_pan_move) ||
2640        (down && it->old_scrl_y >= it->scrl_y) ||
2641        (!down && it->old_scrl_y <= it->scrl_y))
2642      {
2643         it->old_scrl_y = it->scrl_y;
2644         it->move_effect_enabled = EINA_FALSE;
2645         it->wd->reorder_move_animator = NULL;
2646         return ECORE_CALLBACK_CANCEL;
2647      }
2648    return ECORE_CALLBACK_RENEW;
2649 }
2650
2651 static void
2652 _item_position(Elm_Genlist_Item *it,
2653                Evas_Object      *view,
2654                Evas_Coord        it_x,
2655                Evas_Coord        it_y)
2656 {
2657    if (!it) return;
2658    if (!view) return;
2659
2660    evas_event_freeze(evas_object_evas_get(it->wd->obj));
2661    evas_object_resize(view, it->w, it->h);
2662    evas_object_move(view, it_x, it_y);
2663    evas_object_show(view);
2664    evas_event_thaw(evas_object_evas_get(it->wd->obj));
2665    evas_event_thaw_eval(evas_object_evas_get(it->wd->obj));
2666 }
2667
2668 static void
2669 _item_block_position(Item_Block *itb,
2670                      int         in)
2671 {
2672    const Eina_List *l;
2673    Elm_Genlist_Item *it;
2674    Evas_Coord y = 0, ox, oy, ow, oh, cvx, cvy, cvw, cvh;
2675    Evas_Coord minh = 0;
2676    int vis = 0;
2677
2678    evas_event_freeze(evas_object_evas_get(itb->wd->obj));
2679    evas_object_geometry_get(itb->wd->pan_smart, &ox, &oy, &ow, &oh);
2680    evas_output_viewport_get(evas_object_evas_get(itb->wd->obj), &cvx, &cvy,
2681                             &cvw, &cvh);
2682    EINA_LIST_FOREACH(itb->items, l, it)
2683      {
2684         if (it->delete_me) continue;
2685         else if (it->wd->reorder_it == it) continue;
2686
2687         it->x = 0;
2688         it->y = y;
2689         it->w = itb->w;
2690         it->scrl_x = itb->x + it->x - it->wd->pan_x + ox;
2691         it->scrl_y = itb->y + it->y - it->wd->pan_y + oy;
2692
2693         vis = (ELM_RECTS_INTERSECT(it->scrl_x, it->scrl_y, it->w, it->h,
2694                                    cvx, cvy, cvw, cvh));
2695         if ((itb->realized) && (!it->realized))
2696           {
2697              if (vis) _item_realize(it, in, EINA_FALSE);
2698           }
2699         if (it->realized)
2700           {
2701              if (vis)
2702                {
2703                   if (it->wd->reorder_mode)
2704                     y += _get_space_for_reorder_item(it);
2705                   if ((it->wd->reorder_it) && (it->old_scrl_y != it->scrl_y))
2706                     {
2707                        if (!it->move_effect_enabled)
2708                          {
2709                             it->move_effect_enabled = EINA_TRUE;
2710                             it->wd->reorder_move_animator =
2711                                ecore_animator_add(
2712                                   _reorder_move_animator_cb, it);
2713                          }
2714                     }
2715                   if (!it->move_effect_enabled)
2716                     if (!it->wd->effect_mode || it->wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_NONE || ((it->wd->move_effect_mode != ELM_GENLIST_ITEM_MOVE_EFFECT_DELETE) && it->parent == it->wd->expanded_item))
2717                       {
2718                          if (it->wd->edit_mode && it->itc->edit_item_style)
2719                            {
2720                               _item_position(it, it->edit_obj, it->scrl_x, it->scrl_y);
2721                            }
2722                          else
2723                            {
2724                               if((!it->wd->effect_mode || it->wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_NONE) || ((it->wd->move_effect_mode != ELM_GENLIST_ITEM_MOVE_EFFECT_NONE) && (it->old_scrl_y == it->scrl_y)))
2725                                 {
2726                                    if (it->mode_view)
2727                                      _item_position(it, it->mode_view, it->scrl_x,
2728                                                     it->scrl_y);
2729                                    else
2730                                      _item_position(it, VIEW(it), it->scrl_x,
2731                                                     it->scrl_y);
2732                                 }
2733                               else
2734                                 {
2735                                    evas_object_resize(VIEW(it), it->w, it->h);
2736                                    evas_object_move(VIEW(it), it->scrl_x, it->scrl_y);
2737                                    evas_object_hide(VIEW(it));
2738                                 }
2739                            }
2740                       }
2741                }
2742              else
2743                {
2744                   if ((!it->dragging && (!it->wd->item_moving_effect_timer)) || (it->wd->expanded_item->block != itb)) _item_unrealize(it, EINA_FALSE);
2745                }
2746           }
2747         in++;
2748         if (!it->wd->effect_mode || it->wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_NONE || ((it->wd->move_effect_mode != ELM_GENLIST_ITEM_MOVE_EFFECT_DELETE) && it->parent == it->wd->expanded_item))
2749           {
2750              it->old_scrl_y = it->scrl_y;
2751           }
2752         y += it->h;
2753         minh += it->minh;
2754      }
2755    if (!itb->wd->reorder_it) itb->minh = minh;
2756    evas_event_thaw(evas_object_evas_get(itb->wd->obj));
2757    evas_event_thaw_eval(evas_object_evas_get(itb->wd->obj));
2758 }
2759
2760 static void
2761 _changed_size_hints(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
2762 {
2763    Elm_Genlist_Item *it = data;
2764    if (!it) return;
2765    it->mincalcd = EINA_FALSE;
2766    it->block->changeme = EINA_TRUE;
2767    if (it->wd->changed_job) ecore_job_del(it->wd->changed_job);
2768    it->wd->changed_job = ecore_job_add(_changed_job, it->wd);
2769 }
2770
2771 static Eina_Bool
2772 _must_recalc_idler(void *data)
2773 {
2774    Widget_Data *wd = data;
2775    if (wd->calc_job) ecore_job_del(wd->calc_job);
2776    wd->calc_job = ecore_job_add(_calc_job, wd);
2777    wd->must_recalc_idler = NULL;
2778    return ECORE_CALLBACK_CANCEL;
2779 }
2780
2781 static void
2782 _scroll_item(Widget_Data *wd)
2783 {
2784    Elm_Genlist_Item *it = NULL;
2785    Evas_Coord ow, oh, dx = 0, dy = 0, dw = 0, dh = 0;
2786
2787    evas_object_geometry_get(wd->pan_smart, NULL, NULL, &ow, &oh);
2788    it = wd->show_item;
2789    dx = it->x + it->block->x;
2790    dy = it->y + it->block->y;
2791    dw = it->block->w;
2792    dh = oh;
2793    switch (wd->scrollto_type)
2794      {
2795       case ELM_GENLIST_ITEM_SCROLLTO_TOP:
2796          break;
2797       case ELM_GENLIST_ITEM_SCROLLTO_MIDDLE:
2798          dy += (it->h / 2 - oh / 2);
2799          break;
2800       case ELM_GENLIST_ITEM_SCROLLTO_IN:
2801       default:
2802          if ((wd->expanded_item) &&
2803              ((wd->show_item->y + wd->show_item->block->y + wd->show_item->h)
2804               - (wd->expanded_item->y + wd->expanded_item->block->y) > oh))
2805            {
2806               it = wd->expanded_item;
2807               dx = it->x + it->block->x;
2808               dy = it->y + it->block->y;
2809               dw = it->block->w;
2810            }
2811          else
2812            {
2813               dh = it->h;
2814            }
2815          break;
2816      }
2817    if (wd->bring_in)
2818      elm_smart_scroller_region_bring_in(wd->scr, dx, dy, dw, dh);
2819    else
2820      elm_smart_scroller_child_region_show(wd->scr, dx, dy, dw, dh);
2821
2822    it->showme = EINA_FALSE;
2823    wd->show_item = NULL;
2824    wd->auto_scroll_enabled = EINA_FALSE;
2825    wd->check_scroll = EINA_FALSE;
2826 }
2827
2828 static void
2829 _calc_job(void *data)
2830 {
2831    Widget_Data *wd = data;
2832    Item_Block *itb, *chb = NULL;
2833    Evas_Coord minw = -1, minh = 0, y = 0, ow, dy = 0, vw;
2834    Evas_Coord pan_w = 0, pan_h = 0;
2835    int in = 0;
2836    double t0, t;
2837    Eina_Bool minw_change = EINA_FALSE;
2838    Eina_Bool did_must_recalc = EINA_FALSE;
2839    if (!wd) return;
2840
2841    t0 = ecore_time_get();
2842    evas_object_geometry_get(wd->pan_smart, NULL, NULL, &ow, &wd->h);
2843    elm_smart_scroller_child_viewport_size_get(wd->scr, &vw, NULL);
2844
2845    if (wd->w != ow)
2846      wd->w = ow;
2847
2848    evas_event_freeze(evas_object_evas_get(wd->obj));
2849    EINA_INLIST_FOREACH(wd->blocks, itb)
2850      {
2851         Eina_Bool showme = EINA_FALSE;
2852
2853         itb->num = in;
2854         showme = itb->showme;
2855         itb->showme = EINA_FALSE;
2856         if (chb)
2857           {
2858              if (itb->realized) _item_block_unrealize(itb);
2859           }
2860         if ((itb->changed) || ((itb->must_recalc) && (!did_must_recalc)))
2861           {
2862              if (itb->must_recalc)
2863                {
2864                   Eina_List *l;
2865                   Elm_Genlist_Item *it;
2866                   EINA_LIST_FOREACH(itb->items, l, it)
2867                     if (it->mincalcd) it->mincalcd = EINA_FALSE;
2868                   itb->changed = EINA_TRUE;
2869                   if (itb->must_recalc) did_must_recalc = EINA_TRUE;
2870                   if (itb->realized) _item_block_unrealize(itb);
2871                   itb->must_recalc = EINA_FALSE;
2872                }
2873              showme = _item_block_recalc(itb, in, EINA_FALSE);
2874              chb = itb;
2875           }
2876         itb->y = y;
2877         itb->x = 0;
2878         minh += itb->minh;
2879         if (minw == -1) minw = itb->minw;
2880         else if ((!itb->must_recalc) && (minw < itb->minw))
2881           {
2882              minw = itb->minw;
2883              minw_change = EINA_TRUE;
2884           }
2885         if (minw > vw)
2886           {
2887              minw = vw;
2888              minw_change = EINA_TRUE;
2889           }
2890         itb->w = minw;
2891         itb->h = itb->minh;
2892         y += itb->h;
2893         in += itb->count;
2894         if ((showme) && (wd->show_item) && (!wd->show_item->queued))
2895           wd->check_scroll = EINA_TRUE;
2896      }
2897    if (minw_change)
2898      {
2899         EINA_INLIST_FOREACH(wd->blocks, itb)
2900           {
2901              itb->minw = minw;
2902              itb->w = itb->minw;
2903           }
2904      }
2905    if ((chb) && (EINA_INLIST_GET(chb)->next))
2906      {
2907         EINA_INLIST_FOREACH(EINA_INLIST_GET(chb)->next, itb)
2908           {
2909              if (itb->realized) _item_block_unrealize(itb);
2910           }
2911      }
2912    wd->realminw = minw;
2913    if (minw < wd->w) minw = wd->w;
2914    if ((minw != wd->minw) || (minh != wd->minh))
2915      {
2916         wd->minw = minw;
2917         wd->minh = minh;
2918         if (wd->move_effect_mode != ELM_GENLIST_ITEM_MOVE_EFFECT_DELETE)
2919            evas_object_smart_callback_call(wd->pan_smart, "changed", NULL);
2920         _sizing_eval(wd->obj);
2921 #ifdef ANCHOR_ITEM
2922         if ((wd->anchor_item) && (wd->anchor_item->block) && (!wd->auto_scroll_enabled))
2923           {
2924              Elm_Genlist_Item *it;
2925              Evas_Coord it_y;
2926
2927              it = wd->anchor_item;
2928              it_y = wd->anchor_y;
2929              elm_smart_scroller_child_pos_set(wd->scr, wd->pan_x,
2930                                               it->block->y + it->y + it_y);
2931              wd->anchor_item = it;
2932              wd->anchor_y = it_y;
2933           }
2934 #endif
2935      }
2936    t = ecore_time_get();
2937    if (did_must_recalc)
2938      {
2939         if (!wd->must_recalc_idler)
2940           wd->must_recalc_idler = ecore_idler_add(_must_recalc_idler, wd);
2941      }
2942    if (wd->check_scroll)
2943      {
2944         _pan_child_size_get(wd->pan_smart, &pan_w, &pan_h);
2945         if (EINA_INLIST_GET(wd->show_item) == wd->items->last)
2946           wd->scrollto_type = ELM_GENLIST_ITEM_SCROLLTO_IN;
2947        switch (wd->scrollto_type)
2948           {
2949            case ELM_GENLIST_ITEM_SCROLLTO_TOP:
2950               dy = wd->h;
2951               break;
2952            case ELM_GENLIST_ITEM_SCROLLTO_MIDDLE:
2953               dy = wd->h / 2;
2954               break;
2955            case ELM_GENLIST_ITEM_SCROLLTO_IN:
2956            default:
2957               dy = 0;
2958               break;
2959           }
2960         if ((pan_w > (wd->show_item->x + wd->show_item->block->x)) &&
2961             (pan_h > (wd->show_item->y + wd->show_item->block->y + dy)))
2962           {
2963              _scroll_item(wd);
2964           }
2965      }
2966    wd->calc_job = NULL;
2967    evas_object_smart_changed(wd->pan_smart);
2968    evas_event_thaw(evas_object_evas_get(wd->obj));
2969    evas_event_thaw_eval(evas_object_evas_get(wd->obj));
2970 }
2971
2972 static void
2973 _update_job(void *data)
2974 {
2975    Widget_Data *wd = data;
2976    Eina_List *l2;
2977    Item_Block *itb;
2978    int num, num0;
2979    Eina_Bool position = EINA_FALSE, recalc = EINA_FALSE;
2980    if (!wd) return;
2981    wd->update_job = NULL;
2982    num = 0;
2983
2984    evas_event_freeze(evas_object_evas_get(wd->obj));
2985    EINA_INLIST_FOREACH(wd->blocks, itb)
2986      {
2987         Evas_Coord itminw, itminh;
2988         Elm_Genlist_Item *it;
2989
2990         if (!itb->updateme)
2991           {
2992              num += itb->count;
2993              if (position)
2994                _item_block_position(itb, num);
2995              continue;
2996           }
2997         num0 = num;
2998         recalc = EINA_FALSE;
2999         EINA_LIST_FOREACH(itb->items, l2, it)
3000           {
3001              if (it->updateme)
3002                {
3003                   itminw = it->minw;
3004                   itminh = it->minh;
3005
3006                   if (it->realized)
3007                     {
3008                        _item_unrealize(it, EINA_FALSE);
3009                        _item_realize(it, num, EINA_FALSE);
3010                        position = EINA_TRUE;
3011                     }
3012                   else
3013                     {
3014                        _item_realize(it, num, EINA_TRUE);
3015                        _item_unrealize(it, EINA_TRUE);
3016                     }
3017                   if ((it->minw != itminw) || (it->minh != itminh))
3018                     recalc = EINA_TRUE;
3019                   it->updateme = EINA_FALSE;
3020                }
3021              num++;
3022           }
3023         itb->updateme = EINA_FALSE;
3024         if (recalc)
3025           {
3026              position = EINA_TRUE;
3027              itb->changed = EINA_TRUE;
3028              _item_block_recalc(itb, num0, EINA_FALSE);
3029              _item_block_position(itb, num0);
3030           }
3031      }
3032    if (position)
3033      {
3034         if (wd->calc_job) ecore_job_del(wd->calc_job);
3035         wd->calc_job = ecore_job_add(_calc_job, wd);
3036      }
3037    evas_event_thaw(evas_object_evas_get(wd->obj));
3038    evas_event_thaw_eval(evas_object_evas_get(wd->obj));
3039 }
3040
3041 static void
3042 _changed_job(void *data)
3043 {
3044    Widget_Data *wd = data; Eina_List *l2;
3045    Item_Block *itb;
3046    int num, num0, position = 0;
3047    Eina_Bool width_changed = EINA_FALSE, height_changed = EINA_FALSE;
3048    if (!wd) return;
3049    wd->changed_job = NULL;
3050    num = 0;
3051    EINA_INLIST_FOREACH(wd->blocks, itb)
3052      {
3053         Evas_Coord itminw, itminh;
3054         Elm_Genlist_Item *it;
3055
3056         if (!itb->changeme)
3057           {
3058              num += itb->count;
3059              if (position)
3060                _item_block_position(itb, num);
3061              continue;
3062           }
3063         num0 = num;
3064         width_changed = height_changed = EINA_FALSE;
3065         EINA_LIST_FOREACH(itb->items, l2, it)
3066           {
3067              if ((!it->mincalcd) && (it->realized))
3068                {
3069                   Evas_Coord mw = -1, mh = -1;
3070                   itminw = it->minw;
3071                   itminh = it->minh;
3072
3073                   if (it->wd->height_for_width) mw = it->wd->w;
3074                   if (!it->display_only)
3075                     elm_coords_finger_size_adjust(1, &mw, 1, &mh);
3076                   if (it->wd->height_for_width) mw = it->wd->prev_viewport_w;
3077                   edje_object_size_min_restricted_calc(VIEW(it), &mw, &mh, mw, mh);
3078                   if (!it->display_only)
3079                     elm_coords_finger_size_adjust(1, &mw, 1, &mh);
3080                   if (mw > itb->w)
3081                     width_changed = EINA_TRUE;
3082                   else
3083                     mw = itb->w;
3084                   it->w = it->minw = mw;
3085                   it->h = it->minh = mh;
3086                   it->mincalcd = EINA_TRUE;
3087
3088                   //if ((it->minw != itminw) || (it->minh != itminh))
3089                   //if ((it->minh != itminh))
3090                   //  recalc = 1;
3091                   if ((it->minw != itminw))
3092                     width_changed = EINA_TRUE;
3093                   if ((it->minh != itminh))
3094                     height_changed = EINA_TRUE;
3095
3096                   if ((!it->wd->group_item_width) && (it->flags == ELM_GENLIST_ITEM_GROUP))
3097                     {
3098                        it->wd->group_item_width = mw;
3099                        it->wd->group_item_height = mh;
3100                     }
3101                   else if ((!it->wd->item_width) && (it->flags == ELM_GENLIST_ITEM_NONE))
3102                     {
3103                        it->wd->item_width = mw;
3104                        it->wd->item_height = mh;
3105                     }
3106                }
3107              num++;
3108           }
3109         itb->changeme = EINA_FALSE;
3110         // TODO: why this is separated.
3111         if (height_changed)
3112           {
3113              position = 1;
3114              itb->changed = EINA_TRUE;
3115              _item_block_recalc(itb, num0, 0);
3116              _item_block_position(itb, num0);
3117           }
3118         else if (width_changed)
3119           {
3120              _item_block_position(itb, num0);
3121           }
3122      }
3123    if (position)
3124      {
3125         if (wd->calc_job) ecore_job_del(wd->calc_job);
3126         wd->calc_job = ecore_job_add(_calc_job, wd);
3127      }
3128 }
3129
3130 static void
3131 _pan_set(Evas_Object *obj,
3132          Evas_Coord   x,
3133          Evas_Coord   y)
3134 {
3135    Pan *sd = evas_object_smart_data_get(obj);
3136 #ifdef ANCHOR_ITEM
3137    Item_Block *itb;
3138 #endif
3139
3140    if (!sd) return;
3141    //   Evas_Coord ow, oh;
3142    //   evas_object_geometry_get(obj, NULL, NULL, &ow, &oh);
3143    //   ow = sd->wd->minw - ow;
3144    //   if (ow < 0) ow = 0;
3145    //   oh = sd->wd->minh - oh;
3146    //   if (oh < 0) oh = 0;
3147    //   if (x < 0) x = 0;
3148    //   if (y < 0) y = 0;
3149    //   if (x > ow) x = ow;
3150    //   if (y > oh) y = oh;
3151    if ((x == sd->wd->pan_x) && (y == sd->wd->pan_y)) return;
3152    sd->wd->pan_x = x;
3153    sd->wd->pan_y = y;
3154
3155 #ifdef ANCHOR_ITEM
3156    EINA_INLIST_FOREACH(sd->wd->blocks, itb)
3157      {
3158         if ((itb->y + itb->h) > y)
3159           {
3160              Elm_Genlist_Item *it;
3161              Eina_List *l2;
3162
3163              EINA_LIST_FOREACH(itb->items, l2, it)
3164                {
3165                   if ((itb->y + it->y) >= y)
3166                     {
3167                        sd->wd->anchor_item = it;
3168                        sd->wd->anchor_y = -(itb->y + it->y - y);
3169                        goto done;
3170                     }
3171                }
3172           }
3173      }
3174 done:
3175 #endif
3176    if (!sd->wd->reorder_move_animator) evas_object_smart_changed(obj);
3177 }
3178
3179 static void
3180 _pan_get(Evas_Object *obj,
3181          Evas_Coord  *x,
3182          Evas_Coord  *y)
3183 {
3184    Pan *sd = evas_object_smart_data_get(obj);
3185
3186    if (!sd) return;
3187    if (x) *x = sd->wd->pan_x;
3188    if (y) *y = sd->wd->pan_y;
3189 }
3190
3191 static void
3192 _pan_max_get(Evas_Object *obj,
3193              Evas_Coord  *x,
3194              Evas_Coord  *y)
3195 {
3196    Pan *sd = evas_object_smart_data_get(obj);
3197    Evas_Coord ow, oh;
3198
3199    if (!sd) return;
3200    evas_object_geometry_get(obj, NULL, NULL, &ow, &oh);
3201    ow = sd->wd->minw - ow;
3202    if (ow < 0) ow = 0;
3203    oh = sd->wd->minh - oh;
3204    if (oh < 0) oh = 0;
3205    if (x) *x = ow;
3206    if (y) *y = oh;
3207 }
3208
3209 static void
3210 _pan_min_get(Evas_Object *obj __UNUSED__,
3211              Evas_Coord  *x,
3212              Evas_Coord  *y)
3213 {
3214    if (x) *x = 0;
3215    if (y) *y = 0;
3216 }
3217
3218 static void
3219 _pan_child_size_get(Evas_Object *obj,
3220                     Evas_Coord  *w,
3221                     Evas_Coord  *h)
3222 {
3223    Pan *sd = evas_object_smart_data_get(obj);
3224
3225    if (!sd) return;
3226    if (w) *w = sd->wd->minw;
3227    if (h) *h = sd->wd->minh;
3228 }
3229
3230 static void
3231 _pan_add(Evas_Object *obj)
3232 {
3233    Pan *sd;
3234    Evas_Object_Smart_Clipped_Data *cd;
3235
3236    _pan_sc.add(obj);
3237    cd = evas_object_smart_data_get(obj);
3238    sd = ELM_NEW(Pan);
3239    if (!sd) return;
3240    sd->__clipped_data = *cd;
3241    free(cd);
3242    evas_object_smart_data_set(obj, sd);
3243 }
3244
3245 static void
3246 _pan_del(Evas_Object *obj)
3247 {
3248    Pan *sd = evas_object_smart_data_get(obj);
3249
3250    if (!sd) return;
3251    if (sd->resize_job)
3252      {
3253         ecore_job_del(sd->resize_job);
3254         sd->resize_job = NULL;
3255      }
3256    _pan_sc.del(obj);
3257 }
3258
3259 static void
3260 _pan_resize_job(void *data)
3261 {
3262    Pan *sd = data;
3263    if (!sd) return;
3264    _sizing_eval(sd->wd->obj);
3265    sd->resize_job = NULL;
3266 }
3267
3268 static void
3269 _pan_resize(Evas_Object *obj,
3270             Evas_Coord   w,
3271             Evas_Coord   h)
3272 {
3273    Pan *sd = evas_object_smart_data_get(obj);
3274    Evas_Coord ow, oh;
3275
3276    if (!sd) return;
3277    evas_object_geometry_get(obj, NULL, NULL, &ow, &oh);
3278    if ((ow == w) && (oh == h)) return;
3279    if ((sd->wd->height_for_width) && (ow != w))
3280      {
3281         sd->wd->pan_resize = EINA_TRUE;
3282         if (sd->resize_job) ecore_job_del(sd->resize_job);
3283         sd->resize_job = ecore_job_add(_pan_resize_job, sd);
3284      }
3285    sd->wd->pan_changed = EINA_TRUE;
3286    evas_object_smart_changed(obj);
3287    if (sd->wd->calc_job) ecore_job_del(sd->wd->calc_job);
3288    sd->wd->calc_job = NULL;
3289 }
3290
3291 static void
3292 _pan_calculate(Evas_Object *obj)
3293 {
3294    Pan *sd = evas_object_smart_data_get(obj);
3295    Item_Block *itb;
3296    Evas_Coord ox, oy, ow, oh, cvx, cvy, cvw, cvh;
3297    int in = 0;
3298    Elm_Genlist_Item *git;
3299    Eina_List *l;
3300
3301    if (!sd) return;
3302    evas_event_freeze(evas_object_evas_get(obj));
3303
3304    if (sd->wd->pan_changed)
3305      {
3306         _calc_job(sd->wd);
3307         sd->wd->pan_changed = EINA_FALSE;
3308      }
3309
3310    evas_object_geometry_get(obj, &ox, &oy, &ow, &oh);
3311    sd->wd->prev_viewport_w = ow;
3312    evas_output_viewport_get(evas_object_evas_get(obj), &cvx, &cvy, &cvw, &cvh);
3313    EINA_INLIST_FOREACH(sd->wd->blocks, itb)
3314      {
3315         itb->w = sd->wd->minw;
3316         if (ELM_RECTS_INTERSECT(itb->x - sd->wd->pan_x + ox,
3317                                 itb->y - sd->wd->pan_y + oy,
3318                                 itb->w, itb->h,
3319                                 cvx, cvy, cvw, cvh))
3320           {
3321              if ((!itb->realized) || (itb->changed))
3322                _item_block_realize(itb);
3323              _item_block_position(itb, in);
3324           }
3325         else
3326           {
3327              if (itb->realized) _item_block_unrealize(itb);
3328           }
3329         in += itb->count;
3330      }
3331
3332    if ((sd->wd->reorder_mode) && (sd->wd->reorder_it))
3333      {
3334         if (sd->wd->pan_y != sd->wd->old_pan_y)
3335           sd->wd->reorder_pan_move = EINA_TRUE;
3336         else sd->wd->reorder_pan_move = EINA_FALSE;
3337         evas_object_raise(sd->wd->reorder_it->edit_obj);
3338         sd->wd->old_pan_y = sd->wd->pan_y;
3339         sd->wd->start_time = ecore_loop_time_get();
3340      }
3341
3342       if (sd->wd->effect_mode &&
3343           ((sd->wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_EXPAND) ||
3344            (sd->wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_CONTRACT) ||
3345            (sd->wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_DELETE)))
3346         {
3347            if (!sd->wd->item_moving_effect_timer)
3348              {
3349                 if (sd->wd->move_effect_mode != ELM_GENLIST_ITEM_MOVE_EFFECT_DELETE)
3350                    _item_flip_effect_show(sd->wd->expanded_item);
3351
3352                 evas_object_raise(sd->wd->alpha_bg);
3353                 evas_object_show(sd->wd->alpha_bg);
3354                 elm_smart_scroller_bounce_animator_disabled_set(sd->wd->scr, EINA_TRUE);
3355                 sd->wd->start_time = current_time_get();
3356                 sd->wd->item_moving_effect_timer = ecore_animator_add(_item_moving_effect_timer_cb, sd->wd);
3357              }
3358         }
3359       else _item_auto_scroll(sd->wd);
3360    sd->wd->pan_resize = EINA_FALSE;
3361    evas_event_thaw(evas_object_evas_get(obj));
3362    evas_event_thaw_eval(evas_object_evas_get(obj));
3363 }
3364
3365 static void
3366 _pan_move(Evas_Object *obj,
3367           Evas_Coord   x __UNUSED__,
3368           Evas_Coord   y __UNUSED__)
3369 {
3370    Pan *sd = evas_object_smart_data_get(obj);
3371
3372    if (!sd) return;
3373
3374    sd->wd->pan_changed = EINA_TRUE;
3375    evas_object_smart_changed(obj);
3376    if (sd->wd->calc_job) ecore_job_del(sd->wd->calc_job);
3377    sd->wd->calc_job = NULL;
3378 }
3379
3380 static void
3381 _hold_on(void        *data __UNUSED__,
3382          Evas_Object *obj,
3383          void        *event_info __UNUSED__)
3384 {
3385    Widget_Data *wd = elm_widget_data_get(obj);
3386    if (!wd) return;
3387    elm_smart_scroller_hold_set(wd->scr, 1);
3388 }
3389
3390 static void
3391 _hold_off(void        *data __UNUSED__,
3392           Evas_Object *obj,
3393           void        *event_info __UNUSED__)
3394 {
3395    Widget_Data *wd = elm_widget_data_get(obj);
3396    if (!wd) return;
3397    elm_smart_scroller_hold_set(wd->scr, 0);
3398 }
3399
3400 static void
3401 _freeze_on(void        *data __UNUSED__,
3402            Evas_Object *obj,
3403            void        *event_info __UNUSED__)
3404 {
3405    Widget_Data *wd = elm_widget_data_get(obj);
3406    if (!wd) return;
3407    elm_smart_scroller_freeze_set(wd->scr, 1);
3408 }
3409
3410 static void
3411 _freeze_off(void        *data __UNUSED__,
3412             Evas_Object *obj,
3413             void        *event_info __UNUSED__)
3414 {
3415    Widget_Data *wd = elm_widget_data_get(obj);
3416    if (!wd) return;
3417    elm_smart_scroller_freeze_set(wd->scr, 0);
3418 }
3419
3420 static void
3421 _scr_anim_start(void        *data,
3422                 Evas_Object *obj __UNUSED__,
3423                 void        *event_info __UNUSED__)
3424 {
3425    evas_object_smart_callback_call(data, SIG_SCROLL_ANIM_START, NULL);
3426 }
3427
3428 static void
3429 _scr_anim_stop(void        *data,
3430                Evas_Object *obj __UNUSED__,
3431                void        *event_info __UNUSED__)
3432 {
3433    evas_object_smart_callback_call(data, SIG_SCROLL_ANIM_STOP, NULL);
3434 }
3435
3436 static void
3437 _scr_drag_start(void            *data,
3438                 Evas_Object     *obj __UNUSED__,
3439                 void            *event_info __UNUSED__)
3440 {
3441    evas_object_smart_callback_call(data, SIG_SCROLL_DRAG_START, NULL);
3442 }
3443
3444 static void
3445 _scr_drag_stop(void            *data,
3446                Evas_Object     *obj __UNUSED__,
3447                void            *event_info __UNUSED__)
3448 {
3449    evas_object_smart_callback_call(data, SIG_SCROLL_DRAG_STOP, NULL);
3450 }
3451
3452 static void
3453 _edge_left(void        *data,
3454                   Evas_Object *scr __UNUSED__,
3455                   void        *event_info __UNUSED__)
3456 {
3457    Evas_Object *obj = data;
3458    evas_object_smart_callback_call(obj, SIG_EDGE_LEFT, NULL);
3459 }
3460
3461 static void
3462 _edge_right(void        *data,
3463                    Evas_Object *scr __UNUSED__,
3464                    void        *event_info __UNUSED__)
3465 {
3466    Evas_Object *obj = data;
3467    evas_object_smart_callback_call(obj, SIG_EDGE_RIGHT, NULL);
3468 }
3469
3470 static void
3471 _edge_top(void        *data,
3472                  Evas_Object *scr __UNUSED__,
3473                  void        *event_info __UNUSED__)
3474 {
3475    Evas_Object *obj = data;
3476    evas_object_smart_callback_call(obj, SIG_EDGE_TOP, NULL);
3477 }
3478
3479 static void
3480 _edge_bottom(void        *data,
3481                     Evas_Object *scr __UNUSED__,
3482                     void        *event_info __UNUSED__)
3483 {
3484    Evas_Object *obj = data;
3485    evas_object_smart_callback_call(obj, SIG_EDGE_BOTTOM, NULL);
3486 }
3487
3488 static void
3489 _mode_item_realize(Elm_Genlist_Item *it)
3490 {
3491    char buf[1024];
3492
3493    if ((it->mode_view) || (it->delete_me)) return;
3494
3495    evas_event_freeze(evas_object_evas_get(it->wd->obj));
3496    it->mode_view = edje_object_add(evas_object_evas_get(WIDGET(it)));
3497    edje_object_scale_set(it->mode_view,
3498                          elm_widget_scale_get(WIDGET(it)) *
3499                          _elm_config->scale);
3500    evas_object_smart_member_add(it->mode_view, it->wd->pan_smart);
3501    elm_widget_sub_object_add(WIDGET(it), it->mode_view);
3502
3503    strncpy(buf, "item", sizeof(buf));
3504    if (it->wd->compress)
3505      strncat(buf, "_compress", sizeof(buf) - strlen(buf));
3506
3507    if (it->order_num_in & 0x1) strncat(buf, "_odd", sizeof(buf) - strlen(buf));
3508    strncat(buf, "/", sizeof(buf) - strlen(buf));
3509    strncat(buf, it->itc->mode_item_style, sizeof(buf) - strlen(buf));
3510
3511    _elm_theme_object_set(WIDGET(it), it->mode_view, "genlist", buf,
3512                          elm_widget_style_get(WIDGET(it)));
3513    edje_object_mirrored_set(it->mode_view,
3514                             elm_widget_mirrored_get(WIDGET(it)));
3515
3516    /* signal callback add */
3517    evas_object_event_callback_add(it->mode_view, EVAS_CALLBACK_MOUSE_DOWN,
3518                                   _mouse_down, it);
3519    evas_object_event_callback_add(it->mode_view, EVAS_CALLBACK_MOUSE_UP,
3520                                   _mouse_up, it);
3521    evas_object_event_callback_add(it->mode_view, EVAS_CALLBACK_MOUSE_MOVE,
3522                                   _mouse_move, it);
3523
3524    /* text_get, content_get, state_get */
3525    _item_text_realize(it, it->mode_view, &it->mode_texts, NULL);
3526    if (it->itc->func.content_get)
3527      {
3528         const Eina_List *l;
3529         const char *key;
3530
3531         it->mode_contents =
3532            elm_widget_stringlist_get(edje_object_data_get(it->mode_view,
3533                                                           "contents"));
3534         EINA_LIST_FOREACH(it->mode_contents, l, key)
3535          {
3536            Evas_Object *ic = it->itc->func.content_get
3537                 ((void *)it->base.data, WIDGET(it), l->data);
3538
3539              if (ic)
3540                {
3541                   it->mode_content_objs = eina_list_append(it->mode_content_objs, ic);
3542                   edje_object_part_swallow(it->mode_view, key, ic);
3543                   evas_object_show(ic);
3544                   elm_widget_sub_object_add(WIDGET(it), ic);
3545                   if (elm_widget_item_disabled_get(it))
3546                     elm_widget_disabled_set(ic, EINA_TRUE);
3547                }
3548           }
3549      }
3550    _item_state_realize(it, it->mode_view, &it->mode_states, NULL);
3551    edje_object_part_swallow(it->mode_view,
3552                             edje_object_data_get(it->mode_view, "mode_part"),
3553                             VIEW(it));
3554
3555    it->want_unrealize = EINA_FALSE;
3556    evas_event_thaw(evas_object_evas_get(it->wd->obj));
3557    evas_event_thaw_eval(evas_object_evas_get(it->wd->obj));
3558 }
3559
3560 static void
3561 _mode_item_unrealize(Elm_Genlist_Item *it)
3562 {
3563    Widget_Data *wd = it->wd;
3564    Evas_Object *content;
3565    if (!it->mode_view) return;
3566
3567    evas_event_freeze(evas_object_evas_get(it->wd->obj));
3568    elm_widget_stringlist_free(it->mode_texts);
3569    it->mode_texts = NULL;
3570    elm_widget_stringlist_free(it->mode_contents);
3571    it->mode_contents = NULL;
3572    elm_widget_stringlist_free(it->mode_states);
3573
3574    EINA_LIST_FREE(it->mode_content_objs, content)
3575      evas_object_del(content);
3576
3577    edje_object_part_unswallow(it->mode_view, VIEW(it));
3578    evas_object_smart_member_add(VIEW(it), wd->pan_smart);
3579    evas_object_del(it->mode_view);
3580    it->mode_view = NULL;
3581
3582    if (wd->mode_item == it)
3583      wd->mode_item = NULL;
3584    evas_event_thaw(evas_object_evas_get(it->wd->obj));
3585    evas_event_thaw_eval(evas_object_evas_get(it->wd->obj));
3586 }
3587
3588 static void
3589 _item_mode_set(Elm_Genlist_Item *it)
3590 {
3591    if (!it) return;
3592    Widget_Data *wd = it->wd;
3593    if (!wd) return;
3594    char buf[1024];
3595
3596    wd->mode_item = it;
3597    it->nocache = EINA_TRUE;
3598
3599    if (wd->scr_hold_timer)
3600      {
3601         ecore_timer_del(wd->scr_hold_timer);
3602         wd->scr_hold_timer = NULL;
3603      }
3604    elm_smart_scroller_hold_set(wd->scr, EINA_TRUE);
3605    wd->scr_hold_timer = ecore_timer_add(0.1, _scr_hold_timer_cb, wd);
3606
3607    evas_event_freeze(evas_object_evas_get(it->wd->obj));
3608    _mode_item_realize(it);
3609    if (it->group_item)
3610      evas_object_raise(it->VIEW(group_item));
3611    _item_position(it, it->mode_view, it->scrl_x, it->scrl_y);
3612    evas_event_thaw(evas_object_evas_get(it->wd->obj));
3613    evas_event_thaw_eval(evas_object_evas_get(it->wd->obj));
3614
3615    snprintf(buf, sizeof(buf), "elm,state,%s,active", wd->mode_type);
3616    edje_object_signal_emit(it->mode_view, buf, "elm");
3617 }
3618
3619 static void
3620 _item_mode_unset(Widget_Data *wd)
3621 {
3622    if (!wd) return;
3623    if (!wd->mode_item) return;
3624    char buf[1024], buf2[1024];
3625    Elm_Genlist_Item *it;
3626
3627    it = wd->mode_item;
3628    it->nocache = EINA_TRUE;
3629
3630    snprintf(buf, sizeof(buf), "elm,state,%s,passive", wd->mode_type);
3631    snprintf(buf2, sizeof(buf2), "elm,state,%s,passive,finished", wd->mode_type);
3632
3633    edje_object_signal_emit(it->mode_view, buf, "elm");
3634    edje_object_signal_callback_add(it->mode_view, buf2, "elm", _mode_finished_signal_cb, it);
3635
3636    wd->mode_item = NULL;
3637 }
3638
3639 static void
3640 _item_auto_scroll(Widget_Data *wd)
3641 {
3642    if (!wd) return;
3643    Elm_Genlist_Item  *tmp_item = NULL;
3644
3645    if ((wd->expanded_item) && (wd->auto_scroll_enabled))
3646      {
3647         tmp_item = eina_list_data_get(eina_list_last(wd->expanded_item->items));
3648         if (!tmp_item) return;
3649         wd->show_item = tmp_item;
3650         wd->bring_in = EINA_TRUE;
3651         wd->scrollto_type = ELM_GENLIST_ITEM_SCROLLTO_IN;
3652         if ((wd->show_item->queued) || (!wd->show_item->mincalcd))
3653           {
3654              wd->show_item->showme = EINA_TRUE;
3655              wd->auto_scroll_enabled = EINA_FALSE;
3656           }
3657         else
3658           _scroll_item(wd);
3659      }
3660 }
3661
3662 EAPI Evas_Object *
3663 elm_genlist_add(Evas_Object *parent)
3664 {
3665    Evas_Object *obj;
3666    Evas *e;
3667    Widget_Data *wd;
3668    Evas_Coord minw, minh;
3669    static Evas_Smart *smart = NULL;
3670
3671    if (!smart)
3672      {
3673         static Evas_Smart_Class sc;
3674
3675         evas_object_smart_clipped_smart_set(&_pan_sc);
3676         sc = _pan_sc;
3677         sc.name = "elm_genlist_pan";
3678         sc.version = EVAS_SMART_CLASS_VERSION;
3679         sc.add = _pan_add;
3680         sc.del = _pan_del;
3681         sc.resize = _pan_resize;
3682         sc.move = _pan_move;
3683         sc.calculate = _pan_calculate;
3684         if (!(smart = evas_smart_class_new(&sc))) return NULL;
3685      }
3686
3687    ELM_WIDGET_STANDARD_SETUP(wd, Widget_Data, parent, e, obj, NULL);
3688
3689    ELM_SET_WIDTYPE(widtype, "genlist");
3690    elm_widget_type_set(obj, "genlist");
3691    elm_widget_sub_object_add(parent, obj);
3692    elm_widget_on_focus_hook_set(obj, _on_focus_hook, NULL);
3693    elm_widget_signal_emit_hook_set(obj, _signal_emit_hook);
3694    elm_widget_data_set(obj, wd);
3695    elm_widget_del_hook_set(obj, _del_hook);
3696    elm_widget_del_pre_hook_set(obj, _del_pre_hook);
3697    elm_widget_theme_hook_set(obj, _theme_hook);
3698    elm_widget_can_focus_set(obj, EINA_TRUE);
3699    elm_widget_event_hook_set(obj, _event_hook);
3700    elm_widget_on_show_region_hook_set(obj, _show_region_hook, obj);
3701    elm_widget_translate_hook_set(obj, _translate_hook);
3702
3703    wd->scr = elm_smart_scroller_add(e);
3704    evas_object_event_callback_add(wd->scr, EVAS_CALLBACK_MOUSE_DOWN,
3705                                   _mouse_down_scroller, obj);
3706    evas_object_event_callback_add(wd->scr, EVAS_CALLBACK_MOUSE_UP,
3707                                   _mouse_up_scroller, obj);
3708    evas_object_event_callback_add(wd->scr, EVAS_CALLBACK_MOUSE_MOVE,
3709                                   _mouse_move_scroller, obj);
3710    elm_smart_scroller_widget_set(wd->scr, obj);
3711    elm_smart_scroller_object_theme_set(obj, wd->scr, "genlist", "base",
3712                                        elm_widget_style_get(obj));
3713    elm_smart_scroller_bounce_allow_set(wd->scr, EINA_FALSE,
3714                                        _elm_config->thumbscroll_bounce_enable);
3715    elm_widget_resize_object_set(obj, wd->scr);
3716
3717    evas_object_smart_callback_add(wd->scr, "animate,start", _scr_anim_start, obj);
3718    evas_object_smart_callback_add(wd->scr, "animate,stop", _scr_anim_stop, obj);
3719    evas_object_smart_callback_add(wd->scr, "drag,start", _scr_drag_start, obj);
3720    evas_object_smart_callback_add(wd->scr, "drag,stop", _scr_drag_stop, obj);
3721    evas_object_smart_callback_add(wd->scr, "edge,left", _edge_left, obj);
3722    evas_object_smart_callback_add(wd->scr, "edge,right", _edge_right, obj);
3723    evas_object_smart_callback_add(wd->scr, "edge,top", _edge_top, obj);
3724    evas_object_smart_callback_add(wd->scr, "edge,bottom", _edge_bottom, obj);
3725
3726    wd->obj = obj;
3727    wd->mode = ELM_LIST_SCROLL;
3728    wd->max_items_per_block = MAX_ITEMS_PER_BLOCK;
3729    wd->item_cache_max = wd->max_items_per_block * 2;
3730    wd->longpress_timeout = _elm_config->longpress_timeout;
3731    wd->effect_mode = _elm_config->effect_enable;
3732
3733    evas_object_smart_callback_add(obj, "scroll-hold-on", _hold_on, obj);
3734    evas_object_smart_callback_add(obj, "scroll-hold-off", _hold_off, obj);
3735    evas_object_smart_callback_add(obj, "scroll-freeze-on", _freeze_on, obj);
3736    evas_object_smart_callback_add(obj, "scroll-freeze-off", _freeze_off, obj);
3737
3738    wd->pan_smart = evas_object_smart_add(e, smart);
3739    wd->pan = evas_object_smart_data_get(wd->pan_smart);
3740    wd->pan->wd = wd;
3741
3742    elm_smart_scroller_extern_pan_set(wd->scr, wd->pan_smart,
3743                                      _pan_set, _pan_get, _pan_max_get,
3744                                      _pan_min_get, _pan_child_size_get);
3745
3746    edje_object_size_min_calc(elm_smart_scroller_edje_object_get(wd->scr),
3747                              &minw, &minh);
3748    evas_object_size_hint_min_set(obj, minw, minh);
3749
3750    evas_object_smart_callbacks_descriptions_set(obj, _signals);
3751    
3752    _mirrored_set(obj, elm_widget_mirrored_get(obj));
3753    _sizing_eval(obj);
3754    return obj;
3755 }
3756
3757 static Evas_Object *
3758 _item_content_get_hook(Elm_Genlist_Item *it, const char *part)
3759 {
3760    return edje_object_part_swallow_get(VIEW(it), part);
3761 }
3762
3763 static void
3764 _item_content_set_hook(Elm_Genlist_Item *it, const char *part, Evas_Object *content)
3765 {
3766    if (content && part)
3767      {
3768         it->content_objs = eina_list_append(it->content_objs, content);
3769         edje_object_part_swallow(VIEW(it), part, content);
3770      }
3771 }
3772
3773 static Evas_Object *
3774 _item_content_unset_hook(Elm_Genlist_Item *it, const char *part)
3775 {
3776    Evas_Object *obj;
3777    obj = edje_object_part_swallow_get(VIEW(it), part);
3778    if (!obj) return NULL;
3779    it->content_objs = eina_list_remove(it->content_objs, obj);
3780    edje_object_part_unswallow(VIEW(it), obj);
3781    return obj;
3782 }
3783
3784 static const char *
3785 _item_text_hook(Elm_Genlist_Item *it, const char *part)
3786 {
3787    if (!it->itc->func.text_get) return NULL;
3788    return edje_object_part_text_get(VIEW(it), part);
3789 }
3790
3791 static void
3792 _item_disable_hook(Elm_Object_Item *it)
3793 {
3794    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
3795    Eina_List *l;
3796    Evas_Object *obj;
3797    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
3798
3799    if (_it->delete_me) return;
3800
3801    if (_it->selected)
3802      elm_genlist_item_selected_set(it, EINA_FALSE);
3803
3804    if (_it->realized)
3805      {
3806         if (elm_widget_item_disabled_get(it))
3807           {
3808              edje_object_signal_emit(VIEW(_it), "elm,state,disabled", "elm");
3809              if (_it->edit_obj)
3810                edje_object_signal_emit(_it->edit_obj, "elm,state,disabled", "elm");
3811           }
3812         else
3813           {
3814              edje_object_signal_emit(VIEW(_it), "elm,state,enabled", "elm");
3815              if (_it->edit_obj)
3816                edje_object_signal_emit(_it->edit_obj, "elm,state,enabled", "elm");
3817           }
3818         EINA_LIST_FOREACH(_it->content_objs, l, obj)
3819           elm_widget_disabled_set(obj, elm_widget_item_disabled_get(_it));
3820      }
3821 }
3822
3823 static Eina_Bool
3824 _item_del_pre_hook(Elm_Object_Item *it)
3825 {
3826    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, EINA_FALSE);
3827    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
3828
3829    if ((_it->relcount > 0) || (_it->walking > 0))
3830      {
3831         elm_genlist_item_subitems_clear(it);
3832         if (_it->wd->show_item == _it) _it->wd->show_item = NULL;
3833         _elm_genlist_item_del_notserious(_it);
3834         if (_it->block)
3835           {
3836              if (_it->realized) _item_unrealize(_it, EINA_FALSE);
3837              _it->block->changed = EINA_TRUE;
3838              if (_it->wd->calc_job) ecore_job_del(_it->wd->calc_job);
3839              _it->wd->calc_job = ecore_job_add(_calc_job, _it->wd);
3840           }
3841         if (_it->parent)
3842           {
3843              _it->parent->items =
3844                 eina_list_remove(_it->parent->items, it);
3845              _it->parent = NULL;
3846           }
3847         return EINA_FALSE;
3848      }
3849    _item_del(_it);
3850    return EINA_TRUE;
3851 }
3852
3853 Elm_Genlist_Item *
3854 _elm_genlist_item_new(Widget_Data              *wd,
3855                       const Elm_Genlist_Item_Class *itc,
3856                       const void               *data,
3857                       Elm_Genlist_Item             *parent,
3858                       Evas_Smart_Cb             func,
3859                       const void               *func_data)
3860 {
3861    Elm_Genlist_Item *it;
3862
3863    it = elm_widget_item_new(wd->obj, Elm_Genlist_Item);
3864    if (!it) return NULL;
3865    it->wd = wd;
3866    //it->generation = wd->generation; //upstream
3867    it->itc = itc;
3868    it->base.data = data;
3869    it->parent = parent;
3870    it->func.func = func;
3871    it->func.data = func_data;
3872    elm_widget_item_content_get_hook_set(it, _item_content_get_hook);
3873    elm_widget_item_content_set_hook_set(it, _item_content_set_hook);
3874    elm_widget_item_content_unset_hook_set(it, _item_content_unset_hook);
3875    elm_widget_item_disable_hook_set(it, _item_disable_hook);
3876    elm_widget_item_del_pre_hook_set(it, _item_del_pre_hook);
3877    /* TEMPORARY */
3878 //   it->sel_cb = (Ecore_Cb)_item_select;
3879
3880    elm_widget_item_text_get_hook_set(it, _item_text_hook);
3881    return it;
3882 }
3883
3884 static Elm_Genlist_Item *
3885 _item_new(Widget_Data                  *wd,
3886           const Elm_Genlist_Item_Class *itc,
3887           const void                   *data,
3888           Elm_Genlist_Item             *parent,
3889           Elm_Genlist_Item_Flags        flags,
3890           Evas_Smart_Cb                 func,
3891           const void                   *func_data)
3892 {
3893    Elm_Genlist_Item *it, *it2;
3894    int depth = 0;
3895
3896    it = _elm_genlist_item_new(wd, itc, data, parent, func, func_data);
3897    if (!it) return NULL;
3898    it->flags = flags;
3899    it->mouse_cursor = NULL;
3900    it->expanded_depth = 0;
3901    it->can_focus = EINA_TRUE;
3902    if (it->parent)
3903      {
3904         if (it->parent->flags & ELM_GENLIST_ITEM_GROUP)
3905           it->group_item = parent;
3906         else if (it->parent->group_item)
3907           it->group_item = it->parent->group_item;
3908      }
3909    for (it2 = it, depth = 0; it2->parent; it2 = it2->parent)
3910      {
3911         if (it2->parent->flags != ELM_GENLIST_ITEM_GROUP) depth += 1;
3912      }
3913    it->expanded_depth = depth;
3914    return it;
3915 }
3916
3917 static void
3918 _item_block_add(Widget_Data      *wd,
3919                 Elm_Genlist_Item *it)
3920 {
3921    Item_Block *itb = NULL;
3922
3923    if (!it->rel)
3924      {
3925 newblock:
3926         if (it->rel)
3927           {
3928              itb = calloc(1, sizeof(Item_Block));
3929              if (!itb) return;
3930              itb->wd = wd;
3931              if (!it->rel->block)
3932                {
3933                   wd->blocks =
3934                     eina_inlist_append(wd->blocks, EINA_INLIST_GET(itb));
3935                   itb->items = eina_list_append(itb->items, it);
3936                   itb->position_update = EINA_TRUE;
3937                   it->position = eina_list_count(itb->items);
3938                   it->position_update = EINA_TRUE;
3939
3940                   if (wd->blocks != EINA_INLIST_GET(itb))
3941                     {
3942                        itb->position = ((Item_Block *) (EINA_INLIST_GET(itb)->prev))->position + 1;
3943                     }
3944                   else
3945                     {
3946                        itb->position = 0;
3947                     }
3948                }
3949              else
3950                {
3951                   Eina_List *tmp;
3952
3953                   tmp = eina_list_data_find_list(itb->items, it->rel);
3954                   if (it->before)
3955                     {
3956                        wd->blocks = eina_inlist_prepend_relative
3957                            (wd->blocks, EINA_INLIST_GET(itb),
3958                            EINA_INLIST_GET(it->rel->block));
3959                        itb->items =
3960                          eina_list_prepend_relative_list(itb->items, it, tmp);
3961
3962                        /* Update index from where we prepended */
3963                        _item_position_update(eina_list_prev(tmp), it->rel->position);
3964                        _item_block_position_update(EINA_INLIST_GET(itb),
3965                                                    it->rel->block->position);
3966                     }
3967                   else
3968                     {
3969                        wd->blocks = eina_inlist_append_relative
3970                            (wd->blocks, EINA_INLIST_GET(itb),
3971                            EINA_INLIST_GET(it->rel->block));
3972                        itb->items =
3973                          eina_list_append_relative_list(itb->items, it, tmp);
3974
3975                        /* Update block index from where we appended */
3976                        _item_position_update(eina_list_next(tmp), it->rel->position + 1);
3977                        _item_block_position_update(EINA_INLIST_GET(itb),
3978                                                    it->rel->block->position + 1);
3979                     }
3980                }
3981           }
3982         else
3983           {
3984              if (it->before)
3985                {
3986                   if (wd->blocks)
3987                     {
3988                        itb = (Item_Block *)(wd->blocks);
3989                        if (itb->count >= wd->max_items_per_block)
3990                          {
3991                             itb = calloc(1, sizeof(Item_Block));
3992                             if (!itb) return;
3993                             itb->wd = wd;
3994                             wd->blocks =
3995                               eina_inlist_prepend(wd->blocks,
3996                                                   EINA_INLIST_GET(itb));
3997                             _item_block_position_update(wd->blocks, 0);
3998                          }
3999                     }
4000                   else
4001                     {
4002                        itb = calloc(1, sizeof(Item_Block));
4003                        if (!itb) return;
4004                        itb->wd = wd;
4005                        wd->blocks =
4006                          eina_inlist_prepend(wd->blocks, EINA_INLIST_GET(itb));
4007                        _item_block_position_update(wd->blocks, 0);
4008                     }
4009                   itb->items = eina_list_prepend(itb->items, it);
4010
4011                   _item_position_update(itb->items, 0);
4012                }
4013              else
4014                {
4015                   if (wd->blocks)
4016                     {
4017                        itb = (Item_Block *)(wd->blocks->last);
4018                        if (itb->count >= wd->max_items_per_block)
4019                          {
4020                             itb = calloc(1, sizeof(Item_Block));
4021                             if (!itb) return;
4022                             itb->wd = wd;
4023                             wd->blocks =
4024                               eina_inlist_append(wd->blocks,
4025                                                  EINA_INLIST_GET(itb));
4026                             itb->position_update = EINA_TRUE;
4027                             if (wd->blocks != EINA_INLIST_GET(itb))
4028                               {
4029                                  itb->position = ((Item_Block *) (EINA_INLIST_GET(itb)->prev))->position + 1;
4030                               }
4031                             else
4032                               {
4033                                  itb->position = 0;
4034                               }
4035                          }
4036                     }
4037                   else
4038                     {
4039                        itb = calloc(1, sizeof(Item_Block));
4040                        if (!itb) return;
4041                        itb->wd = wd;
4042                        wd->blocks =
4043                          eina_inlist_append(wd->blocks, EINA_INLIST_GET(itb));
4044                        itb->position_update = EINA_TRUE;
4045                        if (wd->blocks != EINA_INLIST_GET(itb))
4046                          {
4047                             itb->position = ((Item_Block *) (EINA_INLIST_GET(itb)->prev))->position + 1;
4048                          }
4049                        else
4050                          {
4051                             itb->position = 0;
4052                          }
4053                     }
4054                   itb->items = eina_list_append(itb->items, it);
4055                   it->position = eina_list_count(itb->items);
4056                }
4057           }
4058      }
4059    else
4060      {
4061         Eina_List *tmp;
4062
4063         itb = it->rel->block;
4064         tmp = eina_list_data_find_list(itb->items, it->rel);
4065         if (!itb) goto newblock;
4066         if (it->before)
4067           {
4068              itb->items = eina_list_prepend_relative_list(itb->items, it, tmp);
4069              _item_position_update(eina_list_prev(tmp), it->rel->position);
4070           }
4071         else
4072           {
4073              itb->items = eina_list_append_relative_list(itb->items, it, tmp);
4074              _item_position_update(eina_list_next(tmp), it->rel->position + 1);
4075           }
4076      }
4077    itb->count++;
4078    itb->changed = EINA_TRUE;
4079    it->block = itb;
4080    if (itb->wd->calc_job) ecore_job_del(itb->wd->calc_job);
4081    itb->wd->calc_job = ecore_job_add(_calc_job, itb->wd);
4082    if (it->rel)
4083      {
4084         it->rel->relcount--;
4085         if ((it->rel->delete_me) && (!it->rel->relcount))
4086           _item_del(it->rel);
4087         it->rel = NULL;
4088      }
4089    if (itb->count > itb->wd->max_items_per_block)
4090      {
4091         int newc;
4092         Item_Block *itb2;
4093         Elm_Genlist_Item *it2;
4094
4095         newc = itb->count / 2;
4096         itb2 = calloc(1, sizeof(Item_Block));
4097         if (!itb2) return;
4098         itb2->wd = wd;
4099         wd->blocks =
4100           eina_inlist_append_relative(wd->blocks, EINA_INLIST_GET(itb2),
4101                                       EINA_INLIST_GET(itb));
4102         itb2->changed = EINA_TRUE;
4103         while ((itb->count > newc) && (itb->items))
4104           {
4105              Eina_List *l;
4106
4107              l = eina_list_last(itb->items);
4108              it2 = l->data;
4109              itb->items = eina_list_remove_list(itb->items, l);
4110              itb->count--;
4111
4112              itb2->items = eina_list_prepend(itb2->items, it2);
4113              it2->block = itb2;
4114              itb2->count++;
4115           }
4116      }
4117 }
4118
4119 static int
4120 _queue_process(Widget_Data *wd)
4121 {
4122    int n;
4123    Eina_Bool showme = EINA_FALSE;
4124    double t0, t;
4125
4126    t0 = ecore_loop_time_get();
4127    //evas_event_freeze(evas_object_evas_get(wd->obj));
4128    for (n = 0; (wd->queue) && (n < 128); n++)
4129      {
4130         Elm_Genlist_Item *it;
4131
4132         it = eina_list_data_get(wd->queue);
4133         wd->queue = eina_list_remove_list(wd->queue, wd->queue);
4134         it->queued = EINA_FALSE;
4135         _item_block_add(wd, it);
4136         if (!wd->blocks)
4137           _item_block_realize(it->block);
4138         t = ecore_time_get();
4139         if (it->block->changed)
4140           {
4141              showme = _item_block_recalc(it->block, it->block->num, EINA_TRUE);
4142              it->block->changed = 0;
4143              if (wd->pan_changed)
4144                {
4145                   if (wd->calc_job) ecore_job_del(wd->calc_job);
4146                   wd->calc_job = NULL;
4147                   _calc_job(wd);
4148                   wd->pan_changed = EINA_FALSE;
4149                }
4150           }
4151         if (showme) it->block->showme = EINA_TRUE;
4152         if (eina_inlist_count(wd->blocks) > 1)
4153           {
4154              if ((t - t0) > (ecore_animator_frametime_get())) break;
4155           }
4156      }
4157    //evas_event_thaw(evas_object_evas_get(wd->obj));
4158    //evas_event_thaw_eval(evas_object_evas_get(wd->obj));
4159    return n;
4160 }
4161
4162 static Eina_Bool
4163 _idle_process(void *data, Eina_Bool *wakeup)
4164 {
4165    Widget_Data *wd = data;
4166
4167    //xxx
4168    //static double q_start = 0.0;
4169    //if (q_start == 0.0) q_start = ecore_time_get();
4170    //xxx
4171    if (_queue_process(wd) > 0) *wakeup = EINA_TRUE;
4172    if (!wd->queue)
4173      {
4174         //xxx
4175         //printf("PROCESS TIME: %3.3f\n", ecore_time_get() - q_start);
4176         //xxx
4177         return ECORE_CALLBACK_CANCEL;
4178      }
4179    return ECORE_CALLBACK_RENEW;
4180 }
4181
4182 static Eina_Bool
4183 _item_idle_enterer(void *data)
4184 {
4185    Widget_Data *wd = data;
4186    Eina_Bool wakeup = EINA_FALSE;
4187    Eina_Bool ok = _idle_process(data, &wakeup);
4188
4189    if (wakeup)
4190      {
4191         // wake up mainloop
4192         if (wd->calc_job) ecore_job_del(wd->calc_job);
4193         wd->calc_job = ecore_job_add(_calc_job, wd);
4194      }
4195    if (ok == ECORE_CALLBACK_CANCEL) wd->queue_idle_enterer = NULL;
4196    return ok;
4197 }
4198
4199 static void
4200 _item_queue(Widget_Data      *wd,
4201             Elm_Genlist_Item *it)
4202 {
4203    if (it->queued) return;
4204    it->queued = EINA_TRUE;
4205    wd->queue = eina_list_append(wd->queue, it);
4206 // FIXME: why does a freeze then thaw here cause some genlist
4207 // elm_genlist_item_append() to be much much slower?
4208 //   evas_event_freeze(evas_object_evas_get(wd->obj));
4209    while ((wd->queue) && ((!wd->blocks) || (!wd->blocks->next)))
4210      {
4211         if (wd->queue_idle_enterer)
4212           {
4213              ecore_idle_enterer_del(wd->queue_idle_enterer);
4214              wd->queue_idle_enterer = NULL;
4215           }
4216         _queue_process(wd);
4217      }
4218 //   evas_event_thaw(evas_object_evas_get(wd->obj));
4219 //   evas_event_thaw_eval(evas_object_evas_get(wd->obj));
4220    if (!wd->queue_idle_enterer)
4221      wd->queue_idle_enterer = ecore_idle_enterer_add(_item_idle_enterer, wd);
4222 }
4223
4224 static void
4225 _item_move_after(Elm_Genlist_Item *it, Elm_Genlist_Item *after)
4226 {
4227    if (!it) return;
4228    if (!after) return;
4229    Elm_Object_Item *_it = (Elm_Object_Item *) it;
4230    Elm_Object_Item *_after = (Elm_Object_Item *) after;
4231
4232    it->wd->items = eina_inlist_remove(it->wd->items, EINA_INLIST_GET(it));
4233    it->wd->reorder_deleted = EINA_TRUE;
4234    _item_block_del(it);
4235
4236    it->wd->items = eina_inlist_append_relative(it->wd->items, EINA_INLIST_GET(it), EINA_INLIST_GET(after));
4237    it->rel = after;
4238    it->rel->relcount++;
4239    it->before = EINA_FALSE;
4240    if (after->group_item) it->group_item = after->group_item;
4241    _item_queue(it->wd, it);
4242
4243    // TODO: change this to smart callback
4244    if (it->itc->func.moved)
4245      it->itc->func.moved(WIDGET(it), _it, _after, EINA_TRUE);
4246 }
4247
4248 static void
4249 _item_move_before(Elm_Genlist_Item *it, Elm_Genlist_Item *before)
4250 {
4251    if (!it) return;
4252    if (!before) return;
4253    Elm_Object_Item *_it = (Elm_Object_Item *) it;
4254    Elm_Object_Item *_before = (Elm_Object_Item *) before;
4255
4256    it->wd->items = eina_inlist_remove(it->wd->items, EINA_INLIST_GET(it));
4257    it->wd->reorder_deleted = EINA_TRUE;
4258    _item_block_del(it);
4259    it->wd->items = eina_inlist_prepend_relative(it->wd->items, EINA_INLIST_GET(it), EINA_INLIST_GET(before));
4260    it->rel = before;
4261    it->rel->relcount++;
4262    it->before = EINA_TRUE;
4263    if (before->group_item) it->group_item = before->group_item;
4264    _item_queue(it->wd, it);
4265
4266    // TODO: change this to smart callback
4267    if (it->itc->func.moved)
4268      it->itc->func.moved(WIDGET(it), _it, _before, EINA_FALSE);
4269 }
4270
4271 EAPI Elm_Object_Item *
4272 elm_genlist_item_append(Evas_Object                  *obj,
4273                         const Elm_Genlist_Item_Class *itc,
4274                         const void                   *data,
4275                         Elm_Object_Item              *parent,
4276                         Elm_Genlist_Item_Flags        flags,
4277                         Evas_Smart_Cb                 func,
4278                         const void                   *func_data)
4279 {
4280    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4281    Widget_Data *wd = elm_widget_data_get(obj);
4282    if (!wd) return NULL;
4283    Elm_Genlist_Item *it = _item_new(wd, itc, data, (Elm_Genlist_Item *)parent,
4284                                     flags, func, func_data);
4285    if (!it) return NULL;
4286    if (!it->parent)
4287      {
4288         if (flags & ELM_GENLIST_ITEM_GROUP)
4289           wd->group_items = eina_list_append(wd->group_items, it);
4290         wd->items = eina_inlist_append(wd->items, EINA_INLIST_GET(it));
4291         it->rel = NULL;
4292      }
4293    else
4294      {
4295         Elm_Genlist_Item *it2 = NULL;
4296         Eina_List *ll = eina_list_last(it->parent->items);
4297         if (ll) it2 = ll->data;
4298         it->parent->items = eina_list_append(it->parent->items, it);
4299         if (!it2) it2 = it->parent;
4300         if (it2->delete_me)
4301            it2 = (Elm_Genlist_Item *) elm_genlist_item_prev_get((Elm_Object_Item *) it2);
4302         wd->items =
4303           eina_inlist_append_relative(wd->items, EINA_INLIST_GET(it),
4304                                       EINA_INLIST_GET(it2));
4305         it->rel = it2;
4306         it->rel->relcount++;
4307      }
4308    it->before = EINA_FALSE;
4309    _item_queue(wd, it);
4310    return (Elm_Object_Item *) it;
4311 }
4312
4313 EAPI Elm_Object_Item *
4314 elm_genlist_item_prepend(Evas_Object                  *obj,
4315                          const Elm_Genlist_Item_Class *itc,
4316                          const void                   *data,
4317                          Elm_Object_Item              *parent,
4318                          Elm_Genlist_Item_Flags        flags,
4319                          Evas_Smart_Cb                 func,
4320                          const void                   *func_data)
4321 {
4322    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4323    Widget_Data *wd = elm_widget_data_get(obj);
4324    if (!wd) return NULL;
4325    Elm_Genlist_Item *it = _item_new(wd, itc, data, (Elm_Genlist_Item *)parent,
4326                                     flags, func, func_data);
4327
4328    if (!it) return NULL;
4329    if (!it->parent)
4330      {
4331         if (flags & ELM_GENLIST_ITEM_GROUP)
4332           wd->group_items = eina_list_prepend(wd->group_items, it);
4333         wd->items = eina_inlist_prepend(wd->items, EINA_INLIST_GET(it));
4334         it->rel = NULL;
4335      }
4336    else
4337      {
4338         Elm_Genlist_Item *it2 = NULL;
4339         Eina_List *ll = it->parent->items;
4340         if (ll) it2 = ll->data;
4341         it->parent->items = eina_list_prepend(it->parent->items, it);
4342         if (!it2) it2 = it->parent;
4343         if (it2->delete_me)
4344            it2 = (Elm_Genlist_Item *) elm_genlist_item_next_get((Elm_Object_Item *) it2);
4345         wd->items =
4346           eina_inlist_prepend_relative(wd->items, EINA_INLIST_GET(it),
4347                                        EINA_INLIST_GET(it2));
4348         it->rel = it2;
4349         it->rel->relcount++;
4350      }
4351    it->before = EINA_TRUE;
4352    _item_queue(wd, it);
4353    return (Elm_Object_Item *) it;
4354 }
4355
4356 EAPI Elm_Object_Item *
4357 elm_genlist_item_insert_after(Evas_Object                  *obj,
4358                               const Elm_Genlist_Item_Class *itc,
4359                               const void                   *data,
4360                               Elm_Object_Item              *parent,
4361                               Elm_Object_Item              *after,
4362                               Elm_Genlist_Item_Flags        flags,
4363                               Evas_Smart_Cb                 func,
4364                               const void                   *func_data)
4365 {
4366    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4367    ELM_OBJ_ITEM_CHECK_OR_RETURN(after, NULL);
4368    Widget_Data *wd = elm_widget_data_get(obj);
4369    Elm_Genlist_Item *_after = (Elm_Genlist_Item *) after;
4370    if (!wd) return NULL;
4371    EINA_SAFETY_ON_NULL_RETURN_VAL(wd->items, NULL);
4372
4373    Elm_Genlist_Item *it = _item_new(wd, itc, data, (Elm_Genlist_Item *)parent,
4374                                     flags, func, func_data);
4375    if (!it) return NULL;
4376    if (!it->parent)
4377      {
4378         if ((flags & ELM_GENLIST_ITEM_GROUP) &&
4379             (((Elm_Genlist_Item * )after)->flags & ELM_GENLIST_ITEM_GROUP))
4380           wd->group_items = eina_list_append_relative(wd->group_items, it,
4381                                                       after);
4382      }
4383    else
4384      {
4385         it->parent->items = eina_list_append_relative(it->parent->items, it,
4386                                                       after);
4387      }
4388    wd->items = eina_inlist_append_relative(wd->items, EINA_INLIST_GET(it),
4389                                            EINA_INLIST_GET(_after));
4390    it->rel = _after;
4391    it->rel->relcount++;
4392    it->before = EINA_FALSE;
4393    _item_queue(wd, it);
4394    return (Elm_Object_Item *) it;
4395 }
4396
4397 EAPI Elm_Object_Item *
4398 elm_genlist_item_insert_before(Evas_Object                  *obj,
4399                                const Elm_Genlist_Item_Class *itc,
4400                                const void                   *data,
4401                                Elm_Object_Item              *parent,
4402                                Elm_Object_Item              *before,
4403                                Elm_Genlist_Item_Flags        flags,
4404                                Evas_Smart_Cb                 func,
4405                                const void                   *func_data)
4406 {
4407    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4408    ELM_OBJ_ITEM_CHECK_OR_RETURN(before, NULL);
4409    Widget_Data *wd = elm_widget_data_get(obj);
4410    Elm_Genlist_Item *_before = (Elm_Genlist_Item *) before;
4411    if (!wd) return NULL;
4412
4413    EINA_SAFETY_ON_NULL_RETURN_VAL(wd->items, NULL);
4414
4415    Elm_Genlist_Item *it = _item_new(wd, itc, data, (Elm_Genlist_Item *)parent,
4416                                     flags, func, func_data);
4417    if (!it) return NULL;
4418    if (!it->parent)
4419      {
4420         if ((flags & ELM_GENLIST_ITEM_GROUP) &&
4421             (_before->flags & ELM_GENLIST_ITEM_GROUP))
4422           wd->group_items = eina_list_prepend_relative(wd->group_items, it,
4423                                                        _before);
4424      }
4425    else
4426      {
4427         it->parent->items = eina_list_prepend_relative(it->parent->items, it,
4428                                                        _before);
4429      }
4430    wd->items = eina_inlist_prepend_relative(wd->items, EINA_INLIST_GET(it),
4431                                             EINA_INLIST_GET(_before));
4432    it->rel = _before;
4433    it->rel->relcount++;
4434    it->before = EINA_TRUE;
4435    _item_queue(wd, it);
4436    return (Elm_Object_Item *) it;
4437 }
4438
4439 static void
4440 _elm_genlist_clear(Evas_Object *obj, Eina_Bool standby)
4441 {
4442    ELM_CHECK_WIDTYPE(obj, widtype);
4443    Widget_Data *wd = elm_widget_data_get(obj);
4444    if (!wd) return;
4445    if (wd->walking > 0)
4446      {
4447         Elm_Genlist_Item *it;
4448
4449         wd->clear_me = EINA_TRUE;
4450         EINA_INLIST_FOREACH(wd->items, it)
4451           {
4452              it->delete_me = EINA_TRUE;
4453           }
4454         return;
4455      }
4456    evas_event_freeze(evas_object_evas_get(wd->obj));
4457    while (wd->items)
4458      {
4459         Elm_Genlist_Item *it = ELM_GENLIST_ITEM_FROM_INLIST(wd->items);
4460         it->nocache = EINA_TRUE;
4461         it->highlighted = EINA_FALSE;
4462 #ifdef ANCHOR_ITEM
4463         if (wd->anchor_item == it)
4464           {
4465              wd->anchor_item = ELM_GENLIST_ITEM_FROM_INLIST(EINA_INLIST_GET(it)->next);
4466              if (!wd->anchor_item)
4467                wd->anchor_item =
4468                  ELM_GENLIST_ITEM_FROM_INLIST(EINA_INLIST_GET(it)->prev);
4469           }
4470 #endif
4471         wd->items = eina_inlist_remove(wd->items, wd->items);
4472         if (it->flags & ELM_GENLIST_ITEM_GROUP)
4473           it->wd->group_items = eina_list_remove(it->wd->group_items, it);
4474         elm_widget_item_pre_notify_del(it);
4475         if (it->realized) _item_unrealize(it, EINA_FALSE);
4476         if (((wd->clear_me) || (!it->delete_me)) && (it->itc->func.del))
4477           it->itc->func.del((void *)it->base.data, WIDGET(it));
4478         if (it->long_timer) ecore_timer_del(it->long_timer);
4479         if (it->swipe_timer) ecore_timer_del(it->swipe_timer);
4480 //        elm_widget_item_del(it);
4481      }
4482    wd->clear_me = EINA_FALSE;
4483    wd->anchor_item = NULL;
4484    while (wd->blocks)
4485      {
4486         Item_Block *itb = (Item_Block *)(wd->blocks);
4487
4488         wd->blocks = eina_inlist_remove(wd->blocks, wd->blocks);
4489         if (itb->items) eina_list_free(itb->items);
4490         free(itb);
4491      }
4492    wd->pan_changed = EINA_TRUE;
4493    if (wd->calc_job)
4494      {
4495         ecore_job_del(wd->calc_job);
4496         wd->calc_job = NULL;
4497      }
4498    if (wd->queue_idle_enterer)
4499      {
4500         ecore_idle_enterer_del(wd->queue_idle_enterer);
4501         wd->queue_idle_enterer = NULL;
4502      }
4503    if (wd->must_recalc_idler)
4504      {
4505         ecore_idler_del(wd->must_recalc_idler);
4506         wd->must_recalc_idler = NULL;
4507      }
4508    if (wd->queue)
4509      {
4510         eina_list_free(wd->queue);
4511         wd->queue = NULL;
4512      }
4513    if (wd->selected)
4514      {
4515         eina_list_free(wd->selected);
4516         wd->selected = NULL;
4517      }
4518    if (wd->reorder_move_animator)
4519      {
4520         ecore_animator_del(wd->reorder_move_animator);
4521         wd->reorder_move_animator = NULL;
4522      }
4523    wd->show_item = NULL;
4524    wd->pan_x = 0;
4525    wd->pan_y = 0;
4526    wd->old_pan_y = 0;
4527    wd->minw = 0;
4528    wd->minh = 0;
4529
4530    if (wd->alpha_bg)
4531       evas_object_del(wd->alpha_bg);
4532    wd->alpha_bg = NULL;
4533
4534    if (wd->pan_smart)
4535      {
4536         evas_object_size_hint_min_set(wd->pan_smart, wd->minw, wd->minh);
4537         evas_object_smart_callback_call(wd->pan_smart, "changed", NULL);
4538      }
4539    _sizing_eval(obj);
4540    elm_smart_scroller_child_region_show(wd->scr, 0, 0, 0, 0);
4541    evas_event_thaw(evas_object_evas_get(wd->obj));
4542    evas_event_thaw_eval(evas_object_evas_get(wd->obj));
4543 }
4544
4545 EAPI void
4546 elm_genlist_clear(Evas_Object *obj)
4547 {
4548    _elm_genlist_clear(obj, EINA_FALSE);
4549 }
4550
4551 EAPI void
4552 elm_genlist_multi_select_set(Evas_Object *obj,
4553                              Eina_Bool    multi)
4554 {
4555    ELM_CHECK_WIDTYPE(obj, widtype);
4556    Widget_Data *wd = elm_widget_data_get(obj);
4557    if (!wd) return;
4558    wd->multi = multi;
4559 }
4560
4561 EAPI Eina_Bool
4562 elm_genlist_multi_select_get(const Evas_Object *obj)
4563 {
4564    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
4565    Widget_Data *wd = elm_widget_data_get(obj);
4566    if (!wd) return EINA_FALSE;
4567    return wd->multi;
4568 }
4569
4570 EAPI Elm_Object_Item *
4571 elm_genlist_selected_item_get(const Evas_Object *obj)
4572 {
4573    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4574    Widget_Data *wd = elm_widget_data_get(obj);
4575    if (!wd) return NULL;
4576    if (wd->selected) return wd->selected->data;
4577    return NULL;
4578 }
4579
4580 EAPI const Eina_List *
4581 elm_genlist_selected_items_get(const Evas_Object *obj)
4582 {
4583    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4584    Widget_Data *wd = elm_widget_data_get(obj);
4585    if (!wd) return NULL;
4586    return wd->selected;
4587 }
4588
4589 EAPI Eina_List *
4590 elm_genlist_realized_items_get(const Evas_Object *obj)
4591 {
4592    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4593    Widget_Data *wd = elm_widget_data_get(obj);
4594    Eina_List *list = NULL;
4595    Item_Block *itb;
4596    Eina_Bool done = EINA_FALSE;
4597    if (!wd) return NULL;
4598    EINA_INLIST_FOREACH(wd->blocks, itb)
4599      {
4600         if (itb->realized)
4601           {
4602              Eina_List *l;
4603              Elm_Genlist_Item *it;
4604
4605              done = 1;
4606              EINA_LIST_FOREACH(itb->items, l, it)
4607                {
4608                   if (it->realized) list = eina_list_append(list, it);
4609                }
4610           }
4611         else
4612           {
4613              if (done) break;
4614           }
4615      }
4616    return list;
4617 }
4618
4619 EAPI Elm_Object_Item *
4620 elm_genlist_at_xy_item_get(const Evas_Object *obj,
4621                            Evas_Coord         x,
4622                            Evas_Coord         y,
4623                            int               *posret)
4624 {
4625    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4626    Widget_Data *wd = elm_widget_data_get(obj);
4627    Evas_Coord ox, oy, ow, oh;
4628    Item_Block *itb;
4629    Evas_Coord lasty;
4630    if (!wd) return NULL;
4631    evas_object_geometry_get(wd->pan_smart, &ox, &oy, &ow, &oh);
4632    lasty = oy;
4633    EINA_INLIST_FOREACH(wd->blocks, itb)
4634      {
4635         Eina_List *l;
4636         Elm_Genlist_Item *it;
4637
4638         if (!ELM_RECTS_INTERSECT(ox + itb->x - itb->wd->pan_x,
4639                                  oy + itb->y - itb->wd->pan_y,
4640                                  itb->w, itb->h, x, y, 1, 1))
4641           continue;
4642         EINA_LIST_FOREACH(itb->items, l, it)
4643           {
4644              Evas_Coord itx, ity;
4645
4646              itx = ox + itb->x + it->x - itb->wd->pan_x;
4647              ity = oy + itb->y + it->y - itb->wd->pan_y;
4648              if (ELM_RECTS_INTERSECT(itx, ity, it->w, it->h, x, y, 1, 1))
4649                {
4650                   if (posret)
4651                     {
4652                        if (y <= (ity + (it->h / 4))) *posret = -1;
4653                        else if (y >= (ity + it->h - (it->h / 4)))
4654                          *posret = 1;
4655                        else *posret = 0;
4656                     }
4657                   return (Elm_Object_Item *) it;
4658                }
4659              lasty = ity + it->h;
4660           }
4661      }
4662    if (posret)
4663      {
4664         if (y > lasty) *posret = 1;
4665         else *posret = -1;
4666      }
4667    return NULL;
4668 }
4669
4670 EAPI Elm_Object_Item *
4671 elm_genlist_first_item_get(const Evas_Object *obj)
4672 {
4673    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4674    Widget_Data *wd = elm_widget_data_get(obj);
4675    if (!wd) return NULL;
4676    if (!wd->items) return NULL;
4677    Elm_Genlist_Item *it = ELM_GENLIST_ITEM_FROM_INLIST(wd->items);
4678    while ((it) && (it->delete_me))
4679      it = ELM_GENLIST_ITEM_FROM_INLIST(EINA_INLIST_GET(it)->next);
4680    return (Elm_Object_Item *) it;
4681 }
4682
4683 EAPI Elm_Object_Item *
4684 elm_genlist_last_item_get(const Evas_Object *obj)
4685 {
4686    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
4687    Widget_Data *wd = elm_widget_data_get(obj);
4688    if (!wd) return NULL;
4689    if (!wd->items) return NULL;
4690    Elm_Genlist_Item *it = ELM_GENLIST_ITEM_FROM_INLIST(wd->items->last);
4691    while ((it) && (it->delete_me))
4692      it = ELM_GENLIST_ITEM_FROM_INLIST(EINA_INLIST_GET(it)->prev);
4693    return (Elm_Object_Item *) it;
4694 }
4695
4696 EAPI Elm_Object_Item *
4697 elm_genlist_item_next_get(const Elm_Object_Item *it)
4698 {
4699    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, NULL);
4700    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
4701    while (_it)
4702      {
4703         _it = ELM_GENLIST_ITEM_FROM_INLIST(EINA_INLIST_GET(_it)->next);
4704         if ((_it) && (!_it->delete_me)) break;
4705      }
4706    return (Elm_Object_Item *) _it;
4707 }
4708
4709 EAPI Elm_Object_Item *
4710 elm_genlist_item_prev_get(const Elm_Object_Item *it)
4711 {
4712    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, NULL);
4713    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
4714    while (_it)
4715      {
4716         _it = ELM_GENLIST_ITEM_FROM_INLIST(EINA_INLIST_GET(_it)->prev);
4717         if ((_it) && (!_it->delete_me)) break;
4718      }
4719    return (Elm_Object_Item *)_it;
4720 }
4721
4722 EAPI Evas_Object *
4723 elm_genlist_item_genlist_get(const Elm_Object_Item *it)
4724 {
4725    return elm_object_item_widget_get(it);
4726 }
4727
4728 EAPI Elm_Object_Item *
4729 elm_genlist_item_parent_get(const Elm_Object_Item *it)
4730 {
4731    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, NULL);
4732    return (Elm_Object_Item *) ((Elm_Genlist_Item *) it)->parent;
4733 }
4734
4735 EAPI void
4736 elm_genlist_item_subitems_clear(Elm_Object_Item *it)
4737 {
4738    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
4739    Widget_Data *wd = elm_widget_data_get(WIDGET(it));
4740    if (!wd) return;
4741    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
4742    Elm_Genlist_Item *it2, *it3;
4743    Evas_Coord y, h;
4744
4745    if (!wd->effect_mode || !wd->move_effect_mode)
4746      _item_subitems_clear(_it);
4747    else
4748      {
4749         if ((!wd->item_moving_effect_timer) && (_it->flags != ELM_GENLIST_ITEM_GROUP) &&
4750              wd->move_effect_mode != ELM_GENLIST_ITEM_MOVE_EFFECT_DELETE)
4751           {
4752              wd->expanded_item = _it;
4753              _item_flip_effect_show(_it);
4754              evas_object_geometry_get(VIEW(it), NULL, &y, NULL, &h);
4755              wd->expand_item_end = y + h;
4756
4757              it2 = _it;
4758              do {
4759                 it2 = (Elm_Genlist_Item *) elm_genlist_item_next_get((Elm_Object_Item *) it2);
4760                 if (!it2) break;
4761              } while (it2->expanded_depth > _it->expanded_depth);
4762              if (it2)
4763                {
4764                   int diff;
4765                   Evas_Coord vy, vh;
4766                   evas_output_viewport_get(evas_object_evas_get(wd->pan_smart), NULL, &vy, NULL, &vh);
4767                   diff = it2->old_scrl_y - (vy + vh);
4768                   if (diff > 0)
4769                     {
4770                        it3 = it2;
4771                        while (it3)
4772                          {
4773                             it3->old_scrl_y = it3->old_scrl_y - diff;
4774                             it3 = (Elm_Genlist_Item *) elm_genlist_item_next_get((Elm_Object_Item *) it3);
4775                          }
4776                     }
4777
4778                   wd->expand_item_gap = wd->expand_item_end - it2->old_scrl_y;
4779                }
4780              else
4781                 wd->expand_item_gap = 0;
4782
4783              evas_object_raise(wd->alpha_bg);
4784              evas_object_show(wd->alpha_bg);
4785
4786              wd->start_time = current_time_get();
4787              wd->item_moving_effect_timer = ecore_animator_add(_item_moving_effect_timer_cb, wd);
4788           }
4789         else
4790            _item_subitems_clear(_it);
4791      }
4792 }
4793
4794 EAPI void
4795 elm_genlist_item_selected_set(Elm_Object_Item *it,
4796                               Eina_Bool         selected)
4797 {
4798    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
4799    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
4800    Widget_Data *wd = _it->wd;
4801    if (!wd) return;
4802    if ((_it->delete_me) || elm_widget_item_disabled_get(_it))
4803      return;
4804    selected = !!selected;
4805    if (_it->selected == selected) return;
4806
4807    if (selected)
4808      {
4809         if (!wd->multi)
4810           {
4811              while (wd->selected)
4812                {
4813                   _item_unhighlight(wd->selected->data);
4814                   _item_unselect(wd->selected->data);
4815                }
4816           }
4817         _item_highlight(_it);
4818         _item_select(_it);
4819         return;
4820      }
4821    _item_unhighlight(_it);
4822    _item_unselect(_it);
4823 }
4824
4825 EAPI Eina_Bool
4826 elm_genlist_item_selected_get(const Elm_Object_Item *it)
4827 {
4828    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, EINA_FALSE);
4829    return ((Elm_Genlist_Item *) it)->selected;
4830 }
4831
4832 EAPI void
4833 elm_genlist_item_expanded_set(Elm_Object_Item *it,
4834                               Eina_Bool         expanded)
4835 {
4836    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
4837    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
4838    if (_it->flags != ELM_GENLIST_ITEM_SUBITEMS) return;
4839    if (_it->expanded == expanded) return;
4840    _it->expanded = expanded;
4841    _it->wd->expanded_item = _it;
4842    if (_it->wd->effect_mode && !_it->wd->alpha_bg)
4843       _it->wd->alpha_bg = _create_tray_alpha_bg(WIDGET(_it));
4844
4845    if (_it->expanded)
4846      {
4847         _it->wd->move_effect_mode = ELM_GENLIST_ITEM_MOVE_EFFECT_EXPAND;
4848         _it->wd->expanded_move_effect_set = EINA_TRUE;
4849         _it->wd->expanded_next_item = (Elm_Genlist_Item *) elm_genlist_item_next_get(it);
4850         if (_it->realized)
4851           edje_object_signal_emit(VIEW(_it), "elm,state,expanded", "elm");
4852         evas_object_smart_callback_call(WIDGET(_it), SIG_EXPANDED, _it);
4853         _it->wd->auto_scroll_enabled = EINA_TRUE;
4854      }
4855    else
4856      {
4857         _it->wd->move_effect_mode = ELM_GENLIST_ITEM_MOVE_EFFECT_CONTRACT;
4858         _it->wd->expanded_next_item = NULL;
4859         if (_it->realized)
4860           edje_object_signal_emit(VIEW(_it), "elm,state,contracted", "elm");
4861         evas_object_smart_callback_call(WIDGET(_it), SIG_CONTRACTED, _it);
4862         _it->wd->auto_scroll_enabled = EINA_FALSE;
4863      }
4864 }
4865
4866 EAPI Eina_Bool
4867 elm_genlist_item_expanded_get(const Elm_Object_Item *it)
4868 {
4869    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, EINA_FALSE);
4870    return ((Elm_Genlist_Item *) it)->expanded;
4871 }
4872
4873 EAPI int
4874 elm_genlist_item_expanded_depth_get(const Elm_Object_Item *it)
4875 {
4876    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, 0);
4877    return ((Elm_Genlist_Item *) it)->expanded_depth;
4878 }
4879
4880 EAPI void
4881 elm_genlist_item_disabled_set(Elm_Object_Item *it,
4882                               Eina_Bool         disabled)
4883 {
4884    elm_object_item_disabled_set(it, disabled);
4885 }
4886
4887 EAPI Eina_Bool
4888 elm_genlist_item_disabled_get(const Elm_Object_Item *it)
4889 {
4890    return elm_object_item_disabled_get(it);
4891 }
4892
4893 EAPI void
4894 elm_genlist_item_display_only_set(Elm_Object_Item *it,
4895                                   Eina_Bool         display_only)
4896 {
4897    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
4898    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
4899    display_only = !!display_only;
4900 if (_it->display_only == display_only) return;
4901    if (_it->delete_me) return;
4902    _it->display_only = display_only;
4903    _it->mincalcd = EINA_FALSE;
4904    _it->updateme = EINA_TRUE;
4905    if (_it->block) _it->block->updateme = EINA_TRUE;
4906    if (_it->wd->update_job) ecore_job_del(_it->wd->update_job);
4907    _it->wd->update_job = ecore_job_add(_update_job, _it->wd);
4908 }
4909
4910 EAPI Eina_Bool
4911 elm_genlist_item_display_only_get(const Elm_Object_Item *it)
4912 {
4913    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, EINA_FALSE);
4914    if (((Elm_Genlist_Item *) it)->delete_me) return EINA_FALSE;
4915    return ((Elm_Genlist_Item *) it)->display_only;
4916 }
4917
4918 EAPI void
4919 elm_genlist_item_show(Elm_Object_Item *it)
4920 {
4921    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
4922    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
4923    if (_it->delete_me) return;
4924    if ((_it->queued) || (!_it->mincalcd))
4925      {
4926         _it->wd->show_item = _it;
4927         _it->wd->bring_in = EINA_FALSE;
4928         _it->wd->scrollto_type = ELM_GENLIST_ITEM_SCROLLTO_IN;
4929         _it->showme = EINA_TRUE;
4930         return;
4931      }
4932    if (_it->wd->show_item)
4933      {
4934         _it->wd->show_item->showme = EINA_FALSE;
4935         _it->wd->show_item = NULL;
4936      }
4937    elm_smart_scroller_child_region_show(_it->wd->scr,
4938                                         _it->x + _it->block->x,
4939                                         _it->y + _it->block->y,
4940                                         _it->block->w, _it->h);
4941 }
4942
4943 EAPI void
4944 elm_genlist_item_promote(Elm_Object_Item *it)
4945 {
4946    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
4947    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
4948    if (_it->delete_me) return;
4949    _item_move_before(_it,
4950                      (Elm_Genlist_Item *) elm_genlist_first_item_get(WIDGET(_it)));
4951 }
4952
4953 EAPI void
4954 elm_genlist_item_demote(Elm_Object_Item *it)
4955 {
4956    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
4957    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
4958    if (_it->delete_me) return;
4959    _item_move_after(_it,
4960                     (Elm_Genlist_Item *) elm_genlist_last_item_get(WIDGET(_it)));
4961 }
4962
4963 EAPI void
4964 elm_genlist_item_bring_in(Elm_Object_Item *it)
4965 {
4966    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
4967    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
4968    if (_it->delete_me) return;
4969    if ((_it->queued) || (!_it->mincalcd))
4970      {
4971         _it->wd->show_item = _it;
4972         _it->wd->bring_in = EINA_TRUE;
4973         _it->wd->scrollto_type = ELM_GENLIST_ITEM_SCROLLTO_IN;
4974         _it->showme = EINA_TRUE;
4975         return;
4976      }
4977    if (_it->wd->show_item)
4978      {
4979         _it->wd->show_item->showme = EINA_FALSE;
4980         _it->wd->show_item = NULL;
4981      }
4982    elm_smart_scroller_region_bring_in(_it->wd->scr,
4983                                       _it->x + _it->block->x,
4984                                       _it->y + _it->block->y,
4985                                       _it->block->w, _it->h);
4986 }
4987
4988 EAPI void
4989 elm_genlist_item_top_show(Elm_Object_Item *it)
4990 {
4991    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
4992    Evas_Coord ow, oh;
4993    Evas_Coord pan_w = 0, pan_h = 0;
4994    Eina_Bool pan_size_ready = EINA_FALSE;
4995    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
4996
4997    if (_it->delete_me) return;
4998    evas_object_geometry_get(_it->wd->pan_smart, NULL, NULL, &ow, &oh);
4999    if (_it->block)
5000      {
5001         _pan_child_size_get(_it->wd->pan_smart, &pan_w, &pan_h);
5002         if ((pan_w > (_it->x + _it->block->x)) && (pan_h > (_it->y + _it->block->y + oh)))
5003           {
5004              pan_size_ready = EINA_TRUE;
5005           }
5006      }
5007    if ((_it->queued) || (!_it->mincalcd) || (!pan_size_ready))
5008      {
5009         _it->wd->show_item = _it;
5010         _it->wd->bring_in = EINA_FALSE;
5011         _it->wd->scrollto_type = ELM_GENLIST_ITEM_SCROLLTO_TOP;
5012         _it->showme = EINA_TRUE;
5013         return;
5014      }
5015    if (_it->wd->show_item)
5016      {
5017         _it->wd->show_item->showme = EINA_FALSE;
5018         _it->wd->show_item = NULL;
5019      }
5020    elm_smart_scroller_child_region_show(_it->wd->scr,
5021                                         _it->x + _it->block->x,
5022                                         _it->y + _it->block->y,
5023                                         _it->block->w, oh);
5024 }
5025
5026 EAPI void
5027 elm_genlist_item_top_bring_in(Elm_Object_Item *it)
5028 {
5029    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5030    Evas_Coord ow, oh;
5031    Evas_Coord pan_w = 0, pan_h = 0;
5032    Eina_Bool pan_size_ready = EINA_FALSE;
5033    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5034
5035    if (_it->delete_me) return;
5036    evas_object_geometry_get(_it->wd->pan_smart, NULL, NULL, &ow, &oh);
5037    if (_it->block)
5038      {
5039         _pan_child_size_get(_it->wd->pan_smart, &pan_w, &pan_h);
5040         if ((pan_w > (_it->x + _it->block->x)) && (pan_h > (_it->y + _it->block->y + oh)))
5041           {
5042              pan_size_ready = EINA_TRUE;
5043           }
5044      }
5045    if ((_it->queued) || (!_it->mincalcd) || (!pan_size_ready))
5046      {
5047         _it->wd->show_item = _it;
5048         _it->wd->bring_in = EINA_TRUE;
5049         _it->wd->scrollto_type = ELM_GENLIST_ITEM_SCROLLTO_TOP;
5050         _it->showme = EINA_TRUE;
5051         return;
5052      }
5053    if (_it->wd->show_item)
5054      {
5055         _it->wd->show_item->showme = EINA_FALSE;
5056         _it->wd->show_item = NULL;
5057      }
5058    elm_smart_scroller_region_bring_in(_it->wd->scr,
5059                                       _it->x + _it->block->x,
5060                                       _it->y + _it->block->y,
5061                                       _it->block->w, oh);
5062 }
5063
5064 EAPI void
5065 elm_genlist_item_middle_show(Elm_Object_Item *it)
5066 {
5067    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5068    Evas_Coord ow, oh;
5069    Evas_Coord pan_w = 0, pan_h = 0;
5070    Eina_Bool pan_size_ready = EINA_FALSE;
5071    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5072
5073    if (_it->delete_me) return;
5074    evas_object_geometry_get(_it->wd->pan_smart, NULL, NULL, &ow, &oh);
5075    if (_it->block)
5076      {
5077         _pan_child_size_get(_it->wd->pan_smart, &pan_w, &pan_h);
5078         if ((pan_w > (_it->x + _it->block->x)) && (pan_h > (_it->y + _it->block->y + oh / 2)))
5079           {
5080              pan_size_ready = EINA_TRUE;
5081           }
5082      }
5083    if ((_it->queued) || (!_it->mincalcd) || (!pan_size_ready))
5084      {
5085         _it->wd->show_item = _it;
5086         _it->wd->bring_in = EINA_FALSE;
5087         _it->wd->scrollto_type = ELM_GENLIST_ITEM_SCROLLTO_MIDDLE;
5088         _it->showme = EINA_TRUE;
5089         return;
5090      }
5091    if (_it->wd->show_item)
5092      {
5093         _it->wd->show_item->showme = EINA_FALSE;
5094         _it->wd->show_item = NULL;
5095      }
5096    elm_smart_scroller_child_region_show(_it->wd->scr,
5097                                         _it->x + _it->block->x,
5098                                         _it->y + _it->block->y - oh / 2 +
5099                                         _it->h / 2, _it->block->w, oh);
5100 }
5101
5102 EAPI void
5103 elm_genlist_item_middle_bring_in(Elm_Object_Item *it)
5104 {
5105    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5106    Evas_Coord ow, oh;
5107    Evas_Coord pan_w = 0, pan_h = 0;
5108    Eina_Bool pan_size_ready = EINA_FALSE;
5109    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5110
5111    if (_it->delete_me) return;
5112    evas_object_geometry_get(_it->wd->pan_smart, NULL, NULL, &ow, &oh);
5113    if (_it->block)
5114      {
5115         _pan_child_size_get(_it->wd->pan_smart, &pan_w, &pan_h);
5116         if ((pan_w > (_it->x + _it->block->x)) && (pan_h > (_it->y + _it->block->y + oh / 2)))
5117           {
5118              pan_size_ready = EINA_TRUE;
5119           }
5120      }
5121    if ((_it->queued) || (!_it->mincalcd) || (!pan_size_ready))
5122      {
5123         _it->wd->show_item = _it;
5124         _it->wd->bring_in = EINA_TRUE;
5125         _it->wd->scrollto_type = ELM_GENLIST_ITEM_SCROLLTO_MIDDLE;
5126         _it->showme = EINA_TRUE;
5127         return;
5128      }
5129    if (_it->wd->show_item)
5130      {
5131         _it->wd->show_item->showme = EINA_FALSE;
5132         _it->wd->show_item = NULL;
5133      }
5134    elm_smart_scroller_region_bring_in(_it->wd->scr,
5135                                       _it->x + _it->block->x,
5136                                       _it->y + _it->block->y - oh / 2 + _it->h / 2,
5137                                       _it->block->w, oh);
5138 }
5139
5140 EAPI void
5141 elm_genlist_item_del(Elm_Object_Item *it)
5142 {
5143    elm_object_item_del(it);
5144 }
5145
5146 EAPI void
5147 elm_genlist_item_data_set(Elm_Object_Item *it,
5148                           const void       *data)
5149 {
5150    elm_object_item_data_set(it, (void *) data);
5151 }
5152
5153 EAPI void *
5154 elm_genlist_item_data_get(const Elm_Object_Item *it)
5155 {
5156    return elm_object_item_data_get(it);
5157 }
5158
5159 EAPI void
5160 elm_genlist_item_contents_orphan(Elm_Object_Item *it)
5161 {
5162    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5163
5164    Evas_Object *content;
5165    EINA_LIST_FREE(((Elm_Genlist_Item *) it)->content_objs, content)
5166      {
5167         elm_widget_sub_object_del(WIDGET(it), content);
5168         evas_object_smart_member_del(content);
5169         evas_object_hide(content);
5170      }
5171 }
5172
5173 EAPI const Evas_Object *
5174 elm_genlist_item_object_get(const Elm_Object_Item *it)
5175 {
5176    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, NULL);
5177    return VIEW(it);
5178 }
5179
5180 EAPI void
5181 elm_genlist_item_update(Elm_Object_Item *it)
5182 {
5183    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5184    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5185
5186    if (!_it->block) return;
5187    if (_it->delete_me) return;
5188    _it->mincalcd = EINA_FALSE;
5189    _it->updateme = EINA_TRUE;
5190    _it->block->updateme = EINA_TRUE;
5191    if (_it->wd->update_job) ecore_job_del(_it->wd->update_job);
5192    _it->wd->update_job = ecore_job_add(_update_job, _it->wd);
5193    _it->defer_unrealize = EINA_FALSE;
5194 }
5195
5196 EAPI void
5197 elm_genlist_item_fields_update(Elm_Object_Item *it,
5198                                const char *parts,
5199                                Elm_Genlist_Item_Field_Flags itf)
5200 {
5201    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5202    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5203
5204    if (!_it->block) return;
5205    if (_it->delete_me) return;
5206
5207    if ((!itf) || (itf & ELM_GENLIST_ITEM_FIELD_TEXT))
5208      _item_text_realize(_it, VIEW(_it), &_it->texts, parts);
5209    if ((!itf) || (itf & ELM_GENLIST_ITEM_FIELD_CONTENT))
5210      {
5211         _it->content_objs = _item_content_unrealize(_it, VIEW(_it),
5212                                                     &_it->contents, parts);
5213         _it->content_objs = _item_content_realize(_it, VIEW(_it),
5214                                                   &_it->contents, parts);
5215
5216         if (_it->renamed)
5217           {
5218              _it->content_objs = _item_mode_content_unrealize(_it, VIEW(_it),
5219                                                               &_it->contents, parts, &_it->content_objs);
5220              _it->content_objs = _item_mode_content_realize(_it, VIEW(_it),
5221                                                             &_it->contents, parts, &_it->content_objs, "renames");
5222           }
5223
5224         if (_it->mode_view)
5225           {
5226              _it->mode_content_objs = _item_mode_content_unrealize(_it, _it->mode_view,
5227                                                                    &_it->mode_contents, parts, &_it->mode_content_objs);
5228              _it->mode_content_objs = _item_mode_content_realize(_it, _it->mode_view,
5229                                                                  &_it->mode_contents, parts, &_it->mode_content_objs, "contents");
5230           }
5231
5232         if (_it->wd->edit_mode)
5233           {
5234              _it->edit_content_objs = _item_mode_content_unrealize(_it, _it->edit_obj,
5235                                                                    &_it->contents, parts, &_it->edit_content_objs);
5236              _it->edit_content_objs = _item_mode_content_realize(_it, _it->edit_obj,
5237                                                                  &_it->contents, parts, &_it->edit_content_objs, "edit_contents");
5238           }
5239      }
5240    if ((!itf) || (itf & ELM_GENLIST_ITEM_FIELD_STATE))
5241      _item_state_realize(_it, VIEW(_it), &_it->states, parts);
5242 }
5243
5244 EAPI void
5245 elm_genlist_item_item_class_update(Elm_Object_Item             *it,
5246                                    const Elm_Genlist_Item_Class *itc)
5247 {
5248    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5249    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5250
5251    if (!_it->block) return;
5252    EINA_SAFETY_ON_NULL_RETURN(itc);
5253    if (_it->delete_me) return;
5254    _it->itc = itc;
5255    _it->nocache = EINA_TRUE;
5256    elm_genlist_item_update(it);
5257 }
5258
5259 EAPI const Elm_Genlist_Item_Class *
5260 elm_genlist_item_item_class_get(const Elm_Object_Item *it)
5261 {
5262    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, NULL);
5263    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5264    if (_it->delete_me) return NULL;
5265    return _it->itc;
5266 }
5267
5268 static Evas_Object *
5269 _elm_genlist_item_label_create(void        *data,
5270                                Evas_Object *obj __UNUSED__,
5271                                Evas_Object *tooltip,
5272                                void        *it __UNUSED__)
5273 {
5274    Evas_Object *label = elm_label_add(tooltip);
5275    if (!label)
5276      return NULL;
5277    elm_object_style_set(label, "tooltip");
5278    elm_object_text_set(label, data);
5279    return label;
5280 }
5281
5282 static void
5283 _elm_genlist_item_label_del_cb(void        *data,
5284                                Evas_Object *obj __UNUSED__,
5285                                void        *event_info __UNUSED__)
5286 {
5287    eina_stringshare_del(data);
5288 }
5289
5290 EAPI void
5291 elm_genlist_item_tooltip_text_set(Elm_Object_Item *it,
5292                                   const char      *text)
5293 {
5294    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5295    text = eina_stringshare_add(text);
5296    elm_genlist_item_tooltip_content_cb_set(it, _elm_genlist_item_label_create,
5297                                            text,
5298                                            _elm_genlist_item_label_del_cb);
5299 }
5300
5301 EAPI void
5302 elm_genlist_item_tooltip_content_cb_set(Elm_Object_Item            *it,
5303                                         Elm_Tooltip_Item_Content_Cb func,
5304                                         const void                 *data,
5305                                         Evas_Smart_Cb               del_cb)
5306 {
5307    ELM_OBJ_ITEM_CHECK_OR_GOTO(it, error);
5308    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5309
5310    if ((_it->tooltip.content_cb == func) && (_it->tooltip.data == data))
5311      return;
5312
5313    if (_it->tooltip.del_cb)
5314      _it->tooltip.del_cb((void *) _it->tooltip.data, WIDGET(it), it);
5315
5316    _it->tooltip.content_cb = func;
5317    _it->tooltip.data = data;
5318    _it->tooltip.del_cb = del_cb;
5319
5320    if (VIEW(_it))
5321      {
5322         elm_widget_item_tooltip_content_cb_set(_it,
5323                                                _it->tooltip.content_cb,
5324                                                _it->tooltip.data, NULL);
5325         elm_widget_item_tooltip_style_set(_it, _it->tooltip.style);
5326         //elm_widget_item_tooltip_window_mode_set(_it, _it->tooltip.free_size); //upstream
5327      }
5328
5329    return;
5330
5331 error:
5332    if (del_cb) del_cb((void *)data, NULL, NULL);
5333 }
5334
5335 EAPI void
5336 elm_genlist_item_tooltip_unset(Elm_Object_Item *it)
5337 {
5338    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5339    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5340
5341    if ((VIEW(_it)) && (_it->tooltip.content_cb))
5342      elm_widget_item_tooltip_unset(_it);
5343
5344    if (_it->tooltip.del_cb)
5345      _it->tooltip.del_cb((void *)_it->tooltip.data, WIDGET(_it), _it);
5346    _it->tooltip.del_cb = NULL;
5347    _it->tooltip.content_cb = NULL;
5348    _it->tooltip.data = NULL;
5349    //_it->tooltip.free_size = EINA_FALSE; //upstream
5350    if (_it->tooltip.style)
5351      elm_genlist_item_tooltip_style_set(it, NULL);
5352 }
5353
5354 EAPI void
5355 elm_genlist_item_tooltip_style_set(Elm_Object_Item *it,
5356                                    const char       *style)
5357 {
5358    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5359    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5360
5361    eina_stringshare_replace(&_it->tooltip.style, style);
5362    if (VIEW(_it)) elm_widget_item_tooltip_style_set(_it, style);
5363 }
5364
5365 EAPI const char *
5366 elm_genlist_item_tooltip_style_get(const Elm_Object_Item *it)
5367 {
5368    return elm_object_item_tooltip_style_get(it);
5369 }
5370
5371 EAPI Eina_Bool
5372 elm_genlist_item_tooltip_window_mode_set(Elm_Object_Item *it,
5373                                          Eina_Bool disable)
5374 {
5375    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, EINA_FALSE);
5376    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5377
5378 //   _it->tooltip.free_size = disable; //temp
5379    if (VIEW(_it)) return elm_widget_item_tooltip_window_mode_set(_it, disable);
5380    return EINA_TRUE;
5381 }
5382
5383 EAPI Eina_Bool
5384 elm_genlist_item_tooltip_window_mode_get(const Elm_Object_Item *it)
5385 {
5386    return elm_object_tooltip_window_mode_get(VIEW(it));
5387 }
5388
5389 EAPI void
5390 elm_genlist_item_cursor_set(Elm_Object_Item *it,
5391                             const char       *cursor)
5392 {
5393    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5394    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5395    eina_stringshare_replace(&_it->mouse_cursor, cursor);
5396    if (VIEW(_it)) elm_widget_item_cursor_set(it, cursor);
5397 }
5398
5399 EAPI const char *
5400 elm_genlist_item_cursor_get(const Elm_Object_Item *it)
5401 {
5402    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, NULL);
5403    return elm_widget_item_cursor_get(it);
5404 }
5405
5406 EAPI void
5407 elm_genlist_item_cursor_unset(Elm_Object_Item *it)
5408 {
5409    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5410    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5411
5412    if (!_it->mouse_cursor) return;
5413
5414    if (VIEW(_it)) elm_widget_item_cursor_unset(_it);
5415
5416    eina_stringshare_del(_it->mouse_cursor);
5417    _it->mouse_cursor = NULL;
5418 }
5419
5420 EAPI void
5421 elm_genlist_item_cursor_style_set(Elm_Object_Item *it,
5422                                   const char       *style)
5423 {
5424    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5425    elm_widget_item_cursor_style_set(it, style);
5426 }
5427
5428 EAPI const char *
5429 elm_genlist_item_cursor_style_get(const Elm_Object_Item *it)
5430 {
5431    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, NULL);
5432    return elm_widget_item_cursor_style_get(it);
5433 }
5434
5435 EAPI void
5436 elm_genlist_item_cursor_engine_only_set(Elm_Object_Item *it,
5437                                         Eina_Bool         engine_only)
5438 {
5439    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5440    elm_widget_item_cursor_engine_only_set(it, engine_only);
5441 }
5442
5443 EAPI Eina_Bool
5444 elm_genlist_item_cursor_engine_only_get(const Elm_Object_Item *it)
5445 {
5446    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, EINA_FALSE);
5447    return elm_widget_item_cursor_engine_only_get(it);
5448 }
5449
5450 EAPI int
5451 elm_genlist_item_index_get(Elm_Object_Item *it)
5452 {
5453    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, -1);
5454    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5455
5456    if (_it->block)
5457      return _it->position + (_it->block->position * _it->wd->max_items_per_block);
5458    return -1;
5459 }
5460
5461 EAPI void
5462 elm_genlist_mode_set(Evas_Object  *obj,
5463                                 Elm_List_Mode mode)
5464 {
5465    ELM_CHECK_WIDTYPE(obj, widtype);
5466    Widget_Data *wd = elm_widget_data_get(obj);
5467    if (!wd) return;
5468    if (wd->mode == mode) return;
5469    wd->mode = mode;
5470    _sizing_eval(obj);
5471 }
5472
5473 EAPI void
5474 elm_genlist_horizontal_set(Evas_Object  *obj,
5475                            Elm_List_Mode mode)
5476 {
5477    elm_genlist_mode_set(obj, mode);
5478 }
5479
5480 EAPI void
5481 elm_genlist_horizontal_mode_set(Evas_Object  *obj,
5482                                 Elm_List_Mode mode)
5483 {
5484    elm_genlist_mode_set(obj, mode);
5485 }
5486
5487 EAPI Elm_List_Mode
5488 elm_genlist_mode_get(const Evas_Object *obj)
5489 {
5490    ELM_CHECK_WIDTYPE(obj, widtype) ELM_LIST_LAST;
5491    Widget_Data *wd = elm_widget_data_get(obj);
5492    if (!wd) return ELM_LIST_LAST;
5493    return wd->mode;
5494 }
5495
5496 EAPI Elm_List_Mode
5497 elm_genlist_horizontal_get(const Evas_Object *obj)
5498 {
5499    return elm_genlist_mode_get(obj);
5500 }
5501
5502 EAPI Elm_List_Mode
5503 elm_genlist_horizontal_mode_get(const Evas_Object *obj)
5504 {
5505    return elm_genlist_mode_get(obj);
5506 }
5507
5508 EAPI void
5509 elm_genlist_always_select_mode_set(Evas_Object *obj,
5510                                    Eina_Bool    always_select)
5511 {
5512    ELM_CHECK_WIDTYPE(obj, widtype);
5513    Widget_Data *wd = elm_widget_data_get(obj);
5514    if (!wd) return;
5515    wd->always_select = always_select;
5516 }
5517
5518 EAPI Eina_Bool
5519 elm_genlist_always_select_mode_get(const Evas_Object *obj)
5520 {
5521    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
5522    Widget_Data *wd = elm_widget_data_get(obj);
5523    if (!wd) return EINA_FALSE;
5524    return wd->always_select;
5525 }
5526
5527 EAPI void
5528 elm_genlist_no_select_mode_set(Evas_Object *obj,
5529                                Eina_Bool    no_select)
5530 {
5531    ELM_CHECK_WIDTYPE(obj, widtype);
5532    Widget_Data *wd = elm_widget_data_get(obj);
5533    if (!wd) return;
5534    wd->no_select = no_select;
5535 }
5536
5537 EAPI Eina_Bool
5538 elm_genlist_no_select_mode_get(const Evas_Object *obj)
5539 {
5540    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
5541    Widget_Data *wd = elm_widget_data_get(obj);
5542    if (!wd) return EINA_FALSE;
5543    return wd->no_select;
5544 }
5545
5546 EAPI void
5547 elm_genlist_compress_mode_set(Evas_Object *obj,
5548                               Eina_Bool    compress)
5549 {
5550    ELM_CHECK_WIDTYPE(obj, widtype);
5551    Widget_Data *wd = elm_widget_data_get(obj);
5552    if (!wd) return;
5553    wd->compress = compress;
5554    if (!compress) elm_genlist_homogeneous_set(obj, EINA_FALSE);
5555 }
5556
5557 EAPI Eina_Bool
5558 elm_genlist_compress_mode_get(const Evas_Object *obj)
5559 {
5560    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
5561    Widget_Data *wd = elm_widget_data_get(obj);
5562    if (!wd) return EINA_FALSE;
5563    return wd->compress;
5564 }
5565
5566 EAPI void
5567 elm_genlist_height_for_width_mode_set(Evas_Object *obj,
5568                                       Eina_Bool    height_for_width)
5569 {
5570    ELM_CHECK_WIDTYPE(obj, widtype);
5571    Widget_Data *wd = elm_widget_data_get(obj);
5572    if (!wd) return;
5573    wd->height_for_width = !!height_for_width;
5574    if (wd->height_for_width)
5575      {
5576         elm_genlist_homogeneous_set(obj, EINA_FALSE);
5577         elm_genlist_compress_mode_set(obj, EINA_TRUE);
5578      }
5579 }
5580
5581 EAPI Eina_Bool
5582 elm_genlist_height_for_width_mode_get(const Evas_Object *obj)
5583 {
5584    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
5585    Widget_Data *wd = elm_widget_data_get(obj);
5586    if (!wd) return EINA_FALSE;
5587    return wd->height_for_width;
5588 }
5589
5590 EAPI void
5591 elm_genlist_bounce_set(Evas_Object *obj,
5592                        Eina_Bool    h_bounce,
5593                        Eina_Bool    v_bounce)
5594 {
5595    ELM_CHECK_WIDTYPE(obj, widtype);
5596    Widget_Data *wd = elm_widget_data_get(obj);
5597    if (!wd) return;
5598    elm_smart_scroller_bounce_allow_set(wd->scr, h_bounce, v_bounce);
5599 }
5600
5601 EAPI void
5602 elm_genlist_bounce_get(const Evas_Object *obj,
5603                        Eina_Bool         *h_bounce,
5604                        Eina_Bool         *v_bounce)
5605 {
5606    ELM_CHECK_WIDTYPE(obj, widtype);
5607    Widget_Data *wd = elm_widget_data_get(obj);
5608    if (!wd) return;
5609    elm_smart_scroller_bounce_allow_get(wd->scr, h_bounce, v_bounce);
5610 }
5611
5612 EAPI void
5613 elm_genlist_homogeneous_set(Evas_Object *obj,
5614                             Eina_Bool    homogeneous)
5615 {
5616    ELM_CHECK_WIDTYPE(obj, widtype);
5617    Widget_Data *wd = elm_widget_data_get(obj);
5618    if (!wd) return;
5619    if (homogeneous) elm_genlist_compress_mode_set(obj, EINA_TRUE);
5620    wd->homogeneous = homogeneous;
5621 }
5622
5623 EAPI Eina_Bool
5624 elm_genlist_homogeneous_get(const Evas_Object *obj)
5625 {
5626    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
5627    Widget_Data *wd = elm_widget_data_get(obj);
5628    if (!wd) return EINA_FALSE;
5629    return wd->homogeneous;
5630 }
5631
5632 EAPI void
5633 elm_genlist_block_count_set(Evas_Object *obj,
5634                             int          n)
5635 {
5636    ELM_CHECK_WIDTYPE(obj, widtype);
5637    Widget_Data *wd = elm_widget_data_get(obj);
5638    if (!wd) return;
5639    wd->max_items_per_block = n;
5640    wd->item_cache_max = wd->max_items_per_block * 2;
5641    _item_cache_clean(wd);
5642 }
5643
5644 EAPI int
5645 elm_genlist_block_count_get(const Evas_Object *obj)
5646 {
5647    ELM_CHECK_WIDTYPE(obj, widtype) 0;
5648    Widget_Data *wd = elm_widget_data_get(obj);
5649    if (!wd) return 0;
5650    return wd->max_items_per_block;
5651 }
5652
5653 EAPI void
5654 elm_genlist_longpress_timeout_set(Evas_Object *obj,
5655                                   double       timeout)
5656 {
5657    ELM_CHECK_WIDTYPE(obj, widtype);
5658    Widget_Data *wd = elm_widget_data_get(obj);
5659    if (!wd) return;
5660    wd->longpress_timeout = timeout;
5661 }
5662
5663 EAPI double
5664 elm_genlist_longpress_timeout_get(const Evas_Object *obj)
5665 {
5666    ELM_CHECK_WIDTYPE(obj, widtype) 0;
5667    Widget_Data *wd = elm_widget_data_get(obj);
5668    if (!wd) return 0;
5669    return wd->longpress_timeout;
5670 }
5671
5672 EAPI void
5673 elm_genlist_scroller_policy_set(Evas_Object        *obj,
5674                                 Elm_Scroller_Policy policy_h,
5675                                 Elm_Scroller_Policy policy_v)
5676 {
5677    ELM_CHECK_WIDTYPE(obj, widtype);
5678    Widget_Data *wd = elm_widget_data_get(obj);
5679    if ((!wd) || (!wd->scr)) return;
5680    if ((policy_h >= ELM_SCROLLER_POLICY_LAST) ||
5681        (policy_v >= ELM_SCROLLER_POLICY_LAST))
5682      return;
5683    elm_smart_scroller_policy_set(wd->scr, policy_h, policy_v);
5684 }
5685
5686 EAPI void
5687 elm_genlist_scroller_policy_get(const Evas_Object   *obj,
5688                                 Elm_Scroller_Policy *policy_h,
5689                                 Elm_Scroller_Policy *policy_v)
5690 {
5691    ELM_CHECK_WIDTYPE(obj, widtype);
5692    Widget_Data *wd = elm_widget_data_get(obj);
5693    Elm_Smart_Scroller_Policy s_policy_h, s_policy_v;
5694    if ((!wd) || (!wd->scr)) return;
5695    elm_smart_scroller_policy_get(wd->scr, &s_policy_h, &s_policy_v);
5696    if (policy_h) *policy_h = (Elm_Scroller_Policy)s_policy_h;
5697    if (policy_v) *policy_v = (Elm_Scroller_Policy)s_policy_v;
5698 }
5699
5700 EAPI void
5701 elm_genlist_realized_items_update(Evas_Object *obj)
5702 {
5703    ELM_CHECK_WIDTYPE(obj, widtype);
5704
5705    Eina_List *list, *l;
5706    Elm_Object_Item *it;
5707
5708    list = elm_genlist_realized_items_get(obj);
5709    EINA_LIST_FOREACH(list, l, it)
5710      elm_genlist_item_update(it);
5711 }
5712
5713 EAPI void
5714 elm_genlist_item_mode_set(Elm_Object_Item *it,
5715                           const char       *mode_type,
5716                           Eina_Bool         mode_set)
5717 {
5718    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
5719    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5720
5721    Widget_Data *wd = _it->wd;
5722    Eina_List *l;
5723    Elm_Object_Item *it2;
5724
5725    if (!wd) return;
5726    if (!mode_type) return;
5727    if ((_it->delete_me) ||
5728        elm_widget_item_disabled_get(_it)) return;
5729    if (wd->edit_mode) return;
5730
5731    if ((wd->mode_item == _it) &&
5732        (!strcmp(mode_type, wd->mode_type)) &&
5733        (mode_set))
5734       return;
5735    if (!_it->itc->mode_item_style) return;
5736
5737    if (wd->multi)
5738      {
5739         EINA_LIST_FOREACH(wd->selected, l, it2)
5740           if (((Elm_Genlist_Item *) it2)->realized)
5741             elm_genlist_item_selected_set(it2, EINA_FALSE);
5742      }
5743    else
5744      {
5745         it2 = elm_genlist_selected_item_get(wd->obj);
5746         if ((it2) && (((Elm_Genlist_Item *) it2)->realized))
5747           elm_genlist_item_selected_set(it2, EINA_FALSE);
5748      }
5749
5750    if (((wd->mode_type) && (strcmp(mode_type, wd->mode_type))) ||
5751        (mode_set) ||
5752        ((_it == wd->mode_item) && (!mode_set)))
5753      _item_mode_unset(wd);
5754
5755    eina_stringshare_replace(&wd->mode_type, mode_type);
5756    if (mode_set) _item_mode_set(_it);
5757 }
5758
5759
5760 EAPI const char *
5761 elm_genlist_mode_item_style_get(const Evas_Object *obj)
5762 {
5763    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
5764    Widget_Data *wd = elm_widget_data_get(obj);
5765    if (!wd) return NULL;
5766    return wd->mode_item->itc->mode_item_style;
5767 }
5768
5769 EAPI void
5770 elm_genlist_mode_item_style_set(Evas_Object *obj, const char *style)
5771 {
5772    ELM_CHECK_WIDTYPE(obj, widtype);
5773    Widget_Data *wd = elm_widget_data_get(obj);
5774    if (!wd) return;
5775    if ((style == wd->mode_item->itc->mode_item_style) ||
5776        (style && wd->mode_item->itc->mode_item_style &&
5777        (!strcmp(style, wd->mode_item->itc->mode_item_style))))
5778      return;
5779    eina_stringshare_replace(&wd->mode_item->itc->mode_item_style, style);
5780    elm_genlist_realized_items_update(obj);
5781 }
5782
5783 EAPI const char *
5784 elm_genlist_mode_type_get(const Evas_Object *obj)
5785 {
5786    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
5787    Widget_Data *wd = elm_widget_data_get(obj);
5788    if (!wd) return NULL;
5789    return wd->mode_type;
5790 }
5791
5792 EAPI const Elm_Object_Item *
5793 elm_genlist_mode_item_get(const Evas_Object *obj)
5794 {
5795    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
5796    Widget_Data *wd = elm_widget_data_get(obj);
5797    if (!wd) return NULL;
5798    return (Elm_Object_Item *) wd->mode_item;
5799 }
5800
5801 EAPI void
5802 elm_genlist_reorder_mode_set(Evas_Object *obj,
5803                              Eina_Bool    reorder_mode)
5804 {
5805    ELM_CHECK_WIDTYPE(obj, widtype);
5806    Widget_Data *wd = elm_widget_data_get(obj);
5807    if (!wd) return;
5808    wd->reorder_mode = reorder_mode;
5809 }
5810
5811 EAPI Eina_Bool
5812 elm_genlist_reorder_mode_get(const Evas_Object *obj)
5813 {
5814    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
5815    Widget_Data *wd = elm_widget_data_get(obj);
5816    if (!wd) return EINA_FALSE;
5817    return wd->reorder_mode;
5818 }
5819
5820 EAPI Elm_Genlist_Item_Flags
5821 elm_genlist_item_flags_get(const Elm_Object_Item *it)
5822 {
5823    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, ELM_GENLIST_ITEM_MAX);
5824    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
5825    return _it->flags;
5826 }
5827
5828 /* for gengrid as of now */
5829 void
5830 _elm_genlist_page_relative_set(Evas_Object *obj,
5831                                double       h_pagerel,
5832                                double       v_pagerel)
5833 {
5834    Evas_Coord pagesize_h;
5835    Evas_Coord pagesize_v;
5836
5837    ELM_CHECK_WIDTYPE(obj, widtype);
5838    Widget_Data *wd = elm_widget_data_get(obj);
5839    if (!wd) return;
5840
5841    elm_smart_scroller_paging_get(wd->scr, NULL, NULL, &pagesize_h, &pagesize_v);
5842    elm_smart_scroller_paging_set(wd->scr, h_pagerel, v_pagerel, pagesize_h,
5843                                  pagesize_v);
5844 }
5845
5846 /* for gengrid as of now */
5847 void
5848 _elm_genlist_page_relative_get(const Evas_Object *obj,
5849                                double            *h_pagerel,
5850                                double            *v_pagerel)
5851 {
5852    ELM_CHECK_WIDTYPE(obj, widtype);
5853    Widget_Data *wd = elm_widget_data_get(obj);
5854    if (!wd) return;
5855
5856    elm_smart_scroller_paging_get(wd->scr, h_pagerel, v_pagerel, NULL, NULL);
5857 }
5858
5859 /* for gengrid as of now */
5860 void
5861 _elm_genlist_page_size_set(Evas_Object *obj,
5862                            Evas_Coord   h_pagesize,
5863                            Evas_Coord   v_pagesize)
5864 {
5865    double pagerel_h;
5866    double pagerel_v;
5867
5868    ELM_CHECK_WIDTYPE(obj, widtype);
5869    Widget_Data *wd = elm_widget_data_get(obj);
5870    if (!wd) return;
5871    elm_smart_scroller_paging_get(wd->scr, &pagerel_h, &pagerel_v, NULL, NULL);
5872    elm_smart_scroller_paging_set(wd->scr, pagerel_h, pagerel_v, h_pagesize,
5873                                  v_pagesize);
5874 }
5875
5876 /* for gengrid as of now */
5877 void
5878 _elm_genlist_current_page_get(const Evas_Object *obj,
5879                               int               *h_pagenumber,
5880                               int               *v_pagenumber)
5881 {
5882    ELM_CHECK_WIDTYPE(obj, widtype);
5883    Widget_Data *wd = elm_widget_data_get(obj);
5884    if (!wd) return;
5885    if (wd->scr)
5886      elm_smart_scroller_current_page_get(wd->scr, h_pagenumber, v_pagenumber);
5887 }
5888
5889 /* for gengrid as of now */
5890 void
5891 _elm_genlist_last_page_get(const Evas_Object *obj,
5892                            int               *h_pagenumber,
5893                            int               *v_pagenumber)
5894 {
5895    ELM_CHECK_WIDTYPE(obj, widtype);
5896    Widget_Data *wd = elm_widget_data_get(obj);
5897    if (!wd) return;
5898    if (wd->scr)
5899      elm_smart_scroller_last_page_get(wd->scr, h_pagenumber, v_pagenumber);
5900 }
5901
5902 /* for gengrid as of now */
5903 void
5904 _elm_genlist_page_show(const Evas_Object *obj,
5905                        int                h_pagenumber,
5906                        int                v_pagenumber)
5907 {
5908    ELM_CHECK_WIDTYPE(obj, widtype);
5909    Widget_Data *wd = elm_widget_data_get(obj);
5910    if (!wd) return;
5911    if (wd->scr)
5912      elm_smart_scroller_page_show(wd->scr, h_pagenumber, v_pagenumber);
5913 }
5914
5915 /* for gengrid as of now */
5916 void
5917 _elm_genlist_page_bring_in(const Evas_Object *obj,
5918                            int                h_pagenumber,
5919                            int                v_pagenumber)
5920 {
5921    ELM_CHECK_WIDTYPE(obj, widtype);
5922    Widget_Data *wd = elm_widget_data_get(obj);
5923    if (!wd) return;
5924    if (wd->scr)
5925      elm_smart_scroller_page_bring_in(wd->scr, h_pagenumber, v_pagenumber);
5926 }
5927
5928 void
5929 _elm_genlist_item_unrealize(Elm_Genlist_Item *it,
5930                             Eina_Bool     calc)
5931 {
5932    Evas_Object *content;
5933
5934    if (!it->realized) return;
5935    if (it->wd->reorder_it == it) return;
5936    evas_event_freeze(evas_object_evas_get(WIDGET(it)));
5937    if (!calc)
5938      evas_object_smart_callback_call(WIDGET(it), SIG_UNREALIZED, it);
5939    if (it->long_timer)
5940      {
5941         ecore_timer_del(it->long_timer);
5942         it->long_timer = NULL;
5943      }
5944
5945    elm_widget_stringlist_free(it->texts);
5946    it->texts = NULL;
5947    elm_widget_stringlist_free(it->contents);
5948    it->contents = NULL;
5949    elm_widget_stringlist_free(it->states);
5950    it->states = NULL;
5951
5952    EINA_LIST_FREE(it->content_objs, content)
5953      evas_object_del(content);
5954
5955     _item_unrealize(it, EINA_FALSE);//it->unrealize_cb(it); //upstream
5956
5957    it->realized = EINA_FALSE;
5958    it->want_unrealize = EINA_FALSE;
5959    evas_event_thaw(evas_object_evas_get(WIDGET(it)));
5960    evas_event_thaw_eval(evas_object_evas_get(WIDGET(it)));
5961 }
5962
5963 void
5964 _elm_genlist_item_del_notserious(Elm_Genlist_Item *it)
5965 {
5966    elm_widget_item_pre_notify_del(it);
5967    it->delete_me = EINA_TRUE;
5968    //it->generation = it->wd->generation - 1; /* This means that the item is deleted */ //upstream
5969    if (it->selected) it->wd->selected = eina_list_remove(it->wd->selected, it);
5970
5971    if (it->itc->func.del)
5972      it->itc->func.del((void *)it->base.data, WIDGET(it));
5973 }
5974
5975 void
5976 _elm_genlist_item_del_serious(Elm_Genlist_Item *it)
5977 {
5978    _elm_genlist_item_del_notserious(it);
5979    it->wd->items = eina_inlist_remove(it->wd->items, EINA_INLIST_GET(it));
5980    if (it->tooltip.del_cb)
5981      it->tooltip.del_cb((void *)it->tooltip.data, WIDGET(it), it);
5982    it->wd->walking -= it->walking;
5983    if (it->long_timer) ecore_timer_del(it->long_timer);
5984    if (it->flags & ELM_GENLIST_ITEM_GROUP)
5985      it->wd->group_items = eina_list_remove(it->wd->group_items, it);
5986
5987 /*
5988    if (it->wd->state)
5989      {
5990         eina_inlist_sorted_state_free(it->wd->state);
5991         it->wd->state = NULL;
5992      }
5993 */
5994    if (it->wd->calc_job) ecore_job_del(it->wd->calc_job);
5995    it->wd->calc_job = ecore_job_add(_calc_job, it->wd);
5996    //free(it->item); //temp
5997
5998    //it->item = NULL; //temp
5999    if (it->wd->last_selected_item == (Elm_Object_Item *) it)
6000      it->wd->last_selected_item = NULL;
6001 }
6002
6003 EAPI void
6004 elm_genlist_item_move_after(Elm_Object_Item *it __UNUSED__, Elm_Object_Item *after __UNUSED__)
6005 {
6006    return;
6007 }
6008
6009 EAPI void
6010 elm_genlist_item_move_before(Elm_Object_Item *it __UNUSED__, Elm_Object_Item *before __UNUSED__)
6011 {
6012    return;
6013 }
6014
6015 EAPI void
6016 elm_genlist_effect_set(const Evas_Object *obj, Eina_Bool emode)
6017 {
6018    ELM_CHECK_WIDTYPE(obj, widtype);
6019    Widget_Data *wd = elm_widget_data_get(obj);
6020    if (!wd) return;
6021    wd->effect_mode = emode;
6022    //   wd->point_rect = evas_object_rectangle_add(evas_object_evas_get(wd->obj));
6023    //   evas_object_resize(wd->point_rect, 10, 25);
6024    //   evas_object_color_set(wd->point_rect, 255, 0, 0, 130);
6025    //   evas_object_show(wd->point_rect);
6026    //   evas_object_hide(wd->point_rect);
6027 }
6028
6029 static Evas_Object*
6030 _create_tray_alpha_bg(const Evas_Object *obj)
6031 {
6032    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
6033    Widget_Data *wd = elm_widget_data_get(obj);
6034    if (!wd) return NULL;
6035
6036    Evas_Object *bg = NULL;
6037    Evas_Coord ox, oy, ow, oh;
6038
6039    evas_object_geometry_get(wd->pan_smart, &ox, &oy, &ow, &oh);
6040    bg  =  evas_object_rectangle_add(evas_object_evas_get(wd->obj));
6041    evas_object_color_set(bg,0,0,0,0);
6042    evas_object_resize(bg , ow, oh);
6043    evas_object_move(bg , ox, oy);
6044    evas_object_show(bg);
6045    evas_object_hide(bg);
6046    return bg ;
6047 }
6048
6049 static unsigned int
6050 current_time_get()
6051 {
6052    struct timeval timev;
6053
6054    gettimeofday(&timev, NULL);
6055    return ((timev.tv_sec * 1000) + ((timev.tv_usec) / 1000));
6056 }
6057
6058 // added for item moving animation.
6059 static Eina_Bool
6060 _item_moving_effect_timer_cb(void *data)
6061 {
6062    Widget_Data *wd = data;
6063    if (!wd) return EINA_FALSE;
6064    Item_Block *itb;
6065    Evas_Coord ox, oy, ow, oh, cvx, cvy, cvw, cvh;
6066    Elm_Genlist_Item *it = NULL, *it2;
6067    const Eina_List *l;
6068    double time = 0.5, t;
6069    int y, dy;
6070    Eina_Bool check, end = EINA_FALSE, vis = EINA_TRUE;
6071    int in = 0;
6072
6073    t = ((0.0 > (t = current_time_get() - wd->start_time)) ? 0.0 : t) / 1000;
6074
6075    evas_object_geometry_get(wd->pan_smart, &ox, &oy, &ow, &oh);
6076    evas_output_viewport_get(evas_object_evas_get(wd->pan_smart), &cvx, &cvy, &cvw, &cvh);
6077    if (t > time) end = EINA_TRUE;
6078
6079    // Below while statement is needed, when the genlist is resized.
6080    it2 = wd->expanded_item;
6081    while (it2 && vis)
6082      {
6083         evas_object_move(VIEW(it2), it2->scrl_x, it2->scrl_y);
6084         vis = (ELM_RECTS_INTERSECT(it2->scrl_x, it2->scrl_y, it2->w, it2->h,
6085                                    cvx, cvy, cvw, cvh));
6086         it2 = (Elm_Genlist_Item *) elm_genlist_item_prev_get((Elm_Object_Item *) it2);
6087      }
6088    if (wd->expanded_next_item)
6089      {
6090         Item_Block *itb2;
6091         it2 = wd->expanded_next_item;
6092         itb2 = wd->expanded_next_item->block;
6093         if ((wd->expanded_move_effect_set) && (!(ELM_RECTS_INTERSECT(itb2->x - wd->pan_x + ox,
6094                                                                      itb2->y - wd->pan_y + oy,
6095                                                                      itb2->w, itb2->h,
6096                                                                      cvx, cvy, cvw, cvh))))
6097           {
6098              _item_block_position(itb2, 0);
6099              wd->expanded_move_effect_set = EINA_FALSE;
6100           }
6101      }
6102    else
6103      {
6104         it2 = (Elm_Genlist_Item *) elm_genlist_item_next_get((Elm_Object_Item *) wd->expanded_item);
6105      }
6106    while (it2)
6107      {
6108         if (wd->expanded_item->expanded_depth >= it2->expanded_depth) break;
6109         it2 = (Elm_Genlist_Item *) elm_genlist_item_next_get((Elm_Object_Item *) it2);
6110      }
6111    dy = 0;
6112    if (it2)
6113      {
6114         if (wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_EXPAND)
6115           dy = ((it2->scrl_y > cvy + cvh) ? cvy + cvh : it2->scrl_y) - it2->old_scrl_y;
6116         else if (wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_CONTRACT)
6117           dy = wd->expand_item_gap;
6118
6119         if (t <= time)
6120           {
6121              y = ((1 - (1 - (t / time)) * (1 - (t /time))) * dy);
6122           }
6123         else
6124           {
6125              end = EINA_TRUE;
6126              y = dy;
6127           }
6128
6129         check = EINA_FALSE;
6130         EINA_INLIST_FOREACH(wd->blocks, itb)
6131           {
6132              itb->w = wd->minw;
6133              if (ELM_RECTS_INTERSECT(itb->x - wd->pan_x + ox,
6134                                      itb->y - wd->pan_y + oy,
6135                                      itb->w, itb->h,
6136                                      cvx, cvy, cvw, cvh))
6137                {
6138                   EINA_LIST_FOREACH(itb->items, l, it)
6139                     {
6140                        if (it == it2) check = EINA_TRUE;
6141                        if (!check) continue;
6142
6143                        if (!it->old_scrl_y)
6144                          it->old_scrl_y  = it->scrl_y;
6145                        if (it->old_scrl_y + y < oy + oh)
6146                          {
6147                             if (!it->realized) _item_realize(it, in, 0);
6148                          }
6149                        if (wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_DELETE && it->old_scrl_y + y < it->scrl_y)
6150                          it->old_scrl_y = it->scrl_y - y;
6151                        in++;
6152
6153                        if (wd->move_effect_mode != ELM_GENLIST_ITEM_MOVE_EFFECT_DELETE ||
6154                            (wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_DELETE && it->old_scrl_y + y >= it->scrl_y))
6155                          {
6156                             if (wd->edit_mode) _effect_item_controls(it, it->scrl_x, it->old_scrl_y + y);
6157                             else
6158                               {
6159                                  evas_object_resize(VIEW(it), it->w, it->h);
6160                                  evas_object_move(VIEW(it), it->scrl_x, it->old_scrl_y + y);
6161                                  evas_object_show(VIEW(it));
6162                                  evas_object_raise(VIEW(it));
6163                               }
6164
6165                             if (it->group_item) evas_object_raise(it->group_item->base.view);
6166                          }
6167                     }
6168                }
6169           }
6170
6171         if (wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_EXPAND)
6172           {
6173              it = (Elm_Genlist_Item *) elm_genlist_item_prev_get((Elm_Object_Item *) it2);
6174              while (it)
6175                {
6176                   if (it->expanded_depth <= it2->expanded_depth) break;
6177                   if (it->scrl_y && (it->scrl_y < it2->old_scrl_y + y) && (it->expanded_depth > it2->expanded_depth))
6178                     {
6179                        if (!it->effect_done)
6180                          {
6181                             edje_object_signal_emit(VIEW(it), "flip_item", "");
6182                             evas_object_show(VIEW(it));
6183                             it->effect_done = EINA_TRUE;
6184                          }
6185                     }
6186                   it = (Elm_Genlist_Item *) elm_genlist_item_prev_get((Elm_Object_Item *) it);
6187                }
6188           }
6189         else if (wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_CONTRACT)
6190           {
6191              it = (Elm_Genlist_Item *) elm_genlist_item_prev_get((Elm_Object_Item *) it2);
6192              while (it)
6193                {
6194                   if ((it->scrl_y > it2->old_scrl_y + y) && (it->expanded_depth > it2->expanded_depth))
6195                     {
6196                        if (!it->effect_done)
6197                          {
6198                             edje_object_signal_emit(VIEW(it), "elm,state,hide", "");
6199                             it->effect_done = EINA_TRUE;
6200                          }
6201                     }
6202                   else
6203                     break;
6204                   it = (Elm_Genlist_Item *) elm_genlist_item_prev_get((Elm_Object_Item *) it);
6205                }
6206           }
6207      }
6208    else
6209      {
6210         int expand_num = 0;
6211         int expand_order = 0;
6212         if (wd->expanded_item) it = (Elm_Genlist_Item *) elm_genlist_item_next_get((Elm_Object_Item *) wd->expanded_item);
6213         while (it)
6214           {
6215              expand_num++;
6216              it = (Elm_Genlist_Item *) elm_genlist_item_next_get((Elm_Object_Item *) it);
6217           }
6218         if (wd->expanded_item) it = (Elm_Genlist_Item *) elm_genlist_item_next_get((Elm_Object_Item *) wd->expanded_item);
6219         while (it)
6220           {
6221              expand_order++;
6222              if (wd->expanded_item->expanded_depth >= it->expanded_depth) break;
6223              if (wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_EXPAND)
6224                {
6225                   if (!it->effect_done)
6226                     {
6227                        if (t >= (((expand_order - 1) * time) / expand_num))
6228                          {
6229                             edje_object_signal_emit(VIEW(it), "flip_item", "");
6230                             evas_object_show(VIEW(it));
6231                             it->effect_done = EINA_TRUE;
6232                          }
6233                     }
6234                }
6235              it = (Elm_Genlist_Item *) elm_genlist_item_next_get((Elm_Object_Item *) it);
6236           }
6237      }
6238
6239    if (end)
6240      {
6241         if (wd->item_moving_effect_timer)
6242           {
6243              if (wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_CONTRACT)
6244                 _item_subitems_clear(wd->expanded_item);
6245              EINA_INLIST_FOREACH(wd->blocks, itb)
6246                {
6247                   EINA_LIST_FOREACH(itb->items, l, it)
6248                     {
6249                        it->effect_done = EINA_TRUE;
6250                        it->old_scrl_y = it->scrl_y;
6251                     }
6252                }
6253           }
6254         wd->item_moving_effect_timer = NULL;
6255         wd->expanded_move_effect_set = EINA_FALSE;
6256
6257         _item_auto_scroll(wd);
6258         evas_object_lower(wd->alpha_bg);
6259         evas_object_hide(wd->alpha_bg);
6260         elm_smart_scroller_bounce_animator_disabled_set(wd->scr, EINA_FALSE);
6261         wd->move_effect_mode = ELM_GENLIST_ITEM_MOVE_EFFECT_NONE;
6262
6263         evas_object_smart_callback_call(wd->pan_smart, "changed", NULL);
6264         evas_object_smart_callback_call(wd->obj, "effect_done", NULL);
6265         evas_object_smart_changed(wd->pan_smart);
6266         return ECORE_CALLBACK_CANCEL;
6267      }
6268    return ECORE_CALLBACK_RENEW;
6269 }
6270
6271 static void
6272 _emit_contract(Elm_Genlist_Item *it)
6273 {
6274    Elm_Genlist_Item *it2;
6275    Eina_List *l;
6276
6277    edje_object_signal_emit(VIEW(it), "elm,state,contract_flip", "");
6278    it->effect_done = EINA_FALSE;
6279
6280    EINA_LIST_FOREACH(it->items, l, it2)
6281       if (it2)
6282          _emit_contract(it2);
6283 }
6284
6285 // added for item moving animation.
6286 static int
6287 _item_flip_effect_show(Elm_Genlist_Item *it)
6288 {
6289    Elm_Genlist_Item *it2;
6290    Eina_List *l;
6291    Widget_Data *wd = it->wd;
6292
6293    EINA_LIST_FOREACH(it->items, l, it2)
6294      {
6295         if (it2->parent && it == it2->parent)
6296           {
6297              if (wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_EXPAND)
6298                edje_object_signal_emit(it2->base.view, "elm,state,hide", "");
6299              else if (wd->move_effect_mode == ELM_GENLIST_ITEM_MOVE_EFFECT_CONTRACT)
6300                _emit_contract(it2);
6301           }
6302      }
6303
6304    return ECORE_CALLBACK_CANCEL;
6305 }
6306
6307 /*
6308 static void
6309 _elm_genlist_pinch_zoom_execute(Evas_Object *obj, Eina_Bool emode)
6310 {
6311    printf("!!! NOW FIXING \n");
6312 }
6313 */
6314
6315 /**
6316  * Set pinch zoom mode
6317  *
6318  * @param obj The genlist object
6319  * @param emode
6320  * (EINA_TRUE = pinch contract (zoom in), EINA_FALSE = pinch expand (zoom out)
6321  *
6322  * @ingroup Genlist
6323  */
6324 EAPI void
6325 elm_genlist_pinch_zoom_mode_set(Evas_Object *obj __UNUSED__, Eina_Bool emode __UNUSED__)
6326 {
6327    printf("!!! NOW FIXING \n");
6328 }
6329
6330 /**
6331  * Get pinch zoom mode
6332  *
6333  * @param obj The genlist object
6334  * @return The pinch mode
6335  * (EINA_TRUE = pinch contract (zoom in), EINA_FALSE = pinch expand (zoom out)
6336  *
6337  * @ingroup Genlist
6338  */
6339 EAPI Eina_Bool
6340 elm_genlist_pinch_zoom_mode_get(const Evas_Object *obj __UNUSED__)
6341 {
6342    printf("!!! NOW FIXING \n");
6343    return EINA_FALSE;
6344 }
6345
6346 EAPI void
6347 elm_genlist_pinch_zoom_set(Evas_Object *obj __UNUSED__, Eina_Bool emode __UNUSED__)
6348 {
6349    printf("!!! NOW FIXING \n");
6350 }
6351
6352
6353 ////////////////////////////////////////////////////////////////////////
6354 //  EDIT  MODE
6355 ////////////////////////////////////////////////////////////////////////
6356
6357 static void
6358 _effect_item_controls(Elm_Genlist_Item *it, int itx, int ity)
6359 {
6360    if (!it->wd->edit_mode) return;
6361    evas_object_resize(it->edit_obj,it->w, it->h);
6362    evas_object_move(it->edit_obj, itx, ity);
6363 }
6364
6365 static void
6366 _effect_item_realize(Elm_Genlist_Item *it, Eina_Bool effect_on)
6367 {
6368    if ((it->effect_item_realized) || (it->delete_me)) return;
6369    char buf[1024];
6370
6371    if (it->mode_view)
6372      _mode_finished_signal_cb(it, (Evas_Object *)elm_genlist_item_object_get((Elm_Object_Item *) it), NULL, NULL);
6373
6374    it->edit_obj = edje_object_add(evas_object_evas_get(WIDGET(it)));
6375    edje_object_scale_set(it->edit_obj, elm_widget_scale_get(WIDGET(it)) *
6376                          _elm_config->scale);
6377    evas_object_smart_member_add(it->edit_obj, it->wd->pan_smart);
6378    elm_widget_sub_object_add(WIDGET(it), it->edit_obj);
6379
6380    if (it->flags & ELM_GENLIST_ITEM_SUBITEMS) strncpy(buf, "tree", sizeof(buf));
6381    else strncpy(buf, "item", sizeof(buf));
6382    if (it->wd->compress) strncat(buf, "_compress", sizeof(buf) - strlen(buf));
6383
6384    strncat(buf, "/", sizeof(buf) - strlen(buf));
6385
6386    if (it->itc->edit_item_style && strcmp(it->itc->edit_item_style, "default"))
6387      {
6388         strncat(buf, it->itc->edit_item_style, sizeof(buf) - strlen(buf));
6389         _elm_theme_object_set(WIDGET(it), it->edit_obj, "genlist", buf, elm_widget_style_get(WIDGET(it)));
6390      }
6391    else
6392      {
6393         _elm_theme_object_set(WIDGET(it), it->edit_obj, "genlist", "item/edit_default", elm_widget_style_get(WIDGET(it)));
6394      }
6395
6396    if (it->wd->reorder_mode)
6397      {
6398         if (effect_on) edje_object_signal_emit(it->edit_obj, "elm,state,reorder_enabled_effect", "elm");
6399         else edje_object_signal_emit(it->edit_obj, "elm,state,reorder_enabled", "elm");
6400      }
6401    if (effect_on) edje_object_signal_emit(it->edit_obj, "elm,state,emode_enabled_effect", "elm");
6402    else edje_object_signal_emit(it->edit_obj, "elm,state,emode_enabled", "elm");
6403
6404    if (elm_widget_item_disabled_get(it)) edje_object_signal_emit(it->edit_obj, "elm,state,disabled", "elm");
6405    else edje_object_signal_emit(it->edit_obj, "elm,state,enabled", "elm");
6406
6407
6408    if ((it->wd->edit_mode))
6409      {
6410         if (it->itc->func.content_get)
6411           {
6412              const Eina_List *l;
6413              const char *key;
6414
6415              if (it->renamed) edje_object_signal_emit(it->edit_obj, "elm,state,rename,enabled", "elm");
6416              it->contents = elm_widget_stringlist_get(edje_object_data_get(it->edit_obj, "edit_contents"));
6417              EINA_LIST_FOREACH(it->contents, l, key)
6418                {
6419                   Evas_Object *ic = it->itc->func.content_get
6420                      ((void *)it->base.data, WIDGET(it), l->data);
6421
6422                   if (ic)
6423                     {
6424                        it->edit_content_objs = eina_list_append(it->edit_content_objs, ic);
6425                        edje_object_part_swallow(it->edit_obj, key, ic);
6426                        evas_object_show(ic);
6427                        elm_widget_sub_object_add(WIDGET(it), ic);
6428                        if (elm_widget_item_disabled_get(it))
6429                          elm_widget_disabled_set(ic, EINA_TRUE);
6430                     }
6431                }
6432           }
6433      }
6434    edje_object_part_swallow(it->edit_obj, "original_edc", VIEW(it));
6435    _effect_item_controls(it,it->scrl_x, it->scrl_y);
6436    evas_object_show(it->edit_obj);
6437
6438    it->effect_item_realized = EINA_TRUE;
6439    it->want_unrealize = EINA_FALSE;
6440 }
6441
6442 static void
6443 _effect_item_unrealize(Elm_Genlist_Item *it)
6444 {
6445    Evas_Object *content;
6446
6447    if (!it->effect_item_realized) return;
6448    if (it->wd->reorder_it && it->wd->reorder_it == it) return;
6449
6450    edje_object_signal_emit(it->edit_obj, "elm,state,reorder_disable_effect", "elm");
6451    edje_object_signal_emit(it->edit_obj, "elm,state,sel,disable", "elm");
6452    edje_object_message_signal_process(it->edit_obj);
6453    edje_object_part_unswallow(it->edit_obj, VIEW(it));
6454    evas_object_smart_member_add(VIEW(it), it->wd->pan_smart);
6455    elm_widget_sub_object_add(WIDGET(it), VIEW(it));
6456    //   evas_object_smart_callback_call(it->edit_obj, "unrealized", it);
6457    //   _item_cache_add(it);
6458    evas_object_del(it->edit_obj);
6459    it->edit_obj = NULL;
6460    EINA_LIST_FREE(it->edit_content_objs, content)
6461       evas_object_del(content);
6462
6463 //   edje_object_signal_emit(it->edit_obj, "elm,state,edit_end,disable", "elm");
6464    it->effect_item_realized = EINA_FALSE;
6465 }
6466
6467 /**
6468  * Get Genlist edit mode
6469  *
6470  * @param obj The genlist object
6471  * @return The edit mode status
6472  * (EINA_TRUE = edit mode, EINA_FALSE = normal mode
6473  *
6474  * @ingroup Genlist
6475  */
6476 EAPI Eina_Bool
6477 elm_genlist_edit_mode_get(const Evas_Object *obj)
6478 {
6479    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
6480    Widget_Data *wd = elm_widget_data_get(obj);
6481    if (!wd) return EINA_FALSE;
6482
6483    if (wd->edit_mode) return EINA_TRUE;
6484    else return EINA_FALSE;
6485 }
6486
6487 /**
6488  * Set Genlist edit mode
6489  *
6490  * This sets Genlist edit mode.
6491  *
6492  * @param obj The Genlist object
6493  * @param emode ELM_GENLIST_EDIT_MODE_{NONE & REORDER & INSERT & DELETE & SELECT & SELECT_ALL}
6494  * @param edit_class Genlist edit class (Elm_Genlist_Edit_Class structure)
6495  *
6496  * @ingroup Genlist
6497  */
6498 EAPI void
6499 elm_genlist_edit_mode_set(Evas_Object *obj, Eina_Bool edit_mode)
6500 {
6501    ELM_CHECK_WIDTYPE(obj, widtype);
6502
6503    Item_Block *itb;
6504    Eina_List *l;
6505    Elm_Genlist_Item *it;
6506
6507    Widget_Data *wd = elm_widget_data_get(obj);
6508    if (!wd) return;
6509    if (wd->edit_mode == edit_mode) return;
6510
6511    wd->edit_mode = edit_mode;
6512    if (!wd->edit_mode)
6513      {
6514         EINA_INLIST_FOREACH(wd->blocks, itb)
6515           {
6516              if (itb->realized)
6517                {
6518                   EINA_LIST_FOREACH(itb->items, l, it)
6519                     {
6520                        if (it->flags != ELM_GENLIST_ITEM_GROUP && it->realized)
6521                          {
6522                             _effect_item_unrealize(it);
6523                          }
6524                     }
6525                }
6526           }
6527         _item_cache_zero(wd);
6528      }
6529    else
6530      {
6531
6532         EINA_INLIST_FOREACH(wd->blocks, itb)
6533           {
6534              if (itb->realized)
6535                {
6536                   EINA_LIST_FOREACH(itb->items, l, it)
6537                     {
6538                        if (it->flags != ELM_GENLIST_ITEM_GROUP && it->realized)
6539                          {
6540                             if(it->selected) _item_unselect(it);
6541                             if (it->itc->edit_item_style) _effect_item_realize(it, EINA_TRUE);
6542                          }
6543                     }
6544                 }
6545            }
6546       }
6547
6548    if (wd->calc_job) ecore_job_del(wd->calc_job);
6549    wd->calc_job = ecore_job_add(_calc_job, wd);
6550 }
6551
6552 /**
6553  * Set a given item's rename mode
6554  *
6555  * This renames the item's text from genlist
6556  *
6557  * @param it The item
6558  * @param renamed set if emode is EINA_TRUE, unset if emode is EINA_FALSE
6559  *
6560  * @ingroup Genlist
6561  */
6562 EAPI void
6563 elm_genlist_item_rename_mode_set(Elm_Object_Item *it, Eina_Bool renamed)
6564 {
6565    if (!it) return;
6566    Elm_Genlist_Item *_it = (Elm_Genlist_Item *) it;
6567
6568    if (renamed)
6569      {
6570         _item_unrealize(_it, EINA_FALSE);
6571         _it->renamed = EINA_TRUE;
6572         _it->wd->rename_it = _it;
6573         _it->nocache = EINA_TRUE;
6574         if (_it->selected)  _item_unselect(_it);
6575
6576         if (_it->wd->calc_job) ecore_job_del(_it->wd->calc_job);
6577         _it->wd->calc_job = ecore_job_add(_calc_job, _it->wd);
6578      }
6579    else
6580      {
6581         if (_it->renamed)
6582           {
6583              _it->renamed = EINA_FALSE;
6584              _it->nocache = EINA_TRUE;
6585              _it->wd->rename_it = NULL;
6586              _item_cache_zero(_it->wd);
6587              elm_genlist_item_update(it);
6588           }
6589      }
6590 }
6591
6592 EAPI Eina_Bool
6593 elm_genlist_item_rename_mode_get(Elm_Object_Item *it)
6594 {
6595    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, EINA_FALSE);
6596    return ((Elm_Genlist_Item *) it)->renamed;
6597 }
6598
6599 EAPI void
6600 elm_genlist_item_no_select_mode_set(Elm_Object_Item *it,
6601                                     Eina_Bool    no_select)
6602 {
6603    ELM_OBJ_ITEM_CHECK_OR_RETURN(it);
6604    ((Elm_Genlist_Item *) it)->no_select = no_select;
6605 }
6606
6607 EAPI Eina_Bool
6608 elm_genlist_item_no_select_mode_get(const Elm_Object_Item *it)
6609 {
6610    ELM_OBJ_ITEM_CHECK_OR_RETURN(it, EINA_FALSE);
6611    return ((Elm_Genlist_Item *) it)->no_select;
6612 }