2 * image.c - image object
4 * Copyright © 2008 Julien Danjou <julien@danjou.info>
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License along
17 * with this program; if not, write to the Free Software Foundation, Inc.,
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
24 extern awesome_t globalconf
;
26 DO_LUA_NEW(extern, image_t
, image
, "image", image_ref
)
27 DO_LUA_GC(image_t
, image
, "image", image_unref
)
28 DO_LUA_EQ(image_t
, image
, "image")
31 image_imlib_load_strerror(Imlib_Load_Error e
)
35 case IMLIB_LOAD_ERROR_FILE_DOES_NOT_EXIST
:
36 return "no such file or directory";
37 case IMLIB_LOAD_ERROR_FILE_IS_DIRECTORY
:
38 return "file is a directory";
39 case IMLIB_LOAD_ERROR_PERMISSION_DENIED_TO_READ
:
40 return "read permission denied";
41 case IMLIB_LOAD_ERROR_NO_LOADER_FOR_FILE_FORMAT
:
42 return "no loader for file format";
43 case IMLIB_LOAD_ERROR_PATH_TOO_LONG
:
44 return "path too long";
45 case IMLIB_LOAD_ERROR_PATH_COMPONENT_NON_EXISTANT
:
46 return "path component non existant";
47 case IMLIB_LOAD_ERROR_PATH_COMPONENT_NOT_DIRECTORY
:
48 return "path compoment not a directory";
49 case IMLIB_LOAD_ERROR_PATH_POINTS_OUTSIDE_ADDRESS_SPACE
:
50 return "path points oustide address space";
51 case IMLIB_LOAD_ERROR_TOO_MANY_SYMBOLIC_LINKS
:
52 return "too many symbolic links";
53 case IMLIB_LOAD_ERROR_OUT_OF_MEMORY
:
54 return "out of memory";
55 case IMLIB_LOAD_ERROR_OUT_OF_FILE_DESCRIPTORS
:
56 return "out of file descriptors";
57 case IMLIB_LOAD_ERROR_PERMISSION_DENIED_TO_WRITE
:
58 return "write permission denied";
59 case IMLIB_LOAD_ERROR_OUT_OF_DISK_SPACE
:
60 return "out of disk space";
61 case IMLIB_LOAD_ERROR_UNKNOWN
:
62 return "unknown error, that's really bad";
63 case IMLIB_LOAD_ERROR_NONE
:
64 return "no error, oops";
67 return "unknown error";
70 /** Recompute the ARGB32 data from an image.
71 * \param image The image.
75 image_compute(image_t
*image
)
82 imlib_context_set_image(image
->image
);
84 data
= imlib_image_get_data_for_reading_only();
86 image
->width
= imlib_image_get_width();
87 image
->height
= imlib_image_get_height();
89 size
= image
->width
* image
->height
;
91 p_delete(&image
->data
);
92 image
->data
= dataimg
= p_new(uint8_t, size
* 4);
94 for(i
= 0; i
< size
; i
++, dataimg
+= 4)
96 dataimg
[3] = (data
[i
] >> 24) & 0xff; /* A */
97 /* cairo wants pre-multiplied alpha */
98 alpha
= dataimg
[3] / 255.0;
99 dataimg
[2] = ((data
[i
] >> 16) & 0xff) * alpha
; /* R */
100 dataimg
[1] = ((data
[i
] >> 8) & 0xff) * alpha
; /* G */
101 dataimg
[0] = (data
[i
] & 0xff) * alpha
; /* B */
105 /** Create a new image from ARGB32 data.
106 * \param width The image width.
107 * \param height The image height.
108 * \param data The image data.
109 * \return A brand new image.
112 image_new_from_argb32(int width
, int height
, uint32_t *data
)
115 image_t
*image
= NULL
;
117 if((imimage
= imlib_create_image_using_data(width
, height
, data
)))
119 image
= p_new(image_t
, 1);
120 image
->image
= imimage
;
121 image_compute(image
);
127 /** Load an image from filename.
128 * \param filename The image file to load.
129 * \return A new image.
132 image_new_from_file(const char *filename
)
135 Imlib_Load_Error e
= IMLIB_LOAD_ERROR_NONE
;
141 if(!(imimage
= imlib_load_image_with_error_return(filename
, &e
)))
143 warn("cannot load image %s: %s", filename
, image_imlib_load_strerror(e
));
147 image
= p_new(image_t
, 1);
148 image
->image
= imimage
;
150 image_compute(image
);
155 /** Create a new image object.
156 * \param L The Lua stack.
157 * \return The number of elements pushed on stack.
159 * \lparam The image path, or nil to create an empty image.
160 * \lparam The image width if nil was set as first arg.
161 * \lparam The image height if nil was set as first arg.
162 * \lreturn An image object.
165 luaA_image_new(lua_State
*L
)
167 const char *filename
;
169 if((filename
= lua_tostring(L
, 2)))
172 if((image
= image_new_from_file(filename
)))
173 return luaA_image_userdata_new(L
, image
);
175 else if(lua_isnil(L
, 2))
177 int width
= luaL_checknumber(L
, 3);
178 int height
= luaL_checknumber(L
, 4);
180 if(width
<= 0 || height
<= 0)
181 luaL_error(L
, "request image has invalid size");
183 Imlib_Image imimage
= imlib_create_image(width
, height
);
184 image_t
*image
= p_new(image_t
, 1);
185 image
->image
= imimage
;
186 image_compute(image
);
187 return luaA_image_userdata_new(L
, image
);
193 /** Create a new image object from ARGB32 data.
194 * \param L The Lua stack.
195 * \return The number of elements pushed on stack.
197 * \lparam The image width.
198 * \lparam The image height.
199 * \lparam The image data as a string in ARGB32 format.
200 * \lreturn An image object.
203 luaA_image_argb32_new(lua_State
*L
)
207 unsigned int width
= luaL_checknumber(L
, 1);
208 unsigned int height
= luaL_checknumber(L
, 2);
209 const char *data
= luaL_checklstring(L
, 3, &len
);
211 if(width
* height
* 4 != len
)
212 luaL_error(L
, "string size does not match image size");
214 if((image
= image_new_from_argb32(width
, height
, (uint32_t *) data
)))
215 return luaA_image_userdata_new(L
, image
);
220 /** Rotate an image with specified angle radians and return a new image.
221 * \param L The Lua VM state.
222 * \return The number of elements pushed on stack.
225 * \lparam The angle in radians.
226 * \lreturn A rotated image.
229 luaA_image_rotate(lua_State
*L
)
231 image_t
**image
= luaA_checkudata(L
, 1, "image"), *new;
232 double angle
= luaL_checknumber(L
, 2);
234 new = p_new(image_t
, 1);
236 imlib_context_set_image((*image
)->image
);
237 new->image
= imlib_create_rotated_image(angle
);
241 return luaA_image_userdata_new(L
, new);
244 /** Crop an image to the given rectangle.
245 * \return The number of elements pushed on stack.
248 * \lparam The top left x coordinate of the rectangle.
249 * \lparam The top left y coordinate of the rectangle.
250 * \lparam The width of the rectangle.
251 * \lparam The height of the rectangle.
252 * \lreturn A cropped image.
255 luaA_image_crop(lua_State
*L
)
257 image_t
**image
= luaA_checkudata(L
, 1, "image"), *new;
258 int x
= luaL_checkint(L
, 2);
259 int y
= luaL_checkint(L
, 3);
260 int w
= luaL_checkint(L
, 4);
261 int h
= luaL_checkint(L
, 5);
263 new = p_new(image_t
, 1);
265 imlib_context_set_image((*image
)->image
);
266 new->image
= imlib_create_cropped_image(x
, y
, w
, h
);
270 return luaA_image_userdata_new(L
, new);
273 /** Crop the image to the given rectangle and scales it.
274 * \param L The Lua VM state.
275 * \return The number of elements pushed on stack.
278 * \lparam The top left x coordinate of the source rectangle.
279 * \lparam The top left y coordinate of the source rectangle.
280 * \lparam The width of the source rectangle.
281 * \lparam The height of the source rectangle.
282 * \lparam The width of the destination rectangle.
283 * \lparam The height of the destination rectangle.
284 * \lreturn A cropped image.
287 luaA_image_crop_and_scale(lua_State
*L
)
289 image_t
**image
= luaA_checkudata(L
, 1, "image"), *new;
290 int source_x
= luaL_checkint(L
, 2);
291 int source_y
= luaL_checkint(L
, 3);
292 int w
= luaL_checkint(L
, 4);
293 int h
= luaL_checkint(L
, 5);
294 int dest_w
= luaL_checkint(L
, 6);
295 int dest_h
= luaL_checkint(L
, 7);
297 new = p_new(image_t
, 1);
299 imlib_context_set_image((*image
)->image
);
300 new->image
= imlib_create_cropped_scaled_image(source_x
,
307 return luaA_image_userdata_new(L
, new);
311 * \param L The Lua VM state.
312 * \return The number of elements pushed on stack.
315 * \lfield width The image width.
316 * \lfield height The image height.
319 luaA_image_index(lua_State
*L
)
321 if(luaA_usemetatable(L
, 1, 2))
324 image_t
**image
= luaA_checkudata(L
, 1, "image");
326 const char *attr
= luaL_checklstring(L
, 2, &len
);
328 switch(a_tokenize(attr
, len
))
331 imlib_context_set_image((*image
)->image
);
332 lua_pushnumber(L
, imlib_image_get_width());
335 imlib_context_set_image((*image
)->image
);
336 lua_pushnumber(L
, imlib_image_get_height());
345 const struct luaL_reg awesome_image_methods
[] =
347 { "__call", luaA_image_new
},
348 { "argb32", luaA_image_argb32_new
},
351 const struct luaL_reg awesome_image_meta
[] =
353 { "__index", luaA_image_index
},
354 { "rotate", luaA_image_rotate
},
355 { "crop", luaA_image_crop
},
356 { "crop_and_scale", luaA_image_crop_and_scale
},
357 { "__gc", luaA_image_gc
},
358 { "__eq", luaA_image_eq
},
359 { "__tostring", luaA_image_tostring
},
363 // vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80