Mercurial > MadButterfly
view src/img_ldr_imlib2.c @ 1504:d5b677831671
Make grab_bg() stackable
author | Thinker K.F. Li <thinker@codemud.net> |
---|---|
date | Mon, 09 May 2011 00:27:10 +0800 |
parents | 1c64a9cec2f2 |
children |
line wrap: on
line source
// -*- indent-tabs-mode: t; tab-width: 8; c-basic-offset: 4; -*- // vim: sw=4:ts=8:sts=4 #include <stdio.h> #include <stdint.h> #include <string.h> #include <Imlib2.h> #include "mb_graph_engine.h" #include "mb_tools.h" #include "mb_paint.h" #include "mb_img_ldr.h" /*! \brief Simple image loader. * */ struct _simple_mb_img_ldr { mb_img_ldr_t ldr; const char *repo; /*!< \brief The directory of repository. */ }; typedef struct _simple_mb_img_ldr simple_mb_img_ldr_t; struct _simple_mb_img_data { mb_img_data_t img; }; typedef struct _simple_mb_img_data simple_mb_img_data_t; static void simple_mb_img_ldr_img_free(mb_img_data_t *img); static mb_img_data_t *simple_mb_img_ldr_load(mb_img_ldr_t *ldr, const char *img_id) { simple_mb_img_ldr_t *sldr = (simple_mb_img_ldr_t *)ldr; simple_mb_img_data_t *img; Imlib_Image img_hdl; int w, h; int i, j, pos; uint32_t alpha; uint32_t value; uint32_t *data; uint32_t *premultiple_data; char *fname; int sz; sz = strlen(sldr->repo); sz += strlen(img_id); fname = (char *)malloc(sz + 2); if (img_id[0] != '/') strcpy(fname, sldr->repo); else fname[0] = 0; strcat(fname, img_id); img_hdl = imlib_load_image(fname); if(!img_hdl) return NULL; imlib_context_set_image(img_hdl); w = imlib_image_get_width(); h = imlib_image_get_height(); data = (uint32_t *)imlib_image_get_data_for_reading_only(); img = O_ALLOC(simple_mb_img_data_t); if(img == NULL) { imlib_free_image(); return NULL; } premultiple_data = (uint32_t *)malloc(4 * w * h); if(premultiple_data == NULL) { free(img); imlib_free_image(); return NULL; } memcpy(premultiple_data, data, 4 * w * h); /* Pre-multiply * * Our reference model is Cairo. In Cairo model, image is * pre-multiplied with alpha, but imlib2 is not. So, we do it * here. */ pos = 0; for(i = 0; i < h; i++) { for(j = 0; j < w; j++) { value = data[pos]; alpha = value >> 24; if(alpha != 0xff) { if(alpha == 0) value = 0; else { value = (value & 0xff000000) | (((value & 0xff0000) * alpha / 0xff) & 0xff0000) | (((value & 0xff00) * alpha / 0xff) & 0xff00) | ((value & 0xff) * alpha / 0xff); } } premultiple_data[pos++] = value; } } img->img.content = premultiple_data; img->img.w = w; img->img.h = h; img->img.stride = w * 4; img->img.fmt = MB_IFMT_ARGB32; img->img.free = simple_mb_img_ldr_img_free; imlib_free_image(); return (mb_img_data_t *)img; } static void simple_mb_img_ldr_img_free(mb_img_data_t *img) { simple_mb_img_data_t *simg = (simple_mb_img_data_t *)img; free(simg->img.content); free(img); } static void simple_mb_img_ldr_free(mb_img_ldr_t *ldr) { simple_mb_img_ldr_t *defldr = (simple_mb_img_ldr_t *)ldr; free((void *)defldr->repo); } mb_img_ldr_t *simple_mb_img_ldr_new(const char *img_repository) { simple_mb_img_ldr_t *ldr; int sz; if(img_repository == NULL) return NULL; ldr = O_ALLOC(simple_mb_img_ldr_t); if(ldr == NULL) return NULL; /* * Copy and formalize path of image repository. */ sz = strlen(img_repository); ldr->repo = (const char *)malloc(sz + 2); if(ldr->repo == NULL) { free(ldr); return NULL; } strcpy((char *)ldr->repo, img_repository); if(img_repository[sz - 1] != '/' && strlen(img_repository) != 0) { ((char *)ldr->repo)[sz] = '/'; ((char *)ldr->repo)[sz + 1] = 0; } ldr->ldr.load = simple_mb_img_ldr_load; ldr->ldr.free = simple_mb_img_ldr_free; return (mb_img_ldr_t *)ldr; }