2007-03-22 10:30:00 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (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/MPL/
|
|
|
|
*
|
|
|
|
* 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.org code.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* Netscape Communications Corporation.
|
|
|
|
* Portions created by the Initial Developer are Copyright (C) 1998
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
* Pierre Phaneuf <pp@ludusdesign.com>
|
2007-07-25 21:03:29 -07:00
|
|
|
* Elika J. Etemad ("fantasai") <fantasai@inkedblade.net>
|
2007-03-22 10:30:00 -07:00
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
|
|
* either of the GNU General Public License Version 2 or later (the "GPL"),
|
|
|
|
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
|
|
* the provisions above, a recipient may use your version of this file under
|
|
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
|
|
*
|
|
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
|
|
|
|
/* base class #1 for rendering objects that have child lists */
|
|
|
|
|
|
|
|
#include "nsContainerFrame.h"
|
2007-07-25 21:03:29 -07:00
|
|
|
#include "nsHTMLContainerFrame.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
#include "nsIContent.h"
|
|
|
|
#include "nsIDocument.h"
|
|
|
|
#include "nsPresContext.h"
|
|
|
|
#include "nsIRenderingContext.h"
|
|
|
|
#include "nsStyleContext.h"
|
|
|
|
#include "nsRect.h"
|
|
|
|
#include "nsPoint.h"
|
|
|
|
#include "nsGUIEvent.h"
|
|
|
|
#include "nsStyleConsts.h"
|
|
|
|
#include "nsIView.h"
|
|
|
|
#include "nsHTMLContainerFrame.h"
|
|
|
|
#include "nsFrameManager.h"
|
|
|
|
#include "nsIPresShell.h"
|
|
|
|
#include "nsCOMPtr.h"
|
|
|
|
#include "nsGkAtoms.h"
|
|
|
|
#include "nsCSSAnonBoxes.h"
|
|
|
|
#include "nsIViewManager.h"
|
|
|
|
#include "nsIWidget.h"
|
|
|
|
#include "nsGfxCIID.h"
|
|
|
|
#include "nsIServiceManager.h"
|
|
|
|
#include "nsCSSRendering.h"
|
|
|
|
#include "nsTransform2D.h"
|
|
|
|
#include "nsRegion.h"
|
|
|
|
#include "nsLayoutErrors.h"
|
|
|
|
#include "nsDisplayList.h"
|
|
|
|
#include "nsContentErrors.h"
|
|
|
|
#include "nsIEventStateManager.h"
|
2007-12-22 13:07:58 -08:00
|
|
|
#include "nsListControlFrame.h"
|
2008-01-09 14:50:29 -08:00
|
|
|
#include "nsIBaseWindow.h"
|
2008-08-12 17:44:14 -07:00
|
|
|
#include "nsThemeConstants.h"
|
|
|
|
#include "nsCSSFrameConstructor.h"
|
|
|
|
#include "nsThemeConstants.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
#ifdef NS_DEBUG
|
|
|
|
#undef NOISY
|
|
|
|
#else
|
|
|
|
#undef NOISY
|
|
|
|
#endif
|
|
|
|
|
2009-09-12 09:49:24 -07:00
|
|
|
NS_IMPL_FRAMEARENA_HELPERS(nsContainerFrame)
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
nsContainerFrame::~nsContainerFrame()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2009-09-18 04:09:35 -07:00
|
|
|
NS_QUERYFRAME_HEAD(nsContainerFrame)
|
|
|
|
NS_QUERYFRAME_ENTRY(nsContainerFrame)
|
|
|
|
NS_QUERYFRAME_TAIL_INHERITING(nsSplittableFrame)
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsContainerFrame::Init(nsIContent* aContent,
|
|
|
|
nsIFrame* aParent,
|
|
|
|
nsIFrame* aPrevInFlow)
|
|
|
|
{
|
|
|
|
nsresult rv = nsSplittableFrame::Init(aContent, aParent, aPrevInFlow);
|
|
|
|
if (aPrevInFlow) {
|
|
|
|
// Make sure we copy bits from our prev-in-flow that will affect
|
|
|
|
// us. A continuation for a container frame needs to know if it
|
|
|
|
// has a child with a view so that we'll properly reposition it.
|
|
|
|
if (aPrevInFlow->GetStateBits() & NS_FRAME_HAS_CHILD_WITH_VIEW)
|
|
|
|
AddStateBits(NS_FRAME_HAS_CHILD_WITH_VIEW);
|
|
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2009-07-28 05:53:20 -07:00
|
|
|
nsContainerFrame::SetInitialChildList(nsIAtom* aListName,
|
|
|
|
nsFrameList& aChildList)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
nsresult result;
|
2009-09-18 04:09:36 -07:00
|
|
|
if (mFrames.NotEmpty()) {
|
2007-03-22 10:30:00 -07:00
|
|
|
// We already have child frames which means we've already been
|
|
|
|
// initialized
|
|
|
|
NS_NOTREACHED("unexpected second call to SetInitialChildList");
|
|
|
|
result = NS_ERROR_UNEXPECTED;
|
|
|
|
} else if (aListName) {
|
|
|
|
// All we know about is the unnamed principal child list
|
|
|
|
NS_NOTREACHED("unknown frame list");
|
|
|
|
result = NS_ERROR_INVALID_ARG;
|
|
|
|
} else {
|
|
|
|
#ifdef NS_DEBUG
|
|
|
|
nsFrame::VerifyDirtyBitSet(aChildList);
|
|
|
|
#endif
|
|
|
|
mFrames.SetFrames(aChildList);
|
|
|
|
result = NS_OK;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsContainerFrame::AppendFrames(nsIAtom* aListName,
|
2009-07-30 10:23:32 -07:00
|
|
|
nsFrameList& aFrameList)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
if (nsnull != aListName) {
|
|
|
|
#ifdef IBMBIDI
|
|
|
|
if (aListName != nsGkAtoms::nextBidi)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
NS_ERROR("unexpected child list");
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
}
|
2009-07-30 10:23:32 -07:00
|
|
|
if (aFrameList.NotEmpty()) {
|
2007-03-22 10:30:00 -07:00
|
|
|
mFrames.AppendFrames(this, aFrameList);
|
|
|
|
|
|
|
|
// Ask the parent frame to reflow me.
|
|
|
|
#ifdef IBMBIDI
|
|
|
|
if (nsnull == aListName)
|
|
|
|
#endif
|
|
|
|
{
|
2007-03-30 14:11:41 -07:00
|
|
|
PresContext()->PresShell()->
|
2009-04-10 13:42:29 -07:00
|
|
|
FrameNeedsReflow(this, nsIPresShell::eTreeChange,
|
|
|
|
NS_FRAME_HAS_DIRTY_CHILDREN);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsContainerFrame::InsertFrames(nsIAtom* aListName,
|
|
|
|
nsIFrame* aPrevFrame,
|
2009-07-30 10:23:32 -07:00
|
|
|
nsFrameList& aFrameList)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
NS_ASSERTION(!aPrevFrame || aPrevFrame->GetParent() == this,
|
|
|
|
"inserting after sibling frame with different parent");
|
|
|
|
|
|
|
|
if (nsnull != aListName) {
|
|
|
|
#ifdef IBMBIDI
|
|
|
|
if (aListName != nsGkAtoms::nextBidi)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
NS_ERROR("unexpected child list");
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
}
|
2009-07-30 10:23:32 -07:00
|
|
|
if (aFrameList.NotEmpty()) {
|
2007-03-22 10:30:00 -07:00
|
|
|
// Insert frames after aPrevFrame
|
|
|
|
mFrames.InsertFrames(this, aPrevFrame, aFrameList);
|
|
|
|
|
|
|
|
#ifdef IBMBIDI
|
|
|
|
if (nsnull == aListName)
|
|
|
|
#endif
|
|
|
|
{
|
2007-03-30 14:11:41 -07:00
|
|
|
PresContext()->PresShell()->
|
2009-04-10 13:42:29 -07:00
|
|
|
FrameNeedsReflow(this, nsIPresShell::eTreeChange,
|
|
|
|
NS_FRAME_HAS_DIRTY_CHILDREN);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsContainerFrame::RemoveFrame(nsIAtom* aListName,
|
|
|
|
nsIFrame* aOldFrame)
|
|
|
|
{
|
|
|
|
if (nsnull != aListName) {
|
|
|
|
#ifdef IBMBIDI
|
|
|
|
if (nsGkAtoms::nextBidi != aListName)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
NS_ERROR("unexpected child list");
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aOldFrame) {
|
|
|
|
// Loop and destroy the frame and all of its continuations.
|
|
|
|
// If the frame we are removing is a brFrame, we need a reflow so
|
|
|
|
// the line the brFrame was on can attempt to pull up any frames
|
|
|
|
// that can fit from lines below it.
|
2007-12-10 17:14:25 -08:00
|
|
|
PRBool generateReflowCommand = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
#ifdef IBMBIDI
|
2007-12-10 17:14:25 -08:00
|
|
|
if (nsGkAtoms::nextBidi == aListName) {
|
|
|
|
generateReflowCommand = PR_FALSE;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
#endif
|
2007-12-10 17:14:25 -08:00
|
|
|
nsContainerFrame* parent = static_cast<nsContainerFrame*>(aOldFrame->GetParent());
|
|
|
|
while (aOldFrame) {
|
2007-03-22 10:30:00 -07:00
|
|
|
// 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* oldFrameNextContinuation = aOldFrame->GetNextContinuation();
|
2007-07-25 21:03:29 -07:00
|
|
|
//XXXfr probably should use StealFrame here. I'm not sure if we need to
|
|
|
|
// check the overflow lists atm, but we'll need a prescontext lookup
|
|
|
|
// for overflow containers once we can split abspos elements with
|
|
|
|
// inline containing blocks.
|
2007-10-02 20:44:18 -07:00
|
|
|
if (parent == this) {
|
2009-09-18 04:09:36 -07:00
|
|
|
if (!parent->mFrames.DestroyFrameIfPresent(aOldFrame)) {
|
2008-01-14 17:12:01 -08:00
|
|
|
// Try to remove it from our overflow list, if we have one.
|
|
|
|
// The simplest way is to reuse StealFrame.
|
2009-09-18 04:09:36 -07:00
|
|
|
StealFrame(PresContext(), aOldFrame, PR_TRUE);
|
2008-01-14 17:12:01 -08:00
|
|
|
aOldFrame->Destroy();
|
|
|
|
}
|
2007-10-02 20:44:18 -07:00
|
|
|
} else {
|
|
|
|
// This recursive call takes care of all continuations after aOldFrame,
|
|
|
|
// so we don't need to loop anymore.
|
|
|
|
parent->RemoveFrame(nsnull, aOldFrame);
|
|
|
|
break;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
aOldFrame = oldFrameNextContinuation;
|
|
|
|
if (aOldFrame) {
|
2007-07-08 00:08:04 -07:00
|
|
|
parent = static_cast<nsContainerFrame*>(aOldFrame->GetParent());
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (generateReflowCommand) {
|
2007-03-30 14:11:41 -07:00
|
|
|
PresContext()->PresShell()->
|
2009-04-10 13:42:29 -07:00
|
|
|
FrameNeedsReflow(this, nsIPresShell::eTreeChange,
|
|
|
|
NS_FRAME_HAS_DIRTY_CHILDREN);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsContainerFrame::Destroy()
|
|
|
|
{
|
|
|
|
// Prevent event dispatch during destruction
|
|
|
|
if (HasView()) {
|
|
|
|
GetView()->SetClientData(nsnull);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete the primary child list
|
|
|
|
mFrames.DestroyFrames();
|
2007-07-25 21:03:29 -07:00
|
|
|
|
|
|
|
// Destroy auxiliary frame lists
|
|
|
|
nsPresContext* prescontext = PresContext();
|
|
|
|
|
2009-07-28 05:51:09 -07:00
|
|
|
DestroyOverflowList(prescontext);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2007-07-25 21:03:29 -07:00
|
|
|
if (IsFrameOfType(nsIFrame::eCanContainOverflowContainers)) {
|
|
|
|
nsFrameList* frameList = RemovePropTableFrames(prescontext,
|
|
|
|
nsGkAtoms::overflowContainersProperty);
|
|
|
|
if (frameList)
|
|
|
|
frameList->Destroy();
|
|
|
|
|
|
|
|
frameList = RemovePropTableFrames(prescontext,
|
|
|
|
nsGkAtoms::excessOverflowContainersProperty);
|
|
|
|
if (frameList)
|
|
|
|
frameList->Destroy();
|
|
|
|
}
|
|
|
|
|
2007-07-23 16:31:11 -07:00
|
|
|
if (IsGeneratedContentFrame()) {
|
|
|
|
// Make sure all the content nodes for the generated content inside
|
|
|
|
// this frame know it's going away.
|
|
|
|
// See also nsCSSFrameConstructor::CreateGeneratedContentFrame which
|
|
|
|
// created this frame.
|
|
|
|
nsCOMArray<nsIContent>* generatedContent =
|
|
|
|
static_cast<nsCOMArray<nsIContent>*>(
|
|
|
|
UnsetProperty(nsGkAtoms::generatedContent));
|
|
|
|
|
|
|
|
if (generatedContent) {
|
|
|
|
for (int i = generatedContent->Count() - 1; i >= 0; --i) {
|
|
|
|
nsIContent* content = generatedContent->ObjectAt(i);
|
|
|
|
// Tell the ESM that this content is going away now, so it'll update
|
|
|
|
// its hover content, etc.
|
Bug 178324, refactor focus by moving all focus handling into one place and simplifying it, add many tests, fixes many other bugs too numerous to mention in this small checkin comment, r=josh,smichaud,ere,dbaron,marco,neil,gavin,smaug,sr=smaug (CLOSED TREE)
2009-06-10 11:00:39 -07:00
|
|
|
PresContext()->EventStateManager()->
|
|
|
|
ContentRemoved(content->GetCurrentDoc(), content);
|
2007-07-23 16:31:11 -07:00
|
|
|
content->UnbindFromTree();
|
|
|
|
}
|
|
|
|
delete generatedContent;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
// Destroy the frame and remove the flow pointers
|
|
|
|
nsSplittableFrame::Destroy();
|
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Child frame enumeration
|
|
|
|
|
2009-07-28 05:51:09 -07:00
|
|
|
nsFrameList
|
|
|
|
nsContainerFrame::GetChildList(nsIAtom* aListName) const
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
// We only know about the unnamed principal child list and the overflow
|
2009-07-28 05:51:09 -07:00
|
|
|
// lists
|
2007-03-22 10:30:00 -07:00
|
|
|
if (nsnull == aListName) {
|
2009-07-28 05:51:09 -07:00
|
|
|
return mFrames;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nsGkAtoms::overflowList == aListName) {
|
2009-07-28 05:51:09 -07:00
|
|
|
nsFrameList* frameList = GetOverflowFrames();
|
2009-07-28 05:51:09 -07:00
|
|
|
return frameList ? *frameList : nsFrameList::EmptyList();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nsGkAtoms::overflowContainersList == aListName) {
|
2007-07-25 21:03:29 -07:00
|
|
|
nsFrameList* list = GetPropTableFrames(PresContext(),
|
|
|
|
nsGkAtoms::overflowContainersProperty);
|
2009-07-28 05:51:09 -07:00
|
|
|
return list ? *list : nsFrameList::EmptyList();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nsGkAtoms::excessOverflowContainersList == aListName) {
|
2007-07-25 21:03:29 -07:00
|
|
|
nsFrameList* list = GetPropTableFrames(PresContext(),
|
|
|
|
nsGkAtoms::excessOverflowContainersProperty);
|
2009-07-28 05:51:09 -07:00
|
|
|
return list ? *list : nsFrameList::EmptyList();
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2009-07-28 05:51:09 -07:00
|
|
|
|
|
|
|
return nsFrameList::EmptyList();
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2007-07-25 21:03:29 -07:00
|
|
|
#define NS_CONTAINER_FRAME_OVERFLOW_LIST_INDEX 0
|
|
|
|
#define NS_CONTAINER_FRAME_OVERFLOW_CONTAINERS_LIST_INDEX 1
|
|
|
|
#define NS_CONTAINER_FRAME_EXCESS_OVERFLOW_CONTAINERS_LIST_INDEX 2
|
|
|
|
// If adding/removing lists, don't forget to update count in .h file
|
|
|
|
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIAtom*
|
|
|
|
nsContainerFrame::GetAdditionalChildListName(PRInt32 aIndex) const
|
|
|
|
{
|
2007-07-25 21:03:29 -07:00
|
|
|
if (NS_CONTAINER_FRAME_OVERFLOW_LIST_INDEX == aIndex)
|
2007-03-22 10:30:00 -07:00
|
|
|
return nsGkAtoms::overflowList;
|
2007-07-25 21:03:29 -07:00
|
|
|
else if (IsFrameOfType(nsIFrame::eCanContainOverflowContainers)) {
|
|
|
|
if (NS_CONTAINER_FRAME_OVERFLOW_CONTAINERS_LIST_INDEX == aIndex)
|
|
|
|
return nsGkAtoms::overflowContainersList;
|
|
|
|
else if (NS_CONTAINER_FRAME_EXCESS_OVERFLOW_CONTAINERS_LIST_INDEX == aIndex)
|
|
|
|
return nsGkAtoms::excessOverflowContainersList;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2007-07-25 21:03:29 -07:00
|
|
|
return nsnull;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Painting/Events
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsContainerFrame::BuildDisplayList(nsDisplayListBuilder* aBuilder,
|
|
|
|
const nsRect& aDirtyRect,
|
|
|
|
const nsDisplayListSet& aLists)
|
|
|
|
{
|
|
|
|
nsresult rv = DisplayBorderBackgroundOutline(aBuilder, aLists);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
return BuildDisplayListForNonBlockChildren(aBuilder, aDirtyRect, aLists);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsContainerFrame::BuildDisplayListForNonBlockChildren(nsDisplayListBuilder* aBuilder,
|
|
|
|
const nsRect& aDirtyRect,
|
|
|
|
const nsDisplayListSet& aLists,
|
|
|
|
PRUint32 aFlags)
|
|
|
|
{
|
|
|
|
nsIFrame* kid = mFrames.FirstChild();
|
|
|
|
// Put each child's background directly onto the content list
|
|
|
|
nsDisplayListSet set(aLists, aLists.Content());
|
|
|
|
// The children should be in content order
|
|
|
|
while (kid) {
|
|
|
|
nsresult rv = BuildDisplayListForChild(aBuilder, kid, aDirtyRect, set, aFlags);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
kid = kid->GetNextSibling();
|
|
|
|
}
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* virtual */ void
|
|
|
|
nsContainerFrame::ChildIsDirty(nsIFrame* aChild)
|
|
|
|
{
|
|
|
|
AddStateBits(NS_FRAME_HAS_DIRTY_CHILDREN);
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsContainerFrame::IsLeaf() const
|
|
|
|
{
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsContainerFrame::PeekOffsetNoAmount(PRBool aForward, PRInt32* aOffset)
|
|
|
|
{
|
|
|
|
NS_ASSERTION (aOffset && *aOffset <= 1, "aOffset out of range");
|
|
|
|
// Don't allow the caret to stay in an empty (leaf) container frame.
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsContainerFrame::PeekOffsetCharacter(PRBool aForward, PRInt32* aOffset)
|
|
|
|
{
|
|
|
|
NS_ASSERTION (aOffset && *aOffset <= 1, "aOffset out of range");
|
|
|
|
// Don't allow the caret to stay in an empty (leaf) container frame.
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Helper member functions
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Position the view associated with |aKidFrame|, if there is one. A
|
|
|
|
* container frame should call this method after positioning a frame,
|
|
|
|
* but before |Reflow|.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
nsContainerFrame::PositionFrameView(nsIFrame* aKidFrame)
|
|
|
|
{
|
|
|
|
nsIFrame* parentFrame = aKidFrame->GetParent();
|
|
|
|
if (!aKidFrame->HasView() || !parentFrame)
|
|
|
|
return;
|
|
|
|
|
|
|
|
nsIView* view = aKidFrame->GetView();
|
|
|
|
nsIViewManager* vm = view->GetViewManager();
|
|
|
|
nsPoint pt;
|
|
|
|
nsIView* ancestorView = parentFrame->GetClosestView(&pt);
|
|
|
|
|
|
|
|
if (ancestorView != view->GetParent()) {
|
|
|
|
NS_ASSERTION(ancestorView == view->GetParent()->GetParent(),
|
|
|
|
"Allowed only one anonymous view between frames");
|
|
|
|
// parentFrame is responsible for positioning aKidFrame's view
|
|
|
|
// explicitly
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
pt += aKidFrame->GetPosition();
|
|
|
|
vm->MoveViewTo(view, pt.x, pt.y);
|
|
|
|
}
|
|
|
|
|
2009-08-12 13:48:30 -07:00
|
|
|
static nsIWidget*
|
|
|
|
GetPresContextContainerWidget(nsPresContext* aPresContext)
|
2008-01-09 14:50:29 -08:00
|
|
|
{
|
|
|
|
nsCOMPtr<nsISupports> container = aPresContext->Document()->GetContainer();
|
|
|
|
nsCOMPtr<nsIBaseWindow> baseWindow = do_QueryInterface(container);
|
|
|
|
if (!baseWindow)
|
2009-08-12 13:48:30 -07:00
|
|
|
return nsnull;
|
2008-01-09 14:50:29 -08:00
|
|
|
|
|
|
|
nsCOMPtr<nsIWidget> mainWidget;
|
|
|
|
baseWindow->GetMainWidget(getter_AddRefs(mainWidget));
|
2009-08-12 13:48:30 -07:00
|
|
|
return mainWidget;
|
|
|
|
}
|
2008-01-09 14:50:29 -08:00
|
|
|
|
2009-08-12 13:48:30 -07:00
|
|
|
static PRBool
|
|
|
|
IsTopLevelWidget(nsIWidget* aWidget)
|
|
|
|
{
|
2008-01-09 14:50:29 -08:00
|
|
|
nsWindowType windowType;
|
2009-08-12 13:48:30 -07:00
|
|
|
aWidget->GetWindowType(windowType);
|
2008-01-09 14:50:29 -08:00
|
|
|
return windowType == eWindowType_toplevel ||
|
2009-08-12 13:48:30 -07:00
|
|
|
windowType == eWindowType_dialog ||
|
|
|
|
windowType == eWindowType_sheet;
|
2008-01-09 14:50:29 -08:00
|
|
|
// popups aren't toplevel so they're not handled here
|
|
|
|
}
|
|
|
|
|
2009-08-12 13:48:30 -07:00
|
|
|
void
|
|
|
|
nsContainerFrame::SyncWindowProperties(nsPresContext* aPresContext,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIView* aView)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2008-12-29 00:17:08 -08:00
|
|
|
#ifdef MOZ_XUL
|
2009-08-12 13:48:30 -07:00
|
|
|
if (!aView || !nsCSSRendering::IsCanvasFrame(aFrame) || !aView->HasWidget())
|
2009-02-15 17:11:34 -08:00
|
|
|
return;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2009-08-12 13:48:30 -07:00
|
|
|
nsIWidget* windowWidget = GetPresContextContainerWidget(aPresContext);
|
|
|
|
if (!windowWidget || !IsTopLevelWidget(windowWidget))
|
2008-12-29 00:17:08 -08:00
|
|
|
return;
|
|
|
|
|
2009-02-15 17:11:34 -08:00
|
|
|
nsIViewManager* vm = aView->GetViewManager();
|
2008-12-29 00:17:08 -08:00
|
|
|
nsIView* rootView;
|
|
|
|
vm->GetRootView(rootView);
|
|
|
|
|
2009-02-15 17:11:34 -08:00
|
|
|
if (aView != rootView)
|
2008-12-29 00:17:08 -08:00
|
|
|
return;
|
|
|
|
|
|
|
|
nsIContent* rootContent = aPresContext->Document()->GetRootContent();
|
2009-08-24 13:02:07 -07:00
|
|
|
if (!rootContent || !rootContent->IsXUL()) {
|
2008-12-29 00:17:08 -08:00
|
|
|
// Scrollframes use native widgets which don't work well with
|
|
|
|
// translucent windows, at least in Windows XP. So if the document
|
|
|
|
// has a root scrollrame it's useless to try to make it transparent,
|
|
|
|
// we'll just get something broken.
|
|
|
|
// nsCSSFrameConstructor::ConstructRootFrame constructs root
|
|
|
|
// scrollframes whenever the root element is not a XUL element, so
|
|
|
|
// we test for that here. We can't just call
|
|
|
|
// presShell->GetRootScrollFrame() since that might not have
|
|
|
|
// been constructed yet.
|
|
|
|
// We can change this to allow translucent toplevel HTML documents
|
|
|
|
// (e.g. to do something like Dashboard widgets), once we
|
|
|
|
// have broad support for translucent scrolled documents, but be
|
|
|
|
// careful because apparently some Firefox extensions expect
|
|
|
|
// openDialog("something.html") to produce an opaque window
|
|
|
|
// even if the HTML doesn't have a background-color set.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-08-12 13:48:30 -07:00
|
|
|
nsIFrame *rootFrame = aPresContext->PresShell()->FrameConstructor()->GetRootElementStyleFrame();
|
|
|
|
if (!rootFrame)
|
|
|
|
return;
|
|
|
|
|
2009-08-13 15:08:13 -07:00
|
|
|
nsTransparencyMode mode = nsLayoutUtils::GetFrameTransparency(aFrame, rootFrame);
|
2009-08-12 13:48:30 -07:00
|
|
|
nsIWidget* viewWidget = aView->GetWidget();
|
|
|
|
viewWidget->SetTransparencyMode(mode);
|
|
|
|
windowWidget->SetWindowShadowStyle(rootFrame->GetStyleUIReset()->mWindowShadow);
|
2008-12-29 00:17:08 -08:00
|
|
|
#endif
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsContainerFrame::SyncFrameViewAfterReflow(nsPresContext* aPresContext,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIView* aView,
|
|
|
|
const nsRect* aCombinedArea,
|
|
|
|
PRUint32 aFlags)
|
|
|
|
{
|
|
|
|
if (!aView) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-03-25 18:12:49 -07:00
|
|
|
NS_ASSERTION(aCombinedArea, "Combined area must be passed in now");
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
// Make sure the view is sized and positioned correctly
|
|
|
|
if (0 == (aFlags & NS_FRAME_NO_MOVE_VIEW)) {
|
|
|
|
PositionFrameView(aFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (0 == (aFlags & NS_FRAME_NO_SIZE_VIEW)) {
|
|
|
|
nsIViewManager* vm = aView->GetViewManager();
|
|
|
|
|
2007-03-25 18:12:49 -07:00
|
|
|
vm->ResizeView(aView, *aCombinedArea, PR_TRUE);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsContainerFrame::SyncFrameViewProperties(nsPresContext* aPresContext,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsStyleContext* aStyleContext,
|
|
|
|
nsIView* aView,
|
|
|
|
PRUint32 aFlags)
|
|
|
|
{
|
|
|
|
NS_ASSERTION(!aStyleContext || aFrame->GetStyleContext() == aStyleContext,
|
|
|
|
"Wrong style context for frame?");
|
|
|
|
|
|
|
|
if (!aView) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIViewManager* vm = aView->GetViewManager();
|
2008-09-13 02:42:11 -07:00
|
|
|
|
|
|
|
/* If this frame has a -moz-transform property, tell it to invalidate on a scroll
|
|
|
|
* rather than doing a BitBlt.
|
|
|
|
*/
|
|
|
|
if (aFrame->GetStyleDisplay()->HasTransform())
|
|
|
|
aView->SetInvalidateFrameOnScroll();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (nsnull == aStyleContext) {
|
|
|
|
aStyleContext = aFrame->GetStyleContext();
|
|
|
|
}
|
|
|
|
|
2009-08-25 00:44:42 -07:00
|
|
|
// Make sure visibility is correct. This only affects nsSubdocumentFrame.
|
|
|
|
if (0 == (aFlags & NS_FRAME_NO_VISIBILITY) &&
|
|
|
|
!aFrame->SupportsVisibilityHidden()) {
|
2007-03-22 10:30:00 -07:00
|
|
|
// See if the view should be hidden or visible
|
2009-08-25 00:44:42 -07:00
|
|
|
vm->SetViewVisibility(aView,
|
|
|
|
aStyleContext->GetStyleVisibility()->IsVisible()
|
|
|
|
? nsViewVisibility_kShow : nsViewVisibility_kHide);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// See if the frame is being relatively positioned or absolutely
|
|
|
|
// positioned
|
|
|
|
PRBool isPositioned = aStyleContext->GetStyleDisplay()->IsPositioned();
|
|
|
|
|
|
|
|
PRInt32 zIndex = 0;
|
|
|
|
PRBool autoZIndex = PR_FALSE;
|
|
|
|
|
|
|
|
if (!isPositioned) {
|
|
|
|
autoZIndex = PR_TRUE;
|
|
|
|
} else {
|
|
|
|
// Make sure z-index is correct
|
|
|
|
const nsStylePosition* position = aStyleContext->GetStylePosition();
|
|
|
|
|
|
|
|
if (position->mZIndex.GetUnit() == eStyleUnit_Integer) {
|
|
|
|
zIndex = position->mZIndex.GetIntValue();
|
|
|
|
} else if (position->mZIndex.GetUnit() == eStyleUnit_Auto) {
|
|
|
|
autoZIndex = PR_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vm->SetViewZIndex(aView, autoZIndex, zIndex, isPositioned);
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsContainerFrame::FrameNeedsView(nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
// XXX Check needed because frame construction can't properly figure out when
|
|
|
|
// a frame is the child of a scrollframe
|
2009-10-29 14:17:56 -07:00
|
|
|
if (aFrame->GetStyleContext()->GetPseudo() ==
|
2007-03-22 10:30:00 -07:00
|
|
|
nsCSSAnonBoxes::scrolledContent) {
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
2008-09-13 02:42:11 -07:00
|
|
|
return aFrame->NeedsView() || aFrame->GetStyleDisplay()->HasTransform();
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static nscoord GetCoord(const nsStyleCoord& aCoord, nscoord aIfNotCoord)
|
|
|
|
{
|
|
|
|
return aCoord.GetUnit() == eStyleUnit_Coord
|
|
|
|
? aCoord.GetCoordValue()
|
|
|
|
: aIfNotCoord;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsContainerFrame::DoInlineIntrinsicWidth(nsIRenderingContext *aRenderingContext,
|
|
|
|
InlineIntrinsicWidthData *aData,
|
|
|
|
nsLayoutUtils::IntrinsicWidthType aType)
|
|
|
|
{
|
|
|
|
if (GetPrevInFlow())
|
|
|
|
return; // Already added.
|
|
|
|
|
|
|
|
NS_PRECONDITION(aType == nsLayoutUtils::MIN_WIDTH ||
|
|
|
|
aType == nsLayoutUtils::PREF_WIDTH, "bad type");
|
|
|
|
|
|
|
|
PRUint8 startSide, endSide;
|
|
|
|
if (GetStyleVisibility()->mDirection == NS_STYLE_DIRECTION_LTR) {
|
|
|
|
startSide = NS_SIDE_LEFT;
|
|
|
|
endSide = NS_SIDE_RIGHT;
|
|
|
|
} else {
|
|
|
|
startSide = NS_SIDE_RIGHT;
|
|
|
|
endSide = NS_SIDE_LEFT;
|
|
|
|
}
|
|
|
|
|
|
|
|
const nsStylePadding *stylePadding = GetStylePadding();
|
|
|
|
const nsStyleBorder *styleBorder = GetStyleBorder();
|
|
|
|
const nsStyleMargin *styleMargin = GetStyleMargin();
|
|
|
|
|
|
|
|
// This goes at the beginning no matter how things are broken and how
|
|
|
|
// messy the bidi situations are, since per CSS2.1 section 8.6
|
|
|
|
// (implemented in bug 328168), the startSide border is always on the
|
|
|
|
// first line.
|
2008-03-11 14:54:34 -07:00
|
|
|
// This frame is a first-in-flow, but it might have a previous bidi
|
|
|
|
// continuation, in which case that continuation should handle the startSide
|
|
|
|
// border.
|
|
|
|
if (!GetPrevContinuation()) {
|
|
|
|
aData->currentLine +=
|
|
|
|
GetCoord(stylePadding->mPadding.Get(startSide), 0) +
|
2008-07-16 23:30:25 -07:00
|
|
|
styleBorder->GetActualBorderWidth(startSide) +
|
2008-03-11 14:54:34 -07:00
|
|
|
GetCoord(styleMargin->mMargin.Get(startSide), 0);
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2008-01-09 19:18:44 -08:00
|
|
|
const nsLineList_iterator* savedLine = aData->line;
|
2009-02-19 12:18:52 -08:00
|
|
|
nsIFrame* const savedLineContainer = aData->lineContainer;
|
2008-01-09 19:18:44 -08:00
|
|
|
|
2008-03-11 14:54:34 -07:00
|
|
|
nsContainerFrame *lastInFlow;
|
2008-01-09 19:23:35 -08:00
|
|
|
for (nsContainerFrame *nif = this; nif;
|
2009-02-19 12:18:52 -08:00
|
|
|
nif = static_cast<nsContainerFrame*>(nif->GetNextInFlow())) {
|
2007-03-22 10:30:00 -07:00
|
|
|
for (nsIFrame *kid = nif->mFrames.FirstChild(); kid;
|
|
|
|
kid = kid->GetNextSibling()) {
|
|
|
|
if (aType == nsLayoutUtils::MIN_WIDTH)
|
|
|
|
kid->AddInlineMinWidth(aRenderingContext,
|
2007-07-08 00:08:04 -07:00
|
|
|
static_cast<InlineMinWidthData*>(aData));
|
2007-03-22 10:30:00 -07:00
|
|
|
else
|
|
|
|
kid->AddInlinePrefWidth(aRenderingContext,
|
2007-07-08 00:08:04 -07:00
|
|
|
static_cast<InlinePrefWidthData*>(aData));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2008-01-09 19:18:44 -08:00
|
|
|
|
2009-02-19 12:18:52 -08:00
|
|
|
// After we advance to our next-in-flow, the stored line and line container
|
|
|
|
// may no longer be correct. Just forget them.
|
2008-01-09 19:23:35 -08:00
|
|
|
aData->line = nsnull;
|
2009-02-19 12:18:52 -08:00
|
|
|
aData->lineContainer = nsnull;
|
|
|
|
|
2008-03-11 14:54:34 -07:00
|
|
|
lastInFlow = nif;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2008-01-09 19:18:44 -08:00
|
|
|
|
|
|
|
aData->line = savedLine;
|
2009-02-19 12:18:52 -08:00
|
|
|
aData->lineContainer = savedLineContainer;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// This goes at the end no matter how things are broken and how
|
|
|
|
// messy the bidi situations are, since per CSS2.1 section 8.6
|
|
|
|
// (implemented in bug 328168), the endSide border is always on the
|
|
|
|
// last line.
|
2008-03-11 14:54:34 -07:00
|
|
|
// We reached the last-in-flow, but it might have a next bidi
|
|
|
|
// continuation, in which case that continuation should handle
|
|
|
|
// the endSide border.
|
|
|
|
if (!lastInFlow->GetNextContinuation()) {
|
|
|
|
aData->currentLine +=
|
|
|
|
GetCoord(stylePadding->mPadding.Get(endSide), 0) +
|
2008-07-16 23:30:25 -07:00
|
|
|
styleBorder->GetActualBorderWidth(endSide) +
|
2008-03-11 14:54:34 -07:00
|
|
|
GetCoord(styleMargin->mMargin.Get(endSide), 0);
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* virtual */ nsSize
|
|
|
|
nsContainerFrame::ComputeAutoSize(nsIRenderingContext *aRenderingContext,
|
|
|
|
nsSize aCBSize, nscoord aAvailableWidth,
|
|
|
|
nsSize aMargin, nsSize aBorder,
|
|
|
|
nsSize aPadding, PRBool aShrinkWrap)
|
|
|
|
{
|
|
|
|
nsSize result(0xdeadbeef, NS_UNCONSTRAINEDSIZE);
|
|
|
|
nscoord availBased = aAvailableWidth - aMargin.width - aBorder.width -
|
|
|
|
aPadding.width;
|
|
|
|
// replaced elements always shrink-wrap
|
|
|
|
if (aShrinkWrap || IsFrameOfType(eReplaced)) {
|
|
|
|
// don't bother setting it if the result won't be used
|
|
|
|
if (GetStylePosition()->mWidth.GetUnit() == eStyleUnit_Auto) {
|
|
|
|
result.width = ShrinkWidthToFit(aRenderingContext, availBased);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
result.width = availBased;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Invokes the WillReflow() function, positions the frame and its view (if
|
|
|
|
* requested), and then calls Reflow(). If the reflow succeeds and the child
|
|
|
|
* frame is complete, deletes any next-in-flows using DeleteNextInFlowChild()
|
|
|
|
*/
|
|
|
|
nsresult
|
|
|
|
nsContainerFrame::ReflowChild(nsIFrame* aKidFrame,
|
|
|
|
nsPresContext* aPresContext,
|
|
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
|
|
const nsHTMLReflowState& aReflowState,
|
|
|
|
nscoord aX,
|
|
|
|
nscoord aY,
|
|
|
|
PRUint32 aFlags,
|
2007-07-25 21:03:29 -07:00
|
|
|
nsReflowStatus& aStatus,
|
|
|
|
nsOverflowContinuationTracker* aTracker)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
NS_PRECONDITION(aReflowState.frame == aKidFrame, "bad reflow state");
|
|
|
|
|
|
|
|
nsresult result;
|
|
|
|
|
|
|
|
// Send the WillReflow() notification, and position the child frame
|
|
|
|
// and its view if requested
|
|
|
|
aKidFrame->WillReflow(aPresContext);
|
|
|
|
|
2008-09-06 01:52:56 -07:00
|
|
|
if (NS_FRAME_NO_MOVE_FRAME != (aFlags & NS_FRAME_NO_MOVE_FRAME)) {
|
2008-02-08 01:36:32 -08:00
|
|
|
if ((aFlags & NS_FRAME_INVALIDATE_ON_MOVE) &&
|
|
|
|
!(aKidFrame->GetStateBits() & NS_FRAME_FIRST_REFLOW) &&
|
|
|
|
aKidFrame->GetPosition() != nsPoint(aX, aY)) {
|
|
|
|
aKidFrame->InvalidateOverflowRect();
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
aKidFrame->SetPosition(nsPoint(aX, aY));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (0 == (aFlags & NS_FRAME_NO_MOVE_VIEW)) {
|
|
|
|
PositionFrameView(aKidFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reflow the child frame
|
|
|
|
result = aKidFrame->Reflow(aPresContext, aDesiredSize, aReflowState,
|
|
|
|
aStatus);
|
|
|
|
|
|
|
|
// If the reflow was successful and the child frame is complete, delete any
|
|
|
|
// next-in-flows
|
2007-07-25 21:03:29 -07:00
|
|
|
if (NS_SUCCEEDED(result) && NS_FRAME_IS_FULLY_COMPLETE(aStatus)) {
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIFrame* kidNextInFlow = aKidFrame->GetNextInFlow();
|
|
|
|
if (nsnull != kidNextInFlow) {
|
|
|
|
// Remove all of the childs next-in-flows. Make sure that we ask
|
|
|
|
// the right parent to do the removal (it's possible that the
|
|
|
|
// parent is not this because we are executing pullup code)
|
2007-10-01 22:57:45 -07:00
|
|
|
if (aTracker) aTracker->Finish(aKidFrame);
|
2007-07-08 00:08:04 -07:00
|
|
|
static_cast<nsContainerFrame*>(kidNextInFlow->GetParent())
|
2008-12-02 17:27:19 -08:00
|
|
|
->DeleteNextInFlowChild(aPresContext, kidNextInFlow, PR_TRUE);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Position the views of |aFrame|'s descendants. A container frame
|
|
|
|
* should call this method if it moves a frame after |Reflow|.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
nsContainerFrame::PositionChildViews(nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
if (!(aFrame->GetStateBits() & NS_FRAME_HAS_CHILD_WITH_VIEW)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIAtom* childListName = nsnull;
|
|
|
|
PRInt32 childListIndex = 0;
|
|
|
|
|
|
|
|
do {
|
|
|
|
// Recursively walk aFrame's child frames
|
|
|
|
nsIFrame* childFrame = aFrame->GetFirstChild(childListName);
|
|
|
|
while (childFrame) {
|
|
|
|
// Position the frame's view (if it has one) otherwise recursively
|
|
|
|
// process its children
|
|
|
|
if (childFrame->HasView()) {
|
|
|
|
PositionFrameView(childFrame);
|
|
|
|
} else {
|
|
|
|
PositionChildViews(childFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the next sibling child frame
|
|
|
|
childFrame = childFrame->GetNextSibling();
|
|
|
|
}
|
|
|
|
|
2007-07-04 08:49:38 -07:00
|
|
|
// also process the additional child lists, but skip the popup list as the
|
|
|
|
// view for popups is managed by the parent. Currently only nsMenuFrame
|
|
|
|
// has a popupList and during layout will call nsMenuPopupFrame::AdjustView.
|
|
|
|
do {
|
|
|
|
childListName = aFrame->GetAdditionalChildListName(childListIndex++);
|
|
|
|
} while (childListName == nsGkAtoms::popupList);
|
2007-03-22 10:30:00 -07:00
|
|
|
} while (childListName);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The second half of frame reflow. Does the following:
|
|
|
|
* - sets the frame's bounds
|
|
|
|
* - sizes and positions (if requested) the frame's view. If the frame's final
|
|
|
|
* position differs from the current position and the frame itself does not
|
|
|
|
* have a view, then any child frames with views are positioned so they stay
|
|
|
|
* in sync
|
|
|
|
* - sets the view's visibility, opacity, content transparency, and clip
|
|
|
|
* - invoked the DidReflow() function
|
|
|
|
*
|
|
|
|
* Flags:
|
|
|
|
* NS_FRAME_NO_MOVE_FRAME - don't move the frame. aX and aY are ignored in this
|
|
|
|
* case. Also implies NS_FRAME_NO_MOVE_VIEW
|
|
|
|
* NS_FRAME_NO_MOVE_VIEW - don't position the frame's view. Set this if you
|
|
|
|
* don't want to automatically sync the frame and view
|
|
|
|
* NS_FRAME_NO_SIZE_VIEW - don't size the frame's view
|
|
|
|
*/
|
|
|
|
nsresult
|
2008-01-02 20:33:21 -08:00
|
|
|
nsContainerFrame::FinishReflowChild(nsIFrame* aKidFrame,
|
|
|
|
nsPresContext* aPresContext,
|
|
|
|
const nsHTMLReflowState* aReflowState,
|
|
|
|
const nsHTMLReflowMetrics& aDesiredSize,
|
|
|
|
nscoord aX,
|
|
|
|
nscoord aY,
|
|
|
|
PRUint32 aFlags)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
nsPoint curOrigin = aKidFrame->GetPosition();
|
|
|
|
nsRect bounds(aX, aY, aDesiredSize.width, aDesiredSize.height);
|
|
|
|
|
|
|
|
aKidFrame->SetRect(bounds);
|
|
|
|
|
|
|
|
if (aKidFrame->HasView()) {
|
|
|
|
nsIView* view = aKidFrame->GetView();
|
|
|
|
// Make sure the frame's view is properly sized and positioned and has
|
|
|
|
// things like opacity correct
|
|
|
|
SyncFrameViewAfterReflow(aPresContext, aKidFrame, view,
|
|
|
|
&aDesiredSize.mOverflowArea,
|
|
|
|
aFlags);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(aFlags & NS_FRAME_NO_MOVE_VIEW) &&
|
|
|
|
(curOrigin.x != aX || curOrigin.y != aY)) {
|
|
|
|
if (!aKidFrame->HasView()) {
|
|
|
|
// If the frame has moved, then we need to make sure any child views are
|
|
|
|
// correctly positioned
|
|
|
|
PositionChildViews(aKidFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// We also need to redraw everything associated with the frame
|
|
|
|
// because if the frame's Reflow issued any invalidates, then they
|
|
|
|
// will be at the wrong offset ... note that this includes
|
|
|
|
// invalidates issued against the frame's children, so we need to
|
|
|
|
// invalidate the overflow area too.
|
|
|
|
aKidFrame->Invalidate(aDesiredSize.mOverflowArea);
|
|
|
|
}
|
|
|
|
|
|
|
|
return aKidFrame->DidReflow(aPresContext, aReflowState, NS_FRAME_REFLOW_FINISHED);
|
|
|
|
}
|
|
|
|
|
2007-07-25 21:03:29 -07:00
|
|
|
nsresult
|
|
|
|
nsContainerFrame::ReflowOverflowContainerChildren(nsPresContext* aPresContext,
|
|
|
|
const nsHTMLReflowState& aReflowState,
|
|
|
|
nsRect& aOverflowRect,
|
|
|
|
PRUint32 aFlags,
|
|
|
|
nsReflowStatus& aStatus)
|
|
|
|
{
|
|
|
|
NS_PRECONDITION(aPresContext, "null pointer");
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
nsFrameList* overflowContainers =
|
|
|
|
GetPropTableFrames(aPresContext,
|
|
|
|
nsGkAtoms::overflowContainersProperty);
|
|
|
|
|
|
|
|
NS_ASSERTION(!(overflowContainers && GetPrevInFlow()
|
|
|
|
&& static_cast<nsContainerFrame*>(GetPrevInFlow())
|
|
|
|
->GetPropTableFrames(aPresContext,
|
|
|
|
nsGkAtoms::excessOverflowContainersProperty)),
|
|
|
|
"conflicting overflow containers lists");
|
|
|
|
|
|
|
|
if (!overflowContainers) {
|
|
|
|
// Drain excess from previnflow
|
|
|
|
nsContainerFrame* prev = (nsContainerFrame*) GetPrevInFlow();
|
|
|
|
if (prev) {
|
|
|
|
nsFrameList* excessFrames =
|
|
|
|
prev->RemovePropTableFrames(aPresContext,
|
|
|
|
nsGkAtoms::excessOverflowContainersProperty);
|
|
|
|
if (excessFrames) {
|
2009-09-18 04:09:36 -07:00
|
|
|
excessFrames->ApplySetParent(this);
|
|
|
|
nsHTMLContainerFrame::ReparentFrameViewList(aPresContext, *excessFrames,
|
2007-07-25 21:03:29 -07:00
|
|
|
prev, this);
|
|
|
|
overflowContainers = excessFrames;
|
|
|
|
rv = SetPropTableFrames(aPresContext, overflowContainers,
|
|
|
|
nsGkAtoms::overflowContainersProperty);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
excessFrames->DestroyFrames();
|
|
|
|
delete excessFrames;
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!overflowContainers)
|
|
|
|
return NS_OK; // nothing to reflow
|
|
|
|
|
2007-10-01 22:57:45 -07:00
|
|
|
nsOverflowContinuationTracker tracker(aPresContext, this, PR_FALSE, PR_FALSE);
|
2007-07-25 21:03:29 -07:00
|
|
|
for (nsIFrame* frame = overflowContainers->FirstChild(); frame;
|
|
|
|
frame = frame->GetNextSibling()) {
|
2009-05-08 18:53:44 -07:00
|
|
|
if (frame->GetPrevInFlow()->GetParent() != GetPrevInFlow()) {
|
|
|
|
// frame's prevInFlow has moved, skip reflowing this frame;
|
|
|
|
// it will get reflowed once it's been placed
|
|
|
|
continue;
|
|
|
|
}
|
2007-07-25 21:03:29 -07:00
|
|
|
if (NS_SUBTREE_DIRTY(frame)) {
|
|
|
|
// Get prev-in-flow
|
|
|
|
nsIFrame* prevInFlow = frame->GetPrevInFlow();
|
|
|
|
NS_ASSERTION(prevInFlow,
|
|
|
|
"overflow container frame must have a prev-in-flow");
|
|
|
|
NS_ASSERTION(frame->GetStateBits() & NS_FRAME_IS_OVERFLOW_CONTAINER,
|
|
|
|
"overflow container frame must have overflow container bit set");
|
|
|
|
nsRect prevRect = prevInFlow->GetRect();
|
|
|
|
|
|
|
|
// Initialize reflow params
|
|
|
|
nsSize availSpace(prevRect.width, aReflowState.availableHeight);
|
|
|
|
nsHTMLReflowMetrics desiredSize;
|
|
|
|
nsHTMLReflowState frameState(aPresContext, aReflowState,
|
|
|
|
frame, availSpace);
|
|
|
|
nsReflowStatus frameStatus = NS_FRAME_COMPLETE;
|
|
|
|
|
2007-10-01 22:57:45 -07:00
|
|
|
// Cache old bounds
|
|
|
|
nsRect oldRect = frame->GetRect();
|
|
|
|
nsRect oldOverflow = frame->GetOverflowRect();
|
|
|
|
|
2007-07-25 21:03:29 -07:00
|
|
|
// Reflow
|
|
|
|
rv = ReflowChild(frame, aPresContext, desiredSize, frameState,
|
2007-10-01 22:57:45 -07:00
|
|
|
prevRect.x, 0, aFlags, frameStatus, &tracker);
|
2007-07-25 21:03:29 -07:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
//XXXfr Do we need to override any shrinkwrap effects here?
|
|
|
|
// e.g. desiredSize.width = prevRect.width;
|
|
|
|
rv = FinishReflowChild(frame, aPresContext, &frameState, desiredSize,
|
|
|
|
prevRect.x, 0, aFlags);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
2007-10-01 22:57:45 -07:00
|
|
|
// Invalidate if there was a position or size change
|
|
|
|
nsRect rect = frame->GetRect();
|
|
|
|
if (rect != oldRect) {
|
|
|
|
nsRect dirtyRect = oldOverflow;
|
|
|
|
dirtyRect.MoveBy(oldRect.x, oldRect.y);
|
|
|
|
Invalidate(dirtyRect);
|
|
|
|
|
|
|
|
dirtyRect = frame->GetOverflowRect();
|
|
|
|
dirtyRect.MoveBy(rect.x, rect.y);
|
|
|
|
Invalidate(dirtyRect);
|
|
|
|
}
|
|
|
|
|
2007-07-25 21:03:29 -07:00
|
|
|
// Handle continuations
|
|
|
|
if (!NS_FRAME_IS_FULLY_COMPLETE(frameStatus)) {
|
2007-10-01 22:57:45 -07:00
|
|
|
if (frame->GetStateBits() & NS_FRAME_OUT_OF_FLOW) {
|
|
|
|
// Abspos frames can't cause their parent to be incomplete,
|
|
|
|
// only overflow incomplete.
|
|
|
|
NS_FRAME_SET_OVERFLOW_INCOMPLETE(frameStatus);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
NS_ASSERTION(NS_FRAME_IS_COMPLETE(frameStatus),
|
|
|
|
"overflow container frames can't be incomplete, only overflow-incomplete");
|
|
|
|
}
|
|
|
|
|
2007-07-25 21:03:29 -07:00
|
|
|
// Acquire a next-in-flow, creating it if necessary
|
|
|
|
nsIFrame* nif = frame->GetNextInFlow();
|
|
|
|
if (!nif) {
|
|
|
|
NS_ASSERTION(frameStatus & NS_FRAME_REFLOW_NEXTINFLOW,
|
|
|
|
"Someone forgot a REFLOW_NEXTINFLOW flag");
|
2009-09-18 04:09:35 -07:00
|
|
|
rv = aPresContext->PresShell()->FrameConstructor()->
|
|
|
|
CreateContinuingFrame(aPresContext, frame, this, &nif);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
2007-07-25 21:03:29 -07:00
|
|
|
}
|
|
|
|
else if (!(nif->GetStateBits() & NS_FRAME_IS_OVERFLOW_CONTAINER)) {
|
|
|
|
// used to be a normal next-in-flow; steal it from the child list
|
|
|
|
rv = static_cast<nsContainerFrame*>(nif->GetParent())
|
|
|
|
->StealFrame(aPresContext, nif);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
}
|
|
|
|
|
|
|
|
tracker.Insert(nif, frameStatus);
|
|
|
|
}
|
2008-05-28 06:39:41 -07:00
|
|
|
NS_MergeReflowStatusInto(&aStatus, frameStatus);
|
2007-07-25 21:03:29 -07:00
|
|
|
// At this point it would be nice to assert !frame->GetOverflowRect().IsEmpty(),
|
|
|
|
// but we have some unsplittable frames that, when taller than
|
|
|
|
// availableHeight will push zero-height content into a next-in-flow.
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
tracker.Skip(frame, aStatus);
|
2009-08-31 11:25:36 -07:00
|
|
|
if (aReflowState.mFloatManager)
|
|
|
|
nsBlockFrame::RecoverFloatsFor(frame, *aReflowState.mFloatManager);
|
2007-07-25 21:03:29 -07:00
|
|
|
}
|
|
|
|
ConsiderChildOverflow(aOverflowRect, frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsContainerFrame::DisplayOverflowContainers(nsDisplayListBuilder* aBuilder,
|
|
|
|
const nsRect& aDirtyRect,
|
|
|
|
const nsDisplayListSet& aLists)
|
|
|
|
{
|
|
|
|
nsFrameList* overflowconts = GetPropTableFrames(PresContext(),
|
|
|
|
nsGkAtoms::overflowContainersProperty);
|
|
|
|
if (overflowconts) {
|
|
|
|
for (nsIFrame* frame = overflowconts->FirstChild(); frame;
|
|
|
|
frame = frame->GetNextSibling()) {
|
|
|
|
BuildDisplayListForChild(aBuilder, frame, aDirtyRect, aLists);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsContainerFrame::StealFrame(nsPresContext* aPresContext,
|
|
|
|
nsIFrame* aChild,
|
|
|
|
PRBool aForceNormal)
|
|
|
|
{
|
|
|
|
PRBool removed = PR_TRUE;
|
|
|
|
if ((aChild->GetStateBits() & NS_FRAME_IS_OVERFLOW_CONTAINER)
|
|
|
|
&& !aForceNormal) {
|
|
|
|
// Try removing from the overflow container list
|
|
|
|
if (!RemovePropTableFrame(aPresContext, aChild,
|
|
|
|
nsGkAtoms::overflowContainersProperty)) {
|
|
|
|
// It must be in the excess overflow container list
|
|
|
|
removed = RemovePropTableFrame(aPresContext, aChild,
|
|
|
|
nsGkAtoms::excessOverflowContainersProperty);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2009-09-18 04:09:36 -07:00
|
|
|
if (!mFrames.RemoveFrameIfPresent(aChild)) {
|
|
|
|
removed = PR_FALSE;
|
2007-07-25 21:03:29 -07:00
|
|
|
// We didn't find the child in the parent's principal child list.
|
|
|
|
// Maybe it's on the overflow list?
|
2009-07-28 05:51:09 -07:00
|
|
|
nsFrameList* frameList = GetOverflowFrames();
|
|
|
|
if (frameList) {
|
2009-09-18 04:09:36 -07:00
|
|
|
removed = frameList->RemoveFrameIfPresent(aChild);
|
2009-07-28 05:51:09 -07:00
|
|
|
if (frameList->IsEmpty()) {
|
|
|
|
DestroyOverflowList(aPresContext);
|
|
|
|
}
|
2007-07-25 21:03:29 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-09-18 04:09:36 -07:00
|
|
|
|
|
|
|
NS_POSTCONDITION(removed, "StealFrame: can't find aChild");
|
|
|
|
return removed ? NS_OK : NS_ERROR_UNEXPECTED;
|
2007-07-25 21:03:29 -07:00
|
|
|
}
|
|
|
|
|
2009-09-18 04:09:35 -07:00
|
|
|
nsFrameList
|
|
|
|
nsContainerFrame::StealFramesAfter(nsIFrame* aChild)
|
|
|
|
{
|
|
|
|
NS_ASSERTION(!aChild ||
|
|
|
|
!(aChild->GetStateBits() & NS_FRAME_IS_OVERFLOW_CONTAINER),
|
|
|
|
"StealFramesAfter doesn't handle overflow containers");
|
|
|
|
NS_ASSERTION(GetType() != nsGkAtoms::blockFrame, "unexpected call");
|
|
|
|
|
|
|
|
if (!aChild) {
|
|
|
|
nsFrameList copy(mFrames);
|
|
|
|
mFrames.Clear();
|
|
|
|
return copy;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (nsFrameList::FrameLinkEnumerator iter(mFrames); !iter.AtEnd();
|
|
|
|
iter.Next()) {
|
|
|
|
if (iter.PrevFrame() == aChild) {
|
|
|
|
return mFrames.ExtractTail(iter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We didn't find the child in the principal child list.
|
|
|
|
// Maybe it's on the overflow list?
|
|
|
|
nsFrameList* overflowFrames = GetOverflowFrames();
|
|
|
|
if (overflowFrames) {
|
|
|
|
for (nsFrameList::FrameLinkEnumerator iter(*overflowFrames); !iter.AtEnd();
|
|
|
|
iter.Next()) {
|
|
|
|
if (iter.PrevFrame() == aChild) {
|
|
|
|
return overflowFrames->ExtractTail(iter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_ERROR("StealFramesAfter: can't find aChild");
|
|
|
|
return nsFrameList::EmptyList();
|
|
|
|
}
|
|
|
|
|
2009-07-28 05:51:09 -07:00
|
|
|
void
|
|
|
|
nsContainerFrame::DestroyOverflowList(nsPresContext* aPresContext)
|
|
|
|
{
|
|
|
|
nsFrameList* list =
|
2009-08-17 15:01:06 -07:00
|
|
|
RemovePropTableFrames(aPresContext, nsGkAtoms::overflowProperty);
|
2009-07-28 05:51:09 -07:00
|
|
|
if (list)
|
|
|
|
list->Destroy();
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/**
|
|
|
|
* Remove and delete aNextInFlow and its next-in-flows. Updates the sibling and flow
|
|
|
|
* pointers
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
nsContainerFrame::DeleteNextInFlowChild(nsPresContext* aPresContext,
|
2008-12-02 17:27:19 -08:00
|
|
|
nsIFrame* aNextInFlow,
|
|
|
|
PRBool aDeletingEmptyFrames)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2008-09-20 06:42:03 -07:00
|
|
|
#ifdef DEBUG
|
2008-09-08 07:00:32 -07:00
|
|
|
nsIFrame* prevInFlow = aNextInFlow->GetPrevInFlow();
|
2008-09-20 06:42:03 -07:00
|
|
|
#endif
|
2008-09-08 07:00:32 -07:00
|
|
|
NS_PRECONDITION(prevInFlow, "bad prev-in-flow");
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// If the next-in-flow has a next-in-flow then delete it, too (and
|
|
|
|
// delete it first).
|
|
|
|
// Do this in a loop so we don't overflow the stack for frames
|
|
|
|
// with very many next-in-flows
|
|
|
|
nsIFrame* nextNextInFlow = aNextInFlow->GetNextInFlow();
|
|
|
|
if (nextNextInFlow) {
|
2009-02-03 06:42:18 -08:00
|
|
|
nsAutoTArray<nsIFrame*, 8> frames;
|
2007-03-22 10:30:00 -07:00
|
|
|
for (nsIFrame* f = nextNextInFlow; f; f = f->GetNextInFlow()) {
|
|
|
|
frames.AppendElement(f);
|
|
|
|
}
|
2009-02-03 06:42:18 -08:00
|
|
|
for (PRInt32 i = frames.Length() - 1; i >= 0; --i) {
|
|
|
|
nsIFrame* delFrame = frames.ElementAt(i);
|
2007-07-08 00:08:04 -07:00
|
|
|
static_cast<nsContainerFrame*>(delFrame->GetParent())
|
2008-12-02 17:27:19 -08:00
|
|
|
->DeleteNextInFlowChild(aPresContext, delFrame, aDeletingEmptyFrames);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-10-01 22:57:45 -07:00
|
|
|
aNextInFlow->Invalidate(aNextInFlow->GetOverflowRect());
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
// Take the next-in-flow out of the parent's child list
|
2008-09-20 06:42:03 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
nsresult rv =
|
|
|
|
#endif
|
|
|
|
StealFrame(aPresContext, aNextInFlow);
|
2007-07-25 21:03:29 -07:00
|
|
|
NS_ASSERTION(NS_SUCCEEDED(rv), "StealFrame failure");
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2008-12-07 16:39:11 -08:00
|
|
|
// Delete the next-in-flow frame and its descendants. This will also
|
|
|
|
// remove it from its next-in-flow/prev-in-flow chain.
|
2007-03-22 10:30:00 -07:00
|
|
|
aNextInFlow->Destroy();
|
|
|
|
|
|
|
|
NS_POSTCONDITION(!prevInFlow->GetNextInFlow(), "non null next-in-flow");
|
|
|
|
}
|
|
|
|
|
2009-07-28 05:51:09 -07:00
|
|
|
// Destructor function for the proptable-stored framelists
|
2007-03-22 10:30:00 -07:00
|
|
|
static void
|
2009-07-28 05:51:09 -07:00
|
|
|
DestroyFrameList(void* aFrame,
|
|
|
|
nsIAtom* aPropertyName,
|
|
|
|
void* aPropertyValue,
|
|
|
|
void* aDtorData)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2009-07-28 05:51:09 -07:00
|
|
|
if (aPropertyValue)
|
|
|
|
static_cast<nsFrameList*>(aPropertyValue)->Destroy();
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2007-07-03 15:49:00 -07:00
|
|
|
/**
|
|
|
|
* Set the frames on the overflow list
|
|
|
|
*/
|
2007-03-22 10:30:00 -07:00
|
|
|
nsresult
|
|
|
|
nsContainerFrame::SetOverflowFrames(nsPresContext* aPresContext,
|
2009-07-28 05:51:09 -07:00
|
|
|
const nsFrameList& aOverflowFrames)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2009-07-28 05:51:09 -07:00
|
|
|
NS_PRECONDITION(aOverflowFrames.NotEmpty(), "Shouldn't be called");
|
|
|
|
nsFrameList* newList = new nsFrameList(aOverflowFrames);
|
|
|
|
if (!newList) {
|
|
|
|
// XXXbz should really destroy the frames here, but callers are holding
|
|
|
|
// pointers to them.... We should switch all callers to framelists, then
|
|
|
|
// audit and do that.
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
nsresult rv =
|
|
|
|
aPresContext->PropertyTable()->SetProperty(this,
|
|
|
|
nsGkAtoms::overflowProperty,
|
2009-07-28 05:51:09 -07:00
|
|
|
newList,
|
|
|
|
DestroyFrameList,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsnull);
|
2009-07-28 05:51:09 -07:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
newList->Destroy();
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// Verify that we didn't overwrite an existing overflow list
|
|
|
|
NS_ASSERTION(rv != NS_PROPTABLE_PROP_OVERWRITTEN, "existing overflow list");
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2007-07-25 21:03:29 -07:00
|
|
|
nsFrameList*
|
|
|
|
nsContainerFrame::GetPropTableFrames(nsPresContext* aPresContext,
|
|
|
|
nsIAtom* aPropID) const
|
|
|
|
{
|
|
|
|
nsPropertyTable* propTable = aPresContext->PropertyTable();
|
|
|
|
return static_cast<nsFrameList*>(propTable->GetProperty(this, aPropID));
|
|
|
|
}
|
|
|
|
|
|
|
|
nsFrameList*
|
|
|
|
nsContainerFrame::RemovePropTableFrames(nsPresContext* aPresContext,
|
|
|
|
nsIAtom* aPropID) const
|
|
|
|
{
|
|
|
|
nsPropertyTable* propTable = aPresContext->PropertyTable();
|
|
|
|
return static_cast<nsFrameList*>(propTable->UnsetProperty(this, aPropID));
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsContainerFrame::RemovePropTableFrame(nsPresContext* aPresContext,
|
|
|
|
nsIFrame* aFrame,
|
|
|
|
nsIAtom* aPropID) const
|
|
|
|
{
|
|
|
|
nsFrameList* frameList = RemovePropTableFrames(aPresContext, aPropID);
|
2007-12-18 11:22:12 -08:00
|
|
|
if (!frameList) {
|
|
|
|
// No such list
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
2009-09-18 04:09:36 -07:00
|
|
|
if (!frameList->RemoveFrameIfPresent(aFrame)) {
|
2007-12-18 11:22:12 -08:00
|
|
|
// Found list, but it doesn't have the frame. Put list back.
|
|
|
|
SetPropTableFrames(aPresContext, frameList, aPropID);
|
2007-07-25 21:03:29 -07:00
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (frameList->IsEmpty()) {
|
|
|
|
// Removed frame and now list is empty. Delete it.
|
|
|
|
delete frameList;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Removed frame, but list not empty. Put it back.
|
|
|
|
SetPropTableFrames(aPresContext, frameList, aPropID);
|
|
|
|
}
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsContainerFrame::SetPropTableFrames(nsPresContext* aPresContext,
|
|
|
|
nsFrameList* aFrameList,
|
|
|
|
nsIAtom* aPropID) const
|
|
|
|
{
|
|
|
|
NS_PRECONDITION(aPresContext && aPropID && aFrameList, "null ptr");
|
|
|
|
nsresult rv = aPresContext->PropertyTable()->SetProperty(this, aPropID,
|
|
|
|
aFrameList, DestroyFrameList, nsnull);
|
|
|
|
NS_ASSERTION(rv != NS_PROPTABLE_PROP_OVERWRITTEN, "existing framelist");
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/**
|
|
|
|
* Push aFromChild and its next siblings to the next-in-flow. Change the
|
|
|
|
* geometric parent of each frame that's pushed. If there is no next-in-flow
|
|
|
|
* the frames are placed on the overflow list (and the geometric parent is
|
|
|
|
* left unchanged).
|
|
|
|
*
|
|
|
|
* Updates the next-in-flow's child count. Does <b>not</b> update the
|
|
|
|
* pusher's child count.
|
|
|
|
*
|
|
|
|
* @param aFromChild the first child frame to push. It is disconnected from
|
|
|
|
* aPrevSibling
|
|
|
|
* @param aPrevSibling aFromChild's previous sibling. Must not be null. It's
|
|
|
|
* an error to push a parent's first child frame
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
nsContainerFrame::PushChildren(nsPresContext* aPresContext,
|
|
|
|
nsIFrame* aFromChild,
|
|
|
|
nsIFrame* aPrevSibling)
|
|
|
|
{
|
2009-09-18 04:09:35 -07:00
|
|
|
NS_PRECONDITION(aFromChild, "null pointer");
|
|
|
|
NS_PRECONDITION(aPrevSibling, "pushing first child");
|
2007-03-22 10:30:00 -07:00
|
|
|
NS_PRECONDITION(aPrevSibling->GetNextSibling() == aFromChild, "bad prev sibling");
|
|
|
|
|
|
|
|
// Disconnect aFromChild from its previous sibling
|
2009-09-18 04:09:35 -07:00
|
|
|
nsFrameList tail = mFrames.RemoveFramesAfter(aPrevSibling);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2009-09-18 04:09:35 -07:00
|
|
|
nsContainerFrame* nextInFlow =
|
|
|
|
static_cast<nsContainerFrame*>(GetNextInFlow());
|
|
|
|
if (nextInFlow) {
|
2007-03-22 10:30:00 -07:00
|
|
|
// XXX This is not a very good thing to do. If it gets removed
|
|
|
|
// then remove the copy of this routine that doesn't do this from
|
|
|
|
// nsInlineFrame.
|
|
|
|
// When pushing and pulling frames we need to check for whether any
|
|
|
|
// views need to be reparented.
|
|
|
|
for (nsIFrame* f = aFromChild; f; f = f->GetNextSibling()) {
|
|
|
|
nsHTMLContainerFrame::ReparentFrameView(aPresContext, f, this, nextInFlow);
|
|
|
|
}
|
2009-09-18 04:09:35 -07:00
|
|
|
nextInFlow->mFrames.InsertFrames(nextInFlow, nsnull, tail);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Add the frames to our overflow list
|
2009-09-18 04:09:36 -07:00
|
|
|
SetOverflowFrames(aPresContext, tail);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Moves any frames on the overflow lists (the prev-in-flow's overflow list and
|
|
|
|
* the receiver's overflow list) to the child list.
|
|
|
|
*
|
|
|
|
* Updates this frame's child count and content mapping.
|
|
|
|
*
|
|
|
|
* @return PR_TRUE if any frames were moved and PR_FALSE otherwise
|
|
|
|
*/
|
|
|
|
PRBool
|
|
|
|
nsContainerFrame::MoveOverflowToChildList(nsPresContext* aPresContext)
|
|
|
|
{
|
|
|
|
PRBool result = PR_FALSE;
|
|
|
|
|
|
|
|
// Check for an overflow list with our prev-in-flow
|
|
|
|
nsContainerFrame* prevInFlow = (nsContainerFrame*)GetPrevInFlow();
|
|
|
|
if (nsnull != prevInFlow) {
|
2009-07-28 05:51:09 -07:00
|
|
|
nsAutoPtr<nsFrameList> prevOverflowFrames(prevInFlow->StealOverflowFrames());
|
2007-03-22 10:30:00 -07:00
|
|
|
if (prevOverflowFrames) {
|
2009-07-28 05:51:09 -07:00
|
|
|
// Tables are special; they can have repeated header/footer
|
|
|
|
// frames on mFrames at this point.
|
|
|
|
NS_ASSERTION(mFrames.IsEmpty() || GetType() == nsGkAtoms::tableFrame,
|
|
|
|
"bad overflow list");
|
2007-03-22 10:30:00 -07:00
|
|
|
// When pushing and pulling frames we need to check for whether any
|
|
|
|
// views need to be reparented.
|
2009-07-28 05:51:09 -07:00
|
|
|
nsHTMLContainerFrame::ReparentFrameViewList(aPresContext,
|
|
|
|
*prevOverflowFrames,
|
|
|
|
prevInFlow, this);
|
|
|
|
mFrames.AppendFrames(this, *prevOverflowFrames);
|
2007-03-22 10:30:00 -07:00
|
|
|
result = PR_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// It's also possible that we have an overflow list for ourselves
|
2009-07-28 05:51:09 -07:00
|
|
|
nsAutoPtr<nsFrameList> overflowFrames(StealOverflowFrames());
|
2007-03-22 10:30:00 -07:00
|
|
|
if (overflowFrames) {
|
|
|
|
NS_ASSERTION(mFrames.NotEmpty(), "overflow list w/o frames");
|
2009-07-28 05:51:09 -07:00
|
|
|
mFrames.AppendFrames(nsnull, *overflowFrames);
|
2007-03-22 10:30:00 -07:00
|
|
|
result = PR_TRUE;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2007-07-25 21:03:29 -07:00
|
|
|
nsOverflowContinuationTracker::nsOverflowContinuationTracker(nsPresContext* aPresContext,
|
|
|
|
nsContainerFrame* aFrame,
|
2007-10-01 22:57:45 -07:00
|
|
|
PRBool aWalkOOFFrames,
|
2007-07-25 21:03:29 -07:00
|
|
|
PRBool aSkipOverflowContainerChildren)
|
|
|
|
: mOverflowContList(nsnull),
|
|
|
|
mPrevOverflowCont(nsnull),
|
|
|
|
mSentry(nsnull),
|
|
|
|
mParent(aFrame),
|
2007-10-01 22:57:45 -07:00
|
|
|
mSkipOverflowContainerChildren(aSkipOverflowContainerChildren),
|
|
|
|
mWalkOOFFrames(aWalkOOFFrames)
|
2007-07-25 21:03:29 -07:00
|
|
|
{
|
|
|
|
NS_PRECONDITION(aFrame, "null frame pointer");
|
|
|
|
nsContainerFrame* next = static_cast<nsContainerFrame*>
|
|
|
|
(aFrame->GetNextInFlow());
|
|
|
|
if (next) {
|
|
|
|
mOverflowContList =
|
|
|
|
next->GetPropTableFrames(aPresContext,
|
|
|
|
nsGkAtoms::overflowContainersProperty);
|
|
|
|
if (mOverflowContList) {
|
|
|
|
mParent = next;
|
|
|
|
SetUpListWalker();
|
|
|
|
}
|
|
|
|
}
|
2007-10-09 12:46:21 -07:00
|
|
|
if (!mOverflowContList) {
|
2007-07-25 21:03:29 -07:00
|
|
|
mOverflowContList =
|
|
|
|
mParent->GetPropTableFrames(aPresContext,
|
|
|
|
nsGkAtoms::excessOverflowContainersProperty);
|
|
|
|
if (mOverflowContList) {
|
|
|
|
SetUpListWalker();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper function to walk past overflow continuations whose prev-in-flow
|
|
|
|
* isn't a normal child and to set mSentry and mPrevOverflowCont correctly.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
nsOverflowContinuationTracker::SetUpListWalker()
|
|
|
|
{
|
|
|
|
NS_ASSERTION(!mSentry && !mPrevOverflowCont,
|
|
|
|
"forgot to reset mSentry or mPrevOverflowCont");
|
|
|
|
if (mOverflowContList) {
|
|
|
|
nsIFrame* cur = mOverflowContList->FirstChild();
|
|
|
|
if (mSkipOverflowContainerChildren) {
|
|
|
|
while (cur && (cur->GetPrevInFlow()->GetStateBits()
|
|
|
|
& NS_FRAME_IS_OVERFLOW_CONTAINER)) {
|
|
|
|
mPrevOverflowCont = cur;
|
|
|
|
cur = cur->GetNextSibling();
|
|
|
|
}
|
2007-12-01 19:13:23 -08:00
|
|
|
while (cur && (!(cur->GetStateBits() & NS_FRAME_OUT_OF_FLOW)
|
|
|
|
== mWalkOOFFrames)) {
|
2007-10-01 22:57:45 -07:00
|
|
|
mPrevOverflowCont = cur;
|
|
|
|
cur = cur->GetNextSibling();
|
|
|
|
}
|
2007-07-25 21:03:29 -07:00
|
|
|
}
|
|
|
|
if (cur) {
|
|
|
|
mSentry = cur->GetPrevInFlow();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper function to step forward through the overflow continuations list.
|
2007-10-01 22:57:45 -07:00
|
|
|
* Sets mSentry and mPrevOverflowCont, skipping over OOF or non-OOF frames
|
|
|
|
* as appropriate. May only be called when we have already set up an
|
|
|
|
* mOverflowContList; mOverflowContList cannot be null.
|
2007-07-25 21:03:29 -07:00
|
|
|
*/
|
|
|
|
void
|
|
|
|
nsOverflowContinuationTracker::StepForward()
|
|
|
|
{
|
|
|
|
NS_PRECONDITION(mOverflowContList, "null list");
|
|
|
|
|
|
|
|
// Step forward
|
|
|
|
if (mPrevOverflowCont) {
|
|
|
|
mPrevOverflowCont = mPrevOverflowCont->GetNextSibling();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
mPrevOverflowCont = mOverflowContList->FirstChild();
|
|
|
|
}
|
|
|
|
|
2007-10-01 22:57:45 -07:00
|
|
|
// Skip over oof or non-oof frames as appropriate
|
|
|
|
if (mSkipOverflowContainerChildren) {
|
|
|
|
nsIFrame* cur = mPrevOverflowCont->GetNextSibling();
|
2007-12-01 19:13:23 -08:00
|
|
|
while (cur && (!(cur->GetStateBits() & NS_FRAME_OUT_OF_FLOW)
|
|
|
|
== mWalkOOFFrames)) {
|
2007-10-01 22:57:45 -07:00
|
|
|
mPrevOverflowCont = cur;
|
|
|
|
cur = cur->GetNextSibling();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-25 21:03:29 -07:00
|
|
|
// Set up the sentry
|
|
|
|
mSentry = (mPrevOverflowCont->GetNextSibling())
|
|
|
|
? mPrevOverflowCont->GetNextSibling()->GetPrevInFlow()
|
|
|
|
: nsnull;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsOverflowContinuationTracker::Insert(nsIFrame* aOverflowCont,
|
|
|
|
nsReflowStatus& aReflowStatus)
|
|
|
|
{
|
|
|
|
NS_PRECONDITION(aOverflowCont, "null frame pointer");
|
2007-12-01 19:13:23 -08:00
|
|
|
NS_PRECONDITION(!mSkipOverflowContainerChildren || mWalkOOFFrames ==
|
|
|
|
!!(aOverflowCont->GetStateBits() & NS_FRAME_OUT_OF_FLOW),
|
|
|
|
"shouldn't insert frame that doesn't match walker type");
|
2007-07-25 21:03:29 -07:00
|
|
|
NS_PRECONDITION(aOverflowCont->GetPrevInFlow(),
|
|
|
|
"overflow containers must have a prev-in-flow");
|
|
|
|
nsresult rv = NS_OK;
|
2008-09-06 01:48:09 -07:00
|
|
|
PRBool convertedToOverflowContainer = PR_FALSE;
|
|
|
|
nsPresContext* presContext = aOverflowCont->PresContext();
|
2007-07-25 21:03:29 -07:00
|
|
|
if (!mSentry || aOverflowCont != mSentry->GetNextInFlow()) {
|
|
|
|
// Not in our list, so we need to add it
|
2007-11-15 11:23:07 -08:00
|
|
|
if (aOverflowCont->GetStateBits() & NS_FRAME_IS_OVERFLOW_CONTAINER) {
|
|
|
|
// aOverflowCont is in some other overflow container list,
|
2007-07-25 21:03:29 -07:00
|
|
|
// steal it first
|
2007-11-15 11:23:07 -08:00
|
|
|
NS_ASSERTION(!(mOverflowContList &&
|
|
|
|
mOverflowContList->ContainsFrame(aOverflowCont)),
|
|
|
|
"overflow containers out of order");
|
2007-07-25 21:03:29 -07:00
|
|
|
rv = static_cast<nsContainerFrame*>(aOverflowCont->GetParent())
|
|
|
|
->StealFrame(presContext, aOverflowCont);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
aOverflowCont->AddStateBits(NS_FRAME_IS_OVERFLOW_CONTAINER);
|
2008-09-06 01:48:09 -07:00
|
|
|
convertedToOverflowContainer = PR_TRUE;
|
2007-07-25 21:03:29 -07:00
|
|
|
}
|
|
|
|
if (!mOverflowContList) {
|
|
|
|
mOverflowContList = new nsFrameList();
|
|
|
|
rv = mParent->SetPropTableFrames(presContext,
|
|
|
|
mOverflowContList, nsGkAtoms::excessOverflowContainersProperty);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
SetUpListWalker();
|
|
|
|
}
|
|
|
|
if (aOverflowCont->GetParent() != mParent) {
|
|
|
|
nsHTMLContainerFrame::ReparentFrameView(presContext, aOverflowCont,
|
|
|
|
aOverflowCont->GetParent(),
|
|
|
|
mParent);
|
|
|
|
}
|
|
|
|
mOverflowContList->InsertFrame(mParent, mPrevOverflowCont, aOverflowCont);
|
|
|
|
aReflowStatus |= NS_FRAME_REFLOW_NEXTINFLOW;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we need to reflow it, mark it dirty
|
|
|
|
if (aReflowStatus & NS_FRAME_REFLOW_NEXTINFLOW)
|
|
|
|
aOverflowCont->AddStateBits(NS_FRAME_IS_DIRTY);
|
|
|
|
|
|
|
|
// It's in our list, just step forward
|
|
|
|
StepForward();
|
2007-12-01 19:13:23 -08:00
|
|
|
NS_ASSERTION(mPrevOverflowCont == aOverflowCont ||
|
|
|
|
(mSkipOverflowContainerChildren &&
|
|
|
|
(mPrevOverflowCont->GetStateBits() & NS_FRAME_OUT_OF_FLOW) !=
|
|
|
|
(aOverflowCont->GetStateBits() & NS_FRAME_OUT_OF_FLOW)),
|
|
|
|
"OverflowContTracker in unexpected state");
|
2008-09-06 01:48:09 -07:00
|
|
|
|
|
|
|
if (convertedToOverflowContainer) {
|
|
|
|
// Convert all non-overflow-container continuations of aOverflowCont
|
|
|
|
// into overflow containers and move them to our overflow
|
|
|
|
// tracker. This preserves the invariant that the next-continuations
|
|
|
|
// of an overflow container are also overflow containers.
|
|
|
|
nsIFrame* f = aOverflowCont->GetNextContinuation();
|
|
|
|
if (f && !(f->GetStateBits() & NS_FRAME_IS_OVERFLOW_CONTAINER)) {
|
|
|
|
nsContainerFrame* parent = static_cast<nsContainerFrame*>(f->GetParent());
|
|
|
|
rv = parent->StealFrame(presContext, f);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
Insert(f, aReflowStatus);
|
|
|
|
}
|
|
|
|
}
|
2007-07-25 21:03:29 -07:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsOverflowContinuationTracker::Finish(nsIFrame* aChild)
|
|
|
|
{
|
|
|
|
NS_PRECONDITION(aChild, "null ptr");
|
|
|
|
NS_PRECONDITION(aChild->GetNextInFlow(),
|
2007-10-01 22:57:45 -07:00
|
|
|
"supposed to call Finish *before* deleting next-in-flow!");
|
2008-09-06 01:48:09 -07:00
|
|
|
|
|
|
|
for (nsIFrame* f = aChild; f; f = f->GetNextInFlow()) {
|
|
|
|
if (f == mSentry) {
|
|
|
|
// Make sure we drop all references if this was the only frame
|
|
|
|
// in the overflow containers list
|
|
|
|
if (mOverflowContList->FirstChild() == f->GetNextInFlow()
|
|
|
|
&& !f->GetNextInFlow()->GetNextSibling()) {
|
|
|
|
mOverflowContList = nsnull;
|
|
|
|
mPrevOverflowCont = nsnull;
|
|
|
|
mSentry = nsnull;
|
|
|
|
mParent = static_cast<nsContainerFrame*>(f->GetParent());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Step past aChild
|
|
|
|
nsIFrame* prevOverflowCont = mPrevOverflowCont;
|
|
|
|
StepForward();
|
|
|
|
if (mPrevOverflowCont == f->GetNextInFlow()) {
|
|
|
|
// Pull mPrevOverflowChild back to aChild's prevSibling:
|
|
|
|
// aChild will be removed from our list by our caller
|
|
|
|
mPrevOverflowCont = prevOverflowCont;
|
|
|
|
}
|
2007-12-01 19:13:23 -08:00
|
|
|
}
|
2007-07-25 21:03:29 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Debugging
|
|
|
|
|
|
|
|
#ifdef NS_DEBUG
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsContainerFrame::List(FILE* out, PRInt32 aIndent) const
|
|
|
|
{
|
|
|
|
IndentBy(out, aIndent);
|
|
|
|
ListTag(out);
|
|
|
|
#ifdef DEBUG_waterson
|
2007-07-08 00:08:04 -07:00
|
|
|
fprintf(out, " [parent=%p]", static_cast<void*>(mParent));
|
2007-03-22 10:30:00 -07:00
|
|
|
#endif
|
|
|
|
if (HasView()) {
|
2007-07-08 00:08:04 -07:00
|
|
|
fprintf(out, " [view=%p]", static_cast<void*>(GetView()));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2009-09-18 04:09:36 -07:00
|
|
|
if (GetNextSibling()) {
|
|
|
|
fprintf(out, " next=%p", static_cast<void*>(GetNextSibling()));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
if (nsnull != GetPrevContinuation()) {
|
2007-07-08 00:08:04 -07:00
|
|
|
fprintf(out, " prev-continuation=%p", static_cast<void*>(GetPrevContinuation()));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
if (nsnull != GetNextContinuation()) {
|
2007-07-08 00:08:04 -07:00
|
|
|
fprintf(out, " next-continuation=%p", static_cast<void*>(GetNextContinuation()));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
fprintf(out, " {%d,%d,%d,%d}", mRect.x, mRect.y, mRect.width, mRect.height);
|
|
|
|
if (0 != mState) {
|
|
|
|
fprintf(out, " [state=%08x]", mState);
|
|
|
|
}
|
2007-07-08 00:08:04 -07:00
|
|
|
fprintf(out, " [content=%p]", static_cast<void*>(mContent));
|
|
|
|
nsContainerFrame* f = const_cast<nsContainerFrame*>(this);
|
2009-04-05 17:31:50 -07:00
|
|
|
if (f->HasOverflowRect()) {
|
2008-02-19 23:08:55 -08:00
|
|
|
nsRect overflowArea = f->GetOverflowRect();
|
|
|
|
fprintf(out, " [overflow=%d,%d,%d,%d]", overflowArea.x, overflowArea.y,
|
|
|
|
overflowArea.width, overflowArea.height);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2007-07-08 00:08:04 -07:00
|
|
|
fprintf(out, " [sc=%p]", static_cast<void*>(mStyleContext));
|
2009-10-29 14:17:56 -07:00
|
|
|
nsIAtom* pseudoTag = mStyleContext->GetPseudo();
|
2007-03-22 10:30:00 -07:00
|
|
|
if (pseudoTag) {
|
|
|
|
nsAutoString atomString;
|
|
|
|
pseudoTag->ToString(atomString);
|
|
|
|
fprintf(out, " pst=%s",
|
|
|
|
NS_LossyConvertUTF16toASCII(atomString).get());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Output the children
|
|
|
|
nsIAtom* listName = nsnull;
|
|
|
|
PRInt32 listIndex = 0;
|
|
|
|
PRBool outputOneList = PR_FALSE;
|
|
|
|
do {
|
|
|
|
nsIFrame* kid = GetFirstChild(listName);
|
|
|
|
if (nsnull != kid) {
|
|
|
|
if (outputOneList) {
|
|
|
|
IndentBy(out, aIndent);
|
|
|
|
}
|
|
|
|
outputOneList = PR_TRUE;
|
|
|
|
nsAutoString tmp;
|
|
|
|
if (nsnull != listName) {
|
|
|
|
listName->ToString(tmp);
|
|
|
|
fputs(NS_LossyConvertUTF16toASCII(tmp).get(), out);
|
|
|
|
}
|
|
|
|
fputs("<\n", out);
|
|
|
|
while (nsnull != kid) {
|
|
|
|
// Verify the child frame's parent frame pointer is correct
|
|
|
|
NS_ASSERTION(kid->GetParent() == (nsIFrame*)this, "bad parent frame pointer");
|
|
|
|
|
|
|
|
// Have the child frame list
|
2009-08-20 14:52:48 -07:00
|
|
|
kid->List(out, aIndent + 1);
|
2007-03-22 10:30:00 -07:00
|
|
|
kid = kid->GetNextSibling();
|
|
|
|
}
|
|
|
|
IndentBy(out, aIndent);
|
|
|
|
fputs(">\n", out);
|
|
|
|
}
|
|
|
|
listName = GetAdditionalChildListName(listIndex++);
|
|
|
|
} while(nsnull != listName);
|
|
|
|
|
|
|
|
if (!outputOneList) {
|
|
|
|
fputs("<>\n", out);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
#endif
|