forked from AuroraMiddleware/gtk
a11y: Redo indexing
We now index rows by actual expanded row number and don't count them ourselves no more.
This commit is contained in:
parent
92a2284bb2
commit
acc5627e39
@ -125,7 +125,8 @@ static void activate_cell (GtkCellAccessible *
|
||||
static void cell_destroyed (gpointer data);
|
||||
static void cell_info_new (GtkTreeViewAccessible *accessible,
|
||||
GtkTreeModel *tree_model,
|
||||
GtkTreePath *path,
|
||||
GtkRBTree *tree,
|
||||
GtkRBNode *node,
|
||||
GtkTreeViewColumn *tv_col,
|
||||
GtkCellAccessible *cell);
|
||||
static GtkCellAccessible *find_cell (GtkTreeViewAccessible *accessible,
|
||||
@ -147,9 +148,10 @@ static void count_rows (GtkTreeModel
|
||||
gint level,
|
||||
gint depth);
|
||||
|
||||
static gboolean get_path_column_from_index (GtkTreeView *tree_view,
|
||||
static gboolean get_rbtree_column_from_index (GtkTreeView *tree_view,
|
||||
gint index,
|
||||
GtkTreePath **path,
|
||||
GtkRBTree **tree,
|
||||
GtkRBNode **node,
|
||||
GtkTreeViewColumn **column);
|
||||
|
||||
static GtkTreeViewAccessibleCellInfo* find_cell_info (GtkTreeViewAccessible *view,
|
||||
@ -290,12 +292,10 @@ gtk_tree_view_accessible_initialize (AtkObject *obj,
|
||||
G_CALLBACK (focus_out), NULL);
|
||||
|
||||
accessible->tree_model = tree_model;
|
||||
accessible->n_rows = 0;
|
||||
accessible->n_cols = 0;
|
||||
if (tree_model)
|
||||
{
|
||||
g_object_add_weak_pointer (G_OBJECT (accessible->tree_model), (gpointer *)&accessible->tree_model);
|
||||
count_rows (tree_model, NULL, NULL, &accessible->n_rows, 0, G_MAXINT);
|
||||
connect_model_signals (tree_view, accessible);
|
||||
|
||||
if (gtk_tree_model_get_flags (tree_model) & GTK_TREE_MODEL_LIST_ONLY)
|
||||
@ -381,12 +381,10 @@ gtk_tree_view_accessible_notify_gtk (GObject *obj,
|
||||
disconnect_model_signals (accessible);
|
||||
g_hash_table_remove_all (accessible->cell_infos);
|
||||
accessible->tree_model = tree_model;
|
||||
accessible->n_rows = 0;
|
||||
|
||||
if (tree_model)
|
||||
{
|
||||
g_object_add_weak_pointer (G_OBJECT (accessible->tree_model), (gpointer *)&accessible->tree_model);
|
||||
count_rows (tree_model, NULL, NULL, &accessible->n_rows, 0, G_MAXINT);
|
||||
connect_model_signals (tree_view, accessible);
|
||||
|
||||
if (gtk_tree_model_get_flags (tree_model) & GTK_TREE_MODEL_LIST_ONLY)
|
||||
@ -479,18 +477,33 @@ gtk_tree_view_accessible_connect_widget_destroyed (GtkAccessible *accessible)
|
||||
GTK_ACCESSIBLE_CLASS (_gtk_tree_view_accessible_parent_class)->connect_widget_destroyed (accessible);
|
||||
}
|
||||
|
||||
static gint
|
||||
get_n_rows (GtkTreeView *tree_view)
|
||||
{
|
||||
GtkRBTree *tree;
|
||||
|
||||
tree = _gtk_tree_view_get_rbtree (tree_view);
|
||||
|
||||
if (tree == NULL)
|
||||
return 0;
|
||||
|
||||
return tree->root->total_count;
|
||||
}
|
||||
|
||||
static gint
|
||||
gtk_tree_view_accessible_get_n_children (AtkObject *obj)
|
||||
{
|
||||
GtkWidget *widget;
|
||||
GtkTreeViewAccessible *accessible;
|
||||
GtkTreeView *tree_view;
|
||||
|
||||
widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (obj));
|
||||
if (widget == NULL)
|
||||
return 0;
|
||||
|
||||
tree_view = GTK_TREE_VIEW (widget);
|
||||
accessible = GTK_TREE_VIEW_ACCESSIBLE (obj);
|
||||
return (accessible->n_rows + 1) * accessible->n_cols;
|
||||
return (get_n_rows (tree_view) + 1) * accessible->n_cols;
|
||||
}
|
||||
|
||||
static AtkObject *
|
||||
@ -507,6 +520,8 @@ gtk_tree_view_accessible_ref_child (AtkObject *obj,
|
||||
GtkTreeViewColumn *tv_col;
|
||||
GtkTreeSelection *selection;
|
||||
GtkTreePath *path;
|
||||
GtkRBTree *tree;
|
||||
GtkRBNode *node;
|
||||
AtkObject *child;
|
||||
AtkObject *parent;
|
||||
GtkTreeViewColumn *expander_tv;
|
||||
@ -549,10 +564,11 @@ gtk_tree_view_accessible_ref_child (AtkObject *obj,
|
||||
else
|
||||
focus_index = -1;
|
||||
|
||||
/* Find the TreePath and GtkTreeViewColumn for the index */
|
||||
if (!get_path_column_from_index (tree_view, i, &path, &tv_col))
|
||||
/* Find the RBTree and GtkTreeViewColumn for the index */
|
||||
if (!get_rbtree_column_from_index (tree_view, i, &tree, &node, &tv_col))
|
||||
return NULL;
|
||||
|
||||
path = _gtk_tree_view_find_path (tree_view, tree, node);
|
||||
tree_model = gtk_tree_view_get_model (tree_view);
|
||||
retval = gtk_tree_model_get_iter (tree_model, &iter, path);
|
||||
if (!retval)
|
||||
@ -561,12 +577,12 @@ gtk_tree_view_accessible_ref_child (AtkObject *obj,
|
||||
expander_tv = gtk_tree_view_get_expander_column (tree_view);
|
||||
is_expander = FALSE;
|
||||
is_expanded = FALSE;
|
||||
if (gtk_tree_model_iter_has_child (tree_model, &iter))
|
||||
if (GTK_RBNODE_FLAG_SET (node, GTK_RBNODE_IS_PARENT))
|
||||
{
|
||||
if (expander_tv == tv_col)
|
||||
{
|
||||
is_expander = TRUE;
|
||||
is_expanded = gtk_tree_view_row_expanded (tree_view, path);
|
||||
is_expanded = node->children != NULL;
|
||||
}
|
||||
}
|
||||
gtk_tree_view_column_cell_set_cell_data (tv_col, tree_model, &iter,
|
||||
@ -590,7 +606,7 @@ gtk_tree_view_accessible_ref_child (AtkObject *obj,
|
||||
* be before the ones for the cells so that the first one we find for
|
||||
* a position will be for the container
|
||||
*/
|
||||
cell_info_new (accessible, tree_model, path, tv_col, container_cell);
|
||||
cell_info_new (accessible, tree_model, tree, node, tv_col, container_cell);
|
||||
parent = ATK_OBJECT (container);
|
||||
}
|
||||
else
|
||||
@ -612,7 +628,7 @@ gtk_tree_view_accessible_ref_child (AtkObject *obj,
|
||||
renderer_cell->renderer = fake_renderer;
|
||||
|
||||
/* Create the GtkTreeViewAccessibleCellInfo structure for this cell */
|
||||
cell_info_new (accessible, tree_model, path, tv_col, cell);
|
||||
cell_info_new (accessible, tree_model, tree, node, tv_col, cell);
|
||||
|
||||
_gtk_cell_accessible_initialise (cell, widget, parent);
|
||||
|
||||
@ -646,7 +662,7 @@ gtk_tree_view_accessible_ref_child (AtkObject *obj,
|
||||
renderer_cell = GTK_RENDERER_CELL_ACCESSIBLE (child);
|
||||
|
||||
/* Create the GtkTreeViewAccessibleCellInfo for this cell */
|
||||
cell_info_new (accessible, tree_model, path, tv_col, cell);
|
||||
cell_info_new (accessible, tree_model, tree, node, tv_col, cell);
|
||||
|
||||
_gtk_cell_accessible_initialise (cell, widget, parent);
|
||||
|
||||
@ -926,21 +942,19 @@ gtk_tree_view_accessible_get_row_at_index (AtkTable *table,
|
||||
{
|
||||
GtkWidget *widget;
|
||||
GtkTreeView *tree_view;
|
||||
GtkTreePath *path;
|
||||
|
||||
widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (table));
|
||||
if (widget == NULL)
|
||||
return -1;
|
||||
|
||||
tree_view = GTK_TREE_VIEW (widget);
|
||||
if (get_path_column_from_index (tree_view, index, &path, NULL))
|
||||
{
|
||||
gint row = get_row_from_tree_path (tree_view, path);
|
||||
gtk_tree_path_free (path);
|
||||
return row;
|
||||
}
|
||||
else
|
||||
|
||||
index /= GTK_TREE_VIEW_ACCESSIBLE (table)->n_cols;
|
||||
index--;
|
||||
if (index >= get_n_rows (tree_view))
|
||||
return -1;
|
||||
|
||||
return index;
|
||||
}
|
||||
|
||||
static AtkObject *
|
||||
@ -961,20 +975,12 @@ static gint
|
||||
gtk_tree_view_accessible_get_n_rows (AtkTable *table)
|
||||
{
|
||||
GtkWidget *widget;
|
||||
GtkTreeView *tree_view;
|
||||
GtkRBTree *tree;
|
||||
|
||||
widget = gtk_accessible_get_widget (GTK_ACCESSIBLE (table));
|
||||
if (widget == NULL)
|
||||
return 0;
|
||||
|
||||
tree_view = GTK_TREE_VIEW (widget);
|
||||
tree = _gtk_tree_view_get_rbtree (tree_view);
|
||||
|
||||
if (tree == NULL)
|
||||
return 0;
|
||||
|
||||
return tree->root->parity;
|
||||
return get_n_rows (GTK_TREE_VIEW (widget));
|
||||
}
|
||||
|
||||
static gint
|
||||
@ -1800,7 +1806,7 @@ columns_changed (GtkTreeView *tree_view)
|
||||
g_signal_emit_by_name (atk_obj, "column-inserted", column_count, 1);
|
||||
|
||||
/* Generate children-changed signals */
|
||||
for (row = 0; row < accessible->n_rows; row++)
|
||||
for (row = 0; row < get_n_rows (tree_view); row++)
|
||||
{
|
||||
/* Pass NULL as the child object, i.e. 4th argument */
|
||||
g_signal_emit_by_name (atk_obj, "children-changed::add",
|
||||
@ -1842,7 +1848,7 @@ columns_changed (GtkTreeView *tree_view)
|
||||
g_signal_emit_by_name (atk_obj, "column-deleted", i, 1);
|
||||
|
||||
/* Generate children-changed signals */
|
||||
for (row = 0; row < accessible->n_rows; row++)
|
||||
for (row = 0; row < get_n_rows (tree_view); row++)
|
||||
{
|
||||
/* Pass NULL as the child object, 4th argument */
|
||||
g_signal_emit_by_name (atk_obj, "children-changed::remove",
|
||||
@ -2039,7 +2045,6 @@ model_row_inserted (GtkTreeModel *tree_model,
|
||||
|
||||
atk_obj = gtk_widget_get_accessible (GTK_WIDGET (tree_view));
|
||||
accessible = GTK_TREE_VIEW_ACCESSIBLE (atk_obj);
|
||||
accessible->n_rows++;
|
||||
|
||||
if (accessible->idle_expand_id)
|
||||
{
|
||||
@ -2137,8 +2142,6 @@ model_row_deleted (GtkTreeModel *tree_model,
|
||||
atk_obj = gtk_widget_get_accessible (GTK_WIDGET (tree_view));
|
||||
accessible = GTK_TREE_VIEW_ACCESSIBLE (atk_obj);
|
||||
|
||||
accessible->n_rows--;
|
||||
|
||||
if (accessible->idle_expand_id)
|
||||
{
|
||||
g_source_remove (accessible->idle_expand_id);
|
||||
@ -3073,7 +3076,8 @@ cell_info_get_index (GtkTreeView *tree_view,
|
||||
static void
|
||||
cell_info_new (GtkTreeViewAccessible *accessible,
|
||||
GtkTreeModel *tree_model,
|
||||
GtkTreePath *path,
|
||||
GtkRBTree *tree,
|
||||
GtkRBNode *node,
|
||||
GtkTreeViewColumn *tv_col,
|
||||
GtkCellAccessible *cell)
|
||||
{
|
||||
@ -3081,16 +3085,8 @@ cell_info_new (GtkTreeViewAccessible *accessible,
|
||||
|
||||
cell_info = g_new (GtkTreeViewAccessibleCellInfo, 1);
|
||||
|
||||
if (!_gtk_tree_view_find_node (GTK_TREE_VIEW (gtk_accessible_get_widget (GTK_ACCESSIBLE (accessible))),
|
||||
path,
|
||||
&cell_info->tree,
|
||||
&cell_info->node))
|
||||
{
|
||||
/* This check needs to happen way earlier */
|
||||
g_free (cell_info);
|
||||
return;
|
||||
}
|
||||
|
||||
cell_info->tree = tree;
|
||||
cell_info->node = node;
|
||||
cell_info->cell_col_ref = tv_col;
|
||||
cell_info->cell = cell;
|
||||
cell_info->view = accessible;
|
||||
@ -3285,165 +3281,19 @@ count_rows (GtkTreeModel *model,
|
||||
}
|
||||
}
|
||||
|
||||
/* Find the next node, which has children, at the specified depth below
|
||||
* the specified iter. The level is the depth of the current iter.
|
||||
* The position of the node is returned in path and the return value
|
||||
* of TRUE means that a node was found.
|
||||
*/
|
||||
|
||||
static gboolean
|
||||
get_next_node_with_child_at_depth (GtkTreeModel *model,
|
||||
GtkTreeIter *iter,
|
||||
GtkTreePath **path,
|
||||
gint level,
|
||||
gint depth)
|
||||
{
|
||||
GtkTreeIter child_iter;
|
||||
|
||||
*path = NULL;
|
||||
|
||||
if (gtk_tree_model_iter_children (model, &child_iter, iter))
|
||||
{
|
||||
level++;
|
||||
|
||||
while (TRUE)
|
||||
{
|
||||
while (!gtk_tree_model_iter_has_child (model, &child_iter))
|
||||
{
|
||||
if (!gtk_tree_model_iter_next (model, &child_iter))
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
if (level == depth)
|
||||
{
|
||||
/* We have found what we were looking for */
|
||||
*path = gtk_tree_model_get_path (model, &child_iter);
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
if (get_next_node_with_child_at_depth (model, &child_iter, path,
|
||||
level, depth))
|
||||
return TRUE;
|
||||
|
||||
if (!gtk_tree_model_iter_next (model, &child_iter))
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
/* Find the next node, which has children, at the same depth
|
||||
* as the specified GtkTreePath.
|
||||
*/
|
||||
static gboolean
|
||||
get_next_node_with_child (GtkTreeModel *model,
|
||||
GtkTreePath *path,
|
||||
GtkTreePath **return_path)
|
||||
{
|
||||
GtkTreeIter iter;
|
||||
gint depth;
|
||||
|
||||
gtk_tree_model_get_iter (model, &iter, path);
|
||||
|
||||
while (gtk_tree_model_iter_next (model, &iter))
|
||||
{
|
||||
if (gtk_tree_model_iter_has_child (model, &iter))
|
||||
{
|
||||
*return_path = gtk_tree_model_get_path (model, &iter);
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
depth = gtk_tree_path_get_depth (path);
|
||||
while (gtk_tree_path_up (path))
|
||||
{
|
||||
if (gtk_tree_path_get_depth (path) == 0)
|
||||
break;
|
||||
|
||||
gtk_tree_model_get_iter (model, &iter, path);
|
||||
while (gtk_tree_model_iter_next (model, &iter))
|
||||
if (get_next_node_with_child_at_depth (model, &iter, return_path,
|
||||
gtk_tree_path_get_depth (path), depth))
|
||||
return TRUE;
|
||||
}
|
||||
*return_path = NULL;
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
get_tree_path_from_row_index (GtkTreeModel *model,
|
||||
gint row_index,
|
||||
GtkTreePath **tree_path)
|
||||
{
|
||||
GtkTreeIter iter;
|
||||
gint count;
|
||||
gint depth;
|
||||
|
||||
count = gtk_tree_model_iter_n_children (model, NULL);
|
||||
if (count > row_index)
|
||||
{
|
||||
if (gtk_tree_model_iter_nth_child (model, &iter, NULL, row_index))
|
||||
{
|
||||
*tree_path = gtk_tree_model_get_path (model, &iter);
|
||||
return TRUE;
|
||||
}
|
||||
else
|
||||
return FALSE;
|
||||
}
|
||||
else
|
||||
row_index -= count;
|
||||
|
||||
depth = 0;
|
||||
while (TRUE)
|
||||
{
|
||||
depth++;
|
||||
|
||||
if (get_next_node_with_child_at_depth (model, NULL, tree_path, 0, depth))
|
||||
{
|
||||
GtkTreePath *next_path;
|
||||
|
||||
while (TRUE)
|
||||
{
|
||||
gtk_tree_model_get_iter (model, &iter, *tree_path);
|
||||
count = gtk_tree_model_iter_n_children (model, &iter);
|
||||
if (count > row_index)
|
||||
{
|
||||
gtk_tree_path_append_index (*tree_path, row_index);
|
||||
return TRUE;
|
||||
}
|
||||
else
|
||||
row_index -= count;
|
||||
|
||||
if (!get_next_node_with_child (model, *tree_path, &next_path))
|
||||
break;
|
||||
|
||||
gtk_tree_path_free (*tree_path);
|
||||
*tree_path = next_path;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
g_warning ("Index value is too large\n");
|
||||
gtk_tree_path_free (*tree_path);
|
||||
*tree_path = NULL;
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static gboolean
|
||||
get_path_column_from_index (GtkTreeView *tree_view,
|
||||
get_rbtree_column_from_index (GtkTreeView *tree_view,
|
||||
gint index,
|
||||
GtkTreePath **path,
|
||||
GtkRBTree **tree,
|
||||
GtkRBNode **node,
|
||||
GtkTreeViewColumn **column)
|
||||
{
|
||||
GtkTreeModel *tree_model;
|
||||
AtkObject *atk_obj;
|
||||
GtkTreeViewAccessible *accessible;
|
||||
|
||||
atk_obj = gtk_widget_get_accessible (GTK_WIDGET (tree_view));
|
||||
accessible = GTK_TREE_VIEW_ACCESSIBLE (atk_obj);
|
||||
|
||||
tree_model = gtk_tree_view_get_model (tree_view);
|
||||
if (accessible->n_cols == 0)
|
||||
return FALSE;
|
||||
/* First row is the column headers */
|
||||
@ -3451,16 +3301,14 @@ get_path_column_from_index (GtkTreeView *tree_view,
|
||||
if (index < 0)
|
||||
return FALSE;
|
||||
|
||||
if (path)
|
||||
if (tree)
|
||||
{
|
||||
gint row_index;
|
||||
gboolean retval;
|
||||
g_return_val_if_fail (node != NULL, FALSE);
|
||||
|
||||
row_index = index / accessible->n_cols;
|
||||
retval = get_tree_path_from_row_index (tree_model, row_index, path);
|
||||
if (!retval)
|
||||
return FALSE;
|
||||
if (*path == NULL)
|
||||
if (!_gtk_rbtree_find_index (_gtk_tree_view_get_rbtree (tree_view),
|
||||
index / accessible->n_cols,
|
||||
tree,
|
||||
node))
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
@ -3468,12 +3316,8 @@ get_path_column_from_index (GtkTreeView *tree_view,
|
||||
{
|
||||
*column = gtk_tree_view_get_column (tree_view, index % accessible->n_cols);
|
||||
if (*column == NULL)
|
||||
{
|
||||
if (path)
|
||||
gtk_tree_path_free (*path);
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
|
@ -40,7 +40,6 @@ struct _GtkTreeViewAccessible
|
||||
GtkContainerAccessible parent;
|
||||
|
||||
gint n_children_deleted;
|
||||
gint n_rows;
|
||||
gint n_cols;
|
||||
GArray* col_data;
|
||||
GHashTable *cell_infos;
|
||||
|
Loading…
Reference in New Issue
Block a user