1cb93a386Sopenharmony_ci// Copyright 2019 Google LLC.
2cb93a386Sopenharmony_ci// Use of this source code is governed by a BSD-style license that can be found in the LICENSE file.
3cb93a386Sopenharmony_ci
4cb93a386Sopenharmony_ci#include "tests/Test.h"
5cb93a386Sopenharmony_ci
6cb93a386Sopenharmony_ci#if !defined(SK_BUILD_FOR_GOOGLE3)
7cb93a386Sopenharmony_ci
8cb93a386Sopenharmony_ci#include "include/core/SkData.h"
9cb93a386Sopenharmony_ci#include "include/core/SkFont.h"
10cb93a386Sopenharmony_ci#include "include/core/SkPoint.h"
11cb93a386Sopenharmony_ci#include "include/core/SkRefCnt.h"
12cb93a386Sopenharmony_ci#include "include/core/SkSpan.h"
13cb93a386Sopenharmony_ci#include "include/core/SkStream.h"
14cb93a386Sopenharmony_ci#include "include/core/SkTypeface.h"
15cb93a386Sopenharmony_ci#include "include/core/SkTypes.h"
16cb93a386Sopenharmony_ci#include "include/private/base/SkTo.h"
17cb93a386Sopenharmony_ci#include "modules/skshaper/include/SkShaper.h"
18cb93a386Sopenharmony_ci#include "src/base/SkZip.h"
19cb93a386Sopenharmony_ci#include "tools/Resources.h"
20cb93a386Sopenharmony_ci
21cb93a386Sopenharmony_ci#include <cinttypes>
22cb93a386Sopenharmony_ci#include <cstdint>
23cb93a386Sopenharmony_ci#include <memory>
24cb93a386Sopenharmony_ci
25cb93a386Sopenharmony_cinamespace {
26cb93a386Sopenharmony_cistruct RunHandler final : public SkShaper::RunHandler {
27cb93a386Sopenharmony_ci    const char* fResource;
28cb93a386Sopenharmony_ci    skiatest::Reporter* fReporter;
29cb93a386Sopenharmony_ci    const char* fUtf8;
30cb93a386Sopenharmony_ci    size_t fUtf8Size;
31cb93a386Sopenharmony_ci    std::unique_ptr<SkGlyphID[]> fGlyphs;
32cb93a386Sopenharmony_ci    std::unique_ptr<SkPoint[]> fPositions;
33cb93a386Sopenharmony_ci    std::unique_ptr<uint32_t[]> fClusters;
34cb93a386Sopenharmony_ci    SkShaper::RunHandler::Range fRange;
35cb93a386Sopenharmony_ci    unsigned fGlyphCount = 0;
36cb93a386Sopenharmony_ci
37cb93a386Sopenharmony_ci    bool fBeginLine = false;
38cb93a386Sopenharmony_ci    bool fCommitRunInfo = false;
39cb93a386Sopenharmony_ci    bool fCommitLine = false;
40cb93a386Sopenharmony_ci
41cb93a386Sopenharmony_ci    RunHandler(const char* resource, skiatest::Reporter* reporter, const char* utf8,size_t utf8Size)
42cb93a386Sopenharmony_ci        : fResource(resource), fReporter(reporter), fUtf8(utf8), fUtf8Size(utf8Size) {}
43cb93a386Sopenharmony_ci
44cb93a386Sopenharmony_ci    void beginLine() override { fBeginLine = true;}
45cb93a386Sopenharmony_ci    void runInfo(const SkShaper::RunHandler::RunInfo& info) override {}
46cb93a386Sopenharmony_ci    void commitRunInfo() override { fCommitRunInfo = true; }
47cb93a386Sopenharmony_ci    SkShaper::RunHandler::Buffer runBuffer(const SkShaper::RunHandler::RunInfo& info) override {
48cb93a386Sopenharmony_ci        fGlyphCount = SkToUInt(info.glyphCount);
49cb93a386Sopenharmony_ci        fRange = info.utf8Range;
50cb93a386Sopenharmony_ci        fGlyphs = std::make_unique<SkGlyphID[]>(info.glyphCount);
51cb93a386Sopenharmony_ci        fPositions = std::make_unique<SkPoint[]>(info.glyphCount);
52cb93a386Sopenharmony_ci        fClusters = std::make_unique<uint32_t[]>(info.glyphCount);
53cb93a386Sopenharmony_ci        return SkShaper::RunHandler::Buffer{fGlyphs.get(),
54cb93a386Sopenharmony_ci                                            fPositions.get(),
55cb93a386Sopenharmony_ci                                            nullptr,
56cb93a386Sopenharmony_ci                                            fClusters.get(),
57cb93a386Sopenharmony_ci                                            {0, 0}};
58cb93a386Sopenharmony_ci    }
59cb93a386Sopenharmony_ci    void commitRunBuffer(const RunInfo& info) override {
60cb93a386Sopenharmony_ci        REPORTER_ASSERT(fReporter, fGlyphCount == info.glyphCount, "%s", fResource);
61cb93a386Sopenharmony_ci        REPORTER_ASSERT(fReporter, fRange.begin() == info.utf8Range.begin(), "%s", fResource);
62cb93a386Sopenharmony_ci        REPORTER_ASSERT(fReporter, fRange.size() == info.utf8Range.size(), "%s", fResource);
63cb93a386Sopenharmony_ci        if (!(fRange.begin() + fRange.size() <= fUtf8Size)) {
64cb93a386Sopenharmony_ci            REPORTER_ASSERT(fReporter, fRange.begin() + fRange.size() <= fUtf8Size, "%s",fResource);
65cb93a386Sopenharmony_ci            return;
66cb93a386Sopenharmony_ci        }
67cb93a386Sopenharmony_ci
68cb93a386Sopenharmony_ci        if ((false)) {
69cb93a386Sopenharmony_ci            SkString familyName;
70cb93a386Sopenharmony_ci            SkString postscriptName;
71cb93a386Sopenharmony_ci            SkTypeface* typeface = info.fFont.getTypeface();
72cb93a386Sopenharmony_ci            int ttcIndex = 0;
73cb93a386Sopenharmony_ci            size_t fontSize = 0;
74cb93a386Sopenharmony_ci            if (typeface) {
75cb93a386Sopenharmony_ci                typeface->getFamilyName(&familyName);
76cb93a386Sopenharmony_ci                typeface->getPostScriptName(&postscriptName);
77cb93a386Sopenharmony_ci                std::unique_ptr<SkStreamAsset> stream = typeface->openStream(&ttcIndex);
78cb93a386Sopenharmony_ci                if (stream) {
79cb93a386Sopenharmony_ci                    fontSize = stream->getLength();
80cb93a386Sopenharmony_ci                }
81cb93a386Sopenharmony_ci            }
82cb93a386Sopenharmony_ci            SkString glyphs;
83cb93a386Sopenharmony_ci            for (auto&& [glyph, cluster] : SkZip(info.glyphCount, fGlyphs.get(), fClusters.get())) {
84cb93a386Sopenharmony_ci                glyphs.appendU32(glyph);
85cb93a386Sopenharmony_ci                glyphs.append(":");
86cb93a386Sopenharmony_ci                glyphs.appendU32(cluster);
87cb93a386Sopenharmony_ci                glyphs.append(" ");
88cb93a386Sopenharmony_ci            }
89cb93a386Sopenharmony_ci            SkString chars;
90cb93a386Sopenharmony_ci            for (const char c : SkSpan(fUtf8 + fRange.begin(), fRange.size())) {
91cb93a386Sopenharmony_ci                chars.appendHex((unsigned char)c, 2);
92cb93a386Sopenharmony_ci                chars.append(" ");
93cb93a386Sopenharmony_ci            }
94cb93a386Sopenharmony_ci            SkDebugf(
95cb93a386Sopenharmony_ci                "%s range: %zu-%zu(%zu) glyphCount:%u font: \"%s\" \"%s\" #%d %zuB\n"
96cb93a386Sopenharmony_ci                "rangeText: \"%.*s\"\n"
97cb93a386Sopenharmony_ci                "rangeBytes: %s\n"
98cb93a386Sopenharmony_ci                "glyphs:%s\n\n",
99cb93a386Sopenharmony_ci                fResource, fRange.begin(), fRange.end(), fRange.size(), fGlyphCount,
100cb93a386Sopenharmony_ci                familyName.c_str(), postscriptName.c_str(), ttcIndex, fontSize,
101cb93a386Sopenharmony_ci                (int)fRange.size(), fUtf8 + fRange.begin(),
102cb93a386Sopenharmony_ci                chars.c_str(),
103cb93a386Sopenharmony_ci                glyphs.c_str());
104cb93a386Sopenharmony_ci        }
105cb93a386Sopenharmony_ci
106cb93a386Sopenharmony_ci        for (unsigned i = 0; i < fGlyphCount; ++i) {
107cb93a386Sopenharmony_ci            REPORTER_ASSERT(fReporter, fClusters[i] >= fRange.begin(),
108cb93a386Sopenharmony_ci                            "%" PRIu32 " >= %zu %s i:%u glyphCount:%u",
109cb93a386Sopenharmony_ci                            fClusters[i], fRange.begin(), fResource, i, fGlyphCount);
110cb93a386Sopenharmony_ci            REPORTER_ASSERT(fReporter, fClusters[i] < fRange.end(),
111cb93a386Sopenharmony_ci                            "%" PRIu32 " < %zu %s i:%u glyphCount:%u",
112cb93a386Sopenharmony_ci                            fClusters[i], fRange.end(), fResource, i, fGlyphCount);
113cb93a386Sopenharmony_ci        }
114cb93a386Sopenharmony_ci    }
115cb93a386Sopenharmony_ci    void commitLine() override { fCommitLine = true; }
116cb93a386Sopenharmony_ci};
117cb93a386Sopenharmony_ci
118cb93a386Sopenharmony_civoid shaper_test(skiatest::Reporter* reporter, const char* name, SkData* data) {
119cb93a386Sopenharmony_ci    auto shaper = SkShaper::Make();
120cb93a386Sopenharmony_ci    if (!shaper) {
121cb93a386Sopenharmony_ci        ERRORF(reporter, "Could not create shaper.");
122cb93a386Sopenharmony_ci        return;
123cb93a386Sopenharmony_ci    }
124cb93a386Sopenharmony_ci
125cb93a386Sopenharmony_ci    constexpr float kWidth = 400;
126cb93a386Sopenharmony_ci    SkFont font(SkTypeface::MakeDefault());
127cb93a386Sopenharmony_ci    RunHandler rh(name, reporter, (const char*)data->data(), data->size());
128cb93a386Sopenharmony_ci    shaper->shape((const char*)data->data(), data->size(), font, true, kWidth, &rh);
129cb93a386Sopenharmony_ci
130cb93a386Sopenharmony_ci    // Even on empty input, expect that the line is started, that the zero run infos are comitted,
131cb93a386Sopenharmony_ci    // and the empty line is comitted. This allows the user to properly handle empy runs.
132cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, rh.fBeginLine);
133cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, rh.fCommitRunInfo);
134cb93a386Sopenharmony_ci    REPORTER_ASSERT(reporter, rh.fCommitLine);
135cb93a386Sopenharmony_ci
136cb93a386Sopenharmony_ci    constexpr SkFourByteTag latn = SkSetFourByteTag('l','a','t','n');
137cb93a386Sopenharmony_ci    auto fontIterator = SkShaper::TrivialFontRunIterator(font, data->size());
138cb93a386Sopenharmony_ci    auto bidiIterator = SkShaper::TrivialBiDiRunIterator(0, data->size());
139cb93a386Sopenharmony_ci    auto scriptIterator = SkShaper::TrivialScriptRunIterator(latn, data->size());
140cb93a386Sopenharmony_ci    auto languageIterator = SkShaper::TrivialLanguageRunIterator("en-US", data->size());
141cb93a386Sopenharmony_ci    shaper->shape((const char*)data->data(), data->size(),
142cb93a386Sopenharmony_ci                  fontIterator, bidiIterator, scriptIterator, languageIterator, kWidth, &rh);
143cb93a386Sopenharmony_ci}
144cb93a386Sopenharmony_ci
145cb93a386Sopenharmony_civoid cluster_test(skiatest::Reporter* reporter, const char* resource) {
146cb93a386Sopenharmony_ci    auto data = GetResourceAsData(resource);
147cb93a386Sopenharmony_ci    if (!data) {
148cb93a386Sopenharmony_ci        ERRORF(reporter, "Could not get resource %s.", resource);
149cb93a386Sopenharmony_ci        return;
150cb93a386Sopenharmony_ci    }
151cb93a386Sopenharmony_ci
152cb93a386Sopenharmony_ci    shaper_test(reporter, resource, data.get());
153cb93a386Sopenharmony_ci}
154cb93a386Sopenharmony_ci
155cb93a386Sopenharmony_ci}  // namespace
156cb93a386Sopenharmony_ci
157cb93a386Sopenharmony_ciDEF_TEST(Shaper_cluster_empty, r) { shaper_test(r, "empty", SkData::MakeEmpty().get()); }
158cb93a386Sopenharmony_ci
159cb93a386Sopenharmony_ci#define SHAPER_TEST(X) DEF_TEST(Shaper_cluster_ ## X, r) { cluster_test(r, "text/" #X ".txt"); }
160cb93a386Sopenharmony_ciSHAPER_TEST(arabic)
161cb93a386Sopenharmony_ciSHAPER_TEST(armenian)
162cb93a386Sopenharmony_ciSHAPER_TEST(balinese)
163cb93a386Sopenharmony_ciSHAPER_TEST(buginese)
164cb93a386Sopenharmony_ciSHAPER_TEST(cherokee)
165cb93a386Sopenharmony_ciSHAPER_TEST(cyrillic)
166cb93a386Sopenharmony_ciSHAPER_TEST(emoji)
167cb93a386Sopenharmony_ciSHAPER_TEST(english)
168cb93a386Sopenharmony_ciSHAPER_TEST(ethiopic)
169cb93a386Sopenharmony_ciSHAPER_TEST(greek)
170cb93a386Sopenharmony_ciSHAPER_TEST(hangul)
171cb93a386Sopenharmony_ciSHAPER_TEST(han_simplified)
172cb93a386Sopenharmony_ciSHAPER_TEST(han_traditional)
173cb93a386Sopenharmony_ciSHAPER_TEST(hebrew)
174cb93a386Sopenharmony_ciSHAPER_TEST(javanese)
175cb93a386Sopenharmony_ciSHAPER_TEST(kana)
176cb93a386Sopenharmony_ciSHAPER_TEST(lao)
177cb93a386Sopenharmony_ciSHAPER_TEST(mandaic)
178cb93a386Sopenharmony_ciSHAPER_TEST(newtailue)
179cb93a386Sopenharmony_ciSHAPER_TEST(nko)
180cb93a386Sopenharmony_ciSHAPER_TEST(sinhala)
181cb93a386Sopenharmony_ciSHAPER_TEST(sundanese)
182cb93a386Sopenharmony_ciSHAPER_TEST(syriac)
183cb93a386Sopenharmony_ciSHAPER_TEST(thaana)
184cb93a386Sopenharmony_ciSHAPER_TEST(thai)
185cb93a386Sopenharmony_ciSHAPER_TEST(tibetan)
186cb93a386Sopenharmony_ciSHAPER_TEST(tifnagh)
187cb93a386Sopenharmony_ciSHAPER_TEST(vai)
188cb93a386Sopenharmony_ciSHAPER_TEST(bengali)
189cb93a386Sopenharmony_ciSHAPER_TEST(devanagari)
190cb93a386Sopenharmony_ciSHAPER_TEST(khmer)
191cb93a386Sopenharmony_ciSHAPER_TEST(myanmar)
192cb93a386Sopenharmony_ciSHAPER_TEST(taitham)
193cb93a386Sopenharmony_ciSHAPER_TEST(tamil)
194cb93a386Sopenharmony_ci#undef SHAPER_TEST
195cb93a386Sopenharmony_ci
196cb93a386Sopenharmony_ci#endif  // defined(SKSHAPER_IMPLEMENTATION) && !defined(SK_BUILD_FOR_GOOGLE3)
197