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

1792 lines
50 KiB
C

#include <Elementary.h>
#include "elm_priv.h"
#include "elm_widget_layout.h"
static const char LAYOUT_SMART_NAME[] = "elm_layout";
#define ELM_LAYOUT_DATA_GET(o, sd) \
Elm_Layout_Smart_Data * sd = evas_object_smart_data_get(o)
#define ELM_LAYOUT_DATA_GET_OR_RETURN(o, ptr) \
ELM_LAYOUT_DATA_GET(o, ptr); \
if (!ptr) \
{ \
CRITICAL("No widget data for object %p (%s)", \
o, evas_object_type_get(o)); \
return; \
}
#define ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(o, ptr, val) \
ELM_LAYOUT_DATA_GET(o, ptr); \
if (!ptr) \
{ \
CRITICAL("No widget data for object %p (%s)", \
o, evas_object_type_get(o)); \
return val; \
}
#define ELM_LAYOUT_CHECK(obj) \
if (!obj || !elm_widget_type_check((obj), LAYOUT_SMART_NAME, __func__)) \
return
static const char SIG_THEME_CHANGED[] = "theme,changed";
/* no *direct* instantiation of this class, so far */
__UNUSED__ static Evas_Smart *_elm_layout_smart_class_new(void);
/* smart callbacks coming from elm layout objects: */
static const Evas_Smart_Cb_Description _smart_callbacks[] = {
{SIG_THEME_CHANGED, ""},
{NULL, NULL}
};
/* these are data operated by layout's class functions internally, and
* should not be messed up by inhering classes */
typedef struct _Elm_Layout_Sub_Object_Data Elm_Layout_Sub_Object_Data;
typedef struct _Elm_Layout_Sub_Object_Cursor Elm_Layout_Sub_Object_Cursor;
struct _Elm_Layout_Sub_Object_Data
{
const char *part;
Evas_Object *obj;
enum {
SWALLOW,
BOX_APPEND,
BOX_PREPEND,
BOX_INSERT_BEFORE,
BOX_INSERT_AT,
TABLE_PACK,
TEXT
} type;
union {
union {
const Evas_Object *reference;
unsigned int pos;
} box;
struct
{
unsigned short col, row, colspan, rowspan;
} table;
struct
{
const char *text;
} text;
} p;
};
struct _Elm_Layout_Sub_Object_Cursor
{
Evas_Object *obj;
const char *part;
const char *cursor;
const char *style;
Eina_Bool engine_only : 1;
};
/* layout's sizing evaluation is deferred. evaluation requests are
* queued up and only flag the object as 'changed'. when it comes to
* Evas's rendering phase, it will be addressed, finally (see
* _elm_layout_smart_calculate()). */
static void
_elm_layout_smart_sizing_eval(Evas_Object *obj)
{
ELM_LAYOUT_DATA_GET(obj, sd);
if (sd->needs_size_calc) return;
sd->needs_size_calc = EINA_TRUE;
evas_object_smart_changed(obj);
}
static void
_on_sub_object_size_hint_change(void *data,
Evas *e __UNUSED__,
Evas_Object *obj __UNUSED__,
void *event_info __UNUSED__)
{
ELM_LAYOUT_DATA_GET(data, sd);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(data);
}
static void
_part_cursor_free(Elm_Layout_Sub_Object_Cursor *pc)
{
eina_stringshare_del(pc->part);
eina_stringshare_del(pc->style);
eina_stringshare_del(pc->cursor);
free(pc);
}
/* Elementary smart class for all widgets having an Edje layout as a
* building block */
EVAS_SMART_SUBCLASS_NEW
(LAYOUT_SMART_NAME, _elm_layout, Elm_Layout_Smart_Class,
Elm_Container_Smart_Class, elm_container_smart_class_get, _smart_callbacks);
EAPI const Elm_Layout_Smart_Class *
elm_layout_smart_class_get(void)
{
static Elm_Layout_Smart_Class _sc =
ELM_LAYOUT_SMART_CLASS_INIT_NAME_VERSION(LAYOUT_SMART_NAME);
static const Elm_Layout_Smart_Class *class = NULL;
if (class)
return class;
_elm_layout_smart_set(&_sc);
class = &_sc;
return class;
}
static void
_sizing_eval(Evas_Object *obj, Elm_Layout_Smart_Data *sd)
{
Evas_Coord minw = -1, minh = -1;
edje_object_size_min_calc(ELM_WIDGET_DATA(sd)->resize_obj, &minw, &minh);
evas_object_size_hint_min_set(obj, minw, minh);
evas_object_size_hint_max_set(obj, -1, -1);
}
/* common content cases for layout objects: icon and text */
static inline void
_icon_signal_emit(Elm_Layout_Smart_Data *sd,
Elm_Layout_Sub_Object_Data *sub_d,
Eina_Bool visible)
{
char buf[64];
const char *type;
if (sub_d->type != SWALLOW ||
(strcmp("elm.swallow.icon", sub_d->part) &&
(strcmp("elm.swallow.end", sub_d->part))))
return;
type = sub_d->part + 12;
snprintf(buf, sizeof(buf), "elm,state,%s,%s", type,
visible ? "visible" : "hidden");
edje_object_signal_emit(ELM_WIDGET_DATA(sd)->resize_obj, buf, "elm");
/* themes might need imediate action here */
edje_object_message_signal_process(ELM_WIDGET_DATA(sd)->resize_obj);
}
static inline void
_text_signal_emit(Elm_Layout_Smart_Data *sd,
Elm_Layout_Sub_Object_Data *sub_d,
Eina_Bool visible)
{
char buf[64];
if (sub_d->type != TEXT || strcmp("elm.text", sub_d->part))
return;
snprintf(buf, sizeof(buf),
visible ? "elm,state,text,visible" : "elm,state,text,hidden");
edje_object_signal_emit(ELM_WIDGET_DATA(sd)->resize_obj, buf, "elm");
/* themes might need imediate action here */
edje_object_message_signal_process(ELM_WIDGET_DATA(sd)->resize_obj);
}
static void
_parts_signals_emit(Elm_Layout_Smart_Data *sd)
{
const Eina_List *l;
Elm_Layout_Sub_Object_Data *sub_d;
EINA_LIST_FOREACH (sd->subs, l, sub_d)
{
_icon_signal_emit(sd, sub_d, EINA_TRUE);
_text_signal_emit(sd, sub_d, EINA_TRUE);
}
}
static void
_parts_text_fix(Elm_Layout_Smart_Data *sd)
{
const Eina_List *l;
Elm_Layout_Sub_Object_Data *sub_d;
EINA_LIST_FOREACH (sd->subs, l, sub_d)
{
if (sub_d->type == TEXT)
edje_object_part_text_escaped_set
(ELM_WIDGET_DATA(sd)->resize_obj, sub_d->part, sub_d->p.text.text);
}
}
static void
_part_cursor_part_apply(const Elm_Layout_Sub_Object_Cursor *pc)
{
elm_object_cursor_set(pc->obj, pc->cursor);
elm_object_cursor_style_set(pc->obj, pc->style);
elm_object_cursor_theme_search_enabled_set(pc->obj, pc->engine_only);
}
static void
_parts_cursors_apply(Elm_Layout_Smart_Data *sd)
{
const Eina_List *l;
const char *file, *group;
Elm_Layout_Sub_Object_Cursor *pc;
edje_object_file_get(ELM_WIDGET_DATA(sd)->resize_obj, &file, &group);
EINA_LIST_FOREACH (sd->parts_cursors, l, pc)
{
Evas_Object *obj = (Evas_Object *)edje_object_part_object_get
(ELM_WIDGET_DATA(sd)->resize_obj, pc->part);
if (!obj)
{
pc->obj = NULL;
WRN("no part '%s' in group '%s' of file '%s'. "
"Cannot set cursor '%s'",
pc->part, group, file, pc->cursor);
continue;
}
else if (evas_object_pass_events_get(obj))
{
pc->obj = NULL;
WRN("part '%s' in group '%s' of file '%s' has mouse_events: 0. "
"Cannot set cursor '%s'",
pc->part, group, file, pc->cursor);
continue;
}
pc->obj = obj;
_part_cursor_part_apply(pc);
}
}
static void
_reload_theme(void *data, Evas_Object *obj,
const char *emission __UNUSED__, const char *source __UNUSED__)
{
Evas_Object *layout = data;
const char *file;
const char *group;
edje_object_file_get(obj, &file, &group);
elm_layout_file_set(layout, file, group);
}
static void
_visuals_refresh(Evas_Object *obj,
Elm_Layout_Smart_Data *sd)
{
_parts_signals_emit(sd);
_parts_text_fix(sd);
_parts_cursors_apply(sd);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(obj);
edje_object_signal_callback_del(ELM_WIDGET_DATA(sd)->resize_obj,
"edje,change,file", "edje",
_reload_theme);
edje_object_signal_callback_add(ELM_WIDGET_DATA(sd)->resize_obj,
"edje,change,file", "edje",
_reload_theme, obj);
}
static Eina_Bool
_elm_layout_smart_disable(Evas_Object *obj)
{
ELM_LAYOUT_DATA_GET(obj, sd);
if (elm_object_disabled_get(obj))
edje_object_signal_emit
(ELM_WIDGET_DATA(sd)->resize_obj, "elm,state,disabled", "elm");
else
edje_object_signal_emit
(ELM_WIDGET_DATA(sd)->resize_obj, "elm,state,enabled", "elm");
return EINA_TRUE;
}
static Eina_Bool
_elm_layout_smart_theme(Evas_Object *obj)
{
Eina_Bool ret;
const char *fh;
ELM_LAYOUT_DATA_GET(obj, sd);
if (!ELM_WIDGET_CLASS(_elm_layout_parent_sc)->theme(obj)) return EINA_FALSE;
/* function already prints error messages, if any */
ret = elm_widget_theme_object_set
(obj, ELM_WIDGET_DATA(sd)->resize_obj, sd->klass, sd->group,
elm_widget_style_get(obj));
edje_object_mirrored_set
(ELM_WIDGET_DATA(sd)->resize_obj, elm_widget_mirrored_get(obj));
edje_object_scale_set
(ELM_WIDGET_DATA(sd)->resize_obj,
elm_widget_scale_get(obj) * elm_config_scale_get());
fh = edje_object_data_get
(ELM_WIDGET_DATA(sd)->resize_obj, "focus_highlight");
if ((fh) && (!strcmp(fh, "on")))
elm_widget_highlight_in_theme_set(obj, EINA_TRUE);
else
elm_widget_highlight_in_theme_set(obj, EINA_FALSE);
evas_object_smart_callback_call(obj, SIG_THEME_CHANGED, NULL);
_visuals_refresh(obj, sd);
return ret;
}
static void *
_elm_layout_list_data_get(const Eina_List *list)
{
Elm_Layout_Sub_Object_Data *sub_d = eina_list_data_get(list);
return sub_d->obj;
}
static Eina_Bool
_elm_layout_smart_on_focus(Evas_Object *obj)
{
ELM_LAYOUT_DATA_GET(obj, sd);
if (elm_widget_focus_get(obj))
{
elm_layout_signal_emit(obj, "elm,action,focus", "elm");
evas_object_focus_set(ELM_WIDGET_DATA(sd)->resize_obj, EINA_TRUE);
}
else
{
elm_layout_signal_emit(obj, "elm,action,unfocus", "elm");
evas_object_focus_set(ELM_WIDGET_DATA(sd)->resize_obj, EINA_FALSE);
}
return EINA_TRUE;
}
/* WARNING: if you're making a widget *not* supposed to have focusable
* child objects, but still inheriting from elm_layout, just set its
* focus_next smart function back to NULL */
static Eina_Bool
_elm_layout_smart_focus_next(const Evas_Object *obj,
Elm_Focus_Direction dir,
Evas_Object **next)
{
const Eina_List *items;
void *(*list_data_get)(const Eina_List *list);
ELM_LAYOUT_DATA_GET(obj, sd);
if (!sd->subs) return EINA_FALSE;
if ((items = elm_widget_focus_custom_chain_get(obj)))
list_data_get = eina_list_data_get;
else
{
items = sd->subs;
list_data_get = _elm_layout_list_data_get;
}
return elm_widget_focus_list_next_get
(obj, items, list_data_get, dir, next);
}
static Eina_Bool
_elm_layout_smart_sub_object_add(Evas_Object *obj,
Evas_Object *sobj)
{
if (!ELM_WIDGET_CLASS(_elm_layout_parent_sc)->sub_object_add(obj, sobj))
return EINA_FALSE;
evas_object_event_callback_add
(sobj, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
_on_sub_object_size_hint_change, obj);
return EINA_TRUE;
}
static Eina_Bool
_elm_layout_smart_sub_object_del(Evas_Object *obj,
Evas_Object *sobj)
{
Eina_List *l;
Elm_Layout_Sub_Object_Data *sub_d;
ELM_LAYOUT_DATA_GET(obj, sd);
evas_object_event_callback_del_full
(sobj, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
_on_sub_object_size_hint_change, obj);
if (!ELM_WIDGET_CLASS(_elm_layout_parent_sc)->sub_object_del(obj, sobj))
return EINA_FALSE;
EINA_LIST_FOREACH (sd->subs, l, sub_d)
{
if (sub_d->obj != sobj) continue;
sd->subs = eina_list_remove_list(sd->subs, l);
_icon_signal_emit(sd, sub_d, EINA_FALSE);
eina_stringshare_del(sub_d->part);
free(sub_d);
break;
}
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(obj);
return EINA_TRUE;
}
static Eina_Bool
_elm_layout_smart_focus_direction(const Evas_Object *obj,
const Evas_Object *base,
double degree,
Evas_Object **direction,
double *weight)
{
const Eina_List *items;
void *(*list_data_get)(const Eina_List *list);
ELM_LAYOUT_DATA_GET(obj, sd);
if (!sd->subs) return EINA_FALSE;
/* Focus chain (This block is diferent of elm_win cycle) */
if ((items = elm_widget_focus_custom_chain_get(obj)))
list_data_get = eina_list_data_get;
else
{
items = sd->subs;
list_data_get = _elm_layout_list_data_get;
if (!items) return EINA_FALSE;
}
return elm_widget_focus_list_direction_get
(obj, base, items, list_data_get, degree, direction, weight);
}
static void
_elm_layout_smart_signal(Evas_Object *obj,
const char *emission,
const char *source)
{
ELM_LAYOUT_DATA_GET(obj, sd);
edje_object_signal_emit(ELM_WIDGET_DATA(sd)->resize_obj, emission, source);
}
static void
_edje_signal_callback(void *data,
Evas_Object *obj __UNUSED__,
const char *emission,
const char *source)
{
Edje_Signal_Data *esd = data;
esd->func(esd->data, esd->obj, emission, source);
}
static void
_elm_layout_smart_callback_add(Evas_Object *obj,
const char *emission,
const char *source,
Edje_Signal_Cb func_cb,
void *data)
{
Edje_Signal_Data *esd;
ELM_LAYOUT_DATA_GET(obj, sd);
esd = ELM_NEW(Edje_Signal_Data);
if (!esd) return;
esd->obj = obj;
esd->func = func_cb;
esd->emission = eina_stringshare_add(emission);
esd->source = eina_stringshare_add(source);
esd->data = data;
sd->edje_signals = eina_list_append(sd->edje_signals, esd);
edje_object_signal_callback_add
(ELM_WIDGET_DATA(sd)->resize_obj, emission, source,
_edje_signal_callback, esd);
}
static void *
_elm_layout_smart_callback_del(Evas_Object *obj,
const char *emission,
const char *source,
Edje_Signal_Cb func_cb)
{
Edje_Signal_Data *esd = NULL;
void *data = NULL;
Eina_List *l;
ELM_LAYOUT_DATA_GET(obj, sd);
EINA_LIST_FOREACH (sd->edje_signals, l, esd)
{
if ((esd->func == func_cb) && (!strcmp(esd->emission, emission)) &&
(!strcmp(esd->source, source)))
{
sd->edje_signals = eina_list_remove_list(sd->edje_signals, l);
eina_stringshare_del(esd->emission);
eina_stringshare_del(esd->source);
data = esd->data;
free(esd);
edje_object_signal_callback_del_full
(ELM_WIDGET_DATA(sd)->resize_obj, emission, source,
_edje_signal_callback, esd);
return data; /* stop at 1st match */
}
}
return data;
}
static Eina_Bool
_elm_layout_part_aliasing_eval(Elm_Layout_Smart_Data *sd,
const char **part,
Eina_Bool is_text)
{
#define ALIAS_LIST(_sd, _list) \
((ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(_sd)->api))->_list)
const Elm_Layout_Part_Alias_Description *aliases = is_text ?
ALIAS_LIST(sd, text_aliases) : ALIAS_LIST(sd, content_aliases);
#undef ALIAS_LIST
if (!aliases) return EINA_TRUE;
while (aliases->alias && aliases->real_part)
{
/* NULL matches the 1st */
if ((!*part) || (!strcmp(*part, aliases->alias)))
{
*part = aliases->real_part;
break;
}
aliases++;
}
if (!*part)
{
ERR("no default content part set for object %p -- "
"part must not be NULL", ELM_WIDGET_DATA(sd)->obj);
return EINA_FALSE;
}
/* if no match, part goes on with the same value */
return EINA_TRUE;
}
static Eina_Bool
_elm_layout_smart_text_set(Evas_Object *obj,
const char *part,
const char *text)
{
Eina_List *l;
Elm_Layout_Sub_Object_Data *sub_d = NULL;
ELM_LAYOUT_DATA_GET(obj, sd);
if (!_elm_layout_part_aliasing_eval(sd, &part, EINA_TRUE))
return EINA_FALSE;
EINA_LIST_FOREACH (sd->subs, l, sub_d)
{
if ((sub_d->type == TEXT) && (!strcmp(part, sub_d->part)))
{
if (!text)
{
eina_stringshare_del(sub_d->part);
eina_stringshare_del(sub_d->p.text.text);
free(sub_d);
edje_object_part_text_escaped_set
(ELM_WIDGET_DATA(sd)->resize_obj, part, NULL);
sd->subs = eina_list_remove_list(sd->subs, l);
return EINA_TRUE;
}
else
break;
}
sub_d = NULL;
}
if (!edje_object_part_text_escaped_set
(ELM_WIDGET_DATA(sd)->resize_obj, part, text))
return EINA_FALSE;
if (!sub_d)
{
sub_d = ELM_NEW(Elm_Layout_Sub_Object_Data);
if (!sub_d) return EINA_FALSE;
sub_d->type = TEXT;
sub_d->part = eina_stringshare_add(part);
sd->subs = eina_list_append(sd->subs, sub_d);
}
eina_stringshare_replace(&sub_d->p.text.text, text);
_text_signal_emit(sd, sub_d, !!text);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(obj);
return EINA_TRUE;
}
static const char *
_elm_layout_smart_text_get(const Evas_Object *obj,
const char *part)
{
ELM_LAYOUT_DATA_GET(obj, sd);
if (!_elm_layout_part_aliasing_eval(sd, &part, EINA_TRUE))
return EINA_FALSE;
return edje_object_part_text_get(ELM_WIDGET_DATA(sd)->resize_obj, part);
}
static Eina_Bool
_elm_layout_smart_content_set(Evas_Object *obj,
const char *part,
Evas_Object *content)
{
Elm_Layout_Sub_Object_Data *sub_d;
const Eina_List *l;
ELM_LAYOUT_DATA_GET(obj, sd);
if (!_elm_layout_part_aliasing_eval(sd, &part, EINA_FALSE))
return EINA_FALSE;
EINA_LIST_FOREACH (sd->subs, l, sub_d)
{
if ((sub_d->type == SWALLOW))
{
if (!strcmp(part, sub_d->part))
{
if (content == sub_d->obj) return EINA_TRUE;
evas_object_del(sub_d->obj);
break;
}
/* was previously swallowed at another part -- mimic
* edje_object_part_swallow()'s behavior, then */
else if (content == sub_d->obj)
{
elm_widget_sub_object_del(obj, content);
break;
}
}
}
if (content)
{
if (!elm_widget_sub_object_add(obj, content))
{
ERR("could not add %p as sub object of %p", content, obj);
return EINA_FALSE;
}
if (!edje_object_part_swallow
(ELM_WIDGET_DATA(sd)->resize_obj, part, content))
{
ERR("could not swallow %p into part '%s'", content, part);
return EINA_FALSE;
}
sub_d = ELM_NEW(Elm_Layout_Sub_Object_Data);
sub_d->type = SWALLOW;
sub_d->part = eina_stringshare_add(part);
sub_d->obj = content;
sd->subs = eina_list_append(sd->subs, sub_d);
_icon_signal_emit(sd, sub_d, EINA_TRUE);
}
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(obj);
return EINA_TRUE;
}
static Evas_Object *
_elm_layout_smart_content_get(const Evas_Object *obj,
const char *part)
{
const Eina_List *l;
Elm_Layout_Sub_Object_Data *sub_d;
ELM_LAYOUT_DATA_GET(obj, sd);
if (!_elm_layout_part_aliasing_eval(sd, &part, EINA_FALSE))
return EINA_FALSE;
EINA_LIST_FOREACH (sd->subs, l, sub_d)
{
if ((sub_d->type == SWALLOW) && !strcmp(part, sub_d->part))
return sub_d->obj;
}
return NULL;
}
static Evas_Object *
_elm_layout_smart_content_unset(Evas_Object *obj,
const char *part)
{
Elm_Layout_Sub_Object_Data *sub_d;
const Eina_List *l;
ELM_LAYOUT_DATA_GET(obj, sd);
if (!_elm_layout_part_aliasing_eval(sd, &part, EINA_FALSE))
return EINA_FALSE;
EINA_LIST_FOREACH (sd->subs, l, sub_d)
{
if ((sub_d->type == SWALLOW) && (!strcmp(part, sub_d->part)))
{
Evas_Object *content;
if (!sub_d->obj) return NULL;
content = sub_d->obj; /* sub_d will die in
* _elm_layout_smart_sub_object_del */
if (!elm_widget_sub_object_del(obj, content))
{
ERR("could not remove sub object %p from %p", content, obj);
return NULL;
}
edje_object_part_unswallow
(ELM_WIDGET_DATA(sd)->resize_obj, content);
return content;
}
}
return NULL;
}
static Eina_Bool
_elm_layout_smart_box_append(Evas_Object *obj,
const char *part,
Evas_Object *child)
{
Elm_Layout_Sub_Object_Data *sub_d;
ELM_LAYOUT_DATA_GET(obj, sd);
if (!edje_object_part_box_append
(ELM_WIDGET_DATA(sd)->resize_obj, part, child))
{
ERR("child %p could not be appended to box part '%s'", child, part);
return EINA_FALSE;
}
if (!elm_widget_sub_object_add(obj, child))
{
ERR("could not add %p as sub object of %p", child, obj);
edje_object_part_box_remove
(ELM_WIDGET_DATA(sd)->resize_obj, part, child);
return EINA_FALSE;
}
sub_d = ELM_NEW(Elm_Layout_Sub_Object_Data);
sub_d->type = BOX_APPEND;
sub_d->part = eina_stringshare_add(part);
sub_d->obj = child;
sd->subs = eina_list_append(sd->subs, sub_d);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(obj);
return EINA_TRUE;
}
static Eina_Bool
_elm_layout_smart_box_prepend(Evas_Object *obj,
const char *part,
Evas_Object *child)
{
Elm_Layout_Sub_Object_Data *sub_d;
ELM_LAYOUT_DATA_GET(obj, sd);
if (!edje_object_part_box_prepend
(ELM_WIDGET_DATA(sd)->resize_obj, part, child))
{
ERR("child %p could not be prepended to box part '%s'", child, part);
return EINA_FALSE;
}
if (!elm_widget_sub_object_add(obj, child))
{
ERR("could not add %p as sub object of %p", child, obj);
edje_object_part_box_remove
(ELM_WIDGET_DATA(sd)->resize_obj, part, child);
return EINA_FALSE;
}
sub_d = ELM_NEW(Elm_Layout_Sub_Object_Data);
sub_d->type = BOX_PREPEND;
sub_d->part = eina_stringshare_add(part);
sub_d->obj = child;
sd->subs = eina_list_prepend(sd->subs, sub_d);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(obj);
return EINA_TRUE;
}
static void
_box_reference_del(void *data,
Evas *e __UNUSED__,
Evas_Object *obj __UNUSED__,
void *event_info __UNUSED__)
{
Elm_Layout_Sub_Object_Data *sub_d = data;
sub_d->p.box.reference = NULL;
}
static Eina_Bool
_elm_layout_smart_box_insert_before(Evas_Object *obj,
const char *part,
Evas_Object *child,
const Evas_Object *reference)
{
Elm_Layout_Sub_Object_Data *sub_d;
ELM_LAYOUT_DATA_GET(obj, sd);
if (!edje_object_part_box_insert_before
(ELM_WIDGET_DATA(sd)->resize_obj, part, child, reference))
{
ERR("child %p could not be inserted before %p inf box part '%s'",
child, reference, part);
return EINA_FALSE;
}
if (!elm_widget_sub_object_add(obj, child))
{
ERR("could not add %p as sub object of %p", child, obj);
edje_object_part_box_remove
(ELM_WIDGET_DATA(sd)->resize_obj, part, child);
return EINA_FALSE;
}
sub_d = ELM_NEW(Elm_Layout_Sub_Object_Data);
sub_d->type = BOX_INSERT_BEFORE;
sub_d->part = eina_stringshare_add(part);
sub_d->obj = child;
sub_d->p.box.reference = reference;
sd->subs = eina_list_append(sd->subs, sub_d);
evas_object_event_callback_add
((Evas_Object *)reference, EVAS_CALLBACK_DEL, _box_reference_del, sub_d);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(obj);
return EINA_TRUE;
}
static Eina_Bool
_elm_layout_smart_box_insert_at(Evas_Object *obj,
const char *part,
Evas_Object *child,
unsigned int pos)
{
Elm_Layout_Sub_Object_Data *sub_d;
ELM_LAYOUT_DATA_GET(obj, sd);
if (!edje_object_part_box_insert_at
(ELM_WIDGET_DATA(sd)->resize_obj, part, child, pos))
{
ERR("child %p could not be inserted at %u to box part '%s'",
child, pos, part);
return EINA_FALSE;
}
if (!elm_widget_sub_object_add(obj, child))
{
ERR("could not add %p as sub object of %p", child, obj);
edje_object_part_box_remove
(ELM_WIDGET_DATA(sd)->resize_obj, part, child);
return EINA_FALSE;
}
sub_d = ELM_NEW(Elm_Layout_Sub_Object_Data);
sub_d->type = BOX_INSERT_AT;
sub_d->part = eina_stringshare_add(part);
sub_d->obj = child;
sub_d->p.box.pos = pos;
sd->subs = eina_list_append(sd->subs, sub_d);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(obj);
return EINA_TRUE;
}
static Evas_Object *
_sub_box_remove(Evas_Object *obj,
Elm_Layout_Smart_Data *sd,
Elm_Layout_Sub_Object_Data *sub_d)
{
Evas_Object *child = sub_d->obj; /* sub_d will die in
* _elm_layout_smart_sub_object_del */
if (!elm_widget_sub_object_del(obj, child))
{
ERR("could not remove sub object %p from %p", child, obj);
return NULL;
}
if (sub_d->type == BOX_INSERT_BEFORE)
evas_object_event_callback_del_full
((Evas_Object *)sub_d->p.box.reference,
EVAS_CALLBACK_DEL, _box_reference_del, sub_d);
edje_object_part_box_remove
(ELM_WIDGET_DATA(sd)->resize_obj, sub_d->part, child);
return child;
}
static Eina_Bool
_sub_box_is(const Elm_Layout_Sub_Object_Data *sub_d)
{
switch (sub_d->type)
{
case BOX_APPEND:
case BOX_PREPEND:
case BOX_INSERT_BEFORE:
case BOX_INSERT_AT:
return EINA_TRUE;
default:
return EINA_FALSE;
}
}
static Evas_Object *
_elm_layout_smart_box_remove(Evas_Object *obj,
const char *part,
Evas_Object *child)
{
EINA_SAFETY_ON_NULL_RETURN_VAL(part, NULL);
EINA_SAFETY_ON_NULL_RETURN_VAL(child, NULL);
ELM_LAYOUT_DATA_GET(obj, sd);
const Eina_List *l;
Elm_Layout_Sub_Object_Data *sub_d;
EINA_LIST_FOREACH (sd->subs, l, sub_d)
{
if (!_sub_box_is(sub_d)) continue;
if ((sub_d->obj == child) && (!strcmp(sub_d->part, part)))
return _sub_box_remove(obj, sd, sub_d);
}
return NULL;
}
static Eina_Bool
_elm_layout_smart_box_remove_all(Evas_Object *obj,
const char *part,
Eina_Bool clear)
{
EINA_SAFETY_ON_NULL_RETURN_VAL(part, EINA_FALSE);
ELM_LAYOUT_DATA_GET(obj, sd);
Elm_Layout_Sub_Object_Data *sub_d;
Eina_List *lst;
lst = eina_list_clone(sd->subs);
EINA_LIST_FREE (lst, sub_d)
{
if (!_sub_box_is(sub_d)) continue;
if (!strcmp(sub_d->part, part))
{
/* original item's deletion handled at sub-obj-del */
Evas_Object *child = _sub_box_remove(obj, sd, sub_d);
if ((clear) && (child)) evas_object_del(child);
}
}
/* eventually something may not be added with elm_layout, delete them
* as well */
edje_object_part_box_remove_all
(ELM_WIDGET_DATA(sd)->resize_obj, part, clear);
return EINA_TRUE;
}
static Eina_Bool
_elm_layout_smart_table_pack(Evas_Object *obj,
const char *part,
Evas_Object *child,
unsigned short col,
unsigned short row,
unsigned short colspan,
unsigned short rowspan)
{
Elm_Layout_Sub_Object_Data *sub_d;
ELM_LAYOUT_DATA_GET(obj, sd);
if (!edje_object_part_table_pack
(ELM_WIDGET_DATA(sd)->resize_obj, part, child, col,
row, colspan, rowspan))
{
ERR("child %p could not be packed into box part '%s' col=%uh, row=%hu,"
" colspan=%hu, rowspan=%hu", child, part, col, row, colspan,
rowspan);
return EINA_FALSE;
}
if (!elm_widget_sub_object_add(obj, child))
{
ERR("could not add %p as sub object of %p", child, obj);
edje_object_part_table_unpack
(ELM_WIDGET_DATA(sd)->resize_obj, part, child);
return EINA_FALSE;
}
sub_d = ELM_NEW(Elm_Layout_Sub_Object_Data);
sub_d->type = TABLE_PACK;
sub_d->part = eina_stringshare_add(part);
sub_d->obj = child;
sub_d->p.table.col = col;
sub_d->p.table.row = row;
sub_d->p.table.colspan = colspan;
sub_d->p.table.rowspan = rowspan;
sd->subs = eina_list_append(sd->subs, sub_d);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(obj);
return EINA_TRUE;
}
static Evas_Object *
_sub_table_remove(Evas_Object *obj,
Elm_Layout_Smart_Data *sd,
Elm_Layout_Sub_Object_Data *sub_d)
{
Evas_Object *child;
child = sub_d->obj; /* sub_d will die in _elm_layout_smart_sub_object_del */
if (!elm_widget_sub_object_del(obj, child))
{
ERR("could not remove sub object %p from %p", child, obj);
return NULL;
}
edje_object_part_table_unpack
(ELM_WIDGET_DATA(sd)->resize_obj, sub_d->part, child);
return child;
}
static Evas_Object *
_elm_layout_smart_table_unpack(Evas_Object *obj,
const char *part,
Evas_Object *child)
{
EINA_SAFETY_ON_NULL_RETURN_VAL(part, NULL);
EINA_SAFETY_ON_NULL_RETURN_VAL(child, NULL);
ELM_LAYOUT_DATA_GET(obj, sd);
const Eina_List *l;
Elm_Layout_Sub_Object_Data *sub_d;
EINA_LIST_FOREACH (sd->subs, l, sub_d)
{
if (sub_d->type != TABLE_PACK) continue;
if ((sub_d->obj == child) && (!strcmp(sub_d->part, part)))
return _sub_table_remove(obj, sd, sub_d);
}
return NULL;
}
static Eina_Bool
_elm_layout_smart_table_clear(Evas_Object *obj,
const char *part,
Eina_Bool clear)
{
EINA_SAFETY_ON_NULL_RETURN_VAL(part, EINA_FALSE);
ELM_LAYOUT_DATA_GET(obj, sd);
Elm_Layout_Sub_Object_Data *sub_d;
Eina_List *lst;
lst = eina_list_clone(sd->subs);
EINA_LIST_FREE (lst, sub_d)
{
if (sub_d->type != TABLE_PACK) continue;
if (!strcmp(sub_d->part, part))
{
/* original item's deletion handled at sub-obj-del */
Evas_Object *child = _sub_table_remove(obj, sd, sub_d);
if ((clear) && (child)) evas_object_del(child);
}
}
/* eventually something may not be added with elm_layout, delete them
* as well */
edje_object_part_table_clear(ELM_WIDGET_DATA(sd)->resize_obj, part, clear);
return EINA_TRUE;
}
static void
_on_size_evaluate_signal(void *data,
Evas_Object *obj __UNUSED__,
const char *emission __UNUSED__,
const char *source __UNUSED__)
{
ELM_LAYOUT_DATA_GET(data, sd);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(data);
}
static void
_elm_layout_smart_add(Evas_Object *obj)
{
EVAS_SMART_DATA_ALLOC(obj, Elm_Layout_Smart_Data);
/* has to be there *before* parent's smart_add() */
ELM_WIDGET_DATA(priv)->resize_obj =
edje_object_add(evas_object_evas_get(obj));
ELM_WIDGET_CLASS(_elm_layout_parent_sc)->base.add(obj);
elm_widget_can_focus_set(obj, EINA_FALSE);
edje_object_signal_callback_add
(ELM_WIDGET_DATA(priv)->resize_obj, "size,eval", "elm",
_on_size_evaluate_signal, obj);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(priv)->api)->sizing_eval(obj);
}
static void
_elm_layout_smart_del(Evas_Object *obj)
{
Elm_Layout_Sub_Object_Data *sub_d;
Elm_Layout_Sub_Object_Cursor *pc;
Edje_Signal_Data *esd;
Evas_Object *child;
Eina_List *l;
ELM_LAYOUT_DATA_GET(obj, sd);
EINA_LIST_FREE (sd->subs, sub_d)
{
eina_stringshare_del(sub_d->part);
if (sub_d->type == TEXT)
eina_stringshare_del(sub_d->p.text.text);
free(sub_d);
}
EINA_LIST_FREE (sd->parts_cursors, pc)
_part_cursor_free(pc);
EINA_LIST_FREE (sd->edje_signals, esd)
{
eina_stringshare_del(esd->emission);
eina_stringshare_del(esd->source);
free(esd);
}
eina_stringshare_del(sd->klass);
eina_stringshare_del(sd->group);
/* let's make our Edje object the *last* to be processed, since it
* may (smart) parent other sub objects here */
EINA_LIST_FOREACH (ELM_WIDGET_DATA(sd)->subobjs, l, child)
{
if (child == ELM_WIDGET_DATA(sd)->resize_obj)
{
ELM_WIDGET_DATA(sd)->subobjs =
eina_list_demote_list(ELM_WIDGET_DATA(sd)->subobjs, l);
break;
}
}
ELM_WIDGET_CLASS(_elm_layout_parent_sc)->base.del(obj);
}
/* rewrite or extend this one on your derived class as to suit your
* needs */
static void
_elm_layout_smart_calculate(Evas_Object *obj)
{
ELM_LAYOUT_DATA_GET(obj, sd);
if (sd->needs_size_calc)
{
_sizing_eval(obj, sd);
sd->needs_size_calc = EINA_FALSE;
}
}
static void
_elm_layout_smart_set_user(Elm_Layout_Smart_Class *sc)
{
ELM_WIDGET_CLASS(sc)->base.add = _elm_layout_smart_add;
ELM_WIDGET_CLASS(sc)->base.del = _elm_layout_smart_del;
ELM_WIDGET_CLASS(sc)->base.calculate = _elm_layout_smart_calculate;
ELM_WIDGET_CLASS(sc)->theme = _elm_layout_smart_theme;
ELM_WIDGET_CLASS(sc)->disable = _elm_layout_smart_disable;
ELM_WIDGET_CLASS(sc)->focus_next = _elm_layout_smart_focus_next;
ELM_WIDGET_CLASS(sc)->focus_direction = _elm_layout_smart_focus_direction;
ELM_WIDGET_CLASS(sc)->on_focus = _elm_layout_smart_on_focus;
ELM_WIDGET_CLASS(sc)->sub_object_add = _elm_layout_smart_sub_object_add;
ELM_WIDGET_CLASS(sc)->sub_object_del = _elm_layout_smart_sub_object_del;
ELM_CONTAINER_CLASS(sc)->content_set = _elm_layout_smart_content_set;
ELM_CONTAINER_CLASS(sc)->content_get = _elm_layout_smart_content_get;
ELM_CONTAINER_CLASS(sc)->content_unset = _elm_layout_smart_content_unset;
sc->sizing_eval = _elm_layout_smart_sizing_eval;
sc->signal = _elm_layout_smart_signal;
sc->callback_add = _elm_layout_smart_callback_add;
sc->callback_del = _elm_layout_smart_callback_del;
sc->text_set = _elm_layout_smart_text_set;
sc->text_get = _elm_layout_smart_text_get;
sc->box_append = _elm_layout_smart_box_append;
sc->box_prepend = _elm_layout_smart_box_prepend;
sc->box_insert_before = _elm_layout_smart_box_insert_before;
sc->box_insert_at = _elm_layout_smart_box_insert_at;
sc->box_remove = _elm_layout_smart_box_remove;
sc->box_remove_all = _elm_layout_smart_box_remove_all;
sc->table_pack = _elm_layout_smart_table_pack;
sc->table_unpack = _elm_layout_smart_table_unpack;
sc->table_clear = _elm_layout_smart_table_clear;
}
static Elm_Layout_Sub_Object_Cursor *
_parts_cursors_find(Elm_Layout_Smart_Data *sd,
const char *part)
{
const Eina_List *l;
Elm_Layout_Sub_Object_Cursor *pc;
EINA_LIST_FOREACH (sd->parts_cursors, l, pc)
{
if (!strcmp(pc->part, part))
return pc;
}
return NULL;
}
/* The public functions down here are meant to operate on whichever
* widget inheriting from elm_layout */
EAPI Eina_Bool
elm_layout_file_set(Evas_Object *obj,
const char *file,
const char *group)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
Eina_Bool ret =
edje_object_file_set(ELM_WIDGET_DATA(sd)->resize_obj, file, group);
if (ret) _visuals_refresh(obj, sd);
else
ERR("failed to set edje file '%s', group '%s': %s",
file, group,
edje_load_error_str
(edje_object_load_error_get(ELM_WIDGET_DATA(sd)->resize_obj)));
return ret;
}
EAPI Eina_Bool
elm_layout_theme_set(Evas_Object *obj,
const char *klass,
const char *group,
const char *style)
{
Eina_Bool ret;
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
eina_stringshare_replace(&(sd->klass), klass);
eina_stringshare_replace(&(sd->group), group);
eina_stringshare_replace(&(ELM_WIDGET_DATA(sd)->style), style);
/* not issuing smart theme directly here, because one may want to
use this function inside a smart theme routine of its own */
ret = elm_widget_theme_object_set
(obj, ELM_WIDGET_DATA(sd)->resize_obj, sd->klass, sd->group,
elm_widget_style_get(obj));
evas_object_smart_callback_call(obj, SIG_THEME_CHANGED, NULL);
return ret;
}
EAPI void
elm_layout_signal_emit(Evas_Object *obj,
const char *emission,
const char *source)
{
ELM_LAYOUT_CHECK(obj);
ELM_LAYOUT_DATA_GET_OR_RETURN(obj, sd);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->signal(obj, emission, source);
}
EAPI void
elm_layout_signal_callback_add(Evas_Object *obj,
const char *emission,
const char *source,
Edje_Signal_Cb func,
void *data)
{
ELM_LAYOUT_CHECK(obj);
ELM_LAYOUT_DATA_GET_OR_RETURN(obj, sd);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->callback_add
(obj, emission, source, func, data);
}
EAPI void *
elm_layout_signal_callback_del(Evas_Object *obj,
const char *emission,
const char *source,
Edje_Signal_Cb func)
{
ELM_LAYOUT_CHECK(obj) NULL;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, NULL);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->callback_del
(obj, emission, source, func);
}
EAPI Eina_Bool
elm_layout_content_set(Evas_Object *obj,
const char *swallow,
Evas_Object *content)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return ELM_CONTAINER_CLASS(ELM_WIDGET_DATA(sd)->api)->content_set
(obj, swallow, content);
}
EAPI Evas_Object *
elm_layout_content_get(const Evas_Object *obj,
const char *swallow)
{
ELM_LAYOUT_CHECK(obj) NULL;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, NULL);
return ELM_CONTAINER_CLASS(ELM_WIDGET_DATA(sd)->api)->content_get
(obj, swallow);
}
EAPI Evas_Object *
elm_layout_content_unset(Evas_Object *obj,
const char *swallow)
{
ELM_LAYOUT_CHECK(obj) NULL;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, NULL);
return ELM_CONTAINER_CLASS(ELM_WIDGET_DATA(sd)->api)->content_unset
(obj, swallow);
}
EAPI Eina_Bool
elm_layout_text_set(Evas_Object *obj,
const char *part,
const char *text)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->text_set
(obj, part, text);
}
EAPI const char *
elm_layout_text_get(const Evas_Object *obj,
const char *part)
{
ELM_LAYOUT_CHECK(obj) NULL;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, NULL);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->text_get(obj, part);
}
EAPI Eina_Bool
elm_layout_box_append(Evas_Object *obj,
const char *part,
Evas_Object *child)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->box_append
(obj, part, child);
}
EAPI Eina_Bool
elm_layout_box_prepend(Evas_Object *obj,
const char *part,
Evas_Object *child)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->box_prepend
(obj, part, child);
}
EAPI Eina_Bool
elm_layout_box_insert_before(Evas_Object *obj,
const char *part,
Evas_Object *child,
const Evas_Object *reference)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->box_insert_before
(obj, part, child, reference);
}
EAPI Eina_Bool
elm_layout_box_insert_at(Evas_Object *obj,
const char *part,
Evas_Object *child,
unsigned int pos)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->box_insert_at
(obj, part, child, pos);
}
EAPI Evas_Object *
elm_layout_box_remove(Evas_Object *obj,
const char *part,
Evas_Object *child)
{
ELM_LAYOUT_CHECK(obj) NULL;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, NULL);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->box_remove
(obj, part, child);
}
EAPI Eina_Bool
elm_layout_box_remove_all(Evas_Object *obj,
const char *part,
Eina_Bool clear)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->box_remove_all
(obj, part, clear);
}
EAPI Eina_Bool
elm_layout_table_pack(Evas_Object *obj,
const char *part,
Evas_Object *child,
unsigned short col,
unsigned short row,
unsigned short colspan,
unsigned short rowspan)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->table_pack
(obj, part, child, col, row, colspan, rowspan);
}
EAPI Evas_Object *
elm_layout_table_unpack(Evas_Object *obj,
const char *part,
Evas_Object *child)
{
ELM_LAYOUT_CHECK(obj) NULL;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, NULL);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->table_unpack
(obj, part, child);
}
EAPI Eina_Bool
elm_layout_table_clear(Evas_Object *obj,
const char *part,
Eina_Bool clear)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->table_clear
(obj, part, clear);
}
EAPI Evas_Object *
elm_layout_edje_get(const Evas_Object *obj)
{
ELM_LAYOUT_CHECK(obj) NULL;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, NULL);
return ELM_WIDGET_DATA(sd)->resize_obj;
}
EAPI const char *
elm_layout_data_get(const Evas_Object *obj,
const char *key)
{
ELM_LAYOUT_CHECK(obj) NULL;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, NULL);
return edje_object_data_get(ELM_WIDGET_DATA(sd)->resize_obj, key);
}
EAPI void
elm_layout_sizing_eval(Evas_Object *obj)
{
ELM_LAYOUT_CHECK(obj);
ELM_LAYOUT_DATA_GET(obj, sd);
ELM_LAYOUT_CLASS(ELM_WIDGET_DATA(sd)->api)->sizing_eval(obj);
}
EAPI Eina_Bool
elm_layout_part_cursor_set(Evas_Object *obj,
const char *part_name,
const char *cursor)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
EINA_SAFETY_ON_NULL_RETURN_VAL(part_name, EINA_FALSE);
Evas_Object *part_obj;
Elm_Layout_Sub_Object_Cursor *pc;
part_obj = (Evas_Object *)edje_object_part_object_get
(ELM_WIDGET_DATA(sd)->resize_obj, part_name);
if (!part_obj)
{
const char *group, *file;
edje_object_file_get(ELM_WIDGET_DATA(sd)->resize_obj, &file, &group);
ERR("no part '%s' in group '%s' of file '%s'. Cannot set cursor '%s'",
part_name, group, file, cursor);
return EINA_FALSE;
}
if (evas_object_pass_events_get(part_obj))
{
const char *group, *file;
edje_object_file_get(ELM_WIDGET_DATA(sd)->resize_obj, &file, &group);
ERR("part '%s' in group '%s' of file '%s' has mouse_events: 0. "
"Cannot set cursor '%s'",
part_name, group, file, cursor);
return EINA_FALSE;
}
pc = _parts_cursors_find(sd, part_name);
if (pc) eina_stringshare_replace(&pc->cursor, cursor);
else
{
pc = calloc(1, sizeof(*pc));
pc->part = eina_stringshare_add(part_name);
pc->cursor = eina_stringshare_add(cursor);
pc->style = eina_stringshare_add("default");
sd->parts_cursors = eina_list_append(sd->parts_cursors, pc);
}
pc->obj = part_obj;
elm_object_sub_cursor_set(part_obj, obj, pc->cursor);
return EINA_TRUE;
}
EAPI const char *
elm_layout_part_cursor_get(const Evas_Object *obj,
const char *part_name)
{
ELM_LAYOUT_CHECK(obj) NULL;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
EINA_SAFETY_ON_NULL_RETURN_VAL(part_name, NULL);
Elm_Layout_Sub_Object_Cursor *pc = _parts_cursors_find(sd, part_name);
EINA_SAFETY_ON_NULL_RETURN_VAL(pc, NULL);
EINA_SAFETY_ON_NULL_RETURN_VAL(pc->obj, NULL);
return elm_object_cursor_get(pc->obj);
}
EAPI Eina_Bool
elm_layout_part_cursor_unset(Evas_Object *obj,
const char *part_name)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
EINA_SAFETY_ON_NULL_RETURN_VAL(part_name, EINA_FALSE);
Eina_List *l;
Elm_Layout_Sub_Object_Cursor *pc;
EINA_LIST_FOREACH (sd->parts_cursors, l, pc)
{
if (!strcmp(part_name, pc->part))
{
if (pc->obj) elm_object_cursor_unset(pc->obj);
_part_cursor_free(pc);
sd->parts_cursors = eina_list_remove_list(sd->parts_cursors, l);
return EINA_TRUE;
}
}
return EINA_FALSE;
}
EAPI Eina_Bool
elm_layout_part_cursor_style_set(Evas_Object *obj,
const char *part_name,
const char *style)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
EINA_SAFETY_ON_NULL_RETURN_VAL(part_name, EINA_FALSE);
Elm_Layout_Sub_Object_Cursor *pc = _parts_cursors_find(sd, part_name);
EINA_SAFETY_ON_NULL_RETURN_VAL(pc, EINA_FALSE);
EINA_SAFETY_ON_NULL_RETURN_VAL(pc->obj, EINA_FALSE);
eina_stringshare_replace(&pc->style, style);
elm_object_cursor_style_set(pc->obj, pc->style);
return EINA_TRUE;
}
EAPI const char *
elm_layout_part_cursor_style_get(const Evas_Object *obj,
const char *part_name)
{
ELM_LAYOUT_CHECK(obj) NULL;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, NULL);
EINA_SAFETY_ON_NULL_RETURN_VAL(part_name, NULL);
Elm_Layout_Sub_Object_Cursor *pc = _parts_cursors_find(sd, part_name);
EINA_SAFETY_ON_NULL_RETURN_VAL(pc, NULL);
EINA_SAFETY_ON_NULL_RETURN_VAL(pc->obj, NULL);
return elm_object_cursor_style_get(pc->obj);
}
EAPI Eina_Bool
elm_layout_part_cursor_engine_only_set(Evas_Object *obj,
const char *part_name,
Eina_Bool engine_only)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
EINA_SAFETY_ON_NULL_RETURN_VAL(part_name, EINA_FALSE);
Elm_Layout_Sub_Object_Cursor *pc = _parts_cursors_find(sd, part_name);
EINA_SAFETY_ON_NULL_RETURN_VAL(pc, EINA_FALSE);
EINA_SAFETY_ON_NULL_RETURN_VAL(pc->obj, EINA_FALSE);
pc->engine_only = !!engine_only;
elm_object_cursor_theme_search_enabled_set(pc->obj, pc->engine_only);
return EINA_TRUE;
}
EAPI Eina_Bool
elm_layout_part_cursor_engine_only_get(const Evas_Object *obj,
const char *part_name)
{
ELM_LAYOUT_CHECK(obj) EINA_FALSE;
ELM_LAYOUT_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
EINA_SAFETY_ON_NULL_RETURN_VAL(part_name, EINA_FALSE);
Elm_Layout_Sub_Object_Cursor *pc = _parts_cursors_find(sd, part_name);
EINA_SAFETY_ON_NULL_RETURN_VAL(pc, EINA_FALSE);
EINA_SAFETY_ON_NULL_RETURN_VAL(pc->obj, EINA_FALSE);
return elm_object_cursor_theme_search_enabled_get(pc->obj);
}
EVAS_SMART_SUBCLASS_NEW
(LAYOUT_SMART_NAME, _elm_layout_widget, Elm_Layout_Smart_Class,
Elm_Layout_Smart_Class, elm_layout_smart_class_get, NULL);
static const Elm_Layout_Part_Alias_Description _text_aliases[] =
{
{"default", "elm.text"},
{NULL, NULL}
};
/* the layout widget (not the base layout) has this extra bit */
static void
_elm_layout_widget_smart_set_user(Elm_Layout_Smart_Class *sc)
{
sc->text_aliases = _text_aliases;
}
/* And now the basic layout widget itself */
EAPI Evas_Object *
elm_layout_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_layout_widget_smart_class_new());
if (!elm_widget_sub_object_add(parent, obj))
ERR("could not add %p as sub object of %p", obj, parent);
return obj;
}