Blame docs/tools/shooter.c

Packit 98cdb6
#include <gdk/gdk.h>
Packit 98cdb6
#include <gtk/gtk.h>
Packit 98cdb6
#include <gdkx.h>
Packit 98cdb6
#include <stdio.h>
Packit 98cdb6
#include <errno.h>
Packit 98cdb6
#include <sys/wait.h>
Packit 98cdb6
#include <unistd.h>
Packit 98cdb6
#include <X11/extensions/shape.h>
Packit 98cdb6
Packit 98cdb6
#include <gdk-pixbuf/gdk-pixbuf.h>
Packit 98cdb6
#include <sys/types.h>
Packit 98cdb6
#include <sys/stat.h>
Packit 98cdb6
#include <sys/wait.h>
Packit 98cdb6
#include <signal.h>
Packit 98cdb6
#include <unistd.h>
Packit 98cdb6
#include <stdlib.h>
Packit 98cdb6
#include <fcntl.h>
Packit 98cdb6
#include <errno.h>
Packit 98cdb6
#include <locale.h>
Packit 98cdb6
#include "widgets.h"
Packit 98cdb6
#include "shadow.h"
Packit 98cdb6
Packit 98cdb6
#define MAXIMUM_WM_REPARENTING_DEPTH 4
Packit 98cdb6
#ifndef _
Packit 98cdb6
#define _(x) (x)
Packit 98cdb6
#endif
Packit 98cdb6
Packit 98cdb6
static Window
Packit 98cdb6
find_toplevel_window (Window xid)
Packit 98cdb6
{
Packit 98cdb6
  Window root, parent, *children;
Packit 98cdb6
  guint nchildren;
Packit 98cdb6
Packit 98cdb6
  do
Packit 98cdb6
    {
Packit 98cdb6
      if (XQueryTree (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()), xid, &root,
Packit 98cdb6
		      &parent, &children, &nchildren) == 0)
Packit 98cdb6
	{
Packit 98cdb6
	  g_warning ("Couldn't find window manager window");
Packit 98cdb6
	  return 0;
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      if (root == parent)
Packit 98cdb6
	return xid;
Packit 98cdb6
Packit 98cdb6
      xid = parent;
Packit 98cdb6
    }
Packit 98cdb6
  while (TRUE);
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GdkPixbuf *
Packit 98cdb6
add_border_to_shot (GdkPixbuf *pixbuf)
Packit 98cdb6
{
Packit 98cdb6
  GdkPixbuf *retval;
Packit 98cdb6
Packit 98cdb6
  retval = gdk_pixbuf_new (GDK_COLORSPACE_RGB, TRUE, 8,
Packit 98cdb6
			   gdk_pixbuf_get_width (pixbuf) + 2,
Packit 98cdb6
			   gdk_pixbuf_get_height (pixbuf) + 2);
Packit 98cdb6
Packit 98cdb6
  /* Fill with solid black */
Packit 98cdb6
  gdk_pixbuf_fill (retval, 0xFF);
Packit 98cdb6
  gdk_pixbuf_copy_area (pixbuf,
Packit 98cdb6
			0, 0,
Packit 98cdb6
			gdk_pixbuf_get_width (pixbuf),
Packit 98cdb6
			gdk_pixbuf_get_height (pixbuf),
Packit 98cdb6
			retval, 1, 1);
Packit 98cdb6
Packit 98cdb6
  return retval;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GdkPixbuf *
Packit 98cdb6
remove_shaped_area (GdkPixbuf *pixbuf,
Packit 98cdb6
		    Window     window)
Packit 98cdb6
{
Packit 98cdb6
  GdkPixbuf *retval;
Packit 98cdb6
  XRectangle *rectangles;
Packit 98cdb6
  int rectangle_count, rectangle_order;
Packit 98cdb6
  int i;
Packit 98cdb6
Packit 98cdb6
  retval = gdk_pixbuf_new (GDK_COLORSPACE_RGB, TRUE, 8,
Packit 98cdb6
			   gdk_pixbuf_get_width (pixbuf),
Packit 98cdb6
			   gdk_pixbuf_get_height (pixbuf));
Packit 98cdb6
  
Packit 98cdb6
  gdk_pixbuf_fill (retval, 0);
Packit 98cdb6
  rectangles = XShapeGetRectangles (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()), window,
Packit 98cdb6
				    ShapeBounding, &rectangle_count, &rectangle_order);
Packit 98cdb6
Packit 98cdb6
  for (i = 0; i < rectangle_count; i++)
Packit 98cdb6
    {
Packit 98cdb6
      int y, x;
Packit 98cdb6
Packit 98cdb6
      for (y = rectangles[i].y; y < rectangles[i].y + rectangles[i].height; y++)
Packit 98cdb6
	{
Packit 98cdb6
	  guchar *src_pixels, *dest_pixels;
Packit 98cdb6
Packit 98cdb6
	  src_pixels = gdk_pixbuf_get_pixels (pixbuf) +
Packit 98cdb6
	    y * gdk_pixbuf_get_rowstride (pixbuf) +
Packit 98cdb6
	    rectangles[i].x * (gdk_pixbuf_get_has_alpha (pixbuf) ? 4 : 3);
Packit 98cdb6
	  dest_pixels = gdk_pixbuf_get_pixels (retval) +
Packit 98cdb6
	    y * gdk_pixbuf_get_rowstride (retval) +
Packit 98cdb6
	    rectangles[i].x * 4;
Packit 98cdb6
Packit 98cdb6
	  for (x = rectangles[i].x; x < rectangles[i].x + rectangles[i].width; x++)
Packit 98cdb6
	    {
Packit 98cdb6
	      *dest_pixels++ = *src_pixels ++;
Packit 98cdb6
	      *dest_pixels++ = *src_pixels ++;
Packit 98cdb6
	      *dest_pixels++ = *src_pixels ++;
Packit 98cdb6
	      *dest_pixels++ = 255;
Packit 98cdb6
Packit 98cdb6
	      if (gdk_pixbuf_get_has_alpha (pixbuf))
Packit 98cdb6
		src_pixels++;
Packit 98cdb6
	    }
Packit 98cdb6
	}
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return retval;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
static GdkPixbuf *
Packit 98cdb6
take_window_shot (Window   child,
Packit 98cdb6
		  gboolean include_decoration)
Packit 98cdb6
{
Packit 98cdb6
  GdkWindow *window;
Packit 98cdb6
  Display *disp;
Packit 98cdb6
  Window w, xid;
Packit 98cdb6
  gint x_orig, y_orig;
Packit 98cdb6
  gint x = 0, y = 0;
Packit 98cdb6
  gint width, height;
Packit 98cdb6
Packit 98cdb6
  GdkPixbuf *tmp, *tmp2;
Packit 98cdb6
  GdkPixbuf *retval;
Packit 98cdb6
Packit 98cdb6
  disp = GDK_DISPLAY_XDISPLAY (gdk_display_get_default ());
Packit 98cdb6
  w = GDK_ROOT_WINDOW ();
Packit 98cdb6
Packit 98cdb6
  if (include_decoration)
Packit 98cdb6
    xid = find_toplevel_window (child);
Packit 98cdb6
  else
Packit 98cdb6
    xid = child;
Packit 98cdb6
Packit 98cdb6
  window = gdk_window_foreign_new (xid);
Packit 98cdb6
Packit 98cdb6
  gdk_drawable_get_size (window, &width, &height);
Packit 98cdb6
  gdk_window_get_origin (window, &x_orig, &y_orig);
Packit 98cdb6
Packit 98cdb6
  if (x_orig < 0)
Packit 98cdb6
    {
Packit 98cdb6
      x = - x_orig;
Packit 98cdb6
      width = width + x_orig;
Packit 98cdb6
      x_orig = 0;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (y_orig < 0)
Packit 98cdb6
    {
Packit 98cdb6
      y = - y_orig;
Packit 98cdb6
      height = height + y_orig;
Packit 98cdb6
      y_orig = 0;
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  if (x_orig + width > gdk_screen_width ())
Packit 98cdb6
    width = gdk_screen_width () - x_orig;
Packit 98cdb6
Packit 98cdb6
  if (y_orig + height > gdk_screen_height ())
Packit 98cdb6
    height = gdk_screen_height () - y_orig;
Packit 98cdb6
Packit 98cdb6
  tmp = gdk_pixbuf_get_from_drawable (NULL, window, NULL,
Packit 98cdb6
				      x, y, 0, 0, width, height);
Packit 98cdb6
Packit 98cdb6
  if (include_decoration)
Packit 98cdb6
    tmp2 = remove_shaped_area (tmp, xid);
Packit 98cdb6
  else
Packit 98cdb6
    tmp2 = add_border_to_shot (tmp);
Packit 98cdb6
Packit 98cdb6
  retval = create_shadowed_pixbuf (tmp2);
Packit 98cdb6
  g_object_unref (tmp);
Packit 98cdb6
  g_object_unref (tmp2);
Packit 98cdb6
Packit 98cdb6
  return retval;
Packit 98cdb6
}
Packit 98cdb6
Packit 98cdb6
int main (int argc, char **argv)
Packit 98cdb6
{
Packit 98cdb6
  GList *toplevels;
Packit 98cdb6
  GdkPixbuf *screenshot = NULL;
Packit 98cdb6
  GList *node;
Packit 98cdb6
Packit 98cdb6
  /* If there's no DISPLAY, we silently error out.  We don't want to break
Packit 98cdb6
   * headless builds. */
Packit 98cdb6
  if (! gtk_init_check (&argc, &argv))
Packit 98cdb6
    return 0;
Packit 98cdb6
Packit 98cdb6
  toplevels = get_all_widgets ();
Packit 98cdb6
Packit 98cdb6
  for (node = toplevels; node; node = g_list_next (node))
Packit 98cdb6
    {
Packit 98cdb6
      GdkWindow *window;
Packit 98cdb6
      WidgetInfo *info;
Packit 98cdb6
      XID id;
Packit 98cdb6
      char *filename;
Packit 98cdb6
Packit 98cdb6
      info = node->data;
Packit 98cdb6
Packit 98cdb6
      gtk_widget_show (info->window);
Packit 98cdb6
Packit 98cdb6
      window = info->window->window;
Packit 98cdb6
Packit 98cdb6
      gtk_widget_show_now (info->window);
Packit 98cdb6
      gtk_widget_draw (info->window, &(info->window->allocation));
Packit 98cdb6
Packit 98cdb6
      while (gtk_events_pending ())
Packit 98cdb6
	{
Packit 98cdb6
	  gtk_main_iteration ();
Packit 98cdb6
	}
Packit 98cdb6
      sleep (1);
Packit 98cdb6
Packit 98cdb6
      while (gtk_events_pending ())
Packit 98cdb6
	{
Packit 98cdb6
	  gtk_main_iteration ();
Packit 98cdb6
	}
Packit 98cdb6
Packit 98cdb6
      id = gdk_x11_drawable_get_xid (GDK_DRAWABLE (window));
Packit 98cdb6
      screenshot = take_window_shot (id, info->include_decorations);
Packit 98cdb6
      filename = g_strdup_printf ("./%s.png", info->name);
Packit 98cdb6
      gdk_pixbuf_save (screenshot, filename, "png", NULL, NULL);
Packit 98cdb6
      g_free(filename);
Packit 98cdb6
      gtk_widget_hide (info->window);
Packit 98cdb6
    }
Packit 98cdb6
Packit 98cdb6
  return 0;
Packit 98cdb6
}