mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-24 05:44:10 +00:00
8bf0fbd4a8
handling
2141 lines
72 KiB
C++
2141 lines
72 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 "License"); you may not use this file except in
|
|
* compliance with the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/NPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS"
|
|
* basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
|
|
* the License for the specific language governing rights and limitations
|
|
* under the License.
|
|
*
|
|
* The Original Code is Mozilla Communicator client code.
|
|
*
|
|
* The Initial Developer of the Original Code is Netscape Communications
|
|
* Corporation. Portions created by Netscape are Copyright (C) 1998
|
|
* Netscape Communications Corporation. All Rights Reserved.
|
|
*/
|
|
#include "nsInlineFrame.h"
|
|
#include "nsBlockFrame.h"
|
|
#include "nsBlockReflowContext.h"
|
|
#include "nsHTMLIIDs.h"
|
|
#include "nsHTMLAtoms.h"
|
|
#include "nsHTMLParts.h"
|
|
#include "nsCOMPtr.h"
|
|
#include "nsIStyleContext.h"
|
|
#include "nsIPresShell.h"
|
|
#include "nsIPresContext.h"
|
|
#include "nsIRenderingContext.h"
|
|
#include "nsIFontMetrics.h"
|
|
#include "nsAbsoluteContainingBlock.h"
|
|
#include "nsLayoutAtoms.h"
|
|
|
|
// XXX TODO:
|
|
// append/insert/remove floater testing
|
|
|
|
// Theory of operation:
|
|
// XXX write this
|
|
|
|
#ifdef DEBUG
|
|
#undef NOISY_ANON_BLOCK
|
|
#undef NOISY_REFLOW_REASON
|
|
#else
|
|
#undef NOISY_ANON_BLOCK
|
|
#undef NOISY_REFLOW_REASON
|
|
#endif
|
|
|
|
nsIID nsInlineFrame::kInlineFrameCID = NS_INLINE_FRAME_CID;
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
nsresult
|
|
NS_NewPositionedInlineFrame(nsIFrame** aNewFrame)
|
|
{
|
|
NS_PRECONDITION(aNewFrame, "null OUT ptr");
|
|
if (nsnull == aNewFrame) {
|
|
return NS_ERROR_NULL_POINTER;
|
|
}
|
|
nsPositionedInlineFrame* it = new nsPositionedInlineFrame;
|
|
if (nsnull == it) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
*aNewFrame = it;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPositionedInlineFrame::Destroy(nsIPresContext& aPresContext)
|
|
{
|
|
mAbsoluteContainer.DestroyFrames(this, aPresContext);
|
|
return nsInlineFrame::Destroy(aPresContext);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPositionedInlineFrame::SetInitialChildList(nsIPresContext& aPresContext,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aChildList)
|
|
{
|
|
nsresult rv;
|
|
|
|
if (nsLayoutAtoms::absoluteList == aListName) {
|
|
rv = mAbsoluteContainer.SetInitialChildList(this, aPresContext, aListName, aChildList);
|
|
} else {
|
|
rv = nsInlineFrame::SetInitialChildList(aPresContext, aListName, aChildList);
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPositionedInlineFrame::AppendFrames(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
nsresult rv;
|
|
|
|
if (nsLayoutAtoms::absoluteList == aListName) {
|
|
rv = mAbsoluteContainer.AppendFrames(this, aPresContext, aPresShell, aListName,
|
|
aFrameList);
|
|
} else {
|
|
rv = nsInlineFrame::AppendFrames(aPresContext, aPresShell, aListName,
|
|
aFrameList);
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPositionedInlineFrame::InsertFrames(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aPrevFrame,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
nsresult rv;
|
|
|
|
if (nsLayoutAtoms::absoluteList == aListName) {
|
|
rv = mAbsoluteContainer.InsertFrames(this, aPresContext, aPresShell, aListName,
|
|
aPrevFrame, aFrameList);
|
|
} else {
|
|
rv = nsInlineFrame::InsertFrames(aPresContext, aPresShell, aListName, aPrevFrame,
|
|
aFrameList);
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPositionedInlineFrame::RemoveFrame(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aOldFrame)
|
|
{
|
|
nsresult rv;
|
|
|
|
if (nsLayoutAtoms::absoluteList == aListName) {
|
|
rv = mAbsoluteContainer.RemoveFrame(this, aPresContext, aPresShell, aListName, aOldFrame);
|
|
} else {
|
|
rv = nsInlineFrame::RemoveFrame(aPresContext, aPresShell, aListName, aOldFrame);
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPositionedInlineFrame::GetAdditionalChildListName(PRInt32 aIndex,
|
|
nsIAtom** aListName) const
|
|
{
|
|
NS_PRECONDITION(nsnull != aListName, "null OUT parameter pointer");
|
|
*aListName = nsnull;
|
|
if (0 == aIndex) {
|
|
*aListName = nsLayoutAtoms::absoluteList;
|
|
NS_ADDREF(*aListName);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPositionedInlineFrame::FirstChild(nsIAtom* aListName, nsIFrame** aFirstChild) const
|
|
{
|
|
NS_PRECONDITION(nsnull != aFirstChild, "null OUT parameter pointer");
|
|
if (aListName == nsLayoutAtoms::absoluteList) {
|
|
return mAbsoluteContainer.FirstChild(this, aListName, aFirstChild);
|
|
}
|
|
|
|
return nsInlineFrame::FirstChild(aListName, aFirstChild);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPositionedInlineFrame::GetFrameType(nsIAtom** aType) const
|
|
{
|
|
NS_PRECONDITION(nsnull != aType, "null OUT parameter pointer");
|
|
*aType = nsLayoutAtoms::positionedInlineFrame;
|
|
NS_ADDREF(*aType);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsPositionedInlineFrame::Reflow(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus)
|
|
{
|
|
nsresult rv = NS_OK;
|
|
|
|
// See if it's an incremental reflow command
|
|
if (eReflowReason_Incremental == aReflowState.reason) {
|
|
// Give the absolute positioning code a chance to handle it
|
|
PRBool handled;
|
|
|
|
mAbsoluteContainer.IncrementalReflow(this, aPresContext, aReflowState, handled);
|
|
|
|
// If the incremental reflow command was handled by the absolute positioning
|
|
// code, then we're all done
|
|
if (handled) {
|
|
// Just return our current size as our desired size
|
|
// XXX I don't know how to compute that without a reflow, so for the
|
|
// time being pretend a resize reflow occured
|
|
nsHTMLReflowState reflowState(aReflowState);
|
|
reflowState.reason = eReflowReason_Resize;
|
|
reflowState.reflowCommand = nsnull;
|
|
return nsInlineFrame::Reflow(aPresContext, aDesiredSize, reflowState, aStatus);
|
|
}
|
|
}
|
|
|
|
// Let the inline frame do its reflow first
|
|
rv = nsInlineFrame::Reflow(aPresContext, aDesiredSize, aReflowState, aStatus);
|
|
|
|
// Let the absolutely positioned container reflow any absolutely positioned
|
|
// child frames that need to be reflowed
|
|
if (NS_SUCCEEDED(rv)) {
|
|
rv = mAbsoluteContainer.Reflow(this, aPresContext, aReflowState);
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// SectionData implementation
|
|
|
|
nsInlineFrame::SectionData::SectionData(nsIFrame* aFrameList)
|
|
{
|
|
firstBlock = nsnull;
|
|
prevFirstBlock = nsnull;
|
|
lastBlock = nsnull;
|
|
lastFrame = nsnull;
|
|
|
|
// Find the first and last block (if any!). When we exit the loop
|
|
// lastFrame will be the last frame in aList.
|
|
nsIFrame* frame = aFrameList;
|
|
firstFrame = aFrameList;
|
|
while (nsnull != frame) {
|
|
if (nsLineLayout::TreatFrameAsBlock(frame)) {
|
|
if (nsnull == firstBlock) {
|
|
prevFirstBlock = lastFrame;
|
|
firstBlock = frame;
|
|
lastBlock = frame;
|
|
}
|
|
else {
|
|
lastBlock = frame;
|
|
}
|
|
}
|
|
lastFrame = frame;
|
|
frame->GetNextSibling(&frame);
|
|
}
|
|
}
|
|
|
|
PRBool
|
|
nsInlineFrame::SectionData::SplitFrameList(nsFrameList& aSection1,
|
|
nsFrameList& aSection2,
|
|
nsFrameList& aSection3)
|
|
{
|
|
if (nsnull == firstBlock) {
|
|
// There are no blocks
|
|
return PR_FALSE;
|
|
}
|
|
|
|
// We have at least one block
|
|
if (nsnull != prevFirstBlock) {
|
|
// The first block is not the first frame in aList. Setup section1.
|
|
prevFirstBlock->SetNextSibling(nsnull);
|
|
aSection1.SetFrames(firstFrame);
|
|
}
|
|
aSection2.SetFrames(firstBlock);
|
|
|
|
if (lastFrame != lastBlock) {
|
|
// There are inline frames that follow the last block. Setup section3.
|
|
nsIFrame* nextSib;
|
|
lastBlock->GetNextSibling(&nextSib);
|
|
lastBlock->SetNextSibling(nsnull);
|
|
aSection3.SetFrames(nextSib);
|
|
}
|
|
|
|
return PR_TRUE;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
// Basic nsInlineFrame methods
|
|
|
|
nsresult
|
|
NS_NewInlineFrame(nsIFrame** aNewFrame)
|
|
{
|
|
NS_PRECONDITION(aNewFrame, "null OUT ptr");
|
|
if (nsnull == aNewFrame) {
|
|
return NS_ERROR_NULL_POINTER;
|
|
}
|
|
nsInlineFrame* it = new nsInlineFrame;
|
|
if (nsnull == it) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
*aNewFrame = it;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsInlineFrame::nsInlineFrame()
|
|
{
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsInlineFrame::QueryInterface(const nsIID& aIID, void** aInstancePtr)
|
|
{
|
|
if (nsnull == aInstancePtr) {
|
|
return NS_ERROR_NULL_POINTER;
|
|
}
|
|
if (aIID.Equals(kInlineFrameCID)) {
|
|
nsInlineFrame* tmp = this;
|
|
*aInstancePtr = (void*) tmp;
|
|
return NS_OK;
|
|
}
|
|
return nsInlineFrameSuper::QueryInterface(aIID, aInstancePtr);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsInlineFrame::GetFrameName(nsString& aResult) const
|
|
{
|
|
return MakeFrameName("Inline", aResult);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsInlineFrame::GetFrameType(nsIAtom** aType) const
|
|
{
|
|
NS_PRECONDITION(nsnull != aType, "null OUT parameter pointer");
|
|
*aType = nsLayoutAtoms::inlineFrame;
|
|
NS_ADDREF(*aType);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsInlineFrame::Destroy(nsIPresContext& aPresContext)
|
|
{
|
|
mFrames.DestroyFrames(aPresContext);
|
|
return nsInlineFrameSuper::Destroy(aPresContext);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// nsInlineFrame child management
|
|
|
|
// Find the first inline frame, looking backwards starting at "this",
|
|
// that contains an anonymous block. Return nsnull if an anonymous
|
|
// block is not found.
|
|
nsAnonymousBlockFrame*
|
|
nsInlineFrame::FindPrevAnonymousBlock(nsInlineFrame** aBlockParent)
|
|
{
|
|
nsInlineFrame* prevInFlow = this;
|
|
while (nsnull != prevInFlow) {
|
|
// Scan the prev-in-flows frame list, looking for an anonymous
|
|
// block frame.
|
|
nsIFrame* frame = prevInFlow->mFrames.FirstChild();
|
|
while (nsnull != frame) {
|
|
if (nsLineLayout::TreatFrameAsBlock(frame)) {
|
|
*aBlockParent = prevInFlow;
|
|
return (nsAnonymousBlockFrame*) frame;
|
|
}
|
|
frame->GetNextSibling(&frame);
|
|
}
|
|
prevInFlow = (nsInlineFrame*) prevInFlow->mPrevInFlow;
|
|
}
|
|
return nsnull;
|
|
}
|
|
|
|
// Find the first inline frame, looking forwards starting at "this",
|
|
// that contains an anonymous block. Return nsnull if an anonymous
|
|
// block is not found.
|
|
nsAnonymousBlockFrame*
|
|
nsInlineFrame::FindAnonymousBlock(nsInlineFrame** aBlockParent)
|
|
{
|
|
nsInlineFrame* nextInFlow = this;
|
|
while (nsnull != nextInFlow) {
|
|
// Scan the prev-in-flows frame list, looking for an anonymous
|
|
// block frame.
|
|
nsIFrame* frame = nextInFlow->mFrames.FirstChild();
|
|
while (nsnull != frame) {
|
|
if (nsLineLayout::TreatFrameAsBlock(frame)) {
|
|
*aBlockParent = nextInFlow;
|
|
return (nsAnonymousBlockFrame*) frame;
|
|
}
|
|
frame->GetNextSibling(&frame);
|
|
}
|
|
nextInFlow = (nsInlineFrame*) nextInFlow->mNextInFlow;
|
|
}
|
|
return nsnull;
|
|
}
|
|
|
|
nsresult
|
|
nsInlineFrame::CreateAnonymousBlock(nsIPresContext& aPresContext,
|
|
nsIFrame* aInitialFrames,
|
|
nsIFrame** aResult)
|
|
{
|
|
nsIFrame* bf;
|
|
nsresult rv = NS_NewAnonymousBlockFrame(&bf);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
nsCOMPtr<nsIStyleContext> newSC;
|
|
aPresContext.ResolvePseudoStyleContextFor(mContent,
|
|
nsHTMLAtoms::mozAnonymousBlock,
|
|
mStyleContext,
|
|
PR_FALSE,
|
|
getter_AddRefs(newSC));
|
|
rv = bf->Init(aPresContext, mContent, this, newSC, nsnull);
|
|
if (NS_FAILED(rv)) {
|
|
bf->Destroy(aPresContext);
|
|
delete bf;
|
|
}
|
|
else {
|
|
// Set parent for the frames now that the anonymous block has
|
|
// been created.
|
|
nsIFrame* frame = aInitialFrames;
|
|
while (nsnull != frame) {
|
|
frame->SetParent(bf);
|
|
frame->GetNextSibling(&frame);
|
|
}
|
|
rv = bf->SetInitialChildList(aPresContext, nsnull, aInitialFrames);
|
|
}
|
|
*aResult = bf;
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsInlineFrame::SetInitialChildList(nsIPresContext& aPresContext,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
if (nsnull != aListName) {
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
if (nsnull == aFrameList) {
|
|
return NS_OK;
|
|
}
|
|
nsCOMPtr<nsIPresShell> shell;
|
|
nsresult rv = aPresContext.GetShell(getter_AddRefs(shell));
|
|
if (NS_SUCCEEDED(rv) && shell) {
|
|
rv = AppendFrames(aPresContext, *shell, aFrameList, PR_FALSE);
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsInlineFrame::AppendFrames(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
if (nsnull != aListName) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
if (nsnull == aFrameList) {
|
|
return NS_OK;
|
|
}
|
|
return AppendFrames(aPresContext, aPresShell, aFrameList, PR_TRUE);
|
|
}
|
|
|
|
nsresult
|
|
nsInlineFrame::AppendFrames(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIFrame* aFrameList,
|
|
PRBool aGenerateReflowCommands)
|
|
{
|
|
#ifdef NOISY_REFLOW_REASON
|
|
ListTag(stdout);
|
|
printf(": append ");
|
|
nsFrame::ListTag(stdout, aFrameList);
|
|
nsIFrame* lastKid = mFrames.LastChild();
|
|
if (lastKid) {
|
|
printf(" after ");
|
|
nsFrame::ListTag(stdout, lastKid);
|
|
}
|
|
printf("\n");
|
|
#endif
|
|
|
|
nsresult rv = NS_OK;
|
|
SectionData sd(aFrameList);
|
|
if (sd.HasABlock()) {
|
|
nsFrameList section1, section2, section3;
|
|
sd.SplitFrameList(section1, section2, section3);
|
|
|
|
// There is at least one block in the new frames. See if there is
|
|
// an anonymous block frame in a prev-in-flow of this frame.
|
|
nsInlineFrame* prevInline;
|
|
nsAnonymousBlockFrame* anonymousBlock;
|
|
anonymousBlock = FindPrevAnonymousBlock(&prevInline);
|
|
if (nsnull != anonymousBlock) {
|
|
// One of the invariants of nsInlineFrame is that there will be
|
|
// at most one anonymous block frame that reflows the
|
|
// nsInlineFrame's block children (not counting any
|
|
// continuations of the anonymous block frame).
|
|
//
|
|
// We maintain that invariant by noticing when a new block frame
|
|
// enters the list of children and ensuring that all of the
|
|
// frames from the first block to the last block are contained
|
|
// by the anonymous block (or one of its continuations). This
|
|
// can cause some inline frames between here and the inline that
|
|
// contains anonymousBlock to have their children stuffed into
|
|
// the anonymous block.
|
|
|
|
// Build a list of the frames between the anonymous block and
|
|
// this frame, stealing children from our continuation frames as
|
|
// necessary.
|
|
nsIFrame* inlineSiblings;
|
|
anonymousBlock->GetNextSibling(&inlineSiblings);
|
|
nsFrameList newBlockFrames;
|
|
if (nsnull != inlineSiblings) {
|
|
newBlockFrames.AppendFrames(anonymousBlock, inlineSiblings);
|
|
}
|
|
nsInlineFrame* tmp = (nsInlineFrame*) prevInline->mNextInFlow;
|
|
while ((nsnull != tmp) && (this != tmp)) {
|
|
newBlockFrames.AppendFrames(anonymousBlock, tmp->mFrames);
|
|
tmp = (nsInlineFrame*) tmp->mNextInFlow;
|
|
}
|
|
|
|
// Now tack on all of this frame's child frames (unless this
|
|
// frame is the frame that contains the anonymous block)
|
|
if (this != prevInline) {
|
|
newBlockFrames.AppendFrames(anonymousBlock, mFrames);
|
|
}
|
|
|
|
// And then append section1 and section2 frames.
|
|
if (section1.NotEmpty()) {
|
|
newBlockFrames.AppendFrames(anonymousBlock, section1);
|
|
}
|
|
newBlockFrames.AppendFrames(anonymousBlock, section2);
|
|
|
|
// Finally, if there are any frames in section3 then they are
|
|
// appended after the anonymous block. These frames will be
|
|
// reflowed when they are pushed from the prevInline frame to a
|
|
// next-in-flow after the anonymousBlock frame is reflowed.
|
|
anonymousBlock->SetNextSibling(section3.FirstChild());
|
|
|
|
// Now we can append the frames to the anonymous block and it
|
|
// can generate a reflow command.
|
|
rv = anonymousBlock->AppendFrames2(&aPresContext, &aPresShell, nsnull,
|
|
newBlockFrames.FirstChild());
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("AppendFrames: case 1\n");
|
|
#endif
|
|
}
|
|
else {
|
|
// There is no prior block frames that are the children of this
|
|
// inline. Therefore, section 1 is appended to our frame list
|
|
// and we wrap up the frames in section 2 with a new anonymous
|
|
// block and the frames in section 3 get appended to the frame
|
|
// list after the anonymous frame (so that they can be pushed to
|
|
// a next-in-flow after this finishes reflowing its anonymous
|
|
// block).
|
|
nsIFrame* anonBlock;
|
|
rv = CreateAnonymousBlock(aPresContext, section2.FirstChild(),
|
|
&anonBlock);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
if (section1.NotEmpty()) {
|
|
mFrames.AppendFrames(nsnull, section1);
|
|
}
|
|
mFrames.AppendFrame(nsnull, anonBlock);
|
|
if (section3.NotEmpty()) {
|
|
mFrames.AppendFrames(nsnull, section3);
|
|
}
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("AppendFrames: case 2\n");
|
|
#endif
|
|
|
|
if (aGenerateReflowCommands) {
|
|
// generate a reflow command for "this"
|
|
nsIReflowCommand* reflowCmd = nsnull;
|
|
rv = NS_NewHTMLReflowCommand(&reflowCmd, this,
|
|
nsIReflowCommand::ReflowDirty);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
aPresShell.AppendReflowCommand(reflowCmd);
|
|
NS_RELEASE(reflowCmd);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
// The new frames contain no block frames
|
|
mFrames.AppendFrames(this, aFrameList);
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("AppendFrames: case 3\n");
|
|
#endif
|
|
|
|
if (aGenerateReflowCommands) {
|
|
// generate a reflow command for "this"
|
|
nsIReflowCommand* reflowCmd = nsnull;
|
|
rv = NS_NewHTMLReflowCommand(&reflowCmd, this,
|
|
nsIReflowCommand::ReflowDirty);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
aPresShell.AppendReflowCommand(reflowCmd);
|
|
NS_RELEASE(reflowCmd);
|
|
}
|
|
}
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsInlineFrame::InsertFrames(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aPrevFrame,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
if (nsnull != aListName) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
if (nsnull == aFrameList) {
|
|
return NS_OK;
|
|
}
|
|
|
|
#ifdef NOISY_REFLOW_REASON
|
|
ListTag(stdout);
|
|
printf(": insert ");
|
|
nsFrame::ListTag(stdout, aFrameList);
|
|
if (aPrevFrame) {
|
|
printf(" after ");
|
|
nsFrame::ListTag(stdout, aPrevFrame);
|
|
}
|
|
printf("\n");
|
|
#endif
|
|
|
|
nsresult rv = NS_OK;
|
|
SectionData sd(aFrameList);
|
|
if (sd.HasABlock()) {
|
|
// Break insertion up into 3 pieces
|
|
nsFrameList section1, section2, section3;
|
|
sd.SplitFrameList(section1, section2, section3);
|
|
|
|
nsIFrame* prevFrame = aPrevFrame;
|
|
|
|
// First insert the inlines in section1 after prevFrame
|
|
if (section1.NotEmpty()) {
|
|
nsIFrame* newPrevFrame = section1.LastChild();
|
|
rv = InsertInlineFrames(aPresContext, aPresShell, prevFrame,
|
|
section1.FirstChild());
|
|
prevFrame = newPrevFrame;
|
|
}
|
|
|
|
// Next insert the frames in section2 after prevFrame
|
|
if (NS_SUCCEEDED(rv)) {
|
|
nsIFrame* newPrevFrame = section2.LastChild();
|
|
rv = InsertBlockFrames(aPresContext, aPresShell, prevFrame,
|
|
section2.FirstChild());
|
|
prevFrame = newPrevFrame;
|
|
}
|
|
|
|
// Finally, insert the frames in section3 after prevFrame
|
|
if (NS_SUCCEEDED(rv) && section3.NotEmpty()) {
|
|
rv = InsertInlineFrames(aPresContext, aPresShell, prevFrame,
|
|
section3.FirstChild());
|
|
}
|
|
}
|
|
else {
|
|
// Use simpler path when the insertion is only inline frames
|
|
rv = InsertInlineFrames(aPresContext, aPresShell, aPrevFrame, aFrameList);
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
nsresult
|
|
nsInlineFrame::InsertBlockFrames(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIFrame* aPrevFrame,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
nsresult rv = NS_OK;
|
|
PRBool generateReflowCommand = PR_FALSE;
|
|
nsIFrame* target = nsnull;
|
|
|
|
if (nsnull == aPrevFrame) {
|
|
// The block is being inserted at the head of all the child
|
|
// frames.
|
|
nsInlineFrame* flow;
|
|
nsAnonymousBlockFrame* anonymousBlock = FindAnonymousBlock(&flow);
|
|
if (nsnull == anonymousBlock) {
|
|
// There are no anonymous blocks so create one and place the
|
|
// frames into it.
|
|
nsIFrame* anonBlock;
|
|
rv = CreateAnonymousBlock(aPresContext, aFrameList, &anonBlock);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
mFrames.InsertFrames(this, nsnull, anonBlock);
|
|
target = this;
|
|
generateReflowCommand = PR_TRUE;
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("InsertBlockFrames: case 1\n");
|
|
#endif
|
|
}
|
|
else {
|
|
// Take all of the frames before the anonymous block, plus the
|
|
// frames in aFrameList and insert them into the anonymous
|
|
// block.
|
|
nsFrameList frames;
|
|
frames.AppendFrames(anonymousBlock, aFrameList);
|
|
nsInlineFrame* start = this;
|
|
while (start != flow) {
|
|
frames.AppendFrames(anonymousBlock, start->mFrames);
|
|
start->GetNextInFlow((nsIFrame**) &start);
|
|
}
|
|
anonymousBlock->InsertFrames2(&aPresContext, &aPresShell, nsnull,
|
|
nsnull, frames.FirstChild());
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("InsertBlockFrames: case 2\n");
|
|
#endif
|
|
}
|
|
}
|
|
else {
|
|
// First see if the insertion is inside the anonymous block
|
|
nsIFrame* prevFrameParent;
|
|
aPrevFrame->GetParent(&prevFrameParent);
|
|
if (nsLineLayout::TreatFrameAsBlock(prevFrameParent)) {
|
|
// The previous frame's parent is an anonymous block. This means
|
|
// that the new block frames can be safely inserted there.
|
|
nsIFrame* frame = aFrameList;
|
|
while (nsnull != frame) {
|
|
frame->SetParent(prevFrameParent);
|
|
frame->GetNextSibling(&frame);
|
|
}
|
|
nsAnonymousBlockFrame* anonymousBlock;
|
|
anonymousBlock = (nsAnonymousBlockFrame*) prevFrameParent;
|
|
anonymousBlock->InsertFrames2(&aPresContext, &aPresShell, nsnull,
|
|
aPrevFrame, aFrameList);
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("InsertBlockFrames: case 3\n");
|
|
#endif
|
|
}
|
|
else {
|
|
// The previous frame's parent is an inline frame. First see if
|
|
// there is an anonymous block before the insertion point.
|
|
nsInlineFrame* flow = (nsInlineFrame*) prevFrameParent;
|
|
nsAnonymousBlockFrame* anonymousBlock;
|
|
anonymousBlock = flow->FindPrevAnonymousBlock(&flow);
|
|
if (nsnull != anonymousBlock) {
|
|
// We found an anonymous block before the insertion
|
|
// point. Take all of the inline frames between the anonymous
|
|
// block and prevFrameParent and strip them away from the
|
|
// inline frames that contain them.
|
|
nsFrameList frames;
|
|
nsInlineFrame* start;
|
|
flow->GetNextInFlow((nsIFrame**) &start); // start after anon block
|
|
while (start != prevFrameParent) {
|
|
frames.AppendFrames(anonymousBlock, start->mFrames);
|
|
start->GetNextInFlow((nsIFrame**) &start);
|
|
}
|
|
|
|
// Now append the frames just before and including aPrevFrame
|
|
// to "frames".
|
|
flow = (nsInlineFrame*) prevFrameParent;
|
|
nsIFrame* remainingFrames;
|
|
flow->mFrames.Split(aPrevFrame, &remainingFrames);
|
|
frames.AppendFrames(anonymousBlock, flow->mFrames);
|
|
flow->mFrames.SetFrames(remainingFrames);
|
|
generateReflowCommand = PR_TRUE;
|
|
target = flow;
|
|
|
|
// Finally, append the block frames to "frames" and then
|
|
// append the list of frames to the anonymous block.
|
|
frames.AppendFrames(anonymousBlock, aFrameList);
|
|
anonymousBlock->AppendFrames2(&aPresContext, &aPresShell, nsnull,
|
|
frames.FirstChild());
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("InsertBlockFrames: case 4\n");
|
|
#endif
|
|
}
|
|
else {
|
|
// There is no anymous block before the insertion point. See
|
|
// if there is one after the insertion point.
|
|
flow = (nsInlineFrame*) prevFrameParent;
|
|
anonymousBlock = flow->FindAnonymousBlock(&flow);
|
|
if (nsnull != anonymousBlock) {
|
|
// We found an anonymous block after the insertion
|
|
// point. Seed the list of frames to put into the anonymous
|
|
// block with the block frames being inserted.
|
|
nsFrameList frames;
|
|
frames.AppendFrames(anonymousBlock, aFrameList);
|
|
nsInlineFrame* start = (nsInlineFrame*) prevFrameParent;
|
|
|
|
// Take the frames after aPrevFrame and place them at the
|
|
// end of the frames list.
|
|
nsIFrame* remainingFrames;
|
|
start->mFrames.Split(aPrevFrame, &remainingFrames);
|
|
if (remainingFrames) {
|
|
frames.AppendFrames(anonymousBlock, remainingFrames);
|
|
}
|
|
generateReflowCommand = PR_TRUE;
|
|
target = start;
|
|
|
|
// Gather up all of the inline frames from all of the flow
|
|
// between the insertion point and the anonymous block.
|
|
start->GetNextInFlow((nsIFrame**) &start);
|
|
while (start != flow) {
|
|
frames.AppendFrames(anonymousBlock, start->mFrames);
|
|
start->GetNextInFlow((nsIFrame**) &start);
|
|
}
|
|
|
|
// Now update the anonymous block
|
|
anonymousBlock->InsertFrames2(&aPresContext, &aPresShell, nsnull,
|
|
nsnull, frames.FirstChild());
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("InsertBlockFrames: case 5\n");
|
|
#endif
|
|
}
|
|
else {
|
|
// There are no anonymous blocks so create one and place the
|
|
// frames into it.
|
|
nsIFrame* anonBlock;
|
|
rv = CreateAnonymousBlock(aPresContext, aFrameList, &anonBlock);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
// Insert the frame into the correct parent (it will not be
|
|
// this frame when aPrevFrame's parent != this)
|
|
flow = (nsInlineFrame*) prevFrameParent;
|
|
flow->mFrames.InsertFrames(flow, aPrevFrame, anonBlock);
|
|
generateReflowCommand = PR_TRUE;
|
|
target = flow;
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("InsertBlockFrames: case 6\n");
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (generateReflowCommand) {
|
|
// generate a reflow command for "this"
|
|
nsIReflowCommand* reflowCmd = nsnull;
|
|
rv = NS_NewHTMLReflowCommand(&reflowCmd, target,
|
|
nsIReflowCommand::ReflowDirty);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
aPresShell.AppendReflowCommand(reflowCmd);
|
|
NS_RELEASE(reflowCmd);
|
|
}
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
nsresult
|
|
nsInlineFrame::InsertInlineFrames(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIFrame* aPrevFrame,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
nsresult rv = NS_OK;
|
|
PRBool generateReflowCommand = PR_FALSE;
|
|
nsIFrame* target = nsnull;
|
|
|
|
if (nsnull == aPrevFrame) {
|
|
// Insert the frames at the front of our list. Since the frames
|
|
// are all inline frames we just place them there, even if our
|
|
// current first frame is the anonymous block frame. The reflow
|
|
// logic will properly push the anonymous block frame to a
|
|
// next-in-flow.
|
|
mFrames.InsertFrames(this, nsnull, aFrameList);
|
|
generateReflowCommand = PR_TRUE;
|
|
target = this;
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("InsertInlineFrames: case 1\n");
|
|
#endif
|
|
}
|
|
else {
|
|
nsIFrame* prevFrameParent;
|
|
aPrevFrame->GetParent(&prevFrameParent);
|
|
if (nsLineLayout::TreatFrameAsBlock(prevFrameParent)) {
|
|
nsAnonymousBlockFrame* anonymousBlock;
|
|
anonymousBlock = (nsAnonymousBlockFrame*) prevFrameParent;
|
|
|
|
// The previous frame's parent is an anonymous block (otherwise
|
|
// its parent would be this frame). We must not place the inline
|
|
// frames into the anonymous block if they should be in
|
|
// section3.
|
|
nsIFrame* nextSibling;
|
|
aPrevFrame->GetNextSibling(&nextSibling);
|
|
nsIFrame* anonymousBlockNextInFlow;
|
|
prevFrameParent->GetNextInFlow(&anonymousBlockNextInFlow);
|
|
if ((nsnull != nextSibling) || (nsnull != anonymousBlockNextInFlow)) {
|
|
// Easy case: there are more frames following aPrevFrame which
|
|
// means that this insertion lies in the anonymous block.
|
|
nsIFrame* frame = aFrameList;
|
|
while (nsnull != frame) {
|
|
frame->SetParent(anonymousBlock);
|
|
frame->GetNextSibling(&frame);
|
|
}
|
|
anonymousBlock->InsertFrames2(&aPresContext, &aPresShell, nsnull,
|
|
aPrevFrame, aFrameList);
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("InsertInlineFrames: case 2\n");
|
|
#endif
|
|
}
|
|
else {
|
|
// aPrevFrame is the last frame that should be in the
|
|
// anonymous block.
|
|
nsInlineFrame* anonymousBlockParent;
|
|
anonymousBlock->GetParent((nsIFrame**)&anonymousBlockParent);
|
|
|
|
// Place the inline frames after the anonymous block
|
|
nsIFrame* frame = aFrameList;
|
|
while (nsnull != frame) {
|
|
frame->SetParent(anonymousBlockParent);
|
|
frame->GetNextSibling(&frame);
|
|
}
|
|
anonymousBlockParent->mFrames.InsertFrames(nsnull, anonymousBlock,
|
|
aFrameList);
|
|
generateReflowCommand = PR_TRUE;
|
|
target = anonymousBlockParent;
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("InsertInlineFrames: case 3\n");
|
|
#endif
|
|
}
|
|
}
|
|
else {
|
|
// The previous frame's parent is an inline frame. Therefore
|
|
// this is either a section1 or section3 insertion. Insert the
|
|
// frames in the proper flow block (which will be aPrevFrame's
|
|
// parent which is currently stored in anonymousBlock)
|
|
nsInlineFrame* flow = (nsInlineFrame*) prevFrameParent;
|
|
flow->mFrames.InsertFrames(flow, aPrevFrame, aFrameList);
|
|
generateReflowCommand = PR_TRUE;
|
|
target = flow;
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("InsertInlineFrames: case 4\n");
|
|
#endif
|
|
}
|
|
}
|
|
|
|
if (generateReflowCommand) {
|
|
// generate a reflow command for "this"
|
|
nsIReflowCommand* reflowCmd = nsnull;
|
|
rv = NS_NewHTMLReflowCommand(&reflowCmd, target,
|
|
nsIReflowCommand::ReflowDirty);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
aPresShell.AppendReflowCommand(reflowCmd);
|
|
NS_RELEASE(reflowCmd);
|
|
}
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsInlineFrame::RemoveFrame(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aOldFrame)
|
|
{
|
|
if (nsnull != aListName) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
#ifdef NOISY_REFLOW_REASON
|
|
ListTag(stdout);
|
|
printf(": remove ");
|
|
nsFrame::ListTag(stdout, aOldFrame);
|
|
printf("\n");
|
|
#endif
|
|
|
|
nsresult rv = NS_OK;
|
|
PRBool generateReflowCommand = PR_FALSE;
|
|
nsIFrame* target = nsnull;
|
|
|
|
nsIFrame* oldFrameParent;
|
|
if (ParentIsInlineFrame(aOldFrame, &oldFrameParent)) {
|
|
// Loop and destroy the frame and all of its
|
|
// continuations. Because the frame's parent is an inline frame we
|
|
// know that any continuations will also be in an inline frame
|
|
// parent.
|
|
nsInlineFrame* parent = (nsInlineFrame*) oldFrameParent;
|
|
while (nsnull != aOldFrame) {
|
|
// If the frame being removed has zero size then don't bother
|
|
// generating a reflow command.
|
|
nsRect bbox;
|
|
aOldFrame->GetRect(bbox);
|
|
if ((0 == bbox.width) && (0 == bbox.height)) {
|
|
// Don't bother generating a reflow command
|
|
}
|
|
else {
|
|
generateReflowCommand = PR_TRUE;
|
|
target = this;
|
|
}
|
|
|
|
// When the parent is an inline frame we have a simple task -
|
|
// just remove the frame from its parents list and generate a
|
|
// reflow command.
|
|
nsIFrame* oldFrameNextInFlow;
|
|
aOldFrame->GetNextInFlow(&oldFrameNextInFlow);
|
|
nsSplittableType st;
|
|
aOldFrame->IsSplittable(st);
|
|
if (NS_FRAME_NOT_SPLITTABLE != st) {
|
|
aOldFrame->RemoveFromFlow();
|
|
}
|
|
parent->mFrames.DestroyFrame(aPresContext, aOldFrame);
|
|
aOldFrame = oldFrameNextInFlow;
|
|
if (nsnull != aOldFrame) {
|
|
aOldFrame->GetParent((nsIFrame**) &parent);
|
|
}
|
|
}
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("RemoveFrame: case 1\n");
|
|
#endif
|
|
}
|
|
else {
|
|
#ifdef DEBUG
|
|
nsIFrame* oldFrameNextInFlow;
|
|
aOldFrame->GetNextInFlow(&oldFrameNextInFlow);
|
|
NS_ASSERTION(nsnull == oldFrameNextInFlow, "XXX: can't remove continued frames that are in anonymous blocks -- not yet implemented");
|
|
#endif
|
|
|
|
nsIFrame* nextInFlow;
|
|
nsIFrame* prevInFlow;
|
|
|
|
// The parent is not an inline frame which means it is an
|
|
// anonymous block frame.
|
|
nsAnonymousBlockFrame* anonymousBlock =
|
|
(nsAnonymousBlockFrame*) oldFrameParent;
|
|
|
|
// It is possible that we are about to remove the last child of
|
|
// the anonymous block. In this case we remove the anonymous block.
|
|
nsIFrame* kids;
|
|
anonymousBlock->FirstChild(nsnull, &kids);
|
|
nsFrameList blockKids(kids);
|
|
if (1 == blockKids.GetLength()) {
|
|
// Remove the anonymous block
|
|
mFrames.DestroyFrame(aPresContext, anonymousBlock);
|
|
generateReflowCommand = PR_TRUE;
|
|
target = this;
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("RemoveFrame: case 2\n");
|
|
#endif
|
|
}
|
|
else {
|
|
// If the frame being removed is a block frame then we may need to
|
|
// do something fancy.
|
|
if (nsLineLayout::TreatFrameAsBlock(aOldFrame)) {
|
|
// It is possible that we are removing the first block in the
|
|
// anonymous block or the last block. See if its so.
|
|
anonymousBlock->GetPrevInFlow(&prevInFlow);
|
|
nsIFrame* prevSib;
|
|
if ((nsnull != prevInFlow) ||
|
|
(nsnull != (prevSib = blockKids.GetPrevSiblingFor(aOldFrame)))) {
|
|
// There is a block in the anonymous block prior to the
|
|
// block that we are removing. See if we are removing the
|
|
// last block in the anonymous block.
|
|
anonymousBlock->GetNextInFlow(&nextInFlow);
|
|
nsIFrame* nextSib;
|
|
aOldFrame->GetNextSibling(&nextSib);
|
|
if ((nsnull != nextInFlow) || (nsnull != nextSib)) {
|
|
// There is a block in the anonymous block after the block
|
|
// that we are removing. This means that we can let the
|
|
// anonymous block remove the frame.
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("RemoveFrame: case 3\n");
|
|
#endif
|
|
anonymousBlock->RemoveFrame2(&aPresContext, &aPresShell, aListName,
|
|
aOldFrame);
|
|
}
|
|
else {
|
|
// We are removing the last block. We must steal all of
|
|
// the inline frames that preceed the block being removed
|
|
// (up to the closest prior block) so that they can be
|
|
// moved outside the anonymous block and into an inline
|
|
// frame.
|
|
|
|
// First get the last frame out of the picture; delete any
|
|
// continuations it might have.
|
|
nsInlineFrame* anonymousBlockParent;
|
|
anonymousBlock->GetParent((nsIFrame**) &anonymousBlockParent);
|
|
nsAnonymousBlockFrame* ab = anonymousBlock;
|
|
anonymousBlock->RemoveFramesFrom(aOldFrame);
|
|
aOldFrame->Destroy(aPresContext);
|
|
while (nsnull != nextInFlow) {
|
|
nsIFrame* nextParent;
|
|
nextInFlow->GetParent(&nextParent);
|
|
if (nextParent != ab) {
|
|
ab = (nsAnonymousBlockFrame*) nextParent;
|
|
}
|
|
ab->RemoveFirstFrame();
|
|
nsIFrame* nextNextInFlow;
|
|
nextInFlow->GetNextInFlow(&nextNextInFlow);
|
|
nextInFlow->Destroy(aPresContext);
|
|
nextInFlow = nextNextInFlow;
|
|
}
|
|
|
|
// Any inline frames that are between the new last-block
|
|
// inside the anonymous block and the block we just
|
|
// removed need to be taken out of the anonymous block.
|
|
nsFrameList inlines;
|
|
while (nsnull != anonymousBlock) {
|
|
// Find the first inline before the last block
|
|
nsIFrame* abkids;
|
|
anonymousBlock->FirstChild(nsnull, &abkids);
|
|
if (nsnull != abkids) {
|
|
SectionData sd(abkids);
|
|
if (sd.HasABlock()) {
|
|
abkids = sd.lastBlock;
|
|
abkids->GetNextSibling(&abkids);
|
|
if (nsnull != abkids) {
|
|
// Take the frames that follow the last block
|
|
// (which are inline frames) and remove them from
|
|
// the anonymous block. Insert them into the
|
|
// inlines frame-list.
|
|
anonymousBlock->RemoveFramesFrom(abkids);
|
|
inlines.InsertFrames(nsnull, nsnull, abkids);
|
|
}
|
|
}
|
|
else {
|
|
// All of the frames are inline frames -- take them
|
|
// all away.
|
|
anonymousBlock->RemoveFramesFrom(abkids);
|
|
inlines.InsertFrames(nsnull, nsnull, abkids);
|
|
}
|
|
}
|
|
anonymousBlock->GetPrevInFlow((nsIFrame**) &anonymousBlock);
|
|
}
|
|
|
|
// Now we have all of the inline frames that need to be
|
|
// placed into an inline parent instead of the anonymous
|
|
// block parent.
|
|
if (inlines.NotEmpty()) {
|
|
// Place the inline frames after the anonymous block
|
|
// frame in the child list of the anonymousBlockParent.
|
|
anonymousBlockParent->mFrames.AppendFrames(anonymousBlockParent,
|
|
inlines);
|
|
}
|
|
generateReflowCommand = PR_TRUE;
|
|
target = anonymousBlockParent;
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("RemoveFrame: case 4\n");
|
|
#endif
|
|
}
|
|
}
|
|
else {
|
|
// We are removing the first block child of the anonymous
|
|
// block. We must gather all of the inline frames that
|
|
// follow the block being removed from the anonymous block
|
|
// so that they can be moved outside the anonymous block and
|
|
// into an inline frame.
|
|
|
|
// Take away the first frame from the anonymous block (which
|
|
// is the frame we are trying to remove). Make sure we
|
|
// remove aOldFrame's continuations if it has any...
|
|
nsInlineFrame* anonymousBlockParent;
|
|
anonymousBlock->GetParent((nsIFrame**) &anonymousBlockParent);
|
|
anonymousBlock->RemoveFirstFrame();
|
|
aOldFrame->GetNextInFlow(&nextInFlow);
|
|
aOldFrame->Destroy(aPresContext);
|
|
while (nsnull != nextInFlow) {
|
|
nsIFrame* nextParent;
|
|
nextInFlow->GetParent(&nextParent);
|
|
if (nextParent != anonymousBlock) {
|
|
anonymousBlock = (nsAnonymousBlockFrame*) nextParent;
|
|
}
|
|
anonymousBlock->RemoveFirstFrame();
|
|
nsIFrame* nextNextInFlow;
|
|
nextInFlow->GetNextInFlow(&nextNextInFlow);
|
|
nextInFlow->Destroy(aPresContext);
|
|
nextInFlow = nextNextInFlow;
|
|
}
|
|
|
|
// Gather up the inline frames that follow aOldFrame
|
|
nsFrameList frames;
|
|
PRBool done = PR_FALSE;
|
|
while (!done && (nsnull != anonymousBlock)) {
|
|
nsIFrame* kid;
|
|
anonymousBlock->FirstChild(nsnull, &kid);
|
|
while (nsnull != kid) {
|
|
if (nsLineLayout::TreatFrameAsBlock(kid)) {
|
|
done = PR_TRUE;
|
|
break;
|
|
}
|
|
nsIFrame* next;
|
|
kid->GetNextSibling(&next);
|
|
anonymousBlock->RemoveFirstFrame();
|
|
frames.AppendFrame(nsnull, kid);
|
|
kid = next;
|
|
}
|
|
anonymousBlock->GetNextInFlow((nsIFrame**) &anonymousBlock);
|
|
}
|
|
|
|
if (frames.NotEmpty()) {
|
|
// If the anonymousBlockParent has a prev-in-flow then
|
|
// append the inline frames there, otherwise insert them
|
|
// before the anonymousBlock.
|
|
anonymousBlockParent->GetPrevInFlow(&prevInFlow);
|
|
if (nsnull != prevInFlow) {
|
|
anonymousBlockParent = (nsInlineFrame*) prevInFlow;
|
|
anonymousBlockParent->mFrames.AppendFrames(anonymousBlockParent,
|
|
frames);
|
|
}
|
|
else {
|
|
anonymousBlockParent->mFrames.InsertFrames(anonymousBlockParent,
|
|
nsnull,
|
|
frames);
|
|
}
|
|
}
|
|
generateReflowCommand = PR_TRUE;
|
|
target = anonymousBlockParent;
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("RemoveFrame: case 5\n");
|
|
#endif
|
|
}
|
|
}
|
|
else {
|
|
// We can let the anonymousBlock remove the frame directly
|
|
#ifdef NOISY_ANON_BLOCK
|
|
printf("RemoveFrame: case 6\n");
|
|
#endif
|
|
anonymousBlock->RemoveFrame2(&aPresContext, &aPresShell, aListName,
|
|
aOldFrame);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (generateReflowCommand) {
|
|
// generate a reflow command for "this"
|
|
nsIReflowCommand* reflowCmd = nsnull;
|
|
rv = NS_NewHTMLReflowCommand(&reflowCmd, target,
|
|
nsIReflowCommand::ReflowDirty);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
aPresShell.AppendReflowCommand(reflowCmd);
|
|
NS_RELEASE(reflowCmd);
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// Reflow methods
|
|
|
|
NS_IMETHODIMP
|
|
nsInlineFrame::Reflow(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aMetrics,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus)
|
|
{
|
|
if (nsnull == aReflowState.mLineLayout) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
DrainOverflow(&aPresContext);
|
|
|
|
// Set our own reflow state (additional state above and beyond
|
|
// aReflowState)
|
|
InlineReflowState irs;
|
|
irs.mPrevFrame = nsnull;
|
|
irs.mNextInFlow = (nsInlineFrame*) mNextInFlow;
|
|
irs.mNextRCFrame = nsnull;
|
|
if (eReflowReason_Incremental == aReflowState.reason) {
|
|
// Peel off the next frame in the path if this is an incremental
|
|
// reflow aimed at one of the children.
|
|
nsIFrame* target;
|
|
aReflowState.reflowCommand->GetTarget(target);
|
|
if (this != target) {
|
|
aReflowState.reflowCommand->GetNext(irs.mNextRCFrame);
|
|
}
|
|
}
|
|
|
|
nsresult rv;
|
|
if (mFrames.IsEmpty()) {
|
|
// Try to pull over one frame before starting so that we know
|
|
// whether we have an anonymous block or not.
|
|
(void) PullAnyFrame(&aPresContext, irs);
|
|
}
|
|
|
|
if (HaveAnonymousBlock()) {
|
|
if (!aReflowState.mLineLayout->LineIsEmpty()) {
|
|
// This inline frame cannot be placed on the current line
|
|
// because there already is an inline frame on this line (and we
|
|
// contain an anonymous block).
|
|
aStatus = NS_INLINE_LINE_BREAK_BEFORE();
|
|
rv = NS_OK;
|
|
}
|
|
else {
|
|
rv = ReflowBlockFrame(&aPresContext, aReflowState, irs,
|
|
aMetrics, aStatus);
|
|
|
|
// If the combined area of our children exceeds our bounding box
|
|
// then set the NS_FRAME_OUTSIDE_CHILDREN flag, otherwise clear
|
|
// it.
|
|
if ((aMetrics.mCombinedArea.x < 0) ||
|
|
(aMetrics.mCombinedArea.y < 0) ||
|
|
(aMetrics.mCombinedArea.XMost() > aMetrics.width) ||
|
|
(aMetrics.mCombinedArea.YMost() > aMetrics.height)) {
|
|
mState |= NS_FRAME_OUTSIDE_CHILDREN;
|
|
}
|
|
else {
|
|
mState &= ~NS_FRAME_OUTSIDE_CHILDREN;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
rv = ReflowInlineFrames(&aPresContext, aReflowState, irs,
|
|
aMetrics, aStatus);
|
|
// Note: when we are reflowing inline frames the line layout code
|
|
// will properly compute our NS_FRAME_OUTSIDE_CHILDREN state for
|
|
// us.
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsInlineFrame::FindTextRuns(nsLineLayout& aLineLayout)
|
|
{
|
|
if (HaveAnonymousBlock()) {
|
|
aLineLayout.EndTextRun();
|
|
}
|
|
else {
|
|
nsIFrame* frame = mFrames.FirstChild();
|
|
while (nsnull != frame) {
|
|
nsIHTMLReflow* ihr;
|
|
nsresult rv = frame->QueryInterface(kIHTMLReflowIID, (void**)&ihr);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
ihr->FindTextRuns(aLineLayout);
|
|
}
|
|
frame->GetNextSibling(&frame);
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
nsInlineFrame::DrainOverflow(nsIPresContext* aPresContext)
|
|
{
|
|
// Check for an overflow list with our prev-in-flow
|
|
nsInlineFrame* prevInFlow = (nsInlineFrame*)mPrevInFlow;
|
|
if (nsnull != prevInFlow) {
|
|
if (prevInFlow->mOverflowFrames.NotEmpty()) {
|
|
// When pushing and pulling frames we need to check for whether any
|
|
// views need to be reparented.
|
|
// XXX Doing it this way means an extra pass over the frames. We could
|
|
// change InsertFrames() to do this, but that's a general purpose
|
|
// function and it doesn't seem like this functionality belongs there...
|
|
for (nsIFrame* f = prevInFlow->mOverflowFrames.FirstChild(); f; f->GetNextSibling(&f)) {
|
|
nsHTMLContainerFrame::ReparentFrameView(f, prevInFlow, this);
|
|
}
|
|
mFrames.InsertFrames(this, nsnull, prevInFlow->mOverflowFrames);
|
|
}
|
|
}
|
|
|
|
// It's also possible that we have an overflow list for ourselves
|
|
if (mOverflowFrames.NotEmpty()) {
|
|
NS_ASSERTION(mFrames.NotEmpty(), "overflow list w/o frames");
|
|
mFrames.AppendFrames(nsnull, mOverflowFrames);
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
nsInlineFrame::ReflowInlineFrames(nsIPresContext* aPresContext,
|
|
const nsHTMLReflowState& aReflowState,
|
|
InlineReflowState& irs,
|
|
nsHTMLReflowMetrics& aMetrics,
|
|
nsReflowStatus& aStatus)
|
|
{
|
|
nsresult rv = NS_OK;
|
|
aStatus = NS_FRAME_COMPLETE;
|
|
|
|
nsLineLayout* lineLayout = aReflowState.mLineLayout;
|
|
nscoord leftEdge = 0;
|
|
if (nsnull == mPrevInFlow) {
|
|
leftEdge = aReflowState.mComputedBorderPadding.left;
|
|
}
|
|
nscoord availableWidth = aReflowState.availableWidth;
|
|
if (NS_UNCONSTRAINEDSIZE != availableWidth) {
|
|
// Subtract off left and right border+padding from availableWidth
|
|
availableWidth -= leftEdge;
|
|
availableWidth -= aReflowState.mComputedBorderPadding.right;
|
|
}
|
|
lineLayout->BeginSpan(this, &aReflowState,
|
|
leftEdge, leftEdge + availableWidth);
|
|
|
|
// First reflow our current children
|
|
nsIFrame* frame = mFrames.FirstChild();
|
|
PRBool done = PR_FALSE;
|
|
while (nsnull != frame) {
|
|
PRBool reflowingFirstLetter = lineLayout->GetFirstLetterStyleOK();
|
|
rv = ReflowInlineFrame(aPresContext, aReflowState, irs, frame, aStatus);
|
|
if (NS_FAILED(rv)) {
|
|
done = PR_TRUE;
|
|
break;
|
|
}
|
|
if (NS_FRAME_COMPLETE != aStatus) {
|
|
if (!reflowingFirstLetter || NS_INLINE_IS_BREAK(aStatus)) {
|
|
done = PR_TRUE;
|
|
break;
|
|
}
|
|
}
|
|
irs.mPrevFrame = frame;
|
|
frame->GetNextSibling(&frame);
|
|
}
|
|
|
|
// Attempt to pull frames from our next-in-flow until we can't
|
|
if (!done && (nsnull != mNextInFlow)) {
|
|
while (!done) {
|
|
PRBool reflowingFirstLetter = lineLayout->GetFirstLetterStyleOK();
|
|
PRBool isComplete;
|
|
frame = PullInlineFrame(aPresContext, irs, &isComplete);
|
|
if (nsnull == frame) {
|
|
if (!isComplete) {
|
|
aStatus = NS_FRAME_NOT_COMPLETE;
|
|
}
|
|
break;
|
|
}
|
|
rv = ReflowInlineFrame(aPresContext, aReflowState, irs, frame, aStatus);
|
|
if (NS_FAILED(rv)) {
|
|
done = PR_TRUE;
|
|
break;
|
|
}
|
|
if (NS_FRAME_COMPLETE != aStatus) {
|
|
if (!reflowingFirstLetter || NS_INLINE_IS_BREAK(aStatus)) {
|
|
done = PR_TRUE;
|
|
break;
|
|
}
|
|
}
|
|
irs.mPrevFrame = frame;
|
|
}
|
|
}
|
|
#ifdef DEBUG
|
|
if (NS_FRAME_COMPLETE == aStatus) {
|
|
// We can't be complete AND have overflow frames!
|
|
NS_ASSERTION(mOverflowFrames.IsEmpty(), "whoops");
|
|
}
|
|
#endif
|
|
|
|
// If after reflowing our children they take up no area then make
|
|
// sure that we don't either.
|
|
//
|
|
// Note: CSS demands that empty inline elements still affect the
|
|
// line-height calculations. However, continuations of an inline
|
|
// that are empty we force to empty so that things like collapsed
|
|
// whitespace in an inline element don't affect the line-height.
|
|
nsSize size;
|
|
lineLayout->EndSpan(this, size, aMetrics.maxElementSize);
|
|
if ((0 == size.height) && (0 == size.width) &&
|
|
((nsnull != mPrevInFlow) || (nsnull != mNextInFlow))) {
|
|
// This is a continuation of a previous inline. Therefore make
|
|
// sure we don't affect the line-height.
|
|
aMetrics.width = 0;
|
|
aMetrics.height = 0;
|
|
aMetrics.ascent = 0;
|
|
aMetrics.descent = 0;
|
|
if (nsnull != aMetrics.maxElementSize) {
|
|
aMetrics.maxElementSize->width = 0;
|
|
aMetrics.maxElementSize->height = 0;
|
|
}
|
|
}
|
|
else {
|
|
// Compute final width
|
|
aMetrics.width = size.width;
|
|
if (nsnull == mPrevInFlow) {
|
|
aMetrics.width += aReflowState.mComputedBorderPadding.left;
|
|
}
|
|
if (NS_FRAME_IS_COMPLETE(aStatus)) {
|
|
aMetrics.width += aReflowState.mComputedBorderPadding.right;
|
|
}
|
|
|
|
const nsStyleFont* font;
|
|
GetStyleData(eStyleStruct_Font, (const nsStyleStruct*&)font);
|
|
aReflowState.rendContext->SetFont(font->mFont);
|
|
nsIFontMetrics* fm;
|
|
aReflowState.rendContext->GetFontMetrics(fm);
|
|
|
|
// Compute final height. The height of our box is the sum of our
|
|
// font size plus the top and bottom border and padding. The height
|
|
// of children do not affect our height.
|
|
fm->GetMaxAscent(aMetrics.ascent);
|
|
fm->GetMaxDescent(aMetrics.descent);
|
|
fm->GetHeight(aMetrics.height);
|
|
aMetrics.ascent += aReflowState.mComputedBorderPadding.top;
|
|
aMetrics.descent += aReflowState.mComputedBorderPadding.bottom;
|
|
aMetrics.height += aReflowState.mComputedBorderPadding.top +
|
|
aReflowState.mComputedBorderPadding.bottom;
|
|
|
|
// Note: we use the actual font height for sizing our selves
|
|
// instead of the computed font height. On systems where they
|
|
// disagree the actual font height is more appropriate. This
|
|
// little hack lets us override that behavior to allow for more
|
|
// precise layout in the face of imprecise fonts.
|
|
static PRBool useComputedHeight = PR_FALSE;
|
|
#if defined(XP_UNIX) || defined(XP_PC) || defined(XP_BEOS)
|
|
static PRBool firstTime = 1;
|
|
if (firstTime) {
|
|
if (getenv("GECKO_USE_COMPUTED_HEIGHT")) {
|
|
useComputedHeight = PR_TRUE;
|
|
}
|
|
}
|
|
#endif
|
|
if (useComputedHeight) {
|
|
// Special debug code that violates the above CSS2 spec
|
|
// clarification. Why? So that we can predictably compute the
|
|
// values for testing layout.
|
|
nscoord computedHeight = aReflowState.mComputedBorderPadding.top +
|
|
aReflowState.mComputedBorderPadding.bottom +
|
|
font->mFont.size;
|
|
if (computedHeight != aMetrics.height) {
|
|
#ifdef DEBUG
|
|
if (0 == (mState & 0x80000000)) {
|
|
nsFrame::ListTag(stdout, this);
|
|
printf(": using computedHeight %d instead of actual height %d\n",
|
|
computedHeight, aMetrics.height);
|
|
mState |= 0x80000000;
|
|
}
|
|
#endif
|
|
aMetrics.height = computedHeight;
|
|
}
|
|
}
|
|
NS_RELEASE(fm);
|
|
}
|
|
|
|
// For now our combined area is zero. The real value will be
|
|
// computed during vertical alignment of the line we are on.
|
|
aMetrics.mCombinedArea.x = 0;
|
|
aMetrics.mCombinedArea.y = 0;
|
|
aMetrics.mCombinedArea.width = aMetrics.width;
|
|
aMetrics.mCombinedArea.height = aMetrics.height;
|
|
|
|
#ifdef NOISY_FINAL_SIZE
|
|
ListTag(stdout);
|
|
printf(": metrics=%d,%d ascent=%d descent=%d\n",
|
|
aMetrics.width, aMetrics.height, aMetrics.ascent, aMetrics.descent);
|
|
#endif
|
|
|
|
return rv;
|
|
}
|
|
|
|
nsresult
|
|
nsInlineFrame::ReflowInlineFrame(nsIPresContext* aPresContext,
|
|
const nsHTMLReflowState& aReflowState,
|
|
InlineReflowState& irs,
|
|
nsIFrame* aFrame,
|
|
nsReflowStatus& aStatus)
|
|
{
|
|
// Make sure that we don't reflow a block frame when we run across
|
|
// one. This can easily happen if this inline has a mixture of
|
|
// frames (note that an anonymous block frame is used to wrap up the
|
|
// direct block children of this inline therefore when we do run
|
|
// across a block frame its an anonymous block).
|
|
if (nsLineLayout::TreatFrameAsBlock(aFrame)) {
|
|
NS_ASSERTION(aFrame != mFrames.FirstChild(), "bad anon-block status");
|
|
PushFrames(aPresContext, aFrame, irs.mPrevFrame);
|
|
aStatus = NS_INLINE_LINE_BREAK_AFTER(NS_FRAME_NOT_COMPLETE);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsLineLayout* lineLayout = aReflowState.mLineLayout;
|
|
PRBool reflowingFirstLetter = lineLayout->GetFirstLetterStyleOK();
|
|
nsresult rv = lineLayout->ReflowFrame(aFrame, &irs.mNextRCFrame, aStatus);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
if (NS_INLINE_IS_BREAK(aStatus)) {
|
|
if (NS_INLINE_IS_BREAK_BEFORE(aStatus)) {
|
|
if (aFrame != mFrames.FirstChild()) {
|
|
// Change break-before status into break-after since we have
|
|
// already placed at least one child frame. This preserves the
|
|
// break-type so that it can be propogated upward.
|
|
aStatus = NS_FRAME_NOT_COMPLETE |
|
|
NS_INLINE_BREAK | NS_INLINE_BREAK_AFTER |
|
|
(aStatus & NS_INLINE_BREAK_TYPE_MASK);
|
|
PushFrames(aPresContext, aFrame, irs.mPrevFrame);
|
|
}
|
|
else {
|
|
// Preserve reflow status when breaking-before our first child
|
|
// and propogate it upward without modification.
|
|
}
|
|
}
|
|
else {
|
|
// Break-after
|
|
if (NS_FRAME_IS_NOT_COMPLETE(aStatus)) {
|
|
nsIFrame* newFrame;
|
|
rv = CreateNextInFlow(*aPresContext, this, aFrame, newFrame);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
}
|
|
nsIFrame* nextFrame;
|
|
aFrame->GetNextSibling(&nextFrame);
|
|
if (nsnull != nextFrame) {
|
|
aStatus |= NS_FRAME_NOT_COMPLETE;
|
|
PushFrames(aPresContext, nextFrame, aFrame);
|
|
}
|
|
else if (nsnull != mNextInFlow) {
|
|
// We must return an incomplete status if there are more child
|
|
// frames remaining in a next-in-flow that follows this frame.
|
|
nsInlineFrame* nextInFlow = (nsInlineFrame*) mNextInFlow;
|
|
while (nsnull != nextInFlow) {
|
|
if (nextInFlow->mFrames.NotEmpty()) {
|
|
aStatus |= NS_FRAME_NOT_COMPLETE;
|
|
break;
|
|
}
|
|
nextInFlow = (nsInlineFrame*) nextInFlow->mNextInFlow;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if (NS_FRAME_IS_NOT_COMPLETE(aStatus)) {
|
|
nsIFrame* newFrame;
|
|
rv = CreateNextInFlow(*aPresContext, this, aFrame, newFrame);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
if (!reflowingFirstLetter) {
|
|
nsIFrame* nextFrame;
|
|
aFrame->GetNextSibling(&nextFrame);
|
|
if (nsnull != nextFrame) {
|
|
PushFrames(aPresContext, nextFrame, aFrame);
|
|
}
|
|
}
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
nsIFrame*
|
|
nsInlineFrame::PullInlineFrame(nsIPresContext* aPresContext,
|
|
InlineReflowState& irs,
|
|
PRBool* aIsComplete)
|
|
{
|
|
PRBool isComplete = PR_TRUE;
|
|
|
|
nsIFrame* frame = nsnull;
|
|
nsInlineFrame* nextInFlow = irs.mNextInFlow;
|
|
while (nsnull != nextInFlow) {
|
|
if (nextInFlow->HaveAnonymousBlock()) {
|
|
isComplete = PR_FALSE;
|
|
break;
|
|
}
|
|
frame = mFrames.PullFrame(this, irs.mPrevFrame, nextInFlow->mFrames);
|
|
if (nsnull != frame) {
|
|
isComplete = PR_FALSE;
|
|
nsHTMLContainerFrame::ReparentFrameView(frame, nextInFlow, this);
|
|
break;
|
|
}
|
|
nextInFlow = (nsInlineFrame*) nextInFlow->mNextInFlow;
|
|
irs.mNextInFlow = nextInFlow;
|
|
}
|
|
|
|
*aIsComplete = isComplete;
|
|
return frame;
|
|
}
|
|
|
|
nsIFrame*
|
|
nsInlineFrame::PullAnyFrame(nsIPresContext* aPresContext,
|
|
InlineReflowState& irs)
|
|
{
|
|
nsIFrame* frame = nsnull;
|
|
nsInlineFrame* nextInFlow = irs.mNextInFlow;
|
|
while (nsnull != nextInFlow) {
|
|
frame = mFrames.PullFrame(this, irs.mPrevFrame, nextInFlow->mFrames);
|
|
if (nsnull != frame) {
|
|
nsHTMLContainerFrame::ReparentFrameView(frame, nextInFlow, this);
|
|
break;
|
|
}
|
|
|
|
nextInFlow = (nsInlineFrame*) nextInFlow->mNextInFlow;
|
|
irs.mNextInFlow = nextInFlow;
|
|
}
|
|
|
|
return frame;
|
|
}
|
|
|
|
void
|
|
nsInlineFrame::PushFrames(nsIPresContext* aPresContext,
|
|
nsIFrame* aFromChild,
|
|
nsIFrame* aPrevSibling)
|
|
{
|
|
NS_PRECONDITION(nsnull != aFromChild, "null pointer");
|
|
NS_PRECONDITION(nsnull != aPrevSibling, "pushing first child");
|
|
#ifdef DEBUG
|
|
nsIFrame* prevNextSibling;
|
|
aPrevSibling->GetNextSibling(&prevNextSibling);
|
|
NS_PRECONDITION(prevNextSibling == aFromChild, "bad prev sibling");
|
|
#endif
|
|
|
|
// Disconnect aFromChild from its previous sibling
|
|
aPrevSibling->SetNextSibling(nsnull);
|
|
|
|
// Add the frames to our overflow list (let our next in flow drain
|
|
// our overflow list when it is ready)
|
|
NS_ASSERTION(mOverflowFrames.IsEmpty(), "bad overflow list");
|
|
mOverflowFrames.SetFrames(aFromChild);
|
|
}
|
|
|
|
nsresult
|
|
nsInlineFrame::ReflowBlockFrame(nsIPresContext* aPresContext,
|
|
const nsHTMLReflowState& aReflowState,
|
|
InlineReflowState& irs,
|
|
nsHTMLReflowMetrics& aMetrics,
|
|
nsReflowStatus& aStatus)
|
|
{
|
|
nsIFrame* blockFrame = mFrames.FirstChild();
|
|
|
|
// Compute available area
|
|
#if XXX_what_to_do
|
|
nscoord x = aReflowState.mComputedBorderPadding.left;
|
|
nscoord availableWidth = aReflowState.availableWidth;
|
|
if (NS_UNCONSTRAINEDSIZE != availableWidth) {
|
|
if (nsnull != mPrevInFlow) {
|
|
x = 0;
|
|
availableWidth -= aReflowState.mComputedBorderPadding.right;
|
|
}
|
|
else {
|
|
availableWidth -= aReflowState.mComputedBorderPadding.left +
|
|
aReflowState.mComputedBorderPadding.right;
|
|
}
|
|
}
|
|
nscoord y = aReflowState.mComputedBorderPadding.top;
|
|
nscoord availableHeight = aReflowState.availableHeight;
|
|
if (NS_UNCONSTRAINEDSIZE != aReflowState.availableHeight) {
|
|
availableHeight -= aReflowState.mComputedBorderPadding.top +
|
|
aReflowState.mComputedBorderPadding.right;
|
|
}
|
|
#else
|
|
nscoord x = 0;
|
|
nscoord availableWidth = aReflowState.availableWidth;
|
|
nscoord y = 0;
|
|
nscoord availableHeight = aReflowState.availableHeight;
|
|
#endif
|
|
|
|
// XXX_ib write me...
|
|
//XXX nscoord collapsedTopMargin = 0;
|
|
nscoord collapsedBottomMargin = 0;
|
|
|
|
// Reflow the block frame
|
|
nsBlockReflowContext bc(aPresContext, aReflowState,
|
|
nsnull != aMetrics.maxElementSize);
|
|
bc.SetNextRCFrame(irs.mNextRCFrame);
|
|
nsRect availSpace(x, y, availableWidth, availableHeight);
|
|
PRBool isAdjacentWithTop = PR_FALSE;
|
|
nsMargin computedOffsets;
|
|
nsresult rv = bc.ReflowBlock(blockFrame, availSpace, PR_FALSE, 0,
|
|
isAdjacentWithTop,
|
|
computedOffsets, aStatus);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
if (NS_INLINE_IS_BREAK_BEFORE(aStatus)) {
|
|
// We need to break before placing the block so propogate that
|
|
// status outward to our parent.
|
|
}
|
|
else {
|
|
// Place the block (during placement we might discover that none
|
|
// of it fits)
|
|
nsRect bounds;
|
|
PRBool anyFit = bc.PlaceBlock(isAdjacentWithTop,
|
|
computedOffsets, &collapsedBottomMargin,
|
|
bounds, aMetrics.mCombinedArea);
|
|
if (!anyFit) {
|
|
// None of the block fit
|
|
aStatus = NS_INLINE_LINE_BREAK_BEFORE();
|
|
}
|
|
else {
|
|
if (NS_FRAME_IS_NOT_COMPLETE(aStatus)) {
|
|
// When the block isn't complete create a continuation for it
|
|
nsIFrame* newFrame;
|
|
rv = CreateNextInFlow(*aPresContext, this, blockFrame, newFrame);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
}
|
|
|
|
// It's possible that the block frame is followed by one or more
|
|
// inline frames. Make sure we reflect that in our reflow status
|
|
// and push them to our next-in-flow.
|
|
nsIFrame* nextFrame;
|
|
blockFrame->GetNextSibling(&nextFrame);
|
|
if (nsnull != nextFrame) {
|
|
PushFrames(aPresContext, nextFrame, blockFrame);
|
|
aStatus |= NS_FRAME_NOT_COMPLETE;
|
|
}
|
|
else if (NS_FRAME_IS_COMPLETE(aStatus)) {
|
|
// When the block we reflowed is complete then we need to
|
|
// check and see if there are other frames (inline frames)
|
|
// following in our continuations so that we return the proper
|
|
// reflow status.
|
|
nsInlineFrame* nextInFlow = (nsInlineFrame*) mNextInFlow;
|
|
while (nsnull != nextInFlow) {
|
|
if (nextInFlow->mFrames.NotEmpty() ||
|
|
nextInFlow->mOverflowFrames.NotEmpty()) {
|
|
aStatus |= NS_FRAME_NOT_COMPLETE;
|
|
break;
|
|
}
|
|
nextInFlow = (nsInlineFrame*) nextInFlow->mNextInFlow;
|
|
}
|
|
}
|
|
|
|
// What we do here is to fudge the size. <b>This</b> frame will
|
|
// be 0,0 but will contain a single child (the anonymous block)
|
|
// that is properly sized.
|
|
aMetrics.width = bounds.width;
|
|
aMetrics.height = bounds.height;
|
|
aMetrics.ascent = bounds.height;
|
|
aMetrics.descent = 0;
|
|
aMetrics.mCarriedOutBottomMargin = bc.GetCarriedOutBottomMargin();
|
|
if (nsnull != aMetrics.maxElementSize) {
|
|
*aMetrics.maxElementSize = bc.GetMaxElementSize();
|
|
}
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
PRIntn
|
|
nsInlineFrame::GetSkipSides() const
|
|
{
|
|
PRIntn skip = 0;
|
|
if (nsnull != mPrevInFlow) {
|
|
nsInlineFrame* prev = (nsInlineFrame*) mPrevInFlow;
|
|
if (prev->mRect.height || prev->mRect.width) {
|
|
// Prev-in-flow is not empty therefore we don't render our left
|
|
// border edge.
|
|
skip |= 1 << NS_SIDE_LEFT;
|
|
}
|
|
else {
|
|
// If the prev-in-flow is empty, then go ahead and let our right
|
|
// edge border render.
|
|
}
|
|
}
|
|
if (nsnull != mNextInFlow) {
|
|
nsInlineFrame* next = (nsInlineFrame*) mNextInFlow;
|
|
if (next->mRect.height || next->mRect.width) {
|
|
// Next-in-flow is not empty therefore we don't render our right
|
|
// border edge.
|
|
skip |= 1 << NS_SIDE_RIGHT;
|
|
}
|
|
else {
|
|
// If the next-in-flow is empty, then go ahead and let our right
|
|
// edge border render.
|
|
}
|
|
}
|
|
return skip;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// nsLineFrame implementation
|
|
|
|
static void
|
|
ReResolveChildList(nsIPresContext* aPresContext,
|
|
nsIStyleContext* aParentStyleContext,
|
|
nsFrameList& aFrameList)
|
|
{
|
|
nsIFrame* kid = aFrameList.FirstChild();
|
|
while (nsnull != kid) {
|
|
kid->ReResolveStyleContext(aPresContext, aParentStyleContext,
|
|
NS_STYLE_HINT_REFLOW,
|
|
nsnull, nsnull);
|
|
kid->GetNextSibling(&kid);
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
NS_NewFirstLineFrame(nsIFrame** aNewFrame)
|
|
{
|
|
NS_PRECONDITION(nsnull != aNewFrame, "null ptr");
|
|
if (nsnull == aNewFrame) {
|
|
return NS_ERROR_NULL_POINTER;
|
|
}
|
|
nsInlineFrame* it = new nsFirstLineFrame;
|
|
if (nsnull == it) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
*aNewFrame = it;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsFirstLineFrame::nsFirstLineFrame()
|
|
{
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsFirstLineFrame::GetFrameName(nsString& aResult) const
|
|
{
|
|
return MakeFrameName("Line", aResult);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsFirstLineFrame::GetFrameType(nsIAtom** aType) const
|
|
{
|
|
NS_PRECONDITION(nsnull != aType, "null OUT parameter pointer");
|
|
*aType = nsLayoutAtoms::lineFrame;
|
|
NS_ADDREF(*aType);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsFirstLineFrame::AppendFrames(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
return mParent->AppendFrames(aPresContext, aPresShell, aListName,
|
|
aFrameList);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsFirstLineFrame::InsertFrames(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aPrevFrame,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
return mParent->InsertFrames(aPresContext, aPresShell, aListName,
|
|
aPrevFrame, aFrameList);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsFirstLineFrame::RemoveFrame(nsIPresContext& aPresContext,
|
|
nsIPresShell& aPresShell,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aOldFrame)
|
|
{
|
|
return mParent->RemoveFrame(aPresContext, aPresShell, aListName,
|
|
aOldFrame);
|
|
}
|
|
|
|
nsresult
|
|
nsFirstLineFrame::AppendFrames2(nsIPresContext* aPresContext,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
nsFrameList frames(aFrameList);
|
|
ReResolveChildList(aPresContext, mStyleContext, frames);
|
|
// XXX ReparentFrameView
|
|
mFrames.AppendFrames(this, aFrameList);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsFirstLineFrame::InsertFrames2(nsIPresContext* aPresContext,
|
|
nsIFrame* aPrevFrame,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
nsFrameList frames(aFrameList);
|
|
ReResolveChildList(aPresContext, mStyleContext, frames);
|
|
// XXX ReparentFrameView
|
|
mFrames.InsertFrames(this, aPrevFrame, aFrameList);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsFirstLineFrame::RemoveFrame2(nsIPresContext* aPresContext,
|
|
nsIFrame* aOldFrame)
|
|
{
|
|
nsIFrame* nextInFlow;
|
|
aOldFrame->GetNextInFlow(&nextInFlow);
|
|
if (nextInFlow) {
|
|
DeleteChildsNextInFlow(*aPresContext, aOldFrame);
|
|
}
|
|
mFrames.RemoveFrame(aOldFrame);
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
nsFirstLineFrame::RemoveFramesFrom(nsIFrame* aFrame)
|
|
{
|
|
nsIFrame* prevFrame = mFrames.GetPrevSiblingFor(aFrame);
|
|
if (prevFrame) {
|
|
prevFrame->SetNextSibling(nsnull);
|
|
}
|
|
else {
|
|
mFrames.SetFrames(nsnull);
|
|
}
|
|
}
|
|
|
|
nsIFrame*
|
|
nsFirstLineFrame::PullInlineFrame(nsIPresContext* aPresContext,
|
|
InlineReflowState& irs,
|
|
PRBool* aIsComplete)
|
|
{
|
|
nsIFrame* frame =
|
|
nsInlineFrame::PullInlineFrame(aPresContext, irs, aIsComplete);
|
|
if (frame && !mPrevInFlow) {
|
|
// We are a first-line frame. Fixup the child frames
|
|
// style-context that we just pulled.
|
|
frame->ReResolveStyleContext(aPresContext, mStyleContext,
|
|
NS_STYLE_HINT_REFLOW, nsnull, nsnull);
|
|
}
|
|
return frame;
|
|
}
|
|
|
|
void
|
|
nsFirstLineFrame::DrainOverflow(nsIPresContext* aPresContext)
|
|
{
|
|
// Check for an overflow list with our prev-in-flow
|
|
nsFirstLineFrame* prevInFlow = (nsFirstLineFrame*)mPrevInFlow;
|
|
if (nsnull != prevInFlow) {
|
|
if (prevInFlow->mOverflowFrames.NotEmpty()) {
|
|
ReResolveChildList(aPresContext, mStyleContext,
|
|
prevInFlow->mOverflowFrames);
|
|
mFrames.InsertFrames(this, nsnull, prevInFlow->mOverflowFrames);
|
|
}
|
|
}
|
|
|
|
// It's also possible that we have an overflow list for ourselves
|
|
if (mOverflowFrames.NotEmpty()) {
|
|
NS_ASSERTION(mFrames.NotEmpty(), "overflow list w/o frames");
|
|
ReResolveChildList(aPresContext, mStyleContext, mOverflowFrames);
|
|
mFrames.AppendFrames(nsnull, mOverflowFrames);
|
|
}
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsFirstLineFrame::Reflow(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aMetrics,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus)
|
|
{
|
|
if (nsnull == aReflowState.mLineLayout) {
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
DrainOverflow(&aPresContext);
|
|
|
|
// Set our own reflow state (additional state above and beyond
|
|
// aReflowState)
|
|
InlineReflowState irs;
|
|
irs.mPrevFrame = nsnull;
|
|
irs.mNextInFlow = (nsInlineFrame*) mNextInFlow;
|
|
irs.mNextRCFrame = nsnull;
|
|
if (eReflowReason_Incremental == aReflowState.reason) {
|
|
// Peel off the next frame in the path if this is an incremental
|
|
// reflow aimed at one of the children.
|
|
nsIFrame* target;
|
|
aReflowState.reflowCommand->GetTarget(target);
|
|
if (this != target) {
|
|
aReflowState.reflowCommand->GetNext(irs.mNextRCFrame);
|
|
}
|
|
}
|
|
|
|
nsresult rv;
|
|
PRBool wasEmpty = mFrames.IsEmpty();
|
|
if (wasEmpty) {
|
|
// Try to pull over one frame before starting so that we know
|
|
// whether we have an anonymous block or not.
|
|
PullAnyFrame(&aPresContext, irs);
|
|
}
|
|
|
|
if (HaveAnonymousBlock()) {
|
|
if (!aReflowState.mLineLayout->LineIsEmpty()) {
|
|
// This inline frame cannot be placed on the current line
|
|
// because there already is an inline frame on this line (and we
|
|
// contain an anonymous block).
|
|
aStatus = NS_INLINE_LINE_BREAK_BEFORE();
|
|
rv = NS_OK;
|
|
}
|
|
else {
|
|
rv = ReflowBlockFrame(&aPresContext, aReflowState, irs,
|
|
aMetrics, aStatus);
|
|
|
|
// If the combined area of our children exceeds our bounding box
|
|
// then set the NS_FRAME_OUTSIDE_CHILDREN flag, otherwise clear
|
|
// it.
|
|
if ((aMetrics.mCombinedArea.x < 0) ||
|
|
(aMetrics.mCombinedArea.y < 0) ||
|
|
(aMetrics.mCombinedArea.XMost() > aMetrics.width) ||
|
|
(aMetrics.mCombinedArea.YMost() > aMetrics.height)) {
|
|
mState |= NS_FRAME_OUTSIDE_CHILDREN;
|
|
}
|
|
else {
|
|
mState &= ~NS_FRAME_OUTSIDE_CHILDREN;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
if (wasEmpty) {
|
|
// Fixup style of frame just pulled up
|
|
nsIFrame* firstFrame = mFrames.FirstChild();
|
|
if (firstFrame) {
|
|
firstFrame->ReResolveStyleContext(&aPresContext, mStyleContext,
|
|
NS_STYLE_HINT_REFLOW, nsnull,
|
|
nsnull);
|
|
}
|
|
}
|
|
if (nsnull == mPrevInFlow) {
|
|
// XXX This is pretty sick, but what we do here is to pull-up, in
|
|
// advance, all of the next-in-flows children. We re-resolve their
|
|
// style while we are at at it so that when we reflow they have
|
|
// the right style.
|
|
//
|
|
// All of this is so that text-runs reflow properly.
|
|
irs.mPrevFrame = mFrames.LastChild();
|
|
for (;;) {
|
|
PRBool complete;
|
|
nsIFrame* frame = PullInlineFrame(&aPresContext, irs, &complete);
|
|
if (!frame) {
|
|
break;
|
|
}
|
|
irs.mPrevFrame = frame;
|
|
}
|
|
irs.mPrevFrame = nsnull;
|
|
}
|
|
else {
|
|
// XXX do this in the Init method instead
|
|
// For continuations, we need to check and see if our style
|
|
// context is right. If its the same as the first-in-flow, then
|
|
// we need to fix it up (that way :first-line style doesn't leak
|
|
// into this continuation since we aren't the first line).
|
|
nsFirstLineFrame* first = (nsFirstLineFrame*) GetFirstInFlow();
|
|
if (mStyleContext == first->mStyleContext) {
|
|
// Fixup our style context and our children. First get the
|
|
// proper parent context.
|
|
nsIFrame* parentFrame;
|
|
first->GetParent(&parentFrame);
|
|
nsIStyleContext* parentContext;
|
|
parentFrame->GetStyleContext(&parentContext);
|
|
if (parentContext) {
|
|
// Create a new style context that is a child of the parent
|
|
// style context thus removing the :first-line style. This way
|
|
// we behave as if an anonymous (unstyled) span was the child
|
|
// of the parent frame.
|
|
nsIStyleContext* newSC;
|
|
aPresContext.ResolvePseudoStyleContextFor(mContent,
|
|
nsHTMLAtoms::mozLineFrame,
|
|
parentContext,
|
|
PR_FALSE, &newSC);
|
|
if (newSC) {
|
|
// Switch to the new style context.
|
|
SetStyleContext(&aPresContext, newSC);
|
|
|
|
// Re-resolve all children
|
|
ReResolveChildList(&aPresContext, mStyleContext, mFrames);
|
|
|
|
NS_RELEASE(newSC);
|
|
}
|
|
NS_RELEASE(parentContext);
|
|
}
|
|
}
|
|
}
|
|
|
|
rv = ReflowInlineFrames(&aPresContext, aReflowState, irs,
|
|
aMetrics, aStatus);
|
|
// Note: when we are reflowing inline frames the line layout code
|
|
// will properly compute our NS_FRAME_OUTSIDE_CHILDREN state for
|
|
// us.
|
|
}
|
|
|
|
return rv;
|
|
}
|