4 * Copyright (C) 2008 - 2010 The Geeqie Team
8 * This software is released under the GNU General Public License (GNU GPL).
9 * Please read the included file COPYING for more information.
10 * This software comes with no warranty of any kind, use at your own risk!
19 #include "pixbuf-renderer.h"
27 /* comment this out if not using this from within Geeqie
28 * defining GQ_BUILD does these things:
29 * - Sets the shift-click scroller pixbuf to a nice icon instead of a black box
35 #include "pixbuf_util.h"
39 EXIF_ORIENTATION_UNKNOWN = 0,
40 EXIF_ORIENTATION_TOP_LEFT = 1,
41 EXIF_ORIENTATION_TOP_RIGHT = 2,
42 EXIF_ORIENTATION_BOTTOM_RIGHT = 3,
43 EXIF_ORIENTATION_BOTTOM_LEFT = 4,
44 EXIF_ORIENTATION_LEFT_TOP = 5,
45 EXIF_ORIENTATION_RIGHT_TOP = 6,
46 EXIF_ORIENTATION_RIGHT_BOTTOM = 7,
47 EXIF_ORIENTATION_LEFT_BOTTOM = 8
48 } ExifOrientationType;
52 /* size to use when breaking up image pane for rendering */
53 #define PR_TILE_SIZE 128
55 /* default size of tile cache (mb) */
56 #define PR_CACHE_SIZE_DEFAULT 8
58 /* default min and max zoom */
59 #define PR_ZOOM_MIN -32.0
60 #define PR_ZOOM_MAX 32.0
62 /* distance to drag mouse to disable image flip */
63 #define PR_DRAG_SCROLL_THRESHHOLD 4
65 /* increase pan rate when holding down shift */
66 #define PR_PAN_SHIFT_MULTIPLIER 6
69 #define PR_SCROLLER_UPDATES_PER_SEC 30
70 #define PR_SCROLLER_DEAD_ZONE 6
72 /* alpha channel checkerboard background (same as gimp) */
73 #define PR_ALPHA_CHECK1 0x00999999
74 #define PR_ALPHA_CHECK2 0x00666666
75 #define PR_ALPHA_CHECK_SIZE 16
77 /* when scaling image to below this size, use nearest pixel for scaling
78 * (below about 4, the other scale types become slow generating their conversion tables)
80 #define PR_MIN_SCALE_SIZE 8
82 /* round A up/down to integer count of B */
83 #define ROUND_UP(A,B) ((gint)(((A)+(B)-1)/(B))*(B))
84 #define ROUND_DOWN(A,B) ((gint)(((A))/(B))*(B))
87 TILE_RENDER_NONE = 0, /* do nothing */
88 TILE_RENDER_AREA, /* render an area of the tile */
89 TILE_RENDER_ALL /* render the whole tile */
90 } ImageTileRenderType;
92 typedef struct _ImageTile ImageTile;
93 typedef struct _QueueData QueueData;
97 GdkPixmap *pixmap; /* off screen buffer */
98 GdkPixbuf *pixbuf; /* pixbuf area for zooming */
99 gint x; /* x offset into image */
100 gint y; /* y offset into image */
101 gint w; /* width that is visible (may be less if at edge of image) */
102 gint h; /* height '' */
106 /* render_todo: (explanation)
108 AREA render area of tile, usually only used when loading an image
109 note: will jump to an ALL if render_done is not ALL.
110 ALL render entire tile, if never done before w/ ALL, for expose events *only*
113 ImageTileRenderType render_todo; /* what to do (see above) */
114 ImageTileRenderType render_done; /* highest that has been done before on tile */
119 guint size; /* est. memory used by pixmap and pixbuf */
132 typedef struct _SourceTile SourceTile;
141 typedef struct _OverlayData OverlayData;
152 OverlayRendererFlags flags;
158 SIGNAL_SCROLL_NOTIFY,
159 SIGNAL_RENDER_COMPLETE,
177 PROP_CACHE_SIZE_DISPLAY,
178 PROP_CACHE_SIZE_TILES,
181 PROP_WINDOW_LIMIT_VALUE,
183 PROP_AUTOFIT_LIMIT_VALUE
188 PR_ZOOM_FORCE = 1 << 0,
189 PR_ZOOM_NEW = 1 << 1,
190 PR_ZOOM_CENTER = 1 << 2,
191 PR_ZOOM_INVALIDATE = 1 << 3,
192 PR_ZOOM_LAZY = 1 << 4 /* wait with redraw for pixbuf_renderer_area_changed */
195 static guint signals[SIGNAL_COUNT] = { 0 };
196 static GtkEventBoxClass *parent_class = NULL;
200 static void pixbuf_renderer_class_init(PixbufRendererClass *class);
201 static void pixbuf_renderer_init(PixbufRenderer *pr);
202 static void pixbuf_renderer_finalize(GObject *object);
203 static void pixbuf_renderer_set_property(GObject *object, guint prop_id,
204 const GValue *value, GParamSpec *pspec);
205 static void pixbuf_renderer_get_property(GObject *object, guint prop_id,
206 GValue *value, GParamSpec *pspec);
207 static gboolean pixbuf_renderer_expose(GtkWidget *widget, GdkEventExpose *event);
209 static void pr_render_complete_signal(PixbufRenderer *pr);
211 static void pr_overlay_list_clear(PixbufRenderer *pr);
212 static void pr_scroller_timer_set(PixbufRenderer *pr, gboolean start);
213 static void pr_border_draw(PixbufRenderer *pr, gint x, gint y, gint w, gint h);
216 static void pr_source_tile_free_all(PixbufRenderer *pr);
217 static void pr_tile_free_all(PixbufRenderer *pr);
218 static void pr_tile_invalidate_region(PixbufRenderer *pr, gint x, gint y, gint w, gint h);
219 static gboolean pr_tile_is_visible(PixbufRenderer *pr, ImageTile *it);
220 static void pr_queue_clear(PixbufRenderer *pr);
221 static void pr_queue_merge(QueueData *parent, QueueData *qd);
222 static void pr_queue(PixbufRenderer *pr, gint x, gint y, gint w, gint h,
223 gint clamp, ImageTileRenderType render, gboolean new_data, gboolean only_existing);
225 static void pr_redraw(PixbufRenderer *pr, gboolean new_data);
227 static void pr_zoom_sync(PixbufRenderer *pr, gdouble zoom,
228 PrZoomFlags flags, gint px, gint py);
230 static void pr_signals_connect(PixbufRenderer *pr);
231 static void pr_size_cb(GtkWidget *widget, GtkAllocation *allocation, gpointer data);
232 static void pr_hierarchy_changed_cb(GtkWidget *widget, GtkWidget *previous_toplevel, gpointer data);
233 static void pixbuf_renderer_paint(PixbufRenderer *pr, GdkRectangle *area);
234 static gint pr_queue_draw_idle_cb(gpointer data);
238 *-------------------------------------------------------------------
239 * Pixbuf Renderer object
240 *-------------------------------------------------------------------
243 GType pixbuf_renderer_get_type(void)
245 static GType pixbuf_renderer_type = 0;
247 if (!pixbuf_renderer_type)
249 static const GTypeInfo pixbuf_renderer_info =
251 sizeof(PixbufRendererClass), /* class_size */
252 NULL, /* base_init */
253 NULL, /* base_finalize */
254 (GClassInitFunc)pixbuf_renderer_class_init,
255 NULL, /* class_finalize */
256 NULL, /* class_data */
257 sizeof(PixbufRenderer), /* instance_size */
259 (GInstanceInitFunc)pixbuf_renderer_init, /* instance_init */
260 NULL, /* value_table */
263 pixbuf_renderer_type = g_type_register_static(GTK_TYPE_EVENT_BOX, "PixbufRenderer",
264 &pixbuf_renderer_info, 0);
267 return pixbuf_renderer_type;
270 static void pixbuf_renderer_class_init(PixbufRendererClass *class)
272 GObjectClass *gobject_class = G_OBJECT_CLASS(class);
273 GtkWidgetClass *widget_class = GTK_WIDGET_CLASS(class);
275 parent_class = g_type_class_peek_parent(class);
277 gobject_class->set_property = pixbuf_renderer_set_property;
278 gobject_class->get_property = pixbuf_renderer_get_property;
280 gobject_class->finalize = pixbuf_renderer_finalize;
282 widget_class->expose_event = pixbuf_renderer_expose;
284 g_object_class_install_property(gobject_class,
286 g_param_spec_double("zoom_min",
292 G_PARAM_READABLE | G_PARAM_WRITABLE));
294 g_object_class_install_property(gobject_class,
296 g_param_spec_double("zoom_max",
302 G_PARAM_READABLE | G_PARAM_WRITABLE));
304 g_object_class_install_property(gobject_class,
306 g_param_spec_uint("zoom_quality",
312 G_PARAM_READABLE | G_PARAM_WRITABLE));
314 g_object_class_install_property(gobject_class,
316 g_param_spec_boolean("zoom_2pass",
320 G_PARAM_READABLE | G_PARAM_WRITABLE));
322 g_object_class_install_property(gobject_class,
324 g_param_spec_boolean("zoom_expand",
325 "Expand image in autozoom.",
328 G_PARAM_READABLE | G_PARAM_WRITABLE));
330 g_object_class_install_property(gobject_class,
332 g_param_spec_uint("dither_quality",
337 GDK_RGB_DITHER_NORMAL,
338 G_PARAM_READABLE | G_PARAM_WRITABLE));
340 g_object_class_install_property(gobject_class,
342 g_param_spec_uint("scroll_reset",
343 "New image scroll reset",
345 PR_SCROLL_RESET_TOPLEFT,
346 PR_SCROLL_RESET_NOCHANGE,
347 PR_SCROLL_RESET_TOPLEFT,
348 G_PARAM_READABLE | G_PARAM_WRITABLE));
350 g_object_class_install_property(gobject_class,
352 g_param_spec_boolean("delay_flip",
353 "Delay image update",
356 G_PARAM_READABLE | G_PARAM_WRITABLE));
358 g_object_class_install_property(gobject_class,
360 g_param_spec_boolean("loading",
361 "Image actively loading",
364 G_PARAM_READABLE | G_PARAM_WRITABLE));
366 g_object_class_install_property(gobject_class,
368 g_param_spec_boolean("complete",
369 "Image rendering complete",
372 G_PARAM_READABLE | G_PARAM_WRITABLE));
374 g_object_class_install_property(gobject_class,
375 PROP_CACHE_SIZE_DISPLAY,
376 g_param_spec_uint("cache_display",
377 "Display cache size MB",
381 PR_CACHE_SIZE_DEFAULT,
382 G_PARAM_READABLE | G_PARAM_WRITABLE));
384 g_object_class_install_property(gobject_class,
385 PROP_CACHE_SIZE_TILES,
386 g_param_spec_uint("cache_tiles",
388 "Number of tiles to retain in memory at any one time.",
391 PR_CACHE_SIZE_DEFAULT,
392 G_PARAM_READABLE | G_PARAM_WRITABLE));
394 g_object_class_install_property(gobject_class,
396 g_param_spec_boolean("window_fit",
397 "Fit window to image size",
400 G_PARAM_READABLE | G_PARAM_WRITABLE));
402 g_object_class_install_property(gobject_class,
404 g_param_spec_boolean("window_limit",
405 "Limit size of parent window",
408 G_PARAM_READABLE | G_PARAM_WRITABLE));
410 g_object_class_install_property(gobject_class,
411 PROP_WINDOW_LIMIT_VALUE,
412 g_param_spec_uint("window_limit_value",
413 "Size limit of parent window",
418 G_PARAM_READABLE | G_PARAM_WRITABLE));
420 g_object_class_install_property(gobject_class,
422 g_param_spec_boolean("autofit_limit",
423 "Limit size of image when autofitting",
426 G_PARAM_READABLE | G_PARAM_WRITABLE));
428 g_object_class_install_property(gobject_class,
429 PROP_AUTOFIT_LIMIT_VALUE,
430 g_param_spec_uint("autofit_limit_value",
431 "Size limit of image when autofitting",
436 G_PARAM_READABLE | G_PARAM_WRITABLE));
439 signals[SIGNAL_ZOOM] =
441 G_OBJECT_CLASS_TYPE(gobject_class),
443 G_STRUCT_OFFSET(PixbufRendererClass, zoom),
445 g_cclosure_marshal_VOID__DOUBLE,
449 signals[SIGNAL_CLICKED] =
450 g_signal_new("clicked",
451 G_OBJECT_CLASS_TYPE(gobject_class),
453 G_STRUCT_OFFSET(PixbufRendererClass, clicked),
455 g_cclosure_marshal_VOID__BOXED,
459 signals[SIGNAL_SCROLL_NOTIFY] =
460 g_signal_new("scroll-notify",
461 G_OBJECT_CLASS_TYPE(gobject_class),
463 G_STRUCT_OFFSET(PixbufRendererClass, scroll_notify),
465 g_cclosure_marshal_VOID__VOID,
468 signals[SIGNAL_RENDER_COMPLETE] =
469 g_signal_new("render-complete",
470 G_OBJECT_CLASS_TYPE(gobject_class),
472 G_STRUCT_OFFSET(PixbufRendererClass, render_complete),
474 g_cclosure_marshal_VOID__VOID,
477 signals[SIGNAL_DRAG] =
479 G_OBJECT_CLASS_TYPE(gobject_class),
481 G_STRUCT_OFFSET(PixbufRendererClass, drag),
483 g_cclosure_marshal_VOID__BOXED,
487 signals[SIGNAL_UPDATE_PIXEL] =
488 g_signal_new("update-pixel",
489 G_OBJECT_CLASS_TYPE(gobject_class),
491 G_STRUCT_OFFSET(PixbufRendererClass, update_pixel),
493 g_cclosure_marshal_VOID__VOID,
497 static void pixbuf_renderer_init(PixbufRenderer *pr)
501 box = GTK_WIDGET(pr);
503 pr->zoom_min = PR_ZOOM_MIN;
504 pr->zoom_max = PR_ZOOM_MAX;
505 pr->zoom_quality = GDK_INTERP_BILINEAR;
506 pr->zoom_2pass = FALSE;
511 pr->dither_quality = GDK_RGB_DITHER_NORMAL;
513 pr->scroll_reset = PR_SCROLL_RESET_TOPLEFT;
515 pr->draw_idle_id = 0;
517 pr->tile_width = PR_TILE_SIZE;
518 pr->tile_height = PR_TILE_SIZE;
521 pr->tile_cache_size = 0;
523 pr->tile_cache_max = PR_CACHE_SIZE_DEFAULT;
526 pr->scroller_overlay = -1;
531 pr->source_tiles_enabled = FALSE;
532 pr->source_tiles = NULL;
536 pr->norm_center_x = 0.5;
537 pr->norm_center_y = 0.5;
539 gtk_widget_set_double_buffered(box, FALSE);
540 g_signal_connect_after(G_OBJECT(box), "size_allocate",
541 G_CALLBACK(pr_size_cb), pr);
543 g_signal_connect(G_OBJECT(pr), "hierarchy-changed",
544 G_CALLBACK(pr_hierarchy_changed_cb), pr);
546 pr_signals_connect(pr);
549 static void pixbuf_renderer_finalize(GObject *object)
553 pr = PIXBUF_RENDERER(object);
556 pr_tile_free_all(pr);
559 if (pr->pixbuf) g_object_unref(pr->pixbuf);
560 if (pr->spare_tile) g_object_unref(pr->spare_tile);
562 pr_scroller_timer_set(pr, FALSE);
563 pr_overlay_list_clear(pr);
565 pr_source_tile_free_all(pr);
568 PixbufRenderer *pixbuf_renderer_new(void)
570 return g_object_new(TYPE_PIXBUF_RENDERER, NULL);
573 static void pixbuf_renderer_set_property(GObject *object, guint prop_id,
574 const GValue *value, GParamSpec *pspec)
578 pr = PIXBUF_RENDERER(object);
583 pr->zoom_min = g_value_get_double(value);
586 pr->zoom_max = g_value_get_double(value);
588 case PROP_ZOOM_QUALITY:
589 pr->zoom_quality = g_value_get_uint(value);
591 case PROP_ZOOM_2PASS:
592 pr->zoom_2pass = g_value_get_boolean(value);
594 case PROP_ZOOM_EXPAND:
595 pr->zoom_expand = g_value_get_boolean(value);
597 case PROP_DITHER_QUALITY:
598 pr->dither_quality = g_value_get_uint(value);
600 case PROP_SCROLL_RESET:
601 pr->scroll_reset = g_value_get_uint(value);
603 case PROP_DELAY_FLIP:
604 pr->delay_flip = g_value_get_boolean(value);
607 pr->loading = g_value_get_boolean(value);
610 pr->complete = g_value_get_boolean(value);
612 case PROP_CACHE_SIZE_DISPLAY:
613 pr->tile_cache_max = g_value_get_uint(value);
615 case PROP_CACHE_SIZE_TILES:
616 pr->source_tiles_cache_size = g_value_get_uint(value);
618 case PROP_WINDOW_FIT:
619 pr->window_fit = g_value_get_boolean(value);
621 case PROP_WINDOW_LIMIT:
622 pr->window_limit = g_value_get_boolean(value);
624 case PROP_WINDOW_LIMIT_VALUE:
625 pr->window_limit_size = g_value_get_uint(value);
627 case PROP_AUTOFIT_LIMIT:
628 pr->autofit_limit = g_value_get_boolean(value);
630 case PROP_AUTOFIT_LIMIT_VALUE:
631 pr->autofit_limit_size = g_value_get_uint(value);
634 G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
639 static void pixbuf_renderer_get_property(GObject *object, guint prop_id,
640 GValue *value, GParamSpec *pspec)
644 pr = PIXBUF_RENDERER(object);
649 g_value_set_double(value, pr->zoom_min);
652 g_value_set_double(value, pr->zoom_max);
654 case PROP_ZOOM_QUALITY:
655 g_value_set_uint(value, pr->zoom_quality);
657 case PROP_ZOOM_2PASS:
658 g_value_set_boolean(value, pr->zoom_2pass);
660 case PROP_ZOOM_EXPAND:
661 g_value_set_boolean(value, pr->zoom_expand);
663 case PROP_DITHER_QUALITY:
664 g_value_set_uint(value, pr->dither_quality);
666 case PROP_SCROLL_RESET:
667 g_value_set_uint(value, pr->scroll_reset);
669 case PROP_DELAY_FLIP:
670 g_value_set_boolean(value, pr->delay_flip);
673 g_value_set_boolean(value, pr->loading);
676 g_value_set_boolean(value, pr->complete);
678 case PROP_CACHE_SIZE_DISPLAY:
679 g_value_set_uint(value, pr->tile_cache_max);
681 case PROP_CACHE_SIZE_TILES:
682 g_value_set_uint(value, pr->source_tiles_cache_size);
684 case PROP_WINDOW_FIT:
685 g_value_set_boolean(value, pr->window_fit);
687 case PROP_WINDOW_LIMIT:
688 g_value_set_boolean(value, pr->window_limit);
690 case PROP_WINDOW_LIMIT_VALUE:
691 g_value_set_uint(value, pr->window_limit_size);
693 case PROP_AUTOFIT_LIMIT:
694 g_value_set_boolean(value, pr->autofit_limit);
696 case PROP_AUTOFIT_LIMIT_VALUE:
697 g_value_set_uint(value, pr->autofit_limit_size);
700 G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
705 static gboolean pixbuf_renderer_expose(GtkWidget *widget, GdkEventExpose *event)
707 #if GTK_CHECK_VERSION(2,20,0)
708 if (gtk_widget_is_drawable(widget))
710 if (GTK_WIDGET_DRAWABLE(widget))
713 #if GTK_CHECK_VERSION(2,20,0)
714 if (gtk_widget_get_has_window(widget))
716 if (!GTK_WIDGET_NO_WINDOW(widget))
719 if (event->window != widget->window)
723 gdk_window_get_position(event->window, &area.x, &area.y);
725 area.x += event->area.x;
726 area.y += event->area.y;
727 area.width = event->area.width;
728 area.height = event->area.height;
729 pixbuf_renderer_paint(PIXBUF_RENDERER(widget), &area);
733 pixbuf_renderer_paint(PIXBUF_RENDERER(widget), &event->area);
742 *-------------------------------------------------------------------
744 *-------------------------------------------------------------------
747 static void widget_set_cursor(GtkWidget *widget, gint icon)
751 if (!widget->window) return;
759 cursor = gdk_cursor_new(icon);
762 gdk_window_set_cursor(widget->window, cursor);
764 if (cursor) gdk_cursor_unref(cursor);
767 static gint pixmap_calc_size(GdkPixmap *pixmap)
771 d = gdk_drawable_get_depth(pixmap);
772 gdk_drawable_get_size(pixmap, &w, &h);
773 return w * h * (d / 8);
776 static gboolean pr_clip_region(gint x, gint y, gint w, gint h,
777 gint clip_x, gint clip_y, gint clip_w, gint clip_h,
778 gint *rx, gint *ry, gint *rw, gint *rh)
780 if (clip_x + clip_w <= x ||
782 clip_y + clip_h <= y ||
788 *rx = MAX(x, clip_x);
789 *rw = MIN((x + w), (clip_x + clip_w)) - *rx;
791 *ry = MAX(y, clip_y);
792 *rh = MIN((y + h), (clip_y + clip_h)) - *ry;
797 static gboolean pr_parent_window_sizable(PixbufRenderer *pr)
799 GdkWindowState state;
801 if (!pr->parent_window) return FALSE;
802 if (!pr->window_fit) return FALSE;
803 if (!GTK_WIDGET(pr)->window) return FALSE;
805 if (!pr->parent_window->window) return FALSE;
806 state = gdk_window_get_state(pr->parent_window->window);
807 if (state & GDK_WINDOW_STATE_MAXIMIZED) return FALSE;
812 static gboolean pr_parent_window_resize(PixbufRenderer *pr, gint w, gint h)
818 if (!pr_parent_window_sizable(pr)) return FALSE;
820 if (pr->window_limit)
822 gint sw = gdk_screen_width() * pr->window_limit_size / 100;
823 gint sh = gdk_screen_height() * pr->window_limit_size / 100;
829 widget = GTK_WIDGET(pr);
830 parent = GTK_WIDGET(pr->parent_window);
832 w += (parent->allocation.width - widget->allocation.width);
833 h += (parent->allocation.height - widget->allocation.height);
835 gdk_drawable_get_size(parent->window, &ww, &wh);
836 if (w == ww && h == wh) return FALSE;
838 gdk_window_resize(parent->window, w, h);
843 void pixbuf_renderer_set_parent(PixbufRenderer *pr, GtkWindow *window)
845 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
846 g_return_if_fail(window == NULL || GTK_IS_WINDOW(window));
848 pr->parent_window = GTK_WIDGET(window);
851 GtkWindow *pixbuf_renderer_get_parent(PixbufRenderer *pr)
853 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), NULL);
855 return GTK_WINDOW(pr->parent_window);
860 *-------------------------------------------------------------------
862 *-------------------------------------------------------------------
865 static void pr_overlay_get_position(PixbufRenderer *pr, OverlayData *od,
866 gint *x, gint *y, gint *w, gint *h)
870 pw = gdk_pixbuf_get_width(od->pixbuf);
871 ph = gdk_pixbuf_get_height(od->pixbuf);
875 if (od->flags & OVL_RELATIVE)
877 if (px < 0) px = pr->window_width - pw + px;
878 if (py < 0) py = pr->window_height - ph + py;
887 static void pr_overlay_init_window(PixbufRenderer *pr, OverlayData *od)
890 GdkWindowAttr attributes;
891 gint attributes_mask;
893 pr_overlay_get_position(pr, od, &px, &py, &pw, &ph);
895 attributes.window_type = GDK_WINDOW_CHILD;
896 attributes.wclass = GDK_INPUT_OUTPUT;
897 attributes.width = pw;
898 attributes.height = ph;
899 attributes.event_mask = GDK_EXPOSURE_MASK;
902 od->window = gdk_window_new(GTK_WIDGET(pr)->window, &attributes, attributes_mask);
903 gdk_window_set_user_data(od->window, pr);
904 gdk_window_move(od->window, px, py);
905 gdk_window_show(od->window);
908 static void pr_overlay_draw(PixbufRenderer *pr, gint x, gint y, gint w, gint h,
914 box = GTK_WIDGET(pr);
916 work = pr->overlay_list;
926 if (!od->window) pr_overlay_init_window(pr, od);
928 pr_overlay_get_position(pr, od, &px, &py, &pw, &ph);
929 if (pr_clip_region(x, y, w, h, px, py, pw, ph, &rx, &ry, &rw, &rh))
931 if (!pr->overlay_buffer)
933 pr->overlay_buffer = gdk_pixmap_new(((GtkWidget *)pr)->window, pr->tile_width, pr->tile_height, -1);
938 #if GTK_CHECK_VERSION(2,20,0)
939 gdk_draw_drawable(pr->overlay_buffer, box->style->fg_gc[gtk_widget_get_state(box)],
941 gdk_draw_drawable(pr->overlay_buffer, box->style->fg_gc[GTK_WIDGET_STATE(box)],
944 rx - (pr->x_offset + (it->x - pr->x_scroll)),
945 ry - (pr->y_offset + (it->y - pr->y_scroll)),
947 gdk_draw_pixbuf(pr->overlay_buffer,
948 #if GTK_CHECK_VERSION(2,20,0)
949 box->style->fg_gc[gtk_widget_get_state(box)],
951 box->style->fg_gc[GTK_WIDGET_STATE(box)],
956 pr->dither_quality, rx, ry);
957 #if GTK_CHECK_VERSION(2,20,0)
958 gdk_draw_drawable(od->window, box->style->fg_gc[gtk_widget_get_state(box)],
960 gdk_draw_drawable(od->window, box->style->fg_gc[GTK_WIDGET_STATE(box)],
964 rx - px, ry - py, rw, rh);
968 /* no ImageTile means region may be larger than our scratch buffer */
971 for (sx = rx; sx < rx + rw; sx += pr->tile_width)
972 for (sy = ry; sy < ry + rh; sy += pr->tile_height)
976 sw = MIN(rx + rw - sx, pr->tile_width);
977 sh = MIN(ry + rh - sy, pr->tile_height);
979 gdk_draw_rectangle(pr->overlay_buffer,
980 #if GTK_CHECK_VERSION(2,20,0)
981 box->style->bg_gc[gtk_widget_get_state(box)], TRUE,
983 box->style->bg_gc[GTK_WIDGET_STATE(box)], TRUE,
986 gdk_draw_pixbuf(pr->overlay_buffer,
987 #if GTK_CHECK_VERSION(2,20,0)
988 box->style->fg_gc[gtk_widget_get_state(box)],
990 box->style->fg_gc[GTK_WIDGET_STATE(box)],
995 pr->dither_quality, sx, sy);
996 #if GTK_CHECK_VERSION(2,20,0)
997 gdk_draw_drawable(od->window, box->style->fg_gc[gtk_widget_get_state(box)],
999 gdk_draw_drawable(od->window, box->style->fg_gc[GTK_WIDGET_STATE(box)],
1003 sx - px, sy - py, sw, sh);
1010 static void pr_overlay_queue_draw(PixbufRenderer *pr, OverlayData *od)
1014 pr_overlay_get_position(pr, od, &x, &y, &w, &h);
1015 pr_queue(pr, pr->x_scroll - pr->x_offset + x,
1016 pr->y_scroll - pr->y_offset + y,
1018 FALSE, TILE_RENDER_ALL, FALSE, FALSE);
1020 pr_border_draw(pr, x, y, w, h);
1023 static void pr_overlay_queue_all(PixbufRenderer *pr)
1027 work = pr->overlay_list;
1030 OverlayData *od = work->data;
1033 pr_overlay_queue_draw(pr, od);
1037 static void pr_overlay_update_sizes(PixbufRenderer *pr)
1041 work = pr->overlay_list;
1044 OverlayData *od = work->data;
1047 if (!od->window) pr_overlay_init_window(pr, od);
1049 if (od->flags & OVL_RELATIVE)
1053 pr_overlay_get_position(pr, od, &x, &y, &w, &h);
1054 gdk_window_move_resize(od->window, x, y, w, h);
1059 static OverlayData *pr_overlay_find(PixbufRenderer *pr, gint id)
1063 work = pr->overlay_list;
1066 OverlayData *od = work->data;
1069 if (od->id == id) return od;
1076 gint pixbuf_renderer_overlay_add(PixbufRenderer *pr, GdkPixbuf *pixbuf, gint x, gint y,
1077 OverlayRendererFlags flags)
1082 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), -1);
1083 g_return_val_if_fail(pixbuf != NULL, -1);
1086 while (pr_overlay_find(pr, id)) id++;
1088 od = g_new0(OverlayData, 1);
1090 od->pixbuf = pixbuf;
1091 g_object_ref(G_OBJECT(od->pixbuf));
1096 pr_overlay_init_window(pr, od);
1098 pr->overlay_list = g_list_append(pr->overlay_list, od);
1100 pr_overlay_queue_draw(pr, od);
1105 static void pr_overlay_free(PixbufRenderer *pr, OverlayData *od)
1107 pr->overlay_list = g_list_remove(pr->overlay_list, od);
1109 if (od->pixbuf) g_object_unref(G_OBJECT(od->pixbuf));
1110 if (od->window) gdk_window_destroy(od->window);
1113 if (!pr->overlay_list && pr->overlay_buffer)
1115 g_object_unref(pr->overlay_buffer);
1116 pr->overlay_buffer = NULL;
1120 static void pr_overlay_list_clear(PixbufRenderer *pr)
1122 while (pr->overlay_list)
1126 od = pr->overlay_list->data;
1127 pr_overlay_free(pr, od);
1131 static void pr_overlay_list_reset_window(PixbufRenderer *pr)
1135 if (pr->overlay_buffer) g_object_unref(pr->overlay_buffer);
1136 pr->overlay_buffer = NULL;
1138 work = pr->overlay_list;
1141 OverlayData *od = work->data;
1143 if (od->window) gdk_window_destroy(od->window);
1148 void pixbuf_renderer_overlay_set(PixbufRenderer *pr, gint id, GdkPixbuf *pixbuf, gint x, gint y)
1152 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
1154 od = pr_overlay_find(pr, id);
1159 gint px, py, pw, ph;
1161 g_object_ref(G_OBJECT(pixbuf));
1162 g_object_unref(G_OBJECT(od->pixbuf));
1163 od->pixbuf = pixbuf;
1168 if (!od->window) pr_overlay_init_window(pr, od);
1170 pr_overlay_queue_draw(pr, od);
1171 pr_overlay_get_position(pr, od, &px, &py, &pw, &ph);
1172 gdk_window_move_resize(od->window, px, py, pw, ph);
1176 pr_overlay_queue_draw(pr, od);
1177 pr_overlay_free(pr, od);
1181 gboolean pixbuf_renderer_overlay_get(PixbufRenderer *pr, gint id, GdkPixbuf **pixbuf, gint *x, gint *y)
1185 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), FALSE);
1187 od = pr_overlay_find(pr, id);
1188 if (!od) return FALSE;
1190 if (pixbuf) *pixbuf = od->pixbuf;
1197 void pixbuf_renderer_overlay_remove(PixbufRenderer *pr, gint id)
1199 pixbuf_renderer_overlay_set(pr, id, NULL, 0, 0);
1202 static void pr_hierarchy_changed_cb(GtkWidget *widget, GtkWidget *previous_toplevel, gpointer data)
1204 PixbufRenderer *pr = data;
1205 pr_overlay_list_reset_window(pr);
1210 *-------------------------------------------------------------------
1212 *-------------------------------------------------------------------
1216 static gboolean pr_scroller_update_cb(gpointer data)
1218 PixbufRenderer *pr = data;
1222 /* this was a simple scroll by difference between scroller and mouse position,
1223 * but all this math results in a smoother result and accounts for a dead zone.
1226 if (abs(pr->scroller_xpos - pr->scroller_x) < PR_SCROLLER_DEAD_ZONE)
1232 gint shift = PR_SCROLLER_DEAD_ZONE / 2 * PR_SCROLLER_UPDATES_PER_SEC;
1233 x = (pr->scroller_xpos - pr->scroller_x) / 2 * PR_SCROLLER_UPDATES_PER_SEC;
1234 x += (x > 0) ? -shift : shift;
1237 if (abs(pr->scroller_ypos - pr->scroller_y) < PR_SCROLLER_DEAD_ZONE)
1243 gint shift = PR_SCROLLER_DEAD_ZONE / 2 * PR_SCROLLER_UPDATES_PER_SEC;
1244 y = (pr->scroller_ypos - pr->scroller_y) / 2 * PR_SCROLLER_UPDATES_PER_SEC;
1245 y += (y > 0) ? -shift : shift;
1248 if (abs(x) < PR_SCROLLER_DEAD_ZONE * PR_SCROLLER_UPDATES_PER_SEC)
1254 xinc = pr->scroller_xinc;
1258 if (xinc < 0) xinc = 0;
1259 if (x < xinc) xinc = x;
1260 if (x > xinc) xinc = MIN(xinc + x / PR_SCROLLER_UPDATES_PER_SEC, x);
1264 if (xinc > 0) xinc = 0;
1265 if (x > xinc) xinc = x;
1266 if (x < xinc) xinc = MAX(xinc + x / PR_SCROLLER_UPDATES_PER_SEC, x);
1270 if (abs(y) < PR_SCROLLER_DEAD_ZONE * PR_SCROLLER_UPDATES_PER_SEC)
1276 yinc = pr->scroller_yinc;
1280 if (yinc < 0) yinc = 0;
1281 if (y < yinc) yinc = y;
1282 if (y > yinc) yinc = MIN(yinc + y / PR_SCROLLER_UPDATES_PER_SEC, y);
1286 if (yinc > 0) yinc = 0;
1287 if (y > yinc) yinc = y;
1288 if (y < yinc) yinc = MAX(yinc + y / PR_SCROLLER_UPDATES_PER_SEC, y);
1292 pr->scroller_xinc = xinc;
1293 pr->scroller_yinc = yinc;
1295 xinc = xinc / PR_SCROLLER_UPDATES_PER_SEC;
1296 yinc = yinc / PR_SCROLLER_UPDATES_PER_SEC;
1298 pixbuf_renderer_scroll(pr, xinc, yinc);
1303 static void pr_scroller_timer_set(PixbufRenderer *pr, gboolean start)
1305 if (pr->scroller_id)
1307 g_source_remove(pr->scroller_id);
1308 pr->scroller_id = 0;
1313 pr->scroller_id = g_timeout_add(1000 / PR_SCROLLER_UPDATES_PER_SEC,
1314 pr_scroller_update_cb, pr);
1318 static void pr_scroller_start(PixbufRenderer *pr, gint x, gint y)
1320 if (pr->scroller_overlay == -1)
1326 pixbuf = pixbuf_inline(PIXBUF_INLINE_SCROLLER);
1328 pixbuf = gdk_pixbuf_new(GDK_COLORSPACE_RGB, FALSE, 8, 32, 32);
1329 gdk_pixbuf_fill(pixbuf, 0x000000ff);
1331 w = gdk_pixbuf_get_width(pixbuf);
1332 h = gdk_pixbuf_get_height(pixbuf);
1334 pr->scroller_overlay = pixbuf_renderer_overlay_add(pr, pixbuf, x - w / 2, y - h / 2, OVL_NORMAL);
1335 g_object_unref(pixbuf);
1340 pr->scroller_xpos = x;
1341 pr->scroller_ypos = y;
1343 pr_scroller_timer_set(pr, TRUE);
1346 static void pr_scroller_stop(PixbufRenderer *pr)
1348 if (!pr->scroller_id) return;
1350 pixbuf_renderer_overlay_remove(pr, pr->scroller_overlay);
1351 pr->scroller_overlay = -1;
1353 pr_scroller_timer_set(pr, FALSE);
1357 *-------------------------------------------------------------------
1359 *-------------------------------------------------------------------
1362 static void pr_border_draw(PixbufRenderer *pr, gint x, gint y, gint w, gint h)
1365 gint rx, ry, rw, rh;
1367 box = GTK_WIDGET(pr);
1369 if (!box->window) return;
1371 if (!pr->pixbuf && !pr->source_tiles_enabled)
1373 if (pr_clip_region(x, y, w, h,
1375 pr->window_width, pr->window_height,
1376 &rx, &ry, &rw, &rh))
1378 gdk_window_clear_area(box->window, rx, ry, rw, rh);
1379 pr_overlay_draw(pr, rx, ry, rw, rh, NULL);
1384 if (pr->vis_width < pr->window_width)
1386 if (pr->x_offset > 0 &&
1387 pr_clip_region(x, y, w, h,
1389 pr->x_offset, pr->window_height,
1390 &rx, &ry, &rw, &rh))
1392 gdk_window_clear_area(box->window, rx, ry, rw, rh);
1393 pr_overlay_draw(pr, rx, ry, rw, rh, NULL);
1395 if (pr->window_width - pr->vis_width - pr->x_offset > 0 &&
1396 pr_clip_region(x, y, w, h,
1397 pr->x_offset + pr->vis_width, 0,
1398 pr->window_width - pr->vis_width - pr->x_offset, pr->window_height,
1399 &rx, &ry, &rw, &rh))
1401 gdk_window_clear_area(box->window, rx, ry, rw, rh);
1402 pr_overlay_draw(pr, rx, ry, rw, rh, NULL);
1405 if (pr->vis_height < pr->window_height)
1407 if (pr->y_offset > 0 &&
1408 pr_clip_region(x, y, w, h,
1410 pr->vis_width, pr->y_offset,
1411 &rx, &ry, &rw, &rh))
1413 gdk_window_clear_area(box->window, rx, ry, rw, rh);
1414 pr_overlay_draw(pr, rx, ry, rw, rh, NULL);
1416 if (pr->window_height - pr->vis_height - pr->y_offset > 0 &&
1417 pr_clip_region(x, y, w, h,
1418 pr->x_offset, pr->y_offset + pr->vis_height,
1419 pr->vis_width, pr->window_height - pr->vis_height - pr->y_offset,
1420 &rx, &ry, &rw, &rh))
1422 gdk_window_clear_area(box->window, rx, ry, rw, rh);
1423 pr_overlay_draw(pr, rx, ry, rw, rh, NULL);
1428 static void pr_border_clear(PixbufRenderer *pr)
1430 pr_border_draw(pr, 0, 0, pr->window_width, pr->window_height);
1433 void pixbuf_renderer_set_color(PixbufRenderer *pr, GdkColor *color)
1438 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
1440 widget = GTK_WIDGET(pr);
1445 style = gtk_style_copy(gtk_widget_get_style(widget));
1446 slot = &style->bg[GTK_STATE_NORMAL];
1448 slot->red = color->red;
1449 slot->green = color->green;
1450 slot->blue = color->blue;
1453 style = gtk_style_copy(gtk_widget_get_default_style());
1456 gtk_widget_set_style(widget, style);
1458 #if GTK_CHECK_VERSION(2,20,0)
1459 if (gtk_widget_get_visible(widget)) pr_border_clear(pr);
1461 if (GTK_WIDGET_VISIBLE(widget)) pr_border_clear(pr);
1467 *-------------------------------------------------------------------
1469 *-------------------------------------------------------------------
1472 static void pr_source_tile_free(SourceTile *st)
1476 if (st->pixbuf) g_object_unref(st->pixbuf);
1480 static void pr_source_tile_free_all(PixbufRenderer *pr)
1484 work = pr->source_tiles;
1492 pr_source_tile_free(st);
1495 g_list_free(pr->source_tiles);
1496 pr->source_tiles = NULL;
1499 static void pr_source_tile_unset(PixbufRenderer *pr)
1501 pr_source_tile_free_all(pr);
1502 pr->source_tiles_enabled = FALSE;
1505 static gboolean pr_source_tile_visible(PixbufRenderer *pr, SourceTile *st)
1507 gint x1, y1, x2, y2;
1509 if (!st) return FALSE;
1511 x1 = ROUND_DOWN(pr->x_scroll, pr->tile_width);
1512 y1 = ROUND_DOWN(pr->y_scroll, pr->tile_height);
1513 x2 = ROUND_UP(pr->x_scroll + pr->vis_width, pr->tile_width);
1514 y2 = ROUND_UP(pr->y_scroll + pr->vis_height, pr->tile_height);
1516 return !((gdouble)st->x * pr->scale > (gdouble)x2 ||
1517 (gdouble)(st->x + pr->source_tile_width) * pr->scale < (gdouble)x1 ||
1518 (gdouble)st->y * pr->scale > (gdouble)y2 ||
1519 (gdouble)(st->y + pr->source_tile_height) * pr->scale < (gdouble)y1);
1522 static SourceTile *pr_source_tile_new(PixbufRenderer *pr, gint x, gint y)
1524 SourceTile *st = NULL;
1527 g_return_val_if_fail(pr->source_tile_width >= 1 && pr->source_tile_height >= 1, NULL);
1529 if (pr->source_tiles_cache_size < 4) pr->source_tiles_cache_size = 4;
1531 count = g_list_length(pr->source_tiles);
1532 if (count >= pr->source_tiles_cache_size)
1536 work = g_list_last(pr->source_tiles);
1537 while (work && count >= pr->source_tiles_cache_size)
1541 needle = work->data;
1544 if (!pr_source_tile_visible(pr, needle))
1546 pr->source_tiles = g_list_remove(pr->source_tiles, needle);
1548 if (pr->func_tile_dispose)
1550 pr->func_tile_dispose(pr, needle->x, needle->y,
1551 pr->source_tile_width, pr->source_tile_height,
1552 needle->pixbuf, pr->func_tile_data);
1561 pr_source_tile_free(needle);
1571 st = g_new0(SourceTile, 1);
1572 st->pixbuf = gdk_pixbuf_new(GDK_COLORSPACE_RGB, FALSE, 8,
1573 pr->source_tile_width, pr->source_tile_height);
1576 st->x = ROUND_DOWN(x, pr->source_tile_width);
1577 st->y = ROUND_DOWN(y, pr->source_tile_height);
1580 pr->source_tiles = g_list_prepend(pr->source_tiles, st);
1585 static SourceTile *pr_source_tile_request(PixbufRenderer *pr, gint x, gint y)
1589 st = pr_source_tile_new(pr, x, y);
1590 if (!st) return NULL;
1592 if (pr->func_tile_request &&
1593 pr->func_tile_request(pr, st->x, st->y,
1594 pr->source_tile_width, pr->source_tile_height, st->pixbuf, pr->func_tile_data))
1599 pr_tile_invalidate_region(pr, st->x * pr->scale, st->y * pr->scale,
1600 pr->source_tile_width * pr->scale, pr->source_tile_height * pr->scale);
1605 static SourceTile *pr_source_tile_find(PixbufRenderer *pr, gint x, gint y)
1609 work = pr->source_tiles;
1612 SourceTile *st = work->data;
1614 if (x >= st->x && x < st->x + pr->source_tile_width &&
1615 y >= st->y && y < st->y + pr->source_tile_height)
1617 if (work != pr->source_tiles)
1619 pr->source_tiles = g_list_remove_link(pr->source_tiles, work);
1620 pr->source_tiles = g_list_concat(work, pr->source_tiles);
1631 static GList *pr_source_tile_compute_region(PixbufRenderer *pr, gint x, gint y, gint w, gint h, gboolean request)
1639 if (w > pr->image_width) w = pr->image_width;
1640 if (h > pr->image_height) h = pr->image_height;
1642 sx = ROUND_DOWN(x, pr->source_tile_width);
1643 sy = ROUND_DOWN(y, pr->source_tile_height);
1645 for (x1 = sx; x1 < x + w; x1+= pr->source_tile_width)
1647 for (y1 = sy; y1 < y + h; y1 += pr->source_tile_height)
1651 st = pr_source_tile_find(pr, x1, y1);
1652 if (!st && request) st = pr_source_tile_request(pr, x1, y1);
1654 if (st) list = g_list_prepend(list, st);
1658 return g_list_reverse(list);
1661 static void pr_source_tile_changed(PixbufRenderer *pr, gint x, gint y, gint width, gint height)
1665 if (width < 1 || height < 1) return;
1667 work = pr->source_tiles;
1671 gint rx, ry, rw, rh;
1676 if (pr_clip_region(st->x, st->y, pr->source_tile_width, pr->source_tile_height,
1677 x, y, width, height,
1678 &rx, &ry, &rw, &rh))
1682 pixbuf = gdk_pixbuf_new_subpixbuf(st->pixbuf, rx - st->x, ry - st->y, rw, rh);
1683 if (pr->func_tile_request &&
1684 pr->func_tile_request(pr, rx, ry, rw, rh, pixbuf, pr->func_tile_data))
1686 pr_tile_invalidate_region(pr, rx * pr->scale, ry * pr->scale,
1687 rw * pr->scale, rh * pr->scale);
1689 g_object_unref(pixbuf);
1694 static gboolean pr_source_tile_render(PixbufRenderer *pr, ImageTile *it,
1695 gint x, gint y, gint w, gint h,
1696 gboolean new_data, gboolean fast)
1701 gboolean draw = FALSE;
1703 box = GTK_WIDGET(pr);
1705 if (pr->zoom == 1.0 || pr->scale == 1.0)
1707 list = pr_source_tile_compute_region(pr, it->x + x, it->y + y, w, h, TRUE);
1712 gint rx, ry, rw, rh;
1717 if (pr_clip_region(st->x, st->y, pr->source_tile_width, pr->source_tile_height,
1718 it->x + x, it->y + y, w, h,
1719 &rx, &ry, &rw, &rh))
1723 gdk_draw_rectangle(it->pixmap, box->style->black_gc, TRUE,
1724 rx - st->x, ry - st->y, rw, rh);
1726 else /* (pr->zoom == 1.0 || pr->scale == 1.0) */
1728 gdk_draw_pixbuf(it->pixmap,
1729 #if GTK_CHECK_VERSION(2,20,0)
1730 box->style->fg_gc[gtk_widget_get_state(box)],
1732 box->style->fg_gc[GTK_WIDGET_STATE(box)],
1735 rx - st->x, ry - st->y,
1736 rx - it->x, ry - it->y,
1738 pr->dither_quality, rx, ry);
1745 gdouble scale_x, scale_y;
1746 gint sx, sy, sw, sh;
1748 if (pr->image_width == 0 || pr->image_height == 0) return FALSE;
1749 scale_x = (gdouble)pr->width / pr->image_width;
1750 scale_y = (gdouble)pr->height / pr->image_height;
1752 sx = (gdouble)(it->x + x) / scale_x;
1753 sy = (gdouble)(it->y + y) / scale_y;
1754 sw = (gdouble)w / scale_x;
1755 sh = (gdouble)h / scale_y;
1757 if (pr->width < PR_MIN_SCALE_SIZE || pr->height < PR_MIN_SCALE_SIZE) fast = TRUE;
1760 /* draws red over draw region, to check for leaks (regions not filled) */
1761 pixbuf_set_rect_fill(it->pixbuf, x, y, w, h, 255, 0, 0, 255);
1764 list = pr_source_tile_compute_region(pr, sx, sy, sw, sh, TRUE);
1769 gint rx, ry, rw, rh;
1770 gint stx, sty, stw, sth;
1775 stx = floor((gdouble)st->x * scale_x);
1776 sty = floor((gdouble)st->y * scale_y);
1777 stw = ceil((gdouble)(st->x + pr->source_tile_width) * scale_x) - stx;
1778 sth = ceil((gdouble)(st->y + pr->source_tile_height) * scale_y) - sty;
1780 if (pr_clip_region(stx, sty, stw, sth,
1781 it->x + x, it->y + y, w, h,
1782 &rx, &ry, &rw, &rh))
1786 gdk_draw_rectangle(it->pixmap, box->style->black_gc, TRUE,
1787 rx - st->x, ry - st->y, rw, rh);
1794 /* may need to use unfloored stx,sty values here */
1795 offset_x = (gdouble)(stx - it->x);
1796 offset_y = (gdouble)(sty - it->y);
1798 gdk_pixbuf_scale(st->pixbuf, it->pixbuf, rx - it->x, ry - it->y, rw, rh,
1799 (gdouble) 0.0 + offset_x,
1800 (gdouble) 0.0 + offset_y,
1802 (fast) ? GDK_INTERP_NEAREST : pr->zoom_quality);
1814 void pixbuf_renderer_set_tiles(PixbufRenderer *pr, gint width, gint height,
1815 gint tile_width, gint tile_height, gint cache_size,
1816 PixbufRendererTileRequestFunc func_request,
1817 PixbufRendererTileDisposeFunc func_dispose,
1821 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
1822 g_return_if_fail(tile_width >= 32 && tile_width >= 32);
1823 g_return_if_fail(width >= 32 && height > 32);
1824 g_return_if_fail(func_request != NULL);
1826 if (pr->pixbuf) g_object_unref(pr->pixbuf);
1829 pr_source_tile_unset(pr);
1831 if (cache_size < 4) cache_size = 4;
1833 pr->source_tiles_enabled = TRUE;
1834 pr->source_tiles_cache_size = cache_size;
1835 pr->source_tile_width = tile_width;
1836 pr->source_tile_height = tile_height;
1838 pr->image_width = width;
1839 pr->image_height = height;
1841 pr->func_tile_request = func_request;
1842 pr->func_tile_dispose = func_dispose;
1843 pr->func_tile_data = user_data;
1845 pr_zoom_sync(pr, zoom, PR_ZOOM_FORCE | PR_ZOOM_NEW, 0, 0);
1846 pr_redraw(pr, TRUE);
1849 void pixbuf_renderer_set_tiles_size(PixbufRenderer *pr, gint width, gint height)
1851 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
1852 g_return_if_fail(width >= 32 && height > 32);
1854 if (!pr->source_tiles_enabled) return;
1855 if (pr->image_width == width && pr->image_height == height) return;
1857 pr->image_width = width;
1858 pr->image_height = height;
1860 pr_zoom_sync(pr, pr->zoom, PR_ZOOM_FORCE, 0, 0);
1863 gint pixbuf_renderer_get_tiles(PixbufRenderer *pr)
1865 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), FALSE);
1867 return pr->source_tiles_enabled;
1870 static void pr_zoom_adjust_real(PixbufRenderer *pr, gdouble increment,
1871 PrZoomFlags flags, gint x, gint y)
1873 gdouble zoom = pr->zoom;
1875 if (increment == 0.0) return;
1879 if (pr->scale < 1.0)
1881 zoom = 0.0 - 1.0 / pr->scale;
1889 if (increment < 0.0)
1891 if (zoom >= 1.0 && zoom + increment < 1.0)
1893 zoom = zoom + increment - 2.0;
1897 zoom = zoom + increment;
1902 if (zoom <= -1.0 && zoom + increment > -1.0)
1904 zoom = zoom + increment + 2.0;
1908 zoom = zoom + increment;
1912 pr_zoom_sync(pr, zoom, flags, x, y);
1916 *-------------------------------------------------------------------
1918 *-------------------------------------------------------------------
1921 static ImageTile *pr_tile_new(gint x, gint y, gint width, gint height)
1925 it = g_new0(ImageTile, 1);
1932 it->render_done = TILE_RENDER_NONE;
1937 static void pr_tile_free(ImageTile *it)
1941 if (it->pixbuf) g_object_unref(it->pixbuf);
1942 if (it->pixmap) g_object_unref(it->pixmap);
1947 static void pr_tile_free_all(PixbufRenderer *pr)
1962 g_list_free(pr->tiles);
1964 pr->tile_cache_size = 0;
1967 static ImageTile *pr_tile_add(PixbufRenderer *pr, gint x, gint y)
1971 it = pr_tile_new(x, y, pr->tile_width, pr->tile_height);
1973 if (it->x + it->w > pr->width) it->w = pr->width - it->x;
1974 if (it->y + it->h > pr->height) it->h = pr->height - it->y;
1976 pr->tiles = g_list_prepend(pr->tiles, it);
1977 pr->tile_cache_size += it->size;
1982 static void pr_tile_remove(PixbufRenderer *pr, ImageTile *it)
1986 QueueData *qd = it->qd;
1989 pr->draw_queue = g_list_remove(pr->draw_queue, qd);
1995 QueueData *qd = it->qd2;
1998 pr->draw_queue_2pass = g_list_remove(pr->draw_queue_2pass, qd);
2002 pr->tiles = g_list_remove(pr->tiles, it);
2003 pr->tile_cache_size -= it->size;
2008 static void pr_tile_free_space(PixbufRenderer *pr, guint space, ImageTile *it)
2013 work = g_list_last(pr->tiles);
2015 if (pr->source_tiles_enabled && pr->scale < 1.0)
2019 tiles = (pr->vis_width / pr->tile_width + 1) * (pr->vis_height / pr->tile_height + 1);
2020 tile_max = MAX(tiles * pr->tile_width * pr->tile_height * 3,
2021 (gint)((gdouble)pr->tile_cache_max * 1048576.0 * pr->scale));
2025 tile_max = pr->tile_cache_max * 1048576;
2028 while (work && pr->tile_cache_size + space > tile_max)
2032 needle = work->data;
2035 ((!needle->qd && !needle->qd2) || !pr_tile_is_visible(pr, needle))) pr_tile_remove(pr, needle);
2039 static void pr_tile_invalidate_all(PixbufRenderer *pr)
2051 it->render_done = TILE_RENDER_NONE;
2052 it->render_todo = TILE_RENDER_ALL;
2055 it->w = MIN(pr->tile_width, pr->width - it->x);
2056 it->h = MIN(pr->tile_height, pr->height - it->y);
2060 static void pr_tile_invalidate_region(PixbufRenderer *pr, gint x, gint y, gint w, gint h)
2066 x1 = ROUND_DOWN(x, pr->tile_width);
2067 x2 = ROUND_UP(x + w, pr->tile_width);
2069 y1 = ROUND_DOWN(y, pr->tile_height);
2070 y2 = ROUND_UP(y + h, pr->tile_height);
2080 if (it->x < x2 && it->x + it->w > x1 &&
2081 it->y < y2 && it->y + it->h > y1)
2083 it->render_done = TILE_RENDER_NONE;
2084 it->render_todo = TILE_RENDER_ALL;
2089 static ImageTile *pr_tile_get(PixbufRenderer *pr, gint x, gint y, gboolean only_existing)
2099 if (it->x == x && it->y == y)
2101 pr->tiles = g_list_delete_link(pr->tiles, work);
2102 pr->tiles = g_list_prepend(pr->tiles, it);
2109 if (only_existing) return NULL;
2111 return pr_tile_add(pr, x, y);
2114 static void pr_tile_prepare(PixbufRenderer *pr, ImageTile *it)
2121 pixmap = gdk_pixmap_new(((GtkWidget *)pr)->window, pr->tile_width, pr->tile_height, -1);
2123 size = pixmap_calc_size(pixmap);
2124 pr_tile_free_space(pr, size, it);
2126 it->pixmap = pixmap;
2128 pr->tile_cache_size += size;
2131 if ((pr->zoom != 1.0 || pr->source_tiles_enabled || (pr->pixbuf && gdk_pixbuf_get_has_alpha(pr->pixbuf)) ||
2132 pr->orientation != EXIF_ORIENTATION_TOP_LEFT || pr->func_post_process) && !it->pixbuf)
2137 /* I don't think that we need a pixbuf with alpha channel here */
2140 pixbuf = gdk_pixbuf_new(gdk_pixbuf_get_colorspace(pr->pixbuf),
2141 gdk_pixbuf_get_has_alpha(pr->pixbuf),
2142 gdk_pixbuf_get_bits_per_sample(pr->pixbuf),
2143 pr->tile_width, pr->tile_height);
2148 pixbuf = gdk_pixbuf_new(GDK_COLORSPACE_RGB, FALSE, 8, pr->tile_width, pr->tile_height);
2151 size = gdk_pixbuf_get_rowstride(pixbuf) * pr->tile_height;
2152 pr_tile_free_space(pr, size, it);
2154 it->pixbuf = pixbuf;
2156 pr->tile_cache_size += size;
2161 *-------------------------------------------------------------------
2163 *-------------------------------------------------------------------
2167 static void pr_tile_coords_map_orientation(PixbufRenderer *pr,
2168 gdouble tile_x, gdouble tile_y, /* coordinates of the tile */
2169 gdouble image_w, gdouble image_h,
2170 gdouble tile_w, gdouble tile_h,
2171 gdouble *res_x, gdouble *res_y)
2175 switch (pr->orientation)
2177 case EXIF_ORIENTATION_TOP_LEFT:
2178 /* normal -- nothing to do */
2180 case EXIF_ORIENTATION_TOP_RIGHT:
2182 *res_x = image_w - tile_x - tile_w;
2184 case EXIF_ORIENTATION_BOTTOM_RIGHT:
2186 *res_x = image_w - tile_x - tile_w;
2187 *res_y = image_h - tile_y - tile_h;
2189 case EXIF_ORIENTATION_BOTTOM_LEFT:
2191 *res_y = image_h - tile_y - tile_h;
2193 case EXIF_ORIENTATION_LEFT_TOP:
2197 case EXIF_ORIENTATION_RIGHT_TOP:
2198 /* rotated -90 (270) */
2200 *res_y = image_w - tile_x - tile_w;
2202 case EXIF_ORIENTATION_RIGHT_BOTTOM:
2203 *res_x = image_h - tile_y - tile_h;
2204 *res_y = image_w - tile_x - tile_w;
2206 case EXIF_ORIENTATION_LEFT_BOTTOM:
2208 *res_x = image_h - tile_y - tile_h;
2212 /* The other values are out of range */
2215 // log_printf("tile coord y:%f, ih:%d, th:%f ry:%f\n", tile_y, image_h, tile_h, *res_x);
2218 static void pr_tile_region_map_orientation(PixbufRenderer *pr,
2219 gint area_x, gint area_y, /* coordinates of the area inside tile */
2220 gint tile_w, gint tile_h,
2221 gint area_w, gint area_h,
2222 gint *res_x, gint *res_y,
2223 gint *res_w, gint *res_h)
2230 switch (pr->orientation)
2232 case EXIF_ORIENTATION_TOP_LEFT:
2233 /* normal -- nothing to do */
2235 case EXIF_ORIENTATION_TOP_RIGHT:
2237 *res_x = tile_w - area_x - area_w;
2239 case EXIF_ORIENTATION_BOTTOM_RIGHT:
2241 *res_x = tile_w - area_x - area_w;
2242 *res_y = tile_h - area_y - area_h;
2244 case EXIF_ORIENTATION_BOTTOM_LEFT:
2246 *res_y = tile_h - area_y - area_h;
2248 case EXIF_ORIENTATION_LEFT_TOP:
2254 case EXIF_ORIENTATION_RIGHT_TOP:
2255 /* rotated -90 (270) */
2257 *res_y = tile_w - area_x - area_w;
2261 case EXIF_ORIENTATION_RIGHT_BOTTOM:
2262 *res_x = tile_h - area_y - area_h;
2263 *res_y = tile_w - area_x - area_w;
2267 case EXIF_ORIENTATION_LEFT_BOTTOM:
2269 *res_x = tile_h - area_y - area_h;
2275 /* The other values are out of range */
2278 // log_printf("inside y:%d, th:%d, ah:%d ry:%d\n", area_y, tile_h, area_h, *res_x);
2281 static void pr_coords_map_orientation_reverse(PixbufRenderer *pr,
2282 gint area_x, gint area_y,
2283 gint tile_w, gint tile_h,
2284 gint area_w, gint area_h,
2285 gint *res_x, gint *res_y,
2286 gint *res_w, gint *res_h)
2293 switch (pr->orientation)
2295 case EXIF_ORIENTATION_TOP_LEFT:
2296 /* normal -- nothing to do */
2298 case EXIF_ORIENTATION_TOP_RIGHT:
2300 *res_x = tile_w - area_x - area_w;
2302 case EXIF_ORIENTATION_BOTTOM_RIGHT:
2304 *res_x = tile_w - area_x - area_w;
2305 *res_y = tile_h - area_y - area_h;
2307 case EXIF_ORIENTATION_BOTTOM_LEFT:
2309 *res_y = tile_h - area_y - area_h;
2311 case EXIF_ORIENTATION_LEFT_TOP:
2317 case EXIF_ORIENTATION_RIGHT_TOP:
2318 /* rotated -90 (270) */
2319 *res_x = tile_w - area_y - area_h;
2324 case EXIF_ORIENTATION_RIGHT_BOTTOM:
2325 *res_x = tile_w - area_y - area_h;
2326 *res_y = tile_h - area_x - area_w;
2330 case EXIF_ORIENTATION_LEFT_BOTTOM:
2333 *res_y = tile_h - area_x - area_w;
2338 /* The other values are out of range */
2344 static GdkPixbuf *pr_get_spare_tile(PixbufRenderer *pr)
2346 if (!pr->spare_tile) pr->spare_tile = gdk_pixbuf_new(GDK_COLORSPACE_RGB, FALSE, 8, pr->tile_width, pr->tile_height);
2347 return pr->spare_tile;
2350 #define COLOR_BYTES 3 /* rgb */
2352 static void pr_tile_rotate_90_clockwise(PixbufRenderer *pr, GdkPixbuf **tile, gint x, gint y, gint w, gint h)
2354 GdkPixbuf *src = *tile;
2357 guchar *s_pix, *d_pix;
2359 guchar *ip, *spi, *dpi;
2361 gint tw = pr->tile_width;
2363 srs = gdk_pixbuf_get_rowstride(src);
2364 s_pix = gdk_pixbuf_get_pixels(src);
2365 spi = s_pix + (x * COLOR_BYTES);
2367 dest = pr_get_spare_tile(pr);
2368 drs = gdk_pixbuf_get_rowstride(dest);
2369 d_pix = gdk_pixbuf_get_pixels(dest);
2370 dpi = d_pix + (tw - 1) * COLOR_BYTES;
2372 for (i = y; i < y + h; i++)
2374 sp = spi + (i * srs);
2375 ip = dpi - (i * COLOR_BYTES);
2376 for (j = x; j < x + w; j++)
2378 dp = ip + (j * drs);
2379 memcpy(dp, sp, COLOR_BYTES);
2384 pr->spare_tile = src;
2388 static void pr_tile_rotate_90_counter_clockwise(PixbufRenderer *pr, GdkPixbuf **tile, gint x, gint y, gint w, gint h)
2390 GdkPixbuf *src = *tile;
2393 guchar *s_pix, *d_pix;
2395 guchar *ip, *spi, *dpi;
2397 gint th = pr->tile_height;
2399 srs = gdk_pixbuf_get_rowstride(src);
2400 s_pix = gdk_pixbuf_get_pixels(src);
2401 spi = s_pix + (x * COLOR_BYTES);
2403 dest = pr_get_spare_tile(pr);
2404 drs = gdk_pixbuf_get_rowstride(dest);
2405 d_pix = gdk_pixbuf_get_pixels(dest);
2406 dpi = d_pix + (th - 1) * drs;
2408 for (i = y; i < y + h; i++)
2410 sp = spi + (i * srs);
2411 ip = dpi + (i * COLOR_BYTES);
2412 for (j = x; j < x + w; j++)
2414 dp = ip - (j * drs);
2415 memcpy(dp, sp, COLOR_BYTES);
2420 pr->spare_tile = src;
2424 static void pr_tile_mirror_only(PixbufRenderer *pr, GdkPixbuf **tile, gint x, gint y, gint w, gint h)
2426 GdkPixbuf *src = *tile;
2429 guchar *s_pix, *d_pix;
2434 gint tw = pr->tile_width;
2436 srs = gdk_pixbuf_get_rowstride(src);
2437 s_pix = gdk_pixbuf_get_pixels(src);
2438 spi = s_pix + (x * COLOR_BYTES);
2440 dest = pr_get_spare_tile(pr);
2441 drs = gdk_pixbuf_get_rowstride(dest);
2442 d_pix = gdk_pixbuf_get_pixels(dest);
2443 dpi = d_pix + (tw - x - 1) * COLOR_BYTES;
2445 for (i = y; i < y + h; i++)
2447 sp = spi + (i * srs);
2448 dp = dpi + (i * drs);
2449 for (j = 0; j < w; j++)
2451 memcpy(dp, sp, COLOR_BYTES);
2457 pr->spare_tile = src;
2461 static void pr_tile_mirror_and_flip(PixbufRenderer *pr, GdkPixbuf **tile, gint x, gint y, gint w, gint h)
2463 GdkPixbuf *src = *tile;
2466 guchar *s_pix, *d_pix;
2470 gint tw = pr->tile_width;
2471 gint th = pr->tile_height;
2473 srs = gdk_pixbuf_get_rowstride(src);
2474 s_pix = gdk_pixbuf_get_pixels(src);
2475 spi = s_pix + (x * COLOR_BYTES);
2477 dest = pr_get_spare_tile(pr);
2478 drs = gdk_pixbuf_get_rowstride(dest);
2479 d_pix = gdk_pixbuf_get_pixels(dest);
2480 dpi = d_pix + (th - 1) * drs + (tw - 1) * COLOR_BYTES;
2482 for (i = y; i < y + h; i++)
2484 sp = s_pix + (i * srs) + (x * COLOR_BYTES);
2485 dp = dpi - (i * drs) - (x * COLOR_BYTES);
2486 for (j = 0; j < w; j++)
2488 memcpy(dp, sp, COLOR_BYTES);
2494 pr->spare_tile = src;
2498 static void pr_tile_flip_only(PixbufRenderer *pr, GdkPixbuf **tile, gint x, gint y, gint w, gint h)
2500 GdkPixbuf *src = *tile;
2503 guchar *s_pix, *d_pix;
2507 gint th = pr->tile_height;
2509 srs = gdk_pixbuf_get_rowstride(src);
2510 s_pix = gdk_pixbuf_get_pixels(src);
2511 spi = s_pix + (x * COLOR_BYTES);
2513 dest = pr_get_spare_tile(pr);
2514 drs = gdk_pixbuf_get_rowstride(dest);
2515 d_pix = gdk_pixbuf_get_pixels(dest);
2516 dpi = d_pix + (th - 1) * drs + (x * COLOR_BYTES);
2518 for (i = y; i < y + h; i++)
2520 sp = spi + (i * srs);
2521 dp = dpi - (i * drs);
2522 memcpy(dp, sp, w * COLOR_BYTES);
2525 pr->spare_tile = src;
2529 static void pr_tile_apply_orientation(PixbufRenderer *pr, GdkPixbuf **pixbuf, gint x, gint y, gint w, gint h)
2531 switch (pr->orientation)
2533 case EXIF_ORIENTATION_TOP_LEFT:
2534 /* normal -- nothing to do */
2536 case EXIF_ORIENTATION_TOP_RIGHT:
2539 pr_tile_mirror_only(pr, pixbuf, x, y, w, h);
2542 case EXIF_ORIENTATION_BOTTOM_RIGHT:
2545 pr_tile_mirror_and_flip(pr, pixbuf, x, y, w, h);
2548 case EXIF_ORIENTATION_BOTTOM_LEFT:
2551 pr_tile_flip_only(pr, pixbuf, x, y, w, h);
2554 case EXIF_ORIENTATION_LEFT_TOP:
2556 pr_tile_flip_only(pr, pixbuf, x, y, w, h);
2557 pr_tile_rotate_90_clockwise(pr, pixbuf, x, pr->tile_height - y - h, w, h);
2560 case EXIF_ORIENTATION_RIGHT_TOP:
2561 /* rotated -90 (270) */
2563 pr_tile_rotate_90_clockwise(pr, pixbuf, x, y, w, h);
2566 case EXIF_ORIENTATION_RIGHT_BOTTOM:
2568 pr_tile_flip_only(pr, pixbuf, x, y, w, h);
2569 pr_tile_rotate_90_counter_clockwise(pr, pixbuf, x, pr->tile_height - y - h, w, h);
2572 case EXIF_ORIENTATION_LEFT_BOTTOM:
2575 pr_tile_rotate_90_counter_clockwise(pr, pixbuf, x, y, w, h);
2579 /* The other values are out of range */
2585 static void pr_tile_render(PixbufRenderer *pr, ImageTile *it,
2586 gint x, gint y, gint w, gint h,
2587 gboolean new_data, gboolean fast)
2591 gboolean draw = FALSE;
2593 if (it->render_todo == TILE_RENDER_NONE && it->pixmap && !new_data) return;
2595 if (it->render_done != TILE_RENDER_ALL)
2601 if (!fast) it->render_done = TILE_RENDER_ALL;
2603 else if (it->render_todo != TILE_RENDER_AREA)
2605 if (!fast) it->render_todo = TILE_RENDER_NONE;
2609 if (!fast) it->render_todo = TILE_RENDER_NONE;
2611 if (new_data) it->blank = FALSE;
2613 pr_tile_prepare(pr, it);
2614 has_alpha = (pr->pixbuf && gdk_pixbuf_get_has_alpha(pr->pixbuf));
2616 box = GTK_WIDGET(pr);
2618 /* FIXME checker colors for alpha should be configurable,
2619 * also should be drawn for blank = TRUE
2624 /* no data, do fast rect fill */
2625 gdk_draw_rectangle(it->pixmap, box->style->black_gc, TRUE,
2626 0, 0, it->w, it->h);
2628 else if (pr->source_tiles_enabled)
2630 draw = pr_source_tile_render(pr, it, x, y, w, h, new_data, fast);
2632 else if (pr->zoom == 1.0 || pr->scale == 1.0)
2635 gdouble src_x, src_y;
2638 pr_tile_coords_map_orientation(pr, it->x, it->y,
2639 pr->image_width, pr->image_height,
2640 pr->tile_width, pr->tile_height,
2642 pr_tile_region_map_orientation(pr, x, y,
2643 pr->tile_width, pr->tile_height,
2650 gdk_pixbuf_composite_color(pr->pixbuf, it->pixbuf, pb_x, pb_y, pb_w, pb_h,
2651 (gdouble) 0.0 - src_x,
2652 (gdouble) 0.0 - src_y,
2653 1.0, 1.0, GDK_INTERP_NEAREST,
2654 255, it->x + pb_x, it->y + pb_y,
2655 PR_ALPHA_CHECK_SIZE, PR_ALPHA_CHECK1, PR_ALPHA_CHECK2);
2656 pr_tile_apply_orientation(pr, &it->pixbuf, pb_x, pb_y, pb_w, pb_h);
2663 if (pr->orientation == EXIF_ORIENTATION_TOP_LEFT && !(pr->func_post_process && !(pr->post_process_slow && fast)))
2665 /* faster, simple, base orientation, no postprocessing */
2666 gdk_draw_pixbuf(it->pixmap,
2667 #if GTK_CHECK_VERSION(2,20,0)
2668 box->style->fg_gc[gtk_widget_get_state(box)],
2670 box->style->fg_gc[GTK_WIDGET_STATE(box)],
2673 it->x + x, it->y + y,
2676 pr->dither_quality, it->x + x, it->y + y);
2680 gdk_pixbuf_copy_area(pr->pixbuf,
2681 src_x + pb_x, src_y + pb_y,
2685 pr_tile_apply_orientation(pr, &it->pixbuf, pb_x, pb_y, pb_w, pb_h);
2692 gdouble scale_x, scale_y;
2693 gdouble src_x, src_y;
2697 if (pr->image_width == 0 || pr->image_height == 0) return;
2699 scale_x = (gdouble)pr->width / pr->image_width;
2700 scale_y = (gdouble)pr->height / pr->image_height;
2702 pr_tile_coords_map_orientation(pr, it->x, it->y,
2703 pr->image_width * scale_x, pr->image_height * scale_y,
2704 pr->tile_width, pr->tile_height,
2706 pr_tile_region_map_orientation(pr, x, y,
2707 pr->tile_width, pr->tile_height,
2711 switch (pr->orientation)
2714 case EXIF_ORIENTATION_LEFT_TOP:
2715 case EXIF_ORIENTATION_RIGHT_TOP:
2716 case EXIF_ORIENTATION_RIGHT_BOTTOM:
2717 case EXIF_ORIENTATION_LEFT_BOTTOM:
2727 /* HACK: The pixbuf scalers get kinda buggy(crash) with extremely
2728 * small sizes for anything but GDK_INTERP_NEAREST
2730 if (pr->width < PR_MIN_SCALE_SIZE || pr->height < PR_MIN_SCALE_SIZE) fast = TRUE;
2734 gdk_pixbuf_scale(pr->pixbuf, it->pixbuf, pb_x, pb_y, pb_w, pb_h,
2735 (gdouble) 0.0 - src_x,
2736 (gdouble) 0.0 - src_y,
2738 (fast) ? GDK_INTERP_NEAREST : pr->zoom_quality);
2742 gdk_pixbuf_composite_color(pr->pixbuf, it->pixbuf, pb_x, pb_y, pb_w, pb_h,
2743 (gdouble) 0.0 - src_x,
2744 (gdouble) 0.0 - src_y,
2746 (fast) ? GDK_INTERP_NEAREST : pr->zoom_quality,
2747 255, it->x + pb_x, it->y + pb_y,
2748 PR_ALPHA_CHECK_SIZE, PR_ALPHA_CHECK1, PR_ALPHA_CHECK2);
2750 pr_tile_apply_orientation(pr, &it->pixbuf, pb_x, pb_y, pb_w, pb_h);
2754 if (draw && it->pixbuf && !it->blank)
2757 if (pr->func_post_process && !(pr->post_process_slow && fast))
2758 pr->func_post_process(pr, &it->pixbuf, x, y, w, h, pr->post_process_user_data);
2760 gdk_draw_pixbuf(it->pixmap,
2761 #if GTK_CHECK_VERSION(2,20,0)
2762 box->style->fg_gc[gtk_widget_get_state(box)],
2764 box->style->fg_gc[GTK_WIDGET_STATE(box)],
2770 pr->dither_quality, it->x + x, it->y + y);
2774 /* enable this line for debugging the edges of tiles */
2775 gdk_draw_rectangle(it->pixmap, box->style->white_gc,
2776 FALSE, 0, 0, it->w, it->h);
2777 gdk_draw_rectangle(it->pixmap, box->style->white_gc,
2783 static void pr_tile_expose(PixbufRenderer *pr, ImageTile *it,
2784 gint x, gint y, gint w, gint h,
2785 gboolean new_data, gboolean fast)
2789 pr_tile_render(pr, it, x, y, w, h, new_data, fast);
2791 box = GTK_WIDGET(pr);
2793 #if GTK_CHECK_VERSION(2,20,0)
2794 gdk_draw_drawable(box->window, box->style->fg_gc[gtk_widget_get_state(box)],
2796 gdk_draw_drawable(box->window, box->style->fg_gc[GTK_WIDGET_STATE(box)],
2799 pr->x_offset + (it->x - pr->x_scroll) + x, pr->y_offset + (it->y - pr->y_scroll) + y, w, h);
2801 if (pr->overlay_list)
2803 pr_overlay_draw(pr, pr->x_offset + (it->x - pr->x_scroll) + x,
2804 pr->y_offset + (it->y - pr->y_scroll) + y,
2811 static gboolean pr_tile_is_visible(PixbufRenderer *pr, ImageTile *it)
2813 return (it->x + it->w >= pr->x_scroll && it->x < pr->x_scroll + pr->vis_width &&
2814 it->y + it->h >= pr->y_scroll && it->y < pr->y_scroll + pr->vis_height);
2818 *-------------------------------------------------------------------
2820 *-------------------------------------------------------------------
2823 static gint pr_get_queued_area(GList *work)
2829 QueueData *qd = work->data;
2830 area += qd->w * qd->h;
2837 static gboolean pr_queue_schedule_next_draw(PixbufRenderer *pr, gboolean force_set)
2840 gint visible_area = pr->vis_width * pr->vis_height;
2845 DEBUG_2("redraw priority: 2pass");
2846 pr->draw_idle_id = g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, pr_queue_draw_idle_cb, pr, NULL);
2850 if (visible_area == 0)
2857 percent = 100.0 * pr_get_queued_area(pr->draw_queue) / visible_area;
2862 /* we have enough data for starting intensive redrawing */
2863 DEBUG_2("redraw priority: high %.2f %%", percent);
2864 pr->draw_idle_id = g_idle_add_full(GDK_PRIORITY_REDRAW, pr_queue_draw_idle_cb, pr, NULL);
2868 if (percent < 1.0 || force_set)
2870 /* queue is (almost) empty, wait 50 ms*/
2871 DEBUG_2("redraw priority: wait %.2f %%", percent);
2872 pr->draw_idle_id = g_timeout_add_full(G_PRIORITY_DEFAULT_IDLE, 50, pr_queue_draw_idle_cb, pr, NULL);
2876 /* keep the same priority as before */
2877 DEBUG_2("redraw priority: no change %.2f %%", percent);
2882 static gboolean pr_queue_draw_idle_cb(gpointer data)
2884 PixbufRenderer *pr = data;
2889 if ((!pr->pixbuf && !pr->source_tiles_enabled) ||
2890 (!pr->draw_queue && !pr->draw_queue_2pass) ||
2893 pr_render_complete_signal(pr);
2895 pr->draw_idle_id = 0;
2901 qd = pr->draw_queue->data;
2902 fast = (pr->zoom_2pass && ((pr->zoom_quality != GDK_INTERP_NEAREST && pr->scale != 1.0) || pr->post_process_slow));
2908 /* still loading, wait till done (also drops the higher priority) */
2910 return pr_queue_schedule_next_draw(pr, FALSE);
2913 qd = pr->draw_queue_2pass->data;
2917 #if GTK_CHECK_VERSION(2,20,0)
2918 if (gtk_widget_get_realized(pr))
2920 if (GTK_WIDGET_REALIZED(pr))
2923 if (pr_tile_is_visible(pr, qd->it))
2925 pr_tile_expose(pr, qd->it, qd->x, qd->y, qd->w, qd->h, qd->new_data, fast);
2927 else if (qd->new_data)
2929 /* if new pixel data, and we already have a pixmap, update the tile */
2930 qd->it->blank = FALSE;
2931 if (qd->it->pixmap && qd->it->render_done == TILE_RENDER_ALL)
2933 pr_tile_render(pr, qd->it, qd->x, qd->y, qd->w, qd->h, qd->new_data, fast);
2941 pr->draw_queue = g_list_remove(pr->draw_queue, qd);
2946 pr_queue_merge(qd->it->qd2, qd);
2952 pr->draw_queue_2pass = g_list_append(pr->draw_queue_2pass, qd);
2963 pr->draw_queue_2pass = g_list_remove(pr->draw_queue_2pass, qd);
2967 if (!pr->draw_queue && !pr->draw_queue_2pass)
2969 pr_render_complete_signal(pr);
2971 pr->draw_idle_id = 0;
2975 return pr_queue_schedule_next_draw(pr, FALSE);
2978 static void pr_queue_list_free(GList *list)
2998 static void pr_queue_clear(PixbufRenderer *pr)
3000 pr_queue_list_free(pr->draw_queue);
3001 pr->draw_queue = NULL;
3003 pr_queue_list_free(pr->draw_queue_2pass);
3004 pr->draw_queue_2pass = NULL;
3006 if (pr->draw_idle_id)
3008 g_source_remove(pr->draw_idle_id);
3009 pr->draw_idle_id = 0;
3013 static void pr_queue_merge(QueueData *parent, QueueData *qd)
3015 if (parent->x + parent->w < qd->x + qd->w)
3017 parent->w += (qd->x + qd->w) - (parent->x + parent->w);
3019 if (parent->x > qd->x)
3021 parent->w += parent->x - qd->x;
3025 if (parent->y + parent->h < qd->y + qd->h)
3027 parent->h += (qd->y + qd->h) - (parent->y + parent->h);
3029 if (parent->y > qd->y)
3031 parent->h += parent->y - qd->y;
3035 parent->new_data |= qd->new_data;
3038 static gboolean pr_clamp_to_visible(PixbufRenderer *pr, gint *x, gint *y, gint *w, gint *h)
3046 vh = pr->vis_height;
3051 if (*x + *w < vx || *x > vx + vw || *y + *h < vy || *y > vy + vh) return FALSE;
3054 nx = CLAMP(*x, vx, vx + vw);
3055 nw = CLAMP(*w - (nx - *x), 1, vw);
3057 ny = CLAMP(*y, vy, vy + vh);
3058 nh = CLAMP(*h - (ny - *y), 1, vh);
3068 static gboolean pr_queue_to_tiles(PixbufRenderer *pr, gint x, gint y, gint w, gint h,
3069 gboolean clamp, ImageTileRenderType render,
3070 gboolean new_data, gboolean only_existing)
3076 if (clamp && !pr_clamp_to_visible(pr, &x, &y, &w, &h)) return FALSE;
3078 x1 = ROUND_DOWN(x, pr->tile_width);
3079 x2 = ROUND_UP(x + w, pr->tile_width);
3081 y1 = ROUND_DOWN(y, pr->tile_height);
3082 y2 = ROUND_UP(y + h, pr->tile_height);
3084 for (j = y1; j <= y2; j += pr->tile_height)
3086 for (i = x1; i <= x2; i += pr->tile_width)
3090 it = pr_tile_get(pr, i, j,
3092 (i + pr->tile_width < pr->x_scroll ||
3093 i > pr->x_scroll + pr->vis_width ||
3094 j + pr->tile_height < pr->y_scroll ||
3095 j > pr->y_scroll + pr->vis_height)));
3100 if ((render == TILE_RENDER_ALL && it->render_done != TILE_RENDER_ALL) ||
3101 (render == TILE_RENDER_AREA && it->render_todo != TILE_RENDER_ALL))
3103 it->render_todo = render;
3106 qd = g_new(QueueData, 1);
3108 qd->new_data = new_data;
3118 qd->w = x + w - i - qd->x;
3119 if (qd->x + qd->w > pr->tile_width) qd->w = pr->tile_width - qd->x;
3129 qd->h = y + h - j - qd->y;
3130 if (qd->y + qd->h > pr->tile_height) qd->h = pr->tile_height - qd->y;
3132 if (qd->w < 1 || qd->h < 1)
3138 pr_queue_merge(it->qd, qd);
3144 pr->draw_queue = g_list_append(pr->draw_queue, qd);
3153 static void pr_queue(PixbufRenderer *pr, gint x, gint y, gint w, gint h,
3154 gboolean clamp, ImageTileRenderType render,
3155 gboolean new_data, gboolean only_existing)
3159 nx = CLAMP(x, 0, pr->width - 1);
3160 ny = CLAMP(y, 0, pr->height - 1);
3163 w = CLAMP(w, 0, pr->width - nx);
3164 h = CLAMP(h, 0, pr->height - ny);
3165 if (w < 1 || h < 1) return;
3167 if (pr_queue_to_tiles(pr, nx, ny, w, h, clamp, render, new_data, only_existing) &&
3168 ((!pr->draw_queue && !pr->draw_queue_2pass) || !pr->draw_idle_id))
3170 if (pr->draw_idle_id)
3172 g_source_remove(pr->draw_idle_id);
3173 pr->draw_idle_id = 0;
3175 pr_queue_schedule_next_draw(pr, TRUE);
3179 static void pr_redraw(PixbufRenderer *pr, gboolean new_data)
3182 pr_queue(pr, 0, 0, pr->width, pr->height, TRUE, TILE_RENDER_ALL, new_data, FALSE);
3186 *-------------------------------------------------------------------
3188 *-------------------------------------------------------------------
3191 static void pr_update_signal(PixbufRenderer *pr)
3194 log_printf("FIXME: send updated signal\n");
3196 DEBUG_1("%s pixbuf renderer updated - started drawing %p, img: %dx%d", get_exec_time(), pr, pr->image_width, pr->image_height);
3197 pr->debug_updated = TRUE;
3200 static void pr_zoom_signal(PixbufRenderer *pr)
3202 g_signal_emit(pr, signals[SIGNAL_ZOOM], 0, pr->zoom);
3205 static void pr_clicked_signal(PixbufRenderer *pr, GdkEventButton *bevent)
3207 g_signal_emit(pr, signals[SIGNAL_CLICKED], 0, bevent);
3210 static void pr_scroll_notify_signal(PixbufRenderer *pr)
3212 g_signal_emit(pr, signals[SIGNAL_SCROLL_NOTIFY], 0);
3215 static void pr_render_complete_signal(PixbufRenderer *pr)
3219 g_signal_emit(pr, signals[SIGNAL_RENDER_COMPLETE], 0);
3220 g_object_set(G_OBJECT(pr), "complete", TRUE, NULL);
3222 if (pr->debug_updated)
3224 DEBUG_1("%s pixbuf renderer done %p", get_exec_time(), pr);
3225 pr->debug_updated = FALSE;
3229 static void pr_drag_signal(PixbufRenderer *pr, GdkEventButton *bevent)
3231 g_signal_emit(pr, signals[SIGNAL_DRAG], 0, bevent);
3234 static void pr_update_pixel_signal(PixbufRenderer *pr)
3236 g_signal_emit(pr, signals[SIGNAL_UPDATE_PIXEL], 0);
3240 *-------------------------------------------------------------------
3242 *-------------------------------------------------------------------
3245 static void pixbuf_renderer_sync_scroll_center(PixbufRenderer *pr)
3248 if (!pr->width || !pr->height) return;
3251 * Update norm_center only if the image is bigger than the window.
3252 * With this condition the stored center survives also a temporary display
3253 * of the "broken image" icon.
3256 if (pr->width > pr->window_width)
3258 src_x = pr->x_scroll + pr->vis_width / 2;
3259 pr->norm_center_x = (gdouble)src_x / pr->width;
3262 if (pr->height > pr->window_height)
3264 src_y = pr->y_scroll + pr->vis_height / 2;
3265 pr->norm_center_y = (gdouble)src_y / pr->height;
3270 static gboolean pr_scroll_clamp(PixbufRenderer *pr)
3275 if (pr->zoom == 0.0)
3283 old_xs = pr->x_scroll;
3284 old_ys = pr->y_scroll;
3286 if (pr->x_offset > 0)
3292 pr->x_scroll = CLAMP(pr->x_scroll, 0, pr->width - pr->vis_width);
3295 if (pr->y_offset > 0)
3301 pr->y_scroll = CLAMP(pr->y_scroll, 0, pr->height - pr->vis_height);
3304 pixbuf_renderer_sync_scroll_center(pr);
3306 return (old_xs != pr->x_scroll || old_ys != pr->y_scroll);
3309 static gboolean pr_size_clamp(PixbufRenderer *pr)
3311 gint old_vw, old_vh;
3313 old_vw = pr->vis_width;
3314 old_vh = pr->vis_height;
3316 if (pr->width < pr->window_width)
3318 pr->vis_width = pr->width;
3319 pr->x_offset = (pr->window_width - pr->width) / 2;
3323 pr->vis_width = pr->window_width;
3327 if (pr->height < pr->window_height)
3329 pr->vis_height = pr->height;
3330 pr->y_offset = (pr->window_height - pr->height) / 2;
3334 pr->vis_height = pr->window_height;
3338 pixbuf_renderer_sync_scroll_center(pr);
3340 return (old_vw != pr->vis_width || old_vh != pr->vis_height);
3343 static gboolean pr_zoom_clamp(PixbufRenderer *pr, gdouble zoom,
3344 PrZoomFlags flags, gboolean *redrawn)
3349 gboolean force = !!(flags & PR_ZOOM_FORCE);
3350 gboolean new = !!(flags & PR_ZOOM_NEW);
3351 gboolean invalidate = !!(flags & PR_ZOOM_INVALIDATE);
3352 gboolean lazy = !!(flags & PR_ZOOM_LAZY);
3354 zoom = CLAMP(zoom, pr->zoom_min, pr->zoom_max);
3356 if (pr->zoom == zoom && !force) return FALSE;
3358 w = pr->image_width;
3359 h = pr->image_height;
3361 if (zoom == 0.0 && !pr->pixbuf)
3365 else if (zoom == 0.0)
3371 sizeable = (new && pr_parent_window_sizable(pr));
3375 max_w = gdk_screen_width();
3376 max_h = gdk_screen_height();
3378 if (pr->window_limit)
3380 max_w = max_w * pr->window_limit_size / 100;
3381 max_h = max_h * pr->window_limit_size / 100;
3386 max_w = pr->window_width;
3387 max_h = pr->window_height;
3390 if ((pr->zoom_expand && !sizeable) || w > max_w || h > max_h)
3392 if ((gdouble)max_w / w > (gdouble)max_h / h)
3394 scale = (gdouble)max_h / h;
3396 w = w * scale + 0.5;
3397 if (w > max_w) w = max_w;
3401 scale = (gdouble)max_w / w;
3403 h = h * scale + 0.5;
3404 if (h > max_h) h = max_h;
3407 if (pr->autofit_limit)
3409 gdouble factor = (gdouble)pr->autofit_limit_size / 100;
3410 w = w * factor + 0.5;
3411 h = h * factor + 0.5;
3412 scale = scale * factor;
3423 else if (zoom > 0.0) /* zoom orig, in */
3431 scale = 1.0 / (0.0 - zoom);
3436 invalid = (pr->width != w || pr->height != h);
3443 if (invalidate || invalid)
3445 pr_tile_invalidate_all(pr);
3446 if (!lazy) pr_redraw(pr, TRUE);
3448 if (redrawn) *redrawn = (invalidate || invalid);
3450 pixbuf_renderer_sync_scroll_center(pr);
3455 static void pr_zoom_sync(PixbufRenderer *pr, gdouble zoom,
3456 PrZoomFlags flags, gint px, gint py)
3459 gint old_cx, old_cy;
3462 gboolean redrawn = FALSE;
3463 gboolean center_point = !!(flags & PR_ZOOM_CENTER);
3464 gboolean force = !!(flags & PR_ZOOM_FORCE);
3465 gboolean new = !!(flags & PR_ZOOM_NEW);
3466 gboolean lazy = !!(flags & PR_ZOOM_LAZY);
3467 PrZoomFlags clamp_flags = flags;
3468 gdouble old_center_x = pr->norm_center_x;
3469 gdouble old_center_y = pr->norm_center_y;
3471 old_scale = pr->scale;
3474 px = CLAMP(px, 0, pr->width);
3475 py = CLAMP(py, 0, pr->height);
3476 old_cx = pr->x_scroll + (px - pr->x_offset);
3477 old_cy = pr->y_scroll + (py - pr->y_offset);
3482 old_cx = pr->x_scroll + pr->vis_width / 2;
3483 old_cy = pr->y_scroll + pr->vis_height / 2;
3486 if (force) clamp_flags |= PR_ZOOM_INVALIDATE;
3487 if (lazy) clamp_flags |= PR_ZOOM_LAZY;
3488 if (!pr_zoom_clamp(pr, zoom, clamp_flags, &redrawn)) return;
3490 clamped = pr_size_clamp(pr);
3491 sized = pr_parent_window_resize(pr, pr->width, pr->height);
3495 switch (pr->scroll_reset)
3497 case PR_SCROLL_RESET_NOCHANGE:
3498 /* maintain old scroll position */
3499 pr->x_scroll = ((gdouble)pr->image_width * old_center_x * pr->scale) - pr->vis_width / 2;
3500 pr->y_scroll = ((gdouble)pr->image_height * old_center_y * pr->scale) - pr->vis_height / 2;
3502 case PR_SCROLL_RESET_CENTER:
3503 /* center new image */
3504 pr->x_scroll = ((gdouble)pr->image_width / 2.0 * pr->scale) - pr->vis_width / 2;
3505 pr->y_scroll = ((gdouble)pr->image_height / 2.0 * pr->scale) - pr->vis_height / 2;
3507 case PR_SCROLL_RESET_TOPLEFT:
3509 /* reset to upper left */
3517 /* user zoom does not force, so keep visible center point */
3520 pr->x_scroll = old_cx / old_scale * pr->scale - (px - pr->x_offset);
3521 pr->y_scroll = old_cy / old_scale * pr->scale - (py - pr->y_offset);
3525 pr->x_scroll = old_cx / old_scale * pr->scale - (pr->vis_width / 2);
3526 pr->y_scroll = old_cy / old_scale * pr->scale - (pr->vis_height / 2);
3530 pr_scroll_clamp(pr);
3532 /* If the window was not sized, redraw the image - we know there will be no size/expose signal.
3533 * But even if a size is claimed, there is no guarantee that the window manager will allow it,
3534 * so redraw the window anyway :/
3536 if (sized || clamped) pr_border_clear(pr);
3544 pr_redraw(pr, redrawn);
3547 pr_scroll_notify_signal(pr);
3549 pr_update_signal(pr);
3552 static void pr_size_sync(PixbufRenderer *pr, gint new_width, gint new_height)
3554 gboolean zoom_changed = FALSE;
3556 if (pr->window_width == new_width && pr->window_height == new_height) return;
3558 pr->window_width = new_width;
3559 pr->window_height = new_height;
3561 if (pr->zoom == 0.0)
3563 gdouble old_scale = pr->scale;
3564 pr_zoom_clamp(pr, 0.0, PR_ZOOM_FORCE, NULL);
3565 zoom_changed = (old_scale != pr->scale);
3569 pr_scroll_clamp(pr);
3571 pr_overlay_update_sizes(pr);
3573 /* ensure scroller remains visible */
3574 if (pr->scroller_overlay != -1)
3576 gboolean update = FALSE;
3578 if (pr->scroller_x > new_width)
3580 pr->scroller_x = new_width;
3581 pr->scroller_xpos = new_width;
3584 if (pr->scroller_y > new_height)
3586 pr->scroller_y = new_height;
3587 pr->scroller_ypos = new_height;
3595 if (pixbuf_renderer_overlay_get(pr, pr->scroller_overlay, &pixbuf, NULL, NULL))
3599 w = gdk_pixbuf_get_width(pixbuf);
3600 h = gdk_pixbuf_get_height(pixbuf);
3601 pixbuf_renderer_overlay_set(pr, pr->scroller_overlay, pixbuf,
3602 pr->scroller_x - w / 2, pr->scroller_y - h / 2);
3607 pr_border_clear(pr);
3609 pr_scroll_notify_signal(pr);
3610 if (zoom_changed) pr_zoom_signal(pr);
3611 pr_update_signal(pr);
3614 static void pr_size_cb(GtkWidget *widget, GtkAllocation *allocation, gpointer data)
3616 PixbufRenderer *pr = data;
3618 pr_size_sync(pr, allocation->width, allocation->height);
3621 static void pixbuf_renderer_paint(PixbufRenderer *pr, GdkRectangle *area)
3625 pr_border_draw(pr, area->x, area->y, area->width, area->height);
3627 x = MAX(0, (gint)area->x - pr->x_offset + pr->x_scroll);
3628 y = MAX(0, (gint)area->y - pr->y_offset + pr->y_scroll);
3631 MIN((gint)area->width, pr->width - x),
3632 MIN((gint)area->height, pr->height - y),
3633 FALSE, TILE_RENDER_ALL, FALSE, FALSE);
3637 *-------------------------------------------------------------------
3639 *-------------------------------------------------------------------
3642 void pixbuf_renderer_scroll(PixbufRenderer *pr, gint x, gint y)
3648 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
3650 if (!pr->pixbuf && !pr->source_tiles_enabled) return;
3652 old_x = pr->x_scroll;
3653 old_y = pr->y_scroll;
3658 pr_scroll_clamp(pr);
3660 pixbuf_renderer_sync_scroll_center(pr);
3662 if (pr->x_scroll == old_x && pr->y_scroll == old_y) return;
3664 pr_scroll_notify_signal(pr);
3666 x_off = pr->x_scroll - old_x;
3667 y_off = pr->y_scroll - old_y;
3669 w = pr->vis_width - abs(x_off);
3670 h = pr->vis_height - abs(y_off);
3674 /* scrolled completely to new material */
3675 pr_queue(pr, 0, 0, pr->width, pr->height, TRUE, TILE_RENDER_ALL, FALSE, FALSE);
3708 box = GTK_WIDGET(pr);
3710 gc = gdk_gc_new(box->window);
3711 gdk_gc_set_exposures(gc, TRUE);
3712 gdk_draw_drawable(box->window, gc,
3714 x2 + pr->x_offset, y2 + pr->y_offset,
3715 x1 + pr->x_offset, y1 + pr->y_offset, w, h);
3718 if (pr->overlay_list)
3720 pr_overlay_queue_all(pr);
3723 w = pr->vis_width - w;
3724 h = pr->vis_height - h;
3729 x_off > 0 ? pr->x_scroll + (pr->vis_width - w) : pr->x_scroll, pr->y_scroll,
3730 w, pr->vis_height, TRUE, TILE_RENDER_ALL, FALSE, FALSE);
3734 /* FIXME, to optimize this, remove overlap */
3736 pr->x_scroll, y_off > 0 ? pr->y_scroll + (pr->vis_height - h) : pr->y_scroll,
3737 pr->vis_width, h, TRUE, TILE_RENDER_ALL, FALSE, FALSE);
3740 /* process exposures here, "expose_event" seems to miss a few with obstructed windows */
3741 #if ! GTK_CHECK_VERSION(2,18,0)
3742 while ((event = gdk_event_get_graphics_expose(box->window)) != NULL)
3744 pixbuf_renderer_paint(pr, &event->expose.area);
3746 if (event->expose.count == 0)
3748 gdk_event_free(event);
3751 gdk_event_free(event);
3757 void pixbuf_renderer_scroll_to_point(PixbufRenderer *pr, gint x, gint y,
3758 gdouble x_align, gdouble y_align)
3763 x_align = CLAMP(x_align, 0.0, 1.0);
3764 y_align = CLAMP(y_align, 0.0, 1.0);
3766 ax = (gdouble)pr->vis_width * x_align;
3767 ay = (gdouble)pr->vis_height * y_align;
3769 px = (gdouble)x * pr->scale - (pr->x_scroll + ax);
3770 py = (gdouble)y * pr->scale - (pr->y_scroll + ay);
3772 pixbuf_renderer_scroll(pr, px, py);
3775 /* get or set coordinates of viewport center in the image, in range 0.0 - 1.0 */
3777 void pixbuf_renderer_get_scroll_center(PixbufRenderer *pr, gdouble *x, gdouble *y)
3779 *x = pr->norm_center_x;
3780 *y = pr->norm_center_y;
3783 void pixbuf_renderer_set_scroll_center(PixbufRenderer *pr, gdouble x, gdouble y)
3785 gdouble dst_x, dst_y;
3787 dst_x = x * pr->width - pr->vis_width / 2 - pr->x_scroll + CLAMP(pr->subpixel_x_scroll, -1.0, 1.0);
3788 dst_y = y * pr->height - pr->vis_height / 2 - pr->y_scroll + CLAMP(pr->subpixel_y_scroll, -1.0, 1.0);
3790 pr->subpixel_x_scroll = dst_x - (gint)dst_x;
3791 pr->subpixel_y_scroll = dst_y - (gint)dst_y;
3793 pixbuf_renderer_scroll(pr, (gint)dst_x, (gint)dst_y);
3797 *-------------------------------------------------------------------
3799 *-------------------------------------------------------------------
3802 static gboolean pr_mouse_motion_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
3807 /* This is a hack, but work far the best, at least for single pointer systems.
3808 * See http://bugzilla.gnome.org/show_bug.cgi?id=587714 for more. */
3810 gdk_window_get_pointer (bevent->window, &x, &y, NULL);
3814 pr = PIXBUF_RENDERER(widget);
3816 if (pr->scroller_id)
3818 pr->scroller_xpos = bevent->x;
3819 pr->scroller_ypos = bevent->y;
3822 pr->x_mouse = bevent->x;
3823 pr->y_mouse = bevent->y;
3824 pr_update_pixel_signal(pr);
3826 if (!pr->in_drag || !gdk_pointer_is_grabbed()) return FALSE;
3828 if (pr->drag_moved < PR_DRAG_SCROLL_THRESHHOLD)
3834 widget_set_cursor(widget, GDK_FLEUR);
3837 if (bevent->state & GDK_CONTROL_MASK)
3839 accel = PR_PAN_SHIFT_MULTIPLIER;
3847 pixbuf_renderer_scroll(pr, (pr->drag_last_x - bevent->x) * accel,
3848 (pr->drag_last_y - bevent->y) * accel);
3850 pr_drag_signal(pr, bevent);
3852 pr->drag_last_x = bevent->x;
3853 pr->drag_last_y = bevent->y;
3855 /* This is recommended by the GTK+ documentation, but does not work properly.
3856 * Use deprecated way until GTK+ gets a solution for correct motion hint handling:
3857 * http://bugzilla.gnome.org/show_bug.cgi?id=587714
3859 /* gdk_event_request_motions (bevent); */
3863 static gboolean pr_leave_notify_cb(GtkWidget *widget, GdkEventCrossing *cevent, gpointer data)
3867 pr = PIXBUF_RENDERER(widget);
3871 pr_update_pixel_signal(pr);
3875 static gboolean pr_mouse_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
3880 pr = PIXBUF_RENDERER(widget);
3882 if (pr->scroller_id) return TRUE;
3884 switch (bevent->button)
3886 case MOUSE_BUTTON_LEFT:
3888 pr->drag_last_x = bevent->x;
3889 pr->drag_last_y = bevent->y;
3891 gdk_pointer_grab(widget->window, FALSE,
3892 GDK_POINTER_MOTION_MASK | GDK_POINTER_MOTION_HINT_MASK | GDK_BUTTON_RELEASE_MASK,
3893 NULL, NULL, bevent->time);
3894 gtk_grab_add(widget);
3896 case MOUSE_BUTTON_MIDDLE:
3899 case MOUSE_BUTTON_RIGHT:
3900 pr_clicked_signal(pr, bevent);
3906 parent = gtk_widget_get_parent(widget);
3907 #if GTK_CHECK_VERSION(2,20,0)
3908 if (widget && gtk_widget_get_can_focus(parent))
3910 if (widget && GTK_WIDGET_CAN_FOCUS(parent))
3913 gtk_widget_grab_focus(parent);
3919 static gboolean pr_mouse_release_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
3923 pr = PIXBUF_RENDERER(widget);
3925 if (pr->scroller_id)
3927 pr_scroller_stop(pr);
3931 #if GTK_CHECK_VERSION(2,20,0)
3932 if (gdk_pointer_is_grabbed() && gtk_widget_has_grab(pr))
3934 if (gdk_pointer_is_grabbed() && GTK_WIDGET_HAS_GRAB(pr))
3937 gtk_grab_remove(widget);
3938 gdk_pointer_ungrab(bevent->time);
3939 widget_set_cursor(widget, -1);
3942 if (pr->drag_moved < PR_DRAG_SCROLL_THRESHHOLD)
3944 if (bevent->button == MOUSE_BUTTON_LEFT && (bevent->state & GDK_CONTROL_MASK))
3946 pr_scroller_start(pr, bevent->x, bevent->y);
3948 else if (bevent->button == MOUSE_BUTTON_LEFT || bevent->button == MOUSE_BUTTON_MIDDLE)
3950 pr_clicked_signal(pr, bevent);
3954 pr->in_drag = FALSE;
3959 static gboolean pr_mouse_leave_cb(GtkWidget *widget, GdkEventCrossing *event, gpointer data)
3963 pr = PIXBUF_RENDERER(widget);
3965 if (pr->scroller_id)
3967 pr->scroller_xpos = pr->scroller_x;
3968 pr->scroller_ypos = pr->scroller_y;
3969 pr->scroller_xinc = 0;
3970 pr->scroller_yinc = 0;
3976 static void pr_mouse_drag_cb(GtkWidget *widget, GdkDragContext *context, gpointer data)
3980 pr = PIXBUF_RENDERER(widget);
3982 pr->drag_moved = PR_DRAG_SCROLL_THRESHHOLD;
3985 static void pr_signals_connect(PixbufRenderer *pr)
3987 g_signal_connect(G_OBJECT(pr), "motion_notify_event",
3988 G_CALLBACK(pr_mouse_motion_cb), pr);
3989 g_signal_connect(G_OBJECT(pr), "button_press_event",
3990 G_CALLBACK(pr_mouse_press_cb), pr);
3991 g_signal_connect(G_OBJECT(pr), "button_release_event",
3992 G_CALLBACK(pr_mouse_release_cb), pr);
3993 g_signal_connect(G_OBJECT(pr), "leave_notify_event",
3994 G_CALLBACK(pr_mouse_leave_cb), pr);
3995 g_signal_connect(G_OBJECT(pr), "hierarchy-changed",
3996 G_CALLBACK(pr_hierarchy_changed_cb), pr);
3997 g_signal_connect(G_OBJECT(pr), "leave_notify_event",
3998 G_CALLBACK(pr_leave_notify_cb), pr);
4000 gtk_widget_set_events(GTK_WIDGET(pr), GDK_POINTER_MOTION_MASK | GDK_POINTER_MOTION_HINT_MASK |
4001 GDK_BUTTON_RELEASE_MASK | GDK_BUTTON_PRESS_MASK |
4002 GDK_LEAVE_NOTIFY_MASK);
4004 g_signal_connect(G_OBJECT(pr), "drag_begin",
4005 G_CALLBACK(pr_mouse_drag_cb), pr);
4010 *-------------------------------------------------------------------
4012 *-------------------------------------------------------------------
4014 static void pr_pixbuf_size_sync(PixbufRenderer *pr)
4016 if (!pr->pixbuf) return;
4017 switch (pr->orientation)
4019 case EXIF_ORIENTATION_LEFT_TOP:
4020 case EXIF_ORIENTATION_RIGHT_TOP:
4021 case EXIF_ORIENTATION_RIGHT_BOTTOM:
4022 case EXIF_ORIENTATION_LEFT_BOTTOM:
4023 pr->image_width = gdk_pixbuf_get_height(pr->pixbuf);
4024 pr->image_height = gdk_pixbuf_get_width(pr->pixbuf);
4027 pr->image_width = gdk_pixbuf_get_width(pr->pixbuf);
4028 pr->image_height = gdk_pixbuf_get_height(pr->pixbuf);
4032 static void pr_set_pixbuf(PixbufRenderer *pr, GdkPixbuf *pixbuf, gdouble zoom, PrZoomFlags flags)
4034 if (pixbuf) g_object_ref(pixbuf);
4035 if (pr->pixbuf) g_object_unref(pr->pixbuf);
4036 pr->pixbuf = pixbuf;
4042 /* no pixbuf so just clear the window */
4043 pr->image_width = 0;
4044 pr->image_height = 0;
4046 pr->zoom = zoom; /* don't throw away the zoom value, it is set by pixbuf_renderer_move, among others,
4047 and used for pixbuf_renderer_zoom_get */
4049 box = GTK_WIDGET(pr);
4051 #if GTK_CHECK_VERSION(2,20,0)
4052 if (gtk_widget_get_realized(box))
4054 if (GTK_WIDGET_REALIZED(box))
4057 gdk_window_clear(box->window);
4058 pr_overlay_draw(pr, 0, 0, pr->window_width, pr->window_height, NULL);
4061 pr_update_signal(pr);
4066 pr_pixbuf_size_sync(pr);
4067 pr_zoom_sync(pr, zoom, flags | PR_ZOOM_FORCE | PR_ZOOM_NEW, 0, 0);
4070 void pixbuf_renderer_set_pixbuf(PixbufRenderer *pr, GdkPixbuf *pixbuf, gdouble zoom)
4072 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
4074 pr_source_tile_unset(pr);
4076 pr_set_pixbuf(pr, pixbuf, zoom, 0);
4078 pr_update_signal(pr);
4081 void pixbuf_renderer_set_pixbuf_lazy(PixbufRenderer *pr, GdkPixbuf *pixbuf, gdouble zoom, gint orientation)
4083 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
4085 pr_source_tile_unset(pr);
4087 pr->orientation = orientation;
4088 pr_set_pixbuf(pr, pixbuf, zoom, PR_ZOOM_LAZY);
4090 pr_update_signal(pr);
4093 GdkPixbuf *pixbuf_renderer_get_pixbuf(PixbufRenderer *pr)
4095 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), NULL);
4100 void pixbuf_renderer_set_orientation(PixbufRenderer *pr, gint orientation)
4102 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
4104 pr->orientation = orientation;
4106 pr_pixbuf_size_sync(pr);
4107 pr_zoom_sync(pr, pr->zoom, PR_ZOOM_FORCE, 0, 0);
4110 gint pixbuf_renderer_get_orientation(PixbufRenderer *pr)
4113 return pr->orientation;
4116 void pixbuf_renderer_set_post_process_func(PixbufRenderer *pr, PixbufRendererPostProcessFunc func, gpointer user_data, gboolean slow)
4118 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
4120 pr->func_post_process = func;
4121 pr->post_process_user_data = user_data;
4122 pr->post_process_slow = func && slow;
4127 void pixbuf_renderer_move(PixbufRenderer *pr, PixbufRenderer *source)
4130 PixbufRendererScrollResetType scroll_reset;
4132 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
4133 g_return_if_fail(IS_PIXBUF_RENDERER(source));
4135 if (pr == source) return;
4137 object = G_OBJECT(pr);
4139 g_object_set(object, "zoom_min", source->zoom_min, NULL);
4140 g_object_set(object, "zoom_max", source->zoom_max, NULL);
4141 g_object_set(object, "loading", source->loading, NULL);
4143 pr->complete = source->complete;
4145 pr->x_scroll = source->x_scroll;
4146 pr->y_scroll = source->y_scroll;
4147 pr->x_mouse = source->x_mouse;
4148 pr->y_mouse = source->y_mouse;
4150 scroll_reset = pr->scroll_reset;
4151 pr->scroll_reset = PR_SCROLL_RESET_NOCHANGE;
4153 pr->func_post_process = source->func_post_process;
4154 pr->post_process_user_data = source->post_process_user_data;
4155 pr->post_process_slow = source->post_process_slow;
4156 pr->orientation = source->orientation;
4158 if (source->source_tiles_enabled)
4160 pr_source_tile_unset(pr);
4162 pr->source_tiles_enabled = source->source_tiles_enabled;
4163 pr->source_tiles_cache_size = source->source_tiles_cache_size;
4164 pr->source_tile_width = source->source_tile_width;
4165 pr->source_tile_height = source->source_tile_height;
4166 pr->image_width = source->image_width;
4167 pr->image_height = source->image_height;
4169 pr->func_tile_request = source->func_tile_request;
4170 pr->func_tile_dispose = source->func_tile_dispose;
4171 pr->func_tile_data = source->func_tile_data;
4173 pr->source_tiles = source->source_tiles;
4174 source->source_tiles = NULL;
4176 pr_zoom_sync(pr, source->zoom, PR_ZOOM_FORCE | PR_ZOOM_NEW, 0, 0);
4177 pr_redraw(pr, TRUE);
4181 pixbuf_renderer_set_pixbuf(pr, source->pixbuf, source->zoom);
4184 pr->scroll_reset = scroll_reset;
4186 pixbuf_renderer_set_pixbuf(source, NULL, source->zoom);
4187 pr_queue_clear(source);
4188 pr_tile_free_all(source);
4191 void pixbuf_renderer_area_changed(PixbufRenderer *pr, gint src_x, gint src_y, gint src_w, gint src_h)
4193 gint x, y, width, height, x1, y1, x2, y2;
4195 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
4197 pr_coords_map_orientation_reverse(pr,
4199 pr->image_width, pr->image_height,
4204 if (pr->source_tiles_enabled)
4206 pr_source_tile_changed(pr, x, y, width, height);
4209 if (pr->scale != 1.0 && pr->zoom_quality != GDK_INTERP_NEAREST)
4211 /* increase region when using a zoom quality that may access surrounding pixels */
4216 x1 = (gint)floor((gdouble)x * pr->scale);
4217 y1 = (gint)floor((gdouble)y * pr->scale);
4218 x2 = (gint)ceil((gdouble)(x + width) * pr->scale);
4219 y2 = (gint)ceil((gdouble)(y + height) * pr->scale);
4221 pr_queue(pr, x1, y1, x2 - x1, y2 - y1, FALSE, TILE_RENDER_AREA, TRUE, TRUE);
4224 void pixbuf_renderer_zoom_adjust(PixbufRenderer *pr, gdouble increment)
4226 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
4228 pr_zoom_adjust_real(pr, increment, PR_ZOOM_NONE, 0, 0);
4231 void pixbuf_renderer_zoom_adjust_at_point(PixbufRenderer *pr, gdouble increment, gint x, gint y)
4233 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
4235 pr_zoom_adjust_real(pr, increment, PR_ZOOM_CENTER, x, y);
4238 void pixbuf_renderer_zoom_set(PixbufRenderer *pr, gdouble zoom)
4240 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
4242 pr_zoom_sync(pr, zoom, PR_ZOOM_NONE, 0, 0);
4245 gdouble pixbuf_renderer_zoom_get(PixbufRenderer *pr)
4247 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), 1.0);
4252 gdouble pixbuf_renderer_zoom_get_scale(PixbufRenderer *pr)
4254 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), 1.0);
4259 void pixbuf_renderer_zoom_set_limits(PixbufRenderer *pr, gdouble min, gdouble max)
4261 g_return_if_fail(IS_PIXBUF_RENDERER(pr));
4263 if (min > 1.0 || max < 1.0) return;
4264 if (min < 1.0 && min > -1.0) return;
4265 if (min < -200.0 || max > 200.0) return;
4267 if (pr->zoom_min != min)
4270 g_object_notify(G_OBJECT(pr), "zoom_min");
4272 if (pr->zoom_max != max)
4275 g_object_notify(G_OBJECT(pr), "zoom_max");
4279 gboolean pixbuf_renderer_get_pixel_colors(PixbufRenderer *pr, gint x_pixel, gint y_pixel,
4280 gint *r_mouse, gint *g_mouse, gint *b_mouse)
4282 GdkPixbuf *pb = pr->pixbuf;
4285 gint map_x, map_y, map_w, map_h;
4287 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), FALSE);
4288 g_return_val_if_fail(r_mouse != NULL && g_mouse != NULL && b_mouse != NULL, FALSE);
4290 if (!pr->pixbuf && !pr->source_tiles_enabled)
4298 if (!pb) return FALSE;
4300 pr_tile_region_map_orientation(pr,
4302 pr->image_width, pr->image_height,
4303 1, 1, /*single pixel */
4307 if (map_x < 0 || map_x > gdk_pixbuf_get_width(pr->pixbuf) - 1) return FALSE;
4308 if (map_y < 0 || map_y > gdk_pixbuf_get_height(pr->pixbuf) - 1) return FALSE;
4310 p_alpha = gdk_pixbuf_get_has_alpha(pb);
4311 prs = gdk_pixbuf_get_rowstride(pb);
4312 p_pix = gdk_pixbuf_get_pixels(pb);
4314 pp = p_pix + map_y * prs + (map_x * (p_alpha ? 4 : 3));
4324 gboolean pixbuf_renderer_get_mouse_position(PixbufRenderer *pr, gint *x_pixel_return, gint *y_pixel_return)
4326 gint x_pixel, y_pixel, x_pixel_clamped, y_pixel_clamped;
4328 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), FALSE);
4329 g_return_val_if_fail(x_pixel_return != NULL && y_pixel_return != NULL, FALSE);
4331 if (!pr->pixbuf && !pr->source_tiles_enabled)
4333 *x_pixel_return = -1;
4334 *y_pixel_return = -1;
4338 x_pixel = floor((gdouble)(pr->x_mouse - pr->x_offset + pr->x_scroll) / pr->scale);
4339 y_pixel = floor((gdouble)(pr->y_mouse - pr->y_offset + pr->y_scroll) / pr->scale);
4340 x_pixel_clamped = CLAMP(x_pixel, 0, pr->image_width - 1);
4341 y_pixel_clamped = CLAMP(y_pixel, 0, pr->image_height - 1);
4343 if(x_pixel != x_pixel_clamped || y_pixel != y_pixel_clamped)
4345 /* mouse is not on pr */
4346 x_pixel = y_pixel = -1;
4349 *x_pixel_return = x_pixel;
4350 *y_pixel_return = y_pixel;
4355 gboolean pixbuf_renderer_get_image_size(PixbufRenderer *pr, gint *width, gint *height)
4357 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), FALSE);
4358 g_return_val_if_fail(width != NULL && height != NULL, FALSE);
4360 if (!pr->pixbuf && !pr->source_tiles_enabled && (!pr->image_width || !pr->image_height))
4367 *width = pr->image_width;
4368 *height = pr->image_height;
4372 gboolean pixbuf_renderer_get_scaled_size(PixbufRenderer *pr, gint *width, gint *height)
4374 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), FALSE);
4375 g_return_val_if_fail(width != NULL && height != NULL, FALSE);
4377 if (!pr->pixbuf && !pr->source_tiles_enabled && (!pr->image_width || !pr->image_height))
4385 *height = pr->height;
4389 gboolean pixbuf_renderer_get_visible_rect(PixbufRenderer *pr, GdkRectangle *rect)
4391 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), FALSE);
4392 g_return_val_if_fail(rect != NULL, FALSE);
4394 if ((!pr->pixbuf && !pr->source_tiles_enabled) ||
4404 rect->x = (gint)((gdouble)pr->x_scroll / pr->scale);
4405 rect->y = (gint)((gdouble)pr->y_scroll / pr->scale);
4406 rect->width = (gint)((gdouble)pr->vis_width / pr->scale);
4407 rect->height = (gint)((gdouble)pr->vis_height / pr->scale);
4411 gboolean pixbuf_renderer_get_virtual_rect(PixbufRenderer *pr, GdkRectangle *rect)
4413 g_return_val_if_fail(IS_PIXBUF_RENDERER(pr), FALSE);
4414 g_return_val_if_fail(rect != NULL, FALSE);
4416 if ((!pr->pixbuf && !pr->source_tiles_enabled))
4425 rect->x = pr->x_scroll;
4426 rect->y = pr->y_scroll;
4427 rect->width = pr->vis_width;
4428 rect->height = pr->vis_height;
4432 void pixbuf_renderer_set_size_early(PixbufRenderer *pr, guint width, guint height)
4437 zoom = pixbuf_renderer_zoom_get(pr);
4438 pr->image_width = width;
4439 pr->image_height = height;
4441 pr_zoom_clamp(pr, zoom, PR_ZOOM_FORCE, NULL);
4446 //pr->width = width;
4447 //pr->height = height;
4450 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */