spmenu/libs/icon.c
2023-08-07 06:37:38 +02:00

143 lines
4.4 KiB
C

/* See LICENSE file for copyright and license details. */
#if IMAGE
static int draw_icon(struct item *item, int x, int y, int w, int h);
int draw_icon(struct item *item, int x, int y, int w, int h) {
int ich = 0;
int icw = 0;
// cache
int slen = 0, i;
unsigned int digest_len = EVP_MD_size(EVP_md5());
unsigned char *digest = (unsigned char *)OPENSSL_malloc(digest_len);
char *xdg_cache, *home = NULL, *cachesize, *buf = NULL;
struct passwd *user_id = NULL;
if (hideimage) {
return x;
}
if (item->image) {
if (generatecache) {
if(!(xdg_cache = getenv("XDG_CACHE_HOME"))) {
if(!(home = getenv("HOME")) && (user_id = getpwuid(getuid()))) {
home = user_id->pw_dir;
}
if(!home) { // no home directory.. somehow
fprintf(stderr, "spmenu: could not find home directory");
return x;
}
}
cachesize = "icon";
slen = snprintf(NULL, 0, "file://%s", item->image)+1;
if(!(buf = malloc(slen))) {
return x;
}
sprintf(buf, "file://%s", item->image);
EVP_MD_CTX *mdcontext = EVP_MD_CTX_new();
EVP_DigestInit_ex(mdcontext, EVP_md5(), NULL);
EVP_DigestUpdate(mdcontext, buf, slen);
EVP_DigestFinal_ex(mdcontext, digest, &digest_len);
EVP_MD_CTX_free(mdcontext);
free(buf);
char md5sum[digest_len*2+1];
for (i = 0; i < digest_len; ++i)
sprintf(&md5sum[i*2], "%02x", (unsigned int)digest[i]);
if (!cachedir || !strcmp(cachedir, "default")) { // "default" here is from the config file
if (xdg_cache || !strcmp(cachedir, "xdg"))
slen = snprintf(NULL, 0, "%s/thumbnails/%s/%s.png", xdg_cache, cachesize, md5sum)+1;
else
slen = snprintf(NULL, 0, "%s/.cache/thumbnails/%s/%s.png", home, cachesize, md5sum)+1;
} else {
slen = snprintf(NULL, 0, "%s/%s/%s.png", cachedir, cachesize, md5sum)+1;
}
if(!(buf = malloc(slen))) {
return x;
}
if (!cachedir || !strcmp(cachedir, "default")) { // "default" here is from the config file
if (xdg_cache)
sprintf(buf, "%s/thumbnails/%s/%s.png", xdg_cache, cachesize, md5sum);
else
sprintf(buf, "%s/.cache/thumbnails/%s/%s.png", home, cachesize, md5sum);
} else {
sprintf(buf, "%s/%s/%s.png", cachedir, cachesize, md5sum);
}
image = imlib_load_image(buf); // attempt to load image from path
if (image) {
imlib_context_set_image(image);
icw = imlib_image_get_width();
ich = imlib_image_get_height();
}
}
/* If passed w or h has changed, cache is invalid and has to be regenerated. This prevents flickering.
* It could change if sp.bh has changed in some way, such as font size being changed after cache has
* been generated.
*/
if (ich != h || icw != w) {
image = NULL;
}
// load regular image, because an image was not loaded from cache
if (!image || !generatecache) {
image = imlib_load_image(item->image);
if (!image) {
return x;
}
imlib_context_set_image(image);
icw = imlib_image_get_width();
ich = imlib_image_get_height();
if (!image) {
return x;
}
image = imlib_create_cropped_scaled_image(0, 0, icw, ich, w, h); // w = h = bh - lrpad / 2
imlib_free_image();
imlib_context_set_image(image);
// get new width/height
icw = imlib_image_get_width();
ich = imlib_image_get_height();
}
imlib_image_set_format("png");
// create directory and write cache file
if (buf && generatecache && image) {
createifnexist_rec(buf);
imlib_save_image(buf);
free(buf);
}
// draw the image
draw_set_img(draw, imlib_image_get_data(), w, h);
draw_img(draw, x, y);
x += w;
}
return x;
}
#endif