/* GtkPrinterOptionWidget * Copyright (C) 2006 Alexander Larsson * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; 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 #include "gtkintl.h" #include "gtkalignment.h" #include "gtkcheckbutton.h" #include "gtkcelllayout.h" #include "gtkcellrenderertext.h" #include "gtkcombobox.h" #include "gtkfilechooserbutton.h" #include "gtkimage.h" #include "gtklabel.h" #include "gtkliststore.h" #include "gtkradiobutton.h" #include "gtkstock.h" #include "gtktable.h" #include "gtktogglebutton.h" #include "gtkprivate.h" #include "gtkprinteroptionwidget.h" #include "gtkalias.h" #define GTK_PRINTER_OPTION_WIDGET_GET_PRIVATE(o) \ (G_TYPE_INSTANCE_GET_PRIVATE ((o), GTK_TYPE_PRINTER_OPTION_WIDGET, GtkPrinterOptionWidgetPrivate)) static void gtk_printer_option_widget_finalize (GObject *object); static void deconstruct_widgets (GtkPrinterOptionWidget *widget); static void construct_widgets (GtkPrinterOptionWidget *widget); static void update_widgets (GtkPrinterOptionWidget *widget); struct GtkPrinterOptionWidgetPrivate { GtkPrinterOption *source; gulong source_changed_handler; GtkWidget *check; GtkWidget *combo; GtkWidget *entry; GtkWidget *image; GtkWidget *label; GtkWidget *filechooser; GtkWidget *box; }; enum { CHANGED, LAST_SIGNAL }; enum { PROP_0, PROP_SOURCE, }; static guint signals[LAST_SIGNAL] = { 0 }; G_DEFINE_TYPE (GtkPrinterOptionWidget, gtk_printer_option_widget, GTK_TYPE_HBOX) static void gtk_printer_option_widget_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); static void gtk_printer_option_widget_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); static gboolean gtk_printer_option_widget_mnemonic_activate (GtkWidget *widget, gboolean group_cycling); static void gtk_printer_option_widget_class_init (GtkPrinterOptionWidgetClass *class) { GObjectClass *object_class; GtkWidgetClass *widget_class; object_class = (GObjectClass *) class; widget_class = (GtkWidgetClass *) class; object_class->finalize = gtk_printer_option_widget_finalize; object_class->set_property = gtk_printer_option_widget_set_property; object_class->get_property = gtk_printer_option_widget_get_property; widget_class->mnemonic_activate = gtk_printer_option_widget_mnemonic_activate; g_type_class_add_private (class, sizeof (GtkPrinterOptionWidgetPrivate)); signals[CHANGED] = g_signal_new ("changed", G_TYPE_FROM_CLASS (class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GtkPrinterOptionWidgetClass, changed), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); g_object_class_install_property (object_class, PROP_SOURCE, g_param_spec_object ("source", P_("Source option"), P_("The PrinterOption backing this widget"), GTK_TYPE_PRINTER_OPTION, GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT)); } static void gtk_printer_option_widget_init (GtkPrinterOptionWidget *widget) { widget->priv = GTK_PRINTER_OPTION_WIDGET_GET_PRIVATE (widget); gtk_box_set_spacing (GTK_BOX (widget), 12); } static void gtk_printer_option_widget_finalize (GObject *object) { GtkPrinterOptionWidget *widget = GTK_PRINTER_OPTION_WIDGET (object); GtkPrinterOptionWidgetPrivate *priv = widget->priv; if (priv->source) { g_object_unref (priv->source); priv->source = NULL; } G_OBJECT_CLASS (gtk_printer_option_widget_parent_class)->finalize (object); } static void gtk_printer_option_widget_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GtkPrinterOptionWidget *widget; widget = GTK_PRINTER_OPTION_WIDGET (object); switch (prop_id) { case PROP_SOURCE: gtk_printer_option_widget_set_source (widget, g_value_get_object (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gtk_printer_option_widget_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GtkPrinterOptionWidget *widget = GTK_PRINTER_OPTION_WIDGET (object); GtkPrinterOptionWidgetPrivate *priv = widget->priv; switch (prop_id) { case PROP_SOURCE: g_value_set_object (value, priv->source); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static gboolean gtk_printer_option_widget_mnemonic_activate (GtkWidget *widget, gboolean group_cycling) { GtkPrinterOptionWidget *powidget = GTK_PRINTER_OPTION_WIDGET (widget); GtkPrinterOptionWidgetPrivate *priv = powidget->priv; if (priv->check) return gtk_widget_mnemonic_activate (priv->check, group_cycling); if (priv->combo) return gtk_widget_mnemonic_activate (priv->combo, group_cycling); if (priv->entry) return gtk_widget_mnemonic_activate (priv->entry, group_cycling); return FALSE; } static void emit_changed (GtkPrinterOptionWidget *widget) { g_signal_emit (widget, signals[CHANGED], 0); } GtkWidget * gtk_printer_option_widget_new (GtkPrinterOption *source) { return g_object_new (GTK_TYPE_PRINTER_OPTION_WIDGET, "source", source, NULL); } static void source_changed_cb (GtkPrinterOption *source, GtkPrinterOptionWidget *widget) { update_widgets (widget); emit_changed (widget); } void gtk_printer_option_widget_set_source (GtkPrinterOptionWidget *widget, GtkPrinterOption *source) { GtkPrinterOptionWidgetPrivate *priv = widget->priv; if (source) g_object_ref (source); if (priv->source) { g_signal_handler_disconnect (priv->source, priv->source_changed_handler); g_object_unref (priv->source); } priv->source = source; if (source) priv->source_changed_handler = g_signal_connect (source, "changed", G_CALLBACK (source_changed_cb), widget); construct_widgets (widget); update_widgets (widget); g_object_notify (G_OBJECT (widget), "source"); } static GtkWidget * combo_box_new (void) { GtkWidget *combo_box; GtkCellRenderer *cell; GtkListStore *store; store = gtk_list_store_new (2, G_TYPE_STRING, G_TYPE_STRING); combo_box = gtk_combo_box_new_with_model (GTK_TREE_MODEL (store)); g_object_unref (store); cell = gtk_cell_renderer_text_new (); gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo_box), cell, TRUE); gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (combo_box), cell, "text", 0, NULL); return combo_box; } static void combo_box_append (GtkWidget *combo, const char *display_text, const char *value) { GtkTreeModel *model; GtkListStore *store; GtkTreeIter iter; model = gtk_combo_box_get_model (GTK_COMBO_BOX (combo)); store = GTK_LIST_STORE (model); gtk_list_store_append (store, &iter); gtk_list_store_set (store, &iter, 0, display_text, 1, value, -1); } struct ComboSet { GtkComboBox *combo; const char *value; }; static gboolean set_cb (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data) { struct ComboSet *set_data = data; gboolean found; char *value; gtk_tree_model_get (model, iter, 1, &value, -1); found = (strcmp (value, set_data->value) == 0); g_free (value); if (found) gtk_combo_box_set_active_iter (set_data->combo, iter); return found; } static void combo_box_set (GtkWidget *combo, const char *value) { GtkTreeModel *model; GtkListStore *store; struct ComboSet set_data; model = gtk_combo_box_get_model (GTK_COMBO_BOX (combo)); store = GTK_LIST_STORE (model); set_data.combo = GTK_COMBO_BOX (combo); set_data.value = value; gtk_tree_model_foreach (model, set_cb, &set_data); } static char * combo_box_get (GtkWidget *combo) { GtkTreeModel *model; char *val; GtkTreeIter iter; model = gtk_combo_box_get_model (GTK_COMBO_BOX (combo)); val = NULL; if (gtk_combo_box_get_active_iter (GTK_COMBO_BOX (combo), &iter)) gtk_tree_model_get (model, &iter, 1, &val, -1); return val; } static void deconstruct_widgets (GtkPrinterOptionWidget *widget) { GtkPrinterOptionWidgetPrivate *priv = widget->priv; if (priv->check) { gtk_widget_destroy (priv->check); priv->check = NULL; } if (priv->combo) { gtk_widget_destroy (priv->combo); priv->combo = NULL; } if (priv->entry) { gtk_widget_destroy (priv->entry); priv->entry = NULL; } /* make sure entry and combo are destroyed first */ /* as we use the two of them to create the filechooser */ if (priv->filechooser) { gtk_widget_destroy (priv->filechooser); priv->filechooser = NULL; } if (priv->image) { gtk_widget_destroy (priv->image); priv->image = NULL; } if (priv->label) { gtk_widget_destroy (priv->label); priv->label = NULL; } } static void check_toggled_cb (GtkToggleButton *toggle_button, GtkPrinterOptionWidget *widget) { GtkPrinterOptionWidgetPrivate *priv = widget->priv; g_signal_handler_block (priv->source, priv->source_changed_handler); gtk_printer_option_set_boolean (priv->source, gtk_toggle_button_get_active (toggle_button)); g_signal_handler_unblock (priv->source, priv->source_changed_handler); emit_changed (widget); } static void filesave_changed_cb (GtkWidget *w, GtkPrinterOptionWidget *widget) { GtkPrinterOptionWidgetPrivate *priv = widget->priv; char *value; char *directory; const char *file; /* combine the value of the chooser with the value of the entry */ g_signal_handler_block (priv->source, priv->source_changed_handler); /* TODO: how do we support nonlocal file systems? */ directory = gtk_file_chooser_get_current_folder (GTK_FILE_CHOOSER (priv->combo)); file = gtk_entry_get_text (GTK_ENTRY (priv->entry)); if (g_path_is_absolute (file)) value = g_strdup (file); #ifdef G_OS_UNIX else if (file[0] == '~' && file[1] == '/') value = g_build_filename (g_get_home_dir (), file + 2, NULL); #endif else value = g_build_filename (directory, file, NULL); if (value) gtk_printer_option_set (priv->source, value); g_free (directory); g_free (value); g_signal_handler_unblock (priv->source, priv->source_changed_handler); emit_changed (widget); } static void combo_changed_cb (GtkWidget *combo, GtkPrinterOptionWidget *widget) { GtkPrinterOptionWidgetPrivate *priv = widget->priv; char *value; g_signal_handler_block (priv->source, priv->source_changed_handler); value = combo_box_get (combo); if (value) gtk_printer_option_set (priv->source, value); g_free (value); g_signal_handler_unblock (priv->source, priv->source_changed_handler); emit_changed (widget); } static void entry_changed_cb (GtkWidget *entry, GtkPrinterOptionWidget *widget) { GtkPrinterOptionWidgetPrivate *priv = widget->priv; const char *value; g_signal_handler_block (priv->source, priv->source_changed_handler); value = gtk_entry_get_text (GTK_ENTRY (entry)); if (value) gtk_printer_option_set (priv->source, value); g_signal_handler_unblock (priv->source, priv->source_changed_handler); emit_changed (widget); } static void radio_changed_cb (GtkWidget *button, GtkPrinterOptionWidget *widget) { GtkPrinterOptionWidgetPrivate *priv = widget->priv; char *value; g_signal_handler_block (priv->source, priv->source_changed_handler); value = g_object_get_data (G_OBJECT (button), "value"); if (value) gtk_printer_option_set (priv->source, value); g_free (value); g_signal_handler_unblock (priv->source, priv->source_changed_handler); emit_changed (widget); } static void alternative_set (GtkWidget *box, const gchar *value) { GList *children, *l; children = gtk_container_get_children (GTK_CONTAINER (box)); for (l = children; l != NULL; l = l->next) { char *v = g_object_get_data (G_OBJECT (l->data), "value"); if (strcmp (value, v) == 0) gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (l->data), TRUE); } g_list_free (children); } static GSList * alternative_append (GtkWidget *box, const gchar *label, const gchar *value, GtkPrinterOptionWidget *widget, GSList *group) { GtkWidget *button; button = gtk_radio_button_new_with_label (group, label); gtk_widget_show (button); gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0); g_object_set_data (G_OBJECT (button), "value", (gpointer)value); g_signal_connect (button, "toggled", G_CALLBACK (radio_changed_cb), widget); return gtk_radio_button_get_group (GTK_RADIO_BUTTON (button)); } static void construct_widgets (GtkPrinterOptionWidget *widget) { GtkPrinterOptionWidgetPrivate *priv = widget->priv; GtkPrinterOption *source; char *text; int i; GSList *group; source = priv->source; deconstruct_widgets (widget); if (source == NULL) { priv->combo = combo_box_new (); combo_box_append (priv->combo,_("Not available"), "None"); gtk_combo_box_set_active (GTK_COMBO_BOX (priv->combo), 0); gtk_widget_set_sensitive (priv->combo, FALSE); gtk_widget_show (priv->combo); gtk_box_pack_start (GTK_BOX (widget), priv->combo, TRUE, TRUE, 0); } else switch (source->type) { case GTK_PRINTER_OPTION_TYPE_BOOLEAN: priv->check = gtk_check_button_new_with_mnemonic (source->display_text); g_signal_connect (priv->check, "toggled", G_CALLBACK (check_toggled_cb), widget); gtk_widget_show (priv->check); gtk_box_pack_start (GTK_BOX (widget), priv->check, TRUE, TRUE, 0); break; case GTK_PRINTER_OPTION_TYPE_PICKONE: priv->combo = combo_box_new (); for (i = 0; i < source->num_choices; i++) combo_box_append (priv->combo, source->choices_display[i], source->choices[i]); gtk_widget_show (priv->combo); gtk_box_pack_start (GTK_BOX (widget), priv->combo, TRUE, TRUE, 0); g_signal_connect (priv->combo, "changed", G_CALLBACK (combo_changed_cb), widget); text = g_strdup_printf ("%s:", source->display_text); priv->label = gtk_label_new_with_mnemonic (text); g_free (text); gtk_widget_show (priv->label); break; case GTK_PRINTER_OPTION_TYPE_ALTERNATIVE: group = NULL; priv->box = gtk_hbox_new (FALSE, 12); gtk_widget_show (priv->box); gtk_box_pack_start (GTK_BOX (widget), priv->box, TRUE, TRUE, 0); for (i = 0; i < source->num_choices; i++) group = alternative_append (priv->box, source->choices_display[i], source->choices[i], widget, group); if (source->display_text) { text = g_strdup_printf ("%s:", source->display_text); priv->label = gtk_label_new_with_mnemonic (text); g_free (text); gtk_widget_show (priv->label); } break; case GTK_PRINTER_OPTION_TYPE_STRING: priv->entry = gtk_entry_new (); gtk_widget_show (priv->entry); gtk_box_pack_start (GTK_BOX (widget), priv->entry, TRUE, TRUE, 0); g_signal_connect (priv->entry, "changed", G_CALLBACK (entry_changed_cb), widget); text = g_strdup_printf ("%s:", source->display_text); priv->label = gtk_label_new_with_mnemonic (text); g_free (text); gtk_widget_show (priv->label); break; case GTK_PRINTER_OPTION_TYPE_FILESAVE: { GtkWidget *label; priv->filechooser = gtk_table_new (2, 2, FALSE); gtk_table_set_row_spacings (GTK_TABLE (priv->filechooser), 6); gtk_table_set_col_spacings (GTK_TABLE (priv->filechooser), 12); /* TODO: make this a gtkfilechooserentry once we move to GTK */ priv->entry = gtk_entry_new (); priv->combo = gtk_file_chooser_button_new (_("Print to PDF"), GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER); label = gtk_label_new_with_mnemonic (_("_Name:")); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_label_set_mnemonic_widget (GTK_LABEL (label), priv->entry); gtk_table_attach (GTK_TABLE (priv->filechooser), label, 0, 1, 0, 1, GTK_FILL, 0, 0, 0); gtk_table_attach (GTK_TABLE (priv->filechooser), priv->entry, 1, 2, 0, 1, GTK_FILL, 0, 0, 0); label = gtk_label_new_with_mnemonic (_("_Save in folder:")); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_label_set_mnemonic_widget (GTK_LABEL (label), priv->combo); gtk_table_attach (GTK_TABLE (priv->filechooser), label, 0, 1, 1, 2, GTK_FILL, 0, 0, 0); gtk_table_attach (GTK_TABLE (priv->filechooser), priv->combo, 1, 2, 1, 2, GTK_FILL, 0, 0, 0); gtk_widget_show_all (priv->filechooser); gtk_box_pack_start (GTK_BOX (widget), priv->filechooser, TRUE, TRUE, 0); g_signal_connect (priv->entry, "changed", G_CALLBACK (filesave_changed_cb), widget); g_signal_connect (priv->combo, "current-folder-changed", G_CALLBACK (filesave_changed_cb), widget); } break; default: break; } priv->image = gtk_image_new_from_stock (GTK_STOCK_DIALOG_WARNING, GTK_ICON_SIZE_MENU); gtk_box_pack_start (GTK_BOX (widget), priv->image, FALSE, FALSE, 0); } static void update_widgets (GtkPrinterOptionWidget *widget) { GtkPrinterOptionWidgetPrivate *priv = widget->priv; GtkPrinterOption *source; source = priv->source; if (source == NULL) { gtk_widget_hide (priv->image); return; } switch (source->type) { case GTK_PRINTER_OPTION_TYPE_BOOLEAN: if (strcmp (source->value, "True") == 0) gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (priv->check), TRUE); else gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (priv->check), FALSE); break; case GTK_PRINTER_OPTION_TYPE_PICKONE: combo_box_set (priv->combo, source->value); break; case GTK_PRINTER_OPTION_TYPE_ALTERNATIVE: alternative_set (priv->box, source->value); break; case GTK_PRINTER_OPTION_TYPE_STRING: gtk_entry_set_text (GTK_ENTRY (priv->entry), source->value); break; case GTK_PRINTER_OPTION_TYPE_FILESAVE: { char *basename = g_path_get_basename (source->value); char *dirname = g_path_get_dirname (source->value); gtk_entry_set_text (GTK_ENTRY (priv->entry), basename); if (g_path_is_absolute (dirname)) gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (priv->combo), dirname); g_free (basename); g_free (dirname); break; } default: break; } if (source->has_conflict) gtk_widget_show (priv->image); else gtk_widget_hide (priv->image); } gboolean gtk_printer_option_widget_has_external_label (GtkPrinterOptionWidget *widget) { return widget->priv->label != NULL; } GtkWidget * gtk_printer_option_widget_get_external_label (GtkPrinterOptionWidget *widget) { return widget->priv->label; } const char * gtk_printer_option_widget_get_value (GtkPrinterOptionWidget *widget) { GtkPrinterOptionWidgetPrivate *priv = widget->priv; if (priv->source) return priv->source->value; return ""; } #define __GTK_PRINTER_OPTION_WIDGET_C__ #include "gtkaliasdef.c"