efl/src/lib/elementary/elm_factory.c

343 lines
10 KiB
C

#ifdef HAVE_CONFIG_H
# include "elementary_config.h"
#endif
#include <Elementary.h>
#include "elm_priv.h"
// FIXME: handle if canvas resizes
typedef struct _Widget_Data Widget_Data;
struct _Widget_Data
{
Evas_Object *obj;
Evas_Object *content;
int last_calc_count;
Evas_Coord maxminw, maxminh;
Eina_Bool eval : 1;
Eina_Bool szeval : 1;
Eina_Bool maxmin : 1;
};
static const char *widtype = NULL;
static void _del_hook(Evas_Object *obj);
static Eina_Bool _focus_next_hook(const Evas_Object *obj, Elm_Focus_Direction dir, Evas_Object **next);
static void _sizing_eval(Evas_Object *obj);
static void _eval(Evas_Object *obj);
static void _changed(Evas_Object *obj);
static void _move(void *data EINA_UNUSED, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED);
static void _resize(void *data EINA_UNUSED, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED);
static void _child_change(void *data EINA_UNUSED, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED);
static void _child_del(void *data, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED);
static void _content_set_hook(Evas_Object *obj, const char *part, Evas_Object *content);
static Evas_Object *_content_get_hook(const Evas_Object *obj, const char *part);
static Evas_Object *_content_unset_hook(Evas_Object *obj, const char *part);
static const char SIG_REALIZE[] = "realize";
static const char SIG_UNREALIZE[] = "unrealize";
static const Evas_Smart_Cb_Description _signals[] = {
{SIG_REALIZE, ""},
{SIG_UNREALIZE, ""},
{NULL, NULL}
};
static int fac = 0;
static void
_del_hook(Evas_Object *obj)
{
Widget_Data *wd = elm_widget_data_get(obj);
if (!wd) return;
if (wd->content)
{
Evas_Object *o = wd->content;
evas_object_event_callback_del_full(o,
EVAS_CALLBACK_CHANGED_SIZE_HINTS,
_child_change, obj);
evas_object_event_callback_del_full(o,
EVAS_CALLBACK_DEL,
_child_del, obj);
wd->content = NULL;
evas_object_del(o);
fac--;
// DBG("FAC-- = %i", fac);
}
free(wd);
}
static Eina_Bool
_focus_next_hook(const Evas_Object *obj, Elm_Focus_Direction dir, Evas_Object **next)
{
Widget_Data *wd = elm_widget_data_get(obj);
Evas_Object *cur;
if ((!wd) || (!wd->content)) return EINA_FALSE;
cur = wd->content;
return efl_ui_widget_focus_next_get(cur, dir, next);
}
static void
_sizing_eval(Evas_Object *obj)
{
Widget_Data *wd = elm_widget_data_get(obj);
Evas_Coord minw = -1, minh = -1, maxw = -1, maxh = -1;
if (!wd) return;
if (!wd->content) return;
evas_object_size_hint_combined_min_get(wd->content, &minw, &minh);
evas_object_size_hint_max_get(wd->content, &maxw, &maxh);
if (wd->maxmin)
{
if (minw > wd->maxminw) wd->maxminw = minw;
if (minh > wd->maxminh) wd->maxminh = minh;
evas_object_size_hint_min_set(obj, wd->maxminw, wd->maxminh);
}
else
{
evas_object_size_hint_min_set(obj, minw, minh);
}
evas_object_size_hint_max_set(obj, maxw, maxh);
// DBG("FAC SZ: %i %i | %i %i", minw, minh, maxw, maxh);
}
static void
_eval(Evas_Object *obj)
{
Evas_Coord x, y, w, h, cvx, cvy, cvw, cvh;
Widget_Data *wd = elm_widget_data_get(obj);
if (!wd) return;
evas_event_freeze(evas_object_evas_get(obj));
evas_object_geometry_get(obj, &x, &y, &w, &h);
if (w < 1) w = 1;
if (h < 1) h = 1;
evas_output_viewport_get(evas_object_evas_get(obj),
&cvx, &cvy, &cvw, &cvh);
if ((cvw < 1) || (cvh < 1)) return;
// need some fuzz value that's beyond the current viewport
// for now just make it the viewport * 3 in size (so 1 vp in each direction)
/*
cvx -= cvw;
cvy -= cvh;
cvw *= 3;
cvh *= 3;
*/
if (ELM_RECTS_INTERSECT(x, y, w, h, cvx, cvy, cvw, cvh))
{
if (!wd->content)
{
// DBG(" + %i %i %ix%i <> %i %i %ix%i", x, y, w, h, cvx, cvy, cvw, cvh);
evas_object_smart_callback_call(obj, SIG_REALIZE, NULL);
if (wd->content)
{
if (evas_object_smart_data_get(wd->content))
evas_object_smart_calculate(wd->content);
}
wd->last_calc_count =
evas_smart_objects_calculate_count_get(evas_object_evas_get(obj));
}
}
else
{
if (wd->content)
{
if (wd->last_calc_count !=
evas_smart_objects_calculate_count_get(evas_object_evas_get(obj)))
evas_object_smart_callback_call(obj, SIG_UNREALIZE, NULL);
}
}
evas_event_thaw(evas_object_evas_get(obj));
evas_event_thaw_eval(evas_object_evas_get(obj));
}
static void
_changed(Evas_Object *obj)
{
Widget_Data *wd = elm_widget_data_get(obj);
if (!wd) return;
if (wd->eval)
{
_eval(obj);
wd->eval = EINA_FALSE;
}
if (wd->szeval)
{
_sizing_eval(obj);
wd->szeval = EINA_FALSE;
}
}
static void
_move(void *data EINA_UNUSED, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED)
{
Widget_Data *wd = elm_widget_data_get(obj);
if (!wd) return;
wd->eval = EINA_TRUE;
evas_object_smart_changed(obj);
}
static void
_resize(void *data EINA_UNUSED, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED)
{
Widget_Data *wd = elm_widget_data_get(obj);
if (!wd) return;
wd->eval = EINA_TRUE;
evas_object_smart_changed(obj);
}
static void
_child_change(void *data EINA_UNUSED, Evas *e EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED)
{
Widget_Data *wd = elm_widget_data_get(data);
if (!wd) return;
wd->eval = EINA_TRUE;
wd->szeval = EINA_TRUE;
evas_object_smart_changed(data);
}
static void
_child_del(void *data, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED)
{
Evas_Object *fobj = data;
Widget_Data *wd = elm_widget_data_get(fobj);
if (!wd) return;
if (wd->content != obj) return;
evas_object_event_callback_del_full(wd->content,
EVAS_CALLBACK_CHANGED_SIZE_HINTS,
_child_change, obj);
evas_object_event_callback_del_full(wd->content,
EVAS_CALLBACK_DEL,
_child_del, obj);
wd->content = NULL;
fac--;
// DBG("FAC-- = %i", fac);
}
static Evas_Object *
_content_unset_hook(Evas_Object *obj, const char *part)
{
ELM_CHECK_WIDTYPE(obj, widtype) NULL;
Widget_Data *wd;
Evas_Object *content;
if (part && strcmp(part, "default")) return NULL;
wd = elm_widget_data_get(obj);
if (!wd || !wd->content) return NULL;
content = wd->content;
evas_object_event_callback_del_full(content,
EVAS_CALLBACK_CHANGED_SIZE_HINTS,
_child_change, obj);
evas_object_event_callback_del_full(content,
EVAS_CALLBACK_DEL,
_child_del, obj);
wd->content = NULL;
fac--;
// DBG("FAC-- = %i", fac);
return content;
}
static void
_content_set_hook(Evas_Object *obj, const char *part, Evas_Object *content)
{
ELM_CHECK_WIDTYPE(obj, widtype);
Widget_Data *wd;
Evas_Object *prev_content;
if (part && strcmp(part, "default")) return;
wd = elm_widget_data_get(obj);
if (!wd) return;
if (wd->content == content) return;
prev_content = _content_unset_hook(obj, part);
evas_object_del(prev_content);
wd->content = content;
if (!content) return;
elm_widget_resize_object_set(obj, content);
evas_object_event_callback_add(content, EVAS_CALLBACK_DEL, _child_del, obj);
evas_object_event_callback_add(content, EVAS_CALLBACK_CHANGED_SIZE_HINTS,
_child_change, obj);
wd->eval = EINA_TRUE;
wd->szeval = EINA_TRUE;
evas_object_smart_changed(obj);
fac++;
}
static Evas_Object *
_content_get_hook(const Evas_Object *obj, const char *part)
{
ELM_CHECK_WIDTYPE(obj, widtype) NULL;
Widget_Data *wd;
if (part && strcmp(part, "default")) return NULL;
wd = elm_widget_data_get(obj);
if (!wd) return NULL;
return wd->content;
}
EAPI Evas_Object *
elm_factory_add(Evas_Object *parent)
{
Evas_Object *obj;
Evas *e;
Widget_Data *wd;
ELM_WIDGET_STANDARD_SETUP(wd, Widget_Data, parent, e, obj, NULL);
ELM_SET_WIDTYPE(widtype, "factory");
elm_widget_type_set(obj, "factory");
elm_widget_sub_object_add(parent, obj);
elm_widget_data_set(obj, wd);
elm_widget_del_hook_set(obj, _del_hook);
efl_ui_widget_focus_next_hook_set(obj, _focus_next_hook);
elm_widget_content_set_hook_set(obj, _content_set_hook);
elm_widget_content_get_hook_set(obj, _content_get_hook);
elm_widget_content_unset_hook_set(obj, _content_unset_hook);
elm_widget_can_focus_set(obj, EINA_FALSE);
elm_widget_changed_hook_set(obj, _changed);
evas_object_event_callback_add(obj, EVAS_CALLBACK_MOVE, _move, NULL);
evas_object_event_callback_add(obj, EVAS_CALLBACK_RESIZE, _resize, NULL);
evas_object_smart_callbacks_descriptions_set(obj, _signals);
wd->obj = obj;
wd->last_calc_count = -1;
return obj;
}
EAPI void
elm_factory_maxmin_mode_set(Evas_Object *obj, Eina_Bool enabled)
{
ELM_CHECK_WIDTYPE(obj, widtype);
Widget_Data *wd = elm_widget_data_get(obj);
if (!wd) return;
wd->maxmin = !!enabled;
}
EAPI Eina_Bool
elm_factory_maxmin_mode_get(const Evas_Object *obj)
{
ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
Widget_Data *wd = elm_widget_data_get(obj);
if (!wd) return EINA_FALSE;
return wd->maxmin;
}
EAPI void
elm_factory_maxmin_reset_set(Evas_Object *obj)
{
ELM_CHECK_WIDTYPE(obj, widtype);
Widget_Data *wd = elm_widget_data_get(obj);
if (!wd) return;
wd->maxminw = 0;
wd->maxminh = 0;
wd->eval = EINA_TRUE;
wd->szeval = EINA_TRUE;
evas_object_smart_changed(obj);
}