зеркало из https://github.com/mozilla/pjs.git
1639 строки
57 KiB
C++
1639 строки
57 KiB
C++
/* -*- 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>
|
|
* Elika J. Etemad ("fantasai") <fantasai@inkedblade.net>
|
|
*
|
|
* 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"
|
|
#include "nsHTMLContainerFrame.h"
|
|
#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"
|
|
#include "nsListControlFrame.h"
|
|
#include "nsIBaseWindow.h"
|
|
#include "nsThemeConstants.h"
|
|
#include "nsCSSFrameConstructor.h"
|
|
#include "nsThemeConstants.h"
|
|
|
|
#ifdef NS_DEBUG
|
|
#undef NOISY
|
|
#else
|
|
#undef NOISY
|
|
#endif
|
|
|
|
nsContainerFrame::~nsContainerFrame()
|
|
{
|
|
}
|
|
|
|
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
|
|
nsContainerFrame::SetInitialChildList(nsIAtom* aListName,
|
|
nsIFrame* aChildList)
|
|
{
|
|
nsresult result;
|
|
if (!mFrames.IsEmpty()) {
|
|
// 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,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
if (nsnull != aListName) {
|
|
#ifdef IBMBIDI
|
|
if (aListName != nsGkAtoms::nextBidi)
|
|
#endif
|
|
{
|
|
NS_ERROR("unexpected child list");
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
}
|
|
if (aFrameList) {
|
|
mFrames.AppendFrames(this, aFrameList);
|
|
|
|
// Ask the parent frame to reflow me.
|
|
#ifdef IBMBIDI
|
|
if (nsnull == aListName)
|
|
#endif
|
|
{
|
|
PresContext()->PresShell()->
|
|
FrameNeedsReflow(this, nsIPresShell::eTreeChange, NS_FRAME_IS_DIRTY);
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsContainerFrame::InsertFrames(nsIAtom* aListName,
|
|
nsIFrame* aPrevFrame,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
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;
|
|
}
|
|
}
|
|
if (aFrameList) {
|
|
// Insert frames after aPrevFrame
|
|
mFrames.InsertFrames(this, aPrevFrame, aFrameList);
|
|
|
|
#ifdef IBMBIDI
|
|
if (nsnull == aListName)
|
|
#endif
|
|
{
|
|
PresContext()->PresShell()->
|
|
FrameNeedsReflow(this, nsIPresShell::eTreeChange, NS_FRAME_IS_DIRTY);
|
|
}
|
|
}
|
|
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.
|
|
PRBool generateReflowCommand = PR_TRUE;
|
|
#ifdef IBMBIDI
|
|
if (nsGkAtoms::nextBidi == aListName) {
|
|
generateReflowCommand = PR_FALSE;
|
|
}
|
|
#endif
|
|
nsContainerFrame* parent = static_cast<nsContainerFrame*>(aOldFrame->GetParent());
|
|
while (aOldFrame) {
|
|
// 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();
|
|
//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.
|
|
if (parent == this) {
|
|
if (!parent->mFrames.DestroyFrame(aOldFrame)) {
|
|
// Try to remove it from our overflow list, if we have one.
|
|
// The simplest way is to reuse StealFrame.
|
|
#ifdef DEBUG
|
|
nsresult rv =
|
|
#endif
|
|
StealFrame(PresContext(), aOldFrame, PR_TRUE);
|
|
NS_ASSERTION(NS_SUCCEEDED(rv), "Could not find frame to remove!");
|
|
aOldFrame->Destroy();
|
|
}
|
|
} else {
|
|
// This recursive call takes care of all continuations after aOldFrame,
|
|
// so we don't need to loop anymore.
|
|
parent->RemoveFrame(nsnull, aOldFrame);
|
|
break;
|
|
}
|
|
aOldFrame = oldFrameNextContinuation;
|
|
if (aOldFrame) {
|
|
parent = static_cast<nsContainerFrame*>(aOldFrame->GetParent());
|
|
}
|
|
}
|
|
|
|
if (generateReflowCommand) {
|
|
PresContext()->PresShell()->
|
|
FrameNeedsReflow(this, nsIPresShell::eTreeChange, NS_FRAME_IS_DIRTY);
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
nsContainerFrame::Destroy()
|
|
{
|
|
// Prevent event dispatch during destruction
|
|
if (HasView()) {
|
|
GetView()->SetClientData(nsnull);
|
|
}
|
|
|
|
// Delete the primary child list
|
|
mFrames.DestroyFrames();
|
|
|
|
// Destroy auxiliary frame lists
|
|
nsPresContext* prescontext = PresContext();
|
|
|
|
nsFrameList overflowFrames(GetOverflowFrames(prescontext, PR_TRUE));
|
|
overflowFrames.DestroyFrames();
|
|
|
|
if (IsFrameOfType(nsIFrame::eCanContainOverflowContainers)) {
|
|
nsFrameList* frameList = RemovePropTableFrames(prescontext,
|
|
nsGkAtoms::overflowContainersProperty);
|
|
if (frameList)
|
|
frameList->Destroy();
|
|
|
|
frameList = RemovePropTableFrames(prescontext,
|
|
nsGkAtoms::excessOverflowContainersProperty);
|
|
if (frameList)
|
|
frameList->Destroy();
|
|
}
|
|
|
|
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.
|
|
PresContext()->EventStateManager()->ContentRemoved(content);
|
|
content->UnbindFromTree();
|
|
}
|
|
delete generatedContent;
|
|
}
|
|
}
|
|
|
|
// Destroy the frame and remove the flow pointers
|
|
nsSplittableFrame::Destroy();
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// Child frame enumeration
|
|
|
|
nsIFrame*
|
|
nsContainerFrame::GetFirstChild(nsIAtom* aListName) const
|
|
{
|
|
// We only know about the unnamed principal child list and the overflow
|
|
// list
|
|
if (nsnull == aListName) {
|
|
return mFrames.FirstChild();
|
|
} else if (nsGkAtoms::overflowList == aListName) {
|
|
return GetOverflowFrames(PresContext(), PR_FALSE);
|
|
} else if (nsGkAtoms::overflowContainersList == aListName) {
|
|
nsFrameList* list = GetPropTableFrames(PresContext(),
|
|
nsGkAtoms::overflowContainersProperty);
|
|
return (list) ? list->FirstChild() : nsnull;
|
|
} else if (nsGkAtoms::excessOverflowContainersList == aListName) {
|
|
nsFrameList* list = GetPropTableFrames(PresContext(),
|
|
nsGkAtoms::excessOverflowContainersProperty);
|
|
return (list) ? list->FirstChild() : nsnull;
|
|
|
|
} else {
|
|
return nsnull;
|
|
}
|
|
}
|
|
|
|
#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
|
|
|
|
|
|
nsIAtom*
|
|
nsContainerFrame::GetAdditionalChildListName(PRInt32 aIndex) const
|
|
{
|
|
if (NS_CONTAINER_FRAME_OVERFLOW_LIST_INDEX == aIndex)
|
|
return nsGkAtoms::overflowList;
|
|
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;
|
|
}
|
|
return nsnull;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// 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);
|
|
}
|
|
|
|
static PRBool
|
|
IsMenuPopup(nsIFrame *aFrame)
|
|
{
|
|
nsIAtom *frameType = aFrame->GetType();
|
|
|
|
// We're a menupopup if we're the list control frame dropdown for a combobox.
|
|
if (frameType == nsGkAtoms::listControlFrame) {
|
|
nsListControlFrame *listControlFrame = static_cast<nsListControlFrame*>(aFrame);
|
|
|
|
if (listControlFrame) {
|
|
return listControlFrame->IsInDropDownMode();
|
|
}
|
|
}
|
|
|
|
// ... or if we're a XUL menupopup frame.
|
|
return (frameType == nsGkAtoms::menuPopupFrame);
|
|
}
|
|
|
|
static PRBool
|
|
IsTopLevelWidget(nsPresContext* aPresContext)
|
|
{
|
|
nsCOMPtr<nsISupports> container = aPresContext->Document()->GetContainer();
|
|
nsCOMPtr<nsIBaseWindow> baseWindow = do_QueryInterface(container);
|
|
if (!baseWindow)
|
|
return PR_FALSE;
|
|
|
|
nsCOMPtr<nsIWidget> mainWidget;
|
|
baseWindow->GetMainWidget(getter_AddRefs(mainWidget));
|
|
if (!mainWidget)
|
|
return PR_FALSE;
|
|
|
|
nsWindowType windowType;
|
|
mainWidget->GetWindowType(windowType);
|
|
return windowType == eWindowType_toplevel ||
|
|
windowType == eWindowType_dialog;
|
|
// popups aren't toplevel so they're not handled here
|
|
}
|
|
|
|
static void
|
|
SyncFrameViewGeometryDependentProperties(nsPresContext* aPresContext,
|
|
nsIFrame* aFrame,
|
|
nsStyleContext* aStyleContext,
|
|
nsIView* aView,
|
|
PRUint32 aFlags)
|
|
{
|
|
nsIViewManager* vm = aView->GetViewManager();
|
|
|
|
PRBool isCanvas;
|
|
const nsStyleBackground* bg;
|
|
nsCSSRendering::FindBackground(aPresContext, aFrame, &bg, &isCanvas);
|
|
|
|
if (isCanvas) {
|
|
nsIView* rootView;
|
|
vm->GetRootView(rootView);
|
|
|
|
if (aView->HasWidget() && aView == rootView &&
|
|
IsTopLevelWidget(aPresContext)) {
|
|
// The issue here is that the CSS 'background' propagates from the root
|
|
// element's frame (rootFrame) to the real root frame (nsViewportFrame),
|
|
// so we need to call GetFrameTransparency on that. But -moz-appearance
|
|
// does not propagate so we need to check that directly on rootFrame.
|
|
nsTransparencyMode mode = nsLayoutUtils::GetFrameTransparency(aFrame);
|
|
nsIFrame *rootFrame = aPresContext->PresShell()->FrameConstructor()->GetRootElementStyleFrame();
|
|
if(rootFrame && NS_THEME_WIN_GLASS == rootFrame->GetStyleDisplay()->mAppearance)
|
|
mode = eTransparencyGlass;
|
|
nsIWidget* widget = aView->GetWidget();
|
|
widget->SetTransparencyMode(mode);
|
|
if (rootFrame)
|
|
widget->SetWindowShadowStyle(rootFrame->GetStyleUIReset()->mWindowShadow);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
nsContainerFrame::SyncFrameViewAfterReflow(nsPresContext* aPresContext,
|
|
nsIFrame* aFrame,
|
|
nsIView* aView,
|
|
const nsRect* aCombinedArea,
|
|
PRUint32 aFlags)
|
|
{
|
|
if (!aView) {
|
|
return;
|
|
}
|
|
|
|
NS_ASSERTION(aCombinedArea, "Combined area must be passed in now");
|
|
|
|
// 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();
|
|
|
|
vm->ResizeView(aView, *aCombinedArea, PR_TRUE);
|
|
|
|
// Even if the size hasn't changed, we need to sync up the
|
|
// geometry dependent properties, because (kidState &
|
|
// NS_FRAME_OUTSIDE_CHILDREN) might have changed, and we can't
|
|
// detect whether it has or not. Likewise, whether the view size
|
|
// has changed or not, we may need to change the transparency
|
|
// state even if there is no clip.
|
|
nsStyleContext* savedStyleContext = aFrame->GetStyleContext();
|
|
SyncFrameViewGeometryDependentProperties(aPresContext, aFrame, savedStyleContext, aView, aFlags);
|
|
}
|
|
}
|
|
|
|
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();
|
|
|
|
/* 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();
|
|
|
|
if (nsnull == aStyleContext) {
|
|
aStyleContext = aFrame->GetStyleContext();
|
|
}
|
|
|
|
// Make sure visibility is correct
|
|
if (0 == (aFlags & NS_FRAME_NO_VISIBILITY)) {
|
|
// See if the view should be hidden or visible
|
|
PRBool viewIsVisible = PR_TRUE;
|
|
|
|
if (!aStyleContext->GetStyleVisibility()->IsVisible() &&
|
|
!aFrame->SupportsVisibilityHidden()) {
|
|
// If it's a scrollable frame that can't hide its scrollbars,
|
|
// hide the view. This means that child elements can't override
|
|
// their parent's visibility, but it's not practical to leave it
|
|
// visible in all cases because the scrollbars will be showing
|
|
// XXXldb Does the view system really enforce this correctly?
|
|
viewIsVisible = PR_FALSE;
|
|
} else if (IsMenuPopup(aFrame)) {
|
|
// if the view is for a popup, don't show the view if the popup is closed
|
|
nsIWidget* widget = aView->GetWidget();
|
|
if (widget) {
|
|
nsWindowType windowType;
|
|
widget->GetWindowType(windowType);
|
|
if (windowType == eWindowType_popup) {
|
|
widget->IsVisible(viewIsVisible);
|
|
}
|
|
}
|
|
else {
|
|
// widgets for popups can be created later when the popup is opened,
|
|
// so if there is no widget, the popup won't be open.
|
|
viewIsVisible = PR_FALSE;
|
|
}
|
|
}
|
|
|
|
vm->SetViewVisibility(aView, viewIsVisible ? nsViewVisibility_kShow :
|
|
nsViewVisibility_kHide);
|
|
}
|
|
|
|
// 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);
|
|
|
|
SyncFrameViewGeometryDependentProperties(aPresContext, aFrame, aStyleContext, aView, aFlags);
|
|
}
|
|
|
|
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
|
|
if (aFrame->GetStyleContext()->GetPseudoType() ==
|
|
nsCSSAnonBoxes::scrolledContent) {
|
|
return PR_TRUE;
|
|
}
|
|
return aFrame->NeedsView() || aFrame->GetStyleDisplay()->HasTransform();
|
|
}
|
|
|
|
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.
|
|
// 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) +
|
|
styleBorder->GetActualBorderWidth(startSide) +
|
|
GetCoord(styleMargin->mMargin.Get(startSide), 0);
|
|
}
|
|
|
|
const nsLineList_iterator* savedLine = aData->line;
|
|
|
|
nsContainerFrame *lastInFlow;
|
|
for (nsContainerFrame *nif = this; nif;
|
|
nif = (nsContainerFrame*) nif->GetNextInFlow()) {
|
|
for (nsIFrame *kid = nif->mFrames.FirstChild(); kid;
|
|
kid = kid->GetNextSibling()) {
|
|
if (aType == nsLayoutUtils::MIN_WIDTH)
|
|
kid->AddInlineMinWidth(aRenderingContext,
|
|
static_cast<InlineMinWidthData*>(aData));
|
|
else
|
|
kid->AddInlinePrefWidth(aRenderingContext,
|
|
static_cast<InlinePrefWidthData*>(aData));
|
|
}
|
|
|
|
// After we advance to our next-in-flow, the stored line may not
|
|
// longer be the correct line. Just forget it.
|
|
aData->line = nsnull;
|
|
lastInFlow = nif;
|
|
}
|
|
|
|
aData->line = savedLine;
|
|
|
|
// 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.
|
|
// 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) +
|
|
styleBorder->GetActualBorderWidth(endSide) +
|
|
GetCoord(styleMargin->mMargin.Get(endSide), 0);
|
|
}
|
|
}
|
|
|
|
/* 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,
|
|
nsReflowStatus& aStatus,
|
|
nsOverflowContinuationTracker* aTracker)
|
|
{
|
|
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);
|
|
|
|
if (NS_FRAME_NO_MOVE_FRAME != (aFlags & NS_FRAME_NO_MOVE_FRAME)) {
|
|
if ((aFlags & NS_FRAME_INVALIDATE_ON_MOVE) &&
|
|
!(aKidFrame->GetStateBits() & NS_FRAME_FIRST_REFLOW) &&
|
|
aKidFrame->GetPosition() != nsPoint(aX, aY)) {
|
|
aKidFrame->InvalidateOverflowRect();
|
|
}
|
|
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
|
|
if (NS_SUCCEEDED(result) && NS_FRAME_IS_FULLY_COMPLETE(aStatus)) {
|
|
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)
|
|
if (aTracker) aTracker->Finish(aKidFrame);
|
|
static_cast<nsContainerFrame*>(kidNextInFlow->GetParent())
|
|
->DeleteNextInFlowChild(aPresContext, kidNextInFlow);
|
|
}
|
|
}
|
|
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();
|
|
}
|
|
|
|
// 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);
|
|
} 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
|
|
nsContainerFrame::FinishReflowChild(nsIFrame* aKidFrame,
|
|
nsPresContext* aPresContext,
|
|
const nsHTMLReflowState* aReflowState,
|
|
const nsHTMLReflowMetrics& aDesiredSize,
|
|
nscoord aX,
|
|
nscoord aY,
|
|
PRUint32 aFlags)
|
|
{
|
|
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);
|
|
}
|
|
|
|
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) {
|
|
nsFrameList reparenter;
|
|
reparenter.InsertFrames(this, nsnull, excessFrames->FirstChild());
|
|
nsHTMLContainerFrame::ReparentFrameViewList(aPresContext,
|
|
excessFrames->FirstChild(),
|
|
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
|
|
|
|
nsOverflowContinuationTracker tracker(aPresContext, this, PR_FALSE, PR_FALSE);
|
|
for (nsIFrame* frame = overflowContainers->FirstChild(); frame;
|
|
frame = frame->GetNextSibling()) {
|
|
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;
|
|
|
|
// Cache old bounds
|
|
nsRect oldRect = frame->GetRect();
|
|
nsRect oldOverflow = frame->GetOverflowRect();
|
|
|
|
// Reflow
|
|
rv = ReflowChild(frame, aPresContext, desiredSize, frameState,
|
|
prevRect.x, 0, aFlags, frameStatus, &tracker);
|
|
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);
|
|
|
|
// 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);
|
|
}
|
|
|
|
// Handle continuations
|
|
if (!NS_FRAME_IS_FULLY_COMPLETE(frameStatus)) {
|
|
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");
|
|
}
|
|
|
|
// Acquire a next-in-flow, creating it if necessary
|
|
nsIFrame* nif = frame->GetNextInFlow();
|
|
if (!nif) {
|
|
rv = nsHTMLContainerFrame::CreateNextInFlow(aPresContext, this,
|
|
frame, nif);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
NS_ASSERTION(frameStatus & NS_FRAME_REFLOW_NEXTINFLOW,
|
|
"Someone forgot a REFLOW_NEXTINFLOW flag");
|
|
frame->SetNextSibling(nif->GetNextSibling());
|
|
nif->SetNextSibling(nsnull);
|
|
}
|
|
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);
|
|
}
|
|
NS_MergeReflowStatusInto(&aStatus, frameStatus);
|
|
// 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);
|
|
}
|
|
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::AddFrames(nsIFrame* aFrameList,
|
|
nsIFrame* aPrevSibling)
|
|
{
|
|
mFrames.InsertFrames(nsnull, aPrevSibling, aFrameList);
|
|
return NS_OK;
|
|
}
|
|
|
|
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 {
|
|
if (!mFrames.RemoveFrame(aChild)) {
|
|
// We didn't find the child in the parent's principal child list.
|
|
// Maybe it's on the overflow list?
|
|
nsFrameList frameList(GetOverflowFrames(aPresContext, PR_TRUE));
|
|
removed = frameList.RemoveFrame(aChild);
|
|
if (frameList.NotEmpty()) {
|
|
nsresult rv = SetOverflowFrames(aPresContext, frameList.FirstChild());
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
}
|
|
}
|
|
return (removed) ? NS_OK : NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
/**
|
|
* Remove and delete aNextInFlow and its next-in-flows. Updates the sibling and flow
|
|
* pointers
|
|
*/
|
|
void
|
|
nsContainerFrame::DeleteNextInFlowChild(nsPresContext* aPresContext,
|
|
nsIFrame* aNextInFlow)
|
|
{
|
|
#ifdef DEBUG
|
|
nsIFrame* prevInFlow = aNextInFlow->GetPrevInFlow();
|
|
#endif
|
|
NS_PRECONDITION(prevInFlow, "bad prev-in-flow");
|
|
|
|
// 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) {
|
|
nsAutoVoidArray frames;
|
|
for (nsIFrame* f = nextNextInFlow; f; f = f->GetNextInFlow()) {
|
|
frames.AppendElement(f);
|
|
}
|
|
for (PRInt32 i = frames.Count() - 1; i >= 0; --i) {
|
|
nsIFrame* delFrame = static_cast<nsIFrame*>(frames.ElementAt(i));
|
|
static_cast<nsContainerFrame*>(delFrame->GetParent())
|
|
->DeleteNextInFlowChild(aPresContext, delFrame);
|
|
}
|
|
}
|
|
|
|
aNextInFlow->Invalidate(aNextInFlow->GetOverflowRect());
|
|
|
|
// Disconnect the next-in-flow from the flow list
|
|
nsSplittableFrame::BreakFromPrevFlow(aNextInFlow);
|
|
|
|
// Take the next-in-flow out of the parent's child list
|
|
#ifdef DEBUG
|
|
nsresult rv =
|
|
#endif
|
|
StealFrame(aPresContext, aNextInFlow);
|
|
NS_ASSERTION(NS_SUCCEEDED(rv), "StealFrame failure");
|
|
|
|
// Delete the next-in-flow frame and its descendants.
|
|
aNextInFlow->Destroy();
|
|
|
|
NS_POSTCONDITION(!prevInFlow->GetNextInFlow(), "non null next-in-flow");
|
|
}
|
|
|
|
/**
|
|
* Get the frames on the overflow list
|
|
*/
|
|
nsIFrame*
|
|
nsContainerFrame::GetOverflowFrames(nsPresContext* aPresContext,
|
|
PRBool aRemoveProperty) const
|
|
{
|
|
nsPropertyTable *propTable = aPresContext->PropertyTable();
|
|
if (aRemoveProperty) {
|
|
return (nsIFrame*) propTable->UnsetProperty(this,
|
|
nsGkAtoms::overflowProperty);
|
|
}
|
|
return (nsIFrame*) propTable->GetProperty(this,
|
|
nsGkAtoms::overflowProperty);
|
|
}
|
|
|
|
// Destructor function for the overflow frame property
|
|
static void
|
|
DestroyOverflowFrames(void* aFrame,
|
|
nsIAtom* aPropertyName,
|
|
void* aPropertyValue,
|
|
void* aDtorData)
|
|
{
|
|
if (aPropertyValue) {
|
|
nsFrameList frames((nsIFrame*)aPropertyValue);
|
|
|
|
frames.DestroyFrames();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Set the frames on the overflow list
|
|
*/
|
|
nsresult
|
|
nsContainerFrame::SetOverflowFrames(nsPresContext* aPresContext,
|
|
nsIFrame* aOverflowFrames)
|
|
{
|
|
nsresult rv =
|
|
aPresContext->PropertyTable()->SetProperty(this,
|
|
nsGkAtoms::overflowProperty,
|
|
aOverflowFrames,
|
|
DestroyOverflowFrames,
|
|
nsnull);
|
|
|
|
// Verify that we didn't overwrite an existing overflow list
|
|
NS_ASSERTION(rv != NS_PROPTABLE_PROP_OVERWRITTEN, "existing overflow list");
|
|
|
|
return rv;
|
|
}
|
|
|
|
// Destructor function for the proptable-stored framelists
|
|
static void
|
|
DestroyFrameList(void* aFrame,
|
|
nsIAtom* aPropertyName,
|
|
void* aPropertyValue,
|
|
void* aDtorData)
|
|
{
|
|
if (aPropertyValue)
|
|
static_cast<nsFrameList*>(aPropertyValue)->Destroy();
|
|
}
|
|
|
|
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);
|
|
if (!frameList) {
|
|
// No such list
|
|
return PR_FALSE;
|
|
}
|
|
if (!frameList->RemoveFrame(aFrame)) {
|
|
// Found list, but it doesn't have the frame. Put list back.
|
|
SetPropTableFrames(aPresContext, frameList, aPropID);
|
|
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;
|
|
}
|
|
|
|
/**
|
|
* 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)
|
|
{
|
|
NS_PRECONDITION(nsnull != aFromChild, "null pointer");
|
|
NS_PRECONDITION(nsnull != aPrevSibling, "pushing first child");
|
|
NS_PRECONDITION(aPrevSibling->GetNextSibling() == aFromChild, "bad prev sibling");
|
|
|
|
// Disconnect aFromChild from its previous sibling
|
|
aPrevSibling->SetNextSibling(nsnull);
|
|
|
|
if (nsnull != GetNextInFlow()) {
|
|
// 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.
|
|
nsContainerFrame* nextInFlow = (nsContainerFrame*)GetNextInFlow();
|
|
// 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);
|
|
}
|
|
nextInFlow->mFrames.InsertFrames(nextInFlow, nsnull, aFromChild);
|
|
}
|
|
else {
|
|
// Add the frames to our overflow list
|
|
SetOverflowFrames(aPresContext, aFromChild);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 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) {
|
|
nsIFrame* prevOverflowFrames = prevInFlow->GetOverflowFrames(aPresContext,
|
|
PR_TRUE);
|
|
if (prevOverflowFrames) {
|
|
NS_ASSERTION(mFrames.IsEmpty(), "bad overflow list");
|
|
// When pushing and pulling frames we need to check for whether any
|
|
// views need to be reparented.
|
|
for (nsIFrame* f = prevOverflowFrames; f; f = f->GetNextSibling()) {
|
|
nsHTMLContainerFrame::ReparentFrameView(aPresContext, f, prevInFlow, this);
|
|
}
|
|
mFrames.InsertFrames(this, nsnull, prevOverflowFrames);
|
|
result = PR_TRUE;
|
|
}
|
|
}
|
|
|
|
// It's also possible that we have an overflow list for ourselves
|
|
nsIFrame* overflowFrames = GetOverflowFrames(aPresContext, PR_TRUE);
|
|
if (overflowFrames) {
|
|
NS_ASSERTION(mFrames.NotEmpty(), "overflow list w/o frames");
|
|
mFrames.AppendFrames(nsnull, overflowFrames);
|
|
result = PR_TRUE;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
nsOverflowContinuationTracker::nsOverflowContinuationTracker(nsPresContext* aPresContext,
|
|
nsContainerFrame* aFrame,
|
|
PRBool aWalkOOFFrames,
|
|
PRBool aSkipOverflowContainerChildren)
|
|
: mOverflowContList(nsnull),
|
|
mPrevOverflowCont(nsnull),
|
|
mSentry(nsnull),
|
|
mParent(aFrame),
|
|
mSkipOverflowContainerChildren(aSkipOverflowContainerChildren),
|
|
mWalkOOFFrames(aWalkOOFFrames)
|
|
{
|
|
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();
|
|
}
|
|
}
|
|
if (!mOverflowContList) {
|
|
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();
|
|
}
|
|
while (cur && (!(cur->GetStateBits() & NS_FRAME_OUT_OF_FLOW)
|
|
== mWalkOOFFrames)) {
|
|
mPrevOverflowCont = cur;
|
|
cur = cur->GetNextSibling();
|
|
}
|
|
}
|
|
if (cur) {
|
|
mSentry = cur->GetPrevInFlow();
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Helper function to step forward through the overflow continuations list.
|
|
* 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.
|
|
*/
|
|
void
|
|
nsOverflowContinuationTracker::StepForward()
|
|
{
|
|
NS_PRECONDITION(mOverflowContList, "null list");
|
|
|
|
// Step forward
|
|
if (mPrevOverflowCont) {
|
|
mPrevOverflowCont = mPrevOverflowCont->GetNextSibling();
|
|
}
|
|
else {
|
|
mPrevOverflowCont = mOverflowContList->FirstChild();
|
|
}
|
|
|
|
// Skip over oof or non-oof frames as appropriate
|
|
if (mSkipOverflowContainerChildren) {
|
|
nsIFrame* cur = mPrevOverflowCont->GetNextSibling();
|
|
while (cur && (!(cur->GetStateBits() & NS_FRAME_OUT_OF_FLOW)
|
|
== mWalkOOFFrames)) {
|
|
mPrevOverflowCont = cur;
|
|
cur = cur->GetNextSibling();
|
|
}
|
|
}
|
|
|
|
// Set up the sentry
|
|
mSentry = (mPrevOverflowCont->GetNextSibling())
|
|
? mPrevOverflowCont->GetNextSibling()->GetPrevInFlow()
|
|
: nsnull;
|
|
}
|
|
|
|
nsresult
|
|
nsOverflowContinuationTracker::Insert(nsIFrame* aOverflowCont,
|
|
nsReflowStatus& aReflowStatus)
|
|
{
|
|
NS_PRECONDITION(aOverflowCont, "null frame pointer");
|
|
NS_PRECONDITION(!mSkipOverflowContainerChildren || mWalkOOFFrames ==
|
|
!!(aOverflowCont->GetStateBits() & NS_FRAME_OUT_OF_FLOW),
|
|
"shouldn't insert frame that doesn't match walker type");
|
|
NS_PRECONDITION(aOverflowCont->GetPrevInFlow(),
|
|
"overflow containers must have a prev-in-flow");
|
|
nsresult rv = NS_OK;
|
|
PRBool convertedToOverflowContainer = PR_FALSE;
|
|
nsPresContext* presContext = aOverflowCont->PresContext();
|
|
if (!mSentry || aOverflowCont != mSentry->GetNextInFlow()) {
|
|
// Not in our list, so we need to add it
|
|
if (aOverflowCont->GetStateBits() & NS_FRAME_IS_OVERFLOW_CONTAINER) {
|
|
// aOverflowCont is in some other overflow container list,
|
|
// steal it first
|
|
NS_ASSERTION(!(mOverflowContList &&
|
|
mOverflowContList->ContainsFrame(aOverflowCont)),
|
|
"overflow containers out of order");
|
|
rv = static_cast<nsContainerFrame*>(aOverflowCont->GetParent())
|
|
->StealFrame(presContext, aOverflowCont);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
else {
|
|
aOverflowCont->AddStateBits(NS_FRAME_IS_OVERFLOW_CONTAINER);
|
|
convertedToOverflowContainer = PR_TRUE;
|
|
}
|
|
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();
|
|
NS_ASSERTION(mPrevOverflowCont == aOverflowCont ||
|
|
(mSkipOverflowContainerChildren &&
|
|
(mPrevOverflowCont->GetStateBits() & NS_FRAME_OUT_OF_FLOW) !=
|
|
(aOverflowCont->GetStateBits() & NS_FRAME_OUT_OF_FLOW)),
|
|
"OverflowContTracker in unexpected state");
|
|
|
|
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);
|
|
}
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
void
|
|
nsOverflowContinuationTracker::Finish(nsIFrame* aChild)
|
|
{
|
|
NS_PRECONDITION(aChild, "null ptr");
|
|
NS_PRECONDITION(aChild->GetNextInFlow(),
|
|
"supposed to call Finish *before* deleting next-in-flow!");
|
|
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// Debugging
|
|
|
|
#ifdef NS_DEBUG
|
|
NS_IMETHODIMP
|
|
nsContainerFrame::List(FILE* out, PRInt32 aIndent) const
|
|
{
|
|
IndentBy(out, aIndent);
|
|
ListTag(out);
|
|
#ifdef DEBUG_waterson
|
|
fprintf(out, " [parent=%p]", static_cast<void*>(mParent));
|
|
#endif
|
|
if (HasView()) {
|
|
fprintf(out, " [view=%p]", static_cast<void*>(GetView()));
|
|
}
|
|
if (nsnull != mNextSibling) {
|
|
fprintf(out, " next=%p", static_cast<void*>(mNextSibling));
|
|
}
|
|
if (nsnull != GetPrevContinuation()) {
|
|
fprintf(out, " prev-continuation=%p", static_cast<void*>(GetPrevContinuation()));
|
|
}
|
|
if (nsnull != GetNextContinuation()) {
|
|
fprintf(out, " next-continuation=%p", static_cast<void*>(GetNextContinuation()));
|
|
}
|
|
fprintf(out, " {%d,%d,%d,%d}", mRect.x, mRect.y, mRect.width, mRect.height);
|
|
if (0 != mState) {
|
|
fprintf(out, " [state=%08x]", mState);
|
|
}
|
|
fprintf(out, " [content=%p]", static_cast<void*>(mContent));
|
|
nsContainerFrame* f = const_cast<nsContainerFrame*>(this);
|
|
if (f->GetStateBits() & NS_FRAME_OUTSIDE_CHILDREN) {
|
|
nsRect overflowArea = f->GetOverflowRect();
|
|
fprintf(out, " [overflow=%d,%d,%d,%d]", overflowArea.x, overflowArea.y,
|
|
overflowArea.width, overflowArea.height);
|
|
}
|
|
fprintf(out, " [sc=%p]", static_cast<void*>(mStyleContext));
|
|
nsIAtom* pseudoTag = mStyleContext->GetPseudoType();
|
|
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
|
|
nsIFrameDebug* frameDebug;
|
|
if (NS_SUCCEEDED(kid->QueryInterface(NS_GET_IID(nsIFrameDebug), (void**)&frameDebug))) {
|
|
frameDebug->List(out, aIndent + 1);
|
|
}
|
|
kid = kid->GetNextSibling();
|
|
}
|
|
IndentBy(out, aIndent);
|
|
fputs(">\n", out);
|
|
}
|
|
listName = GetAdditionalChildListName(listIndex++);
|
|
} while(nsnull != listName);
|
|
|
|
if (!outputOneList) {
|
|
fputs("<>\n", out);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
#endif
|