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