diff options
| author | Tavmjong Bah <tavmjong@free.fr> | 2012-05-23 12:00:49 +0000 |
|---|---|---|
| committer | tavmjong-free <tavmjong@free.fr> | 2012-05-23 12:00:49 +0000 |
| commit | 0ea9b448ee16fe941d46395c2877f4d2e815b9ed (patch) | |
| tree | a49cb57d691df46475d1e9fe197aeade8b0fa10c /src/widgets/mesh-toolbar.cpp | |
| parent | Correct improper flipping of sRGB transform from RGB to BGR with cairo change... (diff) | |
| download | inkscape-0ea9b448ee16fe941d46395c2877f4d2e815b9ed.tar.gz inkscape-0ea9b448ee16fe941d46395c2877f4d2e815b9ed.zip | |
Add Mesh tool (experimental, requires Cario >= 1.11.4, disabled by default).
(bzr r11406)
Diffstat (limited to 'src/widgets/mesh-toolbar.cpp')
| -rw-r--r-- | src/widgets/mesh-toolbar.cpp | 373 |
1 files changed, 373 insertions, 0 deletions
diff --git a/src/widgets/mesh-toolbar.cpp b/src/widgets/mesh-toolbar.cpp new file mode 100644 index 000000000..7c37d757c --- /dev/null +++ b/src/widgets/mesh-toolbar.cpp @@ -0,0 +1,373 @@ +/* + * Gradient aux toolbar + * + * Authors: + * bulia byak <bulia@dr.com> + * Johan Engelen <j.b.c.engelen@ewi.utwente.nl> + * Abhishek Sharma + * Tavmjong Bah <tavjong@free.fr> + * + * Copyright (C) 2012 Tavmjong Bah + * Copyright (C) 2007 Johan Engelen + * Copyright (C) 2005 authors + * + * Released under GNU GPL, read the file 'COPYING' for more information + */ + +#ifdef HAVE_CONFIG_H +# include "config.h" +#endif + +// REVIEW THESE AT END OF REWRITE +#include "toolbox.h" +#include "mesh-toolbar.h" + +#include "ui/widget/color-preview.h" +#include "verbs.h" + +#include "macros.h" +#include "widgets/button.h" +#include "widgets/widget-sizes.h" +#include "widgets/spw-utilities.h" +#include "widgets/spinbutton-events.h" +#include "widgets/gradient-vector.h" +#include "widgets/gradient-image.h" +#include "style.h" + +#include "preferences.h" +#include "document-private.h" +#include "document-undo.h" +#include "desktop.h" +#include "desktop-handles.h" +#include <glibmm/i18n.h> + +#include "gradient-context.h" +#include "gradient-drag.h" +#include "sp-mesh-gradient.h" +#include "gradient-chemistry.h" +#include "gradient-selector.h" +#include "selection.h" +#include "ui/icon-names.h" + +#include "../ege-adjustment-action.h" +#include "../ege-output-action.h" +#include "../ege-select-one-action.h" +#include "../ink-action.h" +#include "../ink-comboboxentry-action.h" + +#include "sp-stop.h" +#include "svg/css-ostringstream.h" +#include "svg/svg-color.h" +#include "desktop-style.h" +#include "gradient-context.h" + +#include "toolbox.h" + +using Inkscape::DocumentUndo; +using Inkscape::UI::ToolboxFactory; +using Inkscape::UI::PrefPusher; + +static gboolean blocked = FALSE; + +//######################## +//## Mesh ## +//######################## + +/* + * Core function, setup all the widgets whenever something changes on the desktop + */ +static void ms_tb_selection_changed(Inkscape::Selection * /*selection*/, gpointer data) +{ + // DOES NOTHING AT MOMENT + + // std::cout << "ms_tb_selection_changed" << std::endl; + + // if (blocked) + // return; + + // GtkWidget *widget = GTK_WIDGET(data); + + // SPDesktop *desktop = static_cast<SPDesktop *>(g_object_get_data(G_OBJECT(widget), "desktop")); + // if (!desktop) { + // return; + // } + + // Inkscape::Selection *selection = sp_desktop_selection(desktop); // take from desktop, not from args + // if (selection) { + // SPEventContext *ev = sp_desktop_event_context(desktop); + // GrDrag *drag = NULL; + // if (ev) { + // drag = ev->get_drag(); + // // Hide/show handles? + // } + + // } +} + + +static void ms_tb_selection_modified(Inkscape::Selection *selection, guint /*flags*/, gpointer data) +{ + ms_tb_selection_changed(selection, data); +} + +static void ms_drag_selection_changed(gpointer /*dragger*/, gpointer data) +{ + ms_tb_selection_changed(NULL, data); + +} + +static void ms_defs_release(SPObject * defs, GtkWidget *widget) +{ + ms_tb_selection_changed(NULL, (gpointer) widget); +} + +static void ms_defs_modified(SPObject * /*defs*/, guint /*flags*/, GtkWidget *widget) +{ + ms_tb_selection_changed(NULL, (gpointer) widget); +} + +static void ms_disconnect_sigc(GObject * /*obj*/, sigc::connection *connection) { + connection->disconnect(); + delete connection; +} + + +/* + * Callback functions for user actions + */ + +static void ms_new_type_changed( EgeSelectOneAction *act, GObject * /*tbl*/ ) +{ + Inkscape::Preferences *prefs = Inkscape::Preferences::get(); + gint typemode = ege_select_one_action_get_active( act ) == 0 ? SP_GRADIENT_MESH_TYPE_NORMAL : SP_GRADIENT_MESH_TYPE_CONICAL; + prefs->setInt("/tools/mesh/mesh_type", typemode); +} + +static void ms_new_fillstroke_changed( EgeSelectOneAction *act, GObject * /*tbl*/ ) +{ + Inkscape::Preferences *prefs = Inkscape::Preferences::get(); + Inkscape::PaintTarget fsmode = (ege_select_one_action_get_active( act ) == 0) ? Inkscape::FOR_FILL : Inkscape::FOR_STROKE; + prefs->setInt("/tools/gradient/newfillorstroke", (fsmode == Inkscape::FOR_FILL) ? 1 : 0); +} + +static void ms_row_changed (GtkAdjustment *adj, GObject *tbl ) { + + if (blocked) { + return; + } + + blocked = TRUE; + + int rows = gtk_adjustment_get_value(adj); + + Inkscape::Preferences *prefs = Inkscape::Preferences::get(); + + prefs->setInt("/tools/mesh/mesh_rows", rows); + + blocked = FALSE; +} + +static void ms_col_changed (GtkAdjustment *adj, GObject *tbl ) { + + if (blocked) { + return; + } + + blocked = TRUE; + + int cols = gtk_adjustment_get_value(adj); + + Inkscape::Preferences *prefs = Inkscape::Preferences::get(); + + prefs->setInt("/tools/mesh/mesh_cols", cols); + + blocked = FALSE; +} + +static void ms_edit_fillstroke_changed( EgeSelectOneAction *act, GObject * /*tbl*/ ) +{ + Inkscape::Preferences *prefs = Inkscape::Preferences::get(); + Inkscape::PaintTarget fsmode = (ege_select_one_action_get_active( act ) == 0) ? Inkscape::FOR_FILL : Inkscape::FOR_STROKE; + prefs->setInt("/tools/gradient/editfillorstroke", (fsmode == Inkscape::FOR_FILL) ? 1 : 0); +} + +/** + * Mesh auxiliary toolbar construction and setup. + * + */ +void sp_mesh_toolbox_prep(SPDesktop * desktop, GtkActionGroup* mainActions, GObject* holder) +{ + Inkscape::IconSize secondarySize = ToolboxFactory::prefToSize("/toolbox/secondary", 1); + + EgeAdjustmentAction* eact = 0; + + /* New mesh: normal or conical */ + { + GtkListStore* model = gtk_list_store_new( 3, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING ); + + GtkTreeIter iter; + gtk_list_store_append( model, &iter ); + gtk_list_store_set( model, &iter, + 0, _("normal"), 1, _("Create mesh gradient"), 2, INKSCAPE_ICON("paint-gradient-mesh"), -1 ); + + gtk_list_store_append( model, &iter ); + gtk_list_store_set( model, &iter, + 0, _("conical"), 1, _("Create conical gradient"), 2, INKSCAPE_ICON("paint-gradient-conical"), -1 ); + + EgeSelectOneAction* act = ege_select_one_action_new( "MeshNewTypeAction", (""), (""), NULL, GTK_TREE_MODEL(model) ); + g_object_set( act, "short_label", _("New:"), NULL ); + gtk_action_group_add_action( mainActions, GTK_ACTION(act) ); + g_object_set_data( holder, "mesh_new_type_action", act ); + + ege_select_one_action_set_appearance( act, "full" ); + ege_select_one_action_set_radio_action_type( act, INK_RADIO_ACTION_TYPE ); + g_object_set( G_OBJECT(act), "icon-property", "iconId", NULL ); + ege_select_one_action_set_icon_column( act, 2 ); + ege_select_one_action_set_tooltip_column( act, 1 ); + + Inkscape::Preferences *prefs = Inkscape::Preferences::get(); + gint mode = prefs->getInt("/tools/mesh/mesh_type", SP_GRADIENT_MESH_TYPE_NORMAL) != SP_GRADIENT_MESH_TYPE_NORMAL; + ege_select_one_action_set_active( act, mode ); + g_signal_connect_after( G_OBJECT(act), "changed", G_CALLBACK(ms_new_type_changed), holder ); + } + + /* New gradient on fill or stroke*/ + { + GtkListStore* model = gtk_list_store_new( 3, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING ); + + GtkTreeIter iter; + gtk_list_store_append( model, &iter ); + gtk_list_store_set( model, &iter, + 0, _("fill"), 1, _("Create gradient in the fill"), 2, INKSCAPE_ICON("object-fill"), -1 ); + + gtk_list_store_append( model, &iter ); + gtk_list_store_set( model, &iter, + 0, _("stroke"), 1, _("Create gradient in the stroke"), 2, INKSCAPE_ICON("object-stroke"), -1 ); + + EgeSelectOneAction* act = ege_select_one_action_new( "MeshNewFillStrokeAction", (""), (""), NULL, GTK_TREE_MODEL(model) ); + g_object_set( act, "short_label", _("on:"), NULL ); + gtk_action_group_add_action( mainActions, GTK_ACTION(act) ); + g_object_set_data( holder, "mesh_new_fillstroke_action", act ); + + ege_select_one_action_set_appearance( act, "full" ); + ege_select_one_action_set_radio_action_type( act, INK_RADIO_ACTION_TYPE ); + g_object_set( G_OBJECT(act), "icon-property", "iconId", NULL ); + ege_select_one_action_set_icon_column( act, 2 ); + ege_select_one_action_set_tooltip_column( act, 1 ); + + /// @todo Convert to boolean? + Inkscape::Preferences *prefs = Inkscape::Preferences::get(); + bool fillstrokemode = !prefs->getBool("/tools/gradient/newfillorstroke", true); + ege_select_one_action_set_active( act, fillstrokemode ); + g_signal_connect_after( G_OBJECT(act), "changed", G_CALLBACK(ms_new_fillstroke_changed), holder ); + } + + /* Number of mesh rows */ + { + gchar const* labels[] = {}; + gdouble values[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10}; + eact = create_adjustment_action( "MeshRowAction", + _("Rows"), _("Rows:"), _("Number of rows in new mesh"), + "/tools/mesh/mesh_rows", 1, + GTK_WIDGET(desktop->canvas), NULL, holder, FALSE, NULL, + 1, 20, 1, 1, + labels, values, G_N_ELEMENTS(labels), + ms_row_changed, + 1.0, 0 ); + gtk_action_group_add_action( mainActions, GTK_ACTION(eact) ); + gtk_action_set_sensitive( GTK_ACTION(eact), TRUE ); + } + + /* Number of mesh columns */ + { + gchar const* labels[] = {}; + gdouble values[] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10}; + eact = create_adjustment_action( "MeshColumnAction", + _("Columns"), _("Columns:"), _("Number of columns in new mesh"), + "/tools/mesh/mesh_cols", 1, + GTK_WIDGET(desktop->canvas), NULL, holder, FALSE, NULL, + 1, 20, 1, 1, + labels, values, G_N_ELEMENTS(labels), + ms_col_changed, + 1.0, 0 ); + gtk_action_group_add_action( mainActions, GTK_ACTION(eact) ); + gtk_action_set_sensitive( GTK_ACTION(eact), TRUE ); + } + + /* Edit fill mesh */ + { + InkToggleAction* act = ink_toggle_action_new( "MeshEditFillAction", + _("Edit Fill"), + _("Edit fill mesh"), + INKSCAPE_ICON("object-fill"), + secondarySize ); + gtk_action_group_add_action( mainActions, GTK_ACTION( act ) ); + PrefPusher *pusher = new PrefPusher(GTK_TOGGLE_ACTION(act), "/tools/mesh/edit_fill"); + g_signal_connect( holder, "destroy", G_CALLBACK(delete_prefspusher), pusher); + } + + /* Edit stroke mesh */ + { + InkToggleAction* act = ink_toggle_action_new( "MeshEditStrokeAction", + _("Edit Stroke"), + _("Edit stroke mesh"), + INKSCAPE_ICON("object-stroke"), + secondarySize ); + gtk_action_group_add_action( mainActions, GTK_ACTION( act ) ); + PrefPusher *pusher = new PrefPusher(GTK_TOGGLE_ACTION(act), "/tools/mesh/edit_stroke"); + g_signal_connect( holder, "destroy", G_CALLBACK(delete_prefspusher), pusher); + } + + /* Show/hide side and tensor handles */ + { + InkToggleAction* act = ink_toggle_action_new( "MeshShowHandlesAction", + _("Show Handles"), + _("Show side and tensor handles"), + INKSCAPE_ICON("show-node-handles"), + secondarySize ); + gtk_action_group_add_action( mainActions, GTK_ACTION( act ) ); + PrefPusher *pusher = new PrefPusher(GTK_TOGGLE_ACTION(act), "/tools/mesh/show_handles"); + g_signal_connect( holder, "destroy", G_CALLBACK(delete_prefspusher), pusher); + } + + + Inkscape::Selection *selection = sp_desktop_selection (desktop); + SPDocument *document = sp_desktop_document (desktop); + + g_object_set_data(holder, "desktop", desktop); + + // connect to selection modified and changed signals + sigc::connection *conn1 = new sigc::connection( + selection->connectChanged(sigc::bind(sigc::ptr_fun(&ms_tb_selection_changed), (gpointer) holder))); + sigc::connection *conn2 = new sigc::connection( + selection->connectModified(sigc::bind(sigc::ptr_fun(&ms_tb_selection_modified), (gpointer) holder))); + sigc::connection *conn3 = new sigc::connection( + desktop->connectToolSubselectionChanged( sigc::bind(sigc::ptr_fun(&ms_drag_selection_changed), (gpointer) holder))); + + // when holder is destroyed, disconnect + g_signal_connect(G_OBJECT(holder), "destroy", G_CALLBACK(ms_disconnect_sigc), conn1); + g_signal_connect(G_OBJECT(holder), "destroy", G_CALLBACK(ms_disconnect_sigc), conn2); + g_signal_connect(G_OBJECT(holder), "destroy", G_CALLBACK(ms_disconnect_sigc), conn3); + + // connect to release and modified signals of the defs (i.e. when someone changes mesh) + sigc::connection *release_connection = new sigc::connection(); + *release_connection = document->getDefs()->connectRelease(sigc::bind<1>(sigc::ptr_fun(&ms_defs_release), GTK_WIDGET(holder))); + sigc::connection *modified_connection = new sigc::connection(); + *modified_connection = document->getDefs()->connectModified(sigc::bind<2>(sigc::ptr_fun(&ms_defs_modified), GTK_WIDGET(holder))); + + // when holder is destroyed, disconnect + g_signal_connect(G_OBJECT(holder), "destroy", G_CALLBACK(ms_disconnect_sigc), release_connection); + g_signal_connect(G_OBJECT(holder), "destroy", G_CALLBACK(ms_disconnect_sigc), modified_connection); + +} + +/* + Local Variables: + mode:c++ + c-file-style:"stroustrup" + c-file-offsets:((innamespace . 0)(inline-open . 0)(case-label . +)) + indent-tabs-mode:nil + fill-column:99 + End: +*/ +// vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4 : |
