2007-03-22 10:30:00 -07:00
|
|
|
/* -*- 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 Communicator client 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):
|
|
|
|
* Author: Eric D Vaughan <evaughan@netscape.com>
|
|
|
|
* Pierre Phaneuf <pp@ludusdesign.com>
|
|
|
|
*
|
|
|
|
* 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 ***** */
|
|
|
|
|
|
|
|
#include "nsBoxLayoutState.h"
|
|
|
|
#include "nsBox.h"
|
|
|
|
#include "nsBoxFrame.h"
|
|
|
|
#include "nsPresContext.h"
|
|
|
|
#include "nsCOMPtr.h"
|
|
|
|
#include "nsIContent.h"
|
|
|
|
#include "nsHTMLContainerFrame.h"
|
|
|
|
#include "nsINameSpaceManager.h"
|
|
|
|
#include "nsGkAtoms.h"
|
|
|
|
#include "nsFrameManager.h"
|
|
|
|
#include "nsIDOMNode.h"
|
|
|
|
#include "nsIDOMNamedNodeMap.h"
|
|
|
|
#include "nsIDOMAttr.h"
|
|
|
|
#include "nsIRenderingContext.h"
|
|
|
|
#include "nsIDocument.h"
|
|
|
|
#include "nsIDeviceContext.h"
|
|
|
|
#include "nsITheme.h"
|
|
|
|
#include "nsIServiceManager.h"
|
|
|
|
#include "nsIBoxLayout.h"
|
2010-08-30 17:49:13 -07:00
|
|
|
#include "FrameLayerBuilder.h"
|
|
|
|
|
|
|
|
using namespace mozilla;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
#ifdef DEBUG_LAYOUT
|
|
|
|
PRInt32 gIndent = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef DEBUG_LAYOUT
|
|
|
|
void
|
|
|
|
nsBoxAddIndents()
|
|
|
|
{
|
|
|
|
for(PRInt32 i=0; i < gIndent; i++)
|
|
|
|
{
|
|
|
|
printf(" ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef DEBUG_LAYOUT
|
|
|
|
void
|
|
|
|
nsBox::AppendAttribute(const nsAutoString& aAttribute, const nsAutoString& aValue, nsAutoString& aResult)
|
|
|
|
{
|
|
|
|
aResult.Append(aAttribute);
|
|
|
|
aResult.AppendLiteral("='");
|
|
|
|
aResult.Append(aValue);
|
|
|
|
aResult.AppendLiteral("' ");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsBox::ListBox(nsAutoString& aResult)
|
|
|
|
{
|
|
|
|
nsAutoString name;
|
|
|
|
GetBoxName(name);
|
|
|
|
|
|
|
|
char addr[100];
|
2007-07-08 00:08:04 -07:00
|
|
|
sprintf(addr, "[@%p] ", static_cast<void*>(this));
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
aResult.AppendASCII(addr);
|
|
|
|
aResult.Append(name);
|
|
|
|
aResult.AppendLiteral(" ");
|
|
|
|
|
|
|
|
nsIContent* content = GetContent();
|
|
|
|
|
|
|
|
// add on all the set attributes
|
|
|
|
if (content) {
|
|
|
|
nsCOMPtr<nsIDOMNode> node(do_QueryInterface(content));
|
|
|
|
nsCOMPtr<nsIDOMNamedNodeMap> namedMap;
|
|
|
|
|
|
|
|
node->GetAttributes(getter_AddRefs(namedMap));
|
|
|
|
PRUint32 length;
|
|
|
|
namedMap->GetLength(&length);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMNode> attribute;
|
|
|
|
for (PRUint32 i = 0; i < length; ++i)
|
|
|
|
{
|
|
|
|
namedMap->Item(i, getter_AddRefs(attribute));
|
|
|
|
nsCOMPtr<nsIDOMAttr> attr(do_QueryInterface(attribute));
|
|
|
|
attr->GetName(name);
|
|
|
|
nsAutoString value;
|
|
|
|
attr->GetValue(value);
|
|
|
|
AppendAttribute(name, value, aResult);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsBox::DumpBox(FILE* aFile)
|
|
|
|
{
|
|
|
|
nsAutoString s;
|
|
|
|
ListBox(s);
|
|
|
|
fprintf(aFile, "%s", NS_LossyConvertUTF16toASCII(s).get());
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsBox::PropagateDebug(nsBoxLayoutState& aState)
|
|
|
|
{
|
|
|
|
// propagate debug information
|
|
|
|
if (mState & NS_STATE_DEBUG_WAS_SET) {
|
|
|
|
if (mState & NS_STATE_SET_TO_DEBUG)
|
|
|
|
SetDebug(aState, PR_TRUE);
|
|
|
|
else
|
|
|
|
SetDebug(aState, PR_FALSE);
|
|
|
|
} else if (mState & NS_STATE_IS_ROOT) {
|
|
|
|
SetDebug(aState, gDebug);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef DEBUG_LAYOUT
|
|
|
|
void
|
|
|
|
nsBox::GetBoxName(nsAutoString& aName)
|
|
|
|
{
|
|
|
|
aName.AssignLiteral("Box");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsBox::BeginLayout(nsBoxLayoutState& aState)
|
|
|
|
{
|
|
|
|
#ifdef DEBUG_LAYOUT
|
|
|
|
|
|
|
|
nsBoxAddIndents();
|
|
|
|
printf("Layout: ");
|
|
|
|
DumpBox(stdout);
|
|
|
|
printf("\n");
|
|
|
|
gIndent++;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// mark ourselves as dirty so no child under us
|
|
|
|
// can post an incremental layout.
|
|
|
|
// XXXldb Is this still needed?
|
|
|
|
mState |= NS_FRAME_HAS_DIRTY_CHILDREN;
|
|
|
|
|
|
|
|
if (GetStateBits() & NS_FRAME_IS_DIRTY)
|
|
|
|
{
|
|
|
|
// If the parent is dirty, all the children are dirty (nsHTMLReflowState
|
|
|
|
// does this too).
|
|
|
|
nsIFrame* box;
|
|
|
|
for (box = GetChildBox(); box; box = box->GetNextBox())
|
|
|
|
box->AddStateBits(NS_FRAME_IS_DIRTY);
|
|
|
|
}
|
|
|
|
|
2010-03-01 17:57:43 -08:00
|
|
|
// Another copy-over from nsHTMLReflowState.
|
|
|
|
// Since we are in reflow, we don't need to store these properties anymore.
|
2010-03-28 18:46:55 -07:00
|
|
|
FrameProperties props = Properties();
|
|
|
|
props.Delete(UsedBorderProperty());
|
|
|
|
props.Delete(UsedPaddingProperty());
|
|
|
|
props.Delete(UsedMarginProperty());
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
#ifdef DEBUG_LAYOUT
|
|
|
|
PropagateDebug(aState);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsBox::DoLayout(nsBoxLayoutState& aState)
|
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsBox::EndLayout(nsBoxLayoutState& aState)
|
|
|
|
{
|
|
|
|
|
|
|
|
#ifdef DEBUG_LAYOUT
|
|
|
|
--gIndent;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return SyncLayout(aState);
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool nsBox::gGotTheme = PR_FALSE;
|
|
|
|
nsITheme* nsBox::gTheme = nsnull;
|
|
|
|
|
|
|
|
nsBox::nsBox()
|
|
|
|
{
|
|
|
|
MOZ_COUNT_CTOR(nsBox);
|
|
|
|
//mX = 0;
|
|
|
|
//mY = 0;
|
|
|
|
if (!gGotTheme) {
|
|
|
|
gGotTheme = PR_TRUE;
|
|
|
|
CallGetService("@mozilla.org/chrome/chrome-native-theme;1", &gTheme);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
nsBox::~nsBox()
|
|
|
|
{
|
|
|
|
// NOTE: This currently doesn't get called for |nsBoxToBlockAdaptor|
|
|
|
|
// objects, so don't rely on putting anything here.
|
|
|
|
MOZ_COUNT_DTOR(nsBox);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* static */ void
|
|
|
|
nsBox::Shutdown()
|
|
|
|
{
|
|
|
|
gGotTheme = PR_FALSE;
|
|
|
|
NS_IF_RELEASE(gTheme);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsBox::RelayoutChildAtOrdinal(nsBoxLayoutState& aState, nsIBox* aChild)
|
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsIFrame::GetClientRect(nsRect& aClientRect)
|
|
|
|
{
|
|
|
|
aClientRect = mRect;
|
|
|
|
aClientRect.MoveTo(0,0);
|
|
|
|
|
|
|
|
nsMargin borderPadding;
|
|
|
|
GetBorderAndPadding(borderPadding);
|
|
|
|
|
|
|
|
aClientRect.Deflate(borderPadding);
|
|
|
|
|
|
|
|
if (aClientRect.width < 0)
|
|
|
|
aClientRect.width = 0;
|
|
|
|
|
|
|
|
if (aClientRect.height < 0)
|
|
|
|
aClientRect.height = 0;
|
|
|
|
|
|
|
|
// NS_ASSERTION(aClientRect.width >=0 && aClientRect.height >= 0, "Content Size < 0");
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2010-10-06 21:25:47 -07:00
|
|
|
nsBox::SetBounds(nsBoxLayoutState& aState, const nsRect& aRect, PRBool aRemoveOverflowAreas)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
NS_BOX_ASSERTION(this, aRect.width >=0 && aRect.height >= 0, "SetBounds Size < 0");
|
|
|
|
|
|
|
|
nsRect rect(mRect);
|
|
|
|
|
|
|
|
PRUint32 flags = 0;
|
|
|
|
GetLayoutFlags(flags);
|
|
|
|
|
|
|
|
PRUint32 stateFlags = aState.LayoutFlags();
|
|
|
|
|
|
|
|
flags |= stateFlags;
|
|
|
|
|
2008-09-03 14:35:52 -07:00
|
|
|
if ((flags & NS_FRAME_NO_MOVE_FRAME) == NS_FRAME_NO_MOVE_FRAME)
|
2007-03-22 10:30:00 -07:00
|
|
|
SetSize(nsSize(aRect.width, aRect.height));
|
|
|
|
else
|
|
|
|
SetRect(aRect);
|
|
|
|
|
|
|
|
// Nuke the overflow area. The caller is responsible for restoring
|
|
|
|
// it if necessary.
|
2010-10-06 21:25:47 -07:00
|
|
|
if (aRemoveOverflowAreas) {
|
2007-03-22 10:30:00 -07:00
|
|
|
// remove the previously stored overflow area
|
2010-10-06 21:25:47 -07:00
|
|
|
ClearOverflowRects();
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!(flags & NS_FRAME_NO_MOVE_VIEW))
|
|
|
|
{
|
|
|
|
nsContainerFrame::PositionFrameView(this);
|
|
|
|
if ((rect.x != aRect.x) || (rect.y != aRect.y))
|
|
|
|
nsContainerFrame::PositionChildViews(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
// only if the origin changed
|
|
|
|
if ((rect.x != aRect.x) || (rect.y != aRect.y)) {
|
|
|
|
if (frame->HasView()) {
|
|
|
|
nsContainerFrame::PositionFrameView(presContext, frame,
|
|
|
|
frame->GetView());
|
|
|
|
} else {
|
|
|
|
nsContainerFrame::PositionChildViews(presContext, frame);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsBox::GetLayoutFlags(PRUint32& aFlags)
|
|
|
|
{
|
|
|
|
aFlags = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsIFrame::GetBorderAndPadding(nsMargin& aBorderAndPadding)
|
|
|
|
{
|
|
|
|
aBorderAndPadding.SizeTo(0, 0, 0, 0);
|
|
|
|
nsresult rv = GetBorder(aBorderAndPadding);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
nsMargin padding;
|
|
|
|
rv = GetPadding(padding);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
aBorderAndPadding += padding;
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsBox::GetBorder(nsMargin& aMargin)
|
|
|
|
{
|
|
|
|
aMargin.SizeTo(0,0,0,0);
|
|
|
|
|
|
|
|
const nsStyleDisplay* disp = GetStyleDisplay();
|
|
|
|
if (disp->mAppearance && gTheme) {
|
|
|
|
// Go to the theme for the border.
|
2007-03-30 14:11:41 -07:00
|
|
|
nsPresContext *context = PresContext();
|
2007-03-22 10:30:00 -07:00
|
|
|
if (gTheme->ThemeSupportsWidget(context, this, disp->mAppearance)) {
|
2009-01-14 19:27:09 -08:00
|
|
|
nsIntMargin margin(0, 0, 0, 0);
|
2007-03-22 10:30:00 -07:00
|
|
|
gTheme->GetWidgetBorder(context->DeviceContext(), this,
|
|
|
|
disp->mAppearance, &margin);
|
|
|
|
aMargin.top = context->DevPixelsToAppUnits(margin.top);
|
|
|
|
aMargin.right = context->DevPixelsToAppUnits(margin.right);
|
|
|
|
aMargin.bottom = context->DevPixelsToAppUnits(margin.bottom);
|
|
|
|
aMargin.left = context->DevPixelsToAppUnits(margin.left);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-07-16 23:30:25 -07:00
|
|
|
aMargin = GetStyleBorder()->GetActualBorder();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsBox::GetPadding(nsMargin& aMargin)
|
|
|
|
{
|
|
|
|
const nsStyleDisplay *disp = GetStyleDisplay();
|
|
|
|
if (disp->mAppearance && gTheme) {
|
|
|
|
// Go to the theme for the padding.
|
2007-03-30 14:11:41 -07:00
|
|
|
nsPresContext *context = PresContext();
|
2007-03-22 10:30:00 -07:00
|
|
|
if (gTheme->ThemeSupportsWidget(context, this, disp->mAppearance)) {
|
2009-01-14 19:27:09 -08:00
|
|
|
nsIntMargin margin(0, 0, 0, 0);
|
2007-03-22 10:30:00 -07:00
|
|
|
PRBool useThemePadding;
|
|
|
|
|
|
|
|
useThemePadding = gTheme->GetWidgetPadding(context->DeviceContext(),
|
|
|
|
this, disp->mAppearance,
|
|
|
|
&margin);
|
|
|
|
if (useThemePadding) {
|
|
|
|
aMargin.top = context->DevPixelsToAppUnits(margin.top);
|
|
|
|
aMargin.right = context->DevPixelsToAppUnits(margin.right);
|
|
|
|
aMargin.bottom = context->DevPixelsToAppUnits(margin.bottom);
|
|
|
|
aMargin.left = context->DevPixelsToAppUnits(margin.left);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
aMargin.SizeTo(0,0,0,0);
|
|
|
|
GetStylePadding()->GetPadding(aMargin);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsBox::GetMargin(nsMargin& aMargin)
|
|
|
|
{
|
|
|
|
aMargin.SizeTo(0,0,0,0);
|
|
|
|
GetStyleMargin()->GetMargin(aMargin);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsBox::SizeNeedsRecalc(nsSize& aSize)
|
|
|
|
{
|
|
|
|
aSize.width = -1;
|
|
|
|
aSize.height = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2010-04-03 04:36:19 -07:00
|
|
|
nsBox::CoordNeedsRecalc(nscoord& aFlex)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
aFlex = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsBox::DoesNeedRecalc(const nsSize& aSize)
|
|
|
|
{
|
|
|
|
return (aSize.width == -1 || aSize.height == -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsBox::DoesNeedRecalc(nscoord aCoord)
|
|
|
|
{
|
|
|
|
return (aCoord == -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsBox::SetLayoutManager(nsIBoxLayout* aLayout)
|
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsBox::GetLayoutManager(nsIBoxLayout** aLayout)
|
|
|
|
{
|
|
|
|
*aLayout = nsnull;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
nsSize
|
|
|
|
nsBox::GetPrefSize(nsBoxLayoutState& aState)
|
|
|
|
{
|
2007-04-29 15:24:59 -07:00
|
|
|
NS_ASSERTION(aState.GetRenderingContext(), "must have rendering context");
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
nsSize pref(0,0);
|
|
|
|
DISPLAY_PREF_SIZE(this, pref);
|
|
|
|
|
|
|
|
if (IsCollapsed(aState))
|
|
|
|
return pref;
|
|
|
|
|
|
|
|
AddBorderAndPadding(pref);
|
2010-03-18 12:58:15 -07:00
|
|
|
PRBool widthSet, heightSet;
|
|
|
|
nsIBox::AddCSSPrefSize(this, pref, widthSet, heightSet);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
nsSize minSize = GetMinSize(aState);
|
|
|
|
nsSize maxSize = GetMaxSize(aState);
|
2008-01-04 21:49:44 -08:00
|
|
|
return BoundsCheck(minSize, pref, maxSize);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
nsSize
|
|
|
|
nsBox::GetMinSize(nsBoxLayoutState& aState)
|
|
|
|
{
|
2007-04-29 15:24:59 -07:00
|
|
|
NS_ASSERTION(aState.GetRenderingContext(), "must have rendering context");
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
nsSize min(0,0);
|
|
|
|
DISPLAY_MIN_SIZE(this, min);
|
|
|
|
|
|
|
|
if (IsCollapsed(aState))
|
|
|
|
return min;
|
|
|
|
|
|
|
|
AddBorderAndPadding(min);
|
2010-03-18 12:58:15 -07:00
|
|
|
PRBool widthSet, heightSet;
|
|
|
|
nsIBox::AddCSSMinSize(aState, this, min, widthSet, heightSet);
|
2007-03-22 10:30:00 -07:00
|
|
|
return min;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsSize
|
|
|
|
nsBox::GetMinSizeForScrollArea(nsBoxLayoutState& aBoxLayoutState)
|
|
|
|
{
|
|
|
|
return nsSize(0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsSize
|
|
|
|
nsBox::GetMaxSize(nsBoxLayoutState& aState)
|
|
|
|
{
|
2007-04-29 15:24:59 -07:00
|
|
|
NS_ASSERTION(aState.GetRenderingContext(), "must have rendering context");
|
|
|
|
|
2008-02-26 17:51:57 -08:00
|
|
|
nsSize maxSize(NS_INTRINSICSIZE, NS_INTRINSICSIZE);
|
|
|
|
DISPLAY_MAX_SIZE(this, maxSize);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (IsCollapsed(aState))
|
2008-02-26 17:51:57 -08:00
|
|
|
return maxSize;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2008-02-26 17:51:57 -08:00
|
|
|
AddBorderAndPadding(maxSize);
|
2010-03-18 12:58:15 -07:00
|
|
|
PRBool widthSet, heightSet;
|
|
|
|
nsIBox::AddCSSMaxSize(this, maxSize, widthSet, heightSet);
|
2008-02-26 17:51:57 -08:00
|
|
|
return maxSize;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
nscoord
|
|
|
|
nsBox::GetFlex(nsBoxLayoutState& aState)
|
|
|
|
{
|
|
|
|
nscoord flex = 0;
|
|
|
|
|
|
|
|
nsIBox::AddCSSFlex(aState, this, flex);
|
|
|
|
|
|
|
|
return flex;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRUint32
|
|
|
|
nsIFrame::GetOrdinal(nsBoxLayoutState& aState)
|
|
|
|
{
|
|
|
|
PRUint32 ordinal = DEFAULT_ORDINAL_GROUP;
|
2010-03-18 12:58:15 -07:00
|
|
|
|
|
|
|
nsIContent* content = GetContent();
|
|
|
|
if (content) {
|
|
|
|
PRInt32 error;
|
|
|
|
nsAutoString value;
|
|
|
|
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsGkAtoms::ordinal, value);
|
|
|
|
if (!value.IsEmpty()) {
|
|
|
|
ordinal = value.ToInteger(&error);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// No attribute value. Check CSS.
|
|
|
|
const nsStyleXUL* boxInfo = GetStyleXUL();
|
|
|
|
if (boxInfo->mBoxOrdinal > 1) {
|
|
|
|
// The ordinal group was defined in CSS.
|
|
|
|
ordinal = (nscoord)boxInfo->mBoxOrdinal;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
return ordinal;
|
|
|
|
}
|
|
|
|
|
|
|
|
nscoord
|
|
|
|
nsBox::GetBoxAscent(nsBoxLayoutState& aState)
|
|
|
|
{
|
|
|
|
if (IsCollapsed(aState))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return GetPrefSize(aState).height;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsBox::IsCollapsed(nsBoxLayoutState& aState)
|
|
|
|
{
|
2010-03-18 12:58:15 -07:00
|
|
|
return GetStyleVisibility()->mVisible == NS_STYLE_VISIBILITY_COLLAPSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsIFrame::Layout(nsBoxLayoutState& aState)
|
|
|
|
{
|
2007-04-29 15:24:59 -07:00
|
|
|
NS_ASSERTION(aState.GetRenderingContext(), "must have rendering context");
|
|
|
|
|
2007-07-08 00:08:04 -07:00
|
|
|
nsBox *box = static_cast<nsBox*>(this);
|
2007-03-22 10:30:00 -07:00
|
|
|
DISPLAY_LAYOUT(box);
|
|
|
|
|
|
|
|
box->BeginLayout(aState);
|
|
|
|
|
|
|
|
box->DoLayout(aState);
|
|
|
|
|
|
|
|
box->EndLayout(aState);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
nsBox::DoesClipChildren()
|
|
|
|
{
|
|
|
|
const nsStyleDisplay* display = GetStyleDisplay();
|
|
|
|
NS_ASSERTION((display->mOverflowY == NS_STYLE_OVERFLOW_CLIP) ==
|
|
|
|
(display->mOverflowX == NS_STYLE_OVERFLOW_CLIP),
|
|
|
|
"If one overflow is clip, the other should be too");
|
|
|
|
return display->mOverflowX == NS_STYLE_OVERFLOW_CLIP;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsBox::SyncLayout(nsBoxLayoutState& aState)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
PRBool collapsed = PR_FALSE;
|
|
|
|
IsCollapsed(aState, collapsed);
|
|
|
|
if (collapsed) {
|
|
|
|
CollapseChild(aState, this, PR_TRUE);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
if (GetStateBits() & NS_FRAME_IS_DIRTY)
|
|
|
|
Redraw(aState);
|
|
|
|
|
|
|
|
RemoveStateBits(NS_FRAME_HAS_DIRTY_CHILDREN | NS_FRAME_IS_DIRTY
|
|
|
|
| NS_FRAME_FIRST_REFLOW | NS_FRAME_IN_REFLOW);
|
|
|
|
|
|
|
|
nsPresContext* presContext = aState.PresContext();
|
|
|
|
|
|
|
|
PRUint32 flags = 0;
|
|
|
|
GetLayoutFlags(flags);
|
|
|
|
|
|
|
|
PRUint32 stateFlags = aState.LayoutFlags();
|
|
|
|
|
|
|
|
flags |= stateFlags;
|
|
|
|
|
2010-10-06 21:25:46 -07:00
|
|
|
nsRect visualOverflow;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (ComputesOwnOverflowArea()) {
|
2010-10-06 21:25:46 -07:00
|
|
|
visualOverflow = GetVisualOverflowRect();
|
2008-02-19 23:08:55 -08:00
|
|
|
}
|
|
|
|
else {
|
2010-10-06 21:25:46 -07:00
|
|
|
nsRect rect(nsPoint(0, 0), GetSize());
|
|
|
|
nsOverflowAreas overflowAreas(rect, rect);
|
2009-09-13 16:05:55 -07:00
|
|
|
if (!DoesClipChildren() && !IsCollapsed(aState)) {
|
2007-03-22 10:30:00 -07:00
|
|
|
// See if our child frames caused us to overflow after being laid
|
|
|
|
// out. If so, store the overflow area. This normally can't happen
|
|
|
|
// in XUL, but it can happen with the CSS 'outline' property and
|
|
|
|
// possibly with other exotic stuff (e.g. relatively positioned
|
|
|
|
// frames in HTML inside XUL).
|
2010-10-06 21:25:46 -07:00
|
|
|
for (nsIFrame* kid = GetChildBox(); kid; kid = kid->GetNextBox()) {
|
|
|
|
nsOverflowAreas kidOverflow =
|
|
|
|
kid->GetOverflowAreas() + kid->GetPosition();
|
|
|
|
overflowAreas.UnionWith(kidOverflow);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-06 21:25:46 -07:00
|
|
|
FinishAndStoreOverflow(overflowAreas, GetSize());
|
|
|
|
visualOverflow = overflowAreas.VisualOverflow();
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
nsIView* view = GetView();
|
|
|
|
if (view) {
|
|
|
|
// Make sure the frame's view is properly sized and positioned and has
|
|
|
|
// things like opacity correct
|
|
|
|
nsHTMLContainerFrame::SyncFrameViewAfterReflow(
|
|
|
|
presContext,
|
|
|
|
this,
|
|
|
|
view,
|
2010-10-06 21:25:46 -07:00
|
|
|
visualOverflow,
|
2007-03-22 10:30:00 -07:00
|
|
|
flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult
|
|
|
|
nsIFrame::Redraw(nsBoxLayoutState& aState,
|
2010-08-30 17:49:13 -07:00
|
|
|
const nsRect* aDamageRect)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
if (aState.PaintingDisabled())
|
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
nsRect damageRect(0,0,0,0);
|
|
|
|
if (aDamageRect)
|
|
|
|
damageRect = *aDamageRect;
|
|
|
|
else
|
2010-10-06 21:25:47 -07:00
|
|
|
damageRect = GetVisualOverflowRect();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-08-30 17:49:13 -07:00
|
|
|
Invalidate(damageRect);
|
|
|
|
// nsStackLayout, at least, expects us to repaint descendants even
|
|
|
|
// if a damage rect is provided
|
|
|
|
FrameLayerBuilder::InvalidateThebesLayersInSubtree(this);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-03-18 12:58:15 -07:00
|
|
|
PRBool
|
|
|
|
nsIBox::AddCSSPrefSize(nsIBox* aBox, nsSize& aSize, PRBool &aWidthSet, PRBool &aHeightSet)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-03-18 12:58:15 -07:00
|
|
|
aWidthSet = PR_FALSE;
|
|
|
|
aHeightSet = PR_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// add in the css min, max, pref
|
|
|
|
const nsStylePosition* position = aBox->GetStylePosition();
|
|
|
|
|
|
|
|
// see if the width or height was specifically set
|
2007-05-09 01:48:38 -07:00
|
|
|
// XXX Handle eStyleUnit_Enumerated?
|
2007-05-03 16:11:00 -07:00
|
|
|
// (Handling the eStyleUnit_Enumerated types requires
|
|
|
|
// GetPrefSize/GetMinSize methods that don't consider
|
2008-07-17 16:37:12 -07:00
|
|
|
// (min-/max-/)(width/height) properties.)
|
2010-08-11 12:32:53 -07:00
|
|
|
const nsStyleCoord &width = position->mWidth;
|
|
|
|
if (width.GetUnit() == eStyleUnit_Coord) {
|
|
|
|
aSize.width = width.GetCoordValue();
|
2010-03-18 12:58:15 -07:00
|
|
|
aWidthSet = PR_TRUE;
|
2010-08-11 12:32:53 -07:00
|
|
|
} else if (width.IsCalcUnit()) {
|
|
|
|
if (!width.CalcHasPercent()) {
|
|
|
|
// pass 0 for percentage basis since we know there are no %s
|
|
|
|
aSize.width = nsRuleNode::ComputeComputedCalc(width, 0);
|
2010-08-25 03:17:55 -07:00
|
|
|
if (aSize.width < 0)
|
|
|
|
aSize.width = 0;
|
2010-08-11 12:32:53 -07:00
|
|
|
aWidthSet = PR_TRUE;
|
|
|
|
}
|
2008-07-17 16:37:12 -07:00
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-08-25 03:17:55 -07:00
|
|
|
const nsStyleCoord &height = position->mHeight;
|
|
|
|
if (height.GetUnit() == eStyleUnit_Coord) {
|
|
|
|
aSize.height = height.GetCoordValue();
|
2010-03-18 12:58:15 -07:00
|
|
|
aHeightSet = PR_TRUE;
|
2010-08-25 03:17:55 -07:00
|
|
|
} else if (height.IsCalcUnit()) {
|
|
|
|
if (!height.CalcHasPercent()) {
|
|
|
|
// pass 0 for percentage basis since we know there are no %s
|
|
|
|
aSize.height = nsRuleNode::ComputeComputedCalc(height, 0);
|
|
|
|
if (aSize.height < 0)
|
|
|
|
aSize.height = 0;
|
|
|
|
aHeightSet = PR_TRUE;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-08-25 03:17:55 -07:00
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIContent* content = aBox->GetContent();
|
|
|
|
// ignore 'height' and 'width' attributes if the actual element is not XUL
|
|
|
|
// For example, we might be magic XUL frames whose primary content is an HTML
|
|
|
|
// <select>
|
2009-08-24 13:02:07 -07:00
|
|
|
if (content && content->IsXUL()) {
|
2007-03-22 10:30:00 -07:00
|
|
|
nsAutoString value;
|
|
|
|
PRInt32 error;
|
|
|
|
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsGkAtoms::width, value);
|
|
|
|
if (!value.IsEmpty()) {
|
|
|
|
value.Trim("%");
|
|
|
|
|
|
|
|
aSize.width =
|
|
|
|
nsPresContext::CSSPixelsToAppUnits(value.ToInteger(&error));
|
2010-03-18 12:58:15 -07:00
|
|
|
aWidthSet = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsGkAtoms::height, value);
|
|
|
|
if (!value.IsEmpty()) {
|
|
|
|
value.Trim("%");
|
|
|
|
|
|
|
|
aSize.height =
|
|
|
|
nsPresContext::CSSPixelsToAppUnits(value.ToInteger(&error));
|
2010-03-18 12:58:15 -07:00
|
|
|
aHeightSet = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-18 12:58:15 -07:00
|
|
|
return (aWidthSet && aHeightSet);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-18 12:58:15 -07:00
|
|
|
PRBool
|
|
|
|
nsIBox::AddCSSMinSize(nsBoxLayoutState& aState, nsIBox* aBox, nsSize& aSize,
|
|
|
|
PRBool &aWidthSet, PRBool &aHeightSet)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-03-18 12:58:15 -07:00
|
|
|
aWidthSet = PR_FALSE;
|
|
|
|
aHeightSet = PR_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
PRBool canOverride = PR_TRUE;
|
|
|
|
|
|
|
|
// See if a native theme wants to supply a minimum size.
|
|
|
|
const nsStyleDisplay* display = aBox->GetStyleDisplay();
|
|
|
|
if (display->mAppearance) {
|
|
|
|
nsITheme *theme = aState.PresContext()->GetTheme();
|
|
|
|
if (theme && theme->ThemeSupportsWidget(aState.PresContext(), aBox, display->mAppearance)) {
|
2009-01-14 19:27:09 -08:00
|
|
|
nsIntSize size;
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIRenderingContext* rendContext = aState.GetRenderingContext();
|
|
|
|
if (rendContext) {
|
|
|
|
theme->GetMinimumWidgetSize(rendContext, aBox,
|
|
|
|
display->mAppearance, &size, &canOverride);
|
|
|
|
if (size.width) {
|
|
|
|
aSize.width = aState.PresContext()->DevPixelsToAppUnits(size.width);
|
2010-03-18 12:58:15 -07:00
|
|
|
aWidthSet = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
if (size.height) {
|
|
|
|
aSize.height = aState.PresContext()->DevPixelsToAppUnits(size.height);
|
2010-03-18 12:58:15 -07:00
|
|
|
aHeightSet = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// add in the css min, max, pref
|
|
|
|
const nsStylePosition* position = aBox->GetStylePosition();
|
|
|
|
|
|
|
|
// same for min size. Unfortunately min size is always set to 0. So for now
|
2010-08-25 03:17:56 -07:00
|
|
|
// we will assume 0 (as a coord) means not set.
|
|
|
|
const nsStyleCoord &minWidth = position->mMinWidth;
|
|
|
|
if ((minWidth.GetUnit() == eStyleUnit_Coord &&
|
|
|
|
minWidth.GetCoordValue() != 0) ||
|
|
|
|
(minWidth.IsCalcUnit() && !minWidth.CalcHasPercent())) {
|
|
|
|
nscoord min = nsRuleNode::ComputeCoordPercentCalc(minWidth, 0);
|
|
|
|
if (!aWidthSet || (min > aSize.width && canOverride)) {
|
2007-03-22 10:30:00 -07:00
|
|
|
aSize.width = min;
|
2010-03-18 12:58:15 -07:00
|
|
|
aWidthSet = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-08-25 03:17:56 -07:00
|
|
|
} else if (minWidth.GetUnit() == eStyleUnit_Percent) {
|
|
|
|
NS_ASSERTION(minWidth.GetPercentValue() == 0.0f,
|
2008-09-20 06:42:03 -07:00
|
|
|
"Non-zero percentage values not currently supported");
|
2007-03-22 10:30:00 -07:00
|
|
|
aSize.width = 0;
|
2010-08-25 03:17:56 -07:00
|
|
|
aWidthSet = PR_TRUE; // FIXME: should we really do this for
|
|
|
|
// nonzero values?
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2007-05-09 01:48:38 -07:00
|
|
|
// XXX Handle eStyleUnit_Enumerated?
|
2007-05-03 16:11:00 -07:00
|
|
|
// (Handling the eStyleUnit_Enumerated types requires
|
|
|
|
// GetPrefSize/GetMinSize methods that don't consider
|
|
|
|
// (min-/max-/)(width/height) properties.
|
2010-08-25 03:17:56 -07:00
|
|
|
// calc() with percentage is treated like '0' (unset)
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-08-25 03:17:55 -07:00
|
|
|
const nsStyleCoord &minHeight = position->mMinHeight;
|
|
|
|
if ((minHeight.GetUnit() == eStyleUnit_Coord &&
|
|
|
|
minHeight.GetCoordValue() != 0) ||
|
|
|
|
(minHeight.IsCalcUnit() && !minHeight.CalcHasPercent())) {
|
|
|
|
nscoord min = nsRuleNode::ComputeCoordPercentCalc(minHeight, 0);
|
|
|
|
if (!aHeightSet || (min > aSize.height && canOverride)) {
|
2007-03-22 10:30:00 -07:00
|
|
|
aSize.height = min;
|
2010-03-18 12:58:15 -07:00
|
|
|
aHeightSet = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-08-25 03:17:55 -07:00
|
|
|
} else if (minHeight.GetUnit() == eStyleUnit_Percent) {
|
2008-09-20 06:42:03 -07:00
|
|
|
NS_ASSERTION(position->mMinHeight.GetPercentValue() == 0.0f,
|
|
|
|
"Non-zero percentage values not currently supported");
|
2007-03-22 10:30:00 -07:00
|
|
|
aSize.height = 0;
|
2010-08-25 03:17:55 -07:00
|
|
|
aHeightSet = PR_TRUE; // FIXME: should we really do this for
|
|
|
|
// nonzero values?
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-08-25 03:17:55 -07:00
|
|
|
// calc() with percentage is treated like '0' (unset)
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
nsIContent* content = aBox->GetContent();
|
|
|
|
if (content) {
|
|
|
|
nsAutoString value;
|
|
|
|
PRInt32 error;
|
|
|
|
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsGkAtoms::minwidth, value);
|
|
|
|
if (!value.IsEmpty())
|
|
|
|
{
|
|
|
|
value.Trim("%");
|
|
|
|
|
|
|
|
nscoord val =
|
|
|
|
nsPresContext::CSSPixelsToAppUnits(value.ToInteger(&error));
|
|
|
|
if (val > aSize.width)
|
|
|
|
aSize.width = val;
|
2010-03-18 12:58:15 -07:00
|
|
|
aWidthSet = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsGkAtoms::minheight, value);
|
|
|
|
if (!value.IsEmpty())
|
|
|
|
{
|
|
|
|
value.Trim("%");
|
|
|
|
|
|
|
|
nscoord val =
|
|
|
|
nsPresContext::CSSPixelsToAppUnits(value.ToInteger(&error));
|
|
|
|
if (val > aSize.height)
|
|
|
|
aSize.height = val;
|
|
|
|
|
2010-03-18 12:58:15 -07:00
|
|
|
aHeightSet = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-18 12:58:15 -07:00
|
|
|
return (aWidthSet && aHeightSet);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-03-18 12:58:15 -07:00
|
|
|
PRBool
|
|
|
|
nsIBox::AddCSSMaxSize(nsIBox* aBox, nsSize& aSize, PRBool &aWidthSet, PRBool &aHeightSet)
|
|
|
|
{
|
|
|
|
aWidthSet = PR_FALSE;
|
|
|
|
aHeightSet = PR_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// add in the css min, max, pref
|
|
|
|
const nsStylePosition* position = aBox->GetStylePosition();
|
|
|
|
|
|
|
|
// and max
|
2008-07-17 16:37:12 -07:00
|
|
|
// see if the width or height was specifically set
|
2007-05-09 01:48:38 -07:00
|
|
|
// XXX Handle eStyleUnit_Enumerated?
|
2007-05-03 16:11:00 -07:00
|
|
|
// (Handling the eStyleUnit_Enumerated types requires
|
|
|
|
// GetPrefSize/GetMinSize methods that don't consider
|
2008-07-17 16:37:12 -07:00
|
|
|
// (min-/max-/)(width/height) properties.)
|
2010-08-25 03:17:56 -07:00
|
|
|
const nsStyleCoord maxWidth = position->mMaxWidth;
|
|
|
|
if (maxWidth.ConvertsToLength()) {
|
|
|
|
aSize.width = nsRuleNode::ComputeCoordPercentCalc(maxWidth, 0);
|
2010-03-18 12:58:15 -07:00
|
|
|
aWidthSet = PR_TRUE;
|
2008-07-17 16:37:12 -07:00
|
|
|
}
|
2010-08-25 03:17:56 -07:00
|
|
|
// percentages and calc() with percentages are treated like 'none'
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-08-25 03:17:55 -07:00
|
|
|
const nsStyleCoord &maxHeight = position->mMaxHeight;
|
|
|
|
if (maxHeight.ConvertsToLength()) {
|
|
|
|
aSize.height = nsRuleNode::ComputeCoordPercentCalc(maxHeight, 0);
|
2010-03-18 12:58:15 -07:00
|
|
|
aHeightSet = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2010-08-25 03:17:55 -07:00
|
|
|
// percentages and calc() with percentages are treated like 'none'
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
nsIContent* content = aBox->GetContent();
|
|
|
|
if (content) {
|
|
|
|
nsAutoString value;
|
|
|
|
PRInt32 error;
|
|
|
|
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsGkAtoms::maxwidth, value);
|
|
|
|
if (!value.IsEmpty()) {
|
|
|
|
value.Trim("%");
|
|
|
|
|
|
|
|
nscoord val =
|
|
|
|
nsPresContext::CSSPixelsToAppUnits(value.ToInteger(&error));
|
|
|
|
aSize.width = val;
|
2010-03-18 12:58:15 -07:00
|
|
|
aWidthSet = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsGkAtoms::maxheight, value);
|
|
|
|
if (!value.IsEmpty()) {
|
|
|
|
value.Trim("%");
|
|
|
|
|
|
|
|
nscoord val =
|
|
|
|
nsPresContext::CSSPixelsToAppUnits(value.ToInteger(&error));
|
|
|
|
aSize.height = val;
|
|
|
|
|
2010-03-18 12:58:15 -07:00
|
|
|
aHeightSet = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-18 12:58:15 -07:00
|
|
|
return (aWidthSet || aHeightSet);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-03-18 12:58:15 -07:00
|
|
|
PRBool
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIBox::AddCSSFlex(nsBoxLayoutState& aState, nsIBox* aBox, nscoord& aFlex)
|
|
|
|
{
|
|
|
|
PRBool flexSet = PR_FALSE;
|
|
|
|
|
|
|
|
// get the flexibility
|
|
|
|
nsIContent* content = aBox->GetContent();
|
|
|
|
if (content) {
|
|
|
|
PRInt32 error;
|
|
|
|
nsAutoString value;
|
|
|
|
|
|
|
|
content->GetAttr(kNameSpaceID_None, nsGkAtoms::flex, value);
|
|
|
|
if (!value.IsEmpty()) {
|
|
|
|
value.Trim("%");
|
|
|
|
aFlex = value.ToInteger(&error);
|
|
|
|
flexSet = PR_TRUE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// No attribute value. Check CSS.
|
|
|
|
const nsStyleXUL* boxInfo = aBox->GetStyleXUL();
|
|
|
|
if (boxInfo->mBoxFlex > 0.0f) {
|
|
|
|
// The flex was defined in CSS.
|
|
|
|
aFlex = (nscoord)boxInfo->mBoxFlex;
|
|
|
|
flexSet = PR_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aFlex < 0)
|
|
|
|
aFlex = 0;
|
|
|
|
if (aFlex >= nscoord_MAX)
|
|
|
|
aFlex = nscoord_MAX - 1;
|
|
|
|
|
|
|
|
return flexSet;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsBox::AddBorderAndPadding(nsSize& aSize)
|
|
|
|
{
|
|
|
|
AddBorderAndPadding(this, aSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsBox::AddBorderAndPadding(nsIBox* aBox, nsSize& aSize)
|
|
|
|
{
|
|
|
|
nsMargin borderPadding(0,0,0,0);
|
|
|
|
aBox->GetBorderAndPadding(borderPadding);
|
|
|
|
AddMargin(aSize, borderPadding);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsBox::AddMargin(nsIBox* aChild, nsSize& aSize)
|
|
|
|
{
|
|
|
|
nsMargin margin(0,0,0,0);
|
|
|
|
aChild->GetMargin(margin);
|
|
|
|
AddMargin(aSize, margin);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsBox::AddMargin(nsSize& aSize, const nsMargin& aMargin)
|
|
|
|
{
|
|
|
|
if (aSize.width != NS_INTRINSICSIZE)
|
|
|
|
aSize.width += aMargin.left + aMargin.right;
|
|
|
|
|
|
|
|
if (aSize.height != NS_INTRINSICSIZE)
|
|
|
|
aSize.height += aMargin.top + aMargin.bottom;
|
|
|
|
}
|
|
|
|
|
2008-01-04 21:49:44 -08:00
|
|
|
nscoord
|
|
|
|
nsBox::BoundsCheck(nscoord aMin, nscoord aPref, nscoord aMax)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
if (aPref > aMax)
|
|
|
|
aPref = aMax;
|
|
|
|
|
|
|
|
if (aPref < aMin)
|
|
|
|
aPref = aMin;
|
2008-01-04 21:49:44 -08:00
|
|
|
|
|
|
|
return aPref;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2008-01-04 21:49:44 -08:00
|
|
|
nsSize
|
|
|
|
nsBox::BoundsCheckMinMax(const nsSize& aMinSize, const nsSize& aMaxSize)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2009-09-16 08:01:36 -07:00
|
|
|
return nsSize(NS_MAX(aMaxSize.width, aMinSize.width),
|
|
|
|
NS_MAX(aMaxSize.height, aMinSize.height));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2008-01-04 21:49:44 -08:00
|
|
|
nsSize
|
|
|
|
nsBox::BoundsCheck(const nsSize& aMinSize, const nsSize& aPrefSize, const nsSize& aMaxSize)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2008-01-04 21:49:44 -08:00
|
|
|
return nsSize(BoundsCheck(aMinSize.width, aPrefSize.width, aMaxSize.width),
|
|
|
|
BoundsCheck(aMinSize.height, aPrefSize.height, aMaxSize.height));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG_LAYOUT
|
|
|
|
nsresult
|
|
|
|
nsBox::SetDebug(nsBoxLayoutState& aState, PRBool aDebug)
|
|
|
|
{
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsBox::GetDebugBoxAt( const nsPoint& aPoint,
|
|
|
|
nsIBox** aBox)
|
|
|
|
{
|
|
|
|
nsRect thisRect(nsPoint(0,0), GetSize());
|
|
|
|
if (!thisRect.Contains(aPoint))
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
nsIBox* child = GetChildBox();
|
|
|
|
nsIBox* hit = nsnull;
|
|
|
|
|
|
|
|
*aBox = nsnull;
|
|
|
|
while (nsnull != child) {
|
|
|
|
nsresult rv = child->GetDebugBoxAt(aPoint - child->GetOffsetTo(this), &hit);
|
|
|
|
|
|
|
|
if (NS_SUCCEEDED(rv) && hit) {
|
|
|
|
*aBox = hit;
|
|
|
|
}
|
|
|
|
child = child->GetNextBox();
|
|
|
|
}
|
|
|
|
|
|
|
|
// found a child
|
|
|
|
if (*aBox) {
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
nsBox::GetDebug(PRBool& aDebug)
|
|
|
|
{
|
|
|
|
aDebug = PR_FALSE;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|