
2003-11-14 Michael Natterer <mitch@gimp.org> * libgimpwidgets/gimpquerybox.c * app/widgets/gimpwidgets-utils.c: configure the labels in the message dialog and the query boxes to do automatic word wrapping to be HIG compliant. * app/app_procs.c * app/batch.c * app/config/gimpconfig-deserialize.c * app/config/gimpconfig-path.c * app/config/gimpconfig-utils.c * app/config/gimpconfigwriter.c * app/config/gimpscanner.c * app/core/gimpbrush.c * app/core/gimpbrushgenerated.c * app/core/gimpbrushpipe.c * app/core/gimpdatafactory.c * app/core/gimpgradient.c * app/core/gimpimage-merge.c * app/core/gimpimage.c * app/core/gimpimagefile.c * app/core/gimplayer-floating-sel.c * app/core/gimppalette.c * app/core/gimppattern.c * app/core/gimpselection.c * app/display/gimpdisplayshell.c * app/file/file-utils.c * app/gui/brush-select.c * app/gui/dialogs-commands.c * app/gui/drawable-commands.c * app/gui/edit-commands.c * app/gui/file-commands.c * app/gui/file-new-dialog.c * app/gui/font-select.c * app/gui/gradient-select.c * app/gui/gui.c * app/gui/image-commands.c * app/gui/layers-commands.c * app/gui/palette-select.c * app/gui/palettes-commands.c * app/gui/pattern-select.c * app/gui/preferences-dialog.c * app/gui/select-commands.c * app/gui/stroke-dialog.c * app/gui/tool-options-menu.c * app/gui/vectors-commands.c * app/gui/view-commands.c * app/plug-in/plug-in-message.c * app/plug-in/plug-in.c * app/plug-in/plug-ins.c * app/text/gimptextlayer-xcf.c * app/text/gimptextlayer.c * app/tools/gimpcurvestool.c * app/tools/gimphuesaturationtool.c * app/tools/gimplevelstool.c * app/tools/gimptransformtool.c * app/vectors/gimpvectors-export.c * app/widgets/gimpdatafactoryview.c * app/widgets/gimphelp.c * app/widgets/gimptemplateview.c * app/widgets/gimptooloptionseditor.c * app/xcf/xcf.c * tools/pdbgen/pdb/image.pdb: removed explicit newlines from messages. Reduced number of translatable strings by making many file error messages the same. Quote single words and filenames with 'foo', not "foo". Replaced some more "drawable" by "layer". General message cleanup and consistency check. * app/pdb/image_cmds.c: regenerated.
599 lines
16 KiB
C
599 lines
16 KiB
C
/* The GIMP -- an 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 2 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "libgimpwidgets/gimpwidgets.h"
|
|
|
|
#include "gui-types.h"
|
|
|
|
#include "core/gimp.h"
|
|
#include "core/gimpchannel.h"
|
|
#include "core/gimpcontext.h"
|
|
#include "core/gimpimage.h"
|
|
#include "core/gimpimage-crop.h"
|
|
#include "core/gimpimage-duplicate.h"
|
|
#include "core/gimpimage-flip.h"
|
|
#include "core/gimpimage-merge.h"
|
|
#include "core/gimpimage-resize.h"
|
|
#include "core/gimpimage-rotate.h"
|
|
#include "core/gimpimage-scale.h"
|
|
#include "core/gimpimage-undo.h"
|
|
|
|
#include "widgets/gimphelp-ids.h"
|
|
#include "widgets/gimpviewabledialog.h"
|
|
|
|
#include "display/gimpdisplay.h"
|
|
#include "display/gimpdisplayshell.h"
|
|
#include "display/gimpprogress.h"
|
|
|
|
#include "convert-dialog.h"
|
|
#include "image-commands.h"
|
|
#include "grid-dialog.h"
|
|
#include "resize-dialog.h"
|
|
|
|
#include "gimp-intl.h"
|
|
|
|
|
|
typedef struct
|
|
{
|
|
Resize *resize;
|
|
GimpDisplay *gdisp;
|
|
GimpImage *gimage;
|
|
} ImageResize;
|
|
|
|
|
|
#define return_if_no_display(gdisp,data) \
|
|
if (GIMP_IS_DISPLAY (data)) \
|
|
gdisp = data; \
|
|
else if (GIMP_IS_GIMP (data)) \
|
|
gdisp = gimp_context_get_display (gimp_get_user_context (GIMP (data))); \
|
|
else \
|
|
gdisp = NULL; \
|
|
if (! gdisp) \
|
|
return
|
|
|
|
#define return_if_no_image(gimage,data) \
|
|
if (GIMP_IS_DISPLAY (data)) \
|
|
gimage = ((GimpDisplay *) data)->gimage; \
|
|
else if (GIMP_IS_GIMP (data)) \
|
|
gimage = gimp_context_get_image (gimp_get_user_context (GIMP (data))); \
|
|
else \
|
|
gimage = NULL; \
|
|
if (! gimage) \
|
|
return
|
|
|
|
|
|
/* local functions */
|
|
|
|
static void image_resize_callback (GtkWidget *widget,
|
|
gpointer data);
|
|
static void image_scale_callback (GtkWidget *widget,
|
|
gpointer data);
|
|
static void image_scale_warn_callback (GtkWidget *widget,
|
|
gboolean do_scale,
|
|
gpointer data);
|
|
static void image_scale_implement (ImageResize *image_scale);
|
|
|
|
|
|
/* public functions */
|
|
|
|
void
|
|
image_convert_rgb_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
return_if_no_image (gimage, data);
|
|
|
|
convert_to_rgb (gimage);
|
|
}
|
|
|
|
void
|
|
image_convert_grayscale_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
return_if_no_image (gimage, data);
|
|
|
|
convert_to_grayscale (gimage);
|
|
}
|
|
|
|
void
|
|
image_convert_indexed_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
return_if_no_image (gimage, data);
|
|
|
|
convert_to_indexed (gimage, widget);
|
|
}
|
|
|
|
void
|
|
image_resize_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpDisplay *gdisp;
|
|
GimpImage *gimage;
|
|
ImageResize *image_resize;
|
|
return_if_no_display (gdisp, data);
|
|
|
|
gimage = gdisp->gimage;
|
|
|
|
image_resize = g_new0 (ImageResize, 1);
|
|
|
|
image_resize->gdisp = gdisp;
|
|
image_resize->gimage = gimage;
|
|
|
|
image_resize->resize =
|
|
resize_widget_new (GIMP_VIEWABLE (gimage), gdisp->shell,
|
|
ResizeWidget,
|
|
gimage->width,
|
|
gimage->height,
|
|
gimage->xresolution,
|
|
gimage->yresolution,
|
|
gimage->unit,
|
|
GIMP_DISPLAY_SHELL (gdisp->shell)->dot_for_dot,
|
|
G_CALLBACK (image_resize_callback),
|
|
image_resize);
|
|
|
|
g_signal_connect_object (gdisp, "disconnect",
|
|
G_CALLBACK (gtk_widget_destroy),
|
|
image_resize->resize->resize_shell,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
g_object_weak_ref (G_OBJECT (image_resize->resize->resize_shell),
|
|
(GWeakNotify) g_free,
|
|
image_resize);
|
|
|
|
gtk_widget_show (image_resize->resize->resize_shell);
|
|
}
|
|
|
|
void
|
|
image_scale_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpDisplay *gdisp;
|
|
GimpImage *gimage;
|
|
ImageResize *image_scale;
|
|
return_if_no_display (gdisp, data);
|
|
|
|
gimage = gdisp->gimage;
|
|
|
|
image_scale = g_new0 (ImageResize, 1);
|
|
|
|
image_scale->gdisp = gdisp;
|
|
image_scale->gimage = gimage;
|
|
|
|
image_scale->resize =
|
|
resize_widget_new (GIMP_VIEWABLE (gimage), gdisp->shell,
|
|
ScaleWidget,
|
|
gimage->width,
|
|
gimage->height,
|
|
gimage->xresolution,
|
|
gimage->yresolution,
|
|
gimage->unit,
|
|
GIMP_DISPLAY_SHELL (gdisp->shell)->dot_for_dot,
|
|
G_CALLBACK (image_scale_callback),
|
|
image_scale);
|
|
|
|
g_signal_connect_object (gdisp, "disconnect",
|
|
G_CALLBACK (gtk_widget_destroy),
|
|
image_scale->resize->resize_shell,
|
|
G_CONNECT_SWAPPED);
|
|
|
|
g_object_weak_ref (G_OBJECT (image_scale->resize->resize_shell),
|
|
(GWeakNotify) g_free,
|
|
image_scale);
|
|
|
|
gtk_widget_show (image_scale->resize->resize_shell);
|
|
}
|
|
|
|
void
|
|
image_flip_cmd_callback (GtkWidget *widget,
|
|
gpointer data,
|
|
guint action)
|
|
{
|
|
GimpDisplay *gdisp;
|
|
GimpProgress *progress;
|
|
return_if_no_display (gdisp, data);
|
|
|
|
progress = gimp_progress_start (gdisp, _("Flipping..."), TRUE, NULL, NULL);
|
|
|
|
gimp_image_flip (gdisp->gimage, (GimpOrientationType) action,
|
|
gimp_progress_update_and_flush, progress);
|
|
|
|
gimp_progress_end (progress);
|
|
|
|
gimp_image_flush (gdisp->gimage);
|
|
}
|
|
|
|
void
|
|
image_rotate_cmd_callback (GtkWidget *widget,
|
|
gpointer data,
|
|
guint action)
|
|
{
|
|
GimpDisplay *gdisp;
|
|
GimpProgress *progress;
|
|
return_if_no_display (gdisp, data);
|
|
|
|
progress = gimp_progress_start (gdisp, _("Rotating..."), TRUE, NULL, NULL);
|
|
|
|
gimp_image_rotate (gdisp->gimage, (GimpRotationType) action,
|
|
gimp_progress_update_and_flush, progress);
|
|
|
|
gimp_progress_end (progress);
|
|
|
|
gimp_image_flush (gdisp->gimage);
|
|
}
|
|
|
|
void
|
|
image_crop_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpDisplay *gdisp;
|
|
gint x1, y1, x2, y2;
|
|
return_if_no_display (gdisp, data);
|
|
|
|
if (! gimp_channel_bounds (gimp_image_get_mask (gdisp->gimage),
|
|
&x1, &y1, &x2, &y2))
|
|
{
|
|
g_message (_("Cannot crop because the current selection is empty."));
|
|
return;
|
|
}
|
|
|
|
gimp_image_crop (gdisp->gimage, x1, y1, x2, y2, FALSE, TRUE);
|
|
gimp_image_flush (gdisp->gimage);
|
|
}
|
|
|
|
void
|
|
image_duplicate_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
GimpImage *new_gimage;
|
|
return_if_no_image (gimage, data);
|
|
|
|
new_gimage = gimp_image_duplicate (gimage);
|
|
|
|
gimp_create_display (new_gimage->gimp, new_gimage, 0x0101);
|
|
|
|
g_object_unref (new_gimage);
|
|
}
|
|
|
|
void
|
|
image_merge_layers_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
return_if_no_image (gimage, data);
|
|
|
|
image_layers_merge_query (gimage, TRUE, widget);
|
|
}
|
|
|
|
void
|
|
image_flatten_image_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
return_if_no_image (gimage, data);
|
|
|
|
gimp_image_flatten (gimage);
|
|
gimp_image_flush (gimage);
|
|
}
|
|
|
|
void
|
|
image_configure_grid_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpDisplay *gdisp;
|
|
GimpDisplayShell *shell;
|
|
GimpImage *gimage;
|
|
return_if_no_display (gdisp, data);
|
|
|
|
shell = GIMP_DISPLAY_SHELL (gdisp->shell);
|
|
gimage = GIMP_IMAGE (gdisp->gimage);
|
|
|
|
if (! shell->grid_dialog)
|
|
{
|
|
shell->grid_dialog = grid_dialog_new (GIMP_IMAGE (gimage), widget);
|
|
|
|
gtk_window_set_transient_for (GTK_WINDOW (shell->grid_dialog),
|
|
GTK_WINDOW (shell));
|
|
gtk_window_set_destroy_with_parent (GTK_WINDOW (shell->grid_dialog),
|
|
TRUE);
|
|
|
|
g_object_add_weak_pointer (G_OBJECT (shell->grid_dialog),
|
|
(gpointer *) &shell->grid_dialog);
|
|
}
|
|
|
|
gtk_window_present (GTK_WINDOW (shell->grid_dialog));
|
|
}
|
|
|
|
|
|
/****************************/
|
|
/* The layer merge dialog */
|
|
/****************************/
|
|
|
|
typedef struct _LayerMergeOptions LayerMergeOptions;
|
|
|
|
struct _LayerMergeOptions
|
|
{
|
|
GtkWidget *query_box;
|
|
GimpImage *gimage;
|
|
gboolean merge_visible;
|
|
GimpMergeType merge_type;
|
|
};
|
|
|
|
static void
|
|
image_layers_merge_query_response (GtkWidget *widget,
|
|
gint response_id,
|
|
LayerMergeOptions *options)
|
|
{
|
|
GimpImage *gimage = options->gimage;
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
if (response_id == GTK_RESPONSE_OK)
|
|
{
|
|
if (options->merge_visible)
|
|
gimp_image_merge_visible_layers (gimage, options->merge_type);
|
|
|
|
gimp_image_flush (gimage);
|
|
}
|
|
|
|
gtk_widget_destroy (options->query_box);
|
|
}
|
|
|
|
void
|
|
image_layers_merge_query (GimpImage *gimage,
|
|
/* if FALSE, anchor active layer */
|
|
gboolean merge_visible,
|
|
GtkWidget *parent)
|
|
{
|
|
LayerMergeOptions *options;
|
|
GtkWidget *vbox;
|
|
GtkWidget *frame;
|
|
|
|
/* The new options structure */
|
|
options = g_new (LayerMergeOptions, 1);
|
|
options->gimage = gimage;
|
|
options->merge_visible = merge_visible;
|
|
options->merge_type = GIMP_EXPAND_AS_NECESSARY;
|
|
|
|
/* The dialog */
|
|
options->query_box =
|
|
gimp_viewable_dialog_new (GIMP_VIEWABLE (gimage),
|
|
_("Merge Layers"), "gimp-image-merge-layers",
|
|
GIMP_STOCK_MERGE_DOWN,
|
|
_("Layers Merge Options"),
|
|
parent,
|
|
gimp_standard_help_func,
|
|
GIMP_HELP_IMAGE_MERGE_LAYERS,
|
|
|
|
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
|
|
GTK_STOCK_OK, GTK_RESPONSE_OK,
|
|
|
|
NULL);
|
|
|
|
g_signal_connect (options->query_box, "response",
|
|
G_CALLBACK (image_layers_merge_query_response),
|
|
options);
|
|
|
|
g_object_weak_ref (G_OBJECT (options->query_box),
|
|
(GWeakNotify) g_free,
|
|
options);
|
|
|
|
/* The main vbox */
|
|
vbox = gtk_vbox_new (FALSE, 4);
|
|
gtk_container_set_border_width (GTK_CONTAINER (vbox), 6);
|
|
gtk_container_add (GTK_CONTAINER (GTK_DIALOG (options->query_box)->vbox),
|
|
vbox);
|
|
|
|
frame =
|
|
gimp_radio_group_new2 (TRUE,
|
|
merge_visible ?
|
|
_("Final, Merged Layer should be:") :
|
|
_("Final, Anchored Layer should be:"),
|
|
G_CALLBACK (gimp_radio_button_update),
|
|
&options->merge_type,
|
|
GINT_TO_POINTER (options->merge_type),
|
|
|
|
_("Expanded as necessary"),
|
|
GINT_TO_POINTER (GIMP_EXPAND_AS_NECESSARY), NULL,
|
|
|
|
_("Clipped to image"),
|
|
GINT_TO_POINTER (GIMP_CLIP_TO_IMAGE), NULL,
|
|
|
|
_("Clipped to bottom layer"),
|
|
GINT_TO_POINTER (GIMP_CLIP_TO_BOTTOM_LAYER), NULL,
|
|
|
|
NULL);
|
|
|
|
gtk_box_pack_start (GTK_BOX (vbox), frame, FALSE, FALSE, 0);
|
|
gtk_widget_show (frame);
|
|
|
|
gtk_widget_show (vbox);
|
|
gtk_widget_show (options->query_box);
|
|
}
|
|
|
|
|
|
/* private functions */
|
|
|
|
static void
|
|
image_resize_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
ImageResize *image_resize;
|
|
|
|
image_resize = (ImageResize *) data;
|
|
|
|
g_assert (image_resize != NULL);
|
|
g_assert (image_resize->gimage != NULL);
|
|
|
|
gtk_widget_set_sensitive (image_resize->resize->resize_shell, FALSE);
|
|
|
|
if (image_resize->resize->width > 0 &&
|
|
image_resize->resize->height > 0)
|
|
{
|
|
GimpProgress *progress;
|
|
|
|
progress = gimp_progress_start (image_resize->gdisp,
|
|
_("Resizing..."),
|
|
TRUE, NULL, NULL);
|
|
|
|
gimp_image_resize (image_resize->gimage,
|
|
image_resize->resize->width,
|
|
image_resize->resize->height,
|
|
image_resize->resize->offset_x,
|
|
image_resize->resize->offset_y,
|
|
gimp_progress_update_and_flush, progress);
|
|
|
|
gimp_progress_end (progress);
|
|
|
|
gimp_image_flush (image_resize->gimage);
|
|
}
|
|
else
|
|
{
|
|
g_message (_("Resize Error: Both width and height must be "
|
|
"greater than zero."));
|
|
}
|
|
|
|
gtk_widget_destroy (image_resize->resize->resize_shell);
|
|
}
|
|
|
|
static void
|
|
image_scale_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
ImageResize *image_scale = data;
|
|
|
|
g_assert (image_scale != NULL);
|
|
g_assert (image_scale->gimage != NULL);
|
|
|
|
gtk_widget_set_sensitive (image_scale->resize->resize_shell, FALSE);
|
|
|
|
if (gimp_image_check_scaling (image_scale->gimage,
|
|
image_scale->resize->width,
|
|
image_scale->resize->height))
|
|
{
|
|
image_scale_implement (image_scale);
|
|
|
|
gtk_widget_destroy (image_scale->resize->resize_shell);
|
|
}
|
|
else
|
|
{
|
|
GtkWidget *dialog;
|
|
|
|
dialog =
|
|
gimp_query_boolean_box (_("Layer Too Small"),
|
|
image_scale->resize->resize_shell,
|
|
gimp_standard_help_func,
|
|
GIMP_HELP_IMAGE_SCALE_WARNING,
|
|
GTK_STOCK_DIALOG_QUESTION,
|
|
_("The chosen image size will shrink "
|
|
"some layers completely away. "
|
|
"Is this what you want?"),
|
|
GTK_STOCK_OK, GTK_STOCK_CANCEL,
|
|
G_OBJECT (image_scale->resize->resize_shell),
|
|
"destroy",
|
|
image_scale_warn_callback,
|
|
image_scale);
|
|
gtk_widget_show (dialog);
|
|
}
|
|
}
|
|
|
|
static void
|
|
image_scale_warn_callback (GtkWidget *widget,
|
|
gboolean do_scale,
|
|
gpointer data)
|
|
{
|
|
ImageResize *image_scale = data;
|
|
|
|
if (do_scale) /* User doesn't mind losing layers... */
|
|
{
|
|
image_scale_implement (image_scale);
|
|
|
|
gtk_widget_destroy (image_scale->resize->resize_shell);
|
|
}
|
|
else
|
|
{
|
|
gtk_widget_set_sensitive (image_scale->resize->resize_shell, TRUE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
image_scale_implement (ImageResize *image_scale)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
g_assert (image_scale != NULL);
|
|
g_assert (image_scale->gimage != NULL);
|
|
|
|
gimage = image_scale->gimage;
|
|
|
|
if (image_scale->resize->resolution_x == gimage->xresolution &&
|
|
image_scale->resize->resolution_y == gimage->yresolution &&
|
|
image_scale->resize->unit == gimage->unit &&
|
|
image_scale->resize->width == gimage->width &&
|
|
image_scale->resize->height == gimage->height)
|
|
return;
|
|
|
|
gimp_image_undo_group_start (gimage, GIMP_UNDO_GROUP_IMAGE_SCALE,
|
|
_("Scale Image"));
|
|
|
|
gimp_image_set_resolution (gimage,
|
|
image_scale->resize->resolution_x,
|
|
image_scale->resize->resolution_y);
|
|
gimp_image_set_unit (gimage, image_scale->resize->unit);
|
|
|
|
if (image_scale->resize->width != gimage->width ||
|
|
image_scale->resize->height != gimage->height)
|
|
{
|
|
if (image_scale->resize->width > 0 &&
|
|
image_scale->resize->height > 0)
|
|
{
|
|
GimpProgress *progress;
|
|
|
|
progress = gimp_progress_start (image_scale->gdisp,
|
|
_("Scaling..."),
|
|
TRUE, NULL, NULL);
|
|
|
|
gimp_image_scale (gimage,
|
|
image_scale->resize->width,
|
|
image_scale->resize->height,
|
|
image_scale->resize->interpolation,
|
|
gimp_progress_update_and_flush, progress);
|
|
|
|
gimp_progress_end (progress);
|
|
}
|
|
else
|
|
{
|
|
g_message (_("Scale Error: Both width and height must be "
|
|
"greater than zero."));
|
|
return;
|
|
}
|
|
}
|
|
|
|
gimp_image_undo_group_end (gimage);
|
|
|
|
gimp_image_flush (gimage);
|
|
}
|