/* Evolution internal utilities - Glade dialog widget utilities * * Copyright (C) 2000 Ximian, Inc. * Copyright (C) 2000 Ximian, Inc. * * Author: Federico Mena-Quintero <federico@ximian.com> * * This program is free software; you can redistribute it and/or * modify it under the terms of version 2 of the GNU General Public * License as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public * License along with this program; if not, write to the * Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ #include <config.h> #include <math.h> #include <string.h> #include <time.h> #include <gtk/gtkmenu.h> #include <gtk/gtkmenuitem.h> #include <gtk/gtkoptionmenu.h> #include <gtk/gtkradiobutton.h> #include <gtk/gtksignal.h> #include <gtk/gtkspinbutton.h> #include <libgnomeui/gnome-dateedit.h> #include <libgnomeui/gnome-propertybox.h> #include "e-dialog-widgets.h" /* A widget, a pointer to the variable it will modify, and extra information */ typedef struct { GtkWidget *widget; gpointer value_var; gpointer info; } WidgetHook; /* Hook information for a complete dialog */ typedef struct { GSList *whooks; } DialogHooks; /* Destroy handler for the dialog; frees the dialog hooks */ static void dialog_destroy_cb (DialogHooks *hooks, GObject *dialog) { g_slist_free (hooks->whooks); hooks->whooks = NULL; g_free (hooks); g_object_set_data (dialog, "dialog-hooks", NULL); } /* Ensures that the dialog has the necessary attached data to store the widget * hook information. */ static DialogHooks * get_dialog_hooks (GtkWidget *dialog) { DialogHooks *hooks; hooks = g_object_get_data ((GObject *) dialog, "dialog-hooks"); if (!hooks) { hooks = g_new0 (DialogHooks, 1); g_object_set_data ((GObject *) dialog, "dialog-hooks", hooks); g_object_weak_ref ((GObject *) dialog, (GWeakNotify) dialog_destroy_cb, hooks); } return hooks; } /* Converts an mapped value to the appropriate index in an item group. The * values for the items are provided as a -1-terminated array. */ static int value_to_index (const int *value_map, int value) { int i; for (i = 0; value_map[i] != -1; i++) if (value_map[i] == value) return i; return -1; } /* Converts an index in an item group to the appropriate mapped value. See the * function above. */ static int index_to_value (const int *value_map, int index) { int i; /* We do this the hard way, i.e. not as a simple array reference, to * check for correctness. */ for (i = 0; value_map[i] != -1; i++) if (i == index) return value_map[i]; return -1; } /* Callback for the "toggled" signal of toggle buttons */ static void toggled_cb (GtkToggleButton *toggle, gpointer data) { GnomePropertyBox *pbox; pbox = GNOME_PROPERTY_BOX (data); /* For radio buttons, we only notify the property box if the button is * active, because we'll get one call for each of the changed buttons in * the radio group. */ if (!GTK_IS_RADIO_BUTTON (toggle) || toggle->active) gnome_property_box_changed (pbox); } /* Hooks a radio button group */ static void hook_radio (GtkWidget *dialog, GtkRadioButton *radio, gpointer value_var, gpointer info) { const int *value_map; GSList *group, *l; int *value; group = gtk_radio_button_get_group (radio); /* Set the value */ value = (int *) value_var; value_map = (const int *) info; e_dialog_radio_set (GTK_WIDGET (radio), *value, value_map); /* Hook to changed */ if (GNOME_IS_PROPERTY_BOX (dialog)) for (l = group; l; l = l->next) g_signal_connect (l->data, "toggled", G_CALLBACK (toggled_cb), dialog); } /* Gets the value of a radio button group */ static void get_radio_value (GtkRadioButton *radio, gpointer value_var, gpointer info) { int *value; const int *value_map; value = (int *) value_var; value_map = (const int *) info; *value = e_dialog_radio_get (GTK_WIDGET (radio), value_map); } /* Callback for the "activate" signal of menu items */ static void activate_cb (GtkMenuItem *item, gpointer data) { GnomePropertyBox *pbox; pbox = GNOME_PROPERTY_BOX (data); gnome_property_box_changed (pbox); } /* Hooks an option menu */ static void hook_option_menu (GtkWidget *dialog, GtkOptionMenu *omenu, gpointer value_var, gpointer info) { const int *value_map; int *value; /* Set the value */ value = (int *) value_var; value_map = (const int *) info; e_dialog_option_menu_set (GTK_WIDGET (omenu), *value, value_map); /* Hook to changed */ if (GNOME_IS_PROPERTY_BOX (dialog)) { GtkMenu *menu; GList *l; menu = GTK_MENU (gtk_option_menu_get_menu (omenu)); for (l = GTK_MENU_SHELL (menu)->children; l; l = l->next) g_signal_connect (l->data, "activate", G_CALLBACK (activate_cb), dialog); } } /* Gets the value of an option menu */ static void get_option_menu_value (GtkOptionMenu *omenu, gpointer value_var, gpointer info) { const int *value_map; int *value; value = (int *) value_var; value_map = (const int *) info; *value = e_dialog_option_menu_get (GTK_WIDGET (omenu), value_map); } /* Hooks a toggle button */ static void hook_toggle (GtkWidget *dialog, GtkToggleButton *toggle, gpointer value_var, gpointer info) { gboolean *value; /* Set the value */ value = (gboolean *) value_var; e_dialog_toggle_set (GTK_WIDGET (toggle), *value); /* Hook to changed */ if (GNOME_IS_PROPERTY_BOX (dialog)) g_signal_connect (toggle, "toggled", G_CALLBACK (toggled_cb), dialog); } /* Gets the value of a toggle button */ static void get_toggle_value (GtkToggleButton *toggle, gpointer value_var, gpointer info) { gboolean *value; value = (gboolean *) value_var; *value = e_dialog_toggle_get (GTK_WIDGET (toggle)); } /* Callback for the "value_changed" signal of the adjustment of a spin button */ static void value_changed_cb (GtkAdjustment *adj, gpointer data) { GnomePropertyBox *pbox; pbox = GNOME_PROPERTY_BOX (data); gnome_property_box_changed (pbox); } /* Hooks a spin button */ static void hook_spin_button (GtkWidget *dialog, GtkSpinButton *spin, gpointer value_var, gpointer info) { double *value; GtkAdjustment *adj; /* Set the value */ value = (double *) value_var; e_dialog_spin_set (GTK_WIDGET (spin), *value); /* Hook to changed */ adj = gtk_spin_button_get_adjustment (spin); if (GNOME_IS_PROPERTY_BOX (dialog)) g_signal_connect (adj, "value_changed", G_CALLBACK (value_changed_cb), dialog); } /* Gets the value of a spin button */ static void get_spin_button_value (GtkSpinButton *spin, gpointer value_var, gpointer info) { double *value; value = (double *) value_var; *value = e_dialog_spin_get_double (GTK_WIDGET (spin)); } /* Callback for the "changed" signal of a GtkEditable widget */ static void changed_cb (GtkEditable *editable, gpointer data) { GnomePropertyBox *pbox; pbox = GNOME_PROPERTY_BOX (data); gnome_property_box_changed (pbox); } /* Hooks a GtkEditable widget */ static void hook_editable (GtkWidget *dialog, GtkEditable *editable, gpointer value_var, gpointer info) { char **value; /* Set the value */ value = (char **) value_var; e_dialog_editable_set (GTK_WIDGET (editable), *value); /* Hook to changed */ if (GNOME_IS_PROPERTY_BOX (dialog)) g_signal_connect (editable, "changed", G_CALLBACK (changed_cb), dialog); } /* Gets the value of a GtkEditable widget */ static void get_editable_value (GtkEditable *editable, gpointer value_var, gpointer data) { char **value; value = (char **) value_var; if (*value) g_free (*value); *value = e_dialog_editable_get (GTK_WIDGET (editable)); } /** * e_dialog_editable_set: * @widget: A #GtkEditable widget. * @value: String value. * * Sets the string value inside a #GtkEditable-derived widget. **/ void e_dialog_editable_set (GtkWidget *widget, const char *value) { int pos = 0; g_return_if_fail (widget != NULL); g_return_if_fail (GTK_IS_EDITABLE (widget)); gtk_editable_delete_text (GTK_EDITABLE (widget), 0, -1); if (value) gtk_editable_insert_text (GTK_EDITABLE (widget), value, strlen (value), &pos); } /** * e_dialog_editable_get: * @widget: A #GtkEditable widget. * * Queries the string value inside a #GtkEditable-derived widget. * * Return value: String value. You should free it when you are done with it. * This function can return NULL if the string could not be converted from * GTK+'s encoding into UTF8. **/ char * e_dialog_editable_get (GtkWidget *widget) { g_return_val_if_fail (widget != NULL, NULL); g_return_val_if_fail (GTK_IS_EDITABLE (widget), NULL); return gtk_editable_get_chars (GTK_EDITABLE (widget), 0, -1); } /** * e_dialog_radio_set: * @widget: A #GtkRadioButton in a radio button group. * @value: Enumerated value. * @value_map: Map from enumeration values to array indices. * * Sets the selected item in a radio group. The specified @widget can be any of * the #GtkRadioButtons in the group. Each radio button should correspond to an * enumeration value; the specified @value will be mapped to an integer from * zero to the number of items in the group minus 1 by using a mapping table * specified in @value_map. The last element in this table should be -1. Thus * a table to map three possible interpolation values to integers could be * specified as { NEAREST_NEIGHBOR, BILINEAR, HYPERBOLIC, -1 }. **/ void e_dialog_radio_set (GtkWidget *widget, int value, const int *value_map) { GSList *group, *l; int i; g_return_if_fail (widget != NULL); g_return_if_fail (GTK_IS_RADIO_BUTTON (widget)); g_return_if_fail (value_map != NULL); group = gtk_radio_button_get_group (GTK_RADIO_BUTTON (widget)); i = value_to_index (value_map, value); if (i != -1) { /* Groups are built by prepending items, so the list ends up in reverse * order; we need to flip the index around. */ i = g_slist_length (group) - i - 1; l = g_slist_nth (group, i); if (!l) g_message ("e_dialog_radio_set(): could not find index %d in radio group!", i); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (l->data), TRUE); } else g_message ("e_dialog_radio_set(): could not find value %d in value map!", value); } /** * e_dialog_radio_get: * @widget: A #GtkRadioButton in a radio button group. * @value_map: Map from enumeration values to array indices. * * Queries the selected item in a #GtkRadioButton group. Please read the * description of e_dialog_radio_set() to see how @value_map maps enumeration * values to button indices. * * Return value: Enumeration value which corresponds to the selected item in the * radio group. **/ int e_dialog_radio_get (GtkWidget *widget, const int *value_map) { GSList *group, *l; int i, v; g_return_val_if_fail (widget != NULL, -1); g_return_val_if_fail (GTK_IS_RADIO_BUTTON (widget), -1); g_return_val_if_fail (value_map != NULL, -1); group = gtk_radio_button_get_group (GTK_RADIO_BUTTON (widget)); for (i = 0, l = group; l; l = l->next, i++) { widget = GTK_WIDGET (l->data); if (GTK_TOGGLE_BUTTON (widget)->active) break; } g_assert (l != NULL); /* Groups are built by prepending items, so the list ends up in reverse * order; we need to flip the index around. */ i = g_slist_length (group) - i - 1; v = index_to_value (value_map, i); if (v == -1) { g_message ("e_dialog_radio_get(): could not find index %d in value map!", i); return -1; } return v; } /** * e_dialog_toggle_set: * @widget: A #GtkToggleButton. * @value: Toggle value. * * Sets the value of a #GtkToggleButton-derived widget. This should not be used * for radio buttons; it is more convenient to use use e_dialog_radio_set() * instead. **/ void e_dialog_toggle_set (GtkWidget *widget, gboolean value) { g_return_if_fail (widget != NULL); g_return_if_fail (GTK_IS_TOGGLE_BUTTON (widget)); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (widget), value); } /** * e_dialog_toggle_get: * @widget: A #GtkToggleButton. * * Queries the value of a #GtkToggleButton-derived widget. This should not be * used for radio buttons; it is more convenient to use e_dialog_radio_get() * instead. * * Return value: Toggle value. **/ gboolean e_dialog_toggle_get (GtkWidget *widget) { g_return_val_if_fail (widget != NULL, FALSE); g_return_val_if_fail (GTK_IS_TOGGLE_BUTTON (widget), FALSE); return GTK_TOGGLE_BUTTON (widget)->active; } /** * e_dialog_spin_set: * @widget: A #GtkSpinButton. * @value: Numeric value. * * Sets the value of a #GtkSpinButton widget. **/ void e_dialog_spin_set (GtkWidget *widget, double value) { GtkAdjustment *adj; g_return_if_fail (widget != NULL); g_return_if_fail (GTK_IS_SPIN_BUTTON (widget)); adj = gtk_spin_button_get_adjustment (GTK_SPIN_BUTTON (widget)); adj->value = value; g_signal_emit_by_name (adj, "value_changed", 0); } /** * e_dialog_spin_get_double: * @widget: A #GtkSpinButton. * * Queries the floating-point value of a #GtkSpinButton widget. * * Return value: Numeric value. **/ double e_dialog_spin_get_double (GtkWidget *widget) { GtkAdjustment *adj; g_return_val_if_fail (widget != NULL, 0.0); g_return_val_if_fail (GTK_IS_SPIN_BUTTON (widget), 0.0); adj = gtk_spin_button_get_adjustment (GTK_SPIN_BUTTON (widget)); return adj->value; } /** * e_dialog_spin_get_int: * @widget: A #GtkSpinButton. * * Queries the integer value of a #GtkSpinButton widget. * * Return value: Numeric value. **/ int e_dialog_spin_get_int (GtkWidget *widget) { double value; g_return_val_if_fail (widget != NULL, -1); g_return_val_if_fail (GTK_IS_SPIN_BUTTON (widget), -1); value = e_dialog_spin_get_double (widget); return (int) floor (value); } /** * e_dialog_option_menu_set: * @widget: A #GtkOptionMenu. * @value: Enumerated value. * @value_map: Map from enumeration values to array indices. * * Sets the selected item in a #GtkOptionMenu. Please read the description of * e_dialog_radio_set() to see how @value_map maps enumeration values to item * indices. **/ void e_dialog_option_menu_set (GtkWidget *widget, int value, const int *value_map) { int i; g_return_if_fail (widget != NULL); g_return_if_fail (GTK_IS_OPTION_MENU (widget)); g_return_if_fail (value_map != NULL); i = value_to_index (value_map, value); if (i != -1) gtk_option_menu_set_history (GTK_OPTION_MENU (widget), i); else g_message ("e_dialog_option_menu_set(): could not find value %d in value map!", value); } /** * e_dialog_option_menu_get: * @widget: A #GtkOptionMenu. * @value_map: Map from enumeration values to array indices. * * Queries the selected item in a #GtkOptionMenu. Please read the description * of e_dialog_radio_set() to see how @value_map maps enumeration values to item * indices. * * Return value: Enumeration value which corresponds to the selected item in the * option menu. **/ int e_dialog_option_menu_get (GtkWidget *widget, const int *value_map) { GtkMenu *menu; GtkWidget *active; GList *children; GList *l; int i; int v; g_return_val_if_fail (widget != NULL, -1); g_return_val_if_fail (GTK_IS_OPTION_MENU (widget), -1); g_return_val_if_fail (value_map != NULL, -1); menu = GTK_MENU (gtk_option_menu_get_menu (GTK_OPTION_MENU (widget))); active = gtk_menu_get_active (menu); g_assert (active != NULL); children = GTK_MENU_SHELL (menu)->children; for (i = 0, l = children; l; l = l->next, i++) { if (GTK_WIDGET (l->data) == active) break; } g_assert (l != NULL); v = index_to_value (value_map, i); if (v == -1) { g_message ("e_dialog_option_menu_get(): could not find index %d in value map!", i); return -1; } return v; } /** * e_dialog_dateedit_set: * @widget: A #GnomeDateEdit widget. * @t: Date/time value. * * Sets the value of a #GnomeDateEdit widget. **/ void e_dialog_dateedit_set (GtkWidget *widget, time_t t) { g_return_if_fail (widget != NULL); g_return_if_fail (GNOME_IS_DATE_EDIT (widget)); gnome_date_edit_set_time (GNOME_DATE_EDIT (widget), t); } /** * e_dialog_dateedit_get: * @widget: A #GnomeDateEdit widget. * * Queries the value of a #GnomeDateEdit widget. * * Return value: Date/time value. **/ time_t e_dialog_dateedit_get (GtkWidget *widget) { g_return_val_if_fail (widget != NULL, -1); g_return_val_if_fail (GNOME_IS_DATE_EDIT (widget), -1); return gnome_date_edit_get_date (GNOME_DATE_EDIT (widget)); } /** * e_dialog_widget_hook_value: * @dialog: Dialog box in which the @widget lives in. * @widget: A widget that will control a variable. * @value_var: Pointer to the variable that the @widget will control. * @info: NULL for most widgets, or an integer value map array (see * e_dialog_radio_set() for details). * * Hooks a widget from a dialog box to the variable it will modify. Supported * widgets are: #GtkEditable (char *), #GtkRadioButton (int/value_map pair; see * e_dialog_radio_set() for more information), #GtkTogglebutton (gboolean), * #GtkSpinButton (double), #GtkOptionMenu (int/value_map pair), and * #GnomeDateEdit (time_t). * * A pointer to the appropriate variable to modify should be passed in @value_var. * For values that take a value_map array as well, it should be passed in @info. * * The widgets within a dialog that are hooked with this function will set their * respective variables only when e_dialog_get_values() is called. The typical * use is to call that function in the handler for the "OK" button of a dialog * box. * * In addition, if the specified @dialog is a #GnomePropertyBox, the widgets wil * automatically turn on the "Apply" button of the property box when they are * modified by the user. * * Return value: TRUE if the type of the specified @widget is supported, FALSE * otherwise. **/ gboolean e_dialog_widget_hook_value (GtkWidget *dialog, GtkWidget *widget, gpointer value_var, gpointer info) { DialogHooks *hooks; WidgetHook *wh; g_return_val_if_fail (dialog != NULL, FALSE); g_return_val_if_fail (widget != NULL, FALSE); g_return_val_if_fail (GTK_IS_WIDGET (widget), FALSE); g_return_val_if_fail (value_var != NULL, FALSE); hooks = get_dialog_hooks (dialog); /* First check if it is a "group" widget, like a radio button or an * option menu. Then we check for normal ungrouped widgets. */ if (GTK_IS_RADIO_BUTTON (widget)) hook_radio (dialog, GTK_RADIO_BUTTON (widget), value_var, info); else if (GTK_IS_OPTION_MENU (widget)) hook_option_menu (dialog, GTK_OPTION_MENU (widget), value_var, info); else if (GTK_IS_TOGGLE_BUTTON (widget)) hook_toggle (dialog, GTK_TOGGLE_BUTTON (widget), value_var, info); else if (GTK_IS_SPIN_BUTTON (widget)) hook_spin_button (dialog, GTK_SPIN_BUTTON (widget), value_var, info); else if (GTK_IS_EDITABLE (widget)) hook_editable (dialog, GTK_EDITABLE (widget), value_var, info); else return FALSE; wh = g_new (WidgetHook, 1); wh->widget = widget; wh->value_var = value_var; wh->info = info; hooks->whooks = g_slist_prepend (hooks->whooks, wh); return TRUE; } /** * e_dialog_get_values: * @dialog: A dialog box whose widgets have been hooked to the appropriate * variables with e_dialog_widget_hook_value(). * * Makes every widget in a @dialog that was hooked with * e_dialog_widget_hook_value() apply its value to its corresponding variable. * The typical usage is to call this function in the handler for the "OK" button * of a dialog box. **/ void e_dialog_get_values (GtkWidget *dialog) { DialogHooks *hooks; GSList *l; g_return_if_fail (dialog != NULL); hooks = get_dialog_hooks (dialog); for (l = hooks->whooks; l; l = l->next) { WidgetHook *wh; wh = l->data; if (GTK_IS_RADIO_BUTTON (wh->widget)) get_radio_value (GTK_RADIO_BUTTON (wh->widget), wh->value_var, wh->info); else if (GTK_IS_OPTION_MENU (wh->widget)) get_option_menu_value (GTK_OPTION_MENU (wh->widget), wh->value_var, wh->info); else if (GTK_IS_TOGGLE_BUTTON (wh->widget)) get_toggle_value (GTK_TOGGLE_BUTTON (wh->widget), wh->value_var, wh->info); else if (GTK_IS_SPIN_BUTTON (wh->widget)) get_spin_button_value (GTK_SPIN_BUTTON (wh->widget), wh->value_var, wh->info); else if (GTK_IS_EDITABLE (wh->widget)) get_editable_value (GTK_EDITABLE (wh->widget), wh->value_var, wh->info); else g_assert_not_reached (); } } /** * e_dialog_xml_widget_hook_value: * @xml: Glade XML description of a dialog box. * @dialog: Dialog box in which the widget lives in. * @widget_name: Name of the widget in the Glade XML data. * @value_var: Pointer to the variable that the widget will control. * @info: NULL for most widgets, or an integer value map array (see * e_dialog_radio_set() for details). * * Similar to e_dialog_widget_hook_value(), but uses the widget from a #GladeXML * data structure. * * Return value: TRUE if the type of the specified widget is supported, FALSE * otherwise. **/ gboolean e_dialog_xml_widget_hook_value (GladeXML *xml, GtkWidget *dialog, const char *widget_name, gpointer value_var, gpointer info) { GtkWidget *widget; g_return_val_if_fail (xml != NULL, FALSE); g_return_val_if_fail (GLADE_IS_XML (xml), FALSE); g_return_val_if_fail (dialog != NULL, FALSE); g_return_val_if_fail (widget_name != NULL, FALSE); g_return_val_if_fail (value_var != NULL, FALSE); widget = glade_xml_get_widget (xml, widget_name); if (!widget) { g_message ("e_dialog_xml_widget_hook_value(): could not find widget `%s' in " "Glade data!", widget_name); return FALSE; } return e_dialog_widget_hook_value (dialog, widget, value_var, info); }