mirror of https://github.com/GNOME/gimp.git
1314 lines
33 KiB
C
1314 lines
33 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 <string.h>
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "libgimpmath/gimpmath.h"
|
|
#include "libgimpwidgets/gimpwidgets.h"
|
|
#include "libgimp/gimplimits.h"
|
|
|
|
#include "core/core-types.h"
|
|
|
|
#include "paint-funcs/paint-funcs.h"
|
|
|
|
#include "core/gimpimage.h"
|
|
#include "core/gimpimage-mask.h"
|
|
#include "core/gimplayer.h"
|
|
#include "core/gimplayermask.h"
|
|
#include "core/gimplist.h"
|
|
|
|
#include "gdisplay.h"
|
|
#include "gimpui.h"
|
|
#include "layers-commands.h"
|
|
#include "menus.h"
|
|
#include "resize.h"
|
|
|
|
#include "drawable.h"
|
|
#include "floating_sel.h"
|
|
#include "undo.h"
|
|
|
|
#include "libgimp/gimpintl.h"
|
|
|
|
|
|
static void layers_add_mask_query (GimpLayer *layer);
|
|
static void layers_scale_layer_query (GimpImage *gimage,
|
|
GimpLayer *layer);
|
|
static void layers_resize_layer_query (GimpImage *gimage,
|
|
GimpLayer *layer);
|
|
static void layers_menu_set_sensitivity (GimpImage *gimage);
|
|
|
|
|
|
void
|
|
layers_previous_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
GimpLayer *new_layer;
|
|
gint current_layer;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
current_layer =
|
|
gimp_image_get_layer_index (gimage, gimp_image_get_active_layer (gimage));
|
|
|
|
if (current_layer > 0)
|
|
{
|
|
new_layer = (GimpLayer *)
|
|
gimp_container_get_child_by_index (gimage->layers, current_layer - 1);
|
|
|
|
if (new_layer)
|
|
{
|
|
gimp_image_set_active_layer (gimage, new_layer);
|
|
gdisplays_flush ();
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
layers_next_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
GimpLayer *new_layer;
|
|
gint current_layer;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
current_layer =
|
|
gimp_image_get_layer_index (gimage, gimp_image_get_active_layer (gimage));
|
|
|
|
new_layer =
|
|
GIMP_LAYER (gimp_container_get_child_by_index (gimage->layers,
|
|
current_layer + 1));
|
|
|
|
if (new_layer)
|
|
{
|
|
gimp_image_set_active_layer (gimage, new_layer);
|
|
gdisplays_flush ();
|
|
}
|
|
}
|
|
|
|
void
|
|
layers_raise_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
gimp_image_raise_layer (gimage, gimp_image_get_active_layer (gimage));
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_lower_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
gimp_image_lower_layer (gimage, gimp_image_get_active_layer (gimage));
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_raise_to_top_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
gimp_image_raise_layer_to_top (gimage, gimp_image_get_active_layer (gimage));
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_lower_to_bottom_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
gimp_image_lower_layer_to_bottom (gimage,
|
|
gimp_image_get_active_layer (gimage));
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_new_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
GimpLayer *layer;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
/* If there is a floating selection, the new command transforms
|
|
* the current fs into a new layer
|
|
*/
|
|
if ((layer = gimp_image_floating_sel (gimage)))
|
|
{
|
|
floating_sel_to_layer (layer);
|
|
|
|
gdisplays_flush ();
|
|
}
|
|
else
|
|
{
|
|
layers_new_layer_query (gimage);
|
|
}
|
|
}
|
|
|
|
void
|
|
layers_duplicate_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
GimpLayer *active_layer;
|
|
GimpLayer *new_layer;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
active_layer = gimp_image_get_active_layer (gimage);
|
|
new_layer = gimp_layer_copy (active_layer, TRUE);
|
|
gimp_image_add_layer (gimage, new_layer, -1);
|
|
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_delete_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
GimpLayer *layer;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
layer = gimp_image_get_active_layer (gimage);
|
|
|
|
if (! layer)
|
|
return;
|
|
|
|
if (gimp_layer_is_floating_sel (layer))
|
|
floating_sel_remove (layer);
|
|
else
|
|
gimp_image_remove_layer (gimage, layer);
|
|
|
|
gdisplays_flush_now ();
|
|
}
|
|
|
|
void
|
|
layers_scale_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
layers_scale_layer_query (gimage, gimp_image_get_active_layer (gimage));
|
|
}
|
|
|
|
void
|
|
layers_resize_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
layers_resize_layer_query (gimage, gimp_image_get_active_layer (gimage));
|
|
}
|
|
|
|
void
|
|
layers_resize_to_image_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
gimp_layer_resize_to_image (gimp_image_get_active_layer (gimage));
|
|
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_add_layer_mask_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
layers_add_mask_query (gimp_image_get_active_layer (gimage));
|
|
}
|
|
|
|
void
|
|
layers_apply_layer_mask_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
GimpLayer *layer;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
layer = gimp_image_get_active_layer (gimage);
|
|
|
|
/* Make sure there is a layer mask to apply */
|
|
if (layer && gimp_layer_get_mask (layer))
|
|
{
|
|
gboolean flush = ! layer->mask->apply_mask || layer->mask->show_mask;
|
|
|
|
gimp_layer_apply_mask (layer, APPLY, TRUE);
|
|
|
|
if (flush)
|
|
gdisplays_flush ();
|
|
}
|
|
}
|
|
|
|
void
|
|
layers_delete_layer_mask_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
GimpLayer *layer;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
layer = gimp_image_get_active_layer (gimage);
|
|
|
|
/* Make sure there is a layer mask to apply */
|
|
if (layer && gimp_layer_get_mask (layer))
|
|
{
|
|
gboolean flush = layer->mask->apply_mask || layer->mask->show_mask;
|
|
|
|
gimp_layer_apply_mask (layer, DISCARD, TRUE);
|
|
|
|
if (flush)
|
|
gdisplays_flush ();
|
|
}
|
|
}
|
|
|
|
void
|
|
layers_anchor_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
floating_sel_anchor (gimp_image_get_active_layer (gimage));
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_merge_layers_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
layers_layer_merge_query (gimage, TRUE);
|
|
}
|
|
|
|
void
|
|
layers_merge_down_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
gimp_image_merge_down (gimage, gimp_image_get_active_layer (gimage),
|
|
EXPAND_AS_NECESSARY);
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_flatten_image_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
gimp_image_flatten (gimage);
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_alpha_select_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
gimage_mask_layer_alpha (gimage, gimp_image_get_active_layer (gimage));
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_mask_select_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
gimage_mask_layer_mask (gimage, gimp_image_get_active_layer (gimage));
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_add_alpha_channel_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
GimpLayer *layer;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
layer = gimp_image_get_active_layer (gimage);
|
|
|
|
if (! layer)
|
|
return;
|
|
|
|
gimp_layer_add_alpha (layer);
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
void
|
|
layers_edit_attributes_cmd_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
GimpImage *gimage;
|
|
GimpLayer *layer;
|
|
|
|
gimage = (GimpImage *) gimp_widget_get_callback_context (widget);
|
|
|
|
if (! gimage)
|
|
return;
|
|
|
|
layer = gimp_image_get_active_layer (gimage);
|
|
|
|
if (! layer)
|
|
return;
|
|
|
|
layers_edit_layer_query (layer);
|
|
}
|
|
|
|
|
|
/********************************/
|
|
/* The new layer query dialog */
|
|
/********************************/
|
|
|
|
typedef struct _NewLayerOptions NewLayerOptions;
|
|
|
|
struct _NewLayerOptions
|
|
{
|
|
GtkWidget *query_box;
|
|
GtkWidget *name_entry;
|
|
GtkWidget *size_se;
|
|
|
|
GimpFillType fill_type;
|
|
gint xsize;
|
|
gint ysize;
|
|
|
|
GimpImage *gimage;
|
|
};
|
|
|
|
static GimpFillType fill_type = TRANSPARENT_FILL;
|
|
static gchar *layer_name = NULL;
|
|
|
|
static void
|
|
new_layer_query_ok_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
NewLayerOptions *options;
|
|
GimpLayer *layer;
|
|
GimpImage *gimage;
|
|
|
|
options = (NewLayerOptions *) data;
|
|
|
|
if (layer_name)
|
|
g_free (layer_name);
|
|
layer_name = g_strdup (gtk_entry_get_text (GTK_ENTRY (options->name_entry)));
|
|
|
|
options->xsize =
|
|
RINT (gimp_size_entry_get_refval (GIMP_SIZE_ENTRY (options->size_se), 0));
|
|
options->ysize =
|
|
RINT (gimp_size_entry_get_refval (GIMP_SIZE_ENTRY (options->size_se), 1));
|
|
|
|
fill_type = options->fill_type;
|
|
|
|
if ((gimage = options->gimage))
|
|
{
|
|
/* Start a group undo */
|
|
undo_push_group_start (gimage, LAYER_ADD_UNDO);
|
|
|
|
layer = gimp_layer_new (gimage, options->xsize, options->ysize,
|
|
gimp_image_base_type_with_alpha (gimage),
|
|
layer_name, OPAQUE_OPACITY, NORMAL_MODE);
|
|
if (layer)
|
|
{
|
|
drawable_fill (GIMP_DRAWABLE (layer), fill_type);
|
|
gimp_image_add_layer (gimage, layer, -1);
|
|
|
|
/* End the group undo */
|
|
undo_push_group_end (gimage);
|
|
|
|
gdisplays_flush ();
|
|
}
|
|
else
|
|
{
|
|
g_message ("new_layer_query_ok_callback():\n"
|
|
"could not allocate new layer");
|
|
}
|
|
}
|
|
|
|
gtk_widget_destroy (options->query_box);
|
|
}
|
|
|
|
void
|
|
layers_new_layer_query (GimpImage *gimage)
|
|
{
|
|
NewLayerOptions *options;
|
|
GtkWidget *vbox;
|
|
GtkWidget *table;
|
|
GtkWidget *label;
|
|
GtkObject *adjustment;
|
|
GtkWidget *spinbutton;
|
|
GtkWidget *frame;
|
|
|
|
options = g_new0 (NewLayerOptions, 1);
|
|
|
|
options->fill_type = fill_type;
|
|
options->gimage = gimage;
|
|
|
|
/* The dialog */
|
|
options->query_box =
|
|
gimp_dialog_new (_("New Layer Options"), "new_layer_options",
|
|
gimp_standard_help_func,
|
|
"dialogs/layers/new_layer.html",
|
|
GTK_WIN_POS_MOUSE,
|
|
FALSE, TRUE, FALSE,
|
|
|
|
_("OK"), new_layer_query_ok_callback,
|
|
options, NULL, NULL, TRUE, FALSE,
|
|
_("Cancel"), gtk_widget_destroy,
|
|
NULL, 1, NULL, FALSE, TRUE,
|
|
|
|
NULL);
|
|
|
|
gtk_signal_connect_object (GTK_OBJECT (options->query_box), "destroy",
|
|
GTK_SIGNAL_FUNC (g_free),
|
|
(GtkObject *) options);
|
|
|
|
/* The main vbox */
|
|
vbox = gtk_vbox_new (FALSE, 2);
|
|
gtk_container_set_border_width (GTK_CONTAINER (vbox), 4);
|
|
gtk_container_add (GTK_CONTAINER (GTK_DIALOG (options->query_box)->vbox),
|
|
vbox);
|
|
|
|
table = gtk_table_new (3, 2, FALSE);
|
|
gtk_table_set_col_spacing (GTK_TABLE (table), 0, 4);
|
|
gtk_table_set_row_spacing (GTK_TABLE (table), 0, 4);
|
|
gtk_box_pack_start (GTK_BOX (vbox), table, FALSE, FALSE, 0);
|
|
|
|
/* The name label and entry */
|
|
label = gtk_label_new (_("Layer Name:"));
|
|
gtk_misc_set_alignment (GTK_MISC (label), 1.0, 0.5);
|
|
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 0, 1,
|
|
GTK_SHRINK | GTK_FILL, GTK_SHRINK, 0, 1);
|
|
gtk_widget_show (label);
|
|
|
|
options->name_entry = gtk_entry_new ();
|
|
gtk_widget_set_usize (options->name_entry, 75, 0);
|
|
gtk_table_attach_defaults (GTK_TABLE (table),
|
|
options->name_entry, 1, 2, 0, 1);
|
|
gtk_entry_set_text (GTK_ENTRY (options->name_entry),
|
|
(layer_name ? layer_name : _("New Layer")));
|
|
gtk_widget_show (options->name_entry);
|
|
|
|
/* The size labels */
|
|
label = gtk_label_new (_("Layer Width:"));
|
|
gtk_misc_set_alignment (GTK_MISC (label), 1.0, 0.5);
|
|
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 1, 2,
|
|
GTK_SHRINK | GTK_FILL, GTK_SHRINK, 0, 0);
|
|
gtk_widget_show (label);
|
|
|
|
label = gtk_label_new (_("Height:"));
|
|
gtk_misc_set_alignment (GTK_MISC (label), 1.0, 0.5);
|
|
gtk_table_attach (GTK_TABLE (table), label, 0, 1, 2, 3,
|
|
GTK_SHRINK | GTK_FILL, GTK_SHRINK, 0, 0);
|
|
gtk_widget_show (label);
|
|
|
|
/* The size sizeentry */
|
|
adjustment = gtk_adjustment_new (1, 1, 1, 1, 10, 1);
|
|
spinbutton = gtk_spin_button_new (GTK_ADJUSTMENT (adjustment), 1, 2);
|
|
gtk_spin_button_set_shadow_type (GTK_SPIN_BUTTON (spinbutton),
|
|
GTK_SHADOW_NONE);
|
|
gtk_spin_button_set_numeric (GTK_SPIN_BUTTON (spinbutton), TRUE);
|
|
gtk_widget_set_usize (spinbutton, 75, 0);
|
|
|
|
options->size_se = gimp_size_entry_new (1, gimage->unit, "%a",
|
|
TRUE, TRUE, FALSE, 75,
|
|
GIMP_SIZE_ENTRY_UPDATE_SIZE);
|
|
gimp_size_entry_add_field (GIMP_SIZE_ENTRY (options->size_se),
|
|
GTK_SPIN_BUTTON (spinbutton), NULL);
|
|
gtk_table_attach_defaults (GTK_TABLE (options->size_se), spinbutton,
|
|
1, 2, 0, 1);
|
|
gtk_widget_show (spinbutton);
|
|
gtk_table_attach (GTK_TABLE (table), options->size_se, 1, 2, 1, 3,
|
|
GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0);
|
|
gtk_widget_show (options->size_se);
|
|
|
|
gimp_size_entry_set_unit (GIMP_SIZE_ENTRY (options->size_se),
|
|
GIMP_UNIT_PIXEL);
|
|
|
|
gimp_size_entry_set_resolution (GIMP_SIZE_ENTRY (options->size_se), 0,
|
|
gimage->xresolution, FALSE);
|
|
gimp_size_entry_set_resolution (GIMP_SIZE_ENTRY (options->size_se), 1,
|
|
gimage->yresolution, FALSE);
|
|
|
|
gimp_size_entry_set_refval_boundaries (GIMP_SIZE_ENTRY (options->size_se), 0,
|
|
GIMP_MIN_IMAGE_SIZE,
|
|
GIMP_MAX_IMAGE_SIZE);
|
|
gimp_size_entry_set_refval_boundaries (GIMP_SIZE_ENTRY (options->size_se), 1,
|
|
GIMP_MIN_IMAGE_SIZE,
|
|
GIMP_MAX_IMAGE_SIZE);
|
|
|
|
gimp_size_entry_set_size (GIMP_SIZE_ENTRY (options->size_se), 0,
|
|
0, gimage->width);
|
|
gimp_size_entry_set_size (GIMP_SIZE_ENTRY (options->size_se), 1,
|
|
0, gimage->height);
|
|
|
|
gimp_size_entry_set_refval (GIMP_SIZE_ENTRY (options->size_se), 0,
|
|
gimage->width);
|
|
gimp_size_entry_set_refval (GIMP_SIZE_ENTRY (options->size_se), 1,
|
|
gimage->height);
|
|
|
|
gtk_widget_show (table);
|
|
|
|
/* The radio frame */
|
|
frame =
|
|
gimp_radio_group_new2 (TRUE, _("Layer Fill Type"),
|
|
gimp_radio_button_update,
|
|
&options->fill_type, (gpointer) options->fill_type,
|
|
|
|
_("Foreground"), (gpointer) FOREGROUND_FILL, NULL,
|
|
_("Background"), (gpointer) BACKGROUND_FILL, NULL,
|
|
_("White"), (gpointer) WHITE_FILL, NULL,
|
|
_("Transparent"), (gpointer) TRANSPARENT_FILL, 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);
|
|
}
|
|
|
|
/**************************************/
|
|
/* The edit layer attributes dialog */
|
|
/**************************************/
|
|
|
|
typedef struct _EditLayerOptions EditLayerOptions;
|
|
|
|
struct _EditLayerOptions
|
|
{
|
|
GtkWidget *query_box;
|
|
GtkWidget *name_entry;
|
|
GimpLayer *layer;
|
|
GimpImage *gimage;
|
|
};
|
|
|
|
static void
|
|
edit_layer_query_ok_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
EditLayerOptions *options;
|
|
GimpLayer *layer;
|
|
|
|
options = (EditLayerOptions *) data;
|
|
|
|
if ((layer = options->layer))
|
|
{
|
|
/* Set the new layer name */
|
|
if (GIMP_OBJECT (layer)->name && gimp_layer_is_floating_sel (layer))
|
|
{
|
|
/* If the layer is a floating selection, make it a layer */
|
|
floating_sel_to_layer (layer);
|
|
}
|
|
else
|
|
{
|
|
/* We're doing a plain rename */
|
|
undo_push_layer_rename (options->gimage, layer);
|
|
}
|
|
|
|
gimp_object_set_name (GIMP_OBJECT (layer),
|
|
gtk_entry_get_text (GTK_ENTRY (options->name_entry)));
|
|
}
|
|
|
|
gdisplays_flush ();
|
|
|
|
gtk_widget_destroy (options->query_box);
|
|
}
|
|
|
|
void
|
|
layers_edit_layer_query (GimpLayer *layer)
|
|
{
|
|
EditLayerOptions *options;
|
|
GtkWidget *vbox;
|
|
GtkWidget *hbox;
|
|
GtkWidget *label;
|
|
|
|
/* The new options structure */
|
|
options = g_new (EditLayerOptions, 1);
|
|
options->layer = layer;
|
|
options->gimage = gimp_drawable_gimage (GIMP_DRAWABLE (layer));
|
|
|
|
/* The dialog */
|
|
options->query_box =
|
|
gimp_dialog_new (_("Edit Layer Attributes"), "edit_layer_attributes",
|
|
gimp_standard_help_func,
|
|
"dialogs/layers/edit_layer_attributes.html",
|
|
GTK_WIN_POS_MOUSE,
|
|
FALSE, TRUE, FALSE,
|
|
|
|
_("OK"), edit_layer_query_ok_callback,
|
|
options, NULL, NULL, TRUE, FALSE,
|
|
_("Cancel"), gtk_widget_destroy,
|
|
NULL, 1, NULL, FALSE, TRUE,
|
|
|
|
NULL);
|
|
|
|
gtk_signal_connect_object (GTK_OBJECT (options->query_box), "destroy",
|
|
GTK_SIGNAL_FUNC (g_free),
|
|
(GtkObject *) options);
|
|
|
|
gtk_signal_connect_object_while_alive (GTK_OBJECT (layer), "removed",
|
|
GTK_SIGNAL_FUNC (gtk_widget_destroy),
|
|
GTK_OBJECT (options->query_box));
|
|
|
|
/* The main vbox */
|
|
vbox = gtk_vbox_new (FALSE, 2);
|
|
gtk_container_set_border_width (GTK_CONTAINER (vbox), 4);
|
|
gtk_container_add (GTK_CONTAINER (GTK_DIALOG (options->query_box)->vbox),
|
|
vbox);
|
|
|
|
/* The name hbox, label and entry */
|
|
hbox = gtk_hbox_new (FALSE, 4);
|
|
gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, 0);
|
|
|
|
label = gtk_label_new (_("Layer name:"));
|
|
gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
|
|
gtk_widget_show (label);
|
|
|
|
options->name_entry = gtk_entry_new ();
|
|
gtk_box_pack_start (GTK_BOX (hbox), options->name_entry, TRUE, TRUE, 0);
|
|
gtk_entry_set_text (GTK_ENTRY (options->name_entry),
|
|
((gimp_layer_is_floating_sel (layer) ?
|
|
_("Floating Selection") :
|
|
gimp_object_get_name (GIMP_OBJECT (layer)))));
|
|
gtk_signal_connect (GTK_OBJECT (options->name_entry), "activate",
|
|
edit_layer_query_ok_callback,
|
|
options);
|
|
gtk_widget_show (options->name_entry);
|
|
|
|
gtk_widget_show (hbox);
|
|
|
|
gtk_widget_show (vbox);
|
|
gtk_widget_show (options->query_box);
|
|
}
|
|
|
|
/*******************************/
|
|
/* The add mask query dialog */
|
|
/*******************************/
|
|
|
|
typedef struct _AddMaskOptions AddMaskOptions;
|
|
|
|
struct _AddMaskOptions
|
|
{
|
|
GtkWidget *query_box;
|
|
GimpLayer *layer;
|
|
AddMaskType add_mask_type;
|
|
};
|
|
|
|
static void
|
|
add_mask_query_ok_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
AddMaskOptions *options;
|
|
GimpImage *gimage;
|
|
GimpLayerMask *mask;
|
|
GimpLayer *layer;
|
|
|
|
options = (AddMaskOptions *) data;
|
|
|
|
if ((layer = (options->layer)) &&
|
|
(gimage = GIMP_DRAWABLE (layer)->gimage))
|
|
{
|
|
mask = gimp_layer_create_mask (layer, options->add_mask_type);
|
|
gimp_layer_add_mask (layer, mask, TRUE);
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
gtk_widget_destroy (options->query_box);
|
|
}
|
|
|
|
static void
|
|
layers_add_mask_query (GimpLayer *layer)
|
|
{
|
|
AddMaskOptions *options;
|
|
GtkWidget *frame;
|
|
GimpImage *gimage;
|
|
|
|
/* The new options structure */
|
|
options = g_new (AddMaskOptions, 1);
|
|
options->layer = layer;
|
|
options->add_mask_type = ADD_WHITE_MASK;
|
|
|
|
gimage = GIMP_DRAWABLE (layer)->gimage;
|
|
|
|
/* The dialog */
|
|
options->query_box =
|
|
gimp_dialog_new (_("Add Mask Options"), "add_mask_options",
|
|
gimp_standard_help_func,
|
|
"dialogs/layers/add_layer_mask.html",
|
|
GTK_WIN_POS_MOUSE,
|
|
FALSE, TRUE, FALSE,
|
|
|
|
_("OK"), add_mask_query_ok_callback,
|
|
options, NULL, NULL, TRUE, FALSE,
|
|
_("Cancel"), gtk_widget_destroy,
|
|
NULL, 1, NULL, FALSE, TRUE,
|
|
|
|
NULL);
|
|
|
|
gtk_signal_connect_object (GTK_OBJECT (options->query_box), "destroy",
|
|
GTK_SIGNAL_FUNC (g_free),
|
|
(GtkObject *) options);
|
|
|
|
gtk_signal_connect_object_while_alive (GTK_OBJECT (layer), "removed",
|
|
GTK_SIGNAL_FUNC (gtk_widget_destroy),
|
|
GTK_OBJECT (options->query_box));
|
|
|
|
/* The radio frame and box */
|
|
if (gimage->selection_mask)
|
|
{
|
|
options->add_mask_type = ADD_SELECTION_MASK;
|
|
|
|
frame = gimp_radio_group_new2 (TRUE, _("Initialize Layer Mask to:"),
|
|
gimp_radio_button_update,
|
|
&options->add_mask_type,
|
|
(gpointer) options->add_mask_type,
|
|
|
|
_("Selection"),
|
|
(gpointer) ADD_SELECTION_MASK, NULL,
|
|
_("Inverse Selection"),
|
|
(gpointer) ADD_INV_SELECTION_MASK, NULL,
|
|
_("White (Full Opacity)"),
|
|
(gpointer) ADD_WHITE_MASK, NULL,
|
|
_("Black (Full Transparency)"),
|
|
(gpointer) ADD_BLACK_MASK, NULL,
|
|
_("Layer's Alpha Channel"),
|
|
(gpointer) ADD_ALPHA_MASK, NULL,
|
|
NULL);
|
|
}
|
|
else
|
|
{
|
|
frame = gimp_radio_group_new2 (TRUE, _("Initialize Layer Mask to:"),
|
|
gimp_radio_button_update,
|
|
&options->add_mask_type,
|
|
(gpointer) options->add_mask_type,
|
|
|
|
_("White (Full Opacity)"),
|
|
(gpointer) ADD_WHITE_MASK, NULL,
|
|
_("Black (Full Transparency)"),
|
|
(gpointer) ADD_BLACK_MASK, NULL,
|
|
_("Layer's Alpha Channel"),
|
|
(gpointer) ADD_ALPHA_MASK, NULL,
|
|
NULL);
|
|
}
|
|
gtk_container_set_border_width (GTK_CONTAINER (frame), 6);
|
|
gtk_container_add (GTK_CONTAINER (GTK_DIALOG (options->query_box)->vbox),
|
|
frame);
|
|
gtk_widget_show (frame);
|
|
|
|
gtk_widget_show (options->query_box);
|
|
}
|
|
|
|
|
|
/****************************/
|
|
/* The scale layer dialog */
|
|
/****************************/
|
|
|
|
typedef struct _ScaleLayerOptions ScaleLayerOptions;
|
|
|
|
struct _ScaleLayerOptions
|
|
{
|
|
GimpLayer *layer;
|
|
Resize *resize;
|
|
};
|
|
|
|
static void
|
|
scale_layer_query_ok_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
ScaleLayerOptions *options;
|
|
GimpImage *gimage;
|
|
GimpLayer *layer;
|
|
|
|
options = (ScaleLayerOptions *) data;
|
|
|
|
if (options->resize->width > 0 && options->resize->height > 0 &&
|
|
(layer = (options->layer)))
|
|
{
|
|
gtk_widget_set_sensitive (options->resize->resize_shell, FALSE);
|
|
|
|
if ((gimage = GIMP_DRAWABLE (layer)->gimage) != NULL)
|
|
{
|
|
undo_push_group_start (gimage, LAYER_SCALE_UNDO);
|
|
|
|
if (gimp_layer_is_floating_sel (layer))
|
|
floating_sel_relax (layer, TRUE);
|
|
|
|
gimp_layer_scale (layer,
|
|
options->resize->width, options->resize->height,
|
|
TRUE);
|
|
|
|
if (gimp_layer_is_floating_sel (layer))
|
|
floating_sel_rigor (layer, TRUE);
|
|
|
|
undo_push_group_end (gimage);
|
|
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
gtk_widget_destroy (options->resize->resize_shell);
|
|
}
|
|
else
|
|
{
|
|
g_message (_("Invalid width or height.\n"
|
|
"Both must be positive."));
|
|
}
|
|
}
|
|
|
|
static void
|
|
layers_scale_layer_query (GimpImage *gimage,
|
|
GimpLayer *layer)
|
|
{
|
|
ScaleLayerOptions *options;
|
|
|
|
/* the new options structure */
|
|
options = g_new (ScaleLayerOptions, 1);
|
|
options->layer = layer;
|
|
options->resize =
|
|
resize_widget_new (ScaleWidget,
|
|
ResizeLayer,
|
|
GTK_OBJECT (layer),
|
|
"removed",
|
|
gimp_drawable_width (GIMP_DRAWABLE (layer)),
|
|
gimp_drawable_height (GIMP_DRAWABLE (layer)),
|
|
gimage->xresolution,
|
|
gimage->yresolution,
|
|
gimage->unit,
|
|
TRUE,
|
|
scale_layer_query_ok_callback,
|
|
NULL,
|
|
options);
|
|
|
|
gtk_signal_connect_object (GTK_OBJECT (options->resize->resize_shell),
|
|
"destroy",
|
|
GTK_SIGNAL_FUNC (g_free),
|
|
(GtkObject *) options);
|
|
|
|
gtk_widget_show (options->resize->resize_shell);
|
|
}
|
|
|
|
/*****************************/
|
|
/* The resize layer dialog */
|
|
/*****************************/
|
|
|
|
typedef struct _ResizeLayerOptions ResizeLayerOptions;
|
|
|
|
struct _ResizeLayerOptions
|
|
{
|
|
GimpLayer *layer;
|
|
Resize *resize;
|
|
};
|
|
|
|
static void
|
|
resize_layer_query_ok_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
ResizeLayerOptions *options;
|
|
GimpImage *gimage;
|
|
GimpLayer *layer;
|
|
|
|
options = (ResizeLayerOptions *) data;
|
|
|
|
if (options->resize->width > 0 && options->resize->height > 0 &&
|
|
(layer = (options->layer)))
|
|
{
|
|
gtk_widget_set_sensitive (options->resize->resize_shell, FALSE);
|
|
|
|
if ((gimage = GIMP_DRAWABLE (layer)->gimage) != NULL)
|
|
{
|
|
undo_push_group_start (gimage, LAYER_RESIZE_UNDO);
|
|
|
|
if (gimp_layer_is_floating_sel (layer))
|
|
floating_sel_relax (layer, TRUE);
|
|
|
|
gimp_layer_resize (layer,
|
|
options->resize->width,
|
|
options->resize->height,
|
|
options->resize->offset_x,
|
|
options->resize->offset_y);
|
|
|
|
if (gimp_layer_is_floating_sel (layer))
|
|
floating_sel_rigor (layer, TRUE);
|
|
|
|
undo_push_group_end (gimage);
|
|
|
|
gdisplays_flush ();
|
|
}
|
|
|
|
gtk_widget_destroy (options->resize->resize_shell);
|
|
}
|
|
else
|
|
{
|
|
g_message (_("Invalid width or height.\n"
|
|
"Both must be positive."));
|
|
}
|
|
}
|
|
|
|
static void
|
|
layers_resize_layer_query (GimpImage *gimage,
|
|
GimpLayer *layer)
|
|
{
|
|
ResizeLayerOptions *options;
|
|
|
|
/* the new options structure */
|
|
options = g_new (ResizeLayerOptions, 1);
|
|
options->layer = layer;
|
|
options->resize =
|
|
resize_widget_new (ResizeWidget,
|
|
ResizeLayer,
|
|
GTK_OBJECT (layer),
|
|
"removed",
|
|
gimp_drawable_width (GIMP_DRAWABLE (layer)),
|
|
gimp_drawable_height (GIMP_DRAWABLE (layer)),
|
|
gimage->xresolution,
|
|
gimage->yresolution,
|
|
gimage->unit,
|
|
TRUE,
|
|
resize_layer_query_ok_callback,
|
|
NULL,
|
|
options);
|
|
|
|
gtk_signal_connect_object (GTK_OBJECT (options->resize->resize_shell),
|
|
"destroy",
|
|
GTK_SIGNAL_FUNC (g_free),
|
|
(GtkObject *) options);
|
|
|
|
gtk_widget_show (options->resize->resize_shell);
|
|
}
|
|
|
|
/****************************/
|
|
/* The layer merge dialog */
|
|
/****************************/
|
|
|
|
typedef struct _LayerMergeOptions LayerMergeOptions;
|
|
|
|
struct _LayerMergeOptions
|
|
{
|
|
GtkWidget *query_box;
|
|
GimpImage *gimage;
|
|
gboolean merge_visible;
|
|
MergeType merge_type;
|
|
};
|
|
|
|
static void
|
|
layer_merge_query_ok_callback (GtkWidget *widget,
|
|
gpointer data)
|
|
{
|
|
LayerMergeOptions *options;
|
|
GimpImage *gimage;
|
|
|
|
options = (LayerMergeOptions *) data;
|
|
if (! (gimage = options->gimage))
|
|
return;
|
|
|
|
if (options->merge_visible)
|
|
gimp_image_merge_visible_layers (gimage, options->merge_type);
|
|
|
|
gdisplays_flush ();
|
|
|
|
gtk_widget_destroy (options->query_box);
|
|
}
|
|
|
|
void
|
|
layers_layer_merge_query (GimpImage *gimage,
|
|
/* if FALSE, anchor active layer */
|
|
gboolean merge_visible)
|
|
{
|
|
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 = EXPAND_AS_NECESSARY;
|
|
|
|
/* The dialog */
|
|
options->query_box =
|
|
gimp_dialog_new (_("Layer Merge Options"), "layer_merge_options",
|
|
gimp_standard_help_func,
|
|
"dialogs/layers/merge_visible_layers.html",
|
|
GTK_WIN_POS_MOUSE,
|
|
FALSE, TRUE, FALSE,
|
|
|
|
_("OK"), layer_merge_query_ok_callback,
|
|
options, NULL, NULL, TRUE, FALSE,
|
|
_("Cancel"), gtk_widget_destroy,
|
|
NULL, 1, NULL, FALSE, TRUE,
|
|
|
|
NULL);
|
|
|
|
gtk_signal_connect_object (GTK_OBJECT (options->query_box), "destroy",
|
|
GTK_SIGNAL_FUNC (g_free),
|
|
(GtkObject *) options);
|
|
|
|
/* The main vbox */
|
|
vbox = gtk_vbox_new (FALSE, 2);
|
|
gtk_container_set_border_width (GTK_CONTAINER (vbox), 4);
|
|
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:"),
|
|
gimp_radio_button_update,
|
|
&options->merge_type,
|
|
(gpointer) options->merge_type,
|
|
|
|
_("Expanded as necessary"),
|
|
(gpointer) EXPAND_AS_NECESSARY, NULL,
|
|
_("Clipped to image"),
|
|
(gpointer) CLIP_TO_IMAGE, NULL,
|
|
_("Clipped to bottom layer"),
|
|
(gpointer) 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);
|
|
}
|
|
|
|
void
|
|
layers_show_context_menu (GimpImage *gimage)
|
|
{
|
|
GtkItemFactory *item_factory;
|
|
gint x, y;
|
|
|
|
layers_menu_set_sensitivity (gimage);
|
|
|
|
item_factory = menus_get_layers_factory ();
|
|
|
|
gimp_menu_position (GTK_MENU (item_factory->widget), &x, &y);
|
|
|
|
gtk_item_factory_popup_with_data (item_factory,
|
|
gimage, NULL,
|
|
x, y,
|
|
3, 0);
|
|
}
|
|
|
|
static void
|
|
layers_menu_set_sensitivity (GimpImage *gimage)
|
|
{
|
|
GimpLayer *layer;
|
|
gboolean fs = FALSE; /* floating sel */
|
|
gboolean ac = FALSE; /* active channel */
|
|
gboolean lm = FALSE; /* layer mask */
|
|
gboolean lp = FALSE; /* layers present */
|
|
gboolean alpha = FALSE; /* alpha channel present */
|
|
gboolean indexed = FALSE; /* is indexed */
|
|
gboolean next_alpha = FALSE;
|
|
GList *list;
|
|
GList *next = NULL;
|
|
GList *prev = NULL;
|
|
|
|
g_return_if_fail (gimage != NULL);
|
|
g_return_if_fail (GIMP_IS_IMAGE (gimage));
|
|
|
|
layer = gimp_image_get_active_layer (gimage);
|
|
|
|
if (layer)
|
|
lm = (gimp_layer_get_mask (layer)) ? TRUE : FALSE;
|
|
|
|
fs = (gimp_image_floating_sel (gimage) != NULL);
|
|
ac = (gimp_image_get_active_channel (gimage) != NULL);
|
|
|
|
alpha = layer && gimp_layer_has_alpha (layer);
|
|
|
|
lp = ! gimp_image_is_empty (gimage);
|
|
indexed = (gimp_image_base_type (gimage) == INDEXED);
|
|
|
|
for (list = GIMP_LIST (gimage->layers)->list;
|
|
list;
|
|
list = g_list_next (list))
|
|
{
|
|
if (layer == (GimpLayer *) list->data)
|
|
{
|
|
prev = g_list_previous (list);
|
|
next = g_list_next (list);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (next)
|
|
next_alpha = gimp_layer_has_alpha (GIMP_LAYER (next->data));
|
|
else
|
|
next_alpha = FALSE;
|
|
|
|
#define SET_SENSITIVE(menu,condition) \
|
|
menus_set_sensitive ("<Layers>/" menu, (condition) != 0)
|
|
|
|
SET_SENSITIVE ("New Layer...", gimage);
|
|
|
|
SET_SENSITIVE ("Stack/Raise Layer",
|
|
!fs && !ac && gimage && lp && alpha && prev);
|
|
|
|
SET_SENSITIVE ("Stack/Lower Layer",
|
|
!fs && !ac && gimage && lp && next && next_alpha);
|
|
|
|
SET_SENSITIVE ("Stack/Layer to Top",
|
|
!fs && !ac && gimage && lp && alpha && prev);
|
|
SET_SENSITIVE ("Stack/Layer to Bottom",
|
|
!fs && !ac && gimage && lp && next && next_alpha);
|
|
|
|
SET_SENSITIVE ("Duplicate Layer", !fs && !ac && gimage && lp);
|
|
SET_SENSITIVE ("Anchor Layer", !fs && !ac && gimage && lp);
|
|
SET_SENSITIVE ("Delete Layer", !ac && gimage && lp);
|
|
|
|
SET_SENSITIVE ("Layer Boundary Size...", !ac && gimage && lp);
|
|
SET_SENSITIVE ("Layer to Imagesize", !ac && gimage && lp);
|
|
SET_SENSITIVE ("Scale Layer...", !ac && gimage && lp);
|
|
|
|
SET_SENSITIVE ("Merge Visible Layers...", !fs && !ac && gimage && lp);
|
|
SET_SENSITIVE ("Merge Down", !fs && !ac && gimage && lp && next);
|
|
SET_SENSITIVE ("Flatten Image", !fs && !ac && gimage && lp);
|
|
|
|
SET_SENSITIVE ("Add Layer Mask...",
|
|
!fs && !ac && gimage && !lm && lp && alpha && !indexed);
|
|
SET_SENSITIVE ("Apply Layer Mask", !fs && !ac && gimage && lm && lp);
|
|
SET_SENSITIVE ("Delete Layer Mask", !fs && !ac && gimage && lm && lp);
|
|
SET_SENSITIVE ("Mask to Selection", !fs && !ac && gimage && lm && lp);
|
|
|
|
SET_SENSITIVE ("Add Alpha Channel", !fs && !alpha);
|
|
SET_SENSITIVE ("Alpha to Selection", !fs && !ac && gimage && lp && alpha);
|
|
|
|
SET_SENSITIVE ("Edit Layer Attributes...", !fs && !ac && gimage && lp);
|
|
|
|
#undef SET_SENSITIVE
|
|
}
|