197 lines
7.3 KiB
C++
197 lines
7.3 KiB
C++
// Copyright 2019 Google LLC.
|
|
// Use of this source code is governed by a BSD-style license that can be found in the LICENSE file.
|
|
|
|
#include "tests/Test.h"
|
|
|
|
#if !defined(SK_BUILD_FOR_GOOGLE3)
|
|
|
|
#include "include/core/SkData.h"
|
|
#include "include/core/SkFont.h"
|
|
#include "include/core/SkPoint.h"
|
|
#include "include/core/SkRefCnt.h"
|
|
#include "include/core/SkSpan.h"
|
|
#include "include/core/SkStream.h"
|
|
#include "include/core/SkTypeface.h"
|
|
#include "include/core/SkTypes.h"
|
|
#include "include/private/base/SkTo.h"
|
|
#include "modules/skshaper/include/SkShaper.h"
|
|
#include "src/base/SkZip.h"
|
|
#include "tools/Resources.h"
|
|
|
|
#include <cinttypes>
|
|
#include <cstdint>
|
|
#include <memory>
|
|
|
|
namespace {
|
|
struct RunHandler final : public SkShaper::RunHandler {
|
|
const char* fResource;
|
|
skiatest::Reporter* fReporter;
|
|
const char* fUtf8;
|
|
size_t fUtf8Size;
|
|
std::unique_ptr<SkGlyphID[]> fGlyphs;
|
|
std::unique_ptr<SkPoint[]> fPositions;
|
|
std::unique_ptr<uint32_t[]> fClusters;
|
|
SkShaper::RunHandler::Range fRange;
|
|
unsigned fGlyphCount = 0;
|
|
|
|
bool fBeginLine = false;
|
|
bool fCommitRunInfo = false;
|
|
bool fCommitLine = false;
|
|
|
|
RunHandler(const char* resource, skiatest::Reporter* reporter, const char* utf8,size_t utf8Size)
|
|
: fResource(resource), fReporter(reporter), fUtf8(utf8), fUtf8Size(utf8Size) {}
|
|
|
|
void beginLine() override { fBeginLine = true;}
|
|
void runInfo(const SkShaper::RunHandler::RunInfo& info) override {}
|
|
void commitRunInfo() override { fCommitRunInfo = true; }
|
|
SkShaper::RunHandler::Buffer runBuffer(const SkShaper::RunHandler::RunInfo& info) override {
|
|
fGlyphCount = SkToUInt(info.glyphCount);
|
|
fRange = info.utf8Range;
|
|
fGlyphs = std::make_unique<SkGlyphID[]>(info.glyphCount);
|
|
fPositions = std::make_unique<SkPoint[]>(info.glyphCount);
|
|
fClusters = std::make_unique<uint32_t[]>(info.glyphCount);
|
|
return SkShaper::RunHandler::Buffer{fGlyphs.get(),
|
|
fPositions.get(),
|
|
nullptr,
|
|
fClusters.get(),
|
|
{0, 0}};
|
|
}
|
|
void commitRunBuffer(const RunInfo& info) override {
|
|
REPORTER_ASSERT(fReporter, fGlyphCount == info.glyphCount, "%s", fResource);
|
|
REPORTER_ASSERT(fReporter, fRange.begin() == info.utf8Range.begin(), "%s", fResource);
|
|
REPORTER_ASSERT(fReporter, fRange.size() == info.utf8Range.size(), "%s", fResource);
|
|
if (!(fRange.begin() + fRange.size() <= fUtf8Size)) {
|
|
REPORTER_ASSERT(fReporter, fRange.begin() + fRange.size() <= fUtf8Size, "%s",fResource);
|
|
return;
|
|
}
|
|
|
|
if ((false)) {
|
|
SkString familyName;
|
|
SkString postscriptName;
|
|
SkTypeface* typeface = info.fFont.getTypeface();
|
|
int ttcIndex = 0;
|
|
size_t fontSize = 0;
|
|
if (typeface) {
|
|
typeface->getFamilyName(&familyName);
|
|
typeface->getPostScriptName(&postscriptName);
|
|
std::unique_ptr<SkStreamAsset> stream = typeface->openStream(&ttcIndex);
|
|
if (stream) {
|
|
fontSize = stream->getLength();
|
|
}
|
|
}
|
|
SkString glyphs;
|
|
for (auto&& [glyph, cluster] : SkZip(info.glyphCount, fGlyphs.get(), fClusters.get())) {
|
|
glyphs.appendU32(glyph);
|
|
glyphs.append(":");
|
|
glyphs.appendU32(cluster);
|
|
glyphs.append(" ");
|
|
}
|
|
SkString chars;
|
|
for (const char c : SkSpan(fUtf8 + fRange.begin(), fRange.size())) {
|
|
chars.appendHex((unsigned char)c, 2);
|
|
chars.append(" ");
|
|
}
|
|
SkDebugf(
|
|
"%s range: %zu-%zu(%zu) glyphCount:%u font: \"%s\" \"%s\" #%d %zuB\n"
|
|
"rangeText: \"%.*s\"\n"
|
|
"rangeBytes: %s\n"
|
|
"glyphs:%s\n\n",
|
|
fResource, fRange.begin(), fRange.end(), fRange.size(), fGlyphCount,
|
|
familyName.c_str(), postscriptName.c_str(), ttcIndex, fontSize,
|
|
(int)fRange.size(), fUtf8 + fRange.begin(),
|
|
chars.c_str(),
|
|
glyphs.c_str());
|
|
}
|
|
|
|
for (unsigned i = 0; i < fGlyphCount; ++i) {
|
|
REPORTER_ASSERT(fReporter, fClusters[i] >= fRange.begin(),
|
|
"%" PRIu32 " >= %zu %s i:%u glyphCount:%u",
|
|
fClusters[i], fRange.begin(), fResource, i, fGlyphCount);
|
|
REPORTER_ASSERT(fReporter, fClusters[i] < fRange.end(),
|
|
"%" PRIu32 " < %zu %s i:%u glyphCount:%u",
|
|
fClusters[i], fRange.end(), fResource, i, fGlyphCount);
|
|
}
|
|
}
|
|
void commitLine() override { fCommitLine = true; }
|
|
};
|
|
|
|
void shaper_test(skiatest::Reporter* reporter, const char* name, SkData* data) {
|
|
auto shaper = SkShaper::Make();
|
|
if (!shaper) {
|
|
ERRORF(reporter, "Could not create shaper.");
|
|
return;
|
|
}
|
|
|
|
constexpr float kWidth = 400;
|
|
SkFont font(SkTypeface::MakeDefault());
|
|
RunHandler rh(name, reporter, (const char*)data->data(), data->size());
|
|
shaper->shape((const char*)data->data(), data->size(), font, true, kWidth, &rh);
|
|
|
|
// Even on empty input, expect that the line is started, that the zero run infos are comitted,
|
|
// and the empty line is comitted. This allows the user to properly handle empy runs.
|
|
REPORTER_ASSERT(reporter, rh.fBeginLine);
|
|
REPORTER_ASSERT(reporter, rh.fCommitRunInfo);
|
|
REPORTER_ASSERT(reporter, rh.fCommitLine);
|
|
|
|
constexpr SkFourByteTag latn = SkSetFourByteTag('l','a','t','n');
|
|
auto fontIterator = SkShaper::TrivialFontRunIterator(font, data->size());
|
|
auto bidiIterator = SkShaper::TrivialBiDiRunIterator(0, data->size());
|
|
auto scriptIterator = SkShaper::TrivialScriptRunIterator(latn, data->size());
|
|
auto languageIterator = SkShaper::TrivialLanguageRunIterator("en-US", data->size());
|
|
shaper->shape((const char*)data->data(), data->size(),
|
|
fontIterator, bidiIterator, scriptIterator, languageIterator, kWidth, &rh);
|
|
}
|
|
|
|
void cluster_test(skiatest::Reporter* reporter, const char* resource) {
|
|
auto data = GetResourceAsData(resource);
|
|
if (!data) {
|
|
ERRORF(reporter, "Could not get resource %s.", resource);
|
|
return;
|
|
}
|
|
|
|
shaper_test(reporter, resource, data.get());
|
|
}
|
|
|
|
} // namespace
|
|
|
|
DEF_TEST(Shaper_cluster_empty, r) { shaper_test(r, "empty", SkData::MakeEmpty().get()); }
|
|
|
|
#define SHAPER_TEST(X) DEF_TEST(Shaper_cluster_ ## X, r) { cluster_test(r, "text/" #X ".txt"); }
|
|
SHAPER_TEST(arabic)
|
|
SHAPER_TEST(armenian)
|
|
SHAPER_TEST(balinese)
|
|
SHAPER_TEST(buginese)
|
|
SHAPER_TEST(cherokee)
|
|
SHAPER_TEST(cyrillic)
|
|
SHAPER_TEST(emoji)
|
|
SHAPER_TEST(english)
|
|
SHAPER_TEST(ethiopic)
|
|
SHAPER_TEST(greek)
|
|
SHAPER_TEST(hangul)
|
|
SHAPER_TEST(han_simplified)
|
|
SHAPER_TEST(han_traditional)
|
|
SHAPER_TEST(hebrew)
|
|
SHAPER_TEST(javanese)
|
|
SHAPER_TEST(kana)
|
|
SHAPER_TEST(lao)
|
|
SHAPER_TEST(mandaic)
|
|
SHAPER_TEST(newtailue)
|
|
SHAPER_TEST(nko)
|
|
SHAPER_TEST(sinhala)
|
|
SHAPER_TEST(sundanese)
|
|
SHAPER_TEST(syriac)
|
|
SHAPER_TEST(thaana)
|
|
SHAPER_TEST(thai)
|
|
SHAPER_TEST(tibetan)
|
|
SHAPER_TEST(tifnagh)
|
|
SHAPER_TEST(vai)
|
|
SHAPER_TEST(bengali)
|
|
SHAPER_TEST(devanagari)
|
|
SHAPER_TEST(khmer)
|
|
SHAPER_TEST(myanmar)
|
|
SHAPER_TEST(taitham)
|
|
SHAPER_TEST(tamil)
|
|
#undef SHAPER_TEST
|
|
|
|
#endif // defined(SKSHAPER_IMPLEMENTATION) && !defined(SK_BUILD_FOR_GOOGLE3)
|