ba8feb56c3
This distuguishes between glyphs which do not have a path and glyphs which have a path but that path resolves to the empty path. BUG=chromium:816763 Change-Id: Id6c7dd66cdad3868bf3fe15bcb6e5e6f2ca82405 Reviewed-on: https://skia-review.googlesource.com/112484 Reviewed-by: Herb Derby <herb@google.com> Commit-Queue: Ben Wagner <bungeman@google.com>
220 lines
7.4 KiB
C++
220 lines
7.4 KiB
C++
/*
|
|
* 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);
|
|
}
|
|
}
|