[evolution] Bug 469141 - Traverse collapsed threads with Magic space bar
- From: Milan Crha <mcrha src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [evolution] Bug 469141 - Traverse collapsed threads with Magic space bar
- Date: Thu, 4 Dec 2014 07:36:17 +0000 (UTC)
commit 80d9a5f8b1de033a111c51f9e14548f20c8d0b20
Author: Milan Crha <mcrha redhat com>
Date: Thu Dec 4 08:35:10 2014 +0100
Bug 469141 - Traverse collapsed threads with Magic space bar
mail/message-list.c | 161 ++++++++++++++++++++++++++++--
mail/message-list.h | 3 +-
modules/mail/e-mail-shell-view-actions.c | 12 ++-
3 files changed, 162 insertions(+), 14 deletions(-)
---
diff --git a/mail/message-list.c b/mail/message-list.c
index db70ec3..1b026dc 100644
--- a/mail/message-list.c
+++ b/mail/message-list.c
@@ -822,11 +822,110 @@ clear_selection (MessageList *message_list,
}
static GNode *
+ml_get_next_node (GNode *node,
+ GNode *subroot)
+{
+ GNode *next;
+
+ if (!node)
+ return NULL;
+
+ next = g_node_first_child (node);
+
+ if (!next && node != subroot) {
+ next = g_node_next_sibling (node);
+ }
+
+ if (!next && node != subroot) {
+ next = node->parent;
+ while (next) {
+ GNode *sibl = g_node_next_sibling (next);
+
+ if (next == subroot)
+ return NULL;
+
+ if (sibl) {
+ next = sibl;
+ break;
+ } else {
+ next = next->parent;
+ }
+ }
+ }
+
+ return next;
+}
+
+static GNode *
+ml_get_prev_node (GNode *node,
+ GNode *subroot)
+{
+ GNode *prev;
+
+ if (!node)
+ return NULL;
+
+ if (node == subroot)
+ prev = NULL;
+ else
+ prev = g_node_prev_sibling (node);
+
+ if (!prev) {
+ prev = node->parent;
+
+ if (prev == subroot)
+ return NULL;
+
+ if (prev)
+ return prev;
+ }
+
+ if (prev) {
+ GNode *child = g_node_last_child (prev);
+ while (child) {
+ prev = child;
+ child = g_node_last_child (child);
+ }
+ }
+
+ return prev;
+}
+
+static GNode *
+ml_get_last_tree_node (GNode *node,
+ GNode *subroot)
+{
+ GNode *child;
+
+ if (!node)
+ return NULL;
+
+ while (node->parent && node->parent != subroot)
+ node = node->parent;
+
+ if (node == subroot)
+ child = node;
+ else
+ child = g_node_last_sibling (node);
+
+ if (!child)
+ child = node;
+
+ while (child = g_node_last_child (child), child) {
+ node = child;
+ }
+
+ return node;
+}
+
+static GNode *
ml_search_forward (MessageList *message_list,
gint start,
gint end,
guint32 flags,
- guint32 mask)
+ guint32 mask,
+ gboolean include_collapsed,
+ gboolean skip_first)
{
GNode *node;
gint row;
@@ -837,10 +936,22 @@ ml_search_forward (MessageList *message_list,
for (row = start; row <= end; row++) {
node = e_tree_table_adapter_node_at_row (etta, row);
- if (node != NULL
+ if (node != NULL && !skip_first
&& (info = get_message_info (message_list, node))
&& (camel_message_info_flags (info) & mask) == flags)
return node;
+
+ skip_first = FALSE;
+
+ if (node && include_collapsed && !e_tree_table_adapter_node_is_expanded (etta, node) &&
g_node_first_child (node)) {
+ GNode *subnode = node;
+
+ while (subnode = ml_get_next_node (subnode, node), subnode && subnode != node) {
+ if ((info = get_message_info (message_list, subnode)) &&
+ (camel_message_info_flags (info) & mask) == flags)
+ return subnode;
+ }
+ }
}
return NULL;
@@ -851,7 +962,9 @@ ml_search_backward (MessageList *message_list,
gint start,
gint end,
guint32 flags,
- guint32 mask)
+ guint32 mask,
+ gboolean include_collapsed,
+ gboolean skip_first)
{
GNode *node;
gint row;
@@ -862,10 +975,37 @@ ml_search_backward (MessageList *message_list,
for (row = start; row >= end; row--) {
node = e_tree_table_adapter_node_at_row (etta, row);
- if (node != NULL
+ if (node != NULL && !skip_first
&& (info = get_message_info (message_list, node))
- && (camel_message_info_flags (info) & mask) == flags)
+ && (camel_message_info_flags (info) & mask) == flags) {
+ if (include_collapsed && !e_tree_table_adapter_node_is_expanded (etta, node) &&
g_node_first_child (node)) {
+ GNode *subnode = ml_get_last_tree_node (g_node_first_child (node), node);
+
+ while (subnode && subnode != node) {
+ if ((info = get_message_info (message_list, subnode)) &&
+ (camel_message_info_flags (info) & mask) == flags)
+ return subnode;
+
+ subnode = ml_get_prev_node (subnode, node);
+ }
+ }
+
return node;
+ }
+
+ if (node && include_collapsed && !skip_first && !e_tree_table_adapter_node_is_expanded (etta,
node) && g_node_first_child (node)) {
+ GNode *subnode = ml_get_last_tree_node (g_node_first_child (node), node);
+
+ while (subnode && subnode != node) {
+ if ((info = get_message_info (message_list, subnode)) &&
+ (camel_message_info_flags (info) & mask) == flags)
+ return subnode;
+
+ subnode = ml_get_prev_node (subnode, node);
+ }
+ }
+
+ skip_first = FALSE;
}
return NULL;
@@ -878,6 +1018,7 @@ ml_search_path (MessageList *message_list,
guint32 mask)
{
ETreeTableAdapter *adapter;
+ gboolean include_collapsed;
GNode *node;
gint row_count;
gint row;
@@ -898,20 +1039,22 @@ ml_search_path (MessageList *message_list,
if (row == -1)
return NULL;
+ include_collapsed = (direction & MESSAGE_LIST_SELECT_INCLUDE_COLLAPSED) != 0;
+
if ((direction & MESSAGE_LIST_SELECT_DIRECTION) == MESSAGE_LIST_SELECT_NEXT)
node = ml_search_forward (
- message_list, row + 1, row_count - 1, flags, mask);
+ message_list, row, row_count - 1, flags, mask, include_collapsed, TRUE);
else
node = ml_search_backward (
- message_list, row - 1, 0, flags, mask);
+ message_list, row, 0, flags, mask, include_collapsed, TRUE);
if (node == NULL && (direction & MESSAGE_LIST_SELECT_WRAP)) {
if ((direction & MESSAGE_LIST_SELECT_DIRECTION) == MESSAGE_LIST_SELECT_NEXT)
node = ml_search_forward (
- message_list, 0, row, flags, mask);
+ message_list, 0, row, flags, mask, include_collapsed, FALSE);
else
node = ml_search_backward (
- message_list, row_count - 1, row, flags, mask);
+ message_list, row_count - 1, row, flags, mask, include_collapsed, FALSE);
}
return node;
diff --git a/mail/message-list.h b/mail/message-list.h
index ebc8a4e..e5c249a 100644
--- a/mail/message-list.h
+++ b/mail/message-list.h
@@ -144,7 +144,8 @@ typedef enum {
MESSAGE_LIST_SELECT_NEXT = 0,
MESSAGE_LIST_SELECT_PREVIOUS = 1,
MESSAGE_LIST_SELECT_DIRECTION = 1, /* direction mask */
- MESSAGE_LIST_SELECT_WRAP = 1<<1 /* option bit */
+ MESSAGE_LIST_SELECT_WRAP = 1 << 1, /* option bit */
+ MESSAGE_LIST_SELECT_INCLUDE_COLLAPSED = 1 << 2 /* whether to search collapsed nodes as well */
} MessageListSelectDirection;
GType message_list_get_type (void);
diff --git a/modules/mail/e-mail-shell-view-actions.c b/modules/mail/e-mail-shell-view-actions.c
index 17cd2dd..28da400 100644
--- a/modules/mail/e-mail-shell-view-actions.c
+++ b/modules/mail/e-mail-shell-view-actions.c
@@ -1342,14 +1342,16 @@ action_mail_smart_backward_cb (GtkAction *action,
if (message_list_select (
MESSAGE_LIST (message_list),
- MESSAGE_LIST_SELECT_PREVIOUS,
+ MESSAGE_LIST_SELECT_PREVIOUS |
+ MESSAGE_LIST_SELECT_INCLUDE_COLLAPSED,
0, CAMEL_MESSAGE_SEEN))
return;
if (message_list_select (
MESSAGE_LIST (message_list),
MESSAGE_LIST_SELECT_PREVIOUS |
- MESSAGE_LIST_SELECT_WRAP,
+ MESSAGE_LIST_SELECT_WRAP |
+ MESSAGE_LIST_SELECT_INCLUDE_COLLAPSED,
0, CAMEL_MESSAGE_SEEN))
return;
@@ -1427,14 +1429,16 @@ action_mail_smart_forward_cb (GtkAction *action,
if (message_list_select (
MESSAGE_LIST (message_list),
- MESSAGE_LIST_SELECT_NEXT,
+ MESSAGE_LIST_SELECT_NEXT |
+ MESSAGE_LIST_SELECT_INCLUDE_COLLAPSED,
0, CAMEL_MESSAGE_SEEN))
return;
if (message_list_select (
MESSAGE_LIST (message_list),
MESSAGE_LIST_SELECT_NEXT |
- MESSAGE_LIST_SELECT_WRAP,
+ MESSAGE_LIST_SELECT_WRAP |
+ MESSAGE_LIST_SELECT_INCLUDE_COLLAPSED,
0, CAMEL_MESSAGE_SEEN))
return;
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]