summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMatthias Clasen <mclasen@redhat.com>2014-08-30 20:20:00 -0400
committerMatthias Clasen <mclasen@redhat.com>2014-08-30 20:20:00 -0400
commit0068b78276c4ac7df95d97fa270a5e8ca4b58348 (patch)
tree17ea02a2981883c9b230b3022e3fbc32a55bcd81
parent92e7b73e27f54db86d58dcb5b4ddaafb32101980 (diff)
downloadpango-0068b78276c4ac7df95d97fa270a5e8ca4b58348.tar.gz
Add tests for PangoLayout
This tests works similar to the markup parser tests - we populate a layout with known data, and then compare a dump of its properties with expected output.
-rw-r--r--tests/Makefile.am23
-rw-r--r--tests/layouts/valid-1.expected35
-rw-r--r--tests/layouts/valid-1.markup2
-rw-r--r--tests/layouts/valid-2.expected34
-rw-r--r--tests/layouts/valid-2.markup2
-rw-r--r--tests/test-layout.c487
6 files changed, 580 insertions, 3 deletions
diff --git a/tests/Makefile.am b/tests/Makefile.am
index d8891836..b9ed3138 100644
--- a/tests/Makefile.am
+++ b/tests/Makefile.am
@@ -43,7 +43,14 @@ TESTS_ENVIRONMENT = \
G_TEST_BUILDDIR=$(abs_builddir) \
PANGO_RC_FILE=./pangorc
-check_PROGRAMS = testboundaries testboundaries_ucd testcolor testscript markup-parse
+check_PROGRAMS = \
+ testboundaries \
+ testboundaries_ucd \
+ testcolor \
+ testscript \
+ markup-parse \
+ test-layout \
+ $(NULL)
if HAVE_CAIRO
check_PROGRAMS += testiter test-pangocairo-threads
@@ -66,6 +73,7 @@ test_ot_tags_LDADD = $(TEST_PANGOFT2_LIBS) $(GLIB_LIBS)
test_pangocairo_threads_LDADD = $(TEST_PANGOCAIRO_LIBS) $(CAIRO_LIBS) $(GLIB_LIBS)
dump_boundaries_LDADD = $(TEST_PANGO_LIBS) $(GLIB_LIBS)
markup_parse_LDADD = $(TEST_PANGOCAIRO_LIBS) $(GLIB_LIBS)
+test_layout_LDADD = $(TEST_PANGOCAIRO_LIBS) $(GLIB_LIBS)
if HAVE_CXX
check_PROGRAMS += cxx-test
@@ -90,7 +98,15 @@ markup_tests = \
all_markup_data = \
$(addprefix markups/,$(markup_tests:=.markup) $(markup_tests:=.expected))
-EXTRA_DIST += $(all_markup_data)
+layout_tests = \
+ valid-1 valid-2 \
+ $(NULL)
+
+all_layout_data = \
+ $(addprefix layouts/,$(layout_tests:=.markup) $(layout_tests:=.expected))
+
+
+EXTRA_DIST += $(all_markup_data) $(all_layout_data)
if BUILDOPT_INSTALL_TESTS
insttestdir=$(libexecdir)/installed-tests/$(PACKAGE)
@@ -101,7 +117,8 @@ nobase_testdata_DATA = \
all-unicode.txt \
boundaries.utf8 \
GraphemeBreakTest.txt \
- $(all_markup_data)
+ $(all_markup_data) \
+ $(all_layout_data)
testmetadir = $(datadir)/installed-tests/$(PACKAGE)
testmeta_DATA = $(check_PROGRAMS:=.test)
diff --git a/tests/layouts/valid-1.expected b/tests/layouts/valid-1.expected
new file mode 100644
index 00000000..5cf99a08
--- /dev/null
+++ b/tests/layouts/valid-1.expected
@@ -0,0 +1,35 @@
+This is a test of the automatic emergency brake!
+
+---
+
+wrapped: 0
+ellipsized: 1
+lines: 2
+width: 225280
+
+---
+
+range 0 22
+range 22 41
+[22 41] foreground #00000000ffff
+[22 41] underline 1
+range 41 2147483647
+
+---
+
+i=1, index=0, paragraph-start=1, dir=ltr 'This is a test of the automatic emergency brake!
+'
+i=2, index=49, paragraph-start=1, dir=ltr ''
+
+---
+
+i=1, index=0, chars=22, level=0, gravity=south, flags=0, font=Cantarell 11, script=latin, language=c, 'This is a test of the '
+i=2, index=22, chars=11, level=0, gravity=south, flags=0, font=Cantarell 11, script=latin, language=c, 'automatic e'
+ [22 41] foreground #00000000ffff
+ [22 41] underline 1
+i=3, index=33, chars=15, level=0, gravity=south, flags=0, font=Cantarell 11, script=common, language=c, 'mergency brake!'
+ [0 2147483647] fallback 0
+ [22 41] foreground #00000000ffff
+ [22 41] underline 1
+i=4, index=48, no run, line end
+i=5, index=49, no run, line end
diff --git a/tests/layouts/valid-1.markup b/tests/layouts/valid-1.markup
new file mode 100644
index 00000000..780e6831
--- /dev/null
+++ b/tests/layouts/valid-1.markup
@@ -0,0 +1,2 @@
+width=220,ellipsize=end
+This is a test of the <span foreground="#0000ff" underline="single">automatic emergency</span> brake!
diff --git a/tests/layouts/valid-2.expected b/tests/layouts/valid-2.expected
new file mode 100644
index 00000000..c98bfd59
--- /dev/null
+++ b/tests/layouts/valid-2.expected
@@ -0,0 +1,34 @@
+test the blue drink after dinner
+
+---
+
+wrapped: 0
+ellipsized: 0
+lines: 2
+
+---
+
+range 0 9
+range 9 13
+[9 13] style 2
+range 13 20
+range 20 25
+[20 25] underline 1
+range 25 2147483647
+
+---
+
+i=1, index=0, paragraph-start=1, dir=ltr 'test the blue drink after dinner
+'
+i=2, index=33, paragraph-start=1, dir=ltr ''
+
+---
+
+i=1, index=0, chars=9, level=0, gravity=south, flags=0, font=Cantarell 11, script=latin, language=c, 'test the '
+i=2, index=9, chars=4, level=0, gravity=south, flags=0, font=Cantarell Oblique 11, script=latin, language=c, 'blue'
+i=3, index=13, chars=7, level=0, gravity=south, flags=0, font=Cantarell 11, script=latin, language=c, ' drink '
+i=4, index=20, chars=5, level=0, gravity=south, flags=0, font=Cantarell 11, script=latin, language=c, 'after'
+ [20 25] underline 1
+i=5, index=25, chars=7, level=0, gravity=south, flags=0, font=Cantarell 11, script=latin, language=c, ' dinner'
+i=6, index=32, no run, line end
+i=7, index=33, no run, line end
diff --git a/tests/layouts/valid-2.markup b/tests/layouts/valid-2.markup
new file mode 100644
index 00000000..677f26f8
--- /dev/null
+++ b/tests/layouts/valid-2.markup
@@ -0,0 +1,2 @@
+
+test the <i>blue</i> drink <u>after</u> dinner
diff --git a/tests/test-layout.c b/tests/test-layout.c
new file mode 100644
index 00000000..25b81147
--- /dev/null
+++ b/tests/test-layout.c
@@ -0,0 +1,487 @@
+/* Pango
+ * test-layout.c: Test Pango Layout
+ *
+ * Copyright (C) 2014 Red Hat, Inc
+ *
+ * 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, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#include <glib.h>
+#include <string.h>
+#include <locale.h>
+
+#include <pango/pangocairo.h>
+
+static PangoContext *context;
+
+static void
+print_attr (PangoAttribute *attr, GString *string)
+{
+ g_string_append_printf (string, "[%d %d] ", attr->start_index, attr->end_index);
+ switch (attr->klass->type)
+ {
+ case PANGO_ATTR_LANGUAGE:
+ g_string_append_printf (string,"language %s\n", pango_language_to_string (((PangoAttrLanguage *)attr)->value));
+ break;
+ case PANGO_ATTR_FAMILY:
+ g_string_append_printf (string,"family %s\n", ((PangoAttrString *)attr)->value);
+ break;
+ case PANGO_ATTR_STYLE:
+ g_string_append_printf (string,"style %d\n", ((PangoAttrInt *)attr)->value);
+ break;
+ case PANGO_ATTR_WEIGHT:
+ g_string_append_printf (string,"weight %d\n", ((PangoAttrInt *)attr)->value);
+ break;
+ case PANGO_ATTR_VARIANT:
+ g_string_append_printf (string,"variant %d\n", ((PangoAttrInt *)attr)->value);
+ break;
+ case PANGO_ATTR_STRETCH:
+ g_string_append_printf (string,"stretch %d\n", ((PangoAttrInt *)attr)->value);
+ break;
+ case PANGO_ATTR_SIZE:
+ g_string_append_printf (string,"size %d\n", ((PangoAttrSize *)attr)->size);
+ break;
+ case PANGO_ATTR_FONT_DESC:
+ g_string_append_printf (string,"font %s\n", pango_font_description_to_string (((PangoAttrFontDesc *)attr)->desc));
+ break;
+ case PANGO_ATTR_FOREGROUND:
+ g_string_append_printf (string,"foreground %s\n", pango_color_to_string (&((PangoAttrColor *)attr)->color));
+ break;
+ case PANGO_ATTR_BACKGROUND:
+ g_string_append_printf (string,"background %s\n", pango_color_to_string (&((PangoAttrColor *)attr)->color));
+ break;
+ case PANGO_ATTR_UNDERLINE:
+ g_string_append_printf (string,"underline %d\n", ((PangoAttrInt *)attr)->value);
+ break;
+ case PANGO_ATTR_STRIKETHROUGH:
+ g_string_append_printf (string,"strikethrough %d\n", ((PangoAttrInt *)attr)->value);
+ break;
+ case PANGO_ATTR_RISE:
+ g_string_append_printf (string,"rise %d\n", ((PangoAttrInt *)attr)->value);
+ break;
+ case PANGO_ATTR_SHAPE:
+ g_string_append_printf (string,"shape\n");
+ break;
+ case PANGO_ATTR_SCALE:
+ g_string_append_printf (string,"scale %f\n", ((PangoAttrFloat *)attr)->value);
+ break;
+ case PANGO_ATTR_FALLBACK:
+ g_string_append_printf (string,"fallback %d\n", ((PangoAttrInt *)attr)->value);
+ break;
+ case PANGO_ATTR_LETTER_SPACING:
+ g_string_append_printf (string,"letter-spacing %d\n", ((PangoAttrInt *)attr)->value);
+ break;
+ case PANGO_ATTR_UNDERLINE_COLOR:
+ g_string_append_printf (string,"underline-color %s\n", pango_color_to_string (&((PangoAttrColor *)attr)->color));
+ break;
+ case PANGO_ATTR_STRIKETHROUGH_COLOR:
+ g_string_append_printf (string,"strikethrough-color %s\n", pango_color_to_string (&((PangoAttrColor *)attr)->color));
+ break;
+ case PANGO_ATTR_ABSOLUTE_SIZE:
+ g_string_append_printf (string,"absolute-size %d\n", ((PangoAttrSize *)attr)->size);
+ break;
+ case PANGO_ATTR_GRAVITY:
+ g_string_append_printf (string,"gravity %d\n", ((PangoAttrInt *)attr)->value);
+ break;
+ case PANGO_ATTR_GRAVITY_HINT:
+ g_string_append_printf (string,"gravity-hint %d\n", ((PangoAttrInt *)attr)->value);
+ break;
+ default:
+ g_assert_not_reached ();
+ break;
+ }
+}
+
+static void
+dump_attrs (PangoAttrList *attrs, GString *string)
+{
+ PangoAttrIterator *iter;
+
+ iter = pango_attr_list_get_iterator (attrs);
+ do {
+ gint start, end;
+ GSList *list, *l;
+
+ pango_attr_iterator_range (iter, &start, &end);
+ g_string_append_printf (string, "range %d %d\n", start, end);
+ list = pango_attr_iterator_get_attrs (iter);
+ for (l = list; l; l = l->next)
+ {
+ PangoAttribute *attr = l->data;
+ print_attr (attr, string);
+ }
+ g_slist_free_full (list, (GDestroyNotify)pango_attribute_destroy);
+ } while (pango_attr_iterator_next (iter));
+
+ pango_attr_iterator_destroy (iter);
+}
+
+static void
+dump_extra_attrs (GSList *attrs, GString *string)
+{
+ GSList *l;
+
+ for (l = attrs; l; l = l->next)
+ {
+ PangoAttribute *attr = l->data;
+
+ g_string_append (string, " ");
+ print_attr (attr, string);
+ }
+}
+
+static const gchar *
+enum_value_nick (GType type, gint value)
+{
+ GEnumClass *eclass;
+ GEnumValue *ev;
+
+ eclass = g_type_class_ref (type);
+ ev = g_enum_get_value (eclass, value);
+ g_type_class_unref (eclass);
+
+ if (ev)
+ return ev->value_nick;
+ else
+ return "?";
+}
+
+static const gchar *
+direction_name (PangoDirection dir)
+{
+ return enum_value_nick (PANGO_TYPE_DIRECTION, dir);
+}
+
+static const gchar *
+gravity_name (PangoGravity gravity)
+{
+ return enum_value_nick (PANGO_TYPE_GRAVITY, gravity);
+}
+
+static const gchar *
+script_name (PangoScript script)
+{
+ return enum_value_nick (PANGO_TYPE_SCRIPT, script);
+}
+
+static gchar *
+font_name (PangoFont *font)
+{
+ PangoFontDescription *desc;
+ gchar *name;
+
+ desc = pango_font_describe (font);
+ name = pango_font_description_to_string (desc);
+ pango_font_description_free (desc);
+
+ return name;
+}
+
+static void
+dump_lines (PangoLayout *layout, GString *string)
+{
+ PangoLayoutIter *iter;
+ const gchar *text;
+ gint index, index2;
+ gboolean has_more;
+ gchar *char_str;
+ gint i;
+ PangoLayoutLine *line;
+
+ text = pango_layout_get_text (layout);
+ iter = pango_layout_get_iter (layout);
+
+ has_more = TRUE;
+ index = pango_layout_iter_get_index (iter);
+ i = 0;
+ while (has_more)
+ {
+ line = pango_layout_iter_get_line (iter);
+ has_more = pango_layout_iter_next_line (iter);
+ i++;
+
+ if (has_more)
+ {
+ index2 = pango_layout_iter_get_index (iter);
+ char_str = g_strndup (text + index, index2 - index);
+ }
+ else
+ {
+ char_str = g_strdup (text + index);
+ }
+
+ g_string_append_printf (string, "i=%d, index=%d, paragraph-start=%d, dir=%s '%s'\n",
+ i, index, line->is_paragraph_start, direction_name (line->resolved_dir),
+ char_str);
+ g_free (char_str);
+
+ index = index2;
+ }
+ pango_layout_iter_free (iter);
+}
+
+static void
+dump_runs (PangoLayout *layout, GString *string)
+{
+ PangoLayoutIter *iter;
+ PangoLayoutRun *run;
+ PangoItem *item;
+ const gchar *text;
+ gint index, index2;
+ gboolean has_more;
+ gchar *char_str;
+ gint i;
+ gchar *font;
+
+ text = pango_layout_get_text (layout);
+ iter = pango_layout_get_iter (layout);
+
+ has_more = TRUE;
+ index = pango_layout_iter_get_index (iter);
+ i = 0;
+ while (has_more)
+ {
+ run = pango_layout_iter_get_run (iter);
+ has_more = pango_layout_iter_next_run (iter);
+ i++;
+
+ if (has_more)
+ {
+ index2 = pango_layout_iter_get_index (iter);
+ char_str = g_strndup (text + index, index2 - index);
+ }
+ else
+ {
+ char_str = g_strdup (text + index);
+ }
+
+ if (run)
+ {
+ item = ((PangoGlyphItem*)run)->item;
+ font = font_name (item->analysis.font);
+ g_string_append_printf (string, "i=%d, index=%d, chars=%d, level=%d, gravity=%s, flags=%d, font=%s, script=%s, language=%s, '%s'\n",
+ i, index, item->num_chars, item->analysis.level,
+ gravity_name (item->analysis.gravity),
+ item->analysis.flags,
+ font,
+ script_name (item->analysis.script),
+ pango_language_to_string (item->analysis.language),
+ char_str);
+ dump_extra_attrs (item->analysis.extra_attrs, string);
+ g_free (font);
+ }
+ else
+ {
+ g_string_append_printf (string, "i=%d, index=%d, no run, line end\n",
+ i, index);
+ }
+
+ g_free (char_str);
+ index = index2;
+ }
+ pango_layout_iter_free (iter);
+}
+
+static void
+parse_params (const gchar *str,
+ gint *width,
+ gint *ellipsize_at,
+ PangoEllipsizeMode *ellipsize,
+ PangoWrapMode *wrap)
+{
+ gchar **strings;
+ gchar **str2;
+ gint i;
+ GEnumClass *eclass;
+ GEnumValue *ev;
+
+ strings = g_strsplit (str, ",", -1);
+ for (i = 0; strings[i]; i++)
+ {
+ str2 = g_strsplit (strings[i], "=", -1);
+ if (strcmp (str2[0], "width") == 0)
+ *width = (gint) g_ascii_strtoll (str2[1], NULL, 10);
+ else if (strcmp (str2[0], "ellipsize-at") == 0)
+ *ellipsize_at = (gint) g_ascii_strtoll (str2[1], NULL, 10);
+ else if (strcmp (str2[0], "ellipsize") == 0)
+ {
+ eclass = g_type_class_ref (PANGO_TYPE_ELLIPSIZE_MODE);
+ ev = g_enum_get_value_by_name (eclass, str2[1]);
+ if (!ev)
+ ev = g_enum_get_value_by_nick (eclass, str2[1]);
+ if (ev)
+ *ellipsize = ev->value;
+ g_type_class_unref (eclass);
+ }
+ else if (strcmp (str2[0], "wrap") == 0)
+ {
+ eclass = g_type_class_ref (PANGO_TYPE_WRAP_MODE);
+ ev = g_enum_get_value_by_name (eclass, str2[1]);
+ if (!ev)
+ ev = g_enum_get_value_by_nick (eclass, str2[1]);
+ if (ev)
+ *wrap = ev->value;
+ g_type_class_unref (eclass);
+ }
+ g_strfreev (str2);
+ }
+ g_strfreev (strings);
+}
+
+static void
+test_file (const gchar *filename, GString *string)
+{
+ gchar *contents;
+ gchar *markup;
+ gsize length;
+ GError *error = NULL;
+ PangoLayout *layout;
+ gchar *p;
+ gint width = 0;
+ gint ellipsize_at = 0;
+ PangoEllipsizeMode ellipsize = PANGO_ELLIPSIZE_NONE;
+ PangoWrapMode wrap = PANGO_WRAP_WORD;
+
+ if (!g_file_get_contents (filename, &contents, &length, &error))
+ {
+ fprintf (stderr, "%s\n", error->message);
+ g_error_free (error);
+ return;
+ }
+
+ p = strchr (contents, '\n');
+ g_assert (p);
+ markup = p + 1;
+ *p = '\0';
+
+ parse_params (contents, &width, &ellipsize_at, &ellipsize, &wrap);
+
+ layout = pango_layout_new (context);
+ pango_layout_set_markup (layout, markup, length);
+ g_free (contents);
+
+ if (width != 0)
+ pango_layout_set_width (layout, width * PANGO_SCALE);
+ pango_layout_set_ellipsize (layout, ellipsize);
+ pango_layout_set_wrap (layout, wrap);
+
+ g_string_append (string, pango_layout_get_text (layout));
+ g_string_append (string, "\n---\n\n");
+ g_string_append_printf (string, "wrapped: %d\n", pango_layout_is_wrapped (layout));
+ g_string_append_printf (string, "ellipsized: %d\n", pango_layout_is_ellipsized (layout));
+ g_string_append_printf (string, "lines: %d\n", pango_layout_get_line_count (layout));
+ if (width != 0)
+ g_string_append_printf (string, "width: %d\n", pango_layout_get_width (layout));
+ g_string_append (string, "\n---\n\n");
+ dump_attrs (pango_layout_get_attributes (layout), string);
+ g_string_append (string, "\n---\n\n");
+ dump_lines (layout, string);
+ g_string_append (string, "\n---\n\n");
+ dump_runs (layout, string);
+
+ g_object_unref (layout);
+}
+
+static gchar *
+get_expected_filename (const gchar *filename)
+{
+ gchar *f, *p, *expected;
+
+ f = g_strdup (filename);
+ p = strstr (f, ".markup");
+ if (p)
+ *p = 0;
+ expected = g_strconcat (f, ".expected", NULL);
+
+ g_free (f);
+
+ return expected;
+}
+
+static void
+test_layout (gconstpointer d)
+{
+ const gchar *filename = d;
+ gchar *expected_file;
+ gchar *expected;
+ GError *error = NULL;
+ GString *string;
+
+ expected_file = get_expected_filename (filename);
+
+ string = g_string_sized_new (0);
+
+ test_file (filename, string);
+
+ g_file_get_contents (expected_file, &expected, NULL, &error);
+ g_assert_no_error (error);
+ g_assert_cmpstr (string->str, ==, expected);
+ g_free (expected);
+
+ g_string_free (string, TRUE);
+
+ g_free (expected_file);
+}
+
+int
+main (int argc, char *argv[])
+{
+ GDir *dir;
+ GError *error = NULL;
+ const gchar *name;
+ gchar *path;
+ PangoFontDescription *desc;
+
+ g_setenv ("LC_ALL", "C", TRUE);
+ setlocale (LC_ALL, "");
+
+ g_test_init (&argc, &argv, NULL);
+
+ context = pango_font_map_create_context (pango_cairo_font_map_get_default ());
+ desc = pango_font_description_from_string ("Cantarell 11");
+ pango_context_set_font_description (context, desc);
+ pango_font_description_free (desc);
+
+ /* allow to easily generate expected output for new test cases */
+ if (argc > 1)
+ {
+ GString *string;
+
+ string = g_string_sized_new (0);
+ test_file (argv[1], string);
+ g_print ("%s", string->str);
+
+ return 0;
+ }
+
+ path = g_test_build_filename (G_TEST_DIST, "layouts", NULL);
+ dir = g_dir_open (path, 0, &error);
+ g_free (path);
+ g_assert_no_error (error);
+ while ((name = g_dir_read_name (dir)) != NULL)
+ {
+ if (!strstr (name, "markup"))
+ continue;
+
+ path = g_strdup_printf ("/layout/%s", name);
+ g_test_add_data_func_full (path, g_test_build_filename (G_TEST_DIST, "layouts", name, NULL),
+ test_layout, g_free);
+ g_free (path);
+ }
+ g_dir_close (dir);
+
+ return g_test_run ();
+}