You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdelibs/khtml/rendering/render_table.cpp

3071 lines
99 KiB

/**
* This file is part of the DOM implementation for KDE.
*
* Copyright (C) 1997 Martin Jones (mjones@kde.org)
* (C) 1997 Torben Weis (weis@kde.org)
* (C) 1998 Waldo Bastian (bastian@kde.org)
* (C) 1999-2003 Lars Knoll (knoll@kde.org)
* (C) 1999 Antti Koivisto (koivisto@kde.org)
* (C) 2003 Apple Computer, Inc.
* (C) 2005 Allan Sandfeld Jensen (kde@carewolf.com)
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
//#define TABLE_DEBUG
//#define TABLE_PRINT
//#define DEBUG_LAYOUT
//#define BOX_DEBUG
#include "rendering/render_table.h"
#include "rendering/render_replaced.h"
#include "rendering/render_canvas.h"
#include "rendering/table_layout.h"
#include "html/html_tableimpl.h"
#include "html/html_formimpl.h"
#include "misc/htmltags.h"
#include "misc/htmlattrs.h"
#include "rendering/render_line.h"
#include "xml/dom_docimpl.h"
#include <kglobal.h>
#include <tqapplication.h>
#include <tqstyle.h>
#include <kdebug.h>
#include <assert.h>
using namespace khtml;
using namespace DOM;
RenderTable::RenderTable(DOM::NodeImpl* node)
: RenderBlock(node)
{
tCaption = 0;
head = foot = firstBody = 0;
tableLayout = 0;
m_currentBorder = 0;
has_col_elems = false;
hspacing = vspacing = 0;
padding = 0;
needSectionRecalc = false;
padding = 0;
columnPos.resize( 2 );
columnPos.fill( 0 );
columns.resize( 1 );
columns.fill( ColumnStruct() );
columnPos[0] = 0;
}
RenderTable::~RenderTable()
{
delete tableLayout;
}
void RenderTable::setStyle(RenderStyle *_style)
{
ETableLayout oldTableLayout = style() ? style()->tableLayout() : TAUTO;
if ( _style->display() == INLINE ) _style->setDisplay( INLINE_TABLE );
if ( _style->display() != INLINE_TABLE ) _style->setDisplay(TABLE);
if ( !_style->flowAroundFloats() ) _style->setFlowAroundFloats(true);
RenderBlock::setStyle(_style);
// init RenderObject attributes
setInline(style()->display()==INLINE_TABLE && !isPositioned());
setReplaced(style()->display()==INLINE_TABLE);
// In the collapsed border model, there is no cell spacing.
hspacing = collapseBorders() ? 0 : style()->borderHorizontalSpacing();
vspacing = collapseBorders() ? 0 : style()->borderVerticalSpacing();
columnPos[0] = hspacing;
if ( !tableLayout || style()->tableLayout() != oldTableLayout ) {
delete tableLayout;
// According to the CSS2 spec, you only use fixed table layout if an
// explicit width is specified on the table. Auto width implies auto table layout.
if (style()->tableLayout() == TFIXED && !style()->width().isVariable()) {
tableLayout = new FixedTableLayout(this);
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << "using fixed table layout" << endl;
#endif
} else
tableLayout = new AutoTableLayout(this);
}
}
short RenderTable::lineHeight(bool b) const
{
// Inline tables are replaced elements. Otherwise, just pass off to
// the base class.
if (isReplaced())
return height()+marginTop()+marginBottom();
return RenderBlock::lineHeight(b);
}
short RenderTable::baselinePosition(bool b) const
{
// Inline tables are replaced elements. Otherwise, just pass off to
// the base class.
if (isReplaced())
return height()+marginTop()+marginBottom();
return RenderBlock::baselinePosition(b);
}
void RenderTable::addChild(RenderObject *child, RenderObject *beforeChild)
{
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << renderName() << "(Table)::addChild( " << child->renderName() << ", " <<
(beforeChild ? beforeChild->renderName() : "0") << " )" << endl;
#endif
bool wrapInAnonymousSection = false;
switch(child->style()->display())
{
case TABLE_CAPTION:
if (child->isRenderBlock())
tCaption = static_cast<RenderBlock *>(child);
break;
case TABLE_COLUMN:
case TABLE_COLUMN_GROUP:
has_col_elems = true;
break;
case TABLE_HEADER_GROUP:
if ( !head ) {
if (child->isTableSection())
head = static_cast<RenderTableSection *>(child);
}
else if ( !firstBody )
if (child->isTableSection())
firstBody = static_cast<RenderTableSection *>(child);
break;
case TABLE_FOOTER_GROUP:
if ( !foot ) {
if (child->isTableSection())
foot = static_cast<RenderTableSection *>(child);
break;
}
// fall through
case TABLE_ROW_GROUP:
if(!firstBody)
if (child->isTableSection())
firstBody = static_cast<RenderTableSection *>(child);
break;
case TABLE_CELL:
case TABLE_ROW:
wrapInAnonymousSection = true;
break;
case BLOCK:
// case BOX:
case COMPACT:
case INLINE:
case INLINE_BLOCK:
// case INLINE_BOX:
case INLINE_TABLE:
case LIST_ITEM:
case NONE:
case RUN_IN:
case TABLE:
// The special TABLE > FORM quirk allows the form to sit directly under the table
if (child->element() && child->element()->isHTMLElement() && child->element()->id() == ID_FORM)
wrapInAnonymousSection = !static_cast<HTMLFormElementImpl*>(child->element())->isMalformed();
else
wrapInAnonymousSection = true;
break;
}
if (!wrapInAnonymousSection) {
RenderContainer::addChild(child, beforeChild);
return;
}
if (!beforeChild && lastChild() && lastChild()->isTableSection() && lastChild()->isAnonymous()) {
lastChild()->addChild(child);
return;
}
RenderObject *lastBox = beforeChild;
RenderObject *nextToLastBox = beforeChild;
while (lastBox && lastBox->parent()->isAnonymous() &&
!lastBox->isTableSection() && lastBox->style()->display() != TABLE_CAPTION) {
nextToLastBox = lastBox;
lastBox = lastBox->parent();
}
if (lastBox && lastBox->isAnonymous()) {
lastBox->addChild(child, nextToLastBox);
return;
}
if (beforeChild && !beforeChild->isTableSection())
beforeChild = 0;
RenderTableSection* section = new (renderArena()) RenderTableSection(document() /* anonymous */);
RenderStyle* newStyle = new RenderStyle();
newStyle->inheritFrom(style());
newStyle->setDisplay(TABLE_ROW_GROUP);
section->setStyle(newStyle);
addChild(section, beforeChild);
section->addChild(child);
}
void RenderTable::calcWidth()
{
if ( isPositioned() ) {
calcAbsoluteHorizontal();
}
RenderBlock *cb = containingBlock();
int availableWidth = cb->lineWidth( m_y );
LengthType widthType = style()->width().type();
if(widthType > Relative && style()->width().value() > 0) {
// Percent or fixed table
// Percent is calculated from contentWidth, not available width
m_width = calcBoxWidth(style()->width().minWidth( cb->contentWidth() ));
} else {
// Subtract out any fixed margins from our available width for auto width tables.
int marginTotal = 0;
if (!style()->marginLeft().isVariable())
marginTotal += style()->marginLeft().width(availableWidth);
if (!style()->marginRight().isVariable())
marginTotal += style()->marginRight().width(availableWidth);
// Subtract out our margins to get the available content width.
int availContentWidth = kMax(0, availableWidth - marginTotal);
// Ensure we aren't bigger than our max width or smaller than our min width.
m_width = kMin(availContentWidth, m_maxWidth);
}
m_width = kMax (m_width, m_minWidth);
// Finally, with our true width determined, compute our margins for real.
m_marginRight=0;
m_marginLeft=0;
calcHorizontalMargins(style()->marginLeft(),style()->marginRight(),availableWidth);
}
void RenderTable::layout()
{
KHTMLAssert( needsLayout() );
KHTMLAssert( minMaxKnown() );
KHTMLAssert( !needSectionRecalc );
if (posChildNeedsLayout() && !normalChildNeedsLayout() && !selfNeedsLayout()) {
// All we have to is lay out our positioned objects.
layoutPositionedObjects(true);
setNeedsLayout(false);
return;
}
if (markedForRepaint()) {
repaintDuringLayout();
setMarkedForRepaint(false);
}
m_height = 0;
initMaxMarginValues();
int oldWidth = m_width;
calcWidth();
m_overflowWidth = m_width;
if (tCaption && (oldWidth != m_width || tCaption->style()->height().isPercent()))
tCaption->setChildNeedsLayout(true);
// the optimization below doesn't work since the internal table
// layout could have changed. we need to add a flag to the table
// layout that tells us if something has changed in the min max
// calculations to do it correctly.
// if ( oldWidth != m_width || columns.size() + 1 != columnPos.size() )
tableLayout->layout();
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << renderName() << "(Table)::layout1() width=" << width() << ", marginLeft=" << marginLeft() << " marginRight=" << marginRight() << endl;
#endif
setCellWidths();
// layout child objects
int calculatedHeight = 0;
RenderObject *child = firstChild();
while( child ) {
// FIXME: What about a form that has a display value that makes it a table section?
if ( child->needsLayout() && !(child->element() && child->element()->id() == ID_FORM) )
child->layout();
if ( child->isTableSection() ) {
static_cast<RenderTableSection *>(child)->calcRowHeight();
calculatedHeight += static_cast<RenderTableSection *>(child)->layoutRows( 0 );
}
child = child->nextSibling();
}
// ### collapse caption margin
if(tCaption && tCaption->style()->captionSide() != CAPBOTTOM) {
tCaption->setPos(tCaption->marginLeft(), tCaption->marginTop()+m_height);
m_height += tCaption->height() + tCaption->marginTop() + tCaption->marginBottom();
}
int bpTop = borderTop() + (collapseBorders() ? 0 : paddingTop());
int bpBottom = borderBottom() + (collapseBorders() ? 0 : paddingBottom());
m_height += bpTop;
int oldHeight = m_height;
if (isPositioned())
m_height += calculatedHeight + bpBottom;
calcHeight();
int newHeight = m_height;
m_height = oldHeight;
Length h = style()->height();
int th = -(bpTop + bpBottom); // Tables size as though CSS height includes border/padding.
if (isPositioned())
th += newHeight;
else if (h.isFixed())
th += h.value();
else if (h.isPercent())
th += calcPercentageHeight(h);
// layout rows
if ( th > calculatedHeight ) {
// we have to redistribute that height to get the constraint correctly
// just force the first body to the height needed
// ### FIXME This should take height constraints on all table sections into account and distribute
// accordingly. For now this should be good enough
if (firstBody) {
firstBody->calcRowHeight();
firstBody->layoutRows( th - calculatedHeight );
}
else if (!style()->htmlHacks()) {
// Completely empty tables (with no sections or anything) should at least honor specified height
// in strict mode.
m_height += th;
}
}
int bl = borderLeft();
if (!collapseBorders())
bl += paddingLeft();
// position the table sections
if ( head ) {
head->setPos(bl, m_height);
m_height += head->height();
}
RenderObject *body = firstBody;
while ( body ) {
if ( body != head && body != foot && body->isTableSection() ) {
body->setPos(bl, m_height);
m_height += body->height();
}
body = body->nextSibling();
}
if ( foot ) {
foot->setPos(bl, m_height);
m_height += foot->height();
}
m_height += bpBottom;
if(tCaption && tCaption->style()->captionSide()==CAPBOTTOM) {
tCaption->setPos(tCaption->marginLeft(), tCaption->marginTop()+m_height);
m_height += tCaption->height() + tCaption->marginTop() + tCaption->marginBottom();
}
if (canvas()->pagedMode()) {
RenderObject *child = firstChild();
// relayout taking real position into account
while( child ) {
if ( !(child->element() && child->element()->id() == ID_FORM) ) {
child->setNeedsLayout(true);
child->layout();
if (child->containsPageBreak()) setContainsPageBreak(true);
if (child->needsPageClear()) setNeedsPageClear(true);
}
child = child->nextSibling();
}
}
//kdDebug(0) << "table height: " << m_height << endl;
// table can be containing block of positioned elements.
// ### only pass true if width or height changed.
layoutPositionedObjects( true );
m_overflowHeight = m_height;
setNeedsLayout(false);
}
void RenderTable::setCellWidths()
{
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << renderName() << "(Table, this=0x" << this << ")::setCellWidths()" << endl;
#endif
RenderObject *child = firstChild();
while( child ) {
if ( child->isTableSection() )
static_cast<RenderTableSection *>(child)->setCellWidths();
child = child->nextSibling();
}
}
void RenderTable::paint( PaintInfo& pI, int _tx, int _ty)
{
if(needsLayout()) return;
_tx += xPos();
_ty += yPos();
#ifdef TABLE_PRINT
kdDebug( 6040 ) << "RenderTable::paint() w/h = (" << width() << "/" << height() << ")" << endl;
#endif
if (!overhangingContents() && !isRelPositioned() && !isPositioned())
{
int os = 2*maximalOutlineSize(pI.phase);
if((_ty > pI.r.y() + pI.r.height() + os) || (_ty + height() < pI.r.y() - os)) return;
if((_tx > pI.r.x() + pI.r.width() + os) || (_tx + width() < pI.r.x() - os)) return;
}
#ifdef TABLE_PRINT
kdDebug( 6040 ) << "RenderTable::paint(2) " << _tx << "/" << _ty << " (" << _y << "/" << _h << ")" << endl;
#endif
if (pI.phase == PaintActionOutline)
paintOutline(pI.p, _tx, _ty, width(), height(), style());
if(( pI.phase == PaintActionElementBackground || pI.phase == PaintActionChildBackground )
&& shouldPaintBackgroundOrBorder() && style()->visibility() == VISIBLE)
paintBoxDecorations(pI, _tx, _ty);
if ( pI.phase == PaintActionElementBackground )
return;
PaintAction oldphase = pI.phase;
if ( pI.phase == PaintActionChildBackgrounds )
pI.phase = PaintActionChildBackground;
for( RenderObject *child = firstChild(); child; child = child->nextSibling())
if ( child->isTableSection() || child == tCaption )
child->paint( pI, _tx, _ty );
if (collapseBorders() &&
(pI.phase == PaintActionElementBackground || pI.phase == PaintActionChildBackground)
&& style()->visibility() == VISIBLE) {
// Collect all the unique border styles that we want to paint in a sorted list. Once we
// have all the styles sorted, we then do individual passes, painting each style of border
// from lowest precedence to highest precedence.
pI.phase = PaintActionCollapsedTableBorders;
TQValueList<CollapsedBorderValue> borderStyles;
collectBorders(borderStyles);
#if 0
TQString m;
for (uint i = 0; i < borderStyles.count(); i++)
m += TQString("%1 ").arg((*borderStyles.at(i)).width());
kdDebug(6040) << m << endl;
#endif
TQValueListIterator<CollapsedBorderValue> it = borderStyles.begin();
TQValueListIterator<CollapsedBorderValue> end = borderStyles.end();
for (; it != end; ++it) {
m_currentBorder = &*it;
for (RenderObject *child = firstChild(); child; child = child->nextSibling()) {
if (child->isTableSection())
child->paint(pI, _tx, _ty);
}
}
m_currentBorder = 0;
}
pI.phase = oldphase;
#ifdef BOX_DEBUG
outlineBox(p, _tx, _ty, "blue");
#endif
}
void RenderTable::paintBoxDecorations(PaintInfo &pI, int _tx, int _ty)
{
int w = width();
int h = height();
// Account for the caption.
if (tCaption) {
int captionHeight = (tCaption->height() + tCaption->marginBottom() + tCaption->marginTop());
h -= captionHeight;
if (tCaption->style()->captionSide() != CAPBOTTOM)
_ty += captionHeight;
}
int my = kMax(_ty,pI.r.y());
int mh;
if (_ty<pI.r.y())
mh= kMax(0,h-(pI.r.y()-_ty));
else
mh = kMin(pI.r.height(),h);
paintBackground(pI.p, style()->backgroundColor(), style()->backgroundLayers(), my, mh, _tx, _ty, w, h);
if (style()->hasBorder() && !collapseBorders())
paintBorder(pI.p, _tx, _ty, w, h, style());
}
void RenderTable::calcMinMaxWidth()
{
KHTMLAssert( !minMaxKnown() );
if ( needSectionRecalc )
recalcSections();
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << renderName() << "(Table " << this << ")::calcMinMaxWidth()" << endl;
#endif
tableLayout->calcMinMaxWidth();
if (tCaption) {
tCaption->calcWidth();
if (tCaption->marginLeft()+tCaption->marginRight()+tCaption->minWidth() > m_minWidth)
m_minWidth = tCaption->marginLeft()+tCaption->marginRight()+tCaption->minWidth();
}
setMinMaxKnown();
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << renderName() << " END: (Table " << this << ")::calcMinMaxWidth() min = " << m_minWidth << " max = " << m_maxWidth << endl;
#endif
}
void RenderTable::close()
{
// kdDebug( 6040 ) << "RenderTable::close()" << endl;
setNeedsLayoutAndMinMaxRecalc();
}
void RenderTable::splitColumn( int pos, int firstSpan )
{
// we need to add a new columnStruct
int oldSize = columns.size();
columns.resize( oldSize + 1 );
int oldSpan = columns[pos].span;
// qDebug("splitColumn( %d,%d ), oldSize=%d, oldSpan=%d", pos, firstSpan, oldSize, oldSpan );
KHTMLAssert( oldSpan > firstSpan );
columns[pos].span = firstSpan;
memmove( columns.data()+pos+1, columns.data()+pos, (oldSize-pos)*sizeof(ColumnStruct) );
columns[pos+1].span = oldSpan - firstSpan;
// change width of all rows.
RenderObject *child = firstChild();
while ( child ) {
if ( child->isTableSection() ) {
RenderTableSection *section = static_cast<RenderTableSection *>(child);
int size = section->grid.size();
int row = 0;
if ( section->cCol > pos )
section->cCol++;
while ( row < size ) {
section->grid[row].row->resize( oldSize+1 );
RenderTableSection::Row &r = *section->grid[row].row;
memmove( r.data()+pos+1, r.data()+pos, (oldSize-pos)*sizeof( RenderTableCell * ) );
// qDebug("moving from %d to %d, num=%d", pos, pos+1, (oldSize-pos-1) );
r[pos+1] = r[pos] ? (RenderTableCell *)-1 : 0;
row++;
}
}
child = child->nextSibling();
}
columnPos.resize( numEffCols()+1 );
setNeedsLayoutAndMinMaxRecalc();
}
void RenderTable::appendColumn( int span )
{
// easy case.
int pos = columns.size();
// qDebug("appendColumn( %d ), size=%d", span, pos );
int newSize = pos + 1;
columns.resize( newSize );
columns[pos].span = span;
//qDebug("appending column at %d, span %d", pos, span );
// change width of all rows.
RenderObject *child = firstChild();
while ( child ) {
if ( child->isTableSection() ) {
RenderTableSection *section = static_cast<RenderTableSection *>(child);
int size = section->grid.size();
int row = 0;
while ( row < size ) {
section->grid[row].row->resize( newSize );
section->cellAt( row, pos ) = 0;
row++;
}
}
child = child->nextSibling();
}
columnPos.resize( numEffCols()+1 );
setNeedsLayoutAndMinMaxRecalc();
}
RenderTableCol *RenderTable::colElement( int col ) {
if ( !has_col_elems )
return 0;
RenderObject *child = firstChild();
int cCol = 0;
while ( child ) {
if ( child->isTableCol() ) {
RenderTableCol *colElem = static_cast<RenderTableCol *>(child);
int span = colElem->span();
if ( !colElem->firstChild() ) {
cCol += span;
if ( cCol > col )
return colElem;
}
RenderObject *next = child->firstChild();
if ( !next )
next = child->nextSibling();
if ( !next && child->parent()->isTableCol() )
next = child->parent()->nextSibling();
child = next;
} else if (child == tCaption) {
child = child->nextSibling();
} else
break;
}
return 0;
}
void RenderTable::recalcSections()
{
tCaption = 0;
head = foot = firstBody = 0;
has_col_elems = false;
RenderObject *child = firstChild();
// We need to get valid pointers to caption, head, foot and firstbody again
while ( child ) {
switch(child->style()->display()) {
case TABLE_CAPTION:
if ( !tCaption && child->isRenderBlock() ) {
tCaption = static_cast<RenderBlock*>(child);
tCaption->setNeedsLayout(true);
}
break;
case TABLE_COLUMN:
case TABLE_COLUMN_GROUP:
has_col_elems = true;
break;
case TABLE_HEADER_GROUP:
if (child->isTableSection()) {
RenderTableSection *section = static_cast<RenderTableSection *>(child);
if (!head)
head = section;
else if (!firstBody)
firstBody = section;
if (section->needCellRecalc)
section->recalcCells();
}
break;
case TABLE_FOOTER_GROUP:
if (child->isTableSection()) {
RenderTableSection *section = static_cast<RenderTableSection *>(child);
if (!foot)
foot = section;
else if (!firstBody)
firstBody = section;
if (section->needCellRecalc)
section->recalcCells();
}
break;
case TABLE_ROW_GROUP:
if (child->isTableSection()) {
RenderTableSection *section = static_cast<RenderTableSection *>(child);
if (!firstBody)
firstBody = section;
if (section->needCellRecalc)
section->recalcCells();
}
break;
default:
break;
}
child = child->nextSibling();
}
needSectionRecalc = false;
setNeedsLayout(true);
}
RenderObject* RenderTable::removeChildNode(RenderObject* child)
{
setNeedSectionRecalc();
return RenderContainer::removeChildNode( child );
}
int RenderTable::borderLeft() const
{
if (collapseBorders()) {
// FIXME: For strict mode, returning 0 is correct, since the table border half spills into the margin,
// but I'm working to get this changed. For now, follow the spec.
return 0;
}
return RenderBlock::borderLeft();
}
int RenderTable::borderRight() const
{
if (collapseBorders()) {
// FIXME: For strict mode, returning 0 is correct, since the table border half spills into the margin,
// but I'm working to get this changed. For now, follow the spec.
return 0;
}
return RenderBlock::borderRight();
}
int RenderTable::borderTop() const
{
if (collapseBorders()) {
// FIXME: For strict mode, returning 0 is correct, since the table border half spills into the margin,
// but I'm working to get this changed. For now, follow the spec.
return 0;
}
return RenderBlock::borderTop();
}
int RenderTable::borderBottom() const
{
if (collapseBorders()) {
// FIXME: For strict mode, returning 0 is correct, since the table border half spills into the margin,
// but I'm working to get this changed. For now, follow the spec.
return 0;
}
return RenderBlock::borderBottom();
}
RenderTableSection* RenderTable::sectionAbove(const RenderTableSection* section, bool skipEmptySections) const
{
if (section == head)
return 0;
RenderObject *prevSection = (section == foot ? lastChild() : const_cast<RenderTableSection *>(section))->previousSibling();
while (prevSection) {
if (prevSection->isTableSection() && prevSection != head && prevSection != foot && (!skipEmptySections || static_cast<RenderTableSection*>(prevSection)->numRows()))
break;
prevSection = prevSection->previousSibling();
}
if (!prevSection && head && (!skipEmptySections || head->numRows()))
prevSection = head;
return static_cast<RenderTableSection*>(prevSection);
}
RenderTableSection* RenderTable::sectionBelow(const RenderTableSection* section, bool skipEmptySections) const
{
if (section == foot)
return 0;
RenderObject *nextSection = (section == head ? firstChild() : const_cast<RenderTableSection *>(section))->nextSibling();
while (nextSection) {
if (nextSection->isTableSection() && nextSection != head && nextSection != foot && (!skipEmptySections || static_cast<RenderTableSection*>(nextSection)->numRows()))
break;
nextSection = nextSection->nextSibling();
}
if (!nextSection && foot && (!skipEmptySections || foot->numRows()))
nextSection = foot;
return static_cast<RenderTableSection*>(nextSection);
}
RenderTableCell* RenderTable::cellAbove(const RenderTableCell* cell) const
{
// Find the section and row to look in
int r = cell->row();
RenderTableSection* section = 0;
int rAbove = 0;
if (r > 0) {
// cell is not in the first row, so use the above row in its own section
section = cell->section();
rAbove = r-1;
} else {
section = sectionAbove(cell->section(), true);
if (section)
rAbove = section->numRows() - 1;
}
// Look up the cell in the section's grid, which requires effective col index
if (section) {
int effCol = colToEffCol(cell->col());
RenderTableCell* aboveCell;
// If we hit a span back up to a real cell.
do {
aboveCell = section->cellAt(rAbove, effCol);
effCol--;
} while (aboveCell == (RenderTableCell *)-1 && effCol >=0);
return (aboveCell == (RenderTableCell *)-1) ? 0 : aboveCell;
} else {
return 0;
}
}
RenderTableCell* RenderTable::cellBelow(const RenderTableCell* cell) const
{
// Find the section and row to look in
int r = cell->row() + cell->rowSpan() - 1;
RenderTableSection* section = 0;
int rBelow = 0;
if (r < cell->section()->numRows()-1) {
// The cell is not in the last row, so use the next row in the section.
section = cell->section();
rBelow= r+1;
} else {
section = sectionBelow(cell->section(), true);
if (section)
rBelow = 0;
}
// Look up the cell in the section's grid, which requires effective col index
if (section) {
int effCol = colToEffCol(cell->col());
RenderTableCell* belowCell;
// If we hit a colspan back up to a real cell.
do {
belowCell = section->cellAt(rBelow, effCol);
effCol--;
} while (belowCell == (RenderTableCell *)-1 && effCol >=0);
return (belowCell == (RenderTableCell *)-1) ? 0 : belowCell;
} else {
return 0;
}
}
RenderTableCell* RenderTable::cellLeft(const RenderTableCell* cell) const
{
RenderTableSection* section = cell->section();
int effCol = colToEffCol(cell->col());
if (effCol == 0)
return 0;
// If we hit a colspan back up to a real cell.
RenderTableCell* prevCell;
do {
prevCell = section->cellAt(cell->row(), effCol-1);
effCol--;
} while (prevCell == (RenderTableCell *)-1 && effCol >=0);
return (prevCell == (RenderTableCell *)-1) ? 0 : prevCell;
}
RenderTableCell* RenderTable::cellRight(const RenderTableCell* cell) const
{
int effCol = colToEffCol(cell->col()+cell->colSpan());
if (effCol >= numEffCols())
return 0;
RenderTableCell* result = cell->section()->cellAt(cell->row(), effCol);
return (result == (RenderTableCell*)-1) ? 0 : result;
}
#ifdef ENABLE_DUMP
void RenderTable::dump(TQTextStream &stream, const TQString &ind) const
{
RenderBlock::dump(stream, ind);
if (tCaption)
stream << " tCaption";
if (head)
stream << " head";
if (foot)
stream << " foot";
stream << " [cspans:";
for ( unsigned int i = 0; i < columns.size(); i++ )
stream << " " << columns[i].span;
stream << "]";
}
#endif
FindSelectionResult RenderTable::checkSelectionPoint( int _x, int _y, int _tx, int _ty, DOM::NodeImpl*& node, int & offset, SelPointState &state )
{
int off = offset;
DOM::NodeImpl* nod = node;
FindSelectionResult pos;
TableSectionIterator it(this);
for (; *it; ++it) {
pos = (*it)->checkSelectionPoint(_x, _y, _tx + m_x, _ty + m_y, nod, off, state);
switch(pos) {
case SelectionPointBeforeInLine:
case SelectionPointInside:
//kdDebug(6030) << "RenderTable::checkSelectionPoint " << this << " returning SelectionPointInside offset=" << offset << endl;
node = nod;
offset = off;
return SelectionPointInside;
case SelectionPointBefore:
//x,y is before this element -> stop here
if ( state.m_lastNode ) {
node = state.m_lastNode;
offset = state.m_lastOffset;
//kdDebug(6030) << "RenderTable::checkSelectionPoint " << this << " before this child "
// << node << "-> returning SelectionPointInside, offset=" << offset << endl;
return SelectionPointInside;
} else {
node = nod;
offset = off;
//kdDebug(6030) << "RenderTable::checkSelectionPoint " << this << " before us -> returning SelectionPointBefore " << node << "/" << offset << endl;
return SelectionPointBefore;
}
break;
case SelectionPointAfter:
if (state.m_afterInLine) break;
// fall through
case SelectionPointAfterInLine:
if (pos == SelectionPointAfterInLine) state.m_afterInLine = true;
//kdDebug(6030) << "RenderTable::checkSelectionPoint: selection after: " << nod << " offset: " << off << " afterInLine: " << state.m_afterInLine << endl;
state.m_lastNode = nod;
state.m_lastOffset = off;
// No "return" here, obviously. We must keep looking into the children.
break;
}
}
// If we are after the last child, return lastNode/lastOffset
// But lastNode can be 0L if there is no child, for instance.
if ( state.m_lastNode )
{
node = state.m_lastNode;
offset = state.m_lastOffset;
}
// Fallback
return SelectionPointAfter;
}
// --------------------------------------------------------------------------
RenderTableSection::RenderTableSection(DOM::NodeImpl* node)
: RenderBox(node)
{
// init RenderObject attributes
setInline(false); // our object is not Inline
cCol = 0;
cRow = -1;
needCellRecalc = false;
}
RenderTableSection::~RenderTableSection()
{
clearGrid();
}
void RenderTableSection::detach()
{
// recalc cell info because RenderTable has unguarded pointers
// stored that point to this RenderTableSection.
if (table())
table()->setNeedSectionRecalc();
RenderBox::detach();
}
void RenderTableSection::setStyle(RenderStyle* _style)
{
// we don't allow changing this one
if (style())
_style->setDisplay(style()->display());
else if (_style->display() != TABLE_FOOTER_GROUP && _style->display() != TABLE_HEADER_GROUP)
_style->setDisplay(TABLE_ROW_GROUP);
RenderBox::setStyle(_style);
}
void RenderTableSection::addChild(RenderObject *child, RenderObject *beforeChild)
{
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << renderName() << "(TableSection)::addChild( " << child->renderName() << ", beforeChild=" <<
(beforeChild ? beforeChild->renderName() : "0") << " )" << endl;
#endif
if ( !child->isTableRow() ) {
// TBODY > FORM quirk (???)
if (child->element() && child->element()->isHTMLElement() && child->element()->id() == ID_FORM &&
static_cast<HTMLFormElementImpl*>(child->element())->isMalformed())
{
RenderContainer::addChild(child, beforeChild);
return;
}
RenderObject* last = beforeChild;
if (!last)
last = lastChild();
if (last && last->isAnonymous()) {
last->addChild(child);
return;
}
// If beforeChild is inside an anonymous cell/row, insert into the cell or into
// the anonymous row containing it, if there is one.
RenderObject* lastBox = last;
while (lastBox && lastBox->parent()->isAnonymous() && !lastBox->isTableRow())
lastBox = lastBox->parent();
if (lastBox && lastBox->isAnonymous()) {
lastBox->addChild(child, beforeChild);
return;
}
RenderObject* row = new (renderArena()) RenderTableRow(document() /* anonymous table */);
RenderStyle* newStyle = new RenderStyle();
newStyle->inheritFrom(style());
newStyle->setDisplay(TABLE_ROW);
row->setStyle(newStyle);
addChild(row, beforeChild);
row->addChild(child);
return;
}
if (beforeChild)
setNeedCellRecalc();
cRow++;
cCol = 0;
ensureRows( cRow+1 );
KHTMLAssert( child->isTableRow() );
grid[cRow].rowRenderer = static_cast<RenderTableRow*>(child);
if (!beforeChild) {
grid[cRow].height = child->style()->height();
if ( grid[cRow].height.isRelative() )
grid[cRow].height = Length();
}
RenderContainer::addChild(child,beforeChild);
}
void RenderTableSection::ensureRows( int numRows )
{
int nRows = grid.size();
int nCols = table()->numEffCols();
if ( numRows > nRows ) {
grid.resize( numRows );
for ( int r = nRows; r < numRows; r++ ) {
grid[r].row = new Row( nCols );
grid[r].row->fill( 0 );
grid[r].rowRenderer = 0;
grid[r].baseLine = 0;
grid[r].height = Length();
}
}
}
void RenderTableSection::addCell( RenderTableCell *cell, RenderTableRow *row )
{
int rSpan = cell->rowSpan();
int cSpan = cell->colSpan();
TQMemArray<RenderTable::ColumnStruct> &columns = table()->columns;
int nCols = columns.size();
// ### mozilla still seems to do the old HTML way, even for strict DTD
// (see the annotation on table cell layouting in the CSS specs and the testcase below:
// <TABLE border>
// <TR><TD>1 <TD rowspan="2">2 <TD>3 <TD>4
// <TR><TD colspan="2">5
// </TABLE>
while ( cCol < nCols && cellAt( cRow, cCol ) )
cCol++;
// qDebug("adding cell at %d/%d span=(%d/%d)", cRow, cCol, rSpan, cSpan );
if ( rSpan == 1 ) {
// we ignore height settings on rowspan cells
Length height = cell->style()->height();
if ( height.value() > 0 || (height.isRelative() && height.value() >= 0) ) {
Length cRowHeight = grid[cRow].height;
switch( height.type() ) {
case Percent:
if ( !cRowHeight.isPercent() ||
(cRowHeight.isPercent() && cRowHeight.value() < height.value() ) )
grid[cRow].height = height;
break;
case Fixed:
if ( cRowHeight.type() < Percent ||
( cRowHeight.isFixed() && cRowHeight.value() < height.value() ) )
grid[cRow].height = height;
break;
case Relative:
#if 0
// we treat this as variable. This is correct according to HTML4, as it only specifies length for the height.
if ( cRowHeight.type == Variable ||
( cRowHeight.type == Relative && cRowHeight.value < height.value ) )
grid[cRow].height = height;
break;
#endif
default:
break;
}
}
}
// make sure we have enough rows
ensureRows( cRow + rSpan );
grid[cRow].rowRenderer = row;
int col = cCol;
// tell the cell where it is
RenderTableCell *set = cell;
while ( cSpan ) {
int currentSpan;
if ( cCol >= nCols ) {
table()->appendColumn( cSpan );
currentSpan = cSpan;
} else {
if ( cSpan < columns[cCol].span )
table()->splitColumn( cCol, cSpan );
currentSpan = columns[cCol].span;
}
int r = 0;
while ( r < rSpan ) {
if ( !cellAt( cRow + r, cCol ) ) {
// qDebug(" adding cell at %d, %d", cRow + r, cCol );
cellAt( cRow + r, cCol ) = set;
}
r++;
}
cCol++;
cSpan -= currentSpan;
set = (RenderTableCell *)-1;
}
if ( cell ) {
cell->setRow( cRow );
cell->setCol( table()->effColToCol( col ) );
}
}
void RenderTableSection::setCellWidths()
{
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << renderName() << "(Table, this=0x" << this << ")::setCellWidths()" << endl;
#endif
TQMemArray<int> &columnPos = table()->columnPos;
int rows = grid.size();
for ( int i = 0; i < rows; i++ ) {
Row &row = *grid[i].row;
int cols = row.size();
for ( int j = 0; j < cols; j++ ) {
RenderTableCell *cell = row[j];
// qDebug("cell[%d,%d] = %p", i, j, cell );
if ( !cell || cell == (RenderTableCell *)-1 )
continue;
int endCol = j;
int cspan = cell->colSpan();
while ( cspan && endCol < cols ) {
cspan -= table()->columns[endCol].span;
endCol++;
}
int w = columnPos[endCol] - columnPos[j] - table()->borderHSpacing();
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << "setting width of cell " << cell << " " << cell->row() << "/" << cell->col() << " to " << w << " colspan=" << cell->colSpan() << " start=" << j << " end=" << endCol << endl;
#endif
int oldWidth = cell->width();
if ( w != oldWidth ) {
cell->setNeedsLayout(true);
cell->setWidth( w );
}
}
}
}
short RenderTableSection::width() const
{
return table()->width();
}
void RenderTableSection::calcRowHeight()
{
int indx;
RenderTableCell *cell;
int totalRows = grid.size();
int vspacing = table()->borderVSpacing();
rowPos.resize( totalRows + 1 );
rowPos[0] = vspacing + borderTop();
for ( int r = 0; r < totalRows; r++ ) {
rowPos[r+1] = 0;
int baseline=0;
int bdesc = 0;
// qDebug("height of row %d is %d/%d", r, grid[r].height.value, grid[r].height.type );
int ch = grid[r].height.minWidth( 0 );
int pos = rowPos[r] + ch + (grid[r].rowRenderer ? vspacing : 0);
if ( pos > rowPos[r+1] )
rowPos[r+1] = pos;
Row *row = grid[r].row;
int totalCols = row->size();
int totalRows = grid.size();
bool pagedMode = canvas()->pagedMode();
grid[r].needFlex = false;
for ( int c = 0; c < totalCols; c++ ) {
cell = cellAt(r, c);
if ( !cell || cell == (RenderTableCell *)-1 )
continue;
if ( r < totalRows - 1 && cellAt(r+1, c) == cell )
continue;
if ( ( indx = r - cell->rowSpan() + 1 ) < 0 )
indx = 0;
if (cell->cellPercentageHeight() != -1) {
cell->setCellPercentageHeight(-1);
cell->setChildNeedsLayout(true, false);
if (cell->hasFlexedAnonymous()) {
for (RenderObject* o = cell->firstChild(); o ; o = o->nextSibling())
if (o->isAnonymousBlock())
o->setChildNeedsLayout(true, false);
}
if (pagedMode) cell->setNeedsLayout(true);
cell->layoutIfNeeded();
}
ch = cell->style()->height().width(0);
if ( cell->height() > ch)
ch = cell->height();
if (!cell->style()->height().isVariable())
grid[r].needFlex = true;
pos = rowPos[indx] + ch + (grid[r].rowRenderer ? vspacing : 0);
if ( pos > rowPos[r+1] )
rowPos[r+1] = pos;
// find out the baseline
EVerticalAlign va = cell->style()->verticalAlign();
if (va == BASELINE || va == TEXT_BOTTOM || va == TEXT_TOP
|| va == SUPER || va == SUB)
{
int b=cell->baselinePosition();
if (b > cell->borderTop() + cell->paddingTop()) {
if (b>baseline)
baseline=b;
int td = rowPos[ indx ] + ch - b;
if (td>bdesc)
bdesc = td;
}
}
}
//do we have baseline aligned elements?
if (baseline) {
// increase rowheight if baseline requires
int bRowPos = baseline + bdesc + (grid[r].rowRenderer ? vspacing : 0);
if (rowPos[r+1]<bRowPos)
rowPos[r+1]=bRowPos;
grid[r].baseLine = baseline;
}
if ( rowPos[r+1] < rowPos[r] )
rowPos[r+1] = rowPos[r];
// qDebug("rowpos(%d)=%d", r, rowPos[r] );
}
}
int RenderTableSection::layoutRows( int toAdd )
{
int rHeight;
int rindx;
int totalRows = grid.size();
int hspacing = table()->borderHSpacing();
int vspacing = table()->borderVSpacing();
// Set the width of our section now. The rows will also be this width.
m_width = table()->contentWidth();
if (markedForRepaint()) {
repaintDuringLayout();
setMarkedForRepaint(false);
}
if (toAdd && totalRows && (rowPos[totalRows] || !nextSibling())) {
int totalHeight = rowPos[totalRows] + toAdd;
// qDebug("layoutRows: totalHeight = %d", totalHeight );
int dh = toAdd;
int totalPercent = 0;
int numVariable = 0;
for ( int r = 0; r < totalRows; r++ ) {
if ( grid[r].height.isVariable() && !emptyRow(r))
numVariable++;
else if ( grid[r].height.isPercent() )
totalPercent += grid[r].height.value();
}
if ( totalPercent ) {
// qDebug("distributing %d over percent rows totalPercent=%d", dh, totalPercent );
// try to satisfy percent
int add = 0;
if ( totalPercent > 100 )
totalPercent = 100;
int rh = rowPos[1]-rowPos[0];
for ( int r = 0; r < totalRows; r++ ) {
if ( totalPercent > 0 && grid[r].height.isPercent() ) {
int toAdd = kMin( dh, (totalHeight * grid[r].height.value() / 100)-rh );
// If toAdd is negative, then we don't want to shrink the row (this bug
// affected Outlook Web Access).
toAdd = kMax(0, toAdd);
add += toAdd;
dh -= toAdd;
totalPercent -= grid[r].height.value();
// qDebug( "adding %d to row %d", toAdd, r );
}
if ( r < totalRows-1 )
rh = rowPos[r+2] - rowPos[r+1];
rowPos[r+1] += add;
}
}
if ( numVariable ) {
// distribute over non-empty variable rows
// qDebug("distributing %d over variable rows numVariable=%d", dh, numVariable );
int add = 0;
int toAdd = dh/numVariable;
for ( int r = 0; r < totalRows; r++ ) {
if ( grid[r].height.isVariable() && !emptyRow(r)) {
add += toAdd;
}
rowPos[r+1] += add;
}
dh -= add;
}
if (dh>0 && rowPos[totalRows]) {
// if some left overs, distribute weighted.
int tot=rowPos[totalRows];
int add=0;
int prev=rowPos[0];
for ( int r = 0; r < totalRows; r++ ) {
//weight with the original height
add+=dh*(rowPos[r+1]-prev)/tot;
prev=rowPos[r+1];
rowPos[r+1]+=add;
}
dh -= add;
}
if (dh > totalRows) {
// distribute to tables with all empty rows
int add=0;
int toAdd = dh/totalRows;
for ( int r = 0; r < totalRows; r++ ) {
add += toAdd;
rowPos[r+1] += add;
}
dh -= add;
}
// Finally distribute round-off values
if (dh > 0) {
// There is not enough for every row
int add=0;
for ( int r = 0; r < totalRows; r++ ) {
if (add < dh) add++;
rowPos[r+1] += add;
}
dh -= add;
}
assert (dh == 0);
}
int leftOffset = borderLeft() + hspacing;
int nEffCols = table()->numEffCols();
for ( int r = 0; r < totalRows; r++ )
{
Row *row = grid[r].row;
int totalCols = row->size();
#ifdef APPLE_CHANGES
// in WC, rows and cells share the same coordinate space, so that rows can have
// dimensions in the layer system. This is of dubious value, and a heavy maintenance burden
// (RenderObject's coordinates can't be used deterministically anymore) so we'll consider other options.
// Set the row's x/y position and width/height.
if (grid[r].rowRenderer) {
grid[r].rowRenderer->setPos(0, rowPos[r]);
grid[r].rowRenderer->setWidth(m_width);
grid[r].rowRenderer->setHeight(rowPos[r+1] - rowPos[r] - vspacing);
}
#endif
for ( int c = 0; c < nEffCols; c++ )
{
RenderTableCell *cell = cellAt(r, c);
if (!cell || cell == (RenderTableCell *)-1 )
continue;
if ( r < totalRows - 1 && cell == cellAt(r+1, c) )
continue;
if ( ( rindx = r-cell->rowSpan()+1 ) < 0 )
rindx = 0;
rHeight = rowPos[r+1] - rowPos[rindx] - vspacing;
// Force percent height children to lay themselves out again.
// This will cause, e.g., textareas to grow to
// fill the area. FIXME: <div>s and blocks still don't
// work right. We'll need to have an efficient way of
// invalidating all percent height objects in a render subtree.
// For now, we just handle immediate children. -dwh
bool flexAllChildren = grid[r].needFlex || (!table()->style()->height().isVariable() && rHeight != cell->height());
cell->setHasFlexedAnonymous(false);
if ( flexAllChildren && flexCellChildren(cell) ) {
cell->setCellPercentageHeight(kMax(0,
rHeight - cell->borderTop() - cell->paddingTop() -
cell->borderBottom() - cell->paddingBottom()));
cell->layoutIfNeeded();
}
{
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << "setting position " << r << "/" << c << ": "
<< table()->columnPos[c] /*+ padding */ << "/" << rowPos[rindx] << " height=" << rHeight<< endl;
#endif
EVerticalAlign va = cell->style()->verticalAlign();
int te=0;
switch (va)
{
case SUB:
case SUPER:
case TEXT_TOP:
case TEXT_BOTTOM:
case BASELINE:
te = getBaseline(r) - cell->baselinePosition() ;
break;
case TOP:
te = 0;
break;
case MIDDLE:
te = (rHeight - cell->height())/2;
break;
case BOTTOM:
te = rHeight - cell->height();
break;
default:
break;
}
te = kMax( 0, te );
#ifdef DEBUG_LAYOUT
// kdDebug( 6040 ) << "CELL " << cell << " te=" << te << ", be=" << rHeight - cell->height() - te << ", rHeight=" << rHeight << ", valign=" << va << endl;
#endif
cell->setCellTopExtra( te );
cell->setCellBottomExtra( rHeight - cell->height() - te);
}
if (style()->direction()==RTL) {
cell->setPos(
table()->columnPos[(int)totalCols] -
table()->columnPos[table()->colToEffCol(cell->col()+cell->colSpan())] +
leftOffset,
rowPos[rindx] );
} else {
cell->setPos( table()->columnPos[c] + leftOffset, rowPos[rindx] );
}
}
}
m_height = rowPos[totalRows];
return m_height;
}
bool RenderTableSection::flexCellChildren(RenderObject* p) const
{
if (!p)
return false;
RenderObject* o = p->firstChild();
bool didFlex = false;
while (o) {
if (!o->isText() && o->style()->height().isPercent()) {
if (o->isWidget()) {
// cancel resizes from transitory relayouts
static_cast<RenderWidget *>(o)->cancelPendingResize();
}
o->setNeedsLayout(true, false);
p->setChildNeedsLayout(true, false);
didFlex = true;
} else if (o->isAnonymousBlock() && flexCellChildren( o )) {
p->setChildNeedsLayout(true, false);
if (p->isTableCell())
static_cast<RenderTableCell*>(p)->setHasFlexedAnonymous();
didFlex = true;
}
o = o->nextSibling();
}
return didFlex;
}
inline static RenderTableRow *firstTableRow(RenderObject *row)
{
while (row && !row->isTableRow())
row = row->nextSibling();
return static_cast<RenderTableRow *>(row);
}
inline static RenderTableRow *nextTableRow(RenderObject *row)
{
row = row ? row->nextSibling() : row;
while (row && !row->isTableRow())
row = row->nextSibling();
return static_cast<RenderTableRow *>(row);
}
int RenderTableSection::lowestPosition(bool includeOverflowInterior, bool includeSelf) const
{
int bottom = RenderBox::lowestPosition(includeOverflowInterior, includeSelf);
if (!includeOverflowInterior && hasOverflowClip())
return bottom;
for (RenderObject *row = firstChild(); row; row = row->nextSibling()) {
for (RenderObject *cell = row->firstChild(); cell; cell = cell->nextSibling())
if (cell->isTableCell()) {
int bp = cell->yPos() + cell->lowestPosition(false);
bottom = kMax(bottom, bp);
}
}
return bottom;
}
int RenderTableSection::rightmostPosition(bool includeOverflowInterior, bool includeSelf) const
{
int right = RenderBox::rightmostPosition(includeOverflowInterior, includeSelf);
if (!includeOverflowInterior && hasOverflowClip())
return right;
for (RenderObject *row = firstChild(); row; row = row->nextSibling()) {
for (RenderObject *cell = row->firstChild(); cell; cell = cell->nextSibling())
if (cell->isTableCell()) {
int rp = cell->xPos() + cell->rightmostPosition(false);
right = kMax(right, rp);
}
}
return right;
}
int RenderTableSection::leftmostPosition(bool includeOverflowInterior, bool includeSelf) const
{
int left = RenderBox::leftmostPosition(includeOverflowInterior, includeSelf);
if (!includeOverflowInterior && hasOverflowClip())
return left;
for (RenderObject *row = firstChild(); row; row = row->nextSibling()) {
for (RenderObject *cell = row->firstChild(); cell; cell = cell->nextSibling())
if (cell->isTableCell()) {
int lp = cell->xPos() + cell->leftmostPosition(false);
left = kMin(left, lp);
}
}
return left;
}
int RenderTableSection::highestPosition(bool includeOverflowInterior, bool includeSelf) const
{
int top = RenderBox::highestPosition(includeOverflowInterior, includeSelf);
if (!includeOverflowInterior && hasOverflowClip())
return top;
for (RenderObject *row = firstChild(); row; row = row->nextSibling()) {
for (RenderObject *cell = row->firstChild(); cell; cell = cell->nextSibling())
if (cell->isTableCell()) {
int hp = cell->yPos() + cell->highestPosition(false);
top = kMin(top, hp);
}
}
return top;
}
// Search from first_row to last_row for the row containing y
static unsigned int findRow(unsigned int first_row, unsigned int last_row,
const TQMemArray<int> &rowPos, int y)
{
unsigned int under = first_row;
unsigned int over = last_row;
int offset = (over - under)/2;
while (over-under > 1) {
if (rowPos[under+offset] <= y)
under = under+offset;
else
over = under+offset;
offset = (over - under)/2;
}
assert(under == first_row || rowPos[under] <= y);
assert(over == last_row || rowPos[over] > y);
return under;
}
static void findRowCover(unsigned int &startrow, unsigned int &endrow,
const TQMemArray<int> &rowPos,
int min_y, int max_y)
{
assert(max_y >= min_y);
unsigned int totalRows = endrow;
unsigned int index = 0;
// Initial binary search boost:
if (totalRows >= 8) {
int offset = (endrow - startrow)/2;
while (endrow - startrow > 1) {
index = startrow+offset;
if (rowPos[index] <= min_y )
// index is below both min_y and max_y
startrow = index;
else
if (rowPos[index] > max_y)
// index is above both min_y and max_y
endrow = index;
else
// index is within the selection
break;
offset = (endrow - startrow)/2;
}
}
// Binary search for startrow
startrow = findRow(startrow, endrow, rowPos, min_y);
// Binary search for endrow
endrow = findRow(startrow, endrow, rowPos, max_y) + 1;
if (endrow > totalRows) endrow = totalRows;
}
void RenderTableSection::paint( PaintInfo& pI, int tx, int ty )
{
unsigned int totalRows = grid.size();
unsigned int totalCols = table()->columns.size();
tx += m_x;
ty += m_y;
if (pI.phase == PaintActionOutline)
paintOutline(pI.p, tx, ty, width(), height(), style());
CollapsedBorderValue *cbs = table()->currentBorderStyle();
int cbsw2 = cbs ? cbs->width()/2 : 0;
int cbsw21 = cbs ? (cbs->width()+1)/2 : 0;
int x = pI.r.x(), y = pI.r.y(), w = pI.r.width(), h = pI.r.height();
// check which rows and cols are visible and only paint these
int os = 2*maximalOutlineSize(pI.phase);
unsigned int startrow = 0;
unsigned int endrow = totalRows;
findRowCover(startrow, endrow, rowPos, y - os - ty - kMax(cbsw21, os), y + h + os - ty + kMax(cbsw2, os));
// A binary search is probably not worthwhile for coloumns
unsigned int startcol = 0;
unsigned int endcol = totalCols;
if ( style()->direction() == LTR ) {
for ( ; startcol < totalCols; startcol++ ) {
if ( tx + table()->columnPos[startcol+1] + kMax(cbsw21, os) > x - os )
break;
}
for ( ; endcol > 0; endcol-- ) {
if ( tx + table()->columnPos[endcol-1] - kMax(cbsw2, os) < x + w + os )
break;
}
}
if ( startcol < endcol ) {
// draw the cells
for ( unsigned int r = startrow; r < endrow; r++ ) {
// paint the row
if (grid[r].rowRenderer) {
int height = rowPos[r+1] - rowPos[r] - table()->borderVSpacing();
grid[r].rowRenderer->paintRow(pI, tx, ty + rowPos[r], width(), height);
}
unsigned int c = startcol;
Row *row = grid[r].row;
Row *nextrow = (r < endrow - 1) ? grid[r+1].row : 0;
// since a cell can be -1 (indicating a colspan) we might have to search backwards to include it
while ( c && (*row)[c] == (RenderTableCell *)-1 )
c--;
for ( ; c < endcol; c++ ) {
RenderTableCell *cell = (*row)[c];
if ( !cell || cell == (RenderTableCell *)-1 || nextrow && (*nextrow)[c] == cell )
continue;
#ifdef TABLE_PRINT
kdDebug( 6040 ) << "painting cell " << r << "/" << c << endl;
#endif
if (pI.phase == PaintActionElementBackground || pI.phase == PaintActionChildBackground) {
// We need to handle painting a stack of backgrounds. This stack (from bottom to top) consists of
// the column group, column, row group, row, and then the cell.
RenderObject* col = table()->colElement(c);
RenderObject* colGroup = 0;
if (col) {
RenderStyle *style = col->parent()->style();
if (style->display() == TABLE_COLUMN_GROUP)
colGroup = col->parent();
}
RenderObject* row = cell->parent();
// ###
// Column groups and columns first.
// FIXME: Columns and column groups do not currently support opacity, and they are being painted "too late" in
// the stack, since we have already opened a transparency layer (potentially) for the table row group.
// Note that we deliberately ignore whether or not the cell has a layer, since these backgrounds paint "behind" the
// cell.
cell->paintBackgroundsBehindCell(pI, tx, ty, colGroup);
cell->paintBackgroundsBehindCell(pI, tx, ty, col);
// Paint the row group next.
cell->paintBackgroundsBehindCell(pI, tx, ty, this);
// Paint the row next, but only if it doesn't have a layer. If a row has a layer, it will be responsible for
// painting the row background for the cell.
if (!row->layer())
cell->paintBackgroundsBehindCell(pI, tx, ty, row);
}
if ((!cell->layer() && !cell->parent()->layer()) || pI.phase == PaintActionCollapsedTableBorders)
cell->paint(pI, tx, ty);
}
}
}
}
void RenderTableSection::recalcCells()
{
cCol = 0;
cRow = -1;
clearGrid();
grid.resize( 0 );
for (RenderObject *row = firstChild(); row; row = row->nextSibling()) {
if (row->isTableRow()) {
cRow++;
cCol = 0;
ensureRows( cRow+1 );
grid[cRow].rowRenderer = static_cast<RenderTableRow*>(row);
for (RenderObject *cell = row->firstChild(); cell; cell = cell->nextSibling())
if (cell->isTableCell())
addCell( static_cast<RenderTableCell *>(cell), static_cast<RenderTableRow *>(row) );
}
}
needCellRecalc = false;
setNeedsLayout(true);
}
void RenderTableSection::clearGrid()
{
int rows = grid.size();
while ( rows-- ) {
delete grid[rows].row;
}
}
bool RenderTableSection::emptyRow(int rowNum) {
Row &r = *grid[rowNum].row;
const int s = r.size();
RenderTableCell *cell;
for(int i=0; i<s; i++) {
cell = r[i];
if (!cell || cell==(RenderTableCell*)-1)
continue;
return false;
}
return true;
}
RenderObject* RenderTableSection::removeChildNode(RenderObject* child)
{
setNeedCellRecalc();
return RenderContainer::removeChildNode( child );
}
bool RenderTableSection::canClear(RenderObject */*child*/, PageBreakLevel level)
{
// We cannot clear rows yet.
return parent()->canClear(this, level);
}
void RenderTableSection::addSpaceAt(int pos, int dy)
{
const int nEffCols = table()->numEffCols();
const int totalRows = numRows();
for ( int r = 0; r < totalRows; r++ ) {
if (rowPos[r] >= pos) {
rowPos[r] += dy;
int rindx;
for ( int c = 0; c < nEffCols; c++ )
{
RenderTableCell *cell = cellAt(r, c);
if (!cell || cell == (RenderTableCell *)-1 )
continue;
if ( r > 0 && cell == cellAt(r-1, c) )
continue;
if ( ( rindx = r-cell->rowSpan()+1 ) < 0 )
rindx = 0;
cell->setPos(cell->xPos(), rowPos[r]);
}
}
}
if (rowPos[totalRows] >= pos)
rowPos[totalRows] += dy;
m_height = rowPos[totalRows];
setContainsPageBreak(true);
}
#ifdef ENABLE_DUMP
void RenderTableSection::dump(TQTextStream &stream, const TQString &ind) const
{
RenderContainer::dump(stream,ind);
stream << " grid=(" << grid.size() << "," << table()->numEffCols() << ")";
for ( unsigned int r = 0; r < grid.size(); r++ ) {
for ( int c = 0; c < table()->numEffCols(); c++ ) {
if ( cellAt( r, c ) && cellAt( r, c ) != (RenderTableCell *)-1 )
stream << " (" << cellAt( r, c )->row() << "," << cellAt( r, c )->col() << ","
<< cellAt(r, c)->rowSpan() << "," << cellAt(r, c)->colSpan() << ") ";
else
stream << " null cell";
}
}
}
#endif
static RenderTableCell *seekCell(RenderTableSection *section, int row, int col)
{
if (row < 0 || col < 0 || row >= section->numRows()) return 0;
// since a cell can be -1 (indicating a colspan) we might have to search backwards to include it
while ( col && section->cellAt( row, col ) == (RenderTableCell *)-1 )
col--;
return section->cellAt(row, col);
}
/** Looks for the first element suitable for text selection, beginning from
* the last.
* @param base search is restricted within this node. This node must have
* a renderer.
* @return the element or @p base if no suitable element found.
*/
static NodeImpl *findLastSelectableNode(NodeImpl *base)
{
NodeImpl *last = base;
// Look for last text/cdata node that has a renderer,
// or last childless replaced element
while ( last && !(last->renderer()
&& ((last->nodeType() == Node::TEXT_NODE || last->nodeType() == Node::CDATA_SECTION_NODE)
|| (last->renderer()->isReplaced() && !last->renderer()->lastChild()))))
{
NodeImpl *next = last->lastChild();
if ( !next ) next = last->previousSibling();
while ( last && last != base && !next )
{
last = last->parentNode();
if ( last && last != base )
next = last->previousSibling();
}
last = next;
}
return last ? last : base;
}
FindSelectionResult RenderTableSection::checkSelectionPoint( int _x, int _y, int _tx, int _ty, DOM::NodeImpl*& node, int & offset, SelPointState &state )
{
// Table sections need extra treatment for selections. The rows are scanned
// from top to bottom, and within each row, only the cell that matches
// the given position best is descended into.
unsigned int totalRows = grid.size();
unsigned int totalCols = table()->columns.size();
// absolutePosition(_tx, _ty, false);
_tx += m_x;
_ty += m_y;
// bool save_last = false; // true to save last investigated cell
if (needsLayout() || _y < _ty) return SelectionPointBefore;
// else if (_y >= _ty + height()) save_last = true;
// Find the row containing the pointer
int row_idx = findRow(0, totalRows, rowPos, _y - _ty);
int col_idx;
if ( style()->direction() == LTR ) {
for ( col_idx = (int)totalCols - 1; col_idx >= 0; col_idx-- ) {
if ( _tx + table()->columnPos[col_idx] < _x )
break;
}
if (col_idx < 0) col_idx = 0;
} else {
for ( col_idx = 0; col_idx < (int)totalCols; col_idx++ ) {
if ( _tx + table()->columnPos[col_idx] > _x )
break;
}
if (col_idx >= (int)totalCols) col_idx = (int)totalCols + 1;
}
FindSelectionResult pos = SelectionPointBefore;
RenderTableCell *cell = seekCell(this, row_idx, col_idx);
// ### dunno why cell can be 0, maybe due to weird spans? (LS)
if (cell) {
SelPointState localState;
pos = cell->checkSelectionPoint(_x, _y, _tx, _ty, node, offset, localState);
}
if (pos != SelectionPointBefore) return pos;
// store last column of last line
row_idx--;
col_idx = totalCols - 1;
cell = seekCell(this, row_idx, col_idx);
// end of section? take previous section
RenderTableSection *sec = this;
if (!cell) {
sec = *--TableSectionIterator(sec);
if (!sec) return pos;
cell = seekCell(sec, sec->grid.size() - 1, col_idx);
if (!cell) return pos;
}
// take last child of previous cell, and store this one as last node
NodeImpl *element = cell->element();
if (!element) return SelectionPointBefore;
element = findLastSelectableNode(element);
state.m_lastNode = element;
state.m_lastOffset = element->maxOffset();
return SelectionPointBefore;
}
// Hit Testing
bool RenderTableSection::nodeAtPoint(NodeInfo& info, int x, int y, int tx, int ty, HitTestAction action, bool inside)
{
// Table sections cannot ever be hit tested. Effectively they do not exist.
// Just forward to our children always.
tx += m_x;
ty += m_y;
for (RenderObject* child = lastChild(); child; child = child->previousSibling()) {
// FIXME: We have to skip over inline flows, since they can show up inside table rows
// at the moment (a demoted inline <form> for example). If we ever implement a
// table-specific hit-test method (which we should do for performance reasons anyway),
// then we can remove this check.
if (!child->layer() && !child->isInlineFlow() && child->nodeAtPoint(info, x, y, tx, ty, action, inside)) {
return true;
}
}
return false;
}
// -------------------------------------------------------------------------
RenderTableRow::RenderTableRow(DOM::NodeImpl* node)
: RenderContainer(node)
{
// init RenderObject attributes
setInline(false); // our object is not Inline
}
RenderObject* RenderTableRow::removeChildNode(RenderObject* child)
{
RenderTableSection *s = section();
if (s)
s->setNeedCellRecalc();
return RenderContainer::removeChildNode( child );
}
void RenderTableRow::detach()
{
RenderTableSection *s = section();
if (s)
s->setNeedCellRecalc();
RenderContainer::detach();
}
void RenderTableRow::setStyle(RenderStyle* newStyle)
{
if (section() && style() && style()->height() != newStyle->height())
section()->setNeedCellRecalc();
newStyle->setDisplay(TABLE_ROW);
RenderContainer::setStyle(newStyle);
}
void RenderTableRow::addChild(RenderObject *child, RenderObject *beforeChild)
{
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << renderName() << "(TableRow)::addChild( " << child->renderName() << " )" << ", " <<
(beforeChild ? beforeChild->renderName() : "0") << " )" << endl;
#endif
if ( !child->isTableCell() ) {
// TR > FORM quirk (???)
if (child->element() && child->element()->isHTMLElement() && child->element()->id() == ID_FORM &&
static_cast<HTMLFormElementImpl*>(child->element())->isMalformed())
{
RenderContainer::addChild(child, beforeChild);
return;
}
RenderObject* last = beforeChild;
if (!last)
last = lastChild();
if (last && last->isAnonymous() && last->isTableCell()) {
last->addChild(child);
return;
}
// If beforeChild is inside an anonymous cell, insert into the cell.
if (last && !last->isTableCell() && last->parent() && last->parent()->isAnonymous()) {
last->parent()->addChild(child, beforeChild);
return;
}
RenderTableCell* cell = new (renderArena()) RenderTableCell(document() /* anonymous object */);
RenderStyle* newStyle = new RenderStyle();
newStyle->inheritFrom(style());
newStyle->setDisplay(TABLE_CELL);
cell->setStyle(newStyle);
addChild(cell, beforeChild);
cell->addChild(child);
return;
}
RenderTableCell* cell = static_cast<RenderTableCell*>(child);
section()->addCell( cell, this );
RenderContainer::addChild(cell,beforeChild);
if ( beforeChild || nextSibling() )
section()->setNeedCellRecalc();
}
void RenderTableRow::layout()
{
KHTMLAssert( needsLayout() );
KHTMLAssert( minMaxKnown() );
RenderObject *child = firstChild();
const bool pagedMode = canvas()->pagedMode();
while( child ) {
if ( child->isTableCell() ) {
RenderTableCell *cell = static_cast<RenderTableCell *>(child);
if (pagedMode) {
cell->setNeedsLayout(true);
int oldHeight = child->height();
cell->layout();
if (oldHeight > 0 && child->containsPageBreak() && child->height() != oldHeight)
section()->addSpaceAt(child->yPos()+1, child->height() - oldHeight);
} else
if ( child->needsLayout() ) {
if (markedForRepaint())
cell->setMarkedForRepaint( true );
cell->calcVerticalMargins();
cell->layout();
cell->setCellTopExtra(0);
cell->setCellBottomExtra(0);
if (child->containsPageBreak()) setContainsPageBreak(true);
}
}
child = child->nextSibling();
}
setMarkedForRepaint(false);
setNeedsLayout(false);
}
int RenderTableRow::offsetLeft() const
{
RenderObject *child = firstChild();
while (child && !child->isTableCell())
child = child->nextSibling();
if (!child)
return 0;
return child->offsetLeft();
}
int RenderTableRow::offsetTop() const
{
RenderObject *child = firstChild();
while (child && !child->isTableCell())
child = child->nextSibling();
if (!child)
return 0;
return child->offsetTop() -
static_cast<RenderTableCell*>(child)->cellTopExtra();
}
int RenderTableRow::offsetHeight() const
{
RenderObject *child = firstChild();
while (child && !child->isTableCell())
child = child->nextSibling();
if (!child)
return 0;
return child->offsetHeight() +
static_cast<RenderTableCell*>(child)->cellTopExtra() +
static_cast<RenderTableCell*>(child)->cellBottomExtra();
}
short RenderTableRow::offsetWidth() const
{
RenderObject *fc = firstChild();
RenderObject *lc = lastChild();
while (fc && !fc->isTableCell())
fc = fc->nextSibling();
while (lc && !lc->isTableCell())
lc = lc->previousSibling();
if (!lc || !fc)
return 0;
return lc->xPos()+lc->width()-fc->xPos();
}
void RenderTableRow::paintRow( PaintInfo& pI, int tx, int ty, int w, int h )
{
if (pI.phase == PaintActionOutline)
paintOutline(pI.p, tx, ty, w, h, style());
}
void RenderTableRow::paint(PaintInfo& i, int tx, int ty)
{
KHTMLAssert(layer());
if (!layer())
return;
for (RenderObject* child = firstChild(); child; child = child->nextSibling()) {
if (child->isTableCell()) {
// Paint the row background behind the cell.
if (i.phase == PaintActionElementBackground || i.phase == PaintActionChildBackground) {
RenderTableCell* cell = static_cast<RenderTableCell*>(child);
cell->paintBackgroundsBehindCell(i, tx, ty, this);
}
if (!child->layer())
child->paint(i, tx, ty);
}
}
}
// Hit Testing
bool RenderTableRow::nodeAtPoint(NodeInfo& info, int x, int y, int tx, int ty, HitTestAction action, bool inside)
{
// Table rows cannot ever be hit tested. Effectively they do not exist.
// Just forward to our children always.
for (RenderObject* child = lastChild(); child; child = child->previousSibling()) {
// FIXME: We have to skip over inline flows, since they can show up inside table rows
// at the moment (a demoted inline <form> for example). If we ever implement a
// table-specific hit-test method (which we should do for performance reasons anyway),
// then we can remove this check.
if (!child->layer() && !child->isInlineFlow() && child->nodeAtPoint(info, x, y, tx, ty, action, inside)) {
return true;
}
}
return false;
}
// -------------------------------------------------------------------------
RenderTableCell::RenderTableCell(DOM::NodeImpl* _node)
: RenderBlock(_node)
{
_col = -1;
_row = -1;
updateFromElement();
setShouldPaintBackgroundOrBorder(true);
_topExtra = 0;
_bottomExtra = 0;
m_percentageHeight = -1;
m_hasFlexedAnonymous = false;
m_widthChanged = false;
}
void RenderTableCell::detach()
{
if (parent() && section())
section()->setNeedCellRecalc();
RenderBlock::detach();
}
void RenderTableCell::updateFromElement()
{
DOM::NodeImpl *node = element();
if ( node && (node->id() == ID_TD || node->id() == ID_TH) ) {
DOM::HTMLTableCellElementImpl *tc = static_cast<DOM::HTMLTableCellElementImpl *>(node);
cSpan = tc->colSpan();
rSpan = tc->rowSpan();
} else {
cSpan = rSpan = 1;
}
}
Length RenderTableCell::styleOrColWidth()
{
Length w = style()->width();
if (colSpan() > 1 || !w.isVariable())
return w;
RenderTableCol* col = table()->colElement(_col);
if (col)
w = col->style()->width();
return w;
}
void RenderTableCell::calcMinMaxWidth()
{
KHTMLAssert( !minMaxKnown() );
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << renderName() << "(TableCell)::calcMinMaxWidth() known=" << minMaxKnown() << endl;
#endif
if (section()->needCellRecalc)
section()->recalcCells();
RenderBlock::calcMinMaxWidth();
if (element() && style()->whiteSpace() == NORMAL) {
// See if nowrap was set.
Length w = styleOrColWidth();
DOMString nowrap = static_cast<ElementImpl*>(element())->getAttribute(ATTR_NOWRAP);
if (!nowrap.isNull() && w.isFixed() &&
m_minWidth < w.value() )
// Nowrap is set, but we didn't actually use it because of the
// fixed width set on the cell. Even so, it is a WinIE/Moz trait
// to make the minwidth of the cell into the fixed width. They do this
// even in strict mode, so do not make this a quirk. Affected the top
// of hiptop.com.
m_minWidth = w.value();
}
setMinMaxKnown();
}
void RenderTableCell::calcWidth()
{
}
void RenderTableCell::setWidth( int width )
{
if ( width != m_width ) {
m_width = width;
m_widthChanged = true;
}
}
void RenderTableCell::layout()
{
layoutBlock( m_widthChanged );
m_widthChanged = false;
}
void RenderTableCell::close()
{
RenderBlock::close();
#ifdef DEBUG_LAYOUT
kdDebug( 6040 ) << renderName() << "(RenderTableCell)::close() total height =" << m_height << endl;
#endif
}
bool RenderTableCell::requiresLayer() const {
// table-cell display is never positioned (css 2.1-9.7), so the only time a layer is needed
// is when overflow != visible (or when there is opacity when we support it)
return /* style()->opacity() < 1.0f || */ hasOverflowClip() || isRelPositioned();
}
void RenderTableCell::repaintRectangle(int x, int y, int w, int h, Priority p, bool f)
{
RenderBlock::repaintRectangle(x, y, w, h + _topExtra + _bottomExtra, p, f);
}
bool RenderTableCell::absolutePosition(int &xPos, int &yPos, bool f) const
{
bool result = RenderBlock::absolutePosition(xPos, yPos, f);
xPos -= parent()->xPos(); // Rows are in the same coordinate space, so don't add their offset in.
yPos -= parent()->yPos();
return result;
}
int RenderTableCell::pageTopAfter(int y) const
{
return section()->pageTopAfter(y+m_y + _topExtra) - (m_y + _topExtra);
}
short RenderTableCell::baselinePosition( bool ) const
{
RenderObject* o = firstChild();
int offset = paddingTop() + borderTop();
if (!o) return offset + contentHeight();
while (o->firstChild()) {
if (!o->isInline())
offset += o->paddingTop() + o->borderTop();
o = o->firstChild();
}
if (!o->isInline())
return paddingTop() + borderTop() + contentHeight();
offset += o->baselinePosition( true );
return offset;
}
void RenderTableCell::setStyle( RenderStyle *newStyle )
{
if (parent() && section() && style() && style()->height() != newStyle->height())
section()->setNeedCellRecalc();
newStyle->setDisplay(TABLE_CELL);
RenderBlock::setStyle( newStyle );
setShouldPaintBackgroundOrBorder(true);
if (newStyle->whiteSpace() == KHTML_NOWRAP) {
// Figure out if we are really nowrapping or if we should just
// use normal instead. If the width of the cell is fixed, then
// we don't actually use NOWRAP.
if (newStyle->width().isFixed())
newStyle->setWhiteSpace(NORMAL);
else
newStyle->setWhiteSpace(NOWRAP);
}
}
bool RenderTableCell::nodeAtPoint(NodeInfo& info, int _x, int _y, int _tx, int _ty, HitTestAction hitTestAction, bool inside)
{
int tx = _tx + m_x;
int ty = _ty + m_y;
// also include the top and bottom extra space
inside |= style()->visibility() != HIDDEN
&& (_y >= ty) && (_y < ty + height() + _topExtra + _bottomExtra)
&& (_x >= tx) && (_x < tx + width());
return RenderBlock::nodeAtPoint(info, _x, _y, _tx, _ty, hitTestAction, inside);
}
// The following rules apply for resolving conflicts and figuring out which border
// to use.
// (1) Borders with the 'border-style' of 'hidden' take precedence over all other conflicting
// borders. Any border with this value suppresses all borders at this location.
// (2) Borders with a style of 'none' have the lowest priority. Only if the border properties of all
// the elements meeting at this edge are 'none' will the border be omitted (but note that 'none' is
// the default value for the border style.)
// (3) If none of the styles are 'hidden' and at least one of them is not 'none', then narrow borders
// are discarded in favor of wider ones. If several have the same 'border-width' then styles are preferred
// in this order: 'double', 'solid', 'dashed', 'dotted', 'ridge', 'outset', 'groove', and the lowest: 'inset'.
// (4) If border styles differ only in color, then a style set on a cell wins over one on a row,
// which wins over a row group, column, column group and, lastly, table. It is undefined which color
// is used when two elements of the same type disagree.
static CollapsedBorderValue compareBorders(const CollapsedBorderValue& border1,
const CollapsedBorderValue& border2)
{
// Sanity check the values passed in. If either is null, return the other.
if (!border2.exists()) return border1;
if (!border1.exists()) return border2;
// Rule #1 above.
if (border1.style() == BHIDDEN || border2.style() == BHIDDEN)
return CollapsedBorderValue(); // No border should exist at this location.
// Rule #2 above. A style of 'none' has lowest priority and always loses to any other border.
if (border2.style() == BNONE) return border1;
if (border1.style() == BNONE) return border2;
// The first part of rule #3 above. Wider borders win.
if (border1.width() != border2.width())
return border1.width() > border2.width() ? border1 : border2;
// The borders have equal width. Sort by border style.
if (border1.style() != border2.style())
return border1.style() > border2.style() ? border1 : border2;
// The border have the same width and style. Rely on precedence (cell over row over row group, etc.)
return border1.precedence >= border2.precedence ? border1 : border2;
}
CollapsedBorderValue RenderTableCell::collapsedLeftBorder() const
{
// For border left, we need to check, in order of precedence:
// (1) Our left border.
CollapsedBorderValue result(&style()->borderLeft(), BCELL);
// (2) The previous cell's right border.
RenderTableCell* prevCell = table()->cellLeft(this);
if (prevCell) {
result = compareBorders(result, CollapsedBorderValue(&prevCell->style()->borderRight(), BCELL));
if (!result.exists()) return result;
}
else if (col() == 0) {
// (3) Our row's left border.
result = compareBorders(result, CollapsedBorderValue(&parent()->style()->borderLeft(), BROW));
if (!result.exists()) return result;
// (4) Our row group's left border.
result = compareBorders(result, CollapsedBorderValue(&section()->style()->borderLeft(), BROWGROUP));
if (!result.exists()) return result;
}
// (5) Our column's left border.
RenderTableCol* colElt = table()->colElement(col());
if (colElt) {
result = compareBorders(result, CollapsedBorderValue(&colElt->style()->borderLeft(), BCOL));
if (!result.exists()) return result;
}
// (6) The previous column's right border.
if (col() > 0) {
colElt = table()->colElement(col()-1);
if (colElt) {
result = compareBorders(result, CollapsedBorderValue(&colElt->style()->borderRight(), BCOL));
if (!result.exists()) return result;
}
}
if (col() == 0) {
// (7) The table's left border.
result = compareBorders(result, CollapsedBorderValue(&table()->style()->borderLeft(), BTABLE));
if (!result.exists()) return result;
}
return result;
}
CollapsedBorderValue RenderTableCell::collapsedRightBorder() const
{
RenderTable* tableElt = table();
bool inLastColumn = false;
int effCol = tableElt->colToEffCol(col()+colSpan()-1);
if (effCol == tableElt->numEffCols()-1)
inLastColumn = true;
// For border right, we need to check, in order of precedence:
// (1) Our right border.
CollapsedBorderValue result = CollapsedBorderValue(&style()->borderRight(), BCELL);
// (2) The next cell's left border.
if (!inLastColumn) {
RenderTableCell* nextCell = tableElt->cellRight(this);
if (nextCell) {
result = compareBorders(result, CollapsedBorderValue(&nextCell->style()->borderLeft(), BCELL));
if (!result.exists()) return result;
}
}
else {
// (3) Our row's right border.
result = compareBorders(result, CollapsedBorderValue(&parent()->style()->borderRight(), BROW));
if (!result.exists()) return result;
// (4) Our row group's right border.
result = compareBorders(result, CollapsedBorderValue(&section()->style()->borderRight(), BROWGROUP));
if (!result.exists()) return result;
}
// (5) Our column's right border.
RenderTableCol* colElt = table()->colElement(col()+colSpan()-1);
if (colElt) {
result = compareBorders(result, CollapsedBorderValue(&colElt->style()->borderRight(), BCOL));
if (!result.exists()) return result;
}
// (6) The next column's left border.
if (!inLastColumn) {
colElt = tableElt->colElement(col()+colSpan());
if (colElt) {
result = compareBorders(result, CollapsedBorderValue(&colElt->style()->borderLeft(), BCOL));
if (!result.exists()) return result;
}
}
else {
// (7) The table's right border.
result = compareBorders(result, CollapsedBorderValue(&tableElt->style()->borderRight(), BTABLE));
if (!result.exists()) return result;
}
return result;
}
CollapsedBorderValue RenderTableCell::collapsedTopBorder() const
{
// For border top, we need to check, in order of precedence:
// (1) Our top border.
CollapsedBorderValue result = CollapsedBorderValue(&style()->borderTop(), BCELL);
RenderTableCell* prevCell = table()->cellAbove(this);
if (prevCell) {
// (2) A previous cell's bottom border.
result = compareBorders(result, CollapsedBorderValue(&prevCell->style()->borderBottom(), BCELL));
if (!result.exists()) return result;
}
// (3) Our row's top border.
result = compareBorders(result, CollapsedBorderValue(&parent()->style()->borderTop(), BROW));
if (!result.exists()) return result;
// (4) The previous row's bottom border.
if (prevCell) {
RenderObject* prevRow = 0;
if (prevCell->section() == section())
prevRow = parent()->previousSibling();
else
prevRow = prevCell->section()->lastChild();
if (prevRow) {
result = compareBorders(result, CollapsedBorderValue(&prevRow->style()->borderBottom(), BROW));
if (!result.exists()) return result;
}
}
// Now check row groups.
RenderTableSection* currSection = section();
if (row() == 0) {
// (5) Our row group's top border.
result = compareBorders(result, CollapsedBorderValue(&currSection->style()->borderTop(), BROWGROUP));
if (!result.exists()) return result;
// (6) Previous row group's bottom border.
currSection = table()->sectionAbove(currSection);
if (currSection) {
result = compareBorders(result, CollapsedBorderValue(&currSection->style()->borderBottom(), BROWGROUP));
if (!result.exists())
return result;
}
}
if (!currSection) {
// (8) Our column's top border.
RenderTableCol* colElt = table()->colElement(col());
if (colElt) {
result = compareBorders(result, CollapsedBorderValue(&colElt->style()->borderTop(), BCOL));
if (!result.exists()) return result;
}
// (9) The table's top border.
result = compareBorders(result, CollapsedBorderValue(&table()->style()->borderTop(), BTABLE));
if (!result.exists()) return result;
}
return result;
}
CollapsedBorderValue RenderTableCell::collapsedBottomBorder() const
{
// For border top, we need to check, in order of precedence:
// (1) Our bottom border.
CollapsedBorderValue result = CollapsedBorderValue(&style()->borderBottom(), BCELL);
RenderTableCell* nextCell = table()->cellBelow(this);
if (nextCell) {
// (2) A following cell's top border.
result = compareBorders(result, CollapsedBorderValue(&nextCell->style()->borderTop(), BCELL));
if (!result.exists()) return result;
}
// (3) Our row's bottom border. (FIXME: Deal with rowspan!)
result = compareBorders(result, CollapsedBorderValue(&parent()->style()->borderBottom(), BROW));
if (!result.exists()) return result;
// (4) The next row's top border.
if (nextCell) {
result = compareBorders(result, CollapsedBorderValue(&nextCell->parent()->style()->borderTop(), BROW));
if (!result.exists()) return result;
}
// Now check row groups.
RenderTableSection* currSection = section();
if (row()+rowSpan() >= static_cast<RenderTableSection*>(currSection)->numRows()) {
// (5) Our row group's bottom border.
result = compareBorders(result, CollapsedBorderValue(&currSection->style()->borderBottom(), BROWGROUP));
if (!result.exists()) return result;
// (6) Following row group's top border.
currSection = table()->sectionBelow(currSection);
if (currSection) {
result = compareBorders(result, CollapsedBorderValue(&currSection->style()->borderTop(), BROWGROUP));
if (!result.exists())
return result;
}
}
if (!currSection) {
// (8) Our column's bottom border.
RenderTableCol* colElt = table()->colElement(col());
if (colElt) {
result = compareBorders(result, CollapsedBorderValue(&colElt->style()->borderBottom(), BCOL));
if (!result.exists()) return result;
}
// (9) The table's bottom border.
result = compareBorders(result, CollapsedBorderValue(&table()->style()->borderBottom(), BTABLE));
if (!result.exists()) return result;
}
return result;
}
int RenderTableCell::borderLeft() const
{
if (table()->collapseBorders()) {
CollapsedBorderValue border = collapsedLeftBorder();
if (border.exists())
return (border.width()+1)/2; // Give the extra pixel to top and left.
return 0;
}
return RenderBlock::borderLeft();
}
int RenderTableCell::borderRight() const
{
if (table()->collapseBorders()) {
CollapsedBorderValue border = collapsedRightBorder();
if (border.exists())
return border.width()/2;
return 0;
}
return RenderBlock::borderRight();
}
int RenderTableCell::borderTop() const
{
if (table()->collapseBorders()) {
CollapsedBorderValue border = collapsedTopBorder();
if (border.exists())
return (border.width()+1)/2; // Give the extra pixel to top and left.
return 0;
}
return RenderBlock::borderTop();
}
int RenderTableCell::borderBottom() const
{
if (table()->collapseBorders()) {
CollapsedBorderValue border = collapsedBottomBorder();
if (border.exists())
return border.width()/2;
return 0;
}
return RenderBlock::borderBottom();
}
#ifdef BOX_DEBUG
#include <tqpainter.h>
static void outlineBox(TQPainter *p, int _tx, int _ty, int w, int h)
{
p->setPen(TQPen(TQColor("yellow"), 3, Qt::DotLine));
p->setBrush( Qt::NoBrush );
p->drawRect(_tx, _ty, w, h );
}
#endif
void RenderTableCell::paint(PaintInfo& pI, int _tx, int _ty)
{
#ifdef TABLE_PRINT
kdDebug( 6040 ) << renderName() << "(RenderTableCell)::paint() w/h = (" << width() << "/" << height() << ")" << " _y/_h=" << pI.r.y() << "/" << pI.r.height() << endl;
#endif
if (needsLayout()) return;
_tx += m_x;
_ty += m_y/* + _topExtra*/;
RenderTable *tbl = table();
// check if we need to do anything at all...
int os = kMax(tbl->currentBorderStyle() ? (tbl->currentBorderStyle()->border->width+1)/2 : 0, 2*maximalOutlineSize(pI.phase));
if (!overhangingContents() && ((_ty >= pI.r.y() + pI.r.height() + os)
|| (_ty + _topExtra + m_height + _bottomExtra <= pI.r.y() - os))) return;
if (pI.phase == PaintActionOutline) {
paintOutline( pI.p, _tx, _ty, width(), height() + borderTopExtra() + borderBottomExtra(), style());
}
if (pI.phase == PaintActionCollapsedTableBorders && style()->visibility() == VISIBLE) {
int w = width();
int h = height() + borderTopExtra() + borderBottomExtra();
paintCollapsedBorder(pI.p, _tx, _ty, w, h);
}
else
RenderBlock::paintObject(pI, _tx, _ty + _topExtra, false);
#ifdef BOX_DEBUG
::outlineBox( p, _tx, _ty - _topExtra, width(), height() + borderTopExtra() + borderBottomExtra());
#endif
}
static EBorderStyle collapsedBorderStyle(EBorderStyle style)
{
if (style == OUTSET)
style = GROOVE;
else if (style == INSET)
style = RIDGE;
return style;
}
struct CollapsedBorder {
CollapsedBorder(){}
CollapsedBorderValue border;
RenderObject::BorderSide side;
bool shouldPaint;
int x1;
int y1;
int x2;
int y2;
EBorderStyle style;
};
class CollapsedBorders
{
public:
CollapsedBorders() :count(0) {}
void addBorder(const CollapsedBorderValue& b, RenderObject::BorderSide s, bool paint,
int _x1, int _y1, int _x2, int _y2,
EBorderStyle _style)
{
if (b.exists() && paint) {
borders[count].border = b;
borders[count].side = s;
borders[count].shouldPaint = paint;
borders[count].x1 = _x1;
borders[count].x2 = _x2;
borders[count].y1 = _y1;
borders[count].y2 = _y2;
borders[count].style = _style;
count++;
}
}
CollapsedBorder* nextBorder() {
for (int i = 0; i < count; i++) {
if (borders[i].border.exists() && borders[i].shouldPaint) {
borders[i].shouldPaint = false;
return &borders[i];
}
}
return 0;
}
CollapsedBorder borders[4];
int count;
};
static void addBorderStyle(TQValueList<CollapsedBorderValue>& borderStyles, CollapsedBorderValue borderValue)
{
if (!borderValue.exists() || borderStyles.contains(borderValue))
return;
TQValueListIterator<CollapsedBorderValue> it = borderStyles.begin();
TQValueListIterator<CollapsedBorderValue> end = borderStyles.end();
for (; it != end; ++it) {
CollapsedBorderValue result = compareBorders(*it, borderValue);
if (result == *it) {
borderStyles.insert(it, borderValue);
return;
}
}
borderStyles.append(borderValue);
}
void RenderTableCell::collectBorders(TQValueList<CollapsedBorderValue>& borderStyles)
{
addBorderStyle(borderStyles, collapsedLeftBorder());
addBorderStyle(borderStyles, collapsedRightBorder());
addBorderStyle(borderStyles, collapsedTopBorder());
addBorderStyle(borderStyles, collapsedBottomBorder());
}
void RenderTableCell::paintCollapsedBorder(TQPainter* p, int _tx, int _ty, int w, int h)
{
if (!table()->currentBorderStyle())
return;
CollapsedBorderValue leftVal = collapsedLeftBorder();
CollapsedBorderValue rightVal = collapsedRightBorder();
CollapsedBorderValue topVal = collapsedTopBorder();
CollapsedBorderValue bottomVal = collapsedBottomBorder();
// Adjust our x/y/width/height so that we paint the collapsed borders at the correct location.
int topWidth = topVal.width();
int bottomWidth = bottomVal.width();
int leftWidth = leftVal.width();
int rightWidth = rightVal.width();
_tx -= leftWidth/2;
_ty -= topWidth/2;
w += leftWidth/2 + (rightWidth+1)/2;
h += topWidth/2 + (bottomWidth+1)/2;
bool tt = topVal.isTransparent();
bool bt = bottomVal.isTransparent();
bool rt = rightVal.isTransparent();
bool lt = leftVal.isTransparent();
EBorderStyle ts = collapsedBorderStyle(topVal.style());
EBorderStyle bs = collapsedBorderStyle(bottomVal.style());
EBorderStyle ls = collapsedBorderStyle(leftVal.style());
EBorderStyle rs = collapsedBorderStyle(rightVal.style());
bool render_t = ts > BHIDDEN && !tt && (topVal.precedence != BCELL || *topVal.border == style()->borderTop());
bool render_l = ls > BHIDDEN && !lt && (leftVal.precedence != BCELL || *leftVal.border == style()->borderLeft());
bool render_r = rs > BHIDDEN && !rt && (rightVal.precedence != BCELL || *rightVal.border == style()->borderRight());
bool render_b = bs > BHIDDEN && !bt && (bottomVal.precedence != BCELL || *bottomVal.border == style()->borderBottom());
// We never paint diagonals at the joins. We simply let the border with the highest
// precedence paint on top of borders with lower precedence.
CollapsedBorders borders;
borders.addBorder(topVal, BSTop, render_t, _tx, _ty, _tx + w, _ty + topWidth, ts);
borders.addBorder(bottomVal, BSBottom, render_b, _tx, _ty + h - bottomWidth, _tx + w, _ty + h, bs);
borders.addBorder(leftVal, BSLeft, render_l, _tx, _ty, _tx + leftWidth, _ty + h, ls);
borders.addBorder(rightVal, BSRight, render_r, _tx + w - rightWidth, _ty, _tx + w, _ty + h, rs);
for (CollapsedBorder* border = borders.nextBorder(); border; border = borders.nextBorder()) {
if (border->border == *table()->currentBorderStyle())
drawBorder(p, border->x1, border->y1, border->x2, border->y2, border->side,
border->border.color(), style()->color(), border->style, 0, 0);
}
}
void RenderTableCell::paintBackgroundsBehindCell(PaintInfo& pI, int _tx, int _ty, RenderObject* backgroundObject)
{
if (!backgroundObject)
return;
RenderTable* tableElt = table();
if (backgroundObject != this) {
_tx += m_x;
_ty += m_y + _topExtra;
}
int w = width();
int h = height() + borderTopExtra() + borderBottomExtra();
_ty -= borderTopExtra();
int my = kMax(_ty,pI.r.y());
int end = kMin( pI.r.y() + pI.r.height(), _ty + h );
int mh = end - my;
TQColor c = backgroundObject->style()->backgroundColor();
const BackgroundLayer* bgLayer = backgroundObject->style()->backgroundLayers();
if (bgLayer->hasImage() || c.isValid()) {
// We have to clip here because the background would paint
// on top of the borders otherwise. This only matters for cells and rows.
bool hasLayer = backgroundObject->layer() && (backgroundObject == this || backgroundObject == parent());
if (hasLayer && tableElt->collapseBorders()) {
pI.p->save();
TQRect clipRect(_tx + borderLeft(), _ty + borderTop(), w - borderLeft() - borderRight(), h - borderTop() - borderBottom());
clipRect = pI.p->xForm(clipRect);
TQRegion creg(clipRect);
TQRegion old = pI.p->clipRegion();
if (!old.isNull())
creg = old.intersect(creg);
pI.p->setClipRegion(creg);
}
paintBackground(pI.p, c, bgLayer, my, mh, _tx, _ty, w, h);
if (hasLayer && tableElt->collapseBorders())
pI.p->restore();
}
}
void RenderTableCell::paintBoxDecorations(PaintInfo& pI, int _tx, int _ty)
{
RenderTable* tableElt = table();
bool drawBorders = true;
// Moz paints bgcolor/bgimage on <td>s in quirks mode even if
// empty-cells are on. Fixes regression on #43426, attachment #354
if (!tableElt->collapseBorders() && style()->emptyCells() == HIDE && !firstChild())
drawBorders = false;
if (!style()->htmlHacks() && !drawBorders) return;
// Paint our cell background.
paintBackgroundsBehindCell(pI, _tx, _ty, this);
int w = width();
int h = height() + borderTopExtra() + borderBottomExtra();
_ty -= borderTopExtra();
if (drawBorders && style()->hasBorder() && !tableElt->collapseBorders())
paintBorder(pI.p, _tx, _ty, w, h, style());
}
#ifdef ENABLE_DUMP
void RenderTableCell::dump(TQTextStream &stream, const TQString &ind) const
{
RenderFlow::dump(stream,ind);
stream << " row=" << _row;
stream << " col=" << _col;
stream << " rSpan=" << rSpan;
stream << " cSpan=" << cSpan;
// *stream << " nWrap=" << nWrap;
}
#endif
// -------------------------------------------------------------------------
RenderTableCol::RenderTableCol(DOM::NodeImpl* node)
: RenderContainer(node), m_span(1)
{
// init RenderObject attributes
setInline(true); // our object is not Inline
updateFromElement();
}
void RenderTableCol::updateFromElement()
{
DOM::NodeImpl *node = element();
if ( node && (node->id() == ID_COL || node->id() == ID_COLGROUP) ) {
DOM::HTMLTableColElementImpl *tc = static_cast<DOM::HTMLTableColElementImpl *>(node);
m_span = tc->span();
} else
m_span = ! ( style() && style()->display() == TABLE_COLUMN_GROUP );
}
#ifdef ENABLE_DUMP
void RenderTableCol::dump(TQTextStream &stream, const TQString &ind) const
{
RenderContainer::dump(stream,ind);
stream << " _span=" << m_span;
}
#endif
// -------------------------------------------------------------------------
TableSectionIterator::TableSectionIterator(RenderTable *table, bool fromEnd)
{
if (fromEnd) {
sec = table->foot;
if (sec) return;
sec = static_cast<RenderTableSection *>(table->lastChild());
while (sec && (!sec->isTableSection()
|| sec == table->head || sec == table->foot))
sec = static_cast<RenderTableSection *>(sec->previousSibling());
if (sec) return;
sec = table->head;
} else {
sec = table->head;
if (sec) return;
sec = static_cast<RenderTableSection *>(table->firstChild());
while (sec && (!sec->isTableSection()
|| sec == table->head || sec == table->foot))
sec = static_cast<RenderTableSection *>(sec->nextSibling());
if (sec) return;
sec = table->foot;
}/*end if*/
}
TableSectionIterator &TableSectionIterator::operator ++()
{
RenderTable *table = sec->table();
if (sec == table->head) {
sec = static_cast<RenderTableSection *>(table->firstChild());
while (sec && (!sec->isTableSection()
|| sec == table->head || sec == table->foot))
sec = static_cast<RenderTableSection *>(sec->nextSibling());
if (sec) return *this;
} else if (sec == table->foot) {
sec = 0;
return *this;
} else {
do {
sec = static_cast<RenderTableSection *>(sec->nextSibling());
} while (sec && (!sec->isTableSection() || sec == table->head || sec == table->foot));
if (sec) return *this;
}/*end if*/
sec = table->foot;
return *this;
}
TableSectionIterator &TableSectionIterator::operator --()
{
RenderTable *table = sec->table();
if (sec == table->foot) {
sec = static_cast<RenderTableSection *>(table->lastChild());
while (sec && (!sec->isTableSection()
|| sec == table->head || sec == table->foot))
sec = static_cast<RenderTableSection *>(sec->previousSibling());
if (sec) return *this;
} else if (sec == table->head) {
sec = 0;
return *this;
} else {
do {
sec = static_cast<RenderTableSection *>(sec->previousSibling());
} while (sec && (!sec->isTableSection() || sec == table->head || sec == table->foot));
if (sec) return *this;
}/*end if*/
sec = table->foot;
return *this;
}
#undef TABLE_DEBUG
#undef DEBUG_LAYOUT
#undef BOX_DEBUG