162 lines
5.2 KiB
C++
162 lines
5.2 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
*
|
|
* The contents of this file are subject to the Netscape Public License
|
|
* Version 1.0 (the "NPL"); you may not use this file except in
|
|
* compliance with the NPL. You may obtain a copy of the NPL at
|
|
* http://www.mozilla.org/NPL/
|
|
*
|
|
* Software distributed under the NPL is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
|
|
* for the specific language governing rights and limitations under the
|
|
* NPL.
|
|
*
|
|
* The Initial Developer of this code under the NPL is Netscape
|
|
* Communications Corporation. Portions created by Netscape are
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
|
|
* Reserved.
|
|
*/
|
|
#include "nsAbsoluteFrame.h"
|
|
#include "nsBodyFrame.h"
|
|
#include "nsIAbsoluteItems.h"
|
|
#include "nsIStyleContext.h"
|
|
#include "nsStyleConsts.h"
|
|
#include "nsHTMLIIDs.h"
|
|
|
|
nsresult
|
|
nsAbsoluteFrame::NewFrame(nsIFrame** aInstancePtrResult,
|
|
nsIContent* aContent,
|
|
nsIFrame* aParent,
|
|
nsIFrame* aAbsoluteFrame)
|
|
{
|
|
NS_PRECONDITION(nsnull != aInstancePtrResult, "null ptr");
|
|
if (nsnull == aInstancePtrResult) {
|
|
return NS_ERROR_NULL_POINTER;
|
|
}
|
|
nsIFrame* it = new nsAbsoluteFrame(aContent, aParent, aAbsoluteFrame);
|
|
if (nsnull == it) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
*aInstancePtrResult = it;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsAbsoluteFrame::nsAbsoluteFrame(nsIContent* aContent,
|
|
nsIFrame* aParent,
|
|
nsIFrame* aAbsoluteFrame)
|
|
: nsFrame(aContent, aParent)
|
|
{
|
|
mFrame = aAbsoluteFrame;
|
|
}
|
|
|
|
nsAbsoluteFrame::~nsAbsoluteFrame()
|
|
{
|
|
}
|
|
|
|
NS_IMETHODIMP nsAbsoluteFrame::Reflow(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus)
|
|
{
|
|
if (eReflowReason_Initial == aReflowState.reason) {
|
|
// By this point we expect to have been told which absolute frame we're
|
|
// associated with
|
|
NS_ASSERTION(nsnull != mFrame, "no absolute frame");
|
|
|
|
// Get the containing block
|
|
nsIFrame* containingBlock = GetContainingBlock();
|
|
NS_ASSERTION(nsnull != containingBlock, "no initial containing block");
|
|
|
|
// Query for its nsIAbsoluteItems interface
|
|
nsIAbsoluteItems* absoluteItemContainer;
|
|
containingBlock->QueryInterface(kIAbsoluteItemsIID, (void**)&absoluteItemContainer);
|
|
|
|
// Notify it that there's a new absolutely positioned frame, passing it the
|
|
// anchor frame
|
|
NS_ASSERTION(nsnull != absoluteItemContainer, "no nsIAbsoluteItems support");
|
|
absoluteItemContainer->AddAbsoluteItem(this);
|
|
}
|
|
|
|
// Return our desired size as (0, 0)
|
|
return nsFrame::Reflow(aPresContext, aDesiredSize, aReflowState, aStatus);
|
|
}
|
|
|
|
NS_IMETHODIMP nsAbsoluteFrame::ContentChanged(nsIPresContext* aPresContext,
|
|
nsIContent* aChild,
|
|
nsISupports* aSubContent)
|
|
{
|
|
NS_ASSERTION(mContent == aChild, "bad content-changed target");
|
|
|
|
// Forward the notification to the absolutely positioned frame
|
|
if (nsnull != mFrame) {
|
|
return mFrame->ContentChanged(aPresContext, aChild, aSubContent);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP nsAbsoluteFrame::AttributeChanged(nsIPresContext* aPresContext,
|
|
nsIContent* aChild,
|
|
nsIAtom* aAttribute,
|
|
PRInt32 aHint)
|
|
{
|
|
NS_ASSERTION(mContent == aChild, "bad content-changed target");
|
|
|
|
// Forward the notification to the absolutely positioned frame
|
|
if (nsnull != mFrame) {
|
|
return mFrame->AttributeChanged(aPresContext, aChild, aAttribute, aHint);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
nsIFrame* nsAbsoluteFrame::GetContainingBlock() const
|
|
{
|
|
// Look for a containing frame that is absolutely positioned. If we don't
|
|
// find one then use the initial containg block which is the BODY
|
|
nsIFrame* lastFrame = (nsIFrame*)this;
|
|
nsIFrame* result;
|
|
|
|
GetContentParent(result);
|
|
while (nsnull != result) {
|
|
const nsStylePosition* position;
|
|
|
|
// Get the style data
|
|
result->GetStyleData(eStyleStruct_Position, (const nsStyleStruct*&)position);
|
|
|
|
if (position->mPosition == NS_STYLE_POSITION_ABSOLUTE) {
|
|
break;
|
|
}
|
|
|
|
// Get the next contentual parent
|
|
lastFrame = result;
|
|
result->GetContentParent(result);
|
|
}
|
|
|
|
if (nsnull == result) {
|
|
// Walk back down the tree until we find a frame that supports nsIAnchoredItems
|
|
// XXX This is pretty yucky, but there isn't currently a better way to do
|
|
// this...
|
|
lastFrame->FirstChild(result);
|
|
|
|
while (nsnull != result) {
|
|
nsIAbsoluteItems* interface;
|
|
if (NS_OK == result->QueryInterface(kIAbsoluteItemsIID, (void**)&interface)) {
|
|
break;
|
|
}
|
|
|
|
result->FirstChild(result);
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
NS_IMETHODIMP nsAbsoluteFrame::ListTag(FILE* out) const
|
|
{
|
|
fputs("*absolute", out);
|
|
PRInt32 contentIndex;
|
|
GetContentIndex(contentIndex);
|
|
fprintf(out, "(%d)@%p", contentIndex, this);
|
|
return NS_OK;
|
|
}
|