d0a628aec19a47f2d4d24025ad890a205281fc9d
[framework/uifw/elementary.git] / src / lib / elc_ctxpopup.c
1 #include <Elementary.h>
2 #include "elm_priv.h"
3
4 typedef struct _Widget_Data Widget_Data;
5 typedef struct _Elm_Ctxpopup_Item Elm_Ctxpopup_Item;
6
7 struct _Elm_Ctxpopup_Item
8 {
9    ELM_WIDGET_ITEM;
10    const char *label;
11    Evas_Object *icon;
12    Evas_Smart_Cb func;
13 };
14
15 struct _Widget_Data
16 {
17    Evas_Object *parent;
18    Evas_Object *base;
19    Evas_Object *content;
20    Evas_Object *box;
21    Evas_Object *arrow;
22    Evas_Object *scr;
23    Evas_Object *bg;
24    Eina_List *items;
25    Elm_Ctxpopup_Direction dir;
26    Elm_Ctxpopup_Direction dir_priority[4];
27    Evas_Coord max_sc_w, max_sc_h;
28    Eina_Bool horizontal:1;
29    Eina_Bool visible:1;
30    Eina_Bool finished:1;
31 };
32
33 static const char *widtype = NULL;
34
35 static void _freeze_on(void *data, Evas_Object *obj, void *event_info);
36 static void _freeze_off(void *data, Evas_Object *obj, void *event_info);
37 static void _hold_on(void *data, Evas_Object *obj, void *event_info);
38 static void _hold_off(void *data, Evas_Object *obj, void *event_info);
39 static void _scroller_size_reset(Widget_Data *wd);
40 static void _on_focus_hook(void *data, Evas_Object *obj);
41 static Eina_Bool _event_hook(Evas_Object *obj,
42                              Evas_Object *src,
43                              Evas_Callback_Type type,
44                              void *event_info);
45 static void _parent_cut_off(Evas_Object *obj);
46 static void _parent_resize(void *data,
47                            Evas *e,
48                            Evas_Object *obj,
49                            void *event_info);
50 static void _parent_move(void *data,
51                          Evas *e,
52                          Evas_Object *obj,
53                          void *event_info);
54 static void _parent_del(void *data,
55                         Evas *e,
56                         Evas_Object *obj,
57                         void *event_info);
58 static void _item_sizing_eval(Elm_Ctxpopup_Item *item);
59 static void _adjust_pos_x(Evas_Coord_Point *pos,
60                           Evas_Coord_Point *base_size,
61                           Evas_Coord_Rectangle *hover_area);
62 static void _adjust_pos_y(Evas_Coord_Point *pos,
63                           Evas_Coord_Point *base_size,
64                           Evas_Coord_Rectangle *hover_area);
65 static Elm_Ctxpopup_Direction _calc_base_geometry(Evas_Object *obj,
66                                                   Evas_Coord_Rectangle *rect);
67 static void _update_arrow(Evas_Object *obj,
68                           Elm_Ctxpopup_Direction dir,
69                           Evas_Coord_Rectangle rect);
70 static void _sizing_eval(Evas_Object *obj);
71 static void _hide_signal_emit(Evas_Object *obj,
72                               Elm_Ctxpopup_Direction dir);
73 static void _show_signal_emit(Evas_Object *obj,
74                               Elm_Ctxpopup_Direction dir);
75 static void _shift_base_by_arrow(Evas_Object *arrow,
76                                  Elm_Ctxpopup_Direction dir,
77                                  Evas_Coord_Rectangle *rect);
78 static void _del_pre_hook(Evas_Object *obj);
79 static void _del_hook(Evas_Object *obj);
80 static void _theme_hook(Evas_Object *obj);
81 static void _content_set_hook(Evas_Object *obj,
82                               const char *part,
83                               Evas_Object *content);
84 static Evas_Object * _content_unset_hook(Evas_Object *obj,
85                                          const char *part__);
86 static Evas_Object * _content_get_hook(const Evas_Object *obj,
87                                        const char *part);
88 static void _item_text_set_hook(Elm_Object_Item *it,
89                                const char *part,
90                                const char *label);
91 static const char * _item_text_get_hook(const Elm_Object_Item *it,
92                                         const char *part);
93 static void _item_content_set_hook(Elm_Object_Item *it,
94                                    const char *part,
95                                    Evas_Object *content);
96 static Evas_Object * _item_content_get_hook(const Elm_Object_Item *it,
97                                             const char *part);
98 static void _item_disable_hook(Elm_Object_Item *it);
99 static void _item_signal_emit_hook(Elm_Object_Item *it,
100                                    const char *emission,
101                                    const char *source);
102 static void _bg_clicked_cb(void *data, Evas_Object *obj,
103                            const char *emission,
104                            const char *source);
105 static void _ctxpopup_show(void *data,
106                            Evas *e,
107                            Evas_Object *obj,
108                            void *event_info);
109 static void _hide_finished(void *data,
110                            Evas_Object *obj,
111                            const char *emission,
112                            const char *source __UNUSED__);
113 static void _hide(Evas_Object *obj);
114 static void _ctxpopup_hide(void *data,
115                            Evas *e,
116                            Evas_Object *obj,
117                            void *event_info);
118 static void _content_resize(void *data,
119                             Evas *e,
120                             Evas_Object *obj,
121                             void *event_info);
122 static void _scroller_resize(void *data,
123                              Evas *e,
124                              Evas_Object *obj,
125                              void *event_info);
126 static void _ctxpopup_move(void *data,
127                            Evas *e,
128                            Evas_Object *obj,
129                            void *event_info);
130 static void _restack(void *data, Evas *e, Evas_Object *obj, void *event_info);
131 static void _item_select_cb(void *data, Evas_Object *obj,
132                             const char *emission,
133                             const char *source);
134 static void _item_icon_set(Elm_Ctxpopup_Item *item, Evas_Object *icon);
135 static void _item_label_set(Elm_Ctxpopup_Item *item, const char *label);
136 static void _item_new(Elm_Ctxpopup_Item *item, char *group_name);
137 static void _content_del(void *data,
138                          Evas *e,
139                          Evas_Object *obj,
140                          void *event_info);
141 static void _list_del(Widget_Data *wd);
142 static void _list_new(Evas_Object *obj);
143 static void _remove_items(Widget_Data * wd);
144
145 static const char SIG_DISMISSED[] = "dismissed";
146
147 static const Evas_Smart_Cb_Description _signals[] = {
148    {SIG_DISMISSED, ""},
149    {NULL, NULL}
150 };
151
152 static void
153 _freeze_on(void *data __UNUSED__, Evas_Object *obj,
154            void *event_info __UNUSED__)
155 {
156    Widget_Data *wd = elm_widget_data_get(obj);
157
158    if ((!wd) || (!wd->scr)) return;
159    elm_object_scroll_freeze_push(wd->scr);
160 }
161
162 static void
163 _freeze_off(void *data __UNUSED__, Evas_Object *obj,
164             void *event_info __UNUSED__)
165 {
166    Widget_Data *wd = elm_widget_data_get(obj);
167
168    if ((!wd) || (!wd->scr)) return;
169    elm_object_scroll_freeze_pop(wd->scr);
170 }
171
172 static void
173 _hold_on(void *data __UNUSED__, Evas_Object *obj, void *event_info __UNUSED__)
174 {
175    Widget_Data *wd = elm_widget_data_get(obj);
176
177    if ((!wd) || (!wd->scr)) return;
178    elm_object_scroll_hold_push(wd->scr);
179 }
180
181 static void
182 _hold_off(void *data __UNUSED__, Evas_Object *obj, void *event_info __UNUSED__)
183 {
184    Widget_Data *wd = elm_widget_data_get(obj);
185
186    if ((!wd) || (!wd->scr)) return;
187    elm_object_scroll_hold_pop(wd->scr);
188 }
189
190 static void
191 _scroller_size_reset(Widget_Data *wd)
192 {
193    wd->finished = EINA_FALSE;
194    wd->max_sc_h = -1;
195    wd->max_sc_w = -1;
196 }
197
198 static void
199 _on_focus_hook(void *data __UNUSED__, Evas_Object *obj)
200 {
201    Widget_Data *wd = elm_widget_data_get(obj);
202    if (!wd) return;
203
204    if (elm_widget_focus_get(obj))
205      {
206         //FIXME:
207      }
208    else
209      {
210         //FIXME:
211      }
212 }
213
214 static Eina_Bool
215 _event_hook(Evas_Object *obj, Evas_Object *src __UNUSED__,
216             Evas_Callback_Type type, void *event_info)
217 {
218    Evas_Event_Key_Down *ev;
219    Widget_Data *wd;
220
221    if (type != EVAS_CALLBACK_KEY_DOWN)
222      return EINA_FALSE;
223    wd = elm_widget_data_get(obj);
224    if (!wd) return EINA_FALSE;
225
226    ev = event_info;
227    if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) return EINA_FALSE;
228    if (strcmp(ev->keyname, "Escape")) return EINA_FALSE;
229
230    evas_object_hide(obj);
231    ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
232    return EINA_TRUE;
233 }
234
235 static void
236 _parent_cut_off(Evas_Object *obj)
237 {
238    Widget_Data *wd = elm_widget_data_get(obj);
239
240    if (!wd) return;
241
242    evas_object_event_callback_del_full(wd->parent,
243                                        EVAS_CALLBACK_DEL,
244                                        _parent_del,
245                                        obj);
246    evas_object_event_callback_del_full(wd->parent,
247                                        EVAS_CALLBACK_MOVE,
248                                        _parent_move,
249                                        obj);
250    evas_object_event_callback_del_full(wd->parent,
251                                        EVAS_CALLBACK_RESIZE,
252                                        _parent_resize,
253                                        obj);
254
255    elm_widget_sub_object_del(wd->parent, obj);
256 }
257
258 static void
259 _parent_resize(void *data,
260                Evas *e __UNUSED__,
261                Evas_Object *obj __UNUSED__,
262                void *event_info __UNUSED__)
263 {
264    Widget_Data *wd = elm_widget_data_get(data);
265    if (!wd) return;
266
267    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
268
269    _hide(data);
270 }
271
272 static void
273 _parent_move(void *data,
274              Evas *e __UNUSED__,
275              Evas_Object *obj __UNUSED__,
276              void *event_info __UNUSED__)
277 {
278    Widget_Data *wd = elm_widget_data_get(data);
279
280    if (!wd) return;
281
282    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
283
284    if (wd->visible)
285      {
286         _scroller_size_reset(wd);
287         _sizing_eval(obj);
288      }
289 }
290
291 static void
292 _parent_del(void *data,
293             Evas *e __UNUSED__,
294             Evas_Object *obj __UNUSED__,
295             void *event_info __UNUSED__)
296 {
297    evas_object_del(data);
298 }
299
300 static void
301 _item_sizing_eval(Elm_Ctxpopup_Item *item)
302 {
303    Evas_Coord min_w = -1, min_h = -1, max_w = -1, max_h = -1;
304
305    if (!item) return;
306
307    edje_object_size_min_restricted_calc(VIEW(item), &min_w, &min_h, min_w,
308                                         min_h);
309    evas_object_size_hint_min_set(VIEW(item), min_w, min_h);
310    evas_object_size_hint_max_set(VIEW(item), max_w, max_h);
311 }
312
313 static void
314 _adjust_pos_x(Evas_Coord_Point *pos, Evas_Coord_Point *base_size,
315               Evas_Coord_Rectangle *hover_area)
316 {
317    pos->x -= (base_size->x / 2);
318
319    if (pos->x < hover_area->x)
320      pos->x = hover_area->x;
321    else if ((pos->x + base_size->x) > (hover_area->x + hover_area->w))
322      pos->x = (hover_area->x + hover_area->w) - base_size->x;
323
324    if (base_size->x > hover_area->w)
325      base_size->x -= (base_size->x - hover_area->w);
326
327    if (pos->x < hover_area->x)
328      pos->x = hover_area->x;
329 }
330
331 static void
332 _adjust_pos_y(Evas_Coord_Point *pos, Evas_Coord_Point *base_size,
333               Evas_Coord_Rectangle *hover_area)
334 {
335    pos->y -= (base_size->y / 2);
336
337    if (pos->y < hover_area->y)
338      pos->y = hover_area->y;
339    else if ((pos->y + base_size->y) > (hover_area->y + hover_area->h))
340      pos->y = hover_area->y + hover_area->h - base_size->y;
341
342    if (base_size->y > hover_area->h)
343      base_size->y -= (base_size->y - hover_area->h);
344
345    if (pos->y < hover_area->y)
346      pos->y = hover_area->y;
347 }
348
349 static Elm_Ctxpopup_Direction
350 _calc_base_geometry(Evas_Object *obj, Evas_Coord_Rectangle *rect)
351 {
352    Widget_Data *wd;
353    Evas_Coord_Point pos = {0, 0};
354    Evas_Coord_Point base_size;
355    Evas_Coord_Point max_size;
356    Evas_Coord_Point min_size;
357    Evas_Coord_Rectangle hover_area;
358    Evas_Coord_Point arrow_size;
359    Elm_Ctxpopup_Direction dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
360    Evas_Coord_Point temp;
361    int idx;
362
363    wd = elm_widget_data_get(obj);
364
365    if ((!wd) || (!rect))
366      return ELM_CTXPOPUP_DIRECTION_DOWN;
367
368    edje_object_part_geometry_get(wd->arrow, "ctxpopup_arrow", NULL, NULL,
369                                  &arrow_size.x, &arrow_size.y);
370    evas_object_resize(wd->arrow, arrow_size.x, arrow_size.y);
371
372    //Initialize Area Rectangle.
373    evas_object_geometry_get(wd->parent,
374                             &hover_area.x,
375                             &hover_area.y,
376                             &hover_area.w,
377                             &hover_area.h);
378
379    evas_object_geometry_get(obj, &pos.x, &pos.y, NULL, NULL);
380
381    //recalc the edje
382    edje_object_size_min_calc(wd->base, &base_size.x, &base_size.y);
383    evas_object_smart_calculate(wd->base);
384
385    //Limit to Max Size
386    evas_object_size_hint_max_get(obj, &max_size.x, &max_size.y);
387
388    if ((max_size.y > 0) && (base_size.y > max_size.y))
389      base_size.y = max_size.y;
390
391    if ((max_size.x > 0) && (base_size.x > max_size.x))
392      base_size.x = max_size.x;
393
394    //Limit to Min Size
395    evas_object_size_hint_min_get(obj, &min_size.x, &min_size.y);
396
397    if ((min_size.y > 0) && (base_size.y < min_size.y))
398      base_size.y = min_size.y;
399
400    if ((min_size.x > 0) && (base_size.x < min_size.x))
401      base_size.x = min_size.x;
402
403    //Check the Which direction is available.
404    //If find a avaialble direction, it adjusts position and size.
405    for (idx = 0; idx < 4; idx++)
406      {
407         switch (wd->dir_priority[idx])
408           {
409            case ELM_CTXPOPUP_DIRECTION_UNKNOWN:
410            case ELM_CTXPOPUP_DIRECTION_UP:
411               temp.y = (pos.y - base_size.y);
412               if ((temp.y - arrow_size.y) < hover_area.y)
413                 continue;
414               _adjust_pos_x(&pos, &base_size, &hover_area);
415               pos.y -= base_size.y;
416               dir = ELM_CTXPOPUP_DIRECTION_UP;
417               break;
418            case ELM_CTXPOPUP_DIRECTION_LEFT:
419               temp.x = (pos.x - base_size.x);
420               if ((temp.x - arrow_size.x) < hover_area.x)
421                 continue;
422               _adjust_pos_y(&pos, &base_size, &hover_area);
423               pos.x -= base_size.x;
424               dir = ELM_CTXPOPUP_DIRECTION_LEFT;
425               break;
426            case ELM_CTXPOPUP_DIRECTION_RIGHT:
427               temp.x = (pos.x + base_size.x);
428               if ((temp.x + arrow_size.x) >
429                   (hover_area.x + hover_area.w))
430                 continue;
431               _adjust_pos_y(&pos, &base_size, &hover_area);
432               dir = ELM_CTXPOPUP_DIRECTION_RIGHT;
433               break;
434            case ELM_CTXPOPUP_DIRECTION_DOWN:
435               temp.y = (pos.y + base_size.y);
436               if ((temp.y + arrow_size.y) >
437                   (hover_area.y + hover_area.h))
438                 continue;
439               _adjust_pos_x(&pos, &base_size, &hover_area);
440               dir = ELM_CTXPOPUP_DIRECTION_DOWN;
441               break;
442            default:
443               break;
444           }
445         break;
446      }
447
448    //In this case, all directions are invalid because of lack of space.
449    if (idx == 4)
450      {
451         Evas_Coord length[2];
452
453         if(!wd->horizontal)
454           {
455              length[0] = pos.y - hover_area.y;
456              length[1] = (hover_area.y + hover_area.h) - pos.y;
457
458              // ELM_CTXPOPUP_DIRECTION_UP
459              if (length[0] > length[1])
460                {
461                   _adjust_pos_x(&pos, &base_size, &hover_area);
462                   pos.y -= base_size.y;
463                   dir = ELM_CTXPOPUP_DIRECTION_UP;
464                   if (pos.y < (hover_area.y + arrow_size.y))
465                     {
466                        base_size.y -= ((hover_area.y + arrow_size.y) - pos.y);
467                        pos.y = hover_area.y + arrow_size.y;
468                     }
469                }
470              //ELM_CTXPOPUP_DIRECTION_DOWN
471              else
472                {
473                   _adjust_pos_x(&pos, &base_size, &hover_area);
474                   dir = ELM_CTXPOPUP_DIRECTION_DOWN;
475                   if ((pos.y + arrow_size.y + base_size.y) >
476                       (hover_area.y + hover_area.h))
477                      base_size.y -=
478                         ((pos.y + arrow_size.y + base_size.y) -
479                          (hover_area.y + hover_area.h));
480                }
481           }
482         else
483           {
484              length[0] = pos.x - hover_area.x;
485              length[1] = (hover_area.x + hover_area.w) - pos.x;
486
487              //ELM_CTXPOPUP_DIRECTION_LEFT
488              if (length[0] > length[1])
489                {
490                   _adjust_pos_y(&pos, &base_size, &hover_area);
491                   pos.x -= base_size.x;
492                   dir = ELM_CTXPOPUP_DIRECTION_LEFT;
493                   if (pos.x < (hover_area.x + arrow_size.x))
494                     {
495                        base_size.x -= ((hover_area.x + arrow_size.x) - pos.x);
496                        pos.x = hover_area.x + arrow_size.x;
497                     }
498                }
499              //ELM_CTXPOPUP_DIRECTION_RIGHT
500              else
501                {
502                   _adjust_pos_y(&pos, &base_size, &hover_area);
503                   dir = ELM_CTXPOPUP_DIRECTION_RIGHT;
504                   if (pos.x + (arrow_size.x + base_size.x) >
505                       hover_area.x + hover_area.w)
506                      base_size.x -=
507                         ((pos.x + arrow_size.x + base_size.x) -
508                          (hover_area.x + hover_area.w));
509                }
510           }
511      }
512
513    //Final position and size.
514    rect->x = pos.x;
515    rect->y = pos.y;
516    rect->w = base_size.x;
517    rect->h = base_size.y;
518
519    return dir;
520 }
521
522 static void
523 _update_arrow(Evas_Object *obj, Elm_Ctxpopup_Direction dir,
524               Evas_Coord_Rectangle base_size)
525 {
526    Evas_Coord x, y;
527    Evas_Coord_Rectangle arrow_size;
528    Widget_Data *wd;
529    double drag;
530
531    wd = elm_widget_data_get(obj);
532    if (!wd) return;
533
534    evas_object_geometry_get(obj, &x, &y, NULL, NULL);
535    evas_object_geometry_get(wd->arrow, NULL, NULL, &arrow_size.w,
536                             &arrow_size.h);
537
538    edje_object_part_unswallow(wd->base, wd->arrow);
539
540    switch (dir)
541      {
542       case ELM_CTXPOPUP_DIRECTION_RIGHT:
543          edje_object_signal_emit(wd->arrow, "elm,state,left", "elm");
544          edje_object_part_swallow(wd->base,
545                                   "elm.swallow.arrow_left",
546                                   wd->arrow);
547          if (base_size.h > 0)
548            {
549               if (y < ((arrow_size.h * 0.5) + base_size.y))
550                 y = 0;
551               else if (y > base_size.y + base_size.h - (arrow_size.h * 0.5))
552                 y = base_size.h - arrow_size.h;
553               else
554                 y = y - base_size.y - (arrow_size.h * 0.5);
555               drag = (double) (y) / (double) (base_size.h - arrow_size.h);
556               edje_object_part_drag_value_set(wd->base,
557                                               "elm.swallow.arrow_left",
558                                               1,
559                                               drag);
560            }
561          break;
562       case ELM_CTXPOPUP_DIRECTION_LEFT:
563          edje_object_signal_emit(wd->arrow, "elm,state,right", "elm");
564          edje_object_part_swallow(wd->base,
565                                   "elm.swallow.arrow_right",
566                                   wd->arrow);
567          if (base_size.h > 0)
568            {
569               if (y < ((arrow_size.h * 0.5) + base_size.y))
570                 y = 0;
571               else if (y > (base_size.y + base_size.h - (arrow_size.h * 0.5)))
572                 y = base_size.h - arrow_size.h;
573               else
574                 y = y - base_size.y - (arrow_size.h * 0.5);
575               drag = (double) (y) / (double) (base_size.h - arrow_size.h);
576               edje_object_part_drag_value_set(wd->base,
577                                               "elm.swallow.arrow_right",
578                                               0,
579                                               drag);
580            }
581          break;
582       case ELM_CTXPOPUP_DIRECTION_DOWN:
583          edje_object_signal_emit(wd->arrow, "elm,state,top", "elm");
584          edje_object_part_swallow(wd->base, "elm.swallow.arrow_up", wd->arrow);
585          if (base_size.w > 0)
586            {
587               if (x < ((arrow_size.w * 0.5) + base_size.x))
588                 x = 0;
589               else if (x > (base_size.x + base_size.w - (arrow_size.w * 0.5)))
590                 x = base_size.w - arrow_size.w;
591               else
592                 x = x - base_size.x - (arrow_size.w * 0.5);
593               drag = (double) (x) / (double) (base_size.w - arrow_size.w);
594               edje_object_part_drag_value_set(wd->base,
595                                               "elm.swallow.arrow_up",
596                                               drag,
597                                               1);
598            }
599          break;
600       case ELM_CTXPOPUP_DIRECTION_UP:
601          edje_object_signal_emit(wd->arrow, "elm,state,bottom", "elm");
602          edje_object_part_swallow(wd->base,
603                                   "elm.swallow.arrow_down",
604                                   wd->arrow);
605          if (base_size.w > 0)
606            {
607               if (x < ((arrow_size.w * 0.5) + base_size.x))
608                 x = 0;
609               else if (x > (base_size.x + base_size.w - (arrow_size.w * 0.5)))
610                 x = base_size.w - arrow_size.w;
611               else x = x - base_size.x - (arrow_size.w * 0.5);
612               drag = (double) (x) / (double) (base_size.w - arrow_size.w);
613               edje_object_part_drag_value_set(wd->base,
614                                               "elm.swallow.arrow_down",
615                                               drag,
616                                               0);
617            }
618          break;
619       default:
620          break;
621      }
622 }
623
624 static void
625 _hide_signal_emit(Evas_Object *obj, Elm_Ctxpopup_Direction dir)
626 {
627    Widget_Data *wd;
628
629    wd = elm_widget_data_get(obj);
630    if (!wd->visible) return;
631
632    switch (dir)
633      {
634         case ELM_CTXPOPUP_DIRECTION_UP:
635            edje_object_signal_emit(wd->base, "elm,state,hide,up", "elm");
636            break;
637         case ELM_CTXPOPUP_DIRECTION_LEFT:
638            edje_object_signal_emit(wd->base, "elm,state,hide,left", "elm");
639            break;
640         case ELM_CTXPOPUP_DIRECTION_RIGHT:
641            edje_object_signal_emit(wd->base, "elm,state,hide,right", "elm");
642            break;
643         case ELM_CTXPOPUP_DIRECTION_DOWN:
644            edje_object_signal_emit(wd->base, "elm,state,hide,down", "elm");
645            break;
646         default:
647            break;
648      }
649
650    edje_object_signal_emit(wd->bg, "elm,state,hide", "elm");
651 }
652
653 static void
654 _show_signal_emit(Evas_Object *obj, Elm_Ctxpopup_Direction dir)
655 {
656    Widget_Data *wd;
657
658    wd = elm_widget_data_get(obj);
659    if (wd->visible) return;
660
661    switch (dir)
662      {
663         case ELM_CTXPOPUP_DIRECTION_UP:
664            edje_object_signal_emit(wd->base, "elm,state,show,up", "elm");
665            break;
666         case ELM_CTXPOPUP_DIRECTION_LEFT:
667            edje_object_signal_emit(wd->base, "elm,state,show,left", "elm");
668            break;
669         case ELM_CTXPOPUP_DIRECTION_RIGHT:
670            edje_object_signal_emit(wd->base, "elm,state,show,right", "elm");
671            break;
672         case ELM_CTXPOPUP_DIRECTION_DOWN:
673            edje_object_signal_emit(wd->base, "elm,state,show,down", "elm");
674            break;
675         default:
676            break;
677      }
678 }
679
680 static void
681 _sizing_eval(Evas_Object *obj)
682 {
683    Widget_Data *wd;
684    Eina_List *elist;
685    Elm_Ctxpopup_Item *item;
686    Evas_Coord_Rectangle rect = { 0, 0, 1, 1 };
687    Evas_Coord_Point box_size = { 0, 0 };
688    Evas_Coord_Point _box_size = { 0, 0 };
689
690    wd = elm_widget_data_get(obj);
691    if (!wd) return;
692
693    //Box, Scroller
694    EINA_LIST_FOREACH(wd->items, elist, item)
695      {
696         _item_sizing_eval(item);
697         evas_object_size_hint_min_get(VIEW(item), &_box_size.x, &_box_size.y);
698         if (!wd->horizontal)
699           {
700              if (_box_size.x > box_size.x)
701                box_size.x = _box_size.x;
702              if (_box_size.y != -1)
703                box_size.y += _box_size.y;
704           }
705         else
706           {
707              if (_box_size.x != -1)
708                box_size.x += _box_size.x;
709              if (_box_size.y > box_size.y)
710                box_size.y = _box_size.y;
711           }
712      }
713
714    if (!wd->content)
715      {
716         evas_object_size_hint_min_set(wd->box, box_size.x, box_size.y);
717         evas_object_size_hint_min_set(wd->scr, box_size.x, box_size.y);
718      }
719
720    //Base
721    wd->dir = _calc_base_geometry(obj, &rect);
722    _show_signal_emit(obj, wd->dir);
723    _update_arrow(obj, wd->dir, rect);
724    _shift_base_by_arrow(wd->arrow, wd->dir, &rect);
725
726    //resize scroller according to final size.
727    if (!wd->content)
728      evas_object_smart_calculate(wd->scr);
729
730    evas_object_move(wd->base, rect.x, rect.y);
731    evas_object_resize(wd->base, rect.w, rect.h);
732 }
733
734 static void
735 _shift_base_by_arrow(Evas_Object *arrow, Elm_Ctxpopup_Direction dir,
736                      Evas_Coord_Rectangle *rect)
737 {
738    Evas_Coord arrow_w, arrow_h;
739
740    evas_object_geometry_get(arrow, NULL, NULL, &arrow_w, &arrow_h);
741
742    switch (dir)
743      {
744       case ELM_CTXPOPUP_DIRECTION_RIGHT:
745          rect->x += arrow_w;
746          break;
747       case ELM_CTXPOPUP_DIRECTION_LEFT:
748          rect->x -= arrow_w;
749          break;
750       case ELM_CTXPOPUP_DIRECTION_DOWN:
751          rect->y += arrow_h;
752          break;
753       case ELM_CTXPOPUP_DIRECTION_UP:
754          rect->y -= arrow_h;
755          break;
756       default:
757          break;
758      }
759 }
760
761 static void
762 _del_pre_hook(Evas_Object *obj)
763 {
764    Widget_Data *wd;
765
766    wd = elm_widget_data_get(obj);
767    if (!wd) return;
768
769    evas_object_event_callback_del_full(wd->box, EVAS_CALLBACK_RESIZE,
770                                        _content_resize, obj);
771    _parent_cut_off(obj);
772 }
773
774 static void
775 _del_hook(Evas_Object *obj)
776 {
777    Widget_Data *wd;
778
779    wd = elm_widget_data_get(obj);
780    if (!wd) return;
781
782    elm_ctxpopup_clear(obj);
783    evas_object_del(wd->arrow);
784    evas_object_del(wd->base);
785    free(wd);
786 }
787
788 //FIXME: lost the content size when theme hook is called.
789 static void
790 _theme_hook(Evas_Object *obj)
791 {
792    Widget_Data *wd;
793    Eina_List *elist;
794    Elm_Ctxpopup_Item *item;
795    Eina_Bool rtl;
796
797    wd = elm_widget_data_get(obj);
798    if (!wd) return;
799
800    _elm_widget_mirrored_reload(obj);
801    rtl = elm_widget_mirrored_get(obj);
802
803    _elm_theme_object_set(obj, wd->bg, "ctxpopup", "bg",
804                          elm_widget_style_get(obj));
805    _elm_theme_object_set(obj, wd->base, "ctxpopup", "base",
806                          elm_widget_style_get(obj));
807    _elm_theme_object_set(obj, wd->arrow, "ctxpopup", "arrow",
808                          elm_widget_style_get(obj));
809
810    //Items
811    EINA_LIST_FOREACH(wd->items, elist, item)
812      {
813         edje_object_mirrored_set(VIEW(item), rtl);
814
815         if (item->label && item->icon)
816           _elm_theme_object_set(obj, VIEW(item), "ctxpopup",
817                                 "icon_text_style_item",
818                                 elm_widget_style_get(obj));
819         else if (item->label)
820           _elm_theme_object_set(obj, VIEW(item), "ctxpopup", "text_style_item",
821                                 elm_widget_style_get(obj));
822         else if (item->icon)
823           _elm_theme_object_set(obj, VIEW(item), "ctxpopup", "icon_style_item",
824                                 elm_widget_style_get(obj));
825         if (item->label)
826           edje_object_part_text_set(VIEW(item), "elm.text", item->label);
827
828         if (elm_widget_item_disabled_get(item))
829           edje_object_signal_emit(VIEW(item), "elm,state,disabled", "elm");
830
831         edje_object_message_signal_process(VIEW(item));
832      }
833
834    if (wd->scr)
835      {
836         if (!strncmp(elm_object_style_get(obj), "default", strlen("default")))
837            elm_object_style_set(wd->scr, "ctxpopup");
838         else
839            elm_object_style_set(wd->scr, elm_object_style_get(obj));
840      }
841
842    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
843
844    if (wd->visible)
845      {
846         _scroller_size_reset(wd);
847         _sizing_eval(obj);
848      }
849 }
850
851 static void
852 _content_set_hook(Evas_Object *obj, const char *part, Evas_Object *content)
853 {
854    ELM_CHECK_WIDTYPE(obj, widtype);
855    Evas_Coord min_w = -1, min_h = -1;
856
857    Widget_Data *wd;
858
859    if (part && strcmp(part, "default")) return;
860
861    wd = elm_widget_data_get(obj);
862    if ((!wd) || (!content)) return;
863
864    if (wd->items) elm_ctxpopup_clear(obj);
865    if (wd->content) evas_object_del(wd->content);
866
867    //Use Box
868    wd->box = elm_box_add(obj);
869    evas_object_size_hint_weight_set(wd->box, EVAS_HINT_EXPAND,
870                                     EVAS_HINT_EXPAND);
871    evas_object_size_hint_weight_set(content, EVAS_HINT_EXPAND,
872                                     EVAS_HINT_EXPAND);
873    evas_object_size_hint_fill_set(content, EVAS_HINT_FILL,
874                                   EVAS_HINT_FILL);
875    evas_object_show(content);
876    evas_object_size_hint_min_get(content, &min_w, &min_h);
877    evas_object_size_hint_min_set(wd->box, min_w, min_h);
878    elm_box_pack_end(wd->box, content);
879
880    evas_object_event_callback_add(wd->box, EVAS_CALLBACK_RESIZE,
881                                   _content_resize, obj);
882    evas_object_event_callback_add(wd->box, EVAS_CALLBACK_DEL,
883                                   _content_del, obj);
884
885    elm_widget_sub_object_add(obj, wd->box);
886    edje_object_part_swallow(wd->base, "elm.swallow.content", wd->box);
887
888    wd->content = content;
889    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
890
891    if (wd->visible)
892      _sizing_eval(obj);
893 }
894
895 static Evas_Object *
896 _content_unset_hook(Evas_Object *obj, const char *part)
897 {
898    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
899
900    Widget_Data *wd;
901    Evas_Object *content;
902    if (part && strcmp(part, "default")) return NULL;
903    wd = elm_widget_data_get(obj);
904    if (!wd) return NULL;
905
906    content = wd->content;
907    if (!content || !wd->box) return NULL;
908
909    edje_object_part_unswallow(wd->base, wd->box);
910    elm_widget_sub_object_del(obj, wd->box);
911    evas_object_event_callback_del(wd->box, EVAS_CALLBACK_DEL, _content_del);
912    edje_object_signal_emit(wd->base, "elm,state,content,disable", "elm");
913
914    evas_object_del(wd->box);
915    wd->box = NULL;
916    wd->content = NULL;
917    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
918
919    return content;
920 }
921
922 static Evas_Object *
923 _content_get_hook(const Evas_Object *obj, const char *part)
924 {
925    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
926    Widget_Data *wd;
927    if (part && strcmp(part, "default")) return NULL;
928    wd = elm_widget_data_get(obj);
929    if (!wd) return NULL;
930    return wd->content;
931 }
932
933 static void
934 _item_text_set_hook(Elm_Object_Item *it, const char *part, const char *label)
935 {
936    Widget_Data *wd;
937    Elm_Ctxpopup_Item *ctxpopup_it;
938
939    if (part && strcmp(part, "default")) return;
940
941    ctxpopup_it = (Elm_Ctxpopup_Item *)it;
942
943    wd = elm_widget_data_get(WIDGET(ctxpopup_it));
944    if (!wd) return;
945
946    _item_label_set(ctxpopup_it, label);
947    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
948
949    if (wd->visible)
950      {
951         _scroller_size_reset(wd);
952         _sizing_eval(WIDGET(ctxpopup_it));
953      }
954 }
955
956 static const char *
957 _item_text_get_hook(const Elm_Object_Item *it, const char *part)
958 {
959    Elm_Ctxpopup_Item *ctxpopup_it;
960    if (part && strcmp(part, "default")) return NULL;
961    ctxpopup_it = (Elm_Ctxpopup_Item *)it;
962    return ctxpopup_it->label;
963 }
964
965 static void
966 _item_content_set_hook(Elm_Object_Item *it,
967                        const char *part,
968                        Evas_Object *content)
969 {
970    Widget_Data *wd;
971    Elm_Ctxpopup_Item *ctxpopup_it;
972
973    if (part && strcmp(part, "icon")) return;
974
975    ctxpopup_it = (Elm_Ctxpopup_Item *)it;
976
977    wd = elm_widget_data_get(WIDGET(ctxpopup_it));
978    if (!wd) return;
979
980    _item_icon_set(ctxpopup_it, content);
981    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
982
983    if (wd->visible)
984      {
985         _scroller_size_reset(wd);
986         _sizing_eval(WIDGET(ctxpopup_it));
987      }
988 }
989
990 static Evas_Object *
991 _item_content_get_hook(const Elm_Object_Item *it, const char *part)
992 {
993    Elm_Ctxpopup_Item *ctxpopup_it;
994    if (part && strcmp(part, "icon")) return NULL;
995    ctxpopup_it  = (Elm_Ctxpopup_Item *)it;
996    return ctxpopup_it->icon;
997 }
998
999 static void
1000 _item_disable_hook(Elm_Object_Item *it)
1001 {
1002    Widget_Data *wd;
1003    Elm_Ctxpopup_Item *ctxpopup_it = (Elm_Ctxpopup_Item *)it;
1004
1005    wd = elm_widget_data_get(WIDGET(ctxpopup_it));
1006    if (!wd) return;
1007
1008    if (elm_widget_item_disabled_get(it))
1009      edje_object_signal_emit(VIEW(ctxpopup_it), "elm,state,disabled", "elm");
1010    else
1011      edje_object_signal_emit(VIEW(ctxpopup_it), "elm,state,enabled", "elm");
1012 }
1013
1014 static void
1015 _item_signal_emit_hook(Elm_Object_Item *it, const char *emission,
1016                        const char *source)
1017 {
1018    Elm_Ctxpopup_Item *ctxpopup_it = (Elm_Ctxpopup_Item *)it;
1019    edje_object_signal_emit(VIEW(ctxpopup_it), emission, source);
1020 }
1021
1022 static void
1023 _bg_clicked_cb(void *data, Evas_Object *obj __UNUSED__,
1024                const char *emission __UNUSED__, const char *source __UNUSED__)
1025 {
1026    Widget_Data *wd = elm_widget_data_get(data);
1027    if (!wd) return;
1028    _hide_signal_emit(data, wd->dir);
1029 }
1030
1031 static void
1032 _ctxpopup_show(void *data __UNUSED__, Evas *e __UNUSED__, Evas_Object *obj,
1033                void *event_info __UNUSED__)
1034 {
1035    Widget_Data *wd;
1036
1037    wd = elm_widget_data_get(obj);
1038    if (!wd) return;
1039
1040    if ((!wd->items) && (!wd->content)) return;
1041
1042    wd->visible = EINA_TRUE;
1043
1044    evas_object_show(wd->bg);
1045    evas_object_show(wd->base);
1046    evas_object_show(wd->arrow);
1047
1048    edje_object_signal_emit(wd->bg, "elm,state,show", "elm");
1049    edje_object_signal_emit(wd->base, "elm,state,show", "elm");
1050
1051    _sizing_eval(obj);
1052
1053    elm_object_focus_set(obj, EINA_TRUE);
1054 }
1055
1056 static void
1057 _hide_finished(void *data, Evas_Object *obj __UNUSED__,
1058                const char *emission __UNUSED__, const char *source __UNUSED__)
1059 {
1060    _hide(data);
1061 }
1062
1063 static void
1064 _hide(Evas_Object *obj)
1065 {
1066    _ctxpopup_hide(NULL, NULL, obj, NULL);
1067    evas_object_smart_callback_call(obj, SIG_DISMISSED, NULL);
1068 }
1069
1070 static void
1071 _ctxpopup_hide(void *data __UNUSED__, Evas *e __UNUSED__, Evas_Object *obj,
1072                void *event_info __UNUSED__)
1073 {
1074    Widget_Data *wd = elm_widget_data_get(obj);
1075    if ((!wd) || (!wd->visible)) return;
1076
1077    evas_object_hide(wd->bg);
1078    evas_object_hide(wd->arrow);
1079    evas_object_hide(wd->base);
1080
1081    _scroller_size_reset(wd);
1082
1083    wd->visible = EINA_FALSE;
1084 }
1085
1086 static void
1087 _content_resize(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__,
1088                 void *event_info __UNUSED__)
1089 {
1090    Widget_Data *wd = elm_widget_data_get(data);
1091    if (!wd) return;
1092    elm_box_recalculate(wd->box);
1093    _sizing_eval(data);
1094 }
1095
1096 static void
1097 _scroller_resize(void *data, Evas *e __UNUSED__, Evas_Object *obj,
1098                  void *event_info __UNUSED__)
1099 {
1100    Widget_Data *wd;
1101    Evas_Coord w, h;
1102
1103    wd = elm_widget_data_get(data);
1104    if (!wd) return;
1105    if (!wd->visible) return;
1106    if (wd->finished) return;
1107
1108    evas_object_geometry_get(obj, 0, 0, &w, &h);
1109
1110    if (w != 0 && h != 0)
1111      {
1112         if ((w <= wd->max_sc_w) && (h <= wd->max_sc_h))
1113           {
1114              _sizing_eval(data);
1115              wd->finished = EINA_TRUE;
1116              return;
1117           }
1118      }
1119
1120    if (wd->max_sc_w < w)
1121      wd->max_sc_w = w;
1122    if (wd->max_sc_h < h)
1123      wd->max_sc_h = h;
1124
1125    _sizing_eval(data);
1126 }
1127
1128 static void
1129 _ctxpopup_move(void *data __UNUSED__, Evas *e __UNUSED__, Evas_Object *obj,
1130                void *event_info __UNUSED__)
1131 {
1132    Widget_Data *wd;
1133
1134    wd = elm_widget_data_get(obj);
1135
1136    if (!wd) return;
1137
1138    if (wd->visible)
1139      evas_object_show(wd->arrow);
1140
1141    _scroller_size_reset(wd);
1142    _sizing_eval(obj);
1143 }
1144
1145 static void
1146 _restack(void *data __UNUSED__, Evas *e __UNUSED__, Evas_Object *obj, void *event_info __UNUSED__)
1147 {
1148    Widget_Data *wd = elm_widget_data_get(obj);
1149    if (!wd) return;
1150    evas_object_layer_set(wd->bg,
1151                          evas_object_layer_get(obj));
1152    evas_object_layer_set(wd->base,
1153                          evas_object_layer_get(obj));
1154 }
1155
1156 static void
1157 _item_select_cb(void *data, Evas_Object *obj __UNUSED__,
1158                 const char *emission __UNUSED__, const char *source __UNUSED__)
1159 {
1160    Elm_Ctxpopup_Item *item = data;
1161
1162    if (!item) return;
1163    if (elm_widget_item_disabled_get(item)) return;
1164
1165    if (item->func)
1166      item->func((void*)item->base.data, WIDGET(item), data);
1167 }
1168
1169 static void
1170 _item_icon_set(Elm_Ctxpopup_Item *item, Evas_Object *icon)
1171 {
1172    if (item->icon)
1173      evas_object_del(item->icon);
1174
1175    item->icon = icon;
1176    if (!icon) return;
1177
1178    edje_object_part_swallow(VIEW(item), "elm.swallow.icon", item->icon);
1179    edje_object_message_signal_process(VIEW(item));
1180 }
1181
1182 static void
1183 _item_label_set(Elm_Ctxpopup_Item *item, const char *label)
1184 {
1185    if (!eina_stringshare_replace(&item->label, label))
1186      return;
1187
1188    edje_object_part_text_set(VIEW(item), "elm.text", label);
1189    edje_object_message_signal_process(VIEW(item));
1190 }
1191
1192 static void
1193 _item_new(Elm_Ctxpopup_Item *item, char *group_name)
1194 {
1195    Widget_Data *wd;
1196
1197    wd = elm_widget_data_get(WIDGET(item));
1198    if (!wd) return;
1199
1200    VIEW(item) = edje_object_add(evas_object_evas_get(wd->base));
1201    edje_object_mirrored_set(VIEW(item), elm_widget_mirrored_get(WIDGET(item)));
1202    _elm_theme_object_set(WIDGET(item), VIEW(item), "ctxpopup", group_name,
1203                          elm_widget_style_get(WIDGET(item)));
1204    edje_object_signal_callback_add(VIEW(item), "elm,action,click", "",
1205                                    _item_select_cb, item);
1206    evas_object_size_hint_align_set(VIEW(item), EVAS_HINT_FILL, EVAS_HINT_FILL);
1207    evas_object_show(VIEW(item));
1208 }
1209
1210 static void
1211 _content_del(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__,
1212              void *event_info __UNUSED__)
1213 {
1214    elm_object_content_unset(data);
1215 }
1216
1217 static void
1218 _list_del(Widget_Data *wd)
1219 {
1220    if (!wd->scr) return;
1221
1222    edje_object_part_unswallow(wd->base, wd->scr);
1223    evas_object_del(wd->scr);
1224    wd->scr = NULL;
1225    wd->box = NULL;
1226 }
1227
1228 static void
1229 _list_new(Evas_Object *obj)
1230 {
1231    Widget_Data *wd;
1232    wd = elm_widget_data_get(obj);
1233    if (!wd) return;
1234
1235    //scroller
1236    wd->scr = elm_scroller_add(obj);
1237    elm_object_style_set(wd->scr, "ctxpopup");
1238    evas_object_size_hint_align_set(wd->scr, EVAS_HINT_FILL, EVAS_HINT_FILL);
1239    evas_object_event_callback_add(wd->scr, EVAS_CALLBACK_RESIZE,
1240                                   _scroller_resize, obj);
1241    edje_object_part_swallow(wd->base, "elm.swallow.content", wd->scr);
1242
1243    //box
1244    wd->box = elm_box_add(obj);
1245    evas_object_size_hint_weight_set(wd->box, EVAS_HINT_EXPAND,
1246                                     EVAS_HINT_EXPAND);
1247
1248    elm_object_content_set(wd->scr, wd->box);
1249    elm_ctxpopup_horizontal_set(obj, wd->horizontal);
1250 }
1251
1252 static void
1253 _remove_items(Widget_Data *wd)
1254 {
1255    Eina_List *elist;
1256    Elm_Ctxpopup_Item *item;
1257
1258    if (!wd->items) return;
1259
1260    EINA_LIST_FOREACH(wd->items, elist, item)
1261      {
1262         if (item->label)
1263           eina_stringshare_del(item->label);
1264         if (item->icon)
1265           evas_object_del(item->icon);
1266         wd->items = eina_list_remove(wd->items, item);
1267         elm_widget_item_free(item);
1268      }
1269
1270    wd->items = NULL;
1271 }
1272
1273 static Eina_Bool
1274 _item_del_pre_hook(Elm_Object_Item *it)
1275 {
1276    Widget_Data *wd;
1277    Elm_Ctxpopup_Item *ctxpopup_it = (Elm_Ctxpopup_Item *)it;
1278
1279    wd = elm_widget_data_get(WIDGET(ctxpopup_it));
1280    if (!wd) return EINA_FALSE;
1281
1282    if (ctxpopup_it->icon)
1283      evas_object_del(ctxpopup_it->icon);
1284    if (VIEW(ctxpopup_it))
1285      evas_object_del(VIEW(ctxpopup_it));
1286
1287    eina_stringshare_del(ctxpopup_it->label);
1288
1289    wd->items = eina_list_remove(wd->items, ctxpopup_it);
1290
1291    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
1292
1293    if (eina_list_count(wd->items) < 1)
1294      {
1295         evas_object_hide(WIDGET(ctxpopup_it));
1296         return EINA_TRUE;
1297      }
1298
1299    if (wd->visible)
1300      _sizing_eval(WIDGET(ctxpopup_it));
1301
1302    return EINA_TRUE;
1303 }
1304
1305 EAPI Evas_Object *
1306 elm_ctxpopup_add(Evas_Object *parent)
1307 {
1308    Evas_Object *obj;
1309    Evas *e;
1310    Widget_Data *wd;
1311
1312    ELM_WIDGET_STANDARD_SETUP(wd, Widget_Data, parent, e, obj, NULL);
1313
1314    ELM_SET_WIDTYPE(widtype, "ctxpopup");
1315    elm_widget_type_set(obj, "ctxpopup");
1316    elm_widget_data_set(obj, wd);
1317    elm_widget_del_pre_hook_set(obj, _del_pre_hook);
1318    elm_widget_del_hook_set(obj, _del_hook);
1319    elm_widget_theme_hook_set(obj, _theme_hook);
1320    elm_widget_on_focus_hook_set(obj, _on_focus_hook, NULL);
1321    elm_widget_can_focus_set(obj, EINA_TRUE);
1322    elm_widget_event_hook_set(obj, _event_hook);
1323    elm_widget_content_set_hook_set(obj, _content_set_hook);
1324    elm_widget_content_unset_hook_set(obj, _content_unset_hook);
1325    elm_widget_content_get_hook_set(obj, _content_get_hook);
1326
1327    //Background
1328    wd->bg = edje_object_add(e);
1329    elm_widget_sub_object_add(obj, wd->bg);
1330    _elm_theme_object_set(obj, wd->bg, "ctxpopup", "bg", "default");
1331    edje_object_signal_callback_add(wd->bg,
1332                                    "elm,action,click",
1333                                    "",
1334                                    _bg_clicked_cb,
1335                                     obj);
1336    //Base
1337    wd->base = edje_object_add(e);
1338    elm_widget_sub_object_add(obj, wd->base);
1339    _elm_theme_object_set(obj, wd->base, "ctxpopup", "base", "default");
1340    edje_object_signal_callback_add(wd->base, "elm,action,hide,finished", "",
1341                                    _hide_finished, obj);
1342
1343    //Arrow
1344    wd->arrow = edje_object_add(e);
1345    elm_widget_sub_object_add(obj, wd->arrow);
1346    _elm_theme_object_set(obj, wd->arrow, "ctxpopup", "arrow", "default");
1347
1348    wd->dir_priority[0] = ELM_CTXPOPUP_DIRECTION_UP;
1349    wd->dir_priority[1] = ELM_CTXPOPUP_DIRECTION_LEFT;
1350    wd->dir_priority[2] = ELM_CTXPOPUP_DIRECTION_RIGHT;
1351    wd->dir_priority[3] = ELM_CTXPOPUP_DIRECTION_DOWN;
1352    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
1353
1354    evas_object_event_callback_add(obj, EVAS_CALLBACK_SHOW, _ctxpopup_show,
1355                                   NULL);
1356    evas_object_event_callback_add(obj, EVAS_CALLBACK_HIDE, _ctxpopup_hide,
1357                                   NULL);
1358    evas_object_event_callback_add(obj, EVAS_CALLBACK_MOVE, _ctxpopup_move,
1359                                   NULL);
1360    evas_object_event_callback_add(obj, EVAS_CALLBACK_RESTACK, _restack, obj);
1361    evas_object_smart_callback_add(obj, "scroll-freeze-on", _freeze_on, obj);
1362    evas_object_smart_callback_add(obj, "scroll-freeze-off", _freeze_off, obj);
1363    evas_object_smart_callback_add(obj, "scroll-hold-on", _hold_on, obj);
1364    evas_object_smart_callback_add(obj, "scroll-hold-off", _hold_off, obj);
1365
1366    evas_object_smart_callbacks_descriptions_set(obj, _signals);
1367
1368    //default parent is to be hover parent
1369    elm_ctxpopup_hover_parent_set(obj, parent);
1370
1371    return obj;
1372 }
1373
1374 EAPI void
1375 elm_ctxpopup_hover_parent_set(Evas_Object *obj, Evas_Object *parent)
1376 {
1377    ELM_CHECK_WIDTYPE(obj, widtype);
1378
1379    Widget_Data *wd;
1380    Evas_Coord x, y, w, h;
1381
1382    wd = elm_widget_data_get(obj);
1383    if ((!wd) || (!parent)) return;
1384
1385    _parent_cut_off(obj);
1386
1387    if (parent)
1388      {
1389         evas_object_event_callback_add(parent,
1390                                        EVAS_CALLBACK_DEL,
1391                                        _parent_del,
1392                                        obj);
1393         evas_object_event_callback_add(parent,
1394                                        EVAS_CALLBACK_MOVE,
1395                                        _parent_move,
1396                                        obj);
1397         evas_object_event_callback_add(parent,
1398                                        EVAS_CALLBACK_RESIZE,
1399                                        _parent_resize,
1400                                        obj);
1401      }
1402
1403    elm_widget_sub_object_add(parent, obj);
1404    wd->parent = parent;
1405
1406    //Update Background
1407    evas_object_geometry_get(parent, &x, &y, &w, &h);
1408    evas_object_move(wd->bg, x, y);
1409    evas_object_resize(wd->bg, w, h);
1410
1411    if (wd->visible) _sizing_eval(obj);
1412 }
1413
1414 EAPI Evas_Object *
1415 elm_ctxpopup_hover_parent_get(const Evas_Object *obj)
1416 {
1417    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
1418
1419    Widget_Data *wd;
1420
1421    wd = elm_widget_data_get(obj);
1422    if (!wd) return NULL;
1423
1424    return wd->parent;
1425 }
1426
1427 EAPI void
1428 elm_ctxpopup_clear(Evas_Object * obj)
1429 {
1430    ELM_CHECK_WIDTYPE(obj, widtype);
1431
1432    Widget_Data *wd = elm_widget_data_get(obj);
1433    if (!wd) return;
1434
1435    _remove_items(wd);
1436    _list_del(wd);
1437    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
1438 }
1439
1440 EAPI void
1441 elm_ctxpopup_horizontal_set(Evas_Object *obj, Eina_Bool horizontal)
1442 {
1443    ELM_CHECK_WIDTYPE(obj, widtype);
1444
1445    Widget_Data *wd;
1446
1447    wd = elm_widget_data_get(obj);
1448    if (!wd) return;
1449
1450    wd->horizontal = !!horizontal;
1451
1452    if ((!wd->scr) && (!wd->box))
1453       return;
1454
1455    if (!horizontal)
1456      {
1457         elm_box_horizontal_set(wd->box, EINA_FALSE);
1458         elm_scroller_bounce_set(wd->scr, EINA_FALSE, EINA_TRUE);
1459      }
1460    else
1461      {
1462         elm_box_horizontal_set(wd->box, EINA_TRUE);
1463         elm_scroller_bounce_set(wd->scr, EINA_TRUE, EINA_FALSE);
1464      }
1465
1466    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
1467
1468    if (wd->visible)
1469       _sizing_eval(obj);
1470 }
1471
1472 EAPI Eina_Bool
1473 elm_ctxpopup_horizontal_get(const Evas_Object *obj)
1474 {
1475    ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
1476
1477    Widget_Data *wd;
1478
1479    wd = elm_widget_data_get(obj);
1480    if (!wd) return EINA_FALSE;
1481
1482    return wd->horizontal;
1483 }
1484
1485 EAPI Elm_Object_Item *
1486 elm_ctxpopup_item_append(Evas_Object *obj, const char *label,
1487                          Evas_Object *icon, Evas_Smart_Cb func,
1488                          const void *data)
1489 {
1490    ELM_CHECK_WIDTYPE(obj, widtype) NULL;
1491
1492    Widget_Data *wd;
1493    Evas_Object *content;
1494    Elm_Ctxpopup_Item *item;
1495
1496    wd = elm_widget_data_get(obj);
1497    if (!wd) return NULL;
1498
1499    item = elm_widget_item_new(obj, Elm_Ctxpopup_Item);
1500    if (!item) return NULL;
1501
1502    elm_widget_item_del_pre_hook_set(item, _item_del_pre_hook);
1503    elm_widget_item_disable_hook_set(item, _item_disable_hook);
1504    elm_widget_item_text_set_hook_set(item, _item_text_set_hook);
1505    elm_widget_item_text_get_hook_set(item, _item_text_get_hook);
1506    elm_widget_item_content_set_hook_set(item, _item_content_set_hook);
1507    elm_widget_item_content_get_hook_set(item, _item_content_get_hook);
1508    elm_widget_item_signal_emit_hook_set(item, _item_signal_emit_hook);
1509
1510    //The first item is appended.
1511    content = elm_object_content_unset(obj);
1512    if (content) evas_object_del(content);
1513
1514    if (!wd->items)
1515      _list_new(obj);
1516
1517    item->func = func;
1518    item->base.data = data;
1519
1520    if (icon && label)
1521      _item_new(item, "icon_text_style_item");
1522    else if (label)
1523      _item_new(item, "text_style_item");
1524    else
1525      _item_new(item, "icon_style_item");
1526
1527    _item_icon_set(item, icon);
1528    _item_label_set(item, label);
1529    elm_box_pack_end(wd->box, VIEW(item));
1530    wd->items = eina_list_append(wd->items, item);
1531    wd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN;
1532
1533    if (wd->visible)
1534      {
1535         _scroller_size_reset(wd);
1536         _sizing_eval(obj);
1537      }
1538
1539    return (Elm_Object_Item *)item;
1540 }
1541
1542 EAPI void
1543 elm_ctxpopup_direction_priority_set(Evas_Object *obj,
1544                                     Elm_Ctxpopup_Direction first,
1545                                     Elm_Ctxpopup_Direction second,
1546                                     Elm_Ctxpopup_Direction third,
1547                                     Elm_Ctxpopup_Direction fourth)
1548 {
1549    ELM_CHECK_WIDTYPE(obj, widtype);
1550    Widget_Data *wd;
1551
1552    wd = elm_widget_data_get(obj);
1553    if (!wd) return;
1554
1555    wd->dir_priority[0] = first;
1556    wd->dir_priority[1] = second;
1557    wd->dir_priority[2] = third;
1558    wd->dir_priority[3] = fourth;
1559
1560    if (wd->visible)
1561      _sizing_eval(obj);
1562 }
1563
1564 EAPI void
1565 elm_ctxpopup_direction_priority_get(Evas_Object *obj,
1566                                     Elm_Ctxpopup_Direction *first,
1567                                     Elm_Ctxpopup_Direction *second,
1568                                     Elm_Ctxpopup_Direction *third,
1569                                     Elm_Ctxpopup_Direction *fourth)
1570 {
1571    ELM_CHECK_WIDTYPE(obj, widtype);
1572    Widget_Data *wd;
1573
1574    wd = elm_widget_data_get(obj);
1575    if (!wd) return;
1576
1577    if (first) *first = wd->dir_priority[0];
1578    if (second) *second = wd->dir_priority[1];
1579    if (third) *third = wd->dir_priority[2];
1580    if (fourth) *fourth = wd->dir_priority[3];
1581 }
1582
1583 EAPI Elm_Ctxpopup_Direction
1584 elm_ctxpopup_direction_get(const Evas_Object *obj)
1585 {
1586    ELM_CHECK_WIDTYPE(obj, widtype) ELM_CTXPOPUP_DIRECTION_UNKNOWN;
1587    Widget_Data *wd;
1588
1589    wd = elm_widget_data_get(obj);
1590    if (!wd) return ELM_CTXPOPUP_DIRECTION_UNKNOWN;
1591    return wd->dir;
1592 }
1593
1594 EAPI void
1595 elm_ctxpopup_dismiss(Evas_Object *obj)
1596 {
1597    ELM_CHECK_WIDTYPE(obj, widtype);
1598    Widget_Data *wd = elm_widget_data_get(obj);
1599    if (!wd) return;
1600    _hide_signal_emit(obj, wd->dir);
1601 }