1998-04-30 21:57:09 +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 <stdio.h>
|
|
|
|
#include "BasicTableLayoutStrategy.h"
|
|
|
|
#include "nsTableFrame.h"
|
|
|
|
#include "nsColLayoutData.h"
|
|
|
|
#include "nsCellLayoutData.h"
|
|
|
|
#include "nsTableCol.h"
|
|
|
|
#include "nsTableCellFrame.h"
|
|
|
|
#include "nsIStyleContext.h"
|
|
|
|
#include "nsStyleConsts.h"
|
|
|
|
#include "nsIPtr.h"
|
|
|
|
|
|
|
|
NS_DEF_PTR(nsTableCol);
|
|
|
|
NS_DEF_PTR(nsTableCell);
|
|
|
|
NS_DEF_PTR(nsIStyleContext);
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef NS_DEBUG
|
|
|
|
static PRBool gsDebug = PR_FALSE;
|
|
|
|
static PRBool gsDebugCLD = PR_FALSE;
|
|
|
|
static PRBool gsTiming = PR_FALSE;
|
|
|
|
#else
|
|
|
|
static const PRBool gsDebug = PR_FALSE;
|
|
|
|
static const PRBool gsDebugCLD = PR_FALSE;
|
|
|
|
static const PRBool gsTiming = PR_FALSE;
|
|
|
|
#endif
|
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
/* return true if the style indicates that the width is proportional
|
|
|
|
* for the purposes of column width determination
|
|
|
|
*/
|
|
|
|
PRBool BasicTableLayoutStrategy::IsProportionalWidth(nsStylePosition* aStylePosition)
|
|
|
|
{
|
|
|
|
PRBool result = PR_FALSE; // assume that it is not
|
|
|
|
PRInt32 unitType;
|
|
|
|
if (nsnull == aStylePosition) {
|
|
|
|
unitType = eStyleUnit_Auto;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
unitType = aStylePosition->mWidth.GetUnit();
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (unitType) {
|
|
|
|
case eStyleUnit_Coord:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eStyleUnit_Auto:
|
|
|
|
case eStyleUnit_Proportional:
|
|
|
|
case eStyleUnit_Percent:
|
|
|
|
result = PR_TRUE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
// TODO
|
|
|
|
case eStyleUnit_Inherit:
|
|
|
|
break;
|
1998-04-30 21:57:09 +04:00
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
default:
|
|
|
|
NS_ASSERTION(PR_FALSE, "illegal style type in IsProportionalWidth");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return result;
|
1998-04-30 21:57:09 +04:00
|
|
|
}
|
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
PRBool BasicTableLayoutStrategy::IsAutoWidth(nsStylePosition* aStylePosition)
|
|
|
|
{
|
|
|
|
PRBool result = PR_TRUE; // assume that it is
|
|
|
|
if (nsnull!=aStylePosition)
|
|
|
|
{
|
|
|
|
result = (PRBool)(eStyleUnit_Auto==aStylePosition->mWidth.GetUnit());
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// aSpecifiedTableWidth is filled if the table witdth is not auto
|
|
|
|
PRBool BasicTableLayoutStrategy::TableIsAutoWidth(nsIStyleContext *aTableStyle, nscoord & aSpecifiedTableWidth)
|
|
|
|
{
|
|
|
|
NS_ASSERTION(nsnull!=aTableStyle, "bad arg - aTableStyle");
|
|
|
|
PRBool result = PR_TRUE; // the default
|
|
|
|
if (nsnull!=aTableStyle)
|
|
|
|
{
|
|
|
|
//nsStylePosition* tablePosition = (nsStylePosition*)aTableStyle->GetData(eStyleStruct_Position);
|
|
|
|
/* this is sick and wrong, but what the hell
|
|
|
|
we grab the style of our parent (nsTableOuterFrame) and ask it for width info,
|
|
|
|
until the style resolution stuff does the cool stuff about splitting style between outer and inner
|
|
|
|
*/
|
|
|
|
// begin REMOVE_ME_WHEN_TABLE_STYLE_IS_RESOLVED!
|
|
|
|
nsIStyleContext* parentStyle = nsnull;
|
|
|
|
nsIFrame * parent = nsnull;
|
|
|
|
mTableFrame->GetGeometricParent(parent);
|
|
|
|
parent->GetStyleContext(nsnull, parentStyle);
|
|
|
|
nsStylePosition* tablePosition = (nsStylePosition*)parentStyle->GetData(eStyleStruct_Position);
|
|
|
|
// end REMOVE_ME_WHEN_TABLE_STYLE_IS_RESOLVED!
|
|
|
|
switch (tablePosition->mWidth.GetUnit()) {
|
|
|
|
case eStyleUnit_Auto: // specified auto width
|
|
|
|
case eStyleUnit_Proportional: // illegal for table, so ignored
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eStyleUnit_Inherit:
|
|
|
|
// get width of parent and see if it is a specified value or not
|
|
|
|
// XXX for now, just return true
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eStyleUnit_Coord:
|
|
|
|
aSpecifiedTableWidth = tablePosition->mWidth.GetCoordValue();
|
|
|
|
result = PR_FALSE;
|
|
|
|
break;
|
|
|
|
case eStyleUnit_Percent:
|
|
|
|
// get the parent's width (available only from parent frames that claim they can provide it,
|
|
|
|
// and assuming it's already specified; that is, top-down layout sets the widths on the way down.)
|
|
|
|
nscoord parentWidth = 0;
|
|
|
|
nsIFrame *parent=nsnull;
|
|
|
|
nsIFrame *child=nsnull;
|
|
|
|
mTableFrame->GetGeometricParent(child); // start with the outer table frame as the child
|
|
|
|
child->GetGeometricParent(parent); // and the table outer frame's parent as the first parent
|
|
|
|
while(nsnull!=parent)
|
|
|
|
{
|
|
|
|
PRBool isPercentageBase=PR_FALSE;
|
|
|
|
parent->IsPercentageBase(isPercentageBase);
|
|
|
|
if (PR_TRUE==isPercentageBase)
|
|
|
|
{ // found the ancestor who claims to be the guy to base my percentage width on
|
|
|
|
nsRect parentRect;
|
|
|
|
parent->GetRect(parentRect);
|
|
|
|
parentWidth = parentRect.width;
|
|
|
|
if (PR_TRUE==gsDebug) printf(" ** width for parent frame %p = %d\n", parent, parentWidth);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
parent->GetGeometricParent(parent); // get next ancestor
|
|
|
|
}
|
|
|
|
// set aSpecifiedTableWidth to be the given percent of the parent.
|
|
|
|
float percent = tablePosition->mWidth.GetPercentValue();
|
|
|
|
aSpecifiedTableWidth = (PRInt32)(parentWidth*percent);
|
|
|
|
if (PR_TRUE==gsDebug) printf(" ** aSpecifiedTableWidth = %d\n", aSpecifiedTableWidth);
|
|
|
|
result = PR_FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
1998-04-30 21:57:09 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
BasicTableLayoutStrategy::BasicTableLayoutStrategy(nsTableFrame *aFrame)
|
|
|
|
{
|
|
|
|
mTableFrame = aFrame;
|
|
|
|
}
|
|
|
|
|
|
|
|
BasicTableLayoutStrategy::~BasicTableLayoutStrategy()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool BasicTableLayoutStrategy::BalanceColumnWidths(nsIPresContext* aPresContext,
|
1998-05-27 02:03:16 +04:00
|
|
|
nsIStyleContext *aTableStyle,
|
1998-04-30 21:57:09 +04:00
|
|
|
PRInt32 aMaxWidth,
|
|
|
|
PRInt32 aNumCols,
|
|
|
|
PRInt32 &aTotalFixedWidth,
|
|
|
|
PRInt32 &aMinTableWidth,
|
|
|
|
PRInt32 &aMaxTableWidth,
|
|
|
|
nsSize* aMaxElementSize)
|
|
|
|
{
|
|
|
|
PRBool result = PR_TRUE;
|
1998-05-27 02:03:16 +04:00
|
|
|
|
1998-04-30 21:57:09 +04:00
|
|
|
// Step 1 - assign the width of all fixed-width columns
|
|
|
|
AssignFixedColumnWidths(aPresContext, aMaxWidth, aNumCols,
|
|
|
|
aTotalFixedWidth, aMinTableWidth, aMaxTableWidth);
|
|
|
|
|
|
|
|
if (nsnull!=aMaxElementSize)
|
|
|
|
{
|
|
|
|
aMaxElementSize->width = aMinTableWidth;
|
|
|
|
if (gsDebug) printf(" setting aMaxElementSize->width = %d\n", aMaxElementSize->width);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (gsDebug) printf(" nsnull aMaxElementSize\n");
|
|
|
|
}
|
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
// Step 2 - determine how much space is really available
|
1998-04-30 21:57:09 +04:00
|
|
|
PRInt32 availWidth = aMaxWidth - aTotalFixedWidth;
|
1998-05-27 02:03:16 +04:00
|
|
|
nscoord tableWidth = 0;
|
|
|
|
if (PR_FALSE==TableIsAutoWidth(aTableStyle, tableWidth))
|
|
|
|
availWidth = tableWidth - aTotalFixedWidth;
|
|
|
|
|
|
|
|
// Step 3 - assign the width of all proportional-width columns in the remaining space
|
1998-04-30 21:57:09 +04:00
|
|
|
if (gsDebug==PR_TRUE) printf ("Step 2...\n availWidth = %d\n", availWidth);
|
1998-05-27 02:03:16 +04:00
|
|
|
result = BalanceProportionalColumns(aPresContext,
|
|
|
|
availWidth, aMaxWidth,
|
|
|
|
aMinTableWidth, aMaxTableWidth,
|
|
|
|
tableWidth);
|
1998-04-30 21:57:09 +04:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
STEP 1
|
|
|
|
for every col
|
|
|
|
if the col has a fixed width
|
|
|
|
set the width to max(fixed width, maxElementSize)
|
|
|
|
if the cell spans columns, divide the cell width between the columns
|
|
|
|
else skip it for now
|
|
|
|
|
|
|
|
take borders and padding into account
|
|
|
|
|
|
|
|
STEP 2
|
|
|
|
determine the min and max size for the table width
|
|
|
|
if col is proportionately specified
|
|
|
|
if (col width specified to 0)
|
|
|
|
col width = minColWidth
|
|
|
|
else if (minTableWidth >= aMaxSize.width)
|
|
|
|
set col widths to min, install a hor. scroll bar
|
|
|
|
else if (maxTableWidth <= aMaxSize.width)
|
|
|
|
set each col to its max size
|
|
|
|
else
|
|
|
|
W = aMaxSize.width - minTableWidth
|
|
|
|
D = maxTableWidth - minTableWidth
|
|
|
|
for each col
|
|
|
|
d = maxColWidth - minColWidth
|
|
|
|
col width = minColWidth + ((d*W)/D)
|
|
|
|
|
|
|
|
STEP 3
|
|
|
|
if there is space left over
|
|
|
|
for every col
|
|
|
|
if col is proportionately specified
|
|
|
|
add space to col width until it is that proportion of the table width
|
|
|
|
do this non-destructively in case there isn't enough space
|
|
|
|
if there isn't enough space as determined in the prior step,
|
|
|
|
add space in proportion to the proportionate width attribute
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
// Step 1 - assign the width of all fixed-width columns,
|
|
|
|
// and calculate min/max table width
|
|
|
|
PRBool BasicTableLayoutStrategy::AssignFixedColumnWidths(nsIPresContext* aPresContext,
|
1998-04-30 22:04:16 +04:00
|
|
|
PRInt32 maxWidth,
|
|
|
|
PRInt32 aNumCols,
|
|
|
|
PRInt32 &aTotalFixedWidth,
|
|
|
|
PRInt32 &aMinTableWidth,
|
|
|
|
PRInt32 &aMaxTableWidth)
|
1998-04-30 21:57:09 +04:00
|
|
|
{
|
|
|
|
#ifdef DEBUG
|
|
|
|
nsIFrame *tablePIF=nsnull;
|
|
|
|
mTableFrame->GetPrevInFlow(tablePIF);
|
|
|
|
NS_ASSERTION(nsnull==tablePIF, "never ever call me on a continuing frame!");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (gsDebug==PR_TRUE) printf (" AssignFixedColumnWidths\n");
|
|
|
|
nsVoidArray *spanList=nsnull;
|
|
|
|
for (PRInt32 colIndex = 0; colIndex<aNumCols; colIndex++)
|
|
|
|
{
|
|
|
|
nsVoidArray *columnLayoutData = mTableFrame->GetColumnLayoutData();
|
|
|
|
nsColLayoutData * colData = (nsColLayoutData *)(columnLayoutData->ElementAt(colIndex));
|
|
|
|
NS_ASSERTION(nsnull != colData, "bad column data");
|
|
|
|
nsTableColPtr col = colData->GetCol(); // col: ADDREF++
|
|
|
|
NS_ASSERTION(col.IsNotNull(), "bad col");
|
|
|
|
|
|
|
|
// need to track min/max column width for setting min/max table widths
|
|
|
|
PRInt32 minColWidth = 0;
|
|
|
|
PRInt32 maxColWidth = 0;
|
|
|
|
nsVoidArray *cells = colData->GetCells();
|
|
|
|
PRInt32 numCells = cells->Count();
|
|
|
|
if (gsDebug==PR_TRUE) printf (" for column %d numCells = %d\n", colIndex, numCells);
|
|
|
|
|
|
|
|
#if XXX_need_access_to_column_frame_help
|
|
|
|
// Get the columns's style
|
|
|
|
nsIStyleContextPtr colSC;
|
|
|
|
colFrame->GetStyleContext(aPresContext, colSC.AssignRef());
|
|
|
|
nsStylePosition* colPosition = (nsStylePosition*)
|
1998-05-21 06:34:13 +04:00
|
|
|
colSC->GetData(eStyleStruct_Position);
|
1998-04-30 21:57:09 +04:00
|
|
|
|
|
|
|
// Get column width if it has one
|
|
|
|
PRBool haveColWidth = PR_FALSE;
|
|
|
|
nscoord colWidth;
|
1998-04-30 23:55:51 +04:00
|
|
|
switch (colPosition->mWidth.GetUnit()) {
|
1998-04-30 21:57:09 +04:00
|
|
|
default:
|
1998-04-30 23:55:51 +04:00
|
|
|
case eStyleUnit_Auto:
|
|
|
|
case eStyleUnit_Inherit:
|
1998-04-30 21:57:09 +04:00
|
|
|
break;
|
|
|
|
|
1998-04-30 23:55:51 +04:00
|
|
|
case eStyleUnit_Coord:
|
1998-04-30 21:57:09 +04:00
|
|
|
haveColWidth = PR_TRUE;
|
1998-04-30 23:55:51 +04:00
|
|
|
colWidth = colPosition->mWidth.GetCoordValue;
|
1998-04-30 21:57:09 +04:00
|
|
|
break;
|
|
|
|
|
1998-04-30 23:55:51 +04:00
|
|
|
case eStyleUnit_Percent:
|
|
|
|
case eStyleUnit_Proportional:
|
1998-04-30 21:57:09 +04:00
|
|
|
//XXX haveColWidth = PR_TRUE;
|
|
|
|
//XXX colWidth = colPosition->mWidthPCT * something;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Scan the column, simulatneously assigning fixed column widths
|
|
|
|
* and computing the min/max column widths
|
|
|
|
*/
|
|
|
|
// first, deal with any cells that span into this column from a pervious column
|
|
|
|
if (nsnull!=spanList)
|
|
|
|
{
|
|
|
|
PRInt32 spanCount = spanList->Count();
|
|
|
|
// go through the list backwards so we can delete easily
|
|
|
|
for (PRInt32 spanIndex=spanCount-1; 0<=spanIndex; spanIndex--)
|
|
|
|
{
|
|
|
|
SpanInfo *spanInfo = (SpanInfo *)(spanList->ElementAt(spanIndex));
|
|
|
|
if (minColWidth < spanInfo->cellMinWidth)
|
|
|
|
minColWidth = spanInfo->cellMinWidth;
|
|
|
|
if (maxColWidth < spanInfo->cellDesiredWidth)
|
|
|
|
maxColWidth = spanInfo->cellDesiredWidth;
|
|
|
|
spanInfo->span--;
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" for spanning cell %d with remaining span=%d, min = %d and des = %d\n",
|
|
|
|
spanIndex, spanInfo->span, spanInfo->cellMinWidth, spanInfo->cellDesiredWidth);
|
|
|
|
if (0==spanInfo->span)
|
|
|
|
{
|
|
|
|
spanList->RemoveElementAt(spanIndex);
|
|
|
|
delete spanInfo;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (PRInt32 cellIndex = 0; cellIndex<numCells; cellIndex++)
|
|
|
|
{
|
|
|
|
nsCellLayoutData * data = (nsCellLayoutData *)(cells->ElementAt(cellIndex));
|
|
|
|
NS_ASSERTION(nsnull != data, "bad data");
|
1998-05-22 03:43:18 +04:00
|
|
|
|
|
|
|
nsMargin margin;
|
|
|
|
nsresult result = data->GetMargin(margin);
|
|
|
|
|
1998-04-30 21:57:09 +04:00
|
|
|
nsSize * cellMinSize = data->GetMaxElementSize();
|
|
|
|
nsReflowMetrics * cellDesiredSize = data->GetDesiredSize();
|
|
|
|
NS_ASSERTION(nsnull != cellDesiredSize, "bad cellDesiredSize");
|
|
|
|
PRInt32 colSpan = data->GetCellFrame()->GetColSpan();
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" for cell %d with colspan=%d, min = %d,%d and des = %d,%d\n",
|
|
|
|
cellIndex, colSpan, cellMinSize->width, cellMinSize->height,
|
|
|
|
cellDesiredSize->width, cellDesiredSize->height);
|
|
|
|
|
|
|
|
PRBool haveCellWidth = PR_FALSE;
|
|
|
|
nscoord cellWidth;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The first cell in a column (in row 0) has special standing.
|
|
|
|
* if the first cell has a width specification, it overrides the
|
|
|
|
* COL width
|
|
|
|
*/
|
|
|
|
if (0==cellIndex)
|
|
|
|
{
|
|
|
|
nsCellLayoutData * data = (nsCellLayoutData *)(cells->ElementAt(0));
|
1998-05-22 03:43:18 +04:00
|
|
|
nsMargin margin;
|
|
|
|
nsresult result = data->GetMargin(margin);
|
1998-04-30 21:57:09 +04:00
|
|
|
nsTableCellFrame *cellFrame = data->GetCellFrame();
|
|
|
|
nsTableCellPtr cell;
|
|
|
|
cellFrame->GetContent((nsIContent*&)(cell.AssignRef())); // cell: REFCNT++
|
|
|
|
|
|
|
|
// Get the cell's style
|
|
|
|
nsIStyleContextPtr cellSC;
|
|
|
|
cellFrame->GetStyleContext(aPresContext, cellSC.AssignRef());
|
|
|
|
nsStylePosition* cellPosition = (nsStylePosition*)
|
1998-05-21 06:34:13 +04:00
|
|
|
cellSC->GetData(eStyleStruct_Position);
|
1998-04-30 23:55:51 +04:00
|
|
|
switch (cellPosition->mWidth.GetUnit()) {
|
|
|
|
case eStyleUnit_Coord:
|
1998-04-30 21:57:09 +04:00
|
|
|
haveCellWidth = PR_TRUE;
|
1998-04-30 23:55:51 +04:00
|
|
|
cellWidth = cellPosition->mWidth.GetCoordValue();
|
1998-04-30 21:57:09 +04:00
|
|
|
break;
|
|
|
|
|
1998-04-30 23:55:51 +04:00
|
|
|
case eStyleUnit_Percent:
|
|
|
|
case eStyleUnit_Proportional:
|
1998-04-30 21:57:09 +04:00
|
|
|
// XXX write me when pct/proportional are supported
|
|
|
|
// XXX haveCellWidth = PR_TRUE;
|
|
|
|
// XXX cellWidth = cellPosition->mWidth;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
1998-04-30 23:55:51 +04:00
|
|
|
case eStyleUnit_Inherit:
|
|
|
|
case eStyleUnit_Auto:
|
1998-04-30 21:57:09 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if XXX_need_access_to_column_frame_help
|
1998-04-30 23:55:51 +04:00
|
|
|
switch (colPosition->mWidth.GetUnit()) {
|
1998-04-30 21:57:09 +04:00
|
|
|
default:
|
1998-04-30 23:55:51 +04:00
|
|
|
case eStyleUnit_Auto:
|
|
|
|
case eStyleUnit_Inherit:
|
1998-04-30 21:57:09 +04:00
|
|
|
break;
|
|
|
|
|
1998-04-30 23:55:51 +04:00
|
|
|
case eStyleUnit_Coord:
|
1998-04-30 21:57:09 +04:00
|
|
|
{
|
|
|
|
// This col has a fixed width, so set the cell's width to the
|
|
|
|
// larger of (specified width, largest max_element_size of the
|
|
|
|
// cells in the column)
|
1998-04-30 23:55:51 +04:00
|
|
|
PRInt32 widthForThisCell = max(cellMinSize->width, colPosition->mWidth.GetCoordValue());
|
1998-04-30 21:57:09 +04:00
|
|
|
if (mTableFrame->GetColumnWidth(colIndex) < widthForThisCell)
|
|
|
|
{
|
|
|
|
if (gsDebug) printf (" setting fixed width to %d\n",widthForThisCell);
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, widthForThisCell);
|
|
|
|
maxColWidth = widthForThisCell;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NS_STYLE_POSITION_VALUE_PCT:
|
|
|
|
case NS_STYLE_POSITION_VALUE_PROPORTIONAL:
|
|
|
|
// XXX write me when pct/proportional are supported
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// regardless of the width specification, keep track of the
|
|
|
|
// min/max column widths
|
1998-05-22 03:43:18 +04:00
|
|
|
nscoord cellMinWidth = cellMinSize->width/colSpan;
|
|
|
|
nscoord cellDesiredWidth = cellDesiredSize->width/colSpan;
|
|
|
|
|
|
|
|
cellMinWidth += margin.left + margin.right;
|
|
|
|
cellDesiredWidth += margin.left + margin.right;
|
|
|
|
|
1998-04-30 21:57:09 +04:00
|
|
|
if (minColWidth < cellMinWidth)
|
|
|
|
minColWidth = cellMinWidth;
|
|
|
|
if (maxColWidth < cellDesiredWidth)
|
|
|
|
maxColWidth = cellDesiredWidth;
|
|
|
|
if (1<colSpan)
|
|
|
|
{
|
|
|
|
// add the cell to our list of spanners
|
|
|
|
SpanInfo *spanInfo = new SpanInfo(colSpan-1, cellMinWidth, cellDesiredWidth);
|
|
|
|
if (nsnull==spanList)
|
|
|
|
spanList = new nsVoidArray();
|
|
|
|
spanList->AppendElement(spanInfo);
|
|
|
|
}
|
|
|
|
if (gsDebug) {
|
|
|
|
printf (" after cell %d, minColWidth = %d and maxColWidth = %d\n",
|
|
|
|
cellIndex, minColWidth, maxColWidth);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
// if the col is fixed-width, expand the col to the specified
|
|
|
|
// fixed width if necessary
|
|
|
|
if (colStyle->fixedWidth > mTableFrame->GetColumnWidth(colIndex))
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, colStyle->fixedWidth);
|
|
|
|
|
|
|
|
// keep a running total of the amount of space taken up by all
|
|
|
|
// fixed-width columns
|
|
|
|
aTotalFixedWidth += mTableFrame->GetColumnWidths(colIndex);
|
|
|
|
if (gsDebug) {
|
|
|
|
printf (" after col %d, aTotalFixedWidth = %d\n",
|
|
|
|
colIndex, aTotalFixedWidth);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// add col[i] metrics to the running totals for the table min/max width
|
|
|
|
if (NS_UNCONSTRAINEDSIZE!=aMinTableWidth)
|
|
|
|
aMinTableWidth += minColWidth; // SEC: insets!
|
|
|
|
if (aMinTableWidth<=0)
|
|
|
|
aMinTableWidth = NS_UNCONSTRAINEDSIZE; // handle overflow
|
|
|
|
if (NS_UNCONSTRAINEDSIZE!=aMaxTableWidth)
|
|
|
|
aMaxTableWidth += maxColWidth; // SEC: insets!
|
|
|
|
if (aMaxTableWidth<=0)
|
|
|
|
aMaxTableWidth = NS_UNCONSTRAINEDSIZE; // handle overflow
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" after this col, minTableWidth = %d and maxTableWidth = %d\n", aMinTableWidth, aMaxTableWidth);
|
|
|
|
|
|
|
|
} // end Step 1 for fixed-width columns
|
|
|
|
if (nsnull!=spanList)
|
|
|
|
delete spanList;
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
PRBool BasicTableLayoutStrategy::BalanceProportionalColumns(nsIPresContext* aPresContext,
|
|
|
|
PRInt32 aAvailWidth,
|
|
|
|
PRInt32 aMaxWidth,
|
|
|
|
PRInt32 aMinTableWidth,
|
|
|
|
PRInt32 aMaxTableWidth,
|
|
|
|
nscoord aTableFixedWidth)
|
1998-04-30 21:57:09 +04:00
|
|
|
{
|
|
|
|
PRBool result = PR_TRUE;
|
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
if (0==aTableFixedWidth)
|
|
|
|
{
|
|
|
|
if (NS_UNCONSTRAINEDSIZE==aMaxWidth)
|
|
|
|
{ // the max width of the table fits comfortably in the available space
|
|
|
|
if (gsDebug) printf (" * table laying out in NS_UNCONSTRAINEDSIZE, calling BalanceColumnsTableFits\n");
|
|
|
|
result = BalanceColumnsTableFits(aPresContext, aAvailWidth, aTableFixedWidth);
|
|
|
|
}
|
|
|
|
else if (aMinTableWidth > aMaxWidth)
|
|
|
|
{ // the table doesn't fit in the available space
|
|
|
|
if (gsDebug) printf (" * min table does not fit, calling SetColumnsToMinWidth\n");
|
|
|
|
result = SetColumnsToMinWidth(aPresContext);
|
|
|
|
}
|
|
|
|
else if (aMaxTableWidth <= aMaxWidth)
|
|
|
|
{ // the max width of the table fits comfortably in the available space
|
|
|
|
if (gsDebug) printf (" * table desired size fits, calling BalanceColumnsTableFits\n");
|
|
|
|
result = BalanceColumnsTableFits(aPresContext, aAvailWidth, aTableFixedWidth);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // the table fits somewhere between its min and desired size
|
|
|
|
if (gsDebug) printf (" * table desired size does not fit, calling BalanceColumnsConstrained\n");
|
|
|
|
result = BalanceColumnsConstrained(aPresContext, aAvailWidth,
|
|
|
|
aMaxWidth, aMinTableWidth, aMaxTableWidth);
|
|
|
|
}
|
1998-04-30 21:57:09 +04:00
|
|
|
}
|
|
|
|
else
|
1998-05-27 02:03:16 +04:00
|
|
|
{
|
|
|
|
if (aTableFixedWidth<aMinTableWidth)
|
|
|
|
{ // the table doesn't fit in the available space
|
|
|
|
if (gsDebug) printf (" * specified width table with width<minTableWidth, calling SetColumnsToMinWidth\n");
|
|
|
|
result = SetColumnsToMinWidth(aPresContext);
|
|
|
|
}
|
|
|
|
else if (aTableFixedWidth<aMaxTableWidth)
|
|
|
|
{ // the table's specified width is between it's min and max, so fit the columns in proportionately
|
|
|
|
if (gsDebug) printf (" * specified width table < maxTableWidth, calling BalanceColumnsConstrained\n");
|
|
|
|
result = BalanceColumnsConstrained(aPresContext, aAvailWidth,
|
|
|
|
aMaxTableWidth, aMinTableWidth, aMaxTableWidth);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // the table's specified width is >= its max width, so give each column its max requested size
|
|
|
|
if (gsDebug) printf (" * specified width table > maxTableWidth, calling BalanceColumnsTableFits\n");
|
|
|
|
result = BalanceColumnsTableFits(aPresContext, aAvailWidth, aTableFixedWidth);
|
|
|
|
}
|
1998-04-30 21:57:09 +04:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool BasicTableLayoutStrategy::SetColumnsToMinWidth(nsIPresContext* aPresContext)
|
|
|
|
{
|
|
|
|
PRBool result = PR_TRUE;
|
|
|
|
nsVoidArray *columnLayoutData = mTableFrame->GetColumnLayoutData();
|
|
|
|
PRInt32 numCols = columnLayoutData->Count();
|
|
|
|
for (PRInt32 colIndex = 0; colIndex<numCols; colIndex++)
|
|
|
|
{
|
|
|
|
nsColLayoutData * colData = (nsColLayoutData *)(columnLayoutData->ElementAt(colIndex));
|
|
|
|
nsTableColPtr col = colData->GetCol(); // col: ADDREF++
|
|
|
|
nsVoidArray *cells = colData->GetCells();
|
|
|
|
PRInt32 minColWidth = 0;
|
|
|
|
PRInt32 maxColWidth = 0;
|
|
|
|
PRInt32 numCells = cells->Count();
|
|
|
|
if (gsDebug==PR_TRUE) printf (" for col %d\n", colIndex);
|
|
|
|
|
|
|
|
// XXX need column frame to ask this question
|
|
|
|
nsStylePosition* colPosition = nsnull;
|
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
if (PR_TRUE==IsProportionalWidth(colPosition))
|
1998-04-30 21:57:09 +04:00
|
|
|
{
|
|
|
|
for (PRInt32 cellIndex = 0; cellIndex<numCells; cellIndex++)
|
|
|
|
{ // this col has proportional width, so determine its width requirements
|
|
|
|
nsCellLayoutData * data = (nsCellLayoutData *)(cells->ElementAt(cellIndex));
|
|
|
|
NS_ASSERTION(nsnull != data, "bad data");
|
|
|
|
nsSize * cellMinSize = data->GetMaxElementSize();
|
|
|
|
NS_ASSERTION(nsnull != cellMinSize, "bad cellMinSize");
|
|
|
|
nsReflowMetrics * cellDesiredSize = data->GetDesiredSize();
|
|
|
|
NS_ASSERTION(nsnull != cellDesiredSize, "bad cellDesiredSize");
|
|
|
|
if (minColWidth < cellMinSize->width)
|
|
|
|
minColWidth = cellMinSize->width;
|
|
|
|
if (maxColWidth < cellDesiredSize->width)
|
|
|
|
maxColWidth = cellDesiredSize->width;
|
|
|
|
/*
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" after cell %d, minColWidth = %d and maxColWidth = %d\n",
|
|
|
|
cellIndex, minColWidth, maxColWidth);
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
{
|
|
|
|
printf (" for determining width of col %d %s:\n",
|
1998-05-27 02:03:16 +04:00
|
|
|
colIndex, IsProportionalWidth(colPosition)? "(P)":"(A)");
|
1998-04-30 21:57:09 +04:00
|
|
|
printf (" minColWidth = %d and maxColWidth = %d\n", minColWidth, maxColWidth);
|
|
|
|
}
|
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
mTableFrame->SetColumnWidth(colIndex, minColWidth);
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" 2: col %d, set to width = %d\n", colIndex, mTableFrame->GetColumnWidth(colIndex));
|
1998-04-30 21:57:09 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool BasicTableLayoutStrategy::BalanceColumnsTableFits(nsIPresContext* aPresContext,
|
1998-05-27 02:03:16 +04:00
|
|
|
PRInt32 aAvailWidth,
|
|
|
|
nscoord aTableFixedWidth)
|
1998-04-30 21:57:09 +04:00
|
|
|
{
|
|
|
|
#ifdef DEBUG
|
|
|
|
nsIFrame *tablePIF=nsnull;
|
|
|
|
mTableFrame->GetPrevInFlow(tablePIF);
|
|
|
|
NS_ASSERTION(nsnull==tablePIF, "never ever call me on a continuing frame!");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
PRBool result = PR_TRUE;
|
1998-05-27 02:03:16 +04:00
|
|
|
nscoord tableWidth=0;
|
|
|
|
nscoord numProportionalColumns = 0;
|
1998-04-30 21:57:09 +04:00
|
|
|
nsVoidArray *spanList=nsnull;
|
|
|
|
nsVoidArray *columnLayoutData = mTableFrame->GetColumnLayoutData();
|
|
|
|
PRInt32 numCols = columnLayoutData->Count();
|
|
|
|
for (PRInt32 colIndex = 0; colIndex<numCols; colIndex++)
|
|
|
|
{
|
|
|
|
nsColLayoutData * colData = (nsColLayoutData *)(columnLayoutData->ElementAt(colIndex));
|
|
|
|
nsTableColPtr col = colData->GetCol(); // col: ADDREF++
|
|
|
|
nsVoidArray *cells = colData->GetCells();
|
|
|
|
PRInt32 minColWidth = 0;
|
|
|
|
PRInt32 maxColWidth = 0;
|
|
|
|
PRInt32 numCells = cells->Count();
|
|
|
|
if (gsDebug==PR_TRUE) printf (" for col %d\n", colIndex);
|
|
|
|
|
|
|
|
// XXX Need columnFrame to ask the style question
|
|
|
|
nsStylePosition* colPosition = nsnull;
|
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
if (PR_TRUE==IsProportionalWidth(colPosition))
|
1998-04-30 21:57:09 +04:00
|
|
|
{
|
1998-05-27 02:03:16 +04:00
|
|
|
numProportionalColumns++;
|
1998-04-30 21:57:09 +04:00
|
|
|
// first, deal with any cells that span into this column from a pervious column
|
|
|
|
if (nsnull!=spanList)
|
|
|
|
{
|
|
|
|
PRInt32 spanCount = spanList->Count();
|
|
|
|
// go through the list backwards so we can delete easily
|
|
|
|
for (PRInt32 spanIndex=spanCount-1; 0<=spanIndex; spanIndex--)
|
|
|
|
{
|
|
|
|
SpanInfo *spanInfo = (SpanInfo *)(spanList->ElementAt(spanIndex));
|
|
|
|
if (minColWidth < spanInfo->cellMinWidth)
|
|
|
|
minColWidth = spanInfo->cellMinWidth;
|
|
|
|
if (maxColWidth < spanInfo->cellDesiredWidth)
|
|
|
|
maxColWidth = spanInfo->cellDesiredWidth;
|
|
|
|
spanInfo->span--;
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" for spanning cell %d with remaining span=%d, min = %d and des = %d\n",
|
|
|
|
spanIndex, spanInfo->span, spanInfo->cellMinWidth, spanInfo->cellDesiredWidth);
|
|
|
|
if (0==spanInfo->span)
|
|
|
|
{
|
|
|
|
spanList->RemoveElementAt(spanIndex);
|
|
|
|
delete spanInfo;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (PRInt32 cellIndex = 0; cellIndex<numCells; cellIndex++)
|
|
|
|
{ // this col has proportional width, so determine its width requirements
|
|
|
|
nsCellLayoutData * data = (nsCellLayoutData *)(cells->ElementAt(cellIndex));
|
|
|
|
NS_ASSERTION(nsnull != data, "bad data");
|
|
|
|
|
1998-05-22 03:43:18 +04:00
|
|
|
|
1998-04-30 21:57:09 +04:00
|
|
|
PRInt32 colSpan = data->GetCellFrame()->GetColSpan();
|
|
|
|
// distribute a portion of the spanning cell's min and max width to this column
|
|
|
|
nsSize * cellMinSize = data->GetMaxElementSize();
|
|
|
|
NS_ASSERTION(nsnull != cellMinSize, "bad cellMinSize");
|
|
|
|
nsReflowMetrics * cellDesiredSize = data->GetDesiredSize();
|
|
|
|
NS_ASSERTION(nsnull != cellDesiredSize, "bad cellDesiredSize");
|
1998-05-22 03:43:18 +04:00
|
|
|
|
|
|
|
nsMargin margin;
|
|
|
|
nsresult result = data->GetMargin(margin);
|
|
|
|
nscoord marginWidth = margin.left + margin.right;
|
|
|
|
PRInt32 cellMinWidth = cellMinSize->width/colSpan + marginWidth;
|
|
|
|
PRInt32 cellDesiredWidth = cellDesiredSize->width/colSpan + marginWidth;
|
|
|
|
|
1998-04-30 21:57:09 +04:00
|
|
|
if (PR_TRUE==gsDebug)
|
|
|
|
printf("factoring in cell %d with colSpan=%d\n factoring in min=%d and desired=%d\n",
|
|
|
|
cellIndex, colSpan, cellMinWidth, cellDesiredWidth);
|
|
|
|
if (minColWidth < cellMinWidth)
|
|
|
|
minColWidth = cellMinWidth;
|
|
|
|
if (maxColWidth < cellDesiredWidth)
|
|
|
|
maxColWidth = cellDesiredWidth;
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" after cell %d, minColWidth = %d and maxColWidth = %d\n",
|
|
|
|
cellIndex, minColWidth, maxColWidth);
|
|
|
|
if (1<colSpan)
|
|
|
|
{ // add the cell to our list of spanners
|
|
|
|
SpanInfo *spanInfo = new SpanInfo(colSpan-1, cellMinWidth, cellDesiredWidth);
|
|
|
|
if (nsnull==spanList)
|
|
|
|
spanList = new nsVoidArray();
|
|
|
|
spanList->AppendElement(spanInfo);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
{
|
|
|
|
printf (" for determining width of col %d %s:\n",
|
1998-05-27 02:03:16 +04:00
|
|
|
colIndex, IsProportionalWidth(colPosition)? "(P)":"(A)");
|
1998-04-30 21:57:09 +04:00
|
|
|
printf (" minColWidth = %d and maxColWidth = %d\n", minColWidth, maxColWidth);
|
|
|
|
printf (" aAvailWidth = %d\n", aAvailWidth);
|
|
|
|
}
|
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
|
1998-04-30 21:57:09 +04:00
|
|
|
#if XXX_bug_kipp_about_this
|
1998-05-27 02:03:16 +04:00
|
|
|
if (0==colStyle->proportionalWidth)
|
|
|
|
{ // col width is specified to be the minimum
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, minColWidth);
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" 3 (0): col %d set to min width = %d because style set proportionalWidth=0\n",
|
|
|
|
colIndex, mTableFrame->GetColumnWidth(colIndex));
|
|
|
|
}
|
|
|
|
else // BUG? else? other code below has the else
|
1998-04-30 21:57:09 +04:00
|
|
|
#endif
|
1998-05-27 02:03:16 +04:00
|
|
|
if (PR_TRUE==IsAutoWidth(colPosition))
|
|
|
|
{ // give each remaining column it's desired width
|
|
|
|
// if there is width left over, we'll factor that in after this loop is complete
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, maxColWidth);
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" 3a: col %d with availWidth %d, set to width = %d\n",
|
|
|
|
colIndex, aAvailWidth, mTableFrame->GetColumnWidth(colIndex));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // give each remaining column an equal percentage of the remaining space
|
|
|
|
PRInt32 percentage = -1;
|
|
|
|
if (NS_UNCONSTRAINEDSIZE==aAvailWidth)
|
|
|
|
{
|
1998-04-30 21:57:09 +04:00
|
|
|
mTableFrame->SetColumnWidth(colIndex, maxColWidth);
|
|
|
|
}
|
|
|
|
else
|
1998-05-27 02:03:16 +04:00
|
|
|
{
|
1998-04-30 21:57:09 +04:00
|
|
|
#if XXX_bug_kipp_about_this
|
1998-05-27 02:03:16 +04:00
|
|
|
percentage = colStyle->proportionalWidth;
|
|
|
|
if (-1==percentage)
|
1998-04-30 21:57:09 +04:00
|
|
|
#endif
|
1998-05-27 02:03:16 +04:00
|
|
|
percentage = 100/numCols;
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, (percentage*aAvailWidth)/100);
|
|
|
|
// if the column was computed to be too small, enlarge the column
|
|
|
|
if (mTableFrame->GetColumnWidth(colIndex) <= minColWidth)
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, minColWidth);
|
|
|
|
}
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" 3b: col %d given %d percent of availWidth %d, set to width = %d\n",
|
|
|
|
colIndex, percentage, aAvailWidth, mTableFrame->GetColumnWidth(colIndex));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tableWidth += mTableFrame->GetColumnWidth(colIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
// post-process if necessary
|
|
|
|
|
|
|
|
// if the specified width of the table is greater than the table's computed width, expand the
|
|
|
|
// table's computed width to match the specified width, giving the extra space to proportionately-sized
|
|
|
|
// columns if possible.
|
|
|
|
if (aTableFixedWidth > tableWidth)
|
|
|
|
{
|
|
|
|
nscoord excess = aTableFixedWidth - tableWidth;
|
|
|
|
// if there are proportionally-sized columns, give them the extra space
|
|
|
|
if (0!=numProportionalColumns)
|
|
|
|
{
|
|
|
|
nscoord excessPerColumn = excess/numProportionalColumns;
|
|
|
|
if (gsDebug==PR_TRUE) printf(" aTableFixedWidth specified as %d, expanding columns by excess = %d\n", aTableFixedWidth, excess);
|
|
|
|
for (PRInt32 colIndex = 0; colIndex<numCols; colIndex++)
|
|
|
|
{
|
|
|
|
// need the column style here!
|
|
|
|
if (PR_TRUE==IsProportionalWidth(nsnull))
|
|
|
|
{
|
|
|
|
nscoord colWidth = excessPerColumn+mTableFrame->GetColumnWidth(colIndex);
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, colWidth);
|
1998-04-30 21:57:09 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
1998-05-27 02:03:16 +04:00
|
|
|
// otherwise, distribute the space evenly between all the columns
|
|
|
|
else
|
|
|
|
{
|
|
|
|
nscoord excessPerColumn = excess/numCols;
|
|
|
|
if (gsDebug==PR_TRUE) printf(" aTableFixedWidth specified as %d, expanding columns by excess = %d\n", aTableFixedWidth, excess);
|
|
|
|
for (PRInt32 colIndex = 0; colIndex<numCols; colIndex++)
|
|
|
|
{
|
|
|
|
nscoord colWidth = excessPerColumn+mTableFrame->GetColumnWidth(colIndex);
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, colWidth);
|
|
|
|
}
|
|
|
|
}
|
1998-04-30 21:57:09 +04:00
|
|
|
}
|
1998-05-27 02:03:16 +04:00
|
|
|
|
1998-04-30 21:57:09 +04:00
|
|
|
if (nsnull!=spanList)
|
|
|
|
delete spanList;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
1998-04-30 22:04:16 +04:00
|
|
|
PRBool BasicTableLayoutStrategy::BalanceColumnsConstrained( nsIPresContext* aPresContext,
|
|
|
|
PRInt32 aAvailWidth,
|
|
|
|
PRInt32 aMaxWidth,
|
|
|
|
PRInt32 aMinTableWidth,
|
|
|
|
PRInt32 aMaxTableWidth)
|
1998-04-30 21:57:09 +04:00
|
|
|
{
|
|
|
|
#ifdef DEBUG
|
|
|
|
nsIFrame *tablePIF=nsnull;
|
|
|
|
mTableFrame->GetPrevInFlow(tablePIF);
|
|
|
|
NS_ASSERTION(nsnull==tablePIF, "never ever call me on a continuing frame!");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
PRBool result = PR_TRUE;
|
1998-05-27 02:03:16 +04:00
|
|
|
PRBool equalWidthColumns=PR_FALSE; // TODO: compute this via style system
|
1998-04-30 21:57:09 +04:00
|
|
|
PRInt32 maxOfAllMinColWidths = 0;
|
|
|
|
nsVoidArray *spanList=nsnull;
|
|
|
|
nsVoidArray *columnLayoutData = mTableFrame->GetColumnLayoutData();
|
|
|
|
PRInt32 numCols = columnLayoutData->Count();
|
|
|
|
for (PRInt32 colIndex = 0; colIndex<numCols; colIndex++)
|
|
|
|
{
|
|
|
|
nsColLayoutData * colData = (nsColLayoutData *)(columnLayoutData->ElementAt(colIndex));
|
|
|
|
nsTableColPtr col = colData->GetCol(); // col: ADDREF++
|
|
|
|
|
|
|
|
#if XXX_bug_kipp_about_this
|
|
|
|
// XXX BUG: mStyleContext is for the table frame not for the column.
|
|
|
|
nsStyleMolecule* colStyle =
|
1998-05-21 06:34:13 +04:00
|
|
|
(nsStyleMolecule*)mStyleContext->GetData(eStyleStruct_Molecule);
|
1998-04-30 21:57:09 +04:00
|
|
|
#else
|
|
|
|
nsStylePosition* colPosition = nsnull;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
nsVoidArray *cells = colData->GetCells();
|
|
|
|
PRInt32 minColWidth = 0;
|
|
|
|
PRInt32 maxColWidth = 0;
|
|
|
|
PRInt32 numCells = cells->Count();
|
|
|
|
if (gsDebug==PR_TRUE) printf (" for col %d\n", colIndex);
|
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
if (PR_TRUE==IsProportionalWidth(colPosition))
|
1998-04-30 21:57:09 +04:00
|
|
|
{
|
|
|
|
// first, deal with any cells that span into this column from a pervious column
|
|
|
|
if (nsnull!=spanList)
|
|
|
|
{
|
|
|
|
PRInt32 spanCount = spanList->Count();
|
|
|
|
// go through the list backwards so we can delete easily
|
|
|
|
for (PRInt32 spanIndex=spanCount-1; 0<=spanIndex; spanIndex--)
|
|
|
|
{
|
|
|
|
SpanInfo *spanInfo = (SpanInfo *)(spanList->ElementAt(spanIndex));
|
|
|
|
if (minColWidth < spanInfo->cellMinWidth)
|
|
|
|
minColWidth = spanInfo->cellMinWidth;
|
|
|
|
if (maxColWidth < spanInfo->cellDesiredWidth)
|
|
|
|
maxColWidth = spanInfo->cellDesiredWidth;
|
|
|
|
spanInfo->span--;
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" for spanning cell %d with remaining span=%d, min = %d and des = %d\n",
|
|
|
|
spanIndex, spanInfo->span, spanInfo->cellMinWidth, spanInfo->cellDesiredWidth);
|
|
|
|
if (0==spanInfo->span)
|
|
|
|
{
|
|
|
|
spanList->RemoveElementAt(spanIndex);
|
|
|
|
delete spanInfo;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (PRInt32 cellIndex = 0; cellIndex<numCells; cellIndex++)
|
|
|
|
{ // this col has proportional width, so determine its width requirements
|
|
|
|
nsCellLayoutData * data = (nsCellLayoutData *)(cells->ElementAt(cellIndex));
|
|
|
|
NS_ASSERTION(nsnull != data, "bad data");
|
|
|
|
PRInt32 colSpan = data->GetCellFrame()->GetColSpan();
|
|
|
|
nsSize * cellMinSize = data->GetMaxElementSize();
|
|
|
|
NS_ASSERTION(nsnull != cellMinSize, "bad cellMinSize");
|
|
|
|
nsReflowMetrics * cellDesiredSize = data->GetDesiredSize();
|
|
|
|
NS_ASSERTION(nsnull != cellDesiredSize, "bad cellDesiredSize");
|
1998-05-22 03:43:18 +04:00
|
|
|
|
|
|
|
nsMargin margin;
|
|
|
|
nsresult result = data->GetMargin(margin);
|
|
|
|
nscoord marginWidth = margin.left + margin.right;
|
|
|
|
|
|
|
|
PRInt32 cellMinWidth = cellMinSize->width/colSpan + marginWidth;
|
|
|
|
PRInt32 cellDesiredWidth = cellDesiredSize->width/colSpan + marginWidth;
|
1998-04-30 21:57:09 +04:00
|
|
|
if (minColWidth < cellMinWidth)
|
|
|
|
minColWidth = cellMinWidth;
|
|
|
|
if (maxColWidth < cellDesiredWidth)
|
|
|
|
maxColWidth = cellDesiredWidth;
|
|
|
|
if (1<colSpan)
|
|
|
|
{
|
|
|
|
// add the cell to our list of spanners
|
|
|
|
SpanInfo *spanInfo = new SpanInfo(colSpan-1, cellMinWidth, cellDesiredWidth);
|
|
|
|
if (nsnull==spanList)
|
|
|
|
spanList = new nsVoidArray();
|
|
|
|
spanList->AppendElement(spanInfo);
|
|
|
|
}
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" after cell %d, minColWidth = %d and maxColWidth = %d\n",
|
|
|
|
cellIndex, minColWidth, maxColWidth);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
{
|
|
|
|
printf (" for determining width of col %d %s:\n",
|
1998-05-27 02:03:16 +04:00
|
|
|
colIndex, IsProportionalWidth(colPosition)? "(P)":"(A)");
|
1998-04-30 21:57:09 +04:00
|
|
|
printf (" minTableWidth = %d and maxTableWidth = %d\n", aMinTableWidth, aMaxTableWidth);
|
|
|
|
printf (" minColWidth = %d and maxColWidth = %d\n", minColWidth, maxColWidth);
|
|
|
|
printf (" aAvailWidth = %d\n", aAvailWidth);
|
|
|
|
}
|
|
|
|
|
1998-05-27 02:03:16 +04:00
|
|
|
// this col has proportional width, so set its width based on the table width
|
|
|
|
// the table fits in the space somewhere between its min and max size
|
|
|
|
// so dole out the available space appropriately
|
1998-04-30 21:57:09 +04:00
|
|
|
#if XXX_bug_kipp_about_this
|
1998-05-27 02:03:16 +04:00
|
|
|
if (0==colStyle->proportionalWidth)
|
|
|
|
{ // col width is specified to be the minimum
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, minColWidth);
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" 4 (0): col %d set to min width = %d because style set proportionalWidth=0\n",
|
|
|
|
colIndex, mTableFrame->GetColumnWidth(colIndex));
|
|
|
|
}
|
|
|
|
else
|
1998-04-30 21:57:09 +04:00
|
|
|
#endif
|
1998-05-27 02:03:16 +04:00
|
|
|
if (1==numCols)
|
|
|
|
{ // there is only one column, so it should be as wide as the available space allows it to be
|
|
|
|
if (gsDebug==PR_TRUE) printf (" 4 one-column: col %d set to width = %d\n", colIndex, aAvailWidth);
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, aAvailWidth);
|
|
|
|
}
|
|
|
|
else if (IsAutoWidth(colPosition))
|
|
|
|
{ // column's width is determined by its content
|
|
|
|
PRUint32 W = aMaxWidth - aMinTableWidth;
|
|
|
|
PRUint32 D = aMaxTableWidth - aMinTableWidth;
|
|
|
|
if (0==D) // fixed-size table
|
|
|
|
D=1;
|
|
|
|
PRUint32 d = maxColWidth - minColWidth;
|
|
|
|
PRInt32 width = (d*W)/D;
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, minColWidth + width);
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
printf (" 4 auto-width: col %d W=%d D=%d d=%d, set to width = %d\n",
|
|
|
|
colIndex, W, D, d, mTableFrame->GetColumnWidth(colIndex));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // give each remaining column an equal percentage of the remaining space
|
1998-04-30 21:57:09 +04:00
|
|
|
#if XXX_bug_kipp_about_this
|
1998-05-27 02:03:16 +04:00
|
|
|
PRInt32 percentage = colStyle->proportionalWidth;
|
|
|
|
if (-1==percentage)
|
1998-04-30 21:57:09 +04:00
|
|
|
#endif
|
1998-05-27 02:03:16 +04:00
|
|
|
PRInt32 percentage = 100/numCols;
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, (percentage*aAvailWidth)/100);
|
|
|
|
// if the column was computed to be too small, enlarge the column
|
|
|
|
if (mTableFrame->GetColumnWidth(colIndex) <= minColWidth)
|
|
|
|
{
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, minColWidth);
|
|
|
|
if (maxOfAllMinColWidths < minColWidth)
|
|
|
|
maxOfAllMinColWidths = minColWidth;
|
|
|
|
}
|
|
|
|
if (gsDebug==PR_TRUE)
|
|
|
|
{
|
|
|
|
printf (" 4 equal width: col %d given %d percent of availWidth %d, set to width = %d\n",
|
|
|
|
colIndex, percentage, aAvailWidth, mTableFrame->GetColumnWidth(colIndex));
|
|
|
|
if (0!=maxOfAllMinColWidths)
|
|
|
|
printf(" and setting maxOfAllMins to %d\n", maxOfAllMinColWidths);
|
1998-04-30 21:57:09 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// post-process if necessary
|
|
|
|
|
|
|
|
// if columns have equal width, and some column's content couldn't squeeze into the computed size,
|
|
|
|
// then expand every column to the min size of the column with the largest min size
|
1998-05-27 02:03:16 +04:00
|
|
|
if (equalWidthColumns && 0!=maxOfAllMinColWidths)
|
1998-04-30 21:57:09 +04:00
|
|
|
{
|
|
|
|
if (gsDebug==PR_TRUE) printf(" EqualColWidths specified, so setting all col widths to %d\n", maxOfAllMinColWidths);
|
|
|
|
for (PRInt32 colIndex = 0; colIndex<numCols; colIndex++)
|
|
|
|
mTableFrame->SetColumnWidth(colIndex, maxOfAllMinColWidths);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nsnull!=spanList)
|
|
|
|
delete spanList;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|