diff -r 000000000000 -r 6474c204b198 layout/generic/nsLineLayout.h --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/layout/generic/nsLineLayout.h Wed Dec 31 06:09:35 2014 +0100 @@ -0,0 +1,579 @@ +/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- + * vim:cindent:ts=2:et:sw=2: + * + * 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/. + * This Original Code has been modified by IBM Corporation. Modifications made by IBM + * described herein are Copyright (c) International Business Machines Corporation, 2000. + * Modifications to Mozilla code or documentation identified per MPL Section 3.3 + * + * Date Modified by Description of modification + * 04/20/2000 IBM Corp. OS/2 VisualAge build. + */ + +/* state and methods used while laying out a single line of a block frame */ + +#ifndef nsLineLayout_h___ +#define nsLineLayout_h___ + +#include "nsLineBox.h" +#include "nsBlockReflowState.h" +#include "plarena.h" +#include "gfxTypes.h" +#include "WritingModes.h" + +class nsFloatManager; +struct nsStyleText; + +class nsLineLayout { +public: + nsLineLayout(nsPresContext* aPresContext, + nsFloatManager* aFloatManager, + const nsHTMLReflowState* aOuterReflowState, + const nsLineList::iterator* aLine); + ~nsLineLayout(); + + void Init(nsBlockReflowState* aState, nscoord aMinLineBSize, + int32_t aLineNumber) { + mBlockRS = aState; + mMinLineBSize = aMinLineBSize; + mLineNumber = aLineNumber; + } + + int32_t GetLineNumber() const { + return mLineNumber; + } + + void BeginLineReflow(nscoord aICoord, nscoord aBCoord, + nscoord aISize, nscoord aBSize, + bool aImpactedByFloats, + bool aIsTopOfPage, + mozilla::WritingMode aWritingMode, + nscoord aContainerWidth); + + void EndLineReflow(); + + /** + * Called when a float has been placed. This method updates the + * inline frame and span data to account for any change in positions + * due to available space for the line boxes changing. + * @param aX/aY/aWidth/aHeight are the new available + * space rectangle, relative to the containing block. + * @param aFloatFrame the float frame that was placed. + */ + void UpdateBand(const nsRect& aNewAvailableSpace, + nsIFrame* aFloatFrame); + + void BeginSpan(nsIFrame* aFrame, const nsHTMLReflowState* aSpanReflowState, + nscoord aLeftEdge, nscoord aRightEdge, nscoord* aBaseline); + + // Returns the width of the span + nscoord EndSpan(nsIFrame* aFrame); + + int32_t GetCurrentSpanCount() const; + + void SplitLineTo(int32_t aNewCount); + + bool IsZeroBSize(); + + // Reflows the frame and returns the reflow status. aPushedFrame is true + // if the frame is pushed to the next line because it doesn't fit + nsresult ReflowFrame(nsIFrame* aFrame, + nsReflowStatus& aReflowStatus, + nsHTMLReflowMetrics* aMetrics, + bool& aPushedFrame); + + void AddBulletFrame(nsIFrame* aFrame, const nsHTMLReflowMetrics& aMetrics); + + void RemoveBulletFrame(nsIFrame* aFrame) { + PushFrame(aFrame); + } + + void BlockDirAlignLine(); + + bool TrimTrailingWhiteSpace(); + + void InlineDirAlignFrames(nsLineBox* aLine, bool aIsLastLine); + + /** + * Handle all the relative positioning in the line, compute the + * combined area (== overflow area) for the line, and handle view + * sizing/positioning and the setting of the overflow rect. + */ + void RelativePositionFrames(nsOverflowAreas& aOverflowAreas); + + // Support methods for word-wrapping during line reflow + + void SetTextJustificationWeights(int32_t aNumSpaces, int32_t aNumLetters) { + mTextJustificationNumSpaces = aNumSpaces; + mTextJustificationNumLetters = aNumLetters; + } + + /** + * @return true if so far during reflow no non-empty content has been + * placed in the line (according to nsIFrame::IsEmpty()) + */ + bool LineIsEmpty() const + { + return mLineIsEmpty; + } + + /** + * @return true if so far during reflow no non-empty leaf content + * (non-collapsed whitespace, replaced element, inline-block, etc) has been + * placed in the line + */ + bool LineAtStart() const + { + return mLineAtStart; + } + + bool LineIsBreakable() const; + + bool GetLineEndsInBR() const + { + return mLineEndsInBR; + } + + void SetLineEndsInBR(bool aOn) + { + mLineEndsInBR = aOn; + } + + //---------------------------------------- + // Inform the line-layout about the presence of a floating frame + // XXX get rid of this: use get-frame-type? + bool AddFloat(nsIFrame* aFloat, nscoord aAvailableWidth) + { + return mBlockRS->AddFloat(this, aFloat, aAvailableWidth); + } + + void SetTrimmableWidth(nscoord aTrimmableWidth) { + mTrimmableWidth = aTrimmableWidth; + } + + //---------------------------------------- + + bool GetFirstLetterStyleOK() const { + return mFirstLetterStyleOK; + } + + void SetFirstLetterStyleOK(bool aSetting) { + mFirstLetterStyleOK = aSetting; + } + + bool GetInFirstLetter() const { + return mInFirstLetter; + } + + void SetInFirstLetter(bool aSetting) { + mInFirstLetter = aSetting; + } + + bool GetInFirstLine() const { + return mInFirstLine; + } + + void SetInFirstLine(bool aSetting) { + mInFirstLine = aSetting; + } + + // Calling this during block reflow ensures that the next line of inlines + // will be marked dirty, if there is one. + void SetDirtyNextLine() { + mDirtyNextLine = true; + } + bool GetDirtyNextLine() { + return mDirtyNextLine; + } + + //---------------------------------------- + + nsPresContext* mPresContext; + + /** + * Record where an optional break could have been placed. During line reflow, + * frames containing optional break points (e.g., whitespace in text frames) + * can call SetLastOptionalBreakPosition to record where a break could + * have been made, but wasn't because we decided to place more content on + * the line. For non-text frames, offset 0 means + * before the content, offset INT32_MAX means after the content. + * + * Currently this is used to handle cases where a single word comprises + * multiple frames, and the first frame fits on the line but the whole word + * doesn't. We look back to the last optional break position and + * reflow the whole line again, forcing a break at that position. The last + * optional break position could be in a text frame or else after a frame + * that cannot be part of a text run, so those are the positions we record. + * + * @param aFits set to true if the break position is within the available width. + * + * @param aPriority the priority of the break opportunity. If we are + * prioritizing break opportunities, we will not set a break if we have + * already set a break with a higher priority. @see gfxBreakPriority. + * + * @return true if we are actually reflowing with forced break position and we + * should break here + */ + bool NotifyOptionalBreakPosition(nsIContent* aContent, int32_t aOffset, + bool aFits, gfxBreakPriority aPriority) { + NS_ASSERTION(!aFits || !mNeedBackup, + "Shouldn't be updating the break position with a break that fits after we've already flagged an overrun"); + // Remember the last break position that fits; if there was no break that fit, + // just remember the first break + if ((aFits && aPriority >= mLastOptionalBreakPriority) || + !mLastOptionalBreakContent) { + mLastOptionalBreakContent = aContent; + mLastOptionalBreakContentOffset = aOffset; + mLastOptionalBreakPriority = aPriority; + } + return aContent && mForceBreakContent == aContent && + mForceBreakContentOffset == aOffset; + } + /** + * Like NotifyOptionalBreakPosition, but here it's OK for mNeedBackup + * to be set, because the caller is merely pruning some saved break position(s) + * that are actually not feasible. + */ + void RestoreSavedBreakPosition(nsIContent* aContent, int32_t aOffset, + gfxBreakPriority aPriority) { + mLastOptionalBreakContent = aContent; + mLastOptionalBreakContentOffset = aOffset; + mLastOptionalBreakPriority = aPriority; + } + /** + * Signal that no backing up will be required after all. + */ + void ClearOptionalBreakPosition() { + mNeedBackup = false; + mLastOptionalBreakContent = nullptr; + mLastOptionalBreakContentOffset = -1; + mLastOptionalBreakPriority = gfxBreakPriority::eNoBreak; + } + // Retrieve last set optional break position. When this returns null, no + // optional break has been recorded (which means that the line can't break yet). + nsIContent* GetLastOptionalBreakPosition(int32_t* aOffset, + gfxBreakPriority* aPriority) { + *aOffset = mLastOptionalBreakContentOffset; + *aPriority = mLastOptionalBreakPriority; + return mLastOptionalBreakContent; + } + + /** + * Check whether frames overflowed the available width and CanPlaceFrame + * requested backing up to a saved break position. + */ + bool NeedsBackup() { return mNeedBackup; } + + // Line layout may place too much content on a line, overflowing its available + // width. When that happens, if SetLastOptionalBreakPosition has been + // used to record an optional break that wasn't taken, we can reflow the line + // again and force the break to happen at that point (i.e., backtracking + // to the last choice point). + + // Record that we want to break at the given content+offset (which + // should have been previously returned by GetLastOptionalBreakPosition + // from another nsLineLayout). + void ForceBreakAtPosition(nsIContent* aContent, int32_t aOffset) { + mForceBreakContent = aContent; + mForceBreakContentOffset = aOffset; + } + bool HaveForcedBreakPosition() { return mForceBreakContent != nullptr; } + int32_t GetForcedBreakPosition(nsIContent* aContent) { + return mForceBreakContent == aContent ? mForceBreakContentOffset : -1; + } + + /** + * This can't be null. It usually returns a block frame but may return + * some other kind of frame when inline frames are reflowed in a non-block + * context (e.g. MathML or floating first-letter). + */ + nsIFrame* LineContainerFrame() const { return mBlockReflowState->frame; } + const nsHTMLReflowState* LineContainerRS() const { return mBlockReflowState; } + const nsLineList::iterator* GetLine() const { + return mGotLineBox ? &mLineBox : nullptr; + } + nsLineList::iterator* GetLine() { + return mGotLineBox ? &mLineBox : nullptr; + } + + /** + * Returns the accumulated advance width of frames before the current frame + * on the line, plus the line container's left border+padding. + * This is always positive, the advance width is measured from + * the right edge for RTL blocks and from the left edge for LTR blocks. + * In other words, the current frame's distance from the line container's + * start content edge is: + * GetCurrentFrameInlineDistanceFromBlock() - lineContainer->GetUsedBorderAndPadding().left + * Note the use of .left for both LTR and RTL line containers. + */ + nscoord GetCurrentFrameInlineDistanceFromBlock(); + +protected: + // This state is constant for a given block frame doing line layout + nsFloatManager* mFloatManager; + const nsStyleText* mStyleText; // for the block + const nsHTMLReflowState* mBlockReflowState; + + nsIContent* mLastOptionalBreakContent; + nsIContent* mForceBreakContent; + + // XXX remove this when landing bug 154892 (splitting absolute positioned frames) + friend class nsInlineFrame; + + nsBlockReflowState* mBlockRS;/* XXX hack! */ + + nsLineList::iterator mLineBox; + + // Per-frame data recorded by the line-layout reflow logic. This + // state is the state needed to post-process the line after reflow + // has completed (block-direction alignment, inline-direction alignment, + // justification and relative positioning). + + struct PerSpanData; + struct PerFrameData; + friend struct PerSpanData; + friend struct PerFrameData; + struct PerFrameData + { + PerFrameData(mozilla::WritingMode aWritingMode) + : mBounds(aWritingMode) + , mMargin(aWritingMode) + , mBorderPadding(aWritingMode) + , mOffsets(aWritingMode) + {} + + // link to next/prev frame in same span + PerFrameData* mNext; + PerFrameData* mPrev; + + // pointer to child span data if this is an inline container frame + PerSpanData* mSpan; + + // The frame + nsIFrame* mFrame; + + // From metrics + nscoord mAscent; + // note that mBounds is a logical rect in the *line*'s writing mode. + // When setting frame coordinates, we have to convert to the frame's + // writing mode + mozilla::LogicalRect mBounds; + nsOverflowAreas mOverflowAreas; + + // From reflow-state + mozilla::LogicalMargin mMargin; + mozilla::LogicalMargin mBorderPadding; + mozilla::LogicalMargin mOffsets; + + // state for text justification + int32_t mJustificationNumSpaces; + int32_t mJustificationNumLetters; + + // Other state we use + uint8_t mBlockDirAlign; + +// PerFrameData flags +#define PFD_RELATIVEPOS 0x00000001 +#define PFD_ISTEXTFRAME 0x00000002 +#define PFD_ISNONEMPTYTEXTFRAME 0x00000004 +#define PFD_ISNONWHITESPACETEXTFRAME 0x00000008 +#define PFD_ISLETTERFRAME 0x00000010 +#define PFD_RECOMPUTEOVERFLOW 0x00000020 +#define PFD_ISBULLET 0x00000040 +#define PFD_SKIPWHENTRIMMINGWHITESPACE 0x00000080 +#define PFD_LASTFLAG PFD_SKIPWHENTRIMMINGWHITESPACE + + uint8_t mFlags; + + void SetFlag(uint32_t aFlag, bool aValue) + { + NS_ASSERTION(aFlag<=PFD_LASTFLAG, "bad flag"); + NS_ASSERTION(aFlag<=UINT8_MAX, "bad flag"); + if (aValue) { // set flag + mFlags |= aFlag; + } + else { // unset flag + mFlags &= ~aFlag; + } + } + + bool GetFlag(uint32_t aFlag) const + { + NS_ASSERTION(aFlag<=PFD_LASTFLAG, "bad flag"); + return !!(mFlags & aFlag); + } + + + PerFrameData* Last() { + PerFrameData* pfd = this; + while (pfd->mNext) { + pfd = pfd->mNext; + } + return pfd; + } + }; + PerFrameData* mFrameFreeList; + + struct PerSpanData { + union { + PerSpanData* mParent; + PerSpanData* mNextFreeSpan; + }; + PerFrameData* mFrame; + PerFrameData* mFirstFrame; + PerFrameData* mLastFrame; + + const nsHTMLReflowState* mReflowState; + bool mNoWrap; + mozilla::WritingMode mWritingMode; + bool mZeroEffectiveSpanBox; + bool mContainsFloat; + bool mHasNonemptyContent; + + nscoord mIStart; + nscoord mICoord; + nscoord mIEnd; + + nscoord mBStartLeading, mBEndLeading; + nscoord mLogicalBSize; + nscoord mMinBCoord, mMaxBCoord; + nscoord* mBaseline; + + void AppendFrame(PerFrameData* pfd) { + if (nullptr == mLastFrame) { + mFirstFrame = pfd; + } + else { + mLastFrame->mNext = pfd; + pfd->mPrev = mLastFrame; + } + mLastFrame = pfd; + } + }; + PerSpanData* mSpanFreeList; + PerSpanData* mRootSpan; + PerSpanData* mCurrentSpan; + + gfxBreakPriority mLastOptionalBreakPriority; + int32_t mLastOptionalBreakContentOffset; + int32_t mForceBreakContentOffset; + + nscoord mMinLineBSize; + + // The amount of text indent that we applied to this line, needed for + // max-element-size calculation. + nscoord mTextIndent; + + // This state varies during the reflow of a line but is line + // "global" state not span "local" state. + int32_t mLineNumber; + int32_t mTextJustificationNumSpaces; + int32_t mTextJustificationNumLetters; + + int32_t mTotalPlacedFrames; + + nscoord mBStartEdge; + nscoord mMaxStartBoxBSize; + nscoord mMaxEndBoxBSize; + + nscoord mInflationMinFontSize; + + // Final computed line-bSize value after BlockDirAlignFrames for + // the block has been called. + nscoord mFinalLineBSize; + + // Amount of trimmable whitespace width for the trailing text frame, if any + nscoord mTrimmableWidth; + + nscoord mContainerWidth; + + bool mFirstLetterStyleOK : 1; + bool mIsTopOfPage : 1; + bool mImpactedByFloats : 1; + bool mLastFloatWasLetterFrame : 1; + bool mLineIsEmpty : 1; + bool mLineEndsInBR : 1; + bool mNeedBackup : 1; + bool mInFirstLine : 1; + bool mGotLineBox : 1; + bool mInFirstLetter : 1; + bool mHasBullet : 1; + bool mDirtyNextLine : 1; + bool mLineAtStart : 1; + + int32_t mSpanDepth; +#ifdef DEBUG + int32_t mSpansAllocated, mSpansFreed; + int32_t mFramesAllocated, mFramesFreed; +#endif + PLArenaPool mArena; // Per span and per frame data, 4 byte aligned + + /** + * Allocate a PerFrameData from the mArena pool. The allocation is infallible. + */ + PerFrameData* NewPerFrameData(nsIFrame* aFrame); + + /** + * Allocate a PerSpanData from the mArena pool. The allocation is infallible. + */ + PerSpanData* NewPerSpanData(); + + void FreeSpan(PerSpanData* psd); + + bool InBlockContext() const { + return mSpanDepth == 0; + } + + void PushFrame(nsIFrame* aFrame); + + void AllowForStartMargin(PerFrameData* pfd, + nsHTMLReflowState& aReflowState); + + bool CanPlaceFrame(PerFrameData* pfd, + bool aNotSafeToBreak, + bool aFrameCanContinueTextRun, + bool aCanRollBackBeforeFrame, + nsHTMLReflowMetrics& aMetrics, + nsReflowStatus& aStatus, + bool* aOptionalBreakAfterFits); + + void PlaceFrame(PerFrameData* pfd, + nsHTMLReflowMetrics& aMetrics); + + void BlockDirAlignFrames(PerSpanData* psd); + + void PlaceStartEndFrames(PerSpanData* psd, + nscoord aDistanceFromStart, + nscoord aLineBSize); + + void RelativePositionFrames(PerSpanData* psd, nsOverflowAreas& aOverflowAreas); + + bool TrimTrailingWhiteSpaceIn(PerSpanData* psd, nscoord* aDeltaISize); + + void ComputeJustificationWeights(PerSpanData* psd, int32_t* numSpaces, int32_t* numLetters); + + struct FrameJustificationState { + int32_t mTotalNumSpaces; + int32_t mTotalNumLetters; + nscoord mTotalWidthForSpaces; + nscoord mTotalWidthForLetters; + int32_t mNumSpacesProcessed; + int32_t mNumLettersProcessed; + nscoord mWidthForSpacesProcessed; + nscoord mWidthForLettersProcessed; + }; + + // Apply justification. The return value is the amount by which the width of + // the span corresponding to aPSD got increased due to justification. + nscoord ApplyFrameJustification(PerSpanData* aPSD, + FrameJustificationState* aState); + + +#ifdef DEBUG + void DumpPerSpanData(PerSpanData* psd, int32_t aIndent); +#endif +}; + +#endif /* nsLineLayout_h___ */