forked from AuroraMiddleware/gtk
334acbfc39
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
383 lines
11 KiB
C
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 ();
|
|
}
|
|
|