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;
 | 
						|
}
 |