2015-05-03 19:32:37 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
2012-05-21 11:12:37 +00:00
|
|
|
/* 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/. */
|
2006-03-31 08:00:42 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* A class for managing namespace IDs and mapping back and forth
|
|
|
|
* between namespace IDs and namespace URIs.
|
|
|
|
*/
|
|
|
|
|
2014-02-27 23:04:46 +00:00
|
|
|
#include "nsNameSpaceManager.h"
|
|
|
|
|
2001-08-14 07:59:59 +00:00
|
|
|
#include "nscore.h"
|
2014-06-20 02:01:40 +00:00
|
|
|
#include "mozilla/dom/NodeInfo.h"
|
2018-03-20 00:44:28 +00:00
|
|
|
#include "nsAtom.h"
|
2002-11-29 23:44:07 +00:00
|
|
|
#include "nsCOMArray.h"
|
2004-06-25 12:26:02 +00:00
|
|
|
#include "nsContentCreatorFunctions.h"
|
2016-05-13 00:24:07 +00:00
|
|
|
#include "nsGkAtoms.h"
|
2016-06-28 14:24:48 +00:00
|
|
|
#include "nsIDocument.h"
|
1998-12-11 02:35:06 +00:00
|
|
|
#include "nsString.h"
|
2014-06-20 02:01:40 +00:00
|
|
|
#include "mozilla/dom/NodeInfo.h"
|
2014-02-27 23:04:46 +00:00
|
|
|
#include "mozilla/ClearOnShutdown.h"
|
2013-07-01 22:09:37 +00:00
|
|
|
#include "mozilla/dom/XBLChildrenElement.h"
|
2013-12-03 14:40:10 +00:00
|
|
|
#include "mozilla/dom/Element.h"
|
2016-06-28 14:24:48 +00:00
|
|
|
#include "mozilla/Preferences.h"
|
1998-12-11 02:35:06 +00:00
|
|
|
|
2012-03-12 22:53:18 +00:00
|
|
|
using namespace mozilla;
|
2010-10-25 12:17:38 +00:00
|
|
|
using namespace mozilla::dom;
|
|
|
|
|
2016-12-01 07:41:22 +00:00
|
|
|
static const char* kPrefSVGDisabled = "svg.disabled";
|
2016-06-28 14:24:48 +00:00
|
|
|
static const char* kPrefMathMLDisabled = "mathml.disabled";
|
2018-08-11 22:26:49 +00:00
|
|
|
static const char* kObservedNSPrefs[] = {kPrefMathMLDisabled, kPrefSVGDisabled,
|
2016-06-28 14:24:48 +00:00
|
|
|
nullptr};
|
|
|
|
StaticRefPtr<nsNameSpaceManager> nsNameSpaceManager::sInstance;
|
2014-02-27 23:04:46 +00:00
|
|
|
|
|
|
|
/* static */ nsNameSpaceManager* nsNameSpaceManager::GetInstance() {
|
|
|
|
if (!sInstance) {
|
|
|
|
sInstance = new nsNameSpaceManager();
|
|
|
|
if (sInstance->Init()) {
|
|
|
|
ClearOnShutdown(&sInstance);
|
|
|
|
} else {
|
|
|
|
delete sInstance;
|
|
|
|
sInstance = nullptr;
|
|
|
|
}
|
2005-09-11 11:24:16 +00:00
|
|
|
}
|
2004-06-25 12:26:02 +00:00
|
|
|
|
2014-02-27 23:04:46 +00:00
|
|
|
return sInstance;
|
|
|
|
}
|
1998-12-11 02:35:06 +00:00
|
|
|
|
2014-02-27 23:04:46 +00:00
|
|
|
bool nsNameSpaceManager::Init() {
|
2012-05-18 17:30:49 +00:00
|
|
|
nsresult rv;
|
2005-09-11 11:24:16 +00:00
|
|
|
#define REGISTER_NAMESPACE(uri, id) \
|
2016-05-13 00:24:07 +00:00
|
|
|
rv = AddNameSpace(dont_AddRef(uri), id); \
|
2014-02-27 23:04:46 +00:00
|
|
|
NS_ENSURE_SUCCESS(rv, false)
|
2005-09-11 11:24:16 +00:00
|
|
|
|
2016-06-28 14:24:48 +00:00
|
|
|
#define REGISTER_DISABLED_NAMESPACE(uri, id) \
|
|
|
|
rv = AddDisabledNameSpace(dont_AddRef(uri), id); \
|
|
|
|
NS_ENSURE_SUCCESS(rv, false)
|
|
|
|
|
2018-07-05 22:07:12 +00:00
|
|
|
mozilla::Preferences::RegisterCallbacks(
|
|
|
|
PREF_CHANGE_METHOD(nsNameSpaceManager::PrefChanged), kObservedNSPrefs,
|
2018-08-11 22:26:49 +00:00
|
|
|
this);
|
2016-06-28 14:24:48 +00:00
|
|
|
|
2018-07-05 22:07:12 +00:00
|
|
|
PrefChanged(nullptr);
|
2016-06-28 14:24:48 +00:00
|
|
|
|
2002-11-29 23:44:07 +00:00
|
|
|
// Need to be ordered according to ID.
|
2016-08-05 21:21:54 +00:00
|
|
|
MOZ_ASSERT(mURIArray.IsEmpty());
|
2017-07-14 05:31:15 +00:00
|
|
|
REGISTER_NAMESPACE(nsGkAtoms::_empty, kNameSpaceID_None);
|
2016-05-13 00:24:07 +00:00
|
|
|
REGISTER_NAMESPACE(nsGkAtoms::nsuri_xmlns, kNameSpaceID_XMLNS);
|
|
|
|
REGISTER_NAMESPACE(nsGkAtoms::nsuri_xml, kNameSpaceID_XML);
|
|
|
|
REGISTER_NAMESPACE(nsGkAtoms::nsuri_xhtml, kNameSpaceID_XHTML);
|
|
|
|
REGISTER_NAMESPACE(nsGkAtoms::nsuri_xlink, kNameSpaceID_XLink);
|
|
|
|
REGISTER_NAMESPACE(nsGkAtoms::nsuri_xslt, kNameSpaceID_XSLT);
|
|
|
|
REGISTER_NAMESPACE(nsGkAtoms::nsuri_xbl, kNameSpaceID_XBL);
|
|
|
|
REGISTER_NAMESPACE(nsGkAtoms::nsuri_mathml, kNameSpaceID_MathML);
|
|
|
|
REGISTER_NAMESPACE(nsGkAtoms::nsuri_rdf, kNameSpaceID_RDF);
|
|
|
|
REGISTER_NAMESPACE(nsGkAtoms::nsuri_xul, kNameSpaceID_XUL);
|
|
|
|
REGISTER_NAMESPACE(nsGkAtoms::nsuri_svg, kNameSpaceID_SVG);
|
2016-06-28 14:24:48 +00:00
|
|
|
REGISTER_DISABLED_NAMESPACE(nsGkAtoms::nsuri_mathml,
|
|
|
|
kNameSpaceID_disabled_MathML);
|
2016-12-01 07:41:22 +00:00
|
|
|
REGISTER_DISABLED_NAMESPACE(nsGkAtoms::nsuri_svg, kNameSpaceID_disabled_SVG);
|
2005-09-11 11:24:16 +00:00
|
|
|
|
|
|
|
#undef REGISTER_NAMESPACE
|
2016-06-28 14:24:48 +00:00
|
|
|
#undef REGISTER_DISABLED_NAMESPACE
|
2002-11-29 23:44:07 +00:00
|
|
|
|
2014-02-27 23:04:46 +00:00
|
|
|
return true;
|
2002-11-29 23:44:07 +00:00
|
|
|
}
|
|
|
|
|
2014-02-27 23:04:46 +00:00
|
|
|
nsresult nsNameSpaceManager::RegisterNameSpace(const nsAString& aURI,
|
|
|
|
int32_t& aNameSpaceID) {
|
2018-04-04 00:59:03 +00:00
|
|
|
RefPtr<nsAtom> atom = NS_Atomize(aURI);
|
|
|
|
return RegisterNameSpace(atom.forget(), aNameSpaceID);
|
|
|
|
}
|
2002-11-29 23:44:07 +00:00
|
|
|
|
2018-04-04 00:59:03 +00:00
|
|
|
nsresult nsNameSpaceManager::RegisterNameSpace(already_AddRefed<nsAtom> aURI,
|
|
|
|
int32_t& aNameSpaceID) {
|
|
|
|
RefPtr<nsAtom> atom = aURI;
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
if (atom == nsGkAtoms::_empty) {
|
|
|
|
aNameSpaceID = kNameSpaceID_None; // xmlns="", see bug 75700 for details
|
2002-11-29 23:44:07 +00:00
|
|
|
return NS_OK;
|
1998-12-11 02:35:06 +00:00
|
|
|
}
|
2002-11-29 23:44:07 +00:00
|
|
|
|
2016-05-13 00:24:07 +00:00
|
|
|
if (!mURIToIDTable.Get(atom, &aNameSpaceID)) {
|
2016-08-05 21:21:54 +00:00
|
|
|
aNameSpaceID = mURIArray.Length();
|
2002-11-29 23:44:07 +00:00
|
|
|
|
2016-05-13 00:24:07 +00:00
|
|
|
rv = AddNameSpace(atom.forget(), aNameSpaceID);
|
2005-09-11 11:24:16 +00:00
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
aNameSpaceID = kNameSpaceID_Unknown;
|
|
|
|
}
|
2002-11-29 23:44:07 +00:00
|
|
|
}
|
|
|
|
|
2017-11-05 05:48:48 +00:00
|
|
|
MOZ_ASSERT(aNameSpaceID >= -1, "Bogus namespace ID");
|
2016-09-01 07:57:16 +00:00
|
|
|
|
2002-11-29 23:44:07 +00:00
|
|
|
return rv;
|
1998-12-11 02:35:06 +00:00
|
|
|
}
|
|
|
|
|
2014-02-27 23:04:46 +00:00
|
|
|
nsresult nsNameSpaceManager::GetNameSpaceURI(int32_t aNameSpaceID,
|
|
|
|
nsAString& aURI) {
|
2018-04-28 19:50:58 +00:00
|
|
|
MOZ_ASSERT(aNameSpaceID >= 0, "Bogus namespace ID");
|
2016-08-05 21:21:54 +00:00
|
|
|
|
|
|
|
// We have historically treated GetNameSpaceURI calls for kNameSpaceID_None
|
|
|
|
// as erroneous.
|
|
|
|
if (aNameSpaceID <= 0 || aNameSpaceID >= int32_t(mURIArray.Length())) {
|
2002-11-29 23:44:07 +00:00
|
|
|
aURI.Truncate();
|
2001-09-06 06:18:31 +00:00
|
|
|
|
2002-11-29 23:44:07 +00:00
|
|
|
return NS_ERROR_ILLEGAL_VALUE;
|
1998-12-11 02:35:06 +00:00
|
|
|
}
|
2002-11-29 23:44:07 +00:00
|
|
|
|
2016-08-05 21:21:54 +00:00
|
|
|
mURIArray.ElementAt(aNameSpaceID)->ToString(aURI);
|
2002-11-29 23:44:07 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
1998-12-11 02:35:06 +00:00
|
|
|
}
|
|
|
|
|
2016-06-28 14:24:48 +00:00
|
|
|
int32_t nsNameSpaceManager::GetNameSpaceID(const nsAString& aURI,
|
|
|
|
bool aInChromeDoc) {
|
2002-11-29 23:44:07 +00:00
|
|
|
if (aURI.IsEmpty()) {
|
2005-09-11 11:24:16 +00:00
|
|
|
return kNameSpaceID_None; // xmlns="", see bug 75700 for details
|
2002-11-29 23:44:07 +00:00
|
|
|
}
|
|
|
|
|
2017-10-02 22:05:19 +00:00
|
|
|
RefPtr<nsAtom> atom = NS_Atomize(aURI);
|
2016-06-28 14:24:48 +00:00
|
|
|
return GetNameSpaceID(atom, aInChromeDoc);
|
2016-06-30 18:18:28 +00:00
|
|
|
}
|
|
|
|
|
2017-10-02 22:05:19 +00:00
|
|
|
int32_t nsNameSpaceManager::GetNameSpaceID(nsAtom* aURI, bool aInChromeDoc) {
|
2016-06-30 18:18:28 +00:00
|
|
|
if (aURI == nsGkAtoms::_empty) {
|
|
|
|
return kNameSpaceID_None; // xmlns="", see bug 75700 for details
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t nameSpaceID;
|
2016-12-01 07:41:22 +00:00
|
|
|
if (!aInChromeDoc && (mMathMLDisabled || mSVGDisabled) &&
|
|
|
|
mDisabledURIToIDTable.Get(aURI, &nameSpaceID) &&
|
|
|
|
((mMathMLDisabled && kNameSpaceID_disabled_MathML == nameSpaceID) ||
|
|
|
|
(mSVGDisabled && kNameSpaceID_disabled_SVG == nameSpaceID))) {
|
2017-11-05 05:48:48 +00:00
|
|
|
MOZ_ASSERT(nameSpaceID >= 0, "Bogus namespace ID");
|
2016-06-28 14:24:48 +00:00
|
|
|
return nameSpaceID;
|
|
|
|
}
|
2016-06-30 18:18:28 +00:00
|
|
|
if (mURIToIDTable.Get(aURI, &nameSpaceID)) {
|
2017-11-05 05:48:48 +00:00
|
|
|
MOZ_ASSERT(nameSpaceID >= 0, "Bogus namespace ID");
|
2006-07-20 03:25:39 +00:00
|
|
|
return nameSpaceID;
|
|
|
|
}
|
|
|
|
|
|
|
|
return kNameSpaceID_Unknown;
|
1998-12-11 02:35:06 +00:00
|
|
|
}
|
|
|
|
|
2013-12-03 14:40:11 +00:00
|
|
|
nsresult NS_NewElement(Element** aResult,
|
2014-06-20 02:01:40 +00:00
|
|
|
already_AddRefed<mozilla::dom::NodeInfo>&& aNodeInfo,
|
2016-06-06 10:43:39 +00:00
|
|
|
FromParser aFromParser, const nsAString* aIs) {
|
2015-10-18 05:24:48 +00:00
|
|
|
RefPtr<mozilla::dom::NodeInfo> ni = aNodeInfo;
|
2014-03-15 19:00:15 +00:00
|
|
|
int32_t ns = ni->NamespaceID();
|
2018-06-21 18:04:57 +00:00
|
|
|
RefPtr<nsAtom> isAtom = aIs ? NS_AtomizeMainThread(*aIs) : nullptr;
|
2011-11-16 07:50:19 +00:00
|
|
|
if (ns == kNameSpaceID_XHTML) {
|
2017-11-24 08:57:00 +00:00
|
|
|
return NS_NewHTMLElement(aResult, ni.forget(), aFromParser, isAtom);
|
2003-01-15 22:45:39 +00:00
|
|
|
}
|
2004-06-25 12:26:02 +00:00
|
|
|
#ifdef MOZ_XUL
|
2011-11-16 07:50:19 +00:00
|
|
|
if (ns == kNameSpaceID_XUL) {
|
2018-05-11 19:44:46 +00:00
|
|
|
return NS_NewXULElement(aResult, ni.forget(), aFromParser, isAtom);
|
2001-09-05 04:20:54 +00:00
|
|
|
}
|
2004-06-25 12:26:02 +00:00
|
|
|
#endif
|
2011-11-16 07:50:19 +00:00
|
|
|
if (ns == kNameSpaceID_MathML) {
|
2016-06-28 14:24:48 +00:00
|
|
|
// If the mathml.disabled pref. is true, convert all MathML nodes into
|
|
|
|
// disabled MathML nodes by swapping the namespace.
|
2017-07-04 08:00:03 +00:00
|
|
|
if (ni->NodeInfoManager()->MathMLEnabled()) {
|
2016-06-28 14:24:48 +00:00
|
|
|
return NS_NewMathMLElement(aResult, ni.forget());
|
|
|
|
}
|
|
|
|
|
|
|
|
RefPtr<mozilla::dom::NodeInfo> genericXMLNI =
|
|
|
|
ni->NodeInfoManager()->GetNodeInfo(ni->NameAtom(), ni->GetPrefixAtom(),
|
|
|
|
kNameSpaceID_disabled_MathML,
|
|
|
|
ni->NodeType(), ni->GetExtraName());
|
|
|
|
return NS_NewXMLElement(aResult, genericXMLNI.forget());
|
2001-09-05 04:20:54 +00:00
|
|
|
}
|
2011-11-16 07:50:19 +00:00
|
|
|
if (ns == kNameSpaceID_SVG) {
|
2016-12-01 07:41:22 +00:00
|
|
|
// If the svg.disabled pref. is true, convert all SVG nodes into
|
|
|
|
// disabled SVG nodes by swapping the namespace.
|
2017-07-04 08:00:03 +00:00
|
|
|
if (ni->NodeInfoManager()->SVGEnabled()) {
|
2016-12-01 07:41:22 +00:00
|
|
|
return NS_NewSVGElement(aResult, ni.forget(), aFromParser);
|
|
|
|
}
|
|
|
|
RefPtr<mozilla::dom::NodeInfo> genericXMLNI =
|
|
|
|
ni->NodeInfoManager()->GetNodeInfo(ni->NameAtom(), ni->GetPrefixAtom(),
|
|
|
|
kNameSpaceID_disabled_SVG,
|
|
|
|
ni->NodeType(), ni->GetExtraName());
|
|
|
|
return NS_NewXMLElement(aResult, genericXMLNI.forget());
|
2001-09-05 04:20:54 +00:00
|
|
|
}
|
2014-03-15 19:00:15 +00:00
|
|
|
if (ns == kNameSpaceID_XBL && ni->Equals(nsGkAtoms::children)) {
|
|
|
|
NS_ADDREF(*aResult = new XBLChildrenElement(ni.forget()));
|
2013-05-01 22:50:08 +00:00
|
|
|
return NS_OK;
|
|
|
|
}
|
2013-12-03 14:40:11 +00:00
|
|
|
|
2014-03-15 19:00:15 +00:00
|
|
|
return NS_NewXMLElement(aResult, ni.forget());
|
2001-09-05 04:20:54 +00:00
|
|
|
}
|
|
|
|
|
2014-02-27 23:04:46 +00:00
|
|
|
bool nsNameSpaceManager::HasElementCreator(int32_t aNameSpaceID) {
|
2004-06-25 12:26:02 +00:00
|
|
|
return aNameSpaceID == kNameSpaceID_XHTML ||
|
|
|
|
#ifdef MOZ_XUL
|
|
|
|
aNameSpaceID == kNameSpaceID_XUL ||
|
|
|
|
#endif
|
|
|
|
aNameSpaceID == kNameSpaceID_MathML ||
|
|
|
|
aNameSpaceID == kNameSpaceID_SVG || false;
|
2002-09-04 06:57:25 +00:00
|
|
|
}
|
|
|
|
|
2017-10-02 22:05:19 +00:00
|
|
|
nsresult nsNameSpaceManager::AddNameSpace(already_AddRefed<nsAtom> aURI,
|
2014-02-27 23:04:46 +00:00
|
|
|
const int32_t aNameSpaceID) {
|
2017-10-02 22:05:19 +00:00
|
|
|
RefPtr<nsAtom> uri = aURI;
|
2006-07-20 03:25:39 +00:00
|
|
|
if (aNameSpaceID < 0) {
|
|
|
|
// We've wrapped... Can't do anything else here; just bail.
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
}
|
2002-11-29 23:44:07 +00:00
|
|
|
|
2016-08-05 21:21:54 +00:00
|
|
|
MOZ_ASSERT(aNameSpaceID == (int32_t)mURIArray.Length());
|
2016-05-13 00:24:07 +00:00
|
|
|
mURIArray.AppendElement(uri.forget());
|
|
|
|
mURIToIDTable.Put(mURIArray.LastElement(), aNameSpaceID);
|
2002-11-29 23:44:07 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2016-06-28 14:24:48 +00:00
|
|
|
|
2017-10-02 22:05:19 +00:00
|
|
|
nsresult nsNameSpaceManager::AddDisabledNameSpace(already_AddRefed<nsAtom> aURI,
|
2016-06-28 14:24:48 +00:00
|
|
|
const int32_t aNameSpaceID) {
|
2017-10-02 22:05:19 +00:00
|
|
|
RefPtr<nsAtom> uri = aURI;
|
2016-06-28 14:24:48 +00:00
|
|
|
if (aNameSpaceID < 0) {
|
|
|
|
// We've wrapped... Can't do anything else here; just bail.
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
}
|
|
|
|
|
2016-08-05 21:21:54 +00:00
|
|
|
MOZ_ASSERT(aNameSpaceID == (int32_t)mURIArray.Length());
|
2016-06-28 14:24:48 +00:00
|
|
|
mURIArray.AppendElement(uri.forget());
|
|
|
|
mDisabledURIToIDTable.Put(mURIArray.LastElement(), aNameSpaceID);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2018-07-05 22:07:12 +00:00
|
|
|
void nsNameSpaceManager::PrefChanged(const char* aPref) {
|
2016-06-28 14:24:48 +00:00
|
|
|
mMathMLDisabled = mozilla::Preferences::GetBool(kPrefMathMLDisabled);
|
2016-12-01 07:41:22 +00:00
|
|
|
mSVGDisabled = mozilla::Preferences::GetBool(kPrefSVGDisabled);
|
2016-06-28 14:24:48 +00:00
|
|
|
}
|