aboutsummaryrefslogtreecommitdiffstats
path: root/lib/ephy-dialog.c
diff options
context:
space:
mode:
Diffstat (limited to 'lib/ephy-dialog.c')
-rw-r--r--lib/ephy-dialog.c226
1 files changed, 137 insertions, 89 deletions
diff --git a/lib/ephy-dialog.c b/lib/ephy-dialog.c
index 0c5aa4024..60a8d3fa0 100644
--- a/lib/ephy-dialog.c
+++ b/lib/ephy-dialog.c
@@ -59,6 +59,7 @@ typedef enum
typedef struct
{
const char *id;
+ EphyDialog *dialog;
char *pref;
EphyDialogApplyType apply_type;
GtkWidget *widget;
@@ -87,6 +88,14 @@ struct EphyDialogPrivate
#define SPIN_DELAY 0.20
+enum
+{
+ CHANGED,
+ LAST_SIGNAL
+};
+
+static guint signals [LAST_SIGNAL] = { 0, };
+
static void ephy_dialog_class_init (EphyDialogClass *klass);
static void ephy_dialog_init (EphyDialog *window);
@@ -151,40 +160,24 @@ set_sensitivity (PropertyInfo *info, gboolean sensitive)
static void
set_value_from_pref (PropertyInfo *info, GValue *value)
{
- GConfValue *gcvalue;
- GConfValueType value_type;
char *text;
- gcvalue = eel_gconf_get_value (info->pref);
- if (gcvalue == NULL)
- {
- /* ugly hack around what appears to be a gconf bug
- * it returns a NULL GConfValue for a valid string pref
- * which is "" by default */
- value_type = GCONF_VALUE_STRING;
- }
- else
- {
- value_type = gcvalue->type;
- gconf_value_free (gcvalue);
- }
-
- switch (value_type)
+ switch (info->data_type)
{
- case GCONF_VALUE_STRING:
+ case G_TYPE_STRING:
g_value_init (value, G_TYPE_STRING);
text = eel_gconf_get_string (info->pref);
g_value_take_string (value, text ? text : g_strdup (""));
break;
- case GCONF_VALUE_INT:
+ case G_TYPE_INT:
g_value_init (value, G_TYPE_INT);
g_value_set_int (value, eel_gconf_get_integer (info->pref));
break;
- case GCONF_VALUE_FLOAT:
+ case G_TYPE_FLOAT:
g_value_init (value, G_TYPE_FLOAT);
g_value_set_float (value, eel_gconf_get_float (info->pref));
break;
- case GCONF_VALUE_BOOL:
+ case G_TYPE_BOOLEAN:
g_value_init (value, G_TYPE_BOOLEAN);
g_value_set_boolean (value, eel_gconf_get_boolean (info->pref));
break;
@@ -193,19 +186,19 @@ set_value_from_pref (PropertyInfo *info, GValue *value)
break;
}
- if (!G_VALUE_HOLDS (value, info->data_type))
- {
- g_warning ("Pref %s has wrong value type for id %s!\n", info->pref, info->id);
- }
-
- LOG ("id[%s], pref[%s]: %s", info->id, info->pref, g_strdup_value_contents (value))
+ LOG ("id[%s], pref[%s] = %s", info->id, info->pref, g_strdup_value_contents (value))
}
static void
set_pref_from_value (PropertyInfo *info, GValue *value)
{
const char *pref = info->pref;
- g_return_if_fail (G_VALUE_HOLDS (value, info->data_type));
+
+ if (!G_VALUE_HOLDS (value, info->data_type))
+ {
+ g_warning ("Value type mismatch for id[%s], pref[%s]", info->id, info->pref);
+ return;
+ }
switch (info->data_type)
{
@@ -278,8 +271,6 @@ set_value_from_combobox (PropertyInfo *info, GValue *value)
model = gtk_combo_box_get_model (GTK_COMBO_BOX (info->widget));
- g_return_if_fail (gtk_tree_model_get_column_type (model, info->data_col) == info->data_type);
-
if (gtk_tree_model_iter_nth_child (model, &iter, NULL, index))
{
gtk_tree_model_get_value (model, &iter, info->data_col, value);
@@ -544,8 +535,6 @@ set_combo_box_from_value (PropertyInfo *info, const GValue *value)
model = gtk_combo_box_get_model (GTK_COMBO_BOX (info->widget));
- g_return_if_fail (gtk_tree_model_get_column_type (model, info->data_col) == info->data_type);
-
valid = gtk_tree_model_get_iter_first (model, &iter);
while (valid)
{
@@ -582,6 +571,7 @@ set_combo_box_from_value (PropertyInfo *info, const GValue *value)
info->sane_state = FALSE;
g_return_if_fail (index >= 0);
+ return;
}
LOG ("index[%s] is %d", info->id, index)
@@ -615,6 +605,7 @@ set_radiobuttongroup_from_value (PropertyInfo *info, const GValue *value)
{
info->sane_state = FALSE;
g_return_if_fail (index >= 0 && index < length);
+ return;
}
button = GTK_TOGGLE_BUTTON (g_slist_nth_data (list, index));
@@ -708,30 +699,32 @@ set_info_from_value (PropertyInfo *info, const GValue *value)
/* widget changed callbacks */
static void
-set_pref_from_info (PropertyInfo *info)
+set_pref_from_info_and_emit (PropertyInfo *info)
{
GValue value = { 0, };
- if (info->pref != NULL && info->sane_state)
+ if (!info->sane_state)
{
- set_value_from_info (info, &value);
- set_pref_from_value (info, &value);
- g_value_unset (&value);
+ g_warning ("Not emitting/persisting insane state of id[%s]", info->id);
+ return;
}
- if (info->pref != NULL && !info->sane_state)
+ set_value_from_info (info, &value);
+
+ g_signal_emit (info->dialog, signals[CHANGED], g_quark_from_string (info->id), &value);
+
+ if (info->apply_type == PT_AUTOAPPLY && info->pref != NULL)
{
- g_warning ("Not persisting insane state of id[%s] to pref %s!\n", info->id, info->pref);
+ set_pref_from_value (info, &value);
}
+
+ g_value_unset (&value);
}
static void
togglebutton_clicked_cb (GtkWidget *widget, PropertyInfo *info)
{
- if (info->apply_type == PT_AUTOAPPLY)
- {
- set_pref_from_info (info);
- }
+ set_pref_from_info_and_emit (info);
}
static void
@@ -742,10 +735,7 @@ radiobutton_clicked_cb (GtkWidget *widget, PropertyInfo *info)
return;
}
- if (info->apply_type == PT_AUTOAPPLY)
- {
- set_pref_from_info (info);
- }
+ set_pref_from_info_and_emit (info);
}
static gboolean
@@ -774,7 +764,7 @@ spinbutton_timeout_cb (PropertyInfo *info)
* and set in the pref. Otherwise the old value is used */
gtk_spin_button_update (GTK_SPIN_BUTTON (info->widget));
- set_pref_from_info (info);
+ set_pref_from_info_and_emit (info);
/* done, don't run again */
return FALSE;
@@ -810,27 +800,7 @@ spinbutton_changed_cb (GtkWidget *widget, PropertyInfo *info)
static void
changed_cb (GtkWidget *widget, PropertyInfo *info)
{
- if (info->apply_type == PT_AUTOAPPLY)
- {
- set_pref_from_info (info);
- }
-}
-
-static void
-set_info_from_pref (PropertyInfo *info)
-{
- GValue value = { 0, };
-
- g_return_if_fail (info->widget != NULL);
-
- if (info->pref != NULL)
- {
- set_value_from_pref (info, &value);
- set_info_from_value (info, &value);
- g_value_unset (&value);
-
- set_sensitivity (info, eel_gconf_key_is_writable (info->pref));
- }
+ set_pref_from_info_and_emit (info);
}
static void
@@ -895,8 +865,9 @@ init_props (EphyDialog *dialog, const EphyDialogProperty *properties, GladeXML *
for (i = 0 ; properties[i].id != NULL; i++)
{
PropertyInfo *info = g_new0 (PropertyInfo, 1);
-
+
info->id = properties[i].id;
+ info->dialog = dialog;
info->pref = g_strdup (properties[i].pref);
info->apply_type = properties[i].apply_type;
info->string_enum = NULL;
@@ -950,7 +921,21 @@ init_props (EphyDialog *dialog, const EphyDialogProperty *properties, GladeXML *
static void
load_info (gpointer key, PropertyInfo *info, EphyDialog *dialog)
{
- set_info_from_pref (info);
+ GValue value = { 0, };
+
+ g_return_if_fail (info->widget != NULL);
+
+ if (info->pref != NULL)
+ {
+ set_value_from_pref (info, &value);
+ set_info_from_value (info, &value);
+
+ g_signal_emit (info->dialog, signals[CHANGED], g_quark_from_string (info->id), &value);
+
+ g_value_unset (&value);
+
+ set_sensitivity (info, eel_gconf_key_is_writable (info->pref));
+ }
info->loaded = TRUE;
}
@@ -958,10 +943,22 @@ load_info (gpointer key, PropertyInfo *info, EphyDialog *dialog)
static void
save_info (gpointer key, PropertyInfo *info, EphyDialog *dialog)
{
- if (info->apply_type == PT_NORMAL)
+ GValue value = { 0, };
+
+ if (info->pref == NULL || info->apply_type != PT_NORMAL)
+ {
+ return;
+ }
+
+ if (!info->sane_state)
{
- set_pref_from_info (info);
+ g_warning ("Not persisting insane state of id[%s]", info->id);
+ return;
}
+
+ set_value_from_info (info, &value);
+ set_pref_from_value (info, &value);
+ g_value_unset (&value);
}
static void
@@ -1029,7 +1026,7 @@ impl_show (EphyDialog *dialog)
setup_default_size (dialog);
- if (dialog->priv->parent)
+ if (dialog->priv->parent != NULL)
{
/* make the dialog transient again, because it seems to get
* forgotten after gtk_widget_hide
@@ -1111,32 +1108,38 @@ ephy_dialog_set_pref (EphyDialog *dialog,
void
ephy_dialog_set_size_group (EphyDialog *dialog,
- const char **controls_id,
- guint n_controls)
+ const char *first_id,
+ ...)
{
GtkSizeGroup *size_group;
- int i;
+ va_list vl;
size_group = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL);
- for (i = 0; i < n_controls; i++)
+ va_start (vl, first_id);
+
+ while (first_id != NULL)
{
PropertyInfo *info;
- info = lookup_info (dialog, controls_id[i]);
+ info = lookup_info (dialog, first_id);
g_return_if_fail (info != NULL);
g_return_if_fail (info->widget != NULL);
gtk_size_group_add_widget (size_group, info->widget);
+
+ first_id = va_arg (vl, const char*);
}
+
+ va_end (vl);
}
void
ephy_dialog_construct (EphyDialog *dialog,
- const EphyDialogProperty *properties,
- const char *file,
- const char *name)
+ const EphyDialogProperty *properties,
+ const char *file,
+ const char *name)
{
EphyDialogClass *klass = EPHY_DIALOG_GET_CLASS (dialog);
return klass->construct (dialog, properties, file, name);
@@ -1149,11 +1152,50 @@ ephy_dialog_show (EphyDialog *dialog)
klass->show (dialog);
}
+#if 0
+static void
+run_response_cb (GtkWidget *dialog,
+ int response,
+ int *result)
+{
+ *result = response;
+
+ gtk_grab_remove (dialog);
+ LOG ("run_response_cb: leaving gtk level %d", gtk_main_level())
+ gtk_main_quit();
+}
+#endif
+
int
ephy_dialog_run (EphyDialog *dialog)
{
ephy_dialog_show (dialog);
+#if 0
+ if (dialog->priv->parent != NULL && dialog->priv->modal == FALSE)
+ {
+ GtkWindowGroup *group;
+ int response = 0;
+
+ group = GTK_WINDOW (dialog->priv->parent)->group;
+ if (group == NULL)
+ {
+ group = gtk_window_group_new ();
+ gtk_window_group_add_window (group, GTK_WINDOW (dialog->priv->parent));
+ g_object_unref (group);
+ }
+
+ gtk_window_group_add_window (group, GTK_WINDOW (dialog->priv->dialog));
+ g_signal_connect(dialog->priv->dialog, "response",
+ G_CALLBACK (run_response_cb), &response);
+ gtk_grab_add (dialog->priv->dialog);
+ LOG ("ephy_dialog_run before main(): level %d", gtk_main_level())
+ gtk_main ();
+ LOG ("ephy_dialog_run after main(): level %d", gtk_main_level())
+
+ return response;
+ }
+#endif
return gtk_dialog_run (GTK_DIALOG (dialog->priv->dialog));
}
@@ -1254,15 +1296,8 @@ static void
ephy_dialog_set_parent (EphyDialog *dialog,
GtkWidget *parent)
{
- g_return_if_fail (dialog->priv->parent == NULL);
- g_return_if_fail (GTK_IS_WINDOW (parent));
- g_return_if_fail (GTK_IS_WINDOW (dialog->priv->dialog));
-
dialog->priv->parent = parent;
- gtk_window_set_transient_for (GTK_WINDOW (dialog->priv->dialog),
- GTK_WINDOW (parent));
-
g_object_notify (G_OBJECT (dialog), "parent-window");
}
@@ -1318,6 +1353,17 @@ ephy_dialog_class_init (EphyDialogClass *klass)
klass->construct = impl_construct;
klass->show = impl_show;
+ signals[CHANGED] =
+ g_signal_new ("changed",
+ EPHY_TYPE_DIALOG,
+ G_SIGNAL_RUN_FIRST | G_SIGNAL_DETAILED,
+ G_STRUCT_OFFSET (EphyDialogClass, changed),
+ NULL, NULL,
+ g_cclosure_marshal_VOID__POINTER,
+ G_TYPE_NONE,
+ 1,
+ G_TYPE_POINTER);
+
g_object_class_install_property (object_class,
PROP_PARENT_WINDOW,
g_param_spec_object ("parent-window",
@@ -1346,6 +1392,8 @@ ephy_dialog_new (void)
EphyDialog *
ephy_dialog_new_with_parent (GtkWidget *parent_window)
{
+ g_return_val_if_fail (parent_window != NULL, NULL);
+
return EPHY_DIALOG (g_object_new (EPHY_TYPE_DIALOG,
"parent-window", parent_window,
NULL));