forked from AuroraMiddleware/gtk
Work around https://bugs.freedesktop.org/show_bug.cgi?id=4320, which used
2006-02-03 Federico Mena Quintero <federico@ximian.com> Work around https://bugs.freedesktop.org/show_bug.cgi?id=4320, which used to be our own http://bugzilla.gnome.org/show_bug.cgi?id=314616. If one uses a pixmap for a pattern in Cairo, and sets the pattern to CAIRO_EXTEND_REPEAT; and if the destination surface is also a pixmap, Cairo does a slow copy instead of using XCopyArea(). So, we use the same code that we used in GTK+ 2.6 (pre-cairo), by filling the double-buffer pixmap with a tiled GC and XFillRectangle(). * gdk/gdkwindow.c (BackingRectMethod): New structure with a cairo_t and a GdkGC field. Depending on which of these fields gets filled in, we'll use Cairo or GDK to clear the double-buffer pixmap when painting a window. (setup_backing_rect_method): Fill a BackingRectMethod as appropriate, depending on the window's configuration and our knowledge of whether Cairo is fast or slow when doing repeating patterns. (gdk_window_clear_backing_rect): Call setup_backing_rect_method(). Depending on what it returns, use Cairo to clear the double-buffer pixmap, or plain GDK.
This commit is contained in:
parent
32fb5404bf
commit
9bfabba034
24
ChangeLog
24
ChangeLog
@ -1,3 +1,27 @@
|
|||||||
|
2006-02-03 Federico Mena Quintero <federico@ximian.com>
|
||||||
|
|
||||||
|
Work around https://bugs.freedesktop.org/show_bug.cgi?id=4320,
|
||||||
|
which used to be our own
|
||||||
|
http://bugzilla.gnome.org/show_bug.cgi?id=314616. If one uses a
|
||||||
|
pixmap for a pattern in Cairo, and sets the pattern to
|
||||||
|
CAIRO_EXTEND_REPEAT; and if the destination surface is also a
|
||||||
|
pixmap, Cairo does a slow copy instead of using XCopyArea(). So,
|
||||||
|
we use the same code that we used in GTK+ 2.6 (pre-cairo), by
|
||||||
|
filling the double-buffer pixmap with a tiled GC and
|
||||||
|
XFillRectangle().
|
||||||
|
|
||||||
|
* gdk/gdkwindow.c (BackingRectMethod): New structure with a
|
||||||
|
cairo_t and a GdkGC field. Depending on which of these fields
|
||||||
|
gets filled in, we'll use Cairo or GDK to clear the double-buffer
|
||||||
|
pixmap when painting a window.
|
||||||
|
(setup_backing_rect_method): Fill a BackingRectMethod as
|
||||||
|
appropriate, depending on the window's configuration and our
|
||||||
|
knowledge of whether Cairo is fast or slow when doing repeating
|
||||||
|
patterns.
|
||||||
|
(gdk_window_clear_backing_rect): Call
|
||||||
|
setup_backing_rect_method(). Depending on what it returns, use
|
||||||
|
Cairo to clear the double-buffer pixmap, or plain GDK.
|
||||||
|
|
||||||
2006-02-03 Matthias Clasen <mclasen@redhat.com>
|
2006-02-03 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
|
||||||
* gtk/gtklabel.c (get_layout_location): Fix handling
|
* gtk/gtklabel.c (get_layout_location): Fix handling
|
||||||
|
@ -1,3 +1,27 @@
|
|||||||
|
2006-02-03 Federico Mena Quintero <federico@ximian.com>
|
||||||
|
|
||||||
|
Work around https://bugs.freedesktop.org/show_bug.cgi?id=4320,
|
||||||
|
which used to be our own
|
||||||
|
http://bugzilla.gnome.org/show_bug.cgi?id=314616. If one uses a
|
||||||
|
pixmap for a pattern in Cairo, and sets the pattern to
|
||||||
|
CAIRO_EXTEND_REPEAT; and if the destination surface is also a
|
||||||
|
pixmap, Cairo does a slow copy instead of using XCopyArea(). So,
|
||||||
|
we use the same code that we used in GTK+ 2.6 (pre-cairo), by
|
||||||
|
filling the double-buffer pixmap with a tiled GC and
|
||||||
|
XFillRectangle().
|
||||||
|
|
||||||
|
* gdk/gdkwindow.c (BackingRectMethod): New structure with a
|
||||||
|
cairo_t and a GdkGC field. Depending on which of these fields
|
||||||
|
gets filled in, we'll use Cairo or GDK to clear the double-buffer
|
||||||
|
pixmap when painting a window.
|
||||||
|
(setup_backing_rect_method): Fill a BackingRectMethod as
|
||||||
|
appropriate, depending on the window's configuration and our
|
||||||
|
knowledge of whether Cairo is fast or slow when doing repeating
|
||||||
|
patterns.
|
||||||
|
(gdk_window_clear_backing_rect): Call
|
||||||
|
setup_backing_rect_method(). Depending on what it returns, use
|
||||||
|
Cairo to clear the double-buffer pixmap, or plain GDK.
|
||||||
|
|
||||||
2006-02-03 Matthias Clasen <mclasen@redhat.com>
|
2006-02-03 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
|
||||||
* gtk/gtklabel.c (get_layout_location): Fix handling
|
* gtk/gtklabel.c (get_layout_location): Fix handling
|
||||||
|
123
gdk/gdkwindow.c
123
gdk/gdkwindow.c
@ -1730,43 +1730,86 @@ gdk_window_draw_glyphs_transformed (GdkDrawable *drawable,
|
|||||||
RESTORE_GC (gc);
|
RESTORE_GC (gc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
cairo_t *cr; /* if non-null, it means use this cairo context */
|
||||||
|
GdkGC *gc; /* if non-null, it means use this GC instead */
|
||||||
|
} BackingRectMethod;
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gdk_window_set_bg_pattern (GdkWindow *window,
|
setup_backing_rect_method (BackingRectMethod *method, GdkWindow *window, GdkWindowPaint *paint, int x_offset_cairo, int y_offset_cairo)
|
||||||
cairo_t *cr,
|
|
||||||
int x_offset,
|
|
||||||
int y_offset)
|
|
||||||
{
|
{
|
||||||
GdkWindowObject *private = (GdkWindowObject *)window;
|
GdkWindowObject *private = (GdkWindowObject *)window;
|
||||||
|
|
||||||
if (private->bg_pixmap == GDK_PARENT_RELATIVE_BG && private->parent)
|
if (private->bg_pixmap == GDK_PARENT_RELATIVE_BG && private->parent)
|
||||||
{
|
{
|
||||||
x_offset += private->x;
|
GdkWindowPaint tmp_paint;
|
||||||
y_offset += private->y;
|
|
||||||
gdk_window_set_bg_pattern (GDK_WINDOW (private->parent), cr,
|
tmp_paint = *paint;
|
||||||
x_offset, y_offset);
|
tmp_paint.x_offset += private->x;
|
||||||
|
tmp_paint.y_offset += private->y;
|
||||||
|
|
||||||
|
x_offset_cairo += private->x;
|
||||||
|
y_offset_cairo += private->y;
|
||||||
|
|
||||||
|
setup_backing_rect_method (method, GDK_WINDOW (private->parent), &tmp_paint, x_offset_cairo, y_offset_cairo);
|
||||||
}
|
}
|
||||||
else if (private->bg_pixmap &&
|
else if (private->bg_pixmap &&
|
||||||
private->bg_pixmap != GDK_PARENT_RELATIVE_BG &&
|
private->bg_pixmap != GDK_PARENT_RELATIVE_BG &&
|
||||||
private->bg_pixmap != GDK_NO_BG)
|
private->bg_pixmap != GDK_NO_BG)
|
||||||
{
|
{
|
||||||
|
/* This is a workaround for https://bugs.freedesktop.org/show_bug.cgi?id=4320.
|
||||||
|
* In it, using a pixmap as a repeating pattern in Cairo, and painting it to a
|
||||||
|
* pixmap destination surface, can be very slow (on the order of seconds for a
|
||||||
|
* whole-screen copy). The workaround is to use pretty much the same code that
|
||||||
|
* we used in GTK+ 2.6 (pre-Cairo), which clears the double-buffer pixmap with
|
||||||
|
* a tiled GC XFillRectangle().
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* Actually computing this flag is left as an exercise for the reader */
|
||||||
|
#if defined (G_OS_UNIX)
|
||||||
|
# define GDK_CAIRO_REPEAT_IS_FAST 0
|
||||||
|
#else
|
||||||
|
# define GDK_CAIRO_REPEAT_IS_FAST 1
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if GDK_CAIRO_REPEAT_IS_FAST
|
||||||
cairo_surface_t *surface = _gdk_drawable_ref_cairo_surface (private->bg_pixmap);
|
cairo_surface_t *surface = _gdk_drawable_ref_cairo_surface (private->bg_pixmap);
|
||||||
cairo_pattern_t *pattern = cairo_pattern_create_for_surface (surface);
|
cairo_pattern_t *pattern = cairo_pattern_create_for_surface (surface);
|
||||||
cairo_surface_destroy (surface);
|
cairo_surface_destroy (surface);
|
||||||
|
|
||||||
if (x_offset != 0 || y_offset != 0)
|
if (x_offset_cairo != 0 || y_offset_cairo != 0)
|
||||||
{
|
{
|
||||||
cairo_matrix_t matrix;
|
cairo_matrix_t matrix;
|
||||||
cairo_matrix_init_translate (&matrix, x_offset, y_offset);
|
cairo_matrix_init_translate (&matrix, x_offset_cairo, y_offset_cairo);
|
||||||
cairo_pattern_set_matrix (pattern, &matrix);
|
cairo_pattern_set_matrix (pattern, &matrix);
|
||||||
}
|
}
|
||||||
|
|
||||||
cairo_pattern_set_extend (pattern, CAIRO_EXTEND_REPEAT);
|
cairo_pattern_set_extend (pattern, CAIRO_EXTEND_REPEAT);
|
||||||
cairo_set_source (cr, pattern);
|
|
||||||
|
method->cr = cairo_create (paint->surface);
|
||||||
|
method->gc = NULL;
|
||||||
|
|
||||||
|
cairo_set_source (method->cr, pattern);
|
||||||
cairo_pattern_destroy (pattern);
|
cairo_pattern_destroy (pattern);
|
||||||
|
#else
|
||||||
|
guint gc_mask;
|
||||||
|
GdkGCValues gc_values;
|
||||||
|
|
||||||
|
gc_values.fill = GDK_TILED;
|
||||||
|
gc_values.tile = private->bg_pixmap;
|
||||||
|
gc_values.ts_x_origin = -x_offset_cairo;
|
||||||
|
gc_values.ts_y_origin = -y_offset_cairo;
|
||||||
|
|
||||||
|
gc_mask = GDK_GC_FILL | GDK_GC_TILE | GDK_GC_TS_X_ORIGIN | GDK_GC_TS_Y_ORIGIN;
|
||||||
|
|
||||||
|
method->gc = gdk_gc_new_with_values (paint->pixmap, &gc_values, gc_mask);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
gdk_cairo_set_source_color (cr, &private->bg_color);
|
method->cr = cairo_create (paint->surface);
|
||||||
|
|
||||||
|
gdk_cairo_set_source_color (method->cr, &private->bg_color);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1779,22 +1822,56 @@ gdk_window_clear_backing_rect (GdkWindow *window,
|
|||||||
{
|
{
|
||||||
GdkWindowObject *private = (GdkWindowObject *)window;
|
GdkWindowObject *private = (GdkWindowObject *)window;
|
||||||
GdkWindowPaint *paint = private->paint_stack->data;
|
GdkWindowPaint *paint = private->paint_stack->data;
|
||||||
cairo_t *cr;
|
BackingRectMethod method;
|
||||||
|
#if 0
|
||||||
|
GTimer *timer;
|
||||||
|
double elapsed;
|
||||||
|
#endif
|
||||||
|
|
||||||
if (GDK_WINDOW_DESTROYED (window))
|
if (GDK_WINDOW_DESTROYED (window))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
cr = cairo_create (paint->surface);
|
#if 0
|
||||||
|
timer = g_timer_new ();
|
||||||
|
#endif
|
||||||
|
|
||||||
gdk_window_set_bg_pattern (window, cr, 0, 0);
|
method.cr = NULL;
|
||||||
|
method.gc = NULL;
|
||||||
|
setup_backing_rect_method (&method, window, paint, 0, 0);
|
||||||
|
|
||||||
cairo_rectangle (cr, x, y, width, height);
|
if (method.cr)
|
||||||
cairo_clip (cr);
|
{
|
||||||
|
g_assert (method.gc == NULL);
|
||||||
|
|
||||||
gdk_cairo_region (cr, paint->region);
|
cairo_rectangle (method.cr, x, y, width, height);
|
||||||
cairo_fill (cr);
|
cairo_clip (method.cr);
|
||||||
|
|
||||||
cairo_destroy (cr);
|
gdk_cairo_region (method.cr, paint->region);
|
||||||
|
cairo_fill (method.cr);
|
||||||
|
|
||||||
|
cairo_destroy (method.cr);
|
||||||
|
#if 0
|
||||||
|
elapsed = g_timer_elapsed (timer, NULL);
|
||||||
|
g_print ("Draw the background with Cairo: %fs\n", elapsed);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
g_assert (method.gc != NULL);
|
||||||
|
|
||||||
|
gdk_gc_set_clip_region (method.gc, paint->region);
|
||||||
|
gdk_draw_rectangle (window, method.gc, TRUE, x, y, width, height);
|
||||||
|
g_object_unref (method.gc);
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
elapsed = g_timer_elapsed (timer, NULL);
|
||||||
|
g_print ("Draw the background with GDK: %fs\n", elapsed);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
g_timer_destroy (timer);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
Loading…
Reference in New Issue
Block a user