Switch imagebox to use cairo surface from oocairo

For compatibility one can now set an imagebox' .image to both a cairo surface
and an image object. The image will automatically be converted into a cairo
image surface.

Signed-off-by: Uli Schlachter <psychon@znc.in>
This commit is contained in:
Uli Schlachter 2010-08-20 12:13:03 +02:00
parent 7b53a73b0f
commit f97f605fe0
1 changed files with 56 additions and 11 deletions

View File

@ -26,7 +26,7 @@
typedef struct typedef struct
{ {
/** Imagebox image */ /** Imagebox image */
image_t *image; cairo_surface_t *image;
color_t bg; color_t bg;
bool resize; bool resize;
} imagebox_data_t; } imagebox_data_t;
@ -39,8 +39,8 @@ imagebox_extents(lua_State *L, widget_t *widget)
if(d->image) if(d->image)
{ {
geometry.width = image_getwidth(d->image); geometry.width = cairo_image_surface_get_width(d->image);
geometry.height = image_getheight(d->image); geometry.height = cairo_image_surface_get_height(d->image);
} }
else else
{ {
@ -67,8 +67,8 @@ imagebox_draw(widget_t *widget, draw_context_t *ctx, area_t geometry, wibox_t *p
if(d->bg.initialized) if(d->bg.initialized)
draw_rectangle(ctx, geometry, 1.0, true, &d->bg); draw_rectangle(ctx, geometry, 1.0, true, &d->bg);
double ratio = d->resize ? (double) geometry.height / image_getheight(d->image) : 1; double ratio = d->resize ? (double) geometry.height / cairo_image_surface_get_height(d->image) : 1;
draw_image(ctx, geometry.x, geometry.y, ratio, d->image); draw_surface(ctx, geometry.x, geometry.y, ratio, d->image);
} }
} }
@ -79,9 +79,58 @@ static void
imagebox_destructor(widget_t *w) imagebox_destructor(widget_t *w)
{ {
imagebox_data_t *d = w->data; imagebox_data_t *d = w->data;
if(d->image)
cairo_surface_destroy(d->image);
p_delete(&d); p_delete(&d);
} }
static int
imagebox_set_image(lua_State *L, widget_t *widget, int idx)
{
imagebox_data_t *d = widget->data;
if(lua_isnil(L, idx))
{
if(d->image)
cairo_surface_destroy(d->image);
d->image = NULL;
} else {
bool is_surface = false;
/* This inlines luaL_checkudata() but skips luaL_typerror() */
if(lua_getmetatable(L, idx))
{
lua_getfield(L, LUA_REGISTRYINDEX, OOCAIRO_MT_NAME_SURFACE);
if(lua_rawequal(L, -1, -2))
is_surface = true;
lua_pop(L, 2);
}
if(is_surface)
{
/* Ugly lua-OOCairo magic. */
cairo_surface_t **cairo_surface = (cairo_surface_t **)luaL_checkudata(L, idx, OOCAIRO_MT_NAME_SURFACE);
if(d->image)
cairo_surface_destroy(d->image);
d->image = draw_dup_image_surface(*cairo_surface);
} else {
luaA_checkudata(L, idx, &image_class);
image_t *image = (void *) lua_topointer(L, idx);
cairo_surface_t *surface = cairo_image_surface_create_for_data(
image_getdata(image), CAIRO_FORMAT_ARGB32,
image_getwidth(image), image_getheight(image),
image_getwidth(image) * 4);
/* Cairo doesn't copy the data we give to it so we have to make a
* copy of the data. This is the lazy way to do that. */
d->image = draw_dup_image_surface(surface);
cairo_surface_destroy(surface);
}
}
widget_invalidate_bywidget(widget);
return 0;
}
/** Imagebox widget. /** Imagebox widget.
* \param L The Lua VM state. * \param L The Lua VM state.
* \param prop The key that is being indexed. * \param prop The key that is being indexed.
@ -98,7 +147,7 @@ luaA_imagebox_index(lua_State *L, const char *prop)
imagebox_data_t *d = widget->data; imagebox_data_t *d = widget->data;
if(a_strcmp(prop, "image") == 0) if(a_strcmp(prop, "image") == 0)
luaA_object_push_item(L, 1, d->image); oocairo_surface_push(L, d->image);
else if(a_strcmp(prop, "bg") == 0) else if(a_strcmp(prop, "bg") == 0)
luaA_pushcolor(L, &d->bg); luaA_pushcolor(L, &d->bg);
else if(a_strcmp(prop, "resize") == 0) else if(a_strcmp(prop, "resize") == 0)
@ -121,11 +170,7 @@ luaA_imagebox_newindex(lua_State *L, const char *prop)
imagebox_data_t *d = widget->data; imagebox_data_t *d = widget->data;
if(a_strcmp(prop, "image") == 0) if(a_strcmp(prop, "image") == 0)
{ imagebox_set_image(L, widget, 3);
luaA_checkudataornil(L, -1, &image_class);
luaA_object_unref_item(L, 1, d->image);
d->image = luaA_object_ref_item(L, 1, 3);
}
else if(a_strcmp(prop, "bg") == 0) else if(a_strcmp(prop, "bg") == 0)
{ {
const char *buf; const char *buf;