summaryrefslogtreecommitdiffstats
path: root/layout/generic/nsTextFrame.h
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--layout/generic/nsTextFrame.h1075
1 files changed, 1075 insertions, 0 deletions
diff --git a/layout/generic/nsTextFrame.h b/layout/generic/nsTextFrame.h
new file mode 100644
index 0000000000..568d3333c2
--- /dev/null
+++ b/layout/generic/nsTextFrame.h
@@ -0,0 +1,1075 @@
+/* -*- 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 nsTextFrame_h__
+#define nsTextFrame_h__
+
+#include "mozilla/Attributes.h"
+#include "mozilla/UniquePtr.h"
+#include "mozilla/dom/Text.h"
+#include "mozilla/gfx/2D.h"
+
+#include "nsIFrame.h"
+#include "nsISelectionController.h"
+#include "nsSplittableFrame.h"
+#include "gfxSkipChars.h"
+#include "gfxTextRun.h"
+#include "JustificationUtils.h"
+
+// Undo the windows.h damage
+#if defined(XP_WIN) && defined(DrawText)
+# undef DrawText
+#endif
+
+class nsTextPaintStyle;
+class nsLineList_iterator;
+struct SelectionDetails;
+class nsTextFragment;
+
+namespace mozilla {
+class SVGContextPaint;
+class SVGTextFrame;
+class nsDisplayTextGeometry;
+class nsDisplayText;
+} // namespace mozilla
+
+class nsTextFrame : public nsIFrame {
+ typedef mozilla::LayoutDeviceRect LayoutDeviceRect;
+ typedef mozilla::SelectionTypeMask SelectionTypeMask;
+ typedef mozilla::SelectionType SelectionType;
+ typedef mozilla::TextRangeStyle TextRangeStyle;
+ typedef mozilla::gfx::DrawTarget DrawTarget;
+ typedef mozilla::gfx::Point Point;
+ typedef mozilla::gfx::Rect Rect;
+ typedef mozilla::gfx::Size Size;
+ typedef gfxTextRun::Range Range;
+
+ public:
+ enum TextRunType : uint8_t;
+ struct TabWidthStore;
+
+ /**
+ * An implementation of gfxTextRun::PropertyProvider that computes spacing and
+ * hyphenation based on CSS properties for a text frame.
+ */
+ class MOZ_STACK_CLASS PropertyProvider final
+ : public gfxTextRun::PropertyProvider {
+ typedef gfxTextRun::Range Range;
+ typedef gfxTextRun::HyphenType HyphenType;
+ typedef mozilla::gfx::DrawTarget DrawTarget;
+
+ public:
+ /**
+ * Use this constructor for reflow, when we don't know what text is
+ * really mapped by the frame and we have a lot of other data around.
+ *
+ * @param aLength can be INT32_MAX to indicate we cover all the text
+ * associated with aFrame up to where its flow chain ends in the given
+ * textrun. If INT32_MAX is passed, justification and hyphen-related methods
+ * cannot be called, nor can GetOriginalLength().
+ */
+ PropertyProvider(gfxTextRun* aTextRun, const nsStyleText* aTextStyle,
+ const nsTextFragment* aFrag, nsTextFrame* aFrame,
+ const gfxSkipCharsIterator& aStart, int32_t aLength,
+ nsIFrame* aLineContainer,
+ nscoord aOffsetFromBlockOriginForTabs,
+ nsTextFrame::TextRunType aWhichTextRun);
+
+ /**
+ * Use this constructor after the frame has been reflowed and we don't
+ * have other data around. Gets everything from the frame. EnsureTextRun
+ * *must* be called before this!!!
+ */
+ PropertyProvider(nsTextFrame* aFrame, const gfxSkipCharsIterator& aStart,
+ nsTextFrame::TextRunType aWhichTextRun,
+ nsFontMetrics* aFontMetrics);
+
+ /**
+ * As above, but assuming we want the inflated text run and associated
+ * metrics.
+ */
+ PropertyProvider(nsTextFrame* aFrame, const gfxSkipCharsIterator& aStart)
+ : PropertyProvider(aFrame, aStart, nsTextFrame::eInflated,
+ aFrame->InflatedFontMetrics()) {}
+
+ // Call this after construction if you're not going to reflow the text
+ void InitializeForDisplay(bool aTrimAfter);
+
+ void InitializeForMeasure();
+
+ void GetSpacing(Range aRange, Spacing* aSpacing) const final;
+ gfxFloat GetHyphenWidth() const final;
+ void GetHyphenationBreaks(Range aRange,
+ HyphenType* aBreakBefore) const final;
+ mozilla::StyleHyphens GetHyphensOption() const final {
+ return mTextStyle->mHyphens;
+ }
+ mozilla::gfx::ShapedTextFlags GetShapedTextFlags() const final;
+
+ already_AddRefed<DrawTarget> GetDrawTarget() const final;
+
+ uint32_t GetAppUnitsPerDevUnit() const final {
+ return mTextRun->GetAppUnitsPerDevUnit();
+ }
+
+ void GetSpacingInternal(Range aRange, Spacing* aSpacing,
+ bool aIgnoreTabs) const;
+
+ /**
+ * Compute the justification information in given DOM range, return
+ * justification info and assignments if requested.
+ */
+ mozilla::JustificationInfo ComputeJustification(
+ Range aRange,
+ nsTArray<mozilla::JustificationAssignment>* aAssignments = nullptr);
+
+ const nsTextFrame* GetFrame() const { return mFrame; }
+ // This may not be equal to the frame offset/length in because we may have
+ // adjusted for whitespace trimming according to the state bits set in the
+ // frame (for the static provider)
+ const gfxSkipCharsIterator& GetStart() const { return mStart; }
+ // May return INT32_MAX if that was given to the constructor
+ uint32_t GetOriginalLength() const {
+ NS_ASSERTION(mLength != INT32_MAX, "Length not known");
+ return mLength;
+ }
+ const nsTextFragment* GetFragment() const { return mFrag; }
+
+ gfxFontGroup* GetFontGroup() const {
+ if (!mFontGroup) {
+ mFontGroup = GetFontMetrics()->GetThebesFontGroup();
+ }
+ return mFontGroup;
+ }
+
+ nsFontMetrics* GetFontMetrics() const {
+ if (!mFontMetrics) {
+ InitFontGroupAndFontMetrics();
+ }
+ return mFontMetrics;
+ }
+
+ void CalcTabWidths(Range aTransformedRange, gfxFloat aTabWidth) const;
+
+ gfxFloat MinTabAdvance() const;
+
+ const gfxSkipCharsIterator& GetEndHint() const { return mTempIterator; }
+
+ protected:
+ void SetupJustificationSpacing(bool aPostReflow);
+
+ void InitFontGroupAndFontMetrics() const;
+
+ const RefPtr<gfxTextRun> mTextRun;
+ mutable gfxFontGroup* mFontGroup;
+ mutable RefPtr<nsFontMetrics> mFontMetrics;
+ const nsStyleText* mTextStyle;
+ const nsTextFragment* mFrag;
+ const nsIFrame* mLineContainer;
+ nsTextFrame* mFrame;
+ gfxSkipCharsIterator mStart; // Offset in original and transformed string
+ const gfxSkipCharsIterator mTempIterator;
+
+ // Either null, or pointing to the frame's TabWidthProperty.
+ mutable nsTextFrame::TabWidthStore* mTabWidths;
+ // How far we've done tab-width calculation; this is ONLY valid when
+ // mTabWidths is nullptr (otherwise rely on mTabWidths->mLimit instead).
+ // It's a DOM offset relative to the current frame's offset.
+ mutable uint32_t mTabWidthsAnalyzedLimit;
+
+ int32_t mLength; // DOM string length, may be INT32_MAX
+ const gfxFloat mWordSpacing; // space for each whitespace char
+ const gfxFloat mLetterSpacing; // space for each letter
+ mutable gfxFloat mMinTabAdvance; // min advance for <tab> char
+ mutable gfxFloat mHyphenWidth;
+ mutable gfxFloat mOffsetFromBlockOriginForTabs;
+
+ // The values in mJustificationSpacings corresponds to unskipped
+ // characters start from mJustificationArrayStart.
+ uint32_t mJustificationArrayStart;
+ nsTArray<Spacing> mJustificationSpacings;
+
+ const bool mReflowing;
+ const nsTextFrame::TextRunType mWhichTextRun;
+ };
+
+ explicit nsTextFrame(ComputedStyle* aStyle, nsPresContext* aPresContext,
+ ClassID aID = kClassID)
+ : nsIFrame(aStyle, aPresContext, aID) {}
+
+ NS_DECL_FRAMEARENA_HELPERS(nsTextFrame)
+
+ friend class nsContinuingTextFrame;
+
+ // nsQueryFrame
+ NS_DECL_QUERYFRAME
+
+ NS_DECLARE_FRAME_PROPERTY_DELETABLE(ContinuationsProperty,
+ nsTArray<nsTextFrame*>)
+
+ // nsIFrame
+ void BuildDisplayList(nsDisplayListBuilder* aBuilder,
+ const nsDisplayListSet& aLists) final;
+
+ void Init(nsIContent* aContent, nsContainerFrame* aParent,
+ nsIFrame* aPrevInFlow) override;
+
+ void Destroy(DestroyContext&) override;
+
+ Cursor GetCursor(const nsPoint&) final;
+
+ nsresult CharacterDataChanged(const CharacterDataChangeInfo&) final;
+
+ nsTextFrame* FirstContinuation() const override {
+ return const_cast<nsTextFrame*>(this);
+ }
+ nsTextFrame* GetPrevContinuation() const override { return nullptr; }
+ nsTextFrame* GetNextContinuation() const final { return mNextContinuation; }
+ void SetNextContinuation(nsIFrame* aNextContinuation) final {
+ NS_ASSERTION(!aNextContinuation || Type() == aNextContinuation->Type(),
+ "setting a next continuation with incorrect type!");
+ NS_ASSERTION(
+ !nsSplittableFrame::IsInNextContinuationChain(aNextContinuation, this),
+ "creating a loop in continuation chain!");
+ mNextContinuation = static_cast<nsTextFrame*>(aNextContinuation);
+ if (aNextContinuation)
+ aNextContinuation->RemoveStateBits(NS_FRAME_IS_FLUID_CONTINUATION);
+ // Setting a non-fluid continuation might affect our flow length (they're
+ // quite rare so we assume it always does) so we delete our cached value:
+ if (GetContent()->HasFlag(NS_HAS_FLOWLENGTH_PROPERTY)) {
+ GetContent()->RemoveProperty(nsGkAtoms::flowlength);
+ GetContent()->UnsetFlags(NS_HAS_FLOWLENGTH_PROPERTY);
+ }
+ }
+ nsTextFrame* GetNextInFlow() const final {
+ return mNextContinuation && mNextContinuation->HasAnyStateBits(
+ NS_FRAME_IS_FLUID_CONTINUATION)
+ ? mNextContinuation
+ : nullptr;
+ }
+ void SetNextInFlow(nsIFrame* aNextInFlow) final {
+ NS_ASSERTION(!aNextInFlow || Type() == aNextInFlow->Type(),
+ "setting a next in flow with incorrect type!");
+ NS_ASSERTION(
+ !nsSplittableFrame::IsInNextContinuationChain(aNextInFlow, this),
+ "creating a loop in continuation chain!");
+ mNextContinuation = static_cast<nsTextFrame*>(aNextInFlow);
+ if (mNextContinuation &&
+ !mNextContinuation->HasAnyStateBits(NS_FRAME_IS_FLUID_CONTINUATION)) {
+ // Changing from non-fluid to fluid continuation might affect our flow
+ // length, so we delete our cached value:
+ if (GetContent()->HasFlag(NS_HAS_FLOWLENGTH_PROPERTY)) {
+ GetContent()->RemoveProperty(nsGkAtoms::flowlength);
+ GetContent()->UnsetFlags(NS_HAS_FLOWLENGTH_PROPERTY);
+ }
+ }
+ if (aNextInFlow) {
+ aNextInFlow->AddStateBits(NS_FRAME_IS_FLUID_CONTINUATION);
+ }
+ }
+ nsTextFrame* LastInFlow() const final;
+ nsTextFrame* LastContinuation() const final;
+
+ bool ShouldSuppressLineBreak() const;
+
+ void InvalidateFrame(uint32_t aDisplayItemKey = 0,
+ bool aRebuildDisplayItems = true) final;
+ void InvalidateFrameWithRect(const nsRect& aRect,
+ uint32_t aDisplayItemKey = 0,
+ bool aRebuildDisplayItems = true) final;
+
+#ifdef DEBUG_FRAME_DUMP
+ void List(FILE* out = stderr, const char* aPrefix = "",
+ ListFlags aFlags = ListFlags()) const final;
+ nsresult GetFrameName(nsAString& aResult) const final;
+ void ToCString(nsCString& aBuf) const;
+ void ListTextRuns(FILE* out, nsTHashSet<const void*>& aSeen) const final;
+#endif
+
+ // Returns this text frame's content's text fragment.
+ //
+ // Assertions in Init() ensure we only ever get a Text node as content.
+ const nsTextFragment* TextFragment() const {
+ return &mContent->AsText()->TextFragment();
+ }
+
+ /**
+ * Check that the text in this frame is entirely whitespace. Importantly,
+ * this function considers non-breaking spaces (0xa0) to be whitespace,
+ * whereas nsTextFrame::IsEmpty does not. It also considers both one and
+ * two-byte chars.
+ */
+ bool IsEntirelyWhitespace() const;
+
+ ContentOffsets CalcContentOffsetsFromFramePoint(const nsPoint& aPoint) final;
+ ContentOffsets GetCharacterOffsetAtFramePoint(const nsPoint& aPoint);
+
+ /**
+ * This is called only on the primary text frame. It indicates that
+ * the selection state of the given character range has changed.
+ * Frames corresponding to the character range are unconditionally invalidated
+ * (Selection::Repaint depends on this).
+ * @param aStart start of character range.
+ * @param aEnd end (exclusive) of character range.
+ * @param aSelected true iff the character range is now selected.
+ * @param aType the type of the changed selection.
+ */
+ void SelectionStateChanged(uint32_t aStart, uint32_t aEnd, bool aSelected,
+ SelectionType aSelectionType);
+
+ FrameSearchResult PeekOffsetNoAmount(bool aForward, int32_t* aOffset) final;
+ FrameSearchResult PeekOffsetCharacter(
+ bool aForward, int32_t* aOffset,
+ PeekOffsetCharacterOptions aOptions = PeekOffsetCharacterOptions()) final;
+ FrameSearchResult PeekOffsetWord(bool aForward, bool aWordSelectEatSpace,
+ bool aIsKeyboardSelect, int32_t* aOffset,
+ PeekWordState* aState,
+ bool aTrimSpaces) final;
+
+ // Helper method that editor code uses to test for visibility.
+ [[nodiscard]] bool HasVisibleText();
+
+ // Flags for aSetLengthFlags
+ enum { ALLOW_FRAME_CREATION_AND_DESTRUCTION = 0x01 };
+
+ // Update offsets to account for new length. This may clear mTextRun.
+ void SetLength(int32_t aLength, nsLineLayout* aLineLayout,
+ uint32_t aSetLengthFlags = 0);
+
+ std::pair<int32_t, int32_t> GetOffsets() const final;
+
+ void AdjustOffsetsForBidi(int32_t start, int32_t end) final;
+
+ nsresult GetPointFromOffset(int32_t inOffset, nsPoint* outPoint) final;
+ nsresult GetCharacterRectsInRange(int32_t aInOffset, int32_t aLength,
+ nsTArray<nsRect>& aRects) final;
+
+ nsresult GetChildFrameContainingOffset(int32_t inContentOffset, bool inHint,
+ int32_t* outFrameContentOffset,
+ nsIFrame** outChildFrame) final;
+
+ bool IsEmpty() final;
+ bool IsSelfEmpty() final { return IsEmpty(); }
+ Maybe<nscoord> GetNaturalBaselineBOffset(
+ mozilla::WritingMode aWM, BaselineSharingGroup aBaselineGroup,
+ BaselineExportContext) const override;
+
+ bool HasSignificantTerminalNewline() const final;
+
+ /**
+ * Returns true if this text frame is logically adjacent to the end of the
+ * line.
+ */
+ bool IsAtEndOfLine() const;
+
+ /**
+ * Call this only after reflow the frame. Returns true if non-collapsed
+ * characters are present.
+ */
+ bool HasNoncollapsedCharacters() const {
+ return HasAnyStateBits(TEXT_HAS_NONCOLLAPSED_CHARACTERS);
+ }
+
+#ifdef ACCESSIBILITY
+ mozilla::a11y::AccType AccessibleType() final;
+#endif
+
+ float GetFontSizeInflation() const;
+ bool IsCurrentFontInflation(float aInflation) const;
+ bool HasFontSizeInflation() const {
+ return HasAnyStateBits(TEXT_HAS_FONT_INFLATION);
+ }
+ void SetFontSizeInflation(float aInflation);
+
+ void MarkIntrinsicISizesDirty() final;
+ nscoord GetMinISize(gfxContext* aRenderingContext) final;
+ nscoord GetPrefISize(gfxContext* aRenderingContext) final;
+ void AddInlineMinISize(gfxContext* aRenderingContext,
+ InlineMinISizeData* aData) override;
+ void AddInlinePrefISize(gfxContext* aRenderingContext,
+ InlinePrefISizeData* aData) override;
+ SizeComputationResult ComputeSize(
+ gfxContext* aRenderingContext, mozilla::WritingMode aWM,
+ const mozilla::LogicalSize& aCBSize, nscoord aAvailableISize,
+ const mozilla::LogicalSize& aMargin,
+ const mozilla::LogicalSize& aBorderPadding,
+ const mozilla::StyleSizeOverrides& aSizeOverrides,
+ mozilla::ComputeSizeFlags aFlags) final;
+ nsRect ComputeTightBounds(DrawTarget* aDrawTarget) const final;
+ nsresult GetPrefWidthTightBounds(gfxContext* aContext, nscoord* aX,
+ nscoord* aXMost) final;
+ void Reflow(nsPresContext* aPresContext, ReflowOutput& aMetrics,
+ const ReflowInput& aReflowInput, nsReflowStatus& aStatus) final;
+ bool CanContinueTextRun() const final;
+ // Method that is called for a text frame that is logically
+ // adjacent to the end of the line (i.e. followed only by empty text frames,
+ // placeholders or inlines containing such).
+ struct TrimOutput {
+ // true if we trimmed some space or changed metrics in some other way.
+ // In this case, we should call RecomputeOverflow on this frame.
+ bool mChanged;
+ // an amount to *subtract* from the frame's width (zero if !mChanged)
+ nscoord mDeltaWidth;
+ };
+ TrimOutput TrimTrailingWhiteSpace(DrawTarget* aDrawTarget);
+ RenderedText GetRenderedText(
+ uint32_t aStartOffset = 0, uint32_t aEndOffset = UINT32_MAX,
+ TextOffsetType aOffsetType = TextOffsetType::OffsetsInContentText,
+ TrailingWhitespace aTrimTrailingWhitespace =
+ TrailingWhitespace::Trim) final;
+
+ mozilla::OverflowAreas RecomputeOverflow(nsIFrame* aBlockFrame,
+ bool aIncludeShadows = true);
+
+ enum TextRunType : uint8_t {
+ // Anything in reflow (but not intrinsic width calculation) or
+ // painting should use the inflated text run (i.e., with font size
+ // inflation applied).
+ eInflated,
+ // Intrinsic width calculation should use the non-inflated text run.
+ // When there is font size inflation, it will be different.
+ eNotInflated
+ };
+
+ void AddInlineMinISizeForFlow(gfxContext* aRenderingContext,
+ nsIFrame::InlineMinISizeData* aData,
+ TextRunType aTextRunType);
+ void AddInlinePrefISizeForFlow(gfxContext* aRenderingContext,
+ InlinePrefISizeData* aData,
+ TextRunType aTextRunType);
+
+ /**
+ * Calculate the horizontal bounds of the grapheme clusters that fit entirely
+ * inside the given left[top]/right[bottom] edges (which are positive lengths
+ * from the respective frame edge). If an input value is zero it is ignored
+ * and the result for that edge is zero. All out parameter values are
+ * undefined when the method returns false.
+ * @return true if at least one whole grapheme cluster fit between the edges
+ */
+ bool MeasureCharClippedText(nscoord aVisIStartEdge, nscoord aVisIEndEdge,
+ nscoord* aSnappedStartEdge,
+ nscoord* aSnappedEndEdge);
+ /**
+ * Same as above; this method also the returns the corresponding text run
+ * offset and number of characters that fit. All out parameter values are
+ * undefined when the method returns false.
+ * @return true if at least one whole grapheme cluster fit between the edges
+ */
+ bool MeasureCharClippedText(PropertyProvider& aProvider,
+ nscoord aVisIStartEdge, nscoord aVisIEndEdge,
+ uint32_t* aStartOffset, uint32_t* aMaxLength,
+ nscoord* aSnappedStartEdge,
+ nscoord* aSnappedEndEdge);
+
+ /**
+ * Return true if this box has some text to display.
+ * It returns false if at least one of these conditions are met:
+ * a. the frame hasn't been reflowed yet
+ * b. GetContentLength() == 0
+ * c. it contains only non-significant white-space
+ */
+ bool HasNonSuppressedText() const;
+
+ /**
+ * Object with various callbacks for PaintText() to invoke for different parts
+ * of the frame's text rendering, when we're generating paths rather than
+ * painting.
+ *
+ * Callbacks are invoked in the following order:
+ *
+ * NotifySelectionBackgroundNeedsFill?
+ * PaintDecorationLine*
+ * NotifyBeforeText
+ * NotifyGlyphPathEmitted*
+ * NotifyAfterText
+ * PaintDecorationLine*
+ * PaintSelectionDecorationLine*
+ *
+ * The color of each part of the frame's text rendering is passed as an
+ * argument to the NotifyBefore* callback for that part. The nscolor can take
+ * on one of the three selection special colors defined in LookAndFeel.h --
+ * NS_TRANSPARENT, NS_SAME_AS_FOREGROUND_COLOR and
+ * NS_40PERCENT_FOREGROUND_COLOR.
+ */
+ struct DrawPathCallbacks : gfxTextRunDrawCallbacks {
+ /**
+ * @param aShouldPaintSVGGlyphs Whether SVG glyphs should be painted.
+ */
+ explicit DrawPathCallbacks(bool aShouldPaintSVGGlyphs = false)
+ : gfxTextRunDrawCallbacks(aShouldPaintSVGGlyphs) {}
+
+ /**
+ * Called to have the selection highlight drawn before the text is drawn
+ * over the top.
+ */
+ virtual void NotifySelectionBackgroundNeedsFill(const Rect& aBackgroundRect,
+ nscolor aColor,
+ DrawTarget& aDrawTarget) {}
+
+ /**
+ * Called before (for under/over-line) or after (for line-through) the text
+ * is drawn to have a text decoration line drawn.
+ */
+ virtual void PaintDecorationLine(Rect aPath, nscolor aColor) {}
+
+ /**
+ * Called after selected text is drawn to have a decoration line drawn over
+ * the text. (All types of text decoration are drawn after the text when
+ * text is selected.)
+ */
+ virtual void PaintSelectionDecorationLine(Rect aPath, nscolor aColor) {}
+
+ /**
+ * Called just before any paths have been emitted to the gfxContext
+ * for the glyphs of the frame's text.
+ */
+ virtual void NotifyBeforeText(nscolor aColor) {}
+
+ /**
+ * Called just after all the paths have been emitted to the gfxContext
+ * for the glyphs of the frame's text.
+ */
+ virtual void NotifyAfterText() {}
+
+ /**
+ * Called just before a path corresponding to a selection decoration line
+ * has been emitted to the gfxContext.
+ */
+ virtual void NotifyBeforeSelectionDecorationLine(nscolor aColor) {}
+
+ /**
+ * Called just after a path corresponding to a selection decoration line
+ * has been emitted to the gfxContext.
+ */
+ virtual void NotifySelectionDecorationLinePathEmitted() {}
+ };
+
+ struct MOZ_STACK_CLASS PaintTextParams {
+ gfxContext* context;
+ Point framePt;
+ LayoutDeviceRect dirtyRect;
+ mozilla::SVGContextPaint* contextPaint = nullptr;
+ DrawPathCallbacks* callbacks = nullptr;
+ enum {
+ PaintText, // Normal text painting.
+ GenerateTextMask // To generate a mask from a text frame. Should
+ // only paint text itself with opaque color.
+ // Text shadow, text selection color and text
+ // decoration are all discarded in this state.
+ };
+ uint8_t state = PaintText;
+ explicit PaintTextParams(gfxContext* aContext) : context(aContext) {}
+
+ bool IsPaintText() const { return state == PaintText; }
+ bool IsGenerateTextMask() const { return state == GenerateTextMask; }
+ };
+
+ struct PaintTextSelectionParams;
+ struct DrawTextRunParams;
+ struct DrawTextParams;
+ struct ClipEdges;
+ struct PaintShadowParams;
+ struct PaintDecorationLineParams;
+
+ struct PriorityOrderedSelectionsForRange {
+ /// List of Selection Details active for the given range.
+ /// Ordered by priority, i.e. the last element has the highest priority.
+ nsTArray<const SelectionDetails*> mSelectionRanges;
+ Range mRange;
+ };
+
+ // Primary frame paint method called from nsDisplayText. Can also be used
+ // to generate paths rather than paint the frame's text by passing a callback
+ // object. The private DrawText() is what applies the text to a graphics
+ // context.
+ void PaintText(const PaintTextParams& aParams, const nscoord aVisIStartEdge,
+ const nscoord aVisIEndEdge, const nsPoint& aToReferenceFrame,
+ const bool aIsSelected, float aOpacity = 1.0f);
+ // helper: paint text frame when we're impacted by at least one selection.
+ // Return false if the text was not painted and we should continue with
+ // the fast path.
+ bool PaintTextWithSelection(const PaintTextSelectionParams& aParams,
+ const ClipEdges& aClipEdges);
+ // helper: paint text with foreground and background colors determined
+ // by selection(s). Also computes a mask of all selection types applying to
+ // our text, returned in aAllSelectionTypeMask.
+ // Return false if the text was not painted and we should continue with
+ // the fast path.
+ bool PaintTextWithSelectionColors(
+ const PaintTextSelectionParams& aParams,
+ const mozilla::UniquePtr<SelectionDetails>& aDetails,
+ SelectionTypeMask* aAllSelectionTypeMask, const ClipEdges& aClipEdges);
+ // helper: paint text decorations for text selected by aSelectionType
+ void PaintTextSelectionDecorations(
+ const PaintTextSelectionParams& aParams,
+ const mozilla::UniquePtr<SelectionDetails>& aDetails,
+ SelectionType aSelectionType);
+
+ SelectionTypeMask ResolveSelections(
+ const PaintTextSelectionParams& aParams, const SelectionDetails* aDetails,
+ nsTArray<PriorityOrderedSelectionsForRange>& aResult,
+ SelectionType aSelectionType, bool* aAnyBackgrounds = nullptr) const;
+
+ void DrawEmphasisMarks(gfxContext* aContext, mozilla::WritingMode aWM,
+ const mozilla::gfx::Point& aTextBaselinePt,
+ const mozilla::gfx::Point& aFramePt, Range aRange,
+ const nscolor* aDecorationOverrideColor,
+ PropertyProvider* aProvider);
+
+ nscolor GetCaretColorAt(int32_t aOffset) final;
+
+ // @param aSelectionFlags may be multiple of nsISelectionDisplay::DISPLAY_*.
+ // @return nsISelectionController.idl's `getDisplaySelection`.
+ int16_t GetSelectionStatus(int16_t* aSelectionFlags);
+
+ int32_t GetContentOffset() const { return mContentOffset; }
+ int32_t GetContentLength() const {
+ NS_ASSERTION(GetContentEnd() - mContentOffset >= 0, "negative length");
+ return GetContentEnd() - mContentOffset;
+ }
+ int32_t GetContentEnd() const;
+ // This returns the length the frame thinks it *should* have after it was
+ // last reflowed (0 if it hasn't been reflowed yet). This should be used only
+ // when setting up the text offsets for a new continuation frame.
+ int32_t GetContentLengthHint() const { return mContentLengthHint; }
+
+ // Compute the length of the content mapped by this frame
+ // and all its in-flow siblings. Basically this means starting at
+ // mContentOffset and going to the end of the text node or the next bidi
+ // continuation boundary.
+ int32_t GetInFlowContentLength();
+
+ /**
+ * Acquires the text run for this content, if necessary.
+ * @param aWhichTextRun indicates whether to get an inflated or non-inflated
+ * text run
+ * @param aRefDrawTarget the DrawTarget to use as a reference for creating the
+ * textrun, if available (if not, we'll create one which will just be slower)
+ * @param aLineContainer the block ancestor for this frame, or nullptr if
+ * unknown
+ * @param aFlowEndInTextRun if non-null, this returns the textrun offset of
+ * end of the text associated with this frame and its in-flow siblings
+ * @return a gfxSkipCharsIterator set up to map DOM offsets for this frame
+ * to offsets into the textrun; its initial offset is set to this frame's
+ * content offset
+ */
+ gfxSkipCharsIterator EnsureTextRun(TextRunType aWhichTextRun,
+ DrawTarget* aRefDrawTarget = nullptr,
+ nsIFrame* aLineContainer = nullptr,
+ const nsLineList_iterator* aLine = nullptr,
+ uint32_t* aFlowEndInTextRun = nullptr);
+
+ gfxTextRun* GetTextRun(TextRunType aWhichTextRun) const {
+ if (aWhichTextRun == eInflated || !HasFontSizeInflation()) return mTextRun;
+ return GetUninflatedTextRun();
+ }
+ gfxTextRun* GetUninflatedTextRun() const;
+ void SetTextRun(gfxTextRun* aTextRun, TextRunType aWhichTextRun,
+ float aInflation);
+ bool IsInTextRunUserData() const {
+ return HasAnyStateBits(TEXT_IN_TEXTRUN_USER_DATA |
+ TEXT_IN_UNINFLATED_TEXTRUN_USER_DATA);
+ }
+ /**
+ * Notify the frame that it should drop its pointer to a text run.
+ * Returns whether the text run was removed (i.e., whether it was
+ * associated with this frame, either as its inflated or non-inflated
+ * text run.
+ */
+ bool RemoveTextRun(gfxTextRun* aTextRun);
+ /**
+ * Clears out |mTextRun| (or the uninflated text run, when aInflated
+ * is nsTextFrame::eNotInflated and there is inflation) from all frames that
+ * hold a reference to it, starting at |aStartContinuation|, or if it's
+ * nullptr, starting at |this|. Deletes the text run if all references
+ * were cleared and it's not cached.
+ */
+ void ClearTextRun(nsTextFrame* aStartContinuation, TextRunType aWhichTextRun);
+
+ void ClearTextRuns() {
+ ClearTextRun(nullptr, nsTextFrame::eInflated);
+ if (HasFontSizeInflation()) {
+ ClearTextRun(nullptr, nsTextFrame::eNotInflated);
+ }
+ }
+
+ /**
+ * Wipe out references to textrun(s) without deleting the textruns.
+ */
+ void DisconnectTextRuns();
+
+ // Get the DOM content range mapped by this frame after excluding
+ // whitespace subject to start-of-line and end-of-line trimming.
+ // The textrun must have been created before calling this.
+ struct TrimmedOffsets {
+ int32_t mStart;
+ int32_t mLength;
+ int32_t GetEnd() const { return mStart + mLength; }
+ };
+ enum class TrimmedOffsetFlags : uint8_t {
+ Default = 0,
+ NotPostReflow = 1 << 0,
+ NoTrimAfter = 1 << 1,
+ NoTrimBefore = 1 << 2
+ };
+ TrimmedOffsets GetTrimmedOffsets(
+ const nsTextFragment* aFrag,
+ TrimmedOffsetFlags aFlags = TrimmedOffsetFlags::Default) const;
+
+ // Similar to Reflow(), but for use from nsLineLayout
+ void ReflowText(nsLineLayout& aLineLayout, nscoord aAvailableWidth,
+ DrawTarget* aDrawTarget, ReflowOutput& aMetrics,
+ nsReflowStatus& aStatus);
+
+ nscoord ComputeLineHeight() const;
+
+ bool IsFloatingFirstLetterChild() const;
+
+ bool IsInitialLetterChild() const;
+
+ bool ComputeCustomOverflow(mozilla::OverflowAreas& aOverflowAreas) final;
+ bool ComputeCustomOverflowInternal(mozilla::OverflowAreas& aOverflowAreas,
+ bool aIncludeShadows);
+
+ void AssignJustificationGaps(const mozilla::JustificationAssignment& aAssign);
+ mozilla::JustificationAssignment GetJustificationAssignment() const;
+
+ uint32_t CountGraphemeClusters() const;
+
+ bool HasAnyNoncollapsedCharacters() final;
+
+ /**
+ * Call this after you have manually changed the text node contents without
+ * notifying that change. This behaves as if all the text contents changed.
+ * (You should only use this for native anonymous content.)
+ */
+ void NotifyNativeAnonymousTextnodeChange(uint32_t aOldLength);
+
+ nsFontMetrics* InflatedFontMetrics() const;
+
+ nsRect WebRenderBounds();
+
+ // Find the continuation (which may be this frame itself) containing the
+ // given offset. Note that this may return null, if the offset is beyond the
+ // text covered by the continuation chain.
+ // (To be used only on the first textframe in the chain.)
+ nsTextFrame* FindContinuationForOffset(int32_t aOffset);
+
+ void SetHangableISize(nscoord aISize);
+ nscoord GetHangableISize() const;
+ void ClearHangableISize();
+
+ void SetTrimmableWS(gfxTextRun::TrimmableWS aTrimmableWS);
+ gfxTextRun::TrimmableWS GetTrimmableWS() const;
+ void ClearTrimmableWS();
+
+ protected:
+ virtual ~nsTextFrame();
+
+ friend class mozilla::nsDisplayTextGeometry;
+ friend class mozilla::nsDisplayText;
+
+ mutable RefPtr<nsFontMetrics> mFontMetrics;
+ RefPtr<gfxTextRun> mTextRun;
+ nsTextFrame* mNextContinuation = nullptr;
+ // The key invariant here is that mContentOffset never decreases along
+ // a next-continuation chain. And of course mContentOffset is always <= the
+ // the text node's content length, and the mContentOffset for the first frame
+ // is always 0. Furthermore the text mapped by a frame is determined by
+ // GetContentOffset() and GetContentLength()/GetContentEnd(), which get
+ // the length from the difference between this frame's offset and the next
+ // frame's offset, or the text length if there is no next frame. This means
+ // the frames always map the text node without overlapping or leaving any
+ // gaps.
+ int32_t mContentOffset = 0;
+ // This does *not* indicate the length of text currently mapped by the frame;
+ // instead it's a hint saying that this frame *wants* to map this much text
+ // so if we create a new continuation, this is where that continuation should
+ // start.
+ int32_t mContentLengthHint = 0;
+ nscoord mAscent = 0;
+
+ // Cached selection state.
+ enum class SelectionState : uint8_t {
+ Unknown,
+ Selected,
+ NotSelected,
+ };
+ mutable SelectionState mIsSelected = SelectionState::Unknown;
+
+ // Flags used to track whether certain properties are present.
+ // (Public to keep MOZ_MAKE_ENUM_CLASS_BITWISE_OPERATORS happy.)
+ public:
+ enum class PropertyFlags : uint8_t {
+ // Whether a cached continuations array is present.
+ Continuations = 1 << 0,
+ // Whether a HangableWhitespace property is present.
+ HangableWS = 1 << 1,
+ // Whether a TrimmableWhitespace property is present.
+ TrimmableWS = 2 << 1,
+ };
+
+ protected:
+ PropertyFlags mPropertyFlags = PropertyFlags(0);
+
+ /**
+ * Return true if the frame is part of a Selection.
+ * Helper method to implement the public IsSelected() API.
+ */
+ bool IsFrameSelected() const final;
+
+ void InvalidateSelectionState() { mIsSelected = SelectionState::Unknown; }
+
+ mozilla::UniquePtr<SelectionDetails> GetSelectionDetails();
+
+ void UnionAdditionalOverflow(nsPresContext* aPresContext, nsIFrame* aBlock,
+ PropertyProvider& aProvider,
+ nsRect* aInkOverflowRect,
+ bool aIncludeTextDecorations,
+ bool aIncludeShadows);
+
+ // Update information of emphasis marks, and return the visial
+ // overflow rect of the emphasis marks.
+ nsRect UpdateTextEmphasis(mozilla::WritingMode aWM,
+ PropertyProvider& aProvider);
+
+ void PaintOneShadow(const PaintShadowParams& aParams,
+ const mozilla::StyleSimpleShadow& aShadowDetails,
+ gfxRect& aBoundingBox, uint32_t aBlurFlags);
+
+ void PaintShadows(mozilla::Span<const mozilla::StyleSimpleShadow>,
+ const PaintShadowParams& aParams);
+
+ struct LineDecoration {
+ nsIFrame* mFrame;
+
+ // This is represents the offset from our baseline to mFrame's baseline;
+ // positive offsets are *above* the baseline and negative offsets below
+ nscoord mBaselineOffset;
+
+ // This represents the offset from the initial position of the underline
+ const mozilla::LengthPercentageOrAuto mTextUnderlineOffset;
+
+ // for CSS property text-decoration-thickness, the width refers to the
+ // thickness of the decoration line
+ const mozilla::StyleTextDecorationLength mTextDecorationThickness;
+ nscolor mColor;
+ mozilla::StyleTextDecorationStyle mStyle;
+
+ // The text-underline-position property; affects the underline offset only
+ // if mTextUnderlineOffset is auto.
+ const mozilla::StyleTextUnderlinePosition mTextUnderlinePosition;
+
+ LineDecoration(nsIFrame* const aFrame, const nscoord aOff,
+ mozilla::StyleTextUnderlinePosition aUnderlinePosition,
+ const mozilla::LengthPercentageOrAuto& aUnderlineOffset,
+ const mozilla::StyleTextDecorationLength& aDecThickness,
+ const nscolor aColor,
+ const mozilla::StyleTextDecorationStyle aStyle)
+ : mFrame(aFrame),
+ mBaselineOffset(aOff),
+ mTextUnderlineOffset(aUnderlineOffset),
+ mTextDecorationThickness(aDecThickness),
+ mColor(aColor),
+ mStyle(aStyle),
+ mTextUnderlinePosition(aUnderlinePosition) {}
+
+ LineDecoration(const LineDecoration& aOther) = default;
+
+ bool operator==(const LineDecoration& aOther) const {
+ return mFrame == aOther.mFrame && mStyle == aOther.mStyle &&
+ mColor == aOther.mColor &&
+ mBaselineOffset == aOther.mBaselineOffset &&
+ mTextUnderlinePosition == aOther.mTextUnderlinePosition &&
+ mTextUnderlineOffset == aOther.mTextUnderlineOffset &&
+ mTextDecorationThickness == aOther.mTextDecorationThickness;
+ }
+
+ bool operator!=(const LineDecoration& aOther) const {
+ return !(*this == aOther);
+ }
+ };
+ struct TextDecorations {
+ AutoTArray<LineDecoration, 1> mOverlines, mUnderlines, mStrikes;
+
+ TextDecorations() = default;
+
+ bool HasDecorationLines() const {
+ return HasUnderline() || HasOverline() || HasStrikeout();
+ }
+ bool HasUnderline() const { return !mUnderlines.IsEmpty(); }
+ bool HasOverline() const { return !mOverlines.IsEmpty(); }
+ bool HasStrikeout() const { return !mStrikes.IsEmpty(); }
+ bool operator==(const TextDecorations& aOther) const {
+ return mOverlines == aOther.mOverlines &&
+ mUnderlines == aOther.mUnderlines && mStrikes == aOther.mStrikes;
+ }
+ bool operator!=(const TextDecorations& aOther) const {
+ return !(*this == aOther);
+ }
+ };
+ enum TextDecorationColorResolution { eResolvedColors, eUnresolvedColors };
+ void GetTextDecorations(nsPresContext* aPresContext,
+ TextDecorationColorResolution aColorResolution,
+ TextDecorations& aDecorations);
+
+ void DrawTextRun(Range aRange, const mozilla::gfx::Point& aTextBaselinePt,
+ const DrawTextRunParams& aParams);
+
+ void DrawTextRunAndDecorations(Range aRange,
+ const mozilla::gfx::Point& aTextBaselinePt,
+ const DrawTextParams& aParams,
+ const TextDecorations& aDecorations);
+
+ void DrawText(Range aRange, const mozilla::gfx::Point& aTextBaselinePt,
+ const DrawTextParams& aParams);
+
+ // Set non empty rect to aRect, it should be overflow rect or frame rect.
+ // If the result rect is larger than the given rect, this returns true.
+ bool CombineSelectionUnderlineRect(nsPresContext* aPresContext,
+ nsRect& aRect);
+
+ // This sets *aShadows to the appropriate shadows, if any, for the given
+ // type of selection.
+ // If text-shadow was not specified, *aShadows is left untouched.
+ // Note that the returned shadow(s) will only be valid as long as the
+ // textPaintStyle remains in scope.
+ void GetSelectionTextShadow(
+ SelectionType aSelectionType, nsTextPaintStyle& aTextPaintStyle,
+ mozilla::Span<const mozilla::StyleSimpleShadow>* aShadows);
+
+ /**
+ * Utility methods to paint selection.
+ */
+ void DrawSelectionDecorations(
+ gfxContext* aContext, const LayoutDeviceRect& aDirtyRect,
+ mozilla::SelectionType aSelectionType, nsTextPaintStyle& aTextPaintStyle,
+ const TextRangeStyle& aRangeStyle, const Point& aPt,
+ gfxFloat aICoordInFrame, gfxFloat aWidth, gfxFloat aAscent,
+ const gfxFont::Metrics& aFontMetrics, DrawPathCallbacks* aCallbacks,
+ bool aVertical, mozilla::StyleTextDecorationLine aDecoration);
+
+ void PaintDecorationLine(const PaintDecorationLineParams& aParams);
+ /**
+ * ComputeDescentLimitForSelectionUnderline() computes the most far position
+ * where we can put selection underline.
+ *
+ * @return The maximum underline offset from the baseline (positive value
+ * means that the underline can put below the baseline).
+ */
+ gfxFloat ComputeDescentLimitForSelectionUnderline(
+ nsPresContext* aPresContext, const gfxFont::Metrics& aFontMetrics);
+ /**
+ * This function encapsulates all knowledge of how selections affect
+ * foreground and background colors.
+ * @param aForeground the foreground color to use
+ * @param aBackground the background color to use, or RGBA(0,0,0,0) if no
+ * background should be painted
+ * @return true if the selection affects colors, false otherwise
+ */
+ static bool GetSelectionTextColors(SelectionType aSelectionType,
+ nsAtom* aHighlightName,
+ nsTextPaintStyle& aTextPaintStyle,
+ const TextRangeStyle& aRangeStyle,
+ nscolor* aForeground,
+ nscolor* aBackground);
+ /**
+ * ComputeSelectionUnderlineHeight() computes selection underline height of
+ * the specified selection type from the font metrics.
+ */
+ static gfxFloat ComputeSelectionUnderlineHeight(
+ nsPresContext* aPresContext, const gfxFont::Metrics& aFontMetrics,
+ SelectionType aSelectionType);
+
+ /**
+ * @brief Helper struct which contains selection data such as its details,
+ * range and priority.
+ */
+ struct SelectionRange {
+ const SelectionDetails* mDetails{nullptr};
+ gfxTextRun::Range mRange;
+ /// used to determine the order of overlapping selections of the same type.
+ uint32_t mPriority{0};
+ };
+ /**
+ * @brief Helper: Extracts a list of `SelectionRange` structs from given
+ * `SelectionDetails` and computes a priority for overlapping selection
+ * ranges.
+ */
+ static SelectionTypeMask CreateSelectionRangeList(
+ const SelectionDetails* aDetails, SelectionType aSelectionType,
+ const PaintTextSelectionParams& aParams,
+ nsTArray<SelectionRange>& aSelectionRanges, bool* aAnyBackgrounds);
+
+ /**
+ * @brief Creates an array of `CombinedSelectionRange`s from given list
+ * of `SelectionRange`s.
+ * Each instance of `CombinedSelectionRange` represents a piece of text with
+ * constant Selections.
+ *
+ * Example:
+ *
+ * Consider this text fragment, [] and () marking selection ranges:
+ * ab[cd(e]f)g
+ * This results in the following array of combined ranges:
+ * - [0]: range: (2, 4), selections: "[]"
+ * - [1]: range: (4, 5), selections: "[]", "()"
+ * - [2]: range: (5, 6), selections: "()"
+ * Depending on the priorities of the ranges, [1] may have a different order
+ * of its ranges. The given example indicates that "()" has a higher priority
+ * than "[]".
+ *
+ * @param aSelectionRanges Array of `SelectionRange` objects. Must be
+ * sorted by the start offset.
+ * @param aCombinedSelectionRanges Out parameter. Returns the constructed
+ * array of combined selection ranges.
+ */
+ static void CombineSelectionRanges(
+ const nsTArray<SelectionRange>& aSelectionRanges,
+ nsTArray<PriorityOrderedSelectionsForRange>& aCombinedSelectionRanges);
+
+ ContentOffsets GetCharacterOffsetAtFramePointInternal(
+ const nsPoint& aPoint, bool aForInsertionPoint);
+
+ static float GetTextCombineScaleFactor(nsTextFrame* aFrame);
+
+ void ClearFrameOffsetCache();
+
+ void ClearMetrics(ReflowOutput& aMetrics);
+
+ // Return pointer to an array of all frames in the continuation chain, or
+ // null if we're too short of memory.
+ nsTArray<nsTextFrame*>* GetContinuations();
+
+ // Clear any cached continuations array; this should be called whenever the
+ // chain is modified.
+ inline void ClearCachedContinuations();
+
+ /**
+ * UpdateIteratorFromOffset() updates the iterator from a given offset.
+ * Also, aInOffset may be updated to cluster start if aInOffset isn't
+ * the offset of cluster start.
+ */
+ void UpdateIteratorFromOffset(const PropertyProvider& aProperties,
+ int32_t& aInOffset,
+ gfxSkipCharsIterator& aIter);
+
+ nsPoint GetPointFromIterator(const gfxSkipCharsIterator& aIter,
+ PropertyProvider& aProperties);
+};
+
+MOZ_MAKE_ENUM_CLASS_BITWISE_OPERATORS(nsTextFrame::TrimmedOffsetFlags)
+MOZ_MAKE_ENUM_CLASS_BITWISE_OPERATORS(nsTextFrame::PropertyFlags)
+
+inline void nsTextFrame::ClearCachedContinuations() {
+ MOZ_ASSERT(NS_IsMainThread());
+ if (mPropertyFlags & PropertyFlags::Continuations) {
+ RemoveProperty(ContinuationsProperty());
+ mPropertyFlags &= ~PropertyFlags::Continuations;
+ }
+}
+
+#endif