mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-07 11:56:51 +00:00
03e8dd2540
we have nsFrameList
716 lines
26 KiB
C++
716 lines
26 KiB
C++
/* -*- 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 "nsTableColGroupFrame.h"
|
|
#include "nsTableColFrame.h"
|
|
#include "nsTableFrame.h"
|
|
#include "nsIHTMLTableColElement.h"
|
|
#include "nsIReflowCommand.h"
|
|
#include "nsIStyleContext.h"
|
|
#include "nsStyleConsts.h"
|
|
#include "nsIPresContext.h"
|
|
#include "nsIHTMLContent.h"
|
|
#include "nsHTMLParts.h"
|
|
#include "nsIPtr.h"
|
|
#include "nsHTMLAtoms.h"
|
|
#include "nsHTMLIIDs.h"
|
|
#include "nsCOMPtr.h"
|
|
|
|
NS_DEF_PTR(nsIContent);
|
|
|
|
static NS_DEFINE_IID(kIHTMLTableColElementIID, NS_IHTMLTABLECOLELEMENT_IID);
|
|
|
|
|
|
#ifdef NS_DEBUG
|
|
static PRBool gsDebug = PR_FALSE;
|
|
static PRBool gsDebugIR = PR_FALSE;
|
|
#else
|
|
static const PRBool gsDebug = PR_FALSE;
|
|
static const PRBool gsDebugIR = PR_FALSE;
|
|
#endif
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsTableColGroupFrame::InitNewFrames(nsIPresContext& aPresContext, nsIFrame* aChildList)
|
|
{
|
|
nsresult rv=NS_OK;
|
|
nsTableFrame* tableFrame=nsnull;
|
|
rv = nsTableFrame::GetTableFrame(this, tableFrame);
|
|
if ((NS_SUCCEEDED(rv)) && (nsnull!=tableFrame))
|
|
{
|
|
// Process the newly added column frames
|
|
for (nsIFrame* kidFrame = aChildList; nsnull != kidFrame; kidFrame->GetNextSibling(&kidFrame)) {
|
|
// Set the preliminary values for the column frame
|
|
PRInt32 colIndex = mStartColIndex + mColCount;
|
|
((nsTableColFrame *)(kidFrame))->InitColFrame (colIndex);
|
|
PRInt32 repeat = ((nsTableColFrame *)(kidFrame))->GetSpan();
|
|
mColCount += repeat;
|
|
for (PRInt32 i=0; i<repeat; i++)
|
|
{
|
|
tableFrame->AddColumnFrame((nsTableColFrame *)kidFrame);
|
|
}
|
|
}
|
|
// colgroup's span attribute is how many columns the group represents
|
|
// in the absence of any COL children
|
|
// Note that this is the correct, though perhaps unexpected, behavior for the span attribute.
|
|
// The spec says that if there are any COL children, the colgroup's span is ignored.
|
|
if (0==mColCount)
|
|
{
|
|
nsIFrame *firstImplicitColFrame=nsnull;
|
|
nsIFrame *prevColFrame=nsnull;
|
|
nsAutoString colTag;
|
|
nsHTMLAtoms::col->ToString(colTag);
|
|
mColCount = GetSpan();
|
|
for (PRInt32 colIndex=0; colIndex<mColCount; colIndex++)
|
|
{
|
|
nsIHTMLContent *col=nsnull;
|
|
// create an implicit col
|
|
rv = NS_CreateHTMLElement(&col, colTag); // ADDREF: col++
|
|
//XXX mark the col implicit
|
|
mContent->AppendChildTo((nsIContent*)col, PR_FALSE);
|
|
|
|
// Create a new col frame
|
|
nsIFrame* colFrame;
|
|
NS_NewTableColFrame(colFrame);
|
|
|
|
// Set its style context
|
|
nsCOMPtr<nsIStyleContext> colStyleContext;
|
|
aPresContext.ResolveStyleContextFor(col, mStyleContext,
|
|
PR_TRUE,
|
|
getter_AddRefs(colStyleContext));
|
|
colFrame->Init(aPresContext, col, this, colStyleContext, nsnull);
|
|
colFrame->SetInitialChildList(aPresContext, nsnull, nsnull);
|
|
|
|
// Set nsColFrame-specific information
|
|
PRInt32 absColIndex = mStartColIndex + colIndex;
|
|
((nsTableColFrame *)(colFrame))->InitColFrame (absColIndex);
|
|
((nsTableColFrame *)colFrame)->SetColumnIndex(absColIndex);
|
|
tableFrame->AddColumnFrame((nsTableColFrame *)colFrame);
|
|
|
|
//hook into list of children
|
|
if (nsnull==firstImplicitColFrame)
|
|
firstImplicitColFrame = colFrame;
|
|
else
|
|
prevColFrame->SetNextSibling(colFrame);
|
|
prevColFrame = colFrame;
|
|
}
|
|
|
|
// hook new columns into col group child list
|
|
mFrames.AppendFrames(nsnull, firstImplicitColFrame);
|
|
}
|
|
SetStyleContextForFirstPass(aPresContext);
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsTableColGroupFrame::AppendNewFrames(nsIPresContext& aPresContext, nsIFrame* aChildList)
|
|
{
|
|
if (nsnull!=aChildList)
|
|
mFrames.AppendFrames(nsnull, aChildList);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsTableColGroupFrame::SetInitialChildList(nsIPresContext& aPresContext,
|
|
nsIAtom* aListName,
|
|
nsIFrame* aChildList)
|
|
{
|
|
nsresult result = AppendNewFrames(aPresContext, aChildList);
|
|
if (NS_OK==result)
|
|
result = InitNewFrames(aPresContext, aChildList);
|
|
return result;
|
|
}
|
|
|
|
NS_METHOD nsTableColGroupFrame::Paint(nsIPresContext& aPresContext,
|
|
nsIRenderingContext& aRenderingContext,
|
|
const nsRect& aDirtyRect,
|
|
nsFramePaintLayer aWhichLayer)
|
|
{
|
|
if (gsDebug==PR_TRUE) printf("nsTableColGroupFrame::Paint\n");
|
|
PaintChildren(aPresContext, aRenderingContext, aDirtyRect, aWhichLayer);
|
|
return NS_OK;
|
|
}
|
|
|
|
PRIntn
|
|
nsTableColGroupFrame::GetSkipSides() const
|
|
{
|
|
PRIntn skip = 0;
|
|
if (nsnull != mPrevInFlow) {
|
|
skip |= 1 << NS_SIDE_TOP;
|
|
}
|
|
if (nsnull != mNextInFlow) {
|
|
skip |= 1 << NS_SIDE_BOTTOM;
|
|
}
|
|
return skip;
|
|
}
|
|
|
|
NS_METHOD nsTableColGroupFrame::Reflow(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus)
|
|
{
|
|
NS_ASSERTION(nsnull!=mContent, "bad state -- null content for frame");
|
|
nsresult rv=NS_OK;
|
|
// for every content child that (is a column thingy and does not already have a frame)
|
|
// create a frame and adjust it's style
|
|
nsIFrame* kidFrame = nsnull;
|
|
|
|
if (eReflowReason_Incremental == aReflowState.reason) {
|
|
rv = IncrementalReflow(aPresContext, aDesiredSize, aReflowState, aStatus);
|
|
}
|
|
|
|
for (kidFrame = mFrames.FirstChild(); nsnull != kidFrame;
|
|
kidFrame->GetNextSibling(&kidFrame)) {
|
|
// Give the child frame a chance to reflow, even though we know it'll have 0 size
|
|
nsHTMLReflowMetrics kidSize(nsnull);
|
|
// XXX Use a valid reason...
|
|
nsHTMLReflowState kidReflowState(aPresContext, aReflowState, kidFrame,
|
|
nsSize(0,0), eReflowReason_Initial);
|
|
|
|
nsReflowStatus status;
|
|
ReflowChild(kidFrame, aPresContext, kidSize, kidReflowState, status);
|
|
// note that DidReflow is called as the result of some ancestor firing off a DidReflow above me
|
|
kidFrame->SetRect(nsRect(0,0,0,0));
|
|
}
|
|
|
|
aDesiredSize.width=0;
|
|
aDesiredSize.height=0;
|
|
aDesiredSize.ascent=aDesiredSize.height;
|
|
aDesiredSize.descent=0;
|
|
if (nsnull!=aDesiredSize.maxElementSize)
|
|
{
|
|
aDesiredSize.maxElementSize->width=0;
|
|
aDesiredSize.maxElementSize->height=0;
|
|
}
|
|
aStatus = NS_FRAME_COMPLETE;
|
|
return rv;
|
|
}
|
|
|
|
NS_METHOD nsTableColGroupFrame::IncrementalReflow(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus)
|
|
{
|
|
if (PR_TRUE==gsDebugIR) printf("\nTCGF IR: IncrementalReflow\n");
|
|
nsresult rv = NS_OK;
|
|
|
|
// determine if this frame is the target or not
|
|
nsIFrame *target=nsnull;
|
|
rv = aReflowState.reflowCommand->GetTarget(target);
|
|
if ((PR_TRUE==NS_SUCCEEDED(rv)) && (nsnull!=target))
|
|
{
|
|
if (this==target)
|
|
rv = IR_TargetIsMe(aPresContext, aDesiredSize, aReflowState, aStatus);
|
|
else
|
|
{
|
|
// Get the next frame in the reflow chain
|
|
nsIFrame* nextFrame;
|
|
aReflowState.reflowCommand->GetNext(nextFrame);
|
|
rv = IR_TargetIsChild(aPresContext, aDesiredSize, aReflowState, aStatus, nextFrame);
|
|
}
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
NS_METHOD nsTableColGroupFrame::IR_TargetIsMe(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus)
|
|
{
|
|
nsresult rv = NS_OK;
|
|
aStatus = NS_FRAME_COMPLETE;
|
|
nsIReflowCommand::ReflowType type;
|
|
aReflowState.reflowCommand->GetType(type);
|
|
nsIFrame *objectFrame;
|
|
aReflowState.reflowCommand->GetChildFrame(objectFrame);
|
|
const nsStyleDisplay *childDisplay=nsnull;
|
|
if (nsnull!=objectFrame)
|
|
objectFrame->GetStyleData(eStyleStruct_Display, ((const nsStyleStruct *&)childDisplay));
|
|
if (PR_TRUE==gsDebugIR) printf("nTCGF IR: IncrementalReflow_TargetIsMe with type=%d\n", type);
|
|
switch (type)
|
|
{
|
|
case nsIReflowCommand::FrameInserted :
|
|
NS_ASSERTION(nsnull!=objectFrame, "bad objectFrame");
|
|
NS_ASSERTION(nsnull!=childDisplay, "bad childDisplay");
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN == childDisplay->mDisplay)
|
|
{
|
|
rv = IR_ColInserted(aPresContext, aDesiredSize, aReflowState, aStatus,
|
|
(nsTableColFrame*)objectFrame, PR_FALSE);
|
|
}
|
|
else
|
|
{
|
|
rv = AddFrame(aReflowState, objectFrame);
|
|
}
|
|
break;
|
|
|
|
case nsIReflowCommand::FrameAppended :
|
|
NS_ASSERTION(nsnull!=objectFrame, "bad objectFrame");
|
|
NS_ASSERTION(nsnull!=childDisplay, "bad childDisplay");
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN_GROUP == childDisplay->mDisplay)
|
|
{
|
|
rv = IR_ColAppended(aPresContext, aDesiredSize, aReflowState, aStatus,
|
|
(nsTableColFrame*)objectFrame);
|
|
}
|
|
else
|
|
{ // no optimization to be done for Unknown frame types, so just reuse the Inserted method
|
|
rv = AddFrame(aReflowState, objectFrame);
|
|
}
|
|
break;
|
|
|
|
/*
|
|
case nsIReflowCommand::FrameReplaced :
|
|
|
|
*/
|
|
|
|
case nsIReflowCommand::FrameRemoved :
|
|
NS_ASSERTION(nsnull!=objectFrame, "bad objectFrame");
|
|
NS_ASSERTION(nsnull!=childDisplay, "bad childDisplay");
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN_GROUP == childDisplay->mDisplay)
|
|
{
|
|
rv = IR_ColRemoved(aPresContext, aDesiredSize, aReflowState, aStatus,
|
|
(nsTableColFrame*)objectFrame);
|
|
}
|
|
else
|
|
{
|
|
|
|
rv = mFrames.RemoveFrame(objectFrame);
|
|
}
|
|
break;
|
|
|
|
case nsIReflowCommand::StyleChanged :
|
|
rv = IR_StyleChanged(aPresContext, aDesiredSize, aReflowState, aStatus);
|
|
break;
|
|
|
|
case nsIReflowCommand::ContentChanged :
|
|
NS_ASSERTION(PR_FALSE, "illegal reflow type: ContentChanged");
|
|
rv = NS_ERROR_ILLEGAL_VALUE;
|
|
break;
|
|
|
|
case nsIReflowCommand::PullupReflow:
|
|
case nsIReflowCommand::PushReflow:
|
|
case nsIReflowCommand::CheckPullupReflow :
|
|
case nsIReflowCommand::UserDefined :
|
|
NS_NOTYETIMPLEMENTED("unimplemented reflow command type");
|
|
rv = NS_ERROR_NOT_IMPLEMENTED;
|
|
if (PR_TRUE==gsDebugIR) printf("TCGF IR: reflow command not implemented.\n");
|
|
break;
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_METHOD nsTableColGroupFrame::IR_ColInserted(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus,
|
|
nsTableColFrame * aInsertedFrame,
|
|
PRBool aReplace)
|
|
{
|
|
nsresult rv=NS_OK;
|
|
PRBool adjustStartingColIndex=PR_FALSE;
|
|
rv = AddFrame(aReflowState, (nsIFrame *)aInsertedFrame);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
PRInt32 startingColIndex=mStartColIndex;
|
|
startingColIndex += GetColumnCount(); // has the side effect of resetting all column indexes
|
|
|
|
nsIFrame *childFrame=nsnull;
|
|
GetNextSibling(&childFrame);
|
|
while ((NS_SUCCEEDED(rv)) && (nsnull!=childFrame))
|
|
{
|
|
const nsStyleDisplay *display;
|
|
childFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct *&)display);
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN_GROUP == display->mDisplay)
|
|
{
|
|
startingColIndex += ((nsTableColGroupFrame *)childFrame)->SetStartColumnIndex(startingColIndex);
|
|
}
|
|
rv = childFrame->GetNextSibling(&childFrame);
|
|
}
|
|
|
|
nsTableFrame* tableFrame=nsnull;
|
|
rv = nsTableFrame::GetTableFrame(this, tableFrame);
|
|
if ((NS_SUCCEEDED(rv)) && (nsnull!=tableFrame))
|
|
tableFrame->InvalidateColumnCache();
|
|
//XXX: what we want to do here is determine if the new COL information changes anything about layout
|
|
// if not, we can skip rebalancing the columns
|
|
return rv;
|
|
}
|
|
|
|
NS_METHOD nsTableColGroupFrame::IR_ColAppended(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus,
|
|
nsTableColFrame * aAppendedFrame)
|
|
{
|
|
nsresult rv=NS_OK;
|
|
PRBool adjustStartingColIndex=PR_FALSE;
|
|
rv = AddFrame(aReflowState, (nsIFrame*)aAppendedFrame);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
PRInt32 startingColIndex=mStartColIndex;
|
|
// this could be optimized
|
|
startingColIndex += GetColumnCount(); // has the side effect of resetting all column indexes
|
|
|
|
nsIFrame *childFrame=nsnull;
|
|
GetNextSibling(&childFrame);
|
|
while ((NS_SUCCEEDED(rv)) && (nsnull!=childFrame))
|
|
{
|
|
const nsStyleDisplay *display;
|
|
childFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct *&)display);
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN_GROUP == display->mDisplay)
|
|
{
|
|
startingColIndex += ((nsTableColGroupFrame *)childFrame)->SetStartColumnIndex(startingColIndex);
|
|
}
|
|
rv = childFrame->GetNextSibling(&childFrame);
|
|
}
|
|
|
|
// today we need to rebuild the whole column cache
|
|
// if the table frame is ever recoded to build the column cache incrementally, we could take
|
|
// advantage of that here.
|
|
nsTableFrame* tableFrame=nsnull;
|
|
rv = nsTableFrame::GetTableFrame(this, tableFrame);
|
|
if ((NS_SUCCEEDED(rv)) && (nsnull!=tableFrame))
|
|
tableFrame->InvalidateColumnCache();
|
|
return rv;
|
|
}
|
|
|
|
|
|
NS_METHOD nsTableColGroupFrame::IR_ColRemoved(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus,
|
|
nsTableColFrame * aDeletedFrame)
|
|
{
|
|
nsresult rv=NS_OK;
|
|
PRInt32 startingColIndex=mStartColIndex;
|
|
nsIFrame *childFrame=mFrames.FirstChild();
|
|
nsIFrame *prevSib=nsnull;
|
|
while ((NS_SUCCEEDED(rv)) && (nsnull!=childFrame))
|
|
{
|
|
// XXX change to use mFrames.DeleteFrame; and why does it keep
|
|
// looping after finding it?
|
|
if (childFrame==aDeletedFrame)
|
|
{
|
|
nsIFrame *deleteFrameNextSib=nsnull;
|
|
aDeletedFrame->GetNextSibling(&deleteFrameNextSib);
|
|
if (nsnull!=prevSib)
|
|
prevSib->SetNextSibling(deleteFrameNextSib);
|
|
else
|
|
mFrames.SetFrames(deleteFrameNextSib);
|
|
startingColIndex += GetColumnCount(); // resets all column indexes
|
|
}
|
|
const nsStyleDisplay *display;
|
|
childFrame->GetStyleData(eStyleStruct_Display, (const nsStyleStruct *&)display);
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN_GROUP == display->mDisplay)
|
|
{
|
|
// we've removed aDeletedFrame, now adjust the starting col index of all subsequent col groups
|
|
startingColIndex += ((nsTableColGroupFrame *)childFrame)->SetStartColumnIndex(startingColIndex);
|
|
}
|
|
prevSib=childFrame;
|
|
rv = childFrame->GetNextSibling(&childFrame);
|
|
}
|
|
|
|
// today we need to rebuild the whole column cache
|
|
// if the table frame is ever recoded to build the column cache incrementally, we could take
|
|
// advantage of that here.
|
|
nsTableFrame* tableFrame=nsnull;
|
|
rv = nsTableFrame::GetTableFrame(this, tableFrame);
|
|
if ((NS_SUCCEEDED(rv)) && (nsnull!=tableFrame))
|
|
tableFrame->InvalidateColumnCache();
|
|
return rv;
|
|
}
|
|
|
|
NS_METHOD nsTableColGroupFrame::IR_StyleChanged(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus)
|
|
{
|
|
if (PR_TRUE==gsDebugIR) printf("TIF IR: IR_StyleChanged for frame %p\n", this);
|
|
nsresult rv = NS_OK;
|
|
// we presume that all the easy optimizations were done in the nsHTMLStyleSheet before we were called here
|
|
// XXX: we can optimize this when we know which style attribute changed
|
|
nsTableFrame* tableFrame=nsnull;
|
|
rv = nsTableFrame::GetTableFrame(this, tableFrame);
|
|
if ((NS_SUCCEEDED(rv)) && (nsnull!=tableFrame))
|
|
{
|
|
tableFrame->InvalidateColumnCache();
|
|
tableFrame->InvalidateFirstPassCache();
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
NS_METHOD nsTableColGroupFrame::IR_TargetIsChild(nsIPresContext& aPresContext,
|
|
nsHTMLReflowMetrics& aDesiredSize,
|
|
const nsHTMLReflowState& aReflowState,
|
|
nsReflowStatus& aStatus,
|
|
nsIFrame * aNextFrame)
|
|
{
|
|
nsresult rv;
|
|
if (PR_TRUE==gsDebugIR) printf("\nTCGF IR: IR_TargetIsChild\n");
|
|
|
|
// Remember the old col count
|
|
const PRInt32 oldColCount = GetColumnCount();
|
|
|
|
// Pass along the reflow command
|
|
nsHTMLReflowMetrics desiredSize(nsnull);
|
|
nsHTMLReflowState kidReflowState(aPresContext, aReflowState, aNextFrame,
|
|
nsSize(aReflowState.availableWidth,
|
|
aReflowState.availableHeight));
|
|
rv = ReflowChild(aNextFrame, aPresContext, desiredSize, kidReflowState, aStatus);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
nsTableFrame *tableFrame=nsnull;
|
|
rv = nsTableFrame::GetTableFrame(this, tableFrame);
|
|
if ((NS_SUCCEEDED(rv)) && (nsnull!=tableFrame))
|
|
{
|
|
// compare the new col count to the old col count.
|
|
// If they are the same, we just need to rebalance column widths
|
|
// If they differ, we need to fix up other column groups and the column cache
|
|
const PRInt32 newColCount = GetColumnCount(); // this will set the new column indexes if necessary
|
|
if (oldColCount==newColCount)
|
|
tableFrame->InvalidateColumnWidths();
|
|
else
|
|
tableFrame->InvalidateColumnCache();
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
// Subclass hook for style post processing
|
|
NS_METHOD nsTableColGroupFrame::SetStyleContextForFirstPass(nsIPresContext& aPresContext)
|
|
{
|
|
// get the table frame
|
|
nsTableFrame* tableFrame=nsnull;
|
|
nsresult rv = nsTableFrame::GetTableFrame(this, tableFrame);
|
|
if ((NS_SUCCEEDED(rv)) && (nsnull!=tableFrame))
|
|
{
|
|
// get the style for the table frame
|
|
const nsStyleTable *tableStyle;
|
|
tableFrame->GetStyleData(eStyleStruct_Table, (const nsStyleStruct *&)tableStyle);
|
|
|
|
// if COLS is set, then map it into the COL frames
|
|
if (NS_STYLE_TABLE_COLS_NONE != tableStyle->mCols)
|
|
{
|
|
// set numCols to the number of columns effected by the COLS attribute
|
|
PRInt32 numCols=0;
|
|
if (NS_STYLE_TABLE_COLS_ALL == tableStyle->mCols)
|
|
numCols = mFrames.GetLength();
|
|
else
|
|
numCols = tableStyle->mCols;
|
|
|
|
// for every column effected, set its width style
|
|
PRInt32 colIndex=0;
|
|
nsIFrame *colFrame=mFrames.FirstChild();
|
|
while (nsnull!=colFrame)
|
|
{
|
|
const nsStyleDisplay * colDisplay=nsnull;
|
|
colFrame->GetStyleData(eStyleStruct_Display, ((const nsStyleStruct *&)colDisplay));
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN == colDisplay->mDisplay)
|
|
{
|
|
nsCOMPtr<nsIStyleContext> colStyleContext;
|
|
nsStylePosition * colPosition=nsnull;
|
|
colFrame->GetStyleContext(getter_AddRefs(colStyleContext));
|
|
colPosition = (nsStylePosition*)colStyleContext->GetMutableStyleData(eStyleStruct_Position);
|
|
if (colIndex<numCols)
|
|
{
|
|
nsStyleCoord width (1, eStyleUnit_Proportional);
|
|
colPosition->mWidth = width;
|
|
}
|
|
else
|
|
{
|
|
colPosition->mWidth.SetCoordValue(0);
|
|
}
|
|
colStyleContext->RecalcAutomaticData(&aPresContext);
|
|
colIndex++;
|
|
}
|
|
colFrame->GetNextSibling(&colFrame);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// propagate the colgroup width attribute down to the columns if they have no width of their own
|
|
nsStylePosition* position = (nsStylePosition*)mStyleContext->GetStyleData(eStyleStruct_Position);
|
|
if (eStyleUnit_Null!=position->mWidth.GetUnit())
|
|
{
|
|
// now for every column that doesn't have it's own width, set the width style
|
|
nsIFrame *colFrame=mFrames.FirstChild();
|
|
while (nsnull!=colFrame)
|
|
{
|
|
const nsStyleDisplay * colDisplay=nsnull;
|
|
colFrame->GetStyleData(eStyleStruct_Display, ((const nsStyleStruct *&)colDisplay));
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN == colDisplay->mDisplay)
|
|
{
|
|
nsCOMPtr<nsIStyleContext> colStyleContext;
|
|
const nsStylePosition * colPosition=nsnull;
|
|
colFrame->GetStyleData(eStyleStruct_Position, (const nsStyleStruct *&)colPosition); // get a read-only version of the style context
|
|
//XXX: how do I know this is auto because it's defaulted, vs. set explicitly to "auto"?
|
|
if (eStyleUnit_Auto==colPosition->mWidth.GetUnit())
|
|
{
|
|
// notice how we defer getting a mutable style context until we're sure we really need one
|
|
colFrame->GetStyleContext(getter_AddRefs(colStyleContext));
|
|
nsStylePosition * mutableColPosition = (nsStylePosition*)colStyleContext->GetMutableStyleData(eStyleStruct_Position);
|
|
mutableColPosition->mWidth = position->mWidth;
|
|
colStyleContext->RecalcAutomaticData(&aPresContext);
|
|
}
|
|
}
|
|
colFrame->GetNextSibling(&colFrame);
|
|
}
|
|
}
|
|
}
|
|
//mStyleContext->RecalcAutomaticData(&aPresContext);
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
|
|
/** returns the number of columns represented by this group.
|
|
* if there are col children, count them (taking into account the span of each)
|
|
* else, check my own span attribute.
|
|
*/
|
|
int nsTableColGroupFrame::GetColumnCount ()
|
|
{
|
|
mColCount=0;
|
|
nsIFrame *childFrame = mFrames.FirstChild();
|
|
while (nsnull!=childFrame)
|
|
{
|
|
const nsStyleDisplay *childDisplay;
|
|
childFrame->GetStyleData(eStyleStruct_Display, ((const nsStyleStruct *&)childDisplay));
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN == childDisplay->mDisplay)
|
|
{
|
|
nsTableColFrame *col = (nsTableColFrame *)childFrame;
|
|
col->SetColumnIndex (mStartColIndex + mColCount);
|
|
mColCount += col->GetSpan();
|
|
}
|
|
childFrame->GetNextSibling(&childFrame);
|
|
}
|
|
if (0==mColCount)
|
|
{ // there were no children of this colgroup that were columns. So use my span attribute
|
|
const nsStyleTable *tableStyle;
|
|
GetStyleData(eStyleStruct_Table, (const nsStyleStruct *&)tableStyle);
|
|
mColCount = tableStyle->mSpan;
|
|
}
|
|
return mColCount;
|
|
}
|
|
|
|
nsTableColFrame * nsTableColGroupFrame::GetFirstColumn()
|
|
{
|
|
return GetNextColumn(nsnull);
|
|
}
|
|
|
|
nsTableColFrame * nsTableColGroupFrame::GetNextColumn(nsIFrame *aChildFrame)
|
|
{
|
|
nsTableColFrame *result = nsnull;
|
|
nsIFrame *childFrame = aChildFrame;
|
|
if (nsnull==childFrame)
|
|
childFrame = mFrames.FirstChild();
|
|
while (nsnull!=childFrame)
|
|
{
|
|
const nsStyleDisplay *childDisplay;
|
|
childFrame->GetStyleData(eStyleStruct_Display, ((const nsStyleStruct *&)childDisplay));
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN == childDisplay->mDisplay)
|
|
{
|
|
result = (nsTableColFrame *)childFrame;
|
|
break;
|
|
}
|
|
childFrame->GetNextSibling(&childFrame);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
|
|
nsTableColFrame * nsTableColGroupFrame::GetColumnAt (PRInt32 aColIndex)
|
|
{
|
|
nsTableColFrame *result = nsnull;
|
|
PRInt32 count = 0;
|
|
nsIFrame *childFrame = mFrames.FirstChild();
|
|
|
|
while (nsnull!=childFrame) {
|
|
const nsStyleDisplay *childDisplay;
|
|
childFrame->GetStyleData(eStyleStruct_Display, ((const nsStyleStruct *&)childDisplay));
|
|
if (NS_STYLE_DISPLAY_TABLE_COLUMN == childDisplay->mDisplay) {
|
|
nsTableColFrame *col = (nsTableColFrame *)childFrame;
|
|
count += col->GetSpan();
|
|
if (aColIndex<=count) {
|
|
result = col;
|
|
}
|
|
}
|
|
childFrame->GetNextSibling(&childFrame);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
PRInt32 nsTableColGroupFrame::GetSpan()
|
|
{
|
|
PRInt32 span=1;
|
|
const nsStyleTable* tableStyle=nsnull;
|
|
GetStyleData(eStyleStruct_Table, (const nsStyleStruct *&)tableStyle);
|
|
if (nsnull!=tableStyle)
|
|
{
|
|
span = tableStyle->mSpan;
|
|
}
|
|
return span;
|
|
}
|
|
|
|
/* this may be needed when IsSynthetic is properly implemented
|
|
PRBool nsTableColGroupFrame::IsManufactured()
|
|
{
|
|
PRBool result = PR_FALSE;
|
|
nsIFrame *firstCol = GetFirstColumn();
|
|
if (nsTableFrame::IsSynthetic(this) &&
|
|
((nsnull==firstCol) || nsTableFrame::IsSynthetic(firstCol)))
|
|
result = PR_TRUE;
|
|
return result;
|
|
}
|
|
*/
|
|
|
|
/** returns colcount because it is frequently used in the context of
|
|
* shuffling relative colgroup order, and it's convenient to not have to
|
|
* call GetColumnCount redundantly.
|
|
*/
|
|
PRInt32 nsTableColGroupFrame::SetStartColumnIndex (int aIndex)
|
|
{
|
|
PRInt32 result = mColCount;
|
|
if (aIndex != mStartColIndex)
|
|
{
|
|
mStartColIndex = aIndex;
|
|
mColCount=0;
|
|
result = GetColumnCount(); // has the side effect of setting each column index based on new start index
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/* ----- global methods ----- */
|
|
|
|
nsresult
|
|
NS_NewTableColGroupFrame(nsIFrame*& aResult)
|
|
{
|
|
nsIFrame* it = new nsTableColGroupFrame;
|
|
if (nsnull == it) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
aResult = it;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsTableColGroupFrame::GetFrameName(nsString& aResult) const
|
|
{
|
|
return MakeFrameName("TableColGroup", aResult);
|
|
}
|