efl/src/lib/evas/canvas/evas_render.c

4392 lines
155 KiB
C

#include "evas_common_private.h"
#include "evas_private.h"
#include <math.h>
#include <assert.h>
#ifdef EVAS_RENDER_DEBUG_TIMING
#include <sys/time.h>
#endif
// FIXME: Ugly!
#define EFL_CANVAS_FILTER_INTERNAL_PROTECTED
#include "efl_canvas_filter_internal.eo.h"
/* Enable this for extra verbose rendering debug logs */
//#define REND_DBG 1
#define STDOUT_DBG
#ifdef REND_DBG
static FILE *dbf = NULL;
static int __RD_level = 0;
static int __RD_enable = REND_DBG;
static inline Eina_Slstr *
_efl_object_name(const Evas_Object_Protected_Data *obj)
{
const Eo *eo_obj;
const char *name;
const char *klass;
if (!obj) return "(nil)";
eo_obj = obj->object;
name = obj->name ?: efl_name_get(eo_obj);
klass = obj->type; //efl_class_name_get(eo_obj);
if (name)
return eina_slstr_printf("[%p@%p: %s '%s']", eo_obj, obj, klass, name);
else
return eina_slstr_printf("[%p@%p: %s]", eo_obj, obj, klass);
}
#define RDNAME(_obj) _efl_object_name(_obj)
static void
rend_dbg(const char *txt)
{
if (!dbf)
{
#ifdef STDOUT_DBG
dbf = stdout;
#else
dbf = fopen("EVAS-RENDER-DEBUG.log", "wb");
#endif
if (!dbf) return;
}
fputs(txt, dbf);
fflush(dbf);
}
#define RD(xxxx, args...) \
do { if (__RD_enable) { \
char __tmpbuf[4096]; int __tmpi; \
__RD_level = xxxx; \
if (xxxx) { \
for (__tmpi = 0; __tmpi < xxxx * 2; __tmpi++) \
__tmpbuf[__tmpi] = ' '; \
__tmpbuf[__tmpi] = 0; \
rend_dbg(__tmpbuf); \
} \
snprintf(__tmpbuf, sizeof(__tmpbuf), ##args); \
rend_dbg(__tmpbuf); \
} } while (0)
#define IFRD(ifcase, xxxx, args...) \
if (__RD_enable && (ifcase)) { \
char __tmpbuf[4096]; int __tmpi; \
__RD_level = xxxx; \
if (xxxx) { \
for (__tmpi = 0; __tmpi < xxxx * 2; __tmpi++) \
__tmpbuf[__tmpi] = ' '; \
__tmpbuf[__tmpi] = 0; \
rend_dbg(__tmpbuf); \
} \
snprintf(__tmpbuf, sizeof(__tmpbuf), ##args); \
rend_dbg(__tmpbuf); \
}
#else
#define RD(xxx, args...)
#define IFRD(ifcase, xxx, args...)
#define RDNAME(xxx) ""
#endif
#define OBJ_ARRAY_PUSH(array, obj) eina_array_push(array, obj)
#define OBJS_ARRAY_CLEAN(array) eina_array_clean(array)
#define OBJS_ARRAY_FLUSH(array) eina_array_flush(array)
/* save typing */
#undef ENFN
#undef ENC
#define ENFN evas->engine.func
#define ENC _evas_engine_context(evas)
typedef struct _Render_Updates Render_Updates;
typedef struct _Cutout_Margin Cutout_Margin;
struct _Render_Updates
{
void *surface;
Eina_Rectangle *area;
};
struct _Cutout_Margin
{
int l, r, t, b;
};
static void
evas_render_pipe_wakeup(void *data);
static Eina_Bool
evas_render_updates_internal(Evas *eo_e, unsigned char make_updates, unsigned char do_draw, Eina_Bool do_async);
static void
evas_render_mask_subrender(Evas_Public_Data *evas,
void *output,
Evas_Object_Protected_Data *mask,
Evas_Object_Protected_Data *prev_mask,
int level, Eina_Bool do_async);
static Eina_List *_rendering_evases = NULL;
#ifdef EVAS_RENDER_DEBUG_TIMING
static double
_time_get()
{
struct timeval tv;
gettimeofday(&tv, NULL);
return (tv.tv_sec + tv.tv_usec / 1000000.0) * 1000.0;
}
struct accumulator {
double total, min, max, draw_start_time;
int samples;
const char *what;
};
static struct accumulator async_accumulator = {
.total = 0,
.min = 1000000,
.max = 0,
.samples = 0,
.what = "async render"
};
static struct accumulator sync_accumulator = {
.total = 0,
.min = 1000000,
.max = 0,
.samples = 0,
.what = "sync render"
};
static void
_accumulate_time(double before, Eina_Bool async)
{
static Eina_Bool async_start = EINA_TRUE;
static double cache_before;
struct accumulator *acc = &sync_accumulator;
if (async)
{
acc = &async_accumulator;
if (async_start)
{
async_start = EINA_FALSE;
cache_before = before;
return;
}
else
{
async_start = EINA_TRUE;
before = cache_before;
}
}
double diff = _time_get() - before;
acc->total += diff;
if (diff > acc->max) acc->max = diff;
if (diff < acc->min) acc->min = diff;
acc->samples++;
if (acc->samples % 100 == 0)
{
fprintf(stderr, "*** %s: avg %fms min %fms max %fms\n",
acc->what, acc->total / 100.0, acc->min, acc->max);
acc->total = 0.0;
acc->max = 0.0;
acc->min = 1000000;
}
}
#endif
static int _render_busy = 0;
static inline Eina_Bool
_is_obj_in_framespace(Evas_Object_Protected_Data *obj, Evas_Public_Data *evas EINA_UNUSED)
{
return obj->is_frame;
}
static inline void
_evas_render_framespace_context_clip_clip(Evas_Public_Data *evas, void *ctx,
int ox, int oy)
{
int fx, fy, fw, fh;
fx = evas->framespace.x;
fy = evas->framespace.y;
fw = evas->viewport.w - evas->framespace.w;
fh = evas->viewport.h - evas->framespace.h;
ENFN->context_clip_clip(ENC, ctx, fx + ox, fy + oy, fw, fh);
}
static void
_evas_render_cleanup(void)
{
if (_render_busy != 0) return;
evas_common_rgba_pending_unloads_cleanup();
}
static void
_evas_render_busy_begin(void)
{
_render_busy++;
}
static void
_evas_render_busy_end(void)
{
_render_busy--;
_evas_render_cleanup();
}
EOLIAN void
_evas_canvas_damage_rectangle_add(Eo *eo_e EINA_UNUSED, Evas_Public_Data *e, int x, int y, int w, int h)
{
Eina_Rectangle *r;
evas_canvas_async_block(e);
NEW_RECT(r, x, y, w, h);
if (!r) return;
e->damages = eina_list_append(e->damages, r);
e->changed = EINA_TRUE;
}
EOLIAN void
_evas_canvas_obscured_rectangle_add(Eo *eo_e EINA_UNUSED, Evas_Public_Data *e, int x, int y, int w, int h)
{
Eina_Rectangle *r;
evas_canvas_async_block(e);
NEW_RECT(r, x, y, w, h);
if (!r) return;
e->obscures = eina_list_append(e->obscures, r);
}
EOLIAN void
_evas_canvas_obscured_clear(Eo *eo_e EINA_UNUSED, Evas_Public_Data *e)
{
Eina_Rectangle *r;
evas_canvas_async_block(e);
EINA_LIST_FREE(e->obscures, r)
{
eina_rectangle_free(r);
}
}
static Eina_Bool
_evas_clip_changes_free(const void *container EINA_UNUSED, void *data, void *fdata EINA_UNUSED)
{
eina_rectangle_free(data);
return EINA_TRUE;
}
static inline Eina_Rectangle
_evas_render_smallest_static_clipped_geometry_get(const Evas_Object_Protected_State *state)
{
int cx, cy, cw, ch;
cx = state->geometry.x;
cy = state->geometry.y;
cw = state->geometry.w;
ch = state->geometry.h;
while (state->clipper && state->has_fixed_size)
{
/* walk up the clipper tree as long as the clippers are static */
RECTS_CLIP_TO_RECT(cx, cy, cw, ch,
state->clipper->cur->geometry.x,
state->clipper->cur->geometry.y,
state->clipper->cur->geometry.w,
state->clipper->cur->geometry.h);
if (!state->clipper) break;
state = state->clipper->cur;
}
return (Eina_Rectangle){cx, cy, cw, ch};
}
static Eina_Bool
_evas_render_had_map(Evas_Object_Protected_Data *obj)
{
return ((obj->map->prev.map) && (obj->map->prev.usemap));
}
static Eina_Bool
_evas_render_is_relevant(Evas_Object_Protected_Data *obj)
{
return ((evas_object_is_visible(obj) && (!obj->cur->have_clipees)) ||
(evas_object_was_visible(obj) && (!obj->prev->have_clipees)));
}
static Eina_Bool
_evas_render_can_render(Evas_Object_Protected_Data *obj)
{
return (evas_object_is_visible(obj) && (!obj->cur->have_clipees) &&
!obj->no_render);
}
static void
_evas_render_prev_cur_clip_cache_add(Evas_Public_Data *evas, Evas_Object_Protected_Data *obj)
{
// FIXME: Iterate over each output
ENFN->output_redraws_rect_add(ENC,
obj->prev->cache.clip.x + evas->framespace.x,
obj->prev->cache.clip.y + evas->framespace.y,
obj->prev->cache.clip.w,
obj->prev->cache.clip.h);
ENFN->output_redraws_rect_add(ENC,
obj->cur->cache.clip.x + evas->framespace.x,
obj->cur->cache.clip.y + evas->framespace.y,
obj->cur->cache.clip.w,
obj->cur->cache.clip.h);
}
static void
_evas_render_cur_clip_cache_del(Evas_Public_Data *e, Evas_Object_Protected_Data *obj)
{
Evas_Coord x, y, w, h;
x = obj->cur->cache.clip.x;
y = obj->cur->cache.clip.y;
w = obj->cur->cache.clip.w;
h = obj->cur->cache.clip.h;
if (obj->cur->clipper)
{
RECTS_CLIP_TO_RECT(x, y, w, h,
obj->cur->clipper->cur->cache.clip.x,
obj->cur->clipper->cur->cache.clip.y,
obj->cur->clipper->cur->cache.clip.w,
obj->cur->clipper->cur->cache.clip.h);
}
evas_render_update_del(e, x + e->framespace.x, y + e->framespace.y, w, h);
}
/* sets the redraw flag for all the proxies depending on this obj as a source */
static void
_evas_proxy_redraw_set(Evas_Public_Data *e, Evas_Object_Protected_Data *obj,
Eina_Bool render)
{
Evas_Object *eo_proxy;
Evas_Object_Protected_Data *proxy;
Eina_List *l;
EINA_LIST_FOREACH(obj->proxy->proxies, l, eo_proxy)
{
proxy = efl_data_scope_get(eo_proxy, EFL_CANVAS_OBJECT_CLASS);
/* Flag need redraw on proxy too */
EINA_COW_WRITE_BEGIN(evas_object_proxy_cow, proxy->proxy,
Evas_Object_Proxy_Data, proxy_write)
proxy_write->redraw = EINA_TRUE;
EINA_COW_WRITE_END(evas_object_proxy_cow, proxy->proxy, proxy_write);
if (render)
{
/* Not good... but make it sure if the proxies have missed at update
if its sources are remained changed as pending objects in the prev frame. */
evas_object_change(eo_proxy, proxy);
proxy->func->render_pre(eo_proxy, proxy, proxy->private_data);
_evas_render_prev_cur_clip_cache_add(e, proxy);
}
//Update the proxies recursively.
_evas_proxy_redraw_set(e, proxy, render);
}
}
/* sets the mask redraw flag for all the objects clipped by this mask */
static void
_evas_mask_redraw_set(Evas_Public_Data *e EINA_UNUSED,
Evas_Object_Protected_Data *obj)
{
Evas_Object_Protected_Data *clippee;
Eina_List *l;
if (!(obj->mask->redraw))
{
EINA_COW_WRITE_BEGIN(evas_object_mask_cow, obj->mask,
Evas_Object_Mask_Data, mask)
mask->redraw = EINA_TRUE;
EINA_COW_WRITE_END(evas_object_mask_cow, obj->mask, mask);
}
if (!obj->cur->cache.clip.dirty)
{
EINA_COW_STATE_WRITE_BEGIN(obj, state_write, cur)
state_write->cache.clip.dirty = EINA_TRUE;
EINA_COW_STATE_WRITE_END(obj, state_write, cur);
}
EINA_LIST_FOREACH(obj->clip.clipees, l, clippee)
{
evas_object_clip_recalc(clippee);
}
}
static inline Eina_Bool
_evas_render_object_changed_get(Evas_Object_Protected_Data *obj)
{
if (obj->smart.smart)
return evas_object_smart_changed_get(obj);
else
return obj->changed;
}
static inline Eina_Bool
_evas_render_object_is_mask(Evas_Object_Protected_Data *obj)
{
if (!obj) return EINA_FALSE;
if (obj->mask->is_mask && obj->clip.clipees)
return EINA_TRUE;
return EINA_FALSE;
}
static void
_evas_render_phase1_direct(Evas_Public_Data *e,
Eina_Inarray *active_objects,
Eina_Array *restack_objects EINA_UNUSED,
Eina_Array *delete_objects EINA_UNUSED,
Eina_Array *render_objects)
{
unsigned int i;
Evas_Object *eo_obj;
RD(0, " [--- PHASE 1 DIRECT\n");
for (i = 0; i < active_objects->len; i++)
{
Evas_Active_Entry *ent = eina_inarray_nth(active_objects, i);
Evas_Object_Protected_Data *obj = ent->obj;
EINA_PREFETCH(&(obj->cur->clipper));
if (obj->changed) evas_object_clip_recalc(obj);
if (obj->proxy->proxies)
{
/* is proxy source */
if (_evas_render_object_changed_get(obj))
_evas_proxy_redraw_set(e, obj, EINA_FALSE);
}
if (_evas_render_object_is_mask(obj))
{
/* is image clipper */
if (_evas_render_object_changed_get(obj))
_evas_mask_redraw_set(e, obj);
}
}
for (i = 0; i < render_objects->count; i++)
{
Evas_Object_Protected_Data *obj =
eina_array_data_get(render_objects, i);
eo_obj = obj->object;
RD(0, " OBJ %s changed %i\n", RDNAME(obj), obj->changed);
if (obj->changed)
{
evas_object_clip_recalc(obj);
obj->func->render_pre(eo_obj, obj, obj->private_data);
if (obj->proxy->redraw || obj->mask->redraw)
_evas_render_prev_cur_clip_cache_add(e, obj);
if (!obj->smart.smart || evas_object_smart_changed_get(obj))
{
/* proxy sources */
if (obj->proxy->proxies)
{
EINA_COW_WRITE_BEGIN(evas_object_proxy_cow, obj->proxy,
Evas_Object_Proxy_Data, proxy_write)
proxy_write->redraw = EINA_TRUE;
EINA_COW_WRITE_END(evas_object_proxy_cow, obj->proxy,
proxy_write);
_evas_proxy_redraw_set(e, obj, EINA_TRUE);
}
/* clipper objects (image masks) */
if (_evas_render_object_is_mask(obj))
_evas_mask_redraw_set(e, obj);
}
_evas_object_gfx_mapping_update(obj);
Eina_Bool has_map = _evas_render_has_map(obj);
RD(0, " pre-render-done smart:%p|%p [%p, %i] | [%p, %i] has_map:%i had_map:%i\n",
obj->smart.smart,
obj->is_smart ? evas_object_smart_members_get_direct(eo_obj) : NULL,
obj->map->cur.map, obj->map->cur.usemap,
obj->map->prev.map, obj->map->prev.usemap,
has_map, _evas_render_had_map(obj));
if ((obj->is_smart) &&
((has_map && !_evas_render_can_map(obj)) ||
obj->changed_src_visible))
{
RD(0, " has map + smart\n");
_evas_render_prev_cur_clip_cache_add(e, obj);
}
/* This is the only case that map was just turned off,
Need to redraw the previous obj region as well. */
else if (!has_map && obj->changed_map &&
_evas_render_object_changed_get(obj))
{
_evas_render_prev_cur_clip_cache_add(e, obj);
obj->changed_map = EINA_FALSE;
}
}
else
{
if (obj->is_smart)
{
}
else if (evas_object_is_visible(obj) &&
((obj->rect_del) ||
(evas_object_is_opaque(obj))) &&
(!evas_object_is_source_invisible(eo_obj, obj)))
{
RD(0, " rect del\n");
_evas_render_cur_clip_cache_del(e, obj);
}
}
}
RD(0, " ---]\n");
}
static void
_evas_render_object_map_change_update(Evas_Public_Data *evas,
Evas_Object_Protected_Data *obj,
Eina_Bool map, Eina_Bool hmap,
int *redraw_all)
{
// FIXME: handle multiple output
Evas_Coord x = 0, y = 0, w = 0, h = 0;
const int fx = evas->framespace.x;
const int fy = evas->framespace.y;
if (map == hmap) return;
if (!obj->map)
{
*redraw_all = 1;
return;
}
if (map)
{
x = obj->prev->cache.clip.x;
y = obj->prev->cache.clip.y;
w = obj->prev->cache.clip.w;
h = obj->prev->cache.clip.h;
if (obj->prev->clipper)
{
RECTS_CLIP_TO_RECT(x, y, w, h,
obj->prev->clipper->prev->cache.clip.x,
obj->prev->clipper->prev->cache.clip.y,
obj->prev->clipper->prev->cache.clip.w,
obj->prev->clipper->prev->cache.clip.h);
}
ENFN->output_redraws_rect_add(ENC, x + fx, y + fy, w, h);
x = obj->map->cur.map->normal_geometry.x;
y = obj->map->cur.map->normal_geometry.y;
w = obj->map->cur.map->normal_geometry.w;
h = obj->map->cur.map->normal_geometry.h;
if (obj->cur->clipper)
{
RECTS_CLIP_TO_RECT(x, y, w, h,
obj->cur->clipper->cur->cache.clip.x,
obj->cur->clipper->cur->cache.clip.y,
obj->cur->clipper->cur->cache.clip.w,
obj->cur->clipper->cur->cache.clip.h);
}
ENFN->output_redraws_rect_add(ENC, x + fx, y + fy, w, h);
}
else if (hmap)
{
x = obj->map->prev.map->normal_geometry.x;
y = obj->map->prev.map->normal_geometry.y;
w = obj->map->prev.map->normal_geometry.w;
h = obj->map->prev.map->normal_geometry.h;
if (obj->prev->clipper)
{
RECTS_CLIP_TO_RECT(x, y, w, h,
obj->prev->clipper->prev->cache.clip.x,
obj->prev->clipper->prev->cache.clip.y,
obj->prev->clipper->prev->cache.clip.w,
obj->prev->clipper->prev->cache.clip.h);
}
ENFN->output_redraws_rect_add(ENC, x + fx, y + fy, w, h);
x = obj->cur->cache.clip.x;
y = obj->cur->cache.clip.y;
w = obj->cur->cache.clip.w;
h = obj->cur->cache.clip.h;
if (obj->cur->clipper)
{
RECTS_CLIP_TO_RECT(x, y, w, h,
obj->cur->clipper->cur->cache.clip.x,
obj->cur->clipper->cur->cache.clip.y,
obj->cur->clipper->cur->cache.clip.w,
obj->cur->clipper->cur->cache.clip.h);
}
ENFN->output_redraws_rect_add(ENC, x + fx, y + fy, w, h);
}
}
////////////////////////////////////////////////////////////////////////////
//
// object render update phase 1 code -> figure out updates and built object
// render/active/delete etc. lists/arrays.
//
typedef struct
{
Eina_Inarray *active_objects;
Eina_Array *render_objects;
Eina_Array *snapshot_objects;
Eina_Inarray *update_del;
} Render_Cache;
void
evas_render_update_del(Evas_Public_Data *evas, int x, int y, int w, int h)
{
// FIXME: handle multiple output
if (EINA_LIKELY((evas->update_del_redirect_array == NULL)))
{
ENFN->output_redraws_rect_del(ENC, x, y, w, h);
}
else
{
Eina_Rectangle r;
r.x = x; r.y = y; r.w = w; r.h = h;
eina_inarray_push(evas->update_del_redirect_array, &r);
}
}
void
evas_render_object_render_cache_free(Evas_Object *eo_obj EINA_UNUSED,
void *data)
{
Render_Cache *rc;
if (!data) return;
rc = data;
eina_inarray_free(rc->active_objects);
eina_array_free(rc->render_objects);
eina_array_free(rc->snapshot_objects);
eina_inarray_free(rc->update_del);
free(rc);
}
typedef struct
{
Evas_Public_Data *e;
Eina_Inarray *active_objects;
Eina_Array *render_objects;
Eina_Array *snapshot_objects;
Eina_Array *restack_objects;
Eina_Array *delete_objects;
int redraw_all;
} Phase1_Context;
#define RENDCACHE 1
#ifdef RENDCACHE
static Render_Cache *
_evas_render_phase1_object_render_cache_new(void)
{
Render_Cache *rc;
rc = malloc(sizeof(Render_Cache));
rc->active_objects = eina_inarray_new(sizeof(Evas_Active_Entry), 32);
rc->render_objects = eina_array_new(32);
rc->snapshot_objects = eina_array_new(32);
rc->update_del = eina_inarray_new(sizeof(Eina_Rectangle), 16);
return rc;
}
static void
_evas_render_phase1_object_ctx_render_cache_fill(Phase1_Context *ctx,
Render_Cache *rc)
{
ctx->active_objects = rc->active_objects;
ctx->render_objects = rc->render_objects;
ctx->snapshot_objects = rc->snapshot_objects;
}
static void
_evas_render_phase1_object_ctx_render_cache_append(Phase1_Context *ctx,
Render_Cache *rc)
{
void *obj;
unsigned int i, c;
Eina_Rectangle *r;
Evas_Active_Entry *ent;
#define ARR_APPEND(x) \
if (rc->x != ctx->x) { \
do { \
c = eina_array_count_get(rc->x); \
for (i = 0; i < c; i++) { \
obj = eina_array_data_get(rc->x, i); \
eina_array_push(ctx->x, obj); \
} \
} while (0); \
}
ARR_APPEND(render_objects);
ARR_APPEND(snapshot_objects);
c = eina_inarray_count(rc->active_objects);
for (i = 0; i < c; i++)
{
ent = eina_inarray_nth(rc->active_objects, i);
eina_inarray_push(ctx->active_objects, ent);
}
c = eina_inarray_count(rc->update_del);
for (i = 0; i < c; i++)
{
r = eina_inarray_nth(rc->update_del, i);
evas_render_update_del(ctx->e, r->x, r->y, r->w, r->h);
}
}
#endif
static Eina_Bool
_evas_render_phase1_object_process(Phase1_Context *p1ctx,
Evas_Object_Protected_Data *obj,
Eina_Bool restack,
Eina_Bool mapped_parent,
Eina_Bool src_changed,
int level);
static void
_evas_render_phase1_object_restack_handle(Phase1_Context *p1ctx,
Evas_Object_Protected_Data *obj,
Eina_Bool obj_changed)
{
if (!obj_changed)
{
OBJ_ARRAY_PUSH(&(p1ctx->e->pending_objects), obj);
obj->changed = EINA_TRUE;
obj->in_pending_objects = EINA_TRUE;
}
obj->restack = EINA_TRUE;
}
static void
_evas_render_phase1_object_map_clipper_fix(Evas_Object *eo_obj,
Evas_Object_Protected_Data *obj)
{
evas_object_change(obj->cur->clipper->object, obj->cur->clipper);
evas_object_clip_dirty(obj->cur->clipper->object, obj->cur->clipper);
evas_object_clip_recalc(obj->cur->clipper);
evas_object_update_bounding_box(eo_obj, obj, NULL);
}
static void
_evas_render_phase1_object_mapped(Phase1_Context *p1ctx,
Evas_Object_Protected_Data *obj,
Eina_Bool src_changed,
Eina_Bool hmap,
Eina_Bool is_active,
Eina_Bool obj_changed,
int level)
{
Evas_Object_Protected_Data *obj2;
Evas_Object *eo_obj = obj->object;
RD(level, " obj mapped\n");
if (!obj_changed) return;
if (!hmap && obj->cur->clipper)
// Fix some bad clipping issues before an evas map animation starts
_evas_render_phase1_object_map_clipper_fix(eo_obj, obj);
_evas_render_object_map_change_update(p1ctx->e, obj, EINA_TRUE, hmap, &(p1ctx->redraw_all));
if (!((is_active) &&
(!obj->clip.clipees) &&
((evas_object_is_visible(obj) &&
(!obj->cur->have_clipees)) ||
(evas_object_was_visible(obj) &&
(!obj->prev->have_clipees)))))
return;
OBJ_ARRAY_PUSH(p1ctx->render_objects, obj);
_evas_render_prev_cur_clip_cache_add(p1ctx->e, obj);
obj->render_pre = EINA_TRUE;
if (!obj->is_smart) return;
if (obj_changed) evas_object_smart_render_cache_clear(eo_obj);
EINA_INLIST_FOREACH(evas_object_smart_members_get_direct(eo_obj), obj2)
{
_evas_render_phase1_object_process(p1ctx, obj2, obj->restack,
EINA_TRUE, src_changed, level + 1);
}
/* Restacked mapped object should be used as a redraw rect.
The "phase 2. force updates for restack" will use restack_objects. */
if (EINA_UNLIKELY(obj->restack))
OBJ_ARRAY_PUSH(p1ctx->restack_objects, obj);
}
static void
_evas_render_phase1_object_mapped_had_restack(Phase1_Context *p1ctx,
Evas_Object_Protected_Data *obj,
Eina_Bool map,
Eina_Bool obj_changed,
int level)
{
Evas_Object *eo_obj = obj->object;
(void) level;
RD(level, " had map - restack objs\n");
_evas_render_prev_cur_clip_cache_add(p1ctx->e, obj);
if (obj_changed)
{
if (!map)
{
if ((obj->map->cur.map) && (obj->map->cur.usemap))
map = EINA_TRUE;
}
_evas_render_object_map_change_update(p1ctx->e, obj, map, EINA_TRUE, &(p1ctx->redraw_all));
}
if (!(!map && obj->cur->clipper)) return;
// Fix some bad clipping issues after an evas_map animation finishes
evas_object_change(obj->cur->clipper->object, obj->cur->clipper);
evas_object_clip_dirty(obj->cur->clipper->object, obj->cur->clipper);
evas_object_clip_recalc(obj->cur->clipper);
evas_object_update_bounding_box(eo_obj, obj, NULL);
}
static Eina_Bool
_evas_render_phase1_object_changed_smart(Phase1_Context *p1ctx,
Evas_Object_Protected_Data *obj,
Eina_Bool mapped_parent,
Eina_Bool obj_changed,
Eina_Bool src_changed,
Eina_Bool is_active,
int level)
{
Evas_Object_Protected_Data *obj2;
Evas_Object *eo_obj = obj->object;
RD(level, " changed + smart - render ok\n");
OBJ_ARRAY_PUSH(p1ctx->render_objects, obj);
if (!is_active && obj->proxy->proxies) src_changed = EINA_TRUE;
obj->render_pre = EINA_TRUE;
if (obj_changed)
{
evas_object_smart_render_cache_clear(eo_obj);
EINA_INLIST_FOREACH(evas_object_smart_members_get_direct(eo_obj), obj2)
{
_evas_render_phase1_object_process(p1ctx, obj2, obj->restack,
mapped_parent, src_changed,
level + 1);
}
}
else
{
Phase1_Context *ctx = p1ctx;
#ifdef RENDCACHE
Render_Cache *rc = NULL;
void *p_del_redir;
Phase1_Context tmpctx;
if (obj->no_change_render > 3)
{
rc = evas_object_smart_render_cache_get(eo_obj);
if (!rc)
{
rc = _evas_render_phase1_object_render_cache_new();
evas_object_smart_render_cache_set(eo_obj, rc);
ctx = &tmpctx;
*ctx = *p1ctx;
p_del_redir = p1ctx->e->update_del_redirect_array;
p1ctx->e->update_del_redirect_array = rc->update_del;
_evas_render_phase1_object_ctx_render_cache_fill(ctx, rc);
EINA_INLIST_FOREACH
(evas_object_smart_members_get_direct(eo_obj), obj2)
{
_evas_render_phase1_object_process(ctx, obj2,
obj->restack,
mapped_parent,
src_changed,
level + 1);
}
p1ctx->redraw_all = ctx->redraw_all;
p1ctx->e->update_del_redirect_array = p_del_redir;
}
_evas_render_phase1_object_ctx_render_cache_append(p1ctx, rc);
}
else
#endif
{
EINA_INLIST_FOREACH
(evas_object_smart_members_get_direct(eo_obj), obj2)
{
_evas_render_phase1_object_process(ctx, obj2, obj->restack,
mapped_parent,
src_changed, level + 1);
}
}
}
return src_changed;
}
static void
_evas_render_phase1_object_changed_normal(Phase1_Context *p1ctx,
Evas_Object_Protected_Data *obj,
Eina_Bool is_active,
int level
#ifndef REND_DBG
EINA_UNUSED
#endif
)
{
if ((!obj->clip.clipees) && _evas_render_is_relevant(obj))
{
if (EINA_LIKELY(is_active))
{
RD(level, " relevant + active\n");
if (EINA_UNLIKELY(obj->restack))
OBJ_ARRAY_PUSH(p1ctx->restack_objects, obj);
else
{
OBJ_ARRAY_PUSH(p1ctx->render_objects, obj);
obj->render_pre = EINA_TRUE;
}
}
else
{
/* It goes to be hidden. Prev caching should be replaced
by the current (hidden) state. */
if (evas_object_is_visible(obj) !=
evas_object_was_visible(obj))
evas_object_cur_prev(obj);
RD(level, " skip - not smart, not active or clippees or not relevant\n");
}
}
else if (is_active && _evas_render_object_is_mask(obj) &&
(obj->cur->visible || obj->prev->visible))
{
if (EINA_UNLIKELY(obj->restack))
OBJ_ARRAY_PUSH(p1ctx->restack_objects, obj);
else
{
OBJ_ARRAY_PUSH(p1ctx->render_objects, obj);
obj->render_pre = EINA_TRUE;
}
RD(level, " relevant + active: clipper image\n");
}
else
{
RD(level, " skip - not smart, not active or clippees or not relevant\n");
}
}
static void
_evas_render_phase1_object_no_changed_smart(Phase1_Context *p1ctx,
Evas_Object_Protected_Data *obj,
Eina_Bool restack,
Eina_Bool mapped_parent,
Eina_Bool src_changed,
int level)
{
Evas_Object_Protected_Data *obj2;
Phase1_Context *ctx = p1ctx;
Evas_Object *eo_obj = obj->object;
RD(level, " smart + visible/was visible + not clip\n");
OBJ_ARRAY_PUSH(p1ctx->render_objects, obj);
obj->render_pre = EINA_TRUE;
#ifdef RENDCACHE
void *p_del_redir;
Render_Cache *rc = NULL;
Phase1_Context tmpctx;
if (obj->no_change_render > 3)
{
rc = evas_object_smart_render_cache_get(eo_obj);
if (!rc)
{
rc = _evas_render_phase1_object_render_cache_new();
evas_object_smart_render_cache_set(eo_obj, rc);
ctx = &tmpctx;
*ctx = *p1ctx;
p_del_redir = p1ctx->e->update_del_redirect_array;
p1ctx->e->update_del_redirect_array = rc->update_del;
_evas_render_phase1_object_ctx_render_cache_fill(ctx, rc);
EINA_INLIST_FOREACH
(evas_object_smart_members_get_direct(eo_obj), obj2)
{
_evas_render_phase1_object_process(ctx, obj2, restack,
mapped_parent,
src_changed, level + 1);
}
p1ctx->redraw_all = ctx->redraw_all;
p1ctx->e->update_del_redirect_array = p_del_redir;
}
_evas_render_phase1_object_ctx_render_cache_append(p1ctx, rc);
}
else
#endif
{
EINA_INLIST_FOREACH
(evas_object_smart_members_get_direct(eo_obj), obj2)
{
_evas_render_phase1_object_process(ctx, obj2, restack,
mapped_parent,
src_changed, level + 1);
}
}
}
static void
_evas_render_phase1_object_no_changed_normal(Phase1_Context *p1ctx,
Evas_Object_Protected_Data *obj,
int level
#ifndef REND_DBG
EINA_UNUSED
#endif
)
{
if (evas_object_is_opaque(obj) &&
evas_object_is_visible(obj))
{
RD(level, " opaque + visible\n");
OBJ_ARRAY_PUSH(p1ctx->render_objects, obj);
obj->rect_del = EINA_TRUE;
}
else if (evas_object_is_visible(obj))
{
RD(level, " visible\n");
OBJ_ARRAY_PUSH(p1ctx->render_objects, obj);
obj->render_pre = EINA_TRUE;
}
else
{
RD(level, " skip\n");
}
}
static Eina_Bool
_evas_render_phase1_object_process(Phase1_Context *p1ctx,
Evas_Object_Protected_Data *obj,
Eina_Bool restack,
Eina_Bool mapped_parent,
Eina_Bool src_changed,
int level)
{
Eina_Bool clean_them = EINA_FALSE;
Eina_Bool map, hmap, can_map, map_not_can_map, obj_changed, is_active;
Evas_Object *eo_obj = obj->object;
EINA_PREFETCH(&(obj->cur->clipper));
obj->rect_del = EINA_FALSE;
obj->render_pre = EINA_FALSE;
if (obj->delete_me == 2) OBJ_ARRAY_PUSH(p1ctx->delete_objects, obj);
else if (obj->delete_me != 0) obj->delete_me++;
/* If the object will be removed, we should not cache anything during this run. */
if (obj->delete_me != 0) clean_them = EINA_TRUE;
obj_changed = obj->changed;
if (obj->is_static_clip) goto done;
//Need pre render for the children of mapped object.
//But only when they have changed.
if (mapped_parent && (!obj->changed)) goto done;
/* build active object list */
evas_object_clip_recalc(obj);
if (src_changed) is_active = EINA_TRUE;
else
{
is_active = evas_object_is_active(eo_obj, obj);
if (is_active && obj->proxy->proxies) src_changed = is_active;
}
obj->is_active = is_active;
#ifdef REND_DBG
RD(level, "[--- PROCESS %s active = %i, del = %i | %i,%i %ix%i\n",
RDNAME(obj), is_active, obj->delete_me, obj->cur->geometry.x,
obj->cur->geometry.y, obj->cur->geometry.w, obj->cur->geometry.h);
#endif
if ((!mapped_parent) &&
((is_active) || (obj->delete_me != 0)) &&
(!obj->no_render))
{
Evas_Active_Entry ent;
#ifdef INLINE_ACTIVE_GEOM
if (obj->is_smart)
evas_object_smart_bounding_box_get(obj, &(ent.rect), NULL);
else
{
ent.rect.x = obj->cur->cache.clip.x;
ent.rect.y = obj->cur->cache.clip.y;
ent.rect.w = obj->cur->cache.clip.w;
ent.rect.h = obj->cur->cache.clip.h;
}
#endif
ent.obj = obj;
eina_inarray_push(p1ctx->active_objects, &ent);
}
if (is_active && obj->cur->snapshot && !obj->delete_me &&
evas_object_is_visible(obj))
{
// FIXME: Array should be clean (no need to check if already in there)
int found = 0;
for (unsigned i = 0; !found && (i < p1ctx->snapshot_objects->count); i++)
if (obj == eina_array_data_get(p1ctx->snapshot_objects, i))
found = 1;
if (!found) OBJ_ARRAY_PUSH(p1ctx->snapshot_objects, obj);
}
#ifdef REND_DBG
if (!is_active)
{
RD(level, "%s vis: %i, cache.clip.vis: %i cache.clip.a: %i [func: %p]\n",
RDNAME(obj), obj->cur->visible, obj->cur->cache.clip.visible,
obj->cur->cache.clip.a, obj->func->is_visible);
}
#endif
_evas_object_gfx_mapping_update(obj);
map = _evas_render_has_map(obj);
hmap = _evas_render_had_map(obj);
can_map = _evas_render_can_map(obj);
map_not_can_map = map & !can_map;
if (EINA_UNLIKELY((restack && !map_not_can_map)))
{
_evas_render_phase1_object_restack_handle(p1ctx, obj, obj_changed);
obj_changed = EINA_TRUE;
clean_them = EINA_TRUE;
}
if (EINA_UNLIKELY(map_not_can_map))
{
_evas_render_phase1_object_mapped(p1ctx, obj, src_changed, hmap,
is_active, obj_changed, level);
goto done;
}
else if (EINA_UNLIKELY(hmap && !can_map))
_evas_render_phase1_object_mapped_had_restack(p1ctx, obj, map,
obj_changed, level);
/* handle normal rendering. this object knows how to handle maps */
if (obj_changed)
{
if (obj->is_smart)
src_changed =
_evas_render_phase1_object_changed_smart(p1ctx, obj, mapped_parent,
obj_changed, src_changed,
is_active, level);
else /* non smart object */
_evas_render_phase1_object_changed_normal(p1ctx, obj, is_active,
level);
}
else
{
/* not changed */
RD(level, " not changed... [%i] -> (%i %i %p %i) [%i]\n",
evas_object_is_visible(obj),
obj->cur->visible, obj->cur->cache.clip.visible, obj->smart.smart,
obj->cur->cache.clip.a, evas_object_was_visible(obj));
if ((!obj->clip.clipees) &&
(EINA_LIKELY(obj->delete_me == 0)) &&
(_evas_render_can_render(obj) ||
(evas_object_was_visible(obj) &&
(!obj->prev->have_clipees))))
{
if (obj->is_smart)
_evas_render_phase1_object_no_changed_smart(p1ctx, obj, restack,
mapped_parent,
src_changed, level);
else /* not smart */
_evas_render_phase1_object_no_changed_normal(p1ctx, obj, level);
}
else if (EINA_UNLIKELY(is_active &&
_evas_render_object_is_mask(obj) &&
obj->cur->visible))
{
RD(level, " visible clipper image\n");
OBJ_ARRAY_PUSH(p1ctx->render_objects, obj);
obj->render_pre = EINA_TRUE;
}
}
if (!is_active) obj->restack = EINA_FALSE;
RD(level, "---]\n");
done:
if (obj_changed) obj->no_change_render = 0;
else
{
if (obj->no_change_render < 255) obj->no_change_render++;
}
return clean_them;
}
//
//
//
////////////////////////////////////////////////////////////////////////////
static Eina_Bool
_evas_render_phase1_process(Phase1_Context *p1ctx)
{
Evas_Layer *lay;
Eina_Bool clean_them = EINA_FALSE;
RD(0, " [--- PHASE 1\n");
EINA_INLIST_FOREACH(p1ctx->e->layers, lay)
{
Evas_Object_Protected_Data *obj;
EINA_INLIST_FOREACH(lay->objects, obj)
{
if (evas_object_is_on_plane(obj)) continue;
clean_them |= _evas_render_phase1_object_process
(p1ctx, obj, EINA_FALSE, EINA_FALSE, EINA_FALSE, 2);
}
}
RD(0, " ---]\n");
return clean_them;
}
static void
_evas_render_check_pending_objects(Eina_Array *pending_objects, Evas *eo_e EINA_UNUSED, Evas_Public_Data *e)
{
unsigned int i;
for (i = 0; i < pending_objects->count; ++i)
{
Evas_Object *eo_obj;
int is_active;
Eina_Bool ok = EINA_FALSE;
Evas_Object_Protected_Data *obj = eina_array_data_get(pending_objects, i);
eo_obj = obj->object;
if (!obj->layer) goto clean_stuff;
_evas_object_gfx_mapping_update(obj);
EINA_PREFETCH(&(obj->cur->clipper));
EINA_PREFETCH(&(obj->cur->cache.clip));
//If the children are in active objects, They should be cleaned up.
if (EINA_UNLIKELY((obj->changed_map) &&
(_evas_render_has_map(obj)) &&
(!_evas_render_can_map(obj))))
goto clean_stuff;
evas_object_clip_recalc(obj);
is_active = evas_object_is_active(eo_obj, obj);
if ((!is_active) && (!obj->is_active) && (!obj->render_pre) &&
(!obj->rect_del))
{
ok = EINA_TRUE;
goto clean_stuff;
}
if (obj->is_active == is_active)
{
if (obj->changed)
{
if (obj->is_smart)
{
if (obj->render_pre || obj->rect_del) ok = EINA_TRUE;
}
else
if ((is_active) && (obj->restack) && (!obj->clip.clipees) &&
(_evas_render_can_render(obj) ||
(evas_object_was_visible(obj) && (!obj->prev->have_clipees))))
{
if (!obj->render_pre && !obj->rect_del && !obj->delete_me)
ok = EINA_TRUE;
}
else
if (is_active && (!obj->clip.clipees) &&
(_evas_render_can_render(obj) ||
(evas_object_was_visible(obj) && (!obj->prev->have_clipees))))
{
if (obj->render_pre || obj->rect_del) ok = EINA_TRUE;
}
}
else
{
if ((!obj->clip.clipees) && (obj->delete_me == 0) &&
(!obj->cur->have_clipees || (evas_object_was_visible(obj) && (!obj->prev->have_clipees)))
&& evas_object_is_opaque(obj) && evas_object_is_visible(obj))
{
if (obj->rect_del || obj->is_smart) ok = EINA_TRUE;
}
}
}
clean_stuff:
if (!ok)
{
eina_inarray_flush(&e->active_objects);
OBJS_ARRAY_CLEAN(&e->render_objects);
OBJS_ARRAY_CLEAN(&e->restack_objects);
OBJS_ARRAY_CLEAN(&e->delete_objects);
OBJS_ARRAY_CLEAN(&e->snapshot_objects);
e->invalidate = EINA_TRUE;
return;
}
}
}
Eina_Bool
pending_change(void *data, void *gdata EINA_UNUSED)
{
Evas_Object *eo_obj;
Evas_Object_Protected_Data *obj = data;
eo_obj = obj->object;
if (obj->delete_me) return EINA_FALSE;
if (obj->pre_render_done)
{
RD(0, " OBJ %s pending change %i -> 0, pre %i\n", RDNAME(obj), obj->changed, obj->pre_render_done);
evas_object_change_reset(obj);
obj->func->render_post(eo_obj, obj, obj->private_data);
obj->pre_render_done = EINA_FALSE;
}
else if (!_evas_render_can_render(obj) &&
(!obj->is_active) && (!obj->render_pre) &&
(!obj->rect_del))
{
evas_object_change_reset(obj);
}
//FIXME: after evas_object_change_reset(), obj->changed is always false...
Eina_Bool val = obj->changed ? EINA_TRUE : EINA_FALSE;
obj->in_pending_objects = val;
return val;
}
static Eina_Bool
_evas_render_can_use_overlay(Evas_Public_Data *e, Evas_Object *eo_obj, Efl_Canvas_Output *output)
{
Eina_Rectangle *r;
Evas_Object *eo_tmp;
Eina_List *alphas = NULL;
Eina_List *opaques = NULL;
Evas_Object *video_parent = NULL;
Eina_Rectangle zone;
Evas_Coord xc1, yc1, xc2, yc2;
int i;
Eina_Bool nooverlay;
Evas_Object_Protected_Data *obj = efl_data_scope_get(eo_obj, EFL_CANVAS_OBJECT_CLASS);
Evas_Object_Protected_Data *tmp = NULL;
Evas_Coord imgw, imgh;
unsigned int caps;
Eina_Bool surface_below, stacking_check, object_above = EINA_FALSE;
Eina_Bool ignore_window;
if (!_evas_object_image_can_use_plane(obj, output))
return EINA_FALSE;
video_parent = _evas_object_image_video_parent_get(eo_obj);
/* Check if any one is the stack make this object mapped */
eo_tmp = eo_obj;
tmp = efl_data_scope_get(eo_tmp, EFL_CANVAS_OBJECT_CLASS);
_evas_object_gfx_mapping_update(tmp);
while (tmp && !(_evas_render_has_map(tmp) && !_evas_render_can_map(tmp)))
{
eo_tmp = tmp->smart.parent;
tmp = efl_data_scope_get(eo_tmp, EFL_CANVAS_OBJECT_CLASS);
if (tmp) _evas_object_gfx_mapping_update(tmp);
}
if (tmp && _evas_render_has_map(tmp) && !_evas_render_can_map(tmp))
return EINA_FALSE; /* we are mapped, we can't be an overlay */
if (!evas_object_is_visible(obj))
return EINA_FALSE; /* no need to update the overlay if it's not visible */
/* If any recoloring of the surface is needed, n overlay to */
if ((obj->cur->cache.clip.r != 255) ||
(obj->cur->cache.clip.g != 255) ||
(obj->cur->cache.clip.b != 255) ||
(obj->cur->cache.clip.a != 255))
return EINA_FALSE;
caps = evas_object_image_video_surface_caps_get(eo_obj);
/* check if surface is above the canvas */
surface_below = !!(caps & EVAS_VIDEO_SURFACE_BELOW);
if (!surface_below)
{
/* above canvas, must support resize and clipping */
/* check if video surface supports resize */
evas_object_image_size_get(eo_obj, &imgw, &imgh);
if ((obj->cur->geometry.w != imgw) ||
(obj->cur->geometry.h != imgh))
{
if (!(caps & EVAS_VIDEO_SURFACE_RESIZE))
return EINA_FALSE;
}
/* check if video surface supports clipping */
evas_object_image_size_get(eo_obj, &imgw, &imgh);
if ((obj->cur->cache.clip.x != obj->cur->geometry.x) ||
(obj->cur->cache.clip.y != obj->cur->geometry.y) ||
(obj->cur->cache.clip.w != obj->cur->geometry.w) ||
(obj->cur->cache.clip.h != obj->cur->geometry.h))
{
if (!(caps & EVAS_VIDEO_SURFACE_CLIP))
return EINA_FALSE;
}
}
/* check for window/surface/canvas limits */
ignore_window = !!(caps & EVAS_VIDEO_SURFACE_IGNORE_WINDOW);
if (!ignore_window)
{
Evas_Coord x1, x2, y1, y2;
Evas_Coord fx, fy, fw, fh;
fx = e->framespace.x;
fy = e->framespace.y;
fw = e->framespace.w;
fh = e->framespace.h;
x1 = obj->cur->geometry.x + fx;
y1 = obj->cur->geometry.y + fy;
x2 = obj->cur->geometry.x + obj->cur->geometry.w + fx;
y2 = obj->cur->geometry.y + obj->cur->geometry.h + fy;
if ((x1 < fx) || (y1 < fy) ||
(x2 > e->output.w - (fw - fx)) ||
(y2 > e->output.h - (fh - fy)))
return EINA_FALSE;
}
/* check if there are other objects above the video object? */
stacking_check = !!(caps & EVAS_VIDEO_SURFACE_STACKING_CHECK);
if (!stacking_check)
return EINA_TRUE;
/* Check presence of transparent object on top of the video object */
EINA_RECTANGLE_SET(&zone,
obj->cur->cache.clip.x,
obj->cur->cache.clip.y,
obj->cur->cache.clip.w,
obj->cur->cache.clip.h);
for (i = e->active_objects.len - 1; i > 0; i--)
{
Eina_Rectangle self;
Eina_Rectangle *match;
Evas_Object *eo_current;
Eina_List *l;
int xm1, ym1, xm2, ym2;
Evas_Active_Entry *ent = eina_inarray_nth(&e->active_objects, i);
Evas_Object_Protected_Data *current = ent->obj;
eo_current = current->object;
/* Did we find the video object in the stack ? */
if (eo_current == video_parent || eo_current == eo_obj)
break;
EINA_RECTANGLE_SET(&self,
current->cur->cache.clip.x,
current->cur->cache.clip.y,
current->cur->cache.clip.w,
current->cur->cache.clip.h);
/* This doesn't cover the area of the video object, so don't bother with that object */
if (!eina_rectangles_intersect(&zone, &self))
continue;
xc1 = current->cur->cache.clip.x;
yc1 = current->cur->cache.clip.y;
xc2 = current->cur->cache.clip.x + current->cur->cache.clip.w;
yc2 = current->cur->cache.clip.y + current->cur->cache.clip.h;
if (evas_object_is_visible(current) &&
(!current->clip.clipees) &&
(current->cur->visible) &&
(!current->delete_me) &&
(current->cur->cache.clip.visible) &&
(!efl_isa(eo_current, EFL_CANVAS_GROUP_CLASS)))
{
Eina_Bool included = EINA_FALSE;
if (!surface_below)
{
object_above = EINA_TRUE;
break;
}
if (evas_object_is_opaque(current) ||
((current->func->has_opaque_rect) &&
(current->func->has_opaque_rect(eo_current, current, current->private_data))))
{
/* The object is opaque */
/* Check if the opaque object is inside another opaque object */
EINA_LIST_FOREACH(opaques, l, match)
{
xm1 = match->x;
ym1 = match->y;
xm2 = match->x + match->w;
ym2 = match->y + match->h;
/* Both object are included */
if (xc1 >= xm1 && yc1 >= ym1 && xc2 <= xm2 && yc2 <= ym2)
{
included = EINA_TRUE;
break;
}
}
/* Not included yet */
if (!included)
{
Eina_List *ln;
Evas_Coord xn2, yn2;
r = eina_rectangle_new(current->cur->cache.clip.x, current->cur->cache.clip.y,
current->cur->cache.clip.w, current->cur->cache.clip.h);
opaques = eina_list_append(opaques, r);
xn2 = r->x + r->w;
yn2 = r->y + r->h;
/* Remove all the transparent object that are covered by the new opaque object */
EINA_LIST_FOREACH_SAFE(alphas, l, ln, match)
{
xm1 = match->x;
ym1 = match->y;
xm2 = match->x + match->w;
ym2 = match->y + match->h;
if (xm1 >= r->x && ym1 >= r->y && xm2 <= xn2 && ym2 <= yn2)
{
/* The new rectangle is over some transparent object,
so remove the transparent object */
alphas = eina_list_remove_list(alphas, l);
}
}
}
}
else
{
/* The object has some transparency */
/* Check if the transparent object is inside any other transparent object */
EINA_LIST_FOREACH(alphas, l, match)
{
xm1 = match->x;
ym1 = match->y;
xm2 = match->x + match->w;
ym2 = match->y + match->h;
/* Both object are included */
if (xc1 >= xm1 && yc1 >= ym1 && xc2 <= xm2 && yc2 <= ym2)
{
included = EINA_TRUE;
break;
}
}
/* If not check if it is inside any opaque one */
if (!included)
{
EINA_LIST_FOREACH(opaques, l, match)
{
xm1 = match->x;
ym1 = match->y;
xm2 = match->x + match->w;
ym2 = match->y + match->h;
/* Both object are included */
if (xc1 >= xm1 && yc1 >= ym1 && xc2 <= xm2 && yc2 <= ym2)
{
included = EINA_TRUE;
break;
}
}
}
/* No inclusion at all, so add it */
if (!included)
{
r = eina_rectangle_new(current->cur->cache.clip.x, current->cur->cache.clip.y,
current->cur->cache.clip.w, current->cur->cache.clip.h);
alphas = eina_list_append(alphas, r);
}
}
}
}
/* If there is any pending transparent object, then no overlay */
nooverlay = !!eina_list_count(alphas);
EINA_LIST_FREE(alphas, r)
eina_rectangle_free(r);
EINA_LIST_FREE(opaques, r)
eina_rectangle_free(r);
if (nooverlay || object_above)
return EINA_FALSE;
return EINA_TRUE;
}
static Eina_Bool
_proxy_context_clip(Evas_Public_Data *evas, void *ctx, Evas_Proxy_Render_Data *proxy_render_data, Evas_Object_Protected_Data *obj, int off_x, int off_y)
{
const Eina_Rectangle *clip;
Evas_Object_Protected_Data *clipper;
int cw, ch;
/* cache.clip can not be relied on, since the evas is frozen, but we need
* to set the clip. so we recurse from clipper to clipper until we reach
* the source object's clipper */
if (!proxy_render_data) return EINA_TRUE;
if (proxy_render_data->source_clip)
{
Evas_Object_Protected_Data *src_obj = proxy_render_data->src_obj;
/* trust cache.clip since we clip like the source */
ENFN->context_clip_clip(ENC, ctx,
src_obj->cur->cache.clip.x + off_x,
src_obj->cur->cache.clip.y + off_y,
src_obj->cur->cache.clip.w, src_obj->cur->cache.clip.h);
ENFN->context_clip_get(ENC, ctx, NULL, NULL, &cw, &ch);
return ((cw > 0) && (ch > 0));
}
if (!obj->cur->clipper) return EINA_TRUE;
clipper = obj->cur->clipper;
if (!clipper->cur->visible) return EINA_FALSE;
clip = &clipper->cur->geometry;
ENFN->context_clip_clip(ENC, ctx, clip->x + off_x, clip->y + off_y, clip->w, clip->h);
ENFN->context_clip_get(ENC, ctx, NULL, NULL, &cw, &ch);
if ((cw <= 0) || (ch <= 0)) return EINA_FALSE;
/* stop if we found the source object's clipper */
if (clipper == proxy_render_data->src_obj->cur->clipper) return EINA_TRUE;
/* recurse to the clipper itself.
* origin of clipper's clipper won't be transformed to derivative space. */
return _proxy_context_clip(evas, ctx, proxy_render_data, clipper,
-proxy_render_data->src_obj->cur->geometry.x,
-proxy_render_data->src_obj->cur->geometry.y);
}
static Eina_Bool
_mask_apply_inside_proxy(Evas_Proxy_Render_Data *proxy_render_data,
Evas_Object_Protected_Data *mask)
{
// Trying to find out if the mask should be applied inside the proxy or not.
if (!proxy_render_data || proxy_render_data->source_clip) return EINA_TRUE;
if (!proxy_render_data->src_obj->cur->clipper) return EINA_FALSE;
if (!mask) return EINA_FALSE;
// FIXME: Need to implement a logic similar to _proxy_context_clip
return EINA_FALSE;
}
static void
_evas_render_mapped_context_clip_set(Evas_Public_Data *evas, Evas_Object *eo_obj, Evas_Object_Protected_Data *obj, void *ctx, Evas_Proxy_Render_Data *proxy_render_data, int off_x, int off_y)
{
int x, y, w, h;
Eina_Bool proxy_src_clip = EINA_TRUE;
if (proxy_render_data) proxy_src_clip = proxy_render_data->source_clip;
if (proxy_src_clip)
{
x = obj->cur->cache.clip.x;
y = obj->cur->cache.clip.y;
w = obj->cur->cache.clip.w;
h = obj->cur->cache.clip.h;
RECTS_CLIP_TO_RECT(x, y, w, h,
obj->cur->clipper->cur->cache.clip.x,
obj->cur->clipper->cur->cache.clip.y,
obj->cur->clipper->cur->cache.clip.w,
obj->cur->clipper->cur->cache.clip.h);
ENFN->context_clip_set(ENC, ctx, x + off_x, y + off_y, w, h);
}
else if (evas->is_frozen)
{
/* can't trust cache.clip here - clip should be in ctx already */
}
else
{
//FIXME: Consider to clip by the proxy clipper.
if (proxy_render_data->eo_src != eo_obj)
{
x = obj->cur->clipper->cur->geometry.x + off_x;
y = obj->cur->clipper->cur->geometry.y + off_y;
w = obj->cur->clipper->cur->geometry.w;
h = obj->cur->clipper->cur->geometry.h;
ENFN->context_clip_clip(ENC, ctx, x, y, w, h);
}
}
}
static void
_evas_render_mapped_mask(Evas_Public_Data *evas, Evas_Object_Protected_Data *obj, Evas_Object_Protected_Data *mask,
Evas_Proxy_Render_Data *proxy_render_data, void *output, void *ctx, int off_x, int off_y, int level, Eina_Bool do_async)
{
if (!mask) return;
if (proxy_render_data &&
!proxy_render_data->source_clip &&
proxy_render_data->src_obj->clip.mask == mask) return;
// This path can be hit when we're multiplying masks on top of each other...
Evas_Object_Protected_Data *prev_mask = obj->clip.prev_mask;
Eina_Bool redraw = mask->mask->redraw || !mask->mask->surface;
RD(level, " has mask: %s redraw:%d sfc:%p prev_mask:%p\n",
RDNAME(mask), mask->mask->redraw, mask->mask->surface, prev_mask);
if (prev_mask && !_mask_apply_inside_proxy(proxy_render_data, prev_mask))
{
RD(level, " discard prev mask and redraw (guessed outside proxy)\n");
prev_mask = NULL;
redraw = EINA_TRUE;
}
if (redraw)
evas_render_mask_subrender(evas, output, mask, prev_mask, level + 1, do_async);
if (mask->mask->surface)
{
Eina_Rectangle clip = _evas_render_smallest_static_clipped_geometry_get(mask->cur);
ENFN->context_clip_image_set(ENC, ctx, mask->mask->surface,
clip.x + off_x,
clip.y + off_y,
evas, do_async);
}
}
Eina_Bool
evas_render_mapped(Evas_Public_Data *evas, Evas_Object *eo_obj,
Evas_Object_Protected_Data *obj, void *context,
void *output, void *surface,
int off_x, int off_y, int mapped, int ecx,
int ecy, int ecw, int ech,
Evas_Proxy_Render_Data *proxy_render_data, int level,
Eina_Bool do_async)
{
Evas_Object_Protected_Data *obj2;
Eina_Bool clean_them = EINA_FALSE;
Eina_Bool proxy_src_clip = EINA_TRUE;
Eina_Bool was_pre_render_done = obj->pre_render_done;
void *ctx;
if (!proxy_render_data)
{
/* don't render if the source is invisible */
if ((evas_object_is_source_invisible(eo_obj, obj)))
return EINA_FALSE;
}
else
proxy_src_clip = proxy_render_data->source_clip;
evas_object_clip_recalc(obj);
_evas_object_gfx_mapping_update(obj);
/* leave early if clipper is not visible */
if ((obj->cur->clipper) && (!obj->cur->clipper->cur->visible))
return EINA_FALSE;
eina_evlog("+render_object", eo_obj, 0.0, NULL);
#ifdef REND_DBG
RD(level, "{\n");
RD(level, " evas_render_mapped(evas:%p, obj:%s, ctx:%p, sfc:%p, offset:%i,%i, %s, %s)\n",
evas->evas, RDNAME(obj), context, surface, off_x, off_y,
mapped ? "mapped" : "normal", do_async ? "async" : "sync");
IFRD(obj->is_smart, level, " is_smart");
IFRD(obj->is_frame, level, " in framespace%s", obj->is_frame_top ? " (top object)" : "");
if (obj->cur->clipper)
{
RD(level, " clipper: %s (mask: %s) %d,%d %dx%d ; color: {%d,%d,%d,%d} ; cached: {%d,%d,%d,%d}\n",
RDNAME(obj->cur->clipper), RDNAME(obj->clip.mask),
obj->cur->clipper->cur->geometry.x, obj->cur->clipper->cur->geometry.y,
obj->cur->clipper->cur->geometry.w, obj->cur->clipper->cur->geometry.h,
obj->cur->clipper->cur->color.r, obj->cur->clipper->cur->color.g,
obj->cur->clipper->cur->color.b, obj->cur->clipper->cur->color.a,
obj->cur->clipper->cur->cache.clip.r, obj->cur->clipper->cur->cache.clip.g,
obj->cur->clipper->cur->cache.clip.b, obj->cur->clipper->cur->cache.clip.a);
}
RD(level, " geom: %d,%d %dx%d, cache.clip: (vis: %d) %d,%d %dx%d ; color: {%d,%d,%d,%d} ; cached: {%d,%d,%d,%d}\n",
obj->cur->geometry.x, obj->cur->geometry.y, obj->cur->geometry.w, obj->cur->geometry.h, obj->cur->cache.clip.visible,
obj->cur->cache.clip.x, obj->cur->cache.clip.y, obj->cur->cache.clip.w, obj->cur->cache.clip.h,
obj->cur->color.r, obj->cur->color.g, obj->cur->color.b, obj->cur->color.a,
obj->cur->cache.clip.r, obj->cur->cache.clip.g, obj->cur->cache.clip.b, obj->cur->cache.clip.a);
{
int _cu, _cc, _cm, _cx, _cy, _cw, _ch, _cr, _cg, _cb, _ca, _cmr, _cmg, _cmb, _cma;
_cu = ENFN->context_clip_get(ENC, context, &_cx, &_cy, &_cw, &_ch);
_cc = ENFN->context_color_get(ENC, context, &_cr, &_cg, &_cb, &_ca);
_cm = ENFN->context_multiplier_get(ENC, context, &_cmr, &_cmg, &_cmb, &_cma);
RD(level, " context clip: [%d] %d,%d %dx%d ; color: [%d] {%d,%d,%d,%d} ; mult: [%d] {%d,%d,%d,%d}\n",
_cu, _cx, _cy, _cw, _ch, _cc, _cr, _cg, _cb, _ca, _cm, _cmr, _cmg, _cmb, _cma);
}
#endif
if (mapped)
{
if (_evas_render_object_is_mask(obj))
{
RD(level, " is mask: redraw:%d sfc:%p\n", obj->mask->redraw, obj->mask->surface);
if (surface != obj->mask->surface)
{
RD(level, " not rendering mask surface\n");
RD(level, "}\n");
eina_evlog("-render_object", eo_obj, 0.0, NULL);
return EINA_FALSE;
}
// else don't return: draw mask in its surface
}
else if (proxy_src_clip)
{
if (!evas->is_frozen) /* same as "if (proxy_render_data)" */
{
if ((!evas_object_is_visible(obj)) || (obj->clip.clipees)
|| (obj->cur->have_clipees) || (obj->no_render))
{
IFRD(obj->no_render, level, " no_render\n");
IFRD(obj->clip.clipees || obj->cur->have_clipees, level, " has clippees\n");
IFRD(!evas_object_is_visible(obj), level, " not visible\n");
RD(level, "}\n");
eina_evlog("-render_object", eo_obj, 0.0, NULL);
return EINA_FALSE;
}
}
else
{
/* can not trust cache.clip - evas is frozen */
if (!obj->cur->visible || obj->clip.clipees || (obj->no_render && !proxy_render_data) ||
(!obj->cur->color.a && (obj->cur->render_op == EVAS_RENDER_BLEND)))
{
IFRD(obj->no_render, level, " proxy_src_clip + no_render\n");
IFRD(obj->clip.clipees || obj->cur->have_clipees, level, " proxy_src_clip + has clippees\n");
IFRD(!obj->cur->visible, level, " proxy_src_clip + not visible\n");
IFRD(!obj->cur->color.a && (obj->cur->render_op == EVAS_RENDER_BLEND), level, " proxy_src_clip + 0 alpha\n");
RD(level, "}\n");
eina_evlog("-render_object", eo_obj, 0.0, NULL);
return EINA_FALSE;
}
else if (proxy_render_data && (surface != obj->proxy->surface) &&
obj->proxy->src_invisible)
{
RD(level, " src_invisible + not proxy surface (recursive proxies)\n");
RD(level, "}\n");
eina_evlog("-render_object", eo_obj, 0.0, NULL);
return EINA_FALSE;
}
}
}
else if (!evas_object_is_proxy_visible(obj) ||
(obj->clip.clipees) || (obj->cur->have_clipees))
{
IFRD(!evas_object_is_proxy_visible(obj), level, " proxy not visible\n");
IFRD(obj->clip.clipees || obj->cur->have_clipees, level, " has clippees\n");
RD(level, "}\n");
eina_evlog("-render_object", eo_obj, 0.0, NULL);
return EINA_FALSE;
}
else if (obj->no_render && (surface != obj->proxy->surface))
{
RD(level, " no_render\n");
RD(level, "}\n");
eina_evlog("-render_object", eo_obj, 0.0, NULL);
return EINA_FALSE;
}
}
else if (!(((evas_object_is_active(eo_obj, obj) && (!obj->clip.clipees) &&
(_evas_render_can_render(obj))))
))
{
IFRD(!evas_object_is_active(eo_obj, obj), level, " not active\n");
IFRD(!_evas_render_can_render(obj), level, " can't render\n");
IFRD(obj->clip.clipees, level, " has clippees\n");
RD(level, "}\n");
eina_evlog("-render_object", eo_obj, 0.0, NULL);
return EINA_FALSE;
}
// set render_pre - for child objs that may not have gotten it.
obj->pre_render_done = EINA_TRUE;
RD(level, " hasmap: %s [can_map: %d] cur.map:%p cur.usemap:%d\n",
_evas_render_has_map(obj) ? "yes" : "no",
obj->func->can_map ? obj->func->can_map(eo_obj): -1,
obj->map->cur.map, obj->map->cur.usemap);
if (_evas_render_has_map(obj) && !_evas_render_can_map(obj))
{
int sw, sh;
Eina_Bool changed = EINA_FALSE, rendered = EINA_FALSE, pchanged = EINA_FALSE;
clean_them = EINA_TRUE;
sw = obj->cur->geometry.w;
sh = obj->cur->geometry.h;
RD(level, " surf size: %ix%i\n", sw, sh);
if ((sw <= 0) || (sh <= 0)) goto end;
pchanged = evas_object_map_update(eo_obj, off_x, off_y, sw, sh, sw, sh);
if (obj->map->surface)
{
if ((obj->map->surface_w != sw) ||
(obj->map->surface_h != sh))
{
RD(level, " new surf: %ix%i\n", sw, sh);
EINA_COW_WRITE_BEGIN(evas_object_map_cow, obj->map, Evas_Object_Map_Data, map_write)
{
ENFN->image_free(ENC, map_write->surface);
map_write->surface = NULL;
}
EINA_COW_WRITE_END(evas_object_map_cow, obj->map, map_write);
}
}
if (!obj->map->surface)
{
EINA_COW_WRITE_BEGIN(evas_object_map_cow, obj->map, Evas_Object_Map_Data, map_write)
{
map_write->surface_w = sw;
map_write->surface_h = sh;
map_write->surface = ENFN->image_map_surface_new
(ENC, map_write->surface_w,
map_write->surface_h,
map_write->cur.map->alpha);
}
EINA_COW_WRITE_END(evas_object_map_cow, obj->map, map_write);
RD(level, " first surf: %ix%i\n", sw, sh);
changed = EINA_TRUE;
}
if (!changed) changed = evas_object_smart_changed_get(obj);
/* mark the old map as invalid, so later we don't reuse it as a
* cache. */
if (pchanged && obj->map->prev.map)
{
EINA_COW_WRITE_BEGIN(evas_object_map_cow, obj->map, Evas_Object_Map_Data, map_write)
map_write->prev.valid_map = EINA_FALSE;
EINA_COW_WRITE_END(evas_object_map_cow, obj->map, map_write);
}
// clear surface before re-render
if ((changed) && (obj->map->surface))
{
int off_x2, off_y2;
RD(level, " children redraw\n");
// FIXME: calculate "changes" within map surface and only clear
// and re-render those
if (obj->map->cur.map->alpha)
{
ctx = ENFN->context_new(ENC);
ENFN->context_color_set(ENC, ctx, 0, 0, 0, 0);
ENFN->context_render_op_set(ENC, ctx, EVAS_RENDER_COPY);
ENFN->rectangle_draw(ENC, output, ctx, obj->map->surface,
0, 0, obj->map->surface_w, obj->map->surface_h,
EINA_FALSE);
ENFN->context_free(ENC, ctx);
}
ctx = ENFN->context_new(ENC);
off_x2 = -obj->cur->geometry.x;
off_y2 = -obj->cur->geometry.y;
if (obj->is_smart)
{
EINA_INLIST_FOREACH
(evas_object_smart_members_get_direct(eo_obj), obj2)
{
clean_them |= evas_render_mapped(evas, obj2->object,
obj2, ctx,
output, obj->map->surface,
off_x2, off_y2, 1,
ecx, ecy, ecw, ech,
proxy_render_data,
level + 1, do_async);
/* We aren't sure this object will be rendered by
normal(not proxy) drawing after, we reset this
only in case of normal drawing. For optmizing,
push this object in an array then reset them
in the end of the rendering.*/
if (!proxy_render_data)
evas_object_change_reset(obj2);
}
}
else
{
int x = 0, y = 0, w = 0, h = 0;
w = obj->map->surface_w;
h = obj->map->surface_h;
RECTS_CLIP_TO_RECT(x, y, w, h,
obj->cur->geometry.x + off_x2,
obj->cur->geometry.y + off_y2,
obj->cur->geometry.w,
obj->cur->geometry.h);
ENFN->context_clip_set(ENC, ctx, x, y, w, h);
#ifdef REND_DBG
int _c, _cx, _cy, _cw, _ch;
_c = ENFN->context_clip_get(ENC, ctx, &_cx, &_cy, &_cw, &_ch);
RD(level, " draw mapped obj: render(clip: [%d] %d,%d %dx%d)\n", _c, _cx, _cy, _cw, _ch);
#endif
// FIXME: Should this really be sync render?
obj->func->render(eo_obj, obj, obj->private_data,
ENC, output, ctx,
obj->map->surface, off_x2, off_y2,
EINA_FALSE);
}
ENFN->context_free(ENC, ctx);
rendered = EINA_TRUE;
}
if (rendered)
{
EINA_COW_WRITE_BEGIN(evas_object_map_cow, obj->map, Evas_Object_Map_Data, map_write)
{
map_write->surface = ENFN->image_dirty_region
(ENC, map_write->surface,
0, 0, map_write->surface_w, map_write->surface_h);
map_write->cur.valid_map = EINA_TRUE;
}
EINA_COW_WRITE_END(evas_object_map_cow, obj->map, map_write);
}
/* duplicate context and reset clip */
// FIXME: Shouldn't we use a new, clean context?
ctx = ENFN->context_dup(ENC, context);
ENFN->context_clip_unset(ENC, ctx);
//ENFN->context_multiplier_unset(ENC, ctx); // this probably should be here, too
if (obj->map->surface)
{
Evas_Object_Protected_Data *mask = obj->clip.mask;
if (obj->cur->clipper)
{
evas_object_clip_recalc(obj);
if (obj->is_smart)
{
EINA_COW_STATE_WRITE_BEGIN(obj, state_write, cur)
{
state_write->cache.clip.dirty = EINA_TRUE;
}
EINA_COW_STATE_WRITE_END(obj, state_write, cur);
}
_evas_render_mapped_context_clip_set(evas, eo_obj, obj, ctx,
proxy_render_data, off_x,
off_y);
/* Clipper masks */
_evas_render_mapped_mask(evas, obj, mask, proxy_render_data, output, ctx, off_x, off_y, level, do_async);
}
if (obj->cur->cache.clip.visible || !proxy_src_clip)
{
if (!mapped)
{
ENFN->context_clip_clip(ENC, ctx, ecx, ecy, ecw, ech);
if (!_is_obj_in_framespace(obj, evas))
{
_evas_render_framespace_context_clip_clip
(evas, ctx, off_x - evas->framespace.x, off_y - evas->framespace.y);
}
}
ENFN->context_multiplier_unset(ENC, ctx);
ENFN->context_render_op_set(ENC, ctx, obj->cur->render_op);
#ifdef REND_DBG
int _c, _cx, _cy, _cw, _ch;
_c = ENFN->context_clip_get(ENC, ctx, &_cx, &_cy, &_cw, &_ch);
RD(level, " draw image map(clip: [%d] %d,%d %dx%d)\n", _c, _cx, _cy, _cw, _ch);
#endif
evas_draw_image_map_async_check
(obj, ENC, output, ctx, surface,
obj->map->surface, obj->map->spans,
obj->map->cur.map->smooth, 0, do_async);
}
}
ENFN->context_free(ENC, ctx);
// FIXME: needs to cache these maps and
// keep them only rendering updates
}
else // not "has map"
{
ctx = ENFN->context_dup(ENC, context);
if (mapped)
{
RD(level, " child of mapped obj\n");
if (obj->is_smart)
{
/* Clipper masks */
if (obj->cur->clipper && (mapped > 1) &&
_evas_render_object_is_mask(obj->cur->clipper))
_evas_render_mapped_mask(evas, obj, obj->cur->clipper, proxy_render_data, output, ctx, off_x, off_y, level, do_async);
else if (!proxy_src_clip && proxy_render_data)
{
if (!_proxy_context_clip(evas, ctx, proxy_render_data, obj, off_x, off_y))
goto on_empty_clip;
}
#ifdef REND_DBG
int _c, _cx, _cy, _cw, _ch;
_c = ENFN->context_clip_get(ENC, ctx, &_cx, &_cy, &_cw, &_ch);
RD(level, " draw smart children(clip: [%d] %d,%d %dx%d)\n",
_c, _cx, _cy, _cw, _ch);
#endif
EINA_INLIST_FOREACH
(evas_object_smart_members_get_direct(eo_obj), obj2)
{
/* skip proxy object if its source is its smart parent.
who makes this relation? a proxy object working for
a smart object to set a filter program. the proxy
object should be a member of smart object to sync
stacking changes. */
if (obj2->is_filter_object) continue;
clean_them |= evas_render_mapped(evas, obj2->object,
obj2, ctx,
output, surface,
off_x, off_y, mapped + 1,
ecx, ecy, ecw, ech,
proxy_render_data,
level + 1, do_async);
/* We aren't sure this object will be rendered by
normal(not proxy) drawing after, we reset this
only in case of normal drawing. For optmizing,
push this object in an array then reset them
in the end of the rendering.*/
if (!proxy_render_data)
evas_object_change_reset(obj2);
}
}
else
{
const Eina_Rectangle *clip = &obj->cur->geometry;
ENFN->context_clip_clip(ENC, ctx, clip->x + off_x, clip->y + off_y, clip->w, clip->h);
if (obj->cur->clipper && (mapped > 1))
{
Evas_Object_Protected_Data *mask = obj->clip.mask;
if (obj->mask->surface != surface)
{
if (proxy_src_clip)
{
if ((_evas_render_has_map(obj) && !_evas_render_can_map(obj)) ||
_evas_render_object_is_mask(obj->cur->clipper))
evas_object_clip_recalc(obj);
_evas_render_mapped_context_clip_set(evas, eo_obj, obj, ctx,
proxy_render_data,
off_x, off_y);
}
else if (proxy_render_data)
{
if (!_proxy_context_clip(evas, ctx, proxy_render_data, obj, off_x, off_y))
goto on_empty_clip;
}
}
else
{
// rendering a mask in its own surface:
// we want to render it fully and clip only at
// clippee (maskee) render time
RD(level, " draw mask\n");
}
/* Clipper masks */
_evas_render_mapped_mask(evas, obj, mask, proxy_render_data, output, ctx, off_x, off_y, level, do_async);
}
#ifdef REND_DBG
int _c, _cx, _cy, _cw, _ch;
_c = ENFN->context_clip_get(ENC, ctx, &_cx, &_cy, &_cw, &_ch);
RD(level, " render(clip: [%d] %d,%d %dx%d)\n", _c, _cx, _cy, _cw, _ch);
#endif
obj->func->render(eo_obj, obj, obj->private_data,
ENC, output, ctx, surface, off_x, off_y, do_async);
}
}
else if (!obj->is_smart)
{
if (obj->cur->clipper)
{
Evas_Object_Protected_Data *clipper = obj->cur->clipper;
int x, y, w, h;
if (proxy_src_clip)
{
if ((_evas_render_has_map(obj) && !_evas_render_can_map(obj)) ||
_evas_render_object_is_mask(obj->cur->clipper))
evas_object_clip_recalc(obj);
x = obj->cur->cache.clip.x;
y = obj->cur->cache.clip.y;
w = obj->cur->cache.clip.w;
h = obj->cur->cache.clip.h;
RECTS_CLIP_TO_RECT(x, y, w, h,
clipper->cur->cache.clip.x,
clipper->cur->cache.clip.y,
clipper->cur->cache.clip.w,
clipper->cur->cache.clip.h);
ENFN->context_clip_set(ENC, ctx, x + off_x, y + off_y, w, h);
if (!_is_obj_in_framespace(obj, evas))
{
_evas_render_framespace_context_clip_clip
(evas, ctx, off_x - evas->framespace.x, off_y - evas->framespace.y);
}
ENFN->context_clip_clip(ENC, ctx, ecx, ecy, ecw, ech);
}
else if (proxy_render_data)
{
if (!_proxy_context_clip(evas, ctx, proxy_render_data, obj, off_x, off_y))
goto on_empty_clip;
}
}
else if (!_is_obj_in_framespace(obj, evas))
{
_evas_render_framespace_context_clip_clip
(evas, ctx, off_x - evas->framespace.x, off_y - evas->framespace.y);
}
#ifdef REND_DBG
int _c, _cx, _cy, _cw, _ch;
_c = ENFN->context_clip_get(ENC, context, &_cx, &_cy, &_cw, &_ch);
RD(level, " draw normal obj: render(clip: [%d] %d,%d %dx%d)\n", _c, _cx, _cy, _cw, _ch);
#endif
obj->func->render(eo_obj, obj, obj->private_data,
ENC, output, ctx, surface,
off_x, off_y, do_async);
}
if (obj->changed_map) clean_them = EINA_TRUE;
on_empty_clip:
ENFN->context_free(ENC, ctx);
}
end:
RD(level, "}\n");
obj->pre_render_done = was_pre_render_done;
eina_evlog("-render_object", eo_obj, 0.0, NULL);
return clean_them;
}
/*
* Render the source object when a proxy is set.
* Used to force a draw if necessary, else just makes sure it's available.
* Called from: image objects and text with filters.
*/
void
evas_render_proxy_subrender(Evas *eo_e, void *output, Evas_Object *eo_source, Evas_Object *eo_proxy,
Evas_Object_Protected_Data *proxy_obj,
Eina_Bool source_clip, Eina_Bool do_async)
{
Evas_Public_Data *evas = efl_data_scope_get(eo_e, EVAS_CANVAS_CLASS);
Evas_Object_Protected_Data *source, *proxy;
int level = 1;
void *ctx;
int w, h, off_x = 0, off_y = 0;
Eina_Rectangle lr = {0, 0, 0, 0};
#ifdef REND_DBG
level = __RD_level;
#endif
if (!eo_source) return;
eina_evlog("+proxy_subrender", eo_proxy, 0.0, NULL);
source = efl_data_scope_get(eo_source, EFL_CANVAS_OBJECT_CLASS);
proxy = efl_data_scope_get(eo_proxy, EFL_CANVAS_OBJECT_CLASS);
evas_object_image_load_region_get(eo_proxy, &lr.x, &lr.y, &lr.w, &lr.h);
if (lr.w > 0 && lr.h > 0)
{
w = lr.w;
h = lr.h;
off_x = -lr.x;
off_y = -lr.y;
}
else if (proxy->proxy->proxies || (!proxy->cur->clipper) || (!proxy->cur->has_fixed_size))
{
/* make full surface available if this proxy is being sampled from */
w = source->cur->geometry.w;
h = source->cur->geometry.h;
}
else
{
Eina_Rectangle clip = _evas_render_smallest_static_clipped_geometry_get(proxy->cur);
/* nothing is using this proxy, so the proxy render surface can be sized
* to fit exactly the proxy object's render size, and the proxy render will
* naturally be clipped to this geometry
*/
w = clip.w;
h = clip.h;
/* set the render offset for the proxy offset based on the geometry which will
* be visible on the proxy surface after clipping
*/
off_x = proxy->cur->geometry.x - clip.x;
off_y = proxy->cur->geometry.y - clip.y;
}
RD(level, " proxy_subrender(source: %s, proxy: %s, %dx%d)\n",
RDNAME(source), RDNAME(proxy_obj), w, h);
EINA_COW_WRITE_BEGIN(evas_object_proxy_cow, source->proxy,
Evas_Object_Proxy_Data, proxy_write)
{
proxy_write->redraw = EINA_FALSE;
/* We need to redraw surface then */
if ((proxy_write->surface) &&
((proxy_write->w != w) || (proxy_write->h != h)))
{
RD(level, " free surface: %p\n", proxy_write->surface);
ENFN->image_free(ENC, proxy_write->surface);
proxy_write->surface = NULL;
}
/* FIXME: Hardcoded alpha 'on' */
/* FIXME (cont): Should see if the object has alpha */
if (!proxy_write->surface)
{
if ((w < 1) || (h < 1)) goto end;
proxy_write->surface = ENFN->image_map_surface_new(ENC, w, h, 1);
RD(level, " created surface: %p %dx%d\n", proxy_write->surface, w, h);
if (!proxy_write->surface) goto end;
proxy_write->w = w;
proxy_write->h = h;
}
eina_evlog("+proxy_fill", eo_proxy, 0.0, NULL);
ctx = ENFN->context_new(ENC);
ENFN->context_color_set(ENC, ctx, 0, 0,0, 0);
ENFN->context_render_op_set(ENC, ctx, EVAS_RENDER_COPY);
ENFN->rectangle_draw(ENC, output, ctx, proxy_write->surface, 0, 0, w, h, do_async);
ENFN->context_free(ENC, ctx);
eina_evlog("-proxy_fill", eo_proxy, 0.0, NULL);
Evas_Proxy_Render_Data proxy_render_data = {
.eo_proxy = eo_proxy,
.proxy_obj = proxy_obj,
.eo_src = eo_source,
.src_obj = source,
.source_clip = source_clip
};
/* protect changes to the objects' cache.clip */
evas_event_freeze(evas->evas);
ctx = ENFN->context_new(ENC);
evas_render_mapped(evas, eo_source, source, ctx,
output, proxy_write->surface,
-source->cur->geometry.x + off_x,
-source->cur->geometry.y + off_y,
level + 1, 0, 0, evas->output.w, evas->output.h,
&proxy_render_data, level + 1, do_async);
ENFN->context_free(ENC, ctx);
proxy_write->surface = ENFN->image_dirty_region(ENC, proxy_write->surface, 0, 0, w, h);
/* restore previous state */
evas_event_thaw(evas->evas);
}
end:
EINA_COW_WRITE_END(evas_object_proxy_cow, source->proxy, proxy_write);
eina_evlog("-proxy_subrender", eo_proxy, 0.0, NULL);
}
/* @internal
* Synchronously render a mask image (or smart object) into a surface.
* In SW the target surface will be ALPHA only (GRY8), after conversion.
* In GL the target surface will be RGBA for now. TODO: Find out how to
* render GL to alpha, if that's possible.
*/
static void
evas_render_mask_subrender(Evas_Public_Data *evas,
void *output,
Evas_Object_Protected_Data *mask,
Evas_Object_Protected_Data *prev_mask,
int level, Eina_Bool do_async)
{
int x, y, w, h, r, g, b, a, cr, cg, cb, ca;
Eina_Bool is_image, done = EINA_FALSE, restore_state = EINA_FALSE;
void *ctx;
if (!mask) return;
eina_evlog("+mask_subrender", mask->object, 0.0, NULL);
RD(level, "evas_render_mask_subrender(mask: %s, prev: %s, %s)\n",
RDNAME(mask), RDNAME(prev_mask), do_async ? "async" : "sync");
is_image = efl_isa(mask->object, EFL_CANVAS_IMAGE_INTERNAL_CLASS);
x = mask->cur->geometry.x;
y = mask->cur->geometry.y;
w = mask->cur->geometry.w;
h = mask->cur->geometry.h;
if (mask->cur->clipper && mask->cur->has_fixed_size)
{
Eina_Rectangle clip = _evas_render_smallest_static_clipped_geometry_get(mask->cur);
x = clip.x;
y = clip.y;
w = clip.w;
h = clip.h;
}
r = mask->cur->color.r;
g = mask->cur->color.g;
b = mask->cur->color.b;
a = mask->cur->color.a;
cr = mask->cur->cache.clip.r;
cg = mask->cur->cache.clip.g;
cb = mask->cur->cache.clip.b;
ca = mask->cur->cache.clip.a;
if ((r != 255) || (g != 255) || (b != 255) || (a != 255) ||
(cr != 255) || (cg != 255) || (cb != 255) || (ca != 255))
{
restore_state = EINA_TRUE;
EINA_COW_STATE_WRITE_BEGIN(mask, state_write, cur)
{
state_write->color.r = 255;
state_write->color.g = 255;
state_write->color.b = 255;
state_write->color.a = 255;
state_write->cache.clip.r = 255;
state_write->cache.clip.g = 255;
state_write->cache.clip.b = 255;
state_write->cache.clip.a = 255;
}
EINA_COW_STATE_WRITE_END(mask, state_write, cur);
}
// For SW engine, we want sync render of masks to be able to convert the
// surface from RGBA to Alpha. See below. (this logic could be improved)
if (!ENFN->gl_surface_read_pixels)
do_async = EINA_FALSE;
if (prev_mask == mask)
prev_mask = NULL;
if (prev_mask)
{
if (!prev_mask->mask->is_mask)
{
ERR("Passed invalid mask that is not a mask");
prev_mask = NULL;
}
else if (!prev_mask->mask->surface)
{
// Note: This is preventive code. Never seen it happen.
WRN("Mask render order may be invalid");
evas_render_mask_subrender(evas, output, prev_mask, prev_mask->clip.prev_mask, level + 1, do_async);
}
}
EINA_COW_WRITE_BEGIN(evas_object_mask_cow, mask->mask, Evas_Object_Mask_Data, mdata)
mdata->redraw = EINA_FALSE;
if (is_image && ENFN->image_scaled_update)
{
Eina_Bool filled = EINA_FALSE, border = EINA_FALSE;
int bl = 0, br = 0, bt = 0, bb = 0;
if (evas_object_image_filled_get(mask->object))
filled = EINA_TRUE;
else
{
int fx, fy, fw, fh;
evas_object_image_fill_get(mask->object, &fx, &fy, &fw, &fh);
if ((fx == 0) && (fy == 0) && (fw == mask->cur->geometry.w) && (fh == mask->cur->geometry.h))
filled = EINA_TRUE;
}
evas_object_image_border_get(mask->object, &bl, &br, &bt, &bb);
if (bl || br || bt || bb)
border = EINA_TRUE;
if (!border && filled && !prev_mask && mask->func->engine_data_get)
{
eina_evlog("+mask_scaled_update", mask->object, 0.0, NULL);
/* Fast path (for GL) that avoids creating a map surface, render the
* scaled image in it, when the shaders can just scale on the fly. */
Eina_Bool smooth = evas_object_image_smooth_scale_get(mask->object);
void *original = mask->func->engine_data_get(mask->object);
void *scaled = ENFN->image_scaled_update
(ENC, mdata->surface, original, w, h, smooth, EVAS_COLORSPACE_GRY8);
if (scaled)
{
done = EINA_TRUE;
mdata->surface = scaled;
mdata->w = w;
mdata->h = h;
mdata->is_alpha = (ENFN->image_colorspace_get(ENC, scaled) == EVAS_COLORSPACE_GRY8);
mdata->is_scaled = EINA_TRUE;
}
eina_evlog("-mask_scaled_update", mask->object, 0.0, NULL);
}
}
if (!done)
{
/* delete render surface if changed or if already alpha
* (we don't know how to render objects to alpha) */
if (mdata->surface && ((w != mdata->w) || (h != mdata->h) || mdata->is_alpha || mdata->is_scaled))
{
ENFN->image_free(ENC, mdata->surface);
mdata->surface = NULL;
}
/* create new RGBA render surface if needed */
if (!mdata->surface)
{
eina_evlog("+mask_surface_new", mask->object, 0.0, NULL);
mdata->surface = ENFN->image_map_surface_new(ENC, w, h, EINA_TRUE);
eina_evlog("-mask_surface_new", mask->object, 0.0, NULL);
if (!mdata->surface) goto end;
mdata->is_alpha = EINA_FALSE;
mdata->is_scaled = EINA_FALSE;
mdata->w = w;
mdata->h = h;
}
/* Clear surface with transparency */
eina_evlog("+mask_rect_clear", mask->object, 0.0, NULL);
ctx = ENFN->context_new(ENC);
ENFN->context_color_set(ENC, ctx, 0, 0, 0, 0);
ENFN->context_render_op_set(ENC, ctx, EVAS_RENDER_COPY);
ENFN->rectangle_draw(ENC, output, ctx, mdata->surface, 0, 0, w, h, do_async);
ENFN->context_free(ENC, ctx);
eina_evlog("-mask_rect_clear", mask->object, 0.0, NULL);
/* Render mask to RGBA surface */
ctx = ENFN->context_new(ENC);
if (prev_mask)
{
Eina_Rectangle pclip = _evas_render_smallest_static_clipped_geometry_get(prev_mask->cur);
ENFN->context_clip_image_set(ENC, ctx,
prev_mask->mask->surface,
pclip.x - x,
pclip.y - y,
evas, do_async);
ENFN->context_clip_set(ENC, ctx,
pclip.x - x,
pclip.y - y,
pclip.w,
pclip.h);
}
if (EINA_LIKELY(!mask->is_smart))
{
mask->func->render(mask->object, mask, mask->private_data,
ENC, output, ctx, mdata->surface, -x, -y, do_async);
}
else
{
// Unreachable code until we implement support for smart masks
evas_render_mapped(evas, mask->object, mask, ctx,
output, mdata->surface,
-x, -y, 2, 0, 0, evas->output.w, evas->output.h,
NULL, level, do_async);
}
ENFN->context_free(ENC, ctx);
/* BEGIN HACK */
/* Now we want to convert this RGBA surface to Alpha.
* NOTE: So, this is not going to work with the GL engine but only with
* the SW engine. Here's the detection hack:
* FIXME: If you know of a way to support rendering to GL_ALPHA in GL,
* then we should render directly to an ALPHA surface. A priori,
* GLES FBO does not support this.
*/
if (!ENFN->gl_surface_read_pixels)
{
RGBA_Image *alpha_surface;
DATA32 *rgba;
DATA8* alpha;
eina_evlog("+mask_new_cpy_data", mask->object, 0.0, NULL);
alpha_surface = ENFN->image_new_from_copied_data
(ENC, w, h, NULL, EINA_TRUE, EVAS_COLORSPACE_GRY8);
eina_evlog("-mask_new_cpy_data", mask->object, 0.0, NULL);
if (!alpha_surface) goto end;
eina_evlog("+mask_cpy_data", mask->object, 0.0, NULL);
/* Copy alpha channel */
rgba = ((RGBA_Image *) mdata->surface)->image.data;
alpha = alpha_surface->image.data8;
for (y = h; y; --y)
for (x = w; x; --x, alpha++, rgba++)
*alpha = (DATA8) A_VAL(rgba);
eina_evlog("-mask_cpy_data", mask->object, 0.0, NULL);
/* Now we can drop the original surface */
ENFN->image_free(ENC, mdata->surface);
mdata->surface = alpha_surface;
mdata->is_alpha = EINA_TRUE;
}
/* END OF HACK */
}
mdata->surface = ENFN->image_dirty_region(ENC, mdata->surface, 0, 0, w, h);
end:
EINA_COW_WRITE_END(evas_object_mask_cow, mask->mask, mdata);
if (restore_state)
{
EINA_COW_STATE_WRITE_BEGIN(mask, state_write, cur)
{
state_write->color.r = r;
state_write->color.g = g;
state_write->color.b = b;
state_write->color.a = a;
state_write->cache.clip.r = cr;
state_write->cache.clip.g = cg;
state_write->cache.clip.b = cb;
state_write->cache.clip.a = ca;
}
EINA_COW_STATE_WRITE_END(mask, state_write, cur);
}
eina_evlog("-mask_subrender", mask->object, 0.0, NULL);
}
static void
_evas_render_cutout_add(Evas_Public_Data *evas, void *context,
Evas_Object_Protected_Data *obj, int off_x, int off_y,
Cutout_Margin *cutout_margin)
{
Evas_Coord cox = 0, coy = 0, cow = 0, coh = 0;
if (evas_object_is_source_invisible(obj->object, obj)) return;
if (evas_object_is_opaque(obj))
{
cox = obj->cur->cache.clip.x;
coy = obj->cur->cache.clip.y;
cow = obj->cur->cache.clip.w;
coh = obj->cur->cache.clip.h;
if ((obj->map->cur.map) && (obj->map->cur.usemap))
{
Evas_Object_Protected_Data *oo;
for (oo = obj; oo->cur->clipper; oo = oo->cur->clipper)
{
if ((oo->cur->clipper->map->cur.map_parent
!= oo->map->cur.map_parent) &&
(!((oo->map->cur.map) && (oo->map->cur.usemap))))
break;
RECTS_CLIP_TO_RECT(cox, coy, cow, coh,
oo->cur->geometry.x,
oo->cur->geometry.y,
oo->cur->geometry.w,
oo->cur->geometry.h);
if ((cow <= 0) || (coh <= 0)) return;
}
}
}
else if (obj->func->get_opaque_rect)
{
obj->func->get_opaque_rect(obj->object, obj, obj->private_data, &cox, &coy, &cow, &coh);
if ((cow <= 0) || (coh <= 0)) return;
RECTS_CLIP_TO_RECT(cox, coy, cow, coh,
obj->cur->cache.clip.x, obj->cur->cache.clip.y,
obj->cur->cache.clip.w, obj->cur->cache.clip.h);
}
else return;
if (!_is_obj_in_framespace(obj, evas))
{
int fw, fh;
fw = evas->viewport.w - evas->framespace.w;
fh = evas->viewport.h - evas->framespace.h;
RECTS_CLIP_TO_RECT(cox, coy, cow, coh,
0, 0, fw, fh);
}
if (cutout_margin)
{
cox += cutout_margin->l;
coy += cutout_margin->t;
cow -= cutout_margin->l + cutout_margin->r;
coh -= cutout_margin->t + cutout_margin->b;
}
if ((cow <= 0) || (coh <= 0)) return;
ENFN->context_cutout_add(ENC, context, cox + off_x, coy + off_y, cow, coh);
}
void
evas_render_rendering_wait(Evas_Public_Data *evas)
{
while (evas->rendering) evas_async_events_process_blocking();
}
/*
* Syncs ALL async rendering canvases. Must be called in the main thread.
*/
void
evas_all_sync(void)
{
Evas_Public_Data *evas;
if (!_rendering_evases) return;
evas = eina_list_data_get(eina_list_last(_rendering_evases));
evas_render_rendering_wait(evas);
assert(_rendering_evases == NULL);
}
static Eina_Bool
_drop_scie_ref(const void *container EINA_UNUSED, void *data, void *fdata EINA_UNUSED)
{
evas_common_rgba_image_scalecache_item_unref(data);
return EINA_TRUE;
}
static Eina_Bool
_drop_image_cache_ref(const void *container EINA_UNUSED, void *data, void *fdata EINA_UNUSED)
{
evas_cache_image_drop((Image_Entry *)data);
return EINA_TRUE;
}
static void
_cb_always_call(Evas *eo_e, Evas_Public_Data *e, Evas_Callback_Type type, void *event_info)
{
int freeze_num = 0, i;
switch (type)
{
case EVAS_CALLBACK_RENDER_PRE:
if (!e->cb_render_pre) return;
break;
case EVAS_CALLBACK_RENDER_POST:
if (!e->cb_render_post) return;
break;
case EVAS_CALLBACK_RENDER_FLUSH_PRE:
if (!e->cb_render_flush_pre) return;
break;
case EVAS_CALLBACK_RENDER_FLUSH_POST:
if (!e->cb_render_flush_post) return;
break;
default: break;
}
freeze_num = efl_event_freeze_count_get(eo_e);
for (i = 0; i < freeze_num; i++) efl_event_thaw(eo_e);
evas_event_callback_call(eo_e, type, event_info);
for (i = 0; i < freeze_num; i++) efl_event_freeze(eo_e);
}
#ifndef INLINE_ACTIVE_GEOM
static inline Eina_Bool
_is_obj_in_rect(Evas_Object *eo_obj, Evas_Object_Protected_Data *obj,
int x, int y, int w, int h)
{
if (obj->is_smart && !_evas_render_has_map(obj))
{
Eina_Rectangle rect;
evas_object_smart_bounding_box_get(obj, &rect, NULL);
if (RECTS_INTERSECT(x, y, w, h, rect.x, rect.y, rect.w, rect.h))
return EINA_TRUE;
}
else
{
// FIXME: handle multiple output
if (evas_object_is_in_output_rect(eo_obj, obj, x, y, w, h))
return EINA_TRUE;
}
return EINA_FALSE;
}
#endif
static void
_snapshot_redraw_update(Evas_Public_Data *evas, Evas_Object_Protected_Data *snap)
{
Eina_Bool above = EINA_FALSE, add_rect = EINA_FALSE, need_redraw = EINA_FALSE;
const int x = snap->cur->geometry.x;
const int y = snap->cur->geometry.y;
const int w = snap->cur->geometry.w;
const int h = snap->cur->geometry.h;
Evas_Object_Protected_Data *obj;
Evas_Active_Entry *ent;
Eina_Rectangle snap_clip;
Eina_Tiler *tiler = NULL;
void *surface;
// FIXME: Tiler should be inside snapshot data
// TODO: Also list redraw regions (partial updates)
if (!evas_object_is_visible(snap)) return;
evas_object_clip_recalc(snap);
snap_clip.x = snap->cur->cache.clip.x - x;
snap_clip.y = snap->cur->cache.clip.y - y;
snap_clip.w = snap->cur->cache.clip.w;
snap_clip.h = snap->cur->cache.clip.h;
surface = _evas_object_image_surface_get(snap, EINA_FALSE);
if (!surface) need_redraw = EINA_TRUE;
if (snap->changed) add_rect = EINA_TRUE;
if (!add_rect && !need_redraw)
{
// Objects below snapshot
EINA_INARRAY_FOREACH(&evas->active_objects, ent)
{
obj = ent->obj;
if (obj == snap) break;
if (!obj->is_smart && obj->changed &&
evas_object_is_in_output_rect(obj->object, obj, x, y, w, h))
need_redraw = EINA_TRUE;
}
}
if (snap->snapshot_no_obscure || _evas_render_has_map(snap))
goto skip_obscures;
tiler = eina_tiler_new(w, h);
eina_tiler_tile_size_set(tiler, 1, 1);
eina_tiler_strict_set(tiler, EINA_TRUE);
// Opaque objects above snapshot
for (unsigned i = 0; i < evas->obscuring_objects.count; i++)
{
obj = eina_array_data_get(&evas->obscuring_objects, i);
if (!above)
{
if (obj == snap) above = EINA_TRUE;
continue;
}
if (!obj->cur->snapshot || evas_object_is_opaque(obj))
{
const Eina_Rectangle cur = {
obj->cur->cache.clip.x - x, obj->cur->cache.clip.y - y,
obj->cur->cache.clip.w, obj->cur->cache.clip.h
};
if (_evas_eina_rectangle_inside(&cur, &snap_clip))
goto end; // Totally hidden
eina_tiler_rect_add(tiler, &cur);
}
}
skip_obscures:
need_redraw |= _evas_filter_obscured_regions_set(snap, tiler);
if (memcmp(&snap->cur->geometry, &snap->prev->geometry, sizeof(snap->cur->geometry)))
need_redraw = EINA_TRUE;
snap->snapshot_needs_redraw |= need_redraw;
if (add_rect || need_redraw)
{
// FIXME: Only add necessary rects (if object itself hasn't changed)
// FIXME: handle multiple output
ENFN->output_redraws_rect_add(ENC, x, y, w, h);
}
end:
eina_tiler_free(tiler);
}
static void
evas_render_pre(Evas *eo_e, Evas_Public_Data *evas)
{
Eo *eo_obj;
if (!evas->finalize_objects) return; // avoid evlog
// Finalize EO objects now
eina_evlog("+render_pre_objects_finalize", eo_e, 0.0, NULL);
EINA_LIST_FREE(evas->finalize_objects, eo_obj)
{
Evas_Object_Protected_Data *obj;
obj = efl_data_scope_safe_get(eo_obj, EFL_CANVAS_OBJECT_CLASS);
if (!EVAS_OBJECT_DATA_ALIVE(obj)) continue;
obj->legacy.finalized = EINA_TRUE;
if (!obj->legacy.visible_set)
efl_gfx_entity_visible_set(eo_obj, EINA_TRUE);
// weight should be set during finalize()
if (EINA_UNLIKELY(!obj->legacy.weight_set))
efl_gfx_hint_weight_set(eo_obj, 1.0, 1.0);
}
eina_evlog("-render_pre_objects_finalize", eo_e, 0.0, NULL);
}
EAPI void
evas_render_pending_objects_flush(Evas *eo_e)
{
Evas_Public_Data *evas = efl_data_scope_safe_get(eo_e, EVAS_CANVAS_CLASS);
EINA_SAFETY_ON_NULL_RETURN(evas);
evas_render_pre(eo_e, evas);
}
static Eina_Bool
evas_render_updates_internal_loop(Evas *eo_e, Evas_Public_Data *evas,
void *output, void *surface, void *context,
Evas_Object_Protected_Data *top,
int ux, int uy, int uw, int uh,
int cx, int cy, int cw, int ch,
int fx, int fy,
Eina_Bool skip_cutouts, Cutout_Margin *cutout_margin,
Eina_Bool alpha, Eina_Bool do_async,
unsigned int *offset, int level)
{
Evas_Object *eo_obj;
Evas_Object_Protected_Data *obj;
int off_x, off_y;
unsigned int i, j;
Eina_Bool clean_them = EINA_FALSE;
Eina_Bool above_top = EINA_FALSE;
eina_evlog("+render_setup", eo_e, 0.0, NULL);
RD(level, " [--- UPDATE %i %i %ix%i\n", ux, uy, uw, uh);
off_x = cx - ux;
off_y = cy - uy;
/* build obscuring objects list (in order from bottom to top) */
if (alpha)
{
ENFN->context_clip_set(ENC, context, ux + off_x, uy + off_y, uw, uh);
}
for (i = 0; i < evas->obscuring_objects.count; ++i)
{
obj = eina_array_data_get(&evas->obscuring_objects, i);
if (obj == top) above_top = EINA_TRUE;
if (evas_object_is_in_output_rect(obj->object, obj, ux - fx, uy - fy, uw, uh))
{
OBJ_ARRAY_PUSH(&evas->temporary_objects, obj);
if (above_top) continue;
if (obj->cur->snapshot && !evas_object_is_opaque(obj))
continue;
/* reset the background of the area if needed (using cutout and engine alpha flag to help) */
if (alpha && !skip_cutouts)
_evas_render_cutout_add(evas, context, obj, off_x + fx, off_y + fy, cutout_margin);
}
}
if (alpha)
{
ENFN->context_color_set(ENC, context, 0, 0, 0, 0);
ENFN->context_multiplier_unset(ENC, context);
ENFN->context_render_op_set(ENC, context, EVAS_RENDER_COPY);
ENFN->rectangle_draw(ENC, output, context, surface, cx, cy, cw, ch, do_async);
ENFN->context_cutout_clear(ENC, context);
ENFN->context_clip_unset(ENC, context);
}
eina_evlog("-render_setup", eo_e, 0.0, NULL);
eina_evlog("+render_objects", eo_e, 0.0, NULL);
/* render all object that intersect with rect */
for (i = 0; i < evas->active_objects.len; i++)
{
Evas_Active_Entry *ent = eina_inarray_nth(&evas->active_objects, i);
obj = ent->obj;
eo_obj = obj->object;
if (obj == top) break;
/* if it's in our outpout rect and it doesn't clip anything */
RD(level, " OBJ: %s %i %i %ix%i\n", RDNAME(obj), obj->cur->geometry.x, obj->cur->geometry.y, obj->cur->geometry.w, obj->cur->geometry.h);
if (
(!obj->clip.clipees) &&
(obj->cur->visible) &&
(obj->cur->cache.clip.visible) &&
#ifdef INLINE_ACTIVE_GEOM
RECTS_INTERSECT(ux - fx, uy - fy, uw, uh,
ent->rect.x, ent->rect.y,
ent->rect.w, ent->rect.h) &&
#else
(_is_obj_in_rect(eo_obj, obj, ux - fx, uy - fy, uw, uh)) &&
#endif
(!obj->delete_me) &&
((obj->cur->color.a > 0 || obj->cur->render_op != EVAS_RENDER_BLEND)))
{
int x, y, w, h;
RD(level, " DRAW (vis: %i, a: %i, clipees: %p)\n", obj->cur->visible, obj->cur->color.a, obj->clip.clipees);
if ((evas->temporary_objects.count > *offset) &&
(eina_array_data_get(&evas->temporary_objects, *offset) == obj))
(*offset)++;
x = cx; y = cy; w = cw; h = ch;
if (((w > 0) && (h > 0)) || (obj->is_smart))
{
Evas_Object_Protected_Data *mask;
if (!obj->is_smart)
{
int cfx, cfy;
if (!obj->is_frame)
{
cfx = obj->cur->cache.clip.x + off_x + fx;
cfy = obj->cur->cache.clip.y + off_y + fy;
}
else
{
cfx = obj->cur->cache.clip.x + off_x;
cfy = obj->cur->cache.clip.y + off_y;
}
RECTS_CLIP_TO_RECT(x, y, w, h, cfx, cfy,
obj->cur->cache.clip.w,
obj->cur->cache.clip.h);
}
ENFN->context_clip_set(ENC, context, x, y, w, h);
mask = obj->clip.mask;
if (mask)
{
Evas_Object_Protected_Data *prev_mask = obj->clip.prev_mask;
if (mask->mask->redraw || !mask->mask->surface)
evas_render_mask_subrender(obj->layer->evas, output, mask, prev_mask, 4, do_async);
if (mask->mask->surface)
{
Eina_Rectangle clip = _evas_render_smallest_static_clipped_geometry_get(mask->cur);
ENFN->context_clip_image_set(ENC, context,
mask->mask->surface,
clip.x + off_x,
clip.y + off_y,
evas, do_async);
}
}
eina_evlog("+cutouts_add", obj->object, 0.0, NULL);
above_top = EINA_FALSE;
#if 1 /* FIXME: this can slow things down... figure out optimum... coverage */
if (!skip_cutouts)
{
for (j = *offset; j < evas->temporary_objects.count; ++j)
{
Evas_Object_Protected_Data *obj2;
obj2 = eina_array_data_get(&evas->temporary_objects, j);
if (obj2 == top) above_top = EINA_TRUE;
if (obj2->cur->snapshot)
{
if (above_top) continue;
if (!evas_object_is_opaque(obj2)) continue;
}
#if 1
if (
RECTS_INTERSECT
(obj->cur->cache.clip.x, obj->cur->cache.clip.y,
obj->cur->cache.clip.w, obj->cur->cache.clip.h,
obj2->cur->cache.clip.x, obj2->cur->cache.clip.y,
obj2->cur->cache.clip.w, obj2->cur->cache.clip.h) &&
RECTS_INTERSECT
(obj2->cur->cache.clip.x, obj2->cur->cache.clip.y,
obj2->cur->cache.clip.w, obj2->cur->cache.clip.h,
ux, uy, uw, uh)
)
#endif
_evas_render_cutout_add(evas, context, obj2, off_x + fx, off_y + fy, cutout_margin);
}
}
#endif
ENFN->context_cutout_target(ENC, context,
off_x + obj->cur->cache.clip.x,
off_y + obj->cur->cache.clip.y,
obj->cur->cache.clip.w,
obj->cur->cache.clip.h);
eina_evlog("-cutouts_add", obj->object, 0.0, NULL);
clean_them |= evas_render_mapped(evas, eo_obj, obj, context,
output, surface,
off_x + fx, off_y + fy, 0,
cx, cy, cw, ch,
NULL, level + 3, do_async);
ENFN->context_cutout_clear(ENC, context);
if (mask) ENFN->context_clip_image_unset(ENC, context);
}
}
}
eina_evlog("-render_objects", eo_e, 0.0, NULL);
/* free obscuring objects list */
OBJS_ARRAY_CLEAN(&evas->temporary_objects);
#ifdef REND_DBG
if (top) RD(level, " ---] SNAPSHOT [obj:%s sfc:%p]\n", RDNAME(top), surface);
else RD(level, " ---]\n");
#endif
return clean_them;
}
typedef enum _Evas_3State
{
EVAS_3STATE_OUTSIDE,
EVAS_3STATE_INSIDE,
EVAS_3STATE_OVER
} Evas_3State;
static Evas_3State
_evas_overlay_output_find(Efl_Canvas_Output *output,
Evas_Object_Protected_Data *obj)
{
const Eina_Rectangle geometry = {
obj->cur->geometry.x,
obj->cur->geometry.y,
obj->cur->geometry.w,
obj->cur->geometry.h
};
Eina_Rectangle copy = geometry;
/* A video object can only be in one output at a time, check that first */
if (!eina_rectangle_intersection(&copy, &output->geometry))
return EVAS_3STATE_OUTSIDE;
if (memcmp(&copy, &geometry, sizeof (Eina_Rectangle)) != 0)
/* This means that it does intersect this output and another */
return EVAS_3STATE_OVER;
return EVAS_3STATE_INSIDE;
}
static void
_evas_planes(Evas_Public_Data *evas)
{
Evas_Active_Entry *ao;
/* check if individual image objects can be dropped into hardware planes */
if (ENFN->image_plane_assign)
EINA_INARRAY_FOREACH(&evas->active_objects, ao)
{
Evas_Object_Protected_Data *obj2;
Evas_Object *eo_obj2;
Efl_Canvas_Output *output;
Evas_3State state;
Eina_List *lo;
obj2 = ao->obj;
eo_obj2 = obj2->object;
if (!efl_isa(eo_obj2, EFL_CANVAS_IMAGE_INTERNAL_CLASS)) continue;
/* Find the output the object was in */
EINA_LIST_FOREACH(evas->outputs, lo, output)
{
if (!output->output) continue ;
if (!eina_list_data_find(output->planes, obj2)) continue;
_evas_object_image_plane_release(eo_obj2, obj2, output);
break;
}
if (evas_object_is_visible(obj2))
EINA_LIST_FOREACH(evas->outputs, lo, output)
{
/* A video object can only be in one output at a time, check that first */
state = _evas_overlay_output_find(output, obj2);
if (state == EVAS_3STATE_OUTSIDE) continue;
if (!_evas_render_can_use_overlay(evas, eo_obj2, output))
{
/* This may free up things temporarily allocated by
* _can_use_overlay() testing in the engine */
_evas_object_image_plane_release(eo_obj2, obj2, output);
} else break;
}
if (evas_object_plane_changed(obj2))
{
/* Since we're lifting this object out of the scene graph
* (or putting it back), we need to force redraw of the space
* under it.
*/
_evas_canvas_damage_rectangle_add(NULL, evas,
obj2->cur->geometry.x,
obj2->cur->geometry.y,
obj2->cur->geometry.w,
obj2->cur->geometry.h);
/* We also need to clean its previously drawn position
* but only if we're removing it */
if (evas_object_is_on_plane(obj2))
_evas_canvas_damage_rectangle_add(NULL, evas,
obj2->prev->geometry.x,
obj2->prev->geometry.y,
obj2->prev->geometry.w,
obj2->prev->geometry.h);
}
}
}
static Eina_Bool
evas_render_updates_internal(Evas *eo_e,
unsigned char make_updates,
unsigned char do_draw,
Eina_Bool do_async)
{
// FIXME: handle multiple output
Evas_Object *eo_obj;
Evas_Object_Protected_Data *obj;
Evas_Public_Data *evas, *e;
Efl_Canvas_Output *out;
Eina_List *ll, *l;
Eina_Bool clean_them = EINA_FALSE;
Eina_Bool rendering = EINA_FALSE;
Eina_Bool alpha;
Eina_Rectangle *r;
unsigned int i;
Phase1_Context p1ctx;
int redraw_all = 0;
Evas_Render_Mode render_mode = !do_async ?
EVAS_RENDER_MODE_SYNC :
EVAS_RENDER_MODE_ASYNC_INIT;
Eina_Bool haveup = EINA_FALSE;
static int show_update_boxes = -1;
MAGIC_CHECK(eo_e, Evas, MAGIC_EVAS);
return EINA_FALSE;
MAGIC_CHECK_END();
e = evas = efl_data_scope_get(eo_e, EVAS_CANVAS_CLASS);
if (e->inside_post_render) return EINA_FALSE;
if (!e->changed) return EINA_FALSE;
if (e->rendering)
{
if (do_async)
return EINA_FALSE;
else
{
WRN("Mixing render sync as already doing async "
"render! Syncing! e=%p [%s]", e,
e->engine.module->definition->name);
eina_evlog("+render_wait", eo_e, 0.0, NULL);
evas_render_rendering_wait(e);
eina_evlog("-render_wait", eo_e, 0.0, NULL);
}
}
#ifdef EVAS_RENDER_DEBUG_TIMING
double start_time = _time_get();
#endif
if (show_update_boxes == -1)
{
if (getenv("EVAS_RENDER_DEBUG_UPDATE_BOXES")) show_update_boxes = 1;
else show_update_boxes = 0;
}
evas_render_pre(eo_e, evas);
_evas_planes(e);
eina_evlog("+render_calc", eo_e, 0.0, NULL);
evas_call_smarts_calculate(eo_e);
eina_evlog("-render_calc", eo_e, 0.0, NULL);
RD(0, "[--- RENDER EVAS (size: %ix%i): %p (eo %p)\n", e->viewport.w, e->viewport.h, e, eo_e);
_cb_always_call(eo_e, e, EVAS_CALLBACK_RENDER_PRE, NULL);
/* Check if the modified object mean recalculating every thing */
if (!e->invalidate)
{
eina_evlog("+render_pending", eo_e, 0.0, NULL);
_evas_render_check_pending_objects(&e->pending_objects, eo_e, e);
eina_evlog("-render_pending", eo_e, 0.0, NULL);
}
/* phase 1. add extra updates for changed objects */
if (e->invalidate || e->render_objects.count <= 0)
{
eina_evlog("+render_phase1", eo_e, 0.0, NULL);
p1ctx.e = e;
p1ctx.active_objects = &e->active_objects;
p1ctx.restack_objects = &e->restack_objects;
p1ctx.delete_objects = &e->delete_objects;
p1ctx.render_objects = &e->render_objects;
p1ctx.snapshot_objects = &e->snapshot_objects;
p1ctx.redraw_all = redraw_all;
clean_them = _evas_render_phase1_process(&p1ctx);
redraw_all = p1ctx.redraw_all;
eina_evlog("-render_phase1", eo_e, 0.0, NULL);
}
eina_evlog("+render_phase1_direct", eo_e, 0.0, NULL);
/* phase 1.8. pre render for proxy */
_evas_render_phase1_direct(e, &e->active_objects, &e->restack_objects,
&e->delete_objects, &e->render_objects);
eina_evlog("-render_phase1_direct", eo_e, 0.0, NULL);
/* phase 2. force updates for restacks */
eina_evlog("+render_phase2", eo_e, 0.0, NULL);
for (i = 0; i < e->restack_objects.count; ++i)
{
obj = eina_array_data_get(&e->restack_objects, i);
if (_evas_render_object_is_mask(obj))
_evas_mask_redraw_set(e, obj);
obj->func->render_pre(obj->object, obj, obj->private_data);
_evas_render_prev_cur_clip_cache_add(e, obj);
}
OBJS_ARRAY_CLEAN(&e->restack_objects);
eina_evlog("-render_phase2", eo_e, 0.0, NULL);
/* phase 3. add exposes */
eina_evlog("+render_phase3", eo_e, 0.0, NULL);
EINA_LIST_FREE(e->damages, r)
{
ENFN->output_redraws_rect_add(ENC, r->x, r->y, r->w, r->h);
eina_rectangle_free(r);
}
eina_evlog("-render_phase3", eo_e, 0.0, NULL);
/* phase 4. framespace, output & viewport changes */
eina_evlog("+render_phase4", eo_e, 0.0, NULL);
if (e->viewport.changed)
{
ENFN->output_redraws_rect_add(ENC, 0, 0, e->output.w, e->output.h);
}
if ((e->output.w != e->viewport.w) || (e->output.h != e->viewport.h))
{
ERR("viewport size != output size!");
}
if (e->framespace.changed)
{
/* NB: If the framespace changes, we need to add a redraw rectangle
* which covers the Whole viewport. This is because 'framespace' is
* defined as "the space IN the viewport which is Occupied by the
* window frame" */
ENFN->output_redraws_rect_add(ENC,
e->viewport.x, e->viewport.y,
e->viewport.w, e->viewport.h);
}
if (redraw_all)
{
ENFN->output_redraws_rect_add(ENC, 0, 0, e->output.w, e->output.h);
}
eina_evlog("-render_phase4", eo_e, 0.0, NULL);
/* phase 5. add obscures */
eina_evlog("+render_phase5", eo_e, 0.0, NULL);
EINA_LIST_FOREACH(e->obscures, ll, r)
evas_render_update_del(e, r->x, r->y, r->w, r->h);
static int prepare = -1;
if (prepare == -1)
{
if (getenv("EVAS_PREPARE")) prepare = !!atoi(getenv("EVAS_PREPARE"));
else prepare = 1;
}
/* build obscure objects list of active objects that obscure as well
* as objects that may need data (image data loads, texture updates,
* pre-render buffers/fbo's etc.) that are not up to date yet */
for (i = 0; i < e->active_objects.len; i++)
{
Evas_Active_Entry *ent = eina_inarray_nth(&e->active_objects, i);
obj = ent->obj;
eo_obj = obj->object;
if (UNLIKELY(
(!obj->is_smart) &&
(!obj->clip.clipees) &&
(evas_object_is_opaque(obj) || obj->cur->snapshot ||
((obj->func->has_opaque_rect) &&
(obj->func->has_opaque_rect(eo_obj, obj, obj->private_data)))) &&
evas_object_is_visible(obj) &&
(!obj->mask->is_mask) && (!obj->clip.mask) &&
(!obj->delete_me)))
OBJ_ARRAY_PUSH(&e->obscuring_objects, obj);
if (prepare)
{
if (obj->func->render_prepare)
obj->func->render_prepare(eo_obj, obj, do_async);
}
}
if (!redraw_all)
{
for (i = 0; i < e->snapshot_objects.count; i++)
{
obj = eina_array_data_get(&e->snapshot_objects, i);
_snapshot_redraw_update(evas, obj);
}
}
eina_evlog("-render_phase5", eo_e, 0.0, NULL);
EINA_LIST_FOREACH(e->outputs, l, out)
{
// Avoid processing not ready output until they are
if (!out->output) continue;
// Locked output are output that should not yet be rendered
// because the tick/vsync for it doesn't allow it yet.
if (out->lock > 0) continue;
/* phase 6. Initialize output */
if (out->changed)
{
ENFN->output_resize(ENC, out->output,
out->geometry.w, out->geometry.h);
ENFN->output_redraws_rect_add(ENC,
out->geometry.x, out->geometry.y,
out->geometry.w, out->geometry.h);
out->changed = EINA_FALSE;
}
/* Define the output for Evas_GL operation */
if (ENFN->gl_output_set)
ENFN->gl_output_set(ENC, out->output);
/* phase 7. check if video surface should be inlined or stay in their hardware plane */
eina_evlog("+render_phase7", eo_e, 0.0, NULL);
alpha = ENFN->canvas_alpha_get(out->output);
EINA_LIST_FOREACH(e->video_objects, ll, eo_obj)
{
Evas_Object_Protected_Data *obj2;
Evas_3State state;
obj2 = efl_data_scope_get(eo_obj, EFL_CANVAS_OBJECT_CLASS);
state = _evas_overlay_output_find(out, obj2);
if (state == EVAS_3STATE_OUTSIDE) continue;
/* we need the surface to be transparent to display the underlying overlay */
if (state == EVAS_3STATE_INSIDE &&
alpha &&
_evas_render_can_use_overlay(e, eo_obj, out))
_evas_object_image_video_overlay_show(eo_obj);
else
_evas_object_image_video_overlay_hide(eo_obj);
}
eina_evlog("-render_phase7", eo_e, 0.0, NULL);
/* phase 8. go thru each update rect and render objects in it*/
eina_evlog("+render_phase8", eo_e, 0.0, NULL);
if (do_draw)
{
Render_Updates *ru;
void *surface;
int ux, uy, uw, uh;
int cx, cy, cw, ch;
unsigned int offset = 0;
int fx = e->framespace.x;
int fy = e->framespace.y;
int j;
DBG("Rendering output %p [%i, %i, %i, %i]\n", out,
out->geometry.x, out->geometry.y,
out->geometry.w, out->geometry.h);
if (do_async) _evas_render_busy_begin();
eina_evlog("+render_surface", eo_e, 0.0, NULL);
while ((surface =
ENFN->output_redraws_next_update_get
(ENC, out->output,
&ux, &uy, &uw, &uh,
&cx, &cy, &cw, &ch)))
{
void *ctx;
haveup = EINA_TRUE;
/* adjust the rendering rectangle to the output offset */
ux += out->geometry.x;
uy += out->geometry.y;
DBG("Surface %p offset: [%i, %i, %i, %i]\n", surface,
ux, uy, uw, uh);
/* phase 7.1 render every snapshot that needs to be updated
for this part of the screen */
eina_evlog("+render_snapshots", eo_e, 0.0, NULL);
for (j = e->snapshot_objects.count - 1; j >= 0; j--)
{
Evas_Object_Protected_Data *snap;
Eina_Rectangle output, cr, ur;
snap = eina_array_data_get(&e->snapshot_objects, j);
EINA_RECTANGLE_SET(&output,
snap->cur->geometry.x,
snap->cur->geometry.y,
snap->cur->geometry.w,
snap->cur->geometry.h);
EINA_RECTANGLE_SET(&ur, ux, uy, uw, uh);
// FIXME: We should render snapshots only once per frame,
// not once per update region per output !
if (snap->snapshot_needs_redraw &&
eina_rectangle_intersection(&ur, &output))
{
Cutout_Margin cm = {};
unsigned int restore_offset = offset;
Eina_Bool skip_cutouts = EINA_FALSE;
void *pseudo_canvas;
EINA_RECTANGLE_SET(&cr,
ur.x - output.x, ur.y - output.y,
ur.w, ur.h);
pseudo_canvas = _evas_object_image_surface_get(snap, EINA_TRUE);
// Get required margin for filters (eg. blur radius)
_evas_filter_radius_get(snap, &cm.l, &cm.r, &cm.t, &cm.b);
if (snap->map->cur.usemap || snap->proxy->proxies ||
snap->snapshot_no_obscure ||
((cm.l + cm.r) >= output.w) ||
((cm.t + cm.b) >= output.h))
skip_cutouts = EINA_TRUE;
RD(0, " SNAPSHOT %s [sfc:%p ur:%d,%d %dx%d]\n", RDNAME(snap), pseudo_canvas, ur.x, ur.y, ur.w, ur.h);
ctx = ENFN->context_new(ENC);
clean_them |= evas_render_updates_internal_loop(eo_e, e, out->output, pseudo_canvas, ctx,
snap,
ur.x, ur.y, ur.w, ur.h,
cr.x, cr.y, cr.w, cr.h,
fx, fy, skip_cutouts, &cm,
alpha, do_async,
&offset, 1);
ENFN->context_free(ENC, ctx);
offset = restore_offset;
}
}
eina_evlog("-render_snapshots", eo_e, 0.0, NULL);
eina_evlog("+render_update", eo_e, 0.0, NULL);
/* phase 7.2 render all the object on the target surface */
if ((do_async) || (make_updates))
{
ru = malloc(sizeof(*ru));
ru->surface = surface;
//XXX: need a way of reffing output surfaces
NEW_RECT(ru->area, ux, uy, uw, uh);
eina_spinlock_take(&(e->render.lock));
out->updates = eina_list_append(out->updates, ru);
eina_spinlock_release(&(e->render.lock));
}
ctx = ENFN->context_new(ENC);
clean_them |= evas_render_updates_internal_loop(eo_e, e, out->output, surface,
ctx, NULL,
ux, uy, uw, uh,
cx, cy, cw, ch,
fx, fy,
EINA_FALSE, NULL,
alpha, do_async,
&offset, 0);
ENFN->context_free(ENC, ctx);
eina_evlog("-render_update", eo_e, 0.0, NULL);
if (!do_async)
{
if (show_update_boxes == 1)
{
static int fn = 0;
void *ctx;
fn++;
ctx = ENFN->context_new(ENC);
ENFN->context_color_set
(ENC, ctx, fn & 0xff, 0x40, 0x20, 0xff);
ENFN->rectangle_draw(ENC, out->output,
ctx, surface,
ux - out->geometry.x, uy - out->geometry.y, uw, 1, do_async);
ENFN->rectangle_draw(ENC, out->output,
ctx, surface,
ux - out->geometry.x, uy - out->geometry.y + uh - 1, uw, 1, do_async);
ENFN->rectangle_draw(ENC, out->output,
ctx, surface,
ux - out->geometry.x, uy - out->geometry.y, 1, uh, do_async);
ENFN->rectangle_draw(ENC, out->output,
ctx, surface,
ux - out->geometry.x + uw - 1, uy - out->geometry.y, 1, uh, do_async);
ENFN->context_free(ENC, ctx);
}
eina_evlog("+render_push", eo_e, 0.0, NULL);
ENFN->output_redraws_next_update_push(ENC, out->output, surface,
ux - out->geometry.x, uy - out->geometry.y, uw, uh,
render_mode);
eina_evlog("-render_push", eo_e, 0.0, NULL);
}
}
eina_evlog("-render_surface", eo_e, 0.0, NULL);
}
else if (make_updates)
{
Render_Updates *ru;
void *surface;
int ux, uy, uw, uh;
int cx, cy, cw, ch;
while ((surface =
ENFN->output_redraws_next_update_get
(ENC, out->output,
&ux, &uy, &uw, &uh,
&cx, &cy, &cw, &ch)))
{
haveup = EINA_TRUE;
/* adjust the rendering rectangle to the output offset */
ux += out->geometry.x;
uy += out->geometry.y;
ru = malloc(sizeof(*ru));
ru->surface = surface;
//XXX: need a way of reffing output surfaces
NEW_RECT(ru->area, ux, uy, uw, uh);
eina_spinlock_take(&(e->render.lock));
out->updates = eina_list_append(out->updates, ru);
eina_spinlock_release(&(e->render.lock));
}
}
}
/* First process all output, then flush */
if (do_draw)
{
if (haveup)
{
if (do_async)
{
eina_evlog("+render_output_async_flush", eo_e, 0.0, NULL);
efl_ref(eo_e);
_rendering_evases = eina_list_prepend(_rendering_evases, e);
e->rendering = _rendering_evases;
_cb_always_call(eo_e, e, EVAS_CALLBACK_RENDER_FLUSH_PRE, NULL);
evas_thread_queue_flush((Evas_Thread_Command_Cb)evas_render_pipe_wakeup, e);
eina_evlog("-render_output_async_flush", eo_e, 0.0, NULL);
}
else
{
eina_evlog("+render_output_flush", eo_e, 0.0, NULL);
EINA_LIST_FOREACH(e->video_objects, ll, eo_obj)
{
_evas_object_image_video_overlay_do(eo_obj);
}
_cb_always_call(eo_e, e, EVAS_CALLBACK_RENDER_FLUSH_PRE, NULL);
EINA_LIST_FOREACH(e->outputs, l, out)
if (out->output)
ENFN->output_flush(ENC, out->output, EVAS_RENDER_MODE_SYNC);
_cb_always_call(eo_e, e, EVAS_CALLBACK_RENDER_FLUSH_POST, NULL);
_deferred_callbacks_process(eo_e, evas);
eina_evlog("-render_output_flush", eo_e, 0.0, NULL);
}
}
rendering = haveup;
}
eina_evlog("-render_phase8", eo_e, 0.0, NULL);
eina_evlog("+render_clear", eo_e, 0.0, NULL);
if (!do_async && rendering)
{
/* clear redraws */
EINA_LIST_FOREACH(e->outputs, l, out)
if (out->output)
ENFN->output_redraws_clear(ENC, out->output);
}
eina_evlog("-render_clear", eo_e, 0.0, NULL);
/* and do a post render pass */
eina_evlog("+render_post", eo_e, 0.0, NULL);
IFRD(e->active_objects.len, 0, " [--- POST RENDER\n");
for (i = 0; i < e->active_objects.len; i++)
{
Evas_Active_Entry *ent = eina_inarray_nth(&e->active_objects, i);
obj = ent->obj;
eo_obj = obj->object;
obj->pre_render_done = EINA_FALSE;
RD(0, " OBJ %s changed:%i do_draw:%i\n", RDNAME(obj), obj->changed, do_draw);
if ((clean_them) || (obj->changed && do_draw))
{
RD(0, " OBJ %s render_post()\n", RDNAME(obj));
obj->restack = EINA_FALSE;
evas_object_change_reset(obj);
obj->func->render_post(eo_obj, obj, obj->private_data);
}
}
for (i = 0; i < e->snapshot_objects.count; i++)
{
Evas_Object_Protected_Data *snap;
snap = eina_array_data_get(&e->snapshot_objects, i);
snap->snapshot_needs_redraw = EINA_FALSE;
snap->snapshot_no_obscure = EINA_FALSE;
}
eina_evlog("-render_post", eo_e, 0.0, NULL);
IFRD(e->active_objects.len, 0, " ---]\n");
/* Set back Evas_GL output to NULL */
/* if (ENFN->gl_output_set) */
/* ENFN->gl_output_set(ENC, NULL); */
/* free our obscuring object list */
OBJS_ARRAY_CLEAN(&e->obscuring_objects);
/* If some object are still marked as changed, do not remove
them from the pending list. */
eina_array_remove(&e->pending_objects, pending_change, NULL);
/* Reinsert parent of changed object in the pending changed state */
eina_evlog("+render_post_change", eo_e, 0.0, NULL);
for (i = 0; i < e->pending_objects.count; ++i)
{
obj = eina_array_data_get(&e->pending_objects, i);
if (obj->smart.parent)
{
/* these are all objects which could not be rendered because they were
* added to their smart parent in this cycle and the parent was not changed
*/
Evas_Object_Protected_Data *smart_parent;
smart_parent = efl_data_scope_get(obj->smart.parent,
EFL_CANVAS_OBJECT_CLASS);
evas_object_change(obj->smart.parent, smart_parent);
/* render cache must be invalidated to correctly render subobj on next pass */
evas_object_smart_render_cache_clear(obj->smart.parent);
}
obj->changed = EINA_TRUE;
}
eina_evlog("-render_post_change", eo_e, 0.0, NULL);
eina_evlog("+render_post_reset", eo_e, 0.0, NULL);
for (i = 0; i < e->render_objects.count; ++i)
{
obj = eina_array_data_get(&e->render_objects, i);
eo_obj = obj->object;
obj->pre_render_done = EINA_FALSE;
if ((obj->changed) && (do_draw))
{
obj->restack = EINA_FALSE;
evas_object_change_reset(obj);
obj->func->render_post(eo_obj, obj, obj->private_data);
}
}
eina_evlog("-render_post_reset", eo_e, 0.0, NULL);
for (i = 0; i < e->render_post_change_objects.count; ++i)
{
obj = eina_array_data_get(&e->render_post_change_objects, i);
eo_obj = obj->object;
evas_object_change(eo_obj, obj);
}
OBJS_ARRAY_CLEAN(&e->render_post_change_objects);
eina_evlog("+render_end", eo_e, 0.0, NULL);
e->changed = EINA_FALSE;
e->viewport.changed = EINA_FALSE;
e->framespace.changed = EINA_FALSE;
e->invalidate = EINA_FALSE;
// always clean... lots of mem waste!
/* If their are some object to restack or some object to delete,
* it's useless to keep the render object list around. */
if (clean_them)
{
eina_inarray_flush(&e->active_objects);
OBJS_ARRAY_CLEAN(&e->render_objects);
OBJS_ARRAY_CLEAN(&e->restack_objects);
OBJS_ARRAY_CLEAN(&e->temporary_objects);
OBJS_ARRAY_CLEAN(&e->snapshot_objects);
eina_array_foreach(&e->clip_changes, _evas_clip_changes_free, NULL);
eina_array_clean(&e->clip_changes);
e->invalidate = EINA_TRUE;
}
/* delete all objects flagged for deletion now */
for (i = 0; i < e->delete_objects.count; ++i)
{
obj = eina_array_data_get(&e->delete_objects, i);
evas_object_free(obj, EINA_TRUE);
}
eina_array_clean(&e->delete_objects);
/* if we deleted no objects this frame or we deleted a lot (> 1024) then
* try and reset the deleted objects array to empty (no mem used) for
* efficiency */
if ((e->delete_objects.count == 0) || (e->delete_objects.count > 1024))
eina_array_flush(&e->delete_objects);
evas_module_clean();
/* Send a RENDER_POST when we are rendering synchronously or,
when do_async but no drawing. This gurantees pre-post pair. */
if (!do_async || !rendering)
{
Evas_Event_Render_Post post;
Eina_List *l1, *l2;
Render_Updates *ru;
post.updated_area = NULL;
if (haveup)
{
EINA_LIST_FOREACH(e->outputs, l1, out)
{
if (!out->output) continue ;
EINA_LIST_FOREACH(out->updates, l2, ru)
{
post.updated_area = eina_list_append(post.updated_area, ru->area);
//XXX: need a way of unreffing output surfaces
ru->surface = NULL;
}
}
}
eina_spinlock_take(&(e->render.lock));
e->inside_post_render = EINA_TRUE;
_cb_always_call(eo_e, e, EVAS_CALLBACK_RENDER_POST, &post);
e->inside_post_render = EINA_FALSE;
eina_spinlock_release(&(e->render.lock));
if (post.updated_area) eina_list_free(post.updated_area);
}
RD(0, "---]\n");
#ifdef EVAS_RENDER_DEBUG_TIMING
_accumulate_time(start_time, do_async);
#endif
if (!do_async) _evas_render_cleanup();
eina_evlog("-render_end", eo_e, 0.0, NULL);
return rendering;
}
static Eina_Bool
_drop_glyph_ref(const void *container EINA_UNUSED, void *data, void *fdata EINA_UNUSED)
{
evas_common_font_glyphs_unref(data);
return EINA_TRUE;
}
static Eina_Bool
_drop_texts_ref(const void *container EINA_UNUSED, void *data, void *fdata EINA_UNUSED)
{
evas_common_font_fonts_unref(data);
return EINA_TRUE;
}
static void
evas_render_wakeup(Evas *eo_e)
{
Evas_Event_Render_Post post;
Render_Updates *ru;
Eina_Bool haveup = EINA_FALSE;
Eina_List *ret_updates = NULL, *l;
Evas_Post_Render_Job *job;
Evas_Public_Data *evas;
Efl_Canvas_Output *out;
Eina_Inlist *jobs_il;
evas = efl_data_scope_get(eo_e, EVAS_CANVAS_CLASS);
eina_evlog("+render_wakeup", eo_e, 0.0, NULL);
eina_spinlock_take(&(evas->render.lock));
EINA_LIST_FOREACH(evas->outputs, l, out)
{
if (!out->output) continue ;
EINA_LIST_FREE(out->updates, ru)
{
ret_updates = eina_list_append(ret_updates, ru->area);
free(ru);
haveup = EINA_TRUE;
}
}
eina_spinlock_release(&(evas->render.lock));
/* flush redraws */
if (haveup)
{
Eina_List *ll;
Evas_Object *eo_obj;
EINA_LIST_FOREACH(evas->video_objects, ll, eo_obj)
{
_evas_object_image_video_overlay_do(eo_obj);
}
_cb_always_call(eo_e, evas, EVAS_CALLBACK_RENDER_FLUSH_POST, NULL);
}
/* clear redraws */
EINA_LIST_FOREACH(evas->outputs, l, out)
{
if (!out->output) continue ;
ENFN->output_redraws_clear(ENC, out->output);
}
/* unref queues */
eina_array_foreach(&evas->scie_unref_queue, _drop_scie_ref, NULL);
eina_array_flush(&evas->scie_unref_queue);
evas_common_rgba_image_scalecache_prune();
eina_array_foreach(&evas->image_unref_queue, _drop_image_cache_ref, NULL);
eina_array_flush(&evas->image_unref_queue);
eina_array_foreach(&evas->glyph_unref_queue, _drop_glyph_ref, NULL);
eina_array_flush(&evas->glyph_unref_queue);
eina_array_foreach(&evas->texts_unref_queue, _drop_texts_ref, NULL);
eina_array_flush(&evas->texts_unref_queue);
SLKL(evas->post_render.lock);
jobs_il = EINA_INLIST_GET(evas->post_render.jobs);
evas->post_render.jobs = NULL;
SLKU(evas->post_render.lock);
evas->inside_post_render = EINA_TRUE;
EINA_INLIST_FREE(jobs_il, job)
{
jobs_il = eina_inlist_remove(jobs_il, EINA_INLIST_GET(job));
if (job->func)
job->func(job->data);
free(job);
}
/* post rendering */
_rendering_evases = eina_list_remove_list(_rendering_evases, evas->rendering);
evas->rendering = NULL;
post.updated_area = ret_updates;
_cb_always_call(eo_e, evas, EVAS_CALLBACK_RENDER_POST, &post);
evas->inside_post_render = EINA_FALSE;
_deferred_callbacks_process(eo_e, evas);
evas_render_updates_free(ret_updates);
eina_evlog("-render_wakeup", eo_e, 0.0, NULL);
efl_unref(eo_e);
#ifdef EVAS_RENDER_DEBUG_TIMING
_accumulate_time(0, EINA_TRUE);
#endif
}
static void
evas_render_async_wakeup(void *target, Evas_Callback_Type type EINA_UNUSED, void *event_info EINA_UNUSED)
{
Evas_Public_Data *e = target;
evas_render_wakeup(e->evas);
_evas_render_busy_end();
}
// FIXME: This event should be per output... maybe ? maybe not ?
static void
evas_render_pipe_wakeup(void *data)
{
Eina_List *l, *ll;
Render_Updates *ru;
Evas_Public_Data *evas = data;
Efl_Canvas_Output *out;
Evas *e;
eina_evlog("+render_pipe_wakeup", evas->evas, 0.0, NULL);
eina_spinlock_take(&(evas->render.lock));
e = evas->evas;
EINA_LIST_FOREACH(evas->outputs, ll, out)
{
if (!out->output) continue ;
EINA_LIST_FOREACH(out->updates, l, ru)
{
eina_evlog("+render_push", evas->evas, 0.0, NULL);
ENFN->output_redraws_next_update_push
(ENC, out->output, ru->surface,
ru->area->x - out->geometry.x, ru->area->y - out->geometry.y,
ru->area->w, ru->area->h,
EVAS_RENDER_MODE_ASYNC_END);
eina_evlog("-render_push", evas->evas, 0.0, NULL);
//XXX: need a way to unref render output surfaces
ru->surface = NULL;
}
eina_evlog("+render_output_flush", evas->evas, 0.0, NULL);
ENFN->output_flush(ENC, out->output, EVAS_RENDER_MODE_ASYNC_END);
eina_evlog("-render_output_flush", evas->evas, 0.0, NULL);
}
eina_spinlock_release(&(evas->render.lock));
evas_async_events_put(data, 0, NULL, evas_render_async_wakeup);
/* use local pointer to avoid data race with 'evas' deref after releasing lock */
eina_evlog("-render_pipe_wakeup", e, 0.0, NULL);
}
EAPI void
evas_render_updates_free(Eina_List *updates)
{
Eina_Rectangle *r;
EINA_LIST_FREE(updates, r)
eina_rectangle_free(r);
}
EOLIAN Eina_Bool
_evas_canvas_render_async(Eo *eo_e, Evas_Public_Data *e)
{
Eina_Bool ret;
eina_evlog("+render_block", eo_e, 0.0, NULL);
evas_canvas_async_block(e);
eina_evlog("-render_block", eo_e, 0.0, NULL);
eina_evlog("+render", eo_e, 0.0, NULL);
ret = evas_render_updates_internal(eo_e, 1, 1, EINA_TRUE);
eina_evlog("-render", eo_e, 0.0, NULL);
return ret;
}
static Eina_List *
evas_render_updates_internal_wait(Evas *eo_e,
unsigned char make_updates,
unsigned char do_draw)
{
Eina_List *ret = NULL, *l;
Efl_Canvas_Output *out;
Evas_Public_Data *e;
e = efl_data_scope_get(eo_e, EVAS_CANVAS_CLASS);
if (!evas_render_updates_internal(eo_e, make_updates, do_draw, EINA_FALSE))
return NULL;
eina_spinlock_take(&(e->render.lock));
EINA_LIST_FOREACH(e->outputs, l, out)
{
if (!out->output) continue ;
ret = eina_list_merge(ret, out->updates);
out->updates = NULL;
}
eina_spinlock_release(&(e->render.lock));
return ret;
}
EOLIAN Eina_List*
_evas_canvas_render_updates(Eo *eo_e, Evas_Public_Data *e)
{
Eina_List *ret, *updates = NULL;
Render_Updates *ru;
eina_evlog("+render_block", eo_e, 0.0, NULL);
evas_canvas_async_block(e);
eina_evlog("-render_block", eo_e, 0.0, NULL);
eina_evlog("+render", eo_e, 0.0, NULL);
ret = evas_render_updates_internal_wait(eo_e, 1, 1);
eina_evlog("-render", eo_e, 0.0, NULL);
EINA_LIST_FREE(ret, ru)
{
updates = eina_list_append(updates, ru->area);
free(ru);
}
return updates;
}
EOLIAN void
_evas_canvas_render(Eo *eo_e, Evas_Public_Data *e)
{
Eina_List *ret;
Render_Updates *ru;
if (!e->changed) return;
eina_evlog("+render_block", eo_e, 0.0, NULL);
evas_canvas_async_block(e);
eina_evlog("-render_block", eo_e, 0.0, NULL);
eina_evlog("+render", eo_e, 0.0, NULL);
ret = evas_render_updates_internal_wait(eo_e, 0, 1);
eina_evlog("-render", eo_e, 0.0, NULL);
EINA_LIST_FREE(ret, ru)
{
eina_rectangle_free(ru->area);
free(ru);
}
}
EOLIAN void
_evas_canvas_norender(Eo *eo_e, Evas_Public_Data *e)
{
evas_canvas_async_block(e);
evas_render_updates_internal_wait(eo_e, 0, 0);
}
EAPI void
evas_norender_with_updates(Eo *eo_e)
{
Evas_Public_Data *e = efl_data_scope_get(eo_e, EVAS_CANVAS_CLASS);
Eina_List *ret;
Render_Updates *ru;
evas_canvas_async_block(e);
ret = evas_render_updates_internal_wait(eo_e, 1, 0);
EINA_LIST_FREE(ret, ru)
{
eina_rectangle_free(ru->area);
free(ru);
}
}
EOLIAN void
_evas_canvas_render_idle_flush(Eo *eo_e, Evas_Public_Data *evas)
{
eina_evlog("+idle_flush", eo_e, 0.0, NULL);
evas_canvas_async_block(evas);
evas_render_rendering_wait(evas);
evas_fonts_zero_pressure();
if (ENFN && ENFN->output_idle_flush)
{
Efl_Canvas_Output *output;
Eina_List *l;
EINA_LIST_FOREACH(evas->outputs, l, output)
if (output->output)
ENFN->output_idle_flush(ENC, output->output);
}
eina_inarray_flush(&evas->active_objects);
OBJS_ARRAY_FLUSH(&evas->render_objects);
OBJS_ARRAY_FLUSH(&evas->restack_objects);
OBJS_ARRAY_FLUSH(&evas->delete_objects);
OBJS_ARRAY_FLUSH(&evas->obscuring_objects);
OBJS_ARRAY_FLUSH(&evas->temporary_objects);
OBJS_ARRAY_FLUSH(&evas->map_clip_objects);
eina_array_foreach(&evas->clip_changes, _evas_clip_changes_free, NULL);
eina_array_clean(&evas->clip_changes);
evas->invalidate = EINA_TRUE;
eina_evlog("-idle_flush", eo_e, 0.0, NULL);
}
EOLIAN void
_evas_canvas_sync(Eo *eo_e, Evas_Public_Data *e)
{
eina_evlog("+render_sync", eo_e, 0.0, NULL);
evas_canvas_async_block(e);
evas_render_rendering_wait(e);
eina_evlog("-render_sync", eo_e, 0.0, NULL);
}
void
_evas_render_dump_map_surfaces(Evas_Object *eo_obj)
{
Evas_Object_Protected_Data *obj = efl_data_scope_get(eo_obj, EFL_CANVAS_OBJECT_CLASS);
if ((obj->map->cur.map) && obj->map->surface)
{
Evas_Public_Data *evas = obj->layer->evas;
ENFN->image_free(ENC, obj->map->surface);
EINA_COW_WRITE_BEGIN(evas_object_map_cow, obj->map, Evas_Object_Map_Data, map_write)
map_write->surface = NULL;
EINA_COW_WRITE_END(evas_object_map_cow, obj->map, map_write);
}
if (obj->is_smart)
{
Evas_Object_Protected_Data *obj2;
EINA_INLIST_FOREACH(evas_object_smart_members_get_direct(eo_obj), obj2)
_evas_render_dump_map_surfaces(obj2->object);
}
}
EOLIAN void
_evas_canvas_render_dump(Eo *eo_e EINA_UNUSED, Evas_Public_Data *evas)
{
Evas_Layer *lay;
evas_canvas_async_block(evas);
evas_all_sync();
evas_cache_async_freeze();
EINA_INLIST_FOREACH(evas->layers, lay)
{
Evas_Object_Protected_Data *obj;
lay->walking_objects++;
EINA_INLIST_FOREACH(lay->objects, obj)
{
if (obj->proxy->surface)
{
EINA_COW_WRITE_BEGIN(evas_object_proxy_cow, obj->proxy, Evas_Object_Proxy_Data, proxy_write)
{
ENFN->image_free(ENC, proxy_write->surface);
proxy_write->surface = NULL;
}
EINA_COW_WRITE_END(evas_object_proxy_cow, obj->proxy, proxy_write);
}
if (obj->mask->surface)
{
EINA_COW_WRITE_BEGIN(evas_object_mask_cow, obj->mask, Evas_Object_Mask_Data, mdata)
{
ENFN->image_free(ENC, mdata->surface);
mdata->surface = NULL;
}
EINA_COW_WRITE_END(evas_object_mask_cow, obj->mask, mdata);
}
if ((obj->type) && (!strcmp(obj->type, "image")))
evas_object_inform_call_image_unloaded(obj->object);
_evas_render_dump_map_surfaces(obj->object);
}
lay->walking_objects--;
_evas_layer_flush_removes(lay);
}
if (ENFN && ENFN->output_dump)
{
Efl_Canvas_Output *output;
Eina_List *l;
EINA_LIST_FOREACH(evas->outputs, l, output)
if (output->output)
ENFN->output_dump(ENC, output->output);
}
#define GC_ALL(Cow) \
if (Cow) while (eina_cow_gc(Cow))
GC_ALL(evas_object_proxy_cow);
GC_ALL(evas_object_map_cow);
GC_ALL(evas_object_image_pixels_cow);
GC_ALL(evas_object_image_load_opts_cow);
GC_ALL(evas_object_image_state_cow);
evas_fonts_zero_pressure();
if (ENFN && ENFN->output_idle_flush)
{
Efl_Canvas_Output *output;
Eina_List *l;
EINA_LIST_FOREACH(evas->outputs, l, output)
if (output->output)
ENFN->output_idle_flush(ENC, output->output);
}
eina_inarray_flush(&evas->active_objects);
OBJS_ARRAY_FLUSH(&evas->render_objects);
OBJS_ARRAY_FLUSH(&evas->restack_objects);
OBJS_ARRAY_FLUSH(&evas->delete_objects);
OBJS_ARRAY_FLUSH(&evas->obscuring_objects);
OBJS_ARRAY_FLUSH(&evas->temporary_objects);
OBJS_ARRAY_FLUSH(&evas->map_clip_objects);
eina_array_foreach(&evas->clip_changes, _evas_clip_changes_free, NULL);
eina_array_clean(&evas->clip_changes);
evas->invalidate = EINA_TRUE;
evas_cache_async_thaw();
}
void
evas_render_invalidate(Evas *eo_e)
{
Evas_Public_Data *e;
MAGIC_CHECK(eo_e, Evas, MAGIC_EVAS);
return;
MAGIC_CHECK_END();
e = efl_data_scope_get(eo_e, EVAS_CANVAS_CLASS);
eina_inarray_flush(&e->active_objects);
OBJS_ARRAY_CLEAN(&e->render_objects);
OBJS_ARRAY_FLUSH(&e->restack_objects);
OBJS_ARRAY_FLUSH(&e->delete_objects);
OBJS_ARRAY_FLUSH(&e->snapshot_objects);
OBJS_ARRAY_FLUSH(&e->map_clip_objects);
e->invalidate = EINA_TRUE;
}
void
evas_render_object_recalc(Evas_Object_Protected_Data *obj)
{
if ((!obj->changed) && (obj->delete_me < 2))
{
Evas_Public_Data *e;
e = obj->layer->evas;
if ((!e) || (e->cleanup)) return;
if (!obj->in_pending_objects)
OBJ_ARRAY_PUSH(&e->pending_objects, obj);
obj->in_pending_objects = obj->changed = EINA_TRUE;
}
}
void
evas_unref_queue_image_put(Evas_Public_Data *pd, void *image)
{
eina_array_push(&pd->image_unref_queue, image);
evas_common_rgba_image_scalecache_items_ref(image, &pd->scie_unref_queue);
}
void
evas_unref_queue_glyph_put(Evas_Public_Data *pd, void *glyph)
{
eina_array_push(&pd->glyph_unref_queue, glyph);
}
void
evas_unref_queue_texts_put(Evas_Public_Data *pd, void *texts)
{
eina_array_push(&pd->texts_unref_queue, texts);
}
void
evas_post_render_job_add(Evas_Public_Data *pd, void (*func)(void *), void *data)
{
Evas_Post_Render_Job *job;
if (!pd || pd->delete_me) return;
if (!func) return;
job = malloc(sizeof(*job));
job->func = func;
job->data = data;
SLKL(pd->post_render.lock);
pd->post_render.jobs = (Evas_Post_Render_Job *)
eina_inlist_append(EINA_INLIST_GET(pd->post_render.jobs), EINA_INLIST_GET(job));
SLKU(pd->post_render.lock);
}
void
evas_render_post_change_object_push(Evas_Object_Protected_Data *obj)
{
Evas_Public_Data *e;
if (!obj || !obj->layer || !obj->layer->evas) return;
e = obj->layer->evas;
OBJ_ARRAY_PUSH(&e->render_post_change_objects, obj);
}
/* vim:set ts=8 sw=3 sts=3 expandtab cino=>5n-2f0^-2{2(0W1st0 :*/