/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- eel-debug-drawing.c: Eel drawing debugging aids. Copyright (C) 2000 Eazel, Inc. This program is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. Author: Ramiro Estrugo <ramiro@eazel.com> */ #include <config.h> #include "eel-debug-drawing.h" #include "eel-art-gtk-extensions.h" #include "eel-debug.h" #include "eel-gdk-extensions.h" #include "eel-gdk-extensions.h" #include "eel-gdk-pixbuf-extensions.h" #include "eel-gtk-extensions.h" #include "eel-gtk-extensions.h" #include <gtk/gtk.h> #include <stdlib.h> #include <unistd.h> #include <stdio.h> /* * PixbufViewer is a very simple "private" widget that displays * a GdkPixbuf. It is used by eel_debug_show_pixbuf() to * display a pixbuf in an in process pixbuf debugging window. * * We cant use EelImage for this because part of the reason * for pixbuf debugging is to debug EelImage itself. */ #define DEBUG_TYPE_PIXBUF_VIEWER debug_pixbuf_viewer_get_type() #define DEBUG_PIXBUF_VIEWER(obj) \ (G_TYPE_CHECK_INSTANCE_CAST ((obj), DEBUG_TYPE_PIXBUF_VIEWER, DebugPixbufViewer)) #define DEBUG_IS_PIXBUF_VIEWER(obj) \ (G_TYPE_CHECK_INSTANCE_TYPE ((obj), DEBUG_TYPE_PIXBUF_VIEWER)) #if GTK_CHECK_VERSION (3, 0, 0) #define gtk_vbox_new(X,Y) gtk_box_new(GTK_ORIENTATION_VERTICAL,Y) #endif typedef struct DebugPixbufViewer DebugPixbufViewer; typedef struct DebugPixbufViewerClass DebugPixbufViewerClass; static GType debug_pixbuf_viewer_get_type (void); static void debug_pixbuf_viewer_set_pixbuf (DebugPixbufViewer *viewer, GdkPixbuf *pixbuf); struct DebugPixbufViewer { GtkWidget widget; GdkPixbuf *pixbuf; }; struct DebugPixbufViewerClass { GtkWidgetClass parent_class; }; G_DEFINE_TYPE (DebugPixbufViewer, debug_pixbuf_viewer, GTK_TYPE_WIDGET) static void debug_pixbuf_viewer_init (DebugPixbufViewer *viewer) { gtk_widget_set_can_focus (GTK_WIDGET (viewer), FALSE); gtk_widget_set_has_window (GTK_WIDGET (viewer), FALSE); } static void debug_pixbuf_viewer_finalize (GObject *object) { DebugPixbufViewer *viewer; viewer = DEBUG_PIXBUF_VIEWER (object); eel_gdk_pixbuf_unref_if_not_null (viewer->pixbuf); viewer->pixbuf = NULL; G_OBJECT_CLASS (debug_pixbuf_viewer_parent_class)->finalize (object); } static void debug_pixbuf_viewer_size_request (GtkWidget *widget, GtkRequisition *requisition) { DebugPixbufViewer *viewer; EelDimensions dimensions; g_assert (DEBUG_IS_PIXBUF_VIEWER (widget)); g_assert (requisition != NULL); viewer = DEBUG_PIXBUF_VIEWER (widget); if (viewer->pixbuf != NULL) { dimensions = eel_gdk_pixbuf_get_dimensions (viewer->pixbuf); } else { dimensions = eel_dimensions_empty; } requisition->width = MAX (2, dimensions.width); requisition->height = MAX (2, dimensions.height); } #if GTK_CHECK_VERSION (3, 0, 0) static void debug_pixbuf_viewer_get_preferred_width (GtkWidget *widget, gint *minimum_width, gint *natural_width) { GtkRequisition req; debug_pixbuf_viewer_size_request (widget, &req); *minimum_width = *natural_width = req.width; } static void debug_pixbuf_viewer_get_preferred_height (GtkWidget *widget, gint *minimum_height, gint *natural_height) { GtkRequisition req; debug_pixbuf_viewer_size_request (widget, &req); *minimum_height = *natural_height = req.height; } #endif static int #if GTK_CHECK_VERSION (3, 0, 0) debug_pixbuf_viewer_draw (GtkWidget *widget, cairo_t *cr) #else debug_pixbuf_viewer_expose_event (GtkWidget *widget, GdkEventExpose *event) #endif { DebugPixbufViewer *viewer; EelIRect clipped_dirty_area; EelIRect dirty_area; EelIRect bounds; GtkAllocation allocation; g_assert (DEBUG_IS_PIXBUF_VIEWER (widget)); #if GTK_CHECK_VERSION (3, 0, 0) g_assert (cr != NULL); #else g_assert (event != NULL); g_assert (event->window == gtk_widget_get_window (widget)); #endif g_assert (gtk_widget_get_realized (widget)); viewer = DEBUG_PIXBUF_VIEWER (widget); if (viewer->pixbuf == NULL) { return TRUE; } gtk_widget_get_allocation (widget, &allocation); bounds.x0 = allocation.x + (allocation.width - gdk_pixbuf_get_width (viewer->pixbuf)) / 2; bounds.y0 = allocation.y + (allocation.height - gdk_pixbuf_get_height (viewer->pixbuf)) / 2; bounds.x1 = bounds.x0 + gdk_pixbuf_get_width (viewer->pixbuf); bounds.y1 = bounds.y0 + gdk_pixbuf_get_height (viewer->pixbuf); /* Clip the dirty area to the screen; bail if no work to do */ #if GTK_CHECK_VERSION (3, 0, 0) clipped_dirty_area = eel_gdk_window_clip_dirty_area_to_screen (gtk_widget_get_window (widget), dirty_area); #else dirty_area = eel_gdk_rectangle_to_eel_irect (event->area); clipped_dirty_area = eel_gdk_window_clip_dirty_area_to_screen (event->window, dirty_area); #endif if (!eel_irect_is_empty (&clipped_dirty_area)) { EelIRect clipped_bounds; eel_irect_intersect (&clipped_bounds, &bounds, &clipped_dirty_area); if (!eel_irect_is_empty (&clipped_bounds)) { g_assert (clipped_bounds.x0 >= bounds.x0); g_assert (clipped_bounds.y0 >= bounds.y0); eel_gdk_pixbuf_draw_to_drawable (viewer->pixbuf, #if GTK_CHECK_VERSION (3, 0, 0) cr, #else event->window, #endif clipped_bounds.x0 - bounds.x0, clipped_bounds.y0 - bounds.y0, clipped_bounds); } } bounds.x0 -= 1; bounds.y0 -= 1; bounds.x1 += 1; bounds.y1 += 1; return TRUE; } static void debug_pixbuf_viewer_class_init (DebugPixbufViewerClass *pixbuf_viewer_class) { GObjectClass *object_class = G_OBJECT_CLASS (pixbuf_viewer_class); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (pixbuf_viewer_class); object_class->finalize = debug_pixbuf_viewer_finalize; #if GTK_CHECK_VERSION (3, 0, 0) widget_class->get_preferred_width = debug_pixbuf_viewer_get_preferred_width; widget_class->get_preferred_height = debug_pixbuf_viewer_get_preferred_height; widget_class->draw = debug_pixbuf_viewer_draw; #else widget_class->size_request = debug_pixbuf_viewer_size_request; widget_class->expose_event = debug_pixbuf_viewer_expose_event; #endif } static void debug_pixbuf_viewer_set_pixbuf (DebugPixbufViewer *viewer, GdkPixbuf *pixbuf) { g_assert (DEBUG_IS_PIXBUF_VIEWER (viewer)); if (pixbuf != viewer->pixbuf) { eel_gdk_pixbuf_unref_if_not_null (viewer->pixbuf); eel_gdk_pixbuf_ref_if_not_null (pixbuf); viewer->pixbuf = pixbuf; gtk_widget_queue_resize (GTK_WIDGET (viewer)); } } /** * eel_debug_draw_rectangle_and_cross: * @rectangle: Rectangle bounding the rectangle. * @color: Color to use for the rectangle and cross. * * Draw a rectangle and cross. Useful for debugging exposure events. */ #if !GTK_CHECK_VERSION (3, 0, 0) void eel_debug_draw_rectangle_and_cross (GdkDrawable *drawable, EelIRect rectangle, guint32 color, gboolean draw_cross) { cairo_t *cr; GdkColor color_gdk = { 0 }; int width; int height; g_return_if_fail (drawable != NULL); g_return_if_fail (!eel_irect_is_empty (&rectangle)); width = rectangle.x1 - rectangle.x0; height = rectangle.y1 - rectangle.y0; cr = gdk_cairo_create (drawable); color_gdk.red = ((color >> 16) & 0xff) << 8; color_gdk.green = ((color >> 8) & 0xff) << 8; color_gdk.blue = ((color ) & 0xff) << 8; gdk_cairo_set_source_color (cr, &color_gdk); cairo_set_line_width (cr, 1.0); cairo_rectangle (cr, rectangle.x0 + 0.5, rectangle.y0 + 0.5, width, height); if (draw_cross) { cairo_move_to (cr, rectangle.x0, rectangle.y0); cairo_line_to (cr, rectangle.x0 + width, rectangle.y0 + height); cairo_move_to (cr, rectangle.x0 + width, rectangle.y0); cairo_line_to (cr, rectangle.x0, rectangle.y0 + height); } cairo_stroke (cr); cairo_destroy (cr); } #endif /** * eel_debug_show_pixbuf_in_external_viewer: * @pixbuf: Pixbuf to show. * @viewer_name: Viewer name. * * Show the given pixbuf in an external out of process viewer. * This is very useful for debugging pixbuf stuff. * * Perhaps this function should be #ifdef DEBUG or something like that. */ void eel_debug_show_pixbuf_in_external_viewer (const GdkPixbuf *pixbuf, const char *viewer_name) { char *command; char *file_name; gboolean save_result; int ignore; int fd; g_return_if_fail (pixbuf != NULL); g_return_if_fail (viewer_name != NULL); file_name = g_strdup ("/tmp/eel-debug-png-file-XXXXXX"); fd = g_mkstemp (file_name); if (fd == -1) { g_free (file_name); file_name = g_strdup_printf ("/tmp/isis-debug-png-file-%d", getpid ()); } else { close (fd); } save_result = eel_gdk_pixbuf_save_to_file (pixbuf, file_name); if (save_result == FALSE) { g_warning ("Failed to save '%s'", file_name); g_free (file_name); return; } command = g_strdup_printf ("%s %s", viewer_name, file_name); ignore = system (command); g_free (command); remove (file_name); g_free (file_name); } static GtkWidget *debug_window = NULL; static GtkWidget *debug_image = NULL; static void debug_delete_event (GtkWidget *widget, GdkEvent *event, gpointer callback_data) { gtk_widget_hide (widget); } static void destroy_debug_window (void) { if (debug_window != NULL) { gtk_widget_destroy (debug_window); debug_window = NULL; } } /** * eel_debug_show_pixbuf_in: * @pixbuf: Pixbuf to show. * * Show the given pixbuf in an in process window. */ void eel_debug_show_pixbuf (GdkPixbuf *pixbuf) { if (debug_window == NULL) { GtkWidget *vbox; debug_window = gtk_window_new (GTK_WINDOW_TOPLEVEL); vbox = gtk_vbox_new (FALSE, 0); gtk_container_add (GTK_CONTAINER (debug_window), vbox); gtk_window_set_title (GTK_WINDOW (debug_window), "Pixbuf debugging"); gtk_window_set_resizable (GTK_WINDOW (debug_window), TRUE); gtk_container_set_border_width (GTK_CONTAINER (debug_window), 10); g_signal_connect (debug_window, "delete_event", G_CALLBACK (debug_delete_event), NULL); debug_image = gtk_widget_new (debug_pixbuf_viewer_get_type (), NULL); gtk_box_pack_start (GTK_BOX (vbox), debug_image, TRUE, TRUE, 0); eel_debug_call_at_shutdown (destroy_debug_window); gtk_widget_show (debug_image); gtk_widget_show (vbox); } gtk_widget_show (debug_window); debug_pixbuf_viewer_set_pixbuf (DEBUG_PIXBUF_VIEWER (debug_image), pixbuf); gdk_window_clear_area_e (gtk_widget_get_window (debug_window), 0, 0, -1, -1); } void eel_debug_pixbuf_draw_point (GdkPixbuf *pixbuf, int x, int y, guint32 color, int opacity) { EelDimensions dimensions; guchar *pixels; gboolean has_alpha; guint pixel_offset; guint rowstride; guchar red; guchar green; guchar blue; guchar alpha; guchar *offset; g_return_if_fail (eel_gdk_pixbuf_is_valid (pixbuf)); g_return_if_fail (opacity >= EEL_OPACITY_FULLY_TRANSPARENT); g_return_if_fail (opacity <= EEL_OPACITY_FULLY_OPAQUE); dimensions = eel_gdk_pixbuf_get_dimensions (pixbuf); g_return_if_fail (x >= 0 && x < dimensions.width); g_return_if_fail (y >= 0 && y < dimensions.height); pixels = gdk_pixbuf_get_pixels (pixbuf); rowstride = gdk_pixbuf_get_rowstride (pixbuf); has_alpha = gdk_pixbuf_get_has_alpha (pixbuf); pixel_offset = has_alpha ? 4 : 3; red = EEL_RGBA_COLOR_GET_R (color); green = EEL_RGBA_COLOR_GET_G (color); blue = EEL_RGBA_COLOR_GET_B (color); alpha = (guchar) opacity; offset = pixels + y * rowstride + x * pixel_offset; *(offset + 0) = red; *(offset + 1) = green; *(offset + 2) = blue; if (has_alpha) { *(offset + 3) = alpha; } } void eel_debug_pixbuf_draw_rectangle (GdkPixbuf *pixbuf, gboolean filled, int x0, int y0, int x1, int y1, guint32 color, int opacity) { EelDimensions dimensions; int x; int y; g_return_if_fail (eel_gdk_pixbuf_is_valid (pixbuf)); g_return_if_fail (opacity >= EEL_OPACITY_FULLY_TRANSPARENT); g_return_if_fail (opacity <= EEL_OPACITY_FULLY_OPAQUE); dimensions = eel_gdk_pixbuf_get_dimensions (pixbuf); if (x0 == -1) { x0 = 0; } if (y0 == -1) { y0 = 0; } if (x1 == -1) { x1 = dimensions.width - 1; } if (y1 == -1) { y1 = dimensions.height - 1; } g_return_if_fail (x1 > x0); g_return_if_fail (y1 > y0); g_return_if_fail (x0 >= 0 && x0 < dimensions.width); g_return_if_fail (y0 >= 0 && y0 < dimensions.height); g_return_if_fail (x1 >= 0 && x1 < dimensions.width); g_return_if_fail (y1 >= 0 && y1 < dimensions.height); if (filled) { for (y = y0; y <= y1; y++) { for (x = x0; x <= x1; x++) { eel_debug_pixbuf_draw_point (pixbuf, x, y, color, opacity); } } } else { /* Top / Bottom */ for (x = x0; x <= x1; x++) { eel_debug_pixbuf_draw_point (pixbuf, x, y0, color, opacity); eel_debug_pixbuf_draw_point (pixbuf, x, y1, color, opacity); } /* Left / Right */ for (y = y0; y <= y1; y++) { eel_debug_pixbuf_draw_point (pixbuf, x0, y, color, opacity); eel_debug_pixbuf_draw_point (pixbuf, x1, y, color, opacity); } } } void eel_debug_pixbuf_draw_rectangle_inset (GdkPixbuf *pixbuf, gboolean filled, int x0, int y0, int x1, int y1, guint32 color, int opacity, int inset) { EelDimensions dimensions; g_return_if_fail (eel_gdk_pixbuf_is_valid (pixbuf)); g_return_if_fail (opacity >= EEL_OPACITY_FULLY_TRANSPARENT); g_return_if_fail (opacity <= EEL_OPACITY_FULLY_OPAQUE); dimensions = eel_gdk_pixbuf_get_dimensions (pixbuf); if (x0 == -1) { x0 = 0; } if (y0 == -1) { y0 = 0; } if (x1 == -1) { x1 = dimensions.width - 1; } if (y1 == -1) { y1 = dimensions.height - 1; } x0 += inset; y0 += inset; x1 -= inset; y1 -= inset; g_return_if_fail (x1 > x0); g_return_if_fail (y1 > y0); eel_debug_pixbuf_draw_rectangle (pixbuf, filled, x0, y0, x1, y1, color, opacity); }