/* -*- 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); } 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; } 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", aReflowState.availableWidth, aReflowState.availableHeight, aReflowState.reason)); nsresult rv = NS_OK; // 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(); // 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; PRBool isAbsoluteChild; // See if it's for an absolutely positioned child frame aReflowState.reflowCommand->GetChildListName(listName); isAbsoluteChild = nsLayoutAtoms::absoluteList == listName; NS_IF_RELEASE(listName); if (isAbsoluteChild) { nsIReflowCommand::ReflowType type; aReflowState.reflowCommand->GetType(type); // 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); // 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"); } } } } 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); } // 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 nscoord floaterYMost; mSpaceManager->YMost(floaterYMost); 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) { for (nsIFrame* f = mAbsoluteFrames; nsnull != f; f->GetNextSibling(f)) { nsRect rect; f->GetRect(rect); 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_BLOCK_DOCUMENT_ROOT & 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; } cf->Init(aPresContext, mContent, aParent, mContentParent, aStyleContext); 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) { for (nsIFrame* absoluteFrame = mAbsoluteFrames; nsnull != absoluteFrame; absoluteFrame->GetNextSibling(absoluteFrame)) { PRBool placeFrame = PR_FALSE; #if 0 PRBool reflowFrame = PR_FALSE; #else PRBool reflowFrame = PR_TRUE; #endif 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); #if 0 // 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 { #endif // We need to place the frame if the left-offset or the top-offset are // auto or a percentage if ((eStyleUnit_Coord != position->mOffset.GetLeftUnit()) || (eStyleUnit_Coord != position->mOffset.GetTopUnit())) { 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; } #if 0 } #endif if (placeFrame || reflowFrame) { // Get the rect for the absolutely positioned element nsRect rect; ComputeAbsoluteFrameBounds(aPresContext, absoluteFrame, aReflowState, position, rect); 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); } } void nsAreaFrame::ComputeAbsoluteFrameBounds(nsIPresContext& aPresContext, nsIFrame* aFrame, 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 nsPoint offset(0, 0); if ((eStyleUnit_Auto == aPosition->mOffset.GetLeftUnit()) || (eStyleUnit_Auto == aPosition->mOffset.GetTopUnit())) { // 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); } } // left-offset if (eStyleUnit_Auto == aPosition->mOffset.GetLeftUnit()) { // Use the current x-offset of the anchor frame translated into our // coordinate space aRect.x = offset.x; } else if (eStyleUnit_Coord == aPosition->mOffset.GetLeftUnit()) { nsStyleCoord coord; aRect.x = aPosition->mOffset.GetLeft(coord).GetCoordValue(); } else { NS_ASSERTION(eStyleUnit_Percent == aPosition->mOffset.GetLeftUnit(), "unexpected offset type"); // Percentage values refer to the width of the containing block. If the // width is unconstrained then just use 0 if (NS_UNCONSTRAINEDSIZE == aReflowState.availableWidth) { aRect.x = 0; } else { nsStyleCoord coord; aRect.x = (nscoord)((float)aReflowState.availableWidth * aPosition->mOffset.GetLeft(coord).GetPercentValue()); } } // top-offset if (eStyleUnit_Auto == aPosition->mOffset.GetTopUnit()) { // Use the current y-offset of the anchor frame translated into our // coordinate space aRect.y = offset.y; } else if (eStyleUnit_Coord == aPosition->mOffset.GetTopUnit()) { nsStyleCoord coord; aRect.y = aPosition->mOffset.GetTop(coord).GetCoordValue(); } else { NS_ASSERTION(eStyleUnit_Percent == aPosition->mOffset.GetTopUnit(), "unexpected offset type"); // Percentage values refer to the height of the containing block. If the // height is unconstrained then interpret it like 'auto' if (NS_UNCONSTRAINEDSIZE == aReflowState.availableHeight) { aRect.y = offset.y; } else { nsStyleCoord coord; aRect.y = (nscoord)((float)aReflowState.availableHeight * aPosition->mOffset.GetTop(coord).GetPercentValue()); } } // XXX We aren't properly handling 'auto' width and height // XXX The width/height represent the size of the padding area only, and not // the frame size... // width if (eStyleUnit_Auto == aPosition->mWidth.GetUnit()) { // Use the right-edge of the containing block aRect.width = aReflowState.availableWidth - aRect.x; } 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 if (NS_UNCONSTRAINEDSIZE == aReflowState.availableWidth) { aRect.width = NS_UNCONSTRAINEDSIZE; } else { aRect.width = (nscoord)((float)aReflowState.availableWidth * 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 if (NS_UNCONSTRAINEDSIZE == aReflowState.availableHeight) { aRect.height = NS_UNCONSTRAINEDSIZE; } else { aRect.height = (nscoord)((float)aReflowState.availableHeight * 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; }