Add empty check for mask case. Small cleanups elsewhere.
We were overly cautious with spaces close to a cut. Re-enable empty glyph removal from the Renderer' diff canvas processing. * rename prepareForDrawing -> prepareForDrawingRemoveEmpty Bug: chromium:881505 Change-Id: I9b5d51cd66645bccfc3b1f6fefb317f9eaf621e7 Reviewed-on: https://skia-review.googlesource.com/c/skia/+/233417 Commit-Queue: Herb Derby <herb@google.com> Reviewed-by: Robert Phillips <robertphillips@google.com>
This commit is contained in:
parent
c86cf3cdb3
commit
1436a13881
@ -188,7 +188,8 @@ void SkGlyphRunListPainter::drawForBitmapDevice(
|
|||||||
fPositions,
|
fPositions,
|
||||||
fPackedGlyphIDs);
|
fPackedGlyphIDs);
|
||||||
|
|
||||||
auto glyphPosSpan = strike->prepareForDrawing(packedGlyphIDs.data(),
|
auto glyphPosSpan = strike->prepareForDrawingRemoveEmpty(
|
||||||
|
packedGlyphIDs.data(),
|
||||||
fPositions,
|
fPositions,
|
||||||
glyphRun.runSize(),
|
glyphRun.runSize(),
|
||||||
0,
|
0,
|
||||||
@ -241,7 +242,7 @@ void SkGlyphRunListPainter::drawForBitmapDevice(
|
|||||||
fPositions,
|
fPositions,
|
||||||
fPackedGlyphIDs);
|
fPackedGlyphIDs);
|
||||||
|
|
||||||
SkSpan<const SkGlyphPos> glyphPosSpan = strike->prepareForDrawing(
|
SkSpan<const SkGlyphPos> glyphPosSpan = strike->prepareForDrawingRemoveEmpty(
|
||||||
packedGlyphIDs.data(),
|
packedGlyphIDs.data(),
|
||||||
fPositions,
|
fPositions,
|
||||||
glyphRun.runSize(),
|
glyphRun.runSize(),
|
||||||
@ -333,7 +334,7 @@ void SkGlyphRunListPainter::processARGBFallback(SkScalar maxSourceGlyphDimension
|
|||||||
*cursor++ = SkPackedGlyphID{glyphID};
|
*cursor++ = SkPackedGlyphID{glyphID};
|
||||||
}
|
}
|
||||||
|
|
||||||
SkSpan<const SkGlyphPos> glyphPosSpan = strike->prepareForDrawing(
|
SkSpan<const SkGlyphPos> glyphPosSpan = strike->prepareForDrawingRemoveEmpty(
|
||||||
fPackedGlyphIDs,
|
fPackedGlyphIDs,
|
||||||
fARGBPositions.data(),
|
fARGBPositions.data(),
|
||||||
fARGBGlyphsIDs.size(),
|
fARGBGlyphsIDs.size(),
|
||||||
@ -359,7 +360,8 @@ void SkGlyphRunListPainter::processARGBFallback(SkScalar maxSourceGlyphDimension
|
|||||||
*cursor++ = SkPackedGlyphID{glyphID};
|
*cursor++ = SkPackedGlyphID{glyphID};
|
||||||
}
|
}
|
||||||
|
|
||||||
auto glyphPosSpan = strike->prepareForDrawing(fPackedGlyphIDs,
|
auto glyphPosSpan = strike->prepareForDrawingRemoveEmpty(
|
||||||
|
fPackedGlyphIDs,
|
||||||
fARGBPositions.data(),
|
fARGBPositions.data(),
|
||||||
fARGBGlyphsIDs.size(),
|
fARGBGlyphsIDs.size(),
|
||||||
SkStrikeCommon::kSkSideTooBigForAtlas,
|
SkStrikeCommon::kSkSideTooBigForAtlas,
|
||||||
@ -423,7 +425,7 @@ void SkGlyphRunListPainter::processGlyphRunList(const SkGlyphRunList& glyphRunLi
|
|||||||
fPositions,
|
fPositions,
|
||||||
fPackedGlyphIDs);
|
fPackedGlyphIDs);
|
||||||
|
|
||||||
SkSpan<const SkGlyphPos> glyphPosSpan = strike->prepareForDrawing(
|
SkSpan<const SkGlyphPos> glyphPosSpan = strike->prepareForDrawingRemoveEmpty(
|
||||||
packedGlyphIDs.data(),
|
packedGlyphIDs.data(),
|
||||||
fPositions,
|
fPositions,
|
||||||
glyphRun.runSize(),
|
glyphRun.runSize(),
|
||||||
@ -438,18 +440,13 @@ void SkGlyphRunListPainter::processGlyphRunList(const SkGlyphRunList& glyphRunLi
|
|||||||
|
|
||||||
// The SDF scaler context system ensures that a glyph is empty, kSDF_Format, or
|
// The SDF scaler context system ensures that a glyph is empty, kSDF_Format, or
|
||||||
// kARGB32_Format. The following if statements use this assumption.
|
// kARGB32_Format. The following if statements use this assumption.
|
||||||
SkASSERT(glyph.isEmpty()
|
SkASSERT(glyph.maskFormat() == SkMask::kSDF_Format || glyph.isColor());
|
||||||
|| glyph.maskFormat() == SkMask::kSDF_Format
|
|
||||||
|| glyph.isColor());
|
|
||||||
|
|
||||||
if (glyph.isEmpty()) {
|
if (glyph.maskFormat() == SkMask::kSDF_Format
|
||||||
// do nothing
|
|
||||||
} else if (glyph.maskFormat() == SkMask::kSDF_Format
|
|
||||||
&& glyph.maxDimension() <= SkStrikeCommon::kSkSideTooBigForAtlas) {
|
&& glyph.maxDimension() <= SkStrikeCommon::kSkSideTooBigForAtlas) {
|
||||||
// SDF mask will work.
|
// SDF mask will work.
|
||||||
fGlyphPos[glyphsWithMaskCount++] = glyphPos;
|
fGlyphPos[glyphsWithMaskCount++] = glyphPos;
|
||||||
} else if (!glyph.isColor()
|
} else if (!glyph.isColor() && glyph.path() != nullptr) {
|
||||||
&& glyph.path() != nullptr) {
|
|
||||||
// If not color but too big, use a path.
|
// If not color but too big, use a path.
|
||||||
fPaths.push_back(glyphPos);
|
fPaths.push_back(glyphPos);
|
||||||
} else {
|
} else {
|
||||||
@ -485,7 +482,6 @@ void SkGlyphRunListPainter::processGlyphRunList(const SkGlyphRunList& glyphRunLi
|
|||||||
runPaint, runFont, viewMatrix, process);
|
runPaint, runFont, viewMatrix, process);
|
||||||
}
|
}
|
||||||
} else if (SkStrikeSpec::ShouldDrawAsPath(runPaint, runFont, viewMatrix)) {
|
} else if (SkStrikeSpec::ShouldDrawAsPath(runPaint, runFont, viewMatrix)) {
|
||||||
|
|
||||||
SkStrikeSpec strikeSpec = SkStrikeSpec::MakePath(
|
SkStrikeSpec strikeSpec = SkStrikeSpec::MakePath(
|
||||||
runFont, runPaint, fDeviceProps, fScalerContextFlags);
|
runFont, runPaint, fDeviceProps, fScalerContextFlags);
|
||||||
|
|
||||||
@ -499,7 +495,7 @@ void SkGlyphRunListPainter::processGlyphRunList(const SkGlyphRunList& glyphRunLi
|
|||||||
fPositions,
|
fPositions,
|
||||||
fPackedGlyphIDs);
|
fPackedGlyphIDs);
|
||||||
|
|
||||||
SkSpan<const SkGlyphPos> glyphPosSpan = strike->prepareForDrawing(
|
SkSpan<const SkGlyphPos> glyphPosSpan = strike->prepareForDrawingRemoveEmpty(
|
||||||
packedGlyphIDs.data(),
|
packedGlyphIDs.data(),
|
||||||
fPositions,
|
fPositions,
|
||||||
glyphRun.runSize(),
|
glyphRun.runSize(),
|
||||||
@ -512,10 +508,7 @@ void SkGlyphRunListPainter::processGlyphRunList(const SkGlyphRunList& glyphRunLi
|
|||||||
for (const SkGlyphPos& glyphPos : glyphPosSpan) {
|
for (const SkGlyphPos& glyphPos : glyphPosSpan) {
|
||||||
const SkGlyph& glyph = *glyphPos.glyph;
|
const SkGlyph& glyph = *glyphPos.glyph;
|
||||||
SkPoint position = glyphPos.position;
|
SkPoint position = glyphPos.position;
|
||||||
if (glyph.isEmpty()) {
|
if (!glyph.isColor() && glyph.path() != nullptr) {
|
||||||
// do nothing
|
|
||||||
} else if (!glyph.isColor()
|
|
||||||
&& glyph.path() != nullptr) {
|
|
||||||
// Place paths in fGlyphPos
|
// Place paths in fGlyphPos
|
||||||
fGlyphPos[glyphsWithPathCount++] = glyphPos;
|
fGlyphPos[glyphsWithPathCount++] = glyphPos;
|
||||||
} else {
|
} else {
|
||||||
@ -537,7 +530,6 @@ void SkGlyphRunListPainter::processGlyphRunList(const SkGlyphRunList& glyphRunLi
|
|||||||
runPaint, runFont, viewMatrix, process);
|
runPaint, runFont, viewMatrix, process);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
SkStrikeSpec strikeSpec =
|
SkStrikeSpec strikeSpec =
|
||||||
SkStrikeSpec::MakeMask(runFont, runPaint,
|
SkStrikeSpec::MakeMask(runFont, runPaint,
|
||||||
fDeviceProps, fScalerContextFlags, viewMatrix);
|
fDeviceProps, fScalerContextFlags, viewMatrix);
|
||||||
@ -554,8 +546,8 @@ void SkGlyphRunListPainter::processGlyphRunList(const SkGlyphRunList& glyphRunLi
|
|||||||
fPositions,
|
fPositions,
|
||||||
fPackedGlyphIDs);
|
fPackedGlyphIDs);
|
||||||
|
|
||||||
// Lookup all the glyphs from the cache.
|
// Lookup all the glyphs from the cache. Strip empty glyphs.
|
||||||
SkSpan<const SkGlyphPos> glyphPosSpan = strike->prepareForDrawing(
|
SkSpan<const SkGlyphPos> glyphPosSpan = strike->prepareForDrawingRemoveEmpty(
|
||||||
packedGlyphIDs.data(),
|
packedGlyphIDs.data(),
|
||||||
fPositions,
|
fPositions,
|
||||||
glyphRun.runSize(),
|
glyphRun.runSize(),
|
||||||
@ -569,15 +561,12 @@ void SkGlyphRunListPainter::processGlyphRunList(const SkGlyphRunList& glyphRunLi
|
|||||||
const SkGlyph& glyph = *glyphPos.glyph;
|
const SkGlyph& glyph = *glyphPos.glyph;
|
||||||
const SkPoint position = glyphPos.position;
|
const SkPoint position = glyphPos.position;
|
||||||
|
|
||||||
// Able to position glyph?
|
// Does the glyph have work to do or is the code able to position the glyph?
|
||||||
if (!SkScalarsAreFinite(position.x(), position.y())) {
|
if (!SkScalarsAreFinite(position.x(), position.y())) {
|
||||||
continue;
|
// Do nothing;
|
||||||
}
|
} else if (glyph.maxDimension() <= SkStrikeCommon::kSkSideTooBigForAtlas) {
|
||||||
|
|
||||||
if (glyph.maxDimension() <= SkStrikeCommon::kSkSideTooBigForAtlas) {
|
|
||||||
fGlyphPos[glyphsWithMaskCount++] = glyphPos;
|
fGlyphPos[glyphsWithMaskCount++] = glyphPos;
|
||||||
} else if (!glyph.isColor()
|
} else if (!glyph.isColor() && glyph.path() != nullptr) {
|
||||||
&& glyph.path() != nullptr) {
|
|
||||||
fPaths.push_back(glyphPos);
|
fPaths.push_back(glyphPos);
|
||||||
} else {
|
} else {
|
||||||
addFallback(glyph, origin + glyphRun.positions()[glyphPos.index]);
|
addFallback(glyph, origin + glyphRun.positions()[glyphPos.index]);
|
||||||
|
@ -547,14 +547,18 @@ void SkStrikeServer::SkGlyphCacheState::writeGlyphPath(const SkPackedGlyphID& gl
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Be sure to read and understand the comment for prepareForDrawing in SkStrikeInterface.h before
|
// Be sure to read and understand the comment for prepareForDrawingRemoveEmpty in
|
||||||
// working on this code.
|
// SkStrikeInterface.h before working on this code.
|
||||||
SkSpan<const SkGlyphPos>
|
SkSpan<const SkGlyphPos>
|
||||||
SkStrikeServer::SkGlyphCacheState::prepareForDrawing(const SkPackedGlyphID packedGlyphIDs[],
|
SkStrikeServer::SkGlyphCacheState::prepareForDrawingRemoveEmpty(
|
||||||
const SkPoint positions[], size_t n,
|
const SkPackedGlyphID packedGlyphIDs[],
|
||||||
int maxDimension, PreparationDetail detail,
|
const SkPoint positions[],
|
||||||
|
size_t n,
|
||||||
|
int maxDimension,
|
||||||
|
PreparationDetail detail,
|
||||||
SkGlyphPos results[]) {
|
SkGlyphPos results[]) {
|
||||||
|
|
||||||
|
size_t drawableGlyphCount = 0;
|
||||||
for (size_t i = 0; i < n; i++) {
|
for (size_t i = 0; i < n; i++) {
|
||||||
SkPoint glyphPos = positions[i];
|
SkPoint glyphPos = positions[i];
|
||||||
|
|
||||||
@ -573,9 +577,8 @@ SkStrikeServer::SkGlyphCacheState::prepareForDrawing(const SkPackedGlyphID packe
|
|||||||
if (glyphPtr->maxDimension() <= maxDimension) {
|
if (glyphPtr->maxDimension() <= maxDimension) {
|
||||||
// do nothing
|
// do nothing
|
||||||
} else if (!glyphPtr->isColor()) {
|
} else if (!glyphPtr->isColor()) {
|
||||||
|
// The glyph is too big for the atlas, but it is not color, so it is handled
|
||||||
// The glyph is too big for the atlas, but it is not color, so it is handled with a
|
// with a path.
|
||||||
// path.
|
|
||||||
if (glyphPtr->setPath(&fAlloc, fContext.get())) {
|
if (glyphPtr->setPath(&fAlloc, fContext.get())) {
|
||||||
// Always send the path data, even if its not available, to make sure empty
|
// Always send the path data, even if its not available, to make sure empty
|
||||||
// paths are not incorrectly assumed to be cache misses.
|
// paths are not incorrectly assumed to be cache misses.
|
||||||
@ -592,12 +595,14 @@ SkStrikeServer::SkGlyphCacheState::prepareForDrawing(const SkPackedGlyphID packe
|
|||||||
fPendingGlyphImages.push_back(packedGlyphIDs[i]);
|
fPendingGlyphImages.push_back(packedGlyphIDs[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Each glyph needs to be added as per the contract for prepareForDrawing.
|
// Each non-empty glyph needs to be added as per the contract for
|
||||||
// TODO(herb): check if the empty glyphs need to be added here. They certainly need to be
|
// prepareForDrawingRemoveEmpty.
|
||||||
// sent, but do the need to be processed by the painter?
|
// TODO(herb): Change the code to only send the glyphs for fallback?
|
||||||
results[i] = {i, glyphPtr, glyphPos};
|
if (!glyphPtr->isEmpty()) {
|
||||||
|
results[drawableGlyphCount++] = {i, glyphPtr, glyphPos};
|
||||||
}
|
}
|
||||||
return SkSpan<const SkGlyphPos>{results, n};
|
}
|
||||||
|
return SkMakeSpan(results, drawableGlyphCount);
|
||||||
}
|
}
|
||||||
|
|
||||||
// SkStrikeClient -----------------------------------------
|
// SkStrikeClient -----------------------------------------
|
||||||
|
@ -42,8 +42,12 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
SkSpan<const SkGlyphPos>
|
SkSpan<const SkGlyphPos>
|
||||||
prepareForDrawing(const SkPackedGlyphID packedGlyphIDs[], const SkPoint positions[], size_t n,
|
prepareForDrawingRemoveEmpty(const SkPackedGlyphID packedGlyphIDs[],
|
||||||
int maxDimension, PreparationDetail detail, SkGlyphPos results[]) override;
|
const SkPoint positions[],
|
||||||
|
size_t n,
|
||||||
|
int maxDimension,
|
||||||
|
PreparationDetail detail,
|
||||||
|
SkGlyphPos results[]) override;
|
||||||
|
|
||||||
void onAboutToExitScope() override {}
|
void onAboutToExitScope() override {}
|
||||||
|
|
||||||
|
@ -189,10 +189,12 @@ SkStrike::prepareImages(SkSpan<const SkPackedGlyphID> glyphIDs, const SkGlyph* r
|
|||||||
// N.B. This glyphMetrics call culls all the glyphs which will not display based on a non-finite
|
// N.B. This glyphMetrics call culls all the glyphs which will not display based on a non-finite
|
||||||
// position or that there are no mask pixels.
|
// position or that there are no mask pixels.
|
||||||
SkSpan<const SkGlyphPos>
|
SkSpan<const SkGlyphPos>
|
||||||
SkStrike::prepareForDrawing(const SkPackedGlyphID packedGlyphIDs[], const SkPoint positions[],
|
SkStrike::prepareForDrawingRemoveEmpty(const SkPackedGlyphID packedGlyphIDs[],
|
||||||
|
const SkPoint positions[],
|
||||||
size_t n,
|
size_t n,
|
||||||
int maxDimension, PreparationDetail detail, SkGlyphPos results[]) {
|
int maxDimension,
|
||||||
|
PreparationDetail detail,
|
||||||
|
SkGlyphPos results[]) {
|
||||||
size_t drawableGlyphCount = 0;
|
size_t drawableGlyphCount = 0;
|
||||||
for (size_t i = 0; i < n; i++) {
|
for (size_t i = 0; i < n; i++) {
|
||||||
SkPoint pos = positions[i];
|
SkPoint pos = positions[i];
|
||||||
|
@ -119,7 +119,7 @@ public:
|
|||||||
SkSpan<const SkGlyph*> prepareImages(SkSpan<const SkPackedGlyphID> glyphIDs,
|
SkSpan<const SkGlyph*> prepareImages(SkSpan<const SkPackedGlyphID> glyphIDs,
|
||||||
const SkGlyph* results[]);
|
const SkGlyph* results[]);
|
||||||
|
|
||||||
SkSpan<const SkGlyphPos> prepareForDrawing(const SkPackedGlyphID packedGlyphIDs[],
|
SkSpan<const SkGlyphPos> prepareForDrawingRemoveEmpty(const SkPackedGlyphID packedGlyphIDs[],
|
||||||
const SkPoint positions[],
|
const SkPoint positions[],
|
||||||
size_t n,
|
size_t n,
|
||||||
int maxDimension,
|
int maxDimension,
|
||||||
|
@ -37,9 +37,17 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
SkSpan<const SkGlyphPos>
|
SkSpan<const SkGlyphPos>
|
||||||
prepareForDrawing(const SkPackedGlyphID packedGlyphIDs[], const SkPoint positions[], size_t n,
|
prepareForDrawingRemoveEmpty(const SkPackedGlyphID packedGlyphIDs[],
|
||||||
int maxDimension, PreparationDetail detail, SkGlyphPos results[]) override {
|
const SkPoint positions[],
|
||||||
return fStrike.prepareForDrawing(packedGlyphIDs, positions, n, maxDimension, detail,
|
size_t n,
|
||||||
|
int maxDimension,
|
||||||
|
PreparationDetail detail,
|
||||||
|
SkGlyphPos results[]) override {
|
||||||
|
return fStrike.prepareForDrawingRemoveEmpty(packedGlyphIDs,
|
||||||
|
positions,
|
||||||
|
n,
|
||||||
|
maxDimension,
|
||||||
|
detail,
|
||||||
results);
|
results);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -56,15 +56,16 @@ public:
|
|||||||
kImageIfNeeded,
|
kImageIfNeeded,
|
||||||
};
|
};
|
||||||
|
|
||||||
// prepareForDrawing takes glyphIDs, and position, and returns a list of SkGlyphs and
|
// prepareForDrawingRemoveEmpty takes glyphIDs, and position, and returns a list of SkGlyphs
|
||||||
// positions where all the data to draw the glyph has been created. The maxDimension
|
// and positions where all the data to draw the glyph has been created. The maxDimension
|
||||||
// parameter determines if the mask/SDF version will be created, or an alternate drawing
|
// parameter determines if the mask/SDF version will be created, or an alternate drawing
|
||||||
// format should be used. For path-only drawing set maxDimension to 0, and for bitmap-device
|
// format should be used. For path-only drawing set maxDimension to 0, and for bitmap-device
|
||||||
// drawing (where there is no upper limit to the glyph in the cache) use INT_MAX.
|
// drawing (where there is no upper limit to the glyph in the cache) use INT_MAX.
|
||||||
// * PreparationDetail determines, in the mask case, if the mask/SDF should be generated.
|
// * PreparationDetail determines, in the mask case, if the mask/SDF should be generated.
|
||||||
// This does not affect the path or fallback cases.
|
// This does not affect the path or fallback cases.
|
||||||
|
// prepareForDrawingRemoveEmpty should remove all empty glyphs from the returned span.
|
||||||
virtual SkSpan<const SkGlyphPos>
|
virtual SkSpan<const SkGlyphPos>
|
||||||
prepareForDrawing(const SkPackedGlyphID packedGlyphIDs[],
|
prepareForDrawingRemoveEmpty(const SkPackedGlyphID packedGlyphIDs[],
|
||||||
const SkPoint positions[],
|
const SkPoint positions[],
|
||||||
size_t n,
|
size_t n,
|
||||||
int maxDimension,
|
int maxDimension,
|
||||||
@ -100,5 +101,4 @@ public:
|
|||||||
const SkScalerContextEffects& effects,
|
const SkScalerContextEffects& effects,
|
||||||
const SkTypeface& typeface) = 0;
|
const SkTypeface& typeface) = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif //SkStrikeInterface_DEFINED
|
#endif //SkStrikeInterface_DEFINED
|
||||||
|
Loading…
Reference in New Issue
Block a user