2010-08-27 12:20:26 +00:00
|
|
|
/* -*- mode: c++; c-basic-offset: 4; tab-width: 4; indent-tabs-mode: t -*-
|
|
|
|
* vim: ts=4 sw=4 noet ai cindent syntax=cpp
|
2009-07-28 21:44:22 +00:00
|
|
|
*
|
|
|
|
* Conky, a system monitor, based on torsmo
|
2009-05-08 20:42:34 +00:00
|
|
|
*
|
|
|
|
* Please see COPYING for details
|
|
|
|
*
|
2012-05-03 23:34:44 +00:00
|
|
|
* Copyright (c) 2005-2012 Brenden Matthews, et. al.
|
2009-05-08 20:42:34 +00:00
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
2009-06-15 03:05:11 +00:00
|
|
|
#include "conky.h"
|
2009-05-08 20:42:34 +00:00
|
|
|
#include "logging.h"
|
2009-11-28 21:45:01 +00:00
|
|
|
#include "text_object.h"
|
|
|
|
#include "imlib2.h"
|
2009-05-08 20:42:34 +00:00
|
|
|
|
|
|
|
#include <Imlib2.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2009-05-24 19:47:44 +00:00
|
|
|
#include <limits.h>
|
2009-06-15 03:05:11 +00:00
|
|
|
#include <string.h>
|
2009-06-03 19:24:53 +00:00
|
|
|
#include <time.h>
|
2009-05-08 20:42:34 +00:00
|
|
|
|
2010-08-29 12:31:22 +00:00
|
|
|
#include "x11.h"
|
|
|
|
|
2009-05-08 20:42:34 +00:00
|
|
|
struct image_list_s {
|
2009-06-15 03:05:11 +00:00
|
|
|
char name[1024];
|
2009-05-08 20:42:34 +00:00
|
|
|
Imlib_Image image;
|
2009-05-17 05:47:02 +00:00
|
|
|
int x, y, w, h;
|
|
|
|
int wh_set;
|
2009-06-03 19:24:53 +00:00
|
|
|
char no_cache;
|
|
|
|
int flush_interval;
|
2009-05-08 20:42:34 +00:00
|
|
|
struct image_list_s *next;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct image_list_s *image_list_start, *image_list_end;
|
|
|
|
|
|
|
|
/* areas to update */
|
|
|
|
Imlib_Updates updates, current_update;
|
|
|
|
/* our virtual framebuffer image we draw into */
|
|
|
|
Imlib_Image buffer, image;
|
|
|
|
|
2010-08-29 12:31:22 +00:00
|
|
|
namespace {
|
|
|
|
Imlib_Context context;
|
2009-05-22 17:23:05 +00:00
|
|
|
|
2010-08-29 12:31:22 +00:00
|
|
|
conky::range_config_setting<unsigned int> imlib_cache_flush_interval(
|
|
|
|
"imlib_cache_flush_interval", 0,
|
|
|
|
std::numeric_limits<unsigned int>::max(), 0, true
|
|
|
|
);
|
2009-06-03 19:24:53 +00:00
|
|
|
|
2010-12-23 17:25:31 +00:00
|
|
|
unsigned int cimlib_cache_flush_last = 0;
|
2010-08-29 12:31:22 +00:00
|
|
|
}
|
2009-05-08 20:42:34 +00:00
|
|
|
|
2010-08-29 12:31:22 +00:00
|
|
|
void imlib_cache_size_setting::lua_setter(lua::state &l, bool init)
|
2009-05-08 20:42:34 +00:00
|
|
|
{
|
2010-08-29 12:31:22 +00:00
|
|
|
lua::stack_sentry s(l, -2);
|
|
|
|
|
|
|
|
Base::lua_setter(l, init);
|
|
|
|
|
|
|
|
if(init && out_to_x.get(l)) {
|
|
|
|
image_list_start = image_list_end = NULL;
|
|
|
|
context = imlib_context_new();
|
|
|
|
imlib_context_push(context);
|
|
|
|
imlib_set_cache_size(do_convert(l, -1).first);
|
|
|
|
/* set the maximum number of colors to allocate for 8bpp and less to 256 */
|
|
|
|
imlib_set_color_usage(256);
|
|
|
|
/* dither for depths < 24bpp */
|
|
|
|
imlib_context_set_dither(1);
|
|
|
|
/* set the display , visual, colormap and drawable we are using */
|
|
|
|
imlib_context_set_display(display);
|
|
|
|
imlib_context_set_visual(window.visual);
|
|
|
|
imlib_context_set_colormap(window.colourmap);
|
|
|
|
imlib_context_set_drawable(window.drawable);
|
|
|
|
}
|
|
|
|
|
|
|
|
++s;
|
2009-05-08 20:42:34 +00:00
|
|
|
}
|
|
|
|
|
2010-08-29 12:31:22 +00:00
|
|
|
void imlib_cache_size_setting::cleanup(lua::state &l)
|
2009-06-03 19:24:53 +00:00
|
|
|
{
|
2010-08-29 12:31:22 +00:00
|
|
|
lua::stack_sentry s(l, -1);
|
|
|
|
|
|
|
|
if(out_to_x.get(l)) {
|
|
|
|
cimlib_cleanup();
|
|
|
|
imlib_context_disconnect_display();
|
|
|
|
imlib_context_pop();
|
|
|
|
imlib_context_free(context);
|
2009-06-14 22:12:12 +00:00
|
|
|
}
|
2009-06-03 19:24:53 +00:00
|
|
|
}
|
|
|
|
|
2009-05-08 20:42:34 +00:00
|
|
|
void cimlib_cleanup(void)
|
|
|
|
{
|
|
|
|
struct image_list_s *cur = image_list_start, *last = NULL;
|
|
|
|
while (cur) {
|
|
|
|
last = cur;
|
|
|
|
cur = last->next;
|
|
|
|
free(last);
|
|
|
|
}
|
|
|
|
image_list_start = image_list_end = NULL;
|
|
|
|
}
|
|
|
|
|
2009-05-17 05:47:02 +00:00
|
|
|
void cimlib_add_image(const char *args)
|
2009-05-08 20:42:34 +00:00
|
|
|
{
|
|
|
|
struct image_list_s *cur = NULL;
|
2010-01-10 21:04:15 +00:00
|
|
|
const char *tmp;
|
2009-05-17 05:47:02 +00:00
|
|
|
|
2010-01-10 21:04:15 +00:00
|
|
|
cur = (struct image_list_s *)malloc(sizeof(struct image_list_s));
|
2009-05-08 20:42:34 +00:00
|
|
|
memset(cur, 0, sizeof(struct image_list_s));
|
2009-05-17 05:47:02 +00:00
|
|
|
|
2009-06-15 03:05:11 +00:00
|
|
|
if (!sscanf(args, "%1023s", cur->name)) {
|
2009-08-03 18:39:44 +00:00
|
|
|
NORM_ERR("Invalid args for $image. Format is: '<path to image> (-p"
|
|
|
|
"x,y) (-s WxH) (-n) (-f interval)' (got '%s')", args);
|
2009-06-15 03:05:11 +00:00
|
|
|
free(cur);
|
|
|
|
return;
|
2009-05-17 05:47:02 +00:00
|
|
|
}
|
2010-03-04 10:42:38 +00:00
|
|
|
strncpy(cur->name, to_real_path(cur->name).c_str(), 1024);
|
|
|
|
cur->name[1023] = 0;
|
|
|
|
//
|
2009-05-17 05:47:02 +00:00
|
|
|
// now we check for optional args
|
|
|
|
tmp = strstr(args, "-p ");
|
|
|
|
if (tmp) {
|
|
|
|
tmp += 3;
|
|
|
|
sscanf(tmp, "%i,%i", &cur->x, &cur->y);
|
|
|
|
}
|
|
|
|
tmp = strstr(args, "-s ");
|
|
|
|
if (tmp) {
|
|
|
|
tmp += 3;
|
|
|
|
if (sscanf(tmp, "%ix%i", &cur->w, &cur->h)) {
|
|
|
|
cur->wh_set = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-03 19:24:53 +00:00
|
|
|
tmp = strstr(args, "-n");
|
|
|
|
if (tmp) {
|
|
|
|
cur->no_cache = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp = strstr(args, "-f ");
|
|
|
|
if (tmp) {
|
|
|
|
tmp += 3;
|
|
|
|
if (sscanf(tmp, "%d", &cur->flush_interval)) {
|
|
|
|
cur->no_cache = 0;
|
|
|
|
}
|
|
|
|
}
|
2009-06-14 22:12:12 +00:00
|
|
|
if (cur->flush_interval < 0) {
|
2009-08-01 18:45:43 +00:00
|
|
|
NORM_ERR("Imlib2: flush interval should be >= 0");
|
2009-06-14 22:12:12 +00:00
|
|
|
cur->flush_interval = 0;
|
|
|
|
}
|
2009-06-03 19:24:53 +00:00
|
|
|
|
2009-05-08 20:42:34 +00:00
|
|
|
if (image_list_end) {
|
|
|
|
image_list_end->next = cur;
|
|
|
|
image_list_end = cur;
|
|
|
|
} else {
|
|
|
|
image_list_start = image_list_end = cur;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-03 18:39:44 +00:00
|
|
|
static void cimlib_draw_image(struct image_list_s *cur, int *clip_x, int
|
|
|
|
*clip_y, int *clip_x2, int *clip_y2)
|
2009-05-08 20:42:34 +00:00
|
|
|
{
|
2009-06-19 23:14:57 +00:00
|
|
|
int w, h;
|
|
|
|
time_t now = time(NULL);
|
2009-09-09 23:59:08 +00:00
|
|
|
static int rep = 0;
|
2009-06-19 23:14:57 +00:00
|
|
|
|
2012-04-11 16:06:47 +00:00
|
|
|
if (imlib_context_get_drawable() != window.drawable) {
|
|
|
|
imlib_context_set_drawable(window.drawable);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-08 20:42:34 +00:00
|
|
|
image = imlib_load_image(cur->name);
|
2009-06-19 23:14:57 +00:00
|
|
|
if (!image) {
|
2009-09-09 23:59:08 +00:00
|
|
|
if (!rep)
|
|
|
|
NORM_ERR("Unable to load image '%s'", cur->name);
|
|
|
|
rep = 1;
|
2009-06-19 23:14:57 +00:00
|
|
|
return;
|
|
|
|
}
|
2009-09-09 23:59:08 +00:00
|
|
|
rep = 0; /* reset so disappearing images are reported */
|
2009-06-19 23:14:57 +00:00
|
|
|
|
|
|
|
DBGP("Drawing image '%s' at (%i,%i) scaled to %ix%i, "
|
|
|
|
"caching interval set to %i (with -n opt %i)",
|
|
|
|
cur->name, cur->x, cur->y, cur->w, cur->h,
|
|
|
|
cur->flush_interval, cur->no_cache);
|
|
|
|
|
|
|
|
imlib_context_set_image(image);
|
|
|
|
/* turn alpha channel on */
|
|
|
|
imlib_image_set_has_alpha(1);
|
|
|
|
w = imlib_image_get_width();
|
|
|
|
h = imlib_image_get_height();
|
|
|
|
if (!cur->wh_set) {
|
|
|
|
cur->w = w;
|
|
|
|
cur->h = h;
|
|
|
|
}
|
|
|
|
imlib_context_set_image(buffer);
|
|
|
|
imlib_blend_image_onto_image(image, 1, 0, 0, w, h,
|
|
|
|
cur->x, cur->y, cur->w, cur->h);
|
|
|
|
imlib_context_set_image(image);
|
|
|
|
if (cur->no_cache || (cur->flush_interval &&
|
|
|
|
now % cur->flush_interval == 0)) {
|
|
|
|
imlib_free_image_and_decache();
|
|
|
|
} else {
|
|
|
|
imlib_free_image();
|
2009-05-08 20:42:34 +00:00
|
|
|
}
|
2009-06-19 23:14:57 +00:00
|
|
|
if (cur->x < *clip_x) *clip_x = cur->x;
|
|
|
|
if (cur->y < *clip_y) *clip_y = cur->y;
|
|
|
|
if (cur->x + cur->w > *clip_x2) *clip_x2 = cur->x + cur->w;
|
|
|
|
if (cur->y + cur->h > *clip_y2) *clip_y2 = cur->y + cur->h;
|
2009-05-08 20:42:34 +00:00
|
|
|
}
|
|
|
|
|
2009-05-24 22:52:48 +00:00
|
|
|
static void cimlib_draw_all(int *clip_x, int *clip_y, int *clip_x2, int *clip_y2)
|
2009-05-08 20:42:34 +00:00
|
|
|
{
|
|
|
|
struct image_list_s *cur = image_list_start;
|
|
|
|
while (cur) {
|
2009-05-24 22:52:48 +00:00
|
|
|
cimlib_draw_image(cur, clip_x, clip_y, clip_x2, clip_y2);
|
2009-05-08 20:42:34 +00:00
|
|
|
cur = cur->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-18 05:11:22 +00:00
|
|
|
void cimlib_render(int x, int y, int width, int height)
|
2009-05-08 20:42:34 +00:00
|
|
|
{
|
2009-05-24 19:47:44 +00:00
|
|
|
int clip_x = INT_MAX, clip_y = INT_MAX;
|
2009-05-24 22:52:48 +00:00
|
|
|
int clip_x2 = 0, clip_y2 = 0;
|
2009-06-03 19:24:53 +00:00
|
|
|
time_t now;
|
2009-05-24 19:47:44 +00:00
|
|
|
|
2009-05-08 20:42:34 +00:00
|
|
|
if (!image_list_start) return; /* are we actually drawing anything? */
|
2009-06-03 19:24:53 +00:00
|
|
|
|
|
|
|
/* cheque if it's time to flush our cache */
|
|
|
|
now = time(NULL);
|
2010-08-29 12:31:22 +00:00
|
|
|
if (imlib_cache_flush_interval.get(*state) &&
|
|
|
|
now - imlib_cache_flush_interval.get(*state) > cimlib_cache_flush_last) {
|
2009-06-03 19:31:16 +00:00
|
|
|
int size = imlib_get_cache_size();
|
|
|
|
imlib_set_cache_size(0);
|
|
|
|
imlib_set_cache_size(size);
|
2009-06-03 19:24:53 +00:00
|
|
|
cimlib_cache_flush_last = now;
|
|
|
|
DBGP("Flushing Imlib2 cache (%li)\n", now);
|
|
|
|
}
|
|
|
|
|
2009-05-08 20:42:34 +00:00
|
|
|
/* take all the little rectangles to redraw and merge them into
|
|
|
|
* something sane for rendering */
|
2009-05-17 05:47:02 +00:00
|
|
|
buffer = imlib_create_image(width, height);
|
2009-05-18 05:11:22 +00:00
|
|
|
/* clear our buffer */
|
|
|
|
imlib_context_set_image(buffer);
|
|
|
|
imlib_image_clear();
|
2009-05-17 05:47:02 +00:00
|
|
|
/* we can blend stuff now */
|
|
|
|
imlib_context_set_blend(1);
|
2009-05-24 22:52:48 +00:00
|
|
|
/* turn alpha channel on */
|
|
|
|
imlib_image_set_has_alpha(1);
|
2009-05-17 05:47:02 +00:00
|
|
|
|
2009-05-24 22:52:48 +00:00
|
|
|
cimlib_draw_all(&clip_x, &clip_y, &clip_x2, &clip_y2);
|
2009-05-17 05:47:02 +00:00
|
|
|
|
|
|
|
/* set the buffer image as our current image */
|
|
|
|
imlib_context_set_image(buffer);
|
2009-05-24 19:47:44 +00:00
|
|
|
|
|
|
|
/* setup our clip rect */
|
|
|
|
if (clip_x == INT_MAX) clip_x = 0;
|
|
|
|
if (clip_y == INT_MAX) clip_y = 0;
|
|
|
|
|
2009-05-17 05:47:02 +00:00
|
|
|
/* render the image at 0, 0 */
|
2009-05-24 22:52:48 +00:00
|
|
|
imlib_render_image_part_on_drawable_at_size(clip_x, clip_y, clip_x2 - clip_x,
|
|
|
|
clip_y2 - clip_y, x + clip_x, y + clip_y, clip_x2 - clip_x,
|
|
|
|
clip_y2 - clip_y);
|
2009-05-17 05:47:02 +00:00
|
|
|
/* don't need that temporary buffer image anymore */
|
|
|
|
imlib_free_image();
|
2009-05-08 20:42:34 +00:00
|
|
|
}
|
|
|
|
|
2009-11-28 21:45:01 +00:00
|
|
|
void print_image_callback(struct text_object *obj, char *p, int p_max_size)
|
|
|
|
{
|
|
|
|
(void)p;
|
|
|
|
(void)p_max_size;
|
|
|
|
|
|
|
|
cimlib_add_image(obj->data.s);
|
|
|
|
}
|