/**
* Copyright (C) 2008 Torch Mobile Inc. All rights reserved. (http://www.torchmobile.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.
*
*/
#include "config.h"
#if ENABLE(WML)
#include "WMLTableElement.h"
#include "Attribute.h"
#include "CSSPropertyNames.h"
#include "CSSValueKeywords.h"
#include "Document.h"
#include "HTMLNames.h"
#include "NodeList.h"
#include "RenderObject.h"
#include "Text.h"
#include "WMLErrorHandling.h"
#include "WMLNames.h"
#include <wtf/unicode/CharacterNames.h>
namespace WebCore {
using namespace WMLNames;
WMLTableElement::WMLTableElement(const QualifiedName& tagName, Document* doc)
: WMLElement(tagName, doc)
, m_columns(0)
{
}
PassRefPtr<WMLTableElement> WMLTableElement::create(const QualifiedName& tagName, Document* document)
{
return adoptRef(new WMLTableElement(tagName, document));
}
WMLTableElement::~WMLTableElement()
{
}
bool WMLTableElement::mapToEntry(const QualifiedName& attrName, MappedAttributeEntry& result) const
{
if (attrName == HTMLNames::alignAttr) {
result = eTable;
return false;
}
return WMLElement::mapToEntry(attrName, result);
}
void WMLTableElement::parseMappedAttribute(Attribute* attr)
{
if (attr->name() == columnsAttr) {
bool isNumber = false;
m_columns = attr->value().string().toUIntStrict(&isNumber);
// Spec: This required attribute specifies the number of columns for the table.
// The user agent must create a table with exactly the number of columns specified
// by the attribute value. It is an error to specify a value of zero ("0")
if (!m_columns || !isNumber)
reportWMLError(document(), WMLErrorInvalidColumnsNumberInTable);
} else if (attr->name() == HTMLNames::alignAttr)
m_alignment = parseValueForbiddingVariableReferences(attr->value());
else
WMLElement::parseMappedAttribute(attr);
}
void WMLTableElement::finishParsingChildren()
{
WMLElement::finishParsingChildren();
if (!m_columns) {
reportWMLError(document(), WMLErrorInvalidColumnsNumberInTable);
return;
}
Vector<WMLElement*> rowElements = scanTableChildElements(this, trTag);
if (rowElements.isEmpty())
return;
Vector<WMLElement*>::iterator it = rowElements.begin();
Vector<WMLElement*>::iterator end = rowElements.end();
for (; it != end; ++it) {
WMLElement* rowElement = (*it);
// Squeeze the table to fit in the desired number of columns
Vector<WMLElement*> columnElements = scanTableChildElements(rowElement, tdTag);
unsigned actualNumberOfColumns = columnElements.size();
if (actualNumberOfColumns > m_columns) {
joinSuperflousColumns(columnElements, rowElement);
columnElements = scanTableChildElements(rowElement, tdTag);
} else if (actualNumberOfColumns < m_columns) {
padWithEmptyColumns(columnElements, rowElement);
columnElements = scanTableChildElements(rowElement, tdTag);
}
// Layout cells according to the 'align' attribute
alignCells(columnElements, rowElement);
}
}
Vector<WMLElement*> WMLTableElement::scanTableChildElements(WMLElement* startElement, const QualifiedName& tagName) const
{
Vector<WMLElement*> childElements;
RefPtr<NodeList> children = startElement->childNodes();
if (!children)
return childElements;
unsigned length = children->length();
for (unsigned i = 0; i < length; ++i) {
Node* child = children->item(i);
if (child->hasTagName(tagName))
childElements.append(static_cast<WMLElement*>(child));
}
return childElements;
}
void WMLTableElement::transferAllChildrenOfElementToTargetElement(WMLElement* sourceElement, WMLElement* targetElement, unsigned startOffset) const
{
RefPtr<NodeList> children = sourceElement->childNodes();
if (!children)
return;
ExceptionCode ec = 0;
unsigned length = children->length();
for (unsigned i = startOffset; i < length; ++i) {
RefPtr<Node> clonedNode = children->item(i)->cloneNode(true);
targetElement->appendChild(clonedNode.release(), ec);
ASSERT(ec == 0);
}
}
bool WMLTableElement::tryMergeAdjacentTextCells(Node* item, Node* nextItem) const
{
if (!item || !nextItem)
return false;
if (!item->isTextNode() || !nextItem->isTextNode())
return false;
Text* itemText = static_cast<Text*>(item);
Text* nextItemText = static_cast<Text*>(nextItem);
String newContent = " ";
newContent += nextItemText->data();
ExceptionCode ec = 0;
itemText->appendData(newContent, ec);
ASSERT(ec == 0);
return true;
}
void WMLTableElement::joinSuperflousColumns(Vector<WMLElement*>& columnElements, WMLElement* rowElement) const
{
// Spec: If the actual number of columns in a row is greater than the value specified
// by this attribute, the extra columns of the row must be aggregated into the last
// column such that the row contains exactly the number of columns specified.
WMLElement* lastColumn = columnElements.at(m_columns - 1);
ASSERT(lastColumn);
// Merge superflous columns into a single one
RefPtr<WMLElement> newCell = WMLElement::create(tdTag, document());
transferAllChildrenOfElementToTargetElement(lastColumn, newCell.get(), 0);
ExceptionCode ec = 0;
unsigned actualNumberOfColumns = columnElements.size();
for (unsigned i = m_columns; i < actualNumberOfColumns; ++i) {
WMLElement* columnElement = columnElements.at(i);
unsigned startOffset = 0;
// Spec: A single inter-word space must be inserted between two cells that are being aggregated.
if (tryMergeAdjacentTextCells(newCell->lastChild(), columnElement->firstChild()))
++startOffset;
transferAllChildrenOfElementToTargetElement(columnElement, newCell.get(), startOffset);
}
// Remove the columns, that have just been merged
unsigned i = actualNumberOfColumns;
for (; i > m_columns; --i) {
rowElement->removeChild(columnElements.at(i - 1), ec);
ASSERT(ec == 0);
}
// Replace the last column in the row with the new merged column
rowElement->replaceChild(newCell.release(), lastColumn, ec);
ASSERT(ec == 0);
}
void WMLTableElement::padWithEmptyColumns(Vector<WMLElement*>& columnElements, WMLElement* rowElement) const
{
// Spec: If the actual number of columns in a row is less than the value specified by the columns
// attribute, the row must be padded with empty columns effectively as if the user agent
// appended empty td elements to the row.
ExceptionCode ec = 0;
for (unsigned i = columnElements.size(); i < m_columns; ++i) {
RefPtr<WMLElement> newCell = WMLElement::create(tdTag, document());
rowElement->appendChild(newCell.release(), ec);
ASSERT(ec == 0);
}
}
void WMLTableElement::alignCells(Vector<WMLElement*>& columnElements, WMLElement* rowElement) const
{
// Spec: User agents should consider the current language when determining
// the default alignment and the direction of the table.
bool rtl = false;
if (RenderObject* renderer = rowElement->renderer()) {
if (RenderStyle* style = renderer->style())
rtl = !style->isLeftToRightDirection();
}
rowElement->setAttribute(HTMLNames::alignAttr, rtl ? "right" : "left");
if (m_alignment.isEmpty())
return;
unsigned alignLength = m_alignment.length();
Vector<WMLElement*>::iterator it = columnElements.begin();
Vector<WMLElement*>::iterator end = columnElements.end();
for (unsigned i = 0; it != end; ++it, ++i) {
if (i == alignLength)
break;
String alignmentValue;
switch (m_alignment[i]) {
case 'C':
alignmentValue = "center";
break;
case 'L':
alignmentValue = "left";
break;
case 'R':
alignmentValue = "right";
break;
default:
break;
}
if (alignmentValue.isEmpty())
continue;
(*it)->setAttribute(HTMLNames::alignAttr, alignmentValue);
}
}
}
#endif