/* -*- 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 "nsHTMLDocument.h" #include "nsIContentPolicy.h" #include "mozilla/DebugOnly.h" #include "mozilla/PresShell.h" #include "mozilla/dom/HTMLAllCollection.h" #include "nsCommandManager.h" #include "nsCOMPtr.h" #include "nsGlobalWindow.h" #include "nsString.h" #include "nsPrintfCString.h" #include "nsReadableUtils.h" #include "nsUnicharUtils.h" #include "nsIContentSecurityPolicy.h" #include "nsGlobalWindowInner.h" #include "nsIDocumentLoader.h" #include "nsIHTMLContentSink.h" #include "nsIXMLContentSink.h" #include "nsHTMLParts.h" #include "nsHTMLStyleSheet.h" #include "nsGkAtoms.h" #include "nsPresContext.h" #include "nsPIDOMWindow.h" #include "nsDOMString.h" #include "nsIStreamListener.h" #include "nsIURI.h" #include "nsIURIMutator.h" #include "nsIIOService.h" #include "nsNetUtil.h" #include "nsIContentViewer.h" #include "nsDocShell.h" #include "nsDocShellLoadTypes.h" #include "nsIWebNavigation.h" #include "nsIBaseWindow.h" #include "nsIScriptContext.h" #include "nsIXPConnect.h" #include "nsContentList.h" #include "nsError.h" #include "nsIPrincipal.h" #include "nsJSPrincipals.h" #include "nsIScriptSecurityManager.h" #include "nsAttrName.h" #include "nsNodeUtils.h" #include "nsNetCID.h" #include "nsIServiceManager.h" #include "nsIConsoleService.h" #include "nsIComponentManager.h" #include "nsParserCIID.h" #include "mozilla/parser/PrototypeDocumentParser.h" #include "mozilla/dom/PrototypeDocumentContentSink.h" #include "nsNameSpaceManager.h" #include "nsGenericHTMLElement.h" #include "mozilla/css/Loader.h" #include "nsIHttpChannel.h" #include "nsIFile.h" #include "nsFrameSelection.h" #include "nsContentUtils.h" #include "nsJSUtils.h" #include "DocumentInlines.h" #include "nsIDocumentEncoder.h" //for outputting selection #include "nsICachingChannel.h" #include "nsIContentViewer.h" #include "nsIScriptElement.h" #include "nsIScriptError.h" #include "nsIMutableArray.h" #include "nsArrayUtils.h" #include "nsIEffectiveTLDService.h" // AHMED 12-2 #include "nsBidiUtils.h" #include "mozilla/dom/FallbackEncoding.h" #include "mozilla/Encoding.h" #include "mozilla/EventListenerManager.h" #include "mozilla/IdentifierMapEntry.h" #include "mozilla/LoadInfo.h" #include "nsNodeInfoManager.h" #include "nsRange.h" #include "mozAutoDocUpdate.h" #include "nsCCUncollectableMarker.h" #include "nsHtml5Module.h" #include "mozilla/dom/Element.h" #include "mozilla/Preferences.h" #include "nsMimeTypes.h" #include "nsIRequest.h" #include "nsHtml5TreeOpExecutor.h" #include "nsHtml5Parser.h" #include "nsSandboxFlags.h" #include "nsIImageDocument.h" #include "mozilla/dom/HTMLBodyElement.h" #include "mozilla/dom/HTMLDocumentBinding.h" #include "mozilla/dom/nsCSPContext.h" #include "mozilla/dom/Selection.h" #include "mozilla/dom/ShadowIncludingTreeIterator.h" #include "nsCharsetSource.h" #include "nsIStringBundle.h" #include "nsFocusManager.h" #include "nsIFrame.h" #include "nsIContent.h" #include "nsIStructuredCloneContainer.h" #include "nsLayoutStylesheetCache.h" #include "mozilla/StyleSheet.h" #include "mozilla/StyleSheetInlines.h" #include "mozilla/Unused.h" using namespace mozilla; using namespace mozilla::dom; #include "prtime.h" //#define DEBUG_charset static NS_DEFINE_CID(kCParserCID, NS_PARSER_CID); // ================================================================== // = // ================================================================== static bool IsAsciiCompatible(const Encoding* aEncoding) { return aEncoding->IsAsciiCompatible() || aEncoding == ISO_2022_JP_ENCODING; } nsresult NS_NewHTMLDocument(Document** aInstancePtrResult, bool aLoadedAsData) { RefPtr doc = new nsHTMLDocument(); nsresult rv = doc->Init(); if (NS_FAILED(rv)) { *aInstancePtrResult = nullptr; return rv; } doc->SetLoadedAsData(aLoadedAsData); doc.forget(aInstancePtrResult); return NS_OK; } nsHTMLDocument::nsHTMLDocument() : Document("text/html"), mContentListHolder(nullptr), mNumForms(0), mLoadFlags(0), mWarnedWidthHeight(false), mIsPlainText(false) { mType = eHTML; mDefaultElementType = kNameSpaceID_XHTML; mCompatMode = eCompatibility_NavQuirks; } nsHTMLDocument::~nsHTMLDocument() {} NS_IMPL_CYCLE_COLLECTION_INHERITED(nsHTMLDocument, Document, mAll) NS_IMPL_ISUPPORTS_CYCLE_COLLECTION_INHERITED_0(nsHTMLDocument, Document) JSObject* nsHTMLDocument::WrapNode(JSContext* aCx, JS::Handle aGivenProto) { return HTMLDocument_Binding::Wrap(aCx, this, aGivenProto); } nsresult nsHTMLDocument::Init() { nsresult rv = Document::Init(); NS_ENSURE_SUCCESS(rv, rv); // Now reset the compatibility mode of the CSSLoader // to match our compat mode. CSSLoader()->SetCompatibilityMode(mCompatMode); return NS_OK; } void nsHTMLDocument::Reset(nsIChannel* aChannel, nsILoadGroup* aLoadGroup) { Document::Reset(aChannel, aLoadGroup); if (aChannel) { aChannel->GetLoadFlags(&mLoadFlags); } } void nsHTMLDocument::ResetToURI(nsIURI* aURI, nsILoadGroup* aLoadGroup, nsIPrincipal* aPrincipal, nsIPrincipal* aStoragePrincipal) { mLoadFlags = nsIRequest::LOAD_NORMAL; Document::ResetToURI(aURI, aLoadGroup, aPrincipal, aStoragePrincipal); mImages = nullptr; mApplets = nullptr; mEmbeds = nullptr; mLinks = nullptr; mAnchors = nullptr; mScripts = nullptr; mForms = nullptr; // Make the content type default to "text/html", we are a HTML // document, after all. Once we start getting data, this may be // changed. SetContentTypeInternal(nsDependentCString("text/html")); } void nsHTMLDocument::TryHintCharset(nsIContentViewer* aCv, int32_t& aCharsetSource, NotNull& aEncoding) { if (aCv) { int32_t requestCharsetSource; nsresult rv = aCv->GetHintCharacterSetSource(&requestCharsetSource); if (NS_SUCCEEDED(rv) && kCharsetUninitialized != requestCharsetSource) { auto requestCharset = aCv->GetHintCharset(); aCv->SetHintCharacterSetSource((int32_t)(kCharsetUninitialized)); if (requestCharsetSource <= aCharsetSource) return; if (requestCharset && IsAsciiCompatible(requestCharset)) { aCharsetSource = requestCharsetSource; aEncoding = WrapNotNull(requestCharset); } } } } void nsHTMLDocument::TryUserForcedCharset(nsIContentViewer* aCv, nsIDocShell* aDocShell, int32_t& aCharsetSource, NotNull& aEncoding) { if (kCharsetFromUserForced <= aCharsetSource) return; // mCharacterSet not updated yet for channel, so check aEncoding, too. if (WillIgnoreCharsetOverride() || !IsAsciiCompatible(aEncoding)) { return; } const Encoding* forceCharsetFromDocShell = nullptr; if (aCv) { // XXX mailnews-only forceCharsetFromDocShell = aCv->GetForceCharset(); } if (forceCharsetFromDocShell && IsAsciiCompatible(forceCharsetFromDocShell)) { aEncoding = WrapNotNull(forceCharsetFromDocShell); aCharsetSource = kCharsetFromUserForced; return; } if (aDocShell) { // This is the Character Encoding menu code path in Firefox auto encoding = nsDocShell::Cast(aDocShell)->GetForcedCharset(); if (encoding) { if (!IsAsciiCompatible(encoding)) { return; } aEncoding = WrapNotNull(encoding); aCharsetSource = kCharsetFromUserForced; aDocShell->SetForcedCharset(NS_LITERAL_CSTRING("")); } } } void nsHTMLDocument::TryCacheCharset(nsICachingChannel* aCachingChannel, int32_t& aCharsetSource, NotNull& aEncoding) { nsresult rv; if (kCharsetFromCache <= aCharsetSource) { return; } nsCString cachedCharset; rv = aCachingChannel->GetCacheTokenCachedCharset(cachedCharset); if (NS_FAILED(rv) || cachedCharset.IsEmpty()) { return; } // The canonical names changed, so the cache may have an old name. const Encoding* encoding = Encoding::ForLabelNoReplacement(cachedCharset); if (!encoding) { return; } // Check IsAsciiCompatible() even in the cache case, because the value // might be stale and in the case of a stale charset that is not a rough // ASCII superset, the parser has no way to recover. if (!encoding->IsAsciiCompatible() && encoding != ISO_2022_JP_ENCODING) { return; } aEncoding = WrapNotNull(encoding); aCharsetSource = kCharsetFromCache; } void nsHTMLDocument::TryParentCharset(nsIDocShell* aDocShell, int32_t& aCharsetSource, NotNull& aEncoding) { if (!aDocShell) { return; } if (aCharsetSource >= kCharsetFromParentForced) { return; } int32_t parentSource; const Encoding* parentCharset; nsCOMPtr parentPrincipal; aDocShell->GetParentCharset(parentCharset, &parentSource, getter_AddRefs(parentPrincipal)); if (!parentCharset) { return; } if (kCharsetFromParentForced == parentSource || kCharsetFromUserForced == parentSource) { if (WillIgnoreCharsetOverride() || !IsAsciiCompatible(aEncoding) || // if channel said UTF-16 !IsAsciiCompatible(parentCharset)) { return; } aEncoding = WrapNotNull(parentCharset); aCharsetSource = kCharsetFromParentForced; return; } if (aCharsetSource >= kCharsetFromParentFrame) { return; } if (kCharsetFromCache <= parentSource) { // Make sure that's OK if (!NodePrincipal()->Equals(parentPrincipal) || !IsAsciiCompatible(parentCharset)) { return; } aEncoding = WrapNotNull(parentCharset); aCharsetSource = kCharsetFromParentFrame; } } void nsHTMLDocument::TryTLD(int32_t& aCharsetSource, NotNull& aEncoding) { if (aCharsetSource >= kCharsetFromTopLevelDomain) { return; } if (!FallbackEncoding::sGuessFallbackFromTopLevelDomain) { return; } if (!mDocumentURI) { return; } nsAutoCString host; mDocumentURI->GetAsciiHost(host); if (host.IsEmpty()) { return; } // First let's see if the host is DNS-absolute and ends with a dot and // get rid of that one. if (host.Last() == '.') { host.SetLength(host.Length() - 1); if (host.IsEmpty()) { return; } } // If we still have a dot, the host is weird, so let's continue only // if we have something other than a dot now. if (host.Last() == '.') { return; } int32_t index = host.RFindChar('.'); if (index == kNotFound) { // We have an intranet host, Gecko-internal URL or an IPv6 address. return; } // Since the string didn't end with a dot and we found a dot, // there is at least one character between the dot and the end of // the string, so taking the substring below is safe. nsAutoCString tld; ToLowerCase(Substring(host, index + 1, host.Length() - (index + 1)), tld); // Reject generic TLDs and country TLDs that need more research if (!FallbackEncoding::IsParticipatingTopLevelDomain(tld)) { return; } // Check if we have an IPv4 address bool seenNonDigit = false; for (size_t i = 0; i < tld.Length(); ++i) { char c = tld.CharAt(i); if (c < '0' || c > '9') { seenNonDigit = true; break; } } if (!seenNonDigit) { return; } aCharsetSource = kCharsetFromTopLevelDomain; aEncoding = FallbackEncoding::FromTopLevelDomain(tld); } void nsHTMLDocument::TryFallback(int32_t& aCharsetSource, NotNull& aEncoding) { if (kCharsetFromFallback <= aCharsetSource) return; aCharsetSource = kCharsetFromFallback; aEncoding = FallbackEncoding::FromLocale(); } // Using a prototype document is only allowed with chrome privilege. bool ShouldUsePrototypeDocument(nsIChannel* aChannel, Document* aDoc) { if (!aChannel || !aDoc || !StaticPrefs::dom_prototype_document_cache_enabled()) { return false; } if (!nsContentUtils::IsChromeDoc(aDoc)) { return false; } nsCOMPtr originalURI; aChannel->GetOriginalURI(getter_AddRefs(originalURI)); return IsChromeURI(originalURI); } nsresult nsHTMLDocument::StartDocumentLoad(const char* aCommand, nsIChannel* aChannel, nsILoadGroup* aLoadGroup, nsISupports* aContainer, nsIStreamListener** aDocListener, bool aReset, nsIContentSink* aSink) { if (!aCommand) { MOZ_ASSERT(false, "Command is mandatory"); return NS_ERROR_INVALID_POINTER; } if (aSink) { MOZ_ASSERT(false, "Got a sink override. Should not happen for HTML doc."); return NS_ERROR_INVALID_ARG; } if (mType != eHTML) { MOZ_ASSERT(mType == eXHTML); MOZ_ASSERT(false, "Must not set HTML doc to XHTML mode before load start."); return NS_ERROR_DOM_INVALID_STATE_ERR; } nsAutoCString contentType; aChannel->GetContentType(contentType); bool view = !strcmp(aCommand, "view") || !strcmp(aCommand, "external-resource"); bool viewSource = !strcmp(aCommand, "view-source"); bool asData = !strcmp(aCommand, kLoadAsData); if (!(view || viewSource || asData)) { MOZ_ASSERT(false, "Bad parser command"); return NS_ERROR_INVALID_ARG; } bool html = contentType.EqualsLiteral(TEXT_HTML); bool xhtml = !html && (contentType.EqualsLiteral(APPLICATION_XHTML_XML) || contentType.EqualsLiteral(APPLICATION_WAPXHTML_XML)); mIsPlainText = !html && !xhtml && nsContentUtils::IsPlainTextType(contentType); if (!(html || xhtml || mIsPlainText || viewSource)) { MOZ_ASSERT(false, "Channel with bad content type."); return NS_ERROR_INVALID_ARG; } bool forceUtf8 = mIsPlainText && nsContentUtils::IsUtf8OnlyPlainTextType(contentType); bool loadAsHtml5 = true; if (!viewSource && xhtml) { // We're parsing XHTML as XML, remember that. mType = eXHTML; SetCompatibilityMode(eCompatibility_FullStandards); loadAsHtml5 = false; } // TODO: Proper about:blank treatment is bug 543435 if (loadAsHtml5 && view) { // mDocumentURI hasn't been set, yet, so get the URI from the channel nsCOMPtr uri; aChannel->GetOriginalURI(getter_AddRefs(uri)); // Adapted from nsDocShell: // GetSpec can be expensive for some URIs, so check the scheme first. bool isAbout = false; if (uri && NS_SUCCEEDED(uri->SchemeIs("about", &isAbout)) && isAbout) { if (uri->GetSpecOrDefault().EqualsLiteral("about:blank")) { loadAsHtml5 = false; } } } nsresult rv = Document::StartDocumentLoad(aCommand, aChannel, aLoadGroup, aContainer, aDocListener, aReset); if (NS_FAILED(rv)) { return rv; } // Store the security info for future use. aChannel->GetSecurityInfo(getter_AddRefs(mSecurityInfo)); nsCOMPtr uri; rv = aChannel->GetURI(getter_AddRefs(uri)); if (NS_FAILED(rv)) { return rv; } nsCOMPtr cachingChan = do_QueryInterface(aChannel); nsCOMPtr docShell(do_QueryInterface(aContainer)); bool loadWithPrototype = false; RefPtr html5Parser; if (loadAsHtml5) { html5Parser = nsHtml5Module::NewHtml5Parser(); mParser = html5Parser; if (mIsPlainText) { if (viewSource) { html5Parser->MarkAsNotScriptCreated("view-source-plain"); } else { html5Parser->MarkAsNotScriptCreated("plain-text"); } } else if (viewSource && !html) { html5Parser->MarkAsNotScriptCreated("view-source-xml"); } else { html5Parser->MarkAsNotScriptCreated(aCommand); } } else if (xhtml && ShouldUsePrototypeDocument(aChannel, this)) { loadWithPrototype = true; nsCOMPtr originalURI; aChannel->GetOriginalURI(getter_AddRefs(originalURI)); mParser = new mozilla::parser::PrototypeDocumentParser(originalURI, this); } else { mParser = do_CreateInstance(kCParserCID, &rv); NS_ENSURE_SUCCESS(rv, rv); } // Look for the parent document. Note that at this point we don't have our // content viewer set up yet, and therefore do not have a useful // mParentDocument. // in this block of code, if we get an error result, we return it // but if we get a null pointer, that's perfectly legal for parent // and parentContentViewer nsCOMPtr parentAsItem; if (docShell) { docShell->GetSameTypeParent(getter_AddRefs(parentAsItem)); } nsCOMPtr parent(do_QueryInterface(parentAsItem)); nsCOMPtr parentContentViewer; if (parent) { rv = parent->GetContentViewer(getter_AddRefs(parentContentViewer)); NS_ENSURE_SUCCESS(rv, rv); } nsCOMPtr cv; if (docShell) { docShell->GetContentViewer(getter_AddRefs(cv)); } if (!cv) { cv = parentContentViewer.forget(); } nsAutoCString urlSpec; uri->GetSpec(urlSpec); #ifdef DEBUG_charset printf("Determining charset for %s\n", urlSpec.get()); #endif // These are the charset source and charset for our document int32_t charsetSource; auto encoding = UTF_8_ENCODING; // For error reporting and referrer policy setting nsHtml5TreeOpExecutor* executor = nullptr; if (loadAsHtml5) { executor = static_cast(mParser->GetContentSink()); } if (forceUtf8) { charsetSource = kCharsetFromUtf8OnlyMime; } else if (!IsHTMLDocument() || !docShell) { // no docshell for text/html XHR charsetSource = IsHTMLDocument() ? kCharsetFromFallback : kCharsetFromDocTypeDefault; TryChannelCharset(aChannel, charsetSource, encoding, executor); } else { NS_ASSERTION(docShell, "Unexpected null value"); charsetSource = kCharsetUninitialized; // The following will try to get the character encoding from various // sources. Each Try* function will return early if the source is already // at least as large as any of the sources it might look at. Some of // these functions (like TryHintCharset and TryParentCharset) can set // charsetSource to various values depending on where the charset they // end up finding originally comes from. // Try the channel's charset (e.g., charset from HTTP // "Content-Type" header) first. This way, we get to reject overrides in // TryParentCharset and TryUserForcedCharset if the channel said UTF-16. // This is to avoid socially engineered XSS by adding user-supplied // content to a UTF-16 site such that the byte have a dangerous // interpretation as ASCII and the user can be lured to using the // charset menu. TryChannelCharset(aChannel, charsetSource, encoding, executor); TryUserForcedCharset(cv, docShell, charsetSource, encoding); TryHintCharset(cv, charsetSource, encoding); // XXX mailnews-only TryParentCharset(docShell, charsetSource, encoding); if (cachingChan && !urlSpec.IsEmpty()) { TryCacheCharset(cachingChan, charsetSource, encoding); } TryTLD(charsetSource, encoding); TryFallback(charsetSource, encoding); } SetDocumentCharacterSetSource(charsetSource); SetDocumentCharacterSet(encoding); if (cachingChan) { nsAutoCString charset; encoding->Name(charset); rv = cachingChan->SetCacheTokenCachedCharset(charset); NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "cannot SetMetaDataElement"); rv = NS_OK; // don't propagate error } // Set the parser as the stream listener for the document loader... rv = NS_OK; nsCOMPtr listener = mParser->GetStreamListener(); listener.forget(aDocListener); #ifdef DEBUG_charset printf(" charset = %s source %d\n", charset.get(), charsetSource); #endif mParser->SetDocumentCharset(encoding, charsetSource); mParser->SetCommand(aCommand); if (!IsHTMLDocument()) { MOZ_ASSERT(!loadAsHtml5); if (loadWithPrototype) { nsCOMPtr sink; NS_NewPrototypeDocumentContentSink(getter_AddRefs(sink), this, uri, docShell, aChannel); mParser->SetContentSink(sink); } else { nsCOMPtr xmlsink; NS_NewXMLContentSink(getter_AddRefs(xmlsink), this, uri, docShell, aChannel); mParser->SetContentSink(xmlsink); } } else { if (loadAsHtml5) { html5Parser->Initialize(this, uri, docShell, aChannel); } else { // about:blank *only* nsCOMPtr htmlsink; NS_NewHTMLContentSink(getter_AddRefs(htmlsink), this, uri, docShell, aChannel); mParser->SetContentSink(htmlsink); } } // parser the content of the URI mParser->Parse(uri, nullptr, (void*)this); return rv; } bool nsHTMLDocument::UseWidthDeviceWidthFallbackViewport() const { if (mIsPlainText) { // Plain text documents are simple enough that font inflation doesn't offer // any appreciable advantage over defaulting to "width=device-width" and // subsequently turning on word-wrapping. return true; } return Document::UseWidthDeviceWidthFallbackViewport(); } Element* nsHTMLDocument::GetUnfocusedKeyEventTarget() { if (nsGenericHTMLElement* body = GetBody()) { return body; } return Document::GetUnfocusedKeyEventTarget(); } bool nsHTMLDocument::IsRegistrableDomainSuffixOfOrEqualTo( const nsAString& aHostSuffixString, const nsACString& aOrigHost) { // https://html.spec.whatwg.org/multipage/browsers.html#is-a-registrable-domain-suffix-of-or-is-equal-to if (aHostSuffixString.IsEmpty()) { return false; } nsCOMPtr origURI = CreateInheritingURIForHost(aOrigHost); if (!origURI) { // Error: failed to parse input domain return false; } nsCOMPtr newURI = RegistrableDomainSuffixOfInternal(aHostSuffixString, origURI); if (!newURI) { // Error: illegal domain return false; } return true; } void nsHTMLDocument::AddedForm() { ++mNumForms; } void nsHTMLDocument::RemovedForm() { --mNumForms; } int32_t nsHTMLDocument::GetNumFormsSynchronous() { return mNumForms; } void nsHTMLDocument::CaptureEvents() { WarnOnceAbout(Document::eUseOfCaptureEvents); } void nsHTMLDocument::ReleaseEvents() { WarnOnceAbout(Document::eUseOfReleaseEvents); } bool nsHTMLDocument::ResolveName(JSContext* aCx, const nsAString& aName, JS::MutableHandle aRetval, ErrorResult& aError) { IdentifierMapEntry* entry = mIdentifierMap.GetEntry(aName); if (!entry) { return false; } nsBaseContentList* list = entry->GetNameContentList(); uint32_t length = list ? list->Length() : 0; nsIContent* node; if (length > 0) { if (length > 1) { // The list contains more than one element, return the whole list. if (!ToJSValue(aCx, list, aRetval)) { aError.NoteJSContextException(aCx); return false; } return true; } // Only one element in the list, return the element instead of returning // the list. node = list->Item(0); } else { // No named items were found, see if there's one registerd by id for aName. Element* e = entry->GetIdElement(); if (!e || !nsGenericHTMLElement::ShouldExposeIdAsHTMLDocumentProperty(e)) { return false; } node = e; } if (!ToJSValue(aCx, node, aRetval)) { aError.NoteJSContextException(aCx); return false; } return true; } void nsHTMLDocument::GetSupportedNames(nsTArray& aNames) { for (auto iter = mIdentifierMap.Iter(); !iter.Done(); iter.Next()) { IdentifierMapEntry* entry = iter.Get(); if (entry->HasNameElement() || entry->HasIdElementExposedAsHTMLDocumentProperty()) { aNames.AppendElement(entry->GetKeyAsString()); } } } //---------------------------- // forms related stuff bool nsHTMLDocument::MatchFormControls(Element* aElement, int32_t aNamespaceID, nsAtom* aAtom, void* aData) { return aElement->IsNodeOfType(nsIContent::eHTML_FORM_CONTROL); } HTMLAllCollection* nsHTMLDocument::All() { if (!mAll) { mAll = new HTMLAllCollection(this); } return mAll; } nsresult nsHTMLDocument::Clone(dom::NodeInfo* aNodeInfo, nsINode** aResult) const { NS_ASSERTION(aNodeInfo->NodeInfoManager() == mNodeInfoManager, "Can't import this document into another document!"); RefPtr clone = new nsHTMLDocument(); nsresult rv = CloneDocHelper(clone.get()); NS_ENSURE_SUCCESS(rv, rv); // State from nsHTMLDocument clone->mLoadFlags = mLoadFlags; clone.forget(aResult); return NS_OK; } /* virtual */ void nsHTMLDocument::DocAddSizeOfExcludingThis( nsWindowSizes& aWindowSizes) const { Document::DocAddSizeOfExcludingThis(aWindowSizes); // Measurement of the following members may be added later if DMD finds it is // worthwhile: // - mLinks // - mAnchors } bool nsHTMLDocument::WillIgnoreCharsetOverride() { if (mEncodingMenuDisabled) { return true; } if (mType != eHTML) { MOZ_ASSERT(mType == eXHTML); return true; } if (mCharacterSetSource >= kCharsetFromByteOrderMark) { return true; } if (!mCharacterSet->IsAsciiCompatible() && mCharacterSet != ISO_2022_JP_ENCODING) { return true; } nsIURI* uri = GetOriginalURI(); if (uri) { bool schemeIs = false; uri->SchemeIs("about", &schemeIs); if (schemeIs) { return true; } bool isResource; nsresult rv = NS_URIChainHasFlags( uri, nsIProtocolHandler::URI_IS_UI_RESOURCE, &isResource); if (NS_FAILED(rv) || isResource) { return true; } } return false; } void nsHTMLDocument::GetFormsAndFormControls(nsContentList** aFormList, nsContentList** aFormControlList) { RefPtr holder = mContentListHolder; if (!holder) { // Flush our content model so it'll be up to date // If this becomes unnecessary and the following line is removed, // please also remove the corresponding flush operation from // nsHtml5TreeBuilderCppSupplement.h. (Look for "See bug 497861." there.) // XXXsmaug nsHtml5TreeBuilderCppSupplement doesn't seem to have such flush // anymore. FlushPendingNotifications(FlushType::Content); RefPtr htmlForms = GetExistingForms(); if (!htmlForms) { // If the document doesn't have an existing forms content list, create a // new one which will be released soon by ContentListHolder. The idea is // that we don't have that list hanging around for a long time and slowing // down future DOM mutations. // // Please keep this in sync with Document::Forms(). htmlForms = new nsContentList(this, kNameSpaceID_XHTML, nsGkAtoms::form, nsGkAtoms::form, /* aDeep = */ true, /* aLiveList = */ true); } RefPtr htmlFormControls = new nsContentList( this, nsHTMLDocument::MatchFormControls, nullptr, nullptr, /* aDeep = */ true, /* aMatchAtom = */ nullptr, /* aMatchNameSpaceId = */ kNameSpaceID_None, /* aFuncMayDependOnAttr = */ true, /* aLiveList = */ true); holder = new ContentListHolder(this, htmlForms, htmlFormControls); RefPtr runnable = holder; if (NS_SUCCEEDED( Dispatch(TaskCategory::GarbageCollection, runnable.forget()))) { mContentListHolder = holder; } } NS_ADDREF(*aFormList = holder->mFormList); NS_ADDREF(*aFormControlList = holder->mFormControlList); } void nsHTMLDocument::UserInteractionForTesting() { SetUserHasInteracted(); }