зеркало из https://github.com/mozilla/gecko-dev.git
218 строки
5.9 KiB
C++
218 строки
5.9 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/. */
|
|
|
|
/*
|
|
* Class that represents a prefix/namespace/localName triple; a single
|
|
* nodeinfo is shared by all elements in a document that have that
|
|
* prefix, namespace, and localName.
|
|
*/
|
|
|
|
#include "mozilla/dom/NodeInfo.h"
|
|
#include "mozilla/dom/NodeInfoInlines.h"
|
|
|
|
#include "mozilla/ArrayUtils.h"
|
|
#include "mozilla/Likely.h"
|
|
|
|
#include "nsNodeInfoManager.h"
|
|
#include "nsCOMPtr.h"
|
|
#include "nsString.h"
|
|
#include "nsIAtom.h"
|
|
#include "nsDOMString.h"
|
|
#include "nsCRT.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsReadableUtils.h"
|
|
#include "mozilla/Snprintf.h"
|
|
#include "nsIDocument.h"
|
|
#include "nsGkAtoms.h"
|
|
#include "nsCCUncollectableMarker.h"
|
|
#include "nsNameSpaceManager.h"
|
|
|
|
using namespace mozilla;
|
|
using mozilla::dom::NodeInfo;
|
|
|
|
NodeInfo::~NodeInfo()
|
|
{
|
|
mOwnerManager->RemoveNodeInfo(this);
|
|
}
|
|
|
|
NodeInfo::NodeInfo(nsIAtom *aName, nsIAtom *aPrefix, int32_t aNamespaceID,
|
|
uint16_t aNodeType, nsIAtom* aExtraName,
|
|
nsNodeInfoManager *aOwnerManager)
|
|
{
|
|
CheckValidNodeInfo(aNodeType, aName, aNamespaceID, aExtraName);
|
|
MOZ_ASSERT(aOwnerManager, "Invalid aOwnerManager");
|
|
|
|
// Initialize mInner
|
|
mInner.mName = aName;
|
|
mInner.mPrefix = aPrefix;
|
|
mInner.mNamespaceID = aNamespaceID;
|
|
mInner.mNodeType = aNodeType;
|
|
mOwnerManager = aOwnerManager;
|
|
mInner.mExtraName = aExtraName;
|
|
|
|
mDocument = aOwnerManager->GetDocument();
|
|
|
|
// Now compute our cached members.
|
|
|
|
// Qualified name. If we have no prefix, use ToString on
|
|
// mInner.mName so that we get to share its buffer.
|
|
if (aPrefix) {
|
|
mQualifiedName = nsDependentAtomString(mInner.mPrefix) +
|
|
NS_LITERAL_STRING(":") +
|
|
nsDependentAtomString(mInner.mName);
|
|
} else {
|
|
mInner.mName->ToString(mQualifiedName);
|
|
}
|
|
|
|
MOZ_ASSERT_IF(aNodeType != nsIDOMNode::ELEMENT_NODE &&
|
|
aNodeType != nsIDOMNode::ATTRIBUTE_NODE &&
|
|
aNodeType != UINT16_MAX,
|
|
aNamespaceID == kNameSpaceID_None && !aPrefix);
|
|
|
|
switch (aNodeType) {
|
|
case nsIDOMNode::ELEMENT_NODE:
|
|
case nsIDOMNode::ATTRIBUTE_NODE:
|
|
// Correct the case for HTML
|
|
if (aNodeType == nsIDOMNode::ELEMENT_NODE &&
|
|
aNamespaceID == kNameSpaceID_XHTML && GetDocument() &&
|
|
GetDocument()->IsHTMLDocument()) {
|
|
nsContentUtils::ASCIIToUpper(mQualifiedName, mNodeName);
|
|
} else {
|
|
mNodeName = mQualifiedName;
|
|
}
|
|
mInner.mName->ToString(mLocalName);
|
|
break;
|
|
case nsIDOMNode::TEXT_NODE:
|
|
case nsIDOMNode::CDATA_SECTION_NODE:
|
|
case nsIDOMNode::COMMENT_NODE:
|
|
case nsIDOMNode::DOCUMENT_NODE:
|
|
case nsIDOMNode::DOCUMENT_FRAGMENT_NODE:
|
|
mInner.mName->ToString(mNodeName);
|
|
SetDOMStringToNull(mLocalName);
|
|
break;
|
|
case nsIDOMNode::PROCESSING_INSTRUCTION_NODE:
|
|
case nsIDOMNode::DOCUMENT_TYPE_NODE:
|
|
mInner.mExtraName->ToString(mNodeName);
|
|
SetDOMStringToNull(mLocalName);
|
|
break;
|
|
default:
|
|
MOZ_ASSERT(aNodeType == UINT16_MAX, "Unknown node type");
|
|
}
|
|
}
|
|
|
|
|
|
// nsISupports
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(NodeInfo)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_0(NodeInfo)
|
|
|
|
static const char* kNodeInfoNSURIs[] = {
|
|
" ([none])",
|
|
" (xmlns)",
|
|
" (xml)",
|
|
" (xhtml)",
|
|
" (XLink)",
|
|
" (XSLT)",
|
|
" (XBL)",
|
|
" (MathML)",
|
|
" (RDF)",
|
|
" (XUL)"
|
|
};
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INTERNAL(NodeInfo)
|
|
if (MOZ_UNLIKELY(cb.WantDebugInfo())) {
|
|
char name[72];
|
|
uint32_t nsid = tmp->NamespaceID();
|
|
nsAtomCString localName(tmp->NameAtom());
|
|
if (nsid < ArrayLength(kNodeInfoNSURIs)) {
|
|
snprintf_literal(name, "NodeInfo%s %s", kNodeInfoNSURIs[nsid],
|
|
localName.get());
|
|
}
|
|
else {
|
|
snprintf_literal(name, "NodeInfo %s", localName.get());
|
|
}
|
|
|
|
cb.DescribeRefCountedNode(tmp->mRefCnt.get(), name);
|
|
}
|
|
else {
|
|
NS_IMPL_CYCLE_COLLECTION_DESCRIBE(NodeInfo, tmp->mRefCnt.get())
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mOwnerManager)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_BEGIN(NodeInfo)
|
|
return nsCCUncollectableMarker::sGeneration && tmp->CanSkip();
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_IN_CC_BEGIN(NodeInfo)
|
|
return nsCCUncollectableMarker::sGeneration && tmp->CanSkip();
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_IN_CC_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_THIS_BEGIN(NodeInfo)
|
|
return nsCCUncollectableMarker::sGeneration && tmp->CanSkip();
|
|
NS_IMPL_CYCLE_COLLECTION_CAN_SKIP_THIS_END
|
|
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_ROOT_NATIVE(NodeInfo, AddRef)
|
|
NS_IMPL_CYCLE_COLLECTION_UNROOT_NATIVE(NodeInfo, Release)
|
|
|
|
void
|
|
NodeInfo::GetName(nsAString& aName) const
|
|
{
|
|
mInner.mName->ToString(aName);
|
|
}
|
|
|
|
void
|
|
NodeInfo::GetPrefix(nsAString& aPrefix) const
|
|
{
|
|
if (mInner.mPrefix) {
|
|
mInner.mPrefix->ToString(aPrefix);
|
|
} else {
|
|
SetDOMStringToNull(aPrefix);
|
|
}
|
|
}
|
|
|
|
void
|
|
NodeInfo::GetNamespaceURI(nsAString& aNameSpaceURI) const
|
|
{
|
|
if (mInner.mNamespaceID > 0) {
|
|
nsresult rv =
|
|
nsContentUtils::NameSpaceManager()->GetNameSpaceURI(mInner.mNamespaceID,
|
|
aNameSpaceURI);
|
|
// How can we possibly end up with a bogus namespace ID here?
|
|
if (NS_FAILED(rv)) {
|
|
MOZ_CRASH();
|
|
}
|
|
} else {
|
|
SetDOMStringToNull(aNameSpaceURI);
|
|
}
|
|
}
|
|
|
|
bool
|
|
NodeInfo::NamespaceEquals(const nsAString& aNamespaceURI) const
|
|
{
|
|
int32_t nsid =
|
|
nsContentUtils::NameSpaceManager()->GetNameSpaceID(aNamespaceURI);
|
|
|
|
return mozilla::dom::NodeInfo::NamespaceEquals(nsid);
|
|
}
|
|
|
|
void
|
|
NodeInfo::DeleteCycleCollectable()
|
|
{
|
|
RefPtr<nsNodeInfoManager> kungFuDeathGrip = mOwnerManager;
|
|
delete this;
|
|
}
|
|
|
|
bool
|
|
NodeInfo::CanSkip()
|
|
{
|
|
return mDocument &&
|
|
nsCCUncollectableMarker::InGeneration(mDocument->GetMarkedCCGeneration());
|
|
}
|