efl/legacy/elementary/src/lib/elm_bubble.c

240 lines
6.3 KiB
C

#include <Elementary.h>
#include "elm_priv.h"
#include "elm_widget_layout.h"
static const char BUBBLE_SMART_NAME[] = "elm_bubble";
typedef struct _Elm_Bubble_Smart_Data Elm_Bubble_Smart_Data;
struct _Elm_Bubble_Smart_Data
{
Elm_Layout_Smart_Data base;
Elm_Bubble_Pos pos;
};
static const char SIG_CLICKED[] = "clicked";
static const Elm_Layout_Part_Alias_Description _content_aliases[] =
{
{"default", "elm.swallow.content"},
{"icon", "elm.swallow.icon"},
{NULL, NULL}
};
static const Elm_Layout_Part_Alias_Description _text_aliases[] =
{
{"default", "elm.text"},
{"info", "elm.info"},
{NULL, NULL}
};
static const Evas_Smart_Cb_Description _smart_callbacks[] =
{
{SIG_CLICKED, ""},
{NULL, NULL}
};
static const char *corner_string[] =
{
"top_left",
"top_right",
"bottom_left",
"bottom_right"
};
#define ELM_BUBBLE_DATA_GET(o, sd) \
Elm_Bubble_Smart_Data * sd = evas_object_smart_data_get(o)
#define ELM_BUBBLE_DATA_GET_OR_RETURN(o, ptr) \
ELM_BUBBLE_DATA_GET(o, ptr); \
if (!ptr) \
{ \
CRITICAL("No widget data for object %p (%s)", \
o, evas_object_type_get(o)); \
return; \
}
#define ELM_BUBBLE_DATA_GET_OR_RETURN_VAL(o, ptr, val) \
ELM_BUBBLE_DATA_GET(o, ptr); \
if (!ptr) \
{ \
CRITICAL("No widget data for object %p (%s)", \
o, evas_object_type_get(o)); \
return val; \
}
#define ELM_BUBBLE_CHECK(obj) \
if (!obj || !elm_widget_type_check((obj), BUBBLE_SMART_NAME, __func__)) \
return
/* Inheriting from elm_layout. Besides, we need no more than what is
* there */
EVAS_SMART_SUBCLASS_NEW
(BUBBLE_SMART_NAME, _elm_bubble, Elm_Layout_Smart_Class,
Elm_Layout_Smart_Class, elm_layout_smart_class_get, _smart_callbacks);
static void
_elm_bubble_smart_sizing_eval(Evas_Object *obj)
{
Evas_Coord minw = -1, minh = -1, maxw = -1, maxh = -1;
ELM_BUBBLE_DATA_GET(obj, sd);
elm_coords_finger_size_adjust(1, &minw, 1, &minh);
edje_object_size_min_restricted_calc
(ELM_WIDGET_DATA(sd)->resize_obj, &minw, &minh, minw, minh);
evas_object_size_hint_min_set(obj, minw, minh);
evas_object_size_hint_max_set(obj, maxw, maxh);
}
static void
_on_mouse_up(void *data,
Evas *e __UNUSED__,
Evas_Object *obj __UNUSED__,
void *event_info)
{
Evas_Event_Mouse_Up *ev = event_info;
if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD)
return;
evas_object_smart_callback_call(data, SIG_CLICKED, NULL);
}
/* overriding layout's focus_next() in order to just cycle through the
* content's tree */
static Eina_Bool
_elm_bubble_smart_focus_next(const Evas_Object *obj,
Elm_Focus_Direction dir,
Evas_Object **next)
{
Evas_Object *content;
content = elm_layout_content_get(obj, NULL);
if (!content) return EINA_FALSE;
/* attempt to follow focus cycle into sub-object */
return elm_widget_focus_next_get(content, dir, next);
}
static Eina_Bool
_elm_bubble_smart_focus_direction(const Evas_Object *obj,
const Evas_Object *base,
double degree,
Evas_Object **direction,
double *weight)
{
Evas_Object *content;
content = elm_layout_content_get(obj, NULL);
if (!content) return EINA_FALSE;
/* Try Focus cycle in subitem */
return elm_widget_focus_direction_get
(content, base, degree, direction, weight);
}
static Eina_Bool
_elm_bubble_smart_text_set(Evas_Object *obj,
const char *item,
const char *label)
{
if (item && (!strcmp(item, "info") || !strcmp(item, "elm.info")))
{
if (label)
elm_layout_signal_emit(obj, "elm,state,info,visible", "elm");
else
elm_layout_signal_emit(obj, "elm,state,info,hidden", "elm");
}
else if (!_elm_bubble_parent_sc->text_set(obj, item, label))
return EINA_FALSE;
elm_layout_sizing_eval(obj);
return EINA_TRUE;
}
static void
_elm_bubble_smart_add(Evas_Object *obj)
{
EVAS_SMART_DATA_ALLOC(obj, Elm_Bubble_Smart_Data);
ELM_WIDGET_CLASS(_elm_bubble_parent_sc)->base.add(obj);
priv->pos = ELM_BUBBLE_POS_TOP_LEFT; //default
elm_widget_can_focus_set(obj, EINA_FALSE);
evas_object_event_callback_add
(ELM_WIDGET_DATA(priv)->resize_obj, EVAS_CALLBACK_MOUSE_UP,
_on_mouse_up, obj);
elm_layout_theme_set(obj, "bubble", "base", elm_widget_style_get(obj));
}
static void
_elm_bubble_smart_set_user(Elm_Layout_Smart_Class *sc)
{
ELM_WIDGET_CLASS(sc)->base.add = _elm_bubble_smart_add;
ELM_WIDGET_CLASS(sc)->focus_next = _elm_bubble_smart_focus_next;
ELM_WIDGET_CLASS(sc)->focus_direction = _elm_bubble_smart_focus_direction;
sc->text_set = _elm_bubble_smart_text_set;
sc->sizing_eval = _elm_bubble_smart_sizing_eval;
sc->content_aliases = _content_aliases;
sc->text_aliases = _text_aliases;
}
EAPI Evas_Object *
elm_bubble_add(Evas_Object *parent)
{
Evas *e;
Evas_Object *obj;
EINA_SAFETY_ON_NULL_RETURN_VAL(parent, NULL);
e = evas_object_evas_get(parent);
if (!e) return NULL;
obj = evas_object_smart_add(e, _elm_bubble_smart_class_new());
if (!elm_widget_sub_object_add(parent, obj))
ERR("could not add %p as sub object of %p", obj, parent);
elm_layout_sizing_eval(obj);
return obj;
}
EAPI void
elm_bubble_pos_set(Evas_Object *obj,
Elm_Bubble_Pos pos)
{
ELM_BUBBLE_CHECK(obj);
ELM_BUBBLE_DATA_GET_OR_RETURN(obj, sd);
if (pos < ELM_BUBBLE_POS_TOP_LEFT || pos > ELM_BUBBLE_POS_BOTTOM_RIGHT)
return;
sd->pos = pos;
eina_stringshare_replace
(&(ELM_LAYOUT_DATA(sd)->group), corner_string[sd->pos]);
ELM_WIDGET_DATA(sd)->api->theme(obj);
}
EAPI Elm_Bubble_Pos
elm_bubble_pos_get(const Evas_Object *obj)
{
ELM_BUBBLE_CHECK(obj) ELM_BUBBLE_POS_INVALID;
ELM_BUBBLE_DATA_GET_OR_RETURN_VAL(obj, sd, ELM_BUBBLE_POS_INVALID);
return sd->pos;
}