1 /* 2 * Copyright 2011 Google Inc. 3 * 4 * Use of this source code is governed by a BSD-style license that can be 5 * found in the LICENSE file. 6 */ 7 8 #include "include/core/SkBitmap.h" 9 #include "include/core/SkCanvas.h" 10 #include "include/core/SkColor.h" 11 #include "include/core/SkFont.h" 12 #include "include/core/SkMatrix.h" 13 #include "include/core/SkPaint.h" 14 #include "include/core/SkPathEffect.h" 15 #include "include/core/SkPoint.h" 16 #include "include/core/SkRect.h" 17 #include "include/core/SkRefCnt.h" 18 #include "include/core/SkScalar.h" 19 #include "include/core/SkSurface.h" 20 #include "include/core/SkTextBlob.h" 21 #include "include/core/SkTypeface.h" 22 #include "include/core/SkTypes.h" 23 #include "include/effects/SkDashPathEffect.h" 24 #include "tests/Test.h" 25 26 #include <cmath> 27 28 static const SkColor bgColor = SK_ColorWHITE; 29 create(SkBitmap * bm,SkIRect bound)30 static void create(SkBitmap* bm, SkIRect bound) { 31 bm->allocN32Pixels(bound.width(), bound.height()); 32 } 33 34 /** Assumes that the ref draw was completely inside ref canvas -- 35 implies that everything outside is "bgColor". 36 Checks that all overlap is the same and that all non-overlap on the 37 ref is "bgColor". 38 */ compare(const SkBitmap & ref,const SkIRect & iref,const SkBitmap & test,const SkIRect & itest)39 static bool compare(const SkBitmap& ref, const SkIRect& iref, 40 const SkBitmap& test, const SkIRect& itest) 41 { 42 const int xOff = itest.fLeft - iref.fLeft; 43 const int yOff = itest.fTop - iref.fTop; 44 45 for (int y = 0; y < test.height(); ++y) { 46 for (int x = 0; x < test.width(); ++x) { 47 SkColor testColor = test.getColor(x, y); 48 int refX = x + xOff; 49 int refY = y + yOff; 50 SkColor refColor; 51 if (refX >= 0 && refX < ref.width() && 52 refY >= 0 && refY < ref.height()) 53 { 54 refColor = ref.getColor(refX, refY); 55 } else { 56 refColor = bgColor; 57 } 58 if (refColor != testColor) { 59 return false; 60 } 61 } 62 } 63 return true; 64 } 65 66 /** Test that drawing glyphs with empty paths is different from drawing glyphs without paths. */ DEF_TEST(DrawText_dashout,reporter)67 DEF_TEST(DrawText_dashout, reporter) { 68 SkIRect size = SkIRect::MakeWH(64, 64); 69 70 SkBitmap drawTextBitmap; 71 create(&drawTextBitmap, size); 72 SkCanvas drawTextCanvas(drawTextBitmap); 73 74 SkBitmap drawDashedTextBitmap; 75 create(&drawDashedTextBitmap, size); 76 SkCanvas drawDashedTextCanvas(drawDashedTextBitmap); 77 78 SkBitmap emptyBitmap; 79 create(&emptyBitmap, size); 80 SkCanvas emptyCanvas(emptyBitmap); 81 82 SkPoint point = SkPoint::Make(25.0f, 25.0f); 83 SkFont font(nullptr, 20); 84 font.setEdging(SkFont::Edging::kSubpixelAntiAlias); 85 font.setSubpixel(true); 86 87 SkPaint paint; 88 paint.setColor(SK_ColorGRAY); 89 paint.setStyle(SkPaint::kStroke_Style); 90 91 // Draw a stroked "A" without a dash which will draw something. 92 drawTextCanvas.drawColor(SK_ColorWHITE); 93 drawTextCanvas.drawString("A", point.fX, point.fY, font, paint); 94 95 // Draw an "A" but with a dash which will never draw anything. 96 paint.setStrokeWidth(2); 97 constexpr SkScalar bigInterval = 10000; 98 static constexpr SkScalar intervals[] = { 1, bigInterval }; 99 paint.setPathEffect(SkDashPathEffect::Make(intervals, SK_ARRAY_COUNT(intervals), 2)); 100 101 drawDashedTextCanvas.drawColor(SK_ColorWHITE); 102 drawDashedTextCanvas.drawString("A", point.fX, point.fY, font, paint); 103 104 // Draw nothing. 105 emptyCanvas.drawColor(SK_ColorWHITE); 106 107 REPORTER_ASSERT(reporter, !compare(drawTextBitmap, size, emptyBitmap, size)); 108 REPORTER_ASSERT(reporter, compare(drawDashedTextBitmap, size, emptyBitmap, size)); 109 } 110 111 // Test drawing text at some unusual coordinates. 112 // We measure success by not crashing or asserting. DEF_TEST(DrawText_weirdCoordinates,r)113 DEF_TEST(DrawText_weirdCoordinates, r) { 114 auto surface = SkSurface::MakeRasterN32Premul(10,10); 115 auto canvas = surface->getCanvas(); 116 117 SkScalar oddballs[] = { 0.0f, (float)INFINITY, (float)NAN, 34359738368.0f }; 118 119 for (auto x : oddballs) { 120 canvas->drawString("a", +x, 0.0f, SkFont(), SkPaint()); 121 canvas->drawString("a", -x, 0.0f, SkFont(), SkPaint()); 122 } 123 for (auto y : oddballs) { 124 canvas->drawString("a", 0.0f, +y, SkFont(), SkPaint()); 125 canvas->drawString("a", 0.0f, -y, SkFont(), SkPaint()); 126 } 127 } 128 129 // Test drawing text with some unusual matricies. 130 // We measure success by not crashing or asserting. DEF_TEST(DrawText_weirdMatricies,r)131 DEF_TEST(DrawText_weirdMatricies, r) { 132 auto surface = SkSurface::MakeRasterN32Premul(100,100); 133 auto canvas = surface->getCanvas(); 134 135 SkFont font; 136 font.setEdging(SkFont::Edging::kSubpixelAntiAlias); 137 138 struct { 139 SkScalar textSize; 140 SkScalar matrix[9]; 141 } testCases[] = { 142 // 2x2 singular 143 {10, { 0, 0, 0, 0, 0, 0, 0, 0, 1}}, 144 {10, { 0, 0, 0, 0, 1, 0, 0, 0, 1}}, 145 {10, { 0, 0, 0, 1, 0, 0, 0, 0, 1}}, 146 {10, { 0, 0, 0, 1, 1, 0, 0, 0, 1}}, 147 {10, { 0, 1, 0, 0, 1, 0, 0, 0, 1}}, 148 {10, { 1, 0, 0, 0, 0, 0, 0, 0, 1}}, 149 {10, { 1, 0, 0, 1, 0, 0, 0, 0, 1}}, 150 {10, { 1, 1, 0, 0, 0, 0, 0, 0, 1}}, 151 {10, { 1, 1, 0, 1, 1, 0, 0, 0, 1}}, 152 // See https://bugzilla.mozilla.org/show_bug.cgi?id=1305085 . 153 { 1, {10, 20, 0, 20, 40, 0, 0, 0, 1}}, 154 }; 155 156 for (const auto& testCase : testCases) { 157 font.setSize(testCase.textSize); 158 const SkScalar(&m)[9] = testCase.matrix; 159 SkMatrix mat; 160 mat.setAll(m[0], m[1], m[2], m[3], m[4], m[5], m[6], m[7], m[8]); 161 canvas->setMatrix(mat); 162 canvas->drawString("Hamburgefons", 10, 10, font, SkPaint()); 163 } 164 } 165 166 // This produces no glyphs, and is to check that buffers from previous draws don't get 167 // reused. DEF_TEST(DrawText_noglyphs,r)168 DEF_TEST(DrawText_noglyphs, r) { 169 auto surface = SkSurface::MakeRasterN32Premul(100,100); 170 auto canvas = surface->getCanvas(); 171 auto text = "Hamburgfons"; 172 { 173 // scoped to ensure blob is deleted. 174 auto blob = SkTextBlob::MakeFromText(text, strlen(text), SkFont()); 175 canvas->drawTextBlob(blob, 10, 10, SkPaint()); 176 } 177 canvas->drawString( 178 "\x0d\xf3\xf2\xf2\xe9\x0d\x0d\x0d\x05\x0d\x0d\xe3\xe3\xe3\xe3\xe3\xe3\xe3\xe3\xe3", 179 10, 20, SkFont(), SkPaint()); 180 } 181