/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*- * * Copyright 2009-2010 Red Hat, Inc, * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * * Written by: Matthias Clasen */ #include #include "um-editable-combo.h" #define EMPTY_TEXT "\xe2\x80\x94" struct _UmEditableComboPrivate { GtkNotebook *notebook; GtkLabel *label; GtkButton *button; GtkComboBox *combo; GtkWidget *toplevel; gint active; gint editable; gint text_column; }; #define UM_EDITABLE_COMBO_GET_PRIVATE(obj) (G_TYPE_INSTANCE_GET_PRIVATE ((obj), UM_TYPE_EDITABLE_COMBO, UmEditableComboPrivate)) enum { PROP_0, PROP_EDITABLE, PROP_MODEL, PROP_TEXT_COLUMN }; enum { EDITING_DONE, ACTIVATE, LAST_SIGNAL }; static guint signals [LAST_SIGNAL] = { 0, }; G_DEFINE_TYPE (UmEditableCombo, um_editable_combo, GTK_TYPE_ALIGNMENT); void um_editable_combo_set_editable (UmEditableCombo *combo, gboolean editable) { UmEditableComboPrivate *priv; priv = combo->priv; if (priv->editable != editable) { priv->editable = editable; gtk_notebook_set_current_page (priv->notebook, editable ? 1 : 0); g_object_notify (G_OBJECT (combo), "editable"); } } gboolean um_editable_combo_get_editable (UmEditableCombo *combo) { return combo->priv->editable; } void um_editable_combo_set_model (UmEditableCombo *combo, GtkTreeModel *model) { gtk_combo_box_set_model (combo->priv->combo, model); g_object_notify (G_OBJECT (combo), "model"); } GtkTreeModel * um_editable_combo_get_model (UmEditableCombo *combo) { return gtk_combo_box_get_model (combo->priv->combo); } void um_editable_combo_set_text_column (UmEditableCombo *combo, gint text_column) { UmEditableComboPrivate *priv = combo->priv; GList *cells; if (priv->text_column == text_column) return; priv->text_column = text_column; cells = gtk_cell_layout_get_cells (GTK_CELL_LAYOUT (priv->combo)); gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (priv->combo), cells->data, "text", text_column, NULL); g_list_free (cells); g_object_notify (G_OBJECT (combo), "text-column"); } gint um_editable_combo_get_text_column (UmEditableCombo *combo) { return combo->priv->text_column; } void um_editable_combo_set_active (UmEditableCombo *combo, gint active) { GtkTreeModel *model; GtkTreePath *path; GtkTreeIter iter; if (active == -1) um_editable_combo_set_active_iter (combo, NULL); else { model = gtk_combo_box_get_model (combo->priv->combo); path = gtk_tree_path_new_from_indices (active, -1); gtk_tree_model_get_iter (model, &iter, path); gtk_tree_path_free (path); um_editable_combo_set_active_iter (combo, &iter); } } void um_editable_combo_set_active_iter (UmEditableCombo *combo, GtkTreeIter *iter) { UmEditableComboPrivate *priv = combo->priv; GtkWidget *label; gchar *text; GtkTreeModel *model; gtk_combo_box_set_active_iter (priv->combo, iter); priv->active = gtk_combo_box_get_active (priv->combo); if (priv->text_column == -1) return; if (iter) { model = gtk_combo_box_get_model (priv->combo); gtk_tree_model_get (model, iter, priv->text_column, &text, -1); } else { text = g_strdup (EMPTY_TEXT); } gtk_label_set_text (priv->label, text); label = gtk_bin_get_child ((GtkBin*)priv->button); gtk_label_set_text (GTK_LABEL (label), text); g_free (text); } gboolean um_editable_combo_get_active_iter (UmEditableCombo *combo, GtkTreeIter *iter) { return gtk_combo_box_get_active_iter (combo->priv->combo, iter); } gint um_editable_combo_get_active (UmEditableCombo *combo) { return combo->priv->active; } static void um_editable_combo_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { UmEditableCombo *combo = UM_EDITABLE_COMBO (object); switch (prop_id) { case PROP_EDITABLE: um_editable_combo_set_editable (combo, g_value_get_boolean (value)); break; case PROP_MODEL: um_editable_combo_set_model (combo, g_value_get_object (value)); break; case PROP_TEXT_COLUMN: um_editable_combo_set_text_column (combo, g_value_get_int (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void um_editable_combo_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { UmEditableCombo *combo = UM_EDITABLE_COMBO (object); switch (prop_id) { case PROP_EDITABLE: g_value_set_boolean (value, um_editable_combo_get_editable (combo)); break; case PROP_MODEL: g_value_set_object (value, um_editable_combo_get_model (combo)); break; case PROP_TEXT_COLUMN: g_value_set_int (value, um_editable_combo_get_text_column (combo)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void um_editable_combo_activate (UmEditableCombo *combo); static void um_editable_combo_class_init (UmEditableComboClass *class) { GObjectClass *object_class; GtkWidgetClass *widget_class; object_class = G_OBJECT_CLASS (class); widget_class = GTK_WIDGET_CLASS (class); object_class->set_property = um_editable_combo_set_property; object_class->get_property = um_editable_combo_get_property; signals[EDITING_DONE] = g_signal_new ("editing-done", G_TYPE_FROM_CLASS (class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (UmEditableComboClass, editing_done), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); signals[ACTIVATE] = g_signal_new ("activate", G_TYPE_FROM_CLASS (class), G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION, G_STRUCT_OFFSET (UmEditableComboClass, activate), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); widget_class->activate_signal = signals[ACTIVATE]; class->activate = um_editable_combo_activate; g_object_class_install_property (object_class, PROP_MODEL, g_param_spec_object ("model", "Model", "The options to present in the combobox", GTK_TYPE_TREE_MODEL, G_PARAM_READWRITE)); g_object_class_install_property (object_class, PROP_TEXT_COLUMN, g_param_spec_int ("text-column", "Text Column", "The model column that contains the displayable text", -1, G_MAXINT, -1, G_PARAM_READWRITE)); g_object_class_install_property (object_class, PROP_EDITABLE, g_param_spec_boolean ("editable", "Editable", "Whether the text can be edited", FALSE, G_PARAM_READWRITE)); g_type_class_add_private (class, sizeof (UmEditableComboPrivate)); } static void start_editing (UmEditableCombo *combo) { gtk_notebook_set_current_page (combo->priv->notebook, 2); gtk_combo_box_popup (combo->priv->combo); } static void stop_editing (UmEditableCombo *combo) { um_editable_combo_set_active (combo, gtk_combo_box_get_active (combo->priv->combo)); gtk_notebook_set_current_page (combo->priv->notebook, 1); g_signal_emit (combo, signals[EDITING_DONE], 0); } static void cancel_editing (UmEditableCombo *combo) { gtk_combo_box_set_active (combo->priv->combo, um_editable_combo_get_active (combo)); gtk_notebook_set_current_page (combo->priv->notebook, 1); } static void um_editable_combo_activate (UmEditableCombo *combo) { if (combo->priv->editable) { gtk_notebook_set_current_page (combo->priv->notebook, 2); gtk_widget_grab_focus (GTK_WIDGET (combo->priv->combo)); } } static void button_clicked (GtkWidget *widget, UmEditableCombo *combo) { if (combo->priv->editable) start_editing (combo); } static void combo_changed (GtkWidget *widget, UmEditableCombo *combo) { if (combo->priv->editable) stop_editing (combo); } static gboolean combo_key_press (GtkWidget *widget, GdkEventKey *event, UmEditableCombo *combo) { if (event->keyval == GDK_KEY_Escape) { cancel_editing (combo); return TRUE; } return FALSE; } static void focus_moved (GtkWindow *window, GtkWidget *widget, UmEditableCombo *combo) { if (gtk_notebook_get_current_page (combo->priv->notebook) == 2 && (!widget || !gtk_widget_is_ancestor (widget, (GtkWidget *)combo))) stop_editing (combo); } static void combo_hierarchy_changed (GtkWidget *widget, GtkWidget *previous_toplevel, UmEditableCombo *combo) { UmEditableComboPrivate *priv; GtkWidget *toplevel; priv = combo->priv; toplevel = gtk_widget_get_toplevel (widget); if (priv->toplevel != toplevel) { if (priv->toplevel) g_signal_handlers_disconnect_by_func (priv->toplevel, focus_moved, combo); if (GTK_IS_WINDOW (toplevel)) priv->toplevel = toplevel; else priv->toplevel = NULL; if (priv->toplevel) g_signal_connect (priv->toplevel, "set-focus", G_CALLBACK (focus_moved), combo); } } static void update_button_padding (GtkWidget *widget, GtkAllocation *allocation, UmEditableCombo *combo) { UmEditableComboPrivate *priv = combo->priv; GtkAllocation parent_allocation; gint offset; gint pad; gtk_widget_get_allocation (gtk_widget_get_parent (widget), &parent_allocation); offset = allocation->x - parent_allocation.x; gtk_misc_get_padding (GTK_MISC (priv->label), &pad, NULL); if (offset != pad) gtk_misc_set_padding (GTK_MISC (priv->label), offset, 0); } static void um_editable_combo_init (UmEditableCombo *combo) { UmEditableComboPrivate *priv; GtkCellRenderer *cell; priv = combo->priv = UM_EDITABLE_COMBO_GET_PRIVATE (combo); priv->active = -1; priv->text_column = -1; priv->notebook = (GtkNotebook*)gtk_notebook_new (); gtk_notebook_set_show_tabs (priv->notebook, FALSE); gtk_notebook_set_show_border (priv->notebook, FALSE); priv->label = (GtkLabel*)gtk_label_new (""); gtk_misc_set_alignment (GTK_MISC (priv->label), 0.0, 0.5); gtk_notebook_append_page (priv->notebook, (GtkWidget*)priv->label, NULL); priv->button = (GtkButton*)gtk_button_new_with_label (""); gtk_widget_set_receives_default ((GtkWidget*)priv->button, TRUE); gtk_button_set_relief (priv->button, GTK_RELIEF_NONE); gtk_button_set_alignment (priv->button, 0.0, 0.5); gtk_notebook_append_page (priv->notebook, (GtkWidget*)priv->button, NULL); g_signal_connect (priv->button, "clicked", G_CALLBACK (button_clicked), combo); priv->combo = (GtkComboBox*)gtk_combo_box_new (); cell = gtk_cell_renderer_text_new (); gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (priv->combo), cell, TRUE); gtk_notebook_append_page (priv->notebook, (GtkWidget*)priv->combo, NULL); g_signal_connect (priv->combo, "changed", G_CALLBACK (combo_changed), combo); g_signal_connect (priv->combo, "key-press-event", G_CALLBACK (combo_key_press), combo); g_signal_connect (gtk_bin_get_child (GTK_BIN (priv->button)), "size-allocate", G_CALLBACK (update_button_padding), combo); gtk_container_add (GTK_CONTAINER (combo), (GtkWidget*)priv->notebook); gtk_widget_show ((GtkWidget*)priv->notebook); gtk_widget_show ((GtkWidget*)priv->label); gtk_widget_show ((GtkWidget*)priv->button); gtk_widget_show ((GtkWidget*)priv->combo); gtk_notebook_set_current_page (priv->notebook, 0); /* ugly hack to catch the combo box losing focus */ g_signal_connect (combo, "hierarchy-changed", G_CALLBACK (combo_hierarchy_changed), combo); } GtkWidget * um_editable_combo_new (void) { return (GtkWidget *) g_object_new (UM_TYPE_EDITABLE_COMBO, NULL); }