mirror of https://github.com/GNOME/gimp.git
207 lines
6.2 KiB
C
207 lines
6.2 KiB
C
/* GIMP - The GNU Image Manipulation Program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* Screenshot plug-in
|
|
* Copyright 1998-2007 Sven Neumann <sven@gimp.org>
|
|
* Copyright 2003 Henrik Brix Andersen <brix@gimp.org>
|
|
* Copyright 2016 Michael Natterer <mitch@gimp.org>
|
|
* Copyright 2017 Jehan <jehan@gimp.org>
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 3 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <glib.h>
|
|
#include <glib/gstdio.h> /* g_unlink() */
|
|
|
|
#include <libgimp/gimp.h>
|
|
#include <libgimp/gimpui.h>
|
|
|
|
#include "screenshot.h"
|
|
#include "screenshot-kwin.h"
|
|
|
|
|
|
static GDBusProxy *proxy = NULL;
|
|
|
|
|
|
gboolean
|
|
screenshot_kwin_available (void)
|
|
{
|
|
proxy = g_dbus_proxy_new_for_bus_sync (G_BUS_TYPE_SESSION,
|
|
G_DBUS_PROXY_FLAGS_DO_NOT_AUTO_START,
|
|
NULL,
|
|
"org.kde.KWin",
|
|
"/Screenshot",
|
|
"org.kde.kwin.Screenshot",
|
|
NULL, NULL);
|
|
|
|
if (proxy)
|
|
{
|
|
GError *error = NULL;
|
|
|
|
g_dbus_proxy_call_sync (proxy, "org.freedesktop.DBus.Peer.Ping",
|
|
NULL,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1, NULL, &error);
|
|
if (! error)
|
|
return TRUE;
|
|
|
|
g_clear_error (&error);
|
|
|
|
g_object_unref (proxy);
|
|
proxy = NULL;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
ScreenshotCapabilities
|
|
screenshot_kwin_get_capabilities (void)
|
|
{
|
|
return (SCREENSHOT_CAN_SHOOT_DECORATIONS |
|
|
SCREENSHOT_CAN_SHOOT_POINTER |
|
|
SCREENSHOT_CAN_SHOOT_WINDOW |
|
|
SCREENSHOT_CAN_PICK_WINDOW);
|
|
/* TODO: SCREENSHOT_CAN_SHOOT_REGION.
|
|
* The KDE API has "screenshotArea" method but no method to get
|
|
* coordinates could be found. See below.
|
|
*/
|
|
}
|
|
|
|
GimpPDBStatusType
|
|
screenshot_kwin_shoot (ScreenshotValues *shootvals,
|
|
GdkMonitor *monitor,
|
|
GimpImage **image,
|
|
GError **error)
|
|
{
|
|
gchar *filename = NULL;
|
|
const gchar *method = NULL;
|
|
GVariant *args = NULL;
|
|
GVariant *retval;
|
|
gint32 mask;
|
|
|
|
switch (shootvals->shoot_type)
|
|
{
|
|
case SHOOT_ROOT:
|
|
if (shootvals->screenshot_delay > 0)
|
|
screenshot_delay (shootvals->screenshot_delay);
|
|
else
|
|
{
|
|
/* As an exception, I force a delay of at least 0.5 seconds
|
|
* for KWin. Because of windows effect slowly fading out, the
|
|
* screenshot plug-in GUI was constantly visible (with
|
|
* transparency as it is fading out) in 0s-delay screenshots.
|
|
*/
|
|
g_usleep (500000);
|
|
}
|
|
|
|
method = "screenshotFullscreen";
|
|
args = g_variant_new ("(b)", shootvals->show_cursor);
|
|
|
|
/* FIXME: figure profile */
|
|
break;
|
|
|
|
case SHOOT_REGION:
|
|
break;
|
|
/* FIXME: GNOME-shell has a "SelectArea" returning coordinates
|
|
* which can be fed to "ScreenshotArea". KDE has the equivalent
|
|
* "screenshotArea", but no "SelectArea" equivalent that I could
|
|
* find.
|
|
* Also at first, I expected "interactive" method to take care of
|
|
* the whole selecting-are-then-screenshotting workflow, but this
|
|
* is apparently only made to select interactively a specific
|
|
* window, not an area.
|
|
*/
|
|
method = "screenshotArea";
|
|
args = g_variant_new ("(iiii)",
|
|
shootvals->x1,
|
|
shootvals->y1,
|
|
shootvals->x2 - shootvals->x1,
|
|
shootvals->y2 - shootvals->y1);
|
|
args = NULL;
|
|
|
|
break;
|
|
|
|
case SHOOT_WINDOW:
|
|
if (shootvals->select_delay > 0)
|
|
screenshot_delay (shootvals->select_delay);
|
|
|
|
/* XXX I expected "screenshotWindowUnderCursor" method to be the
|
|
* right one, but it returns nothing, nor is there a file
|
|
* descriptor in argument. So I don't understand how to grab the
|
|
* screenshot. Also "interactive" changes the cursor to a
|
|
* crosshair, waiting for click, which is more helpful than
|
|
* immediate screenshot under cursor.
|
|
*/
|
|
method = "interactive";
|
|
mask = (shootvals->decorate ? 1 : 0) |
|
|
(shootvals->show_cursor ? 1 << 1 : 0);
|
|
args = g_variant_new ("(i)", mask);
|
|
|
|
/* FIXME: figure monitor */
|
|
break;
|
|
}
|
|
|
|
retval = g_dbus_proxy_call_sync (proxy, method, args,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1, NULL, error);
|
|
if (! retval)
|
|
goto failure;
|
|
|
|
g_variant_get (retval, "(s)",
|
|
&filename);
|
|
g_variant_unref (retval);
|
|
|
|
if (filename)
|
|
{
|
|
GimpColorProfile *profile;
|
|
|
|
*image = gimp_file_load (GIMP_RUN_NONINTERACTIVE,
|
|
g_file_new_for_path (filename));
|
|
gimp_image_set_file (*image, g_file_new_for_path ("screenshot.png"));
|
|
|
|
/* This is very wrong in multi-display setups since we have no
|
|
* idea which profile is to be used. Let's keep it anyway and
|
|
* assume always the monitor 0, which will still work in common
|
|
* cases.
|
|
*/
|
|
profile = gimp_monitor_get_color_profile (monitor);
|
|
|
|
if (profile)
|
|
{
|
|
gimp_image_set_color_profile (*image, profile);
|
|
g_object_unref (profile);
|
|
}
|
|
|
|
g_unlink (filename);
|
|
g_free (filename);
|
|
|
|
g_object_unref (proxy);
|
|
proxy = NULL;
|
|
|
|
return GIMP_PDB_SUCCESS;
|
|
}
|
|
|
|
failure:
|
|
|
|
if (filename)
|
|
g_free (filename);
|
|
|
|
g_object_unref (proxy);
|
|
proxy = NULL;
|
|
|
|
return GIMP_PDB_EXECUTION_ERROR;
|
|
}
|