2019-04-30 16:38:28 +00:00
|
|
|
// Copyright 2019 Google LLC.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be found in the LICENSE file.
|
|
|
|
|
2019-05-14 19:01:39 +00:00
|
|
|
#include "experimental/editor/run_handler.h"
|
2019-04-30 16:38:28 +00:00
|
|
|
|
2019-04-04 22:00:05 +00:00
|
|
|
#include "include/core/SkFont.h"
|
2019-04-30 16:38:28 +00:00
|
|
|
#include "include/core/SkFontMetrics.h"
|
2019-04-04 22:00:05 +00:00
|
|
|
#include "include/core/SkString.h"
|
|
|
|
#include "include/private/SkTFitsIn.h"
|
2019-05-14 19:01:39 +00:00
|
|
|
#include "modules/skshaper/include/SkShaper.h"
|
2019-04-30 16:38:28 +00:00
|
|
|
#include "src/core/SkTextBlobPriv.h"
|
|
|
|
|
2019-04-04 22:00:05 +00:00
|
|
|
#include <limits.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2019-04-30 16:38:28 +00:00
|
|
|
using namespace editor;
|
|
|
|
|
|
|
|
void RunHandler::beginLine() {
|
|
|
|
fCurrentPosition = fOffset;
|
|
|
|
fMaxRunAscent = 0;
|
|
|
|
fMaxRunDescent = 0;
|
|
|
|
fMaxRunLeading = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RunHandler::runInfo(const SkShaper::RunHandler::RunInfo& info) {
|
|
|
|
SkFontMetrics metrics;
|
|
|
|
info.fFont.getMetrics(&metrics);
|
|
|
|
fMaxRunAscent = SkTMin(fMaxRunAscent, metrics.fAscent);
|
|
|
|
fMaxRunDescent = SkTMax(fMaxRunDescent, metrics.fDescent);
|
|
|
|
fMaxRunLeading = SkTMax(fMaxRunLeading, metrics.fLeading);
|
|
|
|
}
|
|
|
|
|
|
|
|
void RunHandler::commitRunInfo() {
|
|
|
|
fCurrentPosition.fY -= fMaxRunAscent;
|
|
|
|
}
|
|
|
|
|
|
|
|
SkShaper::RunHandler::Buffer RunHandler::runBuffer(const RunInfo& info) {
|
|
|
|
int glyphCount = SkTFitsIn<int>(info.glyphCount) ? info.glyphCount : INT_MAX;
|
|
|
|
int utf8RangeSize = SkTFitsIn<int>(info.utf8Range.size()) ? info.utf8Range.size() : INT_MAX;
|
|
|
|
|
|
|
|
const auto& runBuffer = SkTextBlobBuilderPriv::AllocRunTextPos(&fBuilder, info.fFont, glyphCount,
|
|
|
|
utf8RangeSize, SkString());
|
|
|
|
fCurrentGlyphs = runBuffer.glyphs;
|
|
|
|
fCurrentPoints = runBuffer.points();
|
|
|
|
|
|
|
|
if (runBuffer.utf8text && fUtf8Text) {
|
|
|
|
memcpy(runBuffer.utf8text, fUtf8Text + info.utf8Range.begin(), utf8RangeSize);
|
|
|
|
}
|
|
|
|
fClusters = runBuffer.clusters;
|
|
|
|
fGlyphCount = glyphCount;
|
|
|
|
fClusterOffset = info.utf8Range.begin();
|
|
|
|
|
|
|
|
return {runBuffer.glyphs,
|
|
|
|
runBuffer.points(),
|
|
|
|
nullptr,
|
|
|
|
runBuffer.clusters,
|
|
|
|
fCurrentPosition};
|
|
|
|
}
|
|
|
|
|
|
|
|
void RunHandler::commitRunBuffer(const RunInfo& info) {
|
|
|
|
// for (size_t i = 0; i < info.glyphCount; ++i) {
|
|
|
|
// SkASSERT(fClusters[i] >= info.utf8Range.begin());
|
|
|
|
// // this fails for khmer example.
|
|
|
|
// SkASSERT(fClusters[i] < info.utf8Range.end());
|
|
|
|
// }
|
|
|
|
if (fCallbackFunction) {
|
|
|
|
fCallbackFunction(fCallbackContext,
|
|
|
|
fUtf8Text,
|
|
|
|
info.utf8Range.end(),
|
|
|
|
info.glyphCount,
|
|
|
|
fCurrentGlyphs,
|
|
|
|
fCurrentPoints,
|
|
|
|
fClusters,
|
|
|
|
info.fFont);
|
|
|
|
}
|
|
|
|
SkASSERT(0 <= fClusterOffset);
|
|
|
|
for (int i = 0; i < fGlyphCount; ++i) {
|
|
|
|
SkASSERT(fClusters[i] >= (unsigned)fClusterOffset);
|
|
|
|
fClusters[i] -= fClusterOffset;
|
|
|
|
}
|
|
|
|
fCurrentPosition += info.fAdvance;
|
2019-07-25 17:58:42 +00:00
|
|
|
fTextOffset = SkTMax(fTextOffset, SkToUInt(info.utf8Range.end()));
|
2019-04-30 16:38:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void RunHandler::commitLine() {
|
2019-07-25 17:58:42 +00:00
|
|
|
if (fLineEndOffsets.empty() || fTextOffset > fLineEndOffsets.back()) {
|
|
|
|
// Ensure that fLineEndOffsets is monotonic.
|
|
|
|
fLineEndOffsets.push_back(fTextOffset);
|
|
|
|
}
|
2019-04-30 16:38:28 +00:00
|
|
|
fOffset += { 0, fMaxRunDescent + fMaxRunLeading - fMaxRunAscent };
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<SkTextBlob> RunHandler::makeBlob() {
|
|
|
|
return fBuilder.make();
|
|
|
|
}
|
2019-07-25 17:58:42 +00:00
|
|
|
|
|
|
|
|