diff options
author | 2015-02-20 13:54:40 -0800 | |
---|---|---|
committer | 2015-02-20 13:54:40 -0800 | |
commit | 7a048690d3cb6b3c1bdef594c3bd4b66f6b400c1 (patch) | |
tree | 2bd30bcc04e8db8db2de40e33a2a4435c7bc4072 /tests/SVGDeviceTest.cpp | |
parent | c1b71d6c30041f01675dd54a77adc9c177afdf44 (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.cpp | 149 |
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); + } +} |