1998-09-23 06:32:15 +04: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 "FixedTableLayoutStrategy.h"
|
|
|
|
#include "nsTableFrame.h"
|
|
|
|
#include "nsTableColFrame.h"
|
|
|
|
#include "nsTableCellFrame.h"
|
|
|
|
#include "nsIStyleContext.h"
|
|
|
|
#include "nsStyleConsts.h"
|
|
|
|
#include "nsVoidArray.h"
|
|
|
|
#include "nsIPtr.h"
|
|
|
|
#include "nsHTMLIIDs.h"
|
|
|
|
|
|
|
|
NS_DEF_PTR(nsIStyleContext);
|
|
|
|
|
1998-10-20 03:43:06 +04:00
|
|
|
FixedTableLayoutStrategy::FixedTableLayoutStrategy(nsTableFrame *aFrame)
|
|
|
|
: BasicTableLayoutStrategy(aFrame)
|
1998-09-23 06:32:15 +04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
FixedTableLayoutStrategy::~FixedTableLayoutStrategy()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
1999-07-28 12:09:02 +04:00
|
|
|
PRBool FixedTableLayoutStrategy::BalanceColumnWidths(nsIStyleContext* aTableStyle,
|
1998-10-02 08:10:00 +04:00
|
|
|
const nsHTMLReflowState& aReflowState,
|
1999-07-28 12:09:02 +04:00
|
|
|
nscoord aMaxWidth)
|
1998-09-23 06:32:15 +04:00
|
|
|
{
|
1999-07-28 12:09:02 +04:00
|
|
|
return PR_TRUE;
|
1998-09-23 06:32:15 +04:00
|
|
|
}
|
|
|
|
|
1999-06-03 06:42:33 +04:00
|
|
|
/*
|
|
|
|
* assign the width of all columns
|
1998-09-23 06:32:15 +04:00
|
|
|
* if there is a colframe with a width attribute, use it as the column width
|
|
|
|
* otherwise if there is a cell in the first row and it has a width attribute, use it
|
|
|
|
* if this cell includes a colspan, width is divided equally among spanned columns
|
|
|
|
* otherwise the cell get a proportion of the remaining space
|
|
|
|
* as determined by the table width attribute. If no table width attribute, it gets 0 width
|
|
|
|
*/
|
1999-06-03 06:42:33 +04:00
|
|
|
PRBool FixedTableLayoutStrategy::AssignPreliminaryColumnWidths(nscoord aComputedWidth)
|
1998-09-23 06:32:15 +04:00
|
|
|
{
|
1999-07-24 00:32:18 +04:00
|
|
|
// NS_ASSERTION(aComputedWidth != NS_UNCONSTRAINEDSIZE, "bad computed width");
|
1999-06-03 06:42:33 +04:00
|
|
|
|
|
|
|
const nsStylePosition* tablePosition;
|
|
|
|
mTableFrame->GetStyleData(eStyleStruct_Position, (const nsStyleStruct*&)tablePosition);
|
|
|
|
PRBool tableIsFixedWidth = eStyleUnit_Coord == tablePosition->mWidth.GetUnit() ||
|
|
|
|
eStyleUnit_Percent == tablePosition->mWidth.GetUnit();
|
|
|
|
|
|
|
|
const nsStyleSpacing* tableSpacing;
|
|
|
|
mTableFrame->GetStyleData(eStyleStruct_Spacing, (const nsStyleStruct*&)tableSpacing);
|
|
|
|
nsMargin borderPadding;
|
|
|
|
tableSpacing->CalcBorderPaddingFor(mTableFrame, borderPadding);
|
|
|
|
|
|
|
|
PRInt32 colX;
|
|
|
|
// availWidth is used as the basis for percentage width columns. It is aComputedWidth
|
|
|
|
// minus table border, padding, & cellspacing
|
|
|
|
nscoord availWidth = aComputedWidth - borderPadding.left - borderPadding.right -
|
|
|
|
((mNumCols + 1) * mTableFrame->GetCellSpacingX());
|
1998-09-23 06:32:15 +04:00
|
|
|
|
1999-06-03 06:42:33 +04:00
|
|
|
PRInt32 specifiedCols = 0; // the number of columns whose width is given
|
|
|
|
nscoord totalColWidth = 0; // the sum of the widths of the columns
|
1998-09-23 06:32:15 +04:00
|
|
|
|
1999-06-03 06:42:33 +04:00
|
|
|
nscoord* colWidths = new PRBool[mNumCols];
|
|
|
|
nsCRT::memset(colWidths, -1, mNumCols*sizeof(nscoord));
|
1998-09-23 06:32:15 +04:00
|
|
|
|
1999-07-24 03:41:05 +04:00
|
|
|
// for every column, determine its specified width
|
1999-06-03 06:42:33 +04:00
|
|
|
for (colX = 0; colX < mNumCols; colX++) {
|
1998-09-23 06:32:15 +04:00
|
|
|
// Get column information
|
1999-06-03 06:42:33 +04:00
|
|
|
nsTableColFrame* colFrame = mTableFrame->GetColFrame(colX);
|
1999-07-28 12:09:02 +04:00
|
|
|
if (!colFrame) {
|
|
|
|
NS_ASSERTION(PR_FALSE, "bad col frame");
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
1998-09-23 06:32:15 +04:00
|
|
|
|
|
|
|
// Get the columns's style
|
|
|
|
const nsStylePosition* colPosition;
|
1998-11-20 12:29:58 +03:00
|
|
|
colFrame->GetStyleData(eStyleStruct_Position, (const nsStyleStruct*&)colPosition);
|
1998-09-23 06:32:15 +04:00
|
|
|
|
1999-06-03 06:42:33 +04:00
|
|
|
// get the fixed width if available
|
|
|
|
if (eStyleUnit_Coord == colPosition->mWidth.GetUnit()) {
|
|
|
|
colWidths[colX] = colPosition->mWidth.GetCoordValue();
|
|
|
|
} // get the percentage width
|
1999-07-24 00:32:18 +04:00
|
|
|
else if ((eStyleUnit_Percent == colPosition->mWidth.GetUnit()) &&
|
|
|
|
(aComputedWidth != NS_UNCONSTRAINEDSIZE)) {
|
|
|
|
// Only apply percentages if we're unconstrained.
|
1999-06-03 06:42:33 +04:00
|
|
|
float percent = colPosition->mWidth.GetPercentValue();
|
|
|
|
colWidths[colX] = NSToCoordRound(percent * (float)availWidth);
|
1998-09-23 06:32:15 +04:00
|
|
|
}
|
1999-06-03 06:42:33 +04:00
|
|
|
else { // get width from the cell
|
|
|
|
nsTableCellFrame* cellFrame = mTableFrame->GetCellFrameAt(0, colX);
|
|
|
|
if (nsnull != cellFrame) {
|
1998-09-23 06:32:15 +04:00
|
|
|
// Get the cell's style
|
|
|
|
const nsStylePosition* cellPosition;
|
1998-11-20 12:29:58 +03:00
|
|
|
cellFrame->GetStyleData(eStyleStruct_Position, (const nsStyleStruct*&)cellPosition);
|
1998-09-23 06:32:15 +04:00
|
|
|
|
1999-07-28 12:09:02 +04:00
|
|
|
PRInt32 colSpan = mTableFrame->GetEffectiveColSpan(cellFrame);
|
1999-06-03 06:42:33 +04:00
|
|
|
// Get fixed cell width if available
|
|
|
|
if (eStyleUnit_Coord == cellPosition->mWidth.GetUnit()) {
|
|
|
|
colWidths[colX] = cellPosition->mWidth.GetCoordValue() / colSpan;
|
|
|
|
}
|
1999-07-24 00:32:18 +04:00
|
|
|
else if ((eStyleUnit_Percent == cellPosition->mWidth.GetUnit()) &&
|
|
|
|
(aComputedWidth != NS_UNCONSTRAINEDSIZE)) {
|
1999-06-03 06:42:33 +04:00
|
|
|
float percent = cellPosition->mWidth.GetPercentValue();
|
|
|
|
colWidths[colX] = NSToCoordRound(percent * (float)availWidth / (float)colSpan);
|
1998-09-23 06:32:15 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
1999-06-03 06:42:33 +04:00
|
|
|
if (colWidths[colX] >= 0) {
|
|
|
|
totalColWidth += colWidths[colX];
|
|
|
|
specifiedCols++;
|
|
|
|
}
|
1998-09-23 06:32:15 +04:00
|
|
|
}
|
1999-06-03 06:42:33 +04:00
|
|
|
|
|
|
|
nscoord lastColAllocated = -1;
|
|
|
|
nscoord remainingWidth = availWidth - totalColWidth;
|
1999-07-24 04:32:21 +04:00
|
|
|
if (remainingWidth >= 500000) {
|
1999-07-24 03:41:05 +04:00
|
|
|
// let's put a cap on the width so that it doesn't become insane.
|
|
|
|
remainingWidth = 100;
|
|
|
|
}
|
|
|
|
|
1999-06-03 06:42:33 +04:00
|
|
|
if (tableIsFixedWidth && (0 < remainingWidth)) {
|
|
|
|
if (mNumCols > specifiedCols) {
|
|
|
|
// allocate the extra space to the columns which have no width specified
|
|
|
|
nscoord colAlloc = NSToCoordRound( ((float)remainingWidth) / (((float)mNumCols) - ((float)specifiedCols)));
|
|
|
|
for (colX = 0; colX < mNumCols; colX++) {
|
|
|
|
if (-1 == colWidths[colX]) {
|
|
|
|
colWidths[colX] = colAlloc;
|
|
|
|
totalColWidth += colAlloc;
|
|
|
|
lastColAllocated = colX;
|
1998-09-23 06:32:15 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
1999-06-03 06:42:33 +04:00
|
|
|
else { // allocate the extra space to the columns which have width specified
|
|
|
|
float divisor = (float)totalColWidth;
|
|
|
|
for (colX = 0; colX < mNumCols; colX++) {
|
|
|
|
if (colWidths[colX] > 0) {
|
|
|
|
nscoord colAlloc = NSToCoordRound(remainingWidth * colWidths[colX] / divisor);
|
|
|
|
colWidths[colX] += colAlloc;
|
|
|
|
totalColWidth += colAlloc;
|
|
|
|
lastColAllocated = colX;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
1998-09-23 06:32:15 +04:00
|
|
|
}
|
1999-06-03 06:42:33 +04:00
|
|
|
|
|
|
|
nscoord overAllocation = (availWidth >= 0)
|
|
|
|
? totalColWidth - availWidth : 0;
|
|
|
|
// set the column widths
|
|
|
|
for (colX = 0; colX < mNumCols; colX++) {
|
|
|
|
if (colWidths[colX] < 0)
|
|
|
|
colWidths[colX] = 0;
|
|
|
|
// if there was too much allocated due to rounding, remove it from the last col
|
|
|
|
if ((colX == lastColAllocated) && (overAllocation != 0)) {
|
|
|
|
colWidths[colX] += overAllocation;
|
|
|
|
colWidths[colX] = PR_MAX(0, colWidths[colX]);
|
|
|
|
}
|
|
|
|
mTableFrame->SetColumnWidth(colX, colWidths[colX]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// min/max TW is min/max of (specified table width, sum of specified column(cell) widths)
|
1999-07-28 12:09:02 +04:00
|
|
|
mMinTableContentWidth = mMaxTableContentWidth = totalColWidth;
|
1998-09-23 06:32:15 +04:00
|
|
|
|
|
|
|
// clean up
|
1999-06-03 06:42:33 +04:00
|
|
|
if (nsnull != colWidths) {
|
|
|
|
delete [] colWidths;
|
1998-09-23 06:32:15 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
1999-08-19 23:52:37 +04:00
|
|
|
PRBool FixedTableLayoutStrategy::ColumnsCanBeInvalidatedBy(nsStyleCoord* aPrevStyleWidth,
|
1999-08-21 03:27:54 +04:00
|
|
|
const nsTableCellFrame& aCellFrame) const
|
1999-08-19 23:52:37 +04:00
|
|
|
{
|
|
|
|
return ColumnsCanBeInvalidatedBy(aCellFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool FixedTableLayoutStrategy::ColumnsCanBeInvalidatedBy(const nsTableCellFrame& aCellFrame,
|
|
|
|
PRBool aConsiderMinWidth) const
|
|
|
|
|
|
|
|
{
|
|
|
|
nscoord rowIndex;
|
|
|
|
aCellFrame.GetRowIndex(rowIndex);
|
|
|
|
if (0 == rowIndex) {
|
|
|
|
// It is not worth the effort to determine if the col or cell determined the col
|
|
|
|
// width. Since rebalancing the columns is fairly trival in this strategy, just force it.
|
1999-08-21 03:27:54 +04:00
|
|
|
return PR_TRUE;
|
1999-08-19 23:52:37 +04:00
|
|
|
}
|
1999-08-21 03:27:54 +04:00
|
|
|
return PR_FALSE;
|
1999-08-19 23:52:37 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
PRBool FixedTableLayoutStrategy::ColumnsAreValidFor(const nsTableCellFrame& aCellFrame,
|
|
|
|
nscoord aPrevCellMin,
|
|
|
|
nscoord aPrevCellDes) const
|
|
|
|
{
|
|
|
|
// take the easy way out, see comments above.
|
1999-08-21 03:27:54 +04:00
|
|
|
return !ColumnsCanBeInvalidatedBy(aCellFrame);
|
1999-08-19 23:52:37 +04:00
|
|
|
}
|
1998-09-23 06:32:15 +04:00
|
|
|
|
|
|
|
|
|
|
|
|