*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
+ * the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
*/
#include "defs.h"
#include "sourcewindow.h"
#include "prefs_common.h"
#include "prefs_summary_column.h"
+#include "prefs_summary_open.h"
#include "prefs_filtering.h"
#include "account.h"
#include "compose.h"
#include "folderutils.h"
#include "quicksearch.h"
#include "partial_download.h"
+#include "tags.h"
#include "timing.h"
#include "gedit-print.h"
#include "log.h"
+#include "edittags.h"
#include "manual.h"
#define SUMMARY_COL_MARK_WIDTH 10
static GdkBitmap *forwardedxpmmask;
static GdkPixmap *ignorethreadxpm;
static GdkBitmap *ignorethreadxpmmask;
+static GdkPixmap *watchthreadxpm;
+static GdkBitmap *watchthreadxpmmask;
static GdkPixmap *lockedxpm;
static GdkBitmap *lockedxpmmask;
static GdkPixmap *spamxpm;
static void summary_set_column_titles (SummaryView *summaryview);
static void summary_set_ctree_from_list (SummaryView *summaryview,
GSList *mlist);
-static void summary_set_header (SummaryView *summaryview,
+static inline void summary_set_header (SummaryView *summaryview,
gchar *text[],
MsgInfo *msginfo);
static void summary_display_msg (SummaryView *summaryview,
static void summary_set_row_marks (SummaryView *summaryview,
GtkCTreeNode *row);
+static gboolean summary_set_row_tag (SummaryView *summaryview,
+ GtkCTreeNode *row,
+ gboolean refresh,
+ gboolean set,
+ gint id);
/* message handling */
static void summary_mark_row (SummaryView *summaryview,
GtkCTreeNode *row);
gpointer data);
static void summary_colorlabel_menu_create(SummaryView *summaryview,
gboolean refresh);
+static void summary_tags_menu_item_activate_cb
+ (GtkWidget *widget,
+ gpointer data);
+static void summary_tags_menu_item_activate_item_cb
+ (GtkMenuItem *label_menu_item,
+ gpointer data);
+static void summary_tags_menu_create(SummaryView *summaryview,
+ gboolean refresh);
static GtkWidget *summary_ctree_create (SummaryView *summaryview);
static gint summary_toggle_pressed (GtkWidget *eventbox,
GdkEventButton *event,
SummaryView *summaryview);
+#ifdef MAEMO
+static void summary_toggle_multiple_pressed
+ (GtkWidget *widget,
+ SummaryView *summaryview);
+#endif
static gint summary_folder_eventbox_pressed
(GtkWidget *eventbox,
GdkEventButton *event,
SummaryView *summaryview);
static void summary_locked_clicked (GtkWidget *button,
SummaryView *summaryview);
+static void summary_tags_clicked (GtkWidget *button,
+ SummaryView *summaryview);
static void summary_start_drag (GtkWidget *widget,
int button,
static gint summary_cmp_by_date (GtkCList *clist,
gconstpointer ptr1,
gconstpointer ptr2);
+static gint summary_cmp_by_thread_date (GtkCList *clist,
+ gconstpointer ptr1,
+ gconstpointer ptr2);
static gint summary_cmp_by_from (GtkCList *clist,
gconstpointer ptr1,
gconstpointer ptr2);
static gint summary_cmp_by_locked (GtkCList *clist,
gconstpointer ptr1,
gconstpointer ptr2);
+static gint summary_cmp_by_tags (GtkCList *clist,
+ gconstpointer ptr1,
+ gconstpointer ptr2);
static void quicksearch_execute_cb (QuickSearch *quicksearch,
gpointer data);
static GtkItemFactoryEntry summary_popup_entries[] =
{
{N_("/_Reply"), "<control>R", summary_reply_cb, COMPOSE_REPLY, NULL},
+#ifndef MAEMO
{N_("/Repl_y to"), NULL, NULL, 0, "<Branch>"},
{N_("/Repl_y to/_all"), "<shift><control>R", summary_reply_cb, COMPOSE_REPLY_TO_ALL, NULL},
{N_("/Repl_y to/_sender"), NULL, summary_reply_cb, COMPOSE_REPLY_TO_SENDER, NULL},
{N_("/Repl_y to/mailing _list"),
"<control>L", summary_reply_cb, COMPOSE_REPLY_TO_LIST, NULL},
{"/---", NULL, NULL, 0, "<Separator>"},
+#endif
{N_("/_Forward"), "<control><alt>F", summary_reply_cb, COMPOSE_FORWARD_INLINE, NULL},
+#ifndef MAEMO
{N_("/For_ward as attachment"), NULL, summary_reply_cb, COMPOSE_FORWARD_AS_ATTACH, NULL},
{N_("/Redirect"), NULL, summary_reply_cb, COMPOSE_REDIRECT, NULL},
+#endif
{"/---", NULL, NULL, 0, "<Separator>"},
{N_("/M_ove..."), "<control>O", summary_move_to, 0, NULL},
{N_("/_Copy..."), "<shift><control>O", summary_copy_to, 0, NULL},
{N_("/Move to _trash"), "<control>D", summary_delete_trash, 0, NULL},
+#ifndef MAEMO
{N_("/_Delete..."), NULL, summary_delete, 0, NULL},
+#endif
{"/---", NULL, NULL, 0, "<Separator>"},
{N_("/_Mark"), NULL, NULL, 0, "<Branch>"},
{N_("/_Mark/_Mark"), NULL, summary_mark, 0, NULL},
{N_("/_Mark/Mark all read"), NULL, summary_mark_all_read, 0, NULL},
{N_("/_Mark/Ignore thread"), NULL, summary_ignore_thread, 0, NULL},
{N_("/_Mark/Unignore thread"), NULL, summary_unignore_thread, 0, NULL},
+ {N_("/_Mark/Watch thread"), NULL, summary_watch_thread, 0, NULL},
+ {N_("/_Mark/Unwatch thread"), NULL, summary_unwatch_thread, 0, NULL},
{N_("/_Mark/---"), NULL, NULL, 0, "<Separator>"},
{N_("/_Mark/Mark as _spam"), NULL, summary_mark_as_spam, 1, NULL},
{N_("/_Mark/Mark as _ham"), NULL, summary_mark_as_spam, 0, NULL},
{N_("/_Mark/Lock"), NULL, summary_msgs_lock, 0, NULL},
{N_("/_Mark/Unlock"), NULL, summary_msgs_unlock, 0, NULL},
{N_("/Color la_bel"), NULL, NULL, 0, NULL},
+ {N_("/Ta_gs"), NULL, NULL, 0, NULL},
{"/---", NULL, NULL, 0, "<Separator>"},
+#ifndef MAEMO
{N_("/Add sender to address boo_k"),
NULL, summary_add_address_cb, 0, NULL},
+#endif
{N_("/Create f_ilter rule"), NULL, NULL, 0, "<Branch>"},
{N_("/Create f_ilter rule/_Automatically"),
NULL, summary_create_filter_cb, FILTER_BY_AUTO, NULL},
NULL, summary_create_filter_cb, FILTER_BY_TO, NULL},
{N_("/Create f_ilter rule/by _Subject"),
NULL, summary_create_filter_cb, FILTER_BY_SUBJECT, NULL},
+#ifndef MAEMO
{N_("/Create processing rule"), NULL, NULL, 0, "<Branch>"},
{N_("/Create processing rule/_Automatically"),
NULL, summary_create_processing_cb, FILTER_BY_AUTO, NULL},
NULL, summary_create_processing_cb, FILTER_BY_TO, NULL},
{N_("/Create processing rule/by _Subject"),
NULL, summary_create_processing_cb, FILTER_BY_SUBJECT, NULL},
+#endif
{"/---", NULL, NULL, 0, "<Separator>"},
{N_("/_View"), NULL, NULL, 0, "<Branch>"},
{N_("/_View/Open in new _window"),
"<control><alt>N", summary_open_msg, 0, NULL},
{N_("/_View/Message _source"), "<control>U", summary_view_source, 0, NULL},
+#ifndef MAEMO
{N_("/_View/All _headers"), "<control>H", summary_show_all_header_cb, 0, "<ToggleItem>"},
+#endif
{"/---", NULL, NULL, 0, "<Separator>"},
{N_("/_Save as..."), "<control>S", summary_save_as, 0, NULL},
+#ifndef MAEMO
{N_("/_Print..."), "<control>P", summary_print, 0, NULL},
+#endif
}; /* see also list in menu_connect_identical_items() in menu.c if this changes */
static const gchar *const col_label[N_SUMMARY_COLS] = {
N_("#"), /* S_COL_NUMBER */
N_("Score"), /* S_COL_SCORE */
"", /* S_COL_LOCKED */
+ N_("Tags"), /* S_COL_TAGS */
};
void summary_freeze(SummaryView *summaryview)
GtkWidget *statlabel_msgs;
GtkWidget *hbox_spc;
GtkWidget *toggle_eventbox;
+#ifdef MAEMO
+ GtkWidget *multiple_sel_togbtn;
+#endif
GtkWidget *toggle_arrow;
GtkWidget *popupmenu;
GtkWidget *toggle_search;
/* toggle view button */
toggle_eventbox = gtk_event_box_new();
gtk_widget_show(toggle_eventbox);
+
gtk_box_pack_end(GTK_BOX(hbox), toggle_eventbox, FALSE, FALSE, 4);
+
toggle_arrow = gtk_arrow_new(GTK_ARROW_DOWN, GTK_SHADOW_OUT);
gtk_widget_show(toggle_arrow);
gtk_container_add(GTK_CONTAINER(toggle_eventbox), toggle_arrow);
g_signal_connect(G_OBJECT(toggle_eventbox), "button_press_event",
G_CALLBACK(summary_toggle_pressed),
summaryview);
-
+
+#ifdef MAEMO
+ multiple_sel_togbtn = gtk_toggle_button_new();
+ gtk_widget_show(multiple_sel_togbtn);
+ gtk_box_pack_end(GTK_BOX(hbox), multiple_sel_togbtn, FALSE, FALSE, 4);
+ gtk_tooltips_set_tip(GTK_TOOLTIPS(summaryview->tips),
+ multiple_sel_togbtn,
+ _("Toggle multiple selection"), NULL);
+ g_signal_connect(G_OBJECT(multiple_sel_togbtn), "toggled",
+ G_CALLBACK(summary_toggle_multiple_pressed),
+ summaryview);
+#endif
statlabel_msgs = gtk_label_new("");
gtk_widget_show(statlabel_msgs);
summaryview->statlabel_msgs = statlabel_msgs;
summaryview->toggle_eventbox = toggle_eventbox;
summaryview->toggle_arrow = toggle_arrow;
+#ifdef MAEMO
+ summaryview->multiple_sel_togbtn = multiple_sel_togbtn;
+#endif
summaryview->toggle_search = toggle_search;
summaryview->popupmenu = popupmenu;
summaryview->popupfactory = popupfactory;
gtk_box_pack_start(GTK_BOX(summaryview->stat_box2), summaryview->statlabel_msgs, FALSE, FALSE, 4);
gtk_widget_show_all(summaryview->stat_box);
gtk_widget_show_all(summaryview->stat_box2);
- if (prefs_common.layout_mode == SMALL_LAYOUT)
+ if (prefs_common.layout_mode == SMALL_LAYOUT) {
gtk_widget_hide(summaryview->toggle_eventbox);
- else
+ gtk_widget_hide(summaryview->statlabel_msgs);
+ } else {
gtk_widget_show(summaryview->toggle_eventbox);
+ gtk_widget_show(summaryview->statlabel_msgs);
+ }
break;
}
gtk_widget_unref(summaryview->hbox_l);
gtk_widget_unref(summaryview->statlabel_msgs);
quicksearch_relayout(summaryview->quicksearch);
+ if (prefs_common.show_searchbar)
+ quicksearch_show(summaryview->quicksearch);
+ else
+ quicksearch_hide(summaryview->quicksearch);
}
static void summary_set_fonts(SummaryView *summaryview)
gtk_widget_modify_font(summaryview->statlabel_folder, font_desc);
gtk_widget_modify_font(summaryview->statlabel_select, font_desc);
gtk_widget_modify_font(summaryview->statlabel_msgs, font_desc);
+ /* ici */
pango_font_description_free(font_desc);
}
&lockedxpm, &lockedxpmmask);
stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_IGNORETHREAD,
&ignorethreadxpm, &ignorethreadxpmmask);
+ stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_WATCHTHREAD,
+ &watchthreadxpm, &watchthreadxpmmask);
stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_CLIP_KEY,
&clipkeyxpm, &clipkeyxpmmask);
stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_KEY,
gtk_container_add (GTK_CONTAINER(summaryview->toggle_search), pixmap);
gtk_widget_show(pixmap);
summaryview->quick_search_pixmap = pixmap;
+
+#ifdef MAEMO
+ pixmap = stock_pixmap_widget(summaryview->hbox, STOCK_PIXMAP_SELECTION);
+ gtk_container_add(GTK_CONTAINER(summaryview->multiple_sel_togbtn), pixmap);
+ gtk_widget_show(pixmap);
+ summaryview->multiple_sel_image = pixmap;
+#endif
/* Init summaryview prefs */
summaryview->sort_key = SORT_BY_NONE;
summary_clear_list(summaryview);
summary_set_column_titles(summaryview);
summary_colorlabel_menu_create(summaryview, FALSE);
+ summary_tags_menu_create(summaryview, FALSE);
main_create_mailing_list_menu (summaryview->mainwin, NULL);
summary_set_menu_sensitive(summaryview);
}
}
if (!prefs_common.summary_quicksearch_sticky
- && !prefs_common.summary_quicksearch_recurse
+ && (!prefs_common.summary_quicksearch_recurse
+ || !quicksearch_is_active(summaryview->quicksearch))
&& !quicksearch_is_running(summaryview->quicksearch)
&& !is_refresh) {
quicksearch_set(summaryview->quicksearch, prefs_common.summary_quicksearch_type, "");
g_free(buf);
debug_print("empty folder (%p %s %p %d)\n\n",
item,
- item?item->path:"NULL",
+ (item && item->path)?item->path:"(null)",
item?folder_item_parent(item):0x0,
item?item->no_select:FALSE);
summary_set_hide_read_msgs_menu(summaryview, FALSE);
gtk_ctree_node_moveto(ctree, node, 0, 0.5, 0);
}
} else {
- switch (prefs_common.select_on_entry) {
- case SELECTONENTRY_MNU:
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_MARKED, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_NEW, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_UNREAD, FALSE);
+ /* backward compat */
+ int i = 0;
+ gboolean set = FALSE, stop = FALSE;
+ for (i = 0; i < 6; i++) {
+ EntryAction act = prefs_common.summary_select_prio[i];
+
+ if (act != ACTION_UNSET) {
+ set = TRUE;
break;
- case SELECTONENTRY_MUN:
+ }
+ }
+ if (!set)
+ prefs_summary_open_set_defaults();
+
+ for (i = 0; i < 6 && node == NULL; i++) {
+ EntryAction act = prefs_common.summary_select_prio[i];
+
+ switch(act) {
+ case ACTION_MARKED:
node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_MARKED, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_UNREAD, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_NEW, FALSE);
+ MSG_MARKED, FALSE);
break;
- case SELECTONENTRY_NMU:
+ case ACTION_NEW:
node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_NEW, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_MARKED, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_UNREAD, FALSE);
+ MSG_NEW, FALSE);
break;
- case SELECTONENTRY_NUM:
+ case ACTION_UNREAD:
node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_NEW, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_UNREAD, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_MARKED, FALSE);
+ MSG_UNREAD, FALSE);
break;
- case SELECTONENTRY_UNM:
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_UNREAD, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_NEW, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_MARKED, FALSE);
+ case ACTION_LAST_OPENED:
+ if (summaryview->folder_item) {
+ node = summary_find_msg_by_msgnum(summaryview,
+ summaryview->folder_item->last_seen);
+ }
break;
- case SELECTONENTRY_UMN:
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_UNREAD, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_MARKED, FALSE);
- if (node == NULL)
- node = summary_find_next_flagged_msg(summaryview, NULL,
- MSG_NEW, FALSE);
+ case ACTION_LAST_LIST:
+ if (GTK_CLIST(ctree)->row_list != NULL) {
+ node = gtk_ctree_node_nth
+ (ctree,
+ item->sort_type == SORT_DESCENDING
+ ? 0 : GTK_CLIST(ctree)->rows - 1);
+ }
break;
- default:
+ case ACTION_NOTHING:
+ case ACTION_UNSET:
node = NULL;
- }
-
- if (node == NULL && GTK_CLIST(ctree)->row_list != NULL) {
- node = gtk_ctree_node_nth
- (ctree,
- item->sort_type == SORT_DESCENDING
- ? 0 : GTK_CLIST(ctree)->rows - 1);
+ stop = TRUE;
+ break;
+ }
+
+ if (stop || node)
+ break;
}
+
summary_unlock(summaryview);
- summary_select_node(summaryview, node,
+ if (node)
+ summary_select_node(summaryview, node,
prefs_common.always_show_msg,
TRUE);
summary_lock(summaryview);
SensitiveCond cond;
} entry[] = {
{"/Reply" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
+#ifndef MAEMO
{"/Reply to" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
{"/Reply to/all" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
{"/Reply to/sender" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
{"/Reply to/mailing list" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
+#endif
{"/Forward" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
+#ifndef MAEMO
{"/Forward as attachment" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
{"/Redirect" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
+#endif
{"/Move..." , M_TARGET_EXIST|M_ALLOW_DELETE|M_NOT_NEWS},
{"/Copy..." , M_TARGET_EXIST|M_EXEC},
{"/Move to trash" , M_TARGET_EXIST|M_ALLOW_DELETE|M_NOT_NEWS},
+#ifndef MAEMO
{"/Delete..." , M_TARGET_EXIST|M_ALLOW_DELETE},
+#endif
{"/Mark" , M_TARGET_EXIST},
{"/Mark/Mark" , M_TARGET_EXIST},
{"/Mark/Mark as spam" , M_TARGET_EXIST|M_CAN_LEARN_SPAM},
{"/Mark/Mark as ham" , M_TARGET_EXIST|M_CAN_LEARN_SPAM},
{"/Color label" , M_TARGET_EXIST},
+ {"/Tags" , M_TARGET_EXIST},
+#ifndef MAEMO
{"/Add sender to address book" , M_SINGLE_TARGET_EXIST},
+#endif
{"/Create filter rule" , M_SINGLE_TARGET_EXIST|M_UNLOCKED},
+#ifndef MAEMO
{"/Create processing rule" , M_SINGLE_TARGET_EXIST|M_UNLOCKED},
+#endif
{"/View" , M_SINGLE_TARGET_EXIST},
{"/View/Open in new window" , M_SINGLE_TARGET_EXIST},
{"/View/Message source" , M_SINGLE_TARGET_EXIST},
+#ifndef MAEMO
{"/View/All headers" , M_SINGLE_TARGET_EXIST},
+#endif
{"/Save as..." , M_TARGET_EXIST},
+#ifndef MAEMO
{"/Print..." , M_TARGET_EXIST},
+#endif
{NULL, 0}
};
summary_lock(summaryview);
+#ifndef MAEMO
menuitem = gtk_item_factory_get_widget(ifactory, "/View/All headers");
if (summaryview->messageview
&& summaryview->messageview->mimeview
gtk_check_menu_item_set_active
(GTK_CHECK_MENU_ITEM(menuitem),
summaryview->messageview->mimeview->textview->show_all_headers);
+#endif
summary_unlock(summaryview);
}
itstr = g_strdup("");
}
- str = g_strconcat(n_selected ? itos(n_selected) : "",
- itstr, sel, spc, del, mv, cp, NULL);
- gtk_label_set_text(GTK_LABEL(summaryview->statlabel_select), str);
- g_free(str);
- g_free(sel);
- g_free(del);
- g_free(mv);
- g_free(cp);
- g_free(itstr);
-
- str = g_strdup_printf(_("%d new, %d unread, %d total (%s)"),
-
- n_new, n_unread, n_total,
- to_human_readable(n_size));
- gtk_label_set_text(GTK_LABEL(summaryview->statlabel_msgs), str);
- g_free(str);
+ if (prefs_common.layout_mode != SMALL_LAYOUT) {
+ str = g_strconcat(n_selected ? itos(n_selected) : "",
+ itstr, sel, spc, del, mv, cp, NULL);
+ g_free(sel);
+ g_free(del);
+ g_free(mv);
+ g_free(cp);
+ g_free(itstr);
+
+ gtk_label_set_text(GTK_LABEL(summaryview->statlabel_select), str);
+ g_free(str);
+
+ str = g_strdup_printf(_("%d new, %d unread, %d total (%s)"),
+
+ n_new, n_unread, n_total,
+ to_human_readable(n_size));
+
+ gtk_label_set_text(GTK_LABEL(summaryview->statlabel_msgs), str);
+ g_free(str);
+ } else {
+ gchar *ssize, *tsize;
+ if (n_selected) {
+ ssize = g_strdup(to_human_readable(sel_size));
+ tsize = g_strdup(to_human_readable(n_size));
+ str = g_strdup_printf(_("%d/%d selected (%s/%s), %d unread"),
+ n_selected, n_total, ssize, tsize, n_unread);
+ g_free(ssize);
+ g_free(tsize);
+ } else
+ str = g_strdup_printf(_("%d new, %d unread, %d total (%s)"),
+ n_new, n_unread, n_total, to_human_readable(n_size));
+ g_free(sel);
+ g_free(del);
+ g_free(mv);
+ g_free(cp);
+ g_free(itstr);
+
+ gtk_label_set_text(GTK_LABEL(summaryview->statlabel_select), str);
+ g_free(str);
+ }
+
+ summary_set_menu_sensitive(summaryview);
toolbar_main_set_sensitive(summaryview->mainwin);
}
SORT_BY_SIZE,
SORT_BY_NUMBER,
SORT_BY_SCORE,
- SORT_BY_LOCKED
+ SORT_BY_LOCKED,
+ SORT_BY_TAGS
};
for (pos = 0; pos < N_SUMMARY_COLS; pos++) {
hbox = gtk_hbox_new(FALSE, 4);
label = gtk_label_new(title);
+#ifdef MAEMO
+ gtk_widget_set_size_request(hbox, -1, 20);
+#endif
if (justify == GTK_JUSTIFY_RIGHT)
gtk_box_pack_end(GTK_BOX(hbox), label,
}
}
+void summary_reflect_tags_changes(SummaryView *summaryview)
+{
+ GtkMenuShell *menu;
+ GList *cur;
+ GtkCTreeNode *node;
+ GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
+ gboolean froze = FALSE;
+ gboolean redisplay = FALSE;
+
+ /* re-create colorlabel submenu */
+ menu = GTK_MENU_SHELL(summaryview->tags_menu);
+ g_return_if_fail(menu != NULL);
+
+ /* clear items. get item pointers. */
+ for (cur = menu->children; cur != NULL && cur->data != NULL; cur = cur->next) {
+ gtk_menu_item_remove_submenu(GTK_MENU_ITEM(cur->data));
+ }
+ summary_tags_menu_create(summaryview, TRUE);
+
+ START_LONG_OPERATION(summaryview, TRUE);
+ for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list); node != NULL;
+ node = gtkut_ctree_node_next(ctree, node)) {
+ redisplay |= summary_set_row_tag(summaryview,
+ node, TRUE, FALSE, 0);
+ }
+ END_LONG_OPERATION(summaryview);
+ if (redisplay)
+ summary_redisplay_msg(summaryview);
+}
+
+
void summary_reflect_prefs(void)
{
static gchar *last_font = NULL;
}
summary_set_column_titles(summaryview);
- summary_show(summaryview, summaryview->folder_item);
+ summary_relayout(summaryview);
+
+ if (summaryview->folder_item)
+ summary_show(summaryview, summaryview->folder_item);
}
void summary_sort(SummaryView *summaryview,
case SORT_BY_DATE:
cmp_func = (GtkCListCompareFunc)summary_cmp_by_date;
break;
+ case SORT_BY_THREAD_DATE:
+ cmp_func = (GtkCListCompareFunc)summary_cmp_by_thread_date;
+ break;
case SORT_BY_FROM:
cmp_func = (GtkCListCompareFunc)summary_cmp_by_from;
break;
case SORT_BY_LOCKED:
cmp_func = (GtkCListCompareFunc)summary_cmp_by_locked;
break;
+ case SORT_BY_TAGS:
+ cmp_func = (GtkCListCompareFunc)summary_cmp_by_tags;
+ break;
case SORT_BY_NONE:
break;
default:
END_TIMING();
}
+static gboolean summary_update_thread_age(GNode *node, gpointer data)
+{
+ MsgInfo *msginfo = node->data;
+ time_t *most_recent = (time_t *)data;
+
+ if (msginfo->date_t > *most_recent) {
+ *most_recent = msginfo->date_t;
+ }
+ return FALSE;
+}
+
+static void summary_find_thread_age(GNode *gnode)
+{
+ MsgInfo *msginfo = (MsgInfo *)gnode->data;
+ time_t most_recent;
+
+ if (!msginfo)
+ return;
+ most_recent = msginfo->thread_date = msginfo->date_t;
+
+ g_node_traverse(gnode, G_IN_ORDER, G_TRAVERSE_ALL, -1, summary_update_thread_age, &most_recent);
+
+ msginfo->thread_date = most_recent;
+}
+
static gboolean summary_insert_gnode_func(GtkCTree *ctree, guint depth, GNode *gnode,
GtkCTreeNode *cnode, gpointer data)
{
SET_TEXT(S_COL_FROM);
if (summaryview->col_state[summaryview->col_pos[S_COL_TO]].visible)
SET_TEXT(S_COL_TO);
+ if (summaryview->col_state[summaryview->col_pos[S_COL_TAGS]].visible)
+ SET_TEXT(S_COL_TAGS);
#undef SET_TEXT
START_TIMING("threaded");
root = procmsg_get_thread_tree(mlist);
+
for (gnode = root->children; gnode != NULL;
gnode = gnode->next) {
+ summary_find_thread_age(gnode);
node = gtk_sctree_insert_gnode
(ctree, NULL, node, gnode,
summary_insert_gnode_func, summaryview);
return res;
}
-static void summary_set_header(SummaryView *summaryview, gchar *text[],
+static inline void summary_set_header(SummaryView *summaryview, gchar *text[],
MsgInfo *msginfo)
{
static gchar date_modified[80];
static gchar col_score[11];
- static gchar buf[BUFFSIZE];
+ static gchar buf[BUFFSIZE], tmp1[BUFFSIZE], tmp2[BUFFSIZE], tmp3[BUFFSIZE];
gint *col_pos = summaryview->col_pos;
- gchar *from_text = NULL, *to_text = NULL;
+ gchar *from_text = NULL, *to_text = NULL, *tags_text = NULL;
gboolean should_swap = FALSE;
text[col_pos[S_COL_FROM]] = "";
text[col_pos[S_COL_MIME]] = "";
text[col_pos[S_COL_LOCKED]] = "";
text[col_pos[S_COL_DATE]] = "";
+ text[col_pos[S_COL_TAGS]] = "";
if (summaryview->col_state[summaryview->col_pos[S_COL_NUMBER]].visible)
text[col_pos[S_COL_NUMBER]] = itos(msginfo->msgnum);
else
else
text[col_pos[S_COL_SCORE]] = "";
+ if (summaryview->col_state[summaryview->col_pos[S_COL_TAGS]].visible) {
+ tags_text = procmsg_msginfo_get_tags_str(msginfo);
+ if (!tags_text) {
+ text[col_pos[S_COL_TAGS]] = "-";
+ } else {
+ strncpy2(tmp1, tags_text, sizeof(tmp1));
+ tmp1[sizeof(tmp1)-1]='\0';
+ g_free(tags_text);
+ text[col_pos[S_COL_TAGS]] = tmp1;
+ }
+ } else
+ text[col_pos[S_COL_TAGS]] = "";
+
/* slow! */
if (summaryview->col_state[summaryview->col_pos[S_COL_DATE]].visible) {
if (msginfo->date_t) {
msginfo->fromname :
_("(No From)");
} else {
- gchar buf[BUFFSIZE];
gchar *tmp = summary_complete_address(msginfo->from);
if (tmp) {
strncpy2(buf, tmp, sizeof(buf));
if (!should_swap) {
text[col_pos[S_COL_FROM]] = from_text;
} else {
- gchar tmp[BUFFSIZE];
- snprintf(tmp, BUFFSIZE-1, "--> %s", to_text);
- tmp[BUFFSIZE-1]='\0';
- text[col_pos[S_COL_FROM]] = tmp;
+ snprintf(tmp2, BUFFSIZE-1, "--> %s", to_text);
+ tmp2[BUFFSIZE-1]='\0';
+ text[col_pos[S_COL_FROM]] = tmp2;
}
if (summaryview->simplify_subject_preg != NULL)
text[col_pos[S_COL_SUBJECT]] = msginfo->subject ?
- string_remove_match(buf, BUFFSIZE, msginfo->subject,
+ string_remove_match(tmp3, BUFFSIZE, msginfo->subject,
summaryview->simplify_subject_preg) :
_("(No Subject)");
else
if (MSG_IS_IGNORE_THREAD(flags)) {
gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
ignorethreadxpm, ignorethreadxpmmask);
+ } else if (MSG_IS_WATCH_THREAD(flags)) {
+ gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
+ watchthreadxpm, watchthreadxpmmask);
} else if (MSG_IS_SPAM(flags)) {
gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
spamxpm, spamxpmmask);
return;
}
+ if (to_folder->no_select) {
+ alertpanel_error(_("The destination folder can only be used to "
+ "store subfolders."));
+ return;
+ }
+
START_LONG_OPERATION(summaryview, FALSE);
for (cur = GTK_CLIST(summaryview->ctree)->selection;
return;
}
+ if (to_folder->no_select) {
+ alertpanel_error(_("The destination folder can only be used to "
+ "store subfolders."));
+ return;
+ }
+
START_LONG_OPERATION(summaryview, FALSE);
for (cur = GTK_CLIST(summaryview->ctree)->selection;
void summary_select_thread(SummaryView *summaryview, gboolean delete_thread)
{
GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
- GtkCTreeNode *node = summaryview->selected;
+ GtkCTreeNode *node = NULL;
gboolean froze = FALSE;
+ GList *cur = NULL;
+ GList *copy = NULL;
+ if (!GTK_CLIST(summaryview->ctree)->selection)
+ return;
- if (!node) return;
-
- while (GTK_CTREE_ROW(node)->parent != NULL)
- node = GTK_CTREE_ROW(node)->parent;
START_LONG_OPERATION(summaryview, FALSE);
- if (node != summaryview->selected)
- summary_select_node
- (summaryview, node,
- messageview_is_visible(summaryview->messageview),
- FALSE);
+ copy = g_list_copy(GTK_CLIST(summaryview->ctree)->selection);
+ for (cur = copy; cur != NULL && cur->data != NULL;
+ cur = cur->next) {
+ node = GTK_CTREE_NODE(cur->data);
+ if (!node)
+ continue;
+ while (GTK_CTREE_ROW(node)->parent != NULL)
+ node = GTK_CTREE_ROW(node)->parent;
- gtk_ctree_select_recursive(ctree, node);
+ gtk_ctree_select_recursive(ctree, node);
+ }
+ g_list_free(copy);
END_LONG_OPERATION(summaryview);
if (delete_thread) {
conv_get_locale_charset_str(),
CS_UTF_8);
if (!filename) {
- g_warning("summary_save_as(): faild to convert character set.");
+ g_warning("summary_save_as(): failed to convert character set.");
filename = g_strdup(oldstr);
}
dest = filesel_select_file_save(_("Save as"), filename);
gchar *p;
#endif
GList *cur;
+ gchar *msg = g_strdup_printf(_("You are about to print %d "
+ "messages, one by one. Do you "
+ "want to continue?"),
+ g_list_length(clist->selection));
+ if (g_list_length(clist->selection) > 9
+ && alertpanel(_("Warning"), msg, GTK_STOCK_CANCEL, "+" GTK_STOCK_YES, NULL)
+ != G_ALERTALTERNATE) {
+ g_free(msg);
+ return;
+ }
+ g_free(msg);
if (clist->selection == NULL) return;
#ifndef USE_GNOMEPRINT
END_LONG_OPERATION(summaryview);
}
+static gboolean summary_set_row_tag(SummaryView *summaryview, GtkCTreeNode *row, gboolean refresh, gboolean set, gint id)
+{
+ GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
+ MsgInfo *msginfo;
+ gchar *tags_str = NULL;
+ msginfo = gtk_ctree_node_get_row_data(ctree, row);
+ g_return_val_if_fail(msginfo, FALSE);
+
+ procmsg_msginfo_update_tags(msginfo, set, id);
+
+
+ if (summaryview->col_state[summaryview->col_pos[S_COL_TAGS]].visible) {
+ tags_str = procmsg_msginfo_get_tags_str(msginfo);
+ gtk_ctree_node_set_text(ctree, row,
+ summaryview->col_pos[S_COL_TAGS],
+ tags_str?tags_str:"-");
+ g_free(tags_str);
+ }
+
+ summary_set_row_marks(summaryview, row);
+ if (row == summaryview->displayed) {
+ return TRUE;
+ }
+ return FALSE;
+}
+
+void summary_set_tag(SummaryView *summaryview, gint tag_id,
+ GtkWidget *widget)
+{
+ GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
+ GList *cur;
+ gboolean set = tag_id > 0;
+ gint real_id = set? tag_id:-tag_id;
+ gboolean froze = FALSE;
+ gboolean redisplay = FALSE;
+ START_LONG_OPERATION(summaryview, FALSE);
+ for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next) {
+ redisplay |= summary_set_row_tag(summaryview,
+ GTK_CTREE_NODE(cur->data), FALSE, set, real_id);
+ }
+ END_LONG_OPERATION(summaryview);
+ if (redisplay)
+ summary_redisplay_msg(summaryview);
+}
+
+static void summary_tags_menu_item_activate_cb(GtkWidget *widget,
+ gpointer data)
+{
+ gint id = GPOINTER_TO_INT(data);
+ gboolean set = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget));
+ SummaryView *summaryview;
+
+ summaryview = g_object_get_data(G_OBJECT(widget), "summaryview");
+ g_return_if_fail(summaryview != NULL);
+
+ /* "dont_toggle" state set? */
+ if (g_object_get_data(G_OBJECT(summaryview->tags_menu),
+ "dont_toggle"))
+ return;
+
+ if (!set)
+ id = -id;
+ summary_set_tag(summaryview, id, NULL);
+}
+
static void summary_colorlabel_menu_item_activate_item_cb(GtkMenuItem *menu_item,
gpointer data)
{
summaryview->colorlabel_menu = menu;
}
+static void summary_tags_menu_item_activate_item_cb(GtkMenuItem *menu_item,
+ gpointer data)
+{
+ GtkMenuShell *menu;
+ GList *cur;
+ GList *sel;
+ GHashTable *menu_table = g_hash_table_new_full(
+ g_direct_hash,
+ g_direct_equal,
+ NULL, NULL);
+ GHashTable *menu_allsel_table = g_hash_table_new_full(
+ g_direct_hash,
+ g_direct_equal,
+ NULL, NULL);
+ gint sel_len;
+ SummaryView *summaryview = (SummaryView *)data;
+ g_return_if_fail(summaryview != NULL);
+
+ sel = GTK_CLIST(summaryview->ctree)->selection;
+ if (!sel) return;
+
+ menu = GTK_MENU_SHELL(summaryview->tags_menu);
+ g_return_if_fail(menu != NULL);
+
+ /* NOTE: don't return prematurely because we set the "dont_toggle"
+ * state for check menu items */
+ g_object_set_data(G_OBJECT(menu), "dont_toggle",
+ GINT_TO_POINTER(1));
+
+ /* clear items. get item pointers. */
+ for (cur = menu->children; cur != NULL && cur->data != NULL; cur = cur->next) {
+ if (GTK_IS_CHECK_MENU_ITEM(cur->data)) {
+ gint id = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(cur->data),
+ "tag_id"));
+ gtk_check_menu_item_set_active
+ (GTK_CHECK_MENU_ITEM(cur->data), FALSE);
+
+ g_hash_table_insert(menu_table, GINT_TO_POINTER(id), GTK_CHECK_MENU_ITEM(cur->data));
+ g_hash_table_insert(menu_allsel_table, GINT_TO_POINTER(id), GINT_TO_POINTER(0));
+ }
+ }
+
+ /* iterate all messages and set the state of the appropriate
+ * items */
+ sel_len = 0;
+ for (; sel != NULL; sel = sel->next) {
+ MsgInfo *msginfo;
+ GSList *tags = NULL;
+ gint id;
+ GtkCheckMenuItem *item;
+ msginfo = gtk_ctree_node_get_row_data
+ (GTK_CTREE(summaryview->ctree),
+ GTK_CTREE_NODE(sel->data));
+ sel_len++;
+ if (msginfo) {
+ tags = msginfo->tags;
+ if (!tags)
+ continue;
+
+ for (; tags; tags = tags->next) {
+ gint num_checked = GPOINTER_TO_INT(g_hash_table_lookup(menu_allsel_table, tags->data));
+ id = GPOINTER_TO_INT(tags->data);
+ item = g_hash_table_lookup(menu_table, GINT_TO_POINTER(tags->data));
+ if (item && !item->active) {
+ gtk_check_menu_item_set_active
+ (item, TRUE);
+ }
+ num_checked++;
+ g_hash_table_replace(menu_allsel_table, tags->data, GINT_TO_POINTER(num_checked));
+ }
+ }
+ }
+
+ for (cur = menu->children; cur != NULL && cur->data != NULL; cur = cur->next) {
+ if (GTK_IS_CHECK_MENU_ITEM(cur->data)) {
+ gint id = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(cur->data),
+ "tag_id"));
+ gint num_checked = GPOINTER_TO_INT(g_hash_table_lookup(menu_allsel_table, GINT_TO_POINTER(id)));
+ if (num_checked < sel_len && num_checked > 0)
+ gtk_check_menu_item_set_inconsistent(GTK_CHECK_MENU_ITEM(cur->data), TRUE);
+ else
+ gtk_check_menu_item_set_inconsistent(GTK_CHECK_MENU_ITEM(cur->data), FALSE);
+ }
+ }
+ g_hash_table_destroy(menu_table);
+ g_hash_table_destroy(menu_allsel_table);
+ /* reset "dont_toggle" state */
+ g_object_set_data(G_OBJECT(menu), "dont_toggle",
+ GINT_TO_POINTER(0));
+
+}
+
+static void summary_tags_menu_item_apply_tags_activate_cb(GtkWidget *widget,
+ gpointer data)
+{
+ SummaryView *summaryview;
+
+ summaryview = g_object_get_data(G_OBJECT(widget), "summaryview");
+ g_return_if_fail(summaryview != NULL);
+
+ /* "dont_toggle" state set? */
+ if (g_object_get_data(G_OBJECT(summaryview->tags_menu),
+ "dont_toggle"))
+ return;
+
+ tag_apply_open(summary_get_selection(summaryview));
+}
+
+static void summary_tags_menu_create(SummaryView *summaryview, gboolean refresh)
+{
+ GtkWidget *label_menuitem;
+ GtkWidget *menu;
+ GtkWidget *item;
+ GSList *cur = tags_get_list();
+ GSList *orig = cur;
+ gboolean existing_tags = FALSE;
+
+ label_menuitem = gtk_item_factory_get_item(summaryview->popupfactory,
+ "/Tags");
+ g_signal_connect(G_OBJECT(label_menuitem), "activate",
+ G_CALLBACK(summary_tags_menu_item_activate_item_cb),
+ summaryview);
+
+ gtk_widget_show(label_menuitem);
+
+ menu = gtk_menu_new();
+
+ /* create tags menu items */
+ for (; cur; cur = cur->next) {
+ gint id = GPOINTER_TO_INT(cur->data);
+ const gchar *tag = tags_get_tag(id);
+
+ item = gtk_check_menu_item_new_with_label(tag);
+ gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
+ g_signal_connect(G_OBJECT(item), "activate",
+ G_CALLBACK(summary_tags_menu_item_activate_cb),
+ GINT_TO_POINTER(id));
+ g_object_set_data(G_OBJECT(item), "summaryview",
+ summaryview);
+ g_object_set_data(G_OBJECT(item), "tag_id",
+ GINT_TO_POINTER(id));
+ gtk_widget_show(item);
+ existing_tags = TRUE;
+ }
+ if (existing_tags) {
+ /* separator */
+ item = gtk_menu_item_new();
+ gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
+ gtk_widget_show(item);
+ }
+
+ item = gtk_menu_item_new_with_label(_("Apply tags..."));
+ gtk_widget_add_accelerator(item, "activate",
+ summaryview->popupfactory->accel_group,
+ GDK_T, GDK_CONTROL_MASK|GDK_SHIFT_MASK,
+ GTK_ACCEL_LOCKED | GTK_ACCEL_VISIBLE);
+ gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
+ g_signal_connect(G_OBJECT(item), "activate",
+ G_CALLBACK(summary_tags_menu_item_apply_tags_activate_cb),
+ NULL);
+ g_object_set_data(G_OBJECT(item), "summaryview",
+ summaryview);
+ gtk_widget_show(item);
+
+ g_slist_free(orig);
+ gtk_widget_show(menu);
+ gtk_menu_item_set_submenu(GTK_MENU_ITEM(label_menuitem), menu);
+ summaryview->tags_menu = menu;
+}
+
static gboolean summary_popup_menu(GtkWidget *widget, gpointer data)
{
SummaryView *summaryview = (SummaryView *)data;
ctree = gtk_sctree_new_with_titles
(N_SUMMARY_COLS, col_pos[S_COL_SUBJECT], titles);
+ if (prefs_common.show_col_headers == FALSE)
+ gtk_clist_column_titles_hide(GTK_CLIST(ctree));
+
gtk_clist_set_selection_mode(GTK_CLIST(ctree), GTK_SELECTION_EXTENDED);
gtk_clist_set_column_justification(GTK_CLIST(ctree), col_pos[S_COL_MARK],
GTK_JUSTIFY_CENTER);
prefs_common.summary_col_size[S_COL_NUMBER]);
gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_SCORE],
prefs_common.summary_col_size[S_COL_SCORE]);
+ gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_TAGS],
+ prefs_common.summary_col_size[S_COL_TAGS]);
if (prefs_common.enable_dotted_lines) {
gtk_ctree_set_line_style(GTK_CTREE(ctree), GTK_CTREE_LINES_DOTTED);
CLIST_BUTTON_SIGNAL_CONNECT(S_COL_SUBJECT, summary_subject_clicked);
CLIST_BUTTON_SIGNAL_CONNECT(S_COL_SCORE, summary_score_clicked);
CLIST_BUTTON_SIGNAL_CONNECT(S_COL_LOCKED, summary_locked_clicked);
+ CLIST_BUTTON_SIGNAL_CONNECT(S_COL_TAGS, summary_tags_clicked);
#undef CLIST_BUTTON_SIGNAL_CONNECT
summary_toggle_view(summaryview);
return TRUE;
}
-
+#ifdef MAEMO
+static void summary_toggle_multiple_pressed(GtkWidget *widget,
+ SummaryView *summaryview)
+{
+ GTK_SCTREE(summaryview->ctree)->force_additive_sel =
+ gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
+}
+#endif
static gboolean summary_button_pressed(GtkWidget *ctree, GdkEventButton *event,
SummaryView *summaryview)
{
mimeview_scroll_page(messageview->mimeview, TRUE);
break;
case GDK_Return: /* Scroll up/down one line */
+ case GDK_KP_Enter:
handled = TRUE;
if (summaryview->displayed != summaryview->selected) {
summary_display_msg(summaryview,
case GDK_y:
case GDK_t:
case GDK_l:
+ case GDK_o:
case GDK_c:
case GDK_a:
if ((event->state & (GDK_MOD1_MASK|GDK_CONTROL_MASK)) == 0) {
summary_sort_by_column_click(summaryview, SORT_BY_LOCKED);
}
+static void summary_tags_clicked(GtkWidget *button,
+ SummaryView *summaryview)
+{
+ summary_sort_by_column_click(summaryview, SORT_BY_TAGS);
+}
+
static void summary_start_drag(GtkWidget *widget, gint button, GdkEvent *event,
SummaryView *summaryview)
{
(msginfo1->subject, msginfo2->subject);
}
+static gint summary_cmp_by_thread_date(GtkCList *clist,
+ gconstpointer ptr1,
+ gconstpointer ptr2)
+{
+ MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data;
+ MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data;
+ gint thread_diff = msginfo1->thread_date - msginfo2->thread_date;
+
+ if (msginfo1->thread_date > 0 && msginfo2->thread_date > 0)
+ return thread_diff;
+ else
+ return msginfo1->date_t - msginfo2->date_t;
+}
+
static gint summary_cmp_by_from(GtkCList *clist, gconstpointer ptr1,
gconstpointer ptr2)
{
return g_utf8_collate(str1, str2);
}
+static gint summary_cmp_by_tags(GtkCList *clist, gconstpointer ptr1,
+ gconstpointer ptr2)
+{
+ const gchar *str1, *str2;
+ const GtkCListRow *r1 = (const GtkCListRow *) ptr1;
+ const GtkCListRow *r2 = (const GtkCListRow *) ptr2;
+ const SummaryView *sv = g_object_get_data(G_OBJECT(clist), "summaryview");
+ gint res;
+ g_return_val_if_fail(sv, -1);
+
+ str1 = GTK_CELL_TEXT(r1->cell[sv->col_pos[S_COL_TAGS]])->text;
+ str2 = GTK_CELL_TEXT(r2->cell[sv->col_pos[S_COL_TAGS]])->text;
+
+ if (!str1)
+ return str2 != NULL;
+
+ if (!str2)
+ return -1;
+
+ res = g_utf8_collate(str1, str2);
+ return (res != 0)? res: summary_cmp_by_date(clist, ptr1, ptr2);
+}
+
static gint summary_cmp_by_simplified_subject
(GtkCList *clist, gconstpointer ptr1, gconstpointer ptr2)
{
msginfo = gtk_ctree_node_get_row_data(ctree, row);
g_return_if_fail(msginfo);
+ summary_msginfo_unset_flags(msginfo, MSG_WATCH_THREAD, 0);
summary_msginfo_change_flags(msginfo, MSG_IGNORE_THREAD, 0, MSG_NEW | MSG_UNREAD, 0);
summary_set_row_marks(summaryview, row);
summary_ignore_thread(summaryview);
}
+static void summary_watch_thread_func(GtkCTree *ctree, GtkCTreeNode *row, gpointer data)
+{
+ SummaryView *summaryview = (SummaryView *) data;
+ MsgInfo *msginfo;
+
+ msginfo = gtk_ctree_node_get_row_data(ctree, row);
+ g_return_if_fail(msginfo);
+
+ summary_msginfo_change_flags(msginfo, MSG_WATCH_THREAD, 0, MSG_IGNORE_THREAD, 0);
+
+ summary_set_row_marks(summaryview, row);
+ debug_print("Message %d is marked as watch thread\n",
+ msginfo->msgnum);
+}
+
+void summary_watch_thread(SummaryView *summaryview)
+{
+ GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
+ GList *cur;
+ gboolean froze = FALSE;
+
+ START_LONG_OPERATION(summaryview, FALSE);
+ for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
+ gtk_ctree_pre_recursive(ctree, GTK_CTREE_NODE(cur->data),
+ GTK_CTREE_FUNC(summary_watch_thread_func),
+ summaryview);
+
+ END_LONG_OPERATION(summaryview);
+
+ summary_status_show(summaryview);
+}
+
+static void summary_unwatch_thread_func(GtkCTree *ctree, GtkCTreeNode *row, gpointer data)
+{
+ SummaryView *summaryview = (SummaryView *) data;
+ MsgInfo *msginfo;
+
+ msginfo = gtk_ctree_node_get_row_data(ctree, row);
+ g_return_if_fail(msginfo);
+
+ summary_msginfo_unset_flags(msginfo, MSG_WATCH_THREAD, 0);
+
+ summary_set_row_marks(summaryview, row);
+ debug_print("Message %d is marked as unwatch thread\n",
+ msginfo->msgnum);
+}
+
+void summary_unwatch_thread(SummaryView *summaryview)
+{
+ GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
+ GList *cur;
+ gboolean froze = FALSE;
+
+ START_LONG_OPERATION(summaryview, FALSE);
+ for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
+ gtk_ctree_pre_recursive(ctree, GTK_CTREE_NODE(cur->data),
+ GTK_CTREE_FUNC(summary_unwatch_thread_func),
+ summaryview);
+
+ END_LONG_OPERATION(summaryview);
+
+ summary_status_show(summaryview);
+}
+
+static void summary_check_watch_thread_func
+ (GtkCTree *ctree, GtkCTreeNode *row, gpointer data)
+{
+ MsgInfo *msginfo;
+ gint *found_watch = (gint *) data;
+
+ if (*found_watch) return;
+ else {
+ msginfo = gtk_ctree_node_get_row_data(ctree, row);
+ *found_watch = msginfo && MSG_IS_WATCH_THREAD(msginfo->flags);
+ }
+}
+
+void summary_toggle_watch_thread(SummaryView *summaryview)
+{
+ GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
+ GList *cur;
+ gint found_watch = 0;
+
+ for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
+ gtk_ctree_pre_recursive(ctree, GTK_CTREE_NODE(cur->data),
+ GTK_CTREE_FUNC(summary_check_watch_thread_func),
+ &found_watch);
+
+ if (found_watch)
+ summary_unwatch_thread(summaryview);
+ else
+ summary_watch_thread(summaryview);
+}
+
void summary_toggle_show_read_messages(SummaryView *summaryview)
{
FolderItemUpdateData source;
stock_pixmap_gdk(ctree, STOCK_PIXMAP_CLIP, &clipxpm, &clipxpmmask);
stock_pixmap_gdk(ctree, STOCK_PIXMAP_LOCKED, &lockedxpm, &lockedxpmmask);
stock_pixmap_gdk(ctree, STOCK_PIXMAP_IGNORETHREAD, &ignorethreadxpm, &ignorethreadxpmmask);
+ stock_pixmap_gdk(ctree, STOCK_PIXMAP_WATCHTHREAD, &watchthreadxpm, &watchthreadxpmmask);
stock_pixmap_gdk(ctree, STOCK_PIXMAP_CLIP_KEY, &clipkeyxpm, &clipkeyxpmmask);
stock_pixmap_gdk(ctree, STOCK_PIXMAP_KEY, &keyxpm, &keyxpmmask);
stock_pixmap_gdk(ctree, STOCK_PIXMAP_GPG_SIGNED, &gpgsignedxpm, &gpgsignedxpmmask);
gtk_widget_show(pixmap);
summaryview->quick_search_pixmap = pixmap;
+#ifdef MAEMO
+ pixmap = stock_pixmap_widget(summaryview->hbox, STOCK_PIXMAP_SELECTION);
+ gtk_container_remove (GTK_CONTAINER(summaryview->multiple_sel_togbtn),
+ summaryview->multiple_sel_image);
+ gtk_container_add(GTK_CONTAINER(summaryview->multiple_sel_togbtn), pixmap);
+ gtk_widget_show(pixmap);
+ summaryview->multiple_sel_togbtn = pixmap;
+#endif
+
folderview_unselect(summaryview->folderview);
folderview_select(summaryview->folderview, summaryview->folder_item);
summary_set_column_titles(summaryview);