1998-12-18 20:54:54 +03:00
|
|
|
/* -*- 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 "nsAreaFrame.h"
|
|
|
|
#include "nsIContent.h"
|
|
|
|
#include "nsIHTMLContent.h"
|
|
|
|
#include "nsIReflowCommand.h"
|
|
|
|
#include "nsIStyleContext.h"
|
|
|
|
#include "nsStyleConsts.h"
|
|
|
|
#include "nsIPresContext.h"
|
|
|
|
#include "nsIPresShell.h"
|
|
|
|
#include "nsIViewManager.h"
|
|
|
|
#include "nsIDeviceContext.h"
|
|
|
|
#include "nsSpaceManager.h"
|
|
|
|
#include "nsHTMLAtoms.h"
|
|
|
|
#include "nsIView.h"
|
|
|
|
#include "nsViewsCID.h"
|
|
|
|
#include "nsHTMLIIDs.h"
|
|
|
|
#include "nsIWebShell.h"
|
|
|
|
#include "nsHTMLValue.h"
|
|
|
|
#include "nsHTMLParts.h"
|
|
|
|
#include "nsLayoutAtoms.h"
|
|
|
|
|
|
|
|
static NS_DEFINE_IID(kIWebShellIID, NS_IWEB_SHELL_IID);
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
NS_NewAreaFrame(nsIFrame*& aResult, PRUint32 aFlags)
|
|
|
|
{
|
|
|
|
nsAreaFrame* it = new nsAreaFrame;
|
|
|
|
if (nsnull == it) {
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
}
|
|
|
|
it->SetFlags(aFlags);
|
|
|
|
aResult = it;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsAreaFrame::nsAreaFrame()
|
|
|
|
{
|
|
|
|
mSpaceManager = new nsSpaceManager(this);
|
|
|
|
NS_ADDREF(mSpaceManager);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsAreaFrame::~nsAreaFrame()
|
|
|
|
{
|
|
|
|
NS_RELEASE(mSpaceManager);
|
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// nsIFrame
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsAreaFrame::DeleteFrame(nsIPresContext& aPresContext)
|
|
|
|
{
|
|
|
|
DeleteFrameList(aPresContext, &mAbsoluteFrames);
|
|
|
|
return nsBlockFrame::DeleteFrame(aPresContext);
|
|
|
|
}
|
|
|
|
|
1998-12-29 06:38:16 +03:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsAreaFrame::SetInitialChildList(nsIPresContext& aPresContext,
|
|
|
|
nsIAtom* aListName,
|
|
|
|
nsIFrame* aChildList)
|
|
|
|
{
|
|
|
|
nsresult rv;
|
|
|
|
|
|
|
|
if (nsLayoutAtoms::absoluteList == aListName) {
|
|
|
|
mAbsoluteFrames = aChildList;
|
|
|
|
rv = NS_OK;
|
|
|
|
} else {
|
|
|
|
rv = nsBlockFrame::SetInitialChildList(aPresContext, aListName, aChildList);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
1998-12-18 20:54:54 +03:00
|
|
|
NS_IMETHODIMP
|
|
|
|
nsAreaFrame::GetAdditionalChildListName(PRInt32 aIndex,
|
|
|
|
nsIAtom*& aListName) const
|
|
|
|
{
|
|
|
|
if (aIndex <= NS_BLOCK_FRAME_LAST_LIST_INDEX) {
|
|
|
|
return nsBlockFrame::GetAdditionalChildListName(aIndex, aListName);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsIAtom* atom = nsnull;
|
|
|
|
if (NS_AREA_FRAME_ABSOLUTE_LIST_INDEX == aIndex) {
|
|
|
|
atom = nsLayoutAtoms::absoluteList;
|
|
|
|
NS_ADDREF(atom);
|
|
|
|
}
|
|
|
|
aListName = atom;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsAreaFrame::FirstChild(nsIAtom* aListName, nsIFrame*& aFirstChild) const
|
|
|
|
{
|
|
|
|
if (aListName == nsLayoutAtoms::absoluteList) {
|
|
|
|
aFirstChild = mAbsoluteFrames;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nsBlockFrame::FirstChild(aListName, aFirstChild);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef NS_DEBUG
|
|
|
|
void
|
|
|
|
nsAreaFrame::BandData::ComputeAvailSpaceRect()
|
|
|
|
{
|
|
|
|
nsBandTrapezoid* trapezoid = data;
|
|
|
|
|
|
|
|
if (count > 1) {
|
|
|
|
// If there's more than one trapezoid that means there are floaters
|
|
|
|
PRInt32 i;
|
|
|
|
|
|
|
|
// Stop when we get to space occupied by a right floater, or when we've
|
|
|
|
// looked at every trapezoid and none are right floaters
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
nsBandTrapezoid* trapezoid = &data[i];
|
|
|
|
if (trapezoid->state != nsBandTrapezoid::Available) {
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
if (nsBandTrapezoid::OccupiedMultiple == trapezoid->state) {
|
|
|
|
PRInt32 j, numFrames = trapezoid->frames->Count();
|
|
|
|
NS_ASSERTION(numFrames > 0, "bad trapezoid frame list");
|
|
|
|
for (j = 0; j < numFrames; j++) {
|
|
|
|
nsIFrame* f = (nsIFrame*)trapezoid->frames->ElementAt(j);
|
|
|
|
f->GetStyleData(eStyleStruct_Display,
|
|
|
|
(const nsStyleStruct*&)display);
|
|
|
|
if (NS_STYLE_FLOAT_RIGHT == display->mFloats) {
|
|
|
|
goto foundRightFloater;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
trapezoid->frame->GetStyleData(eStyleStruct_Display,
|
|
|
|
(const nsStyleStruct*&)display);
|
|
|
|
if (NS_STYLE_FLOAT_RIGHT == display->mFloats) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
foundRightFloater:
|
|
|
|
|
|
|
|
if (i > 0) {
|
|
|
|
trapezoid = &data[i - 1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nsBandTrapezoid::Available == trapezoid->state) {
|
|
|
|
// The trapezoid is available
|
|
|
|
trapezoid->GetRect(availSpace);
|
|
|
|
} else {
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
|
|
|
|
// The trapezoid is occupied. That means there's no available space
|
|
|
|
trapezoid->GetRect(availSpace);
|
|
|
|
|
|
|
|
// XXX Better handle the case of multiple frames
|
|
|
|
if (nsBandTrapezoid::Occupied == trapezoid->state) {
|
|
|
|
trapezoid->frame->GetStyleData(eStyleStruct_Display,
|
|
|
|
(const nsStyleStruct*&)display);
|
|
|
|
if (NS_STYLE_FLOAT_LEFT == display->mFloats) {
|
|
|
|
availSpace.x = availSpace.XMost();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
availSpace.width = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsAreaFrame::Paint(nsIPresContext& aPresContext,
|
|
|
|
nsIRenderingContext& aRenderingContext,
|
|
|
|
const nsRect& aDirtyRect,
|
|
|
|
nsFramePaintLayer aWhichLayer)
|
|
|
|
{
|
|
|
|
// Note: all absolutely positioned elements have views so we don't
|
|
|
|
// need to worry about painting them
|
|
|
|
nsresult rv = nsBlockFrame::Paint(aPresContext, aRenderingContext,
|
|
|
|
aDirtyRect, aWhichLayer);
|
|
|
|
|
|
|
|
if ((eFramePaintLayer_Overlay == aWhichLayer) &&
|
|
|
|
nsIFrame::GetShowFrameBorders()) {
|
|
|
|
// Render the bands in the spacemanager
|
|
|
|
BandData band;
|
|
|
|
nsISpaceManager* sm = mSpaceManager;
|
|
|
|
nscoord y = 0;
|
|
|
|
while (y < mRect.height) {
|
|
|
|
sm->GetBandData(y, nsSize(mRect.width, mRect.height - y), band);
|
|
|
|
band.ComputeAvailSpaceRect();
|
|
|
|
|
|
|
|
// Render a box and a diagonal line through the band
|
|
|
|
aRenderingContext.SetColor(NS_RGB(0,255,0));
|
|
|
|
aRenderingContext.DrawRect(0, band.availSpace.y,
|
|
|
|
mRect.width, band.availSpace.height);
|
|
|
|
aRenderingContext.DrawLine(0, band.availSpace.y,
|
|
|
|
mRect.width, band.availSpace.YMost());
|
|
|
|
|
|
|
|
// Render boxes and opposite diagonal lines around the
|
|
|
|
// unavailable parts of the band.
|
|
|
|
PRInt32 i;
|
|
|
|
for (i = 0; i < band.count; i++) {
|
|
|
|
nsBandTrapezoid* trapezoid = &band.data[i];
|
|
|
|
if (nsBandTrapezoid::Available != trapezoid->state) {
|
|
|
|
nsRect r;
|
|
|
|
trapezoid->GetRect(r);
|
|
|
|
if (nsBandTrapezoid::OccupiedMultiple == trapezoid->state) {
|
|
|
|
aRenderingContext.SetColor(NS_RGB(0,255,128));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
aRenderingContext.SetColor(NS_RGB(128,255,0));
|
|
|
|
}
|
|
|
|
aRenderingContext.DrawRect(r);
|
|
|
|
aRenderingContext.DrawLine(r.x, r.YMost(), r.XMost(), r.y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
y = band.availSpace.YMost();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsAreaFrame::Reflow(nsIPresContext& aPresContext,
|
|
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
|
|
const nsHTMLReflowState& aReflowState,
|
|
|
|
nsReflowStatus& aStatus)
|
|
|
|
{
|
|
|
|
NS_FRAME_TRACE(NS_FRAME_TRACE_CALLS,
|
|
|
|
("enter nsAreaFrame::Reflow: maxSize=%d,%d reason=%d",
|
1999-01-06 02:31:18 +03:00
|
|
|
aReflowState.availableWidth,
|
|
|
|
aReflowState.availableHeight,
|
1998-12-18 20:54:54 +03:00
|
|
|
aReflowState.reason));
|
|
|
|
|
1998-12-29 06:38:16 +03:00
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
1998-12-18 20:54:54 +03:00
|
|
|
// Make a copy of the reflow state so we can set the space manager
|
|
|
|
nsHTMLReflowState reflowState(aReflowState);
|
|
|
|
reflowState.spaceManager = mSpaceManager;
|
|
|
|
|
|
|
|
// Clear the spacemanager's regions.
|
|
|
|
mSpaceManager->ClearRegions();
|
|
|
|
|
1998-12-29 06:38:16 +03:00
|
|
|
// See if the reflow command is for an absolutely positioned frame
|
|
|
|
PRBool wasHandled = PR_FALSE;
|
|
|
|
if (eReflowReason_Incremental == aReflowState.reason) {
|
|
|
|
nsIFrame* targetFrame;
|
|
|
|
|
|
|
|
aReflowState.reflowCommand->GetTarget(targetFrame);
|
|
|
|
if (this == targetFrame) {
|
|
|
|
nsIAtom* listName;
|
1999-01-07 08:46:26 +03:00
|
|
|
PRBool isAbsoluteChild;
|
1998-12-29 06:38:16 +03:00
|
|
|
|
1999-01-07 08:46:26 +03:00
|
|
|
// See if it's for an absolutely positioned child frame
|
1998-12-29 06:38:16 +03:00
|
|
|
aReflowState.reflowCommand->GetChildListName(listName);
|
1999-01-07 08:46:26 +03:00
|
|
|
isAbsoluteChild = nsLayoutAtoms::absoluteList == listName;
|
|
|
|
NS_IF_RELEASE(listName);
|
|
|
|
|
|
|
|
if (isAbsoluteChild) {
|
1998-12-29 06:38:16 +03:00
|
|
|
nsIReflowCommand::ReflowType type;
|
|
|
|
|
|
|
|
aReflowState.reflowCommand->GetType(type);
|
|
|
|
|
1999-01-07 08:46:26 +03:00
|
|
|
// Handle each specific type
|
|
|
|
if (nsIReflowCommand::FrameAppended == type) {
|
|
|
|
// Add the frames to our list of absolutely position frames
|
|
|
|
nsIFrame* childFrames;
|
|
|
|
aReflowState.reflowCommand->GetChildFrame(childFrames);
|
|
|
|
NS_ASSERTION(nsnull != childFrames, "null child list");
|
|
|
|
AddAbsoluteFrame(childFrames);
|
|
|
|
|
|
|
|
// Indicate we handled the reflow command
|
|
|
|
wasHandled = PR_TRUE;
|
|
|
|
|
|
|
|
} else if (nsIReflowCommand::FrameRemoved == type) {
|
|
|
|
// Get the new frame
|
|
|
|
nsIFrame* childFrame;
|
|
|
|
aReflowState.reflowCommand->GetChildFrame(childFrame);
|
|
|
|
|
|
|
|
// Find the frame in our list of absolutely positioned children
|
|
|
|
// and remove it
|
|
|
|
if (mAbsoluteFrames == childFrame) {
|
|
|
|
childFrame->GetNextSibling(mAbsoluteFrames);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
nsIFrame* prevSibling = nsnull;
|
|
|
|
for (nsIFrame* f = mAbsoluteFrames; nsnull != f; f->GetNextSibling(f)) {
|
|
|
|
if (f == childFrame) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
prevSibling = f;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_ASSERTION(nsnull != prevSibling, "didn't find frame");
|
|
|
|
nsIFrame* nextSibling;
|
|
|
|
childFrame->GetNextSibling(nextSibling);
|
|
|
|
prevSibling->SetNextSibling(nextSibling);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now go ahead and delete the child frame
|
|
|
|
childFrame->DeleteFrame(aPresContext);
|
|
|
|
|
|
|
|
// XXX We don't need to reflow all the absolutely positioned
|
|
|
|
// frames. Compute the desired size and exit...
|
|
|
|
wasHandled = PR_TRUE;
|
|
|
|
|
|
|
|
} else if (nsIReflowCommand::FrameInserted == type) {
|
|
|
|
// Get the new frame
|
|
|
|
nsIFrame* childFrame;
|
|
|
|
aReflowState.reflowCommand->GetChildFrame(childFrame);
|
1998-12-29 06:38:16 +03:00
|
|
|
|
1999-01-07 08:46:26 +03:00
|
|
|
// Get the previous sibling
|
|
|
|
nsIFrame* prevSibling;
|
|
|
|
aReflowState.reflowCommand->GetPrevSiblingFrame(prevSibling);
|
|
|
|
|
|
|
|
// Insert the frame
|
|
|
|
if (nsnull == prevSibling) {
|
|
|
|
mAbsoluteFrames = childFrame;
|
|
|
|
} else {
|
|
|
|
nsIFrame* nextSibling;
|
|
|
|
|
|
|
|
prevSibling->GetNextSibling(nextSibling);
|
|
|
|
prevSibling->SetNextSibling(childFrame);
|
|
|
|
childFrame->SetNextSibling(nextSibling);
|
|
|
|
}
|
|
|
|
wasHandled = PR_TRUE;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
NS_ASSERTION(PR_FALSE, "unexpected reflow type");
|
|
|
|
}
|
1998-12-29 06:38:16 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!wasHandled) {
|
|
|
|
// XXX We need to peek at incremental reflow commands and see if the next
|
|
|
|
// frame is one of the absolutely positioned frames...
|
|
|
|
rv = nsBlockFrame::Reflow(aPresContext, aDesiredSize, reflowState, aStatus);
|
|
|
|
}
|
1998-12-18 20:54:54 +03:00
|
|
|
|
|
|
|
// Reflow any absolutely positioned frames that need reflowing
|
|
|
|
// XXX We shouldn't really be doing this for all incremental reflow commands
|
|
|
|
ReflowAbsoluteItems(aPresContext, reflowState);
|
|
|
|
|
|
|
|
// Compute our desired size. Take into account any floaters when computing the
|
|
|
|
// height
|
1999-01-04 07:54:33 +03:00
|
|
|
nscoord floaterYMost;
|
|
|
|
mSpaceManager->YMost(floaterYMost);
|
1998-12-18 20:54:54 +03:00
|
|
|
if (floaterYMost > 0) {
|
|
|
|
// What we need to check for is if the bottom most floater extends below
|
|
|
|
// the content area of the desired size
|
|
|
|
nsMargin borderPadding;
|
|
|
|
nscoord contentYMost;
|
|
|
|
|
|
|
|
nsHTMLReflowState::ComputeBorderPaddingFor(this, aReflowState.parentReflowState,
|
|
|
|
borderPadding);
|
|
|
|
contentYMost = aDesiredSize.height - borderPadding.bottom;
|
|
|
|
|
|
|
|
if (floaterYMost > contentYMost) {
|
|
|
|
aDesiredSize.height += floaterYMost - contentYMost;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Also take into account absolutely positioned elements depending on
|
|
|
|
// the overflow policy
|
|
|
|
const nsStyleDisplay* display = (const nsStyleDisplay*)
|
|
|
|
mStyleContext->GetStyleData(eStyleStruct_Display);
|
|
|
|
|
|
|
|
if (NS_STYLE_OVERFLOW_HIDDEN != display->mOverflow) {
|
1998-12-29 06:38:16 +03:00
|
|
|
for (nsIFrame* f = mAbsoluteFrames; nsnull != f; f->GetNextSibling(f)) {
|
|
|
|
nsRect rect;
|
1998-12-18 20:54:54 +03:00
|
|
|
|
1998-12-29 06:38:16 +03:00
|
|
|
f->GetRect(rect);
|
1998-12-18 20:54:54 +03:00
|
|
|
nscoord xmost = rect.XMost();
|
|
|
|
nscoord ymost = rect.YMost();
|
|
|
|
if (xmost > aDesiredSize.width) {
|
|
|
|
aDesiredSize.width = xmost;
|
|
|
|
}
|
|
|
|
if (ymost > aDesiredSize.height) {
|
|
|
|
aDesiredSize.height = ymost;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// XXX This code is really temporary; the lower level frame
|
|
|
|
// classes need to contribute to the area that needs damage
|
|
|
|
// repair. This class should only worry about damage repairing
|
|
|
|
// it's border+padding area.
|
|
|
|
nsRect damageArea(0, 0, 0, 0);
|
|
|
|
|
|
|
|
// Decide how much to repaint based on the reflow type.
|
|
|
|
// Note: we don't have to handle the initial reflow case and the
|
|
|
|
// resize reflow case, because they're handled by the scroll frame
|
|
|
|
if (eReflowReason_Incremental == aReflowState.reason) {
|
|
|
|
nsIReflowCommand::ReflowType reflowType;
|
|
|
|
aReflowState.reflowCommand->GetType(reflowType);
|
|
|
|
|
|
|
|
// For append reflow commands that target the flowed frames just
|
|
|
|
// repaint the newly added part of the frame.
|
|
|
|
if (nsIReflowCommand::FrameAppended == reflowType) {
|
|
|
|
//this blows. we're repainting everything, but we have no choice
|
|
|
|
//since we don't know how we got here. see the XXX above for a
|
|
|
|
//real fix.
|
|
|
|
#if 1
|
|
|
|
damageArea.y = 0;
|
|
|
|
damageArea.height = aDesiredSize.height;
|
|
|
|
damageArea.width = aDesiredSize.width;
|
|
|
|
#else
|
|
|
|
// It's an append reflow command
|
|
|
|
damageArea.y = mRect.YMost();
|
|
|
|
damageArea.width = aDesiredSize.width;
|
|
|
|
damageArea.height = aDesiredSize.height - mRect.height;
|
|
|
|
if ((damageArea.height < 0) ||
|
|
|
|
(aDesiredSize.height == mRect.height)) {
|
|
|
|
// Since we don't know what changed, assume it all changed.
|
|
|
|
damageArea.y = 0;
|
|
|
|
damageArea.height = aDesiredSize.height;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
// Ideally the frame that is the target of the reflow command
|
|
|
|
// (or its parent frame) would generate a damage rect, but
|
|
|
|
// since none of the frame classes know how to do this then
|
|
|
|
// for the time being just repaint the entire frame
|
|
|
|
damageArea.width = aDesiredSize.width;
|
|
|
|
// If the new height is smaller than the old height then make
|
|
|
|
// sure we erase whatever used to be displayed
|
|
|
|
damageArea.height = PR_MAX(aDesiredSize.height, mRect.height);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If this is really the body, force a repaint of the damage area
|
|
|
|
if ((NS_BODY_THE_BODY & mFlags) && !damageArea.IsEmpty()) {
|
|
|
|
Invalidate(damageArea);
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_METHOD
|
|
|
|
nsAreaFrame::CreateContinuingFrame(nsIPresContext& aPresContext,
|
|
|
|
nsIFrame* aParent,
|
|
|
|
nsIStyleContext* aStyleContext,
|
|
|
|
nsIFrame*& aContinuingFrame)
|
|
|
|
{
|
|
|
|
nsAreaFrame* cf = new nsAreaFrame;
|
|
|
|
if (nsnull == cf) {
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
}
|
1998-12-29 06:38:16 +03:00
|
|
|
cf->Init(aPresContext, mContent, aParent, mContentParent, aStyleContext);
|
1998-12-18 20:54:54 +03:00
|
|
|
cf->SetFlags(mFlags);
|
|
|
|
cf->AppendToFlow(this);
|
|
|
|
aContinuingFrame = cf;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Helper functions
|
|
|
|
|
|
|
|
// Add the frame to the end of the child list
|
|
|
|
void nsAreaFrame::AddAbsoluteFrame(nsIFrame* aFrame)
|
|
|
|
{
|
|
|
|
if (nsnull == mAbsoluteFrames) {
|
|
|
|
mAbsoluteFrames = aFrame;
|
|
|
|
} else {
|
|
|
|
nsIFrame* lastChild = LastFrame(mAbsoluteFrames);
|
|
|
|
|
|
|
|
lastChild->SetNextSibling(aFrame);
|
|
|
|
aFrame->SetNextSibling(nsnull);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Called at the end of the Reflow() member function so we can process
|
|
|
|
// any abolutely positioned items that need to be reflowed
|
|
|
|
void
|
|
|
|
nsAreaFrame::ReflowAbsoluteItems(nsIPresContext& aPresContext,
|
|
|
|
const nsHTMLReflowState& aReflowState)
|
|
|
|
{
|
1998-12-29 06:38:16 +03:00
|
|
|
for (nsIFrame* absoluteFrame = mAbsoluteFrames;
|
|
|
|
nsnull != absoluteFrame; absoluteFrame->GetNextSibling(absoluteFrame)) {
|
|
|
|
|
1998-12-18 20:54:54 +03:00
|
|
|
PRBool placeFrame = PR_FALSE;
|
1998-12-29 06:38:16 +03:00
|
|
|
#if 0
|
1998-12-18 20:54:54 +03:00
|
|
|
PRBool reflowFrame = PR_FALSE;
|
1998-12-29 06:38:16 +03:00
|
|
|
#else
|
|
|
|
PRBool reflowFrame = PR_TRUE;
|
|
|
|
#endif
|
1998-12-18 20:54:54 +03:00
|
|
|
nsReflowReason reflowReason = eReflowReason_Resize;
|
|
|
|
|
|
|
|
// Get its style information
|
|
|
|
const nsStyleDisplay* display;
|
|
|
|
const nsStylePosition* position;
|
|
|
|
|
|
|
|
absoluteFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct*&)display);
|
|
|
|
absoluteFrame->GetStyleData(eStyleStruct_Position, (const nsStyleStruct*&)position);
|
|
|
|
|
1998-12-29 06:38:16 +03:00
|
|
|
#if 0
|
1998-12-18 20:54:54 +03:00
|
|
|
// See whether the frame is a newly added frame
|
|
|
|
if (!IsAbsoluteFrame(absoluteFrame)) {
|
|
|
|
// The absolutely position item hasn't yet been added to our child list
|
|
|
|
absoluteFrame->SetGeometricParent(this);
|
|
|
|
|
|
|
|
// Add the absolutely positioned frame to the end of the child list
|
|
|
|
AddAbsoluteFrame(absoluteFrame);
|
|
|
|
|
|
|
|
// See whether this is the frame's initial reflow
|
|
|
|
nsFrameState frameState;
|
|
|
|
absoluteFrame->GetFrameState(frameState);
|
|
|
|
if (frameState & NS_FRAME_FIRST_REFLOW) {
|
|
|
|
reflowReason = eReflowReason_Initial;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We need to place and reflow the absolutely positioned frame
|
|
|
|
placeFrame = reflowFrame = PR_TRUE;
|
|
|
|
|
|
|
|
} else {
|
1998-12-29 06:38:16 +03:00
|
|
|
#endif
|
1998-12-18 20:54:54 +03:00
|
|
|
// We need to place the frame if the left-offset or the top-offset are
|
|
|
|
// auto or a percentage
|
|
|
|
if ((eStyleUnit_Coord != position->mLeftOffset.GetUnit()) ||
|
|
|
|
(eStyleUnit_Coord != position->mTopOffset.GetUnit())) {
|
|
|
|
placeFrame = PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We need to reflow the frame if its width or its height is auto or
|
|
|
|
// a percentage
|
|
|
|
if ((eStyleUnit_Coord != position->mWidth.GetUnit()) ||
|
|
|
|
(eStyleUnit_Coord != position->mHeight.GetUnit())) {
|
|
|
|
reflowFrame = PR_TRUE;
|
|
|
|
}
|
1998-12-29 06:38:16 +03:00
|
|
|
#if 0
|
1998-12-18 20:54:54 +03:00
|
|
|
}
|
1998-12-29 06:38:16 +03:00
|
|
|
#endif
|
1998-12-18 20:54:54 +03:00
|
|
|
|
|
|
|
if (placeFrame || reflowFrame) {
|
|
|
|
// Get the rect for the absolutely positioned element
|
|
|
|
nsRect rect;
|
1998-12-29 06:38:16 +03:00
|
|
|
ComputeAbsoluteFrameBounds(aPresContext, absoluteFrame, aReflowState,
|
|
|
|
position, rect);
|
1998-12-18 20:54:54 +03:00
|
|
|
|
|
|
|
nsIHTMLReflow* htmlReflow;
|
|
|
|
if (NS_OK == absoluteFrame->QueryInterface(kIHTMLReflowIID, (void**)&htmlReflow)) {
|
|
|
|
htmlReflow->WillReflow(aPresContext);
|
|
|
|
absoluteFrame->MoveTo(rect.x, rect.y);
|
|
|
|
|
|
|
|
if (reflowFrame) {
|
|
|
|
// Resize reflow the absolutely positioned element
|
|
|
|
nsSize availSize(rect.width, rect.height);
|
|
|
|
|
|
|
|
if (NS_STYLE_OVERFLOW_VISIBLE == display->mOverflow) {
|
|
|
|
// Don't constrain the height since the container should be enlarged
|
|
|
|
// to contain overflowing frames
|
|
|
|
availSize.height = NS_UNCONSTRAINEDSIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsHTMLReflowMetrics desiredSize(nsnull);
|
|
|
|
nsHTMLReflowState reflowState(aPresContext, absoluteFrame,
|
|
|
|
aReflowState, availSize,
|
|
|
|
reflowReason);
|
|
|
|
nsReflowStatus status;
|
|
|
|
htmlReflow->Reflow(aPresContext, desiredSize, reflowState, status);
|
|
|
|
|
|
|
|
// Figure out what size to actually use. If we let the child choose its
|
|
|
|
// size, then use what the child requested. Otherwise, use the value
|
|
|
|
// specified in the style information
|
|
|
|
if ((eStyleUnit_Auto == position->mWidth.GetUnit()) ||
|
|
|
|
((desiredSize.width > availSize.width) &&
|
|
|
|
(NS_STYLE_OVERFLOW_VISIBLE == display->mOverflow))) {
|
|
|
|
rect.width = desiredSize.width;
|
|
|
|
}
|
|
|
|
if ((eStyleUnit_Auto == position->mHeight.GetUnit()) ||
|
|
|
|
(NS_UNCONSTRAINEDSIZE == rect.height) ||
|
|
|
|
((desiredSize.height > rect.height) &&
|
|
|
|
(NS_STYLE_OVERFLOW_VISIBLE == display->mOverflow))) {
|
|
|
|
rect.height = desiredSize.height;
|
|
|
|
}
|
|
|
|
absoluteFrame->SetRect(rect);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Translate aPoint from aFrameFrom's coordinate space to our coordinate space
|
|
|
|
void nsAreaFrame::TranslatePoint(nsIFrame* aFrameFrom, nsPoint& aPoint) const
|
|
|
|
{
|
|
|
|
nsIFrame* parent;
|
|
|
|
|
|
|
|
aFrameFrom->GetGeometricParent(parent);
|
|
|
|
while ((nsnull != parent) && (parent != (nsIFrame*)this)) {
|
|
|
|
nsPoint origin;
|
|
|
|
|
|
|
|
parent->GetOrigin(origin);
|
|
|
|
aPoint += origin;
|
|
|
|
parent->GetGeometricParent(parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1998-12-29 06:38:16 +03:00
|
|
|
void nsAreaFrame::ComputeAbsoluteFrameBounds(nsIPresContext& aPresContext,
|
|
|
|
nsIFrame* aFrame,
|
1998-12-18 20:54:54 +03:00
|
|
|
const nsHTMLReflowState& aReflowState,
|
|
|
|
const nsStylePosition* aPosition,
|
|
|
|
nsRect& aRect) const
|
|
|
|
{
|
|
|
|
// Compute the offset and size of the view based on the position properties,
|
|
|
|
// and the inner rect of the containing block (which we get from the reflow
|
|
|
|
// state)
|
|
|
|
//
|
|
|
|
// If either the left or top are 'auto' then get the offset of the anchor
|
|
|
|
// frame from this frame
|
1998-12-29 06:38:16 +03:00
|
|
|
nsPoint offset(0, 0);
|
1998-12-18 20:54:54 +03:00
|
|
|
if ((eStyleUnit_Auto == aPosition->mLeftOffset.GetUnit()) ||
|
|
|
|
(eStyleUnit_Auto == aPosition->mTopOffset.GetUnit())) {
|
1998-12-29 06:38:16 +03:00
|
|
|
// Get the placeholder frame
|
|
|
|
nsIFrame* placeholderFrame;
|
|
|
|
nsIPresShell* presShell = aPresContext.GetShell();
|
|
|
|
|
|
|
|
presShell->GetPlaceholderFrameFor(aFrame, placeholderFrame);
|
|
|
|
NS_RELEASE(presShell);
|
|
|
|
NS_ASSERTION(nsnull != placeholderFrame, "no placeholder frame");
|
|
|
|
if (nsnull != placeholderFrame) {
|
|
|
|
placeholderFrame->GetOrigin(offset);
|
|
|
|
TranslatePoint(placeholderFrame, offset);
|
|
|
|
}
|
1998-12-18 20:54:54 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// left-offset
|
|
|
|
if (eStyleUnit_Auto == aPosition->mLeftOffset.GetUnit()) {
|
|
|
|
// Use the current x-offset of the anchor frame translated into our
|
|
|
|
// coordinate space
|
|
|
|
aRect.x = offset.x;
|
|
|
|
} else if (eStyleUnit_Coord == aPosition->mLeftOffset.GetUnit()) {
|
|
|
|
aRect.x = aPosition->mLeftOffset.GetCoordValue();
|
|
|
|
} else {
|
|
|
|
NS_ASSERTION(eStyleUnit_Percent == aPosition->mLeftOffset.GetUnit(),
|
|
|
|
"unexpected offset type");
|
|
|
|
// Percentage values refer to the width of the containing block. If the
|
|
|
|
// width is unconstrained then just use 0
|
1999-01-06 02:31:18 +03:00
|
|
|
if (NS_UNCONSTRAINEDSIZE == aReflowState.availableWidth) {
|
1998-12-18 20:54:54 +03:00
|
|
|
aRect.x = 0;
|
|
|
|
} else {
|
1999-01-06 02:31:18 +03:00
|
|
|
aRect.x = (nscoord)((float)aReflowState.availableWidth *
|
1998-12-18 20:54:54 +03:00
|
|
|
aPosition->mLeftOffset.GetPercentValue());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// top-offset
|
|
|
|
if (eStyleUnit_Auto == aPosition->mTopOffset.GetUnit()) {
|
|
|
|
// Use the current y-offset of the anchor frame translated into our
|
|
|
|
// coordinate space
|
|
|
|
aRect.y = offset.y;
|
|
|
|
} else if (eStyleUnit_Coord == aPosition->mTopOffset.GetUnit()) {
|
|
|
|
aRect.y = aPosition->mTopOffset.GetCoordValue();
|
|
|
|
} else {
|
|
|
|
NS_ASSERTION(eStyleUnit_Percent == aPosition->mTopOffset.GetUnit(),
|
|
|
|
"unexpected offset type");
|
|
|
|
// Percentage values refer to the height of the containing block. If the
|
|
|
|
// height is unconstrained then interpret it like 'auto'
|
1999-01-06 02:31:18 +03:00
|
|
|
if (NS_UNCONSTRAINEDSIZE == aReflowState.availableHeight) {
|
1998-12-18 20:54:54 +03:00
|
|
|
aRect.y = offset.y;
|
|
|
|
} else {
|
1999-01-06 02:31:18 +03:00
|
|
|
aRect.y = (nscoord)((float)aReflowState.availableHeight *
|
1998-12-18 20:54:54 +03:00
|
|
|
aPosition->mTopOffset.GetPercentValue());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// XXX We aren't properly handling 'auto' width and height
|
1998-12-22 08:02:46 +03:00
|
|
|
// XXX The width/height represent the size of the padding area only, and not
|
1998-12-18 20:54:54 +03:00
|
|
|
// the frame size...
|
|
|
|
|
|
|
|
// width
|
|
|
|
if (eStyleUnit_Auto == aPosition->mWidth.GetUnit()) {
|
|
|
|
// Use the right-edge of the containing block
|
1999-01-06 02:31:18 +03:00
|
|
|
aRect.width = aReflowState.availableWidth - aRect.x;
|
1998-12-18 20:54:54 +03:00
|
|
|
} else if (eStyleUnit_Coord == aPosition->mWidth.GetUnit()) {
|
|
|
|
aRect.width = aPosition->mWidth.GetCoordValue();
|
|
|
|
} else {
|
|
|
|
NS_ASSERTION(eStyleUnit_Percent == aPosition->mWidth.GetUnit(),
|
|
|
|
"unexpected width type");
|
|
|
|
// Percentage values refer to the width of the containing block
|
1999-01-06 02:31:18 +03:00
|
|
|
if (NS_UNCONSTRAINEDSIZE == aReflowState.availableWidth) {
|
1998-12-18 20:54:54 +03:00
|
|
|
aRect.width = NS_UNCONSTRAINEDSIZE;
|
|
|
|
} else {
|
1999-01-06 02:31:18 +03:00
|
|
|
aRect.width = (nscoord)((float)aReflowState.availableWidth *
|
1998-12-18 20:54:54 +03:00
|
|
|
aPosition->mWidth.GetPercentValue());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// height
|
|
|
|
if (eStyleUnit_Auto == aPosition->mHeight.GetUnit()) {
|
|
|
|
// Allow it to be as high as it wants
|
|
|
|
aRect.height = NS_UNCONSTRAINEDSIZE;
|
|
|
|
} else if (eStyleUnit_Coord == aPosition->mHeight.GetUnit()) {
|
|
|
|
aRect.height = aPosition->mHeight.GetCoordValue();
|
|
|
|
} else {
|
|
|
|
NS_ASSERTION(eStyleUnit_Percent == aPosition->mHeight.GetUnit(),
|
|
|
|
"unexpected height type");
|
|
|
|
// Percentage values refer to the height of the containing block. If the
|
|
|
|
// height is unconstrained, then interpret it like 'auto' and make the
|
|
|
|
// height unconstrained
|
1999-01-06 02:31:18 +03:00
|
|
|
if (NS_UNCONSTRAINEDSIZE == aReflowState.availableHeight) {
|
1998-12-18 20:54:54 +03:00
|
|
|
aRect.height = NS_UNCONSTRAINEDSIZE;
|
|
|
|
} else {
|
1999-01-06 02:31:18 +03:00
|
|
|
aRect.height = (nscoord)((float)aReflowState.availableHeight *
|
1998-12-18 20:54:54 +03:00
|
|
|
aPosition->mHeight.GetPercentValue());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Diagnostics
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsAreaFrame::GetFrameName(nsString& aResult) const
|
|
|
|
{
|
|
|
|
return MakeFrameName("Area", aResult);
|
|
|
|
}
|
|
|
|
|
|
|
|
// XXX The base class implementation should handle listing all of
|
|
|
|
// the additional named child lists...
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsAreaFrame::List(FILE* out, PRInt32 aIndent, nsIListFilter* aFilter) const
|
|
|
|
{
|
|
|
|
nsresult rv = nsBlockFrame::List(out, aIndent, aFilter);
|
|
|
|
|
|
|
|
// Output absolutely positioned frames
|
|
|
|
if (nsnull != mAbsoluteFrames) {
|
|
|
|
for (PRInt32 i = aIndent; --i >= 0; ) fputs(" ", out);
|
|
|
|
fprintf(out, "absolute-items <\n");
|
|
|
|
}
|
|
|
|
nsIFrame* f = mAbsoluteFrames;
|
|
|
|
while (nsnull != f) {
|
|
|
|
f->List(out, aIndent+1, aFilter);
|
|
|
|
f->GetNextSibling(f);
|
|
|
|
}
|
|
|
|
for (PRInt32 i = aIndent; --i >= 0; ) fputs(" ", out);
|
|
|
|
fputs(">\n", out);
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
|