| /* | 
 |  * Copyright 2011 Google Inc. | 
 |  * | 
 |  * Use of this source code is governed by a BSD-style license that can be | 
 |  * found in the LICENSE file. | 
 |  */ | 
 |  | 
 | #include "SkBitmap.h" | 
 | #include "SkCanvas.h" | 
 | #include "SkColor.h" | 
 | #include "SkDashPathEffect.h" | 
 | #include "SkMatrix.h" | 
 | #include "SkPaint.h" | 
 | #include "SkPathEffect.h" | 
 | #include "SkPoint.h" | 
 | #include "SkRect.h" | 
 | #include "SkRefCnt.h" | 
 | #include "SkScalar.h" | 
 | #include "SkSurface.h" | 
 | #include "SkTypes.h" | 
 | #include "Test.h" | 
 |  | 
 | #include <math.h> | 
 |  | 
 | static const SkColor bgColor = SK_ColorWHITE; | 
 |  | 
 | static void create(SkBitmap* bm, SkIRect bound) { | 
 |     bm->allocN32Pixels(bound.width(), bound.height()); | 
 | } | 
 |  | 
 | static void drawBG(SkCanvas* canvas) { | 
 |     canvas->drawColor(bgColor); | 
 | } | 
 |  | 
 | /** Assumes that the ref draw was completely inside ref canvas -- | 
 |     implies that everything outside is "bgColor". | 
 |     Checks that all overlap is the same and that all non-overlap on the | 
 |     ref is "bgColor". | 
 |  */ | 
 | static bool compare(const SkBitmap& ref, const SkIRect& iref, | 
 |                     const SkBitmap& test, const SkIRect& itest) | 
 | { | 
 |     const int xOff = itest.fLeft - iref.fLeft; | 
 |     const int yOff = itest.fTop - iref.fTop; | 
 |  | 
 |     for (int y = 0; y < test.height(); ++y) { | 
 |         for (int x = 0; x < test.width(); ++x) { | 
 |             SkColor testColor = test.getColor(x, y); | 
 |             int refX = x + xOff; | 
 |             int refY = y + yOff; | 
 |             SkColor refColor; | 
 |             if (refX >= 0 && refX < ref.width() && | 
 |                 refY >= 0 && refY < ref.height()) | 
 |             { | 
 |                 refColor = ref.getColor(refX, refY); | 
 |             } else { | 
 |                 refColor = bgColor; | 
 |             } | 
 |             if (refColor != testColor) { | 
 |                 return false; | 
 |             } | 
 |         } | 
 |     } | 
 |     return true; | 
 | } | 
 |  | 
 | DEF_TEST(DrawText, reporter) { | 
 |     SkPaint paint; | 
 |     paint.setColor(SK_ColorGRAY); | 
 |     paint.setTextSize(SkIntToScalar(20)); | 
 |  | 
 |     SkIRect drawTextRect = SkIRect::MakeWH(64, 64); | 
 |     SkBitmap drawTextBitmap; | 
 |     create(&drawTextBitmap, drawTextRect); | 
 |     SkCanvas drawTextCanvas(drawTextBitmap); | 
 |  | 
 |     SkIRect drawPosTextRect = SkIRect::MakeWH(64, 64); | 
 |     SkBitmap drawPosTextBitmap; | 
 |     create(&drawPosTextBitmap, drawPosTextRect); | 
 |     SkCanvas drawPosTextCanvas(drawPosTextBitmap); | 
 |  | 
 |     // Two test cases "A" for the normal path through the code, and " " to check the | 
 |     // early return path. | 
 |     const char* cases[] = {"A", " "}; | 
 |     for (auto c : cases) { | 
 |         for (float offsetY = 0.0f; offsetY < 1.0f; offsetY += (1.0f / 16.0f)) { | 
 |             for (float offsetX = 0.0f; offsetX < 1.0f; offsetX += (1.0f / 16.0f)) { | 
 |                 SkPoint point = SkPoint::Make(25.0f + offsetX, | 
 |                                               25.0f + offsetY); | 
 |  | 
 |                 for (int align = 0; align < SkPaint::kAlignCount; ++align) { | 
 |                     paint.setTextAlign(static_cast<SkPaint::Align>(align)); | 
 |  | 
 |                     for (unsigned int flags = 0; flags < (1 << 3); ++flags) { | 
 |                         static const unsigned int antiAliasFlag = 1; | 
 |                         static const unsigned int subpixelFlag = 1 << 1; | 
 |                         static const unsigned int lcdFlag = 1 << 2; | 
 |  | 
 |                         paint.setAntiAlias(SkToBool(flags & antiAliasFlag)); | 
 |                         paint.setSubpixelText(SkToBool(flags & subpixelFlag)); | 
 |                         paint.setLCDRenderText(SkToBool(flags & lcdFlag)); | 
 |  | 
 |                         // Test: drawText and drawPosText draw the same. | 
 |                         drawBG(&drawTextCanvas); | 
 |                         drawTextCanvas.drawText(c, 1, point.fX, point.fY, paint); | 
 |  | 
 |                         drawBG(&drawPosTextCanvas); | 
 |                         drawPosTextCanvas.drawPosText(c, 1, &point, paint); | 
 |  | 
 |                         REPORTER_ASSERT(reporter, | 
 |                                         compare(drawTextBitmap, drawTextRect, | 
 |                                                 drawPosTextBitmap, drawPosTextRect)); | 
 |                     } | 
 |                 } | 
 |             } | 
 |         } | 
 |     } | 
 | } | 
 |  | 
 | /** Test that drawing glyphs with empty paths is different from drawing glyphs without paths. */ | 
 | DEF_TEST(DrawText_dashout, reporter) { | 
 |     SkIRect size = SkIRect::MakeWH(64, 64); | 
 |  | 
 |     SkBitmap drawTextBitmap; | 
 |     create(&drawTextBitmap, size); | 
 |     SkCanvas drawTextCanvas(drawTextBitmap); | 
 |  | 
 |     SkBitmap drawDashedTextBitmap; | 
 |     create(&drawDashedTextBitmap, size); | 
 |     SkCanvas drawDashedTextCanvas(drawDashedTextBitmap); | 
 |  | 
 |     SkBitmap emptyBitmap; | 
 |     create(&emptyBitmap, size); | 
 |     SkCanvas emptyCanvas(emptyBitmap); | 
 |  | 
 |     SkPoint point = SkPoint::Make(25.0f, 25.0f); | 
 |     SkPaint paint; | 
 |     paint.setColor(SK_ColorGRAY); | 
 |     paint.setTextSize(SkIntToScalar(20)); | 
 |     paint.setAntiAlias(true); | 
 |     paint.setSubpixelText(true); | 
 |     paint.setLCDRenderText(true); | 
 |     paint.setStyle(SkPaint::kStroke_Style); | 
 |  | 
 |     // Draw a stroked "A" without a dash which will draw something. | 
 |     drawBG(&drawTextCanvas); | 
 |     drawTextCanvas.drawText("A", 1, point.fX, point.fY, paint); | 
 |  | 
 |     // Draw an "A" but with a dash which will never draw anything. | 
 |     paint.setStrokeWidth(2); | 
 |     constexpr SkScalar bigInterval = 10000; | 
 |     static constexpr SkScalar intervals[] = { 1, bigInterval }; | 
 |     paint.setPathEffect(SkDashPathEffect::Make(intervals, SK_ARRAY_COUNT(intervals), 2)); | 
 |  | 
 |     drawBG(&drawDashedTextCanvas); | 
 |     drawDashedTextCanvas.drawText("A", 1, point.fX, point.fY, paint); | 
 |  | 
 |     // Draw nothing. | 
 |     drawBG(&emptyCanvas); | 
 |  | 
 |     REPORTER_ASSERT(reporter, !compare(drawTextBitmap, size, emptyBitmap, size)); | 
 |     REPORTER_ASSERT(reporter, compare(drawDashedTextBitmap, size, emptyBitmap, size)); | 
 | } | 
 |  | 
 | // Test drawing text at some unusual coordinates. | 
 | // We measure success by not crashing or asserting. | 
 | DEF_TEST(DrawText_weirdCoordinates, r) { | 
 |     auto surface = SkSurface::MakeRasterN32Premul(10,10); | 
 |     auto canvas = surface->getCanvas(); | 
 |  | 
 |     SkScalar oddballs[] = { 0.0f, (float)INFINITY, (float)NAN, 34359738368.0f }; | 
 |  | 
 |     for (auto x : oddballs) { | 
 |         canvas->drawString("a", +x, 0.0f, SkPaint()); | 
 |         canvas->drawString("a", -x, 0.0f, SkPaint()); | 
 |     } | 
 |     for (auto y : oddballs) { | 
 |         canvas->drawString("a", 0.0f, +y, SkPaint()); | 
 |         canvas->drawString("a", 0.0f, -y, SkPaint()); | 
 |     } | 
 | } | 
 |  | 
 | // Test drawing text with some unusual matricies. | 
 | // We measure success by not crashing or asserting. | 
 | DEF_TEST(DrawText_weirdMatricies, r) { | 
 |     auto surface = SkSurface::MakeRasterN32Premul(100,100); | 
 |     auto canvas = surface->getCanvas(); | 
 |  | 
 |     SkPaint paint; | 
 |     paint.setAntiAlias(true); | 
 |     paint.setLCDRenderText(true); | 
 |  | 
 |     struct { | 
 |         SkScalar textSize; | 
 |         SkScalar matrix[9]; | 
 |     } testCases[] = { | 
 |         // 2x2 singular | 
 |         {10, { 0,  0,  0,  0,  0,  0,  0,  0,  1}}, | 
 |         {10, { 0,  0,  0,  0,  1,  0,  0,  0,  1}}, | 
 |         {10, { 0,  0,  0,  1,  0,  0,  0,  0,  1}}, | 
 |         {10, { 0,  0,  0,  1,  1,  0,  0,  0,  1}}, | 
 |         {10, { 0,  1,  0,  0,  1,  0,  0,  0,  1}}, | 
 |         {10, { 1,  0,  0,  0,  0,  0,  0,  0,  1}}, | 
 |         {10, { 1,  0,  0,  1,  0,  0,  0,  0,  1}}, | 
 |         {10, { 1,  1,  0,  0,  0,  0,  0,  0,  1}}, | 
 |         {10, { 1,  1,  0,  1,  1,  0,  0,  0,  1}}, | 
 |         // See https://bugzilla.mozilla.org/show_bug.cgi?id=1305085 . | 
 |         { 1, {10, 20,  0, 20, 40,  0,  0,  0,  1}}, | 
 |     }; | 
 |  | 
 |     for (const auto& testCase : testCases) { | 
 |         paint.setTextSize(testCase.textSize); | 
 |         const SkScalar(&m)[9] = testCase.matrix; | 
 |         SkMatrix mat; | 
 |         mat.setAll(m[0], m[1], m[2], m[3], m[4], m[5], m[6], m[7], m[8]); | 
 |         canvas->setMatrix(mat); | 
 |         canvas->drawString("Hamburgefons", 10, 10, paint); | 
 |     } | 
 | } |