1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009 |
- /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
- /* 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 "nsAttrValueInlines.h"
- #include "nsRuleData.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(nullptr)
- {
- 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::NameChanged(mNodeInfo, nsGkAtoms::thead,
- getter_AddRefs(nodeInfo));
- head = NS_NewHTMLTableSectionElement(nodeInfo.forget());
- if (!head) {
- return nullptr;
- }
- ErrorResult rv;
- nsCOMPtr<nsINode> refNode = nsINode::GetFirstChild();
- nsINode::InsertBefore(*head, refNode, rv);
- }
- return head.forget();
- }
- void
- HTMLTableElement::DeleteTHead()
- {
- RefPtr<HTMLTableSectionElement> tHead = GetTHead();
- if (tHead) {
- mozilla::IgnoredErrorResult rv;
- nsINode::RemoveChild(*tHead, rv);
- }
- }
- already_AddRefed<nsGenericHTMLElement>
- HTMLTableElement::CreateTFoot()
- {
- RefPtr<nsGenericHTMLElement> foot = GetTFoot();
- if (!foot) {
- // create a new foot rowgroup
- RefPtr<mozilla::dom::NodeInfo> nodeInfo;
- nsContentUtils::NameChanged(mNodeInfo, nsGkAtoms::tfoot,
- getter_AddRefs(nodeInfo));
- foot = NS_NewHTMLTableSectionElement(nodeInfo.forget());
- if (!foot) {
- return nullptr;
- }
- AppendChildTo(foot, true);
- }
- return foot.forget();
- }
- void
- HTMLTableElement::DeleteTFoot()
- {
- RefPtr<HTMLTableSectionElement> tFoot = GetTFoot();
- if (tFoot) {
- mozilla::IgnoredErrorResult rv;
- nsINode::RemoveChild(*tFoot, rv);
- }
- }
- already_AddRefed<nsGenericHTMLElement>
- HTMLTableElement::CreateCaption()
- {
- RefPtr<nsGenericHTMLElement> caption = GetCaption();
- if (!caption) {
- // Create a new caption.
- RefPtr<mozilla::dom::NodeInfo> nodeInfo;
- nsContentUtils::NameChanged(mNodeInfo, nsGkAtoms::caption,
- getter_AddRefs(nodeInfo));
- caption = NS_NewHTMLTableCaptionElement(nodeInfo.forget());
- if (!caption) {
- return nullptr;
- }
- AppendChildTo(caption, true);
- }
- return caption.forget();
- }
- void
- HTMLTableElement::DeleteCaption()
- {
- RefPtr<HTMLTableCaptionElement> caption = GetCaption();
- if (caption) {
- mozilla::IgnoredErrorResult rv;
- nsINode::RemoveChild(*caption, rv);
- }
- }
- 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;
- }
- }
- ErrorResult 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::NameChanged(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::NameChanged(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::NameChanged(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,
- nsRuleData* 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->mPresContext;
- nsCompatibility mode = presContext->CompatibilityMode();
- if (aData->mSIDs & NS_STYLE_INHERIT_BIT(TableBorder)) {
- // cellspacing
- const nsAttrValue* value = aAttributes->GetAttr(nsGkAtoms::cellspacing);
- nsCSSValue* borderSpacing = aData->ValueForBorderSpacing();
- if (value && value->Type() == nsAttrValue::eInteger &&
- borderSpacing->GetUnit() == eCSSUnit_Null) {
- borderSpacing->
- SetFloatValue(float(value->GetIntegerValue()), eCSSUnit_Pixel);
- }
- }
- if (aData->mSIDs & 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) {
- nsCSSValue* marginLeft = aData->ValueForMarginLeft();
- if (marginLeft->GetUnit() == eCSSUnit_Null)
- marginLeft->SetAutoValue();
- nsCSSValue* marginRight = aData->ValueForMarginRight();
- if (marginRight->GetUnit() == eCSSUnit_Null)
- marginRight->SetAutoValue();
- }
- }
- // 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) {
- nsCSSValue* marginLeft = aData->ValueForMarginLeft();
- if (marginLeft->GetUnit() == eCSSUnit_Null)
- marginLeft->SetFloatValue((float)value->GetIntegerValue(), eCSSUnit_Pixel);
- nsCSSValue* marginRight = aData->ValueForMarginRight();
- if (marginRight->GetUnit() == eCSSUnit_Null)
- marginRight->SetFloatValue((float)value->GetIntegerValue(), eCSSUnit_Pixel);
- }
- value = aAttributes->GetAttr(nsGkAtoms::vspace);
- if (value && value->Type() == nsAttrValue::eInteger) {
- nsCSSValue* marginTop = aData->ValueForMarginTop();
- if (marginTop->GetUnit() == eCSSUnit_Null)
- marginTop->SetFloatValue((float)value->GetIntegerValue(), eCSSUnit_Pixel);
- nsCSSValue* marginBottom = aData->ValueForMarginBottom();
- if (marginBottom->GetUnit() == eCSSUnit_Null)
- marginBottom->SetFloatValue((float)value->GetIntegerValue(), eCSSUnit_Pixel);
- }
- }
- }
- if (aData->mSIDs & NS_STYLE_INHERIT_BIT(Position)) {
- // width: value
- nsCSSValue* width = aData->ValueForWidth();
- if (width->GetUnit() == eCSSUnit_Null) {
- const nsAttrValue* value = aAttributes->GetAttr(nsGkAtoms::width);
- if (value && value->Type() == nsAttrValue::eInteger)
- width->SetFloatValue((float)value->GetIntegerValue(), eCSSUnit_Pixel);
- else if (value && value->Type() == nsAttrValue::ePercent)
- width->SetPercentValue(value->GetPercentValue());
- }
- // height: value
- nsCSSValue* height = aData->ValueForHeight();
- if (height->GetUnit() == eCSSUnit_Null) {
- const nsAttrValue* value = aAttributes->GetAttr(nsGkAtoms::height);
- if (value && value->Type() == nsAttrValue::eInteger)
- height->SetFloatValue((float)value->GetIntegerValue(), eCSSUnit_Pixel);
- else if (value && value->Type() == nsAttrValue::ePercent)
- height->SetPercentValue(value->GetPercentValue());
- }
- }
- if (aData->mSIDs & NS_STYLE_INHERIT_BIT(Border)) {
- // bordercolor
- const nsAttrValue* value = aAttributes->GetAttr(nsGkAtoms::bordercolor);
- nscolor color;
- if (value && presContext->UseDocumentColors() &&
- value->GetColorValue(color)) {
- nsCSSValue* borderLeftColor = aData->ValueForBorderLeftColor();
- if (borderLeftColor->GetUnit() == eCSSUnit_Null)
- borderLeftColor->SetColorValue(color);
- nsCSSValue* borderRightColor = aData->ValueForBorderRightColor();
- if (borderRightColor->GetUnit() == eCSSUnit_Null)
- borderRightColor->SetColorValue(color);
- nsCSSValue* borderTopColor = aData->ValueForBorderTopColor();
- if (borderTopColor->GetUnit() == eCSSUnit_Null)
- borderTopColor->SetColorValue(color);
- nsCSSValue* borderBottomColor = aData->ValueForBorderBottomColor();
- if (borderBottomColor->GetUnit() == eCSSUnit_Null)
- borderBottomColor->SetColorValue(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
- nsCSSValue* borderLeftWidth = aData->ValueForBorderLeftWidth();
- if (borderLeftWidth->GetUnit() == eCSSUnit_Null)
- borderLeftWidth->SetFloatValue((float)borderThickness, eCSSUnit_Pixel);
- nsCSSValue* borderRightWidth = aData->ValueForBorderRightWidth();
- if (borderRightWidth->GetUnit() == eCSSUnit_Null)
- borderRightWidth->SetFloatValue((float)borderThickness, eCSSUnit_Pixel);
- nsCSSValue* borderTopWidth = aData->ValueForBorderTopWidth();
- if (borderTopWidth->GetUnit() == eCSSUnit_Null)
- borderTopWidth->SetFloatValue((float)borderThickness, eCSSUnit_Pixel);
- nsCSSValue* borderBottomWidth = aData->ValueForBorderBottomWidth();
- if (borderBottomWidth->GetUnit() == eCSSUnit_Null)
- borderBottomWidth->SetFloatValue((float)borderThickness, eCSSUnit_Pixel);
- }
- }
- 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,
- nsRuleData* aData)
- {
- if (aData->mSIDs & 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.
- nsCSSValue padVal(float(value->GetIntegerValue()), eCSSUnit_Pixel);
- nsCSSValue* paddingLeft = aData->ValueForPaddingLeft();
- if (paddingLeft->GetUnit() == eCSSUnit_Null) {
- *paddingLeft = padVal;
- }
- nsCSSValue* paddingRight = aData->ValueForPaddingRight();
- if (paddingRight->GetUnit() == eCSSUnit_Null) {
- *paddingRight = padVal;
- }
- nsCSSValue* paddingTop = aData->ValueForPaddingTop();
- if (paddingTop->GetUnit() == eCSSUnit_Null) {
- *paddingTop = padVal;
- }
- nsCSSValue* paddingBottom = aData->ValueForPaddingBottom();
- if (paddingBottom->GetUnit() == eCSSUnit_Null) {
- *paddingBottom = padVal;
- }
- }
- }
- }
- nsMappedAttributes*
- HTMLTableElement::GetAttributesMappedForCell()
- {
- return mTableInheritedAttributes;
- }
- void
- HTMLTableElement::BuildInheritedAttributes()
- {
- NS_ASSERTION(!mTableInheritedAttributes,
- "potential leak, plus waste of work");
- MOZ_ASSERT(NS_IsMainThread());
- 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);
- bool oldValueSet;
- modifiableMapped->SetAndSwapAttr(nsGkAtoms::cellpadding, val,
- &oldValueSet);
- }
- 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()
- {
- NS_IF_RELEASE(mTableInheritedAttributes);
- }
- nsresult
- HTMLTableElement::BindToTree(nsIDocument* aDocument, nsIContent* aParent,
- nsIContent* aBindingParent,
- bool aCompileEventHandlers)
- {
- ReleaseInheritedAttributes();
- nsresult rv = nsGenericHTMLElement::BindToTree(aDocument, aParent,
- aBindingParent,
- aCompileEventHandlers);
- NS_ENSURE_SUCCESS(rv, rv);
- BuildInheritedAttributes();
- return NS_OK;
- }
- 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,
- const nsAttrValue* aOldValue, bool aNotify)
- {
- if (aName == nsGkAtoms::cellpadding && aNameSpaceID == kNameSpaceID_None) {
- BuildInheritedAttributes();
- }
- return nsGenericHTMLElement::AfterSetAttr(aNameSpaceID, aName, aValue,
- aOldValue, aNotify);
- }
- } // namespace dom
- } // namespace mozilla
|