diff options
Diffstat (limited to 'libempathy-gtk/empathy-profile-chooser.c')
-rw-r--r-- | libempathy-gtk/empathy-profile-chooser.c | 425 |
1 files changed, 252 insertions, 173 deletions
diff --git a/libempathy-gtk/empathy-profile-chooser.c b/libempathy-gtk/empathy-profile-chooser.c index 14965741c..b3cbf90d0 100644 --- a/libempathy-gtk/empathy-profile-chooser.c +++ b/libempathy-gtk/empathy-profile-chooser.c @@ -1,6 +1,6 @@ -/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ +/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */ /* - * Copyright (C) 2007-2008 Collabora Ltd. + * Copyright (C) 2007-2009 Collabora Ltd. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public @@ -17,6 +17,7 @@ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * * Authors: Xavier Claessens <xclaesse@gmail.com> + * Jonny Lamb <jonny.lamb@collabora.co.uk> */ #include <config.h> @@ -27,6 +28,8 @@ #include <libmissioncontrol/mc-profile.h> #include <libmissioncontrol/mc-protocol.h> +#include <libempathy/empathy-utils.h> + #include "empathy-profile-chooser.h" #include "empathy-ui-utils.h" @@ -34,112 +37,277 @@ * SECTION:empathy-profile-chooser * @title: EmpathyProfileChooser * @short_description: A widget used to choose from a list of profiles - * @include: libempathy-gtk/empathy-account-chooser.h + * @include: libempathy-gtk/empathy-profile-chooser.h * - * #EmpathyProfileChooser is a widget which provides a chooser of available + * #EmpathyProfileChooser is a widget which extends #GtkComboBox to provides a + * chooser of available profiles. + */ + +/** + * EmpathyProfileChooser: + * @parent: parent object + * + * Widget which extends #GtkComboBox to provide a chooser of available * profiles. */ -enum { - COL_ICON, - COL_LABEL, - COL_PROFILE, - COL_COUNT +#define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathyProfileChooser) +typedef struct +{ + GtkListStore *store; + gboolean dispose_run; +} EmpathyProfileChooserPriv; + +enum +{ + COL_ICON, + COL_LABEL, + COL_PROFILE, + COL_COUNT }; +G_DEFINE_TYPE (EmpathyProfileChooser, empathy_profile_chooser, + GTK_TYPE_COMBO_BOX); + +static gint +profile_chooser_sort_profile_value (McProfile *profile) +{ + guint i; + const gchar *profile_name; + const gchar *names[] = { + "jabber", + "salut", + "gtalk", + NULL + }; + + profile_name = mc_profile_get_unique_name (profile); + + for (i = 0 ; names[i]; i++) + { + if (strcmp (profile_name, names[i]) == 0) + return i; + } + + return i; +} + +static gint +profile_chooser_sort_func (GtkTreeModel *model, + GtkTreeIter *iter_a, + GtkTreeIter *iter_b, + gpointer user_data) +{ + McProfile *profile_a; + McProfile *profile_b; + gint cmp; + + gtk_tree_model_get (model, iter_a, + COL_PROFILE, &profile_a, + -1); + gtk_tree_model_get (model, iter_b, + COL_PROFILE, &profile_b, + -1); + + cmp = profile_chooser_sort_profile_value (profile_a); + cmp -= profile_chooser_sort_profile_value (profile_b); + if (cmp == 0) + { + cmp = strcmp (mc_profile_get_display_name (profile_a), + mc_profile_get_display_name (profile_b)); + } + + g_object_unref (profile_a); + g_object_unref (profile_b); + + return cmp; +} + +static void +profile_chooser_constructed (GObject *object) +{ + EmpathyProfileChooser *profile_chooser; + EmpathyProfileChooserPriv *priv; + + GList *profiles, *l, *seen; + GtkCellRenderer *renderer; + GtkTreeIter iter; + gboolean iter_set = FALSE; + McManager *btf_cm; + + priv = GET_PRIV (object); + profile_chooser = EMPATHY_PROFILE_CHOOSER (object); + + /* set up combo box with new store */ + priv->store = gtk_list_store_new (COL_COUNT, + G_TYPE_STRING, /* Icon name */ + G_TYPE_STRING, /* Label */ + MC_TYPE_PROFILE); /* Profile */ + + gtk_combo_box_set_model (GTK_COMBO_BOX (object), + GTK_TREE_MODEL (priv->store)); + + renderer = gtk_cell_renderer_pixbuf_new (); + gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (object), renderer, FALSE); + gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (object), renderer, + "icon-name", COL_ICON, + NULL); + g_object_set (renderer, "stock-size", GTK_ICON_SIZE_BUTTON, NULL); + + renderer = gtk_cell_renderer_text_new (); + gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (object), renderer, TRUE); + gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (object), renderer, + "text", COL_LABEL, + NULL); + + btf_cm = mc_manager_lookup ("butterfly"); + profiles = mc_profiles_list (); + seen = NULL; + for (l = profiles; l; l = g_list_next (l)) + { + McProfile *profile; + McProtocol *protocol; + const gchar *unique_name; + + profile = l->data; + + /* Check if the CM is installed, otherwise skip that profile. + * Workaround SF bug #1688779 */ + protocol = mc_profile_get_protocol (profile); + if (!protocol) + continue; + + g_object_unref (protocol); + + /* Skip MSN-Haze if we have butterfly */ + unique_name = mc_profile_get_unique_name (profile); + if (btf_cm && strcmp (unique_name, "msn-haze") == 0) + continue; + + if (g_list_find_custom (seen, unique_name, (GCompareFunc) strcmp)) + continue; + + seen = g_list_append (seen, (char *) unique_name); + + gtk_list_store_insert_with_values (priv->store, &iter, 0, + COL_ICON, mc_profile_get_icon_name (profile), + COL_LABEL, mc_profile_get_display_name (profile), + COL_PROFILE, profile, + -1); + iter_set = TRUE; + } + + g_list_free (seen); + + if (btf_cm) + g_object_unref (btf_cm); + + /* Set the profile sort function */ + gtk_tree_sortable_set_sort_func (GTK_TREE_SORTABLE (priv->store), + COL_PROFILE, + profile_chooser_sort_func, + NULL, NULL); + gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (priv->store), + COL_PROFILE, + GTK_SORT_ASCENDING); + + if (iter_set) + gtk_combo_box_set_active_iter (GTK_COMBO_BOX (object), &iter); + + mc_profiles_free_list (profiles); + + if (G_OBJECT_CLASS (empathy_profile_chooser_parent_class)->constructed) + G_OBJECT_CLASS (empathy_profile_chooser_parent_class)->constructed (object); +} + +static void +empathy_profile_chooser_init (EmpathyProfileChooser *profile_chooser) +{ + EmpathyProfileChooserPriv *priv = + G_TYPE_INSTANCE_GET_PRIVATE (profile_chooser, + EMPATHY_TYPE_PROFILE_CHOOSER, EmpathyProfileChooserPriv); + + priv->dispose_run = FALSE; + + profile_chooser->priv = priv; +} + +static void +profile_chooser_dispose (GObject *object) +{ + EmpathyProfileChooser *profile_chooser = EMPATHY_PROFILE_CHOOSER (object); + EmpathyProfileChooserPriv *priv = GET_PRIV (profile_chooser); + + if (priv->dispose_run) + return; + + priv->dispose_run = TRUE; + + if (priv->store) + { + g_object_unref (priv->store); + priv->store = NULL; + } + + (G_OBJECT_CLASS (empathy_profile_chooser_parent_class)->dispose) (object); +} + +static void +empathy_profile_chooser_class_init (EmpathyProfileChooserClass *klass) +{ + GObjectClass *object_class = G_OBJECT_CLASS (klass); + + object_class->constructed = profile_chooser_constructed; + object_class->dispose = profile_chooser_dispose; + + g_type_class_add_private (object_class, sizeof (EmpathyProfileChooserPriv)); +} + /** * empathy_profile_chooser_dup_selected: - * @widget: an #EmpathyProfileChooser + * @profile_chooser: an #EmpathyProfileChooser * - * Returns a new reference to the selected #McProfile in @widget. The returned - * #McProfile should be unrefed with g_object_unref() when finished with. + * Returns a new reference to the selected #McProfile in @profile_chooser. The + * returned #McProfile should be unrefed with g_object_unref() when finished + * with. * * Return value: a new reference to the selected #McProfile */ McProfile * -empathy_profile_chooser_dup_selected (GtkWidget *widget) +empathy_profile_chooser_dup_selected (EmpathyProfileChooser *profile_chooser) { - GtkTreeModel *model; - GtkTreeIter iter; - McProfile *profile = NULL; - - model = gtk_combo_box_get_model (GTK_COMBO_BOX (widget)); - if (gtk_combo_box_get_active_iter (GTK_COMBO_BOX (widget), &iter)) { - gtk_tree_model_get (model, &iter, - COL_PROFILE, &profile, - -1); - } - - return profile; + EmpathyProfileChooserPriv *priv = GET_PRIV (profile_chooser); + GtkTreeIter iter; + McProfile *profile = NULL; + + g_return_val_if_fail (EMPATHY_IS_PROFILE_CHOOSER (profile_chooser), NULL); + + if (gtk_combo_box_get_active_iter (GTK_COMBO_BOX (profile_chooser), &iter)) + { + gtk_tree_model_get (GTK_TREE_MODEL (priv->store), &iter, + COL_PROFILE, &profile, + -1); + } + + return profile; } /** * empathy_profile_chooser_n_profiles: - * @widget: an #EmpathyProfileChooser + * @profile_chooser: an #EmpathyProfileChooser * - * Returns the number of profiles in @widget. + * Returns the number of profiles in @profile_chooser. * - * Return value: the number of profiles in @widget + * Return value: the number of profiles in @profile_chooser */ gint -empathy_profile_chooser_n_profiles (GtkWidget *widget) +empathy_profile_chooser_n_profiles (EmpathyProfileChooser *profile_chooser) { - GtkTreeModel *model; + EmpathyProfileChooserPriv *priv = GET_PRIV (profile_chooser); - model = gtk_combo_box_get_model (GTK_COMBO_BOX (widget)); - - return gtk_tree_model_iter_n_children (model, NULL); -} + g_return_val_if_fail (EMPATHY_IS_PROFILE_CHOOSER (profile_chooser), 0); -static gint -profile_chooser_sort_profile_value (McProfile *profile) -{ - guint i; - const gchar *profile_name; - const gchar *names[] = {"jabber", - "salut", - "gtalk", - NULL}; - - profile_name = mc_profile_get_unique_name (profile); - - for (i = 0 ; names[i]; i++) { - if (strcmp (profile_name, names[i]) == 0) { - return i; - } - } - - return i; -} - -static gint -profile_chooser_sort_func (GtkTreeModel *model, - GtkTreeIter *iter_a, - GtkTreeIter *iter_b, - gpointer user_data) -{ - McProfile *profile_a; - McProfile *profile_b; - gint cmp; - - gtk_tree_model_get (model, iter_a, - COL_PROFILE, &profile_a, - -1); - gtk_tree_model_get (model, iter_b, - COL_PROFILE, &profile_b, - -1); - - cmp = profile_chooser_sort_profile_value (profile_a); - cmp -= profile_chooser_sort_profile_value (profile_b); - if (cmp == 0) { - cmp = strcmp (mc_profile_get_display_name (profile_a), - mc_profile_get_display_name (profile_b)); - } - - g_object_unref (profile_a); - g_object_unref (profile_b); - - return cmp; + return gtk_tree_model_iter_n_children (GTK_TREE_MODEL (priv->store), NULL); } /** @@ -152,94 +320,5 @@ profile_chooser_sort_func (GtkTreeModel *model, GtkWidget * empathy_profile_chooser_new (void) { - GList *profiles, *l, *seen; - GtkListStore *store; - GtkCellRenderer *renderer; - GtkWidget *combo_box; - GtkTreeIter iter; - gboolean iter_set = FALSE; - McManager *btf_cm; - - /* set up combo box with new store */ - store = gtk_list_store_new (COL_COUNT, - G_TYPE_STRING, /* Icon name */ - G_TYPE_STRING, /* Label */ - MC_TYPE_PROFILE); /* Profile */ - combo_box = gtk_combo_box_new_with_model (GTK_TREE_MODEL (store)); - - - renderer = gtk_cell_renderer_pixbuf_new (); - gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo_box), renderer, FALSE); - gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (combo_box), renderer, - "icon-name", COL_ICON, - NULL); - g_object_set (renderer, "stock-size", GTK_ICON_SIZE_BUTTON, NULL); - - renderer = gtk_cell_renderer_text_new (); - gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo_box), renderer, TRUE); - gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (combo_box), renderer, - "text", COL_LABEL, - NULL); - - btf_cm = mc_manager_lookup ("butterfly"); - profiles = mc_profiles_list (); - seen = NULL; - for (l = profiles; l; l = l->next) { - McProfile *profile; - McProtocol *protocol; - const gchar *unique_name; - - profile = l->data; - - /* Check if the CM is installed, otherwise skip that profile. - * Workaround SF bug #1688779 */ - protocol = mc_profile_get_protocol (profile); - if (!protocol) { - continue; - } - g_object_unref (protocol); - - /* Skip MSN-Haze if we have butterfly */ - unique_name = mc_profile_get_unique_name (profile); - if (btf_cm && strcmp (unique_name, "msn-haze") == 0) { - continue; - } - - if (g_list_find_custom (seen, unique_name, (GCompareFunc) strcmp)) { - continue; - } - seen = g_list_append (seen, (char *) unique_name); - - gtk_list_store_insert_with_values (store, &iter, 0, - COL_ICON, mc_profile_get_icon_name (profile), - COL_LABEL, mc_profile_get_display_name (profile), - COL_PROFILE, profile, - -1); - iter_set = TRUE; - } - - g_list_free (seen); - - if (btf_cm) { - g_object_unref (btf_cm); - } - - /* Set the profile sort function */ - gtk_tree_sortable_set_sort_func (GTK_TREE_SORTABLE (store), - COL_PROFILE, - profile_chooser_sort_func, - NULL, NULL); - gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store), - COL_PROFILE, - GTK_SORT_ASCENDING); - - if (iter_set) { - gtk_combo_box_set_active_iter (GTK_COMBO_BOX (combo_box), &iter); - } - - mc_profiles_free_list (profiles); - g_object_unref (store); - - return combo_box; + return GTK_WIDGET (g_object_new (EMPATHY_TYPE_PROFILE_CHOOSER, NULL)); } - |