2019-10-09 16:26:56 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2019 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "src/core/SkEnumerate.h"
|
|
|
|
#include "src/core/SkGlyphBuffer.h"
|
|
|
|
#include "src/core/SkGlyphRunPainter.h"
|
|
|
|
#include "src/core/SkScalerContext.h"
|
|
|
|
#include "tests/Test.h"
|
|
|
|
|
2019-10-30 20:33:16 +00:00
|
|
|
DEF_TEST(SkPackedGlyphIDCtor, reporter) {
|
2019-11-06 20:47:43 +00:00
|
|
|
using PG = SkPackedGlyphID;
|
|
|
|
// x and y are in one quarter the sub-pixel sampling frequency.
|
|
|
|
// Number of steps on the interval [0, 1)
|
|
|
|
const int perUnit = 1u << (PG::kSubPixelPosLen + 2);
|
|
|
|
const float step = 1.f / perUnit;
|
|
|
|
const int testLimit = 2 * perUnit;
|
|
|
|
auto freqRound = [](uint32_t x) -> uint32_t {
|
|
|
|
return ((x + 2) >> 2) & PG::kSubPixelPosMask;
|
|
|
|
};
|
|
|
|
|
2019-10-30 20:33:16 +00:00
|
|
|
{
|
|
|
|
// Normal sub-pixel with y-axis snapping.
|
|
|
|
auto roundingSpec = SkGlyphPositionRoundingSpec(true, kX_SkAxisAlignment);
|
|
|
|
SkIPoint mask = roundingSpec.ignorePositionFieldMask;
|
2019-11-06 20:47:43 +00:00
|
|
|
for (int x = -testLimit; x < testLimit; x++) {
|
|
|
|
float fx = x * step;
|
|
|
|
SkPoint roundedPos = SkPoint{fx, 0} + roundingSpec.halfAxisSampleFreq;
|
|
|
|
SkPackedGlyphID packedID{3, roundedPos, mask};
|
|
|
|
uint32_t subX = freqRound(x);
|
|
|
|
uint32_t subY = 0;
|
|
|
|
SkPackedGlyphID correctID(3, subX, subY);
|
|
|
|
SkASSERT(packedID == correctID);
|
|
|
|
REPORTER_ASSERT(reporter, packedID == correctID);
|
2019-10-30 20:33:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-06 18:04:54 +00:00
|
|
|
{
|
|
|
|
// No subpixel positioning.
|
|
|
|
auto roundingSpec = SkGlyphPositionRoundingSpec(false, kNone_SkAxisAlignment);
|
|
|
|
SkIPoint mask = roundingSpec.ignorePositionFieldMask;
|
2019-11-06 20:47:43 +00:00
|
|
|
for (int y = -testLimit; y < testLimit; y++) {
|
|
|
|
for (int x = -testLimit; x < testLimit; x++) {
|
2019-11-06 18:04:54 +00:00
|
|
|
float fx = x * step, fy = y * step;
|
|
|
|
SkPoint roundedPos = SkPoint{fx, fy} + roundingSpec.halfAxisSampleFreq;
|
|
|
|
SkPackedGlyphID packedID{3, roundedPos, mask};
|
2019-11-06 20:47:43 +00:00
|
|
|
uint32_t subX = 0;
|
|
|
|
uint32_t subY = 0;
|
2019-11-06 18:04:54 +00:00
|
|
|
SkPackedGlyphID correctID(3, subX, subY);
|
|
|
|
REPORTER_ASSERT(reporter, packedID == correctID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-30 20:33:16 +00:00
|
|
|
{
|
|
|
|
// Subpixel with no axis snapping.
|
|
|
|
auto roundingSpec = SkGlyphPositionRoundingSpec(true, kNone_SkAxisAlignment);
|
|
|
|
SkIPoint mask = roundingSpec.ignorePositionFieldMask;
|
2019-11-06 20:47:43 +00:00
|
|
|
for (int y = -testLimit; y < testLimit; y++) {
|
|
|
|
for (int x = -testLimit; x < testLimit; x++) {
|
2019-10-30 20:33:16 +00:00
|
|
|
float fx = x * step, fy = y * step;
|
|
|
|
SkPoint roundedPos = SkPoint{fx, fy} + roundingSpec.halfAxisSampleFreq;
|
|
|
|
SkPackedGlyphID packedID{3, roundedPos, mask};
|
2019-11-06 20:47:43 +00:00
|
|
|
uint32_t subX = freqRound(x);
|
|
|
|
uint32_t subY = freqRound(y);
|
2019-10-30 20:33:16 +00:00
|
|
|
SkPackedGlyphID correctID(3, subX, subY);
|
|
|
|
REPORTER_ASSERT(reporter, packedID == correctID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2019-11-06 18:04:54 +00:00
|
|
|
// Test dynamic range by transposing a large distance.
|
|
|
|
// Floating point numbers have 24 bits of precision. The largest distance is 24 - 2 (for
|
|
|
|
// sub-pixel) - 1 (for truncation to floor trick in the code). This leaves 21 bits. Large
|
|
|
|
// Distance is 2^21 - 2 (because the test is on the interval [-2, 2).
|
2019-11-06 20:47:43 +00:00
|
|
|
const uint32_t kLogLargeDistance = 24 - PG::kSubPixelPosLen - 1;
|
|
|
|
const int64_t kLargeDistance = (1ull << kLogLargeDistance) - 2;
|
2019-11-06 18:04:54 +00:00
|
|
|
auto roundingSpec = SkGlyphPositionRoundingSpec(true, kNone_SkAxisAlignment);
|
2019-10-30 20:33:16 +00:00
|
|
|
SkIPoint mask = roundingSpec.ignorePositionFieldMask;
|
|
|
|
for (int y = -32; y < 33; y++) {
|
|
|
|
for (int x = -32; x < 33; x++) {
|
2019-11-06 18:04:54 +00:00
|
|
|
float fx = x * step + kLargeDistance, fy = y * step + kLargeDistance;
|
2019-10-30 20:33:16 +00:00
|
|
|
SkPoint roundedPos = SkPoint{fx, fy} + roundingSpec.halfAxisSampleFreq;
|
|
|
|
SkPackedGlyphID packedID{3, roundedPos, mask};
|
2019-11-06 20:47:43 +00:00
|
|
|
uint32_t subX = freqRound(x);
|
|
|
|
uint32_t subY = freqRound(y);
|
2019-10-30 20:33:16 +00:00
|
|
|
SkPackedGlyphID correctID(3, subX, subY);
|
|
|
|
REPORTER_ASSERT(reporter, packedID == correctID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-10-11 14:56:31 +00:00
|
|
|
|
|
|
|
DEF_TEST(SkSourceGlyphBufferBasic, reporter) {
|
|
|
|
SkSourceGlyphBuffer rejects;
|
|
|
|
// Positions are picked to avoid precision problems.
|
|
|
|
const SkPoint positions[] = {{10.25,10.25}, {20.5,10.25}, {30.75,10.25}, {40,10.25}};
|
|
|
|
const SkGlyphID glyphIDs[] = {1, 2, 3, 4};
|
|
|
|
auto source = SkMakeZip(glyphIDs, positions);
|
|
|
|
|
|
|
|
rejects.setSource(source);
|
2019-11-14 18:51:43 +00:00
|
|
|
for (auto [i, glyphID, pos] : SkMakeEnumerate(rejects.source())) {
|
2019-10-11 14:56:31 +00:00
|
|
|
REPORTER_ASSERT(reporter, glyphID == std::get<0>(source[i]));
|
|
|
|
REPORTER_ASSERT(reporter, pos == std::get<1>(source[i]));
|
|
|
|
}
|
|
|
|
// Reject a couple of glyphs.
|
|
|
|
rejects.reject(1);
|
|
|
|
rejects.reject(2, 100);
|
|
|
|
rejects.flipRejectsToSource();
|
2022-01-11 16:21:02 +00:00
|
|
|
REPORTER_ASSERT(reporter, std::get<1>(rejects.maxDimensionHint()) == 100);
|
2019-11-14 18:51:43 +00:00
|
|
|
for (auto [i, glyphID, pos] : SkMakeEnumerate(rejects.source())) {
|
2019-10-11 14:56:31 +00:00
|
|
|
// This will index 1 and 2 from the original source.
|
|
|
|
size_t j = i + 1;
|
|
|
|
REPORTER_ASSERT(reporter, glyphID == std::get<0>(source[j]));
|
|
|
|
REPORTER_ASSERT(reporter, pos == std::get<1>(source[j]));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reject an additional glyph
|
|
|
|
rejects.reject(0, 10);
|
|
|
|
rejects.flipRejectsToSource();
|
2022-01-11 16:21:02 +00:00
|
|
|
REPORTER_ASSERT(reporter, std::get<1>(rejects.maxDimensionHint()) == 10);
|
2019-11-14 18:51:43 +00:00
|
|
|
for (auto [i, glyphID, pos] : SkMakeEnumerate(rejects.source())) {
|
2019-10-11 14:56:31 +00:00
|
|
|
// This will index 1 from the original source.
|
|
|
|
size_t j = i + 1;
|
|
|
|
REPORTER_ASSERT(reporter, glyphID == std::get<0>(source[j]));
|
|
|
|
REPORTER_ASSERT(reporter, pos == std::get<1>(source[j]));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start all over
|
|
|
|
rejects.setSource(source);
|
2019-11-14 18:51:43 +00:00
|
|
|
for (auto [i, glyphID, pos] : SkMakeEnumerate(rejects.source())) {
|
2019-10-11 14:56:31 +00:00
|
|
|
REPORTER_ASSERT(reporter, glyphID == std::get<0>(source[i]));
|
|
|
|
REPORTER_ASSERT(reporter, pos == std::get<1>(source[i]));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that everything is working after calling setSource.
|
|
|
|
rejects.reject(1);
|
|
|
|
rejects.reject(2, 100);
|
|
|
|
rejects.flipRejectsToSource();
|
2022-01-11 16:21:02 +00:00
|
|
|
REPORTER_ASSERT(reporter, std::get<1>(rejects.maxDimensionHint()) == 100);
|
2019-11-14 18:51:43 +00:00
|
|
|
for (auto [i, glyphID, pos] : SkMakeEnumerate(rejects.source())) {
|
2019-10-11 14:56:31 +00:00
|
|
|
// This will index 1 and 2 from the original source.
|
|
|
|
size_t j = i + 1;
|
|
|
|
REPORTER_ASSERT(reporter, glyphID == std::get<0>(source[j]));
|
|
|
|
REPORTER_ASSERT(reporter, pos == std::get<1>(source[j]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-09 16:26:56 +00:00
|
|
|
DEF_TEST(SkDrawableGlyphBufferBasic, reporter) {
|
|
|
|
// Positions are picked to avoid precision problems.
|
|
|
|
const SkPoint positions[] = {{10.25,10.25}, {20.5,10.25}, {30.75,10.25}, {40,10.25}};
|
|
|
|
const SkGlyphID glyphIDs[] = {1, 2, 3, 4};
|
|
|
|
SkGlyph glyphs[100];
|
|
|
|
auto source = SkMakeZip(glyphIDs, positions);
|
|
|
|
|
|
|
|
{
|
|
|
|
SkDrawableGlyphBuffer drawable;
|
|
|
|
drawable.ensureSize(100);
|
2020-05-14 16:26:25 +00:00
|
|
|
drawable.startSource(source);
|
2019-11-14 18:51:43 +00:00
|
|
|
for (auto [i, packedID, pos] : SkMakeEnumerate(drawable.input())) {
|
2019-10-30 15:58:18 +00:00
|
|
|
REPORTER_ASSERT(reporter, packedID.packedID().glyphID() == glyphIDs[i]);
|
2020-05-14 16:26:25 +00:00
|
|
|
REPORTER_ASSERT(reporter, pos == positions[i]);
|
2019-10-09 16:26:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
SkDrawableGlyphBuffer drawable;
|
|
|
|
drawable.ensureSize(100);
|
2020-05-21 16:11:27 +00:00
|
|
|
SkMatrix matrix = SkMatrix::Scale(0.5, 0.5);
|
2019-10-09 16:26:56 +00:00
|
|
|
SkGlyphPositionRoundingSpec rounding{true, kX_SkAxisAlignment};
|
2020-05-15 14:08:49 +00:00
|
|
|
drawable.startBitmapDevice(source, {100, 100}, matrix, rounding);
|
2019-11-14 18:51:43 +00:00
|
|
|
for (auto [i, packedID, pos] : SkMakeEnumerate(drawable.input())) {
|
2019-09-30 20:26:52 +00:00
|
|
|
REPORTER_ASSERT(reporter, glyphIDs[i] == packedID.packedID().glyphID());
|
2019-11-06 20:47:43 +00:00
|
|
|
REPORTER_ASSERT(reporter,
|
|
|
|
pos.x() == positions[i].x() * 0.5 + 50 + SkPackedGlyphID::kSubpixelRound);
|
2019-10-09 16:26:56 +00:00
|
|
|
REPORTER_ASSERT(reporter, pos.y() == positions[i].y() * 0.5 + 50 + 0.5);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
SkDrawableGlyphBuffer drawable;
|
|
|
|
drawable.ensureSize(100);
|
2020-05-14 16:26:25 +00:00
|
|
|
drawable.startSource(source);
|
2019-11-14 18:51:43 +00:00
|
|
|
for (auto [i, packedID, pos] : SkMakeEnumerate(drawable.input())) {
|
2019-10-09 16:26:56 +00:00
|
|
|
drawable.push_back(&glyphs[i], i);
|
|
|
|
}
|
2019-11-14 18:51:43 +00:00
|
|
|
for (auto [i, glyph, pos] : SkMakeEnumerate(drawable.drawable())) {
|
2019-10-09 16:26:56 +00:00
|
|
|
REPORTER_ASSERT(reporter, glyph.glyph() == &glyphs[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|