gtk2/testsuite/a11y/tree-relationships.c
Emmanuele Bassi 334acbfc39 Replace gdk_threads_add_idle* with g_idle_add()
The main GDK thread lock is not portable and deprecated.

The only reason why gdk_threads_add_idle() and
gdk_threads_add_idle_full() exist is to allow invoking a callback with
the GDK lock held, in case 3rd party libraries still use the deprecated
gdk_threads_enter()/gdk_threads_leave() API.

Since we're removing the GDK lock, and we're releasing a new major API,
such code cannot exist any more; this means we can use the GLib API for
installing idle callbacks.

https://bugzilla.gnome.org/show_bug.cgi?id=793124
2018-02-03 12:06:58 +01:00

383 lines
11 KiB
C

/*
* Copyright (C) 2011 SUSE Linux Products GmbH, Nurenberg, Germany
*
* Author:
* Mike Gorse <mgorse@suse.com>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
*/
#include <gtk/gtk.h>
const gchar list_ui[] =
"<interface>"
" <object class='GtkListStore' id='liststore1'>"
" <columns>"
" <column type='gchararray'/>"
" <column type='gchararray'/>"
" <column type='gchararray'/>"
" <column type='gboolean'/>"
" <column type='gint'/>"
" <column type='gint'/>"
" </columns>"
" <data>"
" <row><col id='0'>One</col><col id='1'>Two</col><col id='2'>Three</col><col id='3'>True</col><col id='4'>50</col><col id='5'>50</col></row>"
" </data>"
" </object>"
" <object class='GtkWindow' id='window1'>"
" <child>"
" <object class='GtkTreeView' id='treeview1'>"
" <property name='visible'>True</property>"
" <property name='model'>liststore1</property>"
" <child>"
" <object class='GtkTreeViewColumn' id='column1'>"
" <property name='title' translatable='yes'>First column</property>"
" <child>"
" <object class='GtkCellRendererText' id='renderer1'>"
" </object>"
" <attributes>"
" <attribute name='text'>0</attribute>"
" </attributes>"
" </child>"
" <child>"
" <object class='GtkCellRendererToggle' id='renderer2'>"
" </object>"
" <attributes>"
" <attribute name='active'>3</attribute>"
" </attributes>"
" </child>"
" </object>"
" </child>"
" <child>"
" <object class='GtkTreeViewColumn' id='column2'>"
" <property name='title' translatable='yes'>Second column</property>"
" <child>"
" <object class='GtkCellRendererText' id='renderer3'>"
" </object>"
" <attributes>"
" <attribute name='text'>1</attribute>"
" </attributes>"
" </child>"
" <child>"
" <object class='GtkCellRendererProgress' id='renderer4'>"
" </object>"
" <attributes>"
" <attribute name='value'>4</attribute>"
" </attributes>"
" </child>"
" </object>"
" </child>"
" </object>"
" </child>"
" </object>"
"</interface>";
static GtkWidget *
builder_get_toplevel (GtkBuilder *builder)
{
GSList *list, *walk;
GtkWidget *window = NULL;
list = gtk_builder_get_objects (builder);
for (walk = list; walk; walk = walk->next)
{
if (GTK_IS_WINDOW (walk->data) &&
gtk_widget_get_parent (walk->data) == NULL)
{
window = walk->data;
break;
}
}
g_slist_free (list);
return window;
}
const gchar tree_ui[] =
"<interface>"
" <object class='GtkTreeStore' id='treestore1'>"
" <columns>"
" <column type='gchararray'/>"
" <column type='gchararray'/>"
" <column type='gchararray'/>"
" <column type='gboolean'/>"
" <column type='gint'/>"
" <column type='gint'/>"
" </columns>"
" </object>"
" <object class='GtkWindow' id='window1'>"
" <child>"
" <object class='GtkTreeView' id='treeview1'>"
" <property name='visible'>True</property>"
" <property name='model'>treestore1</property>"
" <child>"
" <object class='GtkTreeViewColumn' id='column1'>"
" <property name='title' translatable='yes'>First column</property>"
" <child>"
" <object class='GtkCellRendererText' id='renderer1'>"
" </object>"
" <attributes>"
" <attribute name='text'>0</attribute>"
" </attributes>"
" </child>"
" <child>"
" <object class='GtkCellRendererToggle' id='renderer2'>"
" </object>"
" <attributes>"
" <attribute name='active'>3</attribute>"
" </attributes>"
" </child>"
" </object>"
" </child>"
" <child>"
" <object class='GtkTreeViewColumn' id='column2'>"
" <property name='title' translatable='yes'>Second column</property>"
" <child>"
" <object class='GtkCellRendererText' id='renderer3'>"
" </object>"
" <attributes>"
" <attribute name='text'>1</attribute>"
" </attributes>"
" </child>"
" <child>"
" <object class='GtkCellRendererProgress' id='renderer4'>"
" </object>"
" <attributes>"
" <attribute name='value'>4</attribute>"
" </attributes>"
" </child>"
" </object>"
" </child>"
" </object>"
" </child>"
" </object>"
"</interface>";
static void
populate_tree (GtkBuilder *builder)
{
GtkTreeView *tv;
GtkTreeStore *store;
GtkTreeIter iter;
tv = (GtkTreeView *)gtk_builder_get_object (builder, "treeview1");
store = (GtkTreeStore *)gtk_tree_view_get_model (tv);
/* append some rows */
gtk_tree_store_append (store, &iter, NULL);
gtk_tree_store_set (store, &iter, 0, "a(1)", 1, "a(2)", 2, "a(3)", 3, TRUE, 4, 0, 5, 0, -1);
gtk_tree_store_append (store, &iter, &iter);
gtk_tree_store_set (store, &iter, 0, "aa(1)", 1, "aa(2)", 2, "aa(3)", 3, TRUE, 4, 0, 5, 0, -1);
gtk_tree_store_append (store, &iter, &iter);
gtk_tree_store_set (store, &iter, 0, "aaa(1)", 1, "aaa(2)", 2, "aaa(3)", 3, TRUE, 4, 0, 5, 0, -1);
gtk_tree_store_append (store, &iter, NULL);
gtk_tree_store_set (store, &iter, 0, "b(1)", 1, "b(2)", 2, "b(3)", 3, TRUE, 4, 0, 5, 0, -1);
gtk_tree_store_append (store, &iter, &iter);
gtk_tree_store_set (store, &iter, 0, "bb(1)", 1, "bb(2)", 2, "bb(3)", 3, TRUE, 4, 0, 5, 0, -1);
gtk_tree_store_append (store, &iter, &iter);
gtk_tree_store_set (store, &iter, 0, "bbb(1)", 1, "bbb(2)", 2, "bbb(3)", 3, TRUE, 4, 0, 5, 0, -1);
}
typedef struct {
gint count;
AtkObject *descendant;
} SignalData;
static void
active_descendant_changed (AtkObject *accessible,
AtkObject *descendant,
SignalData *data)
{
data->count++;
if (data->descendant)
g_object_unref (data->descendant);
data->descendant = g_object_ref (descendant);
}
static gboolean
quit_loop (gpointer data)
{
GMainLoop *loop = data;
g_main_loop_quit (loop);
return FALSE;
}
static void
process_pending_idles ()
{
GMainLoop *loop = g_main_loop_new (NULL, FALSE);
g_idle_add (quit_loop, loop);
g_main_loop_run (loop);
}
static void
test_a11y_tree_focus (void)
{
GtkBuilder *builder;
GtkWidget *window;
GError *error = NULL;
GtkTreeView *tv;
GtkTreePath *path = NULL;
GtkTreeViewColumn *focus_column = NULL;
SignalData data;
AtkObject *accessible;
AtkObject *child;
gchar *text;
builder = gtk_builder_new ();
gtk_builder_add_from_string (builder, tree_ui, -1, &error);
g_assert_no_error (error);
window = builder_get_toplevel (builder);
g_assert (window);
populate_tree (builder);
tv = (GtkTreeView *)gtk_builder_get_object (builder, "treeview1");
gtk_tree_view_expand_all (tv);
gtk_widget_show (window);
gtk_tree_view_get_cursor (tv, &path, &focus_column);
gtk_tree_path_down (path);
data.count = 0;
data.descendant = NULL;
accessible = gtk_widget_get_accessible (GTK_WIDGET (tv));
g_signal_connect (accessible, "active_descendant_changed",
G_CALLBACK (active_descendant_changed), &data);
gtk_tree_view_set_cursor (tv, path, focus_column, FALSE);
/* hack: active_descendant_change gets fired in an idle handler */
process_pending_idles ();
/* Getting only one signal might be ideal, although we get three or so */
g_assert_cmpint (data.count, >=, 1);
child = atk_object_ref_accessible_child (data.descendant, 0);
text = atk_text_get_text (ATK_TEXT (child), 0, -1);
g_assert_cmpstr (text, ==, "aa(1)");
g_free (text);
g_object_unref (child);
g_object_unref (data.descendant);
g_object_unref (builder);
}
static AtkObject *
find_root_accessible (GtkTreeView *tv, const char *name)
{
AtkObject *tvaccessible = gtk_widget_get_accessible (GTK_WIDGET (tv));
int i = 0;
for (i = 0;;i++)
{
AtkObject *child = atk_object_ref_accessible_child (tvaccessible, i);
AtkObject *item;
gchar *text;
if (!child)
return NULL;
item = atk_object_ref_accessible_child (child, 0);
if (!item)
continue;
text = atk_text_get_text (ATK_TEXT (item), 0, -1);
g_object_unref (item);
if (!g_strcmp0 (text, name))
{
g_free (text);
return child;
}
g_free (text);
g_object_unref (child);
}
}
static void
test_node_child_of (AtkObject *child, AtkObject *parent)
{
AtkRelationSet *set = atk_object_ref_relation_set (child);
AtkRelation *relation = atk_relation_set_get_relation_by_type (set, ATK_RELATION_NODE_CHILD_OF);
g_assert (relation != NULL);
g_assert_cmpint (relation->target->len, ==, 1);
g_assert (g_ptr_array_index (relation->target, 0) == parent);
g_object_unref (set);
}
static void
test_node_parent_of (AtkObject *parent, AtkObject *child)
{
AtkRelationSet *set = atk_object_ref_relation_set (parent);
AtkRelation *relation = atk_relation_set_get_relation_by_type (set, ATK_RELATION_NODE_PARENT_OF);
g_assert (relation != NULL);
g_assert_cmpint (relation->target->len, ==, 1);
g_assert (g_ptr_array_index (relation->target, 0) == child);
g_object_unref (set);
}
static void
test_relations (AtkObject *parent, AtkObject *child)
{
test_node_parent_of (parent, child);
test_node_child_of (child, parent);
}
static void
test_a11y_tree_relations (void)
{
GtkBuilder *builder;
GtkWidget *window;
GError *error = NULL;
GtkTreeView *tv;
AtkObject *parent, *child;
builder = gtk_builder_new ();
gtk_builder_add_from_string (builder, tree_ui, -1, &error);
g_assert_no_error (error);
window = builder_get_toplevel (builder);
g_assert (window);
populate_tree (builder);
tv = (GtkTreeView *)gtk_builder_get_object (builder, "treeview1");
gtk_tree_view_expand_all (tv);
gtk_widget_show (window);
parent = find_root_accessible (tv, "a(1)");
child = find_root_accessible (tv, "aa(1)");
test_relations (parent, child);
g_object_unref (parent);
parent = child;
child = find_root_accessible (tv, "aaa(1)");
test_relations (parent, child);
g_object_unref (parent);
g_object_unref (child);
g_object_unref (builder);
}
int
main (int argc, char *argv[])
{
gtk_test_init (&argc, &argv, NULL);
g_test_add_func ("/a11y/tree/focus", test_a11y_tree_focus);
g_test_add_func ("/a11y/tree/relations", test_a11y_tree_relations);
return g_test_run ();
}