 2905fc861a
			
		
	
	2905fc861a
	
	
	
		
			
			This reverts commit 5aedfe048b.
It had a typo that broke the build, only replaced half of the uses, and
replaced them with other functions that are also deprecated anyway.
		
	
		
			
				
	
	
		
			308 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			308 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include <gdk/gdk.h>
 | |
| #include <gtk/gtk.h>
 | |
| #include <gdkx.h>
 | |
| #include <stdio.h>
 | |
| #include <errno.h>
 | |
| #include <sys/wait.h>
 | |
| #include <unistd.h>
 | |
| #include <X11/extensions/shape.h>
 | |
| 
 | |
| #include <gdk-pixbuf/gdk-pixbuf.h>
 | |
| #include <sys/types.h>
 | |
| #include <sys/stat.h>
 | |
| #include <sys/wait.h>
 | |
| #include <signal.h>
 | |
| #include <unistd.h>
 | |
| #include <stdlib.h>
 | |
| #include <fcntl.h>
 | |
| #include <errno.h>
 | |
| #include <locale.h>
 | |
| #include "widgets.h"
 | |
| #include "shadow.h"
 | |
| 
 | |
| #define MAXIMUM_WM_REPARENTING_DEPTH 4
 | |
| #ifndef _
 | |
| #define _(x) (x)
 | |
| #endif
 | |
| 
 | |
| static void queue_show (void);
 | |
| 
 | |
| static Window
 | |
| find_toplevel_window (Window xid)
 | |
| {
 | |
|   Window root, parent, *children;
 | |
|   guint nchildren;
 | |
| 
 | |
|   do
 | |
|     {
 | |
|       if (XQueryTree (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()), xid, &root,
 | |
| 		      &parent, &children, &nchildren) == 0)
 | |
| 	{
 | |
| 	  g_warning ("Couldn't find window manager window");
 | |
| 	  return 0;
 | |
| 	}
 | |
| 
 | |
|       if (root == parent)
 | |
| 	return xid;
 | |
| 
 | |
|       xid = parent;
 | |
|     }
 | |
|   while (TRUE);
 | |
| }
 | |
| 
 | |
| static GdkPixbuf *
 | |
| add_border_to_shot (GdkPixbuf *pixbuf)
 | |
| {
 | |
|   GdkPixbuf *retval;
 | |
|   GdkColorspace colorspace;
 | |
|   int bits;
 | |
| 
 | |
|   colorspace = gdk_pixbuf_get_colorspace (pixbuf);
 | |
|   bits = gdk_pixbuf_get_bits_per_sample (pixbuf);
 | |
|   retval = gdk_pixbuf_new (colorspace, TRUE, bits,
 | |
| 			   gdk_pixbuf_get_width (pixbuf) + 2,
 | |
| 			   gdk_pixbuf_get_height (pixbuf) + 2);
 | |
| 
 | |
|   /* Fill with solid black */
 | |
|   gdk_pixbuf_fill (retval, 0xFF);
 | |
|   gdk_pixbuf_copy_area (pixbuf,
 | |
| 			0, 0,
 | |
| 			gdk_pixbuf_get_width (pixbuf),
 | |
| 			gdk_pixbuf_get_height (pixbuf),
 | |
| 			retval, 1, 1);
 | |
| 
 | |
|   return retval;
 | |
| }
 | |
| 
 | |
| static GdkPixbuf *
 | |
| remove_shaped_area (GdkPixbuf *pixbuf,
 | |
| 		    Window     window)
 | |
| {
 | |
|   GdkPixbuf *retval;
 | |
|   XRectangle *rectangles;
 | |
|   int rectangle_count, rectangle_order;
 | |
|   int i;
 | |
|   GdkColorspace colorspace;
 | |
|   int bits;
 | |
| 
 | |
|   colorspace = gdk_pixbuf_get_colorspace (pixbuf);
 | |
|   bits = gdk_pixbuf_get_bits_per_sample (pixbuf);
 | |
|   retval = gdk_pixbuf_new (colorspace, TRUE, bits,
 | |
| 			   gdk_pixbuf_get_width (pixbuf),
 | |
| 			   gdk_pixbuf_get_height (pixbuf));
 | |
|   
 | |
|   gdk_pixbuf_fill (retval, 0);
 | |
|   rectangles = XShapeGetRectangles (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()), window,
 | |
| 				    ShapeBounding, &rectangle_count, &rectangle_order);
 | |
| 
 | |
|   for (i = 0; i < rectangle_count; i++)
 | |
|     {
 | |
|       int y, x;
 | |
| 
 | |
|       for (y = rectangles[i].y; y < rectangles[i].y + rectangles[i].height; y++)
 | |
| 	{
 | |
| 	  guchar *src_pixels, *dest_pixels;
 | |
| 
 | |
| 	  src_pixels = gdk_pixbuf_get_pixels (pixbuf) +
 | |
| 	    y * gdk_pixbuf_get_rowstride (pixbuf) +
 | |
| 	    rectangles[i].x * (gdk_pixbuf_get_has_alpha (pixbuf) ? 4 : 3);
 | |
| 	  dest_pixels = gdk_pixbuf_get_pixels (retval) +
 | |
| 	    y * gdk_pixbuf_get_rowstride (retval) +
 | |
| 	    rectangles[i].x * 4;
 | |
| 
 | |
| 	  for (x = rectangles[i].x; x < rectangles[i].x + rectangles[i].width; x++)
 | |
| 	    {
 | |
| 	      *dest_pixels++ = *src_pixels ++;
 | |
| 	      *dest_pixels++ = *src_pixels ++;
 | |
| 	      *dest_pixels++ = *src_pixels ++;
 | |
| 	      *dest_pixels++ = 255;
 | |
| 
 | |
| 	      if (gdk_pixbuf_get_has_alpha (pixbuf))
 | |
| 		src_pixels++;
 | |
| 	    }
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|   return retval;
 | |
| }
 | |
| 
 | |
| typedef enum {
 | |
|   DECOR_NONE,
 | |
|   DECOR_FRAME,
 | |
|   DECOR_WINDOW_FRAME
 | |
| } DecorationType;
 | |
| 
 | |
| static GdkPixbuf *
 | |
| take_window_shot (Window         child,
 | |
|                   DecorationType decor)
 | |
| {
 | |
|   GdkWindow *window;
 | |
|   Window xid;
 | |
|   gint x_orig, y_orig;
 | |
|   gint x = 0, y = 0;
 | |
|   gint width, height;
 | |
| 
 | |
|   GdkPixbuf *tmp, *tmp2;
 | |
|   GdkPixbuf *retval = NULL;
 | |
| 
 | |
|   if (decor == DECOR_WINDOW_FRAME)
 | |
|     xid = find_toplevel_window (child);
 | |
|   else
 | |
|     xid = child;
 | |
| 
 | |
|   window = gdk_x11_window_foreign_new_for_display (gdk_display_get_default (), xid);
 | |
| 
 | |
|   width = gdk_window_get_width (window);
 | |
|   height = gdk_window_get_height (window);
 | |
|   gdk_window_get_origin (window, &x_orig, &y_orig);
 | |
| 
 | |
|   if (x_orig < 0)
 | |
|     {
 | |
|       x = - x_orig;
 | |
|       width = width + x_orig;
 | |
|       x_orig = 0;
 | |
|     }
 | |
| 
 | |
|   if (y_orig < 0)
 | |
|     {
 | |
|       y = - y_orig;
 | |
|       height = height + y_orig;
 | |
|       y_orig = 0;
 | |
|     }
 | |
| 
 | |
|   if (x_orig + width > gdk_screen_width ())
 | |
|     width = gdk_screen_width () - x_orig;
 | |
| 
 | |
|   if (y_orig + height > gdk_screen_height ())
 | |
|     height = gdk_screen_height () - y_orig;
 | |
| 
 | |
|   tmp = gdk_pixbuf_get_from_window (window,
 | |
| 				    x, y, width, height);
 | |
| 
 | |
|   if (tmp != NULL)
 | |
|     {
 | |
|       if (decor == DECOR_WINDOW_FRAME)
 | |
|         tmp2 = remove_shaped_area (tmp, xid);
 | |
|       else if (decor == DECOR_FRAME)
 | |
|         tmp2 = add_border_to_shot (tmp);
 | |
|       else
 | |
|         tmp2 = g_object_ref (tmp);
 | |
| 
 | |
|       g_object_unref (tmp);
 | |
| 
 | |
|       if (tmp2 != NULL)
 | |
|         {
 | |
|           retval = create_shadowed_pixbuf (tmp2);
 | |
|           g_object_unref (tmp2);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|   return retval;
 | |
| }
 | |
| 
 | |
| static GList *toplevels;
 | |
| static guint shot_id;
 | |
| static gboolean
 | |
| 
 | |
| window_is_csd (GdkWindow *window)
 | |
| {
 | |
|   gboolean set;
 | |
|   GdkWMDecoration decorations = 0;
 | |
| 
 | |
|   /* FIXME: is this accurate? */
 | |
|   set = gdk_window_get_decorations (window, &decorations);
 | |
|   return (set && (decorations == 0));
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| shoot_one (WidgetInfo *info)
 | |
| {
 | |
|   GdkWindow *window;
 | |
|   XID id;
 | |
|   GdkPixbuf *screenshot = NULL;
 | |
|   DecorationType decor = DECOR_FRAME;
 | |
| 
 | |
|   if (g_list_find (toplevels, info) == NULL)
 | |
|     {
 | |
|       g_warning ("Widget not found in queue");
 | |
|       gtk_main_quit ();
 | |
|     }
 | |
| 
 | |
|   window = gtk_widget_get_window (info->window);
 | |
|   id = gdk_x11_window_get_xid (window);
 | |
|   if (window_is_csd (window))
 | |
|     decor = (info->include_decorations) ? DECOR_NONE : DECOR_WINDOW_FRAME;
 | |
|   screenshot = take_window_shot (id, decor);
 | |
|   if (screenshot != NULL)
 | |
|     {
 | |
|       char *filename;
 | |
|       filename = g_strdup_printf ("./%s.png", info->name);
 | |
|       gdk_pixbuf_save (screenshot, filename, "png", NULL, NULL);
 | |
|       g_free (filename);
 | |
|       g_object_unref (screenshot);
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       g_warning ("unable to save shot of %s", info->name);
 | |
|     }
 | |
|   gtk_widget_destroy (info->window);
 | |
| 
 | |
|   shot_id = 0;
 | |
| 
 | |
|   /* remove from the queue and try to load up another */
 | |
|   toplevels = g_list_remove (toplevels, info);
 | |
|   if (toplevels == NULL)
 | |
|     gtk_main_quit ();
 | |
|   else
 | |
|     queue_show ();
 | |
| 
 | |
|   return G_SOURCE_REMOVE;
 | |
| }
 | |
| 
 | |
| static void
 | |
| on_show (WidgetInfo *info)
 | |
| {
 | |
|   if (shot_id != 0)
 | |
|     return;
 | |
| 
 | |
|   shot_id = g_timeout_add (500, (GSourceFunc) shoot_one, info);
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| show_one (void)
 | |
| {
 | |
|   WidgetInfo *info = toplevels->data;
 | |
| 
 | |
|   g_message ("shooting %s", info->name);
 | |
| 
 | |
|   g_signal_connect_swapped (info->window,
 | |
|                             "show",
 | |
|                             G_CALLBACK (on_show),
 | |
|                             info);
 | |
| 
 | |
|   gtk_widget_show (info->window);
 | |
| 
 | |
|   return G_SOURCE_REMOVE;
 | |
| }
 | |
| 
 | |
| static void
 | |
| queue_show (void)
 | |
| {
 | |
|   g_idle_add ((GSourceFunc) show_one, NULL);
 | |
| }
 | |
| 
 | |
| int main (int argc, char **argv)
 | |
| {
 | |
|   /* If there's no DISPLAY, we silently error out.  We don't want to break
 | |
|    * headless builds. */
 | |
|   if (! gtk_init_check (&argc, &argv))
 | |
|     return 0;
 | |
| 
 | |
|   toplevels = get_all_widgets ();
 | |
| 
 | |
|   queue_show ();
 | |
|   gtk_main ();
 | |
| 
 | |
|   return 0;
 | |
| }
 |