Drawable: Ignore exposes when we have nothing to draw
The X11 server tells us about things that need to be redrawn via expose events. When we get such an expose event before lua drew the drawable, we just fill the exposed area with old data (which is black for newly-created drawables). Fix this by tracking if we have any usable data in a drawable's double buffering pixmap. This flag is unset whenever we throw away the old content (e.g. due to a resize) and is set when lua gave us some new content to display. Signed-off-by: Uli Schlachter <psychon@znc.in>
This commit is contained in:
parent
873358a0b8
commit
86f3b7f11f
|
@ -1511,14 +1511,18 @@ client_get_drawable(client_t *c, int x, int y)
|
||||||
static void
|
static void
|
||||||
client_refresh_titlebar_partial(client_t *c, client_titlebar_t bar, int16_t x, int16_t y, uint16_t width, uint16_t height)
|
client_refresh_titlebar_partial(client_t *c, client_titlebar_t bar, int16_t x, int16_t y, uint16_t width, uint16_t height)
|
||||||
{
|
{
|
||||||
if(c->titlebar[bar].drawable == NULL || c->titlebar[bar].drawable->pixmap == XCB_NONE)
|
if(c->titlebar[bar].drawable == NULL
|
||||||
|
|| c->titlebar[bar].drawable->pixmap == XCB_NONE
|
||||||
|
|| !c->titlebar[bar].drawable->refreshed)
|
||||||
return;
|
return;
|
||||||
area_t area = titlebar_get_area(c, bar);
|
|
||||||
/* Is the titlebar part of the area that should get redrawn? */
|
/* Is the titlebar part of the area that should get redrawn? */
|
||||||
|
area_t area = titlebar_get_area(c, bar);
|
||||||
if (AREA_LEFT(area) >= x + width || AREA_RIGHT(area) <= x)
|
if (AREA_LEFT(area) >= x + width || AREA_RIGHT(area) <= x)
|
||||||
return;
|
return;
|
||||||
if (AREA_TOP(area) >= y + height || AREA_BOTTOM(area) <= y)
|
if (AREA_TOP(area) >= y + height || AREA_BOTTOM(area) <= y)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* Redraw the affected parts */
|
/* Redraw the affected parts */
|
||||||
cairo_surface_flush(c->titlebar[bar].drawable->surface);
|
cairo_surface_flush(c->titlebar[bar].drawable->surface);
|
||||||
xcb_copy_area(globalconf.connection, c->titlebar[bar].drawable->pixmap, c->frame_window,
|
xcb_copy_area(globalconf.connection, c->titlebar[bar].drawable->pixmap, c->frame_window,
|
||||||
|
|
|
@ -34,6 +34,7 @@ drawable_allocator(lua_State *L, drawable_refresh_callback *callback, void *data
|
||||||
drawable_t *d = drawable_new(L);
|
drawable_t *d = drawable_new(L);
|
||||||
d->refresh_callback = callback;
|
d->refresh_callback = callback;
|
||||||
d->refresh_data = data;
|
d->refresh_data = data;
|
||||||
|
d->refreshed = false;
|
||||||
d->surface = NULL;
|
d->surface = NULL;
|
||||||
d->pixmap = XCB_NONE;
|
d->pixmap = XCB_NONE;
|
||||||
return d;
|
return d;
|
||||||
|
@ -46,6 +47,7 @@ drawable_unset_surface(drawable_t *d)
|
||||||
cairo_surface_destroy(d->surface);
|
cairo_surface_destroy(d->surface);
|
||||||
if (d->pixmap)
|
if (d->pixmap)
|
||||||
xcb_free_pixmap(globalconf.connection, d->pixmap);
|
xcb_free_pixmap(globalconf.connection, d->pixmap);
|
||||||
|
d->refreshed = false;
|
||||||
d->surface = NULL;
|
d->surface = NULL;
|
||||||
d->pixmap = XCB_NONE;
|
d->pixmap = XCB_NONE;
|
||||||
}
|
}
|
||||||
|
@ -108,6 +110,7 @@ static int
|
||||||
luaA_drawable_refresh(lua_State *L)
|
luaA_drawable_refresh(lua_State *L)
|
||||||
{
|
{
|
||||||
drawable_t *drawable = luaA_checkudata(L, 1, &drawable_class);
|
drawable_t *drawable = luaA_checkudata(L, 1, &drawable_class);
|
||||||
|
drawable->refreshed = true;
|
||||||
(*drawable->refresh_callback)(drawable->refresh_data);
|
(*drawable->refresh_callback)(drawable->refresh_data);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
|
@ -38,6 +38,8 @@ struct drawable_t
|
||||||
cairo_surface_t *surface;
|
cairo_surface_t *surface;
|
||||||
/** The geometry of the drawable (in root window coordinates). */
|
/** The geometry of the drawable (in root window coordinates). */
|
||||||
area_t geometry;
|
area_t geometry;
|
||||||
|
/** Surface contents are undefined if this is false. */
|
||||||
|
bool refreshed;
|
||||||
/** Callback for refreshing. */
|
/** Callback for refreshing. */
|
||||||
drawable_refresh_callback *refresh_callback;
|
drawable_refresh_callback *refresh_callback;
|
||||||
/** Data for refresh callback. */
|
/** Data for refresh callback. */
|
||||||
|
|
|
@ -177,7 +177,7 @@ drawin_refresh_pixmap_partial(drawin_t *drawin,
|
||||||
int16_t x, int16_t y,
|
int16_t x, int16_t y,
|
||||||
uint16_t w, uint16_t h)
|
uint16_t w, uint16_t h)
|
||||||
{
|
{
|
||||||
if (!drawin->drawable || !drawin->drawable->pixmap)
|
if (!drawin->drawable || !drawin->drawable->pixmap || !drawin->drawable->refreshed)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* Make cairo do all pending drawing */
|
/* Make cairo do all pending drawing */
|
||||||
|
|
Loading…
Reference in New Issue