/* * e-account-combo-box.c * * This program 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) version 3. * * 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with the program; if not, see * * * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) * */ #include "e-account-combo-box.h" #include #define E_ACCOUNT_COMBO_BOX_GET_PRIVATE(obj) \ (G_TYPE_INSTANCE_GET_PRIVATE \ ((obj), E_TYPE_ACCOUNT_COMBO_BOX, EAccountComboBoxPrivate)) enum { COLUMN_STRING, COLUMN_ACCOUNT }; enum { REFRESHED, LAST_SIGNAL }; struct _EAccountComboBoxPrivate { EAccountList *account_list; GHashTable *index; gint num_displayed_accounts; }; static gpointer parent_class; static CamelSession *camel_session; static guint signal_ids[LAST_SIGNAL]; static gboolean account_combo_box_has_dupes (GList *list, const gchar *address) { GList *iter; guint count = 0; /* Look for duplicates of the given email address. */ for (iter = list; iter != NULL; iter = iter->next) { EAccount *account = iter->data; if (g_ascii_strcasecmp (account->id->address, address) == 0) count++; } return (count > 1); } static EAccount * account_combo_box_choose_account (EAccountComboBox *combo_box) { EAccountList *account_list; EAccount *account; GtkTreeModel *model; GtkTreeIter iter; account_list = e_account_combo_box_get_account_list (combo_box); g_return_val_if_fail (account_list != NULL, NULL); /* First try the default account. */ /* XXX EAccountList misuses const. */ account = (EAccount *) e_account_list_get_default (account_list); /* If there is no default account, give up. */ if (account == NULL) return NULL; /* Make sure the default account appears in the combo box. */ if (g_hash_table_lookup (combo_box->priv->index, account) != NULL) return account; /* Default account is disabled or otherwise unusable, * so fall back to the first account in the combo box. */ model = gtk_combo_box_get_model (GTK_COMBO_BOX (combo_box)); if (!gtk_tree_model_get_iter_first (model, &iter)) return NULL; gtk_tree_model_get (model, &iter, COLUMN_ACCOUNT, &account, -1); return account; } static gboolean account_combo_box_test_account (EAccount *account) { CamelStore *store; CamelException ex; const gchar *url; gboolean writable = FALSE; /* Account must be enabled. */ if (!account->enabled) return FALSE; /* Account must have a non-empty email address. */ if (account->id->address == NULL || *account->id->address == '\0') return FALSE; /* XXX Not sure I understand this part. */ if (account->parent_uid == NULL) return TRUE; /* Account must be writable. */ camel_exception_init (&ex); url = e_account_get_string (account, E_ACCOUNT_SOURCE_URL); store = CAMEL_STORE (camel_session_get_service ( camel_session, url, CAMEL_PROVIDER_STORE, &ex)); if (store != NULL) { writable = (store->mode & CAMEL_STORE_WRITE); g_object_unref (store); } camel_exception_clear (&ex); return writable; } static void account_combo_box_refresh_cb (EAccountList *account_list, EAccount *unused, EAccountComboBox *combo_box) { GtkListStore *store; GtkTreeModel *model; EIterator *account_iter; EAccount *account; GHashTable *index; GList *list = NULL; GList *iter; combo_box->priv->num_displayed_accounts = 0; store = gtk_list_store_new (2, G_TYPE_STRING, E_TYPE_ACCOUNT); model = GTK_TREE_MODEL (store); index = combo_box->priv->index; g_hash_table_remove_all (index); if (account_list == NULL) goto skip; /* Build a list of EAccounts to display. */ account_iter = e_list_get_iterator (E_LIST (account_list)); while (e_iterator_is_valid (account_iter)) { EAccount *account; /* XXX EIterator misuses const. */ account = (EAccount *) e_iterator_get (account_iter); if (account_combo_box_test_account (account)) list = g_list_prepend (list, account); e_iterator_next (account_iter); } g_object_unref (account_iter); list = g_list_reverse (list); /* Populate the list store and index. */ for (iter = list; iter != NULL; iter = iter->next) { GtkTreeRowReference *reference; GtkTreeIter tree_iter; GtkTreePath *path; gchar *string; account = iter->data; combo_box->priv->num_displayed_accounts++; /* Show the account name for duplicate email addresses. */ if (account_combo_box_has_dupes (list, account->id->address)) string = g_strdup_printf ( "%s <%s> (%s)", account->id->name, account->id->address, account->name); else string = g_strdup_printf ( "%s <%s>", account->id->name, account->id->address); gtk_list_store_append (store, &tree_iter); gtk_list_store_set ( store, &tree_iter, COLUMN_STRING, string, COLUMN_ACCOUNT, account, -1); path = gtk_tree_model_get_path (model, &tree_iter); reference = gtk_tree_row_reference_new (model, path); g_hash_table_insert (index, account, reference); gtk_tree_path_free (path); g_free (string); } g_list_free (list); skip: /* Restore the previously selected account. */ account = e_account_combo_box_get_active (combo_box); if (account != NULL) g_object_ref (account); gtk_combo_box_set_model (GTK_COMBO_BOX (combo_box), model); e_account_combo_box_set_active (combo_box, account); if (account != NULL) g_object_unref (account); g_signal_emit (combo_box, signal_ids[REFRESHED], 0); } static GObject * account_combo_box_constructor (GType type, guint n_construct_properties, GObjectConstructParam *construct_properties) { GObject *object; GtkCellRenderer *renderer; /* Chain up to parent's constructor() method. */ object = G_OBJECT_CLASS (parent_class)->constructor ( type, n_construct_properties, construct_properties); renderer = gtk_cell_renderer_text_new (); gtk_cell_layout_pack_start ( GTK_CELL_LAYOUT (object), renderer, TRUE); gtk_cell_layout_add_attribute ( GTK_CELL_LAYOUT (object), renderer, "text", COLUMN_STRING); return object; } static void account_combo_box_dispose (GObject *object) { EAccountComboBoxPrivate *priv; priv = E_ACCOUNT_COMBO_BOX_GET_PRIVATE (object); if (priv->account_list != NULL) { g_signal_handlers_disconnect_by_func ( priv->account_list, account_combo_box_refresh_cb, object); g_object_unref (priv->account_list); priv->account_list = NULL; } g_hash_table_remove_all (priv->index); /* Chain up to parent's dispose() method. */ G_OBJECT_CLASS (parent_class)->dispose (object); } static void account_combo_box_finalize (GObject *object) { EAccountComboBoxPrivate *priv; priv = E_ACCOUNT_COMBO_BOX_GET_PRIVATE (object); g_hash_table_destroy (priv->index); /* Chain up to parent's finalize() method. */ G_OBJECT_CLASS (parent_class)->finalize (object); } static void account_combo_box_class_init (EAccountComboBoxClass *class) { GObjectClass *object_class; parent_class = g_type_class_peek_parent (class); g_type_class_add_private (class, sizeof (EAccountComboBoxPrivate)); object_class = G_OBJECT_CLASS (class); object_class->constructor = account_combo_box_constructor; object_class->dispose = account_combo_box_dispose; object_class->finalize = account_combo_box_finalize; signal_ids[REFRESHED] = g_signal_new ( "refreshed", G_TYPE_FROM_CLASS (class), G_SIGNAL_RUN_LAST, 0, NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); } static void account_combo_box_init (EAccountComboBox *combo_box) { GHashTable *index; /* Reverse-lookup index */ index = g_hash_table_new_full ( g_direct_hash, g_direct_equal, (GDestroyNotify) g_object_unref, (GDestroyNotify) gtk_tree_row_reference_free); combo_box->priv = E_ACCOUNT_COMBO_BOX_GET_PRIVATE (combo_box); combo_box->priv->index = index; } GType e_account_combo_box_get_type (void) { static GType type = 0; if (G_UNLIKELY (type == 0)) { static const GTypeInfo type_info = { sizeof (EAccountComboBoxClass), (GBaseInitFunc) NULL, (GBaseFinalizeFunc) NULL, (GClassInitFunc) account_combo_box_class_init, (GClassFinalizeFunc) NULL, NULL, /* class_data */ sizeof (EAccountComboBox), 0, /* n_preallocs */ (GInstanceInitFunc) account_combo_box_init, NULL /* value_table */ }; type = g_type_register_static ( GTK_TYPE_COMBO_BOX, "EAccountComboBox", &type_info, 0); } return type; } GtkWidget * e_account_combo_box_new (void) { return g_object_new (E_TYPE_ACCOUNT_COMBO_BOX, NULL); } void e_account_combo_box_set_session (CamelSession *session) { /* XXX Really gross hack. * * We need a CamelSession to test whether a given EAccount is * writable. The global CamelSession object is defined in the * mailer, but we're too far down the stack to access it. So * we have to rely on someone passing us a reference to it. * * A much cleaner solution would be to store the writeability * of an account directly into the EAccount, but this would likely * require breaking ABI and all the fun that goes along with that. */ camel_session = session; } EAccountList * e_account_combo_box_get_account_list (EAccountComboBox *combo_box) { g_return_val_if_fail (E_IS_ACCOUNT_COMBO_BOX (combo_box), NULL); return combo_box->priv->account_list; } void e_account_combo_box_set_account_list (EAccountComboBox *combo_box, EAccountList *account_list) { EAccountComboBoxPrivate *priv; g_return_if_fail (E_IS_ACCOUNT_COMBO_BOX (combo_box)); if (account_list != NULL) g_return_if_fail (E_IS_ACCOUNT_LIST (account_list)); priv = E_ACCOUNT_COMBO_BOX_GET_PRIVATE (combo_box); if (priv->account_list != NULL) { g_signal_handlers_disconnect_by_func ( priv->account_list, account_combo_box_refresh_cb, combo_box); g_object_unref (priv->account_list); priv->account_list = NULL; } if (account_list != NULL) { priv->account_list = g_object_ref (account_list); /* Listen for changes to the account list. */ g_signal_connect ( priv->account_list, "account-added", G_CALLBACK (account_combo_box_refresh_cb), combo_box); g_signal_connect ( priv->account_list, "account-changed", G_CALLBACK (account_combo_box_refresh_cb), combo_box); g_signal_connect ( priv->account_list, "account-removed", G_CALLBACK (account_combo_box_refresh_cb), combo_box); } account_combo_box_refresh_cb (account_list, NULL, combo_box); } EAccount * e_account_combo_box_get_active (EAccountComboBox *combo_box) { EAccount *account; GtkTreeModel *model; GtkTreeIter iter; gboolean iter_set; g_return_val_if_fail (E_IS_ACCOUNT_COMBO_BOX (combo_box), NULL); iter_set = gtk_combo_box_get_active_iter ( GTK_COMBO_BOX (combo_box), &iter); if (!iter_set) return NULL; model = gtk_combo_box_get_model (GTK_COMBO_BOX (combo_box)); gtk_tree_model_get (model, &iter, COLUMN_ACCOUNT, &account, -1); return account; } gboolean e_account_combo_box_set_active (EAccountComboBox *combo_box, EAccount *account) { EAccountList *account_list; GtkTreeRowReference *reference; GtkTreeModel *model; GtkTreePath *path; GtkTreeIter iter; gboolean iter_set; g_return_val_if_fail (E_IS_ACCOUNT_COMBO_BOX (combo_box), FALSE); if (account != NULL) g_return_val_if_fail (E_IS_ACCOUNT (account), FALSE); account_list = combo_box->priv->account_list; g_return_val_if_fail (account_list != NULL, FALSE); /* NULL means choose an account ourselves. */ if (account == NULL) account = account_combo_box_choose_account (combo_box); if (account == NULL) return FALSE; /* Lookup the tree row reference for the account. */ reference = g_hash_table_lookup (combo_box->priv->index, account); if (reference == NULL) return FALSE; /* Convert the reference to a tree iterator. */ path = gtk_tree_row_reference_get_path (reference); model = gtk_combo_box_get_model (GTK_COMBO_BOX (combo_box)); iter_set = gtk_tree_model_get_iter (model, &iter, path); gtk_tree_path_free (path); if (!iter_set) return FALSE; /* Activate the corresponding combo box item. */ gtk_combo_box_set_active_iter (GTK_COMBO_BOX (combo_box), &iter); return TRUE; } const gchar * e_account_combo_box_get_active_name (EAccountComboBox *combo_box) { EAccount *account; g_return_val_if_fail (E_IS_ACCOUNT_COMBO_BOX (combo_box), NULL); account = e_account_combo_box_get_active (combo_box); return (account != NULL) ? account->name : NULL; } gboolean e_account_combo_box_set_active_name (EAccountComboBox *combo_box, const gchar *account_name) { EAccountList *account_list; EAccount *account; g_return_val_if_fail (E_IS_ACCOUNT_COMBO_BOX (combo_box), FALSE); account_list = combo_box->priv->account_list; g_return_val_if_fail (account_list != NULL, FALSE); /* XXX EAccountList misuses const. */ account = (EAccount *) e_account_list_find ( account_list, E_ACCOUNT_FIND_NAME, account_name); if (account == NULL) return FALSE; return e_account_combo_box_set_active (combo_box, account); } /** * e_account_combo_box_count_displayed_accounts: * @combo_box: an #EAccountComboBox * * Counts the number of accounts that are displayed in the @combo_box. This may not * be the actual number of accounts that are configured, as some of those accounts * may be disabled by the user. * * Return value: number of active and valid accounts as shown in the @combo_box. */ gint e_account_combo_box_count_displayed_accounts (EAccountComboBox *combo_box) { g_return_val_if_fail (E_IS_ACCOUNT_COMBO_BOX (combo_box), -1); return combo_box->priv->num_displayed_accounts; }