enlightenment/src/bin/e_comp.c

5727 lines
179 KiB
C
Raw Normal View History

#include "e.h"
#ifdef HAVE_WAYLAND_CLIENTS
#include "e_comp_wl.h"
#endif
#define ACTION_TIMEOUT 30.0
#define OVER_FLOW 1
//#define SHAPE_DEBUG
//#define BORDER_ZOOMAPS
//////////////////////////////////////////////////////////////////////////
//
2010-01-10 18:59:32 -08:00
// TODO (no specific order):
// 1. abstract evas object and compwin so we can duplicate the object N times
// in N canvases - for winlist, everything, pager etc. too
// 2. implement "unmapped composite cache" -> N pixels worth of unmapped
// windows to be fully composited. only the most active/recent.
// 3. for unmapped windows - when window goes out of unmapped comp cache
// make a miniature copy (1/4 width+height?) and set property on window
// with pixmap id
2011-03-19 23:12:17 -07:00
//
//////////////////////////////////////////////////////////////////////////
static Eina_List *handlers = NULL;
static Eina_List *compositors = NULL;
static Eina_Hash *windows = NULL;
static Eina_Hash *borders = NULL;
static Eina_Hash *damages = NULL;
static Eina_Hash *ignores = NULL;
static Eina_List *actions = NULL;
static E_Comp_Config *conf = NULL;
static E_Config_DD *conf_edd = NULL;
static E_Config_DD *conf_match_edd = NULL;
static Ecore_Timer *action_timeout = NULL;
static Eina_Bool gl_avail = EINA_FALSE;
EAPI int E_EVENT_COMP_SOURCE_VISIBILITY = -1;
EAPI int E_EVENT_COMP_SOURCE_ADD = -1;
EAPI int E_EVENT_COMP_SOURCE_DEL = -1;
EAPI int E_EVENT_COMP_SOURCE_CONFIGURE = -1;
2013-04-19 05:49:34 -07:00
EAPI int E_EVENT_COMP_SOURCE_STACK = -1;
static int _e_comp_log_dom = -1;
//////////////////////////////////////////////////////////////////////////
#undef DBG
#undef INF
#undef WRN
#undef ERR
#undef CRI
#if 1
# ifdef SHAPE_DEBUG
# define SHAPE_DBG(...) EINA_LOG_DOM_DBG(_e_comp_log_dom, __VA_ARGS__)
# define SHAPE_INF(...) EINA_LOG_DOM_INFO(_e_comp_log_dom, __VA_ARGS__)
# define SHAPE_WRN(...) EINA_LOG_DOM_WARN(_e_comp_log_dom, __VA_ARGS__)
# define SHAPE_ERR(...) EINA_LOG_DOM_ERR(_e_comp_log_dom, __VA_ARGS__)
# define SHAPE_CRI(...) EINA_LOG_DOM_CRIT(_e_comp_log_dom, __VA_ARGS__)
# else
# define SHAPE_DBG(f, x ...)
# define SHAPE_INF(f, x ...)
# define SHAPE_WRN(f, x ...)
# define SHAPE_ERR(f, x ...)
# define SHAPE_CRI(f, x ...)
# endif
#define DBG(...) EINA_LOG_DOM_DBG(_e_comp_log_dom, __VA_ARGS__)
#define INF(...) EINA_LOG_DOM_INFO(_e_comp_log_dom, __VA_ARGS__)
#define WRN(...) EINA_LOG_DOM_WARN(_e_comp_log_dom, __VA_ARGS__)
#define ERR(...) EINA_LOG_DOM_ERR(_e_comp_log_dom, __VA_ARGS__)
#define CRI(...) EINA_LOG_DOM_CRIT(_e_comp_log_dom, __VA_ARGS__)
#else
2011-03-19 23:12:17 -07:00
#define DBG(f, x ...)
#define INF(f, x ...)
#define WRN(f, x ...)
#define ERR(f, x ...)
#define CRI(f, x ...)
#endif
static Eina_Bool _e_comp_win_do_shadow(E_Comp_Win *cw);
static void _e_comp_win_ready_timeout_setup(E_Comp_Win *cw);
static void _e_comp_render_queue(E_Comp *c);
static void _e_comp_win_damage(E_Comp_Win *cw, int x, int y, int w, int h, Eina_Bool dmg);
static void _e_comp_win_render_queue(E_Comp_Win *cw);
static void _e_comp_win_release(E_Comp_Win *cw);
static void _e_comp_win_adopt(E_Comp_Win *cw);
static void _e_comp_win_del(E_Comp_Win *cw);
static void _e_comp_win_show(E_Comp_Win *cw);
static void _e_comp_win_real_hide(E_Comp_Win *cw);
static void _e_comp_win_hide(E_Comp_Win *cw);
static void _e_comp_win_configure(E_Comp_Win *cw, int x, int y, int w, int h, int border);
static void _e_comp_shapes_update(void *data, E_Container_Shape *es, E_Container_Shape_Change ch);
static void _e_comp_injected_win_del_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED);
static void _e_comp_injected_win_hide_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED);
static void _e_comp_injected_win_focus_in_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED);
static void _e_comp_injected_win_focus_out_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED);
static void
_e_comp_event_end(void *d EINA_UNUSED, E_Event_Comp *ev)
{
ev->cw->pending_count--;
if (ev->cw->delete_pending && (!ev->cw->pending_count))
_e_comp_win_del(ev->cw);
free(ev);
}
/* FIXME: external ? */
static void
_e_comp_event_source_visibility(E_Comp_Win *cw)
{
E_Event_Comp *ev;
ev = E_NEW(E_Event_Comp, 1);
ev->cw = cw;
ecore_event_add(E_EVENT_COMP_SOURCE_VISIBILITY, ev, (Ecore_End_Cb)_e_comp_event_end, NULL);
}
static void
_e_comp_event_source_add(E_Comp_Win *cw)
{
E_Event_Comp *ev;
ev = E_NEW(E_Event_Comp, 1);
ev->cw = cw;
2013-04-19 05:44:52 -07:00
ecore_event_add(E_EVENT_COMP_SOURCE_ADD, ev, (Ecore_End_Cb)_e_comp_event_end, NULL);
}
static void
_e_comp_event_source_del(E_Comp_Win *cw)
{
E_Event_Comp *ev;
ev = E_NEW(E_Event_Comp, 1);
ev->cw = cw;
2013-04-19 05:44:52 -07:00
ecore_event_add(E_EVENT_COMP_SOURCE_DEL, ev, (Ecore_End_Cb)_e_comp_event_end, NULL);
}
static void
_e_comp_event_source_configure(E_Comp_Win *cw)
{
E_Event_Comp *ev;
ev = E_NEW(E_Event_Comp, 1);
ev->cw = cw;
2013-04-19 05:44:52 -07:00
ecore_event_add(E_EVENT_COMP_SOURCE_CONFIGURE, ev, (Ecore_End_Cb)_e_comp_event_end, NULL);
}
2013-04-19 05:49:34 -07:00
static void
_e_comp_event_source_stack(E_Comp_Win *cw)
{
E_Event_Comp *ev;
ev = E_NEW(E_Event_Comp, 1);
ev->cw = cw;
ecore_event_add(E_EVENT_COMP_SOURCE_STACK, ev, (Ecore_End_Cb)_e_comp_event_end, NULL);
}
static void
_e_comp_child_show(E_Comp_Win *cw)
{
evas_object_show(cw->effect_obj);
if (cw->bd)
{
Eina_List *l;
E_Border *tmp;
EINA_LIST_FOREACH(cw->bd->client.e.state.video_child, l, tmp)
{
E_Comp_Win *tcw;
tcw = eina_hash_find(borders, e_util_winid_str_get(tmp->client.win));
2012-06-28 00:03:44 -07:00
if (!tcw) continue;
evas_object_show(tcw->effect_obj);
}
}
}
static void
_e_comp_child_hide(E_Comp_Win *cw)
{
evas_object_hide(cw->effect_obj);
if (cw->bd)
{
Eina_List *l;
E_Border *tmp;
EINA_LIST_FOREACH(cw->bd->client.e.state.video_child, l, tmp)
{
E_Comp_Win *tcw;
tcw = eina_hash_find(borders, e_util_winid_str_get(tmp->client.win));
2012-06-28 00:03:44 -07:00
if (!tcw) continue;
evas_object_hide(tcw->effect_obj);
}
}
}
static E_Comp_Win *
_e_comp_fullscreen_check(E_Comp *c)
{
E_Comp_Win *cw;
if (!c->wins) return NULL;
EINA_INLIST_REVERSE_FOREACH(c->wins, cw)
{
if ((!cw->visible) || (cw->input_only) || (cw->invalid) || (cw->real_obj))
continue;
if (!cw->bd) continue;
if ((cw->x == 0) && (cw->y == 0) &&
((cw->bd->client.w) >= c->man->w) &&
((cw->bd->client.h) >= c->man->h) &&
(!cw->argb) && (!cw->shaped) && (!cw->bg_win)
)
{
return cw;
}
return NULL;
}
return NULL;
}
static inline Eina_Bool
_e_comp_shaped_check(int w, int h, const Eina_Rectangle *rects, int num)
{
if ((!rects) || (num < 1)) return EINA_FALSE;
if (num > 1) return EINA_TRUE;
if ((rects[0].x == 0) && (rects[0].y == 0) &&
((int)rects[0].w == w) && ((int)rects[0].h == h))
return EINA_FALSE;
return EINA_TRUE;
}
static inline Eina_Bool
_e_comp_win_shaped_check(const E_Comp_Win *cw, const Eina_Rectangle *rects, int num)
{
return _e_comp_shaped_check(cw->w, cw->h, rects, num);
}
static void
_e_comp_win_shape_rectangles_apply(E_Comp_Win *cw, const Eina_Rectangle *rects, int num)
{
Eina_List *l;
Evas_Object *o;
int i;
DBG("SHAPE [0x%x] change, rects=%p (%d)", cw->win, rects, num);
if (!_e_comp_win_shaped_check(cw, rects, num))
{
rects = NULL;
}
if (rects)
{
unsigned int *pix, *p;
unsigned char *spix, *sp;
int w, h, px, py;
evas_object_image_size_get(cw->obj, &w, &h);
if ((w > 0) && (h > 0))
{
2011-03-19 23:12:17 -07:00
if (cw->native)
{
ERR("BUGGER: shape with native surface? cw=%p", cw);
2011-03-19 23:12:17 -07:00
return;
}
evas_object_image_native_surface_set(cw->obj, NULL);
evas_object_image_alpha_set(cw->obj, 1);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_native_surface_set(o, NULL);
evas_object_image_alpha_set(o, 1);
}
pix = evas_object_image_data_get(cw->obj, 1);
if (pix)
{
spix = calloc(w * h, sizeof(unsigned char));
if (spix)
2010-01-07 06:01:35 -08:00
{
DBG("SHAPE [0x%x] rects %i", cw->win, num);
for (i = 0; i < num; i++)
{
int rx, ry, rw, rh;
rx = rects[i].x; ry = rects[i].y;
rw = rects[i].w; rh = rects[i].h;
E_RECTS_CLIP_TO_RECT(rx, ry, rw, rh, 0, 0, w, h);
sp = spix + (w * ry) + rx;
for (py = 0; py < rh; py++)
{
for (px = 0; px < rw; px++)
{
*sp = 0xff; sp++;
}
sp += w - rw;
}
}
sp = spix;
p = pix;
for (py = 0; py < h; py++)
{
for (px = 0; px < w; px++)
{
unsigned int mask, imask;
mask = ((unsigned int)(*sp)) << 24;
imask = mask >> 8;
imask |= imask >> 8;
imask |= imask >> 8;
*p = mask | (*p & imask);
//if (*sp) *p = 0xff000000 | *p;
//else *p = 0x00000000;
sp++;
p++;
}
}
free(spix);
}
evas_object_image_data_set(cw->obj, pix);
evas_object_image_data_update_add(cw->obj, 0, 0, w, h);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_data_set(o, pix);
evas_object_image_data_update_add(o, 0, 0, w, h);
}
}
}
}
else
{
if (cw->shaped)
{
unsigned int *pix, *p;
int w, h, px, py;
evas_object_image_size_get(cw->obj, &w, &h);
if ((w > 0) && (h > 0))
{
if (cw->native)
{
ERR("BUGGER: shape with native surface? cw=%p", cw);
return;
}
evas_object_image_alpha_set(cw->obj, 0);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_alpha_set(o, 1);
}
pix = evas_object_image_data_get(cw->obj, 1);
if (pix)
{
p = pix;
for (py = 0; py < h; py++)
{
for (px = 0; px < w; px++)
*p |= 0xff000000;
}
}
evas_object_image_data_set(cw->obj, pix);
evas_object_image_data_update_add(cw->obj, 0, 0, w, h);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_data_set(o, pix);
evas_object_image_data_update_add(o, 0, 0, w, h);
}
}
}
2011-03-19 23:12:17 -07:00
// dont need to fix alpha chanel as blending
// should be totally off here regardless of
// alpha channel content
}
}
static Eina_Bool
_e_comp_cb_win_show_ready_timeout(void *data)
{
E_Comp_Win *cw = data;
cw->show_ready = 1;
if (cw->visible)
{
2012-06-28 00:03:44 -07:00
if (!cw->update)
{
if (cw->update_timeout)
{
ecore_timer_del(cw->update_timeout);
cw->update_timeout = NULL;
}
cw->update = 1;
cw->c->updates = eina_list_append(cw->c->updates, cw);
}
_e_comp_win_render_queue(cw);
}
cw->ready_timeout = NULL;
return ECORE_CALLBACK_CANCEL;
}
static void
_e_comp_win_ready_timeout_setup(E_Comp_Win *cw)
{
if (cw->ready_timeout)
{
ecore_timer_del(cw->ready_timeout);
cw->ready_timeout = NULL;
}
if (cw->show_ready) return;
if (cw->counter) return;
// FIXME: make show_ready option
if (0)
{
cw->show_ready = 1;
}
else
cw->ready_timeout = ecore_timer_add(conf->first_draw_delay, _e_comp_cb_win_show_ready_timeout, cw);
}
static void
_e_comp_win_layout_populate(E_Comp_Win *cw)
{
e_layout_pack(cw->c->layout, cw->effect_obj);
}
static void
_e_comp_win_restack(E_Comp_Win *cw)
{
Eina_Inlist *prev, *next;
Eina_List *l;
E_Comp_Win *cwp = NULL, *cwn = NULL;
next = EINA_INLIST_GET(cw)->next;
if (next) cwn = EINA_INLIST_CONTAINER_GET(next, E_Comp_Win);
prev = EINA_INLIST_GET(cw)->prev;
if (prev) cwp = EINA_INLIST_CONTAINER_GET(prev, E_Comp_Win);
if (cwp)
{
e_layout_child_raise_above(cw->effect_obj, cwp->effect_obj);
cw->c->wins = eina_inlist_remove(cw->c->wins, EINA_INLIST_GET(cw));
cw->c->wins = eina_inlist_append_relative(cw->c->wins, EINA_INLIST_GET(cw), EINA_INLIST_GET(cwp));
}
else if (cwn)
{
e_layout_child_raise_above(cw->effect_obj, cwn->effect_obj);
cw->c->wins = eina_inlist_remove(cw->c->wins, EINA_INLIST_GET(cw));
cw->c->wins = eina_inlist_append_relative(cw->c->wins, EINA_INLIST_GET(cw), EINA_INLIST_GET(cwn));
}
if (cw->bd)
{
E_Border *tmp;
EINA_LIST_FOREACH(cw->bd->client.e.state.video_child, l, tmp)
{
E_Comp_Win *tcw;
tcw = eina_hash_find(borders, e_util_winid_str_get(tmp->client.win));
if (!tcw) continue;
e_layout_child_lower_below(tcw->effect_obj, cw->effect_obj);
cw->c->wins = eina_inlist_remove(cw->c->wins, EINA_INLIST_GET(tcw));
cw->c->wins = eina_inlist_prepend_relative(cw->c->wins, EINA_INLIST_GET(tcw), EINA_INLIST_GET(cw));
}
}
EINA_LIST_FOREACH(cw->stack_below, l, cwp)
{
e_layout_child_lower_below(cwp->effect_obj, cw->effect_obj);
cw->c->wins = eina_inlist_remove(cw->c->wins, EINA_INLIST_GET(cwp));
cw->c->wins = eina_inlist_prepend_relative(cw->c->wins, EINA_INLIST_GET(cwp), EINA_INLIST_GET(cw));
}
_e_comp_shapes_update(cw->c, NULL, E_CONTAINER_SHAPE_MOVE);
}
static void
_e_comp_win_geometry_update(E_Comp_Win *cw)
{
int x, y, w, h;
if (cw->update)
{
cw->geom_update = 1;
_e_comp_win_render_queue(cw);
return;
}
if (cw->bd)
x = cw->bd->x, y = cw->bd->y;
else if (cw->visible)
x = cw->x, y = cw->y;
else
x = cw->hidden.x, y = cw->hidden.y;
if (cw->real_obj)
w = cw->w, h = cw->h;
else if (cw->bd)
{
if (((!cw->bd->shaded) && (!cw->bd->shading)) && cw->pw && cw->ph)
{
w = cw->pw + cw->bd->client_inset.l + cw->bd->client_inset.r, h = cw->ph + cw->bd->client_inset.t + cw->bd->client_inset.b;
if ((cw->pw != cw->bd->client.w) || (cw->ph != cw->bd->client.h))
{
/* something fucked us and the pixmap came back with the wrong size
* fix it by forcing a resize in e_border
*/
BD_CHANGED(cw->bd);
cw->bd->changes.size = 1;
}
}
else
w = cw->bd->w, h = cw->bd->h;
}
else
w = cw->pw ?: cw->w, h = cw->ph ?: cw->h;
if (cw->zoomobj) e_zoomap_child_resize(cw->zoomobj, w, h);
if (cw->not_in_layout)
{
evas_object_resize(cw->effect_obj, w, h);
evas_object_move(cw->effect_obj, x, y);
}
else
{
e_layout_child_move(cw->effect_obj, x, y);
e_layout_child_resize(cw->effect_obj, w, h);
}
cw->geom_update = 0;
}
static void
_e_comp_win_update(E_Comp_Win *cw)
{
Eina_List *l;
Evas_Object *o;
E_Comp_Render_Update_Rect *r;
int i, num;
int pw, ph;
int pshaped = cw->shaped;
Eina_Rectangle *rects;
2010-02-24 21:12:49 -08:00
DBG("UPDATE [0x%x] pm = %x", cw->win, cw->pixmap);
if (conf->grab) ecore_x_grab();
cw->update = 0;
pw = cw->pw, ph = cw->ph;
if (cw->shape_changed)
{
if (cw->free_shape)
{
ecore_x_pixmap_geometry_get(cw->win, NULL, NULL, &(cw->w), &(cw->h));
rects = (Eina_Rectangle*)ecore_x_window_shape_rectangles_get(cw->win, &num);
e_container_shape_rects_set(cw->shape, rects, num);
if (cw->shape->shape_rects)
e_container_shape_input_rects_set(cw->shape, NULL, 0);
else
{
rects = (Eina_Rectangle*)ecore_x_window_shape_input_rectangles_get(cw->win, &num);
e_container_shape_input_rects_set(cw->shape, rects, num);
}
}
if (cw->shape->shape_rects)
{
for (i = 0; i < cw->shape->shape_rects_num; i++)
{
E_RECTS_CLIP_TO_RECT(cw->shape->shape_rects[i].x, cw->shape->shape_rects[i].y,
cw->shape->shape_rects[i].w, cw->shape->shape_rects[i].h, 0, 0, (int)cw->w, (int)cw->h);
}
}
if (cw->shape->shape_input_rects)
{
for (i = 0; i < cw->shape->shape_input_rects_num; i++)
{
E_RECTS_CLIP_TO_RECT(cw->shape->shape_input_rects[i].x, cw->shape->shape_input_rects[i].y,
cw->shape->shape_input_rects[i].w, cw->shape->shape_input_rects[i].h, 0, 0, (int)cw->w, (int)cw->h);
}
}
cw->shaped = _e_comp_win_shaped_check(cw, cw->shape->shape_rects, cw->shape->shape_rects_num);
}
if (((!cw->pixmap) || (cw->needpix)) &&
(!cw->real_hid) && (!cw->nocomp) && (!cw->c->nocomp))
{
Ecore_X_Pixmap pm = 0;
/* #ifdef HAVE_WAYLAND_CLIENTS */
/* if ((cw->bd) && (cw->bd->borderless)) */
/* pm = e_comp_wl_pixmap_get(cw->win); */
/* #endif */
if (!pm) pm = ecore_x_composite_name_window_pixmap_get(cw->win);
while (pm)
{
Ecore_X_Pixmap oldpm;
oldpm = cw->pixmap;
ecore_x_pixmap_geometry_get(pm, NULL, NULL, &(cw->pw), &(cw->ph));
if ((pw == cw->pw) && (ph == cw->ph))
{
ecore_x_pixmap_free(pm);
break;
}
cw->pixmap = pm;
cw->needpix = 0;
if (cw->xim) cw->needxim = 1;
_e_comp_win_ready_timeout_setup(cw);
if ((cw->pw != pw) || (cw->ph != ph)) cw->geom_update = 1;
DBG("REND [0x%x] pixmap = [0x%x], %ix%i", cw->win, cw->pixmap, cw->pw, cw->ph);
if ((cw->pw <= 0) || (cw->ph <= 0))
{
if (cw->native)
{
DBG(" [0x%x] free native", cw->win);
evas_object_image_native_surface_set(cw->obj, NULL);
cw->native = 0;
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_native_surface_set(o, NULL);
}
}
if (cw->pixmap)
{
DBG(" [0x%x] free pixmap", cw->win);
ecore_x_pixmap_free(cw->pixmap);
cw->pixmap = 0;
//cw->show_ready = 0; // hmm maybe not needed?
}
cw->pw = 0;
cw->ph = 0;
}
ecore_x_e_comp_pixmap_set(cw->win, cw->pixmap);
cw->native = 0;
DBG(" [0x%x] up resize %ix%i", cw->win, cw->pw, cw->ph);
e_comp_render_update_resize(cw->up, cw->pw, cw->ph);
e_comp_render_update_add(cw->up, 0, 0, cw->pw, cw->ph);
if (oldpm)
{
DBG(" [0x%x] free pm %x", cw->win, oldpm);
// XXX the below is unreachable code! :)
/*
if (cw->native)
{
cw->native = 0;
if (!((cw->pw > 0) && (cw->ph > 0)))
{
evas_object_image_native_surface_set(cw->obj, NULL);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_native_surface_set(o, NULL);
}
}
}
*/
ecore_x_pixmap_free(oldpm);
}
break;
}
}
if (!((cw->pw > 0) && (cw->ph > 0)))
{
if (cw->geom_update) _e_comp_win_geometry_update(cw);
if (conf->grab) ecore_x_ungrab();
return;
}
// evas_object_move(cw->effect_obj, cw->x, cw->y);
// was cw->w / cw->h
// evas_object_resize(cw->effect_obj, cw->pw, cw->ph);
if ((cw->c->gl) && (conf->texture_from_pixmap) &&
(!cw->shaped) && (cw->pixmap))
{
/* #ifdef HAVE_WAYLAND_CLIENTS */
/* DBG("DEBUG - pm now %x", e_comp_wl_pixmap_get(cw->win)); */
/* #endif */
/* DBG("DEBUG - pm now %x", ecore_x_composite_name_window_pixmap_get(cw->win)); */
evas_object_image_size_set(cw->obj, cw->pw, cw->ph);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_size_set(o, cw->pw, cw->ph);
}
if (!cw->native)
{
Evas_Native_Surface ns;
ns.version = EVAS_NATIVE_SURFACE_VERSION;
ns.type = EVAS_NATIVE_SURFACE_X11;
ns.data.x11.visual = cw->vis;
ns.data.x11.pixmap = cw->pixmap;
evas_object_image_native_surface_set(cw->obj, &ns);
DBG("NATIVE [0x%x] %x %ix%i", cw->win, cw->pixmap, cw->pw, cw->ph);
cw->native = 1;
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_native_surface_set(o, &ns);
}
}
r = e_comp_render_update_rects_get(cw->up);
2011-03-19 23:12:17 -07:00
if (r)
{
e_comp_render_update_clear(cw->up);
for (i = 0; r[i].w > 0; i++)
{
int x, y, w, h;
x = r[i].x; y = r[i].y;
w = r[i].w; h = r[i].h;
DBG("UPDATE [0x%x] pm [0x%x] %i %i %ix%i", cw->win, cw->pixmap, x, y, w, h);
evas_object_image_data_update_add(cw->obj, x, y, w, h);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_data_update_add(o, x, y, w, h);
}
}
free(r);
}
else
{
DBG("UPDATE [0x%x] NO RECTS!!! %i %i - %i %i",
cw->win, cw->up->w, cw->up->h, cw->up->tw, cw->up->th);
// cw->update = 1;
}
}
else if (cw->pixmap)
{
if (cw->native)
{
evas_object_image_native_surface_set(cw->obj, NULL);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_native_surface_set(o, NULL);
}
cw->native = 0;
}
if (cw->needxim)
{
cw->needxim = 0;
if (cw->xim)
{
evas_object_image_size_set(cw->obj, 1, 1);
evas_object_image_data_set(cw->obj, NULL);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_size_set(o, 1, 1);
evas_object_image_data_set(o, NULL);
}
ecore_x_image_free(cw->xim);
cw->xim = NULL;
}
}
if (!cw->xim)
{
if ((cw->xim = ecore_x_image_new(cw->pw, cw->ph, cw->vis, cw->depth)))
e_comp_render_update_add(cw->up, 0, 0, cw->pw, cw->ph);
}
r = e_comp_render_update_rects_get(cw->up);
2011-03-19 23:12:17 -07:00
if (r)
{
if (cw->xim)
{
if (ecore_x_image_is_argb32_get(cw->xim))
{
unsigned int *pix;
pix = ecore_x_image_data_get(cw->xim, NULL, NULL, NULL);
evas_object_image_data_set(cw->obj, pix);
evas_object_image_size_set(cw->obj, cw->pw, cw->ph);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_data_set(o, pix);
evas_object_image_size_set(o, cw->pw, cw->ph);
}
e_comp_render_update_clear(cw->up);
for (i = 0; r[i].w > 0; i++)
{
int x, y, w, h;
x = r[i].x; y = r[i].y;
w = r[i].w; h = r[i].h;
if (!ecore_x_image_get(cw->xim, cw->pixmap, x, y, x, y, w, h))
{
WRN("UPDATE [0x%x] %i %i %ix%i FAIL!!!!!!!!!!!!!!!!!", cw->win, x, y, w, h);
e_comp_render_update_add(cw->up, x, y, w, h);
cw->update = 1;
}
else
{
// why do we neeed these 2? this smells wrong
pix = ecore_x_image_data_get(cw->xim, NULL, NULL, NULL);
DBG("UPDATE [0x%x] %i %i %ix%i -- pix = %p", cw->win, x, y, w, h, pix);
evas_object_image_data_set(cw->obj, pix);
evas_object_image_data_update_add(cw->obj, x, y, w, h);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_data_set(o, pix);
evas_object_image_data_update_add(o, x, y, w, h);
}
}
}
}
else
{
unsigned int *pix;
int stride;
evas_object_image_size_set(cw->obj, cw->pw, cw->ph);
pix = evas_object_image_data_get(cw->obj, EINA_TRUE);
stride = evas_object_image_stride_get(cw->obj);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_data_set(o, pix);
evas_object_image_size_set(o, cw->pw, cw->ph);
}
e_comp_render_update_clear(cw->up);
for (i = 0; r[i].w > 0; i++)
2010-02-10 23:40:44 -08:00
{
int x, y, w, h;
x = r[i].x; y = r[i].y;
w = r[i].w; h = r[i].h;
if (!ecore_x_image_get(cw->xim, cw->pixmap, x, y, x, y, w, h))
{
WRN("UPDATE [0x%x] %i %i %ix%i FAIL!!!!!!!!!!!!!!!!!", cw->win, x, y, w, h);
e_comp_render_update_add(cw->up, x, y, w, h);
cw->update = 1;
}
else
2012-06-28 00:03:44 -07:00
{
unsigned int *srcpix;
int srcbpp = 0, srcbpl = 0;
// why do we neeed these 2? this smells wrong
srcpix = ecore_x_image_data_get(cw->xim, &srcbpl, NULL, &srcbpp);
ecore_x_image_to_argb_convert(srcpix, srcbpp, srcbpl,
cw->cmap, cw->vis, x, y, w, h, pix, stride, x, y);
DBG("UPDATE [0x%x] %i %i %ix%i -- pix = %p", cw->win, x, y, w, h, pix);
evas_object_image_data_update_add(cw->obj, x, y, w, h);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_data_update_add(o, x, y, w, h);
}
2012-06-28 00:03:44 -07:00
}
2010-02-10 23:40:44 -08:00
}
evas_object_image_data_set(cw->obj, pix);
}
}
free(r);
if (cw->shaped || cw->shape_changed)
_e_comp_win_shape_rectangles_apply(cw, cw->shape->shape_rects, cw->shape->shape_rects_num);
cw->shape_changed = 0;
}
else
2010-06-08 07:58:08 -07:00
{
DBG("UPDATE [0x%x] NO RECTS!!! %i %i - %i %i",
cw->win, cw->up->w, cw->up->h, cw->up->tw, cw->up->th);
// causes updates to be flagged when not needed - disabled
// cw->update = 1;
2010-06-08 07:58:08 -07:00
}
}
// FIXME: below cw update check screws with show
2011-03-19 23:12:17 -07:00
if (/*(!cw->update) &&*/ (cw->visible) && (cw->dmg_updates >= 1) &&
(cw->show_ready) && (!cw->bd || cw->bd->visible))
{
if (!evas_object_visible_get(cw->effect_obj))
{
2012-06-28 00:03:44 -07:00
if (!cw->hidden_override)
{
_e_comp_child_show(cw);
_e_comp_win_render_queue(cw);
2012-06-28 00:03:44 -07:00
}
if (!cw->show_anim)
{
edje_object_signal_emit(cw->shobj, "e,state,visible,on", "e");
if (!cw->animating)
{
cw->c->animating++;
}
cw->animating = 1;
cw->pending_count++;
_e_comp_event_source_visibility(cw);
2012-06-28 00:03:44 -07:00
cw->show_anim = EINA_TRUE;
}
}
}
if (cw->geom_update) _e_comp_win_geometry_update(cw);
if ((cw->shobj) && (cw->obj))
{
if (pshaped != cw->shaped)
{
if (_e_comp_win_do_shadow(cw))
edje_object_signal_emit(cw->shobj, "e,state,shadow,on", "e");
else
edje_object_signal_emit(cw->shobj, "e,state,shadow,off", "e");
}
}
if (conf->grab) ecore_x_ungrab();
}
static void
_e_comp_pre_swap(void *data, Evas *e EINA_UNUSED)
{
E_Comp *c = data;
if (conf->grab)
{
if (c->grabbed)
{
ecore_x_ungrab();
c->grabbed = 0;
}
}
}
static Eina_Bool
_e_comp_cb_delayed_update_timer(void *data)
2010-06-08 07:58:08 -07:00
{
E_Comp *c = data;
_e_comp_render_queue(c);
2010-06-08 07:58:08 -07:00
c->new_up_timer = NULL;
return ECORE_CALLBACK_CANCEL;
2010-06-08 07:58:08 -07:00
}
static void
_e_comp_fps_update(E_Comp *c)
{
if (conf->fps_show)
{
if (!c->fps_bg)
{
c->fps_bg = evas_object_rectangle_add(c->evas);
evas_object_color_set(c->fps_bg, 0, 0, 0, 128);
evas_object_layer_set(c->fps_bg, E_COMP_CANVAS_LAYER_MAX);
evas_object_lower(c->fps_bg);
evas_object_show(c->fps_bg);
c->fps_fg = evas_object_text_add(c->evas);
evas_object_text_font_set(c->fps_fg, "Sans", 10);
evas_object_text_text_set(c->fps_fg, "???");
evas_object_color_set(c->fps_fg, 255, 255, 255, 255);
evas_object_layer_set(c->fps_fg, E_COMP_CANVAS_LAYER_MAX);
evas_object_stack_above(c->fps_fg, c->fps_bg);
evas_object_show(c->fps_fg);
}
}
else
{
if (c->fps_fg)
{
evas_object_del(c->fps_fg);
c->fps_fg = NULL;
}
if (c->fps_bg)
{
evas_object_del(c->fps_bg);
c->fps_bg = NULL;
}
}
}
static void
_e_comp_win_release(E_Comp_Win *cw)
{
Eina_List *l;
Evas_Object *o;
if (cw->xim)
{
evas_object_image_size_set(cw->obj, 1, 1);
evas_object_image_data_set(cw->obj, NULL);
ecore_x_image_free(cw->xim);
cw->xim = NULL;
}
if (!cw->real_obj)
{
evas_object_image_native_surface_set(cw->obj, NULL);
cw->native = 0;
}
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
if (cw->xim)
{
evas_object_image_size_set(o, 1, 1);
evas_object_image_data_set(o, NULL);
}
evas_object_image_native_surface_set(o, NULL);
}
if (cw->pixmap)
{
ecore_x_pixmap_free(cw->pixmap);
cw->pixmap = 0;
cw->pw = 0;
cw->ph = 0;
ecore_x_e_comp_pixmap_set(cw->win, cw->pixmap);
cw->show_ready = 0; // hmm maybe not needed?
}
if (cw->redirected)
{
// we redirect all subwindows anyway
// ecore_x_composite_unredirect_window(cw->win, ECORE_X_COMPOSITE_UPDATE_MANUAL);
cw->redirected = 0;
}
if (cw->damage)
{
Ecore_X_Region parts;
eina_hash_del(damages, e_util_winid_str_get(cw->damage), cw);
parts = ecore_x_region_new(NULL, 0);
ecore_x_damage_subtract(cw->damage, 0, parts);
ecore_x_region_free(parts);
ecore_x_damage_free(cw->damage);
cw->damage = 0;
}
}
static void
_e_comp_win_adopt(E_Comp_Win *cw)
{
if (!cw->damage)
{
cw->damage = ecore_x_damage_new
(cw->win, ECORE_X_DAMAGE_REPORT_DELTA_RECTANGLES);
eina_hash_add(damages, e_util_winid_str_get(cw->damage), cw);
}
if (!cw->update)
{
cw->update = 1;
cw->c->updates = eina_list_append(cw->c->updates, cw);
}
cw->redirected = 1;
if (cw->bd) e_comp_win_reshadow(cw);
e_comp_render_update_resize(cw->up, cw->pw, cw->ph);
e_comp_render_update_add(cw->up, 0, 0, cw->pw, cw->ph);
_e_comp_win_damage(cw, 0, 0, cw->w, cw->h, 0);
_e_comp_win_render_queue(cw);
}
static void
_e_comp_cb_nocomp_begin(E_Comp *c)
{
E_Comp_Win *cw, *cwf;
if (c->nocomp) return;
if (c->nocomp_delay_timer)
{
ecore_timer_del(c->nocomp_delay_timer);
c->nocomp_delay_timer = NULL;
}
cwf = _e_comp_fullscreen_check(c);
if (!cwf) return;
EINA_INLIST_FOREACH(c->wins, cw)
{
_e_comp_win_release(cw);
}
cw = cwf;
INF("NOCOMP win %x effect_obj %p", cw->win, cw->effect_obj);
_e_comp_win_release(cw);
ecore_x_composite_unredirect_subwindows
(c->man->root, ECORE_X_COMPOSITE_UPDATE_MANUAL);
c->nocomp = 1;
c->render_overflow = OVER_FLOW;
ecore_x_window_hide(c->win);
ecore_evas_manual_render_set(c->ee, EINA_TRUE);
ecore_evas_resize(c->ee, 1, 1);
edje_file_cache_flush();
edje_collection_cache_flush();
evas_image_cache_flush(c->evas);
evas_font_cache_flush(c->evas);
evas_render_dump(c->evas);
cw->nocomp = 1;
if (cw->redirected)
{
// ecore_x_composite_unredirect_window(cw->win, ECORE_X_COMPOSITE_UPDATE_MANUAL);
cw->redirected = 0;
}
if (cw->update_timeout)
{
ecore_timer_del(cw->update_timeout);
cw->update_timeout = NULL;
}
if (cw->update)
{
cw->update = 0;
cw->c->updates = eina_list_remove(cw->c->updates, cw);
}
if (cw->counter)
{
if (cw->bd)
ecore_x_e_comp_sync_cancel_send(cw->bd->client.win);
else
ecore_x_e_comp_sync_cancel_send(cw->win);
ecore_x_sync_counter_inc(cw->counter, 1);
}
DBG("JOB2...");
_e_comp_render_queue(c);
}
static void
_e_comp_cb_nocomp_end(E_Comp *c)
{
E_Comp_Win *cw;
if (!c->nocomp) return;
ecore_x_composite_redirect_subwindows
(c->man->root, ECORE_X_COMPOSITE_UPDATE_MANUAL);
INF("COMP RESUME!");
c->nocomp = 0;
c->render_overflow = OVER_FLOW;
// ecore_evas_manual_render_set(c->ee, conf->lock_fps);
ecore_evas_manual_render_set(c->ee, EINA_FALSE);
ecore_evas_resize(c->ee, c->man->w, c->man->h);
ecore_x_window_show(c->win);
EINA_INLIST_FOREACH(c->wins, cw)
{
if (!cw->nocomp)
{
if ((cw->input_only) || (cw->invalid) || cw->real_obj) continue;
if (cw->nocomp_need_update)
{
cw->nocomp_need_update = EINA_FALSE;
e_comp_render_update_resize(cw->up, cw->pw, cw->ph);
e_comp_render_update_add(cw->up, 0, 0, cw->pw, cw->ph);
}
_e_comp_win_adopt(cw);
continue;
}
cw->nocomp = 0;
_e_comp_win_adopt(cw);
INF("restore comp %x --- %p", cw->win, cw->effect_obj);
if (cw->visible)
{
if (!cw->hidden_override) _e_comp_child_show(cw);
cw->pending_count++;
_e_comp_event_source_visibility(cw);
// no need for effect
}
if (cw->counter)
{
if (cw->bd)
ecore_x_e_comp_sync_begin_send(cw->bd->client.win);
else
ecore_x_e_comp_sync_begin_send(cw->win);
}
}
}
static Eina_Bool
_e_comp_cb_nocomp_begin_timeout(void *data)
{
E_Comp *c = data;
c->nocomp_delay_timer = NULL;
if (c->nocomp_override == 0)
{
if (_e_comp_fullscreen_check(c)) c->nocomp_want = 1;
_e_comp_cb_nocomp_begin(c);
}
return EINA_FALSE;
}
static Eina_Bool
_e_comp_cb_update(E_Comp *c)
{
E_Comp_Win *cw;
Eina_List *new_updates = NULL; // for failed pixmap fetches - get them next frame
Eina_List *update_done = NULL;
// static int doframeinfo = -1;
if (!c) return EINA_FALSE;
c->update_job = NULL;
DBG("UPDATE ALL");
if (c->nocomp) goto nocomp;
if (conf->grab)
{
ecore_x_grab();
ecore_x_sync();
c->grabbed = 1;
}
EINA_LIST_FREE(c->updates, cw)
{
if (conf->efl_sync)
{
if (((cw->counter) && (cw->drawme)) || (!cw->counter))
{
_e_comp_win_update(cw);
if (cw->drawme)
{
update_done = eina_list_append(update_done, cw);
cw->drawme = 0;
}
}
else
cw->update = 0;
}
else
_e_comp_win_update(cw);
if (cw->update)
{
new_updates = eina_list_append(new_updates, cw);
}
if (cw->geom_update) _e_comp_win_geometry_update(cw);
}
_e_comp_fps_update(c);
if (conf->fps_show)
{
char buf[128];
double fps = 0.0, t, dt;
int i;
Evas_Coord x = 0, y = 0, w = 0, h = 0;
E_Zone *z;
t = ecore_time_get();
if (conf->fps_average_range < 1)
conf->fps_average_range = 30;
else if (conf->fps_average_range > 120)
conf->fps_average_range = 120;
dt = t - c->frametimes[conf->fps_average_range - 1];
if (dt > 0.0) fps = (double)conf->fps_average_range / dt;
else fps = 0.0;
if (fps > 0.0) snprintf(buf, sizeof(buf), "FPS: %1.1f", fps);
else snprintf(buf, sizeof(buf), "N/A");
2012-06-28 00:03:44 -07:00
for (i = 121; i >= 1; i--)
c->frametimes[i] = c->frametimes[i - 1];
c->frametimes[0] = t;
c->frameskip++;
if (c->frameskip >= conf->fps_average_range)
{
c->frameskip = 0;
evas_object_text_text_set(c->fps_fg, buf);
}
evas_object_geometry_get(c->fps_fg, NULL, NULL, &w, &h);
w += 8;
h += 8;
z = e_util_zone_current_get(c->man);
if (z)
{
switch (conf->fps_corner)
{
case 3: // bottom-right
x = z->x + z->w - w;
y = z->y + z->h - h;
break;
2011-03-19 23:12:17 -07:00
case 2: // bottom-left
x = z->x;
y = z->y + z->h - h;
break;
2011-03-19 23:12:17 -07:00
case 1: // top-right
x = z->x + z->w - w;
y = z->y;
break;
2011-03-19 23:12:17 -07:00
default: // 0 // top-left
x = z->x;
y = z->y;
break;
}
}
evas_object_move(c->fps_bg, x, y);
evas_object_resize(c->fps_bg, w, h);
evas_object_move(c->fps_fg, x + 4, y + 4);
}
if (conf->lock_fps)
2010-06-08 07:58:08 -07:00
{
DBG("MANUAL RENDER...");
// if (!c->nocomp) ecore_evas_manual_render(c->ee);
2010-06-08 07:58:08 -07:00
}
if (conf->efl_sync)
2011-03-19 23:12:17 -07:00
{
EINA_LIST_FREE(update_done, cw)
{
ecore_x_sync_counter_inc(cw->counter, 1);
}
}
if (conf->grab)
{
if (c->grabbed)
{
c->grabbed = 0;
ecore_x_ungrab();
}
}
2010-06-08 07:58:08 -07:00
if (new_updates)
{
DBG("JOB1...");
2010-06-08 07:58:08 -07:00
if (c->new_up_timer) ecore_timer_del(c->new_up_timer);
2011-03-19 23:12:17 -07:00
c->new_up_timer =
ecore_timer_add(0.001, _e_comp_cb_delayed_update_timer, c);
// _e_comp_render_queue(c);
2010-06-08 07:58:08 -07:00
}
c->updates = new_updates;
if (!c->animating) c->render_overflow--;
/*
if (doframeinfo == -1)
{
doframeinfo = 0;
if (getenv("DFI")) doframeinfo = 1;
}
if (doframeinfo)
{
static double t0 = 0.0;
double td, t;
t = ecore_time_get();
td = t - t0;
if (td > 0.0)
{
int fps, i;
fps = 1.0 / td;
for (i = 0; i < fps; i+= 2) putchar('=');
printf(" : %3.3f", 1.0 / td);
}
t0 = t;
}
*/
2011-03-19 23:12:17 -07:00
nocomp:
cw = _e_comp_fullscreen_check(c);
if (cw)
{
if (conf->nocomp_fs)
{
if ((!c->nocomp) && (!c->nocomp_override > 0))
{
if (!c->nocomp_delay_timer)
c->nocomp_delay_timer = ecore_timer_add
(1.0, _e_comp_cb_nocomp_begin_timeout, c);
}
}
}
else
{
c->nocomp_want = 0;
if (c->nocomp_delay_timer)
{
ecore_timer_del(c->nocomp_delay_timer);
c->nocomp_delay_timer = NULL;
}
if (c->nocomp)
_e_comp_cb_nocomp_end(c);
}
DBG("UPDATE ALL DONE: overflow = %i", c->render_overflow);
2010-06-08 07:58:08 -07:00
if (c->render_overflow <= 0)
{
2010-06-08 07:58:08 -07:00
c->render_overflow = 0;
if (c->render_animator) c->render_animator = NULL;
return ECORE_CALLBACK_CANCEL;
}
return ECORE_CALLBACK_RENEW;
}
static void
_e_comp_cb_job(void *data)
{
DBG("UPDATE ALL JOB...");
_e_comp_cb_update(data);
}
static Eina_Bool
_e_comp_cb_animator(void *data)
{
return _e_comp_cb_update(data);
}
static void
_e_comp_render_queue(E_Comp *c)
{
/* FIXME workaround */
if (!c) return;
if (conf->lock_fps)
{
if (c->render_animator)
{
c->render_overflow = OVER_FLOW;
return;
}
c->render_animator = ecore_animator_add(_e_comp_cb_animator, c);
}
else
{
if (c->update_job)
{
DBG("UPDATE JOB DEL...");
ecore_job_del(c->update_job);
c->update_job = NULL;
c->render_overflow = 0;
}
DBG("UPDATE JOB ADD...");
c->update_job = ecore_job_add(_e_comp_cb_job, c);
}
}
static void
_e_comp_win_render_queue(E_Comp_Win *cw)
{
if (cw->real_obj) return;
DBG("JOB3...");
_e_comp_render_queue(cw->c);
}
static E_Comp *
_e_comp_find(Ecore_X_Window root)
{
Eina_List *l;
E_Comp *c;
// fixme: use hash if compositors list > 4
EINA_LIST_FOREACH(compositors, l, c)
{
if (c->man->root == root) return c;
}
return NULL;
}
static E_Comp_Win *
_e_comp_win_find(Ecore_X_Window win)
{
return eina_hash_find(windows, e_util_winid_str_get(win));
}
static E_Comp_Win *
_e_comp_border_client_find(Ecore_X_Window win)
{
return eina_hash_find(borders, e_util_winid_str_get(win));
}
static E_Comp_Win *
_e_comp_win_damage_find(Ecore_X_Damage damage)
{
return eina_hash_find(damages, e_util_winid_str_get(damage));
}
static Eina_Bool
_e_comp_ignore_find(Ecore_X_Window win)
{
return !!eina_hash_find(ignores, e_util_winid_str_get(win));
}
static Eina_Bool
_e_comp_win_is_borderless(E_Comp_Win *cw)
{
if (!cw->bd) return 1;
if ((cw->bd->client.border.name) &&
(!strcmp(cw->bd->client.border.name, "borderless")))
return 1;
return 0;
}
static Eina_Bool
_e_comp_win_do_shadow(E_Comp_Win *cw)
{
if (cw->shaped) return 0;
if (cw->bd && cw->bd->theme_shadow) return 0;
if (cw->real_obj)
{
return ((!!cw->pop) || (!!cw->menu));
}
if (cw->argb)
{
if (_e_comp_win_is_borderless(cw)) return 0;
}
return 1;
}
static Eina_Bool
_e_comp_win_damage_timeout(void *data)
{
E_Comp_Win *cw = data;
if (!cw->update)
{
if (cw->update_timeout)
{
ecore_timer_del(cw->update_timeout);
cw->update_timeout = NULL;
}
cw->update = 1;
cw->c->updates = eina_list_append(cw->c->updates, cw);
}
cw->drawme = 1;
_e_comp_win_render_queue(cw);
cw->update_timeout = NULL;
return ECORE_CALLBACK_CANCEL;
}
static void
_e_comp_object_del(void *data, void *obj)
{
E_Comp_Win *cw = data;
if (!cw) return;
_e_comp_win_render_queue(cw);
if (obj == cw->bd)
{
if (cw->counter)
{
if (cw->bd)
ecore_x_e_comp_sync_cancel_send(cw->bd->client.win);
else
ecore_x_e_comp_sync_cancel_send(cw->win);
ecore_x_sync_counter_inc(cw->counter, 1);
}
if (cw->bd) eina_hash_del(borders, e_util_winid_str_get(cw->bd->client.win), cw);
cw->bd->cw = NULL;
cw->bd = NULL;
evas_object_data_del(cw->shobj, "border");
// hmm - lockup?
// cw->counter = 0;
}
else if (obj == cw->pop)
{
cw->pop->cw = NULL;
cw->pop = NULL;
evas_object_data_del(cw->shobj, "popup");
}
else if (obj == cw->menu)
{
cw->menu->cw = NULL;
cw->menu = NULL;
evas_object_data_del(cw->shobj, "menu");
}
if (cw->dfn)
{
e_object_delfn_del(obj, cw->dfn);
cw->dfn = NULL;
}
cw->eobj = NULL;
e_comp_win_del(cw);
}
static void
_e_comp_done_defer(E_Comp_Win *cw)
{
if (cw->animating)
{
cw->c->animating--;
}
cw->animating = 0;
_e_comp_win_render_queue(cw);
cw->force = 1;
if (cw->defer_hide) _e_comp_win_hide(cw);
cw->force = 1;
if (cw->delete_me)
{
// if (cw->real_obj && cw->eobj)
// e_object_unref(E_OBJECT(cw->eobj));
_e_comp_win_del(cw);
}
else cw->force = 0;
}
static void
_e_comp_show_done(void *data, Evas_Object *obj EINA_UNUSED, const char *emission EINA_UNUSED, const char *source EINA_UNUSED)
{
E_Comp_Win *cw = data;
_e_comp_done_defer(cw);
}
static void
_e_comp_hide_done(void *data, Evas_Object *obj EINA_UNUSED, const char *emission EINA_UNUSED, const char *source EINA_UNUSED)
{
E_Comp_Win *cw = data;
_e_comp_done_defer(cw);
}
static void
_e_comp_win_sync_setup(E_Comp_Win *cw, Ecore_X_Window win)
{
if (!conf->efl_sync) return;
if (cw->bd)
{
if (_e_comp_win_is_borderless(cw) ||
(conf->loose_sync))
cw->counter = ecore_x_e_comp_sync_counter_get(win);
else
ecore_x_e_comp_sync_cancel_send(win);
}
else
cw->counter = ecore_x_e_comp_sync_counter_get(win);
if (cw->counter)
{
ecore_x_e_comp_sync_begin_send(win);
ecore_x_sync_counter_inc(cw->counter, 1);
}
}
static void
_e_comp_win_shadow_setup(E_Comp_Win *cw)
{
Evas_Object *o;
int ok = 0;
char buf[4096];
Eina_List *list = NULL, *l;
E_Comp_Match *m;
Eina_Stringshare *reshadow_group = NULL;
Eina_Bool focus = EINA_FALSE, urgent = EINA_FALSE, skip = EINA_FALSE, fast = EINA_FALSE, reshadow, no_shadow = EINA_FALSE;
const char *title = NULL, *name = NULL, *clas = NULL, *role = NULL;
Ecore_X_Window_Type primary_type = ECORE_X_WINDOW_TYPE_UNKNOWN;
edje_object_file_get(cw->shobj, NULL, &reshadow_group);
if (!cw->real_obj)
evas_object_image_smooth_scale_set(cw->obj, conf->smooth_windows);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_smooth_scale_set(o, conf->smooth_windows);
}
if (cw->bd)
{
list = conf->match.borders;
title = cw->bd->client.icccm.title;
if (cw->bd->client.netwm.name) title = cw->bd->client.netwm.name;
name = cw->bd->client.icccm.name;
clas = cw->bd->client.icccm.class;
role = cw->bd->client.icccm.window_role;
primary_type = cw->bd->client.netwm.type;
skip = (conf->match.disable_borders);
fast = conf->fast_borders;
}
else if (cw->pop)
{
// FIXME: i only added "shelf" as a name for popups that are shelves
// ... need more nmes like for pager popup, evertything, exebuf
// etc. etc.
list = conf->match.popups;
2012-06-28 00:03:44 -07:00
name = cw->pop->name;
skip = conf->match.disable_popups;
if (cw->pop->name && (!skip))
skip = (!strncmp(cw->pop->name, "noshadow", 8));
fast = conf->fast_popups;
}
else if (cw->menu)
{
// FIXME: e has no way to tell e menus apart... need naming
list = conf->match.menus;
skip = (conf->match.disable_menus);
fast = conf->fast_menus;
}
else if (cw->real_obj)
{
list = conf->match.objects;
name = evas_object_name_get(cw->obj);
//skip = conf->match.disable_objects;
skip = 1;
fast = conf->fast_objects;
}
else
{
list = conf->match.overrides;
title = cw->title;
name = cw->name;
clas = cw->clas;
role = cw->role;
primary_type = cw->primary_type;
skip = (conf->match.disable_overrides);
fast = conf->fast_overrides;
}
if (!skip)
{
EINA_LIST_FOREACH(list, l, m)
{
if (((m->title) && (!title)) ||
((title) && (m->title) && (!e_util_glob_match(title, m->title))))
continue;
if (((m->name) && (!name)) ||
((name) && (m->name) && (!e_util_glob_match(name, m->name))))
continue;
if (((m->clas) && (!clas)) ||
((clas) && (m->clas) && (!e_util_glob_match(clas, m->clas))))
continue;
if (((m->role) && (!role)) ||
((role) && (m->role) && (!e_util_glob_match(role, m->role))))
continue;
if ((primary_type != ECORE_X_WINDOW_TYPE_UNKNOWN) &&
(m->primary_type != ECORE_X_WINDOW_TYPE_UNKNOWN) &&
((int)primary_type != m->primary_type))
continue;
if (cw->bd)
{
if (m->borderless != 0)
{
int borderless = 0;
if ((cw->bd->client.mwm.borderless) || (cw->bd->borderless))
borderless = 1;
if (!(((m->borderless == -1) && (!borderless)) ||
((m->borderless == 1) && (borderless))))
continue;
}
if (m->dialog != 0)
{
int dialog = 0;
if (((cw->bd->client.icccm.transient_for != 0) ||
(cw->bd->client.netwm.type == ECORE_X_WINDOW_TYPE_DIALOG)))
dialog = 1;
if (!(((m->dialog == -1) && (!dialog)) ||
((m->dialog == 1) && (dialog))))
continue;
}
if (m->accepts_focus != 0)
{
int accepts_focus = 0;
if (cw->bd->client.icccm.accepts_focus)
accepts_focus = 1;
if (!(((m->accepts_focus == -1) && (!accepts_focus)) ||
((m->accepts_focus == 1) && (accepts_focus))))
continue;
}
if (m->vkbd != 0)
{
int vkbd = 0;
if (cw->bd->client.vkbd.vkbd)
vkbd = 1;
if (!(((m->vkbd == -1) && (!vkbd)) ||
((m->vkbd == 1) && (vkbd))))
continue;
}
if (m->quickpanel != 0)
{
int quickpanel = 0;
if (cw->bd->client.illume.quickpanel.quickpanel)
quickpanel = 1;
if (!(((m->quickpanel == -1) && (!quickpanel)) ||
((m->quickpanel == 1) && (quickpanel))))
continue;
}
if (m->argb != 0)
{
if (!(((m->argb == -1) && (!cw->argb)) ||
((m->argb == 1) && (cw->argb))))
continue;
}
if (m->fullscreen != 0)
{
int fullscreen = 0;
if (cw->bd->client.netwm.state.fullscreen)
fullscreen = 1;
if (!(((m->fullscreen == -1) && (!fullscreen)) ||
((m->fullscreen == 1) && (fullscreen))))
continue;
}
if (m->modal != 0)
{
int modal = 0;
if (cw->bd->client.netwm.state.modal)
modal = 1;
if (!(((m->modal == -1) && (!modal)) ||
((m->modal == 1) && (modal))))
continue;
}
}
focus = m->focus;
urgent = m->urgent;
no_shadow = m->no_shadow;
if (m->shadow_style)
{
if (fast)
{
snprintf(buf, sizeof(buf), "e/comp/border/%s/fast", m->shadow_style);
reshadow = ok = !e_util_strcmp(reshadow_group, buf);
if (!ok)
ok = e_theme_edje_object_set(cw->shobj, "base/theme/borders", buf);
}
if (!ok)
{
snprintf(buf, sizeof(buf), "e/comp/border/%s", m->shadow_style);
reshadow = ok = !e_util_strcmp(reshadow_group, buf);
if (!ok)
ok = e_theme_edje_object_set(cw->shobj, "base/theme/borders", buf);
}
if (ok) break;
}
}
}
while (!ok)
{
if (skip || (cw->bd && cw->bd->client.e.state.video))
{
reshadow = ok = !e_util_strcmp(reshadow_group, "e/comp/border/none");
if (!ok)
ok = e_theme_edje_object_set(cw->shobj, "base/theme/borders", "e/comp/border/none");
}
if (ok) break;
if (conf->shadow_style)
{
if (fast)
{
snprintf(buf, sizeof(buf), "e/comp/border/%s/fast", conf->shadow_style);
reshadow = ok = !e_util_strcmp(reshadow_group, buf);
if (!ok)
ok = e_theme_edje_object_set(cw->shobj, "base/theme/borders", buf);
}
if (!ok)
{
snprintf(buf, sizeof(buf), "e/comp/border/%s", conf->shadow_style);
reshadow = ok = !e_util_strcmp(reshadow_group, buf);
if (!ok)
ok = e_theme_edje_object_set(cw->shobj, "base/theme/borders", buf);
}
}
if (!ok)
{
if (fast)
{
reshadow = ok = !e_util_strcmp(reshadow_group, "e/comp/border/default/fast");
if (!ok)
ok = e_theme_edje_object_set(cw->shobj, "base/theme/borders", "e/comp/border/default/fast");
}
if (!ok)
{
reshadow = ok = !e_util_strcmp(reshadow_group, "e/comp/border/default");
if (!ok)
ok = e_theme_edje_object_set(cw->shobj, "base/theme/borders", "e/comp/border/default");
}
}
break;
}
if (reshadow)
{
if (!cw->bd) return;
#ifdef BORDER_ZOOMAPS
if (cw->bd->bg_object && (e_zoomap_child_get(cw->zoomobj) == cw->bd->bg_object))
#else
if (cw->bd->bg_object && (edje_object_part_swallow_get(cw->shobj, "e.swallow.content") == cw->bd->bg_object))
#endif
return;
}
if (_e_comp_win_do_shadow(cw) && (!no_shadow))
edje_object_signal_emit(cw->shobj, "e,state,shadow,on", "e");
else
edje_object_signal_emit(cw->shobj, "e,state,shadow,off", "e");
if (cw->bd)
{
if (focus || (cw->bd && cw->bd->focused))
edje_object_signal_emit(cw->shobj, "e,state,focus,on", "e");
else
edje_object_signal_emit(cw->shobj, "e,state,focus,off", "e");
if (urgent || (cw->bd && cw->bd->client.icccm.urgent))
edje_object_signal_emit(cw->shobj, "e,state,urgent,on", "e");
else
edje_object_signal_emit(cw->shobj, "e,state,urgent,off", "e");
}
if (cw->visible)
edje_object_signal_emit(cw->shobj, "e,state,visible,on", "e");
else
edje_object_signal_emit(cw->shobj, "e,state,visible,off", "e");
#ifndef BORDER_ZOOMAPS
if (cw->eobj)
#endif
e_zoomap_child_set(cw->zoomobj, NULL);
if (cw->bd && cw->bd->bg_object)
{
edje_object_part_swallow(cw->bd->bg_object, "e.swallow.client", cw->obj);
#ifdef BORDER_ZOOMAPS
e_zoomap_child_set(cw->zoomobj, cw->bd->bg_object);
#else
edje_object_part_swallow(cw->shobj, "e.swallow.content", cw->bd->bg_object);
#endif
no_shadow = 1;
}
else
{
if (cw->bd) no_shadow = 1;
#ifdef BORDER_ZOOMAPS
e_zoomap_child_set(cw->zoomobj, cw->obj);
#else
if (cw->zoomobj)
{
e_zoomap_child_set(cw->zoomobj, cw->obj);
edje_object_part_swallow(cw->shobj, "e.swallow.content", cw->zoomobj);
}
else
edje_object_part_swallow(cw->shobj, "e.swallow.content", cw->obj);
#endif
}
2013-04-23 09:09:51 -07:00
if (cw->menu
#ifdef BORDER_ZOOMAPS
|| cw->bd
#endif
)
e_zoomap_child_edje_solid_setup(cw->zoomobj);
#ifdef BORDER_ZOOMAPS
edje_object_part_swallow(cw->shobj, "e.swallow.content", cw->zoomobj);
#endif
if (!cw->visible) return;
if (!cw->animating)
{
cw->c->animating++;
}
cw->animating = 1;
_e_comp_win_render_queue(cw);
cw->pending_count++;
_e_comp_event_source_visibility(cw);
}
static void
_e_comp_cb_win_mirror_del(void *data, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED)
{
E_Comp_Win *cw;
2011-03-19 23:12:17 -07:00
if (!(cw = data)) return;
cw->obj_mirror = eina_list_remove(cw->obj_mirror, obj);
}
static Evas_Object *
_e_comp_win_mirror_add(E_Comp_Win *cw)
{
Evas_Object *o;
int w, h;
2013-04-19 08:47:06 -07:00
EINA_SAFETY_ON_TRUE_RETURN_VAL(((!cw->pixmap) || (!cw->pw) || (!cw->ph)) && (!cw->real_obj) && ((!cw->w) || (!cw->h)), NULL);
o = evas_object_image_filled_add(cw->c->evas);
evas_object_image_colorspace_set(o, EVAS_COLORSPACE_ARGB8888);
cw->obj_mirror = eina_list_append(cw->obj_mirror, o);
evas_object_image_smooth_scale_set(o, conf->smooth_windows);
evas_object_event_callback_add(o, EVAS_CALLBACK_DEL, _e_comp_cb_win_mirror_del, cw);
2011-03-19 23:12:17 -07:00
if ((cw->pixmap) && (cw->pw > 0) && (cw->ph > 0))
{
unsigned int *pix = NULL;
Eina_Bool alpha, argb = EINA_FALSE;
alpha = evas_object_image_alpha_get(cw->obj);
w = cw->pw, h = cw->ph;
evas_object_image_alpha_set(o, alpha);
evas_object_image_size_set(o, w, h);
if (cw->shaped)
pix = evas_object_image_data_get(cw->obj, 0);
else
{
if (cw->native)
{
Evas_Native_Surface ns;
ns.version = EVAS_NATIVE_SURFACE_VERSION;
ns.type = EVAS_NATIVE_SURFACE_X11;
ns.data.x11.visual = cw->vis;
ns.data.x11.pixmap = cw->pixmap;
evas_object_image_native_surface_set(o, &ns);
}
else if (cw->xim)
{
argb = ecore_x_image_is_argb32_get(cw->xim);
if (argb)
pix = ecore_x_image_data_get(cw->xim, NULL, NULL, NULL);
else
pix = evas_object_image_data_get(cw->obj, EINA_FALSE);
}
}
if (pix)
{
evas_object_image_data_set(o, pix);
if (!argb)
evas_object_image_data_set(cw->obj, pix);
}
2011-03-19 23:12:17 -07:00
}
else if (cw->real_obj)
{
/* FIXME: the first mirror shown with vsync enabled won't render immediately */
evas_object_image_alpha_set(o, 1);
evas_object_geometry_get(cw->zoomobj, NULL, NULL, &w, &h);
evas_object_image_source_set(o, cw->zoomobj);
}
evas_object_image_data_update_add(o, 0, 0, w, h);
return o;
}
static E_Comp_Win *
_e_comp_win_dummy_add(E_Comp *c, Evas_Object *obj, E_Object *eobj, Eina_Bool nolayout)
{
E_Comp_Win *cw;
int x, y, w, h;
cw = calloc(1, sizeof(E_Comp_Win));
if (!cw) return NULL;
cw->not_in_layout = nolayout;
if (eobj)
{
switch (eobj->type)
{
case E_POPUP_TYPE:
cw->pop = (void*)eobj;
cw->pop->cw = cw;
cw->shape = cw->pop->shape;
cw->dfn = e_object_delfn_add(E_OBJECT(cw->pop), _e_comp_object_del, cw);
cw->show_ready = cw->pop->visible;
break;
case E_MENU_TYPE:
cw->menu = (void*)eobj;
cw->menu->cw = cw;
cw->shape = cw->menu->shape;
cw->dfn = e_object_delfn_add(E_OBJECT(cw->menu), _e_comp_object_del, cw);
cw->show_ready = cw->menu->cur.visible;
break;
default:
CRI("UNHANDLED");
}
cw->eobj = eobj;
e_object_ref(E_OBJECT(cw->eobj));
}
else
cw->show_ready = evas_object_visible_get(obj);
cw->obj = obj;
cw->c = c;
cw->real_hid = !cw->show_ready;
cw->real_obj = 1;
cw->argb = 1;
cw->opacity = 255.0;
if (cw->shape) cw->shape->comp_win = cw;
if (conf->grab) ecore_x_grab();
evas_object_geometry_get(obj, &x, &y, &w, &h);
cw->effect_obj = edje_object_add(c->evas);
e_theme_edje_object_set(cw->effect_obj, "base/theme/comp", "e/comp/effects/none");
cw->shobj = edje_object_add(c->evas);
cw->zoomobj = e_zoomap_add(c->evas);
e_zoomap_smooth_set(cw->zoomobj, conf->smooth_windows);
_e_comp_win_shadow_setup(cw);
edje_object_part_swallow(cw->effect_obj, "e.swallow.content", cw->shobj);
edje_object_signal_callback_add(cw->shobj, "e,action,show,done", "e", _e_comp_show_done, cw);
edje_object_signal_callback_add(cw->shobj, "e,action,hide,done", "e", _e_comp_hide_done, cw);
_e_comp_win_configure(cw, x, y, w, h, 0);
if (!nolayout) _e_comp_win_layout_populate(cw);
if (cw->pop)
{
evas_object_data_set(cw->shobj, "popup", cw->pop);
evas_object_data_set(cw->effect_obj, "popup", cw->pop);
evas_object_name_set(cw->zoomobj, "cw->zoomobj::POPUP");
evas_object_name_set(cw->shobj, "cw->shobj::POPUP");
evas_object_name_set(cw->effect_obj, "cw->effect_obj::POPUP");
}
else if (cw->menu)
{
e_zoomap_always_set(cw->zoomobj, EINA_TRUE);
evas_object_data_set(cw->shobj, "menu", cw->menu);
evas_object_data_set(cw->effect_obj, "menu", cw->menu);
evas_object_name_set(cw->zoomobj, "cw->zoomobj::MENU");
evas_object_name_set(cw->shobj, "cw->shobj::MENU");
evas_object_name_set(cw->effect_obj, "cw->effect_obj::MENU");
}
cw->pending_count++;
_e_comp_event_source_add(cw);
evas_object_data_set(cw->effect_obj, "comp_win", cw);
evas_object_data_set(cw->shobj, "comp_win", cw);
evas_object_data_set(cw->obj, "comp_win", cw);
c->wins_invalid = 1;
c->wins = eina_inlist_append(c->wins, EINA_INLIST_GET(cw));
DBG(" [0x%x] add", cw->win);
if (conf->grab) ecore_x_ungrab();
if (cw->show_ready) _e_comp_win_show(cw);
return cw;
}
static void
_e_comp_win_bd_setup(E_Comp_Win *cw, E_Border *bd)
{
cw->bd = bd;
eina_hash_add(borders, e_util_winid_str_get(cw->bd->client.win), cw);
cw->dfn = e_object_delfn_add(E_OBJECT(cw->bd), _e_comp_object_del, cw);
cw->shape = cw->bd->shape;
cw->bd->cw = cw;
cw->opacity = cw->bd->client.netwm.opacity;
cw->eobj = E_OBJECT(cw->bd);
e_object_ref(cw->eobj);
// setup on show
// _e_comp_win_sync_setup(cw, cw->bd->client.win);
cw->input_only = cw->bd->client.initial_attributes.input_only;
cw->override = cw->bd->client.initial_attributes.override;
cw->vis = cw->bd->client.initial_attributes.visual;
cw->cmap = cw->bd->client.initial_attributes.colormap;
cw->depth = cw->bd->client.initial_attributes.depth;
}
static void
_e_comp_win_shape_init(E_Comp_Win *cw, int w, int h)
{
int i;
for (i = 0; i < cw->shape->shape_rects_num; i++)
E_RECTS_CLIP_TO_RECT(cw->shape->shape_rects[i].x, cw->shape->shape_rects[i].y,
cw->shape->shape_rects[i].w, cw->shape->shape_rects[i].h, 0, 0, w, h);
if (_e_comp_shaped_check(w, h, cw->shape->shape_rects, cw->shape->shape_rects_num))
cw->shape_changed = 1;
}
static E_Comp_Win *
_e_comp_win_add(E_Comp *c, Ecore_X_Window win, E_Border *bd)
{
E_Comp_Win *cw;
int w, h;
cw = calloc(1, sizeof(E_Comp_Win));
if (!cw) return NULL;
cw->win = win;
cw->c = c;
cw->real_hid = 1;
cw->opacity = 255.0;
if (conf->grab) ecore_x_grab();
if (bd)
{
_e_comp_win_bd_setup(cw, bd);
w = cw->bd->client.w, h = cw->bd->client.h;
}
else
{
Ecore_X_Window_Attributes att;
memset((&att), 0, sizeof(Ecore_X_Window_Attributes));
if (!ecore_x_window_attributes_get(cw->win, &att))
{
free(cw);
if (conf->grab) ecore_x_ungrab();
return NULL;
}
if ((!att.input_only) &&
((att.depth != 24) && (att.depth != 32)))
{
// printf("WARNING: window 0x%x not 24/32bpp -> %ibpp", cw->win, att.depth);
// cw->invalid = 1;
}
if (!att.override)
{
free(cw);
if (conf->grab) ecore_x_ungrab();
return NULL;
}
cw->input_only = att.input_only;
cw->override = att.override;
cw->vis = att.visual;
cw->cmap = att.colormap;
cw->depth = att.depth;
w = att.w, h = att.h;
if (!(att.event_mask.mine & ECORE_X_EVENT_MASK_WINDOW_PROPERTY))
ecore_x_event_mask_set(cw->win, ECORE_X_EVENT_MASK_WINDOW_PROPERTY);
}
if ((!cw->bd) && (!cw->menu))
{
2011-03-19 23:12:17 -07:00
char *netwm_title = NULL;
2011-03-19 23:12:17 -07:00
cw->title = ecore_x_icccm_title_get(cw->win);
if (ecore_x_netwm_name_get(cw->win, &netwm_title))
{
free(cw->title);
2011-03-19 23:12:17 -07:00
cw->title = netwm_title;
}
ecore_x_icccm_name_class_get(cw->win, &cw->name, &cw->clas);
cw->role = ecore_x_icccm_window_role_get(cw->win);
if (!ecore_x_netwm_window_type_get(cw->win, &cw->primary_type))
cw->primary_type = ECORE_X_WINDOW_TYPE_UNKNOWN;
{
E_Container *con;
con = e_util_container_current_get();
cw->bg_win = (con && (cw->win == con->bg_win));
}
cw->free_shape = 1;
2011-03-19 23:12:17 -07:00
// setup on show
// _e_comp_win_sync_setup(cw, cw->win);
}
if (!cw->counter)
{
// FIXME: config - disable ready timeout for non-counter wins
2011-03-19 23:12:17 -07:00
// cw->show_ready = 1;
}
if (cw->bd)
cw->argb = cw->bd->client.argb;
else
cw->argb = ecore_x_window_argb_get(cw->win);
eina_hash_add(windows, e_util_winid_str_get(cw->win), cw);
cw->inhash = 1;
if ((!cw->input_only) && (!cw->invalid))
{
cw->damage = ecore_x_damage_new
(cw->win, ECORE_X_DAMAGE_REPORT_DELTA_RECTANGLES);
eina_hash_add(damages, e_util_winid_str_get(cw->damage), cw);
cw->effect_obj = edje_object_add(c->evas);
e_theme_edje_object_set(cw->effect_obj, "base/theme/comp", "e/comp/effects/none");
cw->shobj = edje_object_add(c->evas);
#ifdef BORDER_ZOOMAPS
cw->zoomobj = e_zoomap_add(c->evas);
e_zoomap_smooth_set(cw->zoomobj, conf->smooth_windows);
#endif
cw->obj = evas_object_image_filled_add(c->evas);
evas_object_image_colorspace_set(cw->obj, EVAS_COLORSPACE_ARGB8888);
if (cw->argb) evas_object_image_alpha_set(cw->obj, 1);
else evas_object_image_alpha_set(cw->obj, 0);
if (cw->override)
evas_object_pass_events_set(cw->effect_obj, EINA_TRUE);
_e_comp_win_shadow_setup(cw);
edje_object_part_swallow(cw->effect_obj, "e.swallow.content", cw->shobj);
edje_object_signal_callback_add(cw->shobj, "e,action,show,done", "e", _e_comp_show_done, cw);
edje_object_signal_callback_add(cw->shobj, "e,action,hide,done", "e", _e_comp_hide_done, cw);
_e_comp_win_layout_populate(cw);
evas_object_show(cw->obj);
ecore_x_window_shape_events_select(cw->win, 1);
if (cw->bd)
{
_e_comp_win_shape_init(cw, w, h);
evas_object_data_set(cw->shobj, "border", cw->bd);
evas_object_data_set(cw->effect_obj, "border", cw->bd);
#ifdef BORDER_ZOOMAPS
evas_object_name_set(cw->zoomobj, "cw->zoomobj::BORDER");
#endif
evas_object_name_set(cw->shobj, "cw->shobj::BORDER");
evas_object_name_set(cw->effect_obj, "cw->effect_obj::BORDER");
}
else
{
#ifdef BORDER_ZOOMAPS
evas_object_name_set(cw->zoomobj, "cw->zoomobj::WINDOW");
#endif
evas_object_name_set(cw->shobj, "cw->shobj::WINDOW");
evas_object_name_set(cw->effect_obj, "cw->effect_obj::WINDOW");
}
evas_object_name_set(cw->obj, "cw->obj");
cw->pending_count++;
_e_comp_event_source_add(cw);
}
else
{
cw->effect_obj = evas_object_rectangle_add(c->evas);
_e_comp_win_layout_populate(cw);
evas_object_color_set(cw->effect_obj, 0, 0, 0, 0);
}
if (cw->shobj) evas_object_data_set(cw->shobj, "comp_win", cw);
evas_object_data_set(cw->effect_obj, "comp_win", cw);
c->wins_invalid = 1;
c->wins = eina_inlist_append(c->wins, EINA_INLIST_GET(cw));
cw->up = e_comp_render_update_new();
e_comp_render_update_tile_size_set(cw->up, 32, 32);
// for software:
e_comp_render_update_policy_set
(cw->up, E_COMP_RENDER_UPDATE_POLICY_HALF_WIDTH_OR_MORE_ROUND_UP_TO_FULL_WIDTH);
if (((!cw->input_only) && (!cw->invalid)) && (cw->override))
{
2012-06-28 00:03:44 -07:00
cw->redirected = 1;
// we redirect all subwindows anyway
// ecore_x_composite_redirect_window(cw->win, ECORE_X_COMPOSITE_UPDATE_MANUAL);
cw->dmg_updates = 0;
}
if (cw->bd) e_comp_win_opacity_set(cw, cw->opacity);
DBG(" [0x%x] add", cw->win);
if (conf->grab) ecore_x_ungrab();
return cw;
}
static void
_e_comp_win_del(E_Comp_Win *cw)
{
Evas_Object *o;
if (cw->eobj)
{
e_object_unref(E_OBJECT(cw->eobj));
cw->eobj = NULL;
}
E_FREE_FUNC(cw->opacity_set_timer, ecore_timer_del);
if (cw->animating)
{
cw->c->animating--;
}
cw->animating = 0;
if (cw->cw_above)
{
cw->cw_above->stack_below = eina_list_remove(cw->cw_above->stack_below, cw);
cw->cw_above = NULL;
}
if (cw->stack_below)
{
E_Comp_Win *cw2, *cwn = NULL;
Eina_List *l;
if (EINA_INLIST_GET(cw)->next)
cwn = EINA_INLIST_CONTAINER_GET(EINA_INLIST_GET(cw)->next, E_Comp_Win);
else if (EINA_INLIST_GET(cw)->prev)
cwn = EINA_INLIST_CONTAINER_GET(EINA_INLIST_GET(cw)->prev, E_Comp_Win);
if (cwn)
{
cwn->stack_below = cw->stack_below;
cw->stack_below = NULL;
EINA_LIST_FOREACH(cwn->stack_below, l, cw2)
cw2->cw_above = cwn;
_e_comp_win_restack(cwn);
}
else
{
EINA_LIST_FREE(cw->stack_below, cw2)
cw2->cw_above = NULL;
}
}
E_FREE_FUNC(cw->up, e_comp_render_update_free);
DBG(" [0x%x] del", cw->win);
if (cw->update_timeout)
{
ecore_timer_del(cw->update_timeout);
cw->update_timeout = NULL;
}
if (cw->ready_timeout)
{
ecore_timer_del(cw->ready_timeout);
cw->ready_timeout = NULL;
}
if (cw->dfn)
{
if (cw->bd)
{
eina_hash_del(borders, e_util_winid_str_get(cw->bd->client.win), cw);
e_object_delfn_del(E_OBJECT(cw->bd), cw->dfn);
cw->bd->cw = NULL;
cw->bd = NULL;
}
else if (cw->pop)
2011-03-19 23:12:17 -07:00
{
cw->pop->cw = NULL;
2011-03-19 23:12:17 -07:00
e_object_delfn_del(E_OBJECT(cw->pop), cw->dfn);
cw->pop = NULL;
}
else if (cw->menu)
2011-03-19 23:12:17 -07:00
{
e_object_delfn_del(E_OBJECT(cw->menu), cw->dfn);
cw->menu->cw = NULL;
2011-03-19 23:12:17 -07:00
cw->menu = NULL;
}
cw->dfn = NULL;
}
if (cw->update)
{
cw->update = 0;
cw->c->updates = eina_list_remove(cw->c->updates, cw);
}
_e_comp_win_release(cw);
if (cw->inhash)
eina_hash_del(windows, e_util_winid_str_get(cw->win), cw);
cw->inhash = 0;
if (!cw->delete_pending)
{
cw->c->wins_invalid = 1;
cw->c->wins = eina_inlist_remove(cw->c->wins, EINA_INLIST_GET(cw));
}
if ((!cw->delete_pending) && (!cw->input_only) && (!cw->invalid))
{
cw->pending_count++;
_e_comp_event_source_del(cw);
cw->delete_pending = 1;
return;
}
if (cw->pending_count) return;
if (cw->obj_mirror)
{
EINA_LIST_FREE(cw->obj_mirror, o)
{
if (cw->xim) evas_object_image_data_set(o, NULL);
evas_object_event_callback_del(o, EVAS_CALLBACK_DEL, _e_comp_cb_win_mirror_del);
evas_object_del(o);
}
}
if (cw->real_obj && cw->obj)
{
if (evas_object_layer_get(cw->obj) > E_COMP_CANVAS_LAYER_LAYOUT)
e_comp_override_del(cw->c);
evas_object_event_callback_del_full(cw->obj, EVAS_CALLBACK_DEL, _e_comp_injected_win_del_cb, cw);
evas_object_event_callback_del_full(cw->obj, EVAS_CALLBACK_HIDE, _e_comp_injected_win_hide_cb, cw);
evas_object_event_callback_del_full(cw->obj, EVAS_CALLBACK_FOCUS_IN, _e_comp_injected_win_focus_in_cb, cw);
evas_object_event_callback_del_full(cw->obj, EVAS_CALLBACK_FOCUS_OUT, _e_comp_injected_win_focus_out_cb, cw);
}
E_FREE_FUNC(cw->obj, evas_object_del);
E_FREE_FUNC(cw->zoomobj, evas_object_del);
E_FREE_FUNC(cw->shobj, evas_object_del);
E_FREE_FUNC(cw->effect_obj, evas_object_del);
if (cw->free_shape) E_FREE_FUNC(cw->shape, e_object_del);
free(cw->title);
free(cw->name);
free(cw->clas);
free(cw->role);
free(cw);
}
static void
_e_comp_win_show(E_Comp_Win *cw)
{
Eina_List *l;
Evas_Object *o;
int pw, ph;
if (cw->visible || cw->invalid) return;
cw->visible = 1;
pw = cw->pw, ph = cw->ph;
DBG(" [0x%x] sho ++ [redir=%i, pm=%x, dmg_up=%i]",
cw->win, cw->redirected, cw->pixmap, cw->dmg_updates);
_e_comp_win_configure(cw, cw->hidden.x, cw->hidden.y, cw->w, cw->h, cw->border);
if (cw->invalid) return;
if (cw->input_only)
{
if (!cw->shape) return;
cw->real_hid = 0;
cw->shape->visible = 0;
e_container_shape_show(cw->shape);
return;
}
cw->show_anim = EINA_FALSE;
2012-06-28 00:03:44 -07:00
// setup on show
if (cw->bd)
_e_comp_win_sync_setup(cw, cw->bd->client.win);
else if (cw->win)
_e_comp_win_sync_setup(cw, cw->win);
if (cw->real_hid)
{
DBG(" [0x%x] real hid - fix", cw->win);
cw->real_hid = 0;
if (cw->native)
{
evas_object_image_native_surface_set(cw->obj, NULL);
cw->native = 0;
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_native_surface_set(o, NULL);
}
}
if (cw->pixmap)
{
ecore_x_pixmap_free(cw->pixmap);
cw->pixmap = 0;
cw->pw = 0;
cw->ph = 0;
ecore_x_e_comp_pixmap_set(cw->win, cw->pixmap);
}
if (cw->xim)
{
evas_object_image_size_set(cw->obj, 1, 1);
evas_object_image_data_set(cw->obj, NULL);
ecore_x_image_free(cw->xim);
cw->xim = NULL;
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_size_set(o, 1, 1);
evas_object_image_data_set(o, NULL);
}
}
if (cw->redirected)
{
cw->redirected = 0;
cw->pw = 0;
cw->ph = 0;
}
if (!cw->win)
2011-03-19 23:12:17 -07:00
cw->dmg_updates = 1;
}
else
cw->dmg_updates = 1;
cw->geom_update = 1;
if (cw->win && ((!cw->redirected) || (!cw->pixmap)))
{
// we redirect all subwindows anyway
// ecore_x_composite_redirect_window(cw->win, ECORE_X_COMPOSITE_UPDATE_MANUAL);
/* #ifdef HAVE_WAYLAND_CLIENTS */
/* if ((cw->bd) && (cw->bd->borderless)) */
/* cw->pixmap = e_comp_wl_pixmap_get(cw->win); */
/* #endif */
if (!cw->pixmap)
cw->pixmap = ecore_x_composite_name_window_pixmap_get(cw->win);
if (cw->pixmap)
{
ecore_x_pixmap_geometry_get(cw->pixmap, NULL, NULL, &(cw->pw), &(cw->ph));
_e_comp_win_ready_timeout_setup(cw);
if ((cw->pw != pw) || (cw->ph != ph)) cw->geom_update = 1;
}
else
{
cw->pw = 0;
cw->ph = 0;
}
if ((cw->pw <= 0) || (cw->ph <= 0))
{
if (cw->pixmap)
{
ecore_x_pixmap_free(cw->pixmap);
cw->pixmap = 0;
}
// cw->show_ready = 0; // hmm maybe not needed?
}
cw->redirected = 1;
DBG(" [0x%x] up resize %ix%i", cw->win, cw->pw, cw->ph);
e_comp_render_update_resize(cw->up, cw->pw, cw->ph);
e_comp_render_update_add(cw->up, 0, 0, cw->pw, cw->ph);
evas_object_image_size_set(cw->obj, cw->pw, cw->ph);
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_size_set(o, cw->pw, cw->ph);
}
ecore_x_e_comp_pixmap_set(cw->win, cw->pixmap);
}
if ((cw->dmg_updates >= 1) && (cw->show_ready))
{
cw->defer_hide = 0;
if (!cw->hidden_override) _e_comp_child_show(cw);
edje_object_signal_emit(cw->shobj, "e,state,visible,on", "e");
if (!cw->animating)
{
cw->c->animating++;
}
cw->animating = 1;
_e_comp_win_render_queue(cw);
cw->pending_count++;
_e_comp_event_source_visibility(cw);
}
_e_comp_win_render_queue(cw);
if (!cw->shape) return;
cw->shape->visible = 0;
e_container_shape_show(cw->shape);
}
static void
_e_comp_win_real_hide(E_Comp_Win *cw)
{
if (!cw->bd) cw->real_hid = 1;
_e_comp_win_hide(cw);
}
static void
_e_comp_win_hide(E_Comp_Win *cw)
{
Eina_List *l;
Evas_Object *o;
if ((!cw->visible) && (!cw->defer_hide)) return;
cw->visible = 0;
if (cw->invalid) return;
if (cw->input_only)
{
if (cw->shape) e_container_shape_hide(cw->shape);
return;
}
DBG(" [0x%x] hid --", cw->win);
if (!cw->force)
{
cw->defer_hide = 1;
edje_object_signal_emit(cw->shobj, "e,state,visible,off", "e");
if (!cw->animating)
{
cw->c->animating++;
}
cw->animating = 1;
// if (cw->real_obj && cw->eobj)
// e_object_ref(cw->eobj);
_e_comp_win_render_queue(cw);
cw->pending_count++;
_e_comp_event_source_visibility(cw);
if (cw->shape) e_container_shape_hide(cw->shape);
return;
}
cw->defer_hide = 0;
cw->force = 0;
_e_comp_child_hide(cw);
edje_object_signal_emit(cw->shobj, "e,state,visible,off", "e");
if (cw->update_timeout)
{
ecore_timer_del(cw->update_timeout);
cw->update_timeout = NULL;
}
if (conf->keep_unmapped && cw->win)
{
if (conf->send_flush)
{
if (cw->bd) ecore_x_e_comp_flush_send(cw->bd->client.win);
else ecore_x_e_comp_flush_send(cw->win);
}
if (conf->send_dump)
{
if (cw->bd) ecore_x_e_comp_dump_send(cw->bd->client.win);
else ecore_x_e_comp_dump_send(cw->win);
}
return;
}
if (cw->ready_timeout)
{
ecore_timer_del(cw->ready_timeout);
cw->ready_timeout = NULL;
}
if (cw->native)
{
evas_object_image_native_surface_set(cw->obj, NULL);
cw->native = 0;
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_native_surface_set(o, NULL);
}
}
if (cw->pixmap)
{
ecore_x_pixmap_free(cw->pixmap);
cw->pixmap = 0;
cw->pw = 0;
cw->ph = 0;
ecore_x_e_comp_pixmap_set(cw->win, cw->pixmap);
// cw->show_ready = 0; // hmm maybe not needed?
}
if (cw->xim)
{
evas_object_image_size_set(cw->obj, 1, 1);
evas_object_image_data_set(cw->obj, NULL);
ecore_x_image_free(cw->xim);
cw->xim = NULL;
EINA_LIST_FOREACH(cw->obj_mirror, l, o)
{
evas_object_image_size_set(o, 1, 1);
evas_object_image_data_set(o, NULL);
}
}
if (cw->redirected)
{
// we redirect all subwindows anyway
// ecore_x_composite_unredirect_window(cw->win, ECORE_X_COMPOSITE_UPDATE_MANUAL);
2012-06-28 00:03:44 -07:00
cw->redirected = 0;
cw->pw = 0;
cw->ph = 0;
}
_e_comp_win_render_queue(cw);
if (!cw->win) return;
if (conf->send_flush)
{
if (cw->bd) ecore_x_e_comp_flush_send(cw->bd->client.win);
else ecore_x_e_comp_flush_send(cw->win);
}
if (conf->send_dump)
{
if (cw->bd) ecore_x_e_comp_dump_send(cw->bd->client.win);
else ecore_x_e_comp_dump_send(cw->win);
}
}
static void
_e_comp_win_raise_above(E_Comp_Win *cw, E_Comp_Win *cw2)
{
DBG(" [0x%x] abv [0x%x]", cw->win, cw2->win);
cw->c->wins_invalid = 1;
cw->c->wins = eina_inlist_remove(cw->c->wins, EINA_INLIST_GET(cw));
cw->c->wins = eina_inlist_append_relative(cw->c->wins, EINA_INLIST_GET(cw), EINA_INLIST_GET(cw2));
_e_comp_win_restack(cw);
_e_comp_win_render_queue(cw);
cw->pending_count++;
2013-04-19 05:49:34 -07:00
_e_comp_event_source_stack(cw);
}
static void
_e_comp_win_raise(E_Comp_Win *cw)
{
DBG(" [0x%x] rai", cw->win);
cw->c->wins_invalid = 1;
cw->c->wins = eina_inlist_remove(cw->c->wins, EINA_INLIST_GET(cw));
cw->c->wins = eina_inlist_append(cw->c->wins, EINA_INLIST_GET(cw));
_e_comp_win_restack(cw);
_e_comp_win_render_queue(cw);
cw->pending_count++;
2013-04-19 05:49:34 -07:00
_e_comp_event_source_stack(cw);
}
static void
_e_comp_win_lower_below(E_Comp_Win *cw, E_Comp_Win *cw2)
{
DBG(" [0x%x] below [0x%x]", cw->win, cw2->win);
cw->c->wins_invalid = 1;
cw->c->wins = eina_inlist_remove(cw->c->wins, EINA_INLIST_GET(cw));
cw->c->wins = eina_inlist_prepend_relative(cw->c->wins, EINA_INLIST_GET(cw), EINA_INLIST_GET(cw2));
_e_comp_win_restack(cw);
_e_comp_win_render_queue(cw);
cw->pending_count++;
2013-04-19 05:49:34 -07:00
_e_comp_event_source_stack(cw);
}
static void
_e_comp_win_lower(E_Comp_Win *cw)
{
DBG(" [0x%x] low", cw->win);
cw->c->wins_invalid = 1;
cw->c->wins = eina_inlist_remove(cw->c->wins, EINA_INLIST_GET(cw));
cw->c->wins = eina_inlist_prepend(cw->c->wins, EINA_INLIST_GET(cw));
_e_comp_win_restack(cw);
_e_comp_win_render_queue(cw);
cw->pending_count++;
2013-04-19 05:49:34 -07:00
_e_comp_event_source_stack(cw);
}
static void
_e_comp_win_configure(E_Comp_Win *cw, int x, int y, int w, int h, int border)
{
Eina_Bool moved = EINA_FALSE, resized = EINA_FALSE;
if (!cw->visible)
{
cw->hidden.x = x;
cw->hidden.y = y;
cw->border = border;
}
else
{
if (!((x == cw->x) && (y == cw->y)))
{
DBG(" [0x%x] mov %4i %4i", cw->win, x, y);
cw->x = x;
cw->y = y;
// evas_object_move(cw->effect_obj, cw->x, cw->y);
moved = EINA_TRUE;
}
cw->hidden.x = x;
cw->hidden.y = y;
}
cw->hidden.w = w;
cw->hidden.h = h;
if (cw->counter)
{
if (!((w == cw->w) && (h == cw->h)))
{
2012-06-28 00:03:44 -07:00
#if 1
cw->w = w;
cw->h = h;
resized = EINA_TRUE;
2013-05-07 02:32:38 -07:00
if ((!cw->bd) || ((!cw->bd->shading) && (!cw->bd->shaded)))
{
cw->needpix = 1;
// was cw->w / cw->h
// evas_object_resize(cw->effect_obj, cw->pw, cw->ph);
_e_comp_win_damage(cw, 0, 0, cw->w, cw->h, 0);
}
2012-06-28 00:03:44 -07:00
#else
if (cw->bd)
{
if ((cw->bd->shading) || (cw->bd->shaded))
{
resized = EINA_TRUE;
/* don't need pixmap fetch! */
_e_comp_win_damage(cw, 0, 0, cw->w, cw->h, 0);
}
else
2012-06-28 00:03:44 -07:00
cw->update = 0;
}
else
{
cw->update = 0;
//if (cw->ready_timeout) ecore_timer_del(cw->ready_timeout);
//cw->ready_timeout = ecore_timer_add(conf->first_draw_delay, _e_comp_cb_win_show_ready_timeout, cw);
}
2012-06-28 00:03:44 -07:00
#endif
}
if (cw->border != border)
{
cw->border = border;
cw->needpix = 1;
// was cw->w / cw->h
// evas_object_resize(cw->effect_obj, cw->pw, cw->ph);
resized = EINA_TRUE;
_e_comp_win_damage(cw, 0, 0, cw->w, cw->h, 0);
}
if ((cw->input_only) || (cw->invalid)) return;
}
else
{
if (!((w == cw->w) && (h == cw->h)))
{
DBG(" [0x%x] rsz %4ix%4i", cw->win, w, h);
cw->w = w;
cw->h = h;
resized = EINA_TRUE;
2013-05-07 02:32:38 -07:00
if ((!cw->bd) || ((!cw->bd->shading) && (!cw->bd->shaded)))
{
cw->needpix = 1;
// was cw->w / cw->h
// evas_object_resize(cw->effect_obj, cw->pw, cw->ph);
if (!cw->real_obj) _e_comp_win_damage(cw, 0, 0, cw->w, cw->h, 0);
}
}
if (cw->border != border)
{
cw->border = border;
cw->needpix = 1;
// evas_object_resize(cw->effect_obj, cw->pw, cw->ph);
resized = EINA_TRUE;
_e_comp_win_damage(cw, 0, 0, cw->w, cw->h, 0);
}
if ((cw->input_only) || (cw->invalid)) return;
_e_comp_win_render_queue(cw);
}
/* need to block move/resize of the edje for real objects so the external object doesn't
* accidentally get shown and block our show callback
*/
if ((cw->real_obj && cw->visible) || moved || resized) _e_comp_win_geometry_update(cw);
// add pending manager comp event count to match below config send
cw->pending_count++;
_e_comp_event_source_configure(cw);
}
static void
_e_comp_win_damage(E_Comp_Win *cw, int x, int y, int w, int h, Eina_Bool dmg)
{
if ((cw->input_only) || (cw->invalid)) return;
DBG(" [0x%x] dmg [%x] %4i %4i %4ix%4i", cw->win, cw->damage, x, y, w, h);
if ((dmg) && (cw->damage))
{
Ecore_X_Region parts;
parts = ecore_x_region_new(NULL, 0);
ecore_x_damage_subtract(cw->damage, 0, parts);
ecore_x_region_free(parts);
cw->dmg_updates++;
}
if (cw->nocomp) return;
if (cw->c->nocomp)
{
cw->nocomp_need_update = EINA_TRUE;
return;
}
e_comp_render_update_add(cw->up, x, y, w, h);
if (dmg)
{
if (cw->counter)
{
if (!cw->update_timeout)
cw->update_timeout = ecore_timer_add
(ecore_animator_frametime_get() * 2, _e_comp_win_damage_timeout, cw);
return;
}
}
if (!cw->update)
{
cw->update = 1;
cw->c->updates = eina_list_append(cw->c->updates, cw);
}
_e_comp_win_render_queue(cw);
}
static void
_e_comp_win_reshape(E_Comp_Win *cw)
{
if (cw->shape_changed) return;
cw->shape_changed = 1;
if (cw->c->nocomp)
{
cw->nocomp_need_update = EINA_TRUE;
return;
}
if (!cw->update)
{
cw->update = 1;
cw->c->updates = eina_list_append(cw->c->updates, cw);
}
e_comp_render_update_add(cw->up, 0, 0, cw->w, cw->h);
_e_comp_win_render_queue(cw);
}
//////////////////////////////////////////////////////////////////////////
static Eina_Bool
_e_comp_create(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_X_Event_Window_Create *ev = event;
E_Comp_Win *cw;
E_Comp *c;
c = _e_comp_find(ev->parent);
if (!c) return ECORE_CALLBACK_PASS_ON;
if (c->win == ev->win) return ECORE_CALLBACK_PASS_ON;
if (c->ee_win == ev->win) return ECORE_CALLBACK_PASS_ON;
if (c->man->root == ev->win) return ECORE_CALLBACK_PASS_ON;
if (_e_comp_ignore_find(ev->win)) return ECORE_CALLBACK_PASS_ON;
if (_e_comp_win_find(ev->win)) return ECORE_CALLBACK_PASS_ON;
if (!ev->override) return ECORE_CALLBACK_PASS_ON;
cw = _e_comp_win_add(c, ev->win, NULL);
if (!cw) return ECORE_CALLBACK_RENEW;
_e_comp_win_configure(cw, ev->x, ev->y, ev->w, ev->h, ev->border);
if (cw->free_shape)
{
Eina_List *l;
E_Container *con;
Eina_Rectangle *rects;
int num;
EINA_LIST_FOREACH(c->man->containers, l, con)
{
if (!E_INSIDE(ev->x, ev->y, con->x, con->y, con->w, con->h)) continue;
cw->shape = e_container_shape_add(con);
break;
}
if (!cw->shape) cw->shape = e_container_shape_add(eina_list_data_get(c->man->containers));
e_container_shape_resize(cw->shape, ev->w, ev->h);
rects = (Eina_Rectangle*)ecore_x_window_shape_rectangles_get(cw->win, &num);
e_container_shape_rects_set(cw->shape, rects, num);
if (cw->shape->shape_rects)
e_container_shape_input_rects_set(cw->shape, NULL, 0);
else
{
rects = (Eina_Rectangle*)ecore_x_window_shape_input_rectangles_get(cw->win, &num);
e_container_shape_input_rects_set(cw->shape, rects, num);
}
_e_comp_win_shape_init(cw, ev->w, ev->h);
}
if (cw->shape) cw->shape->comp_win = cw;
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_destroy(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_X_Event_Window_Destroy *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->win);
if (!cw)
{
eina_hash_del_by_key(ignores, e_util_winid_str_get(ev->win));
return ECORE_CALLBACK_PASS_ON;
}
if (cw->animating) cw->delete_me = 1;
else _e_comp_win_del(cw);
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_show(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_X_Event_Window_Show *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
cw->defer_hide = 0;
if (cw->visible) return ECORE_CALLBACK_PASS_ON;
_e_comp_win_show(cw);
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_hide(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_X_Event_Window_Hide *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
if (!cw->visible) return ECORE_CALLBACK_PASS_ON;
_e_comp_win_real_hide(cw);
if (cw->free_shape) e_container_shape_hide(cw->shape);
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_reparent(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_X_Event_Window_Reparent *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
DBG("== repar [0x%x] to [0x%x]", ev->win, ev->parent);
if (ev->parent != cw->c->man->root)
_e_comp_win_del(cw);
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_configure(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_X_Event_Window_Configure *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
if (ev->abovewin == 0)
{
if (EINA_INLIST_GET(cw)->prev) _e_comp_win_lower(cw);
}
else
{
E_Comp_Win *cw2 = _e_comp_win_find(ev->abovewin);
if (cw2)
{
E_Comp_Win *cw3 = (E_Comp_Win *)(EINA_INLIST_GET(cw)->prev);
if (cw3 != cw2) _e_comp_win_raise_above(cw, cw2);
}
}
if (!((cw->x == ev->x) && (cw->y == ev->y) &&
(cw->w == ev->w) && (cw->h == ev->h) &&
(cw->border == ev->border)))
{
_e_comp_win_configure(cw, ev->x, ev->y, ev->w, ev->h, ev->border);
if (cw->free_shape)
{
if (cw->visible)
e_container_shape_move(cw->shape, cw->x, cw->y);
else
e_container_shape_move(cw->shape, cw->hidden.x, cw->hidden.y);
e_container_shape_resize(cw->shape, cw->w, cw->h);
}
}
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_stack(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_X_Event_Window_Stack *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
if (ev->detail == ECORE_X_WINDOW_STACK_ABOVE) _e_comp_win_raise(cw);
else _e_comp_win_lower(cw);
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_message(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_X_Event_Client_Message *ev = event;
E_Comp_Win *cw = NULL;
int version, w = 0, h = 0;
Eina_Bool force = 0;
2012-06-28 00:03:44 -07:00
if ((ev->message_type != ECORE_X_ATOM_E_COMP_SYNC_DRAW_DONE) ||
(ev->format != 32)) return ECORE_CALLBACK_PASS_ON;
version = ev->data.l[1];
cw = _e_comp_border_client_find(ev->data.l[0]);
if (cw)
{
if (!cw->bd) return ECORE_CALLBACK_PASS_ON;
2011-10-16 08:35:53 -07:00
if (ev->data.l[0] != (int)cw->bd->client.win) return ECORE_CALLBACK_PASS_ON;
}
else
{
cw = _e_comp_win_find(ev->data.l[0]);
if (!cw) return ECORE_CALLBACK_PASS_ON;
2011-10-16 08:35:53 -07:00
if (ev->data.l[0] != (int)cw->win) return ECORE_CALLBACK_PASS_ON;
}
if (version == 1) // v 0 was first, v1 added size params
{
w = ev->data.l[2];
h = ev->data.l[3];
if (cw->bd)
{
int clw, clh;
2012-06-28 00:03:44 -07:00
if ((cw->bd->shading) || (cw->bd->shaded)) force = 1;
2012-06-28 00:03:44 -07:00
clw = cw->hidden.w -
cw->bd->client_inset.l -
cw->bd->client_inset.r;
2012-06-28 00:03:44 -07:00
clh = cw->hidden.h -
cw->bd->client_inset.t -
cw->bd->client_inset.b;
DBG(" [0x%x] sync draw done @%4ix%4i, bd %4ix%4i",
cw->win, w, h, cw->bd->client.w, cw->bd->client.h);
2012-06-28 00:03:44 -07:00
if ((w != clw) || (h != clh))
{
cw->misses++;
if (cw->misses > 1)
{
cw->misses = 0;
force = 1;
}
else return ECORE_CALLBACK_PASS_ON;
}
cw->misses = 0;
}
else
{
DBG(" [0x%x] sync draw done @%4ix%4i, cw %4ix%4i",
cw->win, w, h, cw->hidden.w, cw->hidden.h);
if ((w != cw->hidden.w) || (h != cw->hidden.h))
{
if (cw->misses > 1)
{
cw->misses = 0;
force = 1;
}
else return ECORE_CALLBACK_PASS_ON;
}
cw->misses = 0;
}
}
DBG(" [0x%x] sync draw done %4ix%4i", cw->win, cw->w, cw->h);
// if (cw->bd)
{
if (cw->counter)
{
DBG(" [0x%x] have counter", cw->win);
cw->show_ready = 1;
if (!cw->update)
{
DBG(" [0x%x] set update", cw->win);
if (cw->update_timeout)
{
DBG(" [0x%x] del timeout", cw->win);
ecore_timer_del(cw->update_timeout);
cw->update_timeout = NULL;
}
cw->update = 1;
cw->c->updates = eina_list_append(cw->c->updates, cw);
}
if ((cw->w != cw->hidden.w) ||
(cw->h != cw->hidden.h) ||
(force))
{
DBG(" [0x%x] rsz done msg %4ix%4i", cw->win, cw->hidden.w, cw->hidden.h);
cw->w = cw->hidden.w;
cw->h = cw->hidden.h;
cw->needpix = 1;
// was cw->w / cw->h
// evas_object_resize(cw->effect_obj, cw->pw, cw->ph);
_e_comp_win_geometry_update(cw);
_e_comp_win_damage(cw, 0, 0, cw->w, cw->h, 0);
}
cw->drawme = 1;
_e_comp_win_render_queue(cw);
}
}
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_shape(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_X_Event_Window_Shape *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
if (ev->type != ECORE_X_SHAPE_BOUNDING) return ECORE_CALLBACK_PASS_ON;
_e_comp_win_reshape(cw);
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_damage(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_X_Event_Damage *ev = event;
E_Comp_Win *cw = _e_comp_win_damage_find(ev->damage);
if (!cw) return ECORE_CALLBACK_PASS_ON;
_e_comp_win_damage(cw, ev->area.x, ev->area.y, ev->area.width, ev->area.height, 1);
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_damage_win(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_X_Event_Window_Damage *ev = event;
Eina_List *l;
E_Comp *c;
// fixme: use hash if compositors list > 4
EINA_LIST_FOREACH(compositors, l, c)
{
if (ev->win == c->ee_win)
{
// expose on comp win - init win or some other bypass win did it
DBG("JOB4...");
_e_comp_render_queue(c);
break;
}
}
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_override_expire(void *data)
{
E_Comp *c = data;
c->nocomp_override_timer = NULL;
c->nocomp_override--;
if (c->nocomp_override <= 0)
{
c->nocomp_override = 0;
if (c->nocomp_want) _e_comp_cb_nocomp_begin(c);
}
return EINA_FALSE;
}
static void
_e_comp_override_timed_pop(E_Comp *c)
{
if (c->nocomp_override <= 0) return;
if (c->nocomp_override_timer)
ecore_timer_del(c->nocomp_override_timer);
c->nocomp_override_timer =
ecore_timer_add(5.0, _e_comp_override_expire, c);
}
static void
_e_comp_fade_handle(E_Comp_Zone *cz, int out, double tim)
{
if (out == 1)
{
if ((e_backlight_exists()) && (!conf->nofade))
{
e_backlight_update();
cz->bloff = EINA_TRUE;
cz->bl = e_backlight_level_get(cz->zone);
e_backlight_level_set(cz->zone, 0.0, tim);
}
}
else
{
if ((e_backlight_exists()) && (!conf->nofade))
{
cz->bloff = EINA_FALSE;
e_backlight_update();
if (e_backlight_mode_get(cz->zone) != E_BACKLIGHT_MODE_NORMAL)
e_backlight_mode_set(cz->zone, E_BACKLIGHT_MODE_NORMAL);
else
e_backlight_level_set(cz->zone, e_config->backlight.normal, tim);
}
}
}
static Eina_Bool
_e_comp_screensaver_on(void *data EINA_UNUSED, int type EINA_UNUSED, void *event EINA_UNUSED)
{
Eina_List *l, *ll;
E_Comp_Zone *cz;
E_Comp *c;
// fixme: use hash if compositors list > 4
EINA_LIST_FOREACH(compositors, l, c)
{
if (!c->saver)
{
c->saver = EINA_TRUE;
EINA_LIST_FOREACH(c->zones, ll, cz)
{
e_comp_override_add(c);
_e_comp_fade_handle(cz, 1, 3.0);
edje_object_signal_emit(cz->base, "e,state,screensaver,on", "e");
edje_object_signal_emit(cz->over, "e,state,screensaver,on", "e");
}
}
}
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_screensaver_off(void *data EINA_UNUSED, int type EINA_UNUSED, void *event EINA_UNUSED)
{
Eina_List *l, *ll;
E_Comp_Zone *cz;
E_Comp *c;
// fixme: use hash if compositors list > 4
EINA_LIST_FOREACH(compositors, l, c)
{
if (c->saver)
{
c->saver = EINA_FALSE;
EINA_LIST_FOREACH(c->zones, ll, cz)
{
edje_object_signal_emit(cz->base, "e,state,screensaver,off", "e");
edje_object_signal_emit(cz->over, "e,state,screensaver,off", "e");
_e_comp_fade_handle(cz, 0, 0.5);
_e_comp_override_timed_pop(c);
}
}
}
return ECORE_CALLBACK_PASS_ON;
}
static void
_e_comp_screens_eval(E_Comp *c)
{
Eina_List *l, *ll;
E_Container *con;
E_Zone *zone;
E_Comp_Zone *cz;
EINA_LIST_FREE(c->zones, cz)
{
evas_object_del(cz->base);
evas_object_del(cz->over);
if (cz->bloff)
{
if (!conf->nofade)
{
if (e_backlight_mode_get(cz->zone) != E_BACKLIGHT_MODE_NORMAL)
e_backlight_mode_set(cz->zone, E_BACKLIGHT_MODE_NORMAL);
e_backlight_level_set(cz->zone, e_config->backlight.normal, 0.0);
}
}
if (cz->zone) cz->zone->comp_zone = NULL;
free(cz);
}
EINA_LIST_FOREACH(c->man->containers, l, con)
{
EINA_LIST_FOREACH(con->zones, ll, zone)
{
cz = E_NEW(E_Comp_Zone, 1);
cz->zone = zone;
cz->comp = c;
zone->comp_zone = cz;
cz->container_num = zone->container->num;
cz->zone_num = zone->num;
cz->x = zone->x;
cz->y = zone->y;
cz->w = zone->w;
cz->h = zone->h;
e_comp_zone_update(cz);
c->zones = eina_list_append(c->zones, cz);
}
}
e_layout_freeze(c->layout);
evas_object_move(c->layout, 0, 0);
evas_object_resize(c->layout, c->man->w, c->man->h);
e_layout_virtual_size_set(c->layout, c->man->w, c->man->h);
e_layout_thaw(c->layout);
}
static void
_e_comp_screen_change(void *data)
{
E_Comp *c = data;
c->screen_job = NULL;
if (!c->nocomp) ecore_evas_resize(c->ee, c->man->w, c->man->h);
_e_comp_screens_eval(c);
}
static Eina_Bool
_e_comp_randr(void *data EINA_UNUSED, int type EINA_UNUSED, EINA_UNUSED void *event)
{
Eina_List *l;
E_Comp *c;
EINA_LIST_FOREACH(compositors, l, c)
{
if (c->screen_job) ecore_job_del(c->screen_job);
c->screen_job = ecore_job_add(_e_comp_screen_change, c);
}
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_zonech(void *data EINA_UNUSED, int type EINA_UNUSED, EINA_UNUSED void *event)
{
Eina_List *l;
E_Comp *c;
EINA_LIST_FOREACH(compositors, l, c)
{
if (c->screen_job) ecore_job_del(c->screen_job);
c->screen_job = ecore_job_add(_e_comp_screen_change, c);
}
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_add(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Add *ev = event;
E_Comp_Win *cw;
E_Container *con;
int x;
cw = ev->border->cw;
if (!cw)
{
cw = _e_comp_win_find(ev->border->win);
if (cw)
{
_e_comp_win_bd_setup(cw, ev->border);
evas_object_data_set(cw->shobj, "border", cw->bd);
evas_object_data_set(cw->effect_obj, "border", cw->bd);
#ifdef BORDER_ZOOMAPS
evas_object_name_set(cw->zoomobj, "cw->zoomobj::BORDER");
#endif
evas_object_name_set(cw->shobj, "cw->shobj::BORDER");
evas_object_name_set(cw->effect_obj, "cw->effect_obj::BORDER");
e_comp_win_reshadow(cw);
}
else
cw = _e_comp_win_add(e_comp_get(ev->border), ev->border->win, ev->border);
}
_e_comp_win_configure(cw, ev->border->x, ev->border->y,
ev->border->w, ev->border->h,
ev->border->client.initial_attributes.border);
if (cw->shape) cw->shape->comp_win = cw;
con = cw->bd->zone->container;
/* we previously ignored potential stacking requests before the border setup,
* so we have to manually stack it here */
for (x = 0; x < E_CONTAINER_LAYER_COUNT; x++)
{
Eina_List *l;
E_Border *bd;
E_Comp_Win *cw2;
if (!con->layers[x].clients) continue;
l = eina_list_data_find_list(con->layers[x].clients, cw->bd);
if (!l) continue;
if (l->prev)
{
bd = eina_list_data_get(l->prev);
cw2 = _e_comp_win_find(bd->win);
if (cw2)
{
_e_comp_win_raise_above(cw, cw2);
break;
}
}
if (l->next)
{
bd = eina_list_data_get(l->next);
cw2 = _e_comp_win_find(bd->win);
if (cw2)
{
_e_comp_win_lower_below(cw, cw2);
break;
}
}
cw2 = _e_comp_win_find(con->layers[x].win);
if (cw2) _e_comp_win_raise_above(cw, cw2);
break;
}
//if (cw->bd->visible)
//{
//_e_comp_win_show(cw);
//}
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_del(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Remove *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (cw) e_comp_win_del(cw);
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_show(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Show *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (cw) _e_comp_win_show(cw);
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_hide(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Hide *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (cw)
{
cw->force |= (ev->manage == 2);
_e_comp_win_hide(cw);
}
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_move(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Move *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
_e_comp_win_configure(cw, ev->border->x, ev->border->y, cw->w, cw->h, cw->border);
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_resize(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Resize *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
_e_comp_win_configure(cw, cw->x, cw->y, ev->border->w, ev->border->h, cw->border);
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_iconify(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Iconify *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
// fimxe: special iconfiy anim
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_uniconify(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Uniconify *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
// fimxe: special uniconfiy anim
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_urgent_change(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Urgent_Change *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
if (cw->bd->client.icccm.urgent)
edje_object_signal_emit(cw->shobj, "e,state,urgent,on", "e");
else
edje_object_signal_emit(cw->shobj, "e,state,urgent,off", "e");
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_focus_in(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Focus_In *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
edje_object_signal_emit(cw->shobj, "e,state,focus,on", "e");
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_focus_out(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Focus_Out *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
edje_object_signal_emit(cw->shobj, "e,state,focus,off", "e");
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_property(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Property *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
// fimxe: other properties?
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_fullscreen(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Property *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
e_comp_win_reshadow(cw);
/* bd->bg_object deletion pending */
#ifdef BORDER_ZOOMAPS
e_zoomap_child_set(cw->zoomobj, cw->obj);
#else
edje_object_part_swallow(cw->shobj, "e.swallow.content", cw->obj);
#endif
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_bd_unfullscreen(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
E_Event_Border_Property *ev = event;
E_Comp_Win *cw = _e_comp_win_find(ev->border->win);
if (!cw) return ECORE_CALLBACK_PASS_ON;
e_comp_win_reshadow(cw);
return ECORE_CALLBACK_PASS_ON;
}
static void
_e_comp_injected_win_focus_out_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED)
{
E_Comp_Win *cw = data;
edje_object_signal_emit(cw->shobj, "e,state,focus,off", "e");
}
static void
_e_comp_injected_win_focus_in_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED)
{
E_Comp_Win *cw = data;
edje_object_signal_emit(cw->shobj, "e,state,focus,on", "e");
}
static void
_e_comp_injected_win_hide_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED)
{
E_Comp_Win *cw = data;
_e_comp_win_real_hide(cw);
}
static void
_e_comp_injected_win_show_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj EINA_UNUSED, void *event_info EINA_UNUSED)
{
E_Comp_Win *cw = data;
cw->defer_hide = 0;
cw->show_ready = 1;
_e_comp_win_geometry_update(cw);
_e_comp_win_show(cw);
}
static void
_e_comp_injected_win_del_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED)
{
E_Comp_Win *cw = data;
if (evas_object_layer_get(obj) > E_COMP_CANVAS_LAYER_LAYOUT)
e_comp_override_del(cw->c);
cw->obj = NULL;
if (cw->animating) cw->delete_me = 1;
else _e_comp_win_del(cw);
}
/*
static void
_e_comp_injected_win_moveresize_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED)
{
E_Comp_Win *cw = data;
int x, y, w, h;
evas_object_geometry_get(obj, &x, &y, &w, &h);
_e_comp_win_configure(cw, x, y, w, h, 0);
}
*/
#ifdef SHAPE_DEBUG
static void
_e_comp_shape_debug_rect(E_Comp *c, Ecore_X_Rectangle *rect, E_Color *color)
{
Evas_Object *o;
#define COLOR_INCREMENT 30
o = evas_object_rectangle_add(c->evas);
if (color->r < 256 - COLOR_INCREMENT)
evas_object_color_set(o, (color->r += COLOR_INCREMENT), 0, 0, 255);
else if (color->g < 256 - COLOR_INCREMENT)
evas_object_color_set(o, 0, (color->g += COLOR_INCREMENT), 0, 255);
else
evas_object_color_set(o, 0, 0, (color->b += COLOR_INCREMENT), 255);
evas_object_repeat_events_set(o, 1);
evas_object_raise(o);
e_layout_pack(c->layout, o);
e_layout_child_move(o, rect->x, rect->y);
e_layout_child_resize(o, rect->width, rect->height);
e_layout_child_raise(o);
c->debug_rects = eina_list_append(c->debug_rects, o);
evas_object_show(o);
}
#endif
static Eina_Bool
_e_comp_shapes_update_object_checker_function_thingy(E_Comp *c, Evas_Object *o)
{
Eina_List *l, *ll;
E_Container *con;
E_Zone *zone;
E_Comp_Zone *cz;
EINA_LIST_FOREACH(c->zones, l, cz)
if ((o == cz->over) || (o == cz->base)) return EINA_TRUE;
EINA_LIST_FOREACH(c->man->containers, l, con)
{
if (o == con->bg_blank_object) return EINA_TRUE;
EINA_LIST_FOREACH(con->zones, ll, zone)
if ((o == zone->bg_object) || (o == zone->bg_event_object) ||
(o == zone->bg_clip_object) || (o == zone->prev_bg_object) ||
(o == zone->transition_object))
return EINA_TRUE;
}
return EINA_FALSE;
}
static void
#ifdef SHAPE_DEBUG
_e_comp_shapes_update_comp_win_shape_comp_helper(E_Comp_Win *cw, Eina_Tiler *tb, Eina_List **rl)
#else
_e_comp_shapes_update_comp_win_shape_comp_helper(E_Comp_Win *cw, Eina_Tiler *tb)
#endif
{
int x, y, w, h;
/* ignore deleted shapes */
if (!cw->shape)
{
SHAPE_INF("IGNORING DELETED: %u", cw->win);
return;
}
if (cw->invalid || cw->real_hid || (!cw->visible) || (!cw->shape->visible) || evas_object_pass_events_get(cw->shobj))
{
SHAPE_DBG("SKIPPING SHAPE");
return;
}
#ifdef SHAPE_DEBUG
if (cw->bd)
INF("COMP BD: %u", cw->win);
else if (cw->pop)
INF("COMP POP: %u", cw->win);
else if (cw->menu)
INF("COMP MENU: %u", cw->win);
else if (cw->real_obj)
INF("COMP OBJ: %s", evas_object_name_get(cw->obj));
else
INF("COMP WIN: %u", cw->win);
#endif
if (cw->shape->shape_input_rects || cw->shape->shape_rects)
{
int num, tot;
Eina_Rectangle *rect, *rects;
/* add the frame */
if (cw->bd)
{
if (cw->bd->client_inset.calc)
{
if (cw->bd->client_inset.t)
{
eina_tiler_rect_add(tb, &(Eina_Rectangle){cw->bd->x, cw->bd->y, cw->bd->w, cw->bd->client_inset.t});
SHAPE_INF("ADD: %d,%d@%dx%d", cw->bd->x, cw->bd->y, cw->bd->w, cw->bd->client_inset.t);
}
if (cw->bd->client_inset.l)
{
eina_tiler_rect_add(tb, &(Eina_Rectangle){cw->bd->x, cw->bd->y, cw->bd->client_inset.l, cw->bd->h});
SHAPE_INF("ADD: %d,%d@%dx%d", cw->bd->x, cw->bd->y, cw->bd->client_inset.l, cw->bd->h);
}
if (cw->bd->client_inset.r)
{
eina_tiler_rect_add(tb, &(Eina_Rectangle){cw->bd->x + cw->bd->client_inset.l + cw->bd->client.w, cw->bd->y, cw->bd->client_inset.r, cw->bd->h});
SHAPE_INF("ADD: %d,%d@%dx%d", cw->bd->x + cw->bd->client_inset.l + cw->bd->client.w, cw->bd->y, cw->bd->client_inset.r, cw->bd->h);
}
if (cw->bd->client_inset.b)
{
eina_tiler_rect_add(tb, &(Eina_Rectangle){cw->bd->x, cw->bd->y + cw->bd->client_inset.t + cw->bd->client.h, cw->bd->w, cw->bd->client_inset.b});
SHAPE_INF("ADD: %d,%d@%dx%d", cw->bd->x, cw->bd->y + cw->bd->client_inset.t + cw->bd->client.h, cw->bd->w, cw->bd->client_inset.b);
}
}
}
rects = cw->shape->shape_rects ?: cw->shape->shape_input_rects;
tot = cw->shape->shape_rects_num ?: cw->shape->shape_input_rects_num;
for (num = 0, rect = rects; num < tot; num++, rect++)
{
x = rect->x, y = rect->y, w = rect->w, h = rect->h;
if (cw->bd)
{
x += cw->bd->x, y += cw->bd->y;
if (cw->bd->client_inset.calc)
x += cw->bd->client_inset.l, y += cw->bd->client_inset.t;
}
else
x += cw->x, y += cw->y;
E_RECTS_CLIP_TO_RECT(x, y, w, h, cw->c->man->x, cw->c->man->y, cw->c->man->w, cw->c->man->h);
if ((w < 1) || (h < 1)) continue;
//#ifdef SHAPE_DEBUG not sure we can shape check these?
//r = E_NEW(Eina_Rectangle, 1);
//EINA_RECTANGLE_SET(r, x, y, w, h);
//rl = eina_list_append(rl, r);
//#endif
eina_tiler_rect_del(tb, &(Eina_Rectangle){x, y, w, h});
SHAPE_INF("DEL: %d,%d@%dx%d", x, y, w, h);
}
return;
}
/* borders and popups sometimes call shape changes before the changes have
* propagated to the comp_win :/
*/
if (cw->bd)
x = cw->bd->x + cw->bd->client_inset.l, y = cw->bd->y + cw->bd->client_inset.t, w = cw->bd->client.w, h = cw->bd->client.h;
else if (cw->pop)
x = cw->pop->x + cw->pop->zone->x, y = cw->pop->y + cw->pop->zone->y, w = cw->pop->w, h = cw->pop->h;
//else if (cw->menu)
//x = cw->menu->x + cw->menu->zone->x, y = cw->menu->y + cw->menu->zone->y, w = cw->menu->w, h = cw->menu->h;
else
x = cw->x, y = cw->y, w = cw->w, h = cw->h;
#ifdef SHAPE_DEBUG
if (!cw->real_obj)
{
Eina_Rectangle *r;
r = E_NEW(Eina_Rectangle, 1);
EINA_RECTANGLE_SET(r, x, y, w, h);
*rl = eina_list_append(*rl, r);
}
#endif
if (cw->real_obj)
{
eina_tiler_rect_add(tb, &(Eina_Rectangle){x, y, w, h});
SHAPE_INF("ADD: %d,%d@%dx%d", x, y, w, h);
}
else
{
if (cw->bd)
{
/* add the frame */
eina_tiler_rect_add(tb, &(Eina_Rectangle){cw->bd->x, cw->bd->y, cw->bd->w, cw->bd->h});
SHAPE_INF("ADD: %d,%d@%dx%d", cw->bd->x, cw->bd->y, cw->bd->w, cw->bd->h);
if (!cw->bd->shaded)
{
/* delete the client if not shaded */
eina_tiler_rect_del(tb, &(Eina_Rectangle){x, y, w, h});
SHAPE_INF("DEL: %d,%d@%dx%d", x, y, w, h);
}
}
else
{
eina_tiler_rect_del(tb, &(Eina_Rectangle){x, y, w, h});
SHAPE_INF("DEL: %d,%d@%dx%d", x, y, w, h);
}
}
}
static void
_e_comp_shapes_update_job(E_Comp *c)
{
Eina_Tiler *tb;
Evas_Object *o;
Eina_Rectangle *tr;
Eina_Iterator *ti;
Ecore_X_Rectangle *exr;
unsigned int i, tile_count;
int x, y, w, h;
Eina_Bool layout = EINA_FALSE;
#ifdef SHAPE_DEBUG
Eina_Rectangle *r;
Eina_List *rl = NULL;
E_Color color = {0};
#endif
E_FREE_LIST(c->debug_rects, evas_object_del);
tb = eina_tiler_new(c->man->w, c->man->h);
eina_tiler_tile_size_set(tb, 1, 1);
eina_tiler_rect_add(tb, &(Eina_Rectangle){c->man->x, c->man->y, c->man->w, c->man->h});
o = evas_object_bottom_get(c->evas);
for (; o; o = evas_object_above_get(o))
{
if (o == c->layout)
{
Evas_Object *ch;
Eina_List *l;
layout = EINA_TRUE; //ignore all objects under layout
l = e_layout_children_get(o);
EINA_LIST_FREE(l, ch)
{
E_Comp_Win *cw;
cw = evas_object_data_get(ch, "comp_win");
if (cw)
_e_comp_shapes_update_comp_win_shape_comp_helper(cw, tb
#ifdef SHAPE_DEBUG
,&rl
#endif
);
2013-02-11 01:39:27 -08:00
else if (evas_object_visible_get(ch) && (!evas_object_pass_events_get(ch)))
{
SHAPE_INF("COMP OBJ: %p", ch);
e_layout_child_geometry_get(ch, &x, &y, &w, &h);
eina_tiler_rect_add(tb, &(Eina_Rectangle){x, y, w, h});
SHAPE_INF("ADD: %d,%d@%dx%d", x, y, w, h);
}
}
}
else if (layout && evas_object_visible_get(o) && (!evas_object_pass_events_get(o)))
{
E_Comp_Win *cw;
if (_e_comp_shapes_update_object_checker_function_thingy(c, o)) continue;
cw = evas_object_data_get(o, "comp_win");
if (cw)
{
_e_comp_shapes_update_comp_win_shape_comp_helper(cw, tb
#ifdef SHAPE_DEBUG
,&rl
#endif
);
continue;
}
SHAPE_INF("OBJ: %p:%s", o, evas_object_name_get(o));
evas_object_geometry_get(o, &x, &y, &w, &h);
eina_tiler_rect_add(tb, &(Eina_Rectangle){x, y, w, h});
SHAPE_INF("ADD: %d,%d@%dx%d", x, y, w, h);
}
}
ti = eina_tiler_iterator_new(tb);
tile_count = 128;
exr = malloc(sizeof(Ecore_X_Rectangle) * tile_count);
i = 0;
EINA_ITERATOR_FOREACH(ti, tr)
{
exr[i++] = *(Ecore_X_Rectangle*)((char*)tr);
if (i == tile_count - 1)
exr = realloc(exr, sizeof(Ecore_X_Rectangle) * (tile_count *= 2));
#ifdef SHAPE_DEBUG
Eina_List *l;
_e_comp_shape_debug_rect(c, &exr[i - 1], &color);
INF("%d,%d @ %dx%d", exr[i - 1].x, exr[i - 1].y, exr[i - 1].width, exr[i - 1].height);
EINA_LIST_FOREACH(rl, l, r)
{
if (E_INTERSECTS(r->x, r->y, r->w, r->h, tr->x, tr->y, tr->w, tr->h))
ERR("POSSIBLE RECT FAIL!!!!");
}
#endif
}
ecore_x_window_shape_input_rectangles_set(c->win, exr, i);
#ifdef SHAPE_DEBUG
E_FREE_LIST(rl, free);
printf("\n");
#endif
free(exr);
eina_iterator_free(ti);
eina_tiler_free(tb);
c->shape_job = NULL;
}
static void
_e_comp_shapes_update(void *data, E_Container_Shape *es, E_Container_Shape_Change change)
{
E_Comp *c = data;
#ifdef SHAPE_DEBUG
const char const *change_text[] =
{
"ADD",
"DEL",
"SHOW",
"HIDE",
"MOVE",
"RESIZE",
"RECTS"
};
#endif
if (change == E_CONTAINER_SHAPE_ADD) return;
if (es)
{
if (!es->comp_win) return;
switch (change)
{
/* these need to always get through since
* it's guaranteed input shape will change when they occur
*/
case E_CONTAINER_SHAPE_SHOW:
case E_CONTAINER_SHAPE_HIDE:
case E_CONTAINER_SHAPE_DEL:
break;
case E_CONTAINER_SHAPE_RECTS:
case E_CONTAINER_SHAPE_INPUT_RECTS:
es->comp_win->shape_changed = 1;
_e_comp_win_render_queue(es->comp_win);
default:
/* any other changes only matter if the
* object is visible
*/
if (!es->visible) return;
}
SHAPE_INF("RESHAPE %u: %s", es->comp_win->win, change_text[change]);
}
if (!c->shape_job) c->shape_job = ecore_job_add((Ecore_Cb)_e_comp_shapes_update_job, c);
}
//////////////////////////////////////////////////////////////////////////
static void
_e_comp_fps_toggle(void)
{
Eina_List *l;
E_Comp *c;
if (conf->fps_show)
{
conf->fps_show = 0;
e_config_save_queue();
}
else
{
conf->fps_show = 1;
e_config_save_queue();
}
EINA_LIST_FOREACH(compositors, l, c)
_e_comp_cb_update(c);
}
static Eina_Bool
_e_comp_key_down(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_Event_Key *ev = event;
if ((!strcasecmp(ev->keyname, "f")) &&
(ev->modifiers & ECORE_EVENT_MODIFIER_SHIFT) &&
(ev->modifiers & ECORE_EVENT_MODIFIER_CTRL) &&
(ev->modifiers & ECORE_EVENT_MODIFIER_ALT))
{
_e_comp_fps_toggle();
}
return ECORE_CALLBACK_PASS_ON;
}
static Eina_Bool
_e_comp_signal_user(void *data EINA_UNUSED, int type EINA_UNUSED, void *event)
{
Ecore_Event_Signal_User *ev = event;
2012-06-28 00:03:44 -07:00
if (ev->number == 1)
{
// e17 uses this to pop up config panel
}
else if (ev->number == 2)
{
_e_comp_fps_toggle();
}
return ECORE_CALLBACK_PASS_ON;
}
//////////////////////////////////////////////////////////////////////////
static void
_e_comp_populate(E_Comp *c)
{
Ecore_X_Window *wins;
Eina_List *l;
E_Container *con;
int i, num;
c->layout = e_layout_add(c->evas);
evas_object_name_set(c->layout, "c->layout");
evas_object_layer_set(c->layout, E_COMP_CANVAS_LAYER_LAYOUT);
evas_object_show(c->layout);
e_drop_xdnd_register_set(c->win, 1);
EINA_LIST_FOREACH(c->man->containers, l, con)
{
e_container_shape_change_callback_add(con, _e_comp_shapes_update, c);
ecore_x_window_configure(c->ee_win,
ECORE_X_WINDOW_CONFIGURE_MASK_SIBLING | ECORE_X_WINDOW_CONFIGURE_MASK_STACK_MODE,
0, 0, 0, 0, 0, con->layers[0].win, ECORE_X_WINDOW_STACK_BELOW);
}
wins = ecore_x_window_children_get(c->man->root, &num);
if (!wins) return;
for (i = 0; i < num; i++)
{
E_Comp_Win *cw;
int x, y, w, h, border;
if (e_border_find_by_client_window(wins[i]) ||
e_border_find_by_window(wins[i])) continue;
cw = _e_comp_win_add(c, wins[i], NULL);
if (!cw) continue;
ecore_x_window_geometry_get(cw->win, &x, &y, &w, &h);
border = ecore_x_window_border_width_get(cw->win);
if (wins[i] == c->win) continue;
_e_comp_win_configure(cw, x, y, w, h, border);
if (cw->free_shape)
{
EINA_LIST_FOREACH(c->man->containers, l, con)
{
if (!E_INSIDE(x, y, con->x, con->y, con->w, con->h)) continue;
cw->shape = e_container_shape_add(con);
break;
}
if (!cw->shape) cw->shape = e_container_shape_add(eina_list_data_get(c->man->containers));
}
if (cw->shape)
{
cw->shape->comp_win = cw;
e_container_shape_move(cw->shape, x, y);
e_container_shape_resize(cw->shape, w, h);
}
if (cw->free_shape)
{
Eina_Rectangle *rects;
int rnum;
rects = (Eina_Rectangle*)ecore_x_window_shape_rectangles_get(cw->win, &rnum);
e_container_shape_rects_set(cw->shape, rects, rnum);
rects = (Eina_Rectangle*)ecore_x_window_shape_input_rectangles_get(cw->win, &rnum);
e_container_shape_input_rects_set(cw->shape, rects, rnum);
_e_comp_win_shape_init(cw, w, h);
}
if ((!cw->bd) && (ecore_x_window_visible_get(wins[i])))
_e_comp_win_show(cw);
}
free(wins);
if (!c->shape_job) c->shape_job = ecore_job_add((Ecore_Cb)_e_comp_shapes_update_job, c);
}
static void
_e_comp_add_fail_job(void *d EINA_UNUSED)
{
e_util_dialog_internal
(_("Compositor Warning"), _("Your display driver does not support OpenGL, or<br>"
"no OpenGL engines were compiled or installed for<br>"
"Evas or Ecore-Evas. Falling back to software engine."));
}
static E_Comp *
_e_comp_add(E_Manager *man)
{
E_Comp *c;
Ecore_X_Window_Attributes att;
Eina_Bool res;
c = calloc(1, sizeof(E_Comp));
if (!c) return NULL;
res = ecore_x_screen_is_composited(man->num);
if (res)
{
e_util_dialog_internal
(_("Compositor Error"), _("Another compositor is already running<br>"
"on your display server."));
free(c);
return NULL;
}
c->cm_selection = ecore_x_window_input_new(man->root, 0, 0, 1, 1);
if (!c->cm_selection)
{
free(c);
return NULL;
}
ecore_x_screen_is_composited_set(man->num, c->cm_selection);
2012-06-28 00:03:44 -07:00
ecore_x_e_comp_sync_supported_set(man->root, conf->efl_sync);
c->man = man;
man->comp = c;
c->win = ecore_x_composite_render_window_enable(man->root);
if (!c->win)
{
e_util_dialog_internal
(_("Compositor Error"), _("Your display server does not support the<br>"
"compositor overlay window. This is needed<br>"
"for it to function."));
free(c);
return NULL;
}
memset((&att), 0, sizeof(Ecore_X_Window_Attributes));
2011-03-19 23:12:17 -07:00
ecore_x_window_attributes_get(c->win, &att);
if ((att.depth != 24) && (att.depth != 32))
{
/*
e_util_dialog_internal
(_("Compositor Error"), _("Your screen is not in 24/32bit display mode.<br>"
"This is required to be your default depth<br>"
"setting for the compositor to work properly."));
ecore_x_composite_render_window_disable(c->win);
free(c);
return NULL;
*/
}
if (c->man->num == 0) e_alert_composite_win(c->man->root, c->win);
if (gl_avail && (conf->engine == E_COMP_ENGINE_GL))
{
int opt[20];
int opt_i = 0;
if (conf->indirect)
{
opt[opt_i] = ECORE_EVAS_GL_X11_OPT_INDIRECT;
opt_i++;
opt[opt_i] = 1;
opt_i++;
}
if (conf->vsync)
{
opt[opt_i] = ECORE_EVAS_GL_X11_OPT_VSYNC;
opt_i++;
opt[opt_i] = 1;
opt_i++;
}
#ifdef ECORE_EVAS_GL_X11_OPT_SWAP_MODE
if (conf->swap_mode)
{
opt[opt_i] = ECORE_EVAS_GL_X11_OPT_SWAP_MODE;
opt_i++;
opt[opt_i] = conf->swap_mode;
opt_i++;
}
#endif
if (opt_i > 0)
{
opt[opt_i] = ECORE_EVAS_GL_X11_OPT_NONE;
c->ee = ecore_evas_gl_x11_options_new
(NULL, c->win, 0, 0, man->w, man->h, opt);
}
if (!c->ee)
c->ee = ecore_evas_gl_x11_new(NULL, c->win, 0, 0, man->w, man->h);
if (c->ee)
{
c->gl = 1;
ecore_evas_gl_x11_pre_post_swap_callback_set
(c->ee, c, _e_comp_pre_swap, NULL);
}
}
if (!c->ee)
{
2011-03-19 23:12:17 -07:00
c->ee = ecore_evas_software_x11_new(NULL, c->win, 0, 0, man->w, man->h);
if (conf->engine == E_COMP_ENGINE_GL)
ecore_job_add(_e_comp_add_fail_job, NULL);
2011-03-19 23:12:17 -07:00
}
ecore_evas_comp_sync_set(c->ee, 0);
ecore_evas_name_class_set(c->ee, "E", "Comp_EE");
// ecore_evas_manual_render_set(c->ee, conf->lock_fps);
c->evas = ecore_evas_get(c->ee);
ecore_evas_data_set(c->ee, "comp", c);
ecore_evas_show(c->ee);
c->ee_win = ecore_evas_window_get(c->ee);
c->pointer = e_pointer_window_new(c->ee_win, 0);
ecore_x_composite_redirect_subwindows
(c->man->root, ECORE_X_COMPOSITE_UPDATE_MANUAL);
ecore_x_window_key_grab(c->man->root, "Home", ECORE_EVENT_MODIFIER_SHIFT |
ECORE_EVENT_MODIFIER_CTRL |
ECORE_EVENT_MODIFIER_ALT, 0);
ecore_x_window_key_grab(c->man->root, "F", ECORE_EVENT_MODIFIER_SHIFT |
ECORE_EVENT_MODIFIER_CTRL |
ECORE_EVENT_MODIFIER_ALT, 0);
return c;
}
EAPI void
e_comp_populate(E_Comp *c)
{
_e_comp_populate(c);
}
static void
_e_comp_del(E_Comp *c)
{
E_Comp_Win *cw;
E_Comp_Zone *cz;
Eina_List *l;
E_Container *con;
c->man->comp = NULL;
evas_event_freeze(c->evas);
edje_freeze();
EINA_LIST_FOREACH(c->man->containers, l, con)
e_container_shape_change_callback_del(con, _e_comp_shapes_update, c);
E_FREE_FUNC(c->fps_fg, evas_object_del);
E_FREE_FUNC(c->fps_bg, evas_object_del);
E_FREE_FUNC(c->shape_job, ecore_job_del);
E_FREE_FUNC(c->pointer, e_object_del);
ecore_x_window_key_ungrab(c->man->root, "F", ECORE_EVENT_MODIFIER_SHIFT |
ECORE_EVENT_MODIFIER_CTRL |
ECORE_EVENT_MODIFIER_ALT, 0);
ecore_x_window_key_ungrab(c->man->root, "Home", ECORE_EVENT_MODIFIER_SHIFT |
ECORE_EVENT_MODIFIER_CTRL |
ECORE_EVENT_MODIFIER_ALT, 0);
if (c->grabbed)
{
c->grabbed = 0;
ecore_x_ungrab();
}
while (c->wins)
{
cw = (E_Comp_Win *)(c->wins);
if (cw->counter)
{
ecore_x_sync_counter_free(cw->counter);
cw->counter = 0;
}
cw->force = 1;
_e_comp_win_hide(cw);
cw->force = 1;
_e_comp_win_del(cw);
}
EINA_LIST_FREE(c->zones, cz)
{
if (cz->zone) cz->zone->comp_zone = NULL;
evas_object_del(cz->base);
evas_object_del(cz->over);
if (!conf->nofade)
{
if (cz->bloff)
{
if (e_backlight_mode_get(cz->zone) != E_BACKLIGHT_MODE_NORMAL)
e_backlight_mode_set(cz->zone, E_BACKLIGHT_MODE_NORMAL);
e_backlight_level_set(cz->zone, e_config->backlight.normal, 0.0);
}
}
free(cz);
}
if (c->layout) evas_object_del(c->layout);
ecore_evas_free(c->ee);
ecore_x_composite_unredirect_subwindows
(c->man->root, ECORE_X_COMPOSITE_UPDATE_MANUAL);
if (c->block_win) ecore_x_window_free(c->block_win);
ecore_x_composite_render_window_disable(c->win);
if (c->man->num == 0) e_alert_composite_win(c->man->root, 0);
if (c->render_animator) ecore_animator_del(c->render_animator);
2010-06-08 07:58:08 -07:00
if (c->new_up_timer) ecore_timer_del(c->new_up_timer);
if (c->update_job) ecore_job_del(c->update_job);
if (c->wins_list) eina_list_free(c->wins_list);
if (c->screen_job) ecore_job_del(c->screen_job);
if (c->nocomp_delay_timer) ecore_timer_del(c->nocomp_delay_timer);
if (c->nocomp_override_timer) ecore_timer_del(c->nocomp_override_timer);
2011-03-19 23:12:17 -07:00
ecore_x_window_free(c->cm_selection);
ecore_x_e_comp_sync_supported_set(c->man->root, 0);
ecore_x_screen_is_composited_set(c->man->num, 0);
free(c);
}
//////////////////////////////////////////////////////////////////////////
static void
_e_comp_sys_done_cb(void *data, Evas_Object *obj, const char *sig, const char *src)
{
edje_object_signal_callback_del(obj, sig, src, _e_comp_sys_done_cb);
e_sys_action_raw_do((E_Sys_Action)(long)data, NULL);
E_FREE_FUNC(action_timeout, ecore_timer_del);
}
static Eina_Bool
_e_comp_sys_action_timeout(void *data)
{
Eina_List *l, *ll;
E_Comp *c;
E_Comp_Zone *cz;
E_Sys_Action a = (long)(intptr_t)data;
const char *sig = NULL;
switch (a)
{
case E_SYS_LOGOUT:
sig = "e,state,sys,logout,done";
break;
case E_SYS_HALT:
sig = "e,state,sys,halt,done";
break;
case E_SYS_REBOOT:
sig = "e,state,sys,reboot,done";
break;
case E_SYS_SUSPEND:
sig = "e,state,sys,suspend,done";
break;
case E_SYS_HIBERNATE:
sig = "e,state,sys,hibernate,done";
break;
default:
break;
}
E_FREE_FUNC(action_timeout, ecore_timer_del);
if (sig)
{
EINA_LIST_FOREACH(compositors, l, c)
EINA_LIST_FOREACH(c->zones, ll, cz)
edje_object_signal_callback_del(cz->over, sig, "e", _e_comp_sys_done_cb);
}
e_sys_action_raw_do(a, NULL);
return EINA_FALSE;
}
static void
_e_comp_sys_emit_cb_wait(E_Sys_Action a, const char *sig, const char *rep, Eina_Bool nocomp_push)
{
Eina_List *l, *ll;
E_Comp_Zone *cz;
E_Comp *c;
Eina_Bool first = EINA_TRUE;
EINA_LIST_FOREACH(compositors, l, c)
{
if (nocomp_push) e_comp_override_add(c);
else _e_comp_override_timed_pop(c);
EINA_LIST_FOREACH(c->zones, ll, cz)
{
_e_comp_fade_handle(cz, nocomp_push, 0.5);
edje_object_signal_emit(cz->base, sig, "e");
edje_object_signal_emit(cz->over, sig, "e");
if ((rep) && (first))
edje_object_signal_callback_add(cz->over, rep, "e", _e_comp_sys_done_cb, (void *)(long)a);
first = EINA_FALSE;
}
}
if (rep)
{
if (action_timeout) ecore_timer_del(action_timeout);
action_timeout = ecore_timer_add(ACTION_TIMEOUT, (Ecore_Task_Cb)_e_comp_sys_action_timeout, (intptr_t*)(long)a);
}
}
static void
_e_comp_sys_suspend(void)
{
_e_comp_sys_emit_cb_wait(E_SYS_SUSPEND, "e,state,sys,suspend", "e,state,sys,suspend,done", EINA_TRUE);
}
static void
_e_comp_sys_hibernate(void)
{
_e_comp_sys_emit_cb_wait(E_SYS_HIBERNATE, "e,state,sys,hibernate", "e,state,sys,hibernate,done", EINA_TRUE);
}
static void
_e_comp_sys_reboot(void)
{
_e_comp_sys_emit_cb_wait(E_SYS_REBOOT, "e,state,sys,reboot", "e,state,sys,reboot,done", EINA_TRUE);
}
static void
_e_comp_sys_shutdown(void)
{
_e_comp_sys_emit_cb_wait(E_SYS_HALT, "e,state,sys,halt", "e,state,sys,halt,done", EINA_TRUE);
}
static void
_e_comp_sys_logout(void)
{
_e_comp_sys_emit_cb_wait(E_SYS_LOGOUT, "e,state,sys,logout", "e,state,sys,logout,done", EINA_TRUE);
}
static void
_e_comp_sys_resume(void)
{
_e_comp_sys_emit_cb_wait(E_SYS_SUSPEND, "e,state,sys,resume", NULL, EINA_FALSE);
}
static Eina_List *
_e_comp_config_engine_info_cb(E_Configure_Option *co)
{
Eina_List *ret = NULL;
E_Configure_Option_Info *oi;
int x;
const char *name[] =
{
"Software", NULL
};
if (gl_avail) name[1] = "OpenGL";
for (x = E_COMP_ENGINE_SW; x <= E_COMP_ENGINE_SW + gl_avail; x++)
{
oi = e_configure_option_info_new(co, _(name[x - 1]), (intptr_t *)(long)x);
oi->current = (*(int *)co->valptr == x);
ret = eina_list_append(ret, oi);
}
return ret;
}
static Eina_Bool
_e_comp_config_style_thumb_timer_cb(void *data)
{
Evas_Object *oo, *ofr = data;
int demo_state;
demo_state = (long)(intptr_t)evas_object_data_get(data, "style_demo_state");
demo_state = (demo_state + 1) % 4;
evas_object_data_set(data, "style_demo_state", (intptr_t *)(long)demo_state);
oo = evas_object_data_get(ofr, "comp_preview");
switch (demo_state)
{
case 0:
edje_object_signal_emit(oo, "e,state,visible,on", "e");
edje_object_signal_emit(oo, "e,state,focus,on", "e");
edje_object_part_text_set(ofr, "e.text.label", _("Visible"));
break;
case 1:
edje_object_signal_emit(oo, "e,state,focus,off", "e");
edje_object_part_text_set(ofr, "e.text.label", _("Focus-Out"));
break;
case 2:
edje_object_signal_emit(oo, "e,state,focus,on", "e");
edje_object_part_text_set(ofr, "e.text.label", _("Focus-In"));
break;
case 3:
edje_object_signal_emit(oo, "e,state,visible,off", "e");
edje_object_part_text_set(ofr, "e.text.label", _("Hidden"));
break;
default:
break;
}
return ECORE_CALLBACK_RENEW;
}
static void
_e_comp_config_style_thumb_del_cb(void *data, Evas *e EINA_UNUSED, Evas_Object *obj, void *event_info EINA_UNUSED)
{
evas_object_del(edje_object_part_swallow_get(obj, "e.swallow.preview"));
ecore_timer_del(data);
}
static Evas_Object *
_e_comp_config_style_thumb_cb(E_Configure_Option_Info *oi, Evas *evas)
{
Evas_Object *ob, *oly, *oo, *obd, *ofr, *orec;
Ecore_Timer *timer;
char buf[4096];
ob = e_livethumb_add(evas);
e_livethumb_vsize_set(ob, 240, 240);
oly = e_layout_add(e_livethumb_evas_get(ob));
e_layout_virtual_size_set(oly, 240, 240);
e_livethumb_thumb_set(ob, oly);
evas_object_show(oly);
oo = edje_object_add(e_livethumb_evas_get(ob));
snprintf(buf, sizeof(buf), "e/comp/border/%s", oi->name);
e_theme_edje_object_set(oo, "base/theme/borders", buf);
e_layout_pack(oly, oo);
e_layout_child_move(oo, 39, 39);
e_layout_child_resize(oo, 162, 162);
edje_object_signal_emit(oo, "e,state,shadow,on", "e");
edje_object_signal_emit(oo, "e,state,visible,on", "e");
evas_object_show(oo);
ofr = edje_object_add(evas);
e_theme_edje_object_set
(ofr, "base/theme/comp", "e/comp/preview");
edje_object_part_swallow(ofr, "e.swallow.preview", ob);
evas_object_show(ofr);
obd = edje_object_add(e_livethumb_evas_get(ob));
e_theme_edje_object_set(obd, "base/theme/borders", "e/widgets/border/default/border");
edje_object_part_text_set(obd, "e.text.title", _("Title"));
edje_object_signal_emit(obd, "e,state,focused", "e");
edje_object_part_swallow(oo, "e.swallow.content", obd);
evas_object_show(obd);
orec = evas_object_rectangle_add(e_livethumb_evas_get(ob));
evas_object_color_set(orec, 0, 0, 0, 128);
edje_object_part_swallow(obd, "e.swallow.client", orec);
evas_object_show(orec);
timer = ecore_timer_add(3.0, _e_comp_config_style_thumb_timer_cb, ofr);
evas_object_data_set(ofr, "style_demo_state", (void *)1);
evas_object_data_set(ofr, "comp_timer", timer);
evas_object_data_set(ofr, "comp_preview", oo);
evas_object_event_callback_add(ofr, EVAS_CALLBACK_DEL, _e_comp_config_style_thumb_del_cb, timer);
return ofr;
}
static Eina_List *
_e_comp_config_style_info_cb(E_Configure_Option *co)
{
Eina_List *ret = NULL, *styles;
Eina_Stringshare *style;
E_Configure_Option_Info *oi;
styles = e_theme_comp_border_list();
EINA_LIST_FREE(styles, style)
{
oi = e_configure_option_info_new(co, style, style);
oi->current = (conf->shadow_style == style);
ret = eina_list_append(ret, oi);
}
return ret;
}
static Eina_List *
_e_comp_fps_corner_info_cb(E_Configure_Option *co)
{
Eina_List *ret = NULL;
E_Configure_Option_Info *oi;
int x;
const char *name[] =
{
"Top left", "Top right", "Bottom left", "Bottom right", };
for (x = 0; x <= 3; x++)
{
oi = e_configure_option_info_new(co, _(name[x]), (intptr_t *)(long)x);
oi->current = (*(int *)co->valptr == x);
ret = eina_list_append(ret, oi);
}
return ret;
}
#ifdef ECORE_EVAS_GL_X11_OPT_SWAP_MODE
static Eina_List *
_e_comp_swap_mode_info_cb(E_Configure_Option *co)
{
Eina_List *ret = NULL;
E_Configure_Option_Info *oi;
int x;
const char *name[] =
{
"Auto", "Invalidate (full redraw)", "Copy from back to front", "Double-buffered", "Triple-buffered"
};
for (x = ECORE_EVAS_GL_X11_SWAP_MODE_AUTO; x <= ECORE_EVAS_GL_X11_SWAP_MODE_TRIPLE; x++)
{
oi = e_configure_option_info_new(co, _(name[x]), (intptr_t *)(long)x);
oi->current = (*(int *)co->valptr == x);
ret = eina_list_append(ret, oi);
}
return ret;
}
#endif
static void
_e_comp_cfg_init(void)
{
E_Configure_Option *co;
e_configure_option_domain_current_set("e_comp");
E_CONFIGURE_OPTION_ADD(co, BOOL, fast_borders, conf, _("Use fast composite effects for windows"), _("composite"), _("border"), _("theme"), _("animate"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, fast_menus, conf, _("Use fast composite effects for menus"), _("composite"), _("menu"), _("theme"), _("animate"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, fast_popups, conf, _("Use fast composite effects for popups"), _("composite"), _("popup"), _("theme"), _("animate"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, fast_objects, conf, _("Use fast composite effects for objects"), _("composite"), _("theme"), _("animate"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, fast_overrides, conf, _("Use fast composite effects for override-redirect windows (tooltips and such)"), _("composite"), _("theme"), _("animate"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, match.disable_borders, conf, _("Disable composite effects for windows"), _("composite"), _("border"), _("theme"), _("animate"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, match.disable_menus, conf, _("Disable composite effects for menus"), _("composite"), _("menu"), _("theme"), _("animate"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, match.disable_popups, conf, _("Disable composite effects for popups"), _("composite"), _("popup"), _("theme"), _("animate"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, match.disable_objects, conf, _("Disable composite effects for objects"), _("composite"), _("theme"), _("animate"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, match.disable_overrides, conf, _("Disable composite effects for override-redirect windows (tooltips and such)"), _("composite"), _("theme"), _("animate"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, disable_screen_effects, conf, _("Disable composite effects for the screen"), _("composite"), _("theme"), _("animate"), _("screen"));
E_CONFIGURE_OPTION_HELP(co, _("This option disables composite effects from themes, such as animating the screen fade when blanking"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, ENUM, engine, conf, _("Compositing engine"), _("composite"), _("border"));
E_CONFIGURE_OPTION_ICON(co, "preferences-engine");
co->info_cb = _e_comp_config_engine_info_cb;
co->requires_restart = 1;
E_CONFIGURE_OPTION_ADD(co, STR, shadow_style, conf, _("Default window composite effect"), _("composite"), _("border"), _("theme"));
co->info_cb = _e_comp_config_style_info_cb;
co->thumb_cb = _e_comp_config_style_thumb_cb;
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, smooth_windows, conf, _("Smooth scaling of composited window content"), _("composite"), _("border"));
/* FIXME: help text */
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, efl_sync, conf, _("Sync composited windows"), _("composite"), _("border"));
/* FIXME: help text */
co->requires_restart = 1;
E_CONFIGURE_OPTION_ADD(co, BOOL, loose_sync, conf, _("Loose sync composited windows"), _("composite"), _("border"));
/* FIXME: help text */
co->requires_restart = 1;
E_CONFIGURE_OPTION_ADD(co, BOOL, grab, conf, _("Grab server during rendering of composited windows"), _("composite"), _("border"));
/* FIXME: help text */
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, DOUBLE, first_draw_delay, conf, _("Initial draw timeout for newly-mapped composited windows"), _("composite"), _("border"), _("delay"));
E_CONFIGURE_OPTION_MINMAX_STEP_FMT(co, 0.01, 0.5, 0.01, _("%1.2f seconds"));
/* FIXME: help text */
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
if (gl_avail)
{
E_CONFIGURE_OPTION_ADD(co, BOOL, vsync, conf, _("Tear-free compositing (VSYNC)"), _("composite"), _("border"));
co->requires_restart = 1;
E_CONFIGURE_OPTION_ADD(co, BOOL, texture_from_pixmap, conf, _("Texture from pixmap rendering for composite"), _("composite"), _("border"));
co->requires_restart = 1;
#ifdef ECORE_EVAS_GL_X11_OPT_SWAP_MODE
if ((evas_version->major >= 1) &&
(evas_version->minor >= 7) &&
(evas_version->micro >= 99))
{
E_CONFIGURE_OPTION_ADD(co, ENUM, swap_mode, conf, _("Composite swapping method"), _("composite"), _("border"));
co->info_cb = _e_comp_swap_mode_info_cb;
co->requires_restart = 1;
}
#endif
}
E_CONFIGURE_OPTION_ADD(co, BOOL, nocomp_fs, conf, _("Don't composite fullscreen windows"), _("composite"), _("border"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, nofade, conf, _("Don't fade backlight"), _("composite"), _("backlight"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, send_flush, conf, _("Send flush when compositing windows"), _("composite"), _("border"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, send_dump, conf, _("Send dump when compositing windows"), _("composite"), _("border"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, BOOL, fps_show, conf, _("Show framerate when compositing windows"), _("composite"), _("border"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, DOUBLE_INT, fps_average_range, conf, _("Rolling average for fps display when compositing"), _("composite"), _("border"));
E_CONFIGURE_OPTION_MINMAX_STEP_FMT(co, 1, 120, 1, _("%1.0f frames"));
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
E_CONFIGURE_OPTION_ADD(co, ENUM, fps_corner, conf, _("Composite framerate display corner"), _("composite"), _("border"));
co->info_cb = _e_comp_fps_corner_info_cb;
co->funcs[1].none = co->funcs[0].none = e_comp_shadows_reset;
e_configure_option_category_tag_add(_("windows"), _("composite"));
e_configure_option_category_tag_add(_("composite"), _("composite"));
e_configure_option_category_icon_set(_("composite"), "preferences-composite");
}
static Eina_Bool
_e_comp_opacity_set_timer_cb(E_Comp_Win *cw)
{
unsigned int opacity;
cw->bd->client.netwm.opacity = cw->opacity;
e_remember_update(cw->bd);
opacity = (cw->opacity << 24);
ecore_x_window_prop_card32_set(cw->bd->client.win, ECORE_X_ATOM_NET_WM_WINDOW_OPACITY, &opacity, 1);
cw->bd->client.netwm.opacity_changed = 1;
cw->opacity_set_timer = NULL;
return EINA_FALSE;
}
static E_Comp_Win *
_e_comp_act_opacity_win_finder(E_Object *obj)
{
E_Border *bd;
switch (obj->type)
{
case E_WIN_TYPE:
bd = ((E_Win*)obj)->border;
if (!bd) return NULL;
return _e_comp_border_client_find(bd->client.win);
case E_BORDER_TYPE:
bd = (E_Border*)obj;
return _e_comp_border_client_find(bd->client.win);
case E_POPUP_TYPE:
return evas_object_data_get(((E_Popup*)obj)->content, "comp_win");
default:
case E_ZONE_TYPE:
case E_CONTAINER_TYPE:
case E_MANAGER_TYPE:
case E_MENU_TYPE:
bd = e_border_focused_get();
if (bd) return _e_comp_border_client_find(bd->client.win);
}
return NULL;
}
static void
_e_comp_act_opacity_change_go(E_Object *obj, const char *params)
{
int opacity;
E_Comp_Win *cw;
if ((!params) || (!params[0])) return;
cw = _e_comp_act_opacity_win_finder(obj);
if (!cw) return;
opacity = atoi(params);
opacity = E_CLAMP(opacity, -255, 255);
opacity += cw->opacity;
opacity = MAX(0, opacity);
e_comp_win_opacity_set(cw, opacity);
}
static void
_e_comp_act_opacity_set_go(E_Object * obj __UNUSED__, const char *params)
{
int opacity;
E_Comp_Win *cw;
if ((!params) || (!params[0])) return;
cw = _e_comp_act_opacity_win_finder(obj);
if (!cw) return;
opacity = atoi(params);
opacity = E_CLAMP(opacity, 0, 255);
e_comp_win_opacity_set(cw, opacity);
}
//////////////////////////////////////////////////////////////////////////
EINTERN Eina_Bool
e_comp_init(void)
{
if (!ecore_x_composite_query())
{
e_error_message_show
(_("Your display server does not support XComposite, "
"or Ecore-X was built without XComposite support. "
"Note that for composite support you will also need "
"XRender and XFixes support in X11 and Ecore."));
return EINA_FALSE;
}
if (!ecore_x_damage_query())
{
e_error_message_show
(_("Your display server does not support XDamage "
"or Ecore was built without XDamage support."));
return EINA_FALSE;
}
_e_comp_log_dom = eina_log_domain_register("e_comp", EINA_COLOR_YELLOW);
eina_log_domain_level_set("e_comp", EINA_LOG_LEVEL_INFO);
e_sys_handlers_set(_e_comp_sys_suspend, _e_comp_sys_hibernate,
_e_comp_sys_reboot, _e_comp_sys_shutdown,
_e_comp_sys_logout, _e_comp_sys_resume);
windows = eina_hash_string_superfast_new(NULL);
borders = eina_hash_string_superfast_new(NULL);
damages = eina_hash_string_superfast_new(NULL);
ignores = eina_hash_string_superfast_new(NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_X_EVENT_WINDOW_CREATE, _e_comp_create, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_X_EVENT_WINDOW_DESTROY, _e_comp_destroy, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_X_EVENT_WINDOW_SHOW, _e_comp_show, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_X_EVENT_WINDOW_HIDE, _e_comp_hide, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_X_EVENT_WINDOW_REPARENT, _e_comp_reparent, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_X_EVENT_WINDOW_CONFIGURE, _e_comp_configure, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_X_EVENT_WINDOW_STACK, _e_comp_stack, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_X_EVENT_CLIENT_MESSAGE, _e_comp_message, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_X_EVENT_WINDOW_SHAPE, _e_comp_shape, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_X_EVENT_DAMAGE_NOTIFY, _e_comp_damage, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_X_EVENT_WINDOW_DAMAGE, _e_comp_damage_win, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_SCREENSAVER_ON, _e_comp_screensaver_on, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_SCREENSAVER_OFF, _e_comp_screensaver_off, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_EVENT_KEY_DOWN, _e_comp_key_down, NULL);
E_LIST_HANDLER_APPEND(handlers, ECORE_EVENT_SIGNAL_USER, _e_comp_signal_user, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_CONTAINER_RESIZE, _e_comp_randr, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_ZONE_MOVE_RESIZE, _e_comp_zonech, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_ZONE_ADD, _e_comp_zonech, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_ZONE_DEL, _e_comp_zonech, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_ADD, _e_comp_bd_add, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_REMOVE, _e_comp_bd_del, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_SHOW, _e_comp_bd_show, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_HIDE, _e_comp_bd_hide, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_MOVE, _e_comp_bd_move, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_RESIZE, _e_comp_bd_resize, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_ICONIFY, _e_comp_bd_iconify, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_UNICONIFY, _e_comp_bd_uniconify, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_URGENT_CHANGE, _e_comp_bd_urgent_change, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_FOCUS_IN, _e_comp_bd_focus_in, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_FOCUS_OUT, _e_comp_bd_focus_out, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_PROPERTY, _e_comp_bd_property, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_FULLSCREEN, _e_comp_bd_fullscreen, NULL);
E_LIST_HANDLER_APPEND(handlers, E_EVENT_BORDER_UNFULLSCREEN, _e_comp_bd_unfullscreen, NULL);
E_EVENT_COMP_SOURCE_VISIBILITY = ecore_event_type_new();
E_EVENT_COMP_SOURCE_ADD = ecore_event_type_new();
E_EVENT_COMP_SOURCE_DEL = ecore_event_type_new();
E_EVENT_COMP_SOURCE_CONFIGURE = ecore_event_type_new();
2013-04-19 05:49:34 -07:00
E_EVENT_COMP_SOURCE_STACK = ecore_event_type_new();
e_comp_cfdata_edd_init(&conf_edd, &conf_match_edd);
conf = e_config_domain_load("e_comp", conf_edd);
if (conf)
{
conf->max_unmapped_pixels = 32 * 1024;
conf->keep_unmapped = 1;
}
else
conf = e_comp_cfdata_config_new();
if (!getenv("ECORE_X_NO_XLIB"))
{
if (ecore_evas_engine_type_supported_get(ECORE_EVAS_ENGINE_OPENGL_X11))
gl_avail = EINA_TRUE;
}
{
E_Action *act;
act = e_action_add("opacity_change");
act->func.go = _e_comp_act_opacity_change_go;
e_action_predef_name_set(N_("Compositor"),
N_("Change current window opacity"), "opacity_change",
NULL, "syntax: +/- the amount to change opacity by (>0 for more opaque)", 1);
actions = eina_list_append(actions, act);
act = e_action_add("opacity_set");
act->func.go = _e_comp_act_opacity_set_go;
e_action_predef_name_set(N_("Compositor"),
N_("Set current window opacity"), "opacity_set",
"255", "syntax: number between 0-255 to set for transparent-opaque", 1);
actions = eina_list_append(actions, act);
}
#ifdef HAVE_WAYLAND_CLIENTS
if (!e_comp_wl_init())
EINA_LOG_ERR("Failed to initialize Wayland Client Support !!");
#endif
_e_comp_cfg_init();
return EINA_TRUE;
}
EINTERN Eina_Bool
e_comp_manager_init(E_Manager *man)
{
E_Comp *c;
c = _e_comp_add(man);
if (c)
compositors = eina_list_append(compositors, c);
return !!c;
}
EAPI int
e_comp_internal_save(void)
{
return e_config_domain_save("e_comp", conf_edd, conf);
}
EINTERN int
e_comp_shutdown(void)
{
if (!compositors) return 1;
E_FREE_FUNC(action_timeout, ecore_timer_del);
E_FREE_LIST(compositors, _e_comp_del);
E_FREE_LIST(handlers, ecore_event_handler_del);
E_FREE_LIST(actions, e_object_del);
#ifdef HAVE_WAYLAND_CLIENTS
e_comp_wl_shutdown();
#endif
e_configure_option_domain_clear("e_comp");
e_configure_option_category_tag_del(_("composite"), _("composite"));
e_configure_option_category_tag_del(_("windows"), _("composite"));
gl_avail = EINA_FALSE;
e_comp_cfdata_config_free(conf);
E_CONFIG_DD_FREE(conf_match_edd);
E_CONFIG_DD_FREE(conf_edd);
conf = NULL;
conf_match_edd = NULL;
conf_edd = NULL;
if (ignores) eina_hash_free(ignores);
if (damages) eina_hash_free(damages);
if (windows) eina_hash_free(windows);
if (borders) eina_hash_free(borders);
ignores = NULL;
damages = NULL;
windows = NULL;
borders = NULL;
e_sys_handlers_set(NULL, NULL, NULL, NULL, NULL, NULL);
return 1;
}
EAPI E_Comp_Config *
e_comp_config_get(void)
{
return conf;
}
EAPI const Eina_List *
e_comp_list(void)
{
return compositors;
}
EAPI void
e_comp_shadows_reset(void)
{
Eina_List *l;
E_Comp *c;
EINA_LIST_FOREACH(compositors, l, c)
{
E_Comp_Win *cw;
// ecore_evas_manual_render_set(c->ee, conf->lock_fps);
_e_comp_fps_update(c);
E_LIST_FOREACH(c->zones, e_comp_zone_update);
EINA_INLIST_FOREACH(c->wins, cw)
{
if ((!cw->shobj) || (!cw->obj)) continue;
_e_comp_win_shadow_setup(cw);
}
}
}
2011-03-19 23:12:17 -07:00
EAPI void
e_comp_render_update(E_Comp *c)
{
_e_comp_render_queue(c);
}
EAPI E_Comp_Win *
e_comp_win_find_client_win(Ecore_X_Window win)
{
return _e_comp_border_client_find(win);
}
EAPI E_Comp_Win *
e_comp_win_find(Ecore_X_Window win)
{
return _e_comp_win_find(win);
}
EAPI const Eina_List *
e_comp_win_list_get(E_Comp *c)
{
E_Comp_Win *cw;
if (!c->wins) return NULL;
if (c->wins_invalid)
{
c->wins_invalid = 0;
if (c->wins_list) eina_list_free(c->wins_list);
c->wins_list = NULL;
EINA_INLIST_FOREACH(c->wins, cw)
{
if ((cw->shobj) && (cw->obj))
c->wins_list = eina_list_append(c->wins_list, cw);
}
}
return c->wins_list;
}
EAPI Evas_Object *
e_comp_win_image_mirror_add(E_Comp_Win *cw)
{
if ((!cw) || (!cw->c)) return NULL;
return _e_comp_win_mirror_add(cw);
}
EAPI void
e_comp_win_hidden_set(E_Comp_Win *cw, Eina_Bool hidden)
{
if (!cw->c) return;
if (cw->hidden_override == hidden) return;
cw->hidden_override = hidden;
if (cw->bd) e_border_comp_hidden_set(cw->bd, cw->hidden_override);
if (cw->visible)
{
if (cw->hidden_override)
_e_comp_child_hide(cw);
else if (!cw->bd || cw->bd->visible)
_e_comp_child_show(cw);
}
else
{
if (cw->hidden_override) _e_comp_child_hide(cw);
}
}
EAPI void
e_comp_win_reshadow(E_Comp_Win *cw)
{
if (cw->visible) evas_object_hide(cw->effect_obj);
_e_comp_win_shadow_setup(cw);
// evas_object_move(cw->effect_obj, cw->x, cw->y);
// evas_object_resize(cw->effect_obj, cw->pw, cw->ph);
if (!cw->visible) return;
cw->geom_update = 1;
evas_object_show(cw->effect_obj);
if (cw->show_ready)
{
cw->defer_hide = 0;
if (!cw->hidden_override) _e_comp_child_show(cw);
edje_object_signal_emit(cw->shobj, "e,state,visible,on", "e");
if (!cw->animating)
{
cw->c->animating++;
}
cw->animating = 1;
_e_comp_win_render_queue(cw);
}
}
EAPI E_Comp *
e_comp_get(void *o)
{
E_Border *bd;
E_Popup *pop;
E_Shelf *es;
E_Menu *m;
2013-04-19 07:18:28 -07:00
E_Desk *desk;
E_Menu_Item *mi;
E_Object *obj = o;
E_Zone *zone = NULL;
E_Container *con = NULL;
E_Manager *man = NULL;
E_Gadcon_Popup *gp;
E_Gadcon *gc;
E_Gadcon_Client *gcc;
E_Drag *drag;
if (!o) obj = (E_Object*)e_manager_current_get();
/* try to get to zone type first */
switch (obj->type)
{
2013-04-19 07:18:28 -07:00
case E_DESK_TYPE:
desk = (E_Desk*)obj;
obj = (void*)desk->zone;
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
break;
case E_BORDER_TYPE:
bd = (E_Border*)obj;
obj = (void*)bd->zone;
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
break;
case E_POPUP_TYPE:
pop = (E_Popup*)obj;
obj = (void*)pop->zone;
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
break;
case E_MENU_TYPE:
m = (E_Menu*)obj;
obj = (void*)m->zone;
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
break;
case E_MENU_ITEM_TYPE:
mi = (E_Menu_Item*)obj;
obj = (void*)mi->menu->zone;
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
break;
case E_SHELF_TYPE:
es = (E_Shelf*)obj;
obj = (void*)es->zone;
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
break;
case E_GADCON_POPUP_TYPE:
gp = (E_Gadcon_Popup*)obj;
obj = (void*)gp->win->zone;
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
break;
case E_DRAG_TYPE:
drag = (E_Drag*)obj;
obj = (void*)drag->container;
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
break;
case E_GADCON_CLIENT_TYPE:
gcc = (E_Gadcon_Client*)obj;
obj = (void*)gcc->gadcon;
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
case E_GADCON_TYPE:
gc = (E_Gadcon*)obj;
obj = (void*)e_gadcon_zone_get(gc);
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
break;
default:
break;
}
switch (obj->type)
{
case E_ZONE_TYPE:
if (!zone) zone = (E_Zone*)obj;
con = zone->container;
EINA_SAFETY_ON_NULL_RETURN_VAL(con, NULL);
case E_CONTAINER_TYPE:
if (!con) con = (E_Container*)obj;
man = con->manager;
EINA_SAFETY_ON_NULL_RETURN_VAL(man, NULL);
case E_MANAGER_TYPE:
if (!man) man = (E_Manager*)obj;
return man->comp;
}
CRI("UNIMPLEMENTED TYPE PASSED! FIXME!");
return NULL;
}
EAPI void
e_comp_zone_update(E_Comp_Zone *cz)
{
Evas_Object *o;
const char *const over_styles[] =
{
"e/comp/screen/overlay/default",
"e/comp/screen/overlay/noeffects"
};
const char *const under_styles[] =
{
"e/comp/screen/base/default",
"e/comp/screen/base/noeffects"
};
if (cz->over && cz->base)
{
e_theme_edje_object_set(cz->base, "base/theme/comp",
under_styles[conf->disable_screen_effects]);
edje_object_part_swallow(cz->base, "e.swallow.background",
cz->zone->transition_object ?: cz->zone->bg_object);
e_theme_edje_object_set(cz->over, "base/theme/comp",
over_styles[conf->disable_screen_effects]);
return;
}
E_FREE_FUNC(cz->base, evas_object_del);
E_FREE_FUNC(cz->over, evas_object_del);
cz->base = o = edje_object_add(cz->comp->evas);
evas_object_repeat_events_set(o, 1);
evas_object_name_set(cz->base, "cz->base");
e_theme_edje_object_set(o, "base/theme/comp", under_styles[conf->disable_screen_effects]);
edje_object_part_swallow(cz->base, "e.swallow.background", cz->zone->transition_object ?: cz->zone->bg_object);
evas_object_move(o, cz->zone->x, cz->zone->y);
evas_object_resize(o, cz->zone->w, cz->zone->h);
evas_object_layer_set(o, E_COMP_CANVAS_LAYER_BG);
evas_object_show(o);
cz->over = o = edje_object_add(cz->comp->evas);
evas_object_layer_set(o, E_COMP_CANVAS_LAYER_MAX);
evas_object_raise(o);
evas_object_name_set(cz->over, "cz->over");
evas_object_pass_events_set(o, 1);
e_theme_edje_object_set(o, "base/theme/comp", over_styles[conf->disable_screen_effects]);
evas_object_move(o, cz->zone->x, cz->zone->y);
evas_object_resize(o, cz->zone->w, cz->zone->h);
evas_object_raise(o);
evas_object_show(o);
}
EAPI Ecore_X_Window
e_comp_top_window_at_xy_get(E_Comp *c, Evas_Coord x, Evas_Coord y, Eina_Bool vis, Ecore_X_Window *ignore, unsigned int ignore_num)
{
E_Comp_Win *cw;
Evas_Object *o;
Eina_List *ignore_list = NULL;
EINA_SAFETY_ON_NULL_RETURN_VAL(c, 0);
o = evas_object_top_at_xy_get(c->evas, x, y, 0, 0);
if (!o) return c->ee_win;
if (o != c->layout) return c->ee_win;
if (ignore && ignore_num)
{
unsigned int i;
for (i = 0; i < ignore_num; i++)
{
cw = e_comp_win_find(ignore[i]);
if (cw)
ignore_list = eina_list_append(ignore_list, cw->effect_obj);
}
}
o = e_layout_top_child_at_xy_get(c->layout, x, y, vis, ignore_list);
eina_list_free(ignore_list);
cw = evas_object_data_get(o, "comp_win");
if (!cw) return c->ee_win;
return cw->real_obj ? c->ee_win : cw->win;
}
EAPI E_Comp_Win *
e_comp_object_inject(E_Comp *c, Evas_Object *obj, E_Object *eobj, E_Layer layer)
{
E_Comp_Win *cw, *cwn;
E_Container *con;
int pos;
EINA_SAFETY_ON_NULL_RETURN_VAL(c, NULL);
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
con = e_container_current_get(c->man);
EINA_SAFETY_ON_NULL_RETURN_VAL(con, NULL);
pos = 1 + (layer / 50);
if (pos > 10) pos = 10;
cw = _e_comp_win_find(con->layers[pos].win);
EINA_SAFETY_ON_NULL_RETURN_VAL(cw, NULL);
cwn = _e_comp_win_dummy_add(c, obj, eobj, 0);
_e_comp_win_lower_below(cwn, (E_Comp_Win*)EINA_INLIST_GET(cw));
cw->stack_below = eina_list_append(cw->stack_below, cwn);
cwn->cw_above = cw;
evas_object_event_callback_add(obj, EVAS_CALLBACK_DEL, _e_comp_injected_win_del_cb, cwn);
evas_object_event_callback_add(obj, EVAS_CALLBACK_FOCUS_IN, _e_comp_injected_win_focus_in_cb, cwn);
evas_object_event_callback_add(obj, EVAS_CALLBACK_FOCUS_OUT, _e_comp_injected_win_focus_out_cb, cwn);
return cwn;
}
EAPI E_Comp_Win *
e_comp_object_add(E_Comp *c, Evas_Object *obj, E_Object *eobj)
{
E_Comp_Win *cw;
EINA_SAFETY_ON_NULL_RETURN_VAL(c, NULL);
EINA_SAFETY_ON_NULL_RETURN_VAL(obj, NULL);
cw = _e_comp_win_dummy_add(c, obj, eobj, 1);
evas_object_event_callback_add(obj, EVAS_CALLBACK_DEL, _e_comp_injected_win_del_cb, cw);
evas_object_event_callback_add(obj, EVAS_CALLBACK_SHOW, _e_comp_injected_win_show_cb, cw);
evas_object_event_callback_add(obj, EVAS_CALLBACK_HIDE, _e_comp_injected_win_hide_cb, cw);
evas_object_event_callback_add(obj, EVAS_CALLBACK_FOCUS_IN, _e_comp_injected_win_focus_in_cb, cw);
evas_object_event_callback_add(obj, EVAS_CALLBACK_FOCUS_OUT, _e_comp_injected_win_focus_out_cb, cw);
return cw;
}
EAPI void
e_comp_win_move(E_Comp_Win *cw, Evas_Coord x, Evas_Coord y)
{
EINA_SAFETY_ON_FALSE_RETURN(cw->real_obj);
if (cw->visible)
{
if ((cw->x == x) && (cw->y == y)) return;
}
else
{
if ((cw->hidden.x == x) && (cw->hidden.y == y)) return;
}
if (cw->shape) e_container_shape_move(cw->shape, x, y);
_e_comp_win_configure(cw, x, y, cw->w, cw->h, 0);
}
EAPI void
e_comp_win_resize(E_Comp_Win *cw, int w, int h)
{
EINA_SAFETY_ON_FALSE_RETURN(cw->real_obj);
if ((cw->w == w) && (cw->h == h)) return;
if (cw->shape) e_container_shape_resize(cw->shape, w, h);
edje_extern_object_min_size_set(cw->obj, w, h);
_e_comp_win_configure(cw, cw->x, cw->y, w, h, 0);
}
EAPI void
e_comp_win_moveresize(E_Comp_Win *cw, Evas_Coord x, Evas_Coord y, int w, int h)
{
EINA_SAFETY_ON_FALSE_RETURN(cw->real_obj);
if ((cw->w == w) && (cw->h == h))
{
if (cw->visible)
{
if ((cw->x == x) && (cw->y == y)) return;
}
else
{
if ((cw->hidden.x == x) && (cw->hidden.y == y)) return;
}
}
if (cw->shape)
{
e_container_shape_move(cw->shape, x, y);
e_container_shape_resize(cw->shape, w, h);
}
edje_extern_object_min_size_set(cw->obj, w, h);
_e_comp_win_configure(cw, x, y, w, h, 0);
}
EAPI void
e_comp_win_del(E_Comp_Win *cw)
{
if (!cw) return;
if (cw->animating) cw->delete_me = 1;
else _e_comp_win_del(cw);
}
EAPI void
e_comp_win_hide(E_Comp_Win *cw)
{
EINA_SAFETY_ON_NULL_RETURN(cw);
EINA_SAFETY_ON_FALSE_RETURN(cw->real_obj);
_e_comp_win_real_hide(cw);
}
EAPI void
e_comp_win_show(E_Comp_Win *cw)
{
EINA_SAFETY_ON_NULL_RETURN(cw);
EINA_SAFETY_ON_FALSE_RETURN(cw->real_obj);
cw->defer_hide = 0;
cw->show_ready = 1;
_e_comp_win_geometry_update(cw);
_e_comp_win_show(cw);
}
EAPI E_Comp_Win *
e_comp_canvas_layer_set(Evas_Object *obj, E_Comp_Canvas_Layer comp_layer, E_Layer layer, E_Comp_Canvas_Stack stack)
{
E_Comp_Win *cw;
E_Comp *c;
c = e_comp_util_evas_object_comp_get(obj);
if (comp_layer == E_COMP_CANVAS_LAYER_LAYOUT)
cw = e_comp_object_inject(c, obj, evas_object_data_get(obj, "eobj"), layer);
else
{
cw = e_comp_object_add(c, obj, evas_object_data_get(obj, "eobj"));
evas_object_layer_set(cw->effect_obj, comp_layer);
if (comp_layer > E_COMP_CANVAS_LAYER_LAYOUT)
e_comp_override_add(c);
else
{
E_Comp_Win *cwn;
E_Container *con;
con = eina_list_data_get(c->man->containers);
cwn = e_comp_win_find(con->layers[0].win);
if (!cwn)
{
ERR("Major error. Cannot find container layer 0 window marker");
c->wins = eina_inlist_prepend(c->wins, EINA_INLIST_GET(cw));
}
else
{
cwn->stack_below = eina_list_append(cwn->stack_below, cw);
cw->cw_above = cwn;
c->wins = eina_inlist_remove(c->wins, EINA_INLIST_GET(cw));
c->wins = eina_inlist_prepend_relative(c->wins, EINA_INLIST_GET(cw), EINA_INLIST_GET(cwn));
}
}
}
if (stack == E_COMP_CANVAS_STACK_ABOVE)
_e_comp_win_raise(cw);
else if (stack == E_COMP_CANVAS_STACK_UNDER)
_e_comp_win_lower(cw);
return cw;
}
EAPI void
e_comp_util_wins_print(const E_Comp *c)
{
E_Comp_Win *cw;
if (!c) c = e_comp_get(NULL);
EINA_INLIST_FOREACH(c->wins, cw)
{
if (cw->bd)
fprintf(stderr, "COMP BD: %p - %u '%s:%s'\n", cw, cw->win, cw->bd->client.icccm.name, cw->bd->client.icccm.class);
else if (cw->pop)
fprintf(stderr, "COMP POP: %p - %s\n", cw, cw->pop->name);
else if (cw->menu)
fprintf(stderr, "COMP MENU: %p - %s\n", cw, cw->menu->header.title);
else if (cw->real_obj)
fprintf(stderr, "COMP OBJ: %p - %s\n", cw, evas_object_name_get(cw->obj));
else
fprintf(stderr, "COMP WIN: %p - %u%s%s\n", cw, cw->win, cw->input_only ? " INPUT" : "", cw->override ? " OVERRIDE" : "");
}
}
EAPI void
e_comp_ignore_win_add(Ecore_X_Window win)
{
EINA_SAFETY_ON_TRUE_RETURN(_e_comp_ignore_find(win));
eina_hash_add(ignores, e_util_winid_str_get(win), (void*)1);
}
EAPI void
e_comp_win_opacity_set(E_Comp_Win *cw, unsigned int opacity)
{
EINA_SAFETY_ON_NULL_RETURN(cw);
if (opacity == cw->opacity) return;
opacity = MIN(opacity, 255);
cw->opacity = opacity;
if (cw->bd)
{
if (cw->opacity_set_timer) ecore_timer_reset(cw->opacity_set_timer);
else cw->opacity_set_timer = ecore_timer_add(5.0, (Ecore_Task_Cb)_e_comp_opacity_set_timer_cb, cw);
}
if (cw->opacity == 255)
{
edje_object_color_class_del(cw->shobj, "comp_alpha");
}
else
{
edje_object_color_class_set(cw->shobj, "comp_alpha",
255, 255, 255, cw->opacity,
255, 255, 255, cw->opacity,
255, 255, 255, cw->opacity);
}
}
EAPI void
e_comp_override_del(E_Comp *c)
{
c->nocomp_override--;
if (c->nocomp_override <= 0)
{
c->nocomp_override = 0;
if (c->nocomp_want) _e_comp_cb_nocomp_begin(c);
}
}
EAPI void
e_comp_override_add(E_Comp *c)
{
c->nocomp_override++;
if ((c->nocomp_override > 0) && (c->nocomp)) _e_comp_cb_nocomp_end(c);
}
EAPI void
e_comp_block_window_add(void)
{
E_Comp *c;
Eina_List *l;
EINA_LIST_FOREACH(compositors, l, c)
{
c->block_count++;
if (c->block_win) continue;
c->block_win = ecore_x_window_new(c->man->root, c->man->x, c->man->y, c->man->w, c->man->h);
INF("BLOCK WIN: %x", c->block_win);
ecore_x_window_background_color_set(c->block_win, 0, 0, 0);
e_comp_ignore_win_add(c->block_win);
ecore_x_window_configure(c->block_win,
ECORE_X_WINDOW_CONFIGURE_MASK_SIBLING | ECORE_X_WINDOW_CONFIGURE_MASK_STACK_MODE,
0, 0, 0, 0, 0, ((E_Comp_Win*)c->wins)->win, ECORE_X_WINDOW_STACK_ABOVE);
ecore_x_window_show(c->block_win);
}
}
EAPI void
e_comp_block_window_del(void)
{
E_Comp *c;
Eina_List *l;
EINA_LIST_FOREACH(compositors, l, c)
{
if (!c->block_count) continue;
c->block_count--;
if (c->block_count) continue;
if (c->block_win) ecore_x_window_free(c->block_win);
c->block_win = 0;
}
}
EAPI void
e_comp_win_effect_set(E_Comp_Win *cw, const char *effect)
{
char buf[4096];
Eina_Stringshare *grp;
EINA_SAFETY_ON_NULL_RETURN(cw);
EINA_SAFETY_ON_NULL_RETURN(effect);
EINA_SAFETY_ON_TRUE_RETURN(!effect[0]);
if (!cw->shobj) return; //input window
snprintf(buf, sizeof(buf), "e/comp/effects/%s", effect);
edje_object_file_get(cw->effect_obj, NULL, &grp);
if (!e_util_strcmp(buf, grp)) return;
if (!e_theme_edje_object_set(cw->effect_obj, "base/theme/comp", buf))
{
snprintf(buf, sizeof(buf), "e/comp/effects/auto/%s", effect);
if (!e_theme_edje_object_set(cw->effect_obj, "base/theme/comp", buf))
if (!e_theme_edje_object_set(cw->effect_obj, "base/theme/comp", "e/comp/effects/none")) return;
}
edje_object_part_swallow(cw->effect_obj, "e.swallow.content", cw->shobj);
if (cw->effect_clip)
{
evas_object_clip_unset(cw->effect_obj);
cw->effect_clip = 0;
}
cw->effect_clip_able = !edje_object_data_get(cw->shobj, "noclip");
}
EAPI void
e_comp_win_effect_params_set(E_Comp_Win *cw, int id, int *params, unsigned int count)
{
Edje_Message_Int_Set *msg;
unsigned int x;
EINA_SAFETY_ON_NULL_RETURN(cw);
EINA_SAFETY_ON_NULL_RETURN(params);
EINA_SAFETY_ON_FALSE_RETURN(count);
msg = alloca(sizeof(Edje_Message_Int_Set) + ((count - 1) * sizeof(int)));
msg->count = (int)count;
for (x = 0; x < count; x++)
msg->val[x] = params[x];
edje_object_message_send(cw->effect_obj, EDJE_MESSAGE_INT_SET, id, msg);
edje_object_message_signal_process(cw->effect_obj);
}
static void
_e_comp_win_effect_end_cb(void *data EINA_UNUSED, Evas_Object *obj, const char *emission, const char *source)
{
Edje_Signal_Cb end_cb;
void *end_data;
end_cb = evas_object_data_get(obj, "_e_comp.end_cb");
end_data = evas_object_data_get(obj, "_e_comp.end_data");
end_cb(end_data, obj, emission, source);
edje_object_signal_callback_del_full(obj, "e,action,done", "e", _e_comp_win_effect_end_cb, NULL);
}
EAPI void
e_comp_win_effect_unclip(E_Comp_Win *cw)
{
EINA_SAFETY_ON_NULL_RETURN(cw);
if (!cw->effect_clip) return;
evas_object_clip_unset(cw->effect_obj);
cw->effect_clip = 0;
}
EAPI void
e_comp_win_effect_start(E_Comp_Win *cw, Edje_Signal_Cb end_cb, const void *end_data)
{
EINA_SAFETY_ON_NULL_RETURN(cw);
EINA_SAFETY_ON_FALSE_RETURN(cw->bd); //FIXME
if (cw->effect_clip)
{
evas_object_clip_unset(cw->effect_obj);
cw->effect_clip = 0;
}
if (cw->effect_clip_able)
{
if (cw->bd->zone)
{
evas_object_clip_set(cw->effect_obj, cw->bd->zone->bg_clip_object);
cw->effect_clip = 1;
}
}
/* this is a stack, so the callbacks added first will be called last */
edje_object_signal_callback_del_full(cw->effect_obj, "e,action,done", "e", _e_comp_win_effect_end_cb, NULL);
edje_object_signal_callback_add(cw->effect_obj, "e,action,done", "e", _e_comp_win_effect_end_cb, NULL);
evas_object_data_set(cw->effect_obj, "_e_comp.end_cb", end_cb);
evas_object_data_set(cw->effect_obj, "_e_comp.end_data", end_data);
edje_object_signal_emit(cw->effect_obj, "e,action,go", "e");
}
EAPI void
e_comp_win_effect_stop(E_Comp_Win *cw, Edje_Signal_Cb end_cb EINA_UNUSED)
{
EINA_SAFETY_ON_NULL_RETURN(cw);
if (cw->effect_clip)
{
evas_object_clip_unset(cw->effect_obj);
cw->effect_clip = 0;
}
edje_object_signal_emit(cw->effect_obj, "e,action,stop", "e");
edje_object_signal_callback_del_full(cw->effect_obj, "e,action,done", "e", _e_comp_win_effect_end_cb, NULL);
}
EAPI unsigned int
e_comp_e_object_layer_get(const E_Object *obj)
{
if (!obj) return 0;
switch (obj->type)
{
E_Gadcon *gc;
case E_GADCON_TYPE:
gc = (E_Gadcon *)obj;
if (gc->shelf)
{
if (gc->shelf->popup)
return E_COMP_CANVAS_LAYER_LAYOUT + gc->shelf->popup->layer;
return E_COMP_CANVAS_LAYER_DESKTOP_TOP;
}
if (!gc->toolbar) return E_COMP_CANVAS_LAYER_DESKTOP;
return E_COMP_CANVAS_LAYER_LAYOUT + gc->toolbar->fwin->border->layer;
case E_GADCON_CLIENT_TYPE:
gc = ((E_Gadcon_Client *)(obj))->gadcon;
if (gc->shelf)
{
if (gc->shelf->popup)
return E_COMP_CANVAS_LAYER_LAYOUT + gc->shelf->popup->layer;
return E_COMP_CANVAS_LAYER_DESKTOP_TOP;
}
if (!gc->toolbar) return E_COMP_CANVAS_LAYER_DESKTOP;
return E_COMP_CANVAS_LAYER_LAYOUT + gc->toolbar->fwin->border->layer;
case E_WIN_TYPE:
return E_COMP_CANVAS_LAYER_LAYOUT + ((E_Win *)(obj))->border->layer;
case E_ZONE_TYPE:
return E_COMP_CANVAS_LAYER_DESKTOP;
case E_BORDER_TYPE:
return E_COMP_CANVAS_LAYER_LAYOUT + ((E_Win *)(obj))->border->layer;
case E_POPUP_TYPE:
return ((E_Popup *)(obj))->comp_layer + ((E_Popup *)(obj))->layer;
/* FIXME: add more types as needed */
default:
break;
}
return 0;
}