aboutsummaryrefslogtreecommitdiffhomepage
path: root/tests/SVGDeviceTest.cpp
diff options
context:
space:
mode:
authorGravatar fmalita <fmalita@chromium.org>2015-02-20 13:54:40 -0800
committerGravatar Commit bot <commit-bot@chromium.org>2015-02-20 13:54:40 -0800
commit7a048690d3cb6b3c1bdef594c3bd4b66f6b400c1 (patch)
tree2bd30bcc04e8db8db2de40e33a2a4435c7bc4072 /tests/SVGDeviceTest.cpp
parentc1b71d6c30041f01675dd54a77adc9c177afdf44 (diff)
[SVGDevice] Text whitespace unittest
Plumb SkDOM as needed to make it suitable for an SkXMLWriter backend. Also fix a potential null typeface issue in SkSVGDevice::AutoElement::addTextAttributes(). R=reed@google.com,mtklein@google.com Review URL: https://codereview.chromium.org/940283002
Diffstat (limited to 'tests/SVGDeviceTest.cpp')
-rw-r--r--tests/SVGDeviceTest.cpp149
1 files changed, 149 insertions, 0 deletions
diff --git a/tests/SVGDeviceTest.cpp b/tests/SVGDeviceTest.cpp
new file mode 100644
index 0000000000..c973f8b15f
--- /dev/null
+++ b/tests/SVGDeviceTest.cpp
@@ -0,0 +1,149 @@
+/*
+ * Copyright 2015 Google Inc.
+ *
+ * Use of this source code is governed by a BSD-style license that can be
+ * found in the LICENSE file.
+ */
+
+#include "SkCanvas.h"
+#include "SkData.h"
+#include "SkDOM.h"
+#include "SkParse.h"
+#include "SkStream.h"
+#include "SkSVGCanvas.h"
+#include "SkXMLWriter.h"
+#include "Test.h"
+
+#include <string.h>
+
+namespace {
+
+void check_text_node(skiatest::Reporter* reporter,
+ const SkDOM& dom,
+ const SkDOM::Node* root,
+ const SkPoint& offset,
+ unsigned scalarsPerPos,
+ const char* expected) {
+ if (root == NULL) {
+ ERRORF(reporter, "root element not found.");
+ return;
+ }
+
+ const SkDOM::Node* textElem = dom.getFirstChild(root, "text");
+ if (textElem == NULL) {
+ ERRORF(reporter, "<text> element not found.");
+ return;
+ }
+ REPORTER_ASSERT(reporter, dom.getType(textElem) == SkDOM::kElement_Type);
+
+ const SkDOM::Node* textNode= dom.getFirstChild(textElem);
+ REPORTER_ASSERT(reporter, textNode != NULL);
+ if (textNode != NULL) {
+ REPORTER_ASSERT(reporter, dom.getType(textNode) == SkDOM::kText_Type);
+ REPORTER_ASSERT(reporter, strcmp(expected, dom.getName(textNode)) == 0);
+ }
+
+ int textLen = SkToInt(strlen(expected));
+
+ const char* x = dom.findAttr(textElem, "x");
+ REPORTER_ASSERT(reporter, x != NULL);
+ if (x != NULL) {
+ int xposCount = (scalarsPerPos < 1) ? 1 : textLen;
+ REPORTER_ASSERT(reporter, SkParse::Count(x) == xposCount);
+
+ SkAutoTMalloc<SkScalar> xpos(xposCount);
+ SkParse::FindScalars(x, xpos.get(), xposCount);
+ if (scalarsPerPos < 1) {
+ REPORTER_ASSERT(reporter, xpos[0] == offset.x());
+ } else {
+ for (int i = 0; i < xposCount; ++i) {
+ REPORTER_ASSERT(reporter, xpos[i] == SkIntToScalar(expected[i]));
+ }
+ }
+ }
+
+ const char* y = dom.findAttr(textElem, "y");
+ REPORTER_ASSERT(reporter, y != NULL);
+ if (y != NULL) {
+ int yposCount = (scalarsPerPos < 2) ? 1 : textLen;
+ REPORTER_ASSERT(reporter, SkParse::Count(y) == yposCount);
+
+ SkAutoTMalloc<SkScalar> ypos(yposCount);
+ SkParse::FindScalars(y, ypos.get(), yposCount);
+ if (scalarsPerPos < 2) {
+ REPORTER_ASSERT(reporter, ypos[0] == offset.y());
+ } else {
+ for (int i = 0; i < yposCount; ++i) {
+ REPORTER_ASSERT(reporter, ypos[i] == -SkIntToScalar(expected[i]));
+ }
+ }
+ }
+}
+
+void test_whitespace_pos(skiatest::Reporter* reporter,
+ const char* txt,
+ const char* expected) {
+ size_t len = strlen(txt);
+
+ SkDOM dom;
+ SkPaint paint;
+ SkPoint offset = SkPoint::Make(10, 20);
+
+ {
+ SkXMLParserWriter writer(dom.beginParsing());
+ SkAutoTUnref<SkCanvas> svgCanvas(SkSVGCanvas::Create(SkRect::MakeWH(100, 100),
+ &writer));
+ svgCanvas->drawText(txt, len, offset.x(), offset.y(), paint);
+ }
+ check_text_node(reporter, dom, dom.finishParsing(), offset, 0, expected);
+
+ {
+ SkAutoTMalloc<SkScalar> xpos(len);
+ for (int i = 0; i < SkToInt(len); ++i) {
+ xpos[i] = SkIntToScalar(txt[i]);
+ }
+
+ SkXMLParserWriter writer(dom.beginParsing());
+ SkAutoTUnref<SkCanvas> svgCanvas(SkSVGCanvas::Create(SkRect::MakeWH(100, 100),
+ &writer));
+ svgCanvas->drawPosTextH(txt, len, xpos, offset.y(), paint);
+ }
+ check_text_node(reporter, dom, dom.finishParsing(), offset, 1, expected);
+
+ {
+ SkAutoTMalloc<SkPoint> pos(len);
+ for (int i = 0; i < SkToInt(len); ++i) {
+ pos[i] = SkPoint::Make(SkIntToScalar(txt[i]), -SkIntToScalar(txt[i]));
+ }
+
+ SkXMLParserWriter writer(dom.beginParsing());
+ SkAutoTUnref<SkCanvas> svgCanvas(SkSVGCanvas::Create(SkRect::MakeWH(100, 100),
+ &writer));
+ svgCanvas->drawPosText(txt, len, pos, paint);
+ }
+ check_text_node(reporter, dom, dom.finishParsing(), offset, 2, expected);
+}
+
+}
+
+DEF_TEST(SVGDevice_whitespace_pos, reporter) {
+ static const struct {
+ const char* tst_in;
+ const char* tst_out;
+ } tests[] = {
+ { "abcd" , "abcd" },
+ { "ab cd" , "ab cd" },
+ { "ab \t\t cd", "ab cd" },
+ { " abcd" , "abcd" },
+ { " abcd" , "abcd" },
+ { " \t\t abcd", "abcd" },
+ { "abcd " , "abcd " }, // we allow one trailing whitespace char
+ { "abcd " , "abcd " }, // because it makes no difference and
+ { "abcd\t " , "abcd\t" }, // simplifies the implementation
+ { "\t\t \t ab \t\t \t cd \t\t \t ", "ab cd " },
+ };
+
+ for (unsigned i = 0; i < SK_ARRAY_COUNT(tests); ++i) {
+ test_whitespace_pos(reporter, tests[i].tst_in, tests[i].tst_out);
+ }
+}