зеркало из https://github.com/mozilla/gecko-dev.git
964 строки
30 KiB
C++
964 строки
30 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "mozilla/dom/HTMLTableElement.h"
|
|
#include "mozilla/GenericSpecifiedValuesInlines.h"
|
|
#include "nsAttrValueInlines.h"
|
|
#include "nsHTMLStyleSheet.h"
|
|
#include "nsMappedAttributes.h"
|
|
#include "mozilla/dom/HTMLCollectionBinding.h"
|
|
#include "mozilla/dom/HTMLTableElementBinding.h"
|
|
#include "nsContentUtils.h"
|
|
#include "jsfriendapi.h"
|
|
|
|
NS_IMPL_NS_NEW_HTML_ELEMENT(Table)
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
/* ------------------------------ TableRowsCollection -------------------------------- */
|
|
/**
|
|
* This class provides a late-bound collection of rows in a table.
|
|
* mParent is NOT ref-counted to avoid circular references
|
|
*/
|
|
class TableRowsCollection final : public nsIHTMLCollection,
|
|
public nsWrapperCache
|
|
{
|
|
public:
|
|
explicit TableRowsCollection(HTMLTableElement* aParent);
|
|
|
|
NS_DECL_CYCLE_COLLECTING_ISUPPORTS
|
|
NS_DECL_NSIDOMHTMLCOLLECTION
|
|
|
|
virtual Element* GetElementAt(uint32_t aIndex) override;
|
|
virtual nsINode* GetParentObject() override
|
|
{
|
|
return mParent;
|
|
}
|
|
|
|
virtual Element*
|
|
GetFirstNamedElement(const nsAString& aName, bool& aFound) override;
|
|
virtual void GetSupportedNames(nsTArray<nsString>& aNames) override;
|
|
|
|
NS_IMETHOD ParentDestroyed();
|
|
|
|
NS_DECL_CYCLE_COLLECTION_SCRIPT_HOLDER_CLASS(TableRowsCollection)
|
|
|
|
// nsWrapperCache
|
|
using nsWrapperCache::GetWrapperPreserveColor;
|
|
virtual JSObject* WrapObject(JSContext* aCx, JS::Handle<JSObject*> aGivenProto) override;
|
|
protected:
|
|
virtual ~TableRowsCollection();
|
|
|
|
virtual JSObject* GetWrapperPreserveColorInternal() override
|
|
{
|
|
return nsWrapperCache::GetWrapperPreserveColor();
|
|
}
|
|
|
|
// Those rows that are not in table sections
|
|
HTMLTableElement* mParent;
|
|
};
|
|
|
|
|
|
TableRowsCollection::TableRowsCollection(HTMLTableElement *aParent)
|
|
: mParent(aParent)
|
|
{
|
|
}
|
|
|
|
TableRowsCollection::~TableRowsCollection()
|
|
{
|
|
// we do NOT have a ref-counted reference to mParent, so do NOT
|
|
// release it! this is to avoid circular references. The
|
|
// instantiator who provided mParent is responsible for managing our
|
|
// reference for us.
|
|
}
|
|
|
|
JSObject*
|
|
TableRowsCollection::WrapObject(JSContext* aCx, JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
return HTMLCollectionBinding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE_0(TableRowsCollection)
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(TableRowsCollection)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(TableRowsCollection)
|
|
|
|
NS_INTERFACE_TABLE_HEAD(TableRowsCollection)
|
|
NS_WRAPPERCACHE_INTERFACE_TABLE_ENTRY
|
|
NS_INTERFACE_TABLE(TableRowsCollection, nsIHTMLCollection,
|
|
nsIDOMHTMLCollection)
|
|
NS_INTERFACE_TABLE_TO_MAP_SEGUE_CYCLE_COLLECTION(TableRowsCollection)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
// Macro that can be used to avoid copy/pasting code to iterate over the
|
|
// rowgroups. _code should be the code to execute for each rowgroup. The
|
|
// rowgroup's rows will be in the nsIDOMHTMLCollection* named "rows".
|
|
// _trCode should be the code to execute for each tr row. Note that
|
|
// this may be null at any time. This macro assumes an nsresult named
|
|
// |rv| is in scope.
|
|
#define DO_FOR_EACH_BY_ORDER(_code, _trCode) \
|
|
do { \
|
|
if (mParent) { \
|
|
HTMLTableSectionElement* rowGroup; \
|
|
nsIHTMLCollection* rows; \
|
|
/* THead */ \
|
|
for (nsIContent* _node = mParent->nsINode::GetFirstChild(); \
|
|
_node; _node = _node->GetNextSibling()) { \
|
|
if (_node->IsHTMLElement(nsGkAtoms::thead)) { \
|
|
rowGroup = static_cast<HTMLTableSectionElement*>(_node);\
|
|
rows = rowGroup->Rows(); \
|
|
do { /* gives scoping */ \
|
|
_code \
|
|
} while (0); \
|
|
} \
|
|
} \
|
|
/* TBodies */ \
|
|
for (nsIContent* _node = mParent->nsINode::GetFirstChild(); \
|
|
_node; _node = _node->GetNextSibling()) { \
|
|
if (_node->IsHTMLElement(nsGkAtoms::tr)) { \
|
|
do { \
|
|
_trCode \
|
|
} while (0); \
|
|
} else if (_node->IsHTMLElement(nsGkAtoms::tbody)) { \
|
|
rowGroup = static_cast<HTMLTableSectionElement*>(_node); \
|
|
rows = rowGroup->Rows(); \
|
|
do { /* gives scoping */ \
|
|
_code \
|
|
} while (0); \
|
|
} \
|
|
} \
|
|
/* TFoot */ \
|
|
for (nsIContent* _node = mParent->nsINode::GetFirstChild(); \
|
|
_node; _node = _node->GetNextSibling()) { \
|
|
if (_node->IsHTMLElement(nsGkAtoms::tfoot)) { \
|
|
rowGroup = static_cast<HTMLTableSectionElement*>(_node);\
|
|
rows = rowGroup->Rows(); \
|
|
do { /* gives scoping */ \
|
|
_code \
|
|
} while (0); \
|
|
} \
|
|
} \
|
|
} \
|
|
} while (0)
|
|
|
|
static uint32_t
|
|
CountRowsInRowGroup(nsIDOMHTMLCollection* rows)
|
|
{
|
|
uint32_t length = 0;
|
|
|
|
if (rows) {
|
|
rows->GetLength(&length);
|
|
}
|
|
|
|
return length;
|
|
}
|
|
|
|
// we re-count every call. A better implementation would be to set
|
|
// ourselves up as an observer of contentAppended, contentInserted,
|
|
// and contentDeleted
|
|
NS_IMETHODIMP
|
|
TableRowsCollection::GetLength(uint32_t* aLength)
|
|
{
|
|
*aLength=0;
|
|
|
|
DO_FOR_EACH_BY_ORDER({
|
|
*aLength += CountRowsInRowGroup(rows);
|
|
}, {
|
|
(*aLength)++;
|
|
});
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
// Returns the item at index aIndex if available. If null is returned,
|
|
// then aCount will be set to the number of rows in this row collection.
|
|
// Otherwise, the value of aCount is undefined.
|
|
static Element*
|
|
GetItemOrCountInRowGroup(nsIDOMHTMLCollection* rows,
|
|
uint32_t aIndex, uint32_t* aCount)
|
|
{
|
|
*aCount = 0;
|
|
|
|
if (rows) {
|
|
rows->GetLength(aCount);
|
|
if (aIndex < *aCount) {
|
|
nsIHTMLCollection* list = static_cast<nsIHTMLCollection*>(rows);
|
|
return list->GetElementAt(aIndex);
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
Element*
|
|
TableRowsCollection::GetElementAt(uint32_t aIndex)
|
|
{
|
|
DO_FOR_EACH_BY_ORDER({
|
|
uint32_t count;
|
|
Element* node = GetItemOrCountInRowGroup(rows, aIndex, &count);
|
|
if (node) {
|
|
return node;
|
|
}
|
|
|
|
NS_ASSERTION(count <= aIndex, "GetItemOrCountInRowGroup screwed up");
|
|
aIndex -= count;
|
|
},{
|
|
if (aIndex == 0) {
|
|
return _node->AsElement();
|
|
}
|
|
aIndex--;
|
|
});
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
TableRowsCollection::Item(uint32_t aIndex, nsIDOMNode** aReturn)
|
|
{
|
|
nsISupports* node = GetElementAt(aIndex);
|
|
if (!node) {
|
|
*aReturn = nullptr;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
return CallQueryInterface(node, aReturn);
|
|
}
|
|
|
|
Element*
|
|
TableRowsCollection::GetFirstNamedElement(const nsAString& aName, bool& aFound)
|
|
{
|
|
aFound = false;
|
|
nsCOMPtr<nsIAtom> nameAtom = NS_Atomize(aName);
|
|
NS_ENSURE_TRUE(nameAtom, nullptr);
|
|
DO_FOR_EACH_BY_ORDER({
|
|
Element* item = rows->NamedGetter(aName, aFound);
|
|
if (aFound) {
|
|
return item;
|
|
}
|
|
}, {
|
|
if (_node->AttrValueIs(kNameSpaceID_None, nsGkAtoms::name,
|
|
nameAtom, eCaseMatters) ||
|
|
_node->AttrValueIs(kNameSpaceID_None, nsGkAtoms::id,
|
|
nameAtom, eCaseMatters)) {
|
|
aFound = true;
|
|
return _node->AsElement();
|
|
}
|
|
});
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
void
|
|
TableRowsCollection::GetSupportedNames(nsTArray<nsString>& aNames)
|
|
{
|
|
DO_FOR_EACH_BY_ORDER({
|
|
nsTArray<nsString> names;
|
|
nsCOMPtr<nsIHTMLCollection> coll = do_QueryInterface(rows);
|
|
if (coll) {
|
|
coll->GetSupportedNames(names);
|
|
for (uint32_t i = 0; i < names.Length(); ++i) {
|
|
if (!aNames.Contains(names[i])) {
|
|
aNames.AppendElement(names[i]);
|
|
}
|
|
}
|
|
}
|
|
}, {
|
|
if (_node->HasID()) {
|
|
nsIAtom* idAtom = _node->GetID();
|
|
MOZ_ASSERT(idAtom != nsGkAtoms::_empty,
|
|
"Empty ids don't get atomized");
|
|
nsDependentAtomString idStr(idAtom);
|
|
if (!aNames.Contains(idStr)) {
|
|
aNames.AppendElement(idStr);
|
|
}
|
|
}
|
|
|
|
nsGenericHTMLElement* el = nsGenericHTMLElement::FromContent(_node);
|
|
if (el) {
|
|
const nsAttrValue* val = el->GetParsedAttr(nsGkAtoms::name);
|
|
if (val && val->Type() == nsAttrValue::eAtom) {
|
|
nsIAtom* nameAtom = val->GetAtomValue();
|
|
MOZ_ASSERT(nameAtom != nsGkAtoms::_empty,
|
|
"Empty names don't get atomized");
|
|
nsDependentAtomString nameStr(nameAtom);
|
|
if (!aNames.Contains(nameStr)) {
|
|
aNames.AppendElement(nameStr);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
TableRowsCollection::NamedItem(const nsAString& aName,
|
|
nsIDOMNode** aReturn)
|
|
{
|
|
bool found;
|
|
nsISupports* node = GetFirstNamedElement(aName, found);
|
|
if (!node) {
|
|
*aReturn = nullptr;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
return CallQueryInterface(node, aReturn);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
TableRowsCollection::ParentDestroyed()
|
|
{
|
|
// see comment in destructor, do NOT release mParent!
|
|
mParent = nullptr;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
/* --------------------------- HTMLTableElement ---------------------------- */
|
|
|
|
HTMLTableElement::HTMLTableElement(already_AddRefed<mozilla::dom::NodeInfo>& aNodeInfo)
|
|
: nsGenericHTMLElement(aNodeInfo),
|
|
mTableInheritedAttributes(TABLE_ATTRS_DIRTY)
|
|
{
|
|
SetHasWeirdParserInsertionMode();
|
|
}
|
|
|
|
HTMLTableElement::~HTMLTableElement()
|
|
{
|
|
if (mRows) {
|
|
mRows->ParentDestroyed();
|
|
}
|
|
ReleaseInheritedAttributes();
|
|
}
|
|
|
|
JSObject*
|
|
HTMLTableElement::WrapNode(JSContext *aCx, JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
return HTMLTableElementBinding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(HTMLTableElement)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(HTMLTableElement, nsGenericHTMLElement)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mTBodies)
|
|
if (tmp->mRows) {
|
|
tmp->mRows->ParentDestroyed();
|
|
}
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mRows)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(HTMLTableElement,
|
|
nsGenericHTMLElement)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTBodies)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRows)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
NS_IMPL_ADDREF_INHERITED(HTMLTableElement, Element)
|
|
NS_IMPL_RELEASE_INHERITED(HTMLTableElement, Element)
|
|
|
|
// QueryInterface implementation for HTMLTableElement
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(HTMLTableElement)
|
|
NS_INTERFACE_MAP_END_INHERITING(nsGenericHTMLElement)
|
|
|
|
|
|
NS_IMPL_ELEMENT_CLONE(HTMLTableElement)
|
|
|
|
|
|
// the DOM spec says border, cellpadding, cellSpacing are all "wstring"
|
|
// in fact, they are integers or they are meaningless. so we store them
|
|
// here as ints.
|
|
|
|
nsIHTMLCollection*
|
|
HTMLTableElement::Rows()
|
|
{
|
|
if (!mRows) {
|
|
mRows = new TableRowsCollection(this);
|
|
}
|
|
|
|
return mRows;
|
|
}
|
|
|
|
nsIHTMLCollection*
|
|
HTMLTableElement::TBodies()
|
|
{
|
|
if (!mTBodies) {
|
|
// Not using NS_GetContentList because this should not be cached
|
|
mTBodies = new nsContentList(this,
|
|
kNameSpaceID_XHTML,
|
|
nsGkAtoms::tbody,
|
|
nsGkAtoms::tbody,
|
|
false);
|
|
}
|
|
|
|
return mTBodies;
|
|
}
|
|
|
|
already_AddRefed<nsGenericHTMLElement>
|
|
HTMLTableElement::CreateTHead()
|
|
{
|
|
RefPtr<nsGenericHTMLElement> head = GetTHead();
|
|
if (!head) {
|
|
// Create a new head rowgroup.
|
|
RefPtr<mozilla::dom::NodeInfo> nodeInfo;
|
|
nsContentUtils::QNameChanged(mNodeInfo, nsGkAtoms::thead,
|
|
getter_AddRefs(nodeInfo));
|
|
|
|
head = NS_NewHTMLTableSectionElement(nodeInfo.forget());
|
|
if (!head) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsCOMPtr<nsIContent> refNode = nullptr;
|
|
for (refNode = nsINode::GetFirstChild();
|
|
refNode;
|
|
refNode = refNode->GetNextSibling()) {
|
|
|
|
if (refNode->IsHTMLElement() &&
|
|
!refNode->IsHTMLElement(nsGkAtoms::caption) &&
|
|
!refNode->IsHTMLElement(nsGkAtoms::colgroup)) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
IgnoredErrorResult rv;
|
|
nsINode::InsertBefore(*head, refNode, rv);
|
|
}
|
|
return head.forget();
|
|
}
|
|
|
|
void
|
|
HTMLTableElement::DeleteTHead()
|
|
{
|
|
HTMLTableSectionElement* tHead = GetTHead();
|
|
if (tHead) {
|
|
mozilla::ErrorResult rv;
|
|
nsINode::RemoveChild(*tHead, rv);
|
|
MOZ_ASSERT(!rv.Failed());
|
|
}
|
|
}
|
|
|
|
already_AddRefed<nsGenericHTMLElement>
|
|
HTMLTableElement::CreateTFoot()
|
|
{
|
|
RefPtr<nsGenericHTMLElement> foot = GetTFoot();
|
|
if (!foot) {
|
|
// create a new foot rowgroup
|
|
RefPtr<mozilla::dom::NodeInfo> nodeInfo;
|
|
nsContentUtils::QNameChanged(mNodeInfo, nsGkAtoms::tfoot,
|
|
getter_AddRefs(nodeInfo));
|
|
|
|
foot = NS_NewHTMLTableSectionElement(nodeInfo.forget());
|
|
if (!foot) {
|
|
return nullptr;
|
|
}
|
|
AppendChildTo(foot, true);
|
|
}
|
|
|
|
return foot.forget();
|
|
}
|
|
|
|
void
|
|
HTMLTableElement::DeleteTFoot()
|
|
{
|
|
HTMLTableSectionElement* tFoot = GetTFoot();
|
|
if (tFoot) {
|
|
mozilla::ErrorResult rv;
|
|
nsINode::RemoveChild(*tFoot, rv);
|
|
MOZ_ASSERT(!rv.Failed());
|
|
}
|
|
}
|
|
|
|
already_AddRefed<nsGenericHTMLElement>
|
|
HTMLTableElement::CreateCaption()
|
|
{
|
|
RefPtr<nsGenericHTMLElement> caption = GetCaption();
|
|
if (!caption) {
|
|
// Create a new caption.
|
|
RefPtr<mozilla::dom::NodeInfo> nodeInfo;
|
|
nsContentUtils::QNameChanged(mNodeInfo, nsGkAtoms::caption,
|
|
getter_AddRefs(nodeInfo));
|
|
|
|
caption = NS_NewHTMLTableCaptionElement(nodeInfo.forget());
|
|
if (!caption) {
|
|
return nullptr;
|
|
}
|
|
|
|
IgnoredErrorResult rv;
|
|
nsCOMPtr<nsINode> firsChild = nsINode::GetFirstChild();
|
|
nsINode::InsertBefore(*caption, firsChild, rv);
|
|
}
|
|
return caption.forget();
|
|
}
|
|
|
|
void
|
|
HTMLTableElement::DeleteCaption()
|
|
{
|
|
HTMLTableCaptionElement* caption = GetCaption();
|
|
if (caption) {
|
|
mozilla::ErrorResult rv;
|
|
nsINode::RemoveChild(*caption, rv);
|
|
MOZ_ASSERT(!rv.Failed());
|
|
}
|
|
}
|
|
|
|
already_AddRefed<nsGenericHTMLElement>
|
|
HTMLTableElement::CreateTBody()
|
|
{
|
|
RefPtr<mozilla::dom::NodeInfo> nodeInfo =
|
|
OwnerDoc()->NodeInfoManager()->GetNodeInfo(nsGkAtoms::tbody, nullptr,
|
|
kNameSpaceID_XHTML,
|
|
nsIDOMNode::ELEMENT_NODE);
|
|
MOZ_ASSERT(nodeInfo);
|
|
|
|
RefPtr<nsGenericHTMLElement> newBody =
|
|
NS_NewHTMLTableSectionElement(nodeInfo.forget());
|
|
MOZ_ASSERT(newBody);
|
|
|
|
nsCOMPtr<nsIContent> referenceNode = nullptr;
|
|
for (nsIContent* child = nsINode::GetLastChild();
|
|
child;
|
|
child = child->GetPreviousSibling()) {
|
|
if (child->IsHTMLElement(nsGkAtoms::tbody)) {
|
|
referenceNode = child->GetNextSibling();
|
|
break;
|
|
}
|
|
}
|
|
|
|
IgnoredErrorResult rv;
|
|
nsINode::InsertBefore(*newBody, referenceNode, rv);
|
|
|
|
return newBody.forget();
|
|
}
|
|
|
|
already_AddRefed<nsGenericHTMLElement>
|
|
HTMLTableElement::InsertRow(int32_t aIndex, ErrorResult& aError)
|
|
{
|
|
/* get the ref row at aIndex
|
|
if there is one,
|
|
get its parent
|
|
insert the new row just before the ref row
|
|
else
|
|
get the first row group
|
|
insert the new row as its first child
|
|
*/
|
|
if (aIndex < -1) {
|
|
aError.Throw(NS_ERROR_DOM_INDEX_SIZE_ERR);
|
|
return nullptr;
|
|
}
|
|
|
|
nsIHTMLCollection* rows = Rows();
|
|
uint32_t rowCount = rows->Length();
|
|
if ((uint32_t)aIndex > rowCount && aIndex != -1) {
|
|
aError.Throw(NS_ERROR_DOM_INDEX_SIZE_ERR);
|
|
return nullptr;
|
|
}
|
|
|
|
// use local variable refIndex so we can remember original aIndex
|
|
uint32_t refIndex = (uint32_t)aIndex;
|
|
|
|
RefPtr<nsGenericHTMLElement> newRow;
|
|
if (rowCount > 0) {
|
|
if (refIndex == rowCount || aIndex == -1) {
|
|
// we set refIndex to the last row so we can get the last row's
|
|
// parent we then do an AppendChild below if (rowCount<aIndex)
|
|
|
|
refIndex = rowCount - 1;
|
|
}
|
|
|
|
RefPtr<Element> refRow = rows->Item(refIndex);
|
|
nsCOMPtr<nsINode> parent = refRow->GetParentNode();
|
|
|
|
// create the row
|
|
RefPtr<mozilla::dom::NodeInfo> nodeInfo;
|
|
nsContentUtils::QNameChanged(mNodeInfo, nsGkAtoms::tr,
|
|
getter_AddRefs(nodeInfo));
|
|
|
|
newRow = NS_NewHTMLTableRowElement(nodeInfo.forget());
|
|
|
|
if (newRow) {
|
|
// If aIndex is -1 or equal to the number of rows, the new row
|
|
// is appended.
|
|
if (aIndex == -1 || uint32_t(aIndex) == rowCount) {
|
|
parent->AppendChild(*newRow, aError);
|
|
} else {
|
|
// insert the new row before the reference row we found above
|
|
parent->InsertBefore(*newRow, refRow, aError);
|
|
}
|
|
|
|
if (aError.Failed()) {
|
|
return nullptr;
|
|
}
|
|
}
|
|
} else {
|
|
// the row count was 0, so
|
|
// find the last row group and insert there as first child
|
|
nsCOMPtr<nsIContent> rowGroup;
|
|
for (nsIContent* child = nsINode::GetLastChild();
|
|
child;
|
|
child = child->GetPreviousSibling()) {
|
|
if (child->IsHTMLElement(nsGkAtoms::tbody)) {
|
|
rowGroup = child;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!rowGroup) { // need to create a TBODY
|
|
RefPtr<mozilla::dom::NodeInfo> nodeInfo;
|
|
nsContentUtils::QNameChanged(mNodeInfo, nsGkAtoms::tbody,
|
|
getter_AddRefs(nodeInfo));
|
|
|
|
rowGroup = NS_NewHTMLTableSectionElement(nodeInfo.forget());
|
|
if (rowGroup) {
|
|
aError = AppendChildTo(rowGroup, true);
|
|
if (aError.Failed()) {
|
|
return nullptr;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (rowGroup) {
|
|
RefPtr<mozilla::dom::NodeInfo> nodeInfo;
|
|
nsContentUtils::QNameChanged(mNodeInfo, nsGkAtoms::tr,
|
|
getter_AddRefs(nodeInfo));
|
|
|
|
newRow = NS_NewHTMLTableRowElement(nodeInfo.forget());
|
|
if (newRow) {
|
|
HTMLTableSectionElement* section =
|
|
static_cast<HTMLTableSectionElement*>(rowGroup.get());
|
|
nsIHTMLCollection* rows = section->Rows();
|
|
nsCOMPtr<nsINode> refNode = rows->Item(0);
|
|
rowGroup->InsertBefore(*newRow, refNode, aError);
|
|
}
|
|
}
|
|
}
|
|
|
|
return newRow.forget();
|
|
}
|
|
|
|
void
|
|
HTMLTableElement::DeleteRow(int32_t aIndex, ErrorResult& aError)
|
|
{
|
|
if (aIndex < -1) {
|
|
aError.Throw(NS_ERROR_DOM_INDEX_SIZE_ERR);
|
|
return;
|
|
}
|
|
|
|
nsIHTMLCollection* rows = Rows();
|
|
uint32_t refIndex;
|
|
if (aIndex == -1) {
|
|
refIndex = rows->Length();
|
|
if (refIndex == 0) {
|
|
return;
|
|
}
|
|
|
|
--refIndex;
|
|
} else {
|
|
refIndex = (uint32_t)aIndex;
|
|
}
|
|
|
|
nsCOMPtr<nsIContent> row = rows->Item(refIndex);
|
|
if (!row) {
|
|
aError.Throw(NS_ERROR_DOM_INDEX_SIZE_ERR);
|
|
return;
|
|
}
|
|
|
|
row->RemoveFromParent();
|
|
}
|
|
|
|
bool
|
|
HTMLTableElement::ParseAttribute(int32_t aNamespaceID,
|
|
nsIAtom* aAttribute,
|
|
const nsAString& aValue,
|
|
nsAttrValue& aResult)
|
|
{
|
|
/* ignore summary, just a string */
|
|
if (aNamespaceID == kNameSpaceID_None) {
|
|
if (aAttribute == nsGkAtoms::cellspacing ||
|
|
aAttribute == nsGkAtoms::cellpadding ||
|
|
aAttribute == nsGkAtoms::border) {
|
|
return aResult.ParseNonNegativeIntValue(aValue);
|
|
}
|
|
if (aAttribute == nsGkAtoms::height) {
|
|
return aResult.ParseSpecialIntValue(aValue);
|
|
}
|
|
if (aAttribute == nsGkAtoms::width) {
|
|
if (aResult.ParseSpecialIntValue(aValue)) {
|
|
// treat 0 width as auto
|
|
nsAttrValue::ValueType type = aResult.Type();
|
|
return !((type == nsAttrValue::eInteger &&
|
|
aResult.GetIntegerValue() == 0) ||
|
|
(type == nsAttrValue::ePercent &&
|
|
aResult.GetPercentValue() == 0.0f));
|
|
}
|
|
return false;
|
|
}
|
|
|
|
if (aAttribute == nsGkAtoms::align) {
|
|
return ParseTableHAlignValue(aValue, aResult);
|
|
}
|
|
if (aAttribute == nsGkAtoms::bgcolor ||
|
|
aAttribute == nsGkAtoms::bordercolor) {
|
|
return aResult.ParseColor(aValue);
|
|
}
|
|
if (aAttribute == nsGkAtoms::hspace ||
|
|
aAttribute == nsGkAtoms::vspace) {
|
|
return aResult.ParseIntWithBounds(aValue, 0);
|
|
}
|
|
}
|
|
|
|
return nsGenericHTMLElement::ParseBackgroundAttribute(aNamespaceID,
|
|
aAttribute, aValue,
|
|
aResult) ||
|
|
nsGenericHTMLElement::ParseAttribute(aNamespaceID, aAttribute, aValue,
|
|
aResult);
|
|
}
|
|
|
|
|
|
|
|
void
|
|
HTMLTableElement::MapAttributesIntoRule(const nsMappedAttributes* aAttributes,
|
|
GenericSpecifiedValues* aData)
|
|
{
|
|
// XXX Bug 211636: This function is used by a single style rule
|
|
// that's used to match two different type of elements -- tables, and
|
|
// table cells. (nsHTMLTableCellElement overrides
|
|
// WalkContentStyleRules so that this happens.) This violates the
|
|
// nsIStyleRule contract, since it's the same style rule object doing
|
|
// the mapping in two different ways. It's also incorrect since it's
|
|
// testing the display type of the style context rather than checking
|
|
// which *element* it's matching (style rules should not stop matching
|
|
// when the display type is changed).
|
|
|
|
nsPresContext* presContext = aData->PresContext();
|
|
nsCompatibility mode = presContext->CompatibilityMode();
|
|
|
|
if (aData->ShouldComputeStyleStruct(NS_STYLE_INHERIT_BIT(TableBorder))) {
|
|
// cellspacing
|
|
const nsAttrValue* value = aAttributes->GetAttr(nsGkAtoms::cellspacing);
|
|
if (value && value->Type() == nsAttrValue::eInteger &&
|
|
!aData->PropertyIsSet(eCSSProperty_border_spacing)) {
|
|
aData->SetPixelValue(eCSSProperty_border_spacing, float(value->GetIntegerValue()));
|
|
}
|
|
}
|
|
if (aData->ShouldComputeStyleStruct(NS_STYLE_INHERIT_BIT(Margin))) {
|
|
// align; Check for enumerated type (it may be another type if
|
|
// illegal)
|
|
const nsAttrValue* value = aAttributes->GetAttr(nsGkAtoms::align);
|
|
|
|
if (value && value->Type() == nsAttrValue::eEnum) {
|
|
if (value->GetEnumValue() == NS_STYLE_TEXT_ALIGN_CENTER ||
|
|
value->GetEnumValue() == NS_STYLE_TEXT_ALIGN_MOZ_CENTER) {
|
|
aData->SetAutoValueIfUnset(eCSSProperty_margin_left);
|
|
aData->SetAutoValueIfUnset(eCSSProperty_margin_right);
|
|
}
|
|
}
|
|
|
|
// hspace is mapped into left and right margin,
|
|
// vspace is mapped into top and bottom margins
|
|
// - *** Quirks Mode only ***
|
|
if (eCompatibility_NavQuirks == mode) {
|
|
value = aAttributes->GetAttr(nsGkAtoms::hspace);
|
|
|
|
if (value && value->Type() == nsAttrValue::eInteger) {
|
|
aData->SetPixelValueIfUnset(eCSSProperty_margin_left, (float)value->GetIntegerValue());
|
|
aData->SetPixelValueIfUnset(eCSSProperty_margin_right, (float)value->GetIntegerValue());
|
|
}
|
|
|
|
value = aAttributes->GetAttr(nsGkAtoms::vspace);
|
|
|
|
if (value && value->Type() == nsAttrValue::eInteger) {
|
|
aData->SetPixelValueIfUnset(eCSSProperty_margin_top, (float)value->GetIntegerValue());
|
|
aData->SetPixelValueIfUnset(eCSSProperty_margin_bottom, (float)value->GetIntegerValue());
|
|
}
|
|
}
|
|
}
|
|
if (aData->ShouldComputeStyleStruct(NS_STYLE_INHERIT_BIT(Border))) {
|
|
// bordercolor
|
|
const nsAttrValue* value = aAttributes->GetAttr(nsGkAtoms::bordercolor);
|
|
nscolor color;
|
|
if (value && presContext->UseDocumentColors() &&
|
|
value->GetColorValue(color)) {
|
|
aData->SetColorValueIfUnset(eCSSProperty_border_top_color, color);
|
|
aData->SetColorValueIfUnset(eCSSProperty_border_left_color, color);
|
|
aData->SetColorValueIfUnset(eCSSProperty_border_bottom_color, color);
|
|
aData->SetColorValueIfUnset(eCSSProperty_border_right_color, color);
|
|
}
|
|
|
|
// border
|
|
const nsAttrValue* borderValue = aAttributes->GetAttr(nsGkAtoms::border);
|
|
if (borderValue) {
|
|
// border = 1 pixel default
|
|
int32_t borderThickness = 1;
|
|
|
|
if (borderValue->Type() == nsAttrValue::eInteger)
|
|
borderThickness = borderValue->GetIntegerValue();
|
|
|
|
// by default, set all border sides to the specified width
|
|
aData->SetPixelValueIfUnset(eCSSProperty_border_top_width, (float)borderThickness);
|
|
aData->SetPixelValueIfUnset(eCSSProperty_border_left_width, (float)borderThickness);
|
|
aData->SetPixelValueIfUnset(eCSSProperty_border_bottom_width, (float)borderThickness);
|
|
aData->SetPixelValueIfUnset(eCSSProperty_border_right_width, (float)borderThickness);
|
|
}
|
|
}
|
|
nsGenericHTMLElement::MapImageSizeAttributesInto(aAttributes, aData);
|
|
nsGenericHTMLElement::MapBackgroundAttributesInto(aAttributes, aData);
|
|
nsGenericHTMLElement::MapCommonAttributesInto(aAttributes, aData);
|
|
}
|
|
|
|
NS_IMETHODIMP_(bool)
|
|
HTMLTableElement::IsAttributeMapped(const nsIAtom* aAttribute) const
|
|
{
|
|
static const MappedAttributeEntry attributes[] = {
|
|
{ &nsGkAtoms::cellpadding },
|
|
{ &nsGkAtoms::cellspacing },
|
|
{ &nsGkAtoms::border },
|
|
{ &nsGkAtoms::width },
|
|
{ &nsGkAtoms::height },
|
|
{ &nsGkAtoms::hspace },
|
|
{ &nsGkAtoms::vspace },
|
|
|
|
{ &nsGkAtoms::bordercolor },
|
|
|
|
{ &nsGkAtoms::align },
|
|
{ nullptr }
|
|
};
|
|
|
|
static const MappedAttributeEntry* const map[] = {
|
|
attributes,
|
|
sCommonAttributeMap,
|
|
sBackgroundAttributeMap,
|
|
};
|
|
|
|
return FindAttributeDependence(aAttribute, map);
|
|
}
|
|
|
|
nsMapRuleToAttributesFunc
|
|
HTMLTableElement::GetAttributeMappingFunction() const
|
|
{
|
|
return &MapAttributesIntoRule;
|
|
}
|
|
|
|
static void
|
|
MapInheritedTableAttributesIntoRule(const nsMappedAttributes* aAttributes,
|
|
GenericSpecifiedValues* aData)
|
|
{
|
|
if (aData->ShouldComputeStyleStruct(NS_STYLE_INHERIT_BIT(Padding))) {
|
|
const nsAttrValue* value = aAttributes->GetAttr(nsGkAtoms::cellpadding);
|
|
if (value && value->Type() == nsAttrValue::eInteger) {
|
|
// We have cellpadding. This will override our padding values if we
|
|
// don't have any set.
|
|
float pad = float(value->GetIntegerValue());
|
|
|
|
aData->SetPixelValueIfUnset(eCSSProperty_padding_top, pad);
|
|
aData->SetPixelValueIfUnset(eCSSProperty_padding_right, pad);
|
|
aData->SetPixelValueIfUnset(eCSSProperty_padding_bottom, pad);
|
|
aData->SetPixelValueIfUnset(eCSSProperty_padding_left, pad);
|
|
}
|
|
}
|
|
}
|
|
|
|
nsMappedAttributes*
|
|
HTMLTableElement::GetAttributesMappedForCell()
|
|
{
|
|
if (mTableInheritedAttributes) {
|
|
if (mTableInheritedAttributes == TABLE_ATTRS_DIRTY)
|
|
BuildInheritedAttributes();
|
|
if (mTableInheritedAttributes != TABLE_ATTRS_DIRTY)
|
|
return mTableInheritedAttributes;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void
|
|
HTMLTableElement::BuildInheritedAttributes()
|
|
{
|
|
NS_ASSERTION(mTableInheritedAttributes == TABLE_ATTRS_DIRTY,
|
|
"potential leak, plus waste of work");
|
|
nsIDocument *document = GetComposedDoc();
|
|
nsHTMLStyleSheet* sheet = document ?
|
|
document->GetAttributeStyleSheet() : nullptr;
|
|
RefPtr<nsMappedAttributes> newAttrs;
|
|
if (sheet) {
|
|
const nsAttrValue* value = mAttrsAndChildren.GetAttr(nsGkAtoms::cellpadding);
|
|
if (value) {
|
|
RefPtr<nsMappedAttributes> modifiableMapped = new
|
|
nsMappedAttributes(sheet, MapInheritedTableAttributesIntoRule);
|
|
|
|
if (modifiableMapped) {
|
|
nsAttrValue val(*value);
|
|
modifiableMapped->SetAndTakeAttr(nsGkAtoms::cellpadding, val);
|
|
}
|
|
newAttrs = sheet->UniqueMappedAttributes(modifiableMapped);
|
|
NS_ASSERTION(newAttrs, "out of memory, but handling gracefully");
|
|
|
|
if (newAttrs != modifiableMapped) {
|
|
// Reset the stylesheet of modifiableMapped so that it doesn't
|
|
// spend time trying to remove itself from the hash. There is no
|
|
// risk that modifiableMapped is in the hash since we created
|
|
// it ourselves and it didn't come from the stylesheet (in which
|
|
// case it would not have been modifiable).
|
|
modifiableMapped->DropStyleSheetReference();
|
|
}
|
|
}
|
|
mTableInheritedAttributes = newAttrs;
|
|
NS_IF_ADDREF(mTableInheritedAttributes);
|
|
}
|
|
}
|
|
|
|
void
|
|
HTMLTableElement::ReleaseInheritedAttributes()
|
|
{
|
|
if (mTableInheritedAttributes &&
|
|
mTableInheritedAttributes != TABLE_ATTRS_DIRTY)
|
|
NS_RELEASE(mTableInheritedAttributes);
|
|
mTableInheritedAttributes = TABLE_ATTRS_DIRTY;
|
|
}
|
|
|
|
nsresult
|
|
HTMLTableElement::BindToTree(nsIDocument* aDocument, nsIContent* aParent,
|
|
nsIContent* aBindingParent,
|
|
bool aCompileEventHandlers)
|
|
{
|
|
ReleaseInheritedAttributes();
|
|
return nsGenericHTMLElement::BindToTree(aDocument, aParent,
|
|
aBindingParent,
|
|
aCompileEventHandlers);
|
|
}
|
|
|
|
void
|
|
HTMLTableElement::UnbindFromTree(bool aDeep, bool aNullParent)
|
|
{
|
|
ReleaseInheritedAttributes();
|
|
nsGenericHTMLElement::UnbindFromTree(aDeep, aNullParent);
|
|
}
|
|
|
|
nsresult
|
|
HTMLTableElement::BeforeSetAttr(int32_t aNameSpaceID, nsIAtom* aName,
|
|
const nsAttrValueOrString* aValue,
|
|
bool aNotify)
|
|
{
|
|
if (aName == nsGkAtoms::cellpadding && aNameSpaceID == kNameSpaceID_None) {
|
|
ReleaseInheritedAttributes();
|
|
}
|
|
return nsGenericHTMLElement::BeforeSetAttr(aNameSpaceID, aName, aValue,
|
|
aNotify);
|
|
}
|
|
|
|
nsresult
|
|
HTMLTableElement::AfterSetAttr(int32_t aNameSpaceID, nsIAtom* aName,
|
|
const nsAttrValue* aValue,
|
|
bool aNotify)
|
|
{
|
|
if (aName == nsGkAtoms::cellpadding && aNameSpaceID == kNameSpaceID_None) {
|
|
BuildInheritedAttributes();
|
|
}
|
|
return nsGenericHTMLElement::AfterSetAttr(aNameSpaceID, aName, aValue,
|
|
aNotify);
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|