aboutsummaryrefslogtreecommitdiffstats
path: root/mail
diff options
context:
space:
mode:
Diffstat (limited to 'mail')
-rw-r--r--mail/e-mail-session.c120
-rw-r--r--mail/em-subscription-editor.c675
2 files changed, 708 insertions, 87 deletions
diff --git a/mail/e-mail-session.c b/mail/e-mail-session.c
index 0687d7561f..c14bdd56d8 100644
--- a/mail/e-mail-session.c
+++ b/mail/e-mail-session.c
@@ -1096,6 +1096,125 @@ mail_session_get_socks_proxy (CamelSession *session,
g_free (uri);
}
+static gboolean
+mail_session_authenticate_sync (CamelSession *session,
+ CamelService *service,
+ const gchar *mechanism,
+ GCancellable *cancellable,
+ GError **error)
+{
+ CamelServiceAuthType *authtype = NULL;
+ CamelAuthenticationResult result;
+ CamelProvider *provider;
+ CamelURL *url;
+ const gchar *password;
+ guint32 password_flags;
+ GError *local_error = NULL;
+
+ /* Do not chain up. Camel's default method is only an example for
+ * subclasses to follow. Instead we mimic most of its logic here. */
+
+ url = camel_service_get_camel_url (service);
+ provider = camel_service_get_provider (service);
+
+ /* APOP is one case where a non-SASL mechanism name is passed, so
+ * don't bail if the CamelServiceAuthType struct comes back NULL. */
+ if (mechanism != NULL)
+ authtype = camel_sasl_authtype (mechanism);
+
+ /* If the SASL mechanism does not involve a user
+ * password, then it gets one shot to authenticate. */
+ if (authtype != NULL && !authtype->need_password) {
+ result = camel_service_authenticate_sync (
+ service, mechanism, cancellable, error);
+ if (result == CAMEL_AUTHENTICATION_REJECTED)
+ g_set_error (
+ error, CAMEL_SERVICE_ERROR,
+ CAMEL_SERVICE_ERROR_CANT_AUTHENTICATE,
+ _("%s authentication failed"), mechanism);
+ return (result == CAMEL_AUTHENTICATION_ACCEPTED);
+ }
+
+ /* Some SASL mechanisms can attempt to authenticate without a
+ * user password being provided (e.g. single-sign-on credentials),
+ * but can fall back to a user password. Handle that case next. */
+ if (mechanism != NULL) {
+ CamelProvider *provider;
+ CamelSasl *sasl;
+ const gchar *service_name;
+ gboolean success = FALSE;
+
+ provider = camel_service_get_provider (service);
+ service_name = provider->protocol;
+
+ /* XXX Would be nice if camel_sasl_try_empty_password_sync()
+ * returned CamelAuthenticationResult so it's easier to
+ * detect errors. */
+ sasl = camel_sasl_new (service_name, mechanism, service);
+ if (sasl != NULL) {
+ success = camel_sasl_try_empty_password_sync (
+ sasl, cancellable, &local_error);
+ g_object_unref (sasl);
+ }
+
+ if (success)
+ return TRUE;
+ }
+
+ /* Abort authentication if we got cancelled.
+ * Otherwise clear any errors and press on. */
+ if (g_error_matches (local_error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
+ return FALSE;
+
+ g_clear_error (&local_error);
+
+ password_flags = CAMEL_SESSION_PASSWORD_SECRET;
+
+retry:
+ password = camel_service_get_password (service);
+
+ if (password == NULL) {
+ gchar *prompt;
+ gchar *new_passwd;
+
+ prompt = camel_session_build_password_prompt (
+ provider->name, url->user, url->host);
+
+ new_passwd = camel_session_get_password (
+ session, service, prompt, "password",
+ password_flags, &local_error);
+ camel_service_set_password (service, new_passwd);
+ password = camel_service_get_password (service);
+ g_free (new_passwd);
+
+ g_free (prompt);
+
+ if (local_error != NULL) {
+ g_propagate_error (error, local_error);
+ return FALSE;
+ }
+
+ if (password == NULL) {
+ g_set_error (
+ error, CAMEL_SERVICE_ERROR,
+ CAMEL_SERVICE_ERROR_CANT_AUTHENTICATE,
+ _("No password was provided"));
+ return FALSE;
+ }
+ }
+
+ result = camel_service_authenticate_sync (
+ service, mechanism, cancellable, error);
+
+ if (result == CAMEL_AUTHENTICATION_REJECTED) {
+ password_flags |= CAMEL_SESSION_PASSWORD_REPROMPT;
+ camel_service_set_password (service, NULL);
+ goto retry;
+ }
+
+ return (result == CAMEL_AUTHENTICATION_ACCEPTED);
+}
+
static void
e_mail_session_class_init (EMailSessionClass *class)
{
@@ -1121,6 +1240,7 @@ e_mail_session_class_init (EMailSessionClass *class)
session_class->lookup_addressbook = mail_session_lookup_addressbook;
session_class->forward_to = mail_session_forward_to;
session_class->get_socks_proxy = mail_session_get_socks_proxy;
+ session_class->authenticate_sync = mail_session_authenticate_sync;
g_object_class_install_property (
object_class,
diff --git a/mail/em-subscription-editor.c b/mail/em-subscription-editor.c
index 09b74bcb3c..d92713f8be 100644
--- a/mail/em-subscription-editor.c
+++ b/mail/em-subscription-editor.c
@@ -48,6 +48,7 @@
((obj), EM_TYPE_SUBSCRIPTION_EDITOR, EMSubscriptionEditorPrivate))
typedef struct _AsyncContext AsyncContext;
+typedef struct _AsyncData AsyncData;
typedef struct _StoreData StoreData;
struct _EMSubscriptionEditorPrivate {
@@ -58,7 +59,9 @@ struct _EMSubscriptionEditorPrivate {
GtkWidget *entry; /* not referenced */
GtkWidget *notebook; /* not referenced */
GtkWidget *subscribe_button; /* not referenced */
+ GtkWidget *subscribe_arrow; /* not referenced */
GtkWidget *unsubscribe_button; /* not referenced */
+ GtkWidget *unsubscribe_arrow; /* not referenced */
GtkWidget *collapse_all_button; /* not referenced */
GtkWidget *expand_all_button; /* not referenced */
GtkWidget *refresh_button; /* not referenced */
@@ -76,12 +79,16 @@ struct _EMSubscriptionEditorPrivate {
guint timeout_id;
};
-struct _AsyncContext {
- EMSubscriptionEditor *editor;
+struct _AsyncData {
CamelFolderInfo *folder_info;
GtkTreeRowReference *reference;
};
+struct _AsyncContext {
+ EMSubscriptionEditor *editor;
+ GSList *async_datas; /* newly allocated AsyncData structures */
+};
+
struct _StoreData {
CamelStore *store;
GtkTreeView *tree_view;
@@ -110,10 +117,21 @@ enum {
G_DEFINE_TYPE (EMSubscriptionEditor, em_subscription_editor, GTK_TYPE_DIALOG)
static void
+async_data_free (AsyncData *data)
+{
+ g_return_if_fail (data != NULL);
+
+ gtk_tree_row_reference_free (data->reference);
+ g_free (data);
+}
+
+static void
async_context_free (AsyncContext *context)
{
g_object_unref (context->editor);
- gtk_tree_row_reference_free (context->reference);
+
+ g_slist_foreach (context->async_datas, (GFunc) async_data_free, NULL);
+ g_slist_free (context->async_datas);
g_slice_free (AsyncContext, context);
}
@@ -293,6 +311,7 @@ subscription_editor_subscribe_folder_done (CamelSubscribable *subscribable,
GtkTreeIter iter;
GdkWindow *window;
GError *error = NULL;
+ AsyncData *async_data;
camel_subscribable_subscribe_folder_finish (
subscribable, result, &error);
@@ -303,14 +322,41 @@ subscription_editor_subscribe_folder_done (CamelSubscribable *subscribable,
goto exit;
}
+ async_data = context->async_datas->data;
+ context->async_datas = g_slist_remove (context->async_datas, async_data);
+
/* XXX Do something smarter with errors. */
if (error == NULL)
- context->folder_info->flags |= CAMEL_FOLDER_SUBSCRIBED;
+ async_data->folder_info->flags |= CAMEL_FOLDER_SUBSCRIBED;
else {
g_warning ("%s", error->message);
g_error_free (error);
+ async_data_free (async_data);
+ goto exit;
+ }
+
+ /* Update the toggle renderer in the selected row. */
+ tree_model = gtk_tree_row_reference_get_model (async_data->reference);
+ path = gtk_tree_row_reference_get_path (async_data->reference);
+ gtk_tree_model_get_iter (tree_model, &iter, path);
+ gtk_tree_model_row_changed (tree_model, path, &iter);
+ gtk_tree_path_free (path);
+
+ async_data_free (async_data);
+
+ if (context->async_datas) {
+ /* continue with the next to subscribe */
+ async_data = context->async_datas->data;
+
+ camel_subscribable_subscribe_folder (
+ subscribable,
+ async_data->folder_info->full_name, G_PRIORITY_DEFAULT,
+ context->editor->priv->active->cancellable, (GAsyncReadyCallback)
+ subscription_editor_subscribe_folder_done, context);
+ return;
}
+ exit:
gtk_widget_set_sensitive (context->editor->priv->notebook, TRUE);
gtk_widget_set_sensitive (context->editor->priv->refresh_button, TRUE);
gtk_widget_set_sensitive (context->editor->priv->stop_button, FALSE);
@@ -323,15 +369,60 @@ subscription_editor_subscribe_folder_done (CamelSubscribable *subscribable,
selection = gtk_tree_view_get_selection (tree_view);
g_signal_emit_by_name (selection, "changed");
- /* Update the toggle renderer in the selected row. */
- tree_model = gtk_tree_row_reference_get_model (context->reference);
- path = gtk_tree_row_reference_get_path (context->reference);
- gtk_tree_model_get_iter (tree_model, &iter, path);
- gtk_tree_model_row_changed (tree_model, path, &iter);
- gtk_tree_path_free (path);
-
-exit:
async_context_free (context);
+
+ gtk_widget_grab_focus (GTK_WIDGET (tree_view));
+}
+
+
+static void
+subscription_editor_subscribe_many (EMSubscriptionEditor *editor,
+ GSList *async_datas)
+{
+ AsyncData *async_data;
+ AsyncContext *context;
+ CamelStore *active_store;
+ GdkCursor *cursor;
+ GdkWindow *window;
+
+ g_return_if_fail (editor != NULL);
+
+ if (!async_datas)
+ return;
+
+ async_data = async_datas->data;
+ g_return_if_fail (async_data != NULL);
+
+ /* Cancel any operation on this store still in progress. */
+ gtk_button_clicked (GTK_BUTTON (editor->priv->stop_button));
+
+ /* Start a new 'subscription' operation. */
+ editor->priv->active->cancellable = g_cancellable_new ();
+
+ gtk_widget_set_sensitive (editor->priv->notebook, FALSE);
+ gtk_widget_set_sensitive (editor->priv->subscribe_button, FALSE);
+ gtk_widget_set_sensitive (editor->priv->subscribe_arrow, FALSE);
+ gtk_widget_set_sensitive (editor->priv->unsubscribe_button, FALSE);
+ gtk_widget_set_sensitive (editor->priv->unsubscribe_arrow, FALSE);
+ gtk_widget_set_sensitive (editor->priv->refresh_button, FALSE);
+ gtk_widget_set_sensitive (editor->priv->stop_button, TRUE);
+
+ cursor = gdk_cursor_new (GDK_WATCH);
+ window = gtk_widget_get_window (GTK_WIDGET (editor));
+ gdk_window_set_cursor (window, cursor);
+ g_object_unref (cursor);
+
+ context = g_slice_new0 (AsyncContext);
+ context->editor = g_object_ref (editor);
+ context->async_datas = async_datas; /* takes ownership of the pointer */
+
+ active_store = editor->priv->active->store;
+
+ camel_subscribable_subscribe_folder (
+ CAMEL_SUBSCRIBABLE (active_store),
+ async_data->folder_info->full_name, G_PRIORITY_DEFAULT,
+ context->editor->priv->active->cancellable, (GAsyncReadyCallback)
+ subscription_editor_subscribe_folder_done, context);
}
static void
@@ -346,6 +437,7 @@ subscription_editor_unsubscribe_folder_done (CamelSubscribable *subscribable,
GtkTreeIter iter;
GdkWindow *window;
GError *error = NULL;
+ AsyncData *async_data;
camel_subscribable_unsubscribe_folder_finish (
subscribable, result, &error);
@@ -356,14 +448,40 @@ subscription_editor_unsubscribe_folder_done (CamelSubscribable *subscribable,
goto exit;
}
+ async_data = context->async_datas->data;
+ context->async_datas = g_slist_remove (context->async_datas, async_data);
+
/* XXX Do something smarter with errors. */
if (error == NULL)
- context->folder_info->flags &= ~CAMEL_FOLDER_SUBSCRIBED;
+ async_data->folder_info->flags &= ~CAMEL_FOLDER_SUBSCRIBED;
else {
g_warning ("%s", error->message);
g_error_free (error);
+ async_data_free (async_data);
+ goto exit;
}
+ /* Update the toggle renderer in the selected row. */
+ tree_model = gtk_tree_row_reference_get_model (async_data->reference);
+ path = gtk_tree_row_reference_get_path (async_data->reference);
+ gtk_tree_model_get_iter (tree_model, &iter, path);
+ gtk_tree_model_row_changed (tree_model, path, &iter);
+ gtk_tree_path_free (path);
+
+ async_data_free (async_data);
+
+ if (context->async_datas) {
+ /* continue with the next to unsubscribe */
+ async_data = context->async_datas->data;
+
+ camel_subscribable_unsubscribe_folder (
+ subscribable,
+ async_data->folder_info->full_name, G_PRIORITY_DEFAULT,
+ context->editor->priv->active->cancellable, (GAsyncReadyCallback)
+ subscription_editor_unsubscribe_folder_done, context);
+ return;
+ }
+ exit:
gtk_widget_set_sensitive (context->editor->priv->notebook, TRUE);
gtk_widget_set_sensitive (context->editor->priv->refresh_button, TRUE);
gtk_widget_set_sensitive (context->editor->priv->stop_button, FALSE);
@@ -376,39 +494,29 @@ subscription_editor_unsubscribe_folder_done (CamelSubscribable *subscribable,
selection = gtk_tree_view_get_selection (tree_view);
g_signal_emit_by_name (selection, "changed");
- /* Update the toggle renderer in the selected row. */
- tree_model = gtk_tree_row_reference_get_model (context->reference);
- path = gtk_tree_row_reference_get_path (context->reference);
- gtk_tree_model_get_iter (tree_model, &iter, path);
- gtk_tree_model_row_changed (tree_model, path, &iter);
- gtk_tree_path_free (path);
-
-exit:
async_context_free (context);
+
+ gtk_widget_grab_focus (GTK_WIDGET (tree_view));
}
+
static void
-subscription_editor_subscribe (EMSubscriptionEditor *editor)
+subscription_editor_unsubscribe_many (EMSubscriptionEditor *editor,
+ GSList *async_datas)
{
+ AsyncData *async_data;
+ AsyncContext *context;
CamelStore *active_store;
- CamelFolderInfo *folder_info;
- GtkTreeRowReference *reference;
- GtkTreeSelection *selection;
- GtkTreeModel *tree_model;
- GtkTreeView *tree_view;
- GtkTreePath *path;
- GtkTreeIter iter;
GdkCursor *cursor;
GdkWindow *window;
- AsyncContext *context;
- gboolean have_selection;
- tree_view = editor->priv->active->tree_view;
- selection = gtk_tree_view_get_selection (tree_view);
+ g_return_if_fail (editor != NULL);
- have_selection = gtk_tree_selection_get_selected (
- selection, &tree_model, &iter);
- g_return_if_fail (have_selection);
+ if (!async_datas)
+ return;
+
+ async_data = async_datas->data;
+ g_return_if_fail (async_data != NULL);
/* Cancel any operation on this store still in progress. */
gtk_button_clicked (GTK_BUTTON (editor->priv->stop_button));
@@ -418,7 +526,9 @@ subscription_editor_subscribe (EMSubscriptionEditor *editor)
gtk_widget_set_sensitive (editor->priv->notebook, FALSE);
gtk_widget_set_sensitive (editor->priv->subscribe_button, FALSE);
+ gtk_widget_set_sensitive (editor->priv->subscribe_arrow, FALSE);
gtk_widget_set_sensitive (editor->priv->unsubscribe_button, FALSE);
+ gtk_widget_set_sensitive (editor->priv->unsubscribe_arrow, FALSE);
gtk_widget_set_sensitive (editor->priv->refresh_button, FALSE);
gtk_widget_set_sensitive (editor->priv->stop_button, TRUE);
@@ -427,42 +537,252 @@ subscription_editor_subscribe (EMSubscriptionEditor *editor)
gdk_window_set_cursor (window, cursor);
g_object_unref (cursor);
- gtk_tree_model_get (
- tree_model, &iter, COL_FOLDER_INFO, &folder_info, -1);
-
- path = gtk_tree_model_get_path (tree_model, &iter);
- reference = gtk_tree_row_reference_new (tree_model, path);
- gtk_tree_path_free (path);
-
context = g_slice_new0 (AsyncContext);
context->editor = g_object_ref (editor);
- context->folder_info = folder_info;
- context->reference = reference;
+ context->async_datas = async_datas; /* takes ownership of the pointer */
active_store = editor->priv->active->store;
- camel_subscribable_subscribe_folder (
+ camel_subscribable_unsubscribe_folder (
CAMEL_SUBSCRIBABLE (active_store),
- folder_info->full_name, G_PRIORITY_DEFAULT,
+ async_data->folder_info->full_name, G_PRIORITY_DEFAULT,
editor->priv->active->cancellable, (GAsyncReadyCallback)
- subscription_editor_subscribe_folder_done, context);
+ subscription_editor_unsubscribe_folder_done, context);
+}
+
+static GtkWidget *
+subscription_editor_create_menu_item (const gchar *label,
+ gboolean sensitive,
+ GCallback activate_cb,
+ EMSubscriptionEditor *editor)
+{
+ GtkWidget *item;
+
+ item = gtk_menu_item_new_with_mnemonic (label);
+ gtk_widget_set_sensitive (item, sensitive);
+
+ gtk_widget_show (item);
+
+ g_signal_connect_swapped (item, "activate", activate_cb, editor);
+
+ return item;
}
static void
-subscription_editor_unsubscribe (EMSubscriptionEditor *editor)
+position_below_widget_cb (GtkMenu *menu,
+ gint *x,
+ gint *y,
+ gboolean *push_in,
+ gpointer under_widget)
{
- CamelStore *active_store;
- CamelFolderInfo *folder_info;
+ GtkRequisition menu_requisition;
+ GtkTextDirection direction;
+ GtkAllocation allocation;
+ GdkRectangle monitor;
+ GdkScreen *screen;
+ GdkWindow *window;
+ GtkWidget *widget;
+ gint monitor_num;
+
+ widget = under_widget;
+ gtk_widget_get_preferred_size (GTK_WIDGET (menu), &menu_requisition, NULL);
+
+ window = gtk_widget_get_parent_window (widget);
+ screen = gtk_widget_get_screen (GTK_WIDGET (menu));
+ monitor_num = gdk_screen_get_monitor_at_window (screen, window);
+ if (monitor_num < 0)
+ monitor_num = 0;
+ gdk_screen_get_monitor_geometry (screen, monitor_num, &monitor);
+
+ gtk_widget_get_allocation (widget, &allocation);
+
+ gdk_window_get_origin (window, x, y);
+ *x += allocation.x;
+ *y += allocation.y + 2 + gtk_widget_get_allocated_height (under_widget);
+
+ direction = gtk_widget_get_direction (widget);
+ if (direction == GTK_TEXT_DIR_LTR)
+ *x += MAX (allocation.width - menu_requisition.width, 0);
+ else if (menu_requisition.width > allocation.width)
+ *x -= menu_requisition.width - allocation.width;
+
+ *push_in = FALSE;
+}
+
+static AsyncData *
+subscription_editor_async_data_from_iter (GtkTreeView *tree_view,
+ GtkTreeModel *model,
+ GtkTreeIter *iter,
+ gboolean *is_expanded)
+{
+ AsyncData *data;
+ CamelFolderInfo *folder_info = NULL;
GtkTreeRowReference *reference;
+ GtkTreePath *path;
+
+ g_return_val_if_fail (tree_view != NULL, NULL);
+ g_return_val_if_fail (model != NULL, NULL);
+ g_return_val_if_fail (iter != NULL, NULL);
+
+ gtk_tree_model_get (
+ model, iter, COL_FOLDER_INFO, &folder_info, -1);
+
+ if (!FOLDER_CAN_SELECT (folder_info))
+ return NULL;
+
+ path = gtk_tree_model_get_path (model, iter);
+ reference = gtk_tree_row_reference_new (model, path);
+ if (is_expanded)
+ *is_expanded = gtk_tree_view_row_expanded (tree_view, path);
+ gtk_tree_path_free (path);
+
+ data = g_new0 (AsyncData, 1);
+ data->folder_info = folder_info;
+ data->reference = reference;
+
+ return data;
+}
+
+typedef enum {
+ PICK_ALL,
+ PICK_SUBSCRIBED,
+ PICK_UNSUBSCRIBED
+} EPickMode;
+
+static gboolean
+can_pick_folder_info (CamelFolderInfo *fi, EPickMode mode)
+{
+ if (!FOLDER_CAN_SELECT (fi))
+ return FALSE;
+
+ if (mode == PICK_ALL)
+ return TRUE;
+
+ return (FOLDER_SUBSCRIBED (fi) ? 1 : 0) == (mode == PICK_SUBSCRIBED ? 1 : 0);
+}
+
+struct PickAllData {
+ GtkTreeView *tree_view;
+ EPickMode mode;
+ GHashTable *skip_folder_infos;
+ GSList *async_datas;
+};
+
+static gboolean
+pick_all_cb (GtkTreeModel *model,
+ GtkTreePath *path,
+ GtkTreeIter *iter,
+ gpointer user_data)
+{
+ struct PickAllData *data = user_data;
+ AsyncData *async_data;
+
+ g_return_val_if_fail (model != NULL, TRUE);
+ g_return_val_if_fail (data != NULL, TRUE);
+ g_return_val_if_fail (data->tree_view != NULL, TRUE);
+
+ async_data = subscription_editor_async_data_from_iter (data->tree_view, model, iter, NULL);
+ if (!async_data)
+ return FALSE;
+
+ if (can_pick_folder_info (async_data->folder_info, data->mode) &&
+ (data->skip_folder_infos == NULL ||
+ !g_hash_table_lookup_extended (data->skip_folder_infos, async_data->folder_info, NULL, NULL))) {
+ data->async_datas = g_slist_prepend (data->async_datas, async_data);
+ } else
+ async_data_free (async_data);
+
+ return FALSE;
+}
+
+/* skip_folder_infos contains CamelFolderInfo-s to skip;
+ these should come from the tree view; can be NULL
+ to include everything.
+*/
+static GSList *
+subscription_editor_pick_all (EMSubscriptionEditor *editor,
+ EPickMode mode,
+ GHashTable *skip_folder_infos)
+{
+ GtkTreeView *tree_view;
+ GtkTreeModel *tree_model;
+ struct PickAllData data;
+
+ tree_view = editor->priv->active->tree_view;
+ tree_model = gtk_tree_view_get_model (tree_view);
+
+ data.tree_view = tree_view;
+ data.mode = mode;
+ data.skip_folder_infos = skip_folder_infos;
+ data.async_datas = NULL;
+
+ gtk_tree_model_foreach (tree_model, pick_all_cb, &data);
+
+ return data.async_datas;
+}
+
+static GSList *
+subscription_editor_pick_shown (EMSubscriptionEditor *editor,
+ EPickMode mode)
+{
+ GtkTreeView *tree_view;
+ GtkTreeModel *tree_model;
+ GtkTreeIter iter, iter2;
+ GSList *async_datas = NULL;
+ gboolean found;
+
+ tree_view = editor->priv->active->tree_view;
+ tree_model = gtk_tree_view_get_model (tree_view);
+
+ if (!gtk_tree_model_get_iter_first (tree_model, &iter))
+ return NULL;
+
+ found = TRUE;
+ while (found) {
+ AsyncData *async_data;
+ gboolean is_expanded = FALSE;
+
+ found = FALSE;
+ async_data = subscription_editor_async_data_from_iter (tree_view, tree_model, &iter, &is_expanded);
+
+ if (async_data && can_pick_folder_info (async_data->folder_info, mode)) {
+ async_datas = g_slist_prepend (async_datas, async_data);
+ } else if (async_data) {
+ async_data_free (async_data);
+ }
+
+ if (is_expanded && gtk_tree_model_iter_children (tree_model, &iter2, &iter)) {
+ iter = iter2;
+ found = TRUE;
+ } else {
+ iter2 = iter;
+ if (gtk_tree_model_iter_next (tree_model, &iter2)) {
+ iter = iter2;
+ found = TRUE;
+ } else {
+ while (found = gtk_tree_model_iter_parent (tree_model, &iter2, &iter), found) {
+ iter = iter2;
+ if (gtk_tree_model_iter_next (tree_model, &iter2)) {
+ iter = iter2;
+ break;
+ }
+ }
+ }
+ }
+ }
+
+ return async_datas;
+}
+
+static void
+subscription_editor_subscribe (EMSubscriptionEditor *editor)
+{
GtkTreeSelection *selection;
GtkTreeModel *tree_model;
GtkTreeView *tree_view;
- GtkTreePath *path;
GtkTreeIter iter;
- GdkCursor *cursor;
- GdkWindow *window;
- AsyncContext *context;
gboolean have_selection;
+ GSList *async_datas;
tree_view = editor->priv->active->tree_view;
selection = gtk_tree_view_get_selection (tree_view);
@@ -471,42 +791,167 @@ subscription_editor_unsubscribe (EMSubscriptionEditor *editor)
selection, &tree_model, &iter);
g_return_if_fail (have_selection);
- /* Cancel any operation on this store still in progress. */
- gtk_button_clicked (GTK_BUTTON (editor->priv->stop_button));
+ async_datas = g_slist_append (NULL,
+ subscription_editor_async_data_from_iter (tree_view, tree_model, &iter, NULL));
- /* Start a new 'unsubscription' operation. */
- editor->priv->active->cancellable = g_cancellable_new ();
+ subscription_editor_subscribe_many (editor, async_datas);
+}
- gtk_widget_set_sensitive (editor->priv->notebook, FALSE);
- gtk_widget_set_sensitive (editor->priv->subscribe_button, FALSE);
- gtk_widget_set_sensitive (editor->priv->unsubscribe_button, FALSE);
- gtk_widget_set_sensitive (editor->priv->refresh_button, FALSE);
- gtk_widget_set_sensitive (editor->priv->stop_button, TRUE);
+static void
+subscription_editor_subscribe_shown (EMSubscriptionEditor *editor)
+{
+ subscription_editor_subscribe_many (editor,
+ subscription_editor_pick_shown (editor, PICK_UNSUBSCRIBED));
+}
- cursor = gdk_cursor_new (GDK_WATCH);
- window = gtk_widget_get_window (GTK_WIDGET (editor));
- gdk_window_set_cursor (window, cursor);
- g_object_unref (cursor);
+static void
+subscription_editor_subscribe_all (EMSubscriptionEditor *editor)
+{
+ subscription_editor_subscribe_many (editor,
+ subscription_editor_pick_all (editor, PICK_UNSUBSCRIBED, NULL));
+}
- gtk_tree_model_get (
- tree_model, &iter, COL_FOLDER_INFO, &folder_info, -1);
+static void
+subscription_editor_subscribe_popup_cb (EMSubscriptionEditor *editor)
+{
+ GtkWidget *menu;
+ GtkTreeIter iter;
+ gboolean tree_filled;
+
+ tree_filled = editor->priv->active &&
+ gtk_tree_model_get_iter_first (
+ editor->priv->active->filtered_view
+ ? editor->priv->active->list_store
+ : editor->priv->active->tree_store,
+ &iter);
+
+ menu = gtk_menu_new ();
+
+ gtk_menu_shell_append (GTK_MENU_SHELL (menu),
+ subscription_editor_create_menu_item (
+ _("_Subscribe"),
+ gtk_widget_get_sensitive (editor->priv->subscribe_button),
+ G_CALLBACK (subscription_editor_subscribe),
+ editor));
+
+ gtk_menu_shell_append (GTK_MENU_SHELL (menu),
+ subscription_editor_create_menu_item (
+ _("Su_bscribe to shown"),
+ tree_filled,
+ G_CALLBACK (subscription_editor_subscribe_shown),
+ editor));
+
+ gtk_menu_shell_append (GTK_MENU_SHELL (menu),
+ subscription_editor_create_menu_item (
+ _("Subscribe to _all"),
+ tree_filled,
+ G_CALLBACK (subscription_editor_subscribe_all),
+ editor));
+
+ gtk_menu_popup (
+ GTK_MENU (menu), NULL, NULL, position_below_widget_cb, editor->priv->subscribe_button,
+ 0, gtk_get_current_event_time ());
+}
- path = gtk_tree_model_get_path (tree_model, &iter);
- reference = gtk_tree_row_reference_new (tree_model, path);
- gtk_tree_path_free (path);
+static void
+subscription_editor_unsubscribe_hidden (EMSubscriptionEditor *editor)
+{
+ GSList *all_shown, *ll;
+ GHashTable *skip_shown;
- context = g_slice_new0 (AsyncContext);
- context->editor = g_object_ref (editor);
- context->folder_info = folder_info;
- context->reference = reference;
+ all_shown = subscription_editor_pick_shown (editor, PICK_ALL);
+ g_return_if_fail (all_shown != NULL);
- active_store = editor->priv->active->store;
+ skip_shown = g_hash_table_new (g_direct_hash, g_direct_equal);
- camel_subscribable_unsubscribe_folder (
- CAMEL_SUBSCRIBABLE (active_store),
- folder_info->full_name, G_PRIORITY_DEFAULT,
- editor->priv->active->cancellable, (GAsyncReadyCallback)
- subscription_editor_unsubscribe_folder_done, context);
+ for (ll = all_shown; ll; ll = ll->next) {
+ AsyncData *data = ll->data;
+
+ if (!data)
+ continue;
+
+ g_hash_table_insert (skip_shown, data->folder_info, GINT_TO_POINTER (1));
+ async_data_free (data);
+ }
+
+ g_slist_free (all_shown);
+
+ subscription_editor_unsubscribe_many (editor,
+ subscription_editor_pick_all (editor, PICK_SUBSCRIBED, skip_shown));
+
+ g_hash_table_destroy (skip_shown);
+}
+
+static void
+subscription_editor_unsubscribe_all (EMSubscriptionEditor *editor)
+{
+ subscription_editor_unsubscribe_many (editor,
+ subscription_editor_pick_all (editor, PICK_SUBSCRIBED, NULL));
+}
+
+static void
+subscription_editor_unsubscribe (EMSubscriptionEditor *editor)
+{
+ GtkTreeSelection *selection;
+ GtkTreeModel *tree_model;
+ GtkTreeView *tree_view;
+ GtkTreeIter iter;
+ gboolean have_selection;
+ GSList *async_datas;
+
+ tree_view = editor->priv->active->tree_view;
+ selection = gtk_tree_view_get_selection (tree_view);
+
+ have_selection = gtk_tree_selection_get_selected (
+ selection, &tree_model, &iter);
+ g_return_if_fail (have_selection);
+
+ async_datas = g_slist_append (NULL,
+ subscription_editor_async_data_from_iter (tree_view, tree_model, &iter, NULL));
+
+ subscription_editor_unsubscribe_many (editor, async_datas);
+}
+
+static void
+subscription_editor_unsubscribe_popup_cb (EMSubscriptionEditor *editor)
+{
+ GtkWidget *menu;
+ GtkTreeIter iter;
+ gboolean tree_filled;
+
+ tree_filled = editor->priv->active &&
+ gtk_tree_model_get_iter_first (
+ editor->priv->active->filtered_view
+ ? editor->priv->active->list_store
+ : editor->priv->active->tree_store,
+ &iter);
+
+ menu = gtk_menu_new ();
+
+ gtk_menu_shell_append (GTK_MENU_SHELL (menu),
+ subscription_editor_create_menu_item (
+ _("_Unsubscribe"),
+ gtk_widget_get_sensitive (editor->priv->unsubscribe_button),
+ G_CALLBACK (subscription_editor_unsubscribe),
+ editor));
+
+ gtk_menu_shell_append (GTK_MENU_SHELL (menu),
+ subscription_editor_create_menu_item (
+ _("Unsu_bscribe from hidden"),
+ tree_filled,
+ G_CALLBACK (subscription_editor_unsubscribe_hidden),
+ editor));
+
+ gtk_menu_shell_append (GTK_MENU_SHELL (menu),
+ subscription_editor_create_menu_item (
+ _("Unsubscribe from _all"),
+ tree_filled,
+ G_CALLBACK (subscription_editor_unsubscribe_all),
+ editor));
+
+ gtk_menu_popup (
+ GTK_MENU (menu), NULL, NULL, position_below_widget_cb, editor->priv->unsubscribe_button,
+ 0, gtk_get_current_event_time ());
}
static void
@@ -535,7 +980,9 @@ subscription_editor_refresh (EMSubscriptionEditor *editor)
gtk_widget_set_sensitive (editor->priv->notebook, FALSE);
gtk_widget_set_sensitive (editor->priv->subscribe_button, FALSE);
+ gtk_widget_set_sensitive (editor->priv->subscribe_arrow, FALSE);
gtk_widget_set_sensitive (editor->priv->unsubscribe_button, FALSE);
+ gtk_widget_set_sensitive (editor->priv->unsubscribe_arrow, FALSE);
gtk_widget_set_sensitive (editor->priv->refresh_button, FALSE);
gtk_widget_set_sensitive (editor->priv->stop_button, TRUE);
@@ -567,9 +1014,12 @@ subscription_editor_stop (EMSubscriptionEditor *editor)
gtk_widget_set_sensitive (editor->priv->notebook, TRUE);
gtk_widget_set_sensitive (editor->priv->subscribe_button, TRUE);
+ gtk_widget_set_sensitive (editor->priv->subscribe_arrow, TRUE);
gtk_widget_set_sensitive (editor->priv->unsubscribe_button, TRUE);
+ gtk_widget_set_sensitive (editor->priv->unsubscribe_arrow, TRUE);
gtk_widget_set_sensitive (editor->priv->refresh_button, TRUE);
gtk_widget_set_sensitive (editor->priv->stop_button, FALSE);
+ gtk_widget_grab_focus (GTK_WIDGET (editor->priv->active->tree_view));
window = gtk_widget_get_window (GTK_WIDGET (editor));
gdk_window_set_cursor (window, NULL);
@@ -811,8 +1261,11 @@ subscription_editor_selection_changed_cb (GtkTreeSelection *selection,
gtk_widget_set_sensitive (
editor->priv->subscribe_button, FALSE);
gtk_widget_set_sensitive (
- editor->priv->subscribe_button, FALSE);
+ editor->priv->unsubscribe_button, FALSE);
}
+
+ gtk_widget_set_sensitive (editor->priv->subscribe_arrow, TRUE);
+ gtk_widget_set_sensitive (editor->priv->unsubscribe_arrow, TRUE);
}
static void
@@ -1264,20 +1717,25 @@ em_subscription_editor_init (EMSubscriptionEditor *editor)
G_BINDING_BIDIRECTIONAL |
G_BINDING_SYNC_CREATE);
- widget = gtk_vbutton_box_new ();
+ widget = gtk_button_box_new (GTK_ORIENTATION_VERTICAL);
gtk_box_set_spacing (GTK_BOX (widget), 6);
gtk_button_box_set_layout (
GTK_BUTTON_BOX (widget), GTK_BUTTONBOX_START);
gtk_box_pack_start (GTK_BOX (container), widget, FALSE, TRUE, 0);
gtk_widget_show (widget);
+ container = box = widget;
+
+ widget = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 2);
+ gtk_box_pack_start (GTK_BOX (container), widget, FALSE, TRUE, 0);
+ gtk_widget_show (widget);
+
container = widget;
tooltip = _("Subscribe to the selected folder");
widget = gtk_button_new_with_mnemonic (_("Su_bscribe"));
gtk_widget_set_sensitive (widget, FALSE);
gtk_widget_set_tooltip_text (widget, tooltip);
- gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
editor->priv->subscribe_button = widget;
gtk_widget_show (widget);
@@ -1285,11 +1743,35 @@ em_subscription_editor_init (EMSubscriptionEditor *editor)
widget, "clicked",
G_CALLBACK (subscription_editor_subscribe), editor);
+ widget = gtk_button_new ();
+ gtk_button_set_image (GTK_BUTTON (widget), gtk_arrow_new (GTK_ARROW_DOWN, GTK_SHADOW_NONE));
+ editor->priv->subscribe_arrow = widget;
+ gtk_widget_show (widget);
+
+ g_signal_connect_swapped (
+ widget, "clicked",
+ G_CALLBACK (subscription_editor_subscribe_popup_cb), editor);
+
+ if (gtk_widget_get_direction (container) == GTK_TEXT_DIR_LTR) {
+ gtk_box_pack_start (GTK_BOX (container), editor->priv->subscribe_button, TRUE, TRUE, 0);
+ gtk_box_pack_start (GTK_BOX (container), editor->priv->subscribe_arrow, FALSE, FALSE, 0);
+ } else {
+ gtk_box_pack_start (GTK_BOX (container), editor->priv->subscribe_arrow, FALSE, FALSE, 0);
+ gtk_box_pack_start (GTK_BOX (container), editor->priv->subscribe_button, TRUE, TRUE, 0);
+ }
+
+ container = box;
+
+ widget = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 2);
+ gtk_box_pack_start (GTK_BOX (container), widget, FALSE, TRUE, 0);
+ gtk_widget_show (widget);
+
+ container = widget;
+
tooltip = _("Unsubscribe from the selected folder");
widget = gtk_button_new_with_mnemonic (_("_Unsubscribe"));
gtk_widget_set_sensitive (widget, FALSE);
gtk_widget_set_tooltip_text (widget, tooltip);
- gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0);
editor->priv->unsubscribe_button = widget;
gtk_widget_show (widget);
@@ -1297,6 +1779,25 @@ em_subscription_editor_init (EMSubscriptionEditor *editor)
widget, "clicked",
G_CALLBACK (subscription_editor_unsubscribe), editor);
+ widget = gtk_button_new ();
+ gtk_button_set_image (GTK_BUTTON (widget), gtk_arrow_new (GTK_ARROW_DOWN, GTK_SHADOW_NONE));
+ editor->priv->unsubscribe_arrow = widget;
+ gtk_widget_show (widget);
+
+ g_signal_connect_swapped (
+ widget, "clicked",
+ G_CALLBACK (subscription_editor_unsubscribe_popup_cb), editor);
+
+ if (gtk_widget_get_direction (container) == GTK_TEXT_DIR_LTR) {
+ gtk_box_pack_start (GTK_BOX (container), editor->priv->unsubscribe_button, TRUE, TRUE, 0);
+ gtk_box_pack_start (GTK_BOX (container), editor->priv->unsubscribe_arrow, FALSE, FALSE, 0);
+ } else {
+ gtk_box_pack_start (GTK_BOX (container), editor->priv->unsubscribe_arrow, FALSE, FALSE, 0);
+ gtk_box_pack_start (GTK_BOX (container), editor->priv->unsubscribe_button, TRUE, TRUE, 0);
+ }
+
+ container = box;
+
tooltip = _("Collapse all folders");
widget = gtk_button_new_with_mnemonic (_("C_ollapse All"));
gtk_widget_set_tooltip_text (widget, tooltip);