diff options
Diffstat (limited to 'layout/generic/nsLineLayout.cpp')
-rw-r--r-- | layout/generic/nsLineLayout.cpp | 3495 |
1 files changed, 3495 insertions, 0 deletions
diff --git a/layout/generic/nsLineLayout.cpp b/layout/generic/nsLineLayout.cpp new file mode 100644 index 0000000000..03aa6e87b3 --- /dev/null +++ b/layout/generic/nsLineLayout.cpp @@ -0,0 +1,3495 @@ +/* -*- 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/. */ + +/* state and methods used while laying out a single line of a block frame */ + +#include "nsLineLayout.h" + +#include "mozilla/ComputedStyle.h" +#include "mozilla/SVGTextFrame.h" + +#include "LayoutLogging.h" +#include "nsBlockFrame.h" +#include "nsFontMetrics.h" +#include "nsStyleConsts.h" +#include "nsContainerFrame.h" +#include "nsFloatManager.h" +#include "nsPresContext.h" +#include "nsGkAtoms.h" +#include "nsIContent.h" +#include "nsLayoutUtils.h" +#include "nsTextFrame.h" +#include "nsStyleStructInlines.h" +#include "nsBidiPresUtils.h" +#include "nsRubyFrame.h" +#include "nsRubyTextFrame.h" +#include "RubyUtils.h" +#include <algorithm> + +#ifdef DEBUG +# undef NOISY_INLINEDIR_ALIGN +# undef NOISY_BLOCKDIR_ALIGN +# undef NOISY_REFLOW +# undef REALLY_NOISY_REFLOW +# undef NOISY_PUSHING +# undef REALLY_NOISY_PUSHING +# undef NOISY_CAN_PLACE_FRAME +# undef NOISY_TRIM +# undef REALLY_NOISY_TRIM +#endif + +using namespace mozilla; + +//---------------------------------------------------------------------- + +nsLineLayout::nsLineLayout(nsPresContext* aPresContext, + nsFloatManager* aFloatManager, + const ReflowInput& aLineContainerRI, + const nsLineList::iterator* aLine, + nsLineLayout* aBaseLineLayout) + : mPresContext(aPresContext), + mFloatManager(aFloatManager), + mLineContainerRI(aLineContainerRI), + mBaseLineLayout(aBaseLineLayout), + mLastOptionalBreakFrame(nullptr), + mForceBreakFrame(nullptr), + mLastOptionalBreakPriority(gfxBreakPriority::eNoBreak), + mLastOptionalBreakFrameOffset(-1), + mForceBreakFrameOffset(-1), + mMinLineBSize(0), + mTextIndent(0), + mMaxStartBoxBSize(0), + mMaxEndBoxBSize(0), + mFinalLineBSize(0), + mFirstLetterStyleOK(false), + mIsTopOfPage(false), + mImpactedByFloats(false), + mLastFloatWasLetterFrame(false), + mLineIsEmpty(false), + mLineEndsInBR(false), + mNeedBackup(false), + mInFirstLine(false), + mGotLineBox(false), + mInFirstLetter(false), + mHasMarker(false), + mDirtyNextLine(false), + mLineAtStart(false), + mHasRuby(false), + mSuppressLineWrap(LineContainerFrame()->IsInSVGTextSubtree()), + mUsedOverflowWrap(false) +#ifdef DEBUG + , + mSpansAllocated(0), + mSpansFreed(0), + mFramesAllocated(0), + mFramesFreed(0) +#endif +{ + NS_ASSERTION(aFloatManager || LineContainerFrame()->IsLetterFrame(), + "float manager should be present"); + MOZ_ASSERT( + !!mBaseLineLayout == LineContainerFrame()->IsRubyTextContainerFrame(), + "Only ruby text container frames have a different base line layout"); + MOZ_COUNT_CTOR(nsLineLayout); + + // Stash away some style data that we need + nsBlockFrame* blockFrame = do_QueryFrame(LineContainerFrame()); + mStyleText = blockFrame ? blockFrame->StyleTextForLineLayout() + : LineContainerFrame()->StyleText(); + + mLineNumber = 0; + mTotalPlacedFrames = 0; + mBStartEdge = 0; + mTrimmableISize = 0; + + mInflationMinFontSize = + nsLayoutUtils::InflationMinFontSizeFor(LineContainerFrame()); + + // Instead of always pre-initializing the free-lists for frames and + // spans, we do it on demand so that situations that only use a few + // frames and spans won't waste a lot of time in unneeded + // initialization. + mFrameFreeList = nullptr; + mSpanFreeList = nullptr; + + mCurrentSpan = mRootSpan = nullptr; + mSpanDepth = 0; + + if (aLine) { + mGotLineBox = true; + mLineBox = *aLine; + } +} + +nsLineLayout::~nsLineLayout() { + MOZ_COUNT_DTOR(nsLineLayout); + + NS_ASSERTION(nullptr == mRootSpan, "bad line-layout user"); +} + +// Find out if the frame has a non-null prev-in-flow, i.e., whether it +// is a continuation. +inline bool HasPrevInFlow(nsIFrame* aFrame) { + nsIFrame* prevInFlow = aFrame->GetPrevInFlow(); + return prevInFlow != nullptr; +} + +void nsLineLayout::BeginLineReflow(nscoord aICoord, nscoord aBCoord, + nscoord aISize, nscoord aBSize, + bool aImpactedByFloats, bool aIsTopOfPage, + WritingMode aWritingMode, + const nsSize& aContainerSize, + nscoord aInset) { + NS_ASSERTION(nullptr == mRootSpan, "bad linelayout user"); + LAYOUT_WARN_IF_FALSE(aISize != NS_UNCONSTRAINEDSIZE, + "have unconstrained width; this should only result from " + "very large sizes, not attempts at intrinsic width " + "calculation"); +#ifdef DEBUG + if ((aISize != NS_UNCONSTRAINEDSIZE) && ABSURD_SIZE(aISize) && + !LineContainerFrame()->GetParent()->IsAbsurdSizeAssertSuppressed()) { + LineContainerFrame()->ListTag(stdout); + printf(": Init: bad caller: width WAS %d(0x%x)\n", aISize, aISize); + } + if ((aBSize != NS_UNCONSTRAINEDSIZE) && ABSURD_SIZE(aBSize) && + !LineContainerFrame()->GetParent()->IsAbsurdSizeAssertSuppressed()) { + LineContainerFrame()->ListTag(stdout); + printf(": Init: bad caller: height WAS %d(0x%x)\n", aBSize, aBSize); + } +#endif +#ifdef NOISY_REFLOW + LineContainerFrame()->ListTag(stdout); + printf(": BeginLineReflow: %d,%d,%d,%d impacted=%s %s\n", aICoord, aBCoord, + aISize, aBSize, aImpactedByFloats ? "true" : "false", + aIsTopOfPage ? "top-of-page" : ""); +#endif +#ifdef DEBUG + mSpansAllocated = mSpansFreed = mFramesAllocated = mFramesFreed = 0; +#endif + + mFirstLetterStyleOK = false; + mIsTopOfPage = aIsTopOfPage; + mImpactedByFloats = aImpactedByFloats; + mTotalPlacedFrames = 0; + if (!mBaseLineLayout) { + mLineIsEmpty = true; + mLineAtStart = true; + } else { + mLineIsEmpty = false; + mLineAtStart = false; + } + mLineEndsInBR = false; + mSpanDepth = 0; + mMaxStartBoxBSize = mMaxEndBoxBSize = 0; + + if (mGotLineBox) { + mLineBox->ClearHasMarker(); + } + + PerSpanData* psd = NewPerSpanData(); + mCurrentSpan = mRootSpan = psd; + psd->mReflowInput = &mLineContainerRI; + psd->mIStart = aICoord; + psd->mICoord = aICoord; + psd->mIEnd = aICoord + aISize; + // Set up inset to be used for text-wrap:balance implementation, but only if + // the available size is greater than inset. + psd->mInset = aISize > aInset ? aInset : 0; + mContainerSize = aContainerSize; + + mBStartEdge = aBCoord; + + psd->mNoWrap = !mStyleText->WhiteSpaceCanWrapStyle() || mSuppressLineWrap; + psd->mWritingMode = aWritingMode; + + // Determine if this is the first line of the block (or first after a hard + // line-break, if `each-line` is in effect). + nsIFrame* containerFrame = LineContainerFrame(); + if (!containerFrame->IsRubyTextContainerFrame()) { + bool isFirstLineOrAfterHardBreak = [&] { + if (mLineNumber > 0) { + return mStyleText->mTextIndent.each_line && GetLine() && + !GetLine()->prev()->IsLineWrapped(); + } + if (nsBlockFrame* prevBlock = + do_QueryFrame(containerFrame->GetPrevInFlow())) { + return mStyleText->mTextIndent.each_line && + (prevBlock->Lines().empty() || + !prevBlock->LinesEnd().prev()->IsLineWrapped()); + } + return true; + }(); + + // Resolve and apply the text-indent value if this line requires it. + // The `hanging` option inverts which lines are to be indented. + if (isFirstLineOrAfterHardBreak != mStyleText->mTextIndent.hanging) { + nscoord pctBasis = mLineContainerRI.ComputedISize(); + mTextIndent = mStyleText->mTextIndent.length.Resolve(pctBasis); + psd->mICoord += mTextIndent; + } + } + + PerFrameData* pfd = NewPerFrameData(containerFrame); + pfd->mAscent = 0; + pfd->mSpan = psd; + psd->mFrame = pfd; + if (containerFrame->IsRubyTextContainerFrame()) { + // Ruby text container won't be reflowed via ReflowFrame, hence the + // relative positioning information should be recorded here. + MOZ_ASSERT(mBaseLineLayout != this); + pfd->mIsRelativelyOrStickyPos = + mLineContainerRI.mStyleDisplay->IsRelativelyOrStickyPositionedStyle(); + if (pfd->mIsRelativelyOrStickyPos) { + MOZ_ASSERT(mLineContainerRI.GetWritingMode() == pfd->mWritingMode, + "mLineContainerRI.frame == frame, " + "hence they should have identical writing mode"); + pfd->mOffsets = + mLineContainerRI.ComputedLogicalOffsets(pfd->mWritingMode); + } + } +} + +bool nsLineLayout::EndLineReflow() { +#ifdef NOISY_REFLOW + LineContainerFrame()->ListTag(stdout); + printf(": EndLineReflow: width=%d\n", + mRootSpan->mICoord - mRootSpan->mIStart); +#endif + + NS_ASSERTION(!mBaseLineLayout || + (!mSpansAllocated && !mSpansFreed && !mSpanFreeList && + !mFramesAllocated && !mFramesFreed && !mFrameFreeList), + "Allocated frames or spans on non-base line layout?"); + MOZ_ASSERT(mRootSpan == mCurrentSpan); + + UnlinkFrame(mRootSpan->mFrame); + mCurrentSpan = mRootSpan = nullptr; + + NS_ASSERTION(mSpansAllocated == mSpansFreed, "leak"); + NS_ASSERTION(mFramesAllocated == mFramesFreed, "leak"); + +#if 0 + static int32_t maxSpansAllocated = NS_LINELAYOUT_NUM_SPANS; + static int32_t maxFramesAllocated = NS_LINELAYOUT_NUM_FRAMES; + if (mSpansAllocated > maxSpansAllocated) { + printf("XXX: saw a line with %d spans\n", mSpansAllocated); + maxSpansAllocated = mSpansAllocated; + } + if (mFramesAllocated > maxFramesAllocated) { + printf("XXX: saw a line with %d frames\n", mFramesAllocated); + maxFramesAllocated = mFramesAllocated; + } +#endif + + return mUsedOverflowWrap; +} + +// XXX swtich to a single mAvailLineWidth that we adjust as each frame +// on the line is placed. Each span can still have a per-span mICoord that +// tracks where a child frame is going in its span; they don't need a +// per-span mIStart? + +void nsLineLayout::UpdateBand(WritingMode aWM, + const LogicalRect& aNewAvailSpace, + nsIFrame* aFloatFrame) { + WritingMode lineWM = mRootSpan->mWritingMode; + // need to convert to our writing mode, because we might have a different + // mode from the caller due to dir: auto + LogicalRect availSpace = + aNewAvailSpace.ConvertTo(lineWM, aWM, ContainerSize()); +#ifdef REALLY_NOISY_REFLOW + printf( + "nsLL::UpdateBand %d, %d, %d, %d, (converted to %d, %d, %d, %d); " + "frame=%p\n will set mImpacted to true\n", + aNewAvailSpace.IStart(aWM), aNewAvailSpace.BStart(aWM), + aNewAvailSpace.ISize(aWM), aNewAvailSpace.BSize(aWM), + availSpace.IStart(lineWM), availSpace.BStart(lineWM), + availSpace.ISize(lineWM), availSpace.BSize(lineWM), aFloatFrame); +#endif +#ifdef DEBUG + if ((availSpace.ISize(lineWM) != NS_UNCONSTRAINEDSIZE) && + ABSURD_SIZE(availSpace.ISize(lineWM)) && + !LineContainerFrame()->GetParent()->IsAbsurdSizeAssertSuppressed()) { + LineContainerFrame()->ListTag(stdout); + printf(": UpdateBand: bad caller: ISize WAS %d(0x%x)\n", + availSpace.ISize(lineWM), availSpace.ISize(lineWM)); + } + if ((availSpace.BSize(lineWM) != NS_UNCONSTRAINEDSIZE) && + ABSURD_SIZE(availSpace.BSize(lineWM)) && + !LineContainerFrame()->GetParent()->IsAbsurdSizeAssertSuppressed()) { + LineContainerFrame()->ListTag(stdout); + printf(": UpdateBand: bad caller: BSize WAS %d(0x%x)\n", + availSpace.BSize(lineWM), availSpace.BSize(lineWM)); + } +#endif + + // Compute the difference between last times width and the new width + NS_WARNING_ASSERTION( + mRootSpan->mIEnd != NS_UNCONSTRAINEDSIZE && + availSpace.ISize(lineWM) != NS_UNCONSTRAINEDSIZE, + "have unconstrained inline size; this should only result from very large " + "sizes, not attempts at intrinsic width calculation"); + // The root span's mIStart moves to aICoord + nscoord deltaICoord = availSpace.IStart(lineWM) - mRootSpan->mIStart; + // The inline size of all spans changes by this much (the root span's + // mIEnd moves to aICoord + aISize, its new inline size is aISize) + nscoord deltaISize = + availSpace.ISize(lineWM) - (mRootSpan->mIEnd - mRootSpan->mIStart); +#ifdef NOISY_REFLOW + LineContainerFrame()->ListTag(stdout); + printf(": UpdateBand: %d,%d,%d,%d deltaISize=%d deltaICoord=%d\n", + availSpace.IStart(lineWM), availSpace.BStart(lineWM), + availSpace.ISize(lineWM), availSpace.BSize(lineWM), deltaISize, + deltaICoord); +#endif + + // Update the root span position + mRootSpan->mIStart += deltaICoord; + mRootSpan->mIEnd += deltaICoord; + mRootSpan->mICoord += deltaICoord; + + // Now update the right edges of the open spans to account for any + // change in available space width + for (PerSpanData* psd = mCurrentSpan; psd; psd = psd->mParent) { + psd->mIEnd += deltaISize; + psd->mContainsFloat = true; +#ifdef NOISY_REFLOW + printf(" span %p: oldIEnd=%d newIEnd=%d\n", psd, psd->mIEnd - deltaISize, + psd->mIEnd); +#endif + } + NS_ASSERTION(mRootSpan->mContainsFloat && + mRootSpan->mIStart == availSpace.IStart(lineWM) && + mRootSpan->mIEnd == availSpace.IEnd(lineWM), + "root span was updated incorrectly?"); + + // Update frame bounds + // Note: Only adjust the outermost frames (the ones that are direct + // children of the block), not the ones in the child spans. The reason + // is simple: the frames in the spans have coordinates local to their + // parent therefore they are moved when their parent span is moved. + if (deltaICoord != 0) { + for (PerFrameData* pfd = mRootSpan->mFirstFrame; pfd; pfd = pfd->mNext) { + pfd->mBounds.IStart(lineWM) += deltaICoord; + } + } + + mBStartEdge = availSpace.BStart(lineWM); + mImpactedByFloats = true; + + mLastFloatWasLetterFrame = aFloatFrame->IsLetterFrame(); +} + +nsLineLayout::PerSpanData* nsLineLayout::NewPerSpanData() { + nsLineLayout* outerLineLayout = GetOutermostLineLayout(); + PerSpanData* psd = outerLineLayout->mSpanFreeList; + if (!psd) { + void* mem = outerLineLayout->mArena.Allocate(sizeof(PerSpanData)); + psd = reinterpret_cast<PerSpanData*>(mem); + } else { + outerLineLayout->mSpanFreeList = psd->mNextFreeSpan; + } + psd->mParent = nullptr; + psd->mFrame = nullptr; + psd->mFirstFrame = nullptr; + psd->mLastFrame = nullptr; + psd->mContainsFloat = false; + psd->mHasNonemptyContent = false; + +#ifdef DEBUG + outerLineLayout->mSpansAllocated++; +#endif + return psd; +} + +void nsLineLayout::BeginSpan(nsIFrame* aFrame, + const ReflowInput* aSpanReflowInput, + nscoord aIStart, nscoord aIEnd, + nscoord* aBaseline) { + NS_ASSERTION(aIEnd != NS_UNCONSTRAINEDSIZE, + "should no longer be using unconstrained sizes"); +#ifdef NOISY_REFLOW + nsIFrame::IndentBy(stdout, mSpanDepth + 1); + aFrame->ListTag(stdout); + printf(": BeginSpan leftEdge=%d rightEdge=%d\n", aIStart, aIEnd); +#endif + + PerSpanData* psd = NewPerSpanData(); + // Link up span frame's pfd to point to its child span data + PerFrameData* pfd = mCurrentSpan->mLastFrame; + NS_ASSERTION(pfd->mFrame == aFrame, "huh?"); + pfd->mSpan = psd; + + // Init new span + psd->mFrame = pfd; + psd->mParent = mCurrentSpan; + psd->mReflowInput = aSpanReflowInput; + psd->mIStart = aIStart; + psd->mICoord = aIStart; + psd->mIEnd = aIEnd; + psd->mInset = mCurrentSpan->mInset; + psd->mBaseline = aBaseline; + + nsIFrame* frame = aSpanReflowInput->mFrame; + psd->mNoWrap = !frame->StyleText()->WhiteSpaceCanWrap(frame) || + mSuppressLineWrap || frame->Style()->ShouldSuppressLineBreak(); + psd->mWritingMode = aSpanReflowInput->GetWritingMode(); + + // Switch to new span + mCurrentSpan = psd; + mSpanDepth++; +} + +nscoord nsLineLayout::EndSpan(nsIFrame* aFrame) { + NS_ASSERTION(mSpanDepth > 0, "end-span without begin-span"); +#ifdef NOISY_REFLOW + nsIFrame::IndentBy(stdout, mSpanDepth); + aFrame->ListTag(stdout); + printf(": EndSpan width=%d\n", mCurrentSpan->mICoord - mCurrentSpan->mIStart); +#endif + PerSpanData* psd = mCurrentSpan; + MOZ_ASSERT(psd->mParent, "We never call this on the root"); + + if (psd->mNoWrap && !psd->mParent->mNoWrap) { + FlushNoWrapFloats(); + } + + nscoord iSizeResult = psd->mLastFrame ? (psd->mICoord - psd->mIStart) : 0; + + mSpanDepth--; + mCurrentSpan->mReflowInput = nullptr; // no longer valid so null it out! + mCurrentSpan = mCurrentSpan->mParent; + return iSizeResult; +} + +void nsLineLayout::AttachFrameToBaseLineLayout(PerFrameData* aFrame) { + MOZ_ASSERT(mBaseLineLayout, + "This method must not be called in a base line layout."); + + PerFrameData* baseFrame = mBaseLineLayout->LastFrame(); + MOZ_ASSERT(aFrame && baseFrame); + MOZ_ASSERT(!aFrame->mIsLinkedToBase, + "The frame must not have been linked with the base"); +#ifdef DEBUG + LayoutFrameType baseType = baseFrame->mFrame->Type(); + LayoutFrameType annotationType = aFrame->mFrame->Type(); + MOZ_ASSERT((baseType == LayoutFrameType::RubyBaseContainer && + annotationType == LayoutFrameType::RubyTextContainer) || + (baseType == LayoutFrameType::RubyBase && + annotationType == LayoutFrameType::RubyText)); +#endif + + aFrame->mNextAnnotation = baseFrame->mNextAnnotation; + baseFrame->mNextAnnotation = aFrame; + aFrame->mIsLinkedToBase = true; +} + +int32_t nsLineLayout::GetCurrentSpanCount() const { + NS_ASSERTION(mCurrentSpan == mRootSpan, "bad linelayout user"); + int32_t count = 0; + PerFrameData* pfd = mRootSpan->mFirstFrame; + while (nullptr != pfd) { + count++; + pfd = pfd->mNext; + } + return count; +} + +void nsLineLayout::SplitLineTo(int32_t aNewCount) { + NS_ASSERTION(mCurrentSpan == mRootSpan, "bad linelayout user"); + +#ifdef REALLY_NOISY_PUSHING + printf("SplitLineTo %d (current count=%d); before:\n", aNewCount, + GetCurrentSpanCount()); + DumpPerSpanData(mRootSpan, 1); +#endif + PerSpanData* psd = mRootSpan; + PerFrameData* pfd = psd->mFirstFrame; + while (nullptr != pfd) { + if (--aNewCount == 0) { + // Truncate list at pfd (we keep pfd, but anything following is freed) + PerFrameData* next = pfd->mNext; + pfd->mNext = nullptr; + psd->mLastFrame = pfd; + + // Now unlink all of the frames following pfd + UnlinkFrame(next); + break; + } + pfd = pfd->mNext; + } +#ifdef NOISY_PUSHING + printf("SplitLineTo %d (current count=%d); after:\n", aNewCount, + GetCurrentSpanCount()); + DumpPerSpanData(mRootSpan, 1); +#endif +} + +void nsLineLayout::PushFrame(nsIFrame* aFrame) { + PerSpanData* psd = mCurrentSpan; + NS_ASSERTION(psd->mLastFrame->mFrame == aFrame, "pushing non-last frame"); + +#ifdef REALLY_NOISY_PUSHING + nsIFrame::IndentBy(stdout, mSpanDepth); + printf("PushFrame %p, before:\n", psd); + DumpPerSpanData(psd, 1); +#endif + + // Take the last frame off of the span's frame list + PerFrameData* pfd = psd->mLastFrame; + if (pfd == psd->mFirstFrame) { + // We are pushing away the only frame...empty the list + psd->mFirstFrame = nullptr; + psd->mLastFrame = nullptr; + } else { + PerFrameData* prevFrame = pfd->mPrev; + prevFrame->mNext = nullptr; + psd->mLastFrame = prevFrame; + } + + // Now unlink the frame + MOZ_ASSERT(!pfd->mNext); + UnlinkFrame(pfd); +#ifdef NOISY_PUSHING + nsIFrame::IndentBy(stdout, mSpanDepth); + printf("PushFrame: %p after:\n", psd); + DumpPerSpanData(psd, 1); +#endif +} + +void nsLineLayout::UnlinkFrame(PerFrameData* pfd) { + while (nullptr != pfd) { + PerFrameData* next = pfd->mNext; + if (pfd->mIsLinkedToBase) { + // This frame is linked to a ruby base, and should not be freed + // now. Just unlink it from the span. It will be freed when its + // base frame gets unlinked. + pfd->mNext = pfd->mPrev = nullptr; + pfd = next; + continue; + } + + // It is a ruby base frame. If there are any annotations + // linked to this frame, free them first. + PerFrameData* annotationPFD = pfd->mNextAnnotation; + while (annotationPFD) { + PerFrameData* nextAnnotation = annotationPFD->mNextAnnotation; + MOZ_ASSERT( + annotationPFD->mNext == nullptr && annotationPFD->mPrev == nullptr, + "PFD in annotations should have been unlinked."); + FreeFrame(annotationPFD); + annotationPFD = nextAnnotation; + } + + FreeFrame(pfd); + pfd = next; + } +} + +void nsLineLayout::FreeFrame(PerFrameData* pfd) { + if (nullptr != pfd->mSpan) { + FreeSpan(pfd->mSpan); + } + nsLineLayout* outerLineLayout = GetOutermostLineLayout(); + pfd->mNext = outerLineLayout->mFrameFreeList; + outerLineLayout->mFrameFreeList = pfd; +#ifdef DEBUG + outerLineLayout->mFramesFreed++; +#endif +} + +void nsLineLayout::FreeSpan(PerSpanData* psd) { + // Unlink its frames + UnlinkFrame(psd->mFirstFrame); + + nsLineLayout* outerLineLayout = GetOutermostLineLayout(); + // Now put the span on the free list since it's free too + psd->mNextFreeSpan = outerLineLayout->mSpanFreeList; + outerLineLayout->mSpanFreeList = psd; +#ifdef DEBUG + outerLineLayout->mSpansFreed++; +#endif +} + +bool nsLineLayout::IsZeroBSize() { + PerSpanData* psd = mCurrentSpan; + PerFrameData* pfd = psd->mFirstFrame; + while (nullptr != pfd) { + if (0 != pfd->mBounds.BSize(psd->mWritingMode)) { + return false; + } + pfd = pfd->mNext; + } + return true; +} + +nsLineLayout::PerFrameData* nsLineLayout::NewPerFrameData(nsIFrame* aFrame) { + nsLineLayout* outerLineLayout = GetOutermostLineLayout(); + PerFrameData* pfd = outerLineLayout->mFrameFreeList; + if (!pfd) { + void* mem = outerLineLayout->mArena.Allocate(sizeof(PerFrameData)); + pfd = reinterpret_cast<PerFrameData*>(mem); + } else { + outerLineLayout->mFrameFreeList = pfd->mNext; + } + pfd->mSpan = nullptr; + pfd->mNext = nullptr; + pfd->mPrev = nullptr; + pfd->mNextAnnotation = nullptr; + pfd->mFrame = aFrame; + + // all flags default to false + pfd->mIsRelativelyOrStickyPos = false; + pfd->mIsTextFrame = false; + pfd->mIsNonEmptyTextFrame = false; + pfd->mIsNonWhitespaceTextFrame = false; + pfd->mIsLetterFrame = false; + pfd->mRecomputeOverflow = false; + pfd->mIsMarker = false; + pfd->mSkipWhenTrimmingWhitespace = false; + pfd->mIsEmpty = false; + pfd->mIsPlaceholder = false; + pfd->mIsLinkedToBase = false; + + pfd->mWritingMode = aFrame->GetWritingMode(); + WritingMode lineWM = mRootSpan->mWritingMode; + pfd->mBounds = LogicalRect(lineWM); + pfd->mOverflowAreas.Clear(); + pfd->mMargin = LogicalMargin(lineWM); + pfd->mBorderPadding = LogicalMargin(lineWM); + pfd->mOffsets = LogicalMargin(pfd->mWritingMode); + + pfd->mJustificationInfo = JustificationInfo(); + pfd->mJustificationAssignment = JustificationAssignment(); + +#ifdef DEBUG + pfd->mBlockDirAlign = 0xFF; + outerLineLayout->mFramesAllocated++; +#endif + return pfd; +} + +bool nsLineLayout::LineIsBreakable() const { + // XXX mTotalPlacedFrames should go away and we should just use + // mLineIsEmpty here instead + if ((0 != mTotalPlacedFrames) || mImpactedByFloats) { + return true; + } + return false; +} + +// Checks all four sides for percentage units. This means it should +// only be used for things (margin, padding) where percentages on top +// and bottom depend on the *width* just like percentages on left and +// right. +template <typename T> +static bool HasPercentageUnitSide(const StyleRect<T>& aSides) { + return aSides.Any([](const auto& aLength) { return aLength.HasPercent(); }); +} + +static bool IsPercentageAware(const nsIFrame* aFrame, WritingMode aWM) { + NS_ASSERTION(aFrame, "null frame is not allowed"); + + LayoutFrameType fType = aFrame->Type(); + if (fType == LayoutFrameType::Text) { + // None of these things can ever be true for text frames. + return false; + } + + // Some of these things don't apply to non-replaced inline frames + // (that is, fType == LayoutFrameType::Inline), but we won't bother making + // things unnecessarily complicated, since they'll probably be set + // quite rarely. + + const nsStyleMargin* margin = aFrame->StyleMargin(); + if (HasPercentageUnitSide(margin->mMargin)) { + return true; + } + + const nsStylePadding* padding = aFrame->StylePadding(); + if (HasPercentageUnitSide(padding->mPadding)) { + return true; + } + + // Note that borders can't be aware of percentages + + const nsStylePosition* pos = aFrame->StylePosition(); + + if ((pos->ISizeDependsOnContainer(aWM) && !pos->ISize(aWM).IsAuto()) || + pos->MaxISizeDependsOnContainer(aWM) || + pos->MinISizeDependsOnContainer(aWM) || + pos->mOffset.GetIStart(aWM).HasPercent() || + pos->mOffset.GetIEnd(aWM).HasPercent()) { + return true; + } + + if (pos->ISize(aWM).IsAuto()) { + // We need to check for frames that shrink-wrap when they're auto + // width. + const nsStyleDisplay* disp = aFrame->StyleDisplay(); + if ((disp->DisplayOutside() == StyleDisplayOutside::Inline && + (disp->DisplayInside() == StyleDisplayInside::FlowRoot || + disp->DisplayInside() == StyleDisplayInside::Table)) || + fType == LayoutFrameType::HTMLButtonControl || + fType == LayoutFrameType::GfxButtonControl || + fType == LayoutFrameType::FieldSet || + fType == LayoutFrameType::ComboboxDisplay) { + return true; + } + + // Per CSS 2.1, section 10.3.2: + // If 'height' and 'width' both have computed values of 'auto' and + // the element has an intrinsic ratio but no intrinsic height or + // width and the containing block's width does not itself depend + // on the replaced element's width, then the used value of 'width' + // is calculated from the constraint equation used for + // block-level, non-replaced elements in normal flow. + nsIFrame* f = const_cast<nsIFrame*>(aFrame); + if (f->GetAspectRatio() && + // Some percents are treated like 'auto', so check != coord + !pos->BSize(aWM).ConvertsToLength()) { + const IntrinsicSize& intrinsicSize = f->GetIntrinsicSize(); + if (!intrinsicSize.width && !intrinsicSize.height) { + return true; + } + } + } + + return false; +} + +void nsLineLayout::ReflowFrame(nsIFrame* aFrame, nsReflowStatus& aReflowStatus, + ReflowOutput* aMetrics, bool& aPushedFrame) { + // Initialize OUT parameter + aPushedFrame = false; + + PerFrameData* pfd = NewPerFrameData(aFrame); + PerSpanData* psd = mCurrentSpan; + psd->AppendFrame(pfd); + +#ifdef REALLY_NOISY_REFLOW + nsIFrame::IndentBy(stdout, mSpanDepth); + printf("%p: Begin ReflowFrame pfd=%p ", psd, pfd); + aFrame->ListTag(stdout); + printf("\n"); +#endif + + if (mCurrentSpan == mRootSpan) { + pfd->mFrame->RemoveProperty(nsIFrame::LineBaselineOffset()); + } else { +#ifdef DEBUG + bool hasLineOffset; + pfd->mFrame->GetProperty(nsIFrame::LineBaselineOffset(), &hasLineOffset); + NS_ASSERTION(!hasLineOffset, + "LineBaselineOffset was set but was not expected"); +#endif + } + + mJustificationInfo = JustificationInfo(); + + // Stash copies of some of the computed state away for later + // (block-direction alignment, for example) + WritingMode frameWM = pfd->mWritingMode; + WritingMode lineWM = mRootSpan->mWritingMode; + + // NOTE: While the inline direction coordinate remains relative to the + // parent span, the block direction coordinate is fixed at the top + // edge for the line. During VerticalAlignFrames we will repair this + // so that the block direction coordinate is properly set and relative + // to the appropriate span. + pfd->mBounds.IStart(lineWM) = psd->mICoord; + pfd->mBounds.BStart(lineWM) = mBStartEdge; + + // We want to guarantee that we always make progress when + // formatting. Therefore, if the object being placed on the line is + // too big for the line, but it is the only thing on the line and is not + // impacted by a float, then we go ahead and place it anyway. (If the line + // is impacted by one or more floats, then it is safe to break because + // we can move the line down below float(s).) + // + // Capture this state *before* we reflow the frame in case it clears + // the state out. We need to know how to treat the current frame + // when breaking. + bool notSafeToBreak = LineIsEmpty() && !mImpactedByFloats; + + // Figure out whether we're talking about a textframe here + LayoutFrameType frameType = aFrame->Type(); + const bool isText = frameType == LayoutFrameType::Text; + + // Inline-ish and text-ish things don't compute their width; + // everything else does. We need to give them an available width that + // reflects the space left on the line. + LAYOUT_WARN_IF_FALSE(psd->mIEnd != NS_UNCONSTRAINEDSIZE, + "have unconstrained width; this should only result from " + "very large sizes, not attempts at intrinsic width " + "calculation"); + nscoord availableSpaceOnLine = psd->mIEnd - psd->mICoord - psd->mInset; + + // Setup reflow input for reflowing the frame + Maybe<ReflowInput> reflowInputHolder; + if (!isText) { + // Compute the available size for the frame. This available width + // includes room for the side margins. + // For now, set the available block-size to unconstrained always. + LogicalSize availSize = mLineContainerRI.ComputedSize(frameWM); + availSize.BSize(frameWM) = NS_UNCONSTRAINEDSIZE; + reflowInputHolder.emplace(mPresContext, *psd->mReflowInput, aFrame, + availSize); + ReflowInput& reflowInput = *reflowInputHolder; + reflowInput.mLineLayout = this; + reflowInput.mFlags.mIsTopOfPage = mIsTopOfPage; + if (reflowInput.ComputedISize() == NS_UNCONSTRAINEDSIZE) { + reflowInput.SetAvailableISize(availableSpaceOnLine); + } + pfd->mMargin = reflowInput.ComputedLogicalMargin(lineWM); + pfd->mBorderPadding = reflowInput.ComputedLogicalBorderPadding(lineWM); + pfd->mIsRelativelyOrStickyPos = + reflowInput.mStyleDisplay->IsRelativelyOrStickyPositionedStyle(); + if (pfd->mIsRelativelyOrStickyPos) { + pfd->mOffsets = reflowInput.ComputedLogicalOffsets(frameWM); + } + + // Calculate whether the the frame should have a start margin and + // subtract the margin from the available width if necessary. + // The margin will be applied to the starting inline coordinates of + // the frame in CanPlaceFrame() after reflowing the frame. + AllowForStartMargin(pfd, reflowInput); + } + // if isText(), no need to propagate NS_FRAME_IS_DIRTY from the parent, + // because reflow doesn't look at the dirty bits on the frame being reflowed. + + // See if this frame depends on the inline-size of its containing block. + // If so, disable resize reflow optimizations for the line. (Note that, + // to be conservative, we do this if we *try* to fit a frame on a + // line, even if we don't succeed.) (Note also that we can only make + // this IsPercentageAware check *after* we've constructed our + // ReflowInput, because that construction may be what forces aFrame + // to lazily initialize its (possibly-percent-valued) intrinsic size.) + if (mGotLineBox && IsPercentageAware(aFrame, lineWM)) { + mLineBox->DisableResizeReflowOptimization(); + } + + // Note that we don't bother positioning the frame yet, because we're probably + // going to end up moving it when we do the block-direction alignment. + + // Adjust float manager coordinate system for the frame. + ReflowOutput reflowOutput(lineWM); +#ifdef DEBUG + reflowOutput.ISize(lineWM) = nscoord(0xdeadbeef); + reflowOutput.BSize(lineWM) = nscoord(0xdeadbeef); +#endif + nscoord tI = pfd->mBounds.LineLeft(lineWM, ContainerSize()); + nscoord tB = pfd->mBounds.BStart(lineWM); + mFloatManager->Translate(tI, tB); + + int32_t savedOptionalBreakOffset; + gfxBreakPriority savedOptionalBreakPriority; + nsIFrame* savedOptionalBreakFrame = GetLastOptionalBreakPosition( + &savedOptionalBreakOffset, &savedOptionalBreakPriority); + + if (!isText) { + aFrame->Reflow(mPresContext, reflowOutput, *reflowInputHolder, + aReflowStatus); + } else { + static_cast<nsTextFrame*>(aFrame)->ReflowText( + *this, availableSpaceOnLine, + psd->mReflowInput->mRenderingContext->GetDrawTarget(), reflowOutput, + aReflowStatus); + } + + pfd->mJustificationInfo = mJustificationInfo; + mJustificationInfo = JustificationInfo(); + + // See if the frame is a placeholderFrame and if it is process + // the float. At the same time, check if the frame has any non-collapsed-away + // content. + bool placedFloat = false; + bool isEmpty; + if (frameType == LayoutFrameType::None) { + isEmpty = pfd->mFrame->IsEmpty(); + } else if (LayoutFrameType::Placeholder == frameType) { + isEmpty = true; + pfd->mIsPlaceholder = true; + pfd->mSkipWhenTrimmingWhitespace = true; + nsIFrame* outOfFlowFrame = nsLayoutUtils::GetFloatFromPlaceholder(aFrame); + if (outOfFlowFrame) { + if (psd->mNoWrap && + // We can always place floats in an empty line. + !LineIsEmpty() && + // We always place floating letter frames. This kinda sucks. They'd + // usually fall into the LineIsEmpty() check anyway, except when + // there's something like a ::marker before or what not. We actually + // need to place them now, because they're pretty nasty and they + // create continuations that are in flow and not a kid of the + // previous continuation's parent. We don't want the deferred reflow + // of the letter frame to kill a continuation after we've stored it + // in the line layout data structures. See bug 1490281 to fix the + // underlying issue. When that's fixed this check should be removed. + !outOfFlowFrame->IsLetterFrame() && + !GetOutermostLineLayout()->mBlockRS->mFlags.mCanHaveOverflowMarkers) { + // We'll do this at the next break opportunity. + RecordNoWrapFloat(outOfFlowFrame); + } else { + placedFloat = TryToPlaceFloat(outOfFlowFrame); + } + } + } else if (isText) { + // Note non-empty text-frames for inline frame compatibility hackery + pfd->mIsTextFrame = true; + auto* textFrame = static_cast<nsTextFrame*>(pfd->mFrame); + isEmpty = !textFrame->HasNoncollapsedCharacters(); + if (!isEmpty) { + pfd->mIsNonEmptyTextFrame = true; + pfd->mIsNonWhitespaceTextFrame = + !textFrame->GetContent()->TextIsOnlyWhitespace(); + } + } else if (LayoutFrameType::Br == frameType) { + pfd->mSkipWhenTrimmingWhitespace = true; + isEmpty = false; + } else { + if (LayoutFrameType::Letter == frameType) { + pfd->mIsLetterFrame = true; + } + if (pfd->mSpan) { + isEmpty = !pfd->mSpan->mHasNonemptyContent && pfd->mFrame->IsSelfEmpty(); + } else { + isEmpty = pfd->mFrame->IsEmpty(); + } + } + pfd->mIsEmpty = isEmpty; + + mFloatManager->Translate(-tI, -tB); + + NS_ASSERTION(reflowOutput.ISize(lineWM) >= 0, "bad inline size"); + NS_ASSERTION(reflowOutput.BSize(lineWM) >= 0, "bad block size"); + if (reflowOutput.ISize(lineWM) < 0) { + reflowOutput.ISize(lineWM) = 0; + } + if (reflowOutput.BSize(lineWM) < 0) { + reflowOutput.BSize(lineWM) = 0; + } + +#ifdef DEBUG + // Note: break-before means ignore the reflow metrics since the + // frame will be reflowed another time. + if (!aReflowStatus.IsInlineBreakBefore()) { + if ((ABSURD_SIZE(reflowOutput.ISize(lineWM)) || + ABSURD_SIZE(reflowOutput.BSize(lineWM))) && + !LineContainerFrame()->GetParent()->IsAbsurdSizeAssertSuppressed()) { + printf("nsLineLayout: "); + aFrame->ListTag(stdout); + printf(" metrics=%d,%d!\n", reflowOutput.Width(), reflowOutput.Height()); + } + if ((reflowOutput.Width() == nscoord(0xdeadbeef)) || + (reflowOutput.Height() == nscoord(0xdeadbeef))) { + printf("nsLineLayout: "); + aFrame->ListTag(stdout); + printf(" didn't set w/h %d,%d!\n", reflowOutput.Width(), + reflowOutput.Height()); + } + } +#endif + + // Unlike with non-inline reflow, the overflow area here does *not* + // include the accumulation of the frame's bounds and its inline + // descendants' bounds. Nor does it include the outline area; it's + // just the union of the bounds of any absolute children. That is + // added in later by nsLineLayout::ReflowInlineFrames. + pfd->mOverflowAreas = reflowOutput.mOverflowAreas; + + pfd->mBounds.ISize(lineWM) = reflowOutput.ISize(lineWM); + pfd->mBounds.BSize(lineWM) = reflowOutput.BSize(lineWM); + + // Size the frame, but |RelativePositionFrames| will size the view. + aFrame->SetRect(lineWM, pfd->mBounds, ContainerSizeForSpan(psd)); + + // Tell the frame that we're done reflowing it + aFrame->DidReflow(mPresContext, isText ? nullptr : reflowInputHolder.ptr()); + + if (aMetrics) { + *aMetrics = reflowOutput; + } + + if (!aReflowStatus.IsInlineBreakBefore()) { + // If frame is complete and has a next-in-flow, we need to delete + // them now. Do not do this when a break-before is signaled because + // the frame is going to get reflowed again (and may end up wanting + // a next-in-flow where it ends up). + if (aReflowStatus.IsComplete()) { + if (nsIFrame* kidNextInFlow = aFrame->GetNextInFlow()) { + // Remove all of the childs next-in-flows. Make sure that we ask + // the right parent to do the removal (it's possible that the + // parent is not this because we are executing pullup code) + FrameDestroyContext context(aFrame->PresShell()); + kidNextInFlow->GetParent()->DeleteNextInFlowChild(context, + kidNextInFlow, true); + } + } + + // Check whether this frame breaks up text runs. All frames break up text + // runs (hence return false here) except for text frames and inline + // containers. + bool continuingTextRun = aFrame->CanContinueTextRun(); + + // Clear any residual mTrimmableISize if this isn't a text frame + if (!continuingTextRun && !pfd->mSkipWhenTrimmingWhitespace) { + mTrimmableISize = 0; + } + + // See if we can place the frame. If we can't fit it, then we + // return now. + bool optionalBreakAfterFits; + NS_ASSERTION(isText || !reflowInputHolder->mStyleDisplay->IsFloating( + reflowInputHolder->mFrame), + "How'd we get a floated inline frame? " + "The frame ctor should've dealt with this."); + if (CanPlaceFrame(pfd, notSafeToBreak, continuingTextRun, + savedOptionalBreakFrame != nullptr, reflowOutput, + aReflowStatus, &optionalBreakAfterFits)) { + if (!isEmpty) { + psd->mHasNonemptyContent = true; + mLineIsEmpty = false; + if (!pfd->mSpan) { + // nonempty leaf content has been placed + mLineAtStart = false; + } + if (LayoutFrameType::Ruby == frameType) { + mHasRuby = true; + SyncAnnotationBounds(pfd); + } + } + + // Place the frame, updating aBounds with the final size and + // location. Then apply the bottom+right margins (as + // appropriate) to the frame. + PlaceFrame(pfd, reflowOutput); + PerSpanData* span = pfd->mSpan; + if (span) { + // The frame we just finished reflowing is an inline + // container. It needs its child frames aligned in the block direction, + // so do most of it now. + VerticalAlignFrames(span); + } + + if (!continuingTextRun && !psd->mNoWrap) { + if (!LineIsEmpty() || placedFloat) { + // record soft break opportunity after this content that can't be + // part of a text run. This is not a text frame so we know + // that offset INT32_MAX means "after the content". + if ((!aFrame->IsPlaceholderFrame() || LineIsEmpty()) && + NotifyOptionalBreakPosition(aFrame, INT32_MAX, + optionalBreakAfterFits, + gfxBreakPriority::eNormalBreak)) { + // If this returns true then we are being told to actually break + // here. + aReflowStatus.SetInlineLineBreakAfter(); + } + } + } + } else { + PushFrame(aFrame); + aPushedFrame = true; + // Undo any saved break positions that the frame might have told us about, + // since we didn't end up placing it + RestoreSavedBreakPosition(savedOptionalBreakFrame, + savedOptionalBreakOffset, + savedOptionalBreakPriority); + } + } else { + PushFrame(aFrame); + aPushedFrame = true; + } + +#ifdef REALLY_NOISY_REFLOW + nsIFrame::IndentBy(stdout, mSpanDepth); + printf("End ReflowFrame "); + aFrame->ListTag(stdout); + printf(" status=%x\n", aReflowStatus); +#endif +} + +void nsLineLayout::AllowForStartMargin(PerFrameData* pfd, + ReflowInput& aReflowInput) { + NS_ASSERTION(!aReflowInput.mStyleDisplay->IsFloating(aReflowInput.mFrame), + "How'd we get a floated inline frame? " + "The frame ctor should've dealt with this."); + + WritingMode lineWM = mRootSpan->mWritingMode; + + // Only apply start-margin on the first-in flow for inline frames, + // and make sure to not apply it to any inline other than the first + // in an ib split. Note that the ib sibling (block-in-inline + // sibling) annotations only live on the first continuation, but we + // don't want to apply the start margin for later continuations + // anyway. For box-decoration-break:clone we apply the start-margin + // on all continuations. + if ((pfd->mFrame->GetPrevContinuation() || + pfd->mFrame->FrameIsNonFirstInIBSplit()) && + aReflowInput.mStyleBorder->mBoxDecorationBreak == + StyleBoxDecorationBreak::Slice) { + // Zero this out so that when we compute the max-element-width of + // the frame we will properly avoid adding in the starting margin. + pfd->mMargin.IStart(lineWM) = 0; + } else if (NS_UNCONSTRAINEDSIZE == aReflowInput.ComputedISize()) { + NS_WARNING_ASSERTION( + NS_UNCONSTRAINEDSIZE != aReflowInput.AvailableISize(), + "have unconstrained inline-size; this should only result from very " + "large sizes, not attempts at intrinsic inline-size calculation"); + // For inline-ish and text-ish things (which don't compute widths + // in the reflow input), adjust available inline-size to account + // for the start margin. The end margin will be accounted for when + // we finish flowing the frame. + WritingMode wm = aReflowInput.GetWritingMode(); + aReflowInput.SetAvailableISize( + aReflowInput.AvailableISize() - + pfd->mMargin.ConvertTo(wm, lineWM).IStart(wm)); + } +} + +nscoord nsLineLayout::GetCurrentFrameInlineDistanceFromBlock() { + PerSpanData* psd; + nscoord x = 0; + for (psd = mCurrentSpan; psd; psd = psd->mParent) { + x += psd->mICoord; + } + return x; +} + +/** + * This method syncs bounds of ruby annotations and ruby annotation + * containers from their rect. It is necessary because: + * Containers are not part of the line in their levels, which means + * their bounds are not set properly before. + * Ruby annotations' position may have been changed when reflowing + * their containers. + */ +void nsLineLayout::SyncAnnotationBounds(PerFrameData* aRubyFrame) { + MOZ_ASSERT(aRubyFrame->mFrame->IsRubyFrame()); + MOZ_ASSERT(aRubyFrame->mSpan); + + PerSpanData* span = aRubyFrame->mSpan; + WritingMode lineWM = mRootSpan->mWritingMode; + for (PerFrameData* pfd = span->mFirstFrame; pfd; pfd = pfd->mNext) { + for (PerFrameData* rtc = pfd->mNextAnnotation; rtc; + rtc = rtc->mNextAnnotation) { + if (lineWM.IsOrthogonalTo(rtc->mFrame->GetWritingMode())) { + // Inter-character case: don't attempt to sync annotation bounds. + continue; + } + // When the annotation container is reflowed, the width of the + // ruby container is unknown so we use a dummy container size; + // in the case of RTL block direction, the final position will be + // fixed up later. + const nsSize dummyContainerSize; + LogicalRect rtcBounds(lineWM, rtc->mFrame->GetRect(), dummyContainerSize); + rtc->mBounds = rtcBounds; + nsSize rtcSize = rtcBounds.Size(lineWM).GetPhysicalSize(lineWM); + for (PerFrameData* rt = rtc->mSpan->mFirstFrame; rt; rt = rt->mNext) { + LogicalRect rtBounds = rt->mFrame->GetLogicalRect(lineWM, rtcSize); + MOZ_ASSERT(rt->mBounds.Size(lineWM) == rtBounds.Size(lineWM), + "Size of the annotation should not have been changed"); + rt->mBounds.SetOrigin(lineWM, rtBounds.Origin(lineWM)); + } + } + } +} + +/** + * See if the frame can be placed now that we know it's desired size. + * We can always place the frame if the line is empty. Note that we + * know that the reflow-status is not a break-before because if it was + * ReflowFrame above would have returned false, preventing this method + * from being called. The logic in this method assumes that. + * + * Note that there is no check against the Y coordinate because we + * assume that the caller will take care of that. + */ +bool nsLineLayout::CanPlaceFrame(PerFrameData* pfd, bool aNotSafeToBreak, + bool aFrameCanContinueTextRun, + bool aCanRollBackBeforeFrame, + ReflowOutput& aMetrics, + nsReflowStatus& aStatus, + bool* aOptionalBreakAfterFits) { + MOZ_ASSERT(pfd && pfd->mFrame, "bad args, null pointers for frame data"); + + *aOptionalBreakAfterFits = true; + + WritingMode lineWM = mRootSpan->mWritingMode; + /* + * We want to only apply the end margin if we're the last continuation and + * either not in an {ib} split or the last inline in it. In all other + * cases we want to zero it out. That means zeroing it out if any of these + * conditions hold: + * 1) The frame is not complete (in this case it will get a next-in-flow) + * 2) The frame is complete but has a non-fluid continuation on its + * continuation chain. Note that if it has a fluid continuation, that + * continuation will get destroyed later, so we don't want to drop the + * end-margin in that case. + * 3) The frame is in an {ib} split and is not the last part. + * + * However, none of that applies if this is a letter frame (XXXbz why?) + * + * For box-decoration-break:clone we apply the end margin on all + * continuations (that are not letter frames). + */ + if ((aStatus.IsIncomplete() || + pfd->mFrame->LastInFlow()->GetNextContinuation() || + pfd->mFrame->FrameIsNonLastInIBSplit()) && + !pfd->mIsLetterFrame && + pfd->mFrame->StyleBorder()->mBoxDecorationBreak == + StyleBoxDecorationBreak::Slice) { + pfd->mMargin.IEnd(lineWM) = 0; + } + + // Apply the start margin to the frame bounds. + nscoord startMargin = pfd->mMargin.IStart(lineWM); + nscoord endMargin = pfd->mMargin.IEnd(lineWM); + + pfd->mBounds.IStart(lineWM) += startMargin; + + PerSpanData* psd = mCurrentSpan; + if (psd->mNoWrap) { + // When wrapping is off, everything fits. + return true; + } + +#ifdef NOISY_CAN_PLACE_FRAME + if (psd->mFrame) { + psd->mFrame->mFrame->ListTag(stdout); + } + printf(": aNotSafeToBreak=%s frame=", aNotSafeToBreak ? "true" : "false"); + pfd->mFrame->ListTag(stdout); + printf(" frameWidth=%d, margins=%d,%d\n", + pfd->mBounds.IEnd(lineWM) + endMargin - psd->mICoord, startMargin, + endMargin); +#endif + + // Set outside to true if the result of the reflow leads to the + // frame sticking outside of our available area. + bool outside = + pfd->mBounds.IEnd(lineWM) - mTrimmableISize + endMargin > psd->mIEnd; + if (!outside) { + // If it fits, it fits +#ifdef NOISY_CAN_PLACE_FRAME + printf(" ==> inside\n"); +#endif + return true; + } + *aOptionalBreakAfterFits = false; + + // When it doesn't fit, check for a few special conditions where we + // allow it to fit anyway. + if (0 == startMargin + pfd->mBounds.ISize(lineWM) + endMargin) { + // Empty frames always fit right where they are +#ifdef NOISY_CAN_PLACE_FRAME + printf(" ==> empty frame fits\n"); +#endif + return true; + } + + // another special case: always place a BR + if (pfd->mFrame->IsBrFrame()) { +#ifdef NOISY_CAN_PLACE_FRAME + printf(" ==> BR frame fits\n"); +#endif + return true; + } + + if (aNotSafeToBreak) { + // There are no frames on the line that take up width and the line is + // not impacted by floats, so we must allow the current frame to be + // placed on the line +#ifdef NOISY_CAN_PLACE_FRAME + printf(" ==> not-safe and not-impacted fits: "); + while (nullptr != psd) { + printf("<psd=%p x=%d left=%d> ", psd, psd->mICoord, psd->mIStart); + psd = psd->mParent; + } + printf("\n"); +#endif + return true; + } + + // Special check for span frames + if (pfd->mSpan && pfd->mSpan->mContainsFloat) { + // If the span either directly or indirectly contains a float then + // it fits. Why? It's kind of complicated, but here goes: + // + // 1. CanPlaceFrame is used for all frame placements on a line, + // and in a span. This includes recursively placement of frames + // inside of spans, and the span itself. Because the logic always + // checks for room before proceeding (the code above here), the + // only things on a line will be those things that "fit". + // + // 2. Before a float is placed on a line, the line has to be empty + // (otherwise it's a "below current line" float and will be placed + // after the line). + // + // Therefore, if the span directly or indirectly has a float + // then it means that at the time of the placement of the float + // the line was empty. Because of #1, only the frames that fit can + // be added after that point, therefore we can assume that the + // current span being placed has fit. + // + // So how do we get here and have a span that should already fit + // and yet doesn't: Simple: span's that have the no-wrap attribute + // set on them and contain a float and are placed where they + // don't naturally fit. + return true; + } + + if (aFrameCanContinueTextRun) { + // Let it fit, but we reserve the right to roll back. + // Note that we usually won't get here because a text frame will break + // itself to avoid exceeding the available width. + // We'll only get here for text frames that couldn't break early enough. +#ifdef NOISY_CAN_PLACE_FRAME + printf(" ==> placing overflowing textrun, requesting backup\n"); +#endif + + // We will want to try backup. + mNeedBackup = true; + return true; + } + +#ifdef NOISY_CAN_PLACE_FRAME + printf(" ==> didn't fit\n"); +#endif + aStatus.SetInlineLineBreakBeforeAndReset(); + return false; +} + +/** + * Place the frame. Update running counters. + */ +void nsLineLayout::PlaceFrame(PerFrameData* pfd, ReflowOutput& aMetrics) { + WritingMode lineWM = mRootSpan->mWritingMode; + + // If the frame's block direction does not match the line's, we can't use + // its ascent; instead, treat it as a block with baseline at the block-end + // edge (or block-begin in the case of an "inverted" line). + if (pfd->mWritingMode.GetBlockDir() != lineWM.GetBlockDir()) { + pfd->mAscent = lineWM.IsAlphabeticalBaseline() + ? lineWM.IsLineInverted() ? 0 : aMetrics.BSize(lineWM) + : aMetrics.BSize(lineWM) / 2; + } else { + // For inline reflow participants, baseline may get assigned as the frame is + // vertically aligned, which happens after this. + const auto baselineSource = pfd->mFrame->StyleDisplay()->mBaselineSource; + if (baselineSource == StyleBaselineSource::Auto || + pfd->mFrame->IsLineParticipant()) { + if (aMetrics.BlockStartAscent() == ReflowOutput::ASK_FOR_BASELINE) { + pfd->mAscent = pfd->mFrame->GetLogicalBaseline(lineWM); + } else { + pfd->mAscent = aMetrics.BlockStartAscent(); + } + } else { + const auto sourceGroup = [baselineSource]() { + switch (baselineSource) { + case StyleBaselineSource::First: + return BaselineSharingGroup::First; + case StyleBaselineSource::Last: + return BaselineSharingGroup::Last; + case StyleBaselineSource::Auto: + break; + } + MOZ_ASSERT_UNREACHABLE("Auto should be already handled?"); + return BaselineSharingGroup::First; + }(); + // We ignore line-layout specific layout quirks by setting + // `BaselineExportContext::Other`. + // Note(dshin): For a lot of frames, the export context does not make a + // difference, and we may be wasting the value cached in + // `BlockStartAscent`. + pfd->mAscent = pfd->mFrame->GetLogicalBaseline( + lineWM, sourceGroup, BaselineExportContext::Other); + } + } + + // Advance to next inline coordinate + mCurrentSpan->mICoord = pfd->mBounds.IEnd(lineWM) + pfd->mMargin.IEnd(lineWM); + + // Count the number of non-placeholder frames on the line... + if (pfd->mFrame->IsPlaceholderFrame()) { + NS_ASSERTION( + pfd->mBounds.ISize(lineWM) == 0 && pfd->mBounds.BSize(lineWM) == 0, + "placeholders should have 0 width/height (checking " + "placeholders were never counted by the old code in " + "this function)"); + } else { + mTotalPlacedFrames++; + } +} + +void nsLineLayout::AddMarkerFrame(nsIFrame* aFrame, + const ReflowOutput& aMetrics) { + NS_ASSERTION(mCurrentSpan == mRootSpan, "bad linelayout user"); + NS_ASSERTION(mGotLineBox, "must have line box"); + + nsBlockFrame* blockFrame = do_QueryFrame(LineContainerFrame()); + MOZ_ASSERT(blockFrame, "must be for block"); + if (!blockFrame->MarkerIsEmpty()) { + mHasMarker = true; + mLineBox->SetHasMarker(); + } + + WritingMode lineWM = mRootSpan->mWritingMode; + PerFrameData* pfd = NewPerFrameData(aFrame); + PerSpanData* psd = mRootSpan; + + MOZ_ASSERT(psd->mFirstFrame, "adding marker to an empty line?"); + // Prepend the marker frame to the line. + psd->mFirstFrame->mPrev = pfd; + pfd->mNext = psd->mFirstFrame; + psd->mFirstFrame = pfd; + + pfd->mIsMarker = true; + if (aMetrics.BlockStartAscent() == ReflowOutput::ASK_FOR_BASELINE) { + pfd->mAscent = aFrame->GetLogicalBaseline(lineWM); + } else { + pfd->mAscent = aMetrics.BlockStartAscent(); + } + + // Note: block-coord value will be updated during block-direction alignment + pfd->mBounds = LogicalRect(lineWM, aFrame->GetRect(), ContainerSize()); + pfd->mOverflowAreas = aMetrics.mOverflowAreas; +} + +void nsLineLayout::RemoveMarkerFrame(nsIFrame* aFrame) { + PerSpanData* psd = mCurrentSpan; + MOZ_ASSERT(psd == mRootSpan, "::marker on non-root span?"); + MOZ_ASSERT(psd->mFirstFrame->mFrame == aFrame, + "::marker is not the first frame?"); + PerFrameData* pfd = psd->mFirstFrame; + MOZ_ASSERT(pfd != psd->mLastFrame, "::marker is the only frame?"); + pfd->mNext->mPrev = nullptr; + psd->mFirstFrame = pfd->mNext; + FreeFrame(pfd); +} +#ifdef DEBUG +void nsLineLayout::DumpPerSpanData(PerSpanData* psd, int32_t aIndent) { + nsIFrame::IndentBy(stdout, aIndent); + printf("%p: left=%d x=%d right=%d\n", static_cast<void*>(psd), psd->mIStart, + psd->mICoord, psd->mIEnd); + PerFrameData* pfd = psd->mFirstFrame; + while (nullptr != pfd) { + nsIFrame::IndentBy(stdout, aIndent + 1); + pfd->mFrame->ListTag(stdout); + nsRect rect = + pfd->mBounds.GetPhysicalRect(psd->mWritingMode, ContainerSize()); + printf(" %d,%d,%d,%d\n", rect.x, rect.y, rect.width, rect.height); + if (pfd->mSpan) { + DumpPerSpanData(pfd->mSpan, aIndent + 1); + } + pfd = pfd->mNext; + } +} +#endif + +void nsLineLayout::RecordNoWrapFloat(nsIFrame* aFloat) { + GetOutermostLineLayout()->mBlockRS->mNoWrapFloats.AppendElement(aFloat); +} + +void nsLineLayout::FlushNoWrapFloats() { + auto& noWrapFloats = GetOutermostLineLayout()->mBlockRS->mNoWrapFloats; + for (nsIFrame* floatedFrame : noWrapFloats) { + TryToPlaceFloat(floatedFrame); + } + noWrapFloats.Clear(); +} + +bool nsLineLayout::TryToPlaceFloat(nsIFrame* aFloat) { + // Add mTrimmableISize to the available width since if the line ends here, the + // width of the inline content will be reduced by mTrimmableISize. + nscoord availableISize = + mCurrentSpan->mIEnd - (mCurrentSpan->mICoord - mTrimmableISize); + NS_ASSERTION(!(aFloat->IsLetterFrame() && GetFirstLetterStyleOK()), + "FirstLetterStyle set on line with floating first letter"); + return GetOutermostLineLayout()->AddFloat(aFloat, availableISize); +} + +bool nsLineLayout::NotifyOptionalBreakPosition(nsIFrame* aFrame, + 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"); + MOZ_ASSERT(mCurrentSpan, "Should be doing line layout"); + if (mCurrentSpan->mNoWrap) { + FlushNoWrapFloats(); + } + + // Remember the last break position that fits; if there was no break that fit, + // just remember the first break + if ((aFits && aPriority >= mLastOptionalBreakPriority) || + !mLastOptionalBreakFrame) { + mLastOptionalBreakFrame = aFrame; + mLastOptionalBreakFrameOffset = aOffset; + mLastOptionalBreakPriority = aPriority; + } + return aFrame && mForceBreakFrame == aFrame && + mForceBreakFrameOffset == aOffset; +} + +#define VALIGN_OTHER 0 +#define VALIGN_TOP 1 +#define VALIGN_BOTTOM 2 + +void nsLineLayout::VerticalAlignLine() { + // Partially place the children of the block frame. The baseline for + // this operation is set to zero so that the y coordinates for all + // of the placed children will be relative to there. + PerSpanData* psd = mRootSpan; + VerticalAlignFrames(psd); + + // *** Note that comments here still use the anachronistic term + // "line-height" when we really mean "size of the line in the block + // direction", "vertical-align" when we really mean "alignment in + // the block direction", and "top" and "bottom" when we really mean + // "block start" and "block end". This is partly for brevity and + // partly to retain the association with the CSS line-height and + // vertical-align properties. + // + // Compute the line-height. The line-height will be the larger of: + // + // [1] maxBCoord - minBCoord (the distance between the first child's + // block-start edge and the last child's block-end edge) + // + // [2] the maximum logical box block size (since not every frame may have + // participated in #1; for example: "top" and "botttom" aligned frames) + // + // [3] the minimum line height ("line-height" property set on the + // block frame) + nscoord lineBSize = psd->mMaxBCoord - psd->mMinBCoord; + + // Now that the line-height is computed, we need to know where the + // baseline is in the line. Position baseline so that mMinBCoord is just + // inside the start of the line box. + nscoord baselineBCoord; + if (psd->mMinBCoord < 0) { + baselineBCoord = mBStartEdge - psd->mMinBCoord; + } else { + baselineBCoord = mBStartEdge; + } + + // It's also possible that the line block-size isn't tall enough because + // of "top" and "bottom" aligned elements that were not accounted for in + // min/max BCoord. + // + // The CSS2 spec doesn't really say what happens when to the + // baseline in this situations. What we do is if the largest start + // aligned box block size is greater than the line block-size then we leave + // the baseline alone. If the largest end aligned box is greater + // than the line block-size then we slide the baseline forward by the extra + // amount. + // + // Navigator 4 gives precedence to the first top/bottom aligned + // object. We just let block end aligned objects win. + if (lineBSize < mMaxEndBoxBSize) { + // When the line is shorter than the maximum block start aligned box + nscoord extra = mMaxEndBoxBSize - lineBSize; + baselineBCoord += extra; + lineBSize = mMaxEndBoxBSize; + } + if (lineBSize < mMaxStartBoxBSize) { + lineBSize = mMaxStartBoxBSize; + } +#ifdef NOISY_BLOCKDIR_ALIGN + printf(" [line]==> lineBSize=%d baselineBCoord=%d\n", lineBSize, + baselineBCoord); +#endif + + // Now position all of the frames in the root span. We will also + // recurse over the child spans and place any frames we find with + // vertical-align: top or bottom. + // XXX PERFORMANCE: set a bit per-span to avoid the extra work + // (propagate it upward too) + WritingMode lineWM = psd->mWritingMode; + for (PerFrameData* pfd = psd->mFirstFrame; pfd; pfd = pfd->mNext) { + if (pfd->mBlockDirAlign == VALIGN_OTHER) { + pfd->mBounds.BStart(lineWM) += baselineBCoord; + pfd->mFrame->SetRect(lineWM, pfd->mBounds, ContainerSize()); + } + } + PlaceTopBottomFrames(psd, -mBStartEdge, lineBSize); + + mFinalLineBSize = lineBSize; + if (mGotLineBox) { + // Fill in returned line-box and max-element-width data + mLineBox->SetBounds(lineWM, psd->mIStart, mBStartEdge, + psd->mICoord - psd->mIStart, lineBSize, + ContainerSize()); + + mLineBox->SetLogicalAscent(baselineBCoord - mBStartEdge); +#ifdef NOISY_BLOCKDIR_ALIGN + printf(" [line]==> bounds{x,y,w,h}={%d,%d,%d,%d} lh=%d a=%d\n", + mLineBox->GetBounds().IStart(lineWM), + mLineBox->GetBounds().BStart(lineWM), + mLineBox->GetBounds().ISize(lineWM), + mLineBox->GetBounds().BSize(lineWM), mFinalLineBSize, + mLineBox->GetLogicalAscent()); +#endif + } +} + +// Place frames with CSS property vertical-align: top or bottom. +void nsLineLayout::PlaceTopBottomFrames(PerSpanData* psd, + nscoord aDistanceFromStart, + nscoord aLineBSize) { + for (PerFrameData* pfd = psd->mFirstFrame; pfd; pfd = pfd->mNext) { + PerSpanData* span = pfd->mSpan; +#ifdef DEBUG + NS_ASSERTION(0xFF != pfd->mBlockDirAlign, "umr"); +#endif + WritingMode lineWM = mRootSpan->mWritingMode; + nsSize containerSize = ContainerSizeForSpan(psd); + switch (pfd->mBlockDirAlign) { + case VALIGN_TOP: + if (span) { + pfd->mBounds.BStart(lineWM) = -aDistanceFromStart - span->mMinBCoord; + } else { + pfd->mBounds.BStart(lineWM) = + -aDistanceFromStart + pfd->mMargin.BStart(lineWM); + } + pfd->mFrame->SetRect(lineWM, pfd->mBounds, containerSize); +#ifdef NOISY_BLOCKDIR_ALIGN + printf(" "); + pfd->mFrame->ListTag(stdout); + printf(": y=%d dTop=%d [bp.top=%d topLeading=%d]\n", + pfd->mBounds.BStart(lineWM), aDistanceFromStart, + span ? pfd->mBorderPadding.BStart(lineWM) : 0, + span ? span->mBStartLeading : 0); +#endif + break; + case VALIGN_BOTTOM: + if (span) { + // Compute bottom leading + pfd->mBounds.BStart(lineWM) = + -aDistanceFromStart + aLineBSize - span->mMaxBCoord; + } else { + pfd->mBounds.BStart(lineWM) = -aDistanceFromStart + aLineBSize - + pfd->mMargin.BEnd(lineWM) - + pfd->mBounds.BSize(lineWM); + } + pfd->mFrame->SetRect(lineWM, pfd->mBounds, containerSize); +#ifdef NOISY_BLOCKDIR_ALIGN + printf(" "); + pfd->mFrame->ListTag(stdout); + printf(": y=%d\n", pfd->mBounds.BStart(lineWM)); +#endif + break; + } + if (span) { + nscoord fromStart = aDistanceFromStart + pfd->mBounds.BStart(lineWM); + PlaceTopBottomFrames(span, fromStart, aLineBSize); + } + } +} + +static nscoord GetBSizeOfEmphasisMarks(nsIFrame* aSpanFrame, float aInflation) { + RefPtr<nsFontMetrics> fm = nsLayoutUtils::GetFontMetricsOfEmphasisMarks( + aSpanFrame->Style(), aSpanFrame->PresContext(), aInflation); + return fm->MaxHeight(); +} + +void nsLineLayout::AdjustLeadings(nsIFrame* spanFrame, PerSpanData* psd, + const nsStyleText* aStyleText, + float aInflation, + bool* aZeroEffectiveSpanBox) { + MOZ_ASSERT(spanFrame == psd->mFrame->mFrame); + nscoord requiredStartLeading = 0; + nscoord requiredEndLeading = 0; + if (spanFrame->IsRubyFrame()) { + // We may need to extend leadings here for ruby annotations as + // required by section Line Spacing in the CSS Ruby spec. + // See http://dev.w3.org/csswg/css-ruby/#line-height + auto rubyFrame = static_cast<nsRubyFrame*>(spanFrame); + RubyBlockLeadings leadings = rubyFrame->GetBlockLeadings(); + requiredStartLeading += leadings.mStart; + requiredEndLeading += leadings.mEnd; + } + if (aStyleText->HasEffectiveTextEmphasis()) { + nscoord bsize = GetBSizeOfEmphasisMarks(spanFrame, aInflation); + LogicalSide side = aStyleText->TextEmphasisSide(mRootSpan->mWritingMode); + if (side == eLogicalSideBStart) { + requiredStartLeading += bsize; + } else { + MOZ_ASSERT(side == eLogicalSideBEnd, + "emphasis marks must be in block axis"); + requiredEndLeading += bsize; + } + } + + nscoord requiredLeading = requiredStartLeading + requiredEndLeading; + // If we do not require any additional leadings, don't touch anything + // here even if it is greater than the original leading, because the + // latter could be negative. + if (requiredLeading != 0) { + nscoord leading = psd->mBStartLeading + psd->mBEndLeading; + nscoord deltaLeading = requiredLeading - leading; + if (deltaLeading > 0) { + // If the total leading is not wide enough for ruby annotations + // and/or emphasis marks, extend the side which is not enough. If + // both sides are not wide enough, replace the leadings with the + // requested values. + if (requiredStartLeading < psd->mBStartLeading) { + psd->mBEndLeading += deltaLeading; + } else if (requiredEndLeading < psd->mBEndLeading) { + psd->mBStartLeading += deltaLeading; + } else { + psd->mBStartLeading = requiredStartLeading; + psd->mBEndLeading = requiredEndLeading; + } + psd->mLogicalBSize += deltaLeading; + // We have adjusted the leadings, it is no longer a zero + // effective span box. + *aZeroEffectiveSpanBox = false; + } + } +} + +static float GetInflationForBlockDirAlignment(nsIFrame* aFrame, + nscoord aInflationMinFontSize) { + if (aFrame->IsInSVGTextSubtree()) { + const nsIFrame* container = + nsLayoutUtils::GetClosestFrameOfType(aFrame, LayoutFrameType::SVGText); + NS_ASSERTION(container, "expected to find an ancestor SVGTextFrame"); + return static_cast<const SVGTextFrame*>(container) + ->GetFontSizeScaleFactor(); + } + return nsLayoutUtils::FontSizeInflationInner(aFrame, aInflationMinFontSize); +} + +bool nsLineLayout::ShouldApplyLineHeightInPreserveWhiteSpace( + const PerSpanData* psd) { + if (psd->mFrame->mFrame->Style()->IsAnonBox()) { + // e.g. An empty `input[type=button]` should still be line-height sized. + return true; + } + + for (PerFrameData* pfd = psd->mFirstFrame; pfd; pfd = pfd->mNext) { + if (!pfd->mIsEmpty) { + return true; + } + } + return false; +} + +#define BLOCKDIR_ALIGN_FRAMES_NO_MINIMUM nscoord_MAX +#define BLOCKDIR_ALIGN_FRAMES_NO_MAXIMUM nscoord_MIN + +// Place frames in the block direction within a given span (CSS property +// vertical-align) Note: this doesn't place frames with vertical-align: +// top or bottom as those have to wait until the entire line box block +// size is known. This is called after the span frame has finished being +// reflowed so that we know its block size. +void nsLineLayout::VerticalAlignFrames(PerSpanData* psd) { + // Get parent frame info + PerFrameData* spanFramePFD = psd->mFrame; + nsIFrame* spanFrame = spanFramePFD->mFrame; + + // Get the parent frame's font for all of the frames in this span + float inflation = + GetInflationForBlockDirAlignment(spanFrame, mInflationMinFontSize); + RefPtr<nsFontMetrics> fm = + nsLayoutUtils::GetFontMetricsForFrame(spanFrame, inflation); + + bool preMode = mStyleText->WhiteSpaceIsSignificant(); + + // See if the span is an empty continuation. It's an empty continuation iff: + // - it has a prev-in-flow + // - it has no next in flow + // - it's zero sized + WritingMode lineWM = mRootSpan->mWritingMode; + bool emptyContinuation = psd != mRootSpan && spanFrame->GetPrevInFlow() && + !spanFrame->GetNextInFlow() && + spanFramePFD->mBounds.IsZeroSize(); + +#ifdef NOISY_BLOCKDIR_ALIGN + printf("[%sSpan]", (psd == mRootSpan) ? "Root" : ""); + spanFrame->ListTag(stdout); + printf(": preMode=%s strictMode=%s w/h=%d,%d emptyContinuation=%s", + preMode ? "yes" : "no", + mPresContext->CompatibilityMode() != eCompatibility_NavQuirks ? "yes" + : "no", + spanFramePFD->mBounds.ISize(lineWM), + spanFramePFD->mBounds.BSize(lineWM), emptyContinuation ? "yes" : "no"); + if (psd != mRootSpan) { + printf(" bp=%d,%d,%d,%d margin=%d,%d,%d,%d", + spanFramePFD->mBorderPadding.Top(lineWM), + spanFramePFD->mBorderPadding.Right(lineWM), + spanFramePFD->mBorderPadding.Bottom(lineWM), + spanFramePFD->mBorderPadding.Left(lineWM), + spanFramePFD->mMargin.Top(lineWM), + spanFramePFD->mMargin.Right(lineWM), + spanFramePFD->mMargin.Bottom(lineWM), + spanFramePFD->mMargin.Left(lineWM)); + } + printf("\n"); +#endif + + // Compute the span's zeroEffectiveSpanBox flag. What we are trying + // to determine is how we should treat the span: should it act + // "normally" according to css2 or should it effectively + // "disappear". + // + // In general, if the document being processed is in full standards + // mode then it should act normally (with one exception). The + // exception case is when a span is continued and yet the span is + // empty (e.g. compressed whitespace). For this kind of span we treat + // it as if it were not there so that it doesn't impact the + // line block-size. + // + // In almost standards mode or quirks mode, we should sometimes make + // it disappear. The cases that matter are those where the span + // contains no real text elements that would provide an ascent and + // descent and height. However, if css style elements have been + // applied to the span (border/padding/margin) so that it's clear the + // document author is intending css2 behavior then we act as if strict + // mode is set. + // + // This code works correctly for preMode, because a blank line + // in PRE mode is encoded as a text node with a LF in it, since + // text nodes with only whitespace are considered in preMode. + // + // Much of this logic is shared with the various implementations of + // nsIFrame::IsEmpty since they need to duplicate the way it makes + // some lines empty. However, nsIFrame::IsEmpty can't be reused here + // since this code sets zeroEffectiveSpanBox even when there are + // non-empty children. + bool zeroEffectiveSpanBox = false; + // XXXldb If we really have empty continuations, then all these other + // checks don't make sense for them. + // XXXldb This should probably just use nsIFrame::IsSelfEmpty, assuming that + // it agrees with this code. (If it doesn't agree, it probably should.) + if ((emptyContinuation || + mPresContext->CompatibilityMode() != eCompatibility_FullStandards) && + ((psd == mRootSpan) || (spanFramePFD->mBorderPadding.IsAllZero() && + spanFramePFD->mMargin.IsAllZero()))) { + // This code handles an issue with compatibility with non-css + // conformant browsers. In particular, there are some cases + // where the font-size and line-height for a span must be + // ignored and instead the span must *act* as if it were zero + // sized. In general, if the span contains any non-compressed + // text then we don't use this logic. + // However, this is not propagated outwards, since (in compatibility + // mode) we don't want big line heights for things like + // <p><font size="-1">Text</font></p> + + // We shouldn't include any whitespace that collapses, unless we're + // preformatted (in which case it shouldn't, but the width=0 test is + // perhaps incorrect). This includes whitespace at the beginning of + // a line and whitespace preceded (?) by other whitespace. + // See bug 134580 and bug 155333. + zeroEffectiveSpanBox = true; + for (PerFrameData* pfd = psd->mFirstFrame; pfd; pfd = pfd->mNext) { + if (pfd->mIsTextFrame && + (pfd->mIsNonWhitespaceTextFrame || preMode || + pfd->mBounds.ISize(mRootSpan->mWritingMode) != 0)) { + zeroEffectiveSpanBox = false; + break; + } + } + } + + // Setup baselineBCoord, minBCoord, and maxBCoord + nscoord baselineBCoord, minBCoord, maxBCoord; + if (psd == mRootSpan) { + // Use a zero baselineBCoord since we don't yet know where the baseline + // will be (until we know how tall the line is; then we will + // know). In addition, use extreme values for the minBCoord and maxBCoord + // values so that only the child frames will impact their values + // (since these are children of the block, there is no span box to + // provide initial values). + baselineBCoord = 0; + minBCoord = BLOCKDIR_ALIGN_FRAMES_NO_MINIMUM; + maxBCoord = BLOCKDIR_ALIGN_FRAMES_NO_MAXIMUM; +#ifdef NOISY_BLOCKDIR_ALIGN + printf("[RootSpan]"); + spanFrame->ListTag(stdout); + printf( + ": pass1 valign frames: topEdge=%d minLineBSize=%d " + "zeroEffectiveSpanBox=%s\n", + mBStartEdge, mMinLineBSize, zeroEffectiveSpanBox ? "yes" : "no"); +#endif + } else { + // Compute the logical block size for this span. The logical block size + // is based on the "line-height" value, not the font-size. Also + // compute the top leading. + float inflation = + GetInflationForBlockDirAlignment(spanFrame, mInflationMinFontSize); + nscoord logicalBSize = ReflowInput::CalcLineHeight( + *spanFrame->Style(), spanFrame->PresContext(), spanFrame->GetContent(), + mLineContainerRI.ComputedHeight(), inflation); + nscoord contentBSize = spanFramePFD->mBounds.BSize(lineWM) - + spanFramePFD->mBorderPadding.BStartEnd(lineWM); + + // Special-case for a ::first-letter frame, set the line height to + // the frame block size if the user has left line-height == normal + if (spanFramePFD->mIsLetterFrame && !spanFrame->GetPrevInFlow() && + spanFrame->StyleFont()->mLineHeight.IsNormal()) { + logicalBSize = spanFramePFD->mBounds.BSize(lineWM); + } + + nscoord leading = logicalBSize - contentBSize; + psd->mBStartLeading = leading / 2; + psd->mBEndLeading = leading - psd->mBStartLeading; + psd->mLogicalBSize = logicalBSize; + AdjustLeadings(spanFrame, psd, spanFrame->StyleText(), inflation, + &zeroEffectiveSpanBox); + + if (zeroEffectiveSpanBox) { + // When the span-box is to be ignored, zero out the initial + // values so that the span doesn't impact the final line + // height. The contents of the span can impact the final line + // height. + + // Note that things are readjusted for this span after its children + // are reflowed + minBCoord = BLOCKDIR_ALIGN_FRAMES_NO_MINIMUM; + maxBCoord = BLOCKDIR_ALIGN_FRAMES_NO_MAXIMUM; + } else { + // The initial values for the min and max block coord values are in the + // span's coordinate space, and cover the logical block size of the span. + // If there are child frames in this span that stick out of this area + // then the minBCoord and maxBCoord are updated by the amount of logical + // blockSize that is outside this range. + minBCoord = + spanFramePFD->mBorderPadding.BStart(lineWM) - psd->mBStartLeading; + maxBCoord = minBCoord + psd->mLogicalBSize; + } + + // This is the distance from the top edge of the parents visual + // box to the baseline. The span already computed this for us, + // so just use it. + *psd->mBaseline = baselineBCoord = spanFramePFD->mAscent; + +#ifdef NOISY_BLOCKDIR_ALIGN + printf("[%sSpan]", (psd == mRootSpan) ? "Root" : ""); + spanFrame->ListTag(stdout); + printf( + ": baseLine=%d logicalBSize=%d topLeading=%d h=%d bp=%d,%d " + "zeroEffectiveSpanBox=%s\n", + baselineBCoord, psd->mLogicalBSize, psd->mBStartLeading, + spanFramePFD->mBounds.BSize(lineWM), + spanFramePFD->mBorderPadding.Top(lineWM), + spanFramePFD->mBorderPadding.Bottom(lineWM), + zeroEffectiveSpanBox ? "yes" : "no"); +#endif + } + + nscoord maxStartBoxBSize = 0; + nscoord maxEndBoxBSize = 0; + PerFrameData* pfd = psd->mFirstFrame; + while (nullptr != pfd) { + nsIFrame* frame = pfd->mFrame; + + // sanity check (see bug 105168, non-reproducible crashes from null frame) + NS_ASSERTION(frame, + "null frame in PerFrameData - something is very very bad"); + if (!frame) { + return; + } + + // Compute the logical block size of the frame + nscoord logicalBSize; + PerSpanData* frameSpan = pfd->mSpan; + if (frameSpan) { + // For span frames the logical-block-size and start-leading were + // pre-computed when the span was reflowed. + logicalBSize = frameSpan->mLogicalBSize; + } else { + // For other elements the logical block size is the same as the + // frame's block size plus its margins. + logicalBSize = + pfd->mBounds.BSize(lineWM) + pfd->mMargin.BStartEnd(lineWM); + if (logicalBSize < 0 && + mPresContext->CompatibilityMode() != eCompatibility_FullStandards) { + pfd->mAscent -= logicalBSize; + logicalBSize = 0; + } + } + + // Get vertical-align property ("vertical-align" is the CSS name for + // block-direction align) + const auto& verticalAlign = frame->StyleDisplay()->mVerticalAlign; + Maybe<StyleVerticalAlignKeyword> verticalAlignEnum = + frame->VerticalAlignEnum(); +#ifdef NOISY_BLOCKDIR_ALIGN + printf(" [frame]"); + frame->ListTag(stdout); + printf(": verticalAlignIsKw=%d (enum == %d", verticalAlign.IsKeyword(), + verticalAlign.IsKeyword() + ? static_cast<int>(verticalAlign.AsKeyword()) + : -1); + if (verticalAlignEnum) { + printf(", after SVG dominant-baseline conversion == %d", + static_cast<int>(*verticalAlignEnum)); + } + printf(")\n"); +#endif + + if (verticalAlignEnum) { + StyleVerticalAlignKeyword keyword = *verticalAlignEnum; + if (lineWM.IsVertical()) { + if (keyword == StyleVerticalAlignKeyword::Middle) { + // For vertical writing mode where the dominant baseline is centered + // (i.e. text-orientation is not sideways-*), we remap 'middle' to + // 'middle-with-baseline' so that images align sensibly with the + // center-baseline-aligned text. + if (!lineWM.IsSideways()) { + keyword = StyleVerticalAlignKeyword::MozMiddleWithBaseline; + } + } else if (lineWM.IsLineInverted()) { + // Swap the meanings of top and bottom when line is inverted + // relative to block direction. + switch (keyword) { + case StyleVerticalAlignKeyword::Top: + keyword = StyleVerticalAlignKeyword::Bottom; + break; + case StyleVerticalAlignKeyword::Bottom: + keyword = StyleVerticalAlignKeyword::Top; + break; + case StyleVerticalAlignKeyword::TextTop: + keyword = StyleVerticalAlignKeyword::TextBottom; + break; + case StyleVerticalAlignKeyword::TextBottom: + keyword = StyleVerticalAlignKeyword::TextTop; + break; + default: + break; + } + } + } + + // baseline coord that may be adjusted for script offset + nscoord revisedBaselineBCoord = baselineBCoord; + + // For superscript and subscript, raise or lower the baseline of the box + // to the proper offset of the parent's box, then proceed as for BASELINE + if (keyword == StyleVerticalAlignKeyword::Sub || + keyword == StyleVerticalAlignKeyword::Super) { + revisedBaselineBCoord += lineWM.FlowRelativeToLineRelativeFactor() * + (keyword == StyleVerticalAlignKeyword::Sub + ? fm->SubscriptOffset() + : -fm->SuperscriptOffset()); + keyword = StyleVerticalAlignKeyword::Baseline; + } + + switch (keyword) { + default: + case StyleVerticalAlignKeyword::Baseline: + pfd->mBounds.BStart(lineWM) = revisedBaselineBCoord - pfd->mAscent; + pfd->mBlockDirAlign = VALIGN_OTHER; + break; + + case StyleVerticalAlignKeyword::Top: { + pfd->mBlockDirAlign = VALIGN_TOP; + nscoord subtreeBSize = logicalBSize; + if (frameSpan) { + subtreeBSize = frameSpan->mMaxBCoord - frameSpan->mMinBCoord; + NS_ASSERTION(subtreeBSize >= logicalBSize, + "unexpected subtree block size"); + } + if (subtreeBSize > maxStartBoxBSize) { + maxStartBoxBSize = subtreeBSize; + } + break; + } + + case StyleVerticalAlignKeyword::Bottom: { + pfd->mBlockDirAlign = VALIGN_BOTTOM; + nscoord subtreeBSize = logicalBSize; + if (frameSpan) { + subtreeBSize = frameSpan->mMaxBCoord - frameSpan->mMinBCoord; + NS_ASSERTION(subtreeBSize >= logicalBSize, + "unexpected subtree block size"); + } + if (subtreeBSize > maxEndBoxBSize) { + maxEndBoxBSize = subtreeBSize; + } + break; + } + + case StyleVerticalAlignKeyword::Middle: { + // Align the midpoint of the frame with 1/2 the parents + // x-height above the baseline. + nscoord parentXHeight = + lineWM.FlowRelativeToLineRelativeFactor() * fm->XHeight(); + if (frameSpan) { + pfd->mBounds.BStart(lineWM) = + baselineBCoord - + (parentXHeight + pfd->mBounds.BSize(lineWM)) / 2; + } else { + pfd->mBounds.BStart(lineWM) = baselineBCoord - + (parentXHeight + logicalBSize) / 2 + + pfd->mMargin.BStart(lineWM); + } + pfd->mBlockDirAlign = VALIGN_OTHER; + break; + } + + case StyleVerticalAlignKeyword::TextTop: { + // The top of the logical box is aligned with the top of + // the parent element's text. + // XXX For vertical text we will need a new API to get the logical + // max-ascent here + nscoord parentAscent = + lineWM.IsLineInverted() ? fm->MaxDescent() : fm->MaxAscent(); + if (frameSpan) { + pfd->mBounds.BStart(lineWM) = baselineBCoord - parentAscent - + pfd->mBorderPadding.BStart(lineWM) + + frameSpan->mBStartLeading; + } else { + pfd->mBounds.BStart(lineWM) = + baselineBCoord - parentAscent + pfd->mMargin.BStart(lineWM); + } + pfd->mBlockDirAlign = VALIGN_OTHER; + break; + } + + case StyleVerticalAlignKeyword::TextBottom: { + // The bottom of the logical box is aligned with the + // bottom of the parent elements text. + nscoord parentDescent = + lineWM.IsLineInverted() ? fm->MaxAscent() : fm->MaxDescent(); + if (frameSpan) { + pfd->mBounds.BStart(lineWM) = + baselineBCoord + parentDescent - pfd->mBounds.BSize(lineWM) + + pfd->mBorderPadding.BEnd(lineWM) - frameSpan->mBEndLeading; + } else { + pfd->mBounds.BStart(lineWM) = baselineBCoord + parentDescent - + pfd->mBounds.BSize(lineWM) - + pfd->mMargin.BEnd(lineWM); + } + pfd->mBlockDirAlign = VALIGN_OTHER; + break; + } + + case StyleVerticalAlignKeyword::MozMiddleWithBaseline: { + // Align the midpoint of the frame with the baseline of the parent. + if (frameSpan) { + pfd->mBounds.BStart(lineWM) = + baselineBCoord - pfd->mBounds.BSize(lineWM) / 2; + } else { + pfd->mBounds.BStart(lineWM) = + baselineBCoord - logicalBSize / 2 + pfd->mMargin.BStart(lineWM); + } + pfd->mBlockDirAlign = VALIGN_OTHER; + break; + } + } + } else { + // We have either a coord, a percent, or a calc(). + nscoord offset = verticalAlign.AsLength().Resolve([&] { + // Percentages are like lengths, except treated as a percentage + // of the elements line block size value. + float inflation = + GetInflationForBlockDirAlignment(frame, mInflationMinFontSize); + return ReflowInput::CalcLineHeight( + *frame->Style(), frame->PresContext(), frame->GetContent(), + mLineContainerRI.ComputedBSize(), inflation); + }); + + // According to the CSS2 spec (10.8.1), a positive value + // "raises" the box by the given distance while a negative value + // "lowers" the box by the given distance (with zero being the + // baseline). Since Y coordinates increase towards the bottom of + // the screen we reverse the sign, unless the line orientation is + // inverted relative to block direction. + nscoord revisedBaselineBCoord = + baselineBCoord - offset * lineWM.FlowRelativeToLineRelativeFactor(); + if (lineWM.IsCentralBaseline()) { + // If we're using a dominant center baseline, we align with the center + // of the frame being placed (bug 1133945). + pfd->mBounds.BStart(lineWM) = + revisedBaselineBCoord - pfd->mBounds.BSize(lineWM) / 2; + } else { + pfd->mBounds.BStart(lineWM) = revisedBaselineBCoord - pfd->mAscent; + } + pfd->mBlockDirAlign = VALIGN_OTHER; + } + + // Update minBCoord/maxBCoord for frames that we just placed. Do not factor + // text into the equation. + if (pfd->mBlockDirAlign == VALIGN_OTHER) { + // Text frames do not contribute to the min/max Y values for the + // line (instead their parent frame's font-size contributes). + // XXXrbs -- relax this restriction because it causes text frames + // to jam together when 'font-size-adjust' is enabled + // and layout is using dynamic font heights (bug 20394) + // -- Note #1: With this code enabled and with the fact that we are + // not using Em[Ascent|Descent] as nsDimensions for text + // metrics in GFX mean that the discussion in bug 13072 cannot + // hold. + // -- Note #2: We still don't want empty-text frames to interfere. + // For example in quirks mode, avoiding empty text frames + // prevents "tall" lines around elements like <hr> since the + // rules of <hr> in quirks.css have pseudo text contents with LF + // in them. + bool canUpdate; + if (pfd->mIsTextFrame) { + // Only consider text frames if they're not empty and + // line-height=normal. + canUpdate = pfd->mIsNonWhitespaceTextFrame && + frame->StyleFont()->mLineHeight.IsNormal(); + } else { + canUpdate = !pfd->mIsPlaceholder; + } + + if (canUpdate) { + nscoord blockStart, blockEnd; + if (frameSpan) { + // For spans that were are now placing, use their position + // plus their already computed min-Y and max-Y values for + // computing blockStart and blockEnd. + blockStart = pfd->mBounds.BStart(lineWM) + frameSpan->mMinBCoord; + blockEnd = pfd->mBounds.BStart(lineWM) + frameSpan->mMaxBCoord; + } else { + blockStart = + pfd->mBounds.BStart(lineWM) - pfd->mMargin.BStart(lineWM); + blockEnd = blockStart + logicalBSize; + } + if (!preMode && + mPresContext->CompatibilityMode() != eCompatibility_FullStandards && + !logicalBSize) { + // Check if it's a BR frame that is not alone on its line (it + // is given a block size of zero to indicate this), and if so reset + // blockStart and blockEnd so that BR frames don't influence the line. + if (frame->IsBrFrame()) { + blockStart = BLOCKDIR_ALIGN_FRAMES_NO_MINIMUM; + blockEnd = BLOCKDIR_ALIGN_FRAMES_NO_MAXIMUM; + } + } + if (blockStart < minBCoord) minBCoord = blockStart; + if (blockEnd > maxBCoord) maxBCoord = blockEnd; +#ifdef NOISY_BLOCKDIR_ALIGN + printf( + " [frame]raw: a=%d h=%d bp=%d,%d logical: h=%d leading=%d y=%d " + "minBCoord=%d maxBCoord=%d\n", + pfd->mAscent, pfd->mBounds.BSize(lineWM), + pfd->mBorderPadding.Top(lineWM), pfd->mBorderPadding.Bottom(lineWM), + logicalBSize, frameSpan ? frameSpan->mBStartLeading : 0, + pfd->mBounds.BStart(lineWM), minBCoord, maxBCoord); +#endif + } + if (psd != mRootSpan) { + frame->SetRect(lineWM, pfd->mBounds, ContainerSizeForSpan(psd)); + } + } + pfd = pfd->mNext; + } + + // Factor in the minimum line block-size when handling the root-span for + // the block. + if (psd == mRootSpan) { + // We should factor in the block element's minimum line-height (as + // defined in section 10.8.1 of the css2 spec) assuming that + // zeroEffectiveSpanBox is not set on the root span. This only happens + // in some cases in quirks mode: + // (1) if the root span contains non-whitespace text directly (this + // is handled by zeroEffectiveSpanBox + // (2) if this line has a ::marker + // (3) if this is the last line of an LI, DT, or DD element + // (The last line before a block also counts, but not before a + // BR) (NN4/IE5 quirk) + + // (1) and (2) above + bool applyMinLH = !zeroEffectiveSpanBox || mHasMarker; + bool isLastLine = + !mGotLineBox || (!mLineBox->IsLineWrapped() && !mLineEndsInBR); + if (!applyMinLH && isLastLine) { + nsIContent* blockContent = mRootSpan->mFrame->mFrame->GetContent(); + if (blockContent) { + // (3) above, if the last line of LI, DT, or DD + if (blockContent->IsAnyOfHTMLElements(nsGkAtoms::li, nsGkAtoms::dt, + nsGkAtoms::dd)) { + applyMinLH = true; + } + } + } + if (applyMinLH) { + if (psd->mHasNonemptyContent || + (preMode && ShouldApplyLineHeightInPreserveWhiteSpace(psd)) || + mHasMarker) { +#ifdef NOISY_BLOCKDIR_ALIGN + printf(" [span]==> adjusting min/maxBCoord: currentValues: %d,%d", + minBCoord, maxBCoord); +#endif + nscoord minimumLineBSize = mMinLineBSize; + nscoord blockStart = -nsLayoutUtils::GetCenteredFontBaseline( + fm, minimumLineBSize, lineWM.IsLineInverted()); + nscoord blockEnd = blockStart + minimumLineBSize; + + if (mStyleText->HasEffectiveTextEmphasis()) { + nscoord fontMaxHeight = fm->MaxHeight(); + nscoord emphasisHeight = + GetBSizeOfEmphasisMarks(spanFrame, inflation); + nscoord delta = fontMaxHeight + emphasisHeight - minimumLineBSize; + if (delta > 0) { + if (minimumLineBSize < fontMaxHeight) { + // If the leadings are negative, fill them first. + nscoord ascent = fm->MaxAscent(); + nscoord descent = fm->MaxDescent(); + if (lineWM.IsLineInverted()) { + std::swap(ascent, descent); + } + blockStart = -ascent; + blockEnd = descent; + delta = emphasisHeight; + } + LogicalSide side = mStyleText->TextEmphasisSide(lineWM); + if (side == eLogicalSideBStart) { + blockStart -= delta; + } else { + blockEnd += delta; + } + } + } + + if (blockStart < minBCoord) minBCoord = blockStart; + if (blockEnd > maxBCoord) maxBCoord = blockEnd; + +#ifdef NOISY_BLOCKDIR_ALIGN + printf(" new values: %d,%d\n", minBCoord, maxBCoord); +#endif +#ifdef NOISY_BLOCKDIR_ALIGN + printf( + " Used mMinLineBSize: %d, blockStart: %d, blockEnd: " + "%d\n", + mMinLineBSize, blockStart, blockEnd); +#endif + } else { + // XXX issues: + // [1] BR's on empty lines stop working + // [2] May not honor css2's notion of handling empty elements + // [3] blank lines in a pre-section ("\n") (handled with preMode) + + // XXX Are there other problems with this? +#ifdef NOISY_BLOCKDIR_ALIGN + printf( + " [span]==> zapping min/maxBCoord: currentValues: %d,%d " + "newValues: 0,0\n", + minBCoord, maxBCoord); +#endif + minBCoord = maxBCoord = 0; + } + } + } + + if ((minBCoord == BLOCKDIR_ALIGN_FRAMES_NO_MINIMUM) || + (maxBCoord == BLOCKDIR_ALIGN_FRAMES_NO_MAXIMUM)) { + minBCoord = maxBCoord = baselineBCoord; + } + + if (psd != mRootSpan && zeroEffectiveSpanBox) { +#ifdef NOISY_BLOCKDIR_ALIGN + printf(" [span]adjusting for zeroEffectiveSpanBox\n"); + printf( + " Original: minBCoord=%d, maxBCoord=%d, bSize=%d, ascent=%d, " + "logicalBSize=%d, topLeading=%d, bottomLeading=%d\n", + minBCoord, maxBCoord, spanFramePFD->mBounds.BSize(lineWM), + spanFramePFD->mAscent, psd->mLogicalBSize, psd->mBStartLeading, + psd->mBEndLeading); +#endif + nscoord goodMinBCoord = + spanFramePFD->mBorderPadding.BStart(lineWM) - psd->mBStartLeading; + nscoord goodMaxBCoord = goodMinBCoord + psd->mLogicalBSize; + + // For cases like the one in bug 714519 (text-decoration placement + // or making nsLineLayout::IsZeroBSize() handle + // vertical-align:top/bottom on a descendant of the line that's not + // a child of it), we want to treat elements that are + // vertical-align: top or bottom somewhat like children for the + // purposes of this quirk. To some extent, this is guessing, since + // they might end up being aligned anywhere. However, we'll guess + // that they'll be placed aligned with the top or bottom of this + // frame (as though this frame is the only thing in the line). + // (Guessing isn't unreasonable, since all we're doing is reducing the + // scope of a quirk and making the behavior more standards-like.) + if (maxStartBoxBSize > maxBCoord - minBCoord) { + // Distribute maxStartBoxBSize to ascent (baselineBCoord - minBCoord), and + // then to descent (maxBCoord - baselineBCoord) by adjusting minBCoord or + // maxBCoord, but not to exceed goodMinBCoord and goodMaxBCoord. + nscoord distribute = maxStartBoxBSize - (maxBCoord - minBCoord); + nscoord ascentSpace = std::max(minBCoord - goodMinBCoord, 0); + if (distribute > ascentSpace) { + distribute -= ascentSpace; + minBCoord -= ascentSpace; + nscoord descentSpace = std::max(goodMaxBCoord - maxBCoord, 0); + if (distribute > descentSpace) { + maxBCoord += descentSpace; + } else { + maxBCoord += distribute; + } + } else { + minBCoord -= distribute; + } + } + if (maxEndBoxBSize > maxBCoord - minBCoord) { + // Likewise, but preferring descent to ascent. + nscoord distribute = maxEndBoxBSize - (maxBCoord - minBCoord); + nscoord descentSpace = std::max(goodMaxBCoord - maxBCoord, 0); + if (distribute > descentSpace) { + distribute -= descentSpace; + maxBCoord += descentSpace; + nscoord ascentSpace = std::max(minBCoord - goodMinBCoord, 0); + if (distribute > ascentSpace) { + minBCoord -= ascentSpace; + } else { + minBCoord -= distribute; + } + } else { + maxBCoord += distribute; + } + } + + if (minBCoord > goodMinBCoord) { + nscoord adjust = minBCoord - goodMinBCoord; // positive + + // shrink the logical extents + psd->mLogicalBSize -= adjust; + psd->mBStartLeading -= adjust; + } + if (maxBCoord < goodMaxBCoord) { + nscoord adjust = goodMaxBCoord - maxBCoord; + psd->mLogicalBSize -= adjust; + psd->mBEndLeading -= adjust; + } + if (minBCoord > 0) { + // shrink the content by moving its block start down. This is tricky, + // since the block start is the 0 for many coordinates, so what we do is + // move everything else up. + spanFramePFD->mAscent -= minBCoord; // move the baseline up + spanFramePFD->mBounds.BSize(lineWM) -= + minBCoord; // move the block end up + psd->mBStartLeading += minBCoord; + *psd->mBaseline -= minBCoord; + + pfd = psd->mFirstFrame; + while (nullptr != pfd) { + pfd->mBounds.BStart(lineWM) -= minBCoord; // move all the children + // back up + pfd->mFrame->SetRect(lineWM, pfd->mBounds, ContainerSizeForSpan(psd)); + pfd = pfd->mNext; + } + maxBCoord -= minBCoord; // since minBCoord is in the frame's own + // coordinate system + minBCoord = 0; + } + if (maxBCoord < spanFramePFD->mBounds.BSize(lineWM)) { + nscoord adjust = spanFramePFD->mBounds.BSize(lineWM) - maxBCoord; + spanFramePFD->mBounds.BSize(lineWM) -= adjust; // move the bottom up + psd->mBEndLeading += adjust; + } +#ifdef NOISY_BLOCKDIR_ALIGN + printf( + " New: minBCoord=%d, maxBCoord=%d, bSize=%d, ascent=%d, " + "logicalBSize=%d, topLeading=%d, bottomLeading=%d\n", + minBCoord, maxBCoord, spanFramePFD->mBounds.BSize(lineWM), + spanFramePFD->mAscent, psd->mLogicalBSize, psd->mBStartLeading, + psd->mBEndLeading); +#endif + } + + psd->mMinBCoord = minBCoord; + psd->mMaxBCoord = maxBCoord; +#ifdef NOISY_BLOCKDIR_ALIGN + printf( + " [span]==> minBCoord=%d maxBCoord=%d delta=%d maxStartBoxBSize=%d " + "maxEndBoxBSize=%d\n", + minBCoord, maxBCoord, maxBCoord - minBCoord, maxStartBoxBSize, + maxEndBoxBSize); +#endif + if (maxStartBoxBSize > mMaxStartBoxBSize) { + mMaxStartBoxBSize = maxStartBoxBSize; + } + if (maxEndBoxBSize > mMaxEndBoxBSize) { + mMaxEndBoxBSize = maxEndBoxBSize; + } +} + +static void SlideSpanFrameRect(nsIFrame* aFrame, nscoord aDeltaWidth) { + // This should not use nsIFrame::MovePositionBy because it happens + // prior to relative positioning. In particular, because + // nsBlockFrame::PlaceLine calls aLineLayout.TrimTrailingWhiteSpace() + // prior to calling aLineLayout.RelativePositionFrames(). + nsPoint p = aFrame->GetPosition(); + p.x -= aDeltaWidth; + aFrame->SetPosition(p); +} + +bool nsLineLayout::TrimTrailingWhiteSpaceIn(PerSpanData* psd, + nscoord* aDeltaISize) { + PerFrameData* pfd = psd->mFirstFrame; + if (!pfd) { + *aDeltaISize = 0; + return false; + } + pfd = pfd->Last(); + while (nullptr != pfd) { +#ifdef REALLY_NOISY_TRIM + psd->mFrame->mFrame->ListTag(stdout); + printf(": attempting trim of "); + pfd->mFrame->ListTag(stdout); + printf("\n"); +#endif + PerSpanData* childSpan = pfd->mSpan; + WritingMode lineWM = mRootSpan->mWritingMode; + if (childSpan) { + // Maybe the child span has the trailing white-space in it? + if (TrimTrailingWhiteSpaceIn(childSpan, aDeltaISize)) { + nscoord deltaISize = *aDeltaISize; + if (deltaISize) { + // Adjust the child spans frame size + pfd->mBounds.ISize(lineWM) -= deltaISize; + if (psd != mRootSpan) { + // When the child span is not a direct child of the block + // we need to update the child spans frame rectangle + // because it most likely will not be done again. Spans + // that are direct children of the block will be updated + // later, however, because the VerticalAlignFrames method + // will be run after this method. + nsSize containerSize = ContainerSizeForSpan(childSpan); + nsIFrame* f = pfd->mFrame; + LogicalRect r(lineWM, f->GetRect(), containerSize); + r.ISize(lineWM) -= deltaISize; + f->SetRect(lineWM, r, containerSize); + } + + // Adjust the inline end edge of the span that contains the child span + psd->mICoord -= deltaISize; + + // Slide any frames that follow the child span over by the + // correct amount. The only thing that can follow the child + // span is empty stuff, so we are just making things + // sensible (keeping the combined area honest). + while (pfd->mNext) { + pfd = pfd->mNext; + pfd->mBounds.IStart(lineWM) -= deltaISize; + if (psd != mRootSpan) { + // When the child span is not a direct child of the block + // we need to update the child span's frame rectangle + // because it most likely will not be done again. Spans + // that are direct children of the block will be updated + // later, however, because the VerticalAlignFrames method + // will be run after this method. + SlideSpanFrameRect(pfd->mFrame, deltaISize); + } + } + } + return true; + } + } else if (!pfd->mIsTextFrame && !pfd->mSkipWhenTrimmingWhitespace) { + // If we hit a frame on the end that's not text and not a placeholder, + // then there is no trailing whitespace to trim. Stop the search. + *aDeltaISize = 0; + return true; + } else if (pfd->mIsTextFrame) { + // Call TrimTrailingWhiteSpace even on empty textframes because they + // might have a soft hyphen which should now appear, changing the frame's + // width + nsTextFrame::TrimOutput trimOutput = + static_cast<nsTextFrame*>(pfd->mFrame) + ->TrimTrailingWhiteSpace( + mLineContainerRI.mRenderingContext->GetDrawTarget()); +#ifdef NOISY_TRIM + psd->mFrame->mFrame->ListTag(stdout); + printf(": trim of "); + pfd->mFrame->ListTag(stdout); + printf(" returned %d\n", trimOutput.mDeltaWidth); +#endif + + if (trimOutput.mChanged) { + pfd->mRecomputeOverflow = true; + } + + // Delta width not being zero means that + // there is trimmed space in the frame. + if (trimOutput.mDeltaWidth) { + pfd->mBounds.ISize(lineWM) -= trimOutput.mDeltaWidth; + + // If any trailing space is trimmed, the justification opportunity + // generated by the space should be removed as well. + pfd->mJustificationInfo.CancelOpportunityForTrimmedSpace(); + + // See if the text frame has already been placed in its parent + if (psd != mRootSpan) { + // The frame was already placed during psd's + // reflow. Update the frames rectangle now. + pfd->mFrame->SetRect(lineWM, pfd->mBounds, ContainerSizeForSpan(psd)); + } + + // Adjust containing span's right edge + psd->mICoord -= trimOutput.mDeltaWidth; + + // Slide any frames that follow the text frame over by the + // right amount. The only thing that can follow the text + // frame is empty stuff, so we are just making things + // sensible (keeping the combined area honest). + while (pfd->mNext) { + pfd = pfd->mNext; + pfd->mBounds.IStart(lineWM) -= trimOutput.mDeltaWidth; + if (psd != mRootSpan) { + // When the child span is not a direct child of the block + // we need to update the child spans frame rectangle + // because it most likely will not be done again. Spans + // that are direct children of the block will be updated + // later, however, because the VerticalAlignFrames method + // will be run after this method. + SlideSpanFrameRect(pfd->mFrame, trimOutput.mDeltaWidth); + } + } + } + + if (pfd->mIsNonEmptyTextFrame || trimOutput.mChanged) { + // Pass up to caller so they can shrink their span + *aDeltaISize = trimOutput.mDeltaWidth; + return true; + } + } + pfd = pfd->mPrev; + } + + *aDeltaISize = 0; + return false; +} + +bool nsLineLayout::TrimTrailingWhiteSpace() { + PerSpanData* psd = mRootSpan; + nscoord deltaISize; + TrimTrailingWhiteSpaceIn(psd, &deltaISize); + return 0 != deltaISize; +} + +bool nsLineLayout::PerFrameData::ParticipatesInJustification() const { + if (mIsMarker || mIsEmpty || mSkipWhenTrimmingWhitespace) { + // Skip ::markers, empty frames, and placeholders + return false; + } + if (mIsTextFrame && !mIsNonWhitespaceTextFrame && + static_cast<nsTextFrame*>(mFrame)->IsAtEndOfLine()) { + // Skip trimmed whitespaces + return false; + } + return true; +} + +struct nsLineLayout::JustificationComputationState { + PerFrameData* mFirstParticipant; + PerFrameData* mLastParticipant; + // When we are going across a boundary of ruby base, i.e. entering + // one, leaving one, or both, the following fields will be set to + // the corresponding ruby base frame for handling ruby-align. + PerFrameData* mLastExitedRubyBase; + PerFrameData* mLastEnteredRubyBase; + + JustificationComputationState() + : mFirstParticipant(nullptr), + mLastParticipant(nullptr), + mLastExitedRubyBase(nullptr), + mLastEnteredRubyBase(nullptr) {} +}; + +static bool IsRubyAlignSpaceAround(nsIFrame* aRubyBase) { + return aRubyBase->StyleText()->mRubyAlign == StyleRubyAlign::SpaceAround; +} + +/** + * Assign justification gaps for justification + * opportunities across two frames. + */ +/* static */ +int nsLineLayout::AssignInterframeJustificationGaps( + PerFrameData* aFrame, JustificationComputationState& aState) { + PerFrameData* prev = aState.mLastParticipant; + MOZ_ASSERT(prev); + + auto& assign = aFrame->mJustificationAssignment; + auto& prevAssign = prev->mJustificationAssignment; + + if (aState.mLastExitedRubyBase || aState.mLastEnteredRubyBase) { + PerFrameData* exitedRubyBase = aState.mLastExitedRubyBase; + if (!exitedRubyBase || IsRubyAlignSpaceAround(exitedRubyBase->mFrame)) { + prevAssign.mGapsAtEnd = 1; + } else { + exitedRubyBase->mJustificationAssignment.mGapsAtEnd = 1; + } + + PerFrameData* enteredRubyBase = aState.mLastEnteredRubyBase; + if (!enteredRubyBase || IsRubyAlignSpaceAround(enteredRubyBase->mFrame)) { + assign.mGapsAtStart = 1; + } else { + enteredRubyBase->mJustificationAssignment.mGapsAtStart = 1; + } + + // We are no longer going across a ruby base boundary. + aState.mLastExitedRubyBase = nullptr; + aState.mLastEnteredRubyBase = nullptr; + return 1; + } + + const auto& info = aFrame->mJustificationInfo; + const auto& prevInfo = prev->mJustificationInfo; + if (!info.mIsStartJustifiable && !prevInfo.mIsEndJustifiable) { + return 0; + } + + if (!info.mIsStartJustifiable) { + prevAssign.mGapsAtEnd = 2; + assign.mGapsAtStart = 0; + } else if (!prevInfo.mIsEndJustifiable) { + prevAssign.mGapsAtEnd = 0; + assign.mGapsAtStart = 2; + } else { + prevAssign.mGapsAtEnd = 1; + assign.mGapsAtStart = 1; + } + return 1; +} + +/** + * Compute the justification info of the given span, and store the + * number of inner opportunities into the frame's justification info. + * It returns the number of non-inner opportunities it detects. + */ +int32_t nsLineLayout::ComputeFrameJustification( + PerSpanData* aPSD, JustificationComputationState& aState) { + NS_ASSERTION(aPSD, "null arg"); + NS_ASSERTION(!aState.mLastParticipant || !aState.mLastParticipant->mSpan, + "Last participant shall always be a leaf frame"); + bool firstChild = true; + int32_t& innerOpportunities = + aPSD->mFrame->mJustificationInfo.mInnerOpportunities; + MOZ_ASSERT(innerOpportunities == 0, + "Justification info should not have been set yet."); + int32_t outerOpportunities = 0; + + for (PerFrameData* pfd = aPSD->mFirstFrame; pfd; pfd = pfd->mNext) { + if (!pfd->ParticipatesInJustification()) { + continue; + } + + bool isRubyBase = pfd->mFrame->IsRubyBaseFrame(); + PerFrameData* outerRubyBase = aState.mLastEnteredRubyBase; + if (isRubyBase) { + aState.mLastEnteredRubyBase = pfd; + } + + int extraOpportunities = 0; + if (pfd->mSpan) { + PerSpanData* span = pfd->mSpan; + extraOpportunities = ComputeFrameJustification(span, aState); + innerOpportunities += pfd->mJustificationInfo.mInnerOpportunities; + } else { + if (pfd->mIsTextFrame) { + innerOpportunities += pfd->mJustificationInfo.mInnerOpportunities; + } + + if (!aState.mLastParticipant) { + aState.mFirstParticipant = pfd; + // It is not an empty ruby base, but we are not assigning gaps + // to the content for now. Clear the last entered ruby base so + // that we can correctly set the last exited ruby base. + aState.mLastEnteredRubyBase = nullptr; + } else { + extraOpportunities = AssignInterframeJustificationGaps(pfd, aState); + } + + aState.mLastParticipant = pfd; + } + + if (isRubyBase) { + if (aState.mLastEnteredRubyBase == pfd) { + // There is no justification participant inside this ruby base. + // Ignore this ruby base completely and restore the outer ruby + // base here. + aState.mLastEnteredRubyBase = outerRubyBase; + } else { + aState.mLastExitedRubyBase = pfd; + } + } + + if (firstChild) { + outerOpportunities = extraOpportunities; + firstChild = false; + } else { + innerOpportunities += extraOpportunities; + } + } + + return outerOpportunities; +} + +void nsLineLayout::AdvanceAnnotationInlineBounds(PerFrameData* aPFD, + const nsSize& aContainerSize, + nscoord aDeltaICoord, + nscoord aDeltaISize) { + nsIFrame* frame = aPFD->mFrame; + LayoutFrameType frameType = frame->Type(); + MOZ_ASSERT(frameType == LayoutFrameType::RubyText || + frameType == LayoutFrameType::RubyTextContainer); + MOZ_ASSERT(aPFD->mSpan, "rt and rtc should have span."); + + PerSpanData* psd = aPFD->mSpan; + WritingMode lineWM = mRootSpan->mWritingMode; + aPFD->mBounds.IStart(lineWM) += aDeltaICoord; + + // Check whether this expansion should be counted into the reserved + // isize or not. When it is a ruby text container, and it has some + // children linked to the base, it must not have reserved isize, + // or its children won't align with their bases. Otherwise, this + // expansion should be reserved. There are two cases a ruby text + // container does not have children linked to the base: + // 1. it is a container for span; 2. its children are collapsed. + // See bug 1055674 for the second case. + if (frameType == LayoutFrameType::RubyText || + // This ruby text container is a span. + (psd->mFirstFrame == psd->mLastFrame && psd->mFirstFrame && + !psd->mFirstFrame->mIsLinkedToBase)) { + // For ruby text frames, only increase frames + // which are not auto-hidden. + if (frameType != LayoutFrameType::RubyText || + !static_cast<nsRubyTextFrame*>(frame)->IsCollapsed()) { + nscoord reservedISize = RubyUtils::GetReservedISize(frame); + RubyUtils::SetReservedISize(frame, reservedISize + aDeltaISize); + } + } else { + // It is a normal ruby text container. Its children will expand + // themselves properly. We only need to expand its own size here. + aPFD->mBounds.ISize(lineWM) += aDeltaISize; + } + aPFD->mFrame->SetRect(lineWM, aPFD->mBounds, aContainerSize); +} + +/** + * This function applies the changes of icoord and isize caused by + * justification to annotations of the given frame. + */ +void nsLineLayout::ApplyLineJustificationToAnnotations(PerFrameData* aPFD, + nscoord aDeltaICoord, + nscoord aDeltaISize) { + PerFrameData* pfd = aPFD->mNextAnnotation; + while (pfd) { + nsSize containerSize = pfd->mFrame->GetParent()->GetSize(); + AdvanceAnnotationInlineBounds(pfd, containerSize, aDeltaICoord, + aDeltaISize); + + // There are two cases where an annotation frame has siblings which + // do not attached to a ruby base-level frame: + // 1. there's an intra-annotation whitespace which has no intra-base + // white-space to pair with; + // 2. there are not enough ruby bases to be paired with annotations. + // In these cases, their size should not be affected, but we still + // need to move them so that they won't overlap other frames. + PerFrameData* sibling = pfd->mNext; + while (sibling && !sibling->mIsLinkedToBase) { + AdvanceAnnotationInlineBounds(sibling, containerSize, + aDeltaICoord + aDeltaISize, 0); + sibling = sibling->mNext; + } + + pfd = pfd->mNextAnnotation; + } +} + +nscoord nsLineLayout::ApplyFrameJustification( + PerSpanData* aPSD, JustificationApplicationState& aState) { + NS_ASSERTION(aPSD, "null arg"); + + nscoord deltaICoord = 0; + for (PerFrameData* pfd = aPSD->mFirstFrame; pfd != nullptr; + pfd = pfd->mNext) { + nscoord dw = 0; + WritingMode lineWM = mRootSpan->mWritingMode; + const auto& assign = pfd->mJustificationAssignment; + bool isInlineText = + pfd->mIsTextFrame && !pfd->mWritingMode.IsOrthogonalTo(lineWM); + + // Don't apply justification if the frame doesn't participate. Same + // as the condition used in ComputeFrameJustification. Note that, + // we still need to move the frame based on deltaICoord even if the + // frame itself doesn't expand. + if (pfd->ParticipatesInJustification()) { + if (isInlineText) { + if (aState.IsJustifiable()) { + // Set corresponding justification gaps here, so that the + // text frame knows how it should add gaps at its sides. + const auto& info = pfd->mJustificationInfo; + auto textFrame = static_cast<nsTextFrame*>(pfd->mFrame); + textFrame->AssignJustificationGaps(assign); + dw = aState.Consume(JustificationUtils::CountGaps(info, assign)); + } + + if (dw) { + pfd->mRecomputeOverflow = true; + } + } else { + if (nullptr != pfd->mSpan) { + dw = ApplyFrameJustification(pfd->mSpan, aState); + } + } + } else { + MOZ_ASSERT(!assign.TotalGaps(), + "Non-participants shouldn't have assigned gaps"); + } + + pfd->mBounds.ISize(lineWM) += dw; + nscoord gapsAtEnd = 0; + if (!isInlineText && assign.TotalGaps()) { + // It is possible that we assign gaps to non-text frame or an + // orthogonal text frame. Apply the gaps as margin for them. + deltaICoord += aState.Consume(assign.mGapsAtStart); + gapsAtEnd = aState.Consume(assign.mGapsAtEnd); + dw += gapsAtEnd; + } + pfd->mBounds.IStart(lineWM) += deltaICoord; + + // The gaps added to the end of the frame should also be + // excluded from the isize added to the annotation. + ApplyLineJustificationToAnnotations(pfd, deltaICoord, dw - gapsAtEnd); + deltaICoord += dw; + pfd->mFrame->SetRect(lineWM, pfd->mBounds, ContainerSizeForSpan(aPSD)); + } + return deltaICoord; +} + +static nsIFrame* FindNearestRubyBaseAncestor(nsIFrame* aFrame) { + MOZ_ASSERT(aFrame->Style()->ShouldSuppressLineBreak()); + while (aFrame && !aFrame->IsRubyBaseFrame()) { + aFrame = aFrame->GetParent(); + } + // XXX It is possible that no ruby base ancestor is found because of + // some edge cases like form control or canvas inside ruby text. + // See bug 1138092 comment 4. + NS_WARNING_ASSERTION(aFrame, "no ruby base ancestor?"); + return aFrame; +} + +/** + * This method expands the given frame by the given reserved isize. + */ +void nsLineLayout::ExpandRubyBox(PerFrameData* aFrame, nscoord aReservedISize, + const nsSize& aContainerSize) { + WritingMode lineWM = mRootSpan->mWritingMode; + auto rubyAlign = aFrame->mFrame->StyleText()->mRubyAlign; + switch (rubyAlign) { + case StyleRubyAlign::Start: + // do nothing for start + break; + case StyleRubyAlign::SpaceBetween: + case StyleRubyAlign::SpaceAround: { + int32_t opportunities = aFrame->mJustificationInfo.mInnerOpportunities; + int32_t gaps = opportunities * 2; + if (rubyAlign == StyleRubyAlign::SpaceAround) { + // Each expandable ruby box with ruby-align space-around has a + // gap at each of its sides. For rb/rbc, see comment in + // AssignInterframeJustificationGaps; for rt/rtc, see comment + // in ExpandRubyBoxWithAnnotations. + gaps += 2; + } + if (gaps > 0) { + JustificationApplicationState state(gaps, aReservedISize); + ApplyFrameJustification(aFrame->mSpan, state); + break; + } + // If there are no justification opportunities for space-between, + // fall-through to center per spec. + [[fallthrough]]; + } + case StyleRubyAlign::Center: + // Indent all children by half of the reserved inline size. + for (PerFrameData* child = aFrame->mSpan->mFirstFrame; child; + child = child->mNext) { + child->mBounds.IStart(lineWM) += aReservedISize / 2; + child->mFrame->SetRect(lineWM, child->mBounds, aContainerSize); + } + break; + default: + MOZ_ASSERT_UNREACHABLE("Unknown ruby-align value"); + } + + aFrame->mBounds.ISize(lineWM) += aReservedISize; + aFrame->mFrame->SetRect(lineWM, aFrame->mBounds, aContainerSize); +} + +/** + * This method expands the given frame by the reserved inline size. + * It also expands its annotations if they are expandable and have + * reserved isize larger than zero. + */ +void nsLineLayout::ExpandRubyBoxWithAnnotations(PerFrameData* aFrame, + const nsSize& aContainerSize) { + nscoord reservedISize = RubyUtils::GetReservedISize(aFrame->mFrame); + if (reservedISize) { + ExpandRubyBox(aFrame, reservedISize, aContainerSize); + } + + WritingMode lineWM = mRootSpan->mWritingMode; + bool isLevelContainer = aFrame->mFrame->IsRubyBaseContainerFrame(); + for (PerFrameData* annotation = aFrame->mNextAnnotation; annotation; + annotation = annotation->mNextAnnotation) { + if (lineWM.IsOrthogonalTo(annotation->mFrame->GetWritingMode())) { + // Inter-character case: don't attempt to expand ruby annotations. + continue; + } + if (isLevelContainer) { + nsIFrame* rtcFrame = annotation->mFrame; + MOZ_ASSERT(rtcFrame->IsRubyTextContainerFrame()); + // It is necessary to set the rect again because the container + // width was unknown, and zero was used instead when we reflow + // them. The corresponding base containers were repositioned in + // VerticalAlignFrames and PlaceTopBottomFrames. + MOZ_ASSERT(rtcFrame->GetLogicalSize(lineWM) == + annotation->mBounds.Size(lineWM)); + rtcFrame->SetPosition(lineWM, annotation->mBounds.Origin(lineWM), + aContainerSize); + } + + nscoord reservedISize = RubyUtils::GetReservedISize(annotation->mFrame); + if (!reservedISize) { + continue; + } + + MOZ_ASSERT(annotation->mSpan); + JustificationComputationState computeState; + ComputeFrameJustification(annotation->mSpan, computeState); + if (!computeState.mFirstParticipant) { + continue; + } + if (IsRubyAlignSpaceAround(annotation->mFrame)) { + // Add one gap at each side of this annotation. + computeState.mFirstParticipant->mJustificationAssignment.mGapsAtStart = 1; + computeState.mLastParticipant->mJustificationAssignment.mGapsAtEnd = 1; + } + nsIFrame* parentFrame = annotation->mFrame->GetParent(); + nsSize containerSize = parentFrame->GetSize(); + MOZ_ASSERT(containerSize == aContainerSize || + parentFrame->IsRubyTextContainerFrame(), + "Container width should only be different when the current " + "annotation is a ruby text frame, whose parent is not same " + "as its base frame."); + ExpandRubyBox(annotation, reservedISize, containerSize); + ExpandInlineRubyBoxes(annotation->mSpan); + } +} + +/** + * This method looks for all expandable ruby box in the given span, and + * calls ExpandRubyBox to expand them in depth-first preorder. + */ +void nsLineLayout::ExpandInlineRubyBoxes(PerSpanData* aSpan) { + nsSize containerSize = ContainerSizeForSpan(aSpan); + for (PerFrameData* pfd = aSpan->mFirstFrame; pfd; pfd = pfd->mNext) { + if (RubyUtils::IsExpandableRubyBox(pfd->mFrame)) { + ExpandRubyBoxWithAnnotations(pfd, containerSize); + } + if (pfd->mSpan) { + ExpandInlineRubyBoxes(pfd->mSpan); + } + } +} + +nscoord nsLineLayout::GetHangFrom(const PerSpanData* aSpan, + bool aLineIsRTL) const { + const PerFrameData* pfd = aSpan->mLastFrame; + nscoord result = 0; + while (pfd) { + if (const PerSpanData* childSpan = pfd->mSpan) { + return GetHangFrom(childSpan, aLineIsRTL); + } + if (pfd->mIsTextFrame) { + auto* lastText = static_cast<nsTextFrame*>(pfd->mFrame); + result = lastText->GetHangableISize(); + if (result) { + // If the hangable space will be at the start edge of the line, due to + // its bidi direction being against the line direction, we flag this by + // negating the advance. + lastText->EnsureTextRun(nsTextFrame::eInflated); + auto* textRun = lastText->GetTextRun(nsTextFrame::eInflated); + if (textRun && textRun->IsRightToLeft() != aLineIsRTL) { + result = -result; + } + } + return result; + } + if (!pfd->mSkipWhenTrimmingWhitespace) { + // If we hit a frame on the end that's not text and not a placeholder or + // <br>, then there is no trailing whitespace to hang. Stop the search. + return result; + } + // Scan back for a preceding frame whose whitespace we can hang. + pfd = pfd->mPrev; + } + return result; +} + +gfxTextRun::TrimmableWS nsLineLayout::GetTrimFrom(const PerSpanData* aSpan, + bool aLineIsRTL) const { + const PerFrameData* pfd = aSpan->mLastFrame; + while (pfd) { + if (const PerSpanData* childSpan = pfd->mSpan) { + return GetTrimFrom(childSpan, aLineIsRTL); + } + if (pfd->mIsTextFrame) { + auto* lastText = static_cast<nsTextFrame*>(pfd->mFrame); + auto result = lastText->GetTrimmableWS(); + if (result.mAdvance) { + lastText->EnsureTextRun(nsTextFrame::eInflated); + auto* textRun = lastText->GetTextRun(nsTextFrame::eInflated); + if (textRun && textRun->IsRightToLeft() != aLineIsRTL) { + result.mAdvance = -result.mAdvance; + } + } + return result; + } + if (!pfd->mSkipWhenTrimmingWhitespace) { + // If we hit a frame on the end that's not text and not a placeholder or + // <br>, then there is no trailing whitespace to trim. Stop the search. + return gfxTextRun::TrimmableWS{}; + } + // Scan back for a preceding frame whose whitespace we can trim. + pfd = pfd->mPrev; + } + return gfxTextRun::TrimmableWS{}; +} + +// Align inline frames within the line according to the CSS text-align +// property. +void nsLineLayout::TextAlignLine(nsLineBox* aLine, bool aIsLastLine) { + /** + * NOTE: aIsLastLine ain't necessarily so: it is correctly set by caller + * only in cases where the last line needs special handling. + */ + PerSpanData* psd = mRootSpan; + WritingMode lineWM = psd->mWritingMode; + LAYOUT_WARN_IF_FALSE(psd->mIEnd != NS_UNCONSTRAINEDSIZE, + "have unconstrained width; this should only result from " + "very large sizes, not attempts at intrinsic width " + "calculation"); + nscoord availISize = psd->mIEnd - psd->mIStart; + nscoord remainingISize = availISize - aLine->ISize(); +#ifdef NOISY_INLINEDIR_ALIGN + LineContainerFrame()->ListTag(stdout); + printf(": availISize=%d lineBounds.IStart=%d lineISize=%d delta=%d\n", + availISize, aLine->IStart(), aLine->ISize(), remainingISize); +#endif + + nscoord dx = 0; + StyleTextAlign textAlign = + aIsLastLine ? mStyleText->TextAlignForLastLine() : mStyleText->mTextAlign; + + // Check if there's trailing whitespace we need to "hang" at line-wrap. + nscoord hang = 0; + uint32_t trimCount = 0; + if (aLine->IsLineWrapped()) { + if (textAlign == StyleTextAlign::Justify) { + auto trim = GetTrimFrom(mRootSpan, lineWM.IsBidiRTL()); + hang = NSToCoordRound(trim.mAdvance); + trimCount = trim.mCount; + } else { + hang = GetHangFrom(mRootSpan, lineWM.IsBidiRTL()); + } + } + + bool isSVG = LineContainerFrame()->IsInSVGTextSubtree(); + bool doTextAlign = remainingISize > 0 || hang != 0; + + int32_t additionalGaps = 0; + if (!isSVG && + (mHasRuby || (doTextAlign && textAlign == StyleTextAlign::Justify))) { + JustificationComputationState computeState; + ComputeFrameJustification(psd, computeState); + if (mHasRuby && computeState.mFirstParticipant) { + PerFrameData* firstFrame = computeState.mFirstParticipant; + if (firstFrame->mFrame->Style()->ShouldSuppressLineBreak()) { + MOZ_ASSERT(!firstFrame->mJustificationAssignment.mGapsAtStart); + nsIFrame* rubyBase = FindNearestRubyBaseAncestor(firstFrame->mFrame); + if (rubyBase && IsRubyAlignSpaceAround(rubyBase)) { + firstFrame->mJustificationAssignment.mGapsAtStart = 1; + additionalGaps++; + } + } + PerFrameData* lastFrame = computeState.mLastParticipant; + if (lastFrame->mFrame->Style()->ShouldSuppressLineBreak()) { + MOZ_ASSERT(!lastFrame->mJustificationAssignment.mGapsAtEnd); + nsIFrame* rubyBase = FindNearestRubyBaseAncestor(lastFrame->mFrame); + if (rubyBase && IsRubyAlignSpaceAround(rubyBase)) { + lastFrame->mJustificationAssignment.mGapsAtEnd = 1; + additionalGaps++; + } + } + } + } + + if (!isSVG && doTextAlign) { + switch (textAlign) { + case StyleTextAlign::Justify: { + int32_t opportunities = + psd->mFrame->mJustificationInfo.mInnerOpportunities - + (hang ? trimCount : 0); + if (opportunities > 0) { + int32_t gaps = opportunities * 2 + additionalGaps; + remainingISize += std::abs(hang); + JustificationApplicationState applyState(gaps, remainingISize); + + // Apply the justification, and make sure to update our linebox + // width to account for it. + aLine->ExpandBy(ApplyFrameJustification(psd, applyState), + ContainerSizeForSpan(psd)); + + // If the trimmable trailing whitespace that we want to hang had + // reverse-inline directionality, adjust line position to account for + // it being at the inline-start side. + // On top of the original "hang" amount, justification will have + // modified its width, so we include that adjustment here. + if (hang < 0) { + dx = hang - trimCount * remainingISize / opportunities; + } + + // Gaps that belong to trimmed whitespace were not included in the + // applyState count, so we need to add them here for the assert. + DebugOnly<int32_t> trimmedGaps = hang ? trimCount * 2 : 0; + MOZ_ASSERT(applyState.mGaps.mHandled == + applyState.mGaps.mCount + trimmedGaps, + "Unprocessed justification gaps"); + // Similarly, account for the adjustment applied to the trimmed + // whitespace, which is in addition to the adjustment that applies + // within the actual width of the line. + DebugOnly<int32_t> trimmedAdjustment = + trimCount * remainingISize / opportunities; + NS_ASSERTION(applyState.mWidth.mConsumed == + applyState.mWidth.mAvailable + trimmedAdjustment, + "Unprocessed justification width"); + break; + } + // Fall through to the default case if we could not justify to fill + // the space. + [[fallthrough]]; + } + + case StyleTextAlign::Start: + case StyleTextAlign::Char: + // Default alignment is to start edge so do nothing, except to apply + // any "reverse-hang" amount resulting from reversed-direction trailing + // space. + // Char is for tables so treat as start if we find it in block layout. + if (hang < 0) { + dx = hang; + } + break; + + case StyleTextAlign::Left: + case StyleTextAlign::MozLeft: + if (lineWM.IsBidiRTL()) { + dx = remainingISize + (hang > 0 ? hang : 0); + } else if (hang < 0) { + dx = hang; + } + break; + + case StyleTextAlign::Right: + case StyleTextAlign::MozRight: + if (lineWM.IsBidiLTR()) { + dx = remainingISize + (hang > 0 ? hang : 0); + } else if (hang < 0) { + dx = hang; + } + break; + + case StyleTextAlign::End: + dx = remainingISize + (hang > 0 ? hang : 0); + break; + + case StyleTextAlign::Center: + case StyleTextAlign::MozCenter: + dx = (remainingISize + hang) / 2; + break; + } + } + + if (mHasRuby) { + ExpandInlineRubyBoxes(mRootSpan); + } + + PerFrameData* startFrame = psd->mFirstFrame; + MOZ_ASSERT(startFrame, "empty line?"); + if (startFrame->mIsMarker) { + // ::marker shouldn't participate in bidi reordering nor text alignment. + startFrame = startFrame->mNext; + MOZ_ASSERT(startFrame, "no frame after ::marker?"); + MOZ_ASSERT(!startFrame->mIsMarker, "multiple ::markers?"); + } + + const bool bidi = mPresContext->BidiEnabled() && + (!mPresContext->IsVisualMode() || lineWM.IsBidiRTL()); + if (bidi) { + nsBidiPresUtils::ReorderFrames(startFrame->mFrame, aLine->GetChildCount(), + lineWM, mContainerSize, + psd->mIStart + mTextIndent + dx); + } + + if (dx) { + // For the bidi case, if startFrame is a ::first-line frame, the mIStart and + // mTextIndent offsets will already have been applied to its position, but + // we still need to apply the text-align adjustment |dx| to its position. + const bool needToAdjustFrames = !bidi || startFrame->mFrame->IsLineFrame(); + MOZ_ASSERT_IF(startFrame->mFrame->IsLineFrame(), !startFrame->mNext); + if (needToAdjustFrames) { + for (PerFrameData* pfd = startFrame; pfd; pfd = pfd->mNext) { + pfd->mBounds.IStart(lineWM) += dx; + pfd->mFrame->SetRect(lineWM, pfd->mBounds, ContainerSizeForSpan(psd)); + } + } + aLine->IndentBy(dx, ContainerSize()); + } +} + +// This method applies any relative positioning to the given frame. +void nsLineLayout::ApplyRelativePositioning(PerFrameData* aPFD) { + if (!aPFD->mIsRelativelyOrStickyPos) { + return; + } + + nsIFrame* frame = aPFD->mFrame; + WritingMode frameWM = aPFD->mWritingMode; + LogicalPoint origin = frame->GetLogicalPosition(ContainerSize()); + // right and bottom are handled by + // ReflowInput::ComputeRelativeOffsets + ReflowInput::ApplyRelativePositioning(frame, frameWM, aPFD->mOffsets, &origin, + ContainerSize()); + frame->SetPosition(frameWM, origin, ContainerSize()); +} + +// This method do relative positioning for ruby annotations. +void nsLineLayout::RelativePositionAnnotations(PerSpanData* aRubyPSD, + OverflowAreas& aOverflowAreas) { + MOZ_ASSERT(aRubyPSD->mFrame->mFrame->IsRubyFrame()); + for (PerFrameData* pfd = aRubyPSD->mFirstFrame; pfd; pfd = pfd->mNext) { + MOZ_ASSERT(pfd->mFrame->IsRubyBaseContainerFrame()); + for (PerFrameData* rtc = pfd->mNextAnnotation; rtc; + rtc = rtc->mNextAnnotation) { + nsIFrame* rtcFrame = rtc->mFrame; + MOZ_ASSERT(rtcFrame->IsRubyTextContainerFrame()); + ApplyRelativePositioning(rtc); + OverflowAreas rtcOverflowAreas; + RelativePositionFrames(rtc->mSpan, rtcOverflowAreas); + aOverflowAreas.UnionWith(rtcOverflowAreas + rtcFrame->GetPosition()); + } + } +} + +void nsLineLayout::RelativePositionFrames(PerSpanData* psd, + OverflowAreas& aOverflowAreas) { + OverflowAreas overflowAreas; + WritingMode wm = psd->mWritingMode; + if (psd != mRootSpan) { + // The span's overflow areas come in three parts: + // -- this frame's width and height + // -- pfd->mOverflowAreas, which is the area of a ::marker or the union + // of a relatively positioned frame's absolute children + // -- the bounds of all inline descendants + // The former two parts are computed right here, we gather the descendants + // below. + // At this point psd->mFrame->mBounds might be out of date since + // bidi reordering can move and resize the frames. So use the frame's + // rect instead of mBounds. + nsRect adjustedBounds(nsPoint(0, 0), psd->mFrame->mFrame->GetSize()); + + overflowAreas.ScrollableOverflow().UnionRect( + psd->mFrame->mOverflowAreas.ScrollableOverflow(), adjustedBounds); + overflowAreas.InkOverflow().UnionRect( + psd->mFrame->mOverflowAreas.InkOverflow(), adjustedBounds); + } else { + LogicalRect rect(wm, psd->mIStart, mBStartEdge, psd->mICoord - psd->mIStart, + mFinalLineBSize); + // The minimum combined area for the frames that are direct + // children of the block starts at the upper left corner of the + // line and is sized to match the size of the line's bounding box + // (the same size as the values returned from VerticalAlignFrames) + overflowAreas.InkOverflow() = rect.GetPhysicalRect(wm, ContainerSize()); + overflowAreas.ScrollableOverflow() = overflowAreas.InkOverflow(); + } + + for (PerFrameData* pfd = psd->mFirstFrame; pfd; pfd = pfd->mNext) { + nsIFrame* frame = pfd->mFrame; + + // Adjust the origin of the frame + ApplyRelativePositioning(pfd); + + // We must position the view correctly before positioning its + // descendants so that widgets are positioned properly (since only + // some views have widgets). + if (frame->HasView()) + nsContainerFrame::SyncFrameViewAfterReflow( + mPresContext, frame, frame->GetView(), + pfd->mOverflowAreas.InkOverflow(), + nsIFrame::ReflowChildFlags::NoSizeView); + + // Note: the combined area of a child is in its coordinate + // system. We adjust the childs combined area into our coordinate + // system before computing the aggregated value by adding in + // <b>x</b> and <b>y</b> which were computed above. + OverflowAreas r; + if (pfd->mSpan) { + // Compute a new combined area for the child span before + // aggregating it into our combined area. + RelativePositionFrames(pfd->mSpan, r); + } else { + r = pfd->mOverflowAreas; + if (pfd->mIsTextFrame) { + // We need to recompute overflow areas in four cases: + // (1) When PFD_RECOMPUTEOVERFLOW is set due to trimming + // (2) When there are text decorations, since we can't recompute the + // overflow area until Reflow and VerticalAlignLine have finished + // (3) When there are text emphasis marks, since the marks may be + // put further away if the text is inside ruby. + // (4) When there are text strokes + if (pfd->mRecomputeOverflow || + frame->Style()->HasTextDecorationLines() || + frame->StyleText()->HasEffectiveTextEmphasis() || + frame->StyleText()->HasWebkitTextStroke()) { + nsTextFrame* f = static_cast<nsTextFrame*>(frame); + r = f->RecomputeOverflow(LineContainerFrame()); + } + frame->FinishAndStoreOverflow(r, frame->GetSize()); + } + + // If we have something that's not an inline but with a complex frame + // hierarchy inside that contains views, they need to be + // positioned. + // All descendant views must be repositioned even if this frame + // does have a view in case this frame's view does not have a + // widget and some of the descendant views do have widgets -- + // otherwise the widgets won't be repositioned. + nsContainerFrame::PositionChildViews(frame); + } + + // Do this here (rather than along with setting the overflow rect + // below) so we get leaf frames as well. No need to worry + // about the root span, since it doesn't have a frame. + if (frame->HasView()) + nsContainerFrame::SyncFrameViewAfterReflow( + mPresContext, frame, frame->GetView(), r.InkOverflow(), + nsIFrame::ReflowChildFlags::NoMoveView); + + overflowAreas.UnionWith(r + frame->GetPosition()); + } + + // Also compute relative position in the annotations. + if (psd->mFrame->mFrame->IsRubyFrame()) { + RelativePositionAnnotations(psd, overflowAreas); + } + + // If we just computed a spans combined area, we need to update its + // overflow rect... + if (psd != mRootSpan) { + PerFrameData* spanPFD = psd->mFrame; + nsIFrame* frame = spanPFD->mFrame; + frame->FinishAndStoreOverflow(overflowAreas, frame->GetSize()); + } + aOverflowAreas = overflowAreas; +} |