/* 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 #include #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-mask.h" #include "core/gimpimage-mask-select.h" #include "core/gimptoolinfo.h" #include "pdb/procedural_db.h" #include "plug-in/plug-in-run.h" #include "vectors/gimpvectors.h" #include "vectors/gimpvectors-import.h" #include "widgets/gimphelp-ids.h" #include "widgets/gimpitemtreeview.h" #include "widgets/gimpviewabledialog.h" #include "display/gimpdisplay.h" #include "tools/gimppainttool.h" #include "tools/gimpvectortool.h" #include "tools/tool_manager.h" #include "vectors-commands.h" #include "gimp-intl.h" #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 if (GIMP_IS_ITEM_TREE_VIEW (data)) \ gimage = ((GimpItemTreeView *) data)->gimage; \ else \ gimage = NULL; \ \ if (! gimage) \ return #define return_if_no_vectors(gimage,vectors,data) \ return_if_no_image (gimage,data); \ vectors = gimp_image_get_active_vectors (gimage); \ if (! vectors) \ return /* public functions */ void vectors_new_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; return_if_no_image (gimage, data); vectors_new_vectors_query (gimage, NULL, TRUE); } void vectors_raise_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; GimpVectors *active_vectors; return_if_no_vectors (gimage, active_vectors, data); gimp_image_raise_vectors (gimage, active_vectors); gimp_image_flush (gimage); } void vectors_lower_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; GimpVectors *active_vectors; return_if_no_vectors (gimage, active_vectors, data); gimp_image_lower_vectors (gimage, active_vectors); gimp_image_flush (gimage); } void vectors_duplicate_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; GimpVectors *active_vectors; GimpVectors *new_vectors; return_if_no_vectors (gimage, active_vectors, data); new_vectors = GIMP_VECTORS (gimp_item_duplicate (GIMP_ITEM (active_vectors), G_TYPE_FROM_INSTANCE (active_vectors), TRUE)); gimp_image_add_vectors (gimage, new_vectors, -1); gimp_image_flush (gimage); } void vectors_delete_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; GimpVectors *active_vectors; return_if_no_vectors (gimage, active_vectors, data); gimp_image_remove_vectors (gimage, active_vectors); gimp_image_flush (gimage); } void vectors_to_selection_cmd_callback (GtkWidget *widget, gpointer data, guint action) { GimpChannelOps op; GimpImage *gimage; GimpVectors *active_vectors; return_if_no_vectors (gimage, active_vectors, data); op = (GimpChannelOps) action; gimp_image_mask_select_vectors (gimage, _("Path to Selection"), active_vectors, op, TRUE, FALSE, 0, 0); gimp_image_flush (gimage); } void vectors_selection_to_vectors_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; return_if_no_image (gimage, data); vectors_selection_to_vectors (gimage, FALSE); } void vectors_stroke_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; GimpVectors *active_vectors; GimpDrawable *active_drawable; GimpToolInfo *tool_info; return_if_no_vectors (gimage, active_vectors, data); active_drawable = gimp_image_active_drawable (gimage); if (! active_drawable) { g_message (_("There is no active layer or channel to stroke to.")); return; } tool_info = gimp_context_get_tool (gimp_get_current_context (gimage->gimp)); gimp_item_stroke (GIMP_ITEM (active_vectors), active_drawable, tool_info->paint_info); gimp_image_flush (gimage); } void vectors_copy_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; GimpVectors *active_vectors; return_if_no_vectors (gimage, active_vectors, data); #ifdef __GNUC__ #warning FIXME: need vectors clipoard #endif } void vectors_paste_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; return_if_no_image (gimage, data); #ifdef __GNUC__ #warning FIXME: need vectors clipoard #endif } void vectors_import_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; GError *error = NULL; return_if_no_image (gimage, data); if (! gimp_vectors_import (gimage, "path.svg", FALSE, &error)) { g_message (error->message); g_error_free (error); } gimp_image_flush (gimage); } void vectors_export_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; GimpVectors *active_vectors; return_if_no_vectors (gimage, active_vectors, data); #ifdef __GNUC__ #warning FIXME: need vectors import/export #endif } void vectors_vectors_tool_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; GimpVectors *active_vectors; return_if_no_vectors (gimage, active_vectors, data); vectors_vectors_tool (active_vectors); } void vectors_edit_attributes_cmd_callback (GtkWidget *widget, gpointer data) { GimpImage *gimage; GimpVectors *active_vectors; return_if_no_vectors (gimage, active_vectors, data); vectors_edit_vectors_query (active_vectors); } void vectors_selection_to_vectors (GimpImage *gimage, gboolean advanced) { ProcRecord *proc_rec; Argument *args; GimpDrawable *drawable; GimpDisplay *gdisp; g_return_if_fail (GIMP_IS_IMAGE (gimage)); drawable = gimp_image_active_drawable (gimage); gdisp = gimp_context_get_display (gimp_get_user_context (gimage->gimp)); if (advanced) proc_rec = procedural_db_lookup (gimage->gimp, "plug_in_sel2path_advanced"); else proc_rec = procedural_db_lookup (gimage->gimp, "plug_in_sel2path"); if (! proc_rec) { g_message ("Selection to path procedure lookup failed."); return; } /* plug-in arguments as if called by /Filters/... */ args = g_new (Argument, 3); args[0].arg_type = GIMP_PDB_INT32; args[0].value.pdb_int = GIMP_RUN_INTERACTIVE; args[1].arg_type = GIMP_PDB_IMAGE; args[1].value.pdb_int = (gint32) gimp_image_get_ID (gimage); args[2].arg_type = GIMP_PDB_DRAWABLE; args[2].value.pdb_int = (gint32) gimp_item_get_ID (GIMP_ITEM (drawable)); plug_in_run (gimage->gimp, proc_rec, args, 3, FALSE, TRUE, gdisp ? gdisp->ID : 0); g_free (args); } void vectors_vectors_tool (GimpVectors *vectors) { GimpImage *gimage; GimpTool *active_tool; g_return_if_fail (GIMP_IS_VECTORS (vectors)); gimage = gimp_item_get_image (GIMP_ITEM (vectors)); active_tool = tool_manager_get_active (gimage->gimp); if (! GIMP_IS_VECTOR_TOOL (active_tool)) { GimpToolInfo *tool_info; tool_info = tool_manager_get_info_by_type (gimage->gimp, GIMP_TYPE_VECTOR_TOOL); gimp_context_set_tool (gimp_get_current_context (gimage->gimp), tool_info); active_tool = tool_manager_get_active (gimage->gimp); } gimp_vector_tool_set_vectors (GIMP_VECTOR_TOOL (active_tool), vectors); } /**********************************/ /* The new vectors query dialog */ /**********************************/ typedef struct _NewVectorsOptions NewVectorsOptions; struct _NewVectorsOptions { GtkWidget *query_box; GtkWidget *name_entry; GimpImage *gimage; }; static gchar *vectors_name = NULL; static void new_vectors_query_ok_callback (GtkWidget *widget, gpointer data) { NewVectorsOptions *options; GimpVectors *new_vectors; GimpImage *gimage; options = (NewVectorsOptions *) data; if (vectors_name) g_free (vectors_name); vectors_name = g_strdup (gtk_entry_get_text (GTK_ENTRY (options->name_entry))); if ((gimage = options->gimage)) { new_vectors = gimp_vectors_new (gimage, vectors_name); gimp_image_add_vectors (gimage, new_vectors, -1); gimp_image_flush (gimage); } gtk_widget_destroy (options->query_box); } void vectors_new_vectors_query (GimpImage *gimage, GimpVectors *template, gboolean interactive) { NewVectorsOptions *options; GtkWidget *hbox; GtkWidget *vbox; GtkWidget *table; GtkWidget *label; g_return_if_fail (GIMP_IS_IMAGE (gimage)); g_return_if_fail (! template || GIMP_IS_VECTORS (template)); if (template || ! interactive) { GimpVectors *new_vectors; new_vectors = gimp_vectors_new (gimage, _("Empty Vectors Copy")); gimp_image_add_vectors (gimage, new_vectors, -1); return; } /* the new options structure */ options = g_new (NewVectorsOptions, 1); options->gimage = gimage; /* The dialog */ options->query_box = gimp_viewable_dialog_new (GIMP_VIEWABLE (gimage), _("New Path"), "new_path_options", GIMP_STOCK_TOOL_PATH, _("New Path Options"), gimp_standard_help_func, GIMP_HELP_PATH_NEW, GTK_STOCK_CANCEL, gtk_widget_destroy, NULL, 1, NULL, FALSE, TRUE, GTK_STOCK_OK, new_vectors_query_ok_callback, options, NULL, NULL, TRUE, FALSE, NULL); g_object_weak_ref (G_OBJECT (options->query_box), (GWeakNotify) g_free, options); /* The main hbox */ hbox = gtk_hbox_new (FALSE, 2); gtk_container_set_border_width (GTK_CONTAINER (hbox), 4); gtk_container_add (GTK_CONTAINER (GTK_DIALOG (options->query_box)->vbox), hbox); /* The vbox */ vbox = gtk_vbox_new (FALSE, 2); gtk_box_pack_start (GTK_BOX (hbox), vbox, TRUE, TRUE, 0); /* The table */ table = gtk_table_new (2, 3, FALSE); gtk_table_set_col_spacings (GTK_TABLE (table), 4); gtk_table_set_row_spacings (GTK_TABLE (table), 2); gtk_box_pack_start (GTK_BOX (vbox), table, FALSE, FALSE, 0); /* The name entry hbox, label and entry */ label = gtk_label_new (_("Path 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 | GTK_FILL, 0, 0); gtk_widget_show (label); options->name_entry = gtk_entry_new (); gtk_widget_set_size_request (options->name_entry, 150, -1); gtk_table_attach_defaults (GTK_TABLE (table), options->name_entry, 1, 2, 0, 1); gtk_entry_set_text (GTK_ENTRY (options->name_entry), (vectors_name ? vectors_name : _("New Path"))); gtk_widget_show (options->name_entry); gtk_widget_show (table); gtk_widget_show (vbox); gtk_widget_show (hbox); gtk_widget_show (options->query_box); } /****************************************/ /* The edit vectors attributes dialog */ /****************************************/ typedef struct _EditVectorsOptions EditVectorsOptions; struct _EditVectorsOptions { GtkWidget *query_box; GtkWidget *name_entry; GimpVectors *vectors; GimpImage *gimage; }; static void edit_vectors_query_ok_callback (GtkWidget *widget, gpointer data) { EditVectorsOptions *options; GimpVectors *vectors; options = (EditVectorsOptions *) data; vectors = options->vectors; if (options->gimage) { const gchar *new_name; new_name = gtk_entry_get_text (GTK_ENTRY (options->name_entry)); if (strcmp (new_name, gimp_object_get_name (GIMP_OBJECT (vectors)))) { gimp_item_rename (GIMP_ITEM (vectors), new_name); gimp_image_flush (options->gimage); } } gtk_widget_destroy (options->query_box); } void vectors_edit_vectors_query (GimpVectors *vectors) { EditVectorsOptions *options; GtkWidget *hbox; GtkWidget *vbox; GtkWidget *table; GtkWidget *label; g_return_if_fail (GIMP_IS_VECTORS (vectors)); options = g_new0 (EditVectorsOptions, 1); options->vectors = vectors; options->gimage = gimp_item_get_image (GIMP_ITEM (vectors)); /* The dialog */ options->query_box = gimp_viewable_dialog_new (GIMP_VIEWABLE (vectors), _("Path Attributes"), "edit_path_attributes", GIMP_STOCK_EDIT, _("Edit Path Attributes"), gimp_standard_help_func, GIMP_HELP_PATH_EDIT, GTK_STOCK_CANCEL, gtk_widget_destroy, NULL, 1, NULL, FALSE, TRUE, GTK_STOCK_OK, edit_vectors_query_ok_callback, options, NULL, NULL, TRUE, FALSE, NULL); g_object_weak_ref (G_OBJECT (options->query_box), (GWeakNotify) g_free, options); /* The main hbox */ hbox = gtk_hbox_new (FALSE, 2); gtk_container_set_border_width (GTK_CONTAINER (hbox), 4); gtk_container_add (GTK_CONTAINER (GTK_DIALOG (options->query_box)->vbox), hbox); /* The vbox */ vbox = gtk_vbox_new (FALSE, 2); gtk_box_pack_start (GTK_BOX (hbox), vbox, TRUE, TRUE, 0); /* The table */ table = gtk_table_new (2, 3, FALSE); gtk_table_set_col_spacings (GTK_TABLE (table), 4); gtk_table_set_row_spacings (GTK_TABLE (table), 2); gtk_box_pack_start (GTK_BOX (vbox), table, FALSE, FALSE, 0); /* The name entry */ label = gtk_label_new (_("Path name:")); gtk_misc_set_alignment (GTK_MISC (label), 1.0, 0.5); gtk_table_attach (GTK_TABLE (table), label, 0, 1, 0, 1, GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0); gtk_widget_show (label); options->name_entry = gtk_entry_new (); gtk_widget_set_size_request (options->name_entry, 150, -1); gtk_table_attach_defaults (GTK_TABLE (table), options->name_entry, 1, 2, 0, 1); gtk_entry_set_text (GTK_ENTRY (options->name_entry), gimp_object_get_name (GIMP_OBJECT (vectors))); gtk_widget_show (options->name_entry); gtk_widget_show (table); gtk_widget_show (vbox); gtk_widget_show (hbox); gtk_widget_show (options->query_box); }