|
|
214acd |
From d928258bb4f3e21973089183463c4dab11558b73 Mon Sep 17 00:00:00 2001
|
|
|
214acd |
From: Milan Crha <mcrha@redhat.com>
|
|
|
214acd |
Date: Wed, 18 Sep 2019 14:12:44 +0200
|
|
|
214acd |
Subject: I#624 - GalA11yETableItem: Incorrect implementation of
|
|
|
214acd |
AtkObjectClass::ref_child()
|
|
|
214acd |
|
|
|
214acd |
Closes https://gitlab.gnome.org/GNOME/evolution/issues/624
|
|
|
214acd |
|
|
|
214acd |
diff --git a/src/e-util/gal-a11y-e-table-item.c b/src/e-util/gal-a11y-e-table-item.c
|
|
|
214acd |
index cf06fb3f4f..be302ed09d 100644
|
|
|
214acd |
--- a/src/e-util/gal-a11y-e-table-item.c
|
|
|
214acd |
+++ b/src/e-util/gal-a11y-e-table-item.c
|
|
|
214acd |
@@ -61,6 +61,7 @@ struct _GalA11yETableItemPrivate {
|
|
|
214acd |
ESelectionModel *selection;
|
|
|
214acd |
AtkStateSet *state_set;
|
|
|
214acd |
GtkWidget *widget;
|
|
|
214acd |
+ GHashTable *a11y_column_headers; /* ETableCol * ~> GalA11yETableColumnHeader * */
|
|
|
214acd |
};
|
|
|
214acd |
|
|
|
214acd |
static gboolean gal_a11y_e_table_item_ref_selection (GalA11yETableItem *a11y,
|
|
|
214acd |
@@ -124,6 +125,11 @@ item_finalized (gpointer user_data,
|
|
|
214acd |
if (priv->selection)
|
|
|
214acd |
gal_a11y_e_table_item_unref_selection (a11y);
|
|
|
214acd |
|
|
|
214acd |
+ if (priv->columns) {
|
|
|
214acd |
+ free_columns (priv->columns);
|
|
|
214acd |
+ priv->columns = NULL;
|
|
|
214acd |
+ }
|
|
|
214acd |
+
|
|
|
214acd |
g_object_unref (a11y);
|
|
|
214acd |
}
|
|
|
214acd |
|
|
|
214acd |
@@ -273,11 +279,60 @@ eti_a11y_reset_focus_object (GalA11yETableItem *a11y,
|
|
|
214acd |
g_signal_emit_by_name (a11y, "active-descendant-changed", cell);
|
|
|
214acd |
}
|
|
|
214acd |
|
|
|
214acd |
+static void eti_column_header_a11y_gone (gpointer user_data, GObject *a11y_col_header);
|
|
|
214acd |
+
|
|
|
214acd |
+static void
|
|
|
214acd |
+eti_table_column_gone (gpointer user_data,
|
|
|
214acd |
+ GObject *col)
|
|
|
214acd |
+{
|
|
|
214acd |
+ GalA11yETableItem *a11y = user_data;
|
|
|
214acd |
+ GalA11yETableItemPrivate *priv;
|
|
|
214acd |
+ GalA11yETableColumnHeader *a11y_col_header;
|
|
|
214acd |
+
|
|
|
214acd |
+ g_return_if_fail (GAL_A11Y_IS_E_TABLE_ITEM (a11y));
|
|
|
214acd |
+
|
|
|
214acd |
+ priv = GET_PRIVATE (a11y);
|
|
|
214acd |
+
|
|
|
214acd |
+ a11y_col_header = g_hash_table_lookup (priv->a11y_column_headers, col);
|
|
|
214acd |
+ g_hash_table_remove (priv->a11y_column_headers, col);
|
|
|
214acd |
+
|
|
|
214acd |
+ if (a11y_col_header)
|
|
|
214acd |
+ g_object_weak_unref (G_OBJECT (a11y_col_header), eti_column_header_a11y_gone, a11y);
|
|
|
214acd |
+}
|
|
|
214acd |
+
|
|
|
214acd |
+static void
|
|
|
214acd |
+eti_column_header_a11y_gone (gpointer user_data,
|
|
|
214acd |
+ GObject *a11y_col_header)
|
|
|
214acd |
+{
|
|
|
214acd |
+ GalA11yETableItem *a11y = user_data;
|
|
|
214acd |
+ GalA11yETableItemPrivate *priv;
|
|
|
214acd |
+ GHashTableIter iter;
|
|
|
214acd |
+ gpointer key, value;
|
|
|
214acd |
+
|
|
|
214acd |
+ g_return_if_fail (GAL_A11Y_IS_E_TABLE_ITEM (a11y));
|
|
|
214acd |
+
|
|
|
214acd |
+ priv = GET_PRIVATE (a11y);
|
|
|
214acd |
+
|
|
|
214acd |
+ g_hash_table_iter_init (&iter, priv->a11y_column_headers);
|
|
|
214acd |
+ while (g_hash_table_iter_next (&iter, &key, &value)) {
|
|
|
214acd |
+ ETableCol *col = key;
|
|
|
214acd |
+ GalA11yETableColumnHeader *stored_a11y_col_header = value;
|
|
|
214acd |
+
|
|
|
214acd |
+ if (((GObject *) stored_a11y_col_header) == a11y_col_header) {
|
|
|
214acd |
+ g_object_weak_unref (G_OBJECT (col), eti_table_column_gone, a11y);
|
|
|
214acd |
+ g_hash_table_remove (priv->a11y_column_headers, col);
|
|
|
214acd |
+ break;
|
|
|
214acd |
+ }
|
|
|
214acd |
+ }
|
|
|
214acd |
+}
|
|
|
214acd |
+
|
|
|
214acd |
static void
|
|
|
214acd |
eti_dispose (GObject *object)
|
|
|
214acd |
{
|
|
|
214acd |
GalA11yETableItem *a11y = GAL_A11Y_E_TABLE_ITEM (object);
|
|
|
214acd |
GalA11yETableItemPrivate *priv = GET_PRIVATE (a11y);
|
|
|
214acd |
+ GHashTableIter iter;
|
|
|
214acd |
+ gpointer key, value;
|
|
|
214acd |
|
|
|
214acd |
if (priv->columns) {
|
|
|
214acd |
free_columns (priv->columns);
|
|
|
214acd |
@@ -289,10 +344,35 @@ eti_dispose (GObject *object)
|
|
|
214acd |
priv->item = NULL;
|
|
|
214acd |
}
|
|
|
214acd |
|
|
|
214acd |
+ g_clear_object (&priv->state_set);
|
|
|
214acd |
+
|
|
|
214acd |
+ g_hash_table_iter_init (&iter, priv->a11y_column_headers);
|
|
|
214acd |
+ while (g_hash_table_iter_next (&iter, &key, &value)) {
|
|
|
214acd |
+ ETableCol *col = key;
|
|
|
214acd |
+ GalA11yETableColumnHeader *a11y_col_header = value;
|
|
|
214acd |
+
|
|
|
214acd |
+ g_object_weak_unref (G_OBJECT (col), eti_table_column_gone, a11y);
|
|
|
214acd |
+ g_object_weak_unref (G_OBJECT (a11y_col_header), eti_column_header_a11y_gone, a11y);
|
|
|
214acd |
+ }
|
|
|
214acd |
+
|
|
|
214acd |
+ g_hash_table_remove_all (priv->a11y_column_headers);
|
|
|
214acd |
+
|
|
|
214acd |
if (parent_class->dispose)
|
|
|
214acd |
parent_class->dispose (object);
|
|
|
214acd |
}
|
|
|
214acd |
|
|
|
214acd |
+static void
|
|
|
214acd |
+eti_finalize (GObject *object)
|
|
|
214acd |
+{
|
|
|
214acd |
+ GalA11yETableItem *a11y = GAL_A11Y_E_TABLE_ITEM (object);
|
|
|
214acd |
+ GalA11yETableItemPrivate *priv = GET_PRIVATE (a11y);
|
|
|
214acd |
+
|
|
|
214acd |
+ g_hash_table_destroy (priv->a11y_column_headers);
|
|
|
214acd |
+
|
|
|
214acd |
+ if (parent_class->finalize)
|
|
|
214acd |
+ parent_class->finalize (object);
|
|
|
214acd |
+}
|
|
|
214acd |
+
|
|
|
214acd |
/* Static functions */
|
|
|
214acd |
static gint
|
|
|
214acd |
eti_get_n_children (AtkObject *accessible)
|
|
|
214acd |
@@ -318,12 +398,24 @@ eti_ref_child (AtkObject *accessible,
|
|
|
214acd |
return NULL;
|
|
|
214acd |
|
|
|
214acd |
if (index < item->cols) {
|
|
|
214acd |
+ GalA11yETableItemPrivate *priv = GET_PRIVATE (accessible);
|
|
|
214acd |
ETableCol *ecol;
|
|
|
214acd |
AtkObject *child;
|
|
|
214acd |
|
|
|
214acd |
ecol = e_table_header_get_column (item->header, index);
|
|
|
214acd |
- child = gal_a11y_e_table_column_header_new (ecol, item, accessible);
|
|
|
214acd |
- return child;
|
|
|
214acd |
+ child = g_hash_table_lookup (priv->a11y_column_headers, ecol);
|
|
|
214acd |
+
|
|
|
214acd |
+ if (!child) {
|
|
|
214acd |
+ child = gal_a11y_e_table_column_header_new (ecol, item, accessible);
|
|
|
214acd |
+ if (child) {
|
|
|
214acd |
+ g_hash_table_insert (priv->a11y_column_headers, ecol, child);
|
|
|
214acd |
+
|
|
|
214acd |
+ g_object_weak_ref (G_OBJECT (ecol), eti_table_column_gone, accessible);
|
|
|
214acd |
+ g_object_weak_ref (G_OBJECT (child), eti_column_header_a11y_gone, accessible);
|
|
|
214acd |
+ }
|
|
|
214acd |
+ }
|
|
|
214acd |
+
|
|
|
214acd |
+ return child ? g_object_ref (child) : NULL;
|
|
|
214acd |
}
|
|
|
214acd |
index -= item->cols;
|
|
|
214acd |
|
|
|
214acd |
@@ -966,6 +1058,7 @@ eti_header_structure_changed (ETableHeader *eth,
|
|
|
214acd |
g_free (state);
|
|
|
214acd |
g_free (reorder);
|
|
|
214acd |
g_free (prev_state);
|
|
|
214acd |
+ free_columns (cols);
|
|
|
214acd |
return;
|
|
|
214acd |
}
|
|
|
214acd |
|
|
|
214acd |
@@ -1051,6 +1144,7 @@ eti_class_init (GalA11yETableItemClass *class)
|
|
|
214acd |
parent_class = g_type_class_ref (PARENT_TYPE);
|
|
|
214acd |
|
|
|
214acd |
object_class->dispose = eti_dispose;
|
|
|
214acd |
+ object_class->finalize = eti_finalize;
|
|
|
214acd |
|
|
|
214acd |
atk_object_class->get_n_children = eti_get_n_children;
|
|
|
214acd |
atk_object_class->ref_child = eti_ref_child;
|
|
|
214acd |
@@ -1069,6 +1163,7 @@ eti_init (GalA11yETableItem *a11y)
|
|
|
214acd |
priv->selection_row_changed_id = 0;
|
|
|
214acd |
priv->cursor_changed_id = 0;
|
|
|
214acd |
priv->selection = NULL;
|
|
|
214acd |
+ priv->a11y_column_headers = g_hash_table_new (g_direct_hash, g_direct_equal);
|
|
|
214acd |
}
|
|
|
214acd |
|
|
|
214acd |
/* atk selection */
|
|
|
214acd |
@@ -1189,14 +1284,17 @@ gal_a11y_e_table_item_new (ETableItem *item)
|
|
|
214acd |
|
|
|
214acd |
accessible = ATK_OBJECT (a11y);
|
|
|
214acd |
|
|
|
214acd |
- GET_PRIVATE (a11y)->item = item;
|
|
|
214acd |
/* Initialize cell data. */
|
|
|
214acd |
GET_PRIVATE (a11y)->cols = item->cols;
|
|
|
214acd |
GET_PRIVATE (a11y)->rows = item->rows >= 0 ? item->rows : 0;
|
|
|
214acd |
|
|
|
214acd |
GET_PRIVATE (a11y)->columns = e_table_header_get_columns (item->header);
|
|
|
214acd |
- if (GET_PRIVATE (a11y)->columns == NULL)
|
|
|
214acd |
+ if (GET_PRIVATE (a11y)->columns == NULL) {
|
|
|
214acd |
+ g_clear_object (&a11y);
|
|
|
214acd |
return NULL;
|
|
|
214acd |
+ }
|
|
|
214acd |
+
|
|
|
214acd |
+ GET_PRIVATE (a11y)->item = item;
|
|
|
214acd |
|
|
|
214acd |
g_signal_connect (
|
|
|
214acd |
item, "selection_model_removed",
|