paint-selector.cpp revision 32dbfdf82f6abb93aeed3f0d1cf66a2cf62164aa
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler/** \file
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler * SPPaintSelector: Generic paint selector widget.
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler */
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler/*
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler * Authors:
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler * Lauris Kaplinski
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler * bulia byak <buliabyak@users.sf.net>
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler * John Cliff <simarilius@yahoo.com>
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler * Jon A. Cruz <jon@joncruz.org>
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński * Abhishek Sharma
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński *
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler * Copyright (C) Lauris Kaplinski 2002
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler * Copyright (C) 2010 Authors
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler*/
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#define noSP_PS_VERBOSE
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#ifdef HAVE_CONFIG_H
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler# include "config.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#endif
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include <cstring>
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include <string>
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
76906e4a110b558e62eab96cd9549c2d8197123bKris#include <gtk/gtk.h>
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "../sp-pattern.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include <glibmm/i18n.h>
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "../widgets/icon.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "widgets/widget-sizes.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "xml/repr.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "sp-color-notebook.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "sp-linear-gradient-fns.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "sp-radial-gradient-fns.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler/* fixme: Move it from dialogs to here */
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "gradient-selector.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include <inkscape.h>
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński#include <document-private.h>
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include <desktop-style.h>
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include <style.h>
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "svg/svg-color.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "svg/css-ostringstream.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "path-prefix.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "io/sys.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "helper/stock-items.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "ui/icon-names.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "widgets/swatch-selector.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "paint-selector.h"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#ifdef SP_PS_VERBOSE
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#include "svg/svg-icc-color.h"
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński#endif // SP_PS_VERBOSE
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerusing Inkscape::Widgets::SwatchSelector;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerenum {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler MODE_CHANGED,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler GRABBED,
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński DRAGGED,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler RELEASED,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler CHANGED,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler FILLRULE_CHANGED,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler LAST_SIGNAL
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler};
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosińskistatic void sp_paint_selector_class_init(SPPaintSelectorClass *klass);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void sp_paint_selector_init(SPPaintSelector *slider);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void sp_paint_selector_destroy(GtkObject *object);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic GtkWidget *sp_paint_selector_style_button_add(SPPaintSelector *psel, gchar const *px, SPPaintSelector::Mode mode, gchar const *tip);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void sp_paint_selector_style_button_toggled(GtkToggleButton *tb, SPPaintSelector *psel);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void sp_paint_selector_fillrule_toggled(GtkToggleButton *tb, SPPaintSelector *psel);
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void sp_paint_selector_set_mode_empty(SPPaintSelector *psel);
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosińskistatic void sp_paint_selector_set_mode_multiple(SPPaintSelector *psel);
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosińskistatic void sp_paint_selector_set_mode_none(SPPaintSelector *psel);
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosińskistatic void sp_paint_selector_set_mode_color(SPPaintSelector *psel, SPPaintSelector::Mode mode);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void sp_paint_selector_set_mode_gradient(SPPaintSelector *psel, SPPaintSelector::Mode mode);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void sp_paint_selector_set_mode_pattern(SPPaintSelector *psel, SPPaintSelector::Mode mode);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void sp_paint_selector_set_mode_swatch(SPPaintSelector *psel, SPPaintSelector::Mode mode);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void sp_paint_selector_set_mode_unset(SPPaintSelector *psel);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void sp_paint_selector_set_style_buttons(SPPaintSelector *psel, GtkWidget *active);
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosińskistatic GtkVBoxClass *parent_class;
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosińskistatic guint psel_signals[LAST_SIGNAL] = {0};
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#ifdef SP_PS_VERBOSE
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic gchar const* modeStrings[] = {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler "MODE_EMPTY",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler "MODE_MULTIPLE",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler "MODE_NONE",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler "MODE_COLOR_RGB",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler "MODE_COLOR_CMYK",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler "MODE_GRADIENT_LINEAR",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler "MODE_GRADIENT_RADIAL",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler "MODE_PATTERN",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler "MODE_SWATCH",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler "MODE_UNSET",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler ".",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler ".",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler ".",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler};
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#endif
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic bool isPaintModeGradient( SPPaintSelector::Mode mode )
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler bool isGrad = (mode == SPPaintSelector::MODE_GRADIENT_LINEAR) ||
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler (mode == SPPaintSelector::MODE_GRADIENT_RADIAL) ||
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler (mode == SPPaintSelector::MODE_SWATCH);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler return isGrad;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic SPGradientSelector *getGradientFromData(SPPaintSelector const *psel)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler SPGradientSelector *grad = 0;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler if (psel->mode == SPPaintSelector::MODE_SWATCH) {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler SwatchSelector *swatchsel = static_cast<SwatchSelector*>(g_object_get_data(G_OBJECT(psel->selector), "swatch-selector"));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler if (swatchsel) {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler grad = swatchsel->getGradientSelector();
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler }
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler } else {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler grad = reinterpret_cast<SPGradientSelector*>(gtk_object_get_data(GTK_OBJECT(psel->selector), "gradient-selector"));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler }
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler return grad;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh AndlerGType sp_paint_selector_get_type(void)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler static GtkType type = 0;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler if (!type) {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler GTypeInfo info = {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler sizeof(SPPaintSelectorClass),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler 0, // base_init
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler 0, // base_finalize
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler (GClassInitFunc)sp_paint_selector_class_init,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler 0, // class_finalize
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler 0, // class_data
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler sizeof(SPPaintSelector),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler 0, // n_preallocs
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski (GInstanceInitFunc)sp_paint_selector_init,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler 0 // value_table
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler };
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler type = g_type_register_static(GTK_TYPE_VBOX, "SPPaintSelector", &info, static_cast<GTypeFlags>(0));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler }
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler return type;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlersp_paint_selector_class_init(SPPaintSelectorClass *klass)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler GtkObjectClass *object_class;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler GtkWidgetClass *widget_class;
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler object_class = (GtkObjectClass *) klass;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler widget_class = (GtkWidgetClass *) klass;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler parent_class = (GtkVBoxClass*)gtk_type_class(GTK_TYPE_VBOX);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel_signals[MODE_CHANGED] = g_signal_new("mode_changed",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_TYPE_FROM_CLASS(object_class),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler (GSignalFlags)(G_SIGNAL_RUN_FIRST | G_SIGNAL_NO_RECURSE),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_STRUCT_OFFSET(SPPaintSelectorClass, mode_changed),
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski NULL, NULL,
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski gtk_marshal_NONE__UINT,
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski G_TYPE_NONE, 1, GTK_TYPE_UINT);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel_signals[GRABBED] = g_signal_new("grabbed",
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski G_TYPE_FROM_CLASS(object_class),
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski (GSignalFlags)(G_SIGNAL_RUN_FIRST | G_SIGNAL_NO_RECURSE),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_STRUCT_OFFSET(SPPaintSelectorClass, grabbed),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler NULL, NULL,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_marshal_NONE__NONE,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_TYPE_NONE, 0);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel_signals[DRAGGED] = g_signal_new("dragged",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_TYPE_FROM_CLASS(object_class),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler (GSignalFlags)(G_SIGNAL_RUN_FIRST | G_SIGNAL_NO_RECURSE),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_STRUCT_OFFSET(SPPaintSelectorClass, dragged),
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski NULL, NULL,
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski gtk_marshal_NONE__NONE,
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski G_TYPE_NONE, 0);
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski psel_signals[RELEASED] = g_signal_new("released",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_TYPE_FROM_CLASS(object_class),
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski (GSignalFlags)(G_SIGNAL_RUN_FIRST | G_SIGNAL_NO_RECURSE),
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski G_STRUCT_OFFSET(SPPaintSelectorClass, released),
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski NULL, NULL,
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski gtk_marshal_NONE__NONE,
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski G_TYPE_NONE, 0);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel_signals[CHANGED] = g_signal_new("changed",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_TYPE_FROM_CLASS(object_class),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler (GSignalFlags)(G_SIGNAL_RUN_FIRST | G_SIGNAL_NO_RECURSE),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_STRUCT_OFFSET(SPPaintSelectorClass, changed),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler NULL, NULL,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_marshal_NONE__NONE,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_TYPE_NONE, 0);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel_signals[FILLRULE_CHANGED] = g_signal_new("fillrule_changed",
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_TYPE_FROM_CLASS(object_class),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler (GSignalFlags)(G_SIGNAL_RUN_FIRST | G_SIGNAL_NO_RECURSE),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_STRUCT_OFFSET(SPPaintSelectorClass, fillrule_changed),
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler NULL, NULL,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_marshal_NONE__UINT,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler G_TYPE_NONE, 1, GTK_TYPE_UINT);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler object_class->destroy = sp_paint_selector_destroy;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#define XPAD 4
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#define YPAD 1
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlersp_paint_selector_init(SPPaintSelector *psel)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel->mode = static_cast<SPPaintSelector::Mode>(-1); // huh? do you mean 0xff? -- I think this means "not in the enum"
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler /* Paint style button box */
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel->style = gtk_hbox_new(FALSE, 0);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_widget_show(psel->style);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_container_set_border_width(GTK_CONTAINER(psel->style), 4);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_box_pack_start(GTK_BOX(psel), psel->style, FALSE, FALSE, 0);
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński /* Buttons */
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński psel->none = sp_paint_selector_style_button_add(psel, INKSCAPE_ICON_PAINT_NONE,
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński SPPaintSelector::MODE_NONE, _("No paint"));
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński psel->solid = sp_paint_selector_style_button_add(psel, INKSCAPE_ICON_PAINT_SOLID,
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński SPPaintSelector::MODE_COLOR_RGB, _("Flat color"));
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński psel->gradient = sp_paint_selector_style_button_add(psel, INKSCAPE_ICON_PAINT_GRADIENT_LINEAR,
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński SPPaintSelector::MODE_GRADIENT_LINEAR, _("Linear gradient"));
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński psel->radial = sp_paint_selector_style_button_add(psel, INKSCAPE_ICON_PAINT_GRADIENT_RADIAL,
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński SPPaintSelector::MODE_GRADIENT_RADIAL, _("Radial gradient"));
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński psel->pattern = sp_paint_selector_style_button_add(psel, INKSCAPE_ICON_PAINT_PATTERN,
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński SPPaintSelector::MODE_PATTERN, _("Pattern"));
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński psel->swatch = sp_paint_selector_style_button_add(psel, INKSCAPE_ICON_PAINT_SWATCH,
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński SPPaintSelector::MODE_SWATCH, _("Swatch"));
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński psel->unset = sp_paint_selector_style_button_add(psel, INKSCAPE_ICON_PAINT_UNKNOWN,
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński SPPaintSelector::MODE_UNSET, _("Unset paint (make it undefined so it can be inherited)"));
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński /* Fillrule */
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński {
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński psel->fillrulebox = gtk_hbox_new(FALSE, 0);
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński gtk_box_pack_end(GTK_BOX(psel->style), psel->fillrulebox, FALSE, FALSE, 0);
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński GtkWidget *w;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel->evenodd = gtk_radio_button_new(NULL);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_button_set_relief(GTK_BUTTON(psel->evenodd), GTK_RELIEF_NONE);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_toggle_button_set_mode(GTK_TOGGLE_BUTTON(psel->evenodd), FALSE);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler // TRANSLATORS: for info, see http://www.w3.org/TR/2000/CR-SVG-20000802/painting.html#FillRuleProperty
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_widget_set_tooltip_text(psel->evenodd, _("Any path self-intersections or subpaths create holes in the fill (fill-rule: evenodd)"));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_object_set_data(GTK_OBJECT(psel->evenodd), "mode", GUINT_TO_POINTER(SPPaintSelector::FILLRULE_EVENODD));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler w = sp_icon_new(Inkscape::ICON_SIZE_DECORATION, INKSCAPE_ICON_FILL_RULE_EVEN_ODD);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_container_add(GTK_CONTAINER(psel->evenodd), w);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_box_pack_start(GTK_BOX(psel->fillrulebox), psel->evenodd, FALSE, FALSE, 0);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_signal_connect(G_OBJECT(psel->evenodd), "toggled", G_CALLBACK(sp_paint_selector_fillrule_toggled), psel);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel->nonzero = gtk_radio_button_new(gtk_radio_button_get_group(GTK_RADIO_BUTTON(psel->evenodd)));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_button_set_relief(GTK_BUTTON(psel->nonzero), GTK_RELIEF_NONE);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_toggle_button_set_mode(GTK_TOGGLE_BUTTON(psel->nonzero), FALSE);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler // TRANSLATORS: for info, see http://www.w3.org/TR/2000/CR-SVG-20000802/painting.html#FillRuleProperty
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_widget_set_tooltip_text(psel->nonzero, _("Fill is solid unless a subpath is counterdirectional (fill-rule: nonzero)"));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_object_set_data(GTK_OBJECT(psel->nonzero), "mode", GUINT_TO_POINTER(SPPaintSelector::FILLRULE_NONZERO));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler w = sp_icon_new(Inkscape::ICON_SIZE_DECORATION, INKSCAPE_ICON_FILL_RULE_NONZERO);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_container_add(GTK_CONTAINER(psel->nonzero), w);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_box_pack_start(GTK_BOX(psel->fillrulebox), psel->nonzero, FALSE, FALSE, 0);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_signal_connect(G_OBJECT(psel->nonzero), "toggled", G_CALLBACK(sp_paint_selector_fillrule_toggled), psel);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler }
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler /* Frame */
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel->frame = gtk_frame_new("");
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_widget_show(psel->frame);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_container_set_border_width(GTK_CONTAINER(psel->frame), 0);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_box_pack_start(GTK_BOX(psel), psel->frame, TRUE, TRUE, 0);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler /* Last used color */
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel->color.set( 0.0, 0.0, 0.0 );
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel->alpha = 1.0;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlersp_paint_selector_destroy(GtkObject *object)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler SPPaintSelector *psel = SP_PAINT_SELECTOR(object);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler // clean up our long-living pattern menu
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_object_set_data(G_OBJECT(psel),"patternmenu",NULL);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler if (((GtkObjectClass *) parent_class)->destroy)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler (* ((GtkObjectClass *) parent_class)->destroy)(object);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic GtkWidget *sp_paint_selector_style_button_add(SPPaintSelector *psel,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gchar const *pixmap, SPPaintSelector::Mode mode,
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gchar const *tip)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler GtkWidget *b, *w;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski b = gtk_toggle_button_new();
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski gtk_widget_set_tooltip_text(b, tip);
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski gtk_widget_show(b);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski gtk_container_set_border_width(GTK_CONTAINER(b), 0);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_button_set_relief(GTK_BUTTON(b), GTK_RELIEF_NONE);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_toggle_button_set_mode(GTK_TOGGLE_BUTTON(b), FALSE);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_object_set_data(GTK_OBJECT(b), "mode", GUINT_TO_POINTER(mode));
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler w = sp_icon_new(Inkscape::ICON_SIZE_BUTTON, pixmap);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_widget_show(w);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_container_add(GTK_CONTAINER(b), w);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_box_pack_start(GTK_BOX(psel->style), b, FALSE, FALSE, 0);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_signal_connect(G_OBJECT(b), "toggled", G_CALLBACK(sp_paint_selector_style_button_toggled), psel);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler return b;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlersp_paint_selector_style_button_toggled(GtkToggleButton *tb, SPPaintSelector *psel)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler if (!psel->update && gtk_toggle_button_get_active(tb)) {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler psel->setMode(static_cast<SPPaintSelector::Mode>(GPOINTER_TO_UINT(gtk_object_get_data(GTK_OBJECT(tb), "mode"))));
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski }
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlerstatic void
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlersp_paint_selector_fillrule_toggled(GtkToggleButton *tb, SPPaintSelector *psel)
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski{
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski if (!psel->update && gtk_toggle_button_get_active(tb)) {
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski SPPaintSelector::FillRule fr = static_cast<SPPaintSelector::FillRule>(GPOINTER_TO_UINT(g_object_get_data(G_OBJECT(tb), "mode")));
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski g_signal_emit(G_OBJECT(psel), psel_signals[FILLRULE_CHANGED], 0, fr);
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski }
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski}
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlervoid
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlersp_paint_selector_show_fillrule(SPPaintSelector *psel, bool is_fill)
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski{
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski if (psel->fillrulebox) {
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski if (is_fill) {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_widget_show_all(psel->fillrulebox);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler } else {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_widget_destroy(psel->fillrulebox);
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski psel->fillrulebox = NULL;
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski }
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler }
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh AndlerSPPaintSelector *sp_paint_selector_new(FillOrStroke kind)
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski{
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski SPPaintSelector *psel = static_cast<SPPaintSelector*>(gtk_type_new(SP_TYPE_PAINT_SELECTOR));
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski psel->setMode(SPPaintSelector::MODE_MULTIPLE);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler // This silliness is here because I don't know how to pass a parameter to the
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler // GtkObject "constructor" (sp_paint_selector_init). Remove it when paint_selector
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski // becomes a normal class.
7bda77e763c0af49270427593108b66455dfd125Krzysztof Kosinski sp_paint_selector_show_fillrule(psel, kind == FILL);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler return psel;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlervoid SPPaintSelector::setMode(Mode mode)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński if (this->mode != mode) {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler update = TRUE;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#ifdef SP_PS_VERBOSE
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_print("Mode change %d -> %d %s -> %s\n", this->mode, mode, modeStrings[this->mode], modeStrings[mode]);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#endif
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler switch (mode) {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler case MODE_EMPTY:
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler sp_paint_selector_set_mode_empty(this);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler break;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler case MODE_MULTIPLE:
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler sp_paint_selector_set_mode_multiple(this);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler break;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler case MODE_NONE:
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński sp_paint_selector_set_mode_none(this);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler break;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler case MODE_COLOR_RGB:
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler case MODE_COLOR_CMYK:
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler sp_paint_selector_set_mode_color(this, mode);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler break;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler case MODE_GRADIENT_LINEAR:
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler case MODE_GRADIENT_RADIAL:
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler sp_paint_selector_set_mode_gradient(this, mode);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler break;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler case MODE_PATTERN:
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler sp_paint_selector_set_mode_pattern(this, mode);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler break;
76addc201c409e81eaaa73fe27cc0f79c4db097cKrzysztof Kosiński case MODE_SWATCH:
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler sp_paint_selector_set_mode_swatch(this, mode);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler break;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler case MODE_UNSET:
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler sp_paint_selector_set_mode_unset(this);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler break;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler default:
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_warning("file %s: line %d: Unknown paint mode %d", __FILE__, __LINE__, mode);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler break;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler }
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler this->mode = mode;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_signal_emit(G_OBJECT(this), psel_signals[MODE_CHANGED], 0, this->mode);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler update = FALSE;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler }
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlervoid SPPaintSelector::setFillrule(FillRule fillrule)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler if (fillrulebox) {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler // TODO this flips widgets but does not use a member to store state. Revisit
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(evenodd), (fillrule == FILLRULE_EVENODD));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(nonzero), (fillrule == FILLRULE_NONZERO));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler }
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlervoid SPPaintSelector::setColorAlpha(SPColor const &color, float alpha)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_return_if_fail( ( 0.0 <= alpha ) && ( alpha <= 1.0 ) );
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler SPColorSelector *csel = 0;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler guint32 rgba = 0;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler/*
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler if ( sp_color_get_colorspace_type(color) == SP_COLORSPACE_TYPE_CMYK )
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler {
40742313779ee5e43be93a9191f1c86412cf183bKrzysztof Kosiński#ifdef SP_PS_VERBOSE
40742313779ee5e43be93a9191f1c86412cf183bKrzysztof Kosiński g_print("PaintSelector set CMYKA\n");
40742313779ee5e43be93a9191f1c86412cf183bKrzysztof Kosiński#endif
40742313779ee5e43be93a9191f1c86412cf183bKrzysztof Kosiński sp_paint_selector_set_mode(psel, MODE_COLOR_CMYK);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler }
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler else
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler*/
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#ifdef SP_PS_VERBOSE
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_print("PaintSelector set RGBA\n");
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#endif
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler setMode(MODE_COLOR_RGB);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler }
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler csel = reinterpret_cast<SPColorSelector*>(gtk_object_get_data(GTK_OBJECT(selector), "color-selector"));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler rgba = color.toRGBA32( alpha );
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler csel->base->setColorAlpha( color, alpha );
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlervoid SPPaintSelector::setSwatch(SPGradient *vector )
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#ifdef SP_PS_VERBOSE
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_print("PaintSelector set SWATCH\n");
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#endif
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler setMode(MODE_SWATCH);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler SwatchSelector *swatchsel = static_cast<SwatchSelector*>(g_object_get_data(G_OBJECT(selector), "swatch-selector"));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler if (swatchsel) {
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler swatchsel->setVector( (vector) ? vector->document : 0, vector );
d1bde559850436556ebee2e70e10f1cfc8aff636Krzysztof Kosiński }
d1bde559850436556ebee2e70e10f1cfc8aff636Krzysztof Kosiński}
d1bde559850436556ebee2e70e10f1cfc8aff636Krzysztof Kosiński
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlervoid SPPaintSelector::setGradientLinear(SPGradient *vector)
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#ifdef SP_PS_VERBOSE
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_print("PaintSelector set GRADIENT LINEAR\n");
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler#endif
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler setMode(MODE_GRADIENT_LINEAR);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler SPGradientSelector *gsel = getGradientFromData(this);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gsel->setMode(SPGradientSelector::MODE_LINEAR);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gsel->setVector((vector) ? vector->document : 0, vector);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosińskivoid SPPaintSelector::setGradientRadial(SPGradient *vector)
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński{
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński#ifdef SP_PS_VERBOSE
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński g_print("PaintSelector set GRADIENT RADIAL\n");
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński#endif
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński setMode(MODE_GRADIENT_RADIAL);
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński SPGradientSelector *gsel = getGradientFromData(this);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gsel->setMode(SPGradientSelector::MODE_RADIAL);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler gsel->setVector((vector) ? vector->document : 0, vector);
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosińskivoid SPPaintSelector::setGradientProperties( SPGradientUnits units, SPGradientSpread spread )
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński{
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński g_return_if_fail(isPaintModeGradient(mode));
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński SPGradientSelector *gsel = getGradientFromData(this);
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński gsel->setUnits(units);
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński gsel->setSpread(spread);
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński}
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński
f6df6236fcf18524e049bba12b69704af32eff90Josh Andlervoid SPPaintSelector::getGradientProperties( SPGradientUnits &units, SPGradientSpread &spread) const
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_return_if_fail(isPaintModeGradient(mode));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński SPGradientSelector *gsel = getGradientFromData(this);
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński units = gsel->getUnits();
e4369b05aaa20df73a37f4d319ce456865cc74f3Krzysztof Kosiński spread = gsel->getSpread();
d1bde559850436556ebee2e70e10f1cfc8aff636Krzysztof Kosiński}
d1bde559850436556ebee2e70e10f1cfc8aff636Krzysztof Kosiński
d1bde559850436556ebee2e70e10f1cfc8aff636Krzysztof Kosiński/**
d1bde559850436556ebee2e70e10f1cfc8aff636Krzysztof Kosiński * \post (alpha == NULL) || (*alpha in [0.0, 1.0]).
d1bde559850436556ebee2e70e10f1cfc8aff636Krzysztof Kosiński */
d1bde559850436556ebee2e70e10f1cfc8aff636Krzysztof Kosińskivoid SPPaintSelector::getColorAlpha(SPColor &color, gfloat &alpha) const
d1bde559850436556ebee2e70e10f1cfc8aff636Krzysztof Kosiński{
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler SPColorSelector *csel;
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler csel = reinterpret_cast<SPColorSelector*>(g_object_get_data(G_OBJECT(selector), "color-selector"));
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler csel->base->getColorAlpha( color, alpha );
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler g_assert( ( 0.0 <= alpha )
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler && ( alpha <= 1.0 ) );
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler}
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler
f6df6236fcf18524e049bba12b69704af32eff90Josh AndlerSPGradient *SPPaintSelector::getGradientVector()
f6df6236fcf18524e049bba12b69704af32eff90Josh Andler{
SPGradient* vect = 0;
if (isPaintModeGradient(mode)) {
SPGradientSelector *gsel = getGradientFromData(this);
vect = gsel->getVector();
}
return vect;
}
void SPPaintSelector::pushAttrsToGradient( SPGradient *gr ) const
{
SPGradientUnits units = SP_GRADIENT_UNITS_OBJECTBOUNDINGBOX;
SPGradientSpread spread = SP_GRADIENT_SPREAD_PAD;
getGradientProperties( units, spread );
gr->setUnits(units);
gr->setSpread(spread);
gr->updateRepr();
}
static void
sp_paint_selector_clear_frame(SPPaintSelector *psel)
{
g_return_if_fail( psel != NULL);
if (psel->selector) {
/* before we destroy the frame contents, we must detach
* the patternmenu so that Gtk doesn't gtk_widget_destroy
* all the children of the menu. (We also have a g_object_ref
* count set on it too so that the gtk_container_remove doesn't
* end up destroying it.
*/
GtkWidget *patterns = (GtkWidget *)g_object_get_data(G_OBJECT(psel), "patternmenu");
if (patterns != NULL) {
GtkWidget * parent = gtk_widget_get_parent( GTK_WIDGET(patterns));
if ( parent != NULL ) {
g_assert( GTK_IS_CONTAINER(parent) );
gtk_container_remove( GTK_CONTAINER(parent), patterns );
}
}
gtk_widget_destroy(psel->selector);
psel->selector = NULL;
}
}
static void
sp_paint_selector_set_mode_empty(SPPaintSelector *psel)
{
sp_paint_selector_set_style_buttons(psel, NULL);
gtk_widget_set_sensitive(psel->style, FALSE);
sp_paint_selector_clear_frame(psel);
gtk_frame_set_label(GTK_FRAME(psel->frame), _("No objects"));
}
static void
sp_paint_selector_set_mode_multiple(SPPaintSelector *psel)
{
sp_paint_selector_set_style_buttons(psel, NULL);
gtk_widget_set_sensitive(psel->style, TRUE);
sp_paint_selector_clear_frame(psel);
gtk_frame_set_label(GTK_FRAME(psel->frame), _("Multiple styles"));
}
static void
sp_paint_selector_set_mode_unset(SPPaintSelector *psel)
{
sp_paint_selector_set_style_buttons(psel, psel->unset);
gtk_widget_set_sensitive(psel->style, TRUE);
sp_paint_selector_clear_frame(psel);
gtk_frame_set_label(GTK_FRAME(psel->frame), _("Paint is undefined"));
}
static void
sp_paint_selector_set_mode_none(SPPaintSelector *psel)
{
sp_paint_selector_set_style_buttons(psel, psel->none);
gtk_widget_set_sensitive(psel->style, TRUE);
sp_paint_selector_clear_frame(psel);
gtk_frame_set_label(GTK_FRAME(psel->frame), _("No paint"));
}
/* Color paint */
static void sp_paint_selector_color_grabbed(SPColorSelector * /*csel*/, SPPaintSelector *psel)
{
g_signal_emit(G_OBJECT(psel), psel_signals[GRABBED], 0);
}
static void sp_paint_selector_color_dragged(SPColorSelector * /*csel*/, SPPaintSelector *psel)
{
g_signal_emit(G_OBJECT(psel), psel_signals[DRAGGED], 0);
}
static void sp_paint_selector_color_released(SPColorSelector * /*csel*/, SPPaintSelector *psel)
{
g_signal_emit(G_OBJECT(psel), psel_signals[RELEASED], 0);
}
static void
sp_paint_selector_color_changed(SPColorSelector *csel, SPPaintSelector *psel)
{
csel->base->getColorAlpha( psel->color, psel->alpha );
g_signal_emit(G_OBJECT(psel), psel_signals[CHANGED], 0);
}
static void sp_paint_selector_set_mode_color(SPPaintSelector *psel, SPPaintSelector::Mode /*mode*/)
{
GtkWidget *csel;
sp_paint_selector_set_style_buttons(psel, psel->solid);
gtk_widget_set_sensitive(psel->style, TRUE);
if ((psel->mode == SPPaintSelector::MODE_COLOR_RGB) || (psel->mode == SPPaintSelector::MODE_COLOR_CMYK)) {
/* Already have color selector */
csel = (GtkWidget*)gtk_object_get_data(GTK_OBJECT(psel->selector), "color-selector");
} else {
sp_paint_selector_clear_frame(psel);
/* Create new color selector */
/* Create vbox */
GtkWidget *vb = gtk_vbox_new(FALSE, 4);
gtk_widget_show(vb);
/* Color selector */
csel = sp_color_selector_new( SP_TYPE_COLOR_NOTEBOOK );
gtk_widget_show(csel);
gtk_object_set_data(GTK_OBJECT(vb), "color-selector", csel);
gtk_box_pack_start(GTK_BOX(vb), csel, TRUE, TRUE, 0);
g_signal_connect(G_OBJECT(csel), "grabbed", G_CALLBACK(sp_paint_selector_color_grabbed), psel);
g_signal_connect(G_OBJECT(csel), "dragged", G_CALLBACK(sp_paint_selector_color_dragged), psel);
g_signal_connect(G_OBJECT(csel), "released", G_CALLBACK(sp_paint_selector_color_released), psel);
g_signal_connect(G_OBJECT(csel), "changed", G_CALLBACK(sp_paint_selector_color_changed), psel);
/* Pack everything to frame */
gtk_container_add(GTK_CONTAINER(psel->frame), vb);
psel->selector = vb;
/* Set color */
SP_COLOR_SELECTOR( csel )->base->setColorAlpha( psel->color, psel->alpha );
}
gtk_frame_set_label(GTK_FRAME(psel->frame), _("Flat color"));
#ifdef SP_PS_VERBOSE
g_print("Color req\n");
#endif
}
/* Gradient */
static void sp_paint_selector_gradient_grabbed(SPColorSelector * /*csel*/, SPPaintSelector *psel)
{
g_signal_emit(G_OBJECT(psel), psel_signals[GRABBED], 0);
}
static void sp_paint_selector_gradient_dragged(SPColorSelector * /*csel*/, SPPaintSelector *psel)
{
g_signal_emit(G_OBJECT(psel), psel_signals[DRAGGED], 0);
}
static void sp_paint_selector_gradient_released(SPColorSelector * /*csel*/, SPPaintSelector *psel)
{
g_signal_emit(G_OBJECT(psel), psel_signals[RELEASED], 0);
}
static void sp_paint_selector_gradient_changed(SPColorSelector * /*csel*/, SPPaintSelector *psel)
{
g_signal_emit(G_OBJECT(psel), psel_signals[CHANGED], 0);
}
static void sp_paint_selector_set_mode_gradient(SPPaintSelector *psel, SPPaintSelector::Mode mode)
{
GtkWidget *gsel;
/* fixme: We do not need function-wide gsel at all */
if (mode == SPPaintSelector::MODE_GRADIENT_LINEAR) {
sp_paint_selector_set_style_buttons(psel, psel->gradient);
} else {
sp_paint_selector_set_style_buttons(psel, psel->radial);
}
gtk_widget_set_sensitive(psel->style, TRUE);
if ((psel->mode == SPPaintSelector::MODE_GRADIENT_LINEAR) || (psel->mode == SPPaintSelector::MODE_GRADIENT_RADIAL)) {
/* Already have gradient selector */
gsel = (GtkWidget*)gtk_object_get_data(GTK_OBJECT(psel->selector), "gradient-selector");
} else {
sp_paint_selector_clear_frame(psel);
/* Create new gradient selector */
gsel = sp_gradient_selector_new();
gtk_widget_show(gsel);
g_signal_connect(G_OBJECT(gsel), "grabbed", G_CALLBACK(sp_paint_selector_gradient_grabbed), psel);
g_signal_connect(G_OBJECT(gsel), "dragged", G_CALLBACK(sp_paint_selector_gradient_dragged), psel);
g_signal_connect(G_OBJECT(gsel), "released", G_CALLBACK(sp_paint_selector_gradient_released), psel);
g_signal_connect(G_OBJECT(gsel), "changed", G_CALLBACK(sp_paint_selector_gradient_changed), psel);
/* Pack everything to frame */
gtk_container_add(GTK_CONTAINER(psel->frame), gsel);
psel->selector = gsel;
gtk_object_set_data(GTK_OBJECT(psel->selector), "gradient-selector", gsel);
}
/* Actually we have to set option menu history here */
if (mode == SPPaintSelector::MODE_GRADIENT_LINEAR) {
SP_GRADIENT_SELECTOR(gsel)->setMode(SPGradientSelector::MODE_LINEAR);
//sp_gradient_selector_set_mode(SP_GRADIENT_SELECTOR(gsel), SP_GRADIENT_SELECTOR_MODE_LINEAR);
gtk_frame_set_label(GTK_FRAME(psel->frame), _("Linear gradient"));
} else {
SP_GRADIENT_SELECTOR(gsel)->setMode(SPGradientSelector::MODE_RADIAL);
gtk_frame_set_label(GTK_FRAME(psel->frame), _("Radial gradient"));
}
#ifdef SP_PS_VERBOSE
g_print("Gradient req\n");
#endif
}
static void
sp_paint_selector_set_style_buttons(SPPaintSelector *psel, GtkWidget *active)
{
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(psel->none), (active == psel->none));
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(psel->solid), (active == psel->solid));
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(psel->gradient), (active == psel->gradient));
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(psel->radial), (active == psel->radial));
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(psel->pattern), (active == psel->pattern));
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(psel->swatch), (active == psel->swatch));
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(psel->unset), (active == psel->unset));
}
static void sp_psel_pattern_destroy(GtkWidget *widget, SPPaintSelector * /*psel*/)
{
// drop our reference to the pattern menu widget
g_object_unref( G_OBJECT(widget) );
}
static void sp_psel_pattern_change(GtkWidget * /*widget*/, SPPaintSelector *psel)
{
g_signal_emit(G_OBJECT(psel), psel_signals[CHANGED], 0);
}
/**
* Returns a list of patterns in the defs of the given source document as a GSList object
* Returns NULL if there are no patterns in the document.
*/
GSList *
ink_pattern_list_get (SPDocument *source)
{
if (source == NULL)
return NULL;
GSList *pl = NULL;
GSList const *patterns = source->getResourceList("pattern");
for (GSList *l = (GSList *) patterns; l != NULL; l = l->next) {
if (SP_PATTERN(l->data) == pattern_getroot(SP_PATTERN(l->data))) { // only if this is a root pattern
pl = g_slist_prepend(pl, l->data);
}
}
pl = g_slist_reverse(pl);
return pl;
}
/**
* Adds menu items for pattern list - derived from marker code, left hb etc in to make addition of previews easier at some point.
*/
static void
sp_pattern_menu_build (GtkWidget *m, GSList *pattern_list, SPDocument */*source*/)
{
for (; pattern_list != NULL; pattern_list = pattern_list->next) {
Inkscape::XML::Node *repr = reinterpret_cast<SPItem *>(pattern_list->data)->getRepr();
GtkWidget *i = gtk_menu_item_new();
gtk_widget_show(i);
if (repr->attribute("inkscape:stockid"))
g_object_set_data (G_OBJECT(i), "stockid", (void *) "true");
else
g_object_set_data (G_OBJECT(i), "stockid", (void *) "false");
gchar const *patid = repr->attribute("id");
g_object_set_data (G_OBJECT(i), "pattern", (void *) patid);
GtkWidget *hb = gtk_hbox_new(FALSE, 4);
gtk_widget_show(hb);
// create label
GtkWidget *l;
if (repr->attribute("inkscape:stockid"))
l = gtk_label_new(_(repr->attribute("inkscape:stockid")));
else
l = gtk_label_new(_(repr->attribute("id")));
gtk_widget_show(l);
gtk_misc_set_alignment(GTK_MISC(l), 0.0, 0.5);
gtk_box_pack_start(GTK_BOX(hb), l, TRUE, TRUE, 0);
gtk_widget_show(hb);
gtk_container_add(GTK_CONTAINER(i), hb);
gtk_menu_append(GTK_MENU(m), i);
}
}
/**
* sp_pattern_list_from_doc()
*
* \brief Pick up all patterns from source, except those that are in
* current_doc (if non-NULL), and add items to the pattern menu
*
*/
static void sp_pattern_list_from_doc (GtkWidget *m, SPDocument * /*current_doc*/, SPDocument *source, SPDocument * /*pattern_doc*/)
{
GSList *pl = ink_pattern_list_get(source);
GSList *clean_pl = NULL;
for (; pl != NULL; pl = pl->next) {
if (!SP_IS_PATTERN(pl->data)) {
continue;
}
// Add to the list of patterns we really do wish to show
clean_pl = g_slist_prepend (clean_pl, pl->data);
}
sp_pattern_menu_build (m, clean_pl, source);
g_slist_free (pl);
g_slist_free (clean_pl);
}
static void
ink_pattern_menu_populate_menu(GtkWidget *m, SPDocument *doc)
{
static SPDocument *patterns_doc = NULL;
// find and load patterns.svg
if (patterns_doc == NULL) {
char *patterns_source = g_build_filename(INKSCAPE_PATTERNSDIR, "patterns.svg", NULL);
if (Inkscape::IO::file_test(patterns_source, G_FILE_TEST_IS_REGULAR)) {
patterns_doc = SPDocument::createNewDoc(patterns_source, FALSE);
}
g_free(patterns_source);
}
// suck in from current doc
sp_pattern_list_from_doc ( m, NULL, doc, patterns_doc );
// add separator
{
GtkWidget *i = gtk_separator_menu_item_new();
gchar const *patid = "";
g_object_set_data (G_OBJECT(i), "pattern", (void *) patid);
gtk_widget_show(i);
gtk_menu_append(GTK_MENU(m), i);
}
// suck in from patterns.svg
if (patterns_doc) {
doc->ensureUpToDate();
sp_pattern_list_from_doc ( m, doc, patterns_doc, NULL );
}
}
static GtkWidget*
ink_pattern_menu(GtkWidget *mnu)
{
/* Create new menu widget */
GtkWidget *m = gtk_menu_new();
gtk_widget_show(m);
SPDocument *doc = SP_ACTIVE_DOCUMENT;
if (!doc) {
GtkWidget *i;
i = gtk_menu_item_new_with_label(_("No document selected"));
gtk_widget_show(i);
gtk_menu_append(GTK_MENU(m), i);
gtk_widget_set_sensitive(mnu, FALSE);
} else {
ink_pattern_menu_populate_menu(m, doc);
gtk_widget_set_sensitive(mnu, TRUE);
}
gtk_option_menu_set_menu(GTK_OPTION_MENU(mnu), m);
/* Set history */
gtk_option_menu_set_history(GTK_OPTION_MENU(mnu), 0);
return mnu;
}
/*update pattern list*/
void SPPaintSelector::updatePatternList( SPPattern *pattern )
{
if (update) {
return;
}
GtkWidget *mnu = GTK_WIDGET(g_object_get_data(G_OBJECT(this), "patternmenu"));
g_assert( mnu != NULL );
/* Clear existing menu if any */
gtk_option_menu_remove_menu(GTK_OPTION_MENU(mnu));
ink_pattern_menu(mnu);
/* Set history */
if (pattern && !gtk_object_get_data(GTK_OBJECT(mnu), "update")) {
gtk_object_set_data(GTK_OBJECT(mnu), "update", GINT_TO_POINTER(TRUE));
gchar const *patname = pattern->getRepr()->attribute("id");
GtkMenu *m = GTK_MENU(gtk_option_menu_get_menu(GTK_OPTION_MENU(mnu)));
GList *kids = GTK_MENU_SHELL(m)->children;
int patpos = 0;
int i = 0;
for (; kids != NULL; kids = kids->next) {
gchar *men_pat = (gchar *) g_object_get_data(G_OBJECT(kids->data), "pattern");
if ( strcmp(men_pat, patname) == 0 ) {
patpos = i;
}
i++;
}
gtk_option_menu_set_history(GTK_OPTION_MENU(mnu), patpos);
gtk_object_set_data(GTK_OBJECT(mnu), "update", GINT_TO_POINTER(FALSE));
}
//gtk_option_menu_set_history(GTK_OPTION_MENU(mnu), 0);
}
static void sp_paint_selector_set_mode_pattern(SPPaintSelector *psel, SPPaintSelector::Mode mode)
{
if (mode == SPPaintSelector::MODE_PATTERN) {
sp_paint_selector_set_style_buttons(psel, psel->pattern);
}
gtk_widget_set_sensitive(psel->style, TRUE);
GtkWidget *tbl = NULL;
if (psel->mode == SPPaintSelector::MODE_PATTERN) {
/* Already have pattern menu */
tbl = (GtkWidget*)gtk_object_get_data(GTK_OBJECT(psel->selector), "pattern-selector");
} else {
sp_paint_selector_clear_frame(psel);
/* Create vbox */
tbl = gtk_vbox_new(FALSE, 4);
gtk_widget_show(tbl);
{
GtkWidget *hb = gtk_hbox_new(FALSE, 1);
GtkWidget *mnu = gtk_option_menu_new();
ink_pattern_menu(mnu);
g_signal_connect(G_OBJECT(mnu), "changed", G_CALLBACK(sp_psel_pattern_change), psel);
g_signal_connect(G_OBJECT(mnu), "destroy", G_CALLBACK(sp_psel_pattern_destroy), psel);
gtk_object_set_data(GTK_OBJECT(psel), "patternmenu", mnu);
g_object_ref( G_OBJECT(mnu));
gtk_container_add(GTK_CONTAINER(hb), mnu);
gtk_box_pack_start(GTK_BOX(tbl), hb, FALSE, FALSE, AUX_BETWEEN_BUTTON_GROUPS);
}
{
GtkWidget *hb = gtk_hbox_new(FALSE, 0);
GtkWidget *l = gtk_label_new(NULL);
gtk_label_set_markup(GTK_LABEL(l), _("Use the <b>Node tool</b> to adjust position, scale, and rotation of the pattern on canvas. Use <b>Object &gt; Pattern &gt; Objects to Pattern</b> to create a new pattern from selection."));
gtk_label_set_line_wrap(GTK_LABEL(l), true);
gtk_widget_set_size_request(l, 180, -1);
gtk_box_pack_start(GTK_BOX(hb), l, TRUE, TRUE, AUX_BETWEEN_BUTTON_GROUPS);
gtk_box_pack_start(GTK_BOX(tbl), hb, FALSE, FALSE, AUX_BETWEEN_BUTTON_GROUPS);
}
gtk_widget_show_all(tbl);
gtk_container_add(GTK_CONTAINER(psel->frame), tbl);
psel->selector = tbl;
gtk_object_set_data(GTK_OBJECT(psel->selector), "pattern-selector", tbl);
gtk_frame_set_label(GTK_FRAME(psel->frame), _("Pattern fill"));
}
#ifdef SP_PS_VERBOSE
g_print("Pattern req\n");
#endif
}
SPPattern *SPPaintSelector::getPattern()
{
SPPattern *pat = 0;
g_return_val_if_fail((mode == MODE_PATTERN) , NULL);
GtkWidget *patmnu = (GtkWidget *) g_object_get_data(G_OBJECT(this), "patternmenu");
/* no pattern menu if we were just selected */
if ( patmnu == NULL ) return NULL;
GtkMenu *m = GTK_MENU(gtk_option_menu_get_menu(GTK_OPTION_MENU(patmnu)));
/* Get Pattern */
if (!g_object_get_data(G_OBJECT(gtk_menu_get_active(m)), "pattern"))
{
return NULL;
}
gchar *patid = (gchar *) g_object_get_data(G_OBJECT(gtk_menu_get_active(m)),
"pattern");
//gchar *pattern = "";
if (strcmp(patid, "none")){
gchar *stockid = (gchar *) g_object_get_data(G_OBJECT(gtk_menu_get_active(m)),
"stockid");
gchar *paturn = patid;
if (!strcmp(stockid,"true")) paturn = g_strconcat("urn:inkscape:pattern:",patid,NULL);
SPObject *pat_obj = get_stock_item(paturn);
if (pat_obj) {
pat = SP_PATTERN(pat_obj);
}
} else {
pat = pattern_getroot(SP_PATTERN(g_object_get_data(G_OBJECT(gtk_menu_get_active(m)), "pattern")));
}
if (pat && !SP_IS_PATTERN(pat)) {
pat = 0;
}
return pat;
}
static void sp_paint_selector_set_mode_swatch(SPPaintSelector *psel, SPPaintSelector::Mode mode)
{
if (mode == SPPaintSelector::MODE_SWATCH) {
sp_paint_selector_set_style_buttons(psel, psel->swatch);
}
gtk_widget_set_sensitive(psel->style, TRUE);
SwatchSelector *swatchsel = 0;
if (psel->mode == SPPaintSelector::MODE_SWATCH){
swatchsel = static_cast<SwatchSelector*>(g_object_get_data(G_OBJECT(psel->selector), "swatch-selector"));
} else {
sp_paint_selector_clear_frame(psel);
// Create new gradient selector
SwatchSelector *swatchsel = new SwatchSelector();
swatchsel->show();
swatchsel->connectGrabbedHandler( G_CALLBACK(sp_paint_selector_gradient_grabbed), psel );
swatchsel->connectDraggedHandler( G_CALLBACK(sp_paint_selector_gradient_dragged), psel );
swatchsel->connectReleasedHandler( G_CALLBACK(sp_paint_selector_gradient_released), psel );
swatchsel->connectchangedHandler( G_CALLBACK(sp_paint_selector_gradient_changed), psel );
// Pack everything to frame
gtk_container_add(GTK_CONTAINER(psel->frame), GTK_WIDGET(swatchsel->gobj()));
psel->selector = GTK_WIDGET(swatchsel->gobj());
gtk_object_set_data(GTK_OBJECT(psel->selector), "swatch-selector", swatchsel);
gtk_frame_set_label(GTK_FRAME(psel->frame), _("Swatch fill"));
}
#ifdef SP_PS_VERBOSE
g_print("Swatch req\n");
#endif
}
// TODO this seems very bad to be taking in a desktop pointer to muck with. Logic probably belongs elsewhere
void SPPaintSelector::setFlatColor( SPDesktop *desktop, gchar const *color_property, gchar const *opacity_property )
{
SPCSSAttr *css = sp_repr_css_attr_new();
SPColor color;
gfloat alpha = 0;
getColorAlpha( color, alpha );
std::string colorStr = color.toString();
#ifdef SP_PS_VERBOSE
guint32 rgba = color.toRGBA32( alpha );
g_message("sp_paint_selector_set_flat_color() to '%s' from 0x%08x::%s",
colorStr.c_str(),
rgba,
(color.icc ? color.icc->colorProfile.c_str():"<null>") );
#endif // SP_PS_VERBOSE
sp_repr_css_set_property(css, color_property, colorStr.c_str());
Inkscape::CSSOStringStream osalpha;
osalpha << alpha;
sp_repr_css_set_property(css, opacity_property, osalpha.str().c_str());
sp_desktop_set_style(desktop, css);
sp_repr_css_attr_unref(css);
}
SPPaintSelector::Mode SPPaintSelector::getModeForStyle(SPStyle const & style, FillOrStroke kind)
{
Mode mode = MODE_UNSET;
SPIPaint const & target = (kind == FILL) ? style.fill : style.stroke;
if ( !target.set ) {
mode = MODE_UNSET;
} else if ( target.isPaintserver() ) {
SPPaintServer const *server = (kind == FILL) ? style.getFillPaintServer() : style.getStrokePaintServer();
#ifdef SP_PS_VERBOSE
g_message("SPPaintSelector::getModeForStyle(%p, %d)", &style, kind);
g_message("==== server:%p %s grad:%s swatch:%s", server, server->getId(), (SP_IS_GRADIENT(server)?"Y":"n"), (SP_IS_GRADIENT(server) && SP_GRADIENT(server)->getVector()->isSwatch()?"Y":"n"));
#endif // SP_PS_VERBOSE
if (server && SP_IS_GRADIENT(server) && SP_GRADIENT(server)->getVector()->isSwatch()) {
mode = MODE_SWATCH;
} else if (SP_IS_LINEARGRADIENT(server)) {
mode = MODE_GRADIENT_LINEAR;
} else if (SP_IS_RADIALGRADIENT(server)) {
mode = MODE_GRADIENT_RADIAL;
} else if (SP_IS_PATTERN(server)) {
mode = MODE_PATTERN;
} else {
g_warning( "file %s: line %d: Unknown paintserver", __FILE__, __LINE__ );
mode = MODE_NONE;
}
} else if ( target.isColor() ) {
// TODO this is no longer a valid assertion:
mode = MODE_COLOR_RGB; // so far only rgb can be read from svg
} else if ( target.isNone() ) {
mode = MODE_NONE;
} else {
g_warning( "file %s: line %d: Unknown paint type", __FILE__, __LINE__ );
mode = MODE_NONE;
}
return mode;
}
/*
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:fileencoding=utf-8:textwidth=99 :