зеркало из https://github.com/mozilla/gecko-dev.git
238 строки
10 KiB
C++
238 строки
10 KiB
C++
/* -*- 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),
|
|
mScriptSizeMultiplier(aStyle->StyleFont()->mScriptSizeMultiplier),
|
|
mTextTransform(aStyle->StyleText()->mTextTransform),
|
|
mMathVariant(aStyle->StyleFont()->mMathVariant),
|
|
mExplicitLanguage(aStyle->StyleFont()->mExplicitLanguage) {}
|
|
|
|
nsFont mFont;
|
|
RefPtr<nsAtom> mLanguage;
|
|
RefPtr<nsPresContext> mPresContext;
|
|
float mScriptSizeMultiplier;
|
|
mozilla::StyleTextTransform mTextTransform;
|
|
uint8_t 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<nsTransformedTextRun> MakeTextRun(
|
|
const uint8_t* aString, uint32_t aLength,
|
|
const gfxFontGroup::Parameters* aParams, gfxFontGroup* aFontGroup,
|
|
mozilla::gfx::ShapedTextFlags aFlags, nsTextFrameUtils::Flags aFlags2,
|
|
nsTArray<RefPtr<nsTransformedCharStyle>>&& aStyles, bool aOwnsFactory);
|
|
|
|
already_AddRefed<nsTransformedTextRun> MakeTextRun(
|
|
const char16_t* aString, uint32_t aLength,
|
|
const gfxFontGroup::Parameters* aParams, gfxFontGroup* aFontGroup,
|
|
mozilla::gfx::ShapedTextFlags aFlags, nsTextFrameUtils::Flags aFlags2,
|
|
nsTArray<RefPtr<nsTransformedCharStyle>>&& 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.
|
|
*/
|
|
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
|
|
explicit nsCaseTransformTextRunFactory(
|
|
mozilla::UniquePtr<nsTransformingTextRunFactory>
|
|
aInnerTransformingTextRunFactory,
|
|
bool aAllUppercase = false)
|
|
: mInnerTransformingTextRunFactory(
|
|
std::move(aInnerTransformingTextRunFactory)),
|
|
mAllUppercase(aAllUppercase) {}
|
|
|
|
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 aAllUppercase is true, the transform is (global)
|
|
// upper-casing, 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 upper-casing usage (no input textrun),
|
|
// these are ignored.
|
|
// 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.
|
|
static bool TransformString(
|
|
const nsAString& aString, nsString& aConvertedString, bool aAllUppercase,
|
|
bool aCaseTransformsOnly, const nsAtom* aLanguage,
|
|
nsTArray<bool>& aCharsToMergeArray, nsTArray<bool>& aDeletedCharsArray,
|
|
const nsTransformedTextRun* aTextRun = nullptr,
|
|
uint32_t aOffsetInTextRun = 0,
|
|
nsTArray<uint8_t>* aCanBreakBeforeArray = nullptr,
|
|
nsTArray<RefPtr<nsTransformedCharStyle>>* aStyleArray = nullptr);
|
|
|
|
protected:
|
|
mozilla::UniquePtr<nsTransformingTextRunFactory>
|
|
mInnerTransformingTextRunFactory;
|
|
bool mAllUppercase;
|
|
};
|
|
|
|
/**
|
|
* 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<nsTransformedTextRun> 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<RefPtr<nsTransformedCharStyle>>&& 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<RefPtr<nsTransformedCharStyle>> mStyles;
|
|
nsTArray<bool> 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<RefPtr<nsTransformedCharStyle>>&& 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<CompressedGlyph*>(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_*/
|