mirror of
https://gitlab.gnome.org/GNOME/gimp.git
synced 2025-07-03 17:33:25 +00:00

- libgimpbase: change GPParam to transfer all information about the GValues we use, in the same way done for GPParamDef. GPParam is now different from GimpParam from libgimp, pointers can't be casted any longer. The protocol is now completely GimpPDBArgType-free. Remove gp_params_destroy() from the public API. - libgimp: add API to convert between an array of GPParams and GimpValueArray, the latter is now the new official API for dealing with procedure arguments and return values, GimpParam is cruft (the wire now talks with GimpPlugIn more directly than with the members of GimpPlugInInfo, which need additional compat conversions). - libgimp, app: rename gimpgpparamspecs.[ch] to simply gimpgpparams.[ch] which is also more accurate because they now contain GValue functions too. The code that used to live in app/plug-in/plug-in-params.h is now completely in libgimp. - app: contains no protocol compat code any longer, the only place that uses GimpPDBArgType is the PDB query procedure implementation, which also needs to change. - app: change some forgotten int32 run-modes to enums.
924 lines
29 KiB
C
924 lines
29 KiB
C
/* GIMP - The GNU Image Manipulation Program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* 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 <string.h>
|
|
#include <errno.h>
|
|
|
|
#include <gegl.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#ifdef GDK_WINDOWING_X11
|
|
#include <gdk/gdkx.h>
|
|
#endif
|
|
|
|
#ifdef G_OS_WIN32
|
|
#include <windows.h>
|
|
#include <fcntl.h>
|
|
#include <io.h>
|
|
|
|
#ifndef pipe
|
|
#define pipe(fds) _pipe(fds, 4096, _O_BINARY)
|
|
#endif
|
|
#else
|
|
#include <sys/types.h>
|
|
#include <sys/wait.h>
|
|
#include <signal.h>
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
#include "libgimpbase/gimpbase.h"
|
|
#include "libgimpwidgets/gimpwidgets.h"
|
|
|
|
#include "gui-types.h"
|
|
|
|
#include "config/gimpguiconfig.h"
|
|
|
|
#include "core/gimp.h"
|
|
#include "core/gimp-parallel.h"
|
|
#include "core/gimp-spawn.h"
|
|
#include "core/gimp-utils.h"
|
|
#include "core/gimpasync.h"
|
|
#include "core/gimpbrush.h"
|
|
#include "core/gimpcancelable.h"
|
|
#include "core/gimpcontainer.h"
|
|
#include "core/gimpcontext.h"
|
|
#include "core/gimpgradient.h"
|
|
#include "core/gimpimage.h"
|
|
#include "core/gimpimagefile.h"
|
|
#include "core/gimplist.h"
|
|
#include "core/gimppalette.h"
|
|
#include "core/gimppattern.h"
|
|
#include "core/gimpprogress.h"
|
|
#include "core/gimpwaitable.h"
|
|
|
|
#include "text/gimpfont.h"
|
|
|
|
#include "pdb/gimppdb.h"
|
|
#include "pdb/gimpprocedure.h"
|
|
|
|
#include "plug-in/gimppluginmanager-file.h"
|
|
|
|
#include "widgets/gimpactiongroup.h"
|
|
#include "widgets/gimpbrushselect.h"
|
|
#include "widgets/gimpdialogfactory.h"
|
|
#include "widgets/gimpdocked.h"
|
|
#include "widgets/gimpfontselect.h"
|
|
#include "widgets/gimpgradientselect.h"
|
|
#include "widgets/gimphelp.h"
|
|
#include "widgets/gimphelp-ids.h"
|
|
#include "widgets/gimpmenufactory.h"
|
|
#include "widgets/gimppaletteselect.h"
|
|
#include "widgets/gimppatternselect.h"
|
|
#include "widgets/gimpprogressdialog.h"
|
|
#include "widgets/gimpuimanager.h"
|
|
#include "widgets/gimpwidgets-utils.h"
|
|
|
|
#include "display/gimpdisplay.h"
|
|
#include "display/gimpdisplay-foreach.h"
|
|
#include "display/gimpdisplayshell.h"
|
|
#include "display/gimpsinglewindowstrategy.h"
|
|
#include "display/gimpmultiwindowstrategy.h"
|
|
|
|
#include "actions/plug-in-actions.h"
|
|
|
|
#include "menus/menus.h"
|
|
|
|
#include "dialogs/color-profile-import-dialog.h"
|
|
|
|
#include "gui.h"
|
|
#include "gui-message.h"
|
|
#include "gui-vtable.h"
|
|
#include "icon-themes.h"
|
|
#include "themes.h"
|
|
|
|
|
|
/* local function prototypes */
|
|
|
|
static void gui_ungrab (Gimp *gimp);
|
|
|
|
static void gui_set_busy (Gimp *gimp);
|
|
static void gui_unset_busy (Gimp *gimp);
|
|
|
|
static void gui_help (Gimp *gimp,
|
|
GimpProgress *progress,
|
|
const gchar *help_domain,
|
|
const gchar *help_id);
|
|
static const gchar * gui_get_program_class (Gimp *gimp);
|
|
static gchar * gui_get_display_name (Gimp *gimp,
|
|
gint display_ID,
|
|
GObject **monitor,
|
|
gint *monitor_number);
|
|
static guint32 gui_get_user_time (Gimp *gimp);
|
|
static GFile * gui_get_theme_dir (Gimp *gimp);
|
|
static GFile * gui_get_icon_theme_dir (Gimp *gimp);
|
|
static GimpObject * gui_get_window_strategy (Gimp *gimp);
|
|
static GimpObject * gui_get_empty_display (Gimp *gimp);
|
|
static GimpObject * gui_display_get_by_ID (Gimp *gimp,
|
|
gint ID);
|
|
static gint gui_display_get_ID (GimpObject *display);
|
|
static guint32 gui_display_get_window_id (GimpObject *display);
|
|
static GimpObject * gui_display_create (Gimp *gimp,
|
|
GimpImage *image,
|
|
GimpUnit unit,
|
|
gdouble scale,
|
|
GObject *monitor);
|
|
static void gui_display_delete (GimpObject *display);
|
|
static void gui_displays_reconnect (Gimp *gimp,
|
|
GimpImage *old_image,
|
|
GimpImage *new_image);
|
|
static gboolean gui_wait (Gimp *gimp,
|
|
GimpWaitable *waitable,
|
|
const gchar *message);
|
|
static GimpProgress * gui_new_progress (Gimp *gimp,
|
|
GimpObject *display);
|
|
static void gui_free_progress (Gimp *gimp,
|
|
GimpProgress *progress);
|
|
static gboolean gui_pdb_dialog_new (Gimp *gimp,
|
|
GimpContext *context,
|
|
GimpProgress *progress,
|
|
GimpContainer *container,
|
|
const gchar *title,
|
|
const gchar *callback_name,
|
|
const gchar *object_name,
|
|
va_list args);
|
|
static gboolean gui_pdb_dialog_set (Gimp *gimp,
|
|
GimpContainer *container,
|
|
const gchar *callback_name,
|
|
const gchar *object_name,
|
|
va_list args);
|
|
static gboolean gui_pdb_dialog_close (Gimp *gimp,
|
|
GimpContainer *container,
|
|
const gchar *callback_name);
|
|
static gboolean gui_recent_list_add_file (Gimp *gimp,
|
|
GFile *file,
|
|
const gchar *mime_type);
|
|
static void gui_recent_list_load (Gimp *gimp);
|
|
|
|
static GMountOperation
|
|
* gui_get_mount_operation (Gimp *gimp,
|
|
GimpProgress *progress);
|
|
|
|
static GimpColorProfilePolicy
|
|
gui_query_profile_policy (Gimp *gimp,
|
|
GimpImage *image,
|
|
GimpContext *context,
|
|
GimpColorProfile **dest_profile,
|
|
GimpColorRenderingIntent *intent,
|
|
gboolean *bpc,
|
|
gboolean *dont_ask);
|
|
|
|
|
|
/* public functions */
|
|
|
|
void
|
|
gui_vtable_init (Gimp *gimp)
|
|
{
|
|
g_return_if_fail (GIMP_IS_GIMP (gimp));
|
|
|
|
gimp->gui.ungrab = gui_ungrab;
|
|
gimp->gui.set_busy = gui_set_busy;
|
|
gimp->gui.unset_busy = gui_unset_busy;
|
|
gimp->gui.show_message = gui_message;
|
|
gimp->gui.help = gui_help;
|
|
gimp->gui.get_program_class = gui_get_program_class;
|
|
gimp->gui.get_display_name = gui_get_display_name;
|
|
gimp->gui.get_user_time = gui_get_user_time;
|
|
gimp->gui.get_theme_dir = gui_get_theme_dir;
|
|
gimp->gui.get_icon_theme_dir = gui_get_icon_theme_dir;
|
|
gimp->gui.get_window_strategy = gui_get_window_strategy;
|
|
gimp->gui.get_empty_display = gui_get_empty_display;
|
|
gimp->gui.display_get_by_id = gui_display_get_by_ID;
|
|
gimp->gui.display_get_id = gui_display_get_ID;
|
|
gimp->gui.display_get_window_id = gui_display_get_window_id;
|
|
gimp->gui.display_create = gui_display_create;
|
|
gimp->gui.display_delete = gui_display_delete;
|
|
gimp->gui.displays_reconnect = gui_displays_reconnect;
|
|
gimp->gui.wait = gui_wait;
|
|
gimp->gui.progress_new = gui_new_progress;
|
|
gimp->gui.progress_free = gui_free_progress;
|
|
gimp->gui.pdb_dialog_new = gui_pdb_dialog_new;
|
|
gimp->gui.pdb_dialog_set = gui_pdb_dialog_set;
|
|
gimp->gui.pdb_dialog_close = gui_pdb_dialog_close;
|
|
gimp->gui.recent_list_add_file = gui_recent_list_add_file;
|
|
gimp->gui.recent_list_load = gui_recent_list_load;
|
|
gimp->gui.get_mount_operation = gui_get_mount_operation;
|
|
gimp->gui.query_profile_policy = gui_query_profile_policy;
|
|
}
|
|
|
|
|
|
/* private functions */
|
|
|
|
static void
|
|
gui_ungrab (Gimp *gimp)
|
|
{
|
|
GdkDisplay *display = gdk_display_get_default ();
|
|
|
|
if (display)
|
|
gdk_seat_ungrab (gdk_display_get_default_seat (display));
|
|
}
|
|
|
|
static void
|
|
gui_set_busy (Gimp *gimp)
|
|
{
|
|
gimp_displays_set_busy (gimp);
|
|
gimp_dialog_factory_set_busy (gimp_dialog_factory_get_singleton ());
|
|
|
|
gdk_display_flush (gdk_display_get_default ());
|
|
}
|
|
|
|
static void
|
|
gui_unset_busy (Gimp *gimp)
|
|
{
|
|
gimp_displays_unset_busy (gimp);
|
|
gimp_dialog_factory_unset_busy (gimp_dialog_factory_get_singleton ());
|
|
|
|
gdk_display_flush (gdk_display_get_default ());
|
|
}
|
|
|
|
static void
|
|
gui_help (Gimp *gimp,
|
|
GimpProgress *progress,
|
|
const gchar *help_domain,
|
|
const gchar *help_id)
|
|
{
|
|
gimp_help_show (gimp, progress, help_domain, help_id);
|
|
}
|
|
|
|
static const gchar *
|
|
gui_get_program_class (Gimp *gimp)
|
|
{
|
|
return gdk_get_program_class ();
|
|
}
|
|
|
|
static gint
|
|
get_monitor_number (GdkMonitor *monitor)
|
|
{
|
|
GdkDisplay *display = gdk_monitor_get_display (monitor);
|
|
gint n_monitors = gdk_display_get_n_monitors (display);
|
|
gint i;
|
|
|
|
for (i = 0; i < n_monitors; i++)
|
|
if (gdk_display_get_monitor (display, i) == monitor)
|
|
return i;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static gchar *
|
|
gui_get_display_name (Gimp *gimp,
|
|
gint display_ID,
|
|
GObject **monitor,
|
|
gint *monitor_number)
|
|
{
|
|
GimpDisplay *display = NULL;
|
|
GdkDisplay *gdk_display;
|
|
|
|
if (display_ID > 0)
|
|
display = gimp_display_get_by_ID (gimp, display_ID);
|
|
|
|
if (display)
|
|
{
|
|
GimpDisplayShell *shell = gimp_display_get_shell (display);
|
|
|
|
gdk_display = gtk_widget_get_display (GTK_WIDGET (shell));
|
|
|
|
*monitor = G_OBJECT (gimp_widget_get_monitor (GTK_WIDGET (shell)));
|
|
}
|
|
else
|
|
{
|
|
*monitor = G_OBJECT (gui_get_initial_monitor (gimp));
|
|
|
|
if (! *monitor)
|
|
*monitor = G_OBJECT (gimp_get_monitor_at_pointer ());
|
|
|
|
gdk_display = gdk_monitor_get_display (GDK_MONITOR (*monitor));
|
|
}
|
|
|
|
*monitor_number = get_monitor_number (GDK_MONITOR (*monitor));
|
|
|
|
return gdk_screen_make_display_name (gdk_display_get_default_screen (gdk_display));
|
|
}
|
|
|
|
static guint32
|
|
gui_get_user_time (Gimp *gimp)
|
|
{
|
|
#ifdef GDK_WINDOWING_X11
|
|
return gdk_x11_display_get_user_time (gdk_display_get_default ());
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
static GFile *
|
|
gui_get_theme_dir (Gimp *gimp)
|
|
{
|
|
return themes_get_theme_dir (gimp, GIMP_GUI_CONFIG (gimp->config)->theme);
|
|
}
|
|
|
|
static GFile *
|
|
gui_get_icon_theme_dir (Gimp *gimp)
|
|
{
|
|
return icon_themes_get_theme_dir (gimp, GIMP_GUI_CONFIG (gimp->config)->icon_theme);
|
|
}
|
|
|
|
static GimpObject *
|
|
gui_get_window_strategy (Gimp *gimp)
|
|
{
|
|
if (GIMP_GUI_CONFIG (gimp->config)->single_window_mode)
|
|
return gimp_single_window_strategy_get_singleton ();
|
|
else
|
|
return gimp_multi_window_strategy_get_singleton ();
|
|
}
|
|
|
|
static GimpObject *
|
|
gui_get_empty_display (Gimp *gimp)
|
|
{
|
|
GimpObject *display = NULL;
|
|
|
|
if (gimp_container_get_n_children (gimp->displays) == 1)
|
|
{
|
|
display = gimp_container_get_first_child (gimp->displays);
|
|
|
|
if (gimp_display_get_image (GIMP_DISPLAY (display)))
|
|
{
|
|
/* The display was not empty */
|
|
display = NULL;
|
|
}
|
|
}
|
|
|
|
return display;
|
|
}
|
|
|
|
static GimpObject *
|
|
gui_display_get_by_ID (Gimp *gimp,
|
|
gint ID)
|
|
{
|
|
return (GimpObject *) gimp_display_get_by_ID (gimp, ID);
|
|
}
|
|
|
|
static gint
|
|
gui_display_get_ID (GimpObject *display)
|
|
{
|
|
return gimp_display_get_ID (GIMP_DISPLAY (display));
|
|
}
|
|
|
|
static guint32
|
|
gui_display_get_window_id (GimpObject *display)
|
|
{
|
|
GimpDisplay *disp = GIMP_DISPLAY (display);
|
|
GimpDisplayShell *shell = gimp_display_get_shell (disp);
|
|
|
|
if (shell)
|
|
{
|
|
GtkWidget *toplevel = gtk_widget_get_toplevel (GTK_WIDGET (shell));
|
|
|
|
if (GTK_IS_WINDOW (toplevel))
|
|
return gimp_window_get_native_id (GTK_WINDOW (toplevel));
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static GimpObject *
|
|
gui_display_create (Gimp *gimp,
|
|
GimpImage *image,
|
|
GimpUnit unit,
|
|
gdouble scale,
|
|
GObject *monitor)
|
|
{
|
|
GimpContext *context = gimp_get_user_context (gimp);
|
|
GimpDisplay *display = GIMP_DISPLAY (gui_get_empty_display (gimp));
|
|
|
|
if (! monitor)
|
|
monitor = G_OBJECT (gimp_get_monitor_at_pointer ());
|
|
|
|
if (display)
|
|
{
|
|
gimp_display_fill (display, image, unit, scale);
|
|
}
|
|
else
|
|
{
|
|
GList *image_managers = gimp_ui_managers_from_name ("<Image>");
|
|
|
|
g_return_val_if_fail (image_managers != NULL, NULL);
|
|
|
|
display = gimp_display_new (gimp, image, unit, scale,
|
|
image_managers->data,
|
|
gimp_dialog_factory_get_singleton (),
|
|
GDK_MONITOR (monitor));
|
|
}
|
|
|
|
if (gimp_context_get_display (context) == display)
|
|
{
|
|
gimp_context_set_image (context, image);
|
|
gimp_context_display_changed (context);
|
|
}
|
|
else
|
|
{
|
|
gimp_context_set_display (context, display);
|
|
}
|
|
|
|
return GIMP_OBJECT (display);
|
|
}
|
|
|
|
static void
|
|
gui_display_delete (GimpObject *display)
|
|
{
|
|
gimp_display_close (GIMP_DISPLAY (display));
|
|
}
|
|
|
|
static void
|
|
gui_displays_reconnect (Gimp *gimp,
|
|
GimpImage *old_image,
|
|
GimpImage *new_image)
|
|
{
|
|
gimp_displays_reconnect (gimp, old_image, new_image);
|
|
}
|
|
|
|
static void
|
|
gui_wait_input_async (GimpAsync *async,
|
|
const gint input_pipe[2])
|
|
{
|
|
guint8 buffer[1];
|
|
|
|
while (read (input_pipe[0], buffer, sizeof (buffer)) == -1 &&
|
|
errno == EINTR);
|
|
|
|
gimp_async_finish (async, NULL);
|
|
}
|
|
|
|
static gboolean
|
|
gui_wait (Gimp *gimp,
|
|
GimpWaitable *waitable,
|
|
const gchar *message)
|
|
{
|
|
GimpProcedure *procedure;
|
|
GimpValueArray *args;
|
|
GimpAsync *input_async = NULL;
|
|
GError *error = NULL;
|
|
gint input_pipe[2];
|
|
gint output_pipe[2];
|
|
|
|
procedure = gimp_pdb_lookup_procedure (gimp->pdb, "plug-in-busy-dialog");
|
|
|
|
if (! procedure)
|
|
return FALSE;
|
|
|
|
if (pipe (input_pipe))
|
|
return FALSE;
|
|
|
|
if (pipe (output_pipe))
|
|
{
|
|
close (input_pipe[0]);
|
|
close (input_pipe[1]);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
gimp_spawn_set_cloexec (input_pipe[0]);
|
|
gimp_spawn_set_cloexec (output_pipe[1]);
|
|
|
|
args = gimp_procedure_get_arguments (procedure);
|
|
gimp_value_array_truncate (args, 5);
|
|
|
|
g_value_set_enum (gimp_value_array_index (args, 0),
|
|
GIMP_RUN_INTERACTIVE);
|
|
g_value_set_int (gimp_value_array_index (args, 1),
|
|
output_pipe[0]);
|
|
g_value_set_int (gimp_value_array_index (args, 2),
|
|
input_pipe[1]);
|
|
g_value_set_string (gimp_value_array_index (args, 3),
|
|
message);
|
|
g_value_set_int (gimp_value_array_index (args, 4),
|
|
GIMP_IS_CANCELABLE (waitable));
|
|
|
|
gimp_procedure_execute_async (procedure, gimp,
|
|
gimp_get_user_context (gimp),
|
|
NULL, args, NULL, &error);
|
|
|
|
gimp_value_array_unref (args);
|
|
|
|
close (input_pipe[1]);
|
|
close (output_pipe[0]);
|
|
|
|
if (error)
|
|
{
|
|
g_clear_error (&error);
|
|
|
|
close (input_pipe[0]);
|
|
close (output_pipe[1]);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
if (GIMP_IS_CANCELABLE (waitable))
|
|
{
|
|
/* listens for a cancellation request */
|
|
input_async = gimp_parallel_run_async_independent (
|
|
(GimpParallelRunAsyncFunc) gui_wait_input_async,
|
|
input_pipe);
|
|
|
|
while (! gimp_waitable_wait_for (waitable, 0.1 * G_TIME_SPAN_SECOND))
|
|
{
|
|
/* check for a cancellation request */
|
|
if (gimp_waitable_try_wait (GIMP_WAITABLE (input_async)))
|
|
{
|
|
gimp_cancelable_cancel (GIMP_CANCELABLE (waitable));
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
gimp_waitable_wait (waitable);
|
|
|
|
/* signal completion to the plug-in */
|
|
close (output_pipe[1]);
|
|
|
|
if (input_async)
|
|
{
|
|
gimp_waitable_wait (GIMP_WAITABLE (input_async));
|
|
|
|
g_object_unref (input_async);
|
|
}
|
|
|
|
close (input_pipe[0]);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static GimpProgress *
|
|
gui_new_progress (Gimp *gimp,
|
|
GimpObject *display)
|
|
{
|
|
g_return_val_if_fail (display == NULL || GIMP_IS_DISPLAY (display), NULL);
|
|
|
|
if (display)
|
|
return GIMP_PROGRESS (display);
|
|
|
|
return GIMP_PROGRESS (gimp_progress_dialog_new ());
|
|
}
|
|
|
|
static void
|
|
gui_free_progress (Gimp *gimp,
|
|
GimpProgress *progress)
|
|
{
|
|
g_return_if_fail (GIMP_IS_PROGRESS_DIALOG (progress));
|
|
|
|
if (GIMP_IS_PROGRESS_DIALOG (progress))
|
|
gtk_widget_destroy (GTK_WIDGET (progress));
|
|
}
|
|
|
|
static gboolean
|
|
gui_pdb_dialog_present (GtkWindow *window)
|
|
{
|
|
gtk_window_present (window);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
gui_pdb_dialog_new (Gimp *gimp,
|
|
GimpContext *context,
|
|
GimpProgress *progress,
|
|
GimpContainer *container,
|
|
const gchar *title,
|
|
const gchar *callback_name,
|
|
const gchar *object_name,
|
|
va_list args)
|
|
{
|
|
GType dialog_type = G_TYPE_NONE;
|
|
const gchar *dialog_role = NULL;
|
|
const gchar *help_id = NULL;
|
|
|
|
if (gimp_container_get_children_type (container) == GIMP_TYPE_BRUSH)
|
|
{
|
|
dialog_type = GIMP_TYPE_BRUSH_SELECT;
|
|
dialog_role = "gimp-brush-selection";
|
|
help_id = GIMP_HELP_BRUSH_DIALOG;
|
|
}
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_FONT)
|
|
{
|
|
dialog_type = GIMP_TYPE_FONT_SELECT;
|
|
dialog_role = "gimp-font-selection";
|
|
help_id = GIMP_HELP_FONT_DIALOG;
|
|
}
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_GRADIENT)
|
|
{
|
|
dialog_type = GIMP_TYPE_GRADIENT_SELECT;
|
|
dialog_role = "gimp-gradient-selection";
|
|
help_id = GIMP_HELP_GRADIENT_DIALOG;
|
|
}
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_PALETTE)
|
|
{
|
|
dialog_type = GIMP_TYPE_PALETTE_SELECT;
|
|
dialog_role = "gimp-palette-selection";
|
|
help_id = GIMP_HELP_PALETTE_DIALOG;
|
|
}
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_PATTERN)
|
|
{
|
|
dialog_type = GIMP_TYPE_PATTERN_SELECT;
|
|
dialog_role = "gimp-pattern-selection";
|
|
help_id = GIMP_HELP_PATTERN_DIALOG;
|
|
}
|
|
|
|
if (dialog_type != G_TYPE_NONE)
|
|
{
|
|
GimpObject *object = NULL;
|
|
|
|
if (object_name && strlen (object_name))
|
|
object = gimp_container_get_child_by_name (container, object_name);
|
|
|
|
if (! object)
|
|
object = gimp_context_get_by_type (context,
|
|
gimp_container_get_children_type (container));
|
|
|
|
if (object)
|
|
{
|
|
gint n_properties = 0;
|
|
gchar **names = NULL;
|
|
GValue *values = NULL;
|
|
GtkWidget *dialog;
|
|
GtkWidget *view;
|
|
|
|
names = gimp_properties_append (dialog_type,
|
|
&n_properties, names, &values,
|
|
"title", title,
|
|
"role", dialog_role,
|
|
"help-func", gimp_standard_help_func,
|
|
"help-id", help_id,
|
|
"pdb", gimp->pdb,
|
|
"context", context,
|
|
"select-type", gimp_container_get_children_type (container),
|
|
"initial-object", object,
|
|
"callback-name", callback_name,
|
|
"menu-factory", global_menu_factory,
|
|
NULL);
|
|
|
|
names = gimp_properties_append_valist (dialog_type,
|
|
&n_properties, names, &values,
|
|
args);
|
|
|
|
dialog = (GtkWidget *)
|
|
g_object_new_with_properties (dialog_type,
|
|
n_properties,
|
|
(const gchar **) names,
|
|
(const GValue *) values);
|
|
|
|
gimp_properties_free (n_properties, names, values);
|
|
|
|
view = GIMP_PDB_DIALOG (dialog)->view;
|
|
if (view)
|
|
gimp_docked_set_show_button_bar (GIMP_DOCKED (view), FALSE);
|
|
|
|
if (progress)
|
|
{
|
|
guint32 window_id = gimp_progress_get_window_id (progress);
|
|
|
|
if (window_id)
|
|
gimp_window_set_transient_for (GTK_WINDOW (dialog), window_id);
|
|
}
|
|
|
|
gtk_widget_show (dialog);
|
|
|
|
/* workaround for bug #360106 */
|
|
{
|
|
GSource *source = g_timeout_source_new (100);
|
|
GClosure *closure;
|
|
|
|
closure = g_cclosure_new_object (G_CALLBACK (gui_pdb_dialog_present),
|
|
G_OBJECT (dialog));
|
|
|
|
g_source_set_closure (source, closure);
|
|
g_source_attach (source, NULL);
|
|
g_source_unref (source);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
gui_pdb_dialog_set (Gimp *gimp,
|
|
GimpContainer *container,
|
|
const gchar *callback_name,
|
|
const gchar *object_name,
|
|
va_list args)
|
|
{
|
|
GimpPdbDialogClass *klass = NULL;
|
|
|
|
if (gimp_container_get_children_type (container) == GIMP_TYPE_BRUSH)
|
|
klass = g_type_class_peek (GIMP_TYPE_BRUSH_SELECT);
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_FONT)
|
|
klass = g_type_class_peek (GIMP_TYPE_FONT_SELECT);
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_GRADIENT)
|
|
klass = g_type_class_peek (GIMP_TYPE_GRADIENT_SELECT);
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_PALETTE)
|
|
klass = g_type_class_peek (GIMP_TYPE_PALETTE_SELECT);
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_PATTERN)
|
|
klass = g_type_class_peek (GIMP_TYPE_PATTERN_SELECT);
|
|
|
|
if (klass)
|
|
{
|
|
GimpPdbDialog *dialog;
|
|
|
|
dialog = gimp_pdb_dialog_get_by_callback (klass, callback_name);
|
|
|
|
if (dialog && dialog->select_type == gimp_container_get_children_type (container))
|
|
{
|
|
GimpObject *object;
|
|
|
|
object = gimp_container_get_child_by_name (container, object_name);
|
|
|
|
if (object)
|
|
{
|
|
const gchar *prop_name = va_arg (args, const gchar *);
|
|
|
|
gimp_context_set_by_type (dialog->context, dialog->select_type,
|
|
object);
|
|
|
|
if (prop_name)
|
|
g_object_set_valist (G_OBJECT (dialog), prop_name, args);
|
|
|
|
gtk_window_present (GTK_WINDOW (dialog));
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
gui_pdb_dialog_close (Gimp *gimp,
|
|
GimpContainer *container,
|
|
const gchar *callback_name)
|
|
{
|
|
GimpPdbDialogClass *klass = NULL;
|
|
|
|
if (gimp_container_get_children_type (container) == GIMP_TYPE_BRUSH)
|
|
klass = g_type_class_peek (GIMP_TYPE_BRUSH_SELECT);
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_FONT)
|
|
klass = g_type_class_peek (GIMP_TYPE_FONT_SELECT);
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_GRADIENT)
|
|
klass = g_type_class_peek (GIMP_TYPE_GRADIENT_SELECT);
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_PALETTE)
|
|
klass = g_type_class_peek (GIMP_TYPE_PALETTE_SELECT);
|
|
else if (gimp_container_get_children_type (container) == GIMP_TYPE_PATTERN)
|
|
klass = g_type_class_peek (GIMP_TYPE_PATTERN_SELECT);
|
|
|
|
if (klass)
|
|
{
|
|
GimpPdbDialog *dialog;
|
|
|
|
dialog = gimp_pdb_dialog_get_by_callback (klass, callback_name);
|
|
|
|
if (dialog && dialog->select_type == gimp_container_get_children_type (container))
|
|
{
|
|
gtk_widget_destroy (GTK_WIDGET (dialog));
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
gui_recent_list_add_file (Gimp *gimp,
|
|
GFile *file,
|
|
const gchar *mime_type)
|
|
{
|
|
GtkRecentData recent;
|
|
const gchar *groups[2] = { "Graphics", NULL };
|
|
gchar *uri;
|
|
gboolean success;
|
|
|
|
g_return_val_if_fail (GIMP_IS_GIMP (gimp), FALSE);
|
|
g_return_val_if_fail (G_IS_FILE (file), FALSE);
|
|
|
|
/* use last part of the URI */
|
|
recent.display_name = NULL;
|
|
|
|
/* no special description */
|
|
recent.description = NULL;
|
|
recent.mime_type = (mime_type ?
|
|
(gchar *) mime_type : "application/octet-stream");
|
|
recent.app_name = "GNU Image Manipulation Program";
|
|
recent.app_exec = GIMP_COMMAND " %u";
|
|
recent.groups = (gchar **) groups;
|
|
recent.is_private = FALSE;
|
|
|
|
uri = g_file_get_uri (file);
|
|
|
|
success = gtk_recent_manager_add_full (gtk_recent_manager_get_default (),
|
|
uri, &recent);
|
|
|
|
g_free (uri);
|
|
|
|
return success;
|
|
}
|
|
|
|
static gint
|
|
gui_recent_list_compare (gconstpointer a,
|
|
gconstpointer b)
|
|
{
|
|
return (gtk_recent_info_get_modified ((GtkRecentInfo *) a) -
|
|
gtk_recent_info_get_modified ((GtkRecentInfo *) b));
|
|
}
|
|
|
|
static void
|
|
gui_recent_list_load (Gimp *gimp)
|
|
{
|
|
GList *items;
|
|
GList *list;
|
|
|
|
g_return_if_fail (GIMP_IS_GIMP (gimp));
|
|
|
|
gimp_container_freeze (gimp->documents);
|
|
gimp_container_clear (gimp->documents);
|
|
|
|
items = gtk_recent_manager_get_items (gtk_recent_manager_get_default ());
|
|
|
|
items = g_list_sort (items, gui_recent_list_compare);
|
|
|
|
for (list = items; list; list = list->next)
|
|
{
|
|
GtkRecentInfo *info = list->data;
|
|
|
|
if (gtk_recent_info_has_application (info,
|
|
"GNU Image Manipulation Program"))
|
|
{
|
|
const gchar *mime_type = gtk_recent_info_get_mime_type (info);
|
|
|
|
if (mime_type &&
|
|
gimp_plug_in_manager_file_procedure_find_by_mime_type (gimp->plug_in_manager,
|
|
GIMP_FILE_PROCEDURE_GROUP_OPEN,
|
|
mime_type))
|
|
{
|
|
GimpImagefile *imagefile;
|
|
GFile *file;
|
|
|
|
file = g_file_new_for_uri (gtk_recent_info_get_uri (info));
|
|
imagefile = gimp_imagefile_new (gimp, file);
|
|
g_object_unref (file);
|
|
|
|
gimp_imagefile_set_mime_type (imagefile, mime_type);
|
|
|
|
gimp_container_add (gimp->documents, GIMP_OBJECT (imagefile));
|
|
g_object_unref (imagefile);
|
|
}
|
|
}
|
|
|
|
gtk_recent_info_unref (info);
|
|
}
|
|
|
|
g_list_free (items);
|
|
|
|
gimp_container_thaw (gimp->documents);
|
|
}
|
|
|
|
static GMountOperation *
|
|
gui_get_mount_operation (Gimp *gimp,
|
|
GimpProgress *progress)
|
|
{
|
|
GtkWidget *toplevel = NULL;
|
|
|
|
if (GTK_IS_WIDGET (progress))
|
|
toplevel = gtk_widget_get_toplevel (GTK_WIDGET (progress));
|
|
|
|
return gtk_mount_operation_new (GTK_WINDOW (toplevel));
|
|
}
|
|
|
|
static GimpColorProfilePolicy
|
|
gui_query_profile_policy (Gimp *gimp,
|
|
GimpImage *image,
|
|
GimpContext *context,
|
|
GimpColorProfile **dest_profile,
|
|
GimpColorRenderingIntent *intent,
|
|
gboolean *bpc,
|
|
gboolean *dont_ask)
|
|
{
|
|
return color_profile_import_dialog_run (image, context, NULL,
|
|
dest_profile,
|
|
intent, bpc,
|
|
dont_ask);
|
|
}
|