gecko-dev/layout/generic/nsInlineFrame.cpp
1998-09-15 00:19:49 +00:00

1090 lines
35 KiB
C++

/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
*
* The contents of this file are subject to the Netscape Public License
* Version 1.0 (the "NPL"); you may not use this file except in
* compliance with the NPL. You may obtain a copy of the NPL at
* http://www.mozilla.org/NPL/
*
* Software distributed under the NPL is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
* for the specific language governing rights and limitations under the
* NPL.
*
* The Initial Developer of this code under the NPL is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
* Reserved.
*/
#include "nsInlineFrame.h"
#include "nsCSSLayout.h"
#include "nsIReflowCommand.h"
#include "nsIStyleContext.h"
#include "nsPlaceholderFrame.h"
#include "nsHTMLIIDs.h"// XXX
#include "nsHTMLBase.h"// XXX rename to nsBase?
// XXX TODO:
// test: <b>...<br></b> : make sure that it works in pullup case and
// non-pullup case
// content appended INCREMENTAL reflow
// content inserted INCREMENTAL reflow
// content deleted INCREMENTAL reflow
nsInlineReflowState::nsInlineReflowState(nsLineLayout& aLineLayout,
nsInlineFrame* aInlineFrame,
nsIStyleContext* aInlineSC,
const nsReflowState& aRS,
PRBool aComputeMaxElementSize)
: nsReflowState(aRS),
mInlineLayout(aLineLayout, aInlineFrame, aInlineSC)
{
// While we skip around the reflow state that our parent gave us so
// that the parentReflowState is linked properly, we don't want to
// skip over it's reason.
reason = aRS.reason;
mPresContext = aLineLayout.mPresContext;
mLastChild = nsnull;
const nsStyleText* styleText = (const nsStyleText*)
aInlineSC->GetStyleData(eStyleStruct_Text);
mNoWrap = PRBool(NS_STYLE_WHITESPACE_NORMAL != styleText->mWhiteSpace);
const nsStyleSpacing* styleSpacing = (const nsStyleSpacing*)
aInlineSC->GetStyleData(eStyleStruct_Spacing);
styleSpacing->CalcBorderPaddingFor(aInlineFrame, mBorderPadding);
// If we're constrained adjust the available size so it excludes space
// needed for border/padding
nscoord width = maxSize.width;
if (NS_UNCONSTRAINEDSIZE != width) {
width -= mBorderPadding.left + mBorderPadding.right;
}
// XXX I don't think we need this!
nscoord height = maxSize.height;
if (NS_UNCONSTRAINEDSIZE != height) {
height -= mBorderPadding.top + mBorderPadding.bottom;
}
PRIntn ss = mStyleSizeFlags =
nsCSSLayout::GetStyleSize(mPresContext, *this, mStyleSize);
if (0 != (ss & NS_SIZE_HAS_WIDTH)) {
// When we are given a width, change the reflow behavior to
// unconstrained.
// XXX is this right?
width = NS_UNCONSTRAINEDSIZE;
}
mInlineLayout.Init(this);
mInlineLayout.Prepare(PRBool(NS_UNCONSTRAINEDSIZE == width),
mNoWrap, aComputeMaxElementSize);
mInlineLayout.SetReflowSpace(mBorderPadding.left, mBorderPadding.top,
width, height/* XXX height??? */);
}
nsInlineReflowState::~nsInlineReflowState()
{
}
//----------------------------------------------------------------------
nsresult NS_NewInlineFrame(nsIFrame** aInstancePtrResult,
nsIContent* aContent,
nsIFrame* aParent)
{
NS_PRECONDITION(nsnull != aInstancePtrResult, "null ptr");
if (nsnull == aInstancePtrResult) {
return NS_ERROR_NULL_POINTER;
}
nsIFrame* it = new nsInlineFrame(aContent, aParent);
if (nsnull == it) {
return NS_ERROR_OUT_OF_MEMORY;
}
*aInstancePtrResult = it;
return NS_OK;
}
nsInlineFrame::nsInlineFrame(nsIContent* aContent, nsIFrame* aParent)
: nsHTMLContainerFrame(aContent, aParent)
{
}
nsInlineFrame::~nsInlineFrame()
{
}
NS_IMETHODIMP
nsInlineFrame::QueryInterface(REFNSIID aIID, void** aInstancePtrResult)
{
NS_PRECONDITION(nsnull != aInstancePtrResult, "null pointer");
if (nsnull == aInstancePtrResult) {
return NS_ERROR_NULL_POINTER;
}
if (aIID.Equals(kIInlineReflowIID)) {
*aInstancePtrResult = (void*) ((nsIInlineReflow*)this);
return NS_OK;
}
return nsFrame::QueryInterface(aIID, aInstancePtrResult);
}
PRIntn
nsInlineFrame::GetSkipSides() const
{
PRIntn skip = 0;
if (nsnull != mPrevInFlow) {
skip |= 1 << NS_SIDE_LEFT;
}
if (nsnull != mNextInFlow) {
skip |= 1 << NS_SIDE_RIGHT;
}
return skip;
}
nsresult
nsInlineFrame::AppendNewFrames(nsIPresContext* aPresContext,
nsIFrame* aNewFrame)
{
nsIFrame* lastFrame;
if (nsnull == mFirstChild) {
lastFrame = nsnull;
mFirstChild = aNewFrame;
} else {
LastChild(lastFrame);
lastFrame->SetNextSibling(aNewFrame);
}
mChildCount += LengthOf(aNewFrame);
// Now walk the new frames and check if there are any elements that want
// to be floated
nsIFrame* prevFrame = lastFrame;
for (nsIFrame* frame = aNewFrame; nsnull != frame; frame->GetNextSibling(frame)) {
const nsStyleDisplay* display;
frame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct*&)display);
// See if the element wants to be floated
if (NS_STYLE_FLOAT_NONE != display->mFloats) {
// Create a placeholder frame that will serve as the anchor point.
nsPlaceholderFrame* placeholder =
CreatePlaceholderFrame(aPresContext, frame);
// Remove the floated element from the flow, and replace it with the
// placeholder frame
if (nsnull == prevFrame) {
mFirstChild = placeholder;
} else {
prevFrame->SetNextSibling(placeholder);
}
nsIFrame* nextSibling;
frame->GetNextSibling(nextSibling);
placeholder->SetNextSibling(nextSibling);
frame->SetNextSibling(nsnull);
frame = placeholder;
}
// Remember the previous frame
prevFrame = frame;
}
return NS_OK;
}
NS_IMETHODIMP
nsInlineFrame::Init(nsIPresContext& aPresContext, nsIFrame* aChildList)
{
NS_PRECONDITION(nsnull == mFirstChild, "already initialized");
return AppendNewFrames(&aPresContext, aChildList);
}
NS_IMETHODIMP
nsInlineFrame::CreateContinuingFrame(nsIPresContext& aCX,
nsIFrame* aParent,
nsIStyleContext* aStyleContext,
nsIFrame*& aContinuingFrame)
{
nsInlineFrame* cf = new nsInlineFrame(mContent, aParent);
if (nsnull == cf) {
return NS_ERROR_OUT_OF_MEMORY;
}
PrepareContinuingFrame(aCX, aParent, aStyleContext, cf);
aContinuingFrame = cf;
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("nsInlineFrame::CreateContinuingFrame: newFrame=%p", cf));
return NS_OK;
}
PRBool
nsInlineFrame::DeleteNextInFlowsFor(nsIPresContext& aPresContext, nsIFrame* aChild)
{
return DeleteChildsNextInFlow(aPresContext, aChild);
}
NS_IMETHODIMP
nsInlineFrame::FindTextRuns(nsLineLayout& aLineLayout,
nsIReflowCommand* aReflowCommand)
{
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("enter nsInlineFrame::FindTextRuns [%d,%d,%c]",
mFirstContentOffset, mLastContentOffset, mLastContentIsComplete?'T':'F'));
nsIFrame* frame;
PRInt32 n;
nsresult rv = NS_OK;
// Gather up children from the overflow lists
DrainOverflowLists();
// XXX CONSTRUCTION
// Create new frames if necessary
if (NS_FRAME_FIRST_REFLOW & mState) {
if ((nsnull == mPrevInFlow) && (nsnull == mNextInFlow) &&
(0 == mChildCount)) {
rv = CreateNewFrames(aLineLayout.mPresContext);
if (NS_OK != rv) {
goto done;
}
}
}
else if (nsnull != aReflowCommand) {
nsIFrame* target;
aReflowCommand->GetTarget(target);
if (this == target) {
nsIReflowCommand::ReflowType type;
aReflowCommand->GetType(type);
if (nsIReflowCommand::FrameAppended == type) {
rv = CreateNewFrames(aLineLayout.mPresContext);
if (NS_OK != rv) {
goto done;
}
}
}
}
// Ask each child frame for its text runs
frame = mFirstChild;
n = mChildCount;
while (--n >= 0) {
nsIInlineReflow* inlineReflow;
if (NS_OK == frame->QueryInterface(kIInlineReflowIID,
(void**)&inlineReflow)) {
rv = inlineReflow->FindTextRuns(aLineLayout, aReflowCommand);
if (NS_OK != rv) {
return rv;
}
}
else {
// A frame that doesn't implement nsIInlineReflow isn't text
// therefore it will end an open text run.
aLineLayout.EndTextRun();
}
frame->GetNextSibling(frame);
}
done:;
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("exit nsInlineFrame::FindTextRuns rv=%x [%d,%d,%c]",
rv, mFirstContentOffset, mLastContentOffset,
mLastContentIsComplete?'T':'F'));
return rv;
}
NS_IMETHODIMP
nsInlineFrame::InlineReflow(nsLineLayout& aLineLayout,
nsReflowMetrics& aMetrics,
const nsReflowState& aReflowState)
{
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("enter nsInlineFrame::InlineReflow maxSize=%d,%d reason=%d kids=%d nif=%p [%d,%d,%c]",
aReflowState.maxSize.width, aReflowState.maxSize.height,
aReflowState.reason, mChildCount, mNextInFlow,
mFirstContentOffset, mLastContentOffset, mLastContentIsComplete?'T':'F'));
// If this is the initial reflow, generate any synthetic content
// that needs generating.
if (eReflowReason_Initial == aReflowState.reason) {
NS_ASSERTION(0 != (NS_FRAME_FIRST_REFLOW & mState), "bad mState");
}
else {
NS_ASSERTION(0 == (NS_FRAME_FIRST_REFLOW & mState), "bad mState");
}
nsInlineReflowState state(aLineLayout, this, mStyleContext,
aReflowState,
PRBool(nsnull != aMetrics.maxElementSize));
nsresult rv = NS_OK;
if (eReflowReason_Initial == state.reason) {
DrainOverflowLists();
rv = InitialReflow(state);
mState &= ~NS_FRAME_FIRST_REFLOW;
}
else if (eReflowReason_Incremental == state.reason) {
// XXX For now we drain our overflow list in case our parent
// reflowed our prev-in-flow and our prev-in-flow pushed some
// children forward to us (e.g. a speculative pullup from us that
// failed)
DrainOverflowLists();
NS_ASSERTION(nsnull == mOverflowList, "unexpected overflow list");
nsIFrame* target;
state.reflowCommand->GetTarget(target);
if (this == target) {
nsIReflowCommand::ReflowType type;
state.reflowCommand->GetType(type);
switch (type) {
case nsIReflowCommand::FrameAppended:
rv = FrameAppendedReflow(state);
break;
default:
// XXX For now map the other incremental operations into full reflows
rv = ResizeReflow(state);
break;
}
}
else {
// Get next frame in reflow command chain
state.reflowCommand->GetNext(state.mInlineLayout.mNextRCFrame);
// Continue the reflow
rv = ChildIncrementalReflow(state);
}
}
else if (eReflowReason_Resize == state.reason) {
DrainOverflowLists();
rv = ResizeReflow(state);
}
ComputeFinalSize(state, aMetrics);
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("exit nsInlineFrame::InlineReflow size=%d,%d rv=%x kids=%d nif=%p [%d,%d,%c]",
aMetrics.width, aMetrics.height, rv, mChildCount, mNextInFlow,
mFirstContentOffset, mLastContentOffset,
mLastContentIsComplete?'T':'F'));
return rv;
}
void
nsInlineFrame::ComputeFinalSize(nsInlineReflowState& aState,
nsReflowMetrics& aMetrics)
{
// Align our child frames. Note that inline frames "shrink wrap"
// around their contents therefore we need to fixup the available
// width in nsInlineLayout so that it doesn't do any horizontal
// alignment.
nsRect bounds;
aState.mInlineLayout.mAvailWidth =
aState.mInlineLayout.mX - aState.mInlineLayout.mLeftEdge;
aState.mInlineLayout.AlignFrames(mFirstChild, mChildCount, bounds);
// Compute the final width. Use the style width when supplied.
PRIntn ss = aState.mStyleSizeFlags;
if (0 != (ss & NS_SIZE_HAS_WIDTH)) {
// XXX check css2 spec: I think the spec dictates the size of the
// entire box, not the content area
aMetrics.width = aState.mBorderPadding.left + aState.mStyleSize.width +
aState.mBorderPadding.right;
}
else {
// Make sure that we collapse into nothingness if our content is
// zero sized
if (0 == bounds.width) {
aMetrics.width = 0;
}
else {
aMetrics.width = aState.mBorderPadding.left + bounds.width +
aState.mBorderPadding.right;
}
}
// Compute the final height. Use the style height when supplied.
#if XXX_fix_me
if (0 != (ss & NS_SIZE_HAS_HEIGHT)) {
aMetrics.height = aState.mBorderPadding.top + aState.mStyleSize.height +
aState.mBorderPadding.bottom;
// XXX Apportion extra height evenly to the ascent/descent, taking
// into account roundoff properly. What about when the height is
// smaller than the computed height? Does the css2 spec say
// anything about this?
aMetrics.ascent = ?;
aMetrics.descent = ?;
}
else
#endif
{
// Make sure that we collapse into nothingness if our content is
// zero sized
if (0 == bounds.height) {
aMetrics.ascent = 0;
aMetrics.descent = 0;
aMetrics.height = 0;
}
else {
aMetrics.ascent = aState.mBorderPadding.top +
aState.mInlineLayout.mMaxAscent;
aMetrics.descent = aState.mInlineLayout.mMaxDescent +
aState.mBorderPadding.bottom;
aMetrics.height = aMetrics.ascent + aMetrics.descent;
}
}
// Adjust max-element size if this frame's no-wrap flag is set.
if (aState.mInlineLayout.mComputeMaxElementSize) {
if (aState.mNoWrap) {
aMetrics.maxElementSize->width = aMetrics.width;
aMetrics.maxElementSize->height = aMetrics.height;
}
else {
*aMetrics.maxElementSize = aState.mInlineLayout.mMaxElementSize;
}
// Add in our border and padding to the max-element-size so that
// we don't shrink too far.
aMetrics.maxElementSize->width += aState.mBorderPadding.left +
aState.mBorderPadding.right;
aMetrics.maxElementSize->height += aState.mBorderPadding.top +
aState.mBorderPadding.bottom;
}
}
nsInlineReflowStatus
nsInlineFrame::InitialReflow(nsInlineReflowState& aState)
{
NS_PRECONDITION(nsnull == mNextInFlow, "bad frame-appended-reflow");
NS_PRECONDITION(mLastContentIsComplete == PR_TRUE, "bad state");
nsresult rv = ProcessInitialReflow(aState.mPresContext);
if (NS_OK != rv) {
return rv;
}
nsInlineReflowStatus rs = NS_FRAME_COMPLETE;
if (0 != mChildCount) {
if (!ReflowMapped(aState, rs)) {
return rs;
}
}
return rs;
}
nsInlineReflowStatus
nsInlineFrame::FrameAppendedReflow(nsInlineReflowState& aState)
{
NS_PRECONDITION(nsnull == mNextInFlow, "bad frame-appended-reflow");
NS_PRECONDITION(mLastContentIsComplete == PR_TRUE, "bad state");
// Get the first of the newly appended frames
nsIFrame* firstAppendedFrame;
aState.reflowCommand->GetChildFrame(firstAppendedFrame);
// Add the new frames
AppendNewFrames(aState.mPresContext, firstAppendedFrame);
nsInlineReflowStatus rs = NS_FRAME_COMPLETE;
if (0 != mChildCount) {
if (!ReflowMapped(aState, rs)) {
return rs;
}
}
return rs;
}
// XXX CONSTRUCTION
nsresult
nsInlineFrame::CreateNewFrames(nsIPresContext* aPresContext)
{
// reason: initial: (a) null nif, pif: create frames
// (b) pullup: don't create frames
// reason: incremental: (a) append targetted at us => create frames
// Get the childPrevInFlow for our eventual first child if we are a
// continuation and we have no children and the last child in our
// prev-in-flow is incomplete. While we are at it, we also compute
// our kidContentIndex.
PRInt32 kidContentIndex;
nsIFrame* childPrevInFlow = nsnull;
nsIFrame* prevChild = nsnull;
if ((nsnull == mFirstChild) && (nsnull != mPrevInFlow)) {
nsInlineFrame* prev = (nsInlineFrame*)mPrevInFlow;
NS_ASSERTION(prev->mLastContentOffset >= prev->mFirstContentOffset,
"bad prevInFlow");
kidContentIndex = prev->NextChildOffset();
if (!prev->mLastContentIsComplete) {
// Our prev-in-flow's last child is not complete
prev->LastChild(childPrevInFlow);
}
}
else {
kidContentIndex = NextChildOffset();
LastChild(prevChild);
}
nsresult rv;
PRInt32 lastContentIndex;
mContent->ChildCount(lastContentIndex);
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("enter nsInlineFrame::CreateNewFrames: kidContentIndex=%d lastContentIndex=%d childPrevInFlow=%p",
kidContentIndex, lastContentIndex, childPrevInFlow));
while (kidContentIndex < lastContentIndex) {
nsIContent* kid;
mContent->ChildAt(kidContentIndex, kid);
if (nsnull == kid) {
// Our content container is bad
break;
}
// Create child
nsIFrame* child;
rv = nsHTMLBase::CreateFrame(aPresContext, this, kid, childPrevInFlow,
child);
NS_RELEASE(kid);
if (NS_OK != rv) {
return rv;
}
if (nsnull == prevChild) {
mFirstChild = child;
mFirstContentOffset = kidContentIndex;
}
else {
prevChild->SetNextSibling(child);
}
mChildCount++;
kidContentIndex++;
childPrevInFlow = nsnull;
prevChild = child;
NS_FRAME_TRACE(NS_FRAME_TRACE_NEW_FRAMES,
("nsInlineFrame::CreateNewFrames: new-frame=%p prev-in-flow=%p",
child, childPrevInFlow));
}
if (kidContentIndex == 0) {
NS_ASSERTION(lastContentIndex == 0, "bad kid content index");
mLastContentOffset = 0;
} else {
mLastContentOffset = kidContentIndex - 1;
}
NS_ASSERTION(mFirstContentOffset <= mLastContentOffset, "bad fco/lco");
mLastContentIsComplete = PR_TRUE;
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("exit nsInlineFrame::CreateNewFrames"));
return NS_OK;
}
nsInlineReflowStatus
nsInlineFrame::ChildIncrementalReflow(nsInlineReflowState& aState)
{
// XXX we can do better SOMEDAY
return ResizeReflow(aState);
}
nsInlineReflowStatus
nsInlineFrame::ResizeReflow(nsInlineReflowState& aState)
{
nsInlineReflowStatus rs = NS_FRAME_COMPLETE;
if (0 != mChildCount) {
if (!ReflowMapped(aState, rs)) {
return rs;
}
}
if (NS_FRAME_IS_COMPLETE(rs)) {
// Try to fit some more children from our next in flow
if (nsnull != mNextInFlow) {
if (!PullUpChildren(aState, rs)) {
return rs;
}
}
}
return rs;
}
PRBool
nsInlineFrame::ReflowMapped(nsInlineReflowState& aState,
nsInlineReflowStatus& aReflowStatus)
{
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("enter nsInlineFrame::ReflowMapped: childCount=%d", mChildCount));
nsresult rv;
PRBool keepGoing = PR_FALSE;
PRInt32 kidContentIndex = mFirstContentOffset;
nsIFrame* prevChild = nsnull;
nsIFrame* child = mFirstChild;
while (nsnull != child) {
aReflowStatus = aState.mInlineLayout.ReflowAndPlaceFrame(child);
if (NS_IS_REFLOW_ERROR(aReflowStatus)) {
return PR_FALSE;
}
// Update our mLastContentIsComplete flag
mLastContentIsComplete = NS_FRAME_IS_COMPLETE(aReflowStatus);
if (!mLastContentIsComplete) {
// Create a continuation frame for the child now
rv = MaybeCreateNextInFlow(aState, child);/* XXX rename method */
if (NS_OK != rv) {
aReflowStatus = nsInlineReflowStatus(rv);
return PR_FALSE;
}
}
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("nsInlineFrame::ReflowMapped: frame=%p c=%c kidRS=%x",
child, mLastContentIsComplete ? 'T' : 'F', aReflowStatus));
// See if a break is needed and if one is needed, what kind of break
if (!NS_INLINE_IS_BREAK(aReflowStatus)) {
// Advance past the child we just reflowed.
prevChild = child;
child->GetNextSibling(child);
// If child is complete then continue reflowing
if (mLastContentIsComplete) {
kidContentIndex++;
continue;
}
// When our child is not complete we need to push any children
// that follow to our next-in-flow.
PushKids(aState, prevChild, child);
mLastContentOffset = kidContentIndex;
goto done;
}
else if (NS_INLINE_IS_BREAK_AFTER(aReflowStatus)) {
// We need to break-after the child. We also need to return
// the break-after state to our parent because it means our
// parent needs to break-after us (and so on until the type of
// break requested is satisfied).
mLastContentOffset = kidContentIndex;
// Advance past the child we are breaking after (if it was
// incomplete then we already created its next-in-flow above)
prevChild = child;
child->GetNextSibling(child);
if (nsnull != child) {
// Push extra children to a next-in-flow (and therefore we
// know we aren't complete)
PushKids(aState, prevChild, child);
aReflowStatus |= NS_FRAME_NOT_COMPLETE;
}
else if (mLastContentIsComplete) {
// Determine our completion status. We might be complete IFF
// the child we just reflowed happens to be the last
// possible child we can contain.
PRInt32 lastContentIndex;
mContent->ChildCount(lastContentIndex);
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("nsInlineFrame::ReflowMapped: HERE kidContentIndex=%d lastContentIndex=%d frame=%p c=%c kidRS=%x",
kidContentIndex, lastContentIndex, child, mLastContentIsComplete ? 'T' : 'F', aReflowStatus));
if (++kidContentIndex == lastContentIndex) {
// We are complete. Yippee. :-)
aReflowStatus &= ~NS_FRAME_NOT_COMPLETE;
}
else {
// We are not complete. Bummer dude.
aReflowStatus |= NS_FRAME_NOT_COMPLETE;
}
}
goto done;
} else {
// We need to break-before the child. Unlink break-after, a
// break-before may be changed into just incomplete if there are
// children preceeding this child (because the other children we
// have do not need to be broken-before).
mLastContentIsComplete = PR_TRUE;
if (nsnull == prevChild) {
// We are breaking before our very first child
// frame. Therefore we need return the break-before status as
// it is. And by definition we are incomplete.
aReflowStatus |= NS_FRAME_NOT_COMPLETE;
mLastContentOffset = kidContentIndex;
}
else {
// We have other children before the child that needs the
// break-before. Therefore map the break-before from the child
// into a break-after for us, preserving the break-type in the
// process. This is important when the break type is not just
// a simple line break.
aReflowStatus = NS_FRAME_NOT_COMPLETE | NS_INLINE_BREAK |
NS_INLINE_BREAK_AFTER | (NS_INLINE_BREAK_TYPE_MASK & aReflowStatus);
mLastContentOffset = kidContentIndex - 1;
PushKids(aState, prevChild, child);
}
goto done;
}
}
if (kidContentIndex == 0) {
mLastContentOffset = 0;
} else {
mLastContentOffset = kidContentIndex - 1;
}
mLastContentIsComplete = PR_TRUE;
keepGoing = PR_TRUE;
done:;
aState.mLastChild = prevChild;
NS_POSTCONDITION(mFirstContentOffset <= mLastContentOffset,
"bad content offsets");
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("exit nsInlineFrame::ReflowMapped: childCount=%d rs=%x",
mChildCount, aReflowStatus));
return keepGoing;
}
PRBool
nsInlineFrame::PullUpChildren(nsInlineReflowState& aState,
nsInlineReflowStatus& aReflowStatus)
{
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("enter nsInlineFrame::PullUpChildren: childCount=%d",
mChildCount));
// Get correct kidContentIndex
PRInt32 kidContentIndex;
if ((nsnull == mFirstChild) && (nsnull != mPrevInFlow)) {
nsInlineFrame* prev = (nsInlineFrame*)mPrevInFlow;
NS_ASSERTION(prev->mLastContentOffset >= prev->mFirstContentOffset,
"bad prevInFlow");
kidContentIndex = prev->NextChildOffset();
}
else {
kidContentIndex = NextChildOffset();
}
PRBool keepGoing = PR_FALSE;
nsresult rv;
aReflowStatus = NS_FRAME_COMPLETE;
nsInlineFrame* nextInFlow = (nsInlineFrame*) mNextInFlow;
nsIFrame* prevChild = aState.mLastChild;
#ifdef NS_DEBUG
if (nsnull == prevChild) {
NS_ASSERTION(nsnull == mFirstChild, "bad prevChild");
NS_ASSERTION(0 == mChildCount, "bad child count");
}
else {
NS_ASSERTION(nsnull != mFirstChild, "bad prevChild");
NS_ASSERTION(0 != mChildCount, "bad child count");
}
#endif
while (nsnull != nextInFlow) {
// Get child from our next-in-flow
nsIFrame* child = PullOneChild(nextInFlow, prevChild);
if (nsnull == child) {
nextInFlow = (nsInlineFrame*) nextInFlow->mNextInFlow;
continue;
}
NS_FRAME_TRACE(NS_FRAME_TRACE_PUSH_PULL,
("nsInlineFrame::PullUpChildren: trying to pullup %p", child));
// Reflow it
aReflowStatus = aState.mInlineLayout.ReflowAndPlaceFrame(child);
if (NS_IS_REFLOW_ERROR(aReflowStatus)) {
goto done;
}
// XXX This code is identical to ReflowMapped; maybe they should
// be one routine?
// Update our mLastContentIsComplete flag
mLastContentIsComplete = NS_FRAME_IS_COMPLETE(aReflowStatus);
if (!mLastContentIsComplete) {
// Create a continuation frame for the child now
rv = MaybeCreateNextInFlow(aState, child);/* XXX rename method */
if (NS_OK != rv) {
aReflowStatus = nsInlineReflowStatus(rv);
return PR_FALSE;
}
}
// See if a break is needed and if one is needed, what kind of break
if (!NS_INLINE_IS_BREAK(aReflowStatus)) {
// Advance past the child we just reflowed.
prevChild = child;
child->GetNextSibling(child);
// If child is complete then continue reflowing
if (mLastContentIsComplete) {
kidContentIndex++;
continue;
}
// When our child is not complete we need to push any children
// that follow to our next-in-flow.
PushKids(aState, prevChild, child);
mLastContentOffset = kidContentIndex;
goto done;
}
else if (NS_INLINE_IS_BREAK_AFTER(aReflowStatus)) {
// We need to break-after the child. We also need to return
// the break-after state to our parent because it means our
// parent needs to break-after us (and so on until the type of
// break requested is satisfied).
mLastContentOffset = kidContentIndex;
// Advance past the child we are breaking after (if it was
// incomplete then we already created its next-in-flow above)
prevChild = child;
child->GetNextSibling(child);
if (nsnull != child) {
// Push extra children to a next-in-flow (and therefore we
// know we aren't complete)
PushKids(aState, prevChild, child);
aReflowStatus |= NS_FRAME_NOT_COMPLETE;
}
else if (mLastContentIsComplete) {
// Determine our completion status. We might be complete IFF
// the child we just reflowed happens to be the last
// possible child we can contain.
PRInt32 lastContentIndex;
mContent->ChildCount(lastContentIndex);
if (++kidContentIndex == lastContentIndex) {
// We are complete. Yippee. :-)
aReflowStatus &= ~NS_FRAME_NOT_COMPLETE;
}
else {
// We are not complete. Bummer dude.
aReflowStatus |= NS_FRAME_NOT_COMPLETE;
}
}
goto done;
} else {
// We need to break-before the child. Unlink break-after, a
// break-before may be changed into just incomplete if there are
// children preceeding this child (because the other children we
// have do not need to be broken-before).
if (nsnull == prevChild) {
// We are breaking before our very first child
// frame. Therefore we need return the break-before status as
// it is. And by definition we are incomplete.
aReflowStatus |= NS_FRAME_NOT_COMPLETE;
mLastContentOffset = kidContentIndex;
}
else {
// We have other children before the child that needs the
// break-before. Therefore map the break-before from the child
// into a break-after for us, preserving the break-type in the
// process. This is important when the break type is not just
// a simple line break.
aReflowStatus = NS_FRAME_NOT_COMPLETE | NS_INLINE_BREAK |
NS_INLINE_BREAK_AFTER | (NS_INLINE_BREAK_TYPE_MASK & aReflowStatus);
mLastContentOffset = kidContentIndex - 1;
PushKids(aState, prevChild, child);
}
goto done;
}
}
if (kidContentIndex == 0) {
mLastContentOffset = 0;
} else {
mLastContentOffset = kidContentIndex - 1;
}
NS_ASSERTION(mFirstContentOffset <= mLastContentOffset, "bad fco/lco");
mLastContentIsComplete = PR_TRUE;
keepGoing = PR_TRUE;
done:;
// XXX Why bother? Assuming our offsets are correct then our
// next-in-flow will pick up where we left off.
#if XXX
nextInFlow = (nsInlineFrame*) mNextInFlow;
if ((nsnull != nextInFlow) && (nsnull == nextInFlow->mFirstChild)) {
if (NS_FRAME_IS_NOT_COMPLETE(aReflowStatus)) {
AdjustOffsetOfEmptyNextInFlows();
}
}
#endif
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
("exit nsInlineFrame::PullUpChildren: childCount=%d rs=%x",
mChildCount, aReflowStatus));
return keepGoing;
}
nsIFrame*
nsInlineFrame::PullOneChild(nsInlineFrame* aNextInFlow,
nsIFrame* aLastChild)
{
NS_PRECONDITION(nsnull != aNextInFlow, "null ptr");
// Get first available frame from the next-in-flow; if it's out of
// frames check it's overflow list.
nsIFrame* kidFrame = aNextInFlow->mFirstChild;
if (nsnull == kidFrame) {
if (nsnull == aNextInFlow->mOverflowList) {
return nsnull;
}
NS_FRAME_LOG(NS_FRAME_TRACE_CALLS,
("nsInlineFrame::PullOneChild: from overflow list, frame=%p",
kidFrame));
kidFrame = aNextInFlow->mOverflowList;
kidFrame->GetNextSibling(aNextInFlow->mOverflowList);
}
else {
NS_FRAME_LOG(NS_FRAME_TRACE_CALLS,
("nsInlineFrame::PullOneChild: frame=%p (%d kids left)",
kidFrame, aNextInFlow->mChildCount - 1));
// Take the frame away from the next-in-flow. Update it's first
// content offset.
kidFrame->GetNextSibling(aNextInFlow->mFirstChild);
aNextInFlow->mChildCount--;
if (nsnull != aNextInFlow->mFirstChild) {
PRInt32 contentIndex;
aNextInFlow->mFirstChild->GetContentIndex(contentIndex);
aNextInFlow->SetFirstContentOffset(contentIndex);
}
}
// Now give the frame to this container
kidFrame->SetGeometricParent(this);
nsIFrame* contentParent;
kidFrame->GetContentParent(contentParent);
if (aNextInFlow == contentParent) {
kidFrame->SetContentParent(this);
}
// Add the frame on our list
if (nsnull == aLastChild) {
mFirstChild = kidFrame;
PRInt32 contentIndex;
kidFrame->GetContentIndex(contentIndex);
SetFirstContentOffset(contentIndex);
} else {
NS_ASSERTION(IsLastChild(aLastChild), "bad last child");
aLastChild->SetNextSibling(kidFrame);
}
kidFrame->SetNextSibling(nsnull);
mChildCount++;
return kidFrame;
}
nsresult
nsInlineFrame::MaybeCreateNextInFlow(nsInlineReflowState& aState,
nsIFrame* aFrame)
{
nsIFrame* nextInFlow;
nsresult rv = aState.mInlineLayout.MaybeCreateNextInFlow(aFrame, nextInFlow);
if (NS_OK != rv) {
return rv;
}
if (nsnull != nextInFlow) {
mChildCount++;
}
return NS_OK;
}
void
nsInlineFrame::PushKids(nsInlineReflowState& aState,
nsIFrame* aPrevChild,
nsIFrame* aPushedChild)
{
NS_ASSERTION(nsnull == mOverflowList, "bad overflow list");
// Count how many children are being pushed
PRInt32 pushCount = mChildCount - aState.mInlineLayout.mFrameNum;
// Only ask container base class to push children if there is
// something to push...
if (0 != pushCount) {
NS_FRAME_TRACE(NS_FRAME_TRACE_PUSH_PULL,
("nsInlineFrame::PushKids: pushing %d children", pushCount));
// Break sibling list
aPrevChild->SetNextSibling(nsnull);
mChildCount -= pushCount;
// Place overflow frames on our overflow list; our next-in-flow
// will pick them up when it is reflowed
mOverflowList = aPushedChild;
}
#ifdef NS_DEBUG
PRInt32 len = LengthOf(mFirstChild);
NS_ASSERTION(len == mChildCount, "child count is wrong");
#endif
}
void
nsInlineFrame::DrainOverflowLists()
{
// Our prev-in-flows overflow list goes before my children and must
// be re-parented.
if (nsnull != mPrevInFlow) {
nsInlineFrame* prevInFlow = (nsInlineFrame*) mPrevInFlow;
if (nsnull != prevInFlow->mOverflowList) {
nsIFrame* frame = prevInFlow->mOverflowList;
nsIFrame* lastFrame = nsnull;
PRInt32 count = 0;
while (nsnull != frame) {
// Reparent the frame
frame->SetGeometricParent(this);
nsIFrame* contentParent;
frame->GetContentParent(contentParent);
if (prevInFlow == contentParent) {
frame->SetContentParent(this);
}
// Advance through the list
count++;
lastFrame = frame;
frame->GetNextSibling(frame);
}
// Join the two frame lists together and update our child count
#if XXX
if (nsnull == mFirstChild) {
// We are a continuation of our prev-in-flow; we assume that
// the last child was complete.
mLastContentIsComplete = PR_TRUE;
}
#endif
nsIFrame* newFirstChild = prevInFlow->mOverflowList;
newFirstChild->GetContentIndex(mFirstContentOffset);
lastFrame->SetNextSibling(mFirstChild);
mFirstChild = newFirstChild;
prevInFlow->mOverflowList = nsnull;
mChildCount += count;
}
}
// Our overflow list goes to the end of our child list
if (nsnull != mOverflowList) {
// Append the overflow list to the end of our child list
nsIFrame* lastFrame;
LastChild(lastFrame);
if (nsnull == lastFrame) {
mFirstChild = mOverflowList;
mFirstChild->GetContentIndex(mFirstContentOffset);
}
else {
lastFrame->SetNextSibling(mOverflowList);
}
// Count how many frames are on the overflow list and then update
// our count
nsIFrame* frame = mOverflowList;
PRInt32 count = 0;
while (nsnull != frame) {
count++;
frame->GetNextSibling(frame);
}
mChildCount += count;
mOverflowList = nsnull;
#if XXX
// We get here when we pushed some children to a potential
// next-in-flow and then our parent decided to reflow us instead
// of continuing us.
NS_ASSERTION(nsnull == mNextInFlow, "huh?");
mLastContentIsComplete = PR_TRUE;
#endif
}
// XXX What do we set mLastContentIsComplete to?
#ifdef NS_DEBUG
PRInt32 len = LengthOf(mFirstChild);
NS_ASSERTION(len == mChildCount, "child count is wrong");
#endif
}