/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef NSTEXTRUNTRANSFORMATIONS_H_ #define NSTEXTRUNTRANSFORMATIONS_H_ #include "mozilla/Attributes.h" #include "mozilla/MemoryReporting.h" #include "mozilla/UniquePtr.h" #include "gfxTextRun.h" #include "mozilla/ComputedStyle.h" #include "nsPresContext.h" #include "nsStyleStruct.h" class nsTransformedTextRun; struct nsTransformedCharStyle final { NS_INLINE_DECL_REFCOUNTING(nsTransformedCharStyle) explicit nsTransformedCharStyle(mozilla::ComputedStyle* aStyle, nsPresContext* aPresContext) : mFont(aStyle->StyleFont()->mFont), mLanguage(aStyle->StyleFont()->mLanguage), mPresContext(aPresContext), mTextTransform(aStyle->StyleText()->mTextTransform), mMathVariant(aStyle->StyleFont()->mMathVariant), mExplicitLanguage(aStyle->StyleFont()->mExplicitLanguage) {} nsFont mFont; RefPtr mLanguage; RefPtr mPresContext; mozilla::StyleTextTransform mTextTransform; mozilla::StyleMathVariant mMathVariant; bool mExplicitLanguage; bool mForceNonFullWidth = false; bool mMaskPassword = false; private: ~nsTransformedCharStyle() = default; nsTransformedCharStyle(const nsTransformedCharStyle& aOther) = delete; nsTransformedCharStyle& operator=(const nsTransformedCharStyle& aOther) = delete; }; class nsTransformingTextRunFactory { public: virtual ~nsTransformingTextRunFactory() = default; // Default 8-bit path just transforms to Unicode and takes that path already_AddRefed MakeTextRun( const uint8_t* aString, uint32_t aLength, const gfxFontGroup::Parameters* aParams, gfxFontGroup* aFontGroup, mozilla::gfx::ShapedTextFlags aFlags, nsTextFrameUtils::Flags aFlags2, nsTArray>&& aStyles, bool aOwnsFactory); already_AddRefed MakeTextRun( const char16_t* aString, uint32_t aLength, const gfxFontGroup::Parameters* aParams, gfxFontGroup* aFontGroup, mozilla::gfx::ShapedTextFlags aFlags, nsTextFrameUtils::Flags aFlags2, nsTArray>&& aStyles, bool aOwnsFactory); virtual void RebuildTextRun(nsTransformedTextRun* aTextRun, mozilla::gfx::DrawTarget* aRefDrawTarget, gfxMissingFontRecorder* aMFR) = 0; }; /** * Builds textruns that transform the text in some way (e.g., capitalize) * and then render the text using some other textrun implementation. * This factory also supports "text-security" transforms that convert all * characters to a single symbol. */ class nsCaseTransformTextRunFactory : public nsTransformingTextRunFactory { public: // We could add an optimization here so that when there is no inner // factory, no title-case conversion, and no upper-casing of SZLIG, we // override MakeTextRun (after making it virtual in the superclass) and have // it just convert the string to uppercase or lowercase and create the textrun // via the fontgroup. // Takes ownership of aInnerTransformTextRunFactory nsCaseTransformTextRunFactory(mozilla::UniquePtr aInnerTransformingTextRunFactory, bool aAllUppercase, char16_t aMaskChar) : mInnerTransformingTextRunFactory( std::move(aInnerTransformingTextRunFactory)), mAllUppercase(aAllUppercase), mMaskChar(aMaskChar) {} virtual void RebuildTextRun(nsTransformedTextRun* aTextRun, mozilla::gfx::DrawTarget* aRefDrawTarget, gfxMissingFontRecorder* aMFR) override; // Perform a transformation on the given string, writing the result into // aConvertedString. If aGlobalTransform is passed, the transform is global // and aLanguage is used to determine any language-specific behavior; // otherwise, an nsTransformedTextRun should be passed in as aTextRun and its // styles will be used to determine the transform(s) to be applied. // If such an input textrun is provided, then its line-breaks and styles // will be copied to the output arrays, which must also be provided by // the caller. For the global transform usage (no input textrun), these are // ignored. // If aMaskChar is non-zero, it is used as a "masking" character to replace // all characters in the text (for -webkit-text-security). // If aCaseTransformsOnly is true, then only the upper/lower/capitalize // transformations are performed; full-width and full-size-kana are ignored. // If `aTextRun` is not nullptr and characters which are styled with setting // `nsTransformedCharStyle::mMaskPassword` to true, they are replaced with // password mask characters and are not transformed (i.e., won't be added // or merged for the specified transform). However, unmasked characters // whose `nsTransformedCharStyle::mMaskPassword` is set to false are // transformed normally. // Note that "masking" behavior may be triggered either by // -webkit-text-security, resulting in a non-zero aMaskChar being passed, // or by , which results in the editor code setting // nsTransformedCharStyle::mMaskPassword. (The latter mechanism enables the // editor to temporarily reveal the just-entered character during typing, // whereas simply setting aMaskChar would unconditionally mask all the text.) static bool TransformString( const nsAString& aString, nsString& aConvertedString, const mozilla::Maybe& aGlobalTransform, char16_t aMaskChar, bool aCaseTransformsOnly, const nsAtom* aLanguage, nsTArray& aCharsToMergeArray, nsTArray& aDeletedCharsArray, const nsTransformedTextRun* aTextRun = nullptr, uint32_t aOffsetInTextRun = 0, nsTArray* aCanBreakBeforeArray = nullptr, nsTArray>* aStyleArray = nullptr); protected: mozilla::UniquePtr mInnerTransformingTextRunFactory; bool mAllUppercase; char16_t mMaskChar; }; /** * So that we can reshape as necessary, we store enough information * to fully rebuild the textrun contents. */ class nsTransformedTextRun final : public gfxTextRun { public: static already_AddRefed Create( const gfxTextRunFactory::Parameters* aParams, nsTransformingTextRunFactory* aFactory, gfxFontGroup* aFontGroup, const char16_t* aString, uint32_t aLength, const mozilla::gfx::ShapedTextFlags aFlags, const nsTextFrameUtils::Flags aFlags2, nsTArray>&& aStyles, bool aOwnsFactory); ~nsTransformedTextRun() { if (mOwnsFactory) { delete mFactory; } } void SetCapitalization(uint32_t aStart, uint32_t aLength, bool* aCapitalization); virtual bool SetPotentialLineBreaks(Range aRange, const uint8_t* aBreakBefore) override; /** * Called after SetCapitalization and SetPotentialLineBreaks * are done and before we request any data from the textrun. Also always * called after a Create. */ void FinishSettingProperties(mozilla::gfx::DrawTarget* aRefDrawTarget, gfxMissingFontRecorder* aMFR) { if (mNeedsRebuild) { mNeedsRebuild = false; mFactory->RebuildTextRun(this, aRefDrawTarget, aMFR); } } // override the gfxTextRun impls to account for additional members here virtual size_t SizeOfExcludingThis( mozilla::MallocSizeOf aMallocSizeOf) override; virtual size_t SizeOfIncludingThis( mozilla::MallocSizeOf aMallocSizeOf) override; nsTransformingTextRunFactory* mFactory; nsTArray> mStyles; nsTArray mCapitalize; nsString mString; bool mOwnsFactory; bool mNeedsRebuild; private: nsTransformedTextRun(const gfxTextRunFactory::Parameters* aParams, nsTransformingTextRunFactory* aFactory, gfxFontGroup* aFontGroup, const char16_t* aString, uint32_t aLength, const mozilla::gfx::ShapedTextFlags aFlags, const nsTextFrameUtils::Flags aFlags2, nsTArray>&& aStyles, bool aOwnsFactory) : gfxTextRun(aParams, aLength, aFontGroup, aFlags, aFlags2), mFactory(aFactory), mStyles(std::move(aStyles)), mString(aString, aLength), mOwnsFactory(aOwnsFactory), mNeedsRebuild(true) { mCharacterGlyphs = reinterpret_cast(this + 1); } }; /** * Copy a given textrun, but merge certain characters into a single logical * character. Glyphs for a character are added to the glyph list for the * previous character and then the merged character is eliminated. Visually the * results are identical. * * This is used for text-transform:uppercase when we encounter a SZLIG, * whose uppercase form is "SS", or other ligature or precomposed form * that expands to multiple codepoints during case transformation, * and for Greek text when combining diacritics have been deleted. * * This function is unable to merge characters when they occur in different * glyph runs. This only happens in tricky edge cases where a character was * decomposed by case-mapping (e.g. there's no precomposed uppercase version * of an accented lowercase letter), and then font-matching caused the * diacritics to be assigned to a different font than the base character. * In this situation, the diacritic(s) get discarded, which is less than * ideal, but they probably weren't going to render very well anyway. * Bug 543200 will improve this by making font-matching operate on entire * clusters instead of individual codepoints. * * For simplicity, this produces a textrun containing all DetailedGlyphs, * no simple glyphs. So don't call it unless you really have merging to do. * * @param aCharsToMerge when aCharsToMerge[i] is true, this character in aSrc * is merged into the previous character * * @param aDeletedChars when aDeletedChars[i] is true, the character at this * position in aDest was deleted (has no corresponding char in aSrc) */ void MergeCharactersInTextRun(gfxTextRun* aDest, gfxTextRun* aSrc, const bool* aCharsToMerge, const bool* aDeletedChars); gfxTextRunFactory::Parameters GetParametersForInner( nsTransformedTextRun* aTextRun, mozilla::gfx::ShapedTextFlags* aFlags, mozilla::gfx::DrawTarget* aRefDrawTarget); #endif /*NSTEXTRUNTRANSFORMATIONS_H_*/