/* * 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 == nullptr) { ERRORF(reporter, "root element not found."); return; } const SkDOM::Node* textElem = dom.getFirstChild(root, "text"); if (textElem == nullptr) { 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 != nullptr); if (textNode != nullptr) { 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 != nullptr); if (x != nullptr) { 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 != nullptr); if (y != nullptr) { 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()); std::unique_ptr<SkCanvas> svgCanvas = SkSVGCanvas::Make(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()); std::unique_ptr<SkCanvas> svgCanvas = SkSVGCanvas::Make(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()); std::unique_ptr<SkCanvas> svgCanvas = SkSVGCanvas::Make(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); } }