gtk/tests/a11y/tree-performance.c
2012-02-27 17:06:11 +00:00

370 lines
11 KiB
C

/*
* Copyright (C) 2011 Red Hat Inc.
*
* Author:
* Matthias Clasen <mclasen@redhat.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>
#define N_ROWS 10000
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 void
walk_accessible_tree (AtkObject *accessible,
gpointer data)
{
gint *count = data;
gint i;
(*count)++;
for (i = 0; i < atk_object_get_n_accessible_children (accessible); i++)
{
AtkObject *child = atk_object_ref_accessible_child (accessible, i);
walk_accessible_tree (child, data);
g_object_unref (child);
}
}
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;
}
static void
populate_list (GtkBuilder *builder)
{
GtkTreeView *tv;
GtkListStore *store;
GtkTreeIter iter;
gint i;
tv = (GtkTreeView *)gtk_builder_get_object (builder, "treeview1");
store = (GtkListStore *)gtk_tree_view_get_model (tv);
/* append a many rows */
for (i = 0; i < N_ROWS; i++)
{
gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter, 0, "Bla", 1, "Bla bla", 2, "Bla bla bla", 3, i % 2 == 0 ? TRUE : FALSE, 4, i % 100, 5, i, -1);
}
}
static void
test_performance_list (void)
{
GtkBuilder *builder;
gdouble elapsed;
GtkWidget *window;
GError *error = NULL;
builder = gtk_builder_new ();
gtk_builder_add_from_string (builder, list_ui, -1, &error);
g_assert_no_error (error);
window = builder_get_toplevel (builder);
g_assert (window);
gtk_widget_show (window);
g_test_timer_start ();
populate_list (builder);
elapsed = g_test_timer_elapsed ();
g_test_minimized_result (elapsed, "large list test: %gsec", elapsed);
g_object_unref (builder);
}
static void
test_a11y_performance_list (void)
{
GtkBuilder *builder;
gdouble elapsed;
GtkWidget *window;
GError *error = NULL;
gint count_before;
gint count_after;
builder = gtk_builder_new ();
gtk_builder_add_from_string (builder, list_ui, -1, &error);
g_assert_no_error (error);
window = builder_get_toplevel (builder);
g_assert (window);
gtk_widget_show (window);
g_test_timer_start ();
/* make sure all accessibles exist */
count_before = 0;
walk_accessible_tree (gtk_widget_get_accessible (window), &count_before);
populate_list (builder);
/* for good measure, do this again */
count_after = 0;
walk_accessible_tree (gtk_widget_get_accessible (window), &count_after);
elapsed = g_test_timer_elapsed ();
g_test_minimized_result (elapsed, "large list with a11y: %gsec", elapsed);
g_object_unref (builder);
g_test_message ("%d accessibles before, %d after\n", count_before, count_after);
}
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;
gint i;
tv = (GtkTreeView *)gtk_builder_get_object (builder, "treeview1");
store = (GtkTreeStore *)gtk_tree_view_get_model (tv);
/* append a thousand rows */
for (i = 0; i < N_ROWS / 3; i++)
{
gtk_tree_store_append (store, &iter, NULL);
gtk_tree_store_set (store, &iter, 0, "Bla", 1, "Bla bla", 2, "Bla bla bla", 3, i % 2 == 0 ? TRUE : FALSE, 4, i % 100, 5, i, -1);
gtk_tree_store_append (store, &iter, &iter);
gtk_tree_store_set (store, &iter, 0, "Bla", 1, "Bla bla", 2, "Bla bla bla", 3, i % 2 == 0 ? TRUE : FALSE, 4, i % 100, 5, i, -1);
gtk_tree_store_append (store, &iter, &iter);
gtk_tree_store_set (store, &iter, 0, "Bla", 1, "Bla bla", 2, "Bla bla bla", 3, i % 2 == 0 ? TRUE : FALSE, 4, i % 100, 5, i, -1);
}
gtk_tree_view_expand_all (tv);
}
static void
test_performance_tree (void)
{
GtkBuilder *builder;
gdouble elapsed;
GtkWidget *window;
GError *error = NULL;
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);
gtk_widget_show (window);
g_test_timer_start ();
populate_tree (builder);
elapsed = g_test_timer_elapsed ();
g_test_minimized_result (elapsed, "large tree test: %gsec", elapsed);
g_object_unref (builder);
}
static void
test_a11y_performance_tree (void)
{
GtkBuilder *builder;
gdouble elapsed;
GtkWidget *window;
GError *error = NULL;
gint count_before;
gint count_after;
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);
gtk_widget_show (window);
g_test_timer_start ();
/* make sure all accessibles exist */
count_before = 0;
walk_accessible_tree (gtk_widget_get_accessible (window), &count_before);
populate_tree (builder);
/* for good measure, do this again */
count_after = 0;
walk_accessible_tree (gtk_widget_get_accessible (window), &count_after);
elapsed = g_test_timer_elapsed ();
g_test_minimized_result (elapsed, "large tree with a11y: %gsec", elapsed);
g_object_unref (builder);
g_test_message ("%d accessibles before, %d after\n", count_before, count_after);
}
int
main (int argc, char *argv[])
{
gtk_test_init (&argc, &argv, NULL);
if (!g_test_perf ())
return 0;
g_test_add_func ("/performance/list", test_performance_list);
g_test_add_func ("/a11y/performance/list", test_a11y_performance_list);
g_test_add_func ("/performance/tree", test_performance_tree);
g_test_add_func ("/a11y/performance/tree", test_a11y_performance_tree);
return g_test_run ();
}