#ifdef HAVE_CONFIG_H # include "elementary_config.h" #endif #include #include "elm_priv.h" #include "elm_widget_ctxpopup.h" EAPI Eo_Op ELM_OBJ_CTXPOPUP_BASE_ID = EO_NOOP; #define MY_CLASS ELM_OBJ_CTXPOPUP_CLASS #define MY_CLASS_NAME "elm_ctxpopup" EAPI const char ELM_CTXPOPUP_SMART_NAME[] = "elm_ctxpopup"; static const char SIG_DISMISSED[] = "dismissed"; static const char SIG_LANG_CHANGED[] = "language,changed"; static const Evas_Smart_Cb_Description _smart_callbacks[] = { {SIG_DISMISSED, ""}, {SIG_LANG_CHANGED, ""}, {NULL, NULL} }; static void _elm_ctxpopup_smart_translate(Eo *obj, void *_pd, va_list *list) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); Elm_Ctxpopup_Smart_Data *sd = _pd; Eina_List *l; Elm_Ctxpopup_Item *it; evas_object_hide(obj); EINA_LIST_FOREACH(sd->items, l, it) elm_widget_item_translate(it); if (ret) *ret = EINA_TRUE; } static void _elm_ctxpopup_smart_focus_next_manager_is(Eo *obj EINA_UNUSED, void *_pd EINA_UNUSED, va_list *list) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); *ret = EINA_TRUE; } static void _elm_ctxpopup_smart_focus_direction_manager_is(Eo *obj EINA_UNUSED, void *_pd EINA_UNUSED, va_list *list) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); *ret = EINA_TRUE; } static void _elm_ctxpopup_smart_focus_next(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Elm_Ctxpopup_Smart_Data *sd = _pd; Elm_Focus_Direction dir = va_arg(*list, Elm_Focus_Direction); Evas_Object **next = va_arg(*list, Evas_Object **); Eina_Bool *ret = va_arg(*list, Eina_Bool *); if (ret) *ret = EINA_FALSE; if (!sd) return; if (!elm_widget_focus_next_get(sd->box, dir, next)) { elm_widget_focused_object_clear(sd->box); elm_widget_focus_next_get(sd->box, dir, next); } if (ret) *ret = EINA_TRUE; } static void _elm_ctxpopup_smart_focus_direction(Eo *obj EINA_UNUSED, void *_pd EINA_UNUSED, va_list *list) { Elm_Ctxpopup_Smart_Data *sd = _pd; Evas_Object *base = va_arg(*list, Evas_Object *); double degree = va_arg(*list, double); Evas_Object **direction = va_arg(*list, Evas_Object **); double *weight = va_arg(*list, double *); Eina_Bool *ret = va_arg(*list, Eina_Bool *); if (ret) *ret = EINA_FALSE; Eina_Bool int_ret; Eina_List *l = NULL; void *(*list_data_get)(const Eina_List *list); if (!sd) return; list_data_get = eina_list_data_get; l = eina_list_append(l, sd->box); int_ret = elm_widget_focus_list_direction_get (obj, base, l, list_data_get, degree, direction, weight); if (ret) *ret = int_ret; eina_list_free(l); } static void _elm_ctxpopup_smart_event(Eo *obj, void *_pd, va_list *list) { Evas_Object *src = va_arg(*list, Evas_Object *); (void)src; Evas_Callback_Type type = va_arg(*list, Evas_Callback_Type); void *event_info = va_arg(*list, void *); Eina_Bool *ret = va_arg(*list, Eina_Bool *); if (ret) *ret = EINA_FALSE; Evas_Event_Key_Down *ev = event_info; Elm_Ctxpopup_Smart_Data *sd = _pd; if (elm_widget_disabled_get(obj)) return; if (type != EVAS_CALLBACK_KEY_DOWN) return; if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) return; if (!strcmp(ev->keyname, "Tab")) { if (evas_key_modifier_is_set(ev->modifiers, "Shift")) elm_widget_focus_cycle(sd->box, ELM_FOCUS_PREVIOUS); else elm_widget_focus_cycle(sd->box, ELM_FOCUS_NEXT); if (ret) *ret = EINA_TRUE; return; } if (strcmp(ev->keyname, "Escape")) return; evas_object_hide(obj); ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD; if (ret) *ret = EINA_TRUE; } static void _x_pos_adjust(Evas_Coord_Point *pos, Evas_Coord_Point *base_size, Evas_Coord_Rectangle *hover_area) { pos->x -= (base_size->x / 2); if (pos->x < hover_area->x) pos->x = hover_area->x; else if ((pos->x + base_size->x) > (hover_area->x + hover_area->w)) pos->x = (hover_area->x + hover_area->w) - base_size->x; if (base_size->x > hover_area->w) base_size->x -= (base_size->x - hover_area->w); if (pos->x < hover_area->x) pos->x = hover_area->x; } static void _y_pos_adjust(Evas_Coord_Point *pos, Evas_Coord_Point *base_size, Evas_Coord_Rectangle *hover_area) { pos->y -= (base_size->y / 2); if (pos->y < hover_area->y) pos->y = hover_area->y; else if ((pos->y + base_size->y) > (hover_area->y + hover_area->h)) pos->y = hover_area->y + hover_area->h - base_size->y; if (base_size->y > hover_area->h) base_size->y -= (base_size->y - hover_area->h); if (pos->y < hover_area->y) pos->y = hover_area->y; } static Elm_Ctxpopup_Direction _base_geometry_calc(Evas_Object *obj, Evas_Coord_Rectangle *rect) { Elm_Ctxpopup_Direction dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; Evas_Coord_Rectangle hover_area; Evas_Coord_Point pos = {0, 0}; Evas_Coord_Point arrow_size; Evas_Coord_Point base_size; Evas_Coord_Point max_size; Evas_Coord_Point min_size; Evas_Coord_Point temp; int idx; ELM_CTXPOPUP_DATA_GET(obj, sd); Elm_Widget_Smart_Data *wd = eo_data_scope_get(obj, ELM_OBJ_WIDGET_CLASS); if (!rect || !sd->parent) return ELM_CTXPOPUP_DIRECTION_DOWN; edje_object_part_geometry_get (sd->arrow, "ctxpopup_arrow", NULL, NULL, &arrow_size.x, &arrow_size.y); evas_object_resize(sd->arrow, arrow_size.x, arrow_size.y); //Initialize Area Rectangle. evas_object_geometry_get (sd->parent, &hover_area.x, &hover_area.y, &hover_area.w, &hover_area.h); if (!strcmp(evas_object_type_get(sd->parent), "elm_win")) hover_area.x = hover_area.y = 0; evas_object_geometry_get(obj, &pos.x, &pos.y, NULL, NULL); //recalc the edje edje_object_size_min_calc (wd->resize_obj, &base_size.x, &base_size.y); evas_object_smart_calculate(wd->resize_obj); //Limit to Max Size evas_object_size_hint_max_get(obj, &max_size.x, &max_size.y); if ((max_size.y > 0) && (base_size.y > max_size.y)) base_size.y = max_size.y; if ((max_size.x > 0) && (base_size.x > max_size.x)) base_size.x = max_size.x; //Limit to Min Size evas_object_size_hint_min_get(obj, &min_size.x, &min_size.y); if ((min_size.y > 0) && (base_size.y < min_size.y)) base_size.y = min_size.y; if ((min_size.x > 0) && (base_size.x < min_size.x)) base_size.x = min_size.x; //Check the Which direction is available. //If find a avaialble direction, it adjusts position and size. for (idx = 0; idx < 4; idx++) { switch (sd->dir_priority[idx]) { case ELM_CTXPOPUP_DIRECTION_UP: temp.y = (pos.y - base_size.y); if ((temp.y - arrow_size.y) < hover_area.y) continue; _x_pos_adjust(&pos, &base_size, &hover_area); pos.y -= base_size.y; dir = ELM_CTXPOPUP_DIRECTION_UP; break; case ELM_CTXPOPUP_DIRECTION_LEFT: temp.x = (pos.x - base_size.x); if ((temp.x - arrow_size.x) < hover_area.x) continue; _y_pos_adjust(&pos, &base_size, &hover_area); pos.x -= base_size.x; dir = ELM_CTXPOPUP_DIRECTION_LEFT; break; case ELM_CTXPOPUP_DIRECTION_RIGHT: temp.x = (pos.x + base_size.x); if ((temp.x + arrow_size.x) > (hover_area.x + hover_area.w)) continue; _y_pos_adjust(&pos, &base_size, &hover_area); dir = ELM_CTXPOPUP_DIRECTION_RIGHT; break; case ELM_CTXPOPUP_DIRECTION_DOWN: temp.y = (pos.y + base_size.y); if ((temp.y + arrow_size.y) > (hover_area.y + hover_area.h)) continue; _x_pos_adjust(&pos, &base_size, &hover_area); dir = ELM_CTXPOPUP_DIRECTION_DOWN; break; default: continue; } break; } //In this case, all directions are invalid because of lack of space. if (idx == 4) { Evas_Coord length[2]; if (!sd->horizontal) { length[0] = pos.y - hover_area.y; length[1] = (hover_area.y + hover_area.h) - pos.y; // ELM_CTXPOPUP_DIRECTION_UP if (length[0] > length[1]) { _x_pos_adjust(&pos, &base_size, &hover_area); pos.y -= base_size.y; dir = ELM_CTXPOPUP_DIRECTION_UP; if (pos.y < (hover_area.y + arrow_size.y)) { base_size.y -= ((hover_area.y + arrow_size.y) - pos.y); pos.y = hover_area.y + arrow_size.y; } } //ELM_CTXPOPUP_DIRECTION_DOWN else { _x_pos_adjust(&pos, &base_size, &hover_area); dir = ELM_CTXPOPUP_DIRECTION_DOWN; if ((pos.y + arrow_size.y + base_size.y) > (hover_area.y + hover_area.h)) base_size.y -= ((pos.y + arrow_size.y + base_size.y) - (hover_area.y + hover_area.h)); } } else { length[0] = pos.x - hover_area.x; length[1] = (hover_area.x + hover_area.w) - pos.x; //ELM_CTXPOPUP_DIRECTION_LEFT if (length[0] > length[1]) { _y_pos_adjust(&pos, &base_size, &hover_area); pos.x -= base_size.x; dir = ELM_CTXPOPUP_DIRECTION_LEFT; if (pos.x < (hover_area.x + arrow_size.x)) { base_size.x -= ((hover_area.x + arrow_size.x) - pos.x); pos.x = hover_area.x + arrow_size.x; } } //ELM_CTXPOPUP_DIRECTION_RIGHT else { _y_pos_adjust(&pos, &base_size, &hover_area); dir = ELM_CTXPOPUP_DIRECTION_RIGHT; if (pos.x + (arrow_size.x + base_size.x) > hover_area.x + hover_area.w) base_size.x -= ((pos.x + arrow_size.x + base_size.x) - (hover_area.x + hover_area.w)); } } } //Final position and size. rect->x = pos.x; rect->y = pos.y; rect->w = base_size.x; rect->h = base_size.y; return dir; } static void _arrow_update(Evas_Object *obj, Elm_Ctxpopup_Direction dir, Evas_Coord_Rectangle base_size) { Evas_Coord_Rectangle arrow_size; Evas_Coord x, y; double drag; ELM_CTXPOPUP_DATA_GET(obj, sd); Elm_Widget_Smart_Data *wd = eo_data_scope_get(obj, ELM_OBJ_WIDGET_CLASS); evas_object_geometry_get(obj, &x, &y, NULL, NULL); evas_object_geometry_get (sd->arrow, NULL, NULL, &arrow_size.w, &arrow_size.h); /* arrow is not being kept as sub-object on purpose, here. the * design of the widget does not help with the contrary */ switch (dir) { case ELM_CTXPOPUP_DIRECTION_RIGHT: edje_object_signal_emit(sd->arrow, "elm,state,left", "elm"); edje_object_part_swallow (wd->resize_obj, (elm_widget_mirrored_get(obj) ? "elm.swallow.arrow_right" : "elm.swallow.arrow_left"), sd->arrow); if (base_size.h > 0) { if (y < ((arrow_size.h * 0.5) + base_size.y)) y = 0; else if (y > base_size.y + base_size.h - (arrow_size.h * 0.5)) y = base_size.h - arrow_size.h; else y = y - base_size.y - (arrow_size.h * 0.5); drag = (double)(y) / (double)(base_size.h - arrow_size.h); edje_object_part_drag_value_set (wd->resize_obj, (elm_widget_mirrored_get(obj) ? "elm.swallow.arrow_right" : "elm.swallow.arrow_left"), 1, drag); } break; case ELM_CTXPOPUP_DIRECTION_LEFT: edje_object_signal_emit(sd->arrow, "elm,state,right", "elm"); edje_object_part_swallow (wd->resize_obj, (elm_widget_mirrored_get(obj) ? "elm.swallow.arrow_left" : "elm.swallow.arrow_right"), sd->arrow); if (base_size.h > 0) { if (y < ((arrow_size.h * 0.5) + base_size.y)) y = 0; else if (y > (base_size.y + base_size.h - (arrow_size.h * 0.5))) y = base_size.h - arrow_size.h; else y = y - base_size.y - (arrow_size.h * 0.5); drag = (double)(y) / (double)(base_size.h - arrow_size.h); edje_object_part_drag_value_set (wd->resize_obj, (elm_widget_mirrored_get(obj) ? "elm.swallow.arrow_left" : "elm.swallow.arrow_right"), 0, drag); } break; case ELM_CTXPOPUP_DIRECTION_DOWN: edje_object_signal_emit(sd->arrow, "elm,state,top", "elm"); edje_object_part_swallow (wd->resize_obj, "elm.swallow.arrow_up", sd->arrow); if (base_size.w > 0) { if (x < ((arrow_size.w * 0.5) + base_size.x)) x = 0; else if (x > (base_size.x + base_size.w - (arrow_size.w * 0.5))) x = base_size.w - arrow_size.w; else x = x - base_size.x - (arrow_size.w * 0.5); drag = (double)(x) / (double)(base_size.w - arrow_size.w); edje_object_part_drag_value_set (wd->resize_obj, "elm.swallow.arrow_up", drag, 1); } break; case ELM_CTXPOPUP_DIRECTION_UP: edje_object_signal_emit(sd->arrow, "elm,state,bottom", "elm"); edje_object_part_swallow (wd->resize_obj, "elm.swallow.arrow_down", sd->arrow); if (base_size.w > 0) { if (x < ((arrow_size.w * 0.5) + base_size.x)) x = 0; else if (x > (base_size.x + base_size.w - (arrow_size.w * 0.5))) x = base_size.w - arrow_size.w; else x = x - base_size.x - (arrow_size.w * 0.5); drag = (double)(x) / (double)(base_size.w - arrow_size.w); edje_object_part_drag_value_set (wd->resize_obj, "elm.swallow.arrow_down", drag, 0); } break; default: break; } //should be here for getting accurate geometry value evas_object_smart_calculate(wd->resize_obj); } static void _show_signals_emit(Evas_Object *obj, Elm_Ctxpopup_Direction dir) { ELM_CTXPOPUP_DATA_GET(obj, sd); if (!sd->visible) return; if ((sd->list) && (!sd->list_visible)) return; switch (dir) { case ELM_CTXPOPUP_DIRECTION_UP: elm_layout_signal_emit(obj, "elm,state,show,up", "elm"); break; case ELM_CTXPOPUP_DIRECTION_LEFT: elm_layout_signal_emit(obj, (elm_widget_mirrored_get(obj) ? "elm,state,show,right" : "elm,state,show,left"), "elm"); break; case ELM_CTXPOPUP_DIRECTION_RIGHT: elm_layout_signal_emit(obj, (elm_widget_mirrored_get(obj) ? "elm,state,show,left" : "elm,state,show,right"), "elm"); break; case ELM_CTXPOPUP_DIRECTION_DOWN: elm_layout_signal_emit(obj, "elm,state,show,down", "elm"); break; default: break; } edje_object_signal_emit(sd->bg, "elm,state,show", "elm"); elm_layout_signal_emit(obj, "elm,state,show", "elm"); } static void _hide_signals_emit(Evas_Object *obj, Elm_Ctxpopup_Direction dir) { ELM_CTXPOPUP_DATA_GET(obj, sd); if (!sd->visible) return; switch (dir) { case ELM_CTXPOPUP_DIRECTION_UP: elm_layout_signal_emit(obj, "elm,state,hide,up", "elm"); break; case ELM_CTXPOPUP_DIRECTION_LEFT: elm_layout_signal_emit(obj, (elm_widget_mirrored_get(obj) ? "elm,state,hide,right" : "elm,state,hide,left"), "elm"); break; case ELM_CTXPOPUP_DIRECTION_RIGHT: elm_layout_signal_emit(obj, (elm_widget_mirrored_get(obj) ? "elm,state,hide,left" : "elm,state,hide,right"), "elm"); break; case ELM_CTXPOPUP_DIRECTION_DOWN: elm_layout_signal_emit(obj, "elm,state,hide,down", "elm"); break; default: break; } edje_object_signal_emit(sd->bg, "elm,state,hide", "elm"); } static void _base_shift_by_arrow(Evas_Object *arrow, Elm_Ctxpopup_Direction dir, Evas_Coord_Rectangle *rect) { Evas_Coord arrow_w, arrow_h; evas_object_geometry_get(arrow, NULL, NULL, &arrow_w, &arrow_h); switch (dir) { case ELM_CTXPOPUP_DIRECTION_RIGHT: rect->x += arrow_w; break; case ELM_CTXPOPUP_DIRECTION_LEFT: rect->x -= arrow_w; break; case ELM_CTXPOPUP_DIRECTION_DOWN: rect->y += arrow_h; break; case ELM_CTXPOPUP_DIRECTION_UP: rect->y -= arrow_h; break; default: break; } } static void _elm_ctxpopup_smart_layout_sub_object_add_enable(Eo *obj EINA_UNUSED, void *_pd EINA_UNUSED, va_list *list) { Eina_Bool *enable = va_arg(*list, Eina_Bool *); *enable = EINA_FALSE; } static void _elm_ctxpopup_smart_sub_object_add(Eo *obj, void *_pd EINA_UNUSED, va_list *list) { Evas_Object *sobj = va_arg(*list, Evas_Object *); Eina_Bool *ret = va_arg(*list, Eina_Bool *); Eina_Bool int_ret; eo_do_super(obj, MY_CLASS, elm_wdg_sub_object_add(sobj, &int_ret)); if (ret) *ret = int_ret; } static void _elm_ctxpopup_smart_sizing_eval(Eo *obj, void *_pd, va_list *list EINA_UNUSED) { Evas_Coord_Rectangle rect = { 0, 0, 1, 1 }; Evas_Coord_Point list_size = { 0, 0 }; Elm_Ctxpopup_Smart_Data *sd = _pd; Elm_Widget_Smart_Data *wd = eo_data_scope_get(obj, ELM_OBJ_WIDGET_CLASS); if (!sd->arrow) return; /* simple way to flag "under deletion" */ if (!sd->parent) return; /* do not calculate sizes unless parent is set */ //Base sd->dir = _base_geometry_calc(obj, &rect); _arrow_update(obj, sd->dir, rect); _base_shift_by_arrow(sd->arrow, sd->dir, &rect); if ((sd->list) && (sd->list_visible)) { evas_object_geometry_get(sd->list, 0, 0, &list_size.x, &list_size.y); if ((list_size.x >= rect.w) || (list_size.y >= rect.h)) { elm_list_mode_set(sd->list, ELM_LIST_COMPRESS); evas_object_size_hint_min_set(sd->box, rect.w, rect.h); evas_object_size_hint_min_set(obj, rect.w, rect.h); } } evas_object_move(wd->resize_obj, rect.x, rect.y); evas_object_resize(wd->resize_obj, rect.w, rect.h); _show_signals_emit(obj, sd->dir); } static void _on_parent_del(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__) { evas_object_del(data); } static void _on_parent_move(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__) { ELM_CTXPOPUP_DATA_GET(data, sd); sd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; if (sd->visible) elm_layout_sizing_eval(data); } static void _on_parent_resize(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__) { ELM_CTXPOPUP_DATA_GET(data, sd); sd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; evas_object_hide(data); evas_object_smart_callback_call(data, SIG_DISMISSED, NULL); } static void _parent_detach(Evas_Object *obj) { ELM_CTXPOPUP_DATA_GET(obj, sd); if (!sd->parent) return; evas_object_event_callback_del_full (sd->parent, EVAS_CALLBACK_DEL, _on_parent_del, obj); evas_object_event_callback_del_full (sd->parent, EVAS_CALLBACK_MOVE, _on_parent_move, obj); evas_object_event_callback_del_full (sd->parent, EVAS_CALLBACK_RESIZE, _on_parent_resize, obj); } static void _on_content_resized(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__) { ELM_CTXPOPUP_DATA_GET(data, sd); elm_box_recalculate(sd->box); elm_layout_sizing_eval(data); } //FIXME: lost the content size when theme hook is called. static void _elm_ctxpopup_smart_theme(Eo *obj, void *_pd, va_list *list) { Elm_Ctxpopup_Smart_Data *sd = _pd; Eina_Bool int_ret; Eina_Bool *ret = va_arg(*list, Eina_Bool *); if (ret) *ret = EINA_FALSE; eo_do_super(obj, MY_CLASS, elm_wdg_theme(&int_ret)); if (!int_ret) return; elm_widget_theme_object_set (obj, sd->bg, "ctxpopup", "bg", elm_widget_style_get(obj)); elm_widget_theme_object_set (obj, sd->arrow, "ctxpopup", "arrow", elm_widget_style_get(obj)); if (sd->list) { if (!strncmp(elm_object_style_get(obj), "default", strlen("default"))) elm_object_style_set(sd->list, "ctxpopup"); else elm_object_style_set(sd->list, elm_object_style_get(obj)); } sd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; if (sd->visible) elm_layout_sizing_eval(obj); if (ret) *ret = EINA_TRUE; } /* kind of a big and tricky override here: an internal box will hold * the actual content. content aliases won't be of much help here */ static void _elm_ctxpopup_smart_content_set(Eo *obj, void *_pd, va_list *list) { Evas_Coord min_w = -1, min_h = -1; Elm_Ctxpopup_Smart_Data *sd = _pd; const char *part = va_arg(*list, const char *); Evas_Object *content = va_arg(*list, Evas_Object *); Eina_Bool *ret = va_arg(*list, Eina_Bool *); Eina_Bool int_ret = EINA_TRUE; if ((part) && (strcmp(part, "default"))) { eo_do_super(obj, MY_CLASS, elm_obj_container_content_set(part, content, &int_ret)); goto end; } if (!content) { int_ret = EINA_FALSE; goto end; } if (content == sd->content) goto end; if (sd->content) evas_object_del(sd->content); if (sd->content == sd->list) sd->list = NULL; evas_object_size_hint_weight_set (content, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND); evas_object_size_hint_fill_set (content, EVAS_HINT_FILL, EVAS_HINT_FILL); /* since it's going to be a box content, not a layout's... */ evas_object_show(content); evas_object_size_hint_min_get(content, &min_w, &min_h); evas_object_size_hint_min_set(sd->box, min_w, min_h); elm_box_pack_end(sd->box, content); sd->content = content; sd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; if (sd->visible) elm_layout_sizing_eval(obj); end: if (ret) *ret = int_ret; } static void _elm_ctxpopup_smart_content_get(Eo *obj, void *_pd, va_list *list) { const char *part = va_arg(*list, const char *); Evas_Object **ret = va_arg(*list, Evas_Object **); *ret = NULL; if ((part) && (strcmp(part, "default"))) { eo_do_super(obj, MY_CLASS, elm_obj_container_content_get(part, ret)); return; } Elm_Ctxpopup_Smart_Data *sd = _pd; *ret = sd->content; } static void _elm_ctxpopup_smart_content_unset(Eo *obj, void *_pd, va_list *list) { Evas_Object *content; const char *part = va_arg(*list, const char *); Evas_Object **ret = va_arg(*list, Evas_Object **); if ((part) && (strcmp(part, "default"))) { eo_do_super(obj, MY_CLASS, elm_obj_container_content_unset(part, &content)); goto end; } Elm_Ctxpopup_Smart_Data *sd = _pd; content = sd->content; if (!content) goto end; sd->content = NULL; sd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; if (sd->visible) elm_layout_sizing_eval(obj); end: if (ret) *ret = content; } static void _item_text_set_hook(Elm_Object_Item *it, const char *part, const char *label) { Elm_Ctxpopup_Item *ctxpopup_it; if ((part) && (strcmp(part, "default"))) return; ctxpopup_it = (Elm_Ctxpopup_Item *)it; ELM_CTXPOPUP_DATA_GET(WIDGET(ctxpopup_it), sd); elm_object_item_part_text_set(ctxpopup_it->list_item, "default", label); sd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; if (sd->visible) elm_layout_sizing_eval(WIDGET(ctxpopup_it)); } static const char * _item_text_get_hook(const Elm_Object_Item *it, const char *part) { Elm_Ctxpopup_Item *ctxpopup_it; if (part && strcmp(part, "default")) return NULL; ctxpopup_it = (Elm_Ctxpopup_Item *)it; return elm_object_item_part_text_get(ctxpopup_it->list_item, "default"); } static void _item_content_set_hook(Elm_Object_Item *it, const char *part, Evas_Object *content) { Elm_Ctxpopup_Item *ctxpopup_it; if ((part) && (strcmp(part, "icon")) && (strcmp(part, "start")) && (strcmp(part, "end"))) return; ctxpopup_it = (Elm_Ctxpopup_Item *)it; ELM_CTXPOPUP_DATA_GET(WIDGET(ctxpopup_it), sd); if ((part) && (!strcmp(part, "end"))) elm_object_item_part_content_set(ctxpopup_it->list_item, "end", content); else elm_object_item_part_content_set (ctxpopup_it->list_item, "start", content); sd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; if (sd->visible) elm_layout_sizing_eval(WIDGET(ctxpopup_it)); } static Evas_Object * _item_content_get_hook(const Elm_Object_Item *it, const char *part) { Elm_Ctxpopup_Item *ctxpopup_it; if (part && strcmp(part, "icon") && strcmp(part, "start") && strcmp(part, "end")) return NULL; ctxpopup_it = (Elm_Ctxpopup_Item *)it; if (part && !strcmp(part, "end")) return elm_object_item_part_content_get(ctxpopup_it->list_item, "end"); else return elm_object_item_part_content_get(ctxpopup_it->list_item, "start"); } static void _item_disable_hook(Elm_Object_Item *it) { Elm_Ctxpopup_Item *ctxpopup_it = (Elm_Ctxpopup_Item *)it; elm_object_item_disabled_set (ctxpopup_it->list_item, elm_widget_item_disabled_get(ctxpopup_it)); } static void _item_signal_emit_hook(Elm_Object_Item *it, const char *emission, const char *source) { Elm_Ctxpopup_Item *ctxpopup_it = (Elm_Ctxpopup_Item *)it; elm_object_item_signal_emit(ctxpopup_it->list_item, emission, source); } static void _bg_clicked_cb(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__) { ELM_CTXPOPUP_DATA_GET(data, sd); _hide_signals_emit(data, sd->dir); } static void _on_show(void *data __UNUSED__, Evas *e __UNUSED__, Evas_Object *obj, void *event_info __UNUSED__) { ELM_CTXPOPUP_DATA_GET(obj, sd); if (sd->list) { elm_list_go(sd->list); sd->visible = EINA_TRUE; elm_object_focus_set(sd->list, EINA_TRUE); return; } if (!sd->content) return; sd->visible = EINA_TRUE; evas_object_show(sd->bg); evas_object_show(sd->arrow); edje_object_signal_emit(sd->bg, "elm,state,show", "elm"); elm_layout_signal_emit(obj, "elm,state,show", "elm"); elm_layout_sizing_eval(obj); elm_object_focus_set(obj, EINA_TRUE); } static void _on_hide(void *data __UNUSED__, Evas *e __UNUSED__, Evas_Object *obj, void *event_info __UNUSED__) { ELM_CTXPOPUP_DATA_GET(obj, sd); if (!sd->visible) return; evas_object_hide(sd->bg); evas_object_hide(sd->arrow); sd->visible = EINA_FALSE; sd->list_visible = EINA_FALSE; } static void _on_move(void *data __UNUSED__, Evas *e __UNUSED__, Evas_Object *obj, void *event_info __UNUSED__) { ELM_CTXPOPUP_DATA_GET(obj, sd); if (sd->visible) evas_object_show(sd->arrow); elm_layout_sizing_eval(obj); } static void _hide_finished_cb(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__) { evas_object_hide(data); evas_object_smart_callback_call(data, SIG_DISMISSED, NULL); } static void _list_resize_cb(void *data, Evas *e __UNUSED__, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__) { ELM_CTXPOPUP_DATA_GET(data, sd); if (!sd->visible) return; if (sd->list_visible) return; sd->list_visible = EINA_TRUE; evas_object_show(sd->bg); evas_object_show(sd->arrow); elm_layout_sizing_eval(data); } static void _ctxpopup_restack_cb(void *data __UNUSED__, Evas *e __UNUSED__, Evas_Object *obj, void *event_info __UNUSED__) { ELM_CTXPOPUP_DATA_GET(obj, sd); evas_object_stack_below(sd->bg, obj); } static void _list_del(Elm_Ctxpopup_Smart_Data *sd) { if (!sd->list) return; evas_object_del(sd->list); sd->list = NULL; } static Eina_Bool _item_del_pre_hook(Elm_Object_Item *it) { Evas_Object *list; Elm_Ctxpopup_Item *ctxpopup_it = (Elm_Ctxpopup_Item *)it; ELM_CTXPOPUP_DATA_GET(WIDGET(ctxpopup_it), sd); sd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; list = elm_object_item_widget_get(ctxpopup_it->list_item); if (eina_list_count(elm_list_items_get(list)) < 2) { elm_object_item_del(ctxpopup_it->list_item); sd->items = eina_list_remove(sd->items, ctxpopup_it); evas_object_hide(WIDGET(ctxpopup_it)); return EINA_TRUE; } elm_object_item_del(ctxpopup_it->list_item); sd->items = eina_list_remove(sd->items, ctxpopup_it); if (sd->list_visible) elm_layout_sizing_eval(WIDGET(ctxpopup_it)); return EINA_TRUE; } static void _elm_ctxpopup_smart_disable(Eo *obj, void *_pd, va_list *list) { Elm_Ctxpopup_Smart_Data *sd = _pd; Eina_Bool *ret = va_arg(*list, Eina_Bool *); if (ret) *ret = EINA_FALSE; Eina_Bool int_ret; eo_do_super(obj, MY_CLASS, elm_wdg_disable(&int_ret)); if (!int_ret) return; elm_object_disabled_set(sd->list, elm_widget_disabled_get(obj)); if (ret) *ret = EINA_TRUE; } static void _elm_ctxpopup_smart_add(Eo *obj, void *_pd, va_list *list EINA_UNUSED) { Elm_Ctxpopup_Smart_Data *priv = _pd; eo_do_super(obj, MY_CLASS, evas_obj_smart_add()); if (!elm_layout_theme_set (obj, "ctxpopup", "base", elm_widget_style_get(obj))) CRITICAL("Failed to set layout!"); elm_layout_signal_callback_add (obj, "elm,action,hide,finished", "", _hide_finished_cb, obj); //Background priv->bg = edje_object_add(evas_object_evas_get(obj)); elm_widget_theme_object_set(obj, priv->bg, "ctxpopup", "bg", "default"); edje_object_signal_callback_add (priv->bg, "elm,action,click", "", _bg_clicked_cb, obj); evas_object_stack_below(priv->bg, obj); //Arrow priv->arrow = edje_object_add(evas_object_evas_get(obj)); elm_widget_theme_object_set (obj, priv->arrow, "ctxpopup", "arrow", "default"); priv->dir_priority[0] = ELM_CTXPOPUP_DIRECTION_UP; priv->dir_priority[1] = ELM_CTXPOPUP_DIRECTION_LEFT; priv->dir_priority[2] = ELM_CTXPOPUP_DIRECTION_RIGHT; priv->dir_priority[3] = ELM_CTXPOPUP_DIRECTION_DOWN; priv->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; evas_object_event_callback_add (obj, EVAS_CALLBACK_RESTACK, _ctxpopup_restack_cb, obj); priv->box = elm_box_add(obj); evas_object_size_hint_weight_set (priv->box, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND); evas_object_event_callback_add (priv->box, EVAS_CALLBACK_RESIZE, _on_content_resized, obj); /* box will be our content placeholder, thus the parent's version call */ Eina_Bool ret; eo_do_super(obj, MY_CLASS, elm_obj_container_content_set("elm.swallow.content", priv->box, &ret)); evas_object_event_callback_add(obj, EVAS_CALLBACK_SHOW, _on_show, NULL); evas_object_event_callback_add(obj, EVAS_CALLBACK_HIDE, _on_hide, NULL); evas_object_event_callback_add(obj, EVAS_CALLBACK_MOVE, _on_move, NULL); elm_widget_can_focus_set(obj, EINA_TRUE); } static void _elm_ctxpopup_smart_del(Eo *obj, void *_pd, va_list *list EINA_UNUSED) { Elm_Ctxpopup_Smart_Data *sd = _pd; Elm_Ctxpopup_Item *it; evas_object_event_callback_del_full (sd->box, EVAS_CALLBACK_RESIZE, _on_content_resized, obj); _parent_detach(obj); elm_ctxpopup_clear(obj); evas_object_del(sd->arrow); sd->arrow = NULL; /* stops _sizing_eval() from going on on deletion */ evas_object_del(sd->bg); sd->bg = NULL; EINA_LIST_FREE(sd->items, it) elm_widget_item_free(it); eo_do_super(obj, MY_CLASS, evas_obj_smart_del()); } static void _elm_ctxpopup_smart_parent_set(Eo *obj, void *_pd EINA_UNUSED, va_list *list) { Evas_Object *parent = va_arg(*list, Evas_Object *); //default parent is to be hover parent elm_ctxpopup_hover_parent_set(obj, parent); } EAPI Evas_Object * elm_ctxpopup_add(Evas_Object *parent) { EINA_SAFETY_ON_NULL_RETURN_VAL(parent, NULL); Evas_Object *obj = eo_add(MY_CLASS, parent); eo_unref(obj); /* access: parent could be any object such as elm_list which does not know elc_ctxpopup as its child object in the focus_next(); */ Elm_Widget_Smart_Data *wd = eo_data_scope_get(obj, ELM_OBJ_WIDGET_CLASS); wd->highlight_root = EINA_TRUE; return obj; } static void _constructor(Eo *obj, void *_pd EINA_UNUSED, va_list *list EINA_UNUSED) { eo_do_super(obj, MY_CLASS, eo_constructor()); eo_do(obj, evas_obj_type_set(MY_CLASS_NAME), evas_obj_smart_callbacks_descriptions_set(_smart_callbacks, NULL)); elm_widget_sub_object_add(eo_parent_get(obj), obj); } EAPI void elm_ctxpopup_hover_parent_set(Evas_Object *obj, Evas_Object *parent) { ELM_CTXPOPUP_CHECK(obj); eo_do(obj, elm_obj_ctxpopup_hover_parent_set(parent)); } static void _hover_parent_set(Eo *obj, void *_pd, va_list *list) { Evas_Coord x, y, w, h; Evas_Object *parent = va_arg(*list, Evas_Object *); Elm_Ctxpopup_Smart_Data *sd = _pd; if (!parent) return; _parent_detach(obj); evas_object_event_callback_add (parent, EVAS_CALLBACK_DEL, _on_parent_del, obj); evas_object_event_callback_add (parent, EVAS_CALLBACK_MOVE, _on_parent_move, obj); evas_object_event_callback_add (parent, EVAS_CALLBACK_RESIZE, _on_parent_resize, obj); sd->parent = parent; //Update Background evas_object_geometry_get(parent, &x, &y, &w, &h); if (!strcmp(evas_object_type_get(parent), "elm_win")) x = y = 0; evas_object_move(sd->bg, x, y); evas_object_resize(sd->bg, w, h); if (sd->visible) elm_layout_sizing_eval(obj); } EAPI Evas_Object * elm_ctxpopup_hover_parent_get(const Evas_Object *obj) { ELM_CTXPOPUP_CHECK(obj) NULL; Evas_Object *ret = NULL; eo_do((Eo *) obj, elm_obj_ctxpopup_hover_parent_get(&ret)); return ret; } static void _hover_parent_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Evas_Object **ret = va_arg(*list, Evas_Object **); Elm_Ctxpopup_Smart_Data *sd = _pd; *ret = sd->parent; } EAPI void elm_ctxpopup_clear(Evas_Object *obj) { ELM_CTXPOPUP_CHECK(obj); eo_do(obj, elm_obj_ctxpopup_clear()); } static void _clear(Eo *obj EINA_UNUSED, void *_pd, va_list *list EINA_UNUSED) { Elm_Ctxpopup_Smart_Data *sd = _pd; _list_del(sd); sd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; } EAPI void elm_ctxpopup_horizontal_set(Evas_Object *obj, Eina_Bool horizontal) { ELM_CTXPOPUP_CHECK(obj); eo_do(obj, elm_obj_ctxpopup_horizontal_set(horizontal)); } static void _horizontal_set(Eo *obj, void *_pd, va_list *list) { Eina_Bool horizontal = va_arg(*list, int); Elm_Ctxpopup_Smart_Data *sd = _pd; sd->horizontal = !!horizontal; if (!sd->list) return; elm_list_horizontal_set(sd->list, sd->horizontal); sd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; if (sd->visible) elm_layout_sizing_eval(obj); } EAPI Eina_Bool elm_ctxpopup_horizontal_get(const Evas_Object *obj) { ELM_CTXPOPUP_CHECK(obj) EINA_FALSE; Eina_Bool ret = EINA_FALSE; eo_do((Eo *) obj, elm_obj_ctxpopup_horizontal_get(&ret)); return ret; } static void _horizontal_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Eina_Bool *ret = va_arg(*list, Eina_Bool *); Elm_Ctxpopup_Smart_Data *sd = _pd; *ret = sd->horizontal; } EAPI Elm_Object_Item * elm_ctxpopup_item_append(Evas_Object *obj, const char *label, Evas_Object *icon, Evas_Smart_Cb func, const void *data) { ELM_CTXPOPUP_CHECK(obj) NULL; Elm_Object_Item *ret = NULL; eo_do(obj, elm_obj_ctxpopup_item_append(label, icon, func, data, &ret)); return ret; } static void _item_wrap_cb(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__) { Elm_Ctxpopup_Item *item = data; if (!item->wcb.org_func_cb) return; item->wcb.org_func_cb((void *)item->wcb.org_data, item->wcb.cobj, item); } static void _item_append(Eo *obj, void *_pd, va_list *list) { const char *label = va_arg(*list, const char *); Evas_Object *icon = va_arg(*list, Evas_Object *); Evas_Smart_Cb func = va_arg(*list, Evas_Smart_Cb); const void *data = va_arg(*list, const void *); Elm_Object_Item **ret = va_arg(*list, Elm_Object_Item **); *ret = NULL; Elm_Ctxpopup_Item *item; Elm_Ctxpopup_Smart_Data *sd = _pd; item = elm_widget_item_new(obj, Elm_Ctxpopup_Item); if (!item) return; elm_widget_item_del_pre_hook_set(item, _item_del_pre_hook); elm_widget_item_disable_hook_set(item, _item_disable_hook); elm_widget_item_text_set_hook_set(item, _item_text_set_hook); elm_widget_item_text_get_hook_set(item, _item_text_get_hook); elm_widget_item_content_set_hook_set(item, _item_content_set_hook); elm_widget_item_content_get_hook_set(item, _item_content_get_hook); elm_widget_item_signal_emit_hook_set(item, _item_signal_emit_hook); if (!sd->list) { //The first item is appended. sd->list = elm_list_add(obj); elm_list_mode_set(sd->list, ELM_LIST_EXPAND); elm_list_horizontal_set(sd->list, sd->horizontal); evas_object_event_callback_add (sd->list, EVAS_CALLBACK_RESIZE, _list_resize_cb, obj); elm_layout_content_set(obj, "default", sd->list); } item->wcb.org_func_cb = func; item->wcb.org_data = data; item->wcb.cobj = obj; item->list_item = elm_list_item_append(sd->list, label, icon, NULL, _item_wrap_cb, item); sd->items = eina_list_append(sd->items, item); sd->dir = ELM_CTXPOPUP_DIRECTION_UNKNOWN; if (sd->visible) elm_layout_sizing_eval(obj); *ret = (Elm_Object_Item *)item; } EAPI void elm_ctxpopup_direction_priority_set(Evas_Object *obj, Elm_Ctxpopup_Direction first, Elm_Ctxpopup_Direction second, Elm_Ctxpopup_Direction third, Elm_Ctxpopup_Direction fourth) { ELM_CTXPOPUP_CHECK(obj); eo_do(obj, elm_obj_ctxpopup_direction_priority_set(first, second, third, fourth)); } static void _direction_priority_set(Eo *obj, void *_pd, va_list *list) { Elm_Ctxpopup_Direction first = va_arg(*list, Elm_Ctxpopup_Direction); Elm_Ctxpopup_Direction second = va_arg(*list, Elm_Ctxpopup_Direction); Elm_Ctxpopup_Direction third = va_arg(*list, Elm_Ctxpopup_Direction); Elm_Ctxpopup_Direction fourth = va_arg(*list, Elm_Ctxpopup_Direction); Elm_Ctxpopup_Smart_Data *sd = _pd; sd->dir_priority[0] = first; sd->dir_priority[1] = second; sd->dir_priority[2] = third; sd->dir_priority[3] = fourth; if (sd->visible) elm_layout_sizing_eval(obj); } EAPI void elm_ctxpopup_direction_priority_get(Evas_Object *obj, Elm_Ctxpopup_Direction *first, Elm_Ctxpopup_Direction *second, Elm_Ctxpopup_Direction *third, Elm_Ctxpopup_Direction *fourth) { ELM_CTXPOPUP_CHECK(obj); eo_do(obj, elm_obj_ctxpopup_direction_priority_get(first, second, third, fourth)); } static void _direction_priority_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Elm_Ctxpopup_Direction *first = va_arg(*list, Elm_Ctxpopup_Direction *); Elm_Ctxpopup_Direction *second = va_arg(*list, Elm_Ctxpopup_Direction *); Elm_Ctxpopup_Direction *third = va_arg(*list, Elm_Ctxpopup_Direction *); Elm_Ctxpopup_Direction *fourth = va_arg(*list, Elm_Ctxpopup_Direction *); Elm_Ctxpopup_Smart_Data *sd = _pd; if (first) *first = sd->dir_priority[0]; if (second) *second = sd->dir_priority[1]; if (third) *third = sd->dir_priority[2]; if (fourth) *fourth = sd->dir_priority[3]; } EAPI Elm_Ctxpopup_Direction elm_ctxpopup_direction_get(const Evas_Object *obj) { ELM_CTXPOPUP_CHECK(obj) ELM_CTXPOPUP_DIRECTION_UNKNOWN; Elm_Ctxpopup_Direction ret = ELM_CTXPOPUP_DIRECTION_UNKNOWN; eo_do((Eo *) obj, elm_obj_ctxpopup_direction_get(&ret)); return ret; } static void _direction_get(Eo *obj EINA_UNUSED, void *_pd, va_list *list) { Elm_Ctxpopup_Direction *ret = va_arg(*list, Elm_Ctxpopup_Direction *); Elm_Ctxpopup_Smart_Data *sd = _pd; *ret = sd->dir; } EAPI void elm_ctxpopup_dismiss(Evas_Object *obj) { ELM_CTXPOPUP_CHECK(obj); eo_do(obj, elm_obj_ctxpopup_dismiss()); } static void _dismiss(Eo *obj, void *_pd, va_list *list EINA_UNUSED) { Elm_Ctxpopup_Smart_Data *sd = _pd; _hide_signals_emit(obj, sd->dir); } static void _class_constructor(Eo_Class *klass) { const Eo_Op_Func_Description func_desc[] = { EO_OP_FUNC(EO_BASE_ID(EO_BASE_SUB_ID_CONSTRUCTOR), _constructor), EO_OP_FUNC(EVAS_OBJ_SMART_ID(EVAS_OBJ_SMART_SUB_ID_ADD), _elm_ctxpopup_smart_add), EO_OP_FUNC(EVAS_OBJ_SMART_ID(EVAS_OBJ_SMART_SUB_ID_DEL), _elm_ctxpopup_smart_del), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_PARENT_SET), _elm_ctxpopup_smart_parent_set), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_DISABLE), _elm_ctxpopup_smart_disable), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_EVENT), _elm_ctxpopup_smart_event), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_THEME), _elm_ctxpopup_smart_theme), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_SUB_OBJECT_ADD), _elm_ctxpopup_smart_sub_object_add), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_FOCUS_NEXT_MANAGER_IS), _elm_ctxpopup_smart_focus_next_manager_is), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_FOCUS_NEXT), _elm_ctxpopup_smart_focus_next), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_FOCUS_DIRECTION_MANAGER_IS), _elm_ctxpopup_smart_focus_direction_manager_is), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_FOCUS_DIRECTION), _elm_ctxpopup_smart_focus_direction), EO_OP_FUNC(ELM_WIDGET_ID(ELM_WIDGET_SUB_ID_TRANSLATE), _elm_ctxpopup_smart_translate), EO_OP_FUNC(ELM_OBJ_CONTAINER_ID(ELM_OBJ_CONTAINER_SUB_ID_CONTENT_SET), _elm_ctxpopup_smart_content_set), EO_OP_FUNC(ELM_OBJ_CONTAINER_ID(ELM_OBJ_CONTAINER_SUB_ID_CONTENT_GET), _elm_ctxpopup_smart_content_get), EO_OP_FUNC(ELM_OBJ_CONTAINER_ID(ELM_OBJ_CONTAINER_SUB_ID_CONTENT_UNSET), _elm_ctxpopup_smart_content_unset), EO_OP_FUNC(ELM_OBJ_LAYOUT_ID(ELM_OBJ_LAYOUT_SUB_ID_SIZING_EVAL), _elm_ctxpopup_smart_sizing_eval), EO_OP_FUNC(ELM_OBJ_LAYOUT_ID(ELM_OBJ_LAYOUT_SUB_ID_SUB_OBJECT_ADD_ENABLE), _elm_ctxpopup_smart_layout_sub_object_add_enable), EO_OP_FUNC(ELM_OBJ_CTXPOPUP_ID(ELM_OBJ_CTXPOPUP_SUB_ID_HOVER_PARENT_SET), _hover_parent_set), EO_OP_FUNC(ELM_OBJ_CTXPOPUP_ID(ELM_OBJ_CTXPOPUP_SUB_ID_HOVER_PARENT_GET), _hover_parent_get), EO_OP_FUNC(ELM_OBJ_CTXPOPUP_ID(ELM_OBJ_CTXPOPUP_SUB_ID_CLEAR), _clear), EO_OP_FUNC(ELM_OBJ_CTXPOPUP_ID(ELM_OBJ_CTXPOPUP_SUB_ID_HORIZONTAL_SET), _horizontal_set), EO_OP_FUNC(ELM_OBJ_CTXPOPUP_ID(ELM_OBJ_CTXPOPUP_SUB_ID_HORIZONTAL_GET), _horizontal_get), EO_OP_FUNC(ELM_OBJ_CTXPOPUP_ID(ELM_OBJ_CTXPOPUP_SUB_ID_ITEM_APPEND), _item_append), EO_OP_FUNC(ELM_OBJ_CTXPOPUP_ID(ELM_OBJ_CTXPOPUP_SUB_ID_DIRECTION_PRIORITY_SET), _direction_priority_set), EO_OP_FUNC(ELM_OBJ_CTXPOPUP_ID(ELM_OBJ_CTXPOPUP_SUB_ID_DIRECTION_PRIORITY_GET), _direction_priority_get), EO_OP_FUNC(ELM_OBJ_CTXPOPUP_ID(ELM_OBJ_CTXPOPUP_SUB_ID_DIRECTION_GET), _direction_get), EO_OP_FUNC(ELM_OBJ_CTXPOPUP_ID(ELM_OBJ_CTXPOPUP_SUB_ID_DISMISS), _dismiss), EO_OP_FUNC_SENTINEL }; eo_class_funcs_set(klass, func_desc); evas_smart_legacy_type_register(MY_CLASS_NAME, klass); } static const Eo_Op_Description op_desc[] = { EO_OP_DESCRIPTION(ELM_OBJ_CTXPOPUP_SUB_ID_HOVER_PARENT_SET, "Set the Ctxpopup's parent."), EO_OP_DESCRIPTION(ELM_OBJ_CTXPOPUP_SUB_ID_HOVER_PARENT_GET, "Get the Ctxpopup's parent."), EO_OP_DESCRIPTION(ELM_OBJ_CTXPOPUP_SUB_ID_CLEAR, "Clear all items in the given ctxpopup object."), EO_OP_DESCRIPTION(ELM_OBJ_CTXPOPUP_SUB_ID_HORIZONTAL_SET, "Change the ctxpopup's orientation to horizontal or vertical."), EO_OP_DESCRIPTION(ELM_OBJ_CTXPOPUP_SUB_ID_HORIZONTAL_GET, "Get the value of current ctxpopup object's orientation."), EO_OP_DESCRIPTION(ELM_OBJ_CTXPOPUP_SUB_ID_ITEM_APPEND, "Add a new item to a ctxpopup object."), EO_OP_DESCRIPTION(ELM_OBJ_CTXPOPUP_SUB_ID_DIRECTION_PRIORITY_SET, "Set the direction priority of a ctxpopup."), EO_OP_DESCRIPTION(ELM_OBJ_CTXPOPUP_SUB_ID_DIRECTION_PRIORITY_GET, "Get the direction priority of a ctxpopup."), EO_OP_DESCRIPTION(ELM_OBJ_CTXPOPUP_SUB_ID_DIRECTION_GET, "Get the current direction of a ctxpopup."), EO_OP_DESCRIPTION(ELM_OBJ_CTXPOPUP_SUB_ID_DISMISS, "Dismiss a ctxpopup object."), EO_OP_DESCRIPTION_SENTINEL }; static const Eo_Class_Description class_desc = { EO_VERSION, MY_CLASS_NAME, EO_CLASS_TYPE_REGULAR, EO_CLASS_DESCRIPTION_OPS(&ELM_OBJ_CTXPOPUP_BASE_ID, op_desc, ELM_OBJ_CTXPOPUP_SUB_ID_LAST), NULL, sizeof(Elm_Ctxpopup_Smart_Data), _class_constructor, NULL }; EO_DEFINE_CLASS(elm_obj_ctxpopup_class_get, &class_desc, ELM_OBJ_LAYOUT_CLASS, NULL);