efl/src/lib/elementary/efl_ui_image.c

2280 lines
60 KiB
C

#ifdef HAVE_CONFIG_H
# include "elementary_config.h"
#endif
#define EFL_ACCESS_IMAGE_PROTECTED
#define EFL_ACCESS_PROTECTED
#define EFL_ACCESS_WIDGET_ACTION_PROTECTED
#define EFL_LAYOUT_CALC_PROTECTED
#include <Elementary.h>
#include "elm_priv.h"
#include "efl_ui_widget_image.h"
#define FMT_SIZE_T "%zu"
#define MY_CLASS EFL_UI_IMAGE_CLASS
#define MY_CLASS_NAME "Efl.Ui.Image"
#define MY_CLASS_NAME_LEGACY "elm_image"
#define NON_EXISTING (void *)-1
static const char *icon_theme = NULL;
static const char SIG_DND[] = "drop";
static const char SIG_CLICKED[] = "clicked";
static const char SIG_DOWNLOAD_START[] = "download,start";
static const char SIG_DOWNLOAD_PROGRESS[] = "download,progress";
static const char SIG_DOWNLOAD_DONE[] = "download,done";
static const char SIG_DOWNLOAD_ERROR[] = "download,error";
static const char SIG_LOAD_OPEN[] = "load,open";
static const char SIG_LOAD_READY[] = "load,ready";
static const char SIG_LOAD_ERROR[] = "load,error";
static const char SIG_LOAD_CANCEL[] = "load,cancel";
static const Evas_Smart_Cb_Description _smart_callbacks[] = {
{SIG_DND, ""},
{SIG_CLICKED, ""},
{SIG_DOWNLOAD_START, ""},
{SIG_DOWNLOAD_PROGRESS, ""},
{SIG_DOWNLOAD_DONE, ""},
{SIG_DOWNLOAD_ERROR, ""},
{SIG_LOAD_OPEN, "Triggered when the file has been opened (image size is known)"},
{SIG_LOAD_READY, "Triggered when the image file is ready for display"},
{SIG_LOAD_ERROR, "Triggered whenener an I/O or decoding error occurred"},
{SIG_LOAD_CANCEL, "Triggered whenener async I/O was cancelled"},
{NULL, NULL}
};
static Eina_Bool _key_action_activate(Evas_Object *obj, const char *params);
static Eina_Bool _efl_ui_image_smart_internal_file_set(Eo *obj, Efl_Ui_Image_Data *sd, const char *file, const Eina_File *f, const char *key);
static void _efl_ui_image_remote_copier_cancel(Eo *obj, Efl_Ui_Image_Data *sd);
void _efl_ui_image_sizing_eval(Evas_Object *obj);
static void _efl_ui_image_model_properties_changed_cb(void *data, const Efl_Event *event);
static void _on_size_hints_changed(void *data, const Efl_Event *e);
static const Elm_Action key_actions[] = {
{"activate", _key_action_activate},
{NULL, NULL}
};
typedef struct _Async_Open_Data Async_Open_Data;
struct _Async_Open_Data
{
Eo *obj;
Eina_Stringshare *file, *key;
Eina_File *f_set, *f_open;
void *map;
};
static void
_prev_img_del(Efl_Ui_Image_Data *sd)
{
elm_widget_sub_object_del(sd->self, sd->prev_img);
evas_object_smart_member_del(sd->prev_img);
evas_object_del(sd->prev_img);
sd->prev_img = NULL;
}
static void
_on_image_preloaded(void *data,
Evas *e EINA_UNUSED,
Evas_Object *obj,
void *event EINA_UNUSED)
{
Efl_Ui_Image_Data *sd = data;
Evas_Load_Error err;
sd->preload_status = EFL_UI_IMAGE_PRELOADED;
if (sd->show) evas_object_show(obj);
_prev_img_del(sd);
err = evas_object_image_load_error_get(obj);
if (!err) evas_object_smart_callback_call(sd->self, SIG_LOAD_READY, NULL);
else evas_object_smart_callback_call(sd->self, SIG_LOAD_ERROR, NULL);
}
static void
_on_mouse_up(void *data,
Evas *e EINA_UNUSED,
Evas_Object *obj EINA_UNUSED,
void *event_info)
{
ELM_WIDGET_DATA_GET_OR_RETURN(data, wd);
Evas_Event_Mouse_Up *ev = event_info;
if (ev->button != 1) return;
if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) return;
if (!wd->still_in) return;
efl_event_callback_legacy_call(data, EFL_UI_EVENT_CLICKED, NULL);
}
static Eina_Bool
_efl_ui_image_animate_cb(void *data)
{
EFL_UI_IMAGE_DATA_GET(data, sd);
if (!sd->anim) return ECORE_CALLBACK_CANCEL;
sd->cur_frame++;
if ((sd->frame_count > 0) && (sd->cur_frame > sd->frame_count))
sd->cur_frame = sd->cur_frame % sd->frame_count;
evas_object_image_animated_frame_set(sd->img, sd->cur_frame);
sd->frame_duration = evas_object_image_animated_frame_duration_get
(sd->img, sd->cur_frame, 0);
if (sd->frame_duration > 0)
ecore_timer_interval_set(sd->anim_timer, sd->frame_duration);
return ECORE_CALLBACK_RENEW;
}
static Evas_Object *
_img_new(Evas_Object *obj)
{
Evas_Object *img;
EFL_UI_IMAGE_DATA_GET(obj, sd);
img = evas_object_image_add(evas_object_evas_get(obj));
evas_object_image_scale_hint_set(img, EVAS_IMAGE_SCALE_HINT_STATIC);
evas_object_event_callback_add
(img, EVAS_CALLBACK_IMAGE_PRELOADED, _on_image_preloaded, sd);
evas_object_smart_member_add(img, obj);
elm_widget_sub_object_add(obj, img);
return img;
}
static void
_image_sizing_eval(Efl_Ui_Image_Data *sd, Evas_Object *img)
{
Evas_Coord x = 0, y = 0, w = 1, h = 1;
if (efl_isa(img, EFL_CANVAS_LAYOUT_CLASS))
{
x = sd->img_x;
y = sd->img_y;
w = sd->img_w;
h = sd->img_h;
goto done;
}
else
{
double alignh = 0.5, alignv = 0.5;
int iw = 0, ih = 0, offset_x = 0, offset_y = 0;
//1. Get the original image size (iw x ih)
evas_object_image_size_get(img, &iw, &ih);
iw = ((double)iw) * sd->scale;
ih = ((double)ih) * sd->scale;
if (iw < 1) iw = 1;
if (ih < 1) ih = 1;
//2. Calculate internal image size (w x h)
// according to (iw x ih), (sd->img_w x sd->img_h), and scale_type
switch (sd->scale_type)
{
case EFL_UI_IMAGE_SCALE_TYPE_NONE:
w = iw;
h = ih;
break;
case EFL_UI_IMAGE_SCALE_TYPE_FILL:
w = sd->img_w;
h = sd->img_h;
break;
case EFL_UI_IMAGE_SCALE_TYPE_FIT_INSIDE:
w = sd->img_w;
h = ((double)ih * w) / (double)iw;
if (h > sd->img_h)
{
h = sd->img_h;
w = ((double)iw * h) / (double)ih;
}
if (((!sd->scale_up) && (w > iw))
|| ((!sd->scale_down) && (w < iw)))
{
w = iw;
h = ih;
}
break;
case EFL_UI_IMAGE_SCALE_TYPE_FIT_OUTSIDE:
w = sd->img_w;
h = ((double)ih * w) / (double)iw;
if (h < sd->img_h)
{
h = sd->img_h;
w = ((double)iw * h) / (double)ih;
}
if (((!sd->scale_up) && (w > iw))
|| ((!sd->scale_down) && (w < iw)))
{
w = iw;
h = ih;
}
break;
case EFL_UI_IMAGE_SCALE_TYPE_TILE:
x = sd->img_x;
y = sd->img_y;
w = sd->img_w;
h = sd->img_h;
evas_object_image_fill_set(img, x, y, iw, ih);
goto done;
}
//3. Calculate offset according to align value
if (!elm_widget_is_legacy(sd->self))
{
offset_x = ((sd->img_w - w) * sd->align_x);
offset_y = ((sd->img_h - h) * sd->align_y);
}
else
{
evas_object_size_hint_align_get(sd->self, &alignh, &alignv);
if (EINA_DBL_EQ(alignh, EVAS_HINT_FILL)) alignh = 0.5;
if (EINA_DBL_EQ(alignv, EVAS_HINT_FILL)) alignv = 0.5;
offset_x = ((sd->img_w - w) * alignh);
offset_y = ((sd->img_h - h) * alignv);
}
x = sd->img_x + offset_x;
y = sd->img_y + offset_y;
//4. Fill, move, resize
if (offset_x >= 0) offset_x = 0;
if (offset_y >= 0) offset_y = 0;
evas_object_image_fill_set(img, offset_x, offset_y, w, h);
if (offset_x < 0)
{
x = sd->img_x;
w = sd->img_w;
}
if (offset_y < 0)
{
y = sd->img_y;
h = sd->img_h;
}
}
done:
evas_object_move(img, x, y);
evas_object_resize(img, w, h);
evas_object_move(sd->hit_rect, x, y);
evas_object_resize(sd->hit_rect, w, h);
}
static void
_efl_ui_image_internal_sizing_eval(Evas_Object *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
if (!sd->img) return;
_image_sizing_eval(sd, sd->img);
if (sd->prev_img) _image_sizing_eval(sd, sd->prev_img);
}
static inline void
_async_open_data_free(Async_Open_Data *data)
{
if (!data) return;
eina_stringshare_del(data->file);
eina_stringshare_del(data->key);
if (data->map) eina_file_map_free(data->f_open, data->map);
if (data->f_open) eina_file_close(data->f_open);
if (data->f_set) eina_file_close(data->f_set);
free(data);
}
static void
_efl_ui_image_async_open_do(void *data, Ecore_Thread *thread)
{
Async_Open_Data *todo = data;
Eina_File *f;
void *map = NULL;
size_t size;
if (ecore_thread_check(thread)) return;
if (todo->f_set) f = eina_file_dup(todo->f_set);
else if (todo->file)
{
// blocking
f = eina_file_open(todo->file, EINA_FALSE);
if (!f) return;
}
else
{
CRI("Async open has no input file!");
return;
}
if (ecore_thread_check(thread))
{
if (!todo->f_set) eina_file_close(f);
return;
}
// Read just enough data for map to actually do something.
size = eina_file_size_get(f);
// Read and ensure all pages are in memory for sure first just
// 1 byte per page will do. also keep a limit on how much we will
// blindly load in here to let's say 32KB (Should be enough to get
// image headers without getting to much data from the hard drive).
size = size > 32 * 1024 ? 32 * 1024 : size;
map = eina_file_map_all(f, EINA_FILE_SEQUENTIAL);
eina_file_map_populate(f, EINA_FILE_POPULATE, map, 0, size);
if (ecore_thread_check(thread))
{
if (map) eina_file_map_free(f, map);
if (!todo->f_set) eina_file_close(f);
return;
}
todo->f_open = f;
todo->map = map;
}
static void
_async_clear(Efl_Ui_Image_Data *sd)
{
sd->async.th = NULL;
sd->async.todo = NULL;
eina_stringshare_del(sd->async.file);
eina_stringshare_del(sd->async.key);
sd->async.file = NULL;
sd->async.key = NULL;
}
static void
_async_cancel(Efl_Ui_Image_Data *sd)
{
if (sd->async.th)
{
ecore_thread_cancel(sd->async.th);
((Async_Open_Data *)(sd->async.todo))->obj = NULL;
_async_clear(sd);
}
}
static void
_efl_ui_image_async_open_cancel(void *data, Ecore_Thread *thread)
{
Async_Open_Data *todo = data;
DBG("Async open thread was canceled");
if (todo->obj)
{
EFL_UI_IMAGE_DATA_GET(todo->obj, sd);
if (sd)
{
evas_object_smart_callback_call(todo->obj, SIG_LOAD_CANCEL, NULL);
if (thread == sd->async.th) _async_clear(sd);
}
}
_async_open_data_free(todo);
}
static void
_efl_ui_image_async_open_done(void *data, Ecore_Thread *thread)
{
Async_Open_Data *todo = data;
Eina_Stringshare *file, *key;
Eina_Bool ok;
Eina_File *f;
void *map;
if (todo->obj)
{
EFL_UI_IMAGE_DATA_GET(todo->obj, sd);
if (sd)
{
if (thread == sd->async.th)
{
DBG("Async open succeeded");
_async_clear(sd);
key = todo->key;
map = todo->map;
f = todo->f_open;
ok = f && map;
if (todo->file) file = todo->file;
else file = f ? eina_file_filename_get(f) : NULL;
if (ok)
{
if (sd->edje)
{
_prev_img_del(sd);
ok = edje_object_mmap_set(sd->img, f, key);
}
else
ok = _efl_ui_image_smart_internal_file_set
(sd->self, sd, file, f, key);
}
if (ok) evas_object_smart_callback_call(sd->self, SIG_LOAD_OPEN, NULL);
else evas_object_smart_callback_call(sd->self, SIG_LOAD_ERROR, NULL);
}
}
}
// close f, map and free strings
_async_open_data_free(todo);
}
static Eina_Bool
_efl_ui_image_async_file_set(Eo *obj, Efl_Ui_Image_Data *sd, const char *file,
const Eina_File *f, const char *key)
{
Async_Open_Data *todo;
if (sd->async.th &&
((file == sd->async.file) ||
(file && sd->async.file && !strcmp(file, sd->async.file))) &&
((key == sd->async.key) ||
(key && sd->async.key && !strcmp(key, sd->async.key))))
return EINA_TRUE;
todo = calloc(1, sizeof(Async_Open_Data));
if (!todo) return EINA_FALSE;
_async_cancel(sd);
todo->obj = obj;
todo->file = eina_stringshare_add(file);
todo->key = eina_stringshare_add(key);
todo->f_set = f ? eina_file_dup(f) : NULL;
eina_stringshare_replace(&sd->async.file, file);
eina_stringshare_replace(&sd->async.key, key);
sd->async.todo = todo;
sd->async.th = ecore_thread_run(_efl_ui_image_async_open_do,
_efl_ui_image_async_open_done,
_efl_ui_image_async_open_cancel, todo);
if (sd->async.th) return EINA_TRUE;
_async_open_data_free(todo);
_async_clear(sd);
DBG("Could not spawn an async thread!");
return EINA_FALSE;
}
static Eina_Bool
_efl_ui_image_edje_file_set(Evas_Object *obj,
const char *file,
const Eina_File *f,
const char *group)
{
Evas_Object *pclip;
EFL_UI_IMAGE_DATA_GET(obj, sd);
_prev_img_del(sd);
if (!sd->edje)
{
pclip = evas_object_clip_get(sd->img);
evas_object_del(sd->img);
/* Edje object instead */
sd->img = edje_object_add(evas_object_evas_get(obj));
evas_object_smart_member_add(sd->img, obj);
if (sd->show) evas_object_show(sd->img);
evas_object_clip_set(sd->img, pclip);
}
_async_cancel(sd);
sd->edje = EINA_TRUE;
if (!sd->async_enable)
{
if (f)
{
if (!edje_object_mmap_set(sd->img, f, group))
{
ERR("failed to set edje file '%s', group '%s': %s", file, group,
edje_load_error_str(edje_object_load_error_get(sd->img)));
return EINA_FALSE;
}
}
else
{
if (!edje_object_file_set(sd->img, file, group))
{
ERR("failed to set edje file '%s', group '%s': %s", file, group,
edje_load_error_str(edje_object_load_error_get(sd->img)));
return EINA_FALSE;
}
}
}
else
return _efl_ui_image_async_file_set(obj, sd, file, f, group);
/* FIXME: do i want to update icon on file change ? */
_efl_ui_image_internal_sizing_eval(obj, sd);
return EINA_TRUE;
}
EOLIAN static void
_efl_ui_image_efl_image_smooth_scale_set(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd, Eina_Bool smooth)
{
sd->smooth = smooth;
if (!sd->edje) evas_object_image_smooth_scale_set(sd->img, smooth);
}
EOLIAN static Eina_Bool
_efl_ui_image_efl_image_smooth_scale_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
return sd->smooth;
}
static Eina_Bool
_efl_ui_image_drag_n_drop_cb(void *elm_obj,
Evas_Object *obj,
Elm_Selection_Data *drop)
{
Eina_Bool ret = EINA_FALSE;
ret = efl_file_set(obj, drop->data, NULL);
if (ret)
{
DBG("dnd: %s, %s, %s", elm_widget_type_get(elm_obj),
SIG_DND, (char *)drop->data);
efl_event_callback_legacy_call(elm_obj, EFL_UI_IMAGE_EVENT_DROP, drop->data);
return EINA_TRUE;
}
return EINA_FALSE;
}
EOLIAN static void
_efl_ui_image_efl_canvas_group_group_add(Eo *obj, Efl_Ui_Image_Data *priv)
{
efl_canvas_group_add(efl_super(obj, MY_CLASS));
elm_widget_sub_object_parent_add(obj);
priv->hit_rect = evas_object_rectangle_add(evas_object_evas_get(obj));
evas_object_smart_member_add(priv->hit_rect, obj);
elm_widget_sub_object_add(obj, priv->hit_rect);
evas_object_color_set(priv->hit_rect, 0, 0, 0, 0);
evas_object_show(priv->hit_rect);
evas_object_repeat_events_set(priv->hit_rect, EINA_TRUE);
evas_object_event_callback_add
(priv->hit_rect, EVAS_CALLBACK_MOUSE_UP, _on_mouse_up, obj);
/* starts as an Evas image. may switch to an Edje object */
priv->img = _img_new(obj);
priv->smooth = EINA_TRUE;
priv->fill_inside = EINA_TRUE;
priv->aspect_fixed = EINA_TRUE;
priv->load_size = 0;
priv->scale = 1.0;
priv->scale_up = EINA_TRUE;
priv->scale_down = EINA_TRUE;
priv->align_x = 0.5;
priv->align_y = 0.5;
elm_widget_can_focus_set(obj, EINA_FALSE);
_efl_ui_image_sizing_eval(obj);
}
EOLIAN static void
_efl_ui_image_efl_canvas_group_group_del(Eo *obj, Efl_Ui_Image_Data *sd)
{
if (elm_widget_is_legacy(obj))
efl_event_callback_del(obj, EFL_GFX_EVENT_CHANGE_SIZE_HINTS,
_on_size_hints_changed, sd);
ecore_timer_del(sd->anim_timer);
evas_object_del(sd->img);
_prev_img_del(sd);
_async_cancel(sd);
if (sd->remote.copier) _efl_ui_image_remote_copier_cancel(obj, sd);
if (sd->remote.binbuf) ELM_SAFE_FREE(sd->remote.binbuf, eina_binbuf_free);
ELM_SAFE_FREE(sd->remote.key, eina_stringshare_del);
if (sd->pfuture)
{
efl_future_cancel(sd->pfuture);
sd->pfuture = NULL;
}
if (sd->model)
{
efl_event_callback_del(sd->model, EFL_MODEL_EVENT_PROPERTIES_CHANGED,
_efl_ui_image_model_properties_changed_cb, obj);
efl_unref(sd->model);
sd->model = NULL;
}
efl_canvas_group_del(efl_super(obj, MY_CLASS));
}
EOLIAN static void
_efl_ui_image_efl_gfx_position_set(Eo *obj, Efl_Ui_Image_Data *sd, Eina_Position2D pos)
{
if (_evas_object_intercept_call(obj, EVAS_OBJECT_INTERCEPT_CB_MOVE, 0, pos.x, pos.y))
return;
efl_gfx_position_set(efl_super(obj, MY_CLASS), pos);
if ((sd->img_x == pos.x) && (sd->img_y == pos.y)) return;
sd->img_x = pos.x;
sd->img_y = pos.y;
/* takes care of moving */
_efl_ui_image_internal_sizing_eval(obj, sd);
}
EOLIAN static void
_efl_ui_image_efl_gfx_size_set(Eo *obj, Efl_Ui_Image_Data *sd, Eina_Size2D sz)
{
if (_evas_object_intercept_call(obj, EVAS_OBJECT_INTERCEPT_CB_RESIZE, 0, sz.w, sz.h))
return;
if ((sd->img_w == sz.w) && (sd->img_h == sz.h)) goto super;
sd->img_w = sz.w;
sd->img_h = sz.h;
/* takes care of resizing */
_efl_ui_image_internal_sizing_eval(obj, sd);
super:
efl_gfx_size_set(efl_super(obj, MY_CLASS), sz);
}
static void
_efl_ui_image_show(Eo *obj, Efl_Ui_Image_Data *sd)
{
sd->show = EINA_TRUE;
efl_gfx_visible_set(efl_super(obj, MY_CLASS), EINA_TRUE);
if (sd->preload_status == EFL_UI_IMAGE_PRELOADING) return;
efl_gfx_visible_set(sd->img, EINA_TRUE);
_prev_img_del(sd);
}
static void
_efl_ui_image_hide(Eo *obj, Efl_Ui_Image_Data *sd)
{
sd->show = EINA_FALSE;
efl_gfx_visible_set(efl_super(obj, MY_CLASS), EINA_FALSE);
efl_gfx_visible_set(sd->img, EINA_FALSE);
_prev_img_del(sd);
}
EOLIAN static void
_efl_ui_image_efl_gfx_visible_set(Eo *obj, Efl_Ui_Image_Data *sd, Eina_Bool vis)
{
if (_evas_object_intercept_call(obj, EVAS_OBJECT_INTERCEPT_CB_VISIBLE, 0, vis))
return;
if (vis) _efl_ui_image_show(obj, sd);
else _efl_ui_image_hide(obj, sd);
}
EOLIAN static void
_efl_ui_image_efl_canvas_group_group_member_add(Eo *obj, Efl_Ui_Image_Data *sd, Evas_Object *member)
{
efl_canvas_group_member_add(efl_super(obj, MY_CLASS), member);
if (sd->hit_rect)
evas_object_raise(sd->hit_rect);
}
EOLIAN static void
_efl_ui_image_efl_gfx_color_set(Eo *obj, Efl_Ui_Image_Data *sd, int r, int g, int b, int a)
{
if (_evas_object_intercept_call(obj, EVAS_OBJECT_INTERCEPT_CB_COLOR_SET, 0, r, g, b, a))
return;
efl_gfx_color_set(efl_super(obj, MY_CLASS), r, g, b, a);
evas_object_color_set(sd->hit_rect, 0, 0, 0, 0);
evas_object_color_set(sd->img, r, g, b, a);
if (sd->prev_img) evas_object_color_set(sd->prev_img, r, g, b, a);
}
EOLIAN static void
_efl_ui_image_efl_canvas_object_clip_set(Eo *obj, Efl_Ui_Image_Data *sd, Evas_Object *clip)
{
if (_evas_object_intercept_call(obj, EVAS_OBJECT_INTERCEPT_CB_CLIP_SET, 0, clip))
return;
efl_canvas_object_clip_set(efl_super(obj, MY_CLASS), clip);
evas_object_clip_set(sd->img, clip);
if (sd->prev_img) evas_object_clip_set(sd->prev_img, clip);
}
EOLIAN static Efl_Ui_Theme_Apply
_efl_ui_image_elm_widget_theme_apply(Eo *obj, Efl_Ui_Image_Data *sd EINA_UNUSED)
{
Efl_Ui_Theme_Apply int_ret = EFL_UI_THEME_APPLY_FAILED;
if (sd->stdicon)
_elm_theme_object_icon_set(obj, sd->stdicon, elm_widget_style_get(obj));
int_ret = efl_ui_widget_theme_apply(efl_super(obj, MY_CLASS));
if (!int_ret) return EFL_UI_THEME_APPLY_FAILED;
_efl_ui_image_sizing_eval(obj);
return int_ret;
}
static Eina_Bool
_key_action_activate(Evas_Object *obj, const char *params EINA_UNUSED)
{
efl_event_callback_legacy_call(obj, EFL_UI_EVENT_CLICKED, NULL);
return EINA_TRUE;
}
// TODO: remove this function after using the widget's scale value instead of image's scale value,
static void
_efl_ui_image_internal_scale_set(Evas_Object *obj, Efl_Ui_Image_Data *sd, double scale)
{
sd->scale = scale;
_efl_ui_image_internal_sizing_eval(obj, sd);
}
void
_efl_ui_image_sizing_eval(Evas_Object *obj)
{
Evas_Coord minw = -1, minh = -1, maxw = -1, maxh = -1;
Eina_Size2D sz;
double ts;
EFL_UI_IMAGE_DATA_GET_OR_RETURN(obj, sd);
_efl_ui_image_internal_sizing_eval(obj, sd);
efl_image_smooth_scale_set(obj, sd->smooth);
if (sd->no_scale)
_efl_ui_image_internal_scale_set(obj, sd, 1.0);
else
_efl_ui_image_internal_scale_set(obj, sd, efl_gfx_scale_get(obj) * elm_config_scale_get());
ts = sd->scale;
sd->scale = 1.0;
sz = efl_gfx_view_size_get(obj);
sd->scale = ts;
evas_object_size_hint_combined_min_get(obj, &minw, &minh);
if (sd->no_scale)
{
maxw = minw = sz.w;
maxh = minh = sz.h;
if ((sd->scale > 1.0) && (sd->scale_up))
{
maxw = minw = sz.w * sd->scale;
maxh = minh = sz.h * sd->scale;
}
else if ((sd->scale < 1.0) && (sd->scale_down))
{
maxw = minw = sz.w * sd->scale;
maxh = minh = sz.h * sd->scale;
}
}
else
{
if (!sd->scale_down)
{
minw = sz.w * sd->scale;
minh = sz.h * sd->scale;
}
if (!sd->scale_up)
{
maxw = sz.w * sd->scale;
maxh = sz.h * sd->scale;
}
}
evas_object_size_hint_min_set(obj, minw, minh);
evas_object_size_hint_max_set(obj, maxw, maxh);
}
static void
_efl_ui_image_file_set_do(Evas_Object *obj)
{
Evas_Object *pclip = NULL;
EFL_UI_IMAGE_DATA_GET(obj, sd);
ELM_SAFE_FREE(sd->prev_img, evas_object_del);
if (sd->img)
{
pclip = evas_object_clip_get(sd->img);
sd->prev_img = sd->img;
}
sd->img = _img_new(obj);
evas_object_image_load_orientation_set(sd->img, EINA_TRUE);
evas_object_clip_set(sd->img, pclip);
sd->edje = EINA_FALSE;
if (sd->load_size > 0)
evas_object_image_load_size_set(sd->img, sd->load_size, sd->load_size);
else
{
Eina_Size2D sz = efl_gfx_view_size_get(obj);
evas_object_image_load_size_set(sd->img, sz.w, sz.h);
}
}
static void
_on_size_hints_changed(void *data, const Efl_Event *ev)
{
_efl_ui_image_internal_sizing_eval(ev->object, data);
}
EAPI Evas_Object *
elm_image_add(Evas_Object *parent)
{
EINA_SAFETY_ON_NULL_RETURN_VAL(parent, NULL);
Evas_Object *obj = elm_legacy_add(MY_CLASS, parent);
EFL_UI_IMAGE_DATA_GET(obj, priv);
efl_event_callback_add(obj, EFL_GFX_EVENT_CHANGE_SIZE_HINTS, _on_size_hints_changed, priv);
return obj;
}
EOLIAN static Eo *
_efl_ui_image_efl_object_constructor(Eo *obj, Efl_Ui_Image_Data *pd)
{
obj = efl_constructor(efl_super(obj, MY_CLASS));
efl_canvas_object_type_set(obj, MY_CLASS_NAME_LEGACY);
evas_object_smart_callbacks_descriptions_set(obj, _smart_callbacks);
efl_access_role_set(obj, EFL_ACCESS_ROLE_IMAGE);
pd->scale_type = EFL_UI_IMAGE_SCALE_TYPE_FIT_INSIDE;
pd->self = obj;
return obj;
}
EOLIAN Eina_Bool
_efl_ui_image_efl_file_mmap_set(Eo *obj, Efl_Ui_Image_Data *sd,
const Eina_File *file, const char *key)
{
Eina_Bool ret = EINA_FALSE;
_async_cancel(sd);
/* stop preloading as it may hit to-be-freed memory */
if (sd->preload_status == EFL_UI_IMAGE_PRELOADING)
evas_object_image_preload(sd->img, EINA_TRUE);
if (sd->remote.copier) _efl_ui_image_remote_copier_cancel(obj, sd);
if (sd->remote.binbuf) ELM_SAFE_FREE(sd->remote.binbuf, eina_binbuf_free);
if (!sd->async_enable)
ret = _efl_ui_image_smart_internal_file_set(obj, sd, eina_file_filename_get(file), file, key);
else
ret = _efl_ui_image_async_file_set(obj, sd, eina_file_filename_get(file), file, key);
return ret;
}
static Eina_Bool
_efl_ui_image_smart_internal_file_set(Eo *obj, Efl_Ui_Image_Data *sd,
const char *file, const Eina_File *f, const char *key)
{
Evas_Load_Error err;
if (eina_str_has_extension(file, ".edj"))
return _efl_ui_image_edje_file_set(obj, file, f, key);
_efl_ui_image_file_set_do(obj);
if (f)
evas_object_image_mmap_set(sd->img, f, key);
else
evas_object_image_file_set(sd->img, file, key);
err = evas_object_image_load_error_get(sd->img);
if (err != EVAS_LOAD_ERROR_NONE)
{
if (file || f)
{
if (key)
ERR("Failed to load image '%s' '%s': %s. (%p)",
file ? file : eina_file_filename_get(f), key,
evas_load_error_str(err), obj);
else
ERR("Failed to load image '%s': %s. (%p)",
file ? file : eina_file_filename_get(f),
evas_load_error_str(err), obj);
}
else
{
ERR("NULL image file passed! (%p)", obj);
}
_prev_img_del(sd);
return EINA_FALSE;
}
if (sd->preload_status == EFL_UI_IMAGE_PRELOAD_DISABLED)
_prev_img_del(sd);
else
{
evas_object_hide(sd->img);
sd->preload_status = EFL_UI_IMAGE_PRELOADING;
evas_object_image_preload(sd->img, EINA_FALSE);
}
_efl_ui_image_internal_sizing_eval(obj, sd);
return EINA_TRUE;
}
static void
_efl_ui_image_remote_copier_del(void *data EINA_UNUSED, const Efl_Event *event)
{
Eo *dialer = efl_io_copier_source_get(event->object);
efl_del(dialer);
}
static void
_efl_ui_image_remote_copier_cancel(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
Eo *copier = sd->remote.copier;
if (!copier) return;
/* copier is flagged as close_on_destructor, thus:
* efl_del()
* -> efl_io_closer_close()
* -> "done" event
* -> _efl_ui_image_remote_copier_done()
*
* flag sd->remote.copier = NULL so _efl_ui_image_remote_copier_done()
* knows about it.
*/
sd->remote.copier = NULL;
efl_del(copier);
}
static void
_efl_ui_image_remote_copier_done(void *data, const Efl_Event *event EINA_UNUSED)
{
Eo *obj = data;
Efl_Ui_Image_Data *sd = efl_data_scope_get(obj, MY_CLASS);
Eina_File *f;
Eo *dialer;
const char *url;
Eina_Bool ret = EINA_FALSE;
/* we're called from _efl_ui_image_remote_copier_cancel() */
if (!sd->remote.copier) return;
/* stop preloading as it may hit to-be-freed memory */
if (sd->preload_status == EFL_UI_IMAGE_PRELOADING)
evas_object_image_preload(sd->img, EINA_TRUE);
if (sd->remote.binbuf) eina_binbuf_free(sd->remote.binbuf);
sd->remote.binbuf = efl_io_copier_binbuf_steal(sd->remote.copier);
dialer = efl_io_copier_source_get(sd->remote.copier);
url = efl_net_dialer_address_dial_get(dialer);
f = eina_file_virtualize(url,
eina_binbuf_string_get(sd->remote.binbuf),
eina_binbuf_length_get(sd->remote.binbuf),
EINA_FALSE);
ret = _efl_ui_image_smart_internal_file_set(obj, sd, url, f, sd->remote.key);
eina_file_close(f);
if (!ret)
{
Efl_Ui_Image_Error err = { 0, EINA_TRUE };
ELM_SAFE_FREE(sd->remote.binbuf, eina_binbuf_free);
evas_object_smart_callback_call(obj, SIG_DOWNLOAD_ERROR, &err);
}
else
{
if (sd->preload_status != EFL_UI_IMAGE_PRELOAD_DISABLED)
{
sd->preload_status = EFL_UI_IMAGE_PRELOADING;
evas_object_image_preload(sd->img, EINA_FALSE);
}
evas_object_smart_callback_call(obj, SIG_DOWNLOAD_DONE, NULL);
}
ELM_SAFE_FREE(sd->remote.key, eina_stringshare_del);
ELM_SAFE_FREE(sd->remote.copier, efl_del);
}
static void
_efl_ui_image_remote_copier_error(void *data, const Efl_Event *event)
{
Eo *obj = data;
Efl_Ui_Image_Data *sd = efl_data_scope_get(obj, MY_CLASS);
Eina_Error *perr = event->info;
Efl_Ui_Image_Error err = { *perr, EINA_FALSE };
evas_object_smart_callback_call(obj, SIG_DOWNLOAD_ERROR, &err);
_efl_ui_image_remote_copier_cancel(obj, sd);
ELM_SAFE_FREE(sd->remote.key, eina_stringshare_del);
}
static void
_efl_ui_image_remote_copier_progress(void *data, const Efl_Event *event)
{
Eo *obj = data;
Efl_Ui_Image_Progress progress;
uint64_t now, total;
efl_io_copier_progress_get(event->object, &now, NULL, &total);
progress.now = now;
progress.total = total;
evas_object_smart_callback_call(obj, SIG_DOWNLOAD_PROGRESS, &progress);
}
EFL_CALLBACKS_ARRAY_DEFINE(_efl_ui_image_remote_copier_cbs,
{ EFL_EVENT_DEL, _efl_ui_image_remote_copier_del },
{ EFL_IO_COPIER_EVENT_DONE, _efl_ui_image_remote_copier_done },
{ EFL_IO_COPIER_EVENT_ERROR, _efl_ui_image_remote_copier_error },
{ EFL_IO_COPIER_EVENT_PROGRESS, _efl_ui_image_remote_copier_progress });
static Eina_Bool
_efl_ui_image_download(Eo *obj, Efl_Ui_Image_Data *sd, const char *url, const char *key)
{
Eo *dialer;
Efl_Ui_Image_Error img_err = { ENOSYS, EINA_FALSE };
Eina_Error err;
dialer = efl_add(EFL_NET_DIALER_HTTP_CLASS, obj,
efl_net_dialer_http_allow_redirects_set(efl_added, EINA_TRUE));
EINA_SAFETY_ON_NULL_GOTO(dialer, error_dialer);
sd->remote.copier = efl_add(EFL_IO_COPIER_CLASS, obj,
efl_io_copier_source_set(efl_added, dialer),
efl_io_closer_close_on_destructor_set(efl_added, EINA_TRUE),
efl_event_callback_array_add(efl_added, _efl_ui_image_remote_copier_cbs(), obj));
EINA_SAFETY_ON_NULL_GOTO(sd->remote.copier, error_copier);
eina_stringshare_replace(&sd->remote.key, key);
err = efl_net_dialer_dial(dialer, url);
if (err)
{
img_err.status = err;
ERR("Could not download %s: %s", url, eina_error_msg_get(err));
evas_object_smart_callback_call(obj, SIG_DOWNLOAD_ERROR, &img_err);
goto error_dial;
}
return EINA_TRUE;
error_dial:
evas_object_smart_callback_call(obj, SIG_DOWNLOAD_ERROR, &img_err);
_efl_ui_image_remote_copier_cancel(obj, sd);
return EINA_FALSE;
error_copier:
efl_del(dialer);
error_dialer:
evas_object_smart_callback_call(obj, SIG_DOWNLOAD_ERROR, &img_err);
return EINA_FALSE;
}
static const Eina_Slice remote_uri[] = {
EINA_SLICE_STR_LITERAL("http://"),
EINA_SLICE_STR_LITERAL("https://"),
EINA_SLICE_STR_LITERAL("ftp://"),
{ }
};
static inline Eina_Bool
_efl_ui_image_is_remote(const char *file)
{
Eina_Slice s = EINA_SLICE_STR(file);
const Eina_Slice *itr;
for (itr = remote_uri; itr->mem; itr++)
if (eina_slice_startswith(s, *itr))
return EINA_TRUE;
return EINA_FALSE;
}
EOLIAN static Eina_Bool
_efl_ui_image_efl_file_file_set(Eo *obj, Efl_Ui_Image_Data *sd, const char *file, const char *key)
{
Eina_Bool ret = EINA_FALSE;
_async_cancel(sd);
/* stop preloading as it may hit to-be-freed memory */
if (sd->preload_status == EFL_UI_IMAGE_PRELOADING)
evas_object_image_preload(sd->img, EINA_TRUE);
if (sd->remote.copier) _efl_ui_image_remote_copier_cancel(obj, sd);
if (sd->remote.binbuf) ELM_SAFE_FREE(sd->remote.binbuf, eina_binbuf_free);
if (sd->anim)
{
ELM_SAFE_FREE(sd->anim_timer, ecore_timer_del);
sd->play = EINA_FALSE;
sd->anim = EINA_FALSE;
}
if (!file && !sd->prev_img) return EINA_FALSE;
else if (!file && sd->prev_img)
{
_prev_img_del(sd);
return EINA_TRUE;
}
if (_efl_ui_image_is_remote(file))
{
evas_object_hide(sd->img);
if (_efl_ui_image_download(obj, sd, file, key))
{
evas_object_smart_callback_call(obj, SIG_DOWNLOAD_START, NULL);
return EINA_TRUE;
}
}
if (!sd->async_enable)
ret = _efl_ui_image_smart_internal_file_set(obj, sd, file, NULL, key);
else
ret = _efl_ui_image_async_file_set(obj, sd, file, NULL, key);
return ret;
}
EOLIAN static void
_efl_ui_image_efl_layout_signal_signal_emit(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd, const char *emission, const char *source)
{
if (sd->edje)
edje_object_signal_emit(sd->img, emission, source);
}
EOLIAN static Eina_Size2D
_efl_ui_image_efl_layout_group_group_size_min_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
if (sd->edje)
return efl_layout_group_size_min_get(sd->img);
else
return EINA_SIZE2D(0, 0);
}
EOLIAN static Eina_Size2D
_efl_ui_image_efl_layout_group_group_size_max_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
if (sd->edje)
return efl_layout_group_size_max_get(sd->img);
else
return EINA_SIZE2D(0, 0);
}
EOLIAN static void
_efl_ui_image_efl_layout_calc_calc_force(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
if (sd->edje)
edje_object_calc_force(sd->img);
}
EOLIAN static Eina_Size2D
_efl_ui_image_efl_layout_calc_calc_size_min(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd, Eina_Size2D restricted)
{
if (sd->edje)
return efl_layout_calc_size_min(sd->img, restricted);
else
{
// Ignore restricted here? Combine with min? Hmm...
return efl_gfx_size_hint_combined_min_get(sd->img);
}
}
EOLIAN static void
_efl_ui_image_efl_file_file_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd, const char **file, const char **key)
{
if (sd->async.th)
{
if (file) *file = sd->async.file;
if (key) *key = sd->async.key;
return;
}
efl_file_get(sd->img, file, key);
}
#if 0
// Kept for reference: wait for async open to complete - probably unused.
static Eina_Bool
_efl_ui_image_efl_file_async_wait(const Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *pd)
{
Eina_Bool ok = EINA_TRUE;
if (!pd->async.th) return ok;
if (!ecore_thread_wait(pd->async.th, 1.0))
{
ERR("Failed to wait on async file open!");
ok = EINA_FALSE;
}
return ok;
}
#endif
/* Legacy style async API. While legacy only, this is new from 1.19.
* Tizen has used elm_image_async_open_set() internally for a while, despite
* EFL upstream not exposing a proper async API. */
EAPI void
elm_image_async_open_set(Eo *obj, Eina_Bool async)
{
Efl_Ui_Image_Data *pd;
EINA_SAFETY_ON_FALSE_RETURN(efl_isa(obj, MY_CLASS));
pd = efl_data_scope_get(obj, MY_CLASS);
if (pd->async_enable == async) return;
pd->async_enable = async;
if (!async) _async_cancel(pd);
}
EOLIAN static Eina_Size2D
_efl_ui_image_efl_gfx_view_view_size_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
int tw, th;
if (efl_isa(sd->img, EFL_CANVAS_LAYOUT_CLASS))
edje_object_size_min_get(sd->img, &tw, &th);
else
evas_object_image_size_get(sd->img, &tw, &th);
return EINA_SIZE2D(tw, th);
}
EOLIAN static Eina_Size2D
_efl_ui_image_efl_image_image_size_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
if (sd->edje)
return EINA_SIZE2D(0, 0);
return efl_image_size_get(sd->img);
}
EAPI void
elm_image_prescale_set(Evas_Object *obj,
int size)
{
EFL_UI_IMAGE_CHECK(obj);
efl_image_load_size_set(obj, EINA_SIZE2D(size, size));
}
EOLIAN static void
_efl_ui_image_efl_image_load_load_size_set(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd, Eina_Size2D sz)
{
// FIXME: This should handle both dimensions!
if (sz.w > sz.h)
sd->load_size = sz.w;
else
sd->load_size = sz.h;
}
EAPI int
elm_image_prescale_get(const Evas_Object *obj)
{
Eina_Size2D sz;
EFL_UI_IMAGE_CHECK(obj) 0;
sz = efl_image_load_size_get(obj);
return sz.w;
}
EOLIAN static Eina_Size2D
_efl_ui_image_efl_image_load_load_size_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
return EINA_SIZE2D(sd->load_size, sd->load_size);
}
EOLIAN static void
_efl_ui_image_efl_orientation_orientation_set(Eo *obj, Efl_Ui_Image_Data *sd, Efl_Orient orient)
{
if (sd->edje) return;
if (sd->orient == orient) return;
efl_orientation_set(sd->img, orient);
sd->orient = orient;
_efl_ui_image_internal_sizing_eval(obj, sd);
}
EOLIAN static Efl_Orient
_efl_ui_image_efl_orientation_orientation_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
return sd->orient;
}
EOLIAN static void
_efl_ui_image_efl_flipable_flip_set(Eo *obj, Efl_Ui_Image_Data *sd, Efl_Flip flip)
{
if (sd->edje) return;
if (sd->flip == flip) return;
efl_flip_set(sd->img, flip);
sd->flip = flip;
_efl_ui_image_internal_sizing_eval(obj, sd);
}
EOLIAN static Efl_Flip
_efl_ui_image_efl_flipable_flip_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
return sd->flip;
}
/**
* Turns on editing through drag and drop and copy and paste.
*/
EOLIAN static void
_efl_ui_image_efl_ui_draggable_drag_target_set(Eo *obj, Efl_Ui_Image_Data *sd, Eina_Bool edit)
{
if (sd->edje)
{
WRN("No editing edje objects yet (ever)\n");
return;
}
edit = !!edit;
if (edit == sd->edit) return;
sd->edit = edit;
if (sd->edit)
elm_drop_target_add
(obj, ELM_SEL_FORMAT_IMAGE,
NULL, NULL,
NULL, NULL,
NULL, NULL,
_efl_ui_image_drag_n_drop_cb, obj);
else
elm_drop_target_del
(obj, ELM_SEL_FORMAT_IMAGE,
NULL, NULL,
NULL, NULL,
NULL, NULL,
_efl_ui_image_drag_n_drop_cb, obj);
}
EOLIAN static Eina_Bool
_efl_ui_image_efl_ui_draggable_drag_target_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
return sd->edit;
}
EAPI Eina_Bool
elm_image_animated_available_get(const Evas_Object *obj)
{
return efl_player_playable_get(obj);
}
EOLIAN static Eina_Bool
_efl_ui_image_efl_player_playable_get(Eo *obj, Efl_Ui_Image_Data *sd)
{
if (sd->edje) return EINA_TRUE;
return evas_object_image_animated_get(elm_image_object_get(obj));
}
static void
_efl_ui_image_animated_set_internal(Eo *obj, Efl_Ui_Image_Data *sd, Eina_Bool anim)
{
anim = !!anim;
if (sd->anim == anim) return;
sd->anim = anim;
if (sd->edje)
{
edje_object_animation_set(sd->img, anim);
return;
}
sd->img = elm_image_object_get(obj);
if (!evas_object_image_animated_get(sd->img)) return;
if (anim)
{
sd->frame_count = evas_object_image_animated_frame_count_get(sd->img);
sd->cur_frame = 1;
sd->frame_duration =
evas_object_image_animated_frame_duration_get
(sd->img, sd->cur_frame, 0);
evas_object_image_animated_frame_set(sd->img, sd->cur_frame);
}
else
{
sd->frame_count = -1;
sd->cur_frame = -1;
sd->frame_duration = -1;
}
}
static Eina_Bool
_efl_ui_image_animated_get_internal(const Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
if (sd->edje)
return edje_object_animation_get(sd->img);
return sd->anim;
}
EAPI void
elm_image_animated_set(Evas_Object *obj, Eina_Bool anim)
{
Efl_Ui_Image_Data *sd = efl_data_scope_get(obj, MY_CLASS);
if (!sd) return;
_efl_ui_image_animated_set_internal(obj, sd, anim);
}
EAPI Eina_Bool
elm_image_animated_get(const Evas_Object *obj)
{
Efl_Ui_Image_Data *sd = efl_data_scope_get(obj, MY_CLASS);
if (!sd) return EINA_FALSE;
return _efl_ui_image_animated_get_internal(obj, sd);
}
static void
_efl_ui_image_animated_play_set_internal(Eo *obj, Efl_Ui_Image_Data *sd, Eina_Bool play)
{
if (!sd->anim) return;
if (sd->play == play) return;
sd->play = play;
if (sd->edje)
{
edje_object_play_set(sd->img, play);
return;
}
if (play)
{
sd->anim_timer = ecore_timer_add
(sd->frame_duration, _efl_ui_image_animate_cb, obj);
}
else
{
ELM_SAFE_FREE(sd->anim_timer, ecore_timer_del);
}
}
static Eina_Bool
_efl_ui_image_animated_play_get_internal(const Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
if (sd->edje)
return edje_object_play_get(sd->img);
return sd->play;
}
EAPI void
elm_image_animated_play_set(Elm_Image *obj, Eina_Bool play)
{
Efl_Ui_Image_Data *sd = efl_data_scope_get(obj, MY_CLASS);
if (!sd) return;
_efl_ui_image_animated_play_set_internal(obj, sd, play);
}
EAPI Eina_Bool
elm_image_animated_play_get(const Elm_Image *obj)
{
Efl_Ui_Image_Data *sd = efl_data_scope_get(obj, MY_CLASS);
if (!sd) return EINA_FALSE;
return _efl_ui_image_animated_play_get_internal(obj, sd);
}
EOLIAN static void
_efl_ui_image_efl_player_play_set(Eo *obj, Efl_Ui_Image_Data *sd, Eina_Bool play)
{
if (play && !_efl_ui_image_animated_get_internal(obj, sd))
_efl_ui_image_animated_set_internal(obj, sd, play);
_efl_ui_image_animated_play_set_internal(obj, sd, play);
}
EOLIAN static Eina_Bool
_efl_ui_image_efl_player_play_get(Eo *obj, Efl_Ui_Image_Data *sd)
{
return _efl_ui_image_animated_play_get_internal(obj, sd);
}
static void
_efl_ui_image_class_constructor(Efl_Class *klass)
{
evas_smart_legacy_type_register(MY_CLASS_NAME_LEGACY, klass);
}
EOLIAN static void
_efl_ui_image_scale_type_set(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd EINA_UNUSED, Efl_Ui_Image_Scale_Type scale_type)
{
if (scale_type == sd->scale_type) return;
sd->scale_type = scale_type;
_efl_ui_image_internal_sizing_eval(obj, sd);
}
EOLIAN static Efl_Ui_Image_Scale_Type
_efl_ui_image_scale_type_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
return sd->scale_type;
}
EOLIAN static void
_efl_ui_image_scalable_set(Eo *obj, Efl_Ui_Image_Data *sd, Eina_Bool up, Eina_Bool down)
{
if ((up == sd->scale_up) && (down == sd->scale_down)) return;
sd->scale_up = !!up;
sd->scale_down = !!down;
_efl_ui_image_internal_sizing_eval(obj, sd);
}
EOLIAN static void
_efl_ui_image_scalable_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd, Eina_Bool *scale_up, Eina_Bool *scale_down)
{
if (scale_up) *scale_up = sd->scale_up;
if (scale_down) *scale_down = sd->scale_down;
}
EOLIAN static void
_efl_ui_image_align_set(Eo *obj, Efl_Ui_Image_Data *sd, double align_x, double align_y)
{
if (align_x > 1.0)
align_x = 1.0;
else if (align_x < 0.0)
align_x = 0.0;
if (align_y > 1.0)
align_y = 1.0;
else if (align_y < 0.0)
align_y = 0.0;
if ((EINA_DBL_EQ(align_x, sd->align_x)) && (EINA_DBL_EQ(align_y, sd->align_y))) return;
sd->align_x = align_x;
sd->align_y = align_y;
_efl_ui_image_internal_sizing_eval(obj, sd);
}
EOLIAN static void
_efl_ui_image_align_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd, double *align_x, double *align_y)
{
if (align_x) *align_x = sd->align_x;
if (align_y) *align_y = sd->align_y;
}
// A11Y
EOLIAN static void
_efl_ui_image_efl_access_image_extents_get(Eo *obj, Efl_Ui_Image_Data *sd EINA_UNUSED, Eina_Bool screen_coords, int *x, int *y, int *w, int *h)
{
int ee_x, ee_y;
Evas_Object *image = elm_image_object_get(obj);
if (!image) return;
evas_object_geometry_get(image, x, y, NULL, NULL);
if (screen_coords)
{
Ecore_Evas *ee = ecore_evas_ecore_evas_get(evas_object_evas_get(image));
if (!ee) return;
ecore_evas_geometry_get(ee, &ee_x, &ee_y, NULL, NULL);
if (x) *x += ee_x;
if (y) *y += ee_y;
}
elm_image_object_size_get(obj, w, h);
}
EOLIAN const Efl_Access_Action_Data *
_efl_ui_image_efl_access_widget_action_elm_actions_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *pd EINA_UNUSED)
{
static Efl_Access_Action_Data atspi_actions[] = {
{ "activate", "activate", NULL, _key_action_activate },
{ NULL, NULL, NULL, NULL },
};
return &atspi_actions[0];
}
static Eina_Bool
_icon_standard_set(Evas_Object *obj, const char *name)
{
EFL_UI_IMAGE_DATA_GET(obj, sd);
if (_elm_theme_object_icon_set(obj, name, "default"))
{
/* TODO: elm_unneed_efreet() */
sd->freedesktop.use = EINA_FALSE;
return EINA_TRUE;
}
return EINA_FALSE;
}
static Eina_Bool
_icon_freedesktop_set(Evas_Object *obj, const char *name, int size)
{
const char *path;
EFL_UI_IMAGE_DATA_GET(obj, sd);
elm_need_efreet();
if (icon_theme == NON_EXISTING) return EINA_FALSE;
if (!icon_theme)
{
Efreet_Icon_Theme *theme;
/* TODO: Listen for EFREET_EVENT_ICON_CACHE_UPDATE */
theme = efreet_icon_theme_find(elm_config_icon_theme_get());
if (!theme)
{
const char **itr;
static const char *themes[] = {
"gnome", "Human", "oxygen", "hicolor", NULL
};
for (itr = themes; *itr; itr++)
{
theme = efreet_icon_theme_find(*itr);
if (theme) break;
}
}
if (!theme)
{
icon_theme = NON_EXISTING;
return EINA_FALSE;
}
else
icon_theme = eina_stringshare_add(theme->name.internal);
}
path = efreet_icon_path_find(icon_theme, name, size);
sd->freedesktop.use = !!path;
if (sd->freedesktop.use)
{
sd->freedesktop.requested_size = size;
efl_file_set(obj, path, NULL);
return EINA_TRUE;
}
return EINA_FALSE;
}
static inline int
_icon_size_min_get(Evas_Object *image)
{
int w, h;
evas_object_geometry_get(image, NULL, NULL, &w, &h);
return MAX(16, MIN(w, h));
}
/* FIXME: move this code to ecore */
#ifdef _WIN32
static Eina_Bool
_path_is_absolute(const char *path)
{
//TODO: Check if this works with all absolute paths in windows
return (isalpha(*path)) && (*(path + 1) == ':') &&
((*(path + 2) == '\\') || (*(path + 2) == '/'));
}
#else
static Eina_Bool
_path_is_absolute(const char *path)
{
return *path == '/';
}
#endif
static Eina_Bool
_internal_efl_ui_image_icon_set(Evas_Object *obj, const char *name, Eina_Bool *fdo)
{
char *tmp;
Eina_Bool ret = EINA_FALSE;
EFL_UI_IMAGE_DATA_GET(obj, sd);
/* try locating the icon using the specified theme */
if (!strcmp(ELM_CONFIG_ICON_THEME_ELEMENTARY, elm_config_icon_theme_get()))
{
ret = _icon_standard_set(obj, name);
if (ret && fdo) *fdo = EINA_FALSE;
}
else
{
ret = _icon_freedesktop_set(obj, name, _icon_size_min_get(obj));
if (ret && fdo) *fdo = EINA_TRUE;
}
if (ret)
{
eina_stringshare_replace(&sd->stdicon, name);
_efl_ui_image_sizing_eval(obj);
return EINA_TRUE;
}
else
eina_stringshare_replace(&sd->stdicon, NULL);
if (_path_is_absolute(name))
{
if (fdo)
*fdo = EINA_FALSE;
return efl_file_set(obj, name, NULL);
}
/* if that fails, see if icon name is in the format size/name. if so,
try locating a fallback without the size specification */
if (!(tmp = strchr(name, '/'))) return EINA_FALSE;
++tmp;
if (*tmp) return _internal_efl_ui_image_icon_set(obj, tmp, fdo);
/* give up */
return EINA_FALSE;
}
static void
_efl_ui_image_icon_resize_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED)
{
EFL_UI_IMAGE_DATA_GET(data, sd);
const char *refup = eina_stringshare_ref(sd->stdicon);
Eina_Bool fdo = EINA_FALSE;
if (!_internal_efl_ui_image_icon_set(obj, sd->stdicon, &fdo) || (!fdo))
evas_object_event_callback_del_full
(obj, EVAS_CALLBACK_RESIZE, _efl_ui_image_icon_resize_cb, data);
eina_stringshare_del(refup);
}
EOLIAN static Eina_Bool
_efl_ui_image_icon_set(Eo *obj, Efl_Ui_Image_Data *_pd EINA_UNUSED, const char *name)
{
Eina_Bool fdo = EINA_FALSE;
if (!name) return EINA_FALSE;
evas_object_event_callback_del_full
(obj, EVAS_CALLBACK_RESIZE, _efl_ui_image_icon_resize_cb, obj);
Eina_Bool int_ret = _internal_efl_ui_image_icon_set(obj, name, &fdo);
if (fdo)
evas_object_event_callback_add
(obj, EVAS_CALLBACK_RESIZE, _efl_ui_image_icon_resize_cb, obj);
return int_ret;
}
EOLIAN static const char*
_efl_ui_image_icon_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *sd)
{
return sd->stdicon;
}
static void
_prop_future_error_cb(void* data, Efl_Event const* event EINA_UNUSED)
{
Eo *obj = data;
EFL_UI_IMAGE_DATA_GET(obj, pd);
pd->pfuture = NULL;
}
static void
_prop_key_future_then_cb(void* data, Efl_Event const * event)
{
Eo *obj = data;
Eina_Accessor *acc = (Eina_Accessor *)((Efl_Future_Event_Success*)event->info)->value;
Eina_Value *value;
char *filename, *key;
EFL_UI_IMAGE_DATA_GET(obj, pd);
pd->pfuture = NULL;
if (eina_accessor_data_get(acc, 0, (void **)&value) && value)
{
filename = eina_value_to_string(value);
}
else return;
if (eina_accessor_data_get(acc, 1, (void **)&value) && value)
{
key = eina_value_to_string(value);
}
else
{
free(filename);
return;
}
elm_image_file_set(obj, filename, key);
free(filename);
free(key);
}
static void
_prop_future_then_cb(void* data, Efl_Event const * event)
{
Eo *obj = data;
Eina_Value *value = (Eina_Value*)((Efl_Future_Event_Success*)event->info)->value;
char *text;
EFL_UI_IMAGE_DATA_GET(obj, pd);
pd->pfuture = NULL;
const Eina_Value_Type *vtype = eina_value_type_get(value);
if (vtype == EINA_VALUE_TYPE_STRING || vtype == EINA_VALUE_TYPE_STRINGSHARE)
{
eina_value_get(value, &text);
if (pd->con_icon) efl_ui_image_icon_set(obj, text);
else elm_image_file_set(obj, text, NULL);
}
else
{
text = eina_value_to_string(value);
if (pd->con_icon) efl_ui_image_icon_set(obj, text);
else elm_image_file_set(obj, text, NULL);
free(text);
}
}
void
_update_viewmodel(Eo *obj, Efl_Ui_Image_Data *pd)
{
if (pd->model && pd->prop_con)
{
if (pd->pfuture) efl_future_cancel(pd->pfuture);
pd->pfuture = efl_model_property_get(pd->model, pd->prop_con);
if (pd->prop_key)
{
const Eina_Array *properties;
Eina_Array_Iterator it;
char *property;
unsigned int i = 0;
properties = efl_model_properties_get(pd->model);
EINA_ARRAY_ITER_NEXT(properties, i, property, it)
{
if (strcmp(property, pd->prop_key) == 0)
{
Efl_Future *futures[2] = {NULL,};
futures[0] = pd->pfuture;
futures[1] = efl_model_property_get(pd->model, pd->prop_key);
pd->pfuture = efl_future_all(futures[0], futures[1]);
efl_future_then(pd->pfuture, &_prop_key_future_then_cb,
&_prop_future_error_cb, NULL, obj);
return;
}
}
}
efl_future_then(pd->pfuture, &_prop_future_then_cb,
&_prop_future_error_cb, NULL, obj);
}
}
static void
_efl_ui_image_model_properties_changed_cb(void *data, const Efl_Event *event)
{
Efl_Model_Property_Event *evt = event->info;
Eo *obj = data;
EFL_UI_IMAGE_DATA_GET(obj, pd);
if (!evt->changed_properties)
return;
if (pd->model && pd->prop_con)
{
Eina_Array_Iterator it;
const char *prop;
unsigned int i;
EINA_ARRAY_ITER_NEXT(evt->changed_properties, i, prop, it)
{
if (!strcmp(pd->prop_con, prop) || (pd->prop_key && !strcmp(pd->prop_key, prop)))
{
_update_viewmodel(obj, pd);
return;
}
}
}
}
EOLIAN static void
_efl_ui_image_efl_ui_view_model_set(Eo *obj, Efl_Ui_Image_Data *pd, Efl_Model *model)
{
if (pd->model)
{
efl_event_callback_del(pd->model, EFL_MODEL_EVENT_PROPERTIES_CHANGED,
_efl_ui_image_model_properties_changed_cb, obj);
efl_unref(pd->model);
pd->model = NULL;
}
if (model)
{
pd->model = model;
efl_ref(pd->model);
efl_event_callback_add(pd->model, EFL_MODEL_EVENT_PROPERTIES_CHANGED,
_efl_ui_image_model_properties_changed_cb, obj);
}
_update_viewmodel(obj, pd);
}
EOLIAN static Efl_Model *
_efl_ui_image_efl_ui_view_model_get(Eo *obj EINA_UNUSED, Efl_Ui_Image_Data *pd)
{
return pd->model;
}
EOLIAN static void
_efl_ui_image_efl_ui_model_connect_connect(Eo *obj, Efl_Ui_Image_Data *pd, const char *name, const char *property)
{
if (strcmp(name, "filename") == 0)
{
pd->con_icon = EINA_FALSE;
eina_stringshare_replace(&pd->prop_con, property);
}
else if (strcmp(name, "icon") == 0)
{
pd->con_icon = EINA_TRUE;
eina_stringshare_replace(&pd->prop_con, property);
eina_stringshare_replace(&pd->prop_key, NULL);
}
else if (strcmp(name, "key") == 0)
{
eina_stringshare_replace(&pd->prop_key, property);
}
else return;
_update_viewmodel(obj, pd);
}
EAPI void
elm_image_smooth_set(Evas_Object *obj, Eina_Bool smooth)
{
EINA_SAFETY_ON_FALSE_RETURN(efl_isa(obj, MY_CLASS));
efl_image_smooth_scale_set(obj, smooth);
_efl_ui_image_sizing_eval(obj);
}
EAPI Eina_Bool
elm_image_smooth_get(const Evas_Object *obj)
{
EINA_SAFETY_ON_FALSE_RETURN_VAL(efl_isa(obj, MY_CLASS), EINA_FALSE);
return efl_image_smooth_scale_get(obj);
}
// A11Y - END
/* Legacy deprecated functions */
EAPI void
elm_image_editable_set(Evas_Object *obj, Eina_Bool edit)
{
efl_ui_draggable_drag_target_set(obj, edit);
}
EAPI Eina_Bool
elm_image_editable_get(const Evas_Object *obj)
{
return efl_ui_draggable_drag_target_get(obj);
}
EAPI Eina_Bool
elm_image_file_set(Evas_Object *obj, const char *file, const char *group)
{
Eina_Bool ret = EINA_FALSE;
EFL_UI_IMAGE_CHECK(obj) EINA_FALSE;
ret = efl_file_set(obj, file, group);
_efl_ui_image_sizing_eval(obj);
return ret;
}
EAPI void
elm_image_file_get(const Eo *obj, const char **file, const char **group)
{
efl_file_get((Eo *) obj, file, group);
}
EAPI Eina_Bool
elm_image_mmap_set(Evas_Object *obj, const Eina_File *file, const char *group)
{
EFL_UI_IMAGE_CHECK(obj) EINA_FALSE;
return efl_file_mmap_set(obj, file, group);
}
EAPI Eina_Bool
elm_image_memfile_set(Evas_Object *obj, const void *img, size_t size, const char *format, const char *key)
{
Evas_Load_Error err;
EFL_UI_IMAGE_CHECK(obj) EINA_FALSE;
EFL_UI_IMAGE_DATA_GET(obj, sd);
_efl_ui_image_file_set_do(obj);
evas_object_image_memfile_set
(sd->img, (void *)img, size, (char *)format, (char *)key);
if (sd->preload_status == EFL_UI_IMAGE_PRELOAD_DISABLED)
_prev_img_del(sd);
else
{
sd->preload_status = EFL_UI_IMAGE_PRELOADING;
evas_object_image_preload(sd->img, EINA_FALSE);
}
err = evas_object_image_load_error_get(sd->img);
if (err != EVAS_LOAD_ERROR_NONE)
{
if (img)
ERR("Failed to load image from memory block (" FMT_SIZE_T
" bytes): %s (%p)", size, evas_load_error_str(err), sd->img);
else
ERR("NULL image data passed (%p)", sd->img);
_prev_img_del(sd);
return EINA_FALSE;
}
_efl_ui_image_internal_sizing_eval(obj, sd);
return EINA_TRUE;
}
EAPI void
elm_image_fill_outside_set(Evas_Object *obj, Eina_Bool fill_outside)
{
EFL_UI_IMAGE_CHECK(obj);
EFL_UI_IMAGE_DATA_GET(obj, sd);
fill_outside = !!fill_outside;
if (sd->fill_inside == !fill_outside) return;
sd->fill_inside = !fill_outside;
if (sd->aspect_fixed)
{
if (sd->fill_inside) sd->scale_type = EFL_UI_IMAGE_SCALE_TYPE_FIT_INSIDE;
else sd->scale_type = EFL_UI_IMAGE_SCALE_TYPE_FIT_OUTSIDE;
}
else
sd->scale_type = EFL_UI_IMAGE_SCALE_TYPE_FILL;
_efl_ui_image_internal_sizing_eval(obj, sd);
}
EAPI Eina_Bool
elm_image_fill_outside_get(const Evas_Object *obj)
{
EFL_UI_IMAGE_CHECK(obj) EINA_FALSE;
EFL_UI_IMAGE_DATA_GET(obj, sd);
return !sd->fill_inside;
}
// TODO: merge preload and async code
EAPI void
elm_image_preload_disabled_set(Evas_Object *obj, Eina_Bool disable)
{
EFL_UI_IMAGE_CHECK(obj);
EFL_UI_IMAGE_DATA_GET(obj, sd);
if (sd->edje || !sd->img) return;
if (disable)
{
if (sd->preload_status == EFL_UI_IMAGE_PRELOADING)
{
evas_object_image_preload(sd->img, disable);
if (sd->show) evas_object_show(sd->img);
_prev_img_del(sd);
}
sd->preload_status = EFL_UI_IMAGE_PRELOAD_DISABLED;
}
else if (sd->preload_status == EFL_UI_IMAGE_PRELOAD_DISABLED)
{
sd->preload_status = EFL_UI_IMAGE_PRELOADING;
evas_object_image_preload(sd->img, disable);
}
}
EAPI void
elm_image_orient_set(Evas_Object *obj, Elm_Image_Orient orient)
{
Efl_Orient dir;
Efl_Flip flip;
EFL_UI_IMAGE_DATA_GET(obj, sd);
sd->image_orient = orient;
switch (orient)
{
case EVAS_IMAGE_ORIENT_0:
dir = EFL_ORIENT_0;
flip = EFL_FLIP_NONE;
break;
case EVAS_IMAGE_ORIENT_90:
dir = EFL_ORIENT_90;
flip = EFL_FLIP_NONE;
break;
case EVAS_IMAGE_ORIENT_180:
dir = EFL_ORIENT_180;
flip = EFL_FLIP_NONE;
break;
case EVAS_IMAGE_ORIENT_270:
dir = EFL_ORIENT_270;
flip = EFL_FLIP_NONE;
break;
case EVAS_IMAGE_FLIP_HORIZONTAL:
dir = EFL_ORIENT_0;
flip = EFL_FLIP_HORIZONTAL;
break;
case EVAS_IMAGE_FLIP_VERTICAL:
dir = EFL_ORIENT_0;
flip = EFL_FLIP_VERTICAL;
break;
case EVAS_IMAGE_FLIP_TRANSVERSE:
dir = EFL_ORIENT_270;
flip = EFL_FLIP_HORIZONTAL;
break;
case EVAS_IMAGE_FLIP_TRANSPOSE:
dir = EFL_ORIENT_270;
flip = EFL_FLIP_VERTICAL;
break;
default:
dir = EFL_ORIENT_0;
flip = EFL_FLIP_NONE;
break;
}
efl_orientation_set(obj, dir);
efl_flip_set(obj, flip);
}
EAPI Elm_Image_Orient
elm_image_orient_get(const Evas_Object *obj)
{
EFL_UI_IMAGE_CHECK(obj) ELM_IMAGE_ORIENT_NONE;
EFL_UI_IMAGE_DATA_GET(obj, sd);
return sd->image_orient;
}
EAPI Evas_Object*
elm_image_object_get(const Evas_Object *obj)
{
EFL_UI_IMAGE_CHECK(obj) NULL;
EFL_UI_IMAGE_DATA_GET(obj, sd);
return sd->img;
}
EAPI void
elm_image_object_size_get(const Evas_Object *obj, int *w, int *h)
{
Eina_Size2D sz;
sz = efl_gfx_view_size_get(obj);
if (w) *w = sz.w;
if (h) *h = sz.h;
}
EAPI void
elm_image_no_scale_set(Evas_Object *obj, Eina_Bool no_scale)
{
EFL_UI_IMAGE_CHECK(obj);
EFL_UI_IMAGE_DATA_GET(obj, sd);
sd->no_scale = no_scale;
_efl_ui_image_sizing_eval(obj);
}
EAPI Eina_Bool
elm_image_no_scale_get(const Evas_Object *obj)
{
EFL_UI_IMAGE_CHECK(obj) EINA_FALSE;
EFL_UI_IMAGE_DATA_GET(obj, sd);
return sd->no_scale;
}
EAPI void
elm_image_resizable_set(Evas_Object *obj, Eina_Bool up, Eina_Bool down)
{
EFL_UI_IMAGE_CHECK(obj);
EFL_UI_IMAGE_DATA_GET(obj, sd);
sd->scale_up = !!up;
sd->scale_down = !!down;
_efl_ui_image_sizing_eval(obj);
}
EAPI void
elm_image_resizable_get(const Evas_Object *obj, Eina_Bool *size_up, Eina_Bool *size_down)
{
EFL_UI_IMAGE_CHECK(obj);
EFL_UI_IMAGE_DATA_GET(obj, sd);
if (size_up) *size_up = sd->scale_up;
if (size_down) *size_down = sd->scale_down;
}
EAPI void
elm_image_aspect_fixed_set(Evas_Object *obj, Eina_Bool fixed)
{
EFL_UI_IMAGE_CHECK(obj);
EFL_UI_IMAGE_DATA_GET(obj, sd);
fixed = !!fixed;
if (sd->aspect_fixed == fixed) return;
sd->aspect_fixed = fixed;
if (sd->aspect_fixed)
{
if (sd->fill_inside) sd->scale_type = EFL_UI_IMAGE_SCALE_TYPE_FIT_INSIDE;
else sd->scale_type = EFL_UI_IMAGE_SCALE_TYPE_FIT_OUTSIDE;
}
else
sd->scale_type = EFL_UI_IMAGE_SCALE_TYPE_FILL;
_efl_ui_image_sizing_eval(obj);
}
EAPI Eina_Bool
elm_image_aspect_fixed_get(const Evas_Object *obj)
{
EFL_UI_IMAGE_CHECK(obj) EINA_FALSE;
EFL_UI_IMAGE_DATA_GET(obj, sd);
return sd->aspect_fixed;
}
/* Standard widget overrides */
ELM_WIDGET_KEY_DOWN_DEFAULT_IMPLEMENT(efl_ui_image, Efl_Ui_Image_Data)
/* Internal EO APIs and hidden overrides */
#define EFL_UI_IMAGE_EXTRA_OPS \
EFL_CANVAS_GROUP_ADD_DEL_OPS(efl_ui_image)
#include "efl_ui_image.eo.c"