/* * * 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 * * * Authors: * Chenthill Palanisamy * * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) * */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include #include #include #include "e-util/e-error.h" #include #include #include #define PRIMARY_TEXT \ N_("Also mark messages in subfolders?") #define SECONDARY_TEXT \ N_("Do you want to mark messages as read in the current folder " \ "only, or in the current folder as well as all subfolders?") gboolean e_plugin_ui_init (GtkUIManager *ui_manager, EShellView *shell_view); static void mar_got_folder (gchar *uri, CamelFolder *folder, gpointer data); static void mar_all_sub_folders (CamelStore *store, CamelFolderInfo *fi, CamelException *ex); static void button_clicked_cb (GtkButton *button, GtkDialog *dialog) { gpointer response; response = g_object_get_data (G_OBJECT (button), "response"); gtk_dialog_response (dialog, GPOINTER_TO_INT (response)); } static void box_mapped_cb (GtkWidget *box, GtkWidget *label) { GtkRequisition requisition; /* In order to get decent line wrapping we need to wait until the * box containing the buttons is mapped, and then resize the label * to the same width as the box. */ gtk_widget_size_request (box, &requisition); gtk_widget_set_size_request (label, requisition.width, -1); } static gint prompt_user (void) { GtkWidget *container; GtkWidget *dialog; GtkWidget *label1; GtkWidget *label2; GtkWidget *table; GtkWidget *widget; GtkWidget *vbox; gchar *markup; gint response; dialog = gtk_dialog_new (); gtk_widget_hide (GTK_DIALOG (dialog)->action_area); gtk_dialog_set_has_separator (GTK_DIALOG (dialog), FALSE); gtk_window_set_title (GTK_WINDOW (dialog), ""); g_signal_connect ( dialog, "map", G_CALLBACK (gtk_widget_queue_resize), NULL); gtk_container_set_border_width (GTK_CONTAINER (dialog), 12); vbox = GTK_DIALOG (dialog)->vbox; /* Table */ widget = gtk_table_new (3, 2, FALSE); gtk_table_set_row_spacings (GTK_TABLE (widget), 12); gtk_table_set_col_spacings (GTK_TABLE (widget), 12); gtk_box_pack_start (GTK_BOX (vbox), widget, TRUE, TRUE, 0); gtk_widget_show (widget); table = widget; /* Question Icon */ widget = gtk_image_new_from_stock ( GTK_STOCK_DIALOG_QUESTION, GTK_ICON_SIZE_DIALOG); gtk_misc_set_alignment (GTK_MISC (widget), 0.5, 0.0); gtk_table_attach ( GTK_TABLE (table), widget, 0, 1, 0, 3, 0, GTK_EXPAND | GTK_FILL, 0, 0); gtk_widget_show (widget); /* Primary Text */ markup = g_markup_printf_escaped ( "%s", gettext (PRIMARY_TEXT)); widget = gtk_label_new (markup); gtk_label_set_line_wrap (GTK_LABEL (widget), TRUE); gtk_label_set_use_markup (GTK_LABEL (widget), TRUE); gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.0); gtk_table_attach ( GTK_TABLE (table), widget, 1, 2, 0, 1, 0, 0, 0, 0); gtk_widget_show (widget); g_free (markup); label1 = widget; /* Secondary Text */ widget = gtk_label_new (gettext (SECONDARY_TEXT)); gtk_label_set_line_wrap (GTK_LABEL (widget), TRUE); gtk_misc_set_alignment (GTK_MISC (widget), 0.0, 0.0); gtk_table_attach ( GTK_TABLE (table), widget, 1, 2, 1, 2, 0, GTK_EXPAND | GTK_FILL, 0, 0); gtk_widget_show (widget); label2 = widget; /* Action Area */ widget = gtk_hbox_new (FALSE, 6); g_signal_connect ( widget, "map", G_CALLBACK (box_mapped_cb), label1); g_signal_connect ( widget, "map", G_CALLBACK (box_mapped_cb), label2); gtk_table_attach ( GTK_TABLE (table), widget, 1, 2, 2, 3, GTK_EXPAND | GTK_FILL, 0, 0, 0); gtk_widget_show (widget); container = widget; /* Cancel button */ widget = gtk_button_new_from_stock (GTK_STOCK_CANCEL); g_object_set_data ( G_OBJECT (widget), "response", GINT_TO_POINTER (GTK_RESPONSE_CANCEL)); g_signal_connect ( widget, "clicked", G_CALLBACK (button_clicked_cb), dialog); gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0); gtk_widget_show (widget); /* To Translators: It's a response button caption on a question "Do you want to mark messages as read in the current folder only, or in the current folder as well as all subfolders?" */ widget = gtk_button_new_with_mnemonic ( _("In Current Folder and _Subfolders")); g_object_set_data ( G_OBJECT (widget), "response", GINT_TO_POINTER (GTK_RESPONSE_YES)); g_signal_connect ( widget, "clicked", G_CALLBACK (button_clicked_cb), dialog); gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0); gtk_widget_show (widget); /* To Translators: It's a response button caption on a question "Do you want to mark messages as read in the current folder only, or in the current folder as well as all subfolders?" */ widget = gtk_button_new_with_mnemonic ( _("In Current _Folder Only")); g_object_set_data ( G_OBJECT (widget), "response", GINT_TO_POINTER (GTK_RESPONSE_NO)); g_signal_connect ( widget, "clicked", G_CALLBACK (button_clicked_cb), dialog); gtk_box_pack_start (GTK_BOX (container), widget, FALSE, FALSE, 0); gtk_widget_show (widget); response = gtk_dialog_run (GTK_DIALOG (dialog)); gtk_widget_destroy (dialog); return response; } static void mark_all_as_read (CamelFolder *folder) { GPtrArray *uids; gint i; uids = camel_folder_get_uids (folder); camel_folder_freeze(folder); for (i=0;ilen;i++) camel_folder_set_message_flags(folder, uids->pdata[i], CAMEL_MESSAGE_SEEN, CAMEL_MESSAGE_SEEN); camel_folder_thaw(folder); camel_folder_free_uids (folder, uids); } static void mar_got_folder (gchar *uri, CamelFolder *folder, gpointer data) { CamelFolderInfo *info; CamelStore *store; CamelException ex; gint response; guint32 flags = CAMEL_STORE_FOLDER_INFO_RECURSIVE | CAMEL_STORE_FOLDER_INFO_FAST; /* FIXME we have to disable the menu item */ if (!folder) return; camel_exception_init (&ex); store = folder->parent_store; info = camel_store_get_folder_info (store, folder->full_name, flags, &ex); if (camel_exception_is_set (&ex)) goto out; if (info && (info->child || info->next)) response = prompt_user (); else response = GTK_RESPONSE_NO; if (response == GTK_RESPONSE_NO) mark_all_as_read (folder); else if (response == GTK_RESPONSE_YES) mar_all_sub_folders (store, info, &ex); out: camel_store_free_folder_info(store, info); } static void mar_all_sub_folders (CamelStore *store, CamelFolderInfo *fi, CamelException *ex) { while (fi) { CamelFolder *folder; if (fi->child) { mar_all_sub_folders (store, fi->child, ex); if (camel_exception_is_set (ex)) return; } if (!(folder = camel_store_get_folder (store, fi->full_name, 0, ex))) return; if (!CAMEL_IS_VEE_FOLDER (folder)) { mark_all_as_read (folder); } fi = fi->next; } } static void has_unread_mail (GtkTreeModel *model, GtkTreeIter *parent, gboolean is_root, gboolean *has_unread, gboolean *applicable) { guint unread = 0; GtkTreeIter iter, child; g_return_if_fail (model != NULL); g_return_if_fail (parent != NULL); g_return_if_fail (has_unread != NULL); g_return_if_fail (applicable != NULL); if (is_root) { gboolean is_draft = FALSE, is_store = FALSE; gtk_tree_model_get (model, parent, COL_UINT_UNREAD, &unread, COL_BOOL_IS_STORE, &is_store, COL_BOOL_IS_DRAFT, &is_draft, -1); *has_unread = *has_unread || (unread > 0 && unread != ~((guint)0)); *applicable = !is_store && !is_draft; if (*has_unread) return; if (!gtk_tree_model_iter_children (model, &iter, parent)) return; } else { iter = *parent; } do { gtk_tree_model_get (model, &iter, COL_UINT_UNREAD, &unread, -1); *has_unread = *has_unread || (unread > 0 && unread != ~((guint)0)); if (*has_unread) break; if (gtk_tree_model_iter_children (model, &child, &iter)) has_unread_mail (model, &child, FALSE, has_unread, applicable); } while (gtk_tree_model_iter_next (model, &iter) && !*has_unread); } static void action_mail_mark_read_recursive_cb (GtkAction *action, EShellView *shell_view) { EShellSidebar *shell_sidebar; EMFolderTree *folder_tree; gchar *folder_uri; shell_sidebar = e_shell_view_get_shell_sidebar (shell_view); g_object_get (shell_sidebar, "folder-tree", &folder_tree, NULL); folder_uri = em_folder_tree_get_selected_uri (folder_tree); g_return_if_fail (folder_uri != NULL); mail_get_folder ( folder_uri, 0, mar_got_folder, NULL, mail_msg_unordered_push); g_free (folder_uri); g_object_unref (folder_tree); } static GtkActionEntry entries[] = { { "mail-mark-read-recursive", "mail-mark-read", N_("Mark Me_ssages as Read"), NULL, NULL, /* XXX Add a tooltip! */ G_CALLBACK (action_mail_mark_read_recursive_cb) } }; static void update_actions_cb (EShellView *shell_view, gpointer user_data) { GtkActionGroup *action_group; EShellWindow *shell_window; GtkAction *action; EShellSidebar *shell_sidebar; EMFolderTree *folder_tree; gchar *folder_uri; gboolean has_unread = FALSE, applicable = FALSE; g_return_if_fail (E_IS_SHELL_VIEW (shell_view)); shell_sidebar = e_shell_view_get_shell_sidebar (shell_view); g_object_get (shell_sidebar, "folder-tree", &folder_tree, NULL); folder_uri = em_folder_tree_get_selected_uri (folder_tree); if (folder_uri != NULL) { EMFolderTreeModel *model; model = em_folder_tree_model_get_default (); if (model) { GtkTreeRowReference *reference = em_folder_tree_model_lookup_uri (model, folder_uri); if (reference != NULL) { GtkTreePath *path = gtk_tree_row_reference_get_path (reference); GtkTreeIter iter; gtk_tree_model_get_iter (GTK_TREE_MODEL (model), &iter, path); has_unread_mail (GTK_TREE_MODEL (model), &iter, TRUE, &has_unread, &applicable); gtk_tree_path_free (path); } } } shell_window = e_shell_view_get_shell_window (shell_view); action_group = e_shell_window_get_action_group (shell_window, "mail"); action = gtk_action_group_get_action (action_group, entries[0].name); g_return_if_fail (action != NULL); gtk_action_set_sensitive (action, has_unread && applicable); g_free (folder_uri); g_object_unref (folder_tree); } gboolean e_plugin_ui_init (GtkUIManager *ui_manager, EShellView *shell_view) { EShellWindow *shell_window; GtkActionGroup *action_group; shell_window = e_shell_view_get_shell_window (shell_view); action_group = e_shell_window_get_action_group (shell_window, "mail"); /* Add actions to the "mail" action group. */ gtk_action_group_add_actions ( action_group, entries, G_N_ELEMENTS (entries), shell_view); g_signal_connect (shell_view, "update-actions", G_CALLBACK (update_actions_cb), NULL); return TRUE; }