/* * Copyright 2016 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #ifndef SkShaper_DEFINED #define SkShaper_DEFINED #include "include/core/SkFontMgr.h" #include "include/core/SkPoint.h" #include "include/core/SkRefCnt.h" #include "include/core/SkScalar.h" #include "include/core/SkTextBlob.h" #include "include/core/SkTypes.h" #include #if !defined(SKSHAPER_IMPLEMENTATION) #define SKSHAPER_IMPLEMENTATION 0 #endif #if !defined(SKSHAPER_API) #if defined(SKSHAPER_DLL) #if defined(_MSC_VER) #if SKSHAPER_IMPLEMENTATION #define SKSHAPER_API __declspec(dllexport) #else #define SKSHAPER_API __declspec(dllimport) #endif #else #define SKSHAPER_API __attribute__((visibility("default"))) #endif #else #define SKSHAPER_API #endif #endif class SkFont; class SkFontMgr; class SkUnicode; class SKSHAPER_API SkShaper { public: static std::unique_ptr MakePrimitive(); #ifdef SK_SHAPER_HARFBUZZ_AVAILABLE static std::unique_ptr MakeShaperDrivenWrapper(sk_sp = nullptr); static std::unique_ptr MakeShapeThenWrap(sk_sp = nullptr); static std::unique_ptr MakeShapeDontWrapOrReorder(sk_sp = nullptr); static void PurgeHarfBuzzCache(); #endif #ifdef SK_SHAPER_CORETEXT_AVAILABLE static std::unique_ptr MakeCoreText(); #endif static std::unique_ptr Make(sk_sp = nullptr); static void PurgeCaches(); SkShaper(); virtual ~SkShaper(); class RunIterator { public: // Required until C++17 copy elision RunIterator(const RunIterator&) = default; // Required because above is required RunIterator() = default; virtual ~RunIterator() = default; /** Set state to that of current run and move iterator to end of that run. */ virtual void consume() = 0; /** Offset to one past the last (utf8) element in the current run. */ virtual size_t endOfCurrentRun() const = 0; /** Return true if consume should no longer be called. */ virtual bool atEnd() const = 0; }; class FontRunIterator : public RunIterator { public: virtual const SkFont& currentFont() const = 0; }; class BiDiRunIterator : public RunIterator { public: /** The unicode bidi embedding level (even ltr, odd rtl) */ virtual uint8_t currentLevel() const = 0; }; class ScriptRunIterator : public RunIterator { public: /** Should be iso15924 codes. */ virtual SkFourByteTag currentScript() const = 0; }; class LanguageRunIterator : public RunIterator { public: /** Should be BCP-47, c locale names may also work. */ virtual const char* currentLanguage() const = 0; }; struct Feature { SkFourByteTag tag; uint32_t value; size_t start; // Offset to the start (utf8) element of the run. size_t end; // Offset to one past the last (utf8) element of the run. }; private: template class TrivialRunIterator : public RunIteratorSubclass { public: static_assert(std::is_base_of::value, ""); TrivialRunIterator(size_t utf8Bytes) : fEnd(utf8Bytes), fAtEnd(fEnd == 0) {} void consume() override { SkASSERT(!fAtEnd); fAtEnd = true; } size_t endOfCurrentRun() const override { return fAtEnd ? fEnd : 0; } bool atEnd() const override { return fAtEnd; } private: size_t fEnd; bool fAtEnd; }; public: static std::unique_ptr MakeFontMgrRunIterator(const char* utf8, size_t utf8Bytes, const SkFont& font, sk_sp fallback); static std::unique_ptr MakeFontMgrRunIterator(const char* utf8, size_t utf8Bytes, const SkFont& font, sk_sp fallback, const char* requestName, SkFontStyle requestStyle, const SkShaper::LanguageRunIterator*); class TrivialFontRunIterator : public TrivialRunIterator { public: TrivialFontRunIterator(const SkFont& font, size_t utf8Bytes) : TrivialRunIterator(utf8Bytes), fFont(font) {} const SkFont& currentFont() const override { return fFont; } private: SkFont fFont; }; static std::unique_ptr MakeBiDiRunIterator(const char* utf8, size_t utf8Bytes, uint8_t bidiLevel); #ifdef SK_UNICODE_AVAILABLE static std::unique_ptr MakeSkUnicodeBidiRunIterator(SkUnicode* unicode, const char* utf8, size_t utf8Bytes, uint8_t bidiLevel); static std::unique_ptr MakeIcuBiDiRunIterator(const char* utf8, size_t utf8Bytes, uint8_t bidiLevel); #endif class TrivialBiDiRunIterator : public TrivialRunIterator { public: TrivialBiDiRunIterator(uint8_t bidiLevel, size_t utf8Bytes) : TrivialRunIterator(utf8Bytes), fBidiLevel(bidiLevel) {} uint8_t currentLevel() const override { return fBidiLevel; } private: uint8_t fBidiLevel; }; static std::unique_ptr MakeScriptRunIterator(const char* utf8, size_t utf8Bytes, SkFourByteTag script); #if defined(SK_SHAPER_HARFBUZZ_AVAILABLE) && defined(SK_UNICODE_AVAILABLE) static std::unique_ptr MakeSkUnicodeHbScriptRunIterator(SkUnicode* unicode, const char* utf8, size_t utf8Bytes); static std::unique_ptr MakeHbIcuScriptRunIterator(const char* utf8, size_t utf8Bytes); #endif class TrivialScriptRunIterator : public TrivialRunIterator { public: TrivialScriptRunIterator(SkFourByteTag script, size_t utf8Bytes) : TrivialRunIterator(utf8Bytes), fScript(script) {} SkFourByteTag currentScript() const override { return fScript; } private: SkFourByteTag fScript; }; static std::unique_ptr MakeStdLanguageRunIterator(const char* utf8, size_t utf8Bytes); class TrivialLanguageRunIterator : public TrivialRunIterator { public: TrivialLanguageRunIterator(const char* language, size_t utf8Bytes) : TrivialRunIterator(utf8Bytes), fLanguage(language) {} const char* currentLanguage() const override { return fLanguage.c_str(); } private: SkString fLanguage; }; class RunHandler { public: virtual ~RunHandler() = default; struct Range { constexpr Range() : fBegin(0), fSize(0) {} constexpr Range(size_t begin, size_t size) : fBegin(begin), fSize(size) {} size_t fBegin; size_t fSize; constexpr size_t begin() const { return fBegin; } constexpr size_t end() const { return begin() + size(); } constexpr size_t size() const { return fSize; } }; struct RunInfo { const SkFont& fFont; uint8_t fBidiLevel; SkVector fAdvance; size_t glyphCount; Range utf8Range; }; struct Buffer { SkGlyphID* glyphs; // required SkPoint* positions; // required, if (!offsets) put glyphs[i] at positions[i] // if ( offsets) positions[i+1]-positions[i] are advances SkPoint* offsets; // optional, if ( offsets) put glyphs[i] at positions[i]+offsets[i] uint32_t* clusters; // optional, utf8+clusters[i] starts run which produced glyphs[i] SkPoint point; // offset to add to all positions }; /** Called when beginning a line. */ virtual void beginLine() = 0; /** Called once for each run in a line. Can compute baselines and offsets. */ virtual void runInfo(const RunInfo&) = 0; /** Called after all runInfo calls for a line. */ virtual void commitRunInfo() = 0; /** Called for each run in a line after commitRunInfo. The buffer will be filled out. */ virtual Buffer runBuffer(const RunInfo&) = 0; /** Called after each runBuffer is filled out. */ virtual void commitRunBuffer(const RunInfo&) = 0; /** Called when ending a line. */ virtual void commitLine() = 0; }; virtual void shape(const char* utf8, size_t utf8Bytes, const SkFont& srcFont, bool leftToRight, SkScalar width, RunHandler*) const = 0; virtual void shape(const char* utf8, size_t utf8Bytes, FontRunIterator&, BiDiRunIterator&, ScriptRunIterator&, LanguageRunIterator&, SkScalar width, RunHandler*) const = 0; virtual void shape(const char* utf8, size_t utf8Bytes, FontRunIterator&, BiDiRunIterator&, ScriptRunIterator&, LanguageRunIterator&, const Feature* features, size_t featuresSize, SkScalar width, RunHandler*) const = 0; private: SkShaper(const SkShaper&) = delete; SkShaper& operator=(const SkShaper&) = delete; }; /** * Helper for shaping text directly into a SkTextBlob. */ class SKSHAPER_API SkTextBlobBuilderRunHandler final : public SkShaper::RunHandler { public: SkTextBlobBuilderRunHandler(const char* utf8Text, SkPoint offset) : fUtf8Text(utf8Text) , fOffset(offset) {} sk_sp makeBlob(); SkPoint endPoint() { return fOffset; } void beginLine() override; void runInfo(const RunInfo&) override; void commitRunInfo() override; Buffer runBuffer(const RunInfo&) override; void commitRunBuffer(const RunInfo&) override; void commitLine() override; private: SkTextBlobBuilder fBuilder; char const * const fUtf8Text; uint32_t* fClusters; int fClusterOffset; int fGlyphCount; SkScalar fMaxRunAscent; SkScalar fMaxRunDescent; SkScalar fMaxRunLeading; SkPoint fCurrentPosition; SkPoint fOffset; }; #endif // SkShaper_DEFINED