1998-12-01 19:13:49 +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 "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/NPL/
|
|
|
|
*
|
|
|
|
* 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 Communicator client code.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is Netscape Communications
|
|
|
|
* Corporation. Portions created by Netscape are Copyright (C) 1998
|
|
|
|
* Netscape Communications Corporation. All Rights Reserved.
|
|
|
|
*/
|
|
|
|
#include "nsBlockReflowContext.h"
|
|
|
|
#include "nsLineLayout.h"
|
|
|
|
#include "nsHTMLIIDs.h"
|
|
|
|
#include "nsISpaceManager.h"
|
|
|
|
#include "nsIFontMetrics.h"
|
|
|
|
#include "nsIPresContext.h"
|
|
|
|
#include "nsIStyleContext.h"
|
|
|
|
#include "nsHTMLContainerFrame.h"
|
|
|
|
|
1999-02-12 20:45:58 +03:00
|
|
|
#ifdef NS_DEBUG
|
|
|
|
#define NOISY_SPECULATIVE_TOP_MARGIN
|
|
|
|
#else
|
|
|
|
#undef NOISY_SPECULATIVE_TOP_MARGIN
|
|
|
|
#endif
|
|
|
|
|
1998-12-01 19:13:49 +03:00
|
|
|
nsBlockReflowContext::nsBlockReflowContext(nsIPresContext& aPresContext,
|
1999-03-05 07:22:11 +03:00
|
|
|
const nsHTMLReflowState& aParentRS,
|
|
|
|
PRBool aComputeMaxElementSize)
|
1998-12-01 19:13:49 +03:00
|
|
|
: mPresContext(aPresContext),
|
|
|
|
mOuterReflowState(aParentRS),
|
1999-03-05 07:22:11 +03:00
|
|
|
mMetrics(aComputeMaxElementSize ? &mMaxElementSize : nsnull)
|
1998-12-01 19:13:49 +03:00
|
|
|
{
|
|
|
|
mCompactMarginWidth = 0;
|
1999-03-05 07:22:11 +03:00
|
|
|
mStyleSpacing = nsnull;
|
1998-12-01 19:13:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
1998-12-12 20:59:30 +03:00
|
|
|
nsBlockReflowContext::ReflowBlock(nsIFrame* aFrame,
|
|
|
|
const nsRect& aSpace,
|
1999-02-12 20:45:58 +03:00
|
|
|
#ifdef SPECULATIVE_TOP_MARGIN
|
|
|
|
PRBool aApplyTopMargin,
|
|
|
|
nscoord aPrevBottomMargin,
|
|
|
|
#endif
|
1998-12-12 20:59:30 +03:00
|
|
|
PRBool aIsAdjacentWithTop,
|
1999-01-16 23:58:17 +03:00
|
|
|
nsMargin& aComputedOffsets,
|
1998-12-05 19:01:11 +03:00
|
|
|
nsReflowStatus& aFrameReflowStatus)
|
1998-12-01 19:13:49 +03:00
|
|
|
{
|
1998-12-05 19:01:11 +03:00
|
|
|
nsresult rv = NS_OK;
|
1998-12-01 19:13:49 +03:00
|
|
|
mFrame = aFrame;
|
|
|
|
mSpace = aSpace;
|
|
|
|
|
1999-02-12 20:45:58 +03:00
|
|
|
#ifdef SPECULATIVE_TOP_MARGIN
|
|
|
|
#ifdef NOISY_SPECULATIVE_TOP_MARGIN
|
|
|
|
PRIntn pass = 0;
|
|
|
|
#endif
|
|
|
|
mSpeculativeTopMargin = 0;
|
|
|
|
if (aApplyTopMargin) {
|
|
|
|
// Compute a "speculative" collapsed top-margin value. Its
|
|
|
|
// speculative because we don't yet have the
|
|
|
|
// carried-out-top-margin value so the final collapsed value isn't
|
|
|
|
// knowable yet. We want to pre-apply the collapsed top-margin
|
|
|
|
// value so that when a block is flowing around floaters that we
|
|
|
|
// don't have to reflow it twice (we can't just slide it down when
|
|
|
|
// its flowing around a floater after discovering the final top
|
|
|
|
// margin value).
|
|
|
|
mSpeculativeTopMargin = MaxMargin(mMargin.top, aPrevBottomMargin);
|
|
|
|
}
|
|
|
|
again:
|
|
|
|
#else
|
|
|
|
mSpeculativeTopMargin = 0;
|
|
|
|
#endif
|
|
|
|
|
1998-12-01 19:13:49 +03:00
|
|
|
// Get reflow reason set correctly. It's possible that a child was
|
|
|
|
// created and then it was decided that it could not be reflowed
|
|
|
|
// (for example, a block frame that isn't at the start of a
|
|
|
|
// line). In this case the reason will be wrong so we need to check
|
|
|
|
// the frame state.
|
|
|
|
nsReflowReason reason = eReflowReason_Resize;
|
|
|
|
nsFrameState state;
|
1999-02-10 07:17:06 +03:00
|
|
|
aFrame->GetFrameState(&state);
|
1998-12-01 19:13:49 +03:00
|
|
|
if (NS_FRAME_FIRST_REFLOW & state) {
|
|
|
|
reason = eReflowReason_Initial;
|
|
|
|
}
|
1999-03-05 07:22:11 +03:00
|
|
|
else if (mNextRCFrame == aFrame) {
|
1998-12-01 19:13:49 +03:00
|
|
|
reason = eReflowReason_Incremental;
|
|
|
|
// Make sure we only incrementally reflow once
|
1999-03-05 07:22:11 +03:00
|
|
|
mNextRCFrame = nsnull;
|
1998-12-01 19:13:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Setup reflow state for reflowing the frame
|
1999-03-05 07:22:11 +03:00
|
|
|
nsSize availSpace(aSpace.width, aSpace.height);
|
|
|
|
nsHTMLReflowState reflowState(mPresContext, mOuterReflowState, aFrame,
|
|
|
|
availSpace, reason);
|
1999-01-16 23:58:17 +03:00
|
|
|
aComputedOffsets = reflowState.computedOffsets;
|
1998-12-01 19:13:49 +03:00
|
|
|
reflowState.lineLayout = nsnull;
|
|
|
|
reflowState.mCompactMarginWidth = mCompactMarginWidth;
|
1998-12-12 20:59:30 +03:00
|
|
|
if (!aIsAdjacentWithTop) {
|
|
|
|
reflowState.isTopOfPage = PR_FALSE; // make sure this is cleared
|
|
|
|
}
|
1999-03-05 07:22:11 +03:00
|
|
|
|
|
|
|
// Compute x/y coordinate where reflow will begin. Use the rules
|
|
|
|
// from 10.3.3 to determine what to apply. At this point in the
|
|
|
|
// reflow auto left/right margins will have a zero value.
|
|
|
|
mMargin = reflowState.computedMargin;
|
|
|
|
mStyleSpacing = reflowState.mStyleSpacing;
|
|
|
|
#ifdef DEBUG_kipp
|
|
|
|
NS_ASSERTION((mMargin.left > -200000) &&
|
|
|
|
(mMargin.left < 200000), "oy");
|
|
|
|
NS_ASSERTION((mMargin.right > -200000) &&
|
|
|
|
(mMargin.right < 200000), "oy");
|
|
|
|
#endif
|
|
|
|
nscoord x = aSpace.x + mMargin.left;
|
|
|
|
nscoord y = aSpace.y + mSpeculativeTopMargin;
|
|
|
|
mX = x;
|
|
|
|
mY = y;
|
1998-12-01 19:13:49 +03:00
|
|
|
|
|
|
|
// Let frame know that we are reflowing it
|
|
|
|
nsIHTMLReflow* htmlReflow;
|
1998-12-05 19:01:11 +03:00
|
|
|
rv = aFrame->QueryInterface(kIHTMLReflowIID, (void**)&htmlReflow);
|
|
|
|
if (NS_FAILED(rv)) {
|
1998-12-01 19:13:49 +03:00
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
htmlReflow->WillReflow(mPresContext);
|
|
|
|
|
1999-03-08 22:25:03 +03:00
|
|
|
#ifdef DEBUG
|
1999-03-20 02:06:20 +03:00
|
|
|
mMetrics.width = nscoord(0xdeadbeef);
|
|
|
|
mMetrics.height = nscoord(0xdeadbeef);
|
|
|
|
mMetrics.ascent = nscoord(0xdeadbeef);
|
|
|
|
mMetrics.descent = nscoord(0xdeadbeef);
|
1999-03-08 22:25:03 +03:00
|
|
|
if (nsnull != mMetrics.maxElementSize) {
|
|
|
|
mMetrics.maxElementSize->width = nscoord(0xdeadbeef);
|
|
|
|
mMetrics.maxElementSize->height = nscoord(0xdeadbeef);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1998-12-01 19:13:49 +03:00
|
|
|
// Adjust spacemanager coordinate system for the frame. The
|
|
|
|
// spacemanager coordinates are <b>inside</b> the callers
|
|
|
|
// border+padding, but the x/y coordinates are not (recall that
|
|
|
|
// frame coordinates are relative to the parents origin and that the
|
|
|
|
// parents border/padding is <b>inside</b> the parent
|
|
|
|
// frame. Therefore we have to subtract out the parents
|
|
|
|
// border+padding before translating.
|
1999-03-05 07:22:11 +03:00
|
|
|
nscoord tx = x - mOuterReflowState.mComputedBorderPadding.left;
|
|
|
|
nscoord ty = y - mOuterReflowState.mComputedBorderPadding.top;
|
1998-12-01 19:13:49 +03:00
|
|
|
mOuterReflowState.spaceManager->Translate(tx, ty);
|
1998-12-05 19:01:11 +03:00
|
|
|
rv = htmlReflow->Reflow(mPresContext, mMetrics, reflowState,
|
|
|
|
aFrameReflowStatus);
|
1998-12-01 19:13:49 +03:00
|
|
|
mOuterReflowState.spaceManager->Translate(-tx, -ty);
|
|
|
|
|
1999-03-05 07:22:11 +03:00
|
|
|
#ifdef DEBUG_kipp
|
|
|
|
NS_ASSERTION((mMetrics.width > -200000) && (mMetrics.width < 200000), "oy");
|
|
|
|
NS_ASSERTION((mMetrics.height > -200000) && (mMetrics.height < 200000), "oy");
|
|
|
|
#endif
|
1999-03-08 22:25:03 +03:00
|
|
|
#ifdef DEBUG
|
|
|
|
if ((nsnull != mMetrics.maxElementSize) &&
|
|
|
|
((nscoord(0xdeadbeef) == mMetrics.maxElementSize->width) ||
|
|
|
|
(nscoord(0xdeadbeef) == mMetrics.maxElementSize->height))) {
|
|
|
|
printf("nsBlockReflowContext: ");
|
|
|
|
nsFrame::ListTag(stdout, aFrame);
|
|
|
|
printf(" didn't set max-element-size!\n");
|
|
|
|
mMetrics.maxElementSize->width = 0;
|
|
|
|
mMetrics.maxElementSize->height = 0;
|
|
|
|
}
|
1999-03-20 02:06:20 +03:00
|
|
|
if ((mMetrics.width == nscoord(0xdeadbeef)) ||
|
|
|
|
(mMetrics.height == nscoord(0xdeadbeef)) ||
|
|
|
|
(mMetrics.ascent == nscoord(0xdeadbeef)) ||
|
|
|
|
(mMetrics.descent == nscoord(0xdeadbeef))) {
|
|
|
|
printf("nsBlockReflowContext: ");
|
|
|
|
nsFrame::ListTag(stdout, aFrame);
|
|
|
|
printf(" didn't set whad %d,%d,%d,%d!\n", mMetrics.width, mMetrics.height,
|
|
|
|
mMetrics.ascent, mMetrics.descent);
|
|
|
|
}
|
1999-03-08 22:25:03 +03:00
|
|
|
#endif
|
1999-03-05 07:22:11 +03:00
|
|
|
|
1999-02-10 07:17:06 +03:00
|
|
|
aFrame->GetFrameState(&state);
|
1998-12-01 19:13:49 +03:00
|
|
|
if (0 == (NS_FRAME_OUTSIDE_CHILDREN & state)) {
|
|
|
|
// Provide combined area for child that doesn't have any
|
|
|
|
mMetrics.mCombinedArea.x = 0;
|
|
|
|
mMetrics.mCombinedArea.y = 0;
|
|
|
|
mMetrics.mCombinedArea.width = mMetrics.width;
|
|
|
|
mMetrics.mCombinedArea.height = mMetrics.height;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now that frame has been reflowed at least one time make sure that
|
|
|
|
// the NS_FRAME_FIRST_REFLOW bit is cleared so that never give it an
|
|
|
|
// initial reflow reason again.
|
|
|
|
if (eReflowReason_Initial == reason) {
|
|
|
|
aFrame->SetFrameState(state & ~NS_FRAME_FIRST_REFLOW);
|
|
|
|
}
|
|
|
|
|
1998-12-05 19:01:11 +03:00
|
|
|
if (!NS_INLINE_IS_BREAK_BEFORE(aFrameReflowStatus)) {
|
1998-12-01 19:13:49 +03:00
|
|
|
// If frame is complete and has a next-in-flow, we need to delete
|
|
|
|
// them now. Do not do this when a break-before is signaled because
|
|
|
|
// the frame is going to get reflowed again (and may end up wanting
|
|
|
|
// a next-in-flow where it ends up).
|
1998-12-05 19:01:11 +03:00
|
|
|
if (NS_FRAME_IS_COMPLETE(aFrameReflowStatus)) {
|
1998-12-01 19:13:49 +03:00
|
|
|
nsIFrame* kidNextInFlow;
|
1999-02-24 07:48:08 +03:00
|
|
|
aFrame->GetNextInFlow(&kidNextInFlow);
|
1998-12-01 19:13:49 +03:00
|
|
|
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)
|
|
|
|
/* XXX promote DeleteChildsNextInFlow to nsIFrame to elminate this cast */
|
|
|
|
nsHTMLContainerFrame* parent;
|
1999-02-10 04:36:30 +03:00
|
|
|
aFrame->GetParent((nsIFrame**)&parent);
|
1998-12-01 19:13:49 +03:00
|
|
|
parent->DeleteChildsNextInFlow(mPresContext, aFrame);
|
|
|
|
}
|
|
|
|
}
|
1999-02-12 20:45:58 +03:00
|
|
|
|
|
|
|
#ifdef SPECULATIVE_TOP_MARGIN
|
|
|
|
if (aApplyTopMargin) {
|
|
|
|
// Compute final collapsed margin value
|
|
|
|
CollapseMargins(mMargin, mMetrics.mCarriedOutTopMargin,
|
|
|
|
mMetrics.mCarriedOutBottomMargin,
|
|
|
|
mMetrics.height, aPrevBottomMargin,
|
|
|
|
mTopMargin, mBottomMargin);
|
|
|
|
|
|
|
|
if (mSpeculativeTopMargin != mTopMargin) {
|
|
|
|
// We lost our speculative gamble. Use new computed margin
|
|
|
|
// value as the speculative value and try the reflow again.
|
|
|
|
#ifdef NOISY_SPECULATIVE_TOP_MARGIN
|
|
|
|
nsFrame::ListTag(stdout, mOuterReflowState.frame);
|
|
|
|
printf(": reflowing again: ");
|
|
|
|
nsFrame::ListTag(stdout, aFrame);
|
|
|
|
printf(" guess=%d actual=%d [pass %d]\n",
|
|
|
|
mSpeculativeTopMargin, mTopMargin, pass);
|
|
|
|
pass++;
|
|
|
|
#endif
|
|
|
|
mSpeculativeTopMargin = mTopMargin;
|
|
|
|
goto again;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
1998-12-01 19:13:49 +03:00
|
|
|
}
|
|
|
|
|
1998-12-05 19:01:11 +03:00
|
|
|
return rv;
|
1998-12-01 19:13:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The CSS2 spec, section 8.3.1 defines margin collapsing to apply to
|
|
|
|
* vertical margins of block boxes. And the spec also indicates that
|
|
|
|
* this should be done for two or more adjacent vertical margins. It
|
|
|
|
* also indicates that the margins collapse between boxes that are
|
|
|
|
* next to each other or nested.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
nsBlockReflowContext::CollapseMargins(const nsMargin& aMargin,
|
|
|
|
nscoord aCarriedOutTopMargin,
|
|
|
|
nscoord aCarriedOutBottomMargin,
|
|
|
|
nscoord aFrameHeight,
|
|
|
|
nscoord aPrevBottomMargin,
|
|
|
|
nscoord& aTopMarginResult,
|
|
|
|
nscoord& aBottomMarginResult)
|
|
|
|
{
|
|
|
|
// Compute the collapsed top margin value. The top margin value is a
|
|
|
|
// 3 way margin collapse. First we collapse the carried out top
|
|
|
|
// margin with the block frames top margin (this is a CSS2 "nested"
|
|
|
|
// collapse). Then we collapse that value with the previous bottom
|
|
|
|
// margin (because the collapsed margin is adjacent to the previous
|
|
|
|
// bottom margin).
|
|
|
|
nscoord carriedTopMargin = aCarriedOutTopMargin;
|
|
|
|
nscoord topMargin = aMargin.top;
|
|
|
|
topMargin = MaxMargin(topMargin, carriedTopMargin);
|
|
|
|
topMargin = MaxMargin(topMargin, aPrevBottomMargin);
|
|
|
|
|
|
|
|
// Compute the collapsed bottom margin value. The bottom margin
|
|
|
|
// value is a 2 way margin collapse. Collapse the carried out bottom
|
|
|
|
// margin with the block frames bottom margin (this is a CSS2
|
|
|
|
// "nested" collapse).
|
|
|
|
nscoord carriedBottomMargin = aCarriedOutBottomMargin;
|
|
|
|
nscoord bottomMargin = aMargin.bottom;
|
|
|
|
bottomMargin = MaxMargin(bottomMargin, carriedBottomMargin);
|
|
|
|
|
|
|
|
// If the block line is empty then we collapse the top and bottom
|
|
|
|
// margin values (because the margins are adjacent).
|
|
|
|
if (0 == aFrameHeight) {
|
|
|
|
nscoord collapsedMargin = MaxMargin(topMargin, bottomMargin);
|
|
|
|
topMargin = 0;
|
|
|
|
bottomMargin = collapsedMargin;
|
|
|
|
}
|
|
|
|
|
|
|
|
aTopMarginResult = topMargin;
|
|
|
|
aBottomMarginResult = bottomMargin;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Attempt to place the block frame within the available space. If
|
|
|
|
* it fits, apply horizontal positioning (CSS 10.3.3), collapse
|
|
|
|
* margins (CSS2 8.3.1). Also apply relative positioning.
|
|
|
|
*/
|
|
|
|
PRBool
|
1999-02-12 20:45:58 +03:00
|
|
|
nsBlockReflowContext::PlaceBlock(PRBool aForceFit,
|
|
|
|
#ifndef SPECULATIVE_TOP_MARGIN
|
|
|
|
PRBool aApplyTopMargin,
|
1998-12-01 19:13:49 +03:00
|
|
|
nscoord aPrevBottomMargin,
|
1999-02-12 20:45:58 +03:00
|
|
|
#endif
|
1999-01-16 23:58:17 +03:00
|
|
|
const nsMargin& aComputedOffsets,
|
1998-12-01 19:13:49 +03:00
|
|
|
nsRect& aInFlowBounds,
|
|
|
|
nsRect& aCombinedRect)
|
|
|
|
{
|
1999-02-12 20:45:58 +03:00
|
|
|
#ifndef SPECULATIVE_TOP_MARGIN
|
|
|
|
// Compute final collapsed margin value
|
1998-12-01 19:13:49 +03:00
|
|
|
CollapseMargins(mMargin, mMetrics.mCarriedOutTopMargin,
|
|
|
|
mMetrics.mCarriedOutBottomMargin,
|
|
|
|
mMetrics.height, aPrevBottomMargin,
|
|
|
|
mTopMargin, mBottomMargin);
|
1999-02-12 20:45:58 +03:00
|
|
|
#endif
|
1998-12-01 19:13:49 +03:00
|
|
|
|
|
|
|
// See if the block will fit in the available space
|
|
|
|
PRBool fits;
|
|
|
|
if (0 == mMetrics.height) {
|
|
|
|
// Empty blocks do not have anything special done to them and they
|
|
|
|
// always fit.
|
|
|
|
nsRect r(mX, mY, 0, 0);
|
|
|
|
mFrame->SetRect(r);
|
|
|
|
aInFlowBounds = r;
|
1999-03-08 22:25:03 +03:00
|
|
|
aCombinedRect = mMetrics.mCombinedArea;
|
1999-03-19 00:03:25 +03:00
|
|
|
mTopMargin = 0;
|
|
|
|
mBottomMargin = 0;
|
1998-12-01 19:13:49 +03:00
|
|
|
fits = PR_TRUE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
nscoord x = mX;
|
|
|
|
nscoord y = mY;
|
|
|
|
|
|
|
|
// Apply top margin unless it's going to be carried out.
|
1999-02-12 20:45:58 +03:00
|
|
|
#ifndef SPECULATIVE_TOP_MARGIN
|
1998-12-01 19:13:49 +03:00
|
|
|
if (aApplyTopMargin) {
|
|
|
|
y += mTopMargin;
|
|
|
|
}
|
1999-02-12 20:45:58 +03:00
|
|
|
#endif
|
1998-12-01 19:13:49 +03:00
|
|
|
|
1999-03-05 07:22:11 +03:00
|
|
|
// See if the frame fit. If its the first frame then it always
|
|
|
|
// fits.
|
|
|
|
if (aForceFit || (y + mMetrics.height <= mSpace.YMost())) {
|
|
|
|
fits = PR_TRUE;
|
|
|
|
|
|
|
|
// Get style unit associated with the left and right margins
|
|
|
|
nsStyleUnit leftUnit = mStyleSpacing->mMargin.GetLeftUnit();
|
|
|
|
if (eStyleUnit_Inherit == leftUnit) {
|
|
|
|
leftUnit = GetRealMarginLeftUnit();
|
|
|
|
}
|
|
|
|
nsStyleUnit rightUnit = mStyleSpacing->mMargin.GetRightUnit();
|
|
|
|
if (eStyleUnit_Inherit == rightUnit) {
|
|
|
|
rightUnit = GetRealMarginRightUnit();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply post-reflow horizontal alignment. When a block element
|
|
|
|
// doesn't use it all of the available width then we need to
|
|
|
|
// align it using the text-align property. Note that
|
|
|
|
// block-non-replaced elements will always take up the available
|
|
|
|
// width (counting the margins!) so we shouldn't be handling
|
|
|
|
// them here.
|
|
|
|
if (NS_UNCONSTRAINEDSIZE != mSpace.width) {
|
|
|
|
nscoord remainder = mSpace.XMost() -
|
|
|
|
(x + mMetrics.width + mMargin.right);
|
|
|
|
if (remainder > 0) {
|
|
|
|
// The block frame didn't use all of the available
|
|
|
|
// space. Synthesize margins for its horizontal placement.
|
|
|
|
if (eStyleUnit_Auto == leftUnit) {
|
|
|
|
if (eStyleUnit_Auto == rightUnit) {
|
|
|
|
// When both margins are auto, we center the block
|
|
|
|
x += remainder / 2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// When the left margin is auto we right align the block
|
|
|
|
x += remainder;
|
|
|
|
}
|
1998-12-15 20:59:49 +03:00
|
|
|
}
|
1999-03-05 07:22:11 +03:00
|
|
|
else if (eStyleUnit_Auto != rightUnit) {
|
|
|
|
#if XXX_not_in_css2
|
|
|
|
// When neither margin is auto then text-align applies
|
|
|
|
const nsStyleText* styleText = mOuterReflowState.mStyleText;
|
|
|
|
switch (styleText->mTextAlign) {
|
|
|
|
case NS_STYLE_TEXT_ALIGN_DEFAULT:
|
|
|
|
case NS_STYLE_TEXT_ALIGN_JUSTIFY:
|
|
|
|
if (NS_STYLE_DIRECTION_RTL == mOuterReflowState.mDirection) {
|
|
|
|
// When given a default alignment, and a right-to-left
|
|
|
|
// direction, right align the frame.
|
|
|
|
x += remainder;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case NS_STYLE_TEXT_ALIGN_RIGHT:
|
|
|
|
x += remainder;
|
|
|
|
break;
|
|
|
|
case NS_STYLE_TEXT_ALIGN_CENTER:
|
|
|
|
x += remainder / 2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
// When neither margin is auto then the block is said to
|
|
|
|
// be over constrained, Depending on the direction, choose
|
|
|
|
// which margin to treat as auto.
|
|
|
|
if (NS_STYLE_DIRECTION_RTL ==
|
|
|
|
mOuterReflowState.mStyleDisplay->mDirection) {
|
|
|
|
// The left margin becomes auto
|
1998-12-01 19:13:49 +03:00
|
|
|
x += remainder;
|
|
|
|
}
|
1999-03-05 07:22:11 +03:00
|
|
|
else {
|
|
|
|
// The right margin becomes auto which is a no-op
|
|
|
|
}
|
|
|
|
#endif
|
1998-12-01 19:13:49 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1998-12-17 21:52:45 +03:00
|
|
|
// Update the in-flow bounding box's bounds. Include the margins.
|
|
|
|
aInFlowBounds.SetRect(x, y,
|
|
|
|
mMetrics.width + mMargin.right,
|
|
|
|
mMetrics.height);
|
1998-12-01 19:13:49 +03:00
|
|
|
|
1999-03-05 07:22:11 +03:00
|
|
|
#ifdef DEBUG_kipp
|
|
|
|
NS_ASSERTION((aInFlowBounds.width > -200000) &&
|
|
|
|
(aInFlowBounds.width < 200000), "oy");
|
|
|
|
NS_ASSERTION((aInFlowBounds.height > -200000) &&
|
|
|
|
(aInFlowBounds.height < 200000), "oy");
|
|
|
|
#endif
|
|
|
|
|
1998-12-01 19:13:49 +03:00
|
|
|
// Apply CSS relative positioning to update x,y coordinates
|
|
|
|
const nsStylePosition* stylePos;
|
|
|
|
mFrame->GetStyleData(eStyleStruct_Position,
|
|
|
|
(const nsStyleStruct*&)stylePos);
|
|
|
|
if (NS_STYLE_POSITION_RELATIVE == stylePos->mPosition) {
|
1999-01-22 07:12:03 +03:00
|
|
|
x += aComputedOffsets.left;
|
|
|
|
y += aComputedOffsets.top;
|
1998-12-01 19:13:49 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Compute combined-rect in callers coordinate system. The value
|
|
|
|
// returned in the reflow metrics is relative to the child
|
|
|
|
// frame.
|
|
|
|
aCombinedRect.x = mMetrics.mCombinedArea.x + x;
|
|
|
|
aCombinedRect.y = mMetrics.mCombinedArea.y + y;
|
|
|
|
aCombinedRect.width = mMetrics.mCombinedArea.width;
|
|
|
|
aCombinedRect.height = mMetrics.mCombinedArea.height;
|
|
|
|
|
|
|
|
// Now place the frame
|
|
|
|
mFrame->SetRect(nsRect(x, y, mMetrics.width, mMetrics.height));
|
|
|
|
|
|
|
|
// If the block frame ended up moving then we need to slide
|
|
|
|
// anything inside of it that impacts the space manager
|
|
|
|
// (otherwise the impacted space in the space manager will be
|
|
|
|
// out of sync with where the frames really are).
|
|
|
|
nscoord dx = x - mX;
|
|
|
|
nscoord dy = y - mY;
|
|
|
|
if ((0 != dx) || (0 != dy)) {
|
|
|
|
nsIHTMLReflow* htmlReflow;
|
|
|
|
nsresult rv;
|
|
|
|
rv = mFrame->QueryInterface(kIHTMLReflowIID, (void**)&htmlReflow);
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
// If the child has any floaters that impact the space manager,
|
|
|
|
// slide them now
|
|
|
|
htmlReflow->MoveInSpaceManager(mPresContext,
|
|
|
|
mOuterReflowState.spaceManager,
|
|
|
|
dx, dy);
|
|
|
|
}
|
|
|
|
}
|
1998-12-17 21:52:45 +03:00
|
|
|
|
|
|
|
// Adjust the max-element-size in the metrics to take into
|
|
|
|
// account the margins around the block element. Note that we
|
|
|
|
// use the collapsed top and bottom margin values.
|
1999-03-05 07:22:11 +03:00
|
|
|
if (nsnull != mMetrics.maxElementSize) {
|
1998-12-17 21:52:45 +03:00
|
|
|
nsSize* m = mMetrics.maxElementSize;
|
1999-03-05 07:22:11 +03:00
|
|
|
// Do not allow auto margins to impact the max-element size
|
|
|
|
// since they are springy and don't really count!
|
|
|
|
if (eStyleUnit_Auto != leftUnit) {
|
|
|
|
m->width += mMargin.left;
|
|
|
|
}
|
|
|
|
if (eStyleUnit_Auto != rightUnit) {
|
|
|
|
m->width += mMargin.right;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Margin height should affect the max-element height (since
|
|
|
|
// auto top/bottom margins are always zero)
|
1998-12-17 21:52:45 +03:00
|
|
|
m->height += mTopMargin + mBottomMargin;
|
|
|
|
}
|
1998-12-01 19:13:49 +03:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
fits = PR_FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return fits;
|
|
|
|
}
|
|
|
|
|
1999-03-05 07:22:11 +03:00
|
|
|
// If we have an inherited margin its possible that its auto all the
|
|
|
|
// way up to the top of the tree. If that is the case, we need to know
|
|
|
|
// it.
|
|
|
|
nsStyleUnit
|
|
|
|
nsBlockReflowContext::GetRealMarginLeftUnit()
|
|
|
|
{
|
|
|
|
nsStyleUnit unit = eStyleUnit_Inherit;
|
|
|
|
nsIStyleContext* sc;
|
|
|
|
mFrame->GetStyleContext(&sc);
|
|
|
|
while ((nsnull != sc) && (eStyleUnit_Inherit == unit)) {
|
|
|
|
// Get parent style context
|
|
|
|
nsIStyleContext* psc;
|
|
|
|
psc = sc->GetParent();
|
|
|
|
NS_RELEASE(sc);
|
|
|
|
sc = psc;
|
|
|
|
if (nsnull != sc) {
|
|
|
|
const nsStyleSpacing* spacing = (const nsStyleSpacing*)
|
|
|
|
sc->GetStyleData(eStyleStruct_Spacing);
|
|
|
|
unit = spacing->mMargin.GetLeftUnit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
NS_IF_RELEASE(sc);
|
|
|
|
return unit;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we have an inherited margin its possible that its auto all the
|
|
|
|
// way up to the top of the tree. If that is the case, we need to know
|
|
|
|
// it.
|
|
|
|
nsStyleUnit
|
|
|
|
nsBlockReflowContext::GetRealMarginRightUnit()
|
|
|
|
{
|
|
|
|
nsStyleUnit unit = eStyleUnit_Inherit;
|
|
|
|
nsIStyleContext* sc;
|
|
|
|
mFrame->GetStyleContext(&sc);
|
|
|
|
while ((nsnull != sc) && (eStyleUnit_Inherit == unit)) {
|
|
|
|
// Get parent style context
|
|
|
|
nsIStyleContext* psc;
|
|
|
|
psc = sc->GetParent();
|
|
|
|
NS_RELEASE(sc);
|
|
|
|
sc = psc;
|
|
|
|
if (nsnull != sc) {
|
|
|
|
const nsStyleSpacing* spacing = (const nsStyleSpacing*)
|
|
|
|
sc->GetStyleData(eStyleStruct_Spacing);
|
|
|
|
unit = spacing->mMargin.GetRightUnit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
NS_IF_RELEASE(sc);
|
|
|
|
return unit;
|
|
|
|
}
|