aboutsummaryrefslogtreecommitdiffstats
path: root/widgets/table
diff options
context:
space:
mode:
Diffstat (limited to 'widgets/table')
-rw-r--r--widgets/table/e-tree-model.c360
-rw-r--r--widgets/table/e-tree-model.h36
2 files changed, 257 insertions, 139 deletions
diff --git a/widgets/table/e-tree-model.c b/widgets/table/e-tree-model.c
index a9c10343e5..508fe1d7c5 100644
--- a/widgets/table/e-tree-model.c
+++ b/widgets/table/e-tree-model.c
@@ -31,13 +31,21 @@
static ETableModel *e_tree_model_parent_class;
-typedef struct {
- gboolean expanded;
- guint visible_descendents;
- char *save_id;
+struct ETreePath {
+ gboolean expanded;
+ guint visible_descendents;
+ char *save_id;
ETreePathCompareFunc compare;
- gpointer node_data;
-} ENode;
+ gpointer node_data;
+
+ /* parent/child/sibling pointers */
+ ETreePath *parent;
+ ETreePath *next_sibling;
+ ETreePath *prev_sibling;
+ ETreePath *first_child;
+ ETreePath *last_child;
+ guint32 num_children;
+};
enum {
NODE_CHANGED,
@@ -50,7 +58,98 @@ enum {
static guint e_tree_model_signals [LAST_SIGNAL] = {0, };
-static void add_visible_descendents_to_array (ETreeModel *etm, GNode *gnode, int *row, int *count);
+static void add_visible_descendents_to_array (ETreeModel *etm, ETreePath *node, int *row, int *count);
+
+
+/* ETreePath functions */
+
+static int
+e_tree_path_depth (ETreePath *path)
+{
+ int depth = 0;
+ while (path) {
+ depth ++;
+ path = path->parent;
+ }
+ return depth;
+}
+
+static void
+e_tree_path_insert (ETreePath *parent, int position, ETreePath *child)
+{
+ g_return_if_fail (position <= parent->num_children || position == -1);
+
+ child->parent = parent;
+
+ if (parent->first_child == NULL)
+ parent->first_child = child;
+
+ if (position == -1 || position == parent->num_children) {
+ child->prev_sibling = parent->last_child;
+ if (parent->last_child)
+ parent->last_child->next_sibling = child;
+ parent->last_child = child;
+ }
+ else {
+ ETreePath *c;
+ for (c = parent->first_child; c; c = c->next_sibling) {
+ if (position == 0) {
+ child->next_sibling = c;
+ child->prev_sibling = c->prev_sibling;
+
+ if (child->next_sibling)
+ child->next_sibling->prev_sibling = child;
+ if (child->prev_sibling)
+ child->prev_sibling->next_sibling = child;
+
+ if (parent->first_child == c)
+ parent->first_child = child;
+ break;
+ }
+ position --;
+ }
+ }
+
+ parent->num_children++;
+}
+
+static void
+e_tree_path_unlink (ETreePath *path)
+{
+ ETreePath *parent = path->parent;
+
+ /* unlink first/last child if applicable */
+ if (parent) {
+ if (path == parent->first_child)
+ parent->first_child = path->next_sibling;
+ if (path == parent->last_child)
+ parent->last_child = path->prev_sibling;
+
+ parent->num_children --;
+ }
+
+ /* unlink prev/next sibling links */
+ if (path->next_sibling)
+ path->next_sibling->prev_sibling = path->prev_sibling;
+ if (path->prev_sibling)
+ path->prev_sibling->next_sibling = path->next_sibling;
+
+ path->parent = NULL;
+ path->next_sibling = NULL;
+ path->prev_sibling = NULL;
+}
+
+void
+e_tree_model_node_traverse (ETreeModel *model, ETreePath *path, ETreePathFunc func, gpointer data)
+{
+ ETreePath *child;
+
+ for (child = path->first_child; child; child = child->next_sibling) {
+ e_tree_model_node_traverse (model, child, func, data);
+ if (func (model, child, data) == TRUE)
+ return;
+ }
+}
/* virtual methods */
@@ -95,7 +194,7 @@ etree_get_next (ETreeModel *etm, ETreePath *node)
{
g_return_val_if_fail (node, NULL);
- return g_node_next_sibling(node);
+ return node->next_sibling;
}
static ETreePath*
@@ -103,7 +202,23 @@ etree_get_prev (ETreeModel *etm, ETreePath *node)
{
g_return_val_if_fail (node, NULL);
- return g_node_prev_sibling (node);
+ return node->prev_sibling;
+}
+
+static ETreePath*
+etree_get_first_child (ETreeModel *etm, ETreePath *node)
+{
+ g_return_val_if_fail (node, NULL);
+
+ return node->first_child;
+}
+
+static ETreePath*
+etree_get_last_child (ETreeModel *etm, ETreePath *node)
+{
+ g_return_val_if_fail (node, NULL);
+
+ return node->last_child;
}
static guint
@@ -113,13 +228,14 @@ etree_get_children (ETreeModel *etm, ETreePath* node, ETreePath ***paths)
g_return_val_if_fail (node, 0);
- n_children = g_node_n_children (node);
+ n_children = node->num_children;
if (paths) {
- int i;
+ ETreePath *p;
+ int i = 0;
(*paths) = g_malloc (sizeof (ETreePath*) * n_children);
- for (i = 0; i < n_children; i ++) {
- (*paths)[i] = g_node_nth_child (node, i);
+ for (p = node->first_child; p; p = p->next_sibling) {
+ (*paths)[i++] = p;
}
}
@@ -129,9 +245,9 @@ etree_get_children (ETreeModel *etm, ETreePath* node, ETreePath ***paths)
static gboolean
etree_is_expanded (ETreeModel *etm, ETreePath* node)
{
- g_return_val_if_fail (node && node->data, FALSE);
+ g_return_val_if_fail (node, FALSE);
- return ((ENode*)node->data)->expanded;
+ return node->expanded;
}
static gboolean
@@ -140,7 +256,7 @@ etree_is_visible (ETreeModel *etm, ETreePath* node)
g_return_val_if_fail (node, FALSE);
for (node = node->parent; node; node = node->parent) {
- if (!((ENode*)node->data)->expanded)
+ if (!node->expanded)
return FALSE;
}
@@ -150,15 +266,12 @@ etree_is_visible (ETreeModel *etm, ETreePath* node)
static void
etree_set_expanded (ETreeModel *etm, ETreePath* node, gboolean expanded)
{
- GNode *child;
- ENode *enode;
+ ETreePath *child;
int row;
- g_return_if_fail (node && node->data);
-
- enode = ((ENode*)node->data);
+ g_return_if_fail (node);
- if (enode->expanded == expanded)
+ if (node->expanded == expanded)
return;
if (expanded) {
@@ -168,9 +281,9 @@ etree_set_expanded (ETreeModel *etm, ETreePath* node, gboolean expanded)
return;
}
- enode->expanded = expanded;
- if (enode->save_id) {
- g_hash_table_insert (etm->expanded_state, enode->save_id, (gpointer)expanded);
+ node->expanded = expanded;
+ if (node->save_id) {
+ g_hash_table_insert (etm->expanded_state, node->save_id, (gpointer)expanded);
if (expanded) {
/* the node previously was collapsed */
@@ -195,29 +308,28 @@ etree_set_expanded (ETreeModel *etm, ETreePath* node, gboolean expanded)
row++;
if (expanded) {
- GNode *parent;
+ ETreePath *parent;
if (e_tree_model_node_is_visible (etm, node)) {
- enode->visible_descendents = 0;
- for (child = g_node_first_child (node); child;
- child = g_node_next_sibling (child)) {
- add_visible_descendents_to_array (etm, child, &row, &enode->visible_descendents);
+ node->visible_descendents = 0;
+ for (child = node->first_child; child;
+ child = child->next_sibling) {
+ add_visible_descendents_to_array (etm, child, &row, &node->visible_descendents);
}
}
/* now iterate back up the tree, adding to our
ancestors' visible descendents */
for (parent = node->parent; parent; parent = parent->parent) {
- ENode *parent_enode = (ENode*)parent->data;
- parent_enode->visible_descendents += enode->visible_descendents;
+ parent->visible_descendents += node->visible_descendents;
}
}
else {
int i;
- GNode *parent;
+ ETreePath *parent;
if (e_tree_model_node_is_visible (etm, node)) {
- for (i = 0; i < enode->visible_descendents; i ++) {
+ for (i = 0; i < node->visible_descendents; i ++) {
etm->row_array = g_array_remove_index (etm->row_array, row);
e_table_model_row_deleted (E_TABLE_MODEL (etm), row);
}
@@ -226,12 +338,10 @@ etree_set_expanded (ETreeModel *etm, ETreePath* node, gboolean expanded)
ancestors' visible descendents */
for (parent = node->parent; parent; parent = parent->parent) {
- ENode *parent_enode = (ENode*)parent->data;
-
- parent_enode->visible_descendents -= enode->visible_descendents;
+ parent->visible_descendents -= node->visible_descendents;
}
- enode->visible_descendents = 0;
+ node->visible_descendents = 0;
e_tree_model_node_collapsed (etm, node);
}
@@ -262,7 +372,7 @@ etree_node_at_row (ETreeModel *etree, int row)
{
g_return_val_if_fail (row < etree->row_array->len, NULL);
- return g_array_index (etree->row_array, GNode*, row);
+ return g_array_index (etree->row_array, ETreePath*, row);
}
@@ -452,6 +562,8 @@ e_tree_model_class_init (GtkObjectClass *klass)
tree_class->get_root = etree_get_root;
tree_class->get_prev = etree_get_prev;
tree_class->get_next = etree_get_next;
+ tree_class->get_first_child = etree_get_first_child;
+ tree_class->get_last_child = etree_get_last_child;
tree_class->get_parent = etree_get_parent;
tree_class->value_at = etree_value_at;
@@ -522,7 +634,7 @@ e_tree_model_node_removed (ETreeModel *tree_model, ETreePath *parent_node, ETre
row = e_tree_model_row_of_node (tree_model, removed_node);
if (row != -1)
- e_table_model_row_inserted (E_TABLE_MODEL (tree_model), row);
+ e_table_model_row_deleted (E_TABLE_MODEL (tree_model), row);
gtk_signal_emit (GTK_OBJECT (tree_model),
e_tree_model_signals [NODE_REMOVED],
@@ -557,7 +669,7 @@ e_tree_model_construct (ETreeModel *etree)
{
etree->root = NULL;
etree->root_visible = TRUE;
- etree->row_array = g_array_new (FALSE, FALSE, sizeof(GNode*));
+ etree->row_array = g_array_new (FALSE, FALSE, sizeof(ETreePath*));
etree->expanded_state = g_hash_table_new (g_str_hash, g_str_equal);
}
@@ -595,7 +707,7 @@ e_tree_model_row_of_node (ETreeModel *etree, ETreePath *node)
int i;
for (i = 0; i < etree->row_array->len; i ++)
- if (g_array_index (etree->row_array, GNode*, i) == node)
+ if (g_array_index (etree->row_array, ETreePath*, i) == node)
return i;
return -1;
@@ -627,6 +739,18 @@ e_tree_model_root_node_is_visible (ETreeModel *etm)
return etm->root_visible;
}
+ETreePath *
+e_tree_model_node_get_first_child (ETreeModel *etree, ETreePath *node)
+{
+ return ETM_CLASS(etree)->get_first_child(etree, node);
+}
+
+ETreePath *
+e_tree_model_node_get_last_child (ETreeModel *etree, ETreePath *node)
+{
+ return ETM_CLASS(etree)->get_last_child(etree, node);
+}
+
ETreePath *
e_tree_model_node_get_next (ETreeModel *etree, ETreePath *node)
@@ -643,7 +767,7 @@ e_tree_model_node_get_prev (ETreeModel *etree, ETreePath *node)
guint
e_tree_model_node_depth (ETreeModel *etree, ETreePath *path)
{
- return g_node_depth (path) - 1;
+ return e_tree_path_depth (path) - 1;
}
ETreePath *
@@ -697,33 +821,23 @@ e_tree_model_node_get_children (ETreeModel *etree, ETreePath *path, ETreePath **
guint
e_tree_model_node_num_visible_descendents (ETreeModel *etm, ETreePath *node)
{
- ENode *enode = (ENode*)node->data;
-
- return enode->visible_descendents;
+ return node->visible_descendents;
}
gpointer
e_tree_model_node_get_data (ETreeModel *etm, ETreePath *node)
{
- ENode *enode;
-
- g_return_val_if_fail (node && node->data, NULL);
-
- enode = (ENode*)node->data;
+ g_return_val_if_fail (node, NULL);
- return enode->node_data;
+ return node->node_data;
}
void
e_tree_model_node_set_data (ETreeModel *etm, ETreePath *node, gpointer node_data)
{
- ENode *enode;
-
- g_return_if_fail (node && node->data);
-
- enode = (ENode*)node->data;
+ g_return_if_fail (node);
- enode->node_data = node_data;
+ node->node_data = node_data;
}
ETreePath*
@@ -732,32 +846,26 @@ e_tree_model_node_insert (ETreeModel *tree_model,
int position,
gpointer node_data)
{
- ENode *node;
ETreePath *new_path;
-
g_return_val_if_fail (parent_path != NULL || tree_model->root == NULL, NULL);
- node = g_new0 (ENode, 1);
+ new_path = g_new0 (ETreePath, 1);
- node->expanded = FALSE;
- node->node_data = node_data;
+ new_path->expanded = FALSE;
+ new_path->node_data = node_data;
if (parent_path != NULL) {
- new_path = g_node_new (node);
-
- g_node_insert (parent_path, position, new_path);
+ e_tree_path_insert (parent_path, position, new_path);
if (e_tree_model_node_is_visible (tree_model, new_path)) {
int parent_row;
- GNode *n;
+ ETreePath *n;
/* we need to iterate back up to the root, incrementing the number of visible
descendents */
for (n = parent_path; n; n = n->parent) {
- ENode *parent_enode = (ENode*)n->data;
-
- parent_enode->visible_descendents ++;
+ n->visible_descendents ++;
}
/* finally, insert a row into the table */
@@ -773,7 +881,7 @@ e_tree_model_node_insert (ETreeModel *tree_model,
}
}
else {
- tree_model->root = g_node_new (node);
+ tree_model->root = new_path;
if (tree_model->root_visible) {
tree_model->row_array = g_array_insert_val (tree_model->row_array, 0, tree_model->root);
e_table_model_row_inserted (E_TABLE_MODEL (tree_model), 0);
@@ -781,9 +889,8 @@ e_tree_model_node_insert (ETreeModel *tree_model,
else {
/* need to mark the new node as expanded or
we'll never see it's children */
- node->expanded = TRUE;
+ new_path->expanded = TRUE;
}
- new_path = tree_model->root;
}
return new_path;
@@ -810,26 +917,31 @@ e_tree_model_node_insert_before (ETreeModel *etree,
ETreePath *sibling,
gpointer node_data)
{
- return e_tree_model_node_insert (etree, parent,
- g_node_child_position (parent, sibling),
- node_data);
+ ETreePath *child;
+ int position = 0;
+ for (child = parent->first_child; child; child = child->next_sibling) {
+ if (child == sibling)
+ break;
+ position ++;
+ }
+ return e_tree_model_node_insert (etree, parent, position, node_data);
}
-static void
-child_remove (GNode *node, gpointer etree)
+static gboolean
+child_remove (ETreeModel *model, ETreePath *node, gpointer data)
{
- e_tree_model_node_remove (etree, node);
+ e_tree_model_node_remove (model, node);
+ return FALSE;
}
gpointer
e_tree_model_node_remove (ETreeModel *etree, ETreePath *path)
{
- GNode *parent = path->parent;
- ENode *enode = (ENode*)path->data;
- gpointer ret = enode->node_data;
+ ETreePath *parent = path->parent;
+ gpointer ret = path->node_data;
/* remove children */
- g_node_children_foreach (path, G_TRAVERSE_ALL, child_remove, etree);
+ e_tree_model_node_traverse (etree, path, child_remove, etree);
/* clean up the display */
if (parent) {
@@ -841,9 +953,7 @@ e_tree_model_node_remove (ETreeModel *etree, ETreePath *path)
/* we need to iterate back up to the root, incrementing the number of visible
descendents */
for (; parent; parent = parent->parent) {
- ENode *parent_enode = (ENode*)parent->data;
-
- parent_enode->visible_descendents --;
+ parent->visible_descendents --;
}
}
}
@@ -859,32 +969,30 @@ e_tree_model_node_remove (ETreeModel *etree, ETreePath *path)
return NULL;
}
+ /* unlink the node */
+ e_tree_path_unlink (path);
/* now free up the storage from that path */
- g_node_destroy (path);
- g_free (enode);
+ g_free (path);
return ret;
}
static void
-add_visible_descendents_to_array (ETreeModel *etm, GNode *gnode, int *row, int *count)
+add_visible_descendents_to_array (ETreeModel *etm, ETreePath *node, int *row, int *count)
{
- GNode *child;
- ENode *enode;
-
/* add a row for this node */
- etm->row_array = g_array_insert_val (etm->row_array, (*row), gnode);
+ etm->row_array = g_array_insert_val (etm->row_array, (*row), node);
e_table_model_row_inserted (E_TABLE_MODEL (etm), (*row)++);
(*count) ++;
/* then loop over its children, calling this routine for each
of them */
- enode = (ENode*)gnode->data;
- if (enode->expanded) {
- for (child = g_node_first_child (gnode); child;
- child = g_node_next_sibling (child)) {
+ if (node->expanded) {
+ ETreePath *child;
+ for (child = node->first_child; child;
+ child = child->next_sibling) {
add_visible_descendents_to_array (etm, child, row, count);
}
}
@@ -1006,14 +1114,11 @@ e_tree_model_load_expanded_state (ETreeModel *etm, const char *filename)
void
e_tree_model_node_set_save_id (ETreeModel *etm, ETreePath *node, const char *id)
{
- ENode *enode;
char *key;
gboolean expanded_state;
g_return_if_fail (E_TREE_MODEL (etm));
- g_return_if_fail (node && node->data);
-
- enode = (ENode*)node->data;
+ g_return_if_fail (node);
if (g_hash_table_lookup_extended (etm->expanded_state,
id, (gpointer*)&key, (gpointer*)&expanded_state)) {
@@ -1027,14 +1132,14 @@ e_tree_model_node_set_save_id (ETreeModel *etm, ETreePath *node, const char *id)
etm->num_collapsed_to_save ++;
/* important that this comes after the e_tree_model_node_set_expanded */
- enode->save_id = key;
+ node->save_id = key;
}
else {
- enode->save_id = g_strdup (id);
+ node->save_id = g_strdup (id);
- g_hash_table_insert (etm->expanded_state, enode->save_id, (gpointer)enode->expanded);
+ g_hash_table_insert (etm->expanded_state, node->save_id, (gpointer)node->expanded);
- if (enode->expanded)
+ if (node->expanded)
etm->num_expanded_to_save ++;
else
etm->num_collapsed_to_save ++;
@@ -1048,17 +1153,14 @@ e_tree_model_node_set_compare_function (ETreeModel *tree_model,
ETreePath *node,
ETreePathCompareFunc compare)
{
- ENode *enode;
gboolean need_sort;
g_return_if_fail (E_TREE_MODEL (tree_model));
- g_return_if_fail (node && node->data);
-
- enode = (ENode*)node->data;
+ g_return_if_fail (node);
- need_sort = (compare != enode->compare);
+ need_sort = (compare != node->compare);
- enode->compare = compare;
+ node->compare = compare;
if (need_sort)
e_tree_model_node_sort (tree_model, node);
@@ -1081,19 +1183,16 @@ void
e_tree_model_node_sort (ETreeModel *tree_model,
ETreePath *node)
{
- int num_nodes = g_node_n_children (node);
+ int num_nodes = node->num_children;
ETreeSortInfo *sort_info;
int i;
int child_index;
gboolean node_expanded = e_tree_model_node_is_expanded (tree_model, node);
- ENode *enode;
-
+
g_return_if_fail (E_TREE_MODEL (tree_model));
- g_return_if_fail (node && node->data);
-
- enode = (ENode*)node->data;
+ g_return_if_fail (node);
- g_return_if_fail (enode->compare);
+ g_return_if_fail (node->compare);
if (num_nodes == 0)
return;
@@ -1102,25 +1201,36 @@ e_tree_model_node_sort (ETreeModel *tree_model,
child_index = e_tree_model_row_of_node (tree_model, node) + 1;
+ /* collect our info and remove the children */
for (i = 0; i < num_nodes; i ++) {
-
- sort_info[i].path = g_node_first_child(node);
+ sort_info[i].path = node->first_child;
sort_info[i].was_expanded = e_tree_model_node_is_expanded (tree_model, sort_info[i].path);
sort_info[i].model = tree_model;
- sort_info[i].compare = enode->compare;
+ sort_info[i].compare = node->compare;
e_tree_model_node_set_expanded(tree_model, sort_info[i].path, FALSE);
if (node_expanded)
tree_model->row_array = g_array_remove_index (tree_model->row_array, child_index);
- g_node_unlink (sort_info[i].path);
+ e_tree_path_unlink (sort_info[i].path);
}
+ /* sort things */
qsort (sort_info, num_nodes, sizeof(ETreeSortInfo), (GCompareFunc)e_tree_model_node_compare);
- for (i = num_nodes - 1; i >= 0; i --) {
- g_node_prepend (node, sort_info[i].path);
+ /* reinsert the children nodes into the tree in the sorted order */
+ for (i = 0; i < num_nodes; i ++) {
+ e_tree_path_insert (node, i, sort_info[i].path);
if (node_expanded)
- tree_model->row_array = g_array_insert_val (tree_model->row_array, child_index, sort_info[i].path);
+ tree_model->row_array = g_array_insert_val (tree_model->row_array, child_index + i,
+ sort_info[i].path);
+ }
+
+ /* make another pass expanding the children as needed.
+
+ XXX this used to be in the loop above, but a recent change
+ (either here or in the etable code) causes an assertion and
+ a crash */
+ for (i = 0; i < num_nodes; i ++) {
e_tree_model_node_set_expanded (tree_model, sort_info[i].path, sort_info[i].was_expanded);
}
diff --git a/widgets/table/e-tree-model.h b/widgets/table/e-tree-model.h
index 8a8e4c177d..13857e6707 100644
--- a/widgets/table/e-tree-model.h
+++ b/widgets/table/e-tree-model.h
@@ -11,18 +11,19 @@
#define E_IS_TREE_MODEL(o) (GTK_CHECK_TYPE ((o), E_TREE_MODEL_TYPE))
#define E_IS_TREE_MODEL_CLASS(k) (GTK_CHECK_CLASS_TYPE ((k), E_TREE_MODEL_TYPE))
-typedef GNode ETreePath;
+typedef struct ETreePath ETreePath;
typedef struct ETreeModel ETreeModel;
typedef struct ETreeModelClass ETreeModelClass;
-typedef gint (*ETreePathCompareFunc)(ETreeModel *, ETreePath *path1, ETreePath *path2);
+typedef gint (*ETreePathCompareFunc)(ETreeModel *model, ETreePath *path1, ETreePath *path2);
+typedef gboolean (*ETreePathFunc)(ETreeModel *model, ETreePath *path, gpointer data);
struct ETreeModel {
ETableModel base;
- GNode *root;
- gboolean root_visible;
+ ETreePath *root;
+ gboolean root_visible;
GArray *row_array; /* used in the mapping between ETable and our tree */
- guint32 num_expanded_to_save;
- guint32 num_collapsed_to_save;
+ guint32 num_expanded_to_save;
+ guint32 num_collapsed_to_save;
GHashTable *expanded_state; /* used for loading/saving expanded state */
GString *sort_group; /* for caching the last sort group info */
};
@@ -35,10 +36,12 @@ struct ETreeModelClass {
*/
ETreePath *(*get_root) (ETreeModel *etm);
- ETreePath *(*get_parent) (ETreeModel *etm, ETreePath* node);
- ETreePath *(*get_next) (ETreeModel *etm, ETreePath* node);
- ETreePath *(*get_prev) (ETreeModel *etm, ETreePath* node);
- guint (*get_children) (ETreeModel *etm, ETreePath* node, ETreePath ***paths);
+ ETreePath *(*get_parent) (ETreeModel *etm, ETreePath* node);
+ ETreePath *(*get_first_child) (ETreeModel *etm, ETreePath* node);
+ ETreePath *(*get_last_child) (ETreeModel *etm, ETreePath* node);
+ ETreePath *(*get_next) (ETreeModel *etm, ETreePath* node);
+ ETreePath *(*get_prev) (ETreeModel *etm, ETreePath* node);
+ guint (*get_children) (ETreeModel *etm, ETreePath* node, ETreePath ***paths);
gboolean (*is_expanded) (ETreeModel *etm, ETreePath* node);
gboolean (*is_visible) (ETreeModel *etm, ETreePath* node);
@@ -73,10 +76,12 @@ void e_tree_model_construct (ETreeModel *etree);
ETreeModel *e_tree_model_new (void);
/* tree traversal operations */
-ETreePath *e_tree_model_get_root (ETreeModel *etree);
-ETreePath *e_tree_model_node_get_parent (ETreeModel *etree, ETreePath *path);
-ETreePath *e_tree_model_node_get_next (ETreeModel *etree, ETreePath *path);
-ETreePath *e_tree_model_node_get_prev (ETreeModel *etree, ETreePath *path);
+ETreePath *e_tree_model_get_root (ETreeModel *etree);
+ETreePath *e_tree_model_node_get_parent (ETreeModel *etree, ETreePath *path);
+ETreePath *e_tree_model_node_get_first_child (ETreeModel *etree, ETreePath *path);
+ETreePath *e_tree_model_node_get_last_child (ETreeModel *etree, ETreePath *path);
+ETreePath *e_tree_model_node_get_next (ETreeModel *etree, ETreePath *path);
+ETreePath *e_tree_model_node_get_prev (ETreeModel *etree, ETreePath *path);
/* node operations */
ETreePath *e_tree_model_node_insert (ETreeModel *etree, ETreePath *parent, int position, gpointer node_data);
@@ -123,4 +128,7 @@ void e_tree_model_node_set_save_id (ETreeModel *etm, ETreePath *node, c
ETreePath* e_tree_model_node_insert_id (ETreeModel *tree_model, ETreePath *parent_path,
int position, gpointer node_data, const char *save_id);
+/* depth first traversal of path's descendents, calling func on each one */
+void e_tree_model_node_traverse (ETreeModel *model, ETreePath *path, ETreePathFunc func, gpointer data);
+
#endif /* _E_TREE_MODEL_H */