spmenu/libs/icon.c

141 lines
4.3 KiB
C
Raw Normal View History

2023-07-03 18:48:38 +02:00
/* See LICENSE file for copyright and license details. */
#if USEIMAGE
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);
2023-07-16 02:57:38 +02:00
char *xdg_cache, *home = NULL, *cachesize, *buf = NULL;
struct passwd *user_id = NULL;
if (hideimage) {
return x;
}
2023-07-03 18:48:38 +02:00
if (item->image) {
if (generatecache) {
if(!(xdg_cache = getenv("XDG_CACHE_HOME"))) {
2023-07-16 02:57:38 +02:00
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;
}
}
2023-07-16 02:57:38 +02:00
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);
2023-07-16 02:57:38 +02:00
char md5sum[digest_len*2+1];
2023-07-03 18:48:38 +02:00
for (i = 0; i < digest_len; ++i)
2023-07-16 02:57:38 +02:00
sprintf(&md5sum[i*2], "%02x", (unsigned int)digest[i]);
2023-07-16 02:57:38 +02:00
if (!cachedir || !strcmp(cachedir, "default")) { // "default" here is from the config file
if (xdg_cache || !strcmp(cachedir, "xdg"))
2023-07-16 02:57:38 +02:00
slen = snprintf(NULL, 0, "%s/thumbnails/%s/%s.png", xdg_cache, cachesize, md5sum)+1;
else
2023-07-16 02:57:38 +02:00
slen = snprintf(NULL, 0, "%s/.cache/thumbnails/%s/%s.png", home, cachesize, md5sum)+1;
} else {
2023-07-16 02:57:38 +02:00
slen = snprintf(NULL, 0, "%s/%s/%s.png", cachedir, cachesize, md5sum)+1;
}
if(!(buf = malloc(slen))) {
return x;
}
2023-07-16 02:57:38 +02:00
if (!cachedir || !strcmp(cachedir, "default")) { // "default" here is from the config file
if (xdg_cache)
2023-07-16 02:57:38 +02:00
sprintf(buf, "%s/thumbnails/%s/%s.png", xdg_cache, cachesize, md5sum);
else
2023-07-16 02:57:38 +02:00
sprintf(buf, "%s/.cache/thumbnails/%s/%s.png", home, cachesize, md5sum);
} else {
2023-07-16 02:57:38 +02:00
sprintf(buf, "%s/%s/%s.png", cachedir, cachesize, md5sum);
}
2023-07-16 02:57:38 +02:00
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();
}
2023-07-03 18:48:38 +02:00
}
2023-07-16 02:57:38 +02:00
/* 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;
}
2023-07-16 02:57:38 +02:00
// load regular image, because an image was not loaded from cache
if (!image || !generatecache) {
image = imlib_load_image(item->image);
if (!image) {
return x;
}
2023-07-03 18:48:38 +02:00
imlib_context_set_image(image);
2023-07-03 18:48:38 +02:00
icw = imlib_image_get_width();
ich = imlib_image_get_height();
2023-07-03 18:48:38 +02:00
if (!image) {
return x;
}
2023-07-03 18:48:38 +02:00
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);
2023-07-03 18:48:38 +02:00
2023-07-16 02:57:38 +02:00
// get new width/height
icw = imlib_image_get_width();
ich = imlib_image_get_height();
2023-07-03 18:48:38 +02:00
}
imlib_image_set_format("png");
2023-07-16 02:57:38 +02:00
// create directory and write cache file
if (buf && generatecache && image) {
createifnexist_rec(buf);
imlib_save_image(buf);
free(buf);
}
2023-07-16 02:57:38 +02:00
// draw the image
2023-07-03 18:48:38 +02:00
draw_set_img(draw, imlib_image_get_data(), w, h);
draw_img(draw, x, y);
x += w;
}
return x;
}
#endif