mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-01 14:45:29 +00:00
3742 lines
109 KiB
C++
3742 lines
109 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 "nsHTMLDocument.h"
|
|
|
|
#include "nsIContentPolicy.h"
|
|
#include "mozilla/DebugOnly.h"
|
|
#include "mozilla/dom/HTMLAllCollection.h"
|
|
#include "nsCOMPtr.h"
|
|
#include "nsGlobalWindow.h"
|
|
#include "nsXPIDLString.h"
|
|
#include "nsPrintfCString.h"
|
|
#include "nsReadableUtils.h"
|
|
#include "nsUnicharUtils.h"
|
|
#include "nsIHTMLContentSink.h"
|
|
#include "nsIXMLContentSink.h"
|
|
#include "nsHTMLParts.h"
|
|
#include "nsHTMLStyleSheet.h"
|
|
#include "nsGkAtoms.h"
|
|
#include "nsIPresShell.h"
|
|
#include "nsPresContext.h"
|
|
#include "nsIDOMNode.h" // for Find
|
|
#include "nsIDOMNodeList.h"
|
|
#include "nsIDOMElement.h"
|
|
#include "nsPIDOMWindow.h"
|
|
#include "nsDOMString.h"
|
|
#include "nsIStreamListener.h"
|
|
#include "nsIURI.h"
|
|
#include "nsIIOService.h"
|
|
#include "nsNetUtil.h"
|
|
#include "nsIContentViewerContainer.h"
|
|
#include "nsIContentViewer.h"
|
|
#include "nsDocShell.h"
|
|
#include "nsDocShellLoadTypes.h"
|
|
#include "nsIWebNavigation.h"
|
|
#include "nsIBaseWindow.h"
|
|
#include "nsIWebShellServices.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 "nsICookieService.h"
|
|
|
|
#include "nsIServiceManager.h"
|
|
#include "nsIConsoleService.h"
|
|
#include "nsIComponentManager.h"
|
|
#include "nsParserCIID.h"
|
|
#include "nsIDOMHTMLElement.h"
|
|
#include "nsIDOMHTMLHeadElement.h"
|
|
#include "nsNameSpaceManager.h"
|
|
#include "nsGenericHTMLElement.h"
|
|
#include "mozilla/css/Loader.h"
|
|
#include "nsIHttpChannel.h"
|
|
#include "nsIFile.h"
|
|
#include "nsFrameSelection.h"
|
|
#include "nsISelectionPrivate.h"//for toStringwithformat code
|
|
|
|
#include "nsContentUtils.h"
|
|
#include "nsJSUtils.h"
|
|
#include "nsIDocumentInlines.h"
|
|
#include "nsIDocumentEncoder.h" //for outputting selection
|
|
#include "nsICachingChannel.h"
|
|
#include "nsIContentViewer.h"
|
|
#include "nsIWyciwygChannel.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/EncodingUtils.h"
|
|
#include "mozilla/dom/FallbackEncoding.h"
|
|
#include "mozilla/LoadInfo.h"
|
|
#include "nsIEditingSession.h"
|
|
#include "nsIEditor.h"
|
|
#include "nsNodeInfoManager.h"
|
|
#include "nsIPlaintextEditor.h"
|
|
#include "nsIHTMLEditor.h"
|
|
#include "nsIEditorStyleSheets.h"
|
|
#include "nsIInlineSpellChecker.h"
|
|
#include "nsRange.h"
|
|
#include "mozAutoDocUpdate.h"
|
|
#include "nsCCUncollectableMarker.h"
|
|
#include "nsHtml5Module.h"
|
|
#include "prprf.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "mozilla/Preferences.h"
|
|
#include "nsMimeTypes.h"
|
|
#include "nsIRequest.h"
|
|
#include "nsHtml5TreeOpExecutor.h"
|
|
#include "nsHtml5Parser.h"
|
|
#include "nsIDOMJSWindow.h"
|
|
#include "nsSandboxFlags.h"
|
|
#include "nsIImageDocument.h"
|
|
#include "mozilla/dom/HTMLBodyElement.h"
|
|
#include "mozilla/dom/HTMLDocumentBinding.h"
|
|
#include "nsCharsetSource.h"
|
|
#include "nsIStringBundle.h"
|
|
#include "nsDOMClassInfo.h"
|
|
#include "nsFocusManager.h"
|
|
#include "nsIFrame.h"
|
|
#include "nsIContent.h"
|
|
#include "nsLayoutStylesheetCache.h"
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::dom;
|
|
|
|
#define NS_MAX_DOCUMENT_WRITE_DEPTH 20
|
|
|
|
#include "prtime.h"
|
|
|
|
//#define DEBUG_charset
|
|
|
|
static NS_DEFINE_CID(kCParserCID, NS_PARSER_CID);
|
|
|
|
uint32_t nsHTMLDocument::gWyciwygSessionCnt = 0;
|
|
|
|
// this function will return false if the command is not recognized
|
|
// inCommandID will be converted as necessary for internal operations
|
|
// inParam will be converted as necessary for internal operations
|
|
// outParam will be Empty if no parameter is needed or if returning a boolean
|
|
// outIsBoolean will determine whether to send param as a boolean or string
|
|
// outBooleanParam will not be set unless outIsBoolean
|
|
static bool ConvertToMidasInternalCommand(const nsAString & inCommandID,
|
|
const nsAString & inParam,
|
|
nsACString& outCommandID,
|
|
nsACString& outParam,
|
|
bool& isBoolean,
|
|
bool& boolValue);
|
|
|
|
static bool ConvertToMidasInternalCommand(const nsAString & inCommandID,
|
|
nsACString& outCommandID);
|
|
|
|
// ==================================================================
|
|
// =
|
|
// ==================================================================
|
|
|
|
nsresult
|
|
NS_NewHTMLDocument(nsIDocument** aInstancePtrResult, bool aLoadedAsData)
|
|
{
|
|
nsRefPtr<nsHTMLDocument> doc = new nsHTMLDocument();
|
|
|
|
nsresult rv = doc->Init();
|
|
|
|
if (NS_FAILED(rv)) {
|
|
*aInstancePtrResult = nullptr;
|
|
return rv;
|
|
}
|
|
|
|
doc->SetLoadedAsData(aLoadedAsData);
|
|
doc.forget(aInstancePtrResult);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
// NOTE! nsDocument::operator new() zeroes out all members, so don't
|
|
// bother initializing members to 0.
|
|
|
|
nsHTMLDocument::nsHTMLDocument()
|
|
: nsDocument("text/html")
|
|
{
|
|
// NOTE! nsDocument::operator new() zeroes out all members, so don't
|
|
// bother initializing members to 0.
|
|
|
|
mType = eHTML;
|
|
mDefaultElementType = kNameSpaceID_XHTML;
|
|
mCompatMode = eCompatibility_NavQuirks;
|
|
}
|
|
|
|
nsHTMLDocument::~nsHTMLDocument()
|
|
{
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_INHERITED(nsHTMLDocument, nsDocument,
|
|
mAll,
|
|
mImages,
|
|
mApplets,
|
|
mEmbeds,
|
|
mLinks,
|
|
mAnchors,
|
|
mScripts,
|
|
mForms,
|
|
mFormControls,
|
|
mWyciwygChannel,
|
|
mMidasCommandManager)
|
|
|
|
NS_IMPL_ADDREF_INHERITED(nsHTMLDocument, nsDocument)
|
|
NS_IMPL_RELEASE_INHERITED(nsHTMLDocument, nsDocument)
|
|
|
|
// QueryInterface implementation for nsHTMLDocument
|
|
NS_INTERFACE_TABLE_HEAD_CYCLE_COLLECTION_INHERITED(nsHTMLDocument)
|
|
NS_INTERFACE_TABLE_INHERITED(nsHTMLDocument, nsIHTMLDocument,
|
|
nsIDOMHTMLDocument)
|
|
NS_INTERFACE_TABLE_TAIL_INHERITING(nsDocument)
|
|
|
|
JSObject*
|
|
nsHTMLDocument::WrapNode(JSContext* aCx, JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
return HTMLDocumentBinding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLDocument::Init()
|
|
{
|
|
nsresult rv = nsDocument::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)
|
|
{
|
|
nsDocument::Reset(aChannel, aLoadGroup);
|
|
|
|
if (aChannel) {
|
|
aChannel->GetLoadFlags(&mLoadFlags);
|
|
}
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::ResetToURI(nsIURI *aURI, nsILoadGroup *aLoadGroup,
|
|
nsIPrincipal* aPrincipal)
|
|
{
|
|
mLoadFlags = nsIRequest::LOAD_NORMAL;
|
|
|
|
nsDocument::ResetToURI(aURI, aLoadGroup, aPrincipal);
|
|
|
|
mImages = nullptr;
|
|
mApplets = nullptr;
|
|
mEmbeds = nullptr;
|
|
mLinks = nullptr;
|
|
mAnchors = nullptr;
|
|
mScripts = nullptr;
|
|
|
|
mForms = nullptr;
|
|
|
|
NS_ASSERTION(!mWyciwygChannel,
|
|
"nsHTMLDocument::Reset() - Wyciwyg Channel still exists!");
|
|
|
|
mWyciwygChannel = 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"));
|
|
}
|
|
|
|
already_AddRefed<nsIPresShell>
|
|
nsHTMLDocument::CreateShell(nsPresContext* aContext,
|
|
nsViewManager* aViewManager,
|
|
nsStyleSet* aStyleSet)
|
|
{
|
|
return doCreateShell(aContext, aViewManager, aStyleSet, mCompatMode);
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::TryHintCharset(nsIContentViewer* aCv,
|
|
int32_t& aCharsetSource, nsACString& aCharset)
|
|
{
|
|
if (aCv) {
|
|
int32_t requestCharsetSource;
|
|
nsresult rv = aCv->GetHintCharacterSetSource(&requestCharsetSource);
|
|
|
|
if(NS_SUCCEEDED(rv) && kCharsetUninitialized != requestCharsetSource) {
|
|
nsAutoCString requestCharset;
|
|
rv = aCv->GetHintCharacterSet(requestCharset);
|
|
aCv->SetHintCharacterSetSource((int32_t)(kCharsetUninitialized));
|
|
|
|
if(requestCharsetSource <= aCharsetSource)
|
|
return;
|
|
|
|
if(NS_SUCCEEDED(rv) && EncodingUtils::IsAsciiCompatible(requestCharset)) {
|
|
aCharsetSource = requestCharsetSource;
|
|
aCharset = requestCharset;
|
|
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
|
|
void
|
|
nsHTMLDocument::TryUserForcedCharset(nsIContentViewer* aCv,
|
|
nsIDocShell* aDocShell,
|
|
int32_t& aCharsetSource,
|
|
nsACString& aCharset)
|
|
{
|
|
nsresult rv = NS_OK;
|
|
|
|
if(kCharsetFromUserForced <= aCharsetSource)
|
|
return;
|
|
|
|
// mCharacterSet not updated yet for channel, so check aCharset, too.
|
|
if (WillIgnoreCharsetOverride() || !EncodingUtils::IsAsciiCompatible(aCharset)) {
|
|
return;
|
|
}
|
|
|
|
nsAutoCString forceCharsetFromDocShell;
|
|
if (aCv) {
|
|
// XXX mailnews-only
|
|
rv = aCv->GetForceCharacterSet(forceCharsetFromDocShell);
|
|
}
|
|
|
|
if(NS_SUCCEEDED(rv) &&
|
|
!forceCharsetFromDocShell.IsEmpty() &&
|
|
EncodingUtils::IsAsciiCompatible(forceCharsetFromDocShell)) {
|
|
aCharset = forceCharsetFromDocShell;
|
|
aCharsetSource = kCharsetFromUserForced;
|
|
return;
|
|
}
|
|
|
|
if (aDocShell) {
|
|
// This is the Character Encoding menu code path in Firefox
|
|
nsAutoCString charset;
|
|
rv = aDocShell->GetForcedCharset(charset);
|
|
|
|
if (NS_SUCCEEDED(rv) && !charset.IsEmpty()) {
|
|
if (!EncodingUtils::IsAsciiCompatible(charset)) {
|
|
return;
|
|
}
|
|
aCharset = charset;
|
|
aCharsetSource = kCharsetFromUserForced;
|
|
aDocShell->SetForcedCharset(NS_LITERAL_CSTRING(""));
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::TryCacheCharset(nsICachingChannel* aCachingChannel,
|
|
int32_t& aCharsetSource,
|
|
nsACString& aCharset)
|
|
{
|
|
nsresult rv;
|
|
|
|
if (kCharsetFromCache <= aCharsetSource) {
|
|
return;
|
|
}
|
|
|
|
nsCString cachedCharset;
|
|
rv = aCachingChannel->GetCacheTokenCachedCharset(cachedCharset);
|
|
// Check EncodingUtils::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 (NS_SUCCEEDED(rv) &&
|
|
!cachedCharset.IsEmpty() &&
|
|
EncodingUtils::IsAsciiCompatible(cachedCharset))
|
|
{
|
|
aCharset = cachedCharset;
|
|
aCharsetSource = kCharsetFromCache;
|
|
}
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::TryParentCharset(nsIDocShell* aDocShell,
|
|
int32_t& aCharsetSource,
|
|
nsACString& aCharset)
|
|
{
|
|
if (!aDocShell) {
|
|
return;
|
|
}
|
|
if (aCharsetSource >= kCharsetFromParentForced) {
|
|
return;
|
|
}
|
|
|
|
int32_t parentSource;
|
|
nsAutoCString parentCharset;
|
|
nsCOMPtr<nsIPrincipal> parentPrincipal;
|
|
aDocShell->GetParentCharset(parentCharset,
|
|
&parentSource,
|
|
getter_AddRefs(parentPrincipal));
|
|
if (parentCharset.IsEmpty()) {
|
|
return;
|
|
}
|
|
if (kCharsetFromParentForced == parentSource ||
|
|
kCharsetFromUserForced == parentSource) {
|
|
if (WillIgnoreCharsetOverride() ||
|
|
!EncodingUtils::IsAsciiCompatible(aCharset) || // if channel said UTF-16
|
|
!EncodingUtils::IsAsciiCompatible(parentCharset)) {
|
|
return;
|
|
}
|
|
aCharset.Assign(parentCharset);
|
|
aCharsetSource = kCharsetFromParentForced;
|
|
return;
|
|
}
|
|
|
|
if (aCharsetSource >= kCharsetFromParentFrame) {
|
|
return;
|
|
}
|
|
|
|
if (kCharsetFromCache <= parentSource) {
|
|
// Make sure that's OK
|
|
if (!NodePrincipal()->Equals(parentPrincipal) ||
|
|
!EncodingUtils::IsAsciiCompatible(parentCharset)) {
|
|
return;
|
|
}
|
|
|
|
aCharset.Assign(parentCharset);
|
|
aCharsetSource = kCharsetFromParentFrame;
|
|
}
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::TryTLD(int32_t& aCharsetSource, nsACString& aCharset)
|
|
{
|
|
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;
|
|
FallbackEncoding::FromTopLevelDomain(tld, aCharset);
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::TryFallback(int32_t& aCharsetSource, nsACString& aCharset)
|
|
{
|
|
if (kCharsetFromFallback <= aCharsetSource)
|
|
return;
|
|
|
|
aCharsetSource = kCharsetFromFallback;
|
|
FallbackEncoding::FromLocale(aCharset);
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::SetDocumentCharacterSet(const nsACString& aCharSetID)
|
|
{
|
|
nsDocument::SetDocumentCharacterSet(aCharSetID);
|
|
// Make sure to stash this charset on our channel as needed if it's a wyciwyg
|
|
// channel.
|
|
nsCOMPtr<nsIWyciwygChannel> wyciwygChannel = do_QueryInterface(mChannel);
|
|
if (wyciwygChannel) {
|
|
wyciwygChannel->SetCharsetAndSource(GetDocumentCharacterSetSource(),
|
|
aCharSetID);
|
|
}
|
|
}
|
|
|
|
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);
|
|
bool import = !strcmp(aCommand, "import");
|
|
if (!(view || viewSource || asData || import)) {
|
|
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);
|
|
bool plainText = !html && !xhtml && nsContentUtils::IsPlainTextType(contentType);
|
|
if (!(html || xhtml || plainText || viewSource)) {
|
|
MOZ_ASSERT(false, "Channel with bad content type.");
|
|
return NS_ERROR_INVALID_ARG;
|
|
}
|
|
|
|
bool loadAsHtml5 = true;
|
|
|
|
if (!viewSource && xhtml) {
|
|
// We're parsing XHTML as XML, remember that.
|
|
mType = eXHTML;
|
|
mCompatMode = 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<nsIURI> 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) {
|
|
nsAutoCString str;
|
|
uri->GetSpec(str);
|
|
if (str.EqualsLiteral("about:blank")) {
|
|
loadAsHtml5 = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
CSSLoader()->SetCompatibilityMode(mCompatMode);
|
|
|
|
nsresult rv = nsDocument::StartDocumentLoad(aCommand,
|
|
aChannel, aLoadGroup,
|
|
aContainer,
|
|
aDocListener, aReset);
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
// Store the security info for future use with wyciwyg channels.
|
|
aChannel->GetSecurityInfo(getter_AddRefs(mSecurityInfo));
|
|
|
|
nsCOMPtr<nsIURI> uri;
|
|
rv = aChannel->GetURI(getter_AddRefs(uri));
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
nsCOMPtr<nsICachingChannel> cachingChan = do_QueryInterface(aChannel);
|
|
|
|
if (loadAsHtml5) {
|
|
mParser = nsHtml5Module::NewHtml5Parser();
|
|
if (plainText) {
|
|
if (viewSource) {
|
|
mParser->MarkAsNotScriptCreated("view-source-plain");
|
|
} else {
|
|
mParser->MarkAsNotScriptCreated("plain-text");
|
|
}
|
|
} else if (viewSource && !html) {
|
|
mParser->MarkAsNotScriptCreated("view-source-xml");
|
|
} else {
|
|
mParser->MarkAsNotScriptCreated(aCommand);
|
|
}
|
|
} 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<nsIDocShell> docShell(do_QueryInterface(aContainer));
|
|
nsCOMPtr<nsIDocShellTreeItem> parentAsItem;
|
|
if (docShell) {
|
|
docShell->GetSameTypeParent(getter_AddRefs(parentAsItem));
|
|
}
|
|
|
|
nsCOMPtr<nsIDocShell> parent(do_QueryInterface(parentAsItem));
|
|
nsCOMPtr<nsIContentViewer> parentContentViewer;
|
|
if (parent) {
|
|
rv = parent->GetContentViewer(getter_AddRefs(parentContentViewer));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
|
|
nsCOMPtr<nsIContentViewer> 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;
|
|
nsAutoCString charset;
|
|
|
|
// These are the charset source and charset for the parser. This can differ
|
|
// from that for the document if the channel is a wyciwyg channel.
|
|
int32_t parserCharsetSource;
|
|
nsAutoCString parserCharset;
|
|
|
|
nsCOMPtr<nsIWyciwygChannel> wyciwygChannel;
|
|
|
|
// For error reporting and referrer policy setting
|
|
nsHtml5TreeOpExecutor* executor = nullptr;
|
|
if (loadAsHtml5) {
|
|
executor = static_cast<nsHtml5TreeOpExecutor*> (mParser->GetContentSink());
|
|
if (mReferrerPolicySet) {
|
|
// CSP may have set the referrer policy, so a speculative parser should
|
|
// start with the new referrer policy.
|
|
executor->SetSpeculationReferrerPolicy(static_cast<ReferrerPolicy>(mReferrerPolicy));
|
|
}
|
|
}
|
|
|
|
if (!IsHTMLDocument() || !docShell) { // no docshell for text/html XHR
|
|
charsetSource = IsHTMLDocument() ? kCharsetFromFallback
|
|
: kCharsetFromDocTypeDefault;
|
|
charset.AssignLiteral("UTF-8");
|
|
TryChannelCharset(aChannel, charsetSource, charset, executor);
|
|
parserCharsetSource = charsetSource;
|
|
parserCharset = charset;
|
|
} else {
|
|
NS_ASSERTION(docShell, "Unexpected null value");
|
|
|
|
charsetSource = kCharsetUninitialized;
|
|
wyciwygChannel = do_QueryInterface(aChannel);
|
|
|
|
// 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.
|
|
|
|
// Don't actually get the charset from the channel if this is a
|
|
// wyciwyg channel; it'll always be UTF-16
|
|
if (!wyciwygChannel) {
|
|
// Otherwise, 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, charset, executor);
|
|
}
|
|
|
|
TryUserForcedCharset(cv, docShell, charsetSource, charset);
|
|
|
|
TryHintCharset(cv, charsetSource, charset); // XXX mailnews-only
|
|
TryParentCharset(docShell, charsetSource, charset);
|
|
|
|
if (cachingChan && !urlSpec.IsEmpty()) {
|
|
TryCacheCharset(cachingChan, charsetSource, charset);
|
|
}
|
|
|
|
TryTLD(charsetSource, charset);
|
|
TryFallback(charsetSource, charset);
|
|
|
|
if (wyciwygChannel) {
|
|
// We know for sure that the parser needs to be using UTF16.
|
|
parserCharset = "UTF-16";
|
|
parserCharsetSource = charsetSource < kCharsetFromChannel ?
|
|
kCharsetFromChannel : charsetSource;
|
|
|
|
nsAutoCString cachedCharset;
|
|
int32_t cachedSource;
|
|
rv = wyciwygChannel->GetCharsetAndSource(&cachedSource, cachedCharset);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
if (cachedSource > charsetSource) {
|
|
charsetSource = cachedSource;
|
|
charset = cachedCharset;
|
|
}
|
|
} else {
|
|
// Don't propagate this error.
|
|
rv = NS_OK;
|
|
}
|
|
|
|
} else {
|
|
parserCharset = charset;
|
|
parserCharsetSource = charsetSource;
|
|
}
|
|
}
|
|
|
|
SetDocumentCharacterSetSource(charsetSource);
|
|
SetDocumentCharacterSet(charset);
|
|
|
|
if (cachingChan) {
|
|
NS_ASSERTION(charset == parserCharset,
|
|
"How did those end up different here? wyciwyg channels are "
|
|
"not nsICachingChannel");
|
|
rv = cachingChan->SetCacheTokenCachedCharset(charset);
|
|
NS_WARN_IF_FALSE(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<nsIStreamListener> listener = mParser->GetStreamListener();
|
|
listener.forget(aDocListener);
|
|
|
|
#ifdef DEBUG_charset
|
|
printf(" charset = %s source %d\n",
|
|
charset.get(), charsetSource);
|
|
#endif
|
|
mParser->SetDocumentCharset(parserCharset, parserCharsetSource);
|
|
mParser->SetCommand(aCommand);
|
|
|
|
if (!IsHTMLDocument()) {
|
|
MOZ_ASSERT(!loadAsHtml5);
|
|
nsCOMPtr<nsIXMLContentSink> xmlsink;
|
|
NS_NewXMLContentSink(getter_AddRefs(xmlsink), this, uri,
|
|
docShell, aChannel);
|
|
mParser->SetContentSink(xmlsink);
|
|
} else {
|
|
if (loadAsHtml5) {
|
|
nsHtml5Module::Initialize(mParser, this, uri, docShell, aChannel);
|
|
} else {
|
|
// about:blank *only*
|
|
nsCOMPtr<nsIHTMLContentSink> htmlsink;
|
|
NS_NewHTMLContentSink(getter_AddRefs(htmlsink), this, uri,
|
|
docShell, aChannel);
|
|
mParser->SetContentSink(htmlsink);
|
|
}
|
|
}
|
|
|
|
if (plainText && !nsContentUtils::IsChildOfSameType(this) &&
|
|
Preferences::GetBool("plain_text.wrap_long_lines")) {
|
|
nsCOMPtr<nsIStringBundleService> bundleService = do_GetService(NS_STRINGBUNDLE_CONTRACTID, &rv);
|
|
NS_ASSERTION(NS_SUCCEEDED(rv) && bundleService, "The bundle service could not be loaded");
|
|
nsCOMPtr<nsIStringBundle> bundle;
|
|
rv = bundleService->CreateBundle("chrome://global/locale/browser.properties",
|
|
getter_AddRefs(bundle));
|
|
NS_ASSERTION(NS_SUCCEEDED(rv) && bundle, "chrome://global/locale/browser.properties could not be loaded");
|
|
nsXPIDLString title;
|
|
if (bundle) {
|
|
bundle->GetStringFromName(MOZ_UTF16("plainText.wordWrap"), getter_Copies(title));
|
|
}
|
|
SetSelectedStyleSheetSet(title);
|
|
}
|
|
|
|
// parser the content of the URI
|
|
mParser->Parse(uri, nullptr, (void *)this);
|
|
|
|
return rv;
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::StopDocumentLoad()
|
|
{
|
|
BlockOnload();
|
|
|
|
// Remove the wyciwyg channel request from the document load group
|
|
// that we added in Open() if Open() was called on this doc.
|
|
RemoveWyciwygChannel();
|
|
NS_ASSERTION(!mWyciwygChannel, "nsHTMLDocument::StopDocumentLoad(): "
|
|
"nsIWyciwygChannel could not be removed!");
|
|
|
|
nsDocument::StopDocumentLoad();
|
|
UnblockOnload(false);
|
|
return;
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::BeginLoad()
|
|
{
|
|
if (IsEditingOn()) {
|
|
// Reset() blows away all event listeners in the document, and our
|
|
// editor relies heavily on those. Midas is turned on, to make it
|
|
// work, re-initialize it to give it a chance to add its event
|
|
// listeners again.
|
|
|
|
TurnEditingOff();
|
|
EditingStateChanged();
|
|
}
|
|
nsDocument::BeginLoad();
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::EndLoad()
|
|
{
|
|
bool turnOnEditing =
|
|
mParser && (HasFlag(NODE_IS_EDITABLE) || mContentEditableCount > 0);
|
|
// Note: nsDocument::EndLoad nulls out mParser.
|
|
nsDocument::EndLoad();
|
|
if (turnOnEditing) {
|
|
EditingStateChanged();
|
|
}
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::SetCompatibilityMode(nsCompatibility aMode)
|
|
{
|
|
NS_ASSERTION(IsHTMLDocument() || aMode == eCompatibility_FullStandards,
|
|
"Bad compat mode for XHTML document!");
|
|
|
|
mCompatMode = aMode;
|
|
CSSLoader()->SetCompatibilityMode(mCompatMode);
|
|
nsCOMPtr<nsIPresShell> shell = GetShell();
|
|
if (shell) {
|
|
nsPresContext *pc = shell->GetPresContext();
|
|
if (pc) {
|
|
pc->CompatibilityModeChanged();
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// nsIDOMHTMLDocument interface implementation
|
|
//
|
|
already_AddRefed<nsIURI>
|
|
nsHTMLDocument::GetDomainURI()
|
|
{
|
|
nsIPrincipal* principal = NodePrincipal();
|
|
|
|
nsCOMPtr<nsIURI> uri;
|
|
principal->GetDomain(getter_AddRefs(uri));
|
|
if (uri) {
|
|
return uri.forget();
|
|
}
|
|
|
|
principal->GetURI(getter_AddRefs(uri));
|
|
return uri.forget();
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetDomain(nsAString& aDomain)
|
|
{
|
|
nsCOMPtr<nsIURI> uri = GetDomainURI();
|
|
|
|
if (!uri) {
|
|
SetDOMStringToNull(aDomain);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsAutoCString hostName;
|
|
|
|
if (NS_SUCCEEDED(uri->GetHost(hostName))) {
|
|
CopyUTF8toUTF16(hostName, aDomain);
|
|
} else {
|
|
// If we can't get the host from the URI (e.g. about:, javascript:,
|
|
// etc), just return an null string.
|
|
SetDOMStringToNull(aDomain);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::SetDomain(const nsAString& aDomain)
|
|
{
|
|
ErrorResult rv;
|
|
SetDomain(aDomain, rv);
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::SetDomain(const nsAString& aDomain, ErrorResult& rv)
|
|
{
|
|
if (mSandboxFlags & SANDBOXED_DOMAIN) {
|
|
// We're sandboxed; disallow setting domain
|
|
rv.Throw(NS_ERROR_DOM_SECURITY_ERR);
|
|
return;
|
|
}
|
|
|
|
if (aDomain.IsEmpty()) {
|
|
rv.Throw(NS_ERROR_DOM_BAD_DOCUMENT_DOMAIN);
|
|
return;
|
|
}
|
|
|
|
// Create new URI
|
|
nsCOMPtr<nsIURI> uri = GetDomainURI();
|
|
|
|
if (!uri) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return;
|
|
}
|
|
|
|
nsAutoCString newURIString;
|
|
if (NS_FAILED(uri->GetScheme(newURIString))) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return;
|
|
}
|
|
nsAutoCString path;
|
|
if (NS_FAILED(uri->GetPath(path))) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return;
|
|
}
|
|
newURIString.AppendLiteral("://");
|
|
AppendUTF16toUTF8(aDomain, newURIString);
|
|
newURIString.Append(path);
|
|
|
|
nsCOMPtr<nsIURI> newURI;
|
|
if (NS_FAILED(NS_NewURI(getter_AddRefs(newURI), newURIString))) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return;
|
|
}
|
|
|
|
// Check new domain - must be a superdomain of the current host
|
|
// For example, a page from foo.bar.com may set domain to bar.com,
|
|
// but not to ar.com, baz.com, or fi.foo.bar.com.
|
|
nsAutoCString current, domain;
|
|
if (NS_FAILED(uri->GetAsciiHost(current)))
|
|
current.Truncate();
|
|
if (NS_FAILED(newURI->GetAsciiHost(domain)))
|
|
domain.Truncate();
|
|
|
|
bool ok = current.Equals(domain);
|
|
if (current.Length() > domain.Length() &&
|
|
StringEndsWith(current, domain) &&
|
|
current.CharAt(current.Length() - domain.Length() - 1) == '.') {
|
|
// We're golden if the new domain is the current page's base domain or a
|
|
// subdomain of it.
|
|
nsCOMPtr<nsIEffectiveTLDService> tldService =
|
|
do_GetService(NS_EFFECTIVETLDSERVICE_CONTRACTID);
|
|
if (!tldService) {
|
|
rv.Throw(NS_ERROR_NOT_AVAILABLE);
|
|
return;
|
|
}
|
|
|
|
nsAutoCString currentBaseDomain;
|
|
ok = NS_SUCCEEDED(tldService->GetBaseDomain(uri, 0, currentBaseDomain));
|
|
NS_ASSERTION(StringEndsWith(domain, currentBaseDomain) ==
|
|
(domain.Length() >= currentBaseDomain.Length()),
|
|
"uh-oh! slight optimization wasn't valid somehow!");
|
|
ok = ok && domain.Length() >= currentBaseDomain.Length();
|
|
}
|
|
if (!ok) {
|
|
// Error: illegal domain
|
|
rv.Throw(NS_ERROR_DOM_BAD_DOCUMENT_DOMAIN);
|
|
return;
|
|
}
|
|
|
|
rv = NodePrincipal()->SetDomain(newURI);
|
|
}
|
|
|
|
nsGenericHTMLElement*
|
|
nsHTMLDocument::GetBody()
|
|
{
|
|
Element* html = GetHtmlElement();
|
|
if (!html) {
|
|
return nullptr;
|
|
}
|
|
|
|
for (nsIContent* child = html->GetFirstChild();
|
|
child;
|
|
child = child->GetNextSibling()) {
|
|
if (child->IsHTMLElement(nsGkAtoms::body) ||
|
|
child->IsHTMLElement(nsGkAtoms::frameset)) {
|
|
return static_cast<nsGenericHTMLElement*>(child);
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetBody(nsIDOMHTMLElement** aBody)
|
|
{
|
|
*aBody = nullptr;
|
|
|
|
nsIContent *body = GetBody();
|
|
|
|
return body ? CallQueryInterface(body, aBody) : NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::SetBody(nsIDOMHTMLElement* aBody)
|
|
{
|
|
nsCOMPtr<nsIContent> newBody = do_QueryInterface(aBody);
|
|
MOZ_ASSERT(!newBody || newBody->IsHTMLElement(),
|
|
"How could we be an nsIContent but not actually HTML here?");
|
|
ErrorResult rv;
|
|
SetBody(static_cast<nsGenericHTMLElement*>(newBody.get()), rv);
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::SetBody(nsGenericHTMLElement* newBody, ErrorResult& rv)
|
|
{
|
|
Element* root = GetRootElement();
|
|
|
|
// The body element must be either a body tag or a frameset tag. And we must
|
|
// have a html root tag, otherwise GetBody will not return the newly set
|
|
// body.
|
|
if (!newBody ||
|
|
!newBody->IsAnyOfHTMLElements(nsGkAtoms::body, nsGkAtoms::frameset) ||
|
|
!root || !root->IsHTMLElement() ||
|
|
!root->IsHTMLElement(nsGkAtoms::html)) {
|
|
rv.Throw(NS_ERROR_DOM_HIERARCHY_REQUEST_ERR);
|
|
return;
|
|
}
|
|
|
|
// Use DOM methods so that we pass through the appropriate security checks.
|
|
nsCOMPtr<Element> currentBody = GetBodyElement();
|
|
if (currentBody) {
|
|
root->ReplaceChild(*newBody, *currentBody, rv);
|
|
} else {
|
|
root->AppendChild(*newBody, rv);
|
|
}
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetHead(nsIDOMHTMLHeadElement** aHead)
|
|
{
|
|
*aHead = nullptr;
|
|
|
|
Element* head = GetHeadElement();
|
|
|
|
return head ? CallQueryInterface(head, aHead) : NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetImages(nsIDOMHTMLCollection** aImages)
|
|
{
|
|
NS_ADDREF(*aImages = Images());
|
|
return NS_OK;
|
|
}
|
|
|
|
nsIHTMLCollection*
|
|
nsHTMLDocument::Images()
|
|
{
|
|
if (!mImages) {
|
|
mImages = new nsContentList(this, kNameSpaceID_XHTML, nsGkAtoms::img, nsGkAtoms::img);
|
|
}
|
|
return mImages;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetApplets(nsIDOMHTMLCollection** aApplets)
|
|
{
|
|
NS_ADDREF(*aApplets = Applets());
|
|
return NS_OK;
|
|
}
|
|
|
|
nsIHTMLCollection*
|
|
nsHTMLDocument::Applets()
|
|
{
|
|
if (!mApplets) {
|
|
mApplets = new nsContentList(this, kNameSpaceID_XHTML, nsGkAtoms::applet, nsGkAtoms::applet);
|
|
}
|
|
return mApplets;
|
|
}
|
|
|
|
bool
|
|
nsHTMLDocument::MatchLinks(nsIContent *aContent, int32_t aNamespaceID,
|
|
nsIAtom* aAtom, void* aData)
|
|
{
|
|
nsIDocument* doc = aContent->GetUncomposedDoc();
|
|
|
|
if (doc) {
|
|
NS_ASSERTION(aContent->IsInDoc(),
|
|
"This method should never be called on content nodes that "
|
|
"are not in a document!");
|
|
#ifdef DEBUG
|
|
{
|
|
nsCOMPtr<nsIHTMLDocument> htmldoc =
|
|
do_QueryInterface(aContent->GetUncomposedDoc());
|
|
NS_ASSERTION(htmldoc,
|
|
"Huh, how did this happen? This should only be used with "
|
|
"HTML documents!");
|
|
}
|
|
#endif
|
|
|
|
mozilla::dom::NodeInfo *ni = aContent->NodeInfo();
|
|
|
|
nsIAtom *localName = ni->NameAtom();
|
|
if (ni->NamespaceID() == kNameSpaceID_XHTML &&
|
|
(localName == nsGkAtoms::a || localName == nsGkAtoms::area)) {
|
|
return aContent->HasAttr(kNameSpaceID_None, nsGkAtoms::href);
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetLinks(nsIDOMHTMLCollection** aLinks)
|
|
{
|
|
NS_ADDREF(*aLinks = Links());
|
|
return NS_OK;
|
|
}
|
|
|
|
nsIHTMLCollection*
|
|
nsHTMLDocument::Links()
|
|
{
|
|
if (!mLinks) {
|
|
mLinks = new nsContentList(this, MatchLinks, nullptr, nullptr);
|
|
}
|
|
return mLinks;
|
|
}
|
|
|
|
bool
|
|
nsHTMLDocument::MatchAnchors(nsIContent *aContent, int32_t aNamespaceID,
|
|
nsIAtom* aAtom, void* aData)
|
|
{
|
|
NS_ASSERTION(aContent->IsInDoc(),
|
|
"This method should never be called on content nodes that "
|
|
"are not in a document!");
|
|
#ifdef DEBUG
|
|
{
|
|
nsCOMPtr<nsIHTMLDocument> htmldoc =
|
|
do_QueryInterface(aContent->GetUncomposedDoc());
|
|
NS_ASSERTION(htmldoc,
|
|
"Huh, how did this happen? This should only be used with "
|
|
"HTML documents!");
|
|
}
|
|
#endif
|
|
|
|
if (aContent->NodeInfo()->Equals(nsGkAtoms::a, kNameSpaceID_XHTML)) {
|
|
return aContent->HasAttr(kNameSpaceID_None, nsGkAtoms::name);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetAnchors(nsIDOMHTMLCollection** aAnchors)
|
|
{
|
|
NS_ADDREF(*aAnchors = Anchors());
|
|
return NS_OK;
|
|
}
|
|
|
|
nsIHTMLCollection*
|
|
nsHTMLDocument::Anchors()
|
|
{
|
|
if (!mAnchors) {
|
|
mAnchors = new nsContentList(this, MatchAnchors, nullptr, nullptr);
|
|
}
|
|
return mAnchors;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetScripts(nsIDOMHTMLCollection** aScripts)
|
|
{
|
|
NS_ADDREF(*aScripts = Scripts());
|
|
return NS_OK;
|
|
}
|
|
|
|
nsIHTMLCollection*
|
|
nsHTMLDocument::Scripts()
|
|
{
|
|
if (!mScripts) {
|
|
mScripts = new nsContentList(this, kNameSpaceID_XHTML, nsGkAtoms::script, nsGkAtoms::script);
|
|
}
|
|
return mScripts;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetCookie(nsAString& aCookie)
|
|
{
|
|
ErrorResult rv;
|
|
GetCookie(aCookie, rv);
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
already_AddRefed<nsIChannel>
|
|
nsHTMLDocument::CreateDummyChannelForCookies(nsIURI* aCodebaseURI)
|
|
{
|
|
// The cookie service reads the privacy status of the channel we pass to it in
|
|
// order to determine which cookie database to query. In some cases we don't
|
|
// have a proper channel to hand it to the cookie service though. This
|
|
// function creates a dummy channel that is not used to load anything, for the
|
|
// sole purpose of handing it to the cookie service. DO NOT USE THIS CHANNEL
|
|
// FOR ANY OTHER PURPOSE.
|
|
MOZ_ASSERT(!mChannel);
|
|
|
|
nsCOMPtr<nsIChannel> channel;
|
|
NS_NewChannel(getter_AddRefs(channel), aCodebaseURI, this,
|
|
nsILoadInfo::SEC_NORMAL,
|
|
nsIContentPolicy::TYPE_INVALID);
|
|
nsCOMPtr<nsIPrivateBrowsingChannel> pbChannel =
|
|
do_QueryInterface(channel);
|
|
nsCOMPtr<nsIDocShell> docShell(mDocumentContainer);
|
|
nsCOMPtr<nsILoadContext> loadContext = do_QueryInterface(docShell);
|
|
if (!pbChannel || !loadContext) {
|
|
return nullptr;
|
|
}
|
|
pbChannel->SetPrivate(loadContext->UsePrivateBrowsing());
|
|
return channel.forget();
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::GetCookie(nsAString& aCookie, ErrorResult& rv)
|
|
{
|
|
aCookie.Truncate(); // clear current cookie in case service fails;
|
|
// no cookie isn't an error condition.
|
|
|
|
if (mDisableCookieAccess) {
|
|
return;
|
|
}
|
|
|
|
// If the document's sandboxed origin flag is set, access to read cookies
|
|
// is prohibited.
|
|
if (mSandboxFlags & SANDBOXED_ORIGIN) {
|
|
rv.Throw(NS_ERROR_DOM_SECURITY_ERR);
|
|
return;
|
|
}
|
|
|
|
// not having a cookie service isn't an error
|
|
nsCOMPtr<nsICookieService> service = do_GetService(NS_COOKIESERVICE_CONTRACTID);
|
|
if (service) {
|
|
// Get a URI from the document principal. We use the original
|
|
// codebase in case the codebase was changed by SetDomain
|
|
nsCOMPtr<nsIURI> codebaseURI;
|
|
NodePrincipal()->GetURI(getter_AddRefs(codebaseURI));
|
|
|
|
if (!codebaseURI) {
|
|
// Document's principal is not a codebase (may be system), so
|
|
// can't set cookies
|
|
|
|
return;
|
|
}
|
|
|
|
nsCOMPtr<nsIChannel> channel(mChannel);
|
|
if (!channel) {
|
|
channel = CreateDummyChannelForCookies(codebaseURI);
|
|
if (!channel) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
nsXPIDLCString cookie;
|
|
service->GetCookieString(codebaseURI, channel, getter_Copies(cookie));
|
|
// CopyUTF8toUTF16 doesn't handle error
|
|
// because it assumes that the input is valid.
|
|
nsContentUtils::ConvertStringFromEncoding(NS_LITERAL_CSTRING("UTF-8"),
|
|
cookie, aCookie);
|
|
}
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::SetCookie(const nsAString& aCookie)
|
|
{
|
|
ErrorResult rv;
|
|
SetCookie(aCookie, rv);
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::SetCookie(const nsAString& aCookie, ErrorResult& rv)
|
|
{
|
|
if (mDisableCookieAccess) {
|
|
return;
|
|
}
|
|
|
|
// If the document's sandboxed origin flag is set, access to write cookies
|
|
// is prohibited.
|
|
if (mSandboxFlags & SANDBOXED_ORIGIN) {
|
|
rv.Throw(NS_ERROR_DOM_SECURITY_ERR);
|
|
return;
|
|
}
|
|
|
|
// not having a cookie service isn't an error
|
|
nsCOMPtr<nsICookieService> service = do_GetService(NS_COOKIESERVICE_CONTRACTID);
|
|
if (service && mDocumentURI) {
|
|
// The for getting the URI matches nsNavigator::GetCookieEnabled
|
|
nsCOMPtr<nsIURI> codebaseURI;
|
|
NodePrincipal()->GetURI(getter_AddRefs(codebaseURI));
|
|
|
|
if (!codebaseURI) {
|
|
// Document's principal is not a codebase (may be system), so
|
|
// can't set cookies
|
|
|
|
return;
|
|
}
|
|
|
|
nsCOMPtr<nsIChannel> channel(mChannel);
|
|
if (!channel) {
|
|
channel = CreateDummyChannelForCookies(codebaseURI);
|
|
if (!channel) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
NS_ConvertUTF16toUTF8 cookie(aCookie);
|
|
service->SetCookieString(codebaseURI, nullptr, cookie.get(), channel);
|
|
}
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::Open(const nsAString& aContentTypeOrUrl,
|
|
const nsAString& aReplaceOrName,
|
|
const nsAString& aFeatures,
|
|
JSContext* cx, uint8_t aOptionalArgCount,
|
|
nsISupports** aReturn)
|
|
{
|
|
// When called with 3 or more arguments, document.open() calls window.open().
|
|
if (aOptionalArgCount > 2) {
|
|
ErrorResult rv;
|
|
*aReturn = Open(cx, aContentTypeOrUrl, aReplaceOrName, aFeatures,
|
|
false, rv).take();
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
nsString type;
|
|
if (aOptionalArgCount > 0) {
|
|
type = aContentTypeOrUrl;
|
|
} else {
|
|
type.AssignLiteral("text/html");
|
|
}
|
|
nsString replace;
|
|
if (aOptionalArgCount > 1) {
|
|
replace = aReplaceOrName;
|
|
}
|
|
ErrorResult rv;
|
|
*aReturn = Open(cx, type, replace, rv).take();
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
already_AddRefed<nsIDOMWindow>
|
|
nsHTMLDocument::Open(JSContext* /* unused */,
|
|
const nsAString& aURL,
|
|
const nsAString& aName,
|
|
const nsAString& aFeatures,
|
|
bool aReplace,
|
|
ErrorResult& rv)
|
|
{
|
|
NS_ASSERTION(nsContentUtils::CanCallerAccess(static_cast<nsIDOMHTMLDocument*>(this)),
|
|
"XOW should have caught this!");
|
|
|
|
nsCOMPtr<nsIDOMWindow> window = GetInnerWindow();
|
|
if (!window) {
|
|
rv.Throw(NS_ERROR_DOM_INVALID_ACCESS_ERR);
|
|
return nullptr;
|
|
}
|
|
nsCOMPtr<nsIDOMJSWindow> win = do_QueryInterface(window);
|
|
nsCOMPtr<nsIDOMWindow> newWindow;
|
|
// XXXbz We ignore aReplace for now.
|
|
rv = win->OpenJS(aURL, aName, aFeatures, getter_AddRefs(newWindow));
|
|
return newWindow.forget();
|
|
}
|
|
|
|
already_AddRefed<nsIDocument>
|
|
nsHTMLDocument::Open(JSContext* cx,
|
|
const nsAString& aType,
|
|
const nsAString& aReplace,
|
|
ErrorResult& rv)
|
|
{
|
|
NS_ASSERTION(nsContentUtils::CanCallerAccess(static_cast<nsIDOMHTMLDocument*>(this)),
|
|
"XOW should have caught this!");
|
|
if (!IsHTMLDocument() || mDisableDocWrite || !IsMasterDocument()) {
|
|
// No calling document.open() on XHTML
|
|
rv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR);
|
|
return nullptr;
|
|
}
|
|
|
|
nsAutoCString contentType;
|
|
contentType.AssignLiteral("text/html");
|
|
|
|
nsAutoString type;
|
|
nsContentUtils::ASCIIToLower(aType, type);
|
|
nsAutoCString actualType, dummy;
|
|
NS_ParseContentType(NS_ConvertUTF16toUTF8(type), actualType, dummy);
|
|
if (!actualType.EqualsLiteral("text/html") &&
|
|
!type.EqualsLiteral("replace")) {
|
|
contentType.AssignLiteral("text/plain");
|
|
}
|
|
|
|
// If we already have a parser we ignore the document.open call.
|
|
if (mParser || mParserAborted) {
|
|
// The WHATWG spec says: "If the document has an active parser that isn't
|
|
// a script-created parser, and the insertion point associated with that
|
|
// parser's input stream is not undefined (that is, it does point to
|
|
// somewhere in the input stream), then the method does nothing. Abort
|
|
// these steps and return the Document object on which the method was
|
|
// invoked."
|
|
// Note that aborting a parser leaves the parser "active" with its
|
|
// insertion point "not undefined". We track this using mParserAborted,
|
|
// because aborting a parser nulls out mParser.
|
|
nsCOMPtr<nsIDocument> ret = this;
|
|
return ret.forget();
|
|
}
|
|
|
|
// No calling document.open() without a script global object
|
|
if (!mScriptGlobalObject) {
|
|
nsCOMPtr<nsIDocument> ret = this;
|
|
return ret.forget();
|
|
}
|
|
|
|
nsPIDOMWindow* outer = GetWindow();
|
|
if (!outer || (GetInnerWindow() != outer->GetCurrentInnerWindow())) {
|
|
nsCOMPtr<nsIDocument> ret = this;
|
|
return ret.forget();
|
|
}
|
|
|
|
// check whether we're in the middle of unload. If so, ignore this call.
|
|
nsCOMPtr<nsIDocShell> shell(mDocumentContainer);
|
|
if (!shell) {
|
|
// We won't be able to create a parser anyway.
|
|
nsCOMPtr<nsIDocument> ret = this;
|
|
return ret.forget();
|
|
}
|
|
|
|
bool inUnload;
|
|
shell->GetIsInUnload(&inUnload);
|
|
if (inUnload) {
|
|
nsCOMPtr<nsIDocument> ret = this;
|
|
return ret.forget();
|
|
}
|
|
|
|
// Note: We want to use GetEntryDocument here because this document
|
|
// should inherit the security information of the document that's opening us,
|
|
// (since if it's secure, then it's presumably trusted).
|
|
nsCOMPtr<nsIDocument> callerDoc = GetEntryDocument();
|
|
if (!callerDoc) {
|
|
// If we're called from C++ or in some other way without an originating
|
|
// document we can't do a document.open w/o changing the principal of the
|
|
// document to something like about:blank (as that's the only sane thing to
|
|
// do when we don't know the origin of this call), and since we can't
|
|
// change the principals of a document for security reasons we'll have to
|
|
// refuse to go ahead with this call.
|
|
|
|
rv.Throw(NS_ERROR_DOM_SECURITY_ERR);
|
|
return nullptr;
|
|
}
|
|
|
|
// Grab a reference to the calling documents security info (if any)
|
|
// and URIs as they may be lost in the call to Reset().
|
|
nsCOMPtr<nsISupports> securityInfo = callerDoc->GetSecurityInfo();
|
|
nsCOMPtr<nsIURI> uri = callerDoc->GetDocumentURI();
|
|
nsCOMPtr<nsIURI> baseURI = callerDoc->GetBaseURI();
|
|
nsCOMPtr<nsIPrincipal> callerPrincipal = callerDoc->NodePrincipal();
|
|
nsCOMPtr<nsIChannel> callerChannel = callerDoc->GetChannel();
|
|
|
|
// We're called from script. Make sure the script is from the same
|
|
// origin, not just that the caller can access the document. This is
|
|
// needed to keep document principals from ever changing, which is
|
|
// needed because of the way we use our XOW code, and is a sane
|
|
// thing to do anyways.
|
|
|
|
bool equals = false;
|
|
if (NS_FAILED(callerPrincipal->Equals(NodePrincipal(), &equals)) ||
|
|
!equals) {
|
|
|
|
#ifdef DEBUG
|
|
nsCOMPtr<nsIURI> callerDocURI = callerDoc->GetDocumentURI();
|
|
nsCOMPtr<nsIURI> thisURI = nsIDocument::GetDocumentURI();
|
|
nsAutoCString callerSpec;
|
|
nsAutoCString thisSpec;
|
|
if (callerDocURI) {
|
|
callerDocURI->GetSpec(callerSpec);
|
|
}
|
|
if (thisURI) {
|
|
thisURI->GetSpec(thisSpec);
|
|
}
|
|
printf("nsHTMLDocument::Open callerDoc %s this %s\n", callerSpec.get(), thisSpec.get());
|
|
#endif
|
|
|
|
rv.Throw(NS_ERROR_DOM_SECURITY_ERR);
|
|
return nullptr;
|
|
}
|
|
|
|
// Stop current loads targeted at the window this document is in.
|
|
if (mScriptGlobalObject) {
|
|
nsCOMPtr<nsIContentViewer> cv;
|
|
shell->GetContentViewer(getter_AddRefs(cv));
|
|
|
|
if (cv) {
|
|
bool okToUnload;
|
|
if (NS_SUCCEEDED(cv->PermitUnload(false, &okToUnload)) && !okToUnload) {
|
|
// We don't want to unload, so stop here, but don't throw an
|
|
// exception.
|
|
nsCOMPtr<nsIDocument> ret = this;
|
|
return ret.forget();
|
|
}
|
|
}
|
|
|
|
nsCOMPtr<nsIWebNavigation> webnav(do_QueryInterface(shell));
|
|
webnav->Stop(nsIWebNavigation::STOP_NETWORK);
|
|
|
|
// The Stop call may have cancelled the onload blocker request or prevented
|
|
// it from getting added, so we need to make sure it gets added to the
|
|
// document again otherwise the document could have a non-zero onload block
|
|
// count without the onload blocker request being in the loadgroup.
|
|
EnsureOnloadBlocker();
|
|
}
|
|
|
|
// The open occurred after the document finished loading.
|
|
// So we reset the document and create a new one.
|
|
nsCOMPtr<nsIChannel> channel;
|
|
nsCOMPtr<nsILoadGroup> group = do_QueryReferent(mDocumentLoadGroup);
|
|
rv = NS_NewChannel(getter_AddRefs(channel),
|
|
uri,
|
|
callerDoc,
|
|
nsILoadInfo::SEC_FORCE_INHERIT_PRINCIPAL,
|
|
nsIContentPolicy::TYPE_OTHER,
|
|
group);
|
|
|
|
if (rv.Failed()) {
|
|
return nullptr;
|
|
}
|
|
|
|
if (callerChannel) {
|
|
nsLoadFlags callerLoadFlags;
|
|
rv = callerChannel->GetLoadFlags(&callerLoadFlags);
|
|
if (rv.Failed()) {
|
|
return nullptr;
|
|
}
|
|
|
|
nsLoadFlags loadFlags;
|
|
rv = channel->GetLoadFlags(&loadFlags);
|
|
if (rv.Failed()) {
|
|
return nullptr;
|
|
}
|
|
|
|
loadFlags |= callerLoadFlags & nsIRequest::INHIBIT_PERSISTENT_CACHING;
|
|
|
|
rv = channel->SetLoadFlags(loadFlags);
|
|
if (rv.Failed()) {
|
|
return nullptr;
|
|
}
|
|
|
|
// If the user has allowed mixed content on the rootDoc, then we should propogate it
|
|
// down to the new document channel.
|
|
bool rootHasSecureConnection = false;
|
|
bool allowMixedContent = false;
|
|
bool isDocShellRoot = false;
|
|
nsresult rvalue = shell->GetAllowMixedContentAndConnectionData(&rootHasSecureConnection, &allowMixedContent, &isDocShellRoot);
|
|
if (NS_SUCCEEDED(rvalue) && allowMixedContent && isDocShellRoot) {
|
|
shell->SetMixedContentChannel(channel);
|
|
}
|
|
}
|
|
|
|
// Before we reset the doc notify the globalwindow of the change,
|
|
// but only if we still have a window (i.e. our window object the
|
|
// current inner window in our outer window).
|
|
|
|
// Hold onto ourselves on the offchance that we're down to one ref
|
|
nsCOMPtr<nsIDocument> kungFuDeathGrip = this;
|
|
|
|
nsPIDOMWindow *window = GetInnerWindow();
|
|
if (window) {
|
|
// Remember the old scope in case the call to SetNewDocument changes it.
|
|
nsCOMPtr<nsIScriptGlobalObject> oldScope(do_QueryReferent(mScopeObject));
|
|
|
|
#ifdef DEBUG
|
|
bool willReparent = mWillReparent;
|
|
mWillReparent = true;
|
|
|
|
nsDocument* templateContentsOwner =
|
|
static_cast<nsDocument*>(mTemplateContentsOwner.get());
|
|
|
|
if (templateContentsOwner) {
|
|
templateContentsOwner->mWillReparent = true;
|
|
}
|
|
#endif
|
|
|
|
// Should this pass true for aForceReuseInnerWindow?
|
|
rv = window->SetNewDocument(this, nullptr, false);
|
|
if (rv.Failed()) {
|
|
return nullptr;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
if (templateContentsOwner) {
|
|
templateContentsOwner->mWillReparent = willReparent;
|
|
}
|
|
|
|
mWillReparent = willReparent;
|
|
#endif
|
|
|
|
// Now make sure we're not flagged as the initial document anymore, now
|
|
// that we've had stuff done to us. From now on, if anyone tries to
|
|
// document.open() us, they get a new inner window.
|
|
SetIsInitialDocument(false);
|
|
|
|
nsCOMPtr<nsIScriptGlobalObject> newScope(do_QueryReferent(mScopeObject));
|
|
JS::Rooted<JSObject*> wrapper(cx, GetWrapper());
|
|
if (oldScope && newScope != oldScope && wrapper) {
|
|
JSAutoCompartment ac(cx, wrapper);
|
|
rv = mozilla::dom::ReparentWrapper(cx, wrapper);
|
|
if (rv.Failed()) {
|
|
return nullptr;
|
|
}
|
|
|
|
// Also reparent the template contents owner document
|
|
// because its global is set to the same as this document.
|
|
if (mTemplateContentsOwner) {
|
|
JS::Rooted<JSObject*> contentsOwnerWrapper(cx,
|
|
mTemplateContentsOwner->GetWrapper());
|
|
if (contentsOwnerWrapper) {
|
|
rv = mozilla::dom::ReparentWrapper(cx, contentsOwnerWrapper);
|
|
if (rv.Failed()) {
|
|
return nullptr;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
mDidDocumentOpen = true;
|
|
|
|
// Call Reset(), this will now do the full reset
|
|
Reset(channel, group);
|
|
if (baseURI) {
|
|
mDocumentBaseURI = baseURI;
|
|
}
|
|
|
|
// Store the security info of the caller now that we're done
|
|
// resetting the document.
|
|
mSecurityInfo = securityInfo;
|
|
|
|
mParserAborted = false;
|
|
mParser = nsHtml5Module::NewHtml5Parser();
|
|
nsHtml5Module::Initialize(mParser, this, uri, shell, channel);
|
|
if (mReferrerPolicySet) {
|
|
// CSP may have set the referrer policy, so a speculative parser should
|
|
// start with the new referrer policy.
|
|
nsHtml5TreeOpExecutor* executor = nullptr;
|
|
executor = static_cast<nsHtml5TreeOpExecutor*> (mParser->GetContentSink());
|
|
if (executor && mReferrerPolicySet) {
|
|
executor->SetSpeculationReferrerPolicy(static_cast<ReferrerPolicy>(mReferrerPolicy));
|
|
}
|
|
}
|
|
|
|
// This will be propagated to the parser when someone actually calls write()
|
|
SetContentTypeInternal(contentType);
|
|
|
|
// Prepare the docshell and the document viewer for the impending
|
|
// out of band document.write()
|
|
shell->PrepareForNewContentModel();
|
|
|
|
// Now check whether we were opened with a "replace" argument. If
|
|
// so, we need to tell the docshell to not create a new history
|
|
// entry for this load. Otherwise, make sure that we're doing a normal load,
|
|
// not whatever type of load was previously done on this docshell.
|
|
shell->SetLoadType(aReplace.LowerCaseEqualsLiteral("replace") ?
|
|
LOAD_NORMAL_REPLACE : LOAD_NORMAL);
|
|
|
|
nsCOMPtr<nsIContentViewer> cv;
|
|
shell->GetContentViewer(getter_AddRefs(cv));
|
|
if (cv) {
|
|
cv->LoadStart(this);
|
|
}
|
|
|
|
// Add a wyciwyg channel request into the document load group
|
|
NS_ASSERTION(!mWyciwygChannel, "nsHTMLDocument::Open(): wyciwyg "
|
|
"channel already exists!");
|
|
|
|
// In case the editor is listening and will see the new channel
|
|
// being added, make sure mWriteLevel is non-zero so that the editor
|
|
// knows that document.open/write/close() is being called on this
|
|
// document.
|
|
++mWriteLevel;
|
|
|
|
CreateAndAddWyciwygChannel();
|
|
|
|
--mWriteLevel;
|
|
|
|
SetReadyStateInternal(nsIDocument::READYSTATE_LOADING);
|
|
|
|
// After changing everything around, make sure that the principal on the
|
|
// document's compartment exactly matches NodePrincipal().
|
|
DebugOnly<JSObject*> wrapper = GetWrapperPreserveColor();
|
|
MOZ_ASSERT_IF(wrapper,
|
|
JS_GetCompartmentPrincipals(js::GetObjectCompartment(wrapper)) ==
|
|
nsJSPrincipals::get(NodePrincipal()));
|
|
|
|
return kungFuDeathGrip.forget();
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::Clear()
|
|
{
|
|
// This method has been deprecated
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::Close()
|
|
{
|
|
ErrorResult rv;
|
|
Close(rv);
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::Close(ErrorResult& rv)
|
|
{
|
|
if (!IsHTMLDocument()) {
|
|
// No calling document.close() on XHTML!
|
|
|
|
rv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR);
|
|
return;
|
|
}
|
|
|
|
if (!mParser || !mParser->IsScriptCreated()) {
|
|
return;
|
|
}
|
|
|
|
++mWriteLevel;
|
|
rv = (static_cast<nsHtml5Parser*>(mParser.get()))->Parse(
|
|
EmptyString(), nullptr, GetContentTypeInternal(), true);
|
|
--mWriteLevel;
|
|
|
|
// Even if that Parse() call failed, do the rest of this method
|
|
|
|
// XXX Make sure that all the document.written content is
|
|
// reflowed. We should remove this call once we change
|
|
// nsHTMLDocument::OpenCommon() so that it completely destroys the
|
|
// earlier document's content and frame hierarchy. Right now, it
|
|
// re-uses the earlier document's root content object and
|
|
// corresponding frame objects. These re-used frame objects think
|
|
// that they have already been reflowed, so they drop initial
|
|
// reflows. For certain cases of document.written content, like a
|
|
// frameset document, the dropping of the initial reflow means
|
|
// that we end up in document.close() without appended any reflow
|
|
// commands to the reflow queue and, consequently, without adding
|
|
// the dummy layout request to the load group. Since the dummy
|
|
// layout request is not added to the load group, the onload
|
|
// handler of the frameset fires before the frames get reflowed
|
|
// and loaded. That is the long explanation for why we need this
|
|
// one line of code here!
|
|
// XXXbz as far as I can tell this may not be needed anymore; all
|
|
// the testcases in bug 57636 pass without this line... Leaving
|
|
// it be for now, though. In any case, there's no reason to do
|
|
// this if we have no presshell, since in that case none of the
|
|
// above about reusing frames applies.
|
|
//
|
|
// XXXhsivonen keeping this around for bug 577508 / 253951 still :-(
|
|
if (GetShell()) {
|
|
FlushPendingNotifications(Flush_Layout);
|
|
}
|
|
|
|
// Removing the wyciwygChannel here is wrong when document.close() is
|
|
// called from within the document itself. However, legacy requires the
|
|
// channel to be removed here. Otherwise, the load event never fires.
|
|
NS_ASSERTION(mWyciwygChannel, "nsHTMLDocument::Close(): Trying to remove "
|
|
"nonexistent wyciwyg channel!");
|
|
RemoveWyciwygChannel();
|
|
NS_ASSERTION(!mWyciwygChannel, "nsHTMLDocument::Close(): "
|
|
"nsIWyciwygChannel could not be removed!");
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::WriteCommon(JSContext *cx,
|
|
const Sequence<nsString>& aText,
|
|
bool aNewlineTerminate,
|
|
mozilla::ErrorResult& rv)
|
|
{
|
|
// Fast path the common case
|
|
if (aText.Length() == 1) {
|
|
rv = WriteCommon(cx, aText[0], aNewlineTerminate);
|
|
} else {
|
|
// XXXbz it would be nice if we could pass all the strings to the parser
|
|
// without having to do all this copying and then ask it to start
|
|
// parsing....
|
|
nsString text;
|
|
for (uint32_t i = 0; i < aText.Length(); ++i) {
|
|
text.Append(aText[i]);
|
|
}
|
|
rv = WriteCommon(cx, text, aNewlineTerminate);
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLDocument::WriteCommon(JSContext *cx,
|
|
const nsAString& aText,
|
|
bool aNewlineTerminate)
|
|
{
|
|
mTooDeepWriteRecursion =
|
|
(mWriteLevel > NS_MAX_DOCUMENT_WRITE_DEPTH || mTooDeepWriteRecursion);
|
|
NS_ENSURE_STATE(!mTooDeepWriteRecursion);
|
|
|
|
if (!IsHTMLDocument() || mDisableDocWrite || !IsMasterDocument()) {
|
|
// No calling document.write*() on XHTML!
|
|
|
|
return NS_ERROR_DOM_INVALID_STATE_ERR;
|
|
}
|
|
|
|
if (mParserAborted) {
|
|
// Hixie says aborting the parser doesn't undefine the insertion point.
|
|
// However, since we null out mParser in that case, we track the
|
|
// theoretically defined insertion point using mParserAborted.
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
void *key = GenerateParserKey();
|
|
if (mParser && !mParser->IsInsertionPointDefined()) {
|
|
if (mExternalScriptsBeingEvaluated) {
|
|
// Instead of implying a call to document.open(), ignore the call.
|
|
nsContentUtils::ReportToConsole(nsIScriptError::warningFlag,
|
|
NS_LITERAL_CSTRING("DOM Events"), this,
|
|
nsContentUtils::eDOM_PROPERTIES,
|
|
"DocumentWriteIgnored",
|
|
nullptr, 0,
|
|
mDocumentURI);
|
|
return NS_OK;
|
|
}
|
|
mParser->Terminate();
|
|
NS_ASSERTION(!mParser, "mParser should have been null'd out");
|
|
}
|
|
|
|
if (!mParser) {
|
|
if (mExternalScriptsBeingEvaluated) {
|
|
// Instead of implying a call to document.open(), ignore the call.
|
|
nsContentUtils::ReportToConsole(nsIScriptError::warningFlag,
|
|
NS_LITERAL_CSTRING("DOM Events"), this,
|
|
nsContentUtils::eDOM_PROPERTIES,
|
|
"DocumentWriteIgnored",
|
|
nullptr, 0,
|
|
mDocumentURI);
|
|
return NS_OK;
|
|
}
|
|
nsCOMPtr<nsISupports> ignored;
|
|
rv = Open(NS_LITERAL_STRING("text/html"), EmptyString(), EmptyString(), cx,
|
|
1, getter_AddRefs(ignored));
|
|
|
|
// If Open() fails, or if it didn't create a parser (as it won't
|
|
// if the user chose to not discard the current document through
|
|
// onbeforeunload), don't write anything.
|
|
if (NS_FAILED(rv) || !mParser) {
|
|
return rv;
|
|
}
|
|
MOZ_ASSERT(!JS_IsExceptionPending(cx),
|
|
"Open() succeeded but JS exception is pending");
|
|
}
|
|
|
|
static NS_NAMED_LITERAL_STRING(new_line, "\n");
|
|
|
|
// Save the data in cache if the write isn't from within the doc
|
|
if (mWyciwygChannel && !key) {
|
|
if (!aText.IsEmpty()) {
|
|
mWyciwygChannel->WriteToCacheEntry(aText);
|
|
}
|
|
|
|
if (aNewlineTerminate) {
|
|
mWyciwygChannel->WriteToCacheEntry(new_line);
|
|
}
|
|
}
|
|
|
|
++mWriteLevel;
|
|
|
|
// This could be done with less code, but for performance reasons it
|
|
// makes sense to have the code for two separate Parse() calls here
|
|
// since the concatenation of strings costs more than we like. And
|
|
// why pay that price when we don't need to?
|
|
if (aNewlineTerminate) {
|
|
rv = (static_cast<nsHtml5Parser*>(mParser.get()))->Parse(
|
|
aText + new_line, key, GetContentTypeInternal(), false);
|
|
} else {
|
|
rv = (static_cast<nsHtml5Parser*>(mParser.get()))->Parse(
|
|
aText, key, GetContentTypeInternal(), false);
|
|
}
|
|
|
|
--mWriteLevel;
|
|
|
|
mTooDeepWriteRecursion = (mWriteLevel != 0 && mTooDeepWriteRecursion);
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::Write(const nsAString& aText, JSContext *cx)
|
|
{
|
|
return WriteCommon(cx, aText, false);
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::Write(JSContext* cx, const Sequence<nsString>& aText,
|
|
ErrorResult& rv)
|
|
{
|
|
WriteCommon(cx, aText, false, rv);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::Writeln(const nsAString& aText, JSContext *cx)
|
|
{
|
|
return WriteCommon(cx, aText, true);
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::Writeln(JSContext* cx, const Sequence<nsString>& aText,
|
|
ErrorResult& rv)
|
|
{
|
|
WriteCommon(cx, aText, true, rv);
|
|
}
|
|
|
|
bool
|
|
nsHTMLDocument::MatchNameAttribute(nsIContent* aContent, int32_t aNamespaceID,
|
|
nsIAtom* aAtom, void* aData)
|
|
{
|
|
NS_PRECONDITION(aContent, "Must have content node to work with!");
|
|
nsString* elementName = static_cast<nsString*>(aData);
|
|
return
|
|
aContent->GetNameSpaceID() == kNameSpaceID_XHTML &&
|
|
aContent->AttrValueIs(kNameSpaceID_None, nsGkAtoms::name,
|
|
*elementName, eCaseMatters);
|
|
}
|
|
|
|
/* static */
|
|
void*
|
|
nsHTMLDocument::UseExistingNameString(nsINode* aRootNode, const nsString* aName)
|
|
{
|
|
return const_cast<nsString*>(aName);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetElementsByName(const nsAString& aElementName,
|
|
nsIDOMNodeList** aReturn)
|
|
{
|
|
*aReturn = GetElementsByName(aElementName).take();
|
|
return NS_OK;
|
|
}
|
|
|
|
static bool MatchItems(nsIContent* aContent, int32_t aNameSpaceID,
|
|
nsIAtom* aAtom, void* aData)
|
|
{
|
|
if (!aContent->IsHTMLElement()) {
|
|
return false;
|
|
}
|
|
|
|
nsGenericHTMLElement* elem = static_cast<nsGenericHTMLElement*>(aContent);
|
|
if (!elem->HasAttr(kNameSpaceID_None, nsGkAtoms::itemscope) ||
|
|
elem->HasAttr(kNameSpaceID_None, nsGkAtoms::itemprop)) {
|
|
return false;
|
|
}
|
|
|
|
nsTArray<nsCOMPtr<nsIAtom> >* tokens = static_cast<nsTArray<nsCOMPtr<nsIAtom> >*>(aData);
|
|
if (tokens->IsEmpty()) {
|
|
return true;
|
|
}
|
|
|
|
const nsAttrValue* attr = elem->GetParsedAttr(nsGkAtoms::itemtype);
|
|
if (!attr)
|
|
return false;
|
|
|
|
for (uint32_t i = 0; i < tokens->Length(); i++) {
|
|
if (!attr->Contains(tokens->ElementAt(i), eCaseMatters)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static void DestroyTokens(void* aData)
|
|
{
|
|
nsTArray<nsCOMPtr<nsIAtom> >* tokens = static_cast<nsTArray<nsCOMPtr<nsIAtom> >*>(aData);
|
|
delete tokens;
|
|
}
|
|
|
|
static void* CreateTokens(nsINode* aRootNode, const nsString* types)
|
|
{
|
|
nsTArray<nsCOMPtr<nsIAtom> >* tokens = new nsTArray<nsCOMPtr<nsIAtom> >();
|
|
nsAString::const_iterator iter, end;
|
|
types->BeginReading(iter);
|
|
types->EndReading(end);
|
|
|
|
// skip initial whitespace
|
|
while (iter != end && nsContentUtils::IsHTMLWhitespace(*iter)) {
|
|
++iter;
|
|
}
|
|
|
|
// parse the tokens
|
|
while (iter != end) {
|
|
nsAString::const_iterator start(iter);
|
|
|
|
do {
|
|
++iter;
|
|
} while (iter != end && !nsContentUtils::IsHTMLWhitespace(*iter));
|
|
|
|
tokens->AppendElement(do_GetAtom(Substring(start, iter)));
|
|
|
|
// skip whitespace
|
|
while (iter != end && nsContentUtils::IsHTMLWhitespace(*iter)) {
|
|
++iter;
|
|
}
|
|
}
|
|
return tokens;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetItems(const nsAString& types, nsIDOMNodeList** aReturn)
|
|
{
|
|
*aReturn = GetItems(types).take();
|
|
return NS_OK;
|
|
}
|
|
|
|
already_AddRefed<nsINodeList>
|
|
nsHTMLDocument::GetItems(const nsAString& aTypeNames)
|
|
{
|
|
return NS_GetFuncStringNodeList(this, MatchItems, DestroyTokens, CreateTokens,
|
|
aTypeNames);
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::AddedForm()
|
|
{
|
|
++mNumForms;
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::RemovedForm()
|
|
{
|
|
--mNumForms;
|
|
}
|
|
|
|
int32_t
|
|
nsHTMLDocument::GetNumFormsSynchronous()
|
|
{
|
|
return mNumForms;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetAlinkColor(nsAString& aAlinkColor)
|
|
{
|
|
aAlinkColor.Truncate();
|
|
|
|
HTMLBodyElement* body = GetBodyElement();
|
|
if (body) {
|
|
body->GetALink(aAlinkColor);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::SetAlinkColor(const nsAString& aAlinkColor)
|
|
{
|
|
HTMLBodyElement* body = GetBodyElement();
|
|
if (body) {
|
|
body->SetALink(aAlinkColor);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetLinkColor(nsAString& aLinkColor)
|
|
{
|
|
aLinkColor.Truncate();
|
|
|
|
HTMLBodyElement* body = GetBodyElement();
|
|
if (body) {
|
|
body->GetLink(aLinkColor);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::SetLinkColor(const nsAString& aLinkColor)
|
|
{
|
|
HTMLBodyElement* body = GetBodyElement();
|
|
if (body) {
|
|
body->SetLink(aLinkColor);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetVlinkColor(nsAString& aVlinkColor)
|
|
{
|
|
aVlinkColor.Truncate();
|
|
|
|
HTMLBodyElement* body = GetBodyElement();
|
|
if (body) {
|
|
body->GetVLink(aVlinkColor);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::SetVlinkColor(const nsAString& aVlinkColor)
|
|
{
|
|
HTMLBodyElement* body = GetBodyElement();
|
|
if (body) {
|
|
body->SetVLink(aVlinkColor);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetBgColor(nsAString& aBgColor)
|
|
{
|
|
aBgColor.Truncate();
|
|
|
|
HTMLBodyElement* body = GetBodyElement();
|
|
if (body) {
|
|
body->GetBgColor(aBgColor);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::SetBgColor(const nsAString& aBgColor)
|
|
{
|
|
HTMLBodyElement* body = GetBodyElement();
|
|
if (body) {
|
|
body->SetBgColor(aBgColor);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetFgColor(nsAString& aFgColor)
|
|
{
|
|
aFgColor.Truncate();
|
|
|
|
HTMLBodyElement* body = GetBodyElement();
|
|
if (body) {
|
|
body->GetText(aFgColor);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::SetFgColor(const nsAString& aFgColor)
|
|
{
|
|
HTMLBodyElement* body = GetBodyElement();
|
|
if (body) {
|
|
body->SetText(aFgColor);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetEmbeds(nsIDOMHTMLCollection** aEmbeds)
|
|
{
|
|
NS_ADDREF(*aEmbeds = Embeds());
|
|
return NS_OK;
|
|
}
|
|
|
|
nsIHTMLCollection*
|
|
nsHTMLDocument::Embeds()
|
|
{
|
|
if (!mEmbeds) {
|
|
mEmbeds = new nsContentList(this, kNameSpaceID_XHTML, nsGkAtoms::embed, nsGkAtoms::embed);
|
|
}
|
|
return mEmbeds;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetSelection(nsISelection** aReturn)
|
|
{
|
|
ErrorResult rv;
|
|
NS_IF_ADDREF(*aReturn = GetSelection(rv));
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
Selection*
|
|
nsHTMLDocument::GetSelection(ErrorResult& aRv)
|
|
{
|
|
nsCOMPtr<nsPIDOMWindow> window = do_QueryInterface(GetScopeObject());
|
|
if (!window) {
|
|
return nullptr;
|
|
}
|
|
|
|
NS_ASSERTION(window->IsInnerWindow(), "Should have inner window here!");
|
|
if (!window->IsCurrentInnerWindow()) {
|
|
return nullptr;
|
|
}
|
|
|
|
return static_cast<nsGlobalWindow*>(window.get())->GetSelection(aRv);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::CaptureEvents()
|
|
{
|
|
WarnOnceAbout(nsIDocument::eUseOfCaptureEvents);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::ReleaseEvents()
|
|
{
|
|
WarnOnceAbout(nsIDocument::eUseOfReleaseEvents);
|
|
return NS_OK;
|
|
}
|
|
|
|
// Mapped to document.embeds for NS4 compatibility
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetPlugins(nsIDOMHTMLCollection** aPlugins)
|
|
{
|
|
*aPlugins = nullptr;
|
|
|
|
return GetEmbeds(aPlugins);
|
|
}
|
|
|
|
nsIHTMLCollection*
|
|
nsHTMLDocument::Plugins()
|
|
{
|
|
return Embeds();
|
|
}
|
|
|
|
nsISupports*
|
|
nsHTMLDocument::ResolveName(const nsAString& aName, nsWrapperCache **aCache)
|
|
{
|
|
nsIdentifierMapEntry *entry = mIdentifierMap.GetEntry(aName);
|
|
if (!entry) {
|
|
*aCache = nullptr;
|
|
return nullptr;
|
|
}
|
|
|
|
nsBaseContentList *list = entry->GetNameContentList();
|
|
uint32_t length = list ? list->Length() : 0;
|
|
|
|
if (length > 0) {
|
|
if (length == 1) {
|
|
// Only one element in the list, return the element instead of returning
|
|
// the list.
|
|
nsIContent *node = list->Item(0);
|
|
*aCache = node;
|
|
return node;
|
|
}
|
|
|
|
// The list contains more than one element, return the whole list.
|
|
*aCache = list;
|
|
return list;
|
|
}
|
|
|
|
// No named items were found, see if there's one registerd by id for aName.
|
|
Element *e = entry->GetIdElement();
|
|
|
|
if (e && nsGenericHTMLElement::ShouldExposeIdAsHTMLDocumentProperty(e)) {
|
|
*aCache = e;
|
|
return e;
|
|
}
|
|
|
|
*aCache = nullptr;
|
|
return nullptr;
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::NamedGetter(JSContext* cx, const nsAString& aName, bool& aFound,
|
|
JS::MutableHandle<JSObject*> aRetval,
|
|
ErrorResult& rv)
|
|
{
|
|
nsWrapperCache* cache;
|
|
nsISupports* supp = ResolveName(aName, &cache);
|
|
if (!supp) {
|
|
aFound = false;
|
|
aRetval.set(nullptr);
|
|
return;
|
|
}
|
|
|
|
JS::Rooted<JS::Value> val(cx);
|
|
// XXXbz Should we call the (slightly misnamed, really) WrapNativeParent
|
|
// here?
|
|
if (!dom::WrapObject(cx, supp, cache, nullptr, &val)) {
|
|
rv.Throw(NS_ERROR_OUT_OF_MEMORY);
|
|
return;
|
|
}
|
|
aFound = true;
|
|
aRetval.set(&val.toObject());
|
|
}
|
|
|
|
bool
|
|
nsHTMLDocument::NameIsEnumerable(const nsAString& aName)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
static PLDHashOperator
|
|
IdentifierMapEntryAddNames(nsIdentifierMapEntry* aEntry, void* aArg)
|
|
{
|
|
nsTArray<nsString>* aNames = static_cast<nsTArray<nsString>*>(aArg);
|
|
if (aEntry->HasNameElement() ||
|
|
aEntry->HasIdElementExposedAsHTMLDocumentProperty()) {
|
|
aNames->AppendElement(aEntry->GetKey());
|
|
}
|
|
return PL_DHASH_NEXT;
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::GetSupportedNames(unsigned, nsTArray<nsString>& aNames)
|
|
{
|
|
mIdentifierMap.EnumerateEntries(IdentifierMapEntryAddNames, &aNames);
|
|
}
|
|
|
|
//----------------------------
|
|
|
|
// forms related stuff
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetForms(nsIDOMHTMLCollection** aForms)
|
|
{
|
|
NS_ADDREF(*aForms = nsHTMLDocument::GetForms());
|
|
return NS_OK;
|
|
}
|
|
|
|
nsContentList*
|
|
nsHTMLDocument::GetForms()
|
|
{
|
|
if (!mForms) {
|
|
mForms = new nsContentList(this, kNameSpaceID_XHTML, nsGkAtoms::form, nsGkAtoms::form);
|
|
}
|
|
|
|
return mForms;
|
|
}
|
|
|
|
static bool MatchFormControls(nsIContent* aContent, int32_t aNamespaceID,
|
|
nsIAtom* aAtom, void* aData)
|
|
{
|
|
return aContent->IsNodeOfType(nsIContent::eHTML_FORM_CONTROL);
|
|
}
|
|
|
|
nsContentList*
|
|
nsHTMLDocument::GetFormControls()
|
|
{
|
|
if (!mFormControls) {
|
|
mFormControls = new nsContentList(this, MatchFormControls, nullptr, nullptr);
|
|
}
|
|
|
|
return mFormControls;
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLDocument::CreateAndAddWyciwygChannel(void)
|
|
{
|
|
nsresult rv = NS_OK;
|
|
nsAutoCString url, originalSpec;
|
|
|
|
mDocumentURI->GetSpec(originalSpec);
|
|
|
|
// Generate the wyciwyg url
|
|
url = NS_LITERAL_CSTRING("wyciwyg://")
|
|
+ nsPrintfCString("%d", gWyciwygSessionCnt++)
|
|
+ NS_LITERAL_CSTRING("/")
|
|
+ originalSpec;
|
|
|
|
nsCOMPtr<nsIURI> wcwgURI;
|
|
NS_NewURI(getter_AddRefs(wcwgURI), url);
|
|
|
|
// Create the nsIWyciwygChannel to store out-of-band
|
|
// document.write() script to cache
|
|
nsCOMPtr<nsIChannel> channel;
|
|
// Create a wyciwyg Channel
|
|
rv = NS_NewChannel(getter_AddRefs(channel),
|
|
wcwgURI,
|
|
NodePrincipal(),
|
|
nsILoadInfo::SEC_FORCE_INHERIT_PRINCIPAL,
|
|
nsIContentPolicy::TYPE_OTHER);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
mWyciwygChannel = do_QueryInterface(channel);
|
|
|
|
mWyciwygChannel->SetSecurityInfo(mSecurityInfo);
|
|
|
|
// Note: we want to treat this like a "previous document" hint so that,
|
|
// e.g. a <meta> tag in the document.write content can override it.
|
|
SetDocumentCharacterSetSource(kCharsetFromHintPrevDoc);
|
|
mWyciwygChannel->SetCharsetAndSource(kCharsetFromHintPrevDoc,
|
|
GetDocumentCharacterSet());
|
|
|
|
// Inherit load flags from the original document's channel
|
|
channel->SetLoadFlags(mLoadFlags);
|
|
|
|
nsCOMPtr<nsILoadGroup> loadGroup = GetDocumentLoadGroup();
|
|
|
|
// Use the Parent document's loadgroup to trigger load notifications
|
|
if (loadGroup && channel) {
|
|
rv = channel->SetLoadGroup(loadGroup);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
nsLoadFlags loadFlags = 0;
|
|
channel->GetLoadFlags(&loadFlags);
|
|
loadFlags |= nsIChannel::LOAD_DOCUMENT_URI;
|
|
channel->SetLoadFlags(loadFlags);
|
|
|
|
channel->SetOriginalURI(wcwgURI);
|
|
|
|
rv = loadGroup->AddRequest(mWyciwygChannel, nullptr);
|
|
NS_ASSERTION(NS_SUCCEEDED(rv), "Failed to add request to load group.");
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLDocument::RemoveWyciwygChannel(void)
|
|
{
|
|
nsCOMPtr<nsILoadGroup> loadGroup = GetDocumentLoadGroup();
|
|
|
|
// note there can be a write request without a load group if
|
|
// this is a synchronously constructed about:blank document
|
|
if (loadGroup && mWyciwygChannel) {
|
|
mWyciwygChannel->CloseCacheEntry(NS_OK);
|
|
loadGroup->RemoveRequest(mWyciwygChannel, nullptr, NS_OK);
|
|
}
|
|
|
|
mWyciwygChannel = nullptr;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void *
|
|
nsHTMLDocument::GenerateParserKey(void)
|
|
{
|
|
if (!mScriptLoader) {
|
|
// If we don't have a script loader, then the parser probably isn't parsing
|
|
// anything anyway, so just return null.
|
|
return nullptr;
|
|
}
|
|
|
|
// The script loader provides us with the currently executing script element,
|
|
// which is guaranteed to be unique per script.
|
|
nsIScriptElement* script = mScriptLoader->GetCurrentParserInsertedScript();
|
|
if (script && mParser && mParser->IsScriptCreated()) {
|
|
nsCOMPtr<nsIParser> creatorParser = script->GetCreatorParser();
|
|
if (creatorParser != mParser) {
|
|
// Make scripts that aren't inserted by the active parser of this document
|
|
// participate in the context of the script that document.open()ed
|
|
// this document.
|
|
return nullptr;
|
|
}
|
|
}
|
|
return script;
|
|
}
|
|
|
|
/* attribute DOMString designMode; */
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::GetDesignMode(nsAString & aDesignMode)
|
|
{
|
|
if (HasFlag(NODE_IS_EDITABLE)) {
|
|
aDesignMode.AssignLiteral("on");
|
|
}
|
|
else {
|
|
aDesignMode.AssignLiteral("off");
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::MaybeEditingStateChanged()
|
|
{
|
|
if (!mPendingMaybeEditingStateChanged &&
|
|
mUpdateNestLevel == 0 && (mContentEditableCount > 0) != IsEditingOn()) {
|
|
if (nsContentUtils::IsSafeToRunScript()) {
|
|
EditingStateChanged();
|
|
} else if (!mInDestructor) {
|
|
nsContentUtils::AddScriptRunner(
|
|
NS_NewRunnableMethod(this, &nsHTMLDocument::MaybeEditingStateChanged));
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::EndUpdate(nsUpdateType aUpdateType)
|
|
{
|
|
const bool reset = !mPendingMaybeEditingStateChanged;
|
|
mPendingMaybeEditingStateChanged = true;
|
|
nsDocument::EndUpdate(aUpdateType);
|
|
if (reset) {
|
|
mPendingMaybeEditingStateChanged = false;
|
|
}
|
|
MaybeEditingStateChanged();
|
|
}
|
|
|
|
|
|
// Helper class, used below in ChangeContentEditableCount().
|
|
class DeferredContentEditableCountChangeEvent : public nsRunnable
|
|
{
|
|
public:
|
|
DeferredContentEditableCountChangeEvent(nsHTMLDocument *aDoc,
|
|
nsIContent *aElement)
|
|
: mDoc(aDoc)
|
|
, mElement(aElement)
|
|
{
|
|
}
|
|
|
|
NS_IMETHOD Run() {
|
|
if (mElement && mElement->OwnerDoc() == mDoc) {
|
|
mDoc->DeferredContentEditableCountChange(mElement);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
nsRefPtr<nsHTMLDocument> mDoc;
|
|
nsCOMPtr<nsIContent> mElement;
|
|
};
|
|
|
|
nsresult
|
|
nsHTMLDocument::ChangeContentEditableCount(nsIContent *aElement,
|
|
int32_t aChange)
|
|
{
|
|
NS_ASSERTION(int32_t(mContentEditableCount) + aChange >= 0,
|
|
"Trying to decrement too much.");
|
|
|
|
mContentEditableCount += aChange;
|
|
|
|
nsContentUtils::AddScriptRunner(
|
|
new DeferredContentEditableCountChangeEvent(this, aElement));
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::DeferredContentEditableCountChange(nsIContent *aElement)
|
|
{
|
|
if (mParser ||
|
|
(mUpdateNestLevel > 0 && (mContentEditableCount > 0) != IsEditingOn())) {
|
|
return;
|
|
}
|
|
|
|
EditingState oldState = mEditingState;
|
|
|
|
nsresult rv = EditingStateChanged();
|
|
NS_ENSURE_SUCCESS_VOID(rv);
|
|
|
|
if (oldState == mEditingState && mEditingState == eContentEditable) {
|
|
// We just changed the contentEditable state of a node, we need to reset
|
|
// the spellchecking state of that node.
|
|
nsCOMPtr<nsIDOMNode> node = do_QueryInterface(aElement);
|
|
if (node) {
|
|
nsPIDOMWindow *window = GetWindow();
|
|
if (!window)
|
|
return;
|
|
|
|
nsIDocShell *docshell = window->GetDocShell();
|
|
if (!docshell)
|
|
return;
|
|
|
|
nsCOMPtr<nsIEditor> editor;
|
|
docshell->GetEditor(getter_AddRefs(editor));
|
|
if (editor) {
|
|
nsRefPtr<nsRange> range = new nsRange(aElement);
|
|
rv = range->SelectNode(node);
|
|
if (NS_FAILED(rv)) {
|
|
// The node might be detached from the document at this point,
|
|
// which would cause this call to fail. In this case, we can
|
|
// safely ignore the contenteditable count change.
|
|
return;
|
|
}
|
|
|
|
nsCOMPtr<nsIInlineSpellChecker> spellChecker;
|
|
rv = editor->GetInlineSpellChecker(false,
|
|
getter_AddRefs(spellChecker));
|
|
NS_ENSURE_SUCCESS_VOID(rv);
|
|
|
|
if (spellChecker) {
|
|
rv = spellChecker->SpellCheckRange(range);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
HTMLAllCollection*
|
|
nsHTMLDocument::All()
|
|
{
|
|
if (!mAll) {
|
|
mAll = new HTMLAllCollection(this);
|
|
}
|
|
return mAll;
|
|
}
|
|
|
|
static void
|
|
NotifyEditableStateChange(nsINode *aNode, nsIDocument *aDocument)
|
|
{
|
|
for (nsIContent* child = aNode->GetFirstChild();
|
|
child;
|
|
child = child->GetNextSibling()) {
|
|
if (child->IsElement()) {
|
|
child->AsElement()->UpdateState(true);
|
|
}
|
|
NotifyEditableStateChange(child, aDocument);
|
|
}
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::TearingDownEditor(nsIEditor *aEditor)
|
|
{
|
|
if (IsEditingOn()) {
|
|
EditingState oldState = mEditingState;
|
|
mEditingState = eTearingDown;
|
|
|
|
nsCOMPtr<nsIPresShell> presShell = GetShell();
|
|
if (!presShell)
|
|
return;
|
|
|
|
nsCOMArray<nsIStyleSheet> agentSheets;
|
|
presShell->GetAgentStyleSheets(agentSheets);
|
|
|
|
agentSheets.RemoveObject(nsLayoutStylesheetCache::ContentEditableSheet());
|
|
if (oldState == eDesignMode)
|
|
agentSheets.RemoveObject(nsLayoutStylesheetCache::DesignModeSheet());
|
|
|
|
presShell->SetAgentStyleSheets(agentSheets);
|
|
|
|
presShell->ReconstructStyleData();
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLDocument::TurnEditingOff()
|
|
{
|
|
NS_ASSERTION(mEditingState != eOff, "Editing is already off.");
|
|
|
|
nsPIDOMWindow *window = GetWindow();
|
|
if (!window)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
nsIDocShell *docshell = window->GetDocShell();
|
|
if (!docshell)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
nsresult rv;
|
|
nsCOMPtr<nsIEditingSession> editSession = do_GetInterface(docshell, &rv);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// turn editing off
|
|
rv = editSession->TearDownEditorOnWindow(window);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
mEditingState = eOff;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
static bool HasPresShell(nsPIDOMWindow *aWindow)
|
|
{
|
|
nsIDocShell *docShell = aWindow->GetDocShell();
|
|
if (!docShell)
|
|
return false;
|
|
return docShell->GetPresShell() != nullptr;
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLDocument::SetEditingState(EditingState aState)
|
|
{
|
|
mEditingState = aState;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLDocument::EditingStateChanged()
|
|
{
|
|
if (mRemovedFromDocShell) {
|
|
return NS_OK;
|
|
}
|
|
|
|
if (mEditingState == eSettingUp || mEditingState == eTearingDown) {
|
|
// XXX We shouldn't recurse
|
|
return NS_OK;
|
|
}
|
|
|
|
bool designMode = HasFlag(NODE_IS_EDITABLE);
|
|
EditingState newState = designMode ? eDesignMode :
|
|
(mContentEditableCount > 0 ? eContentEditable : eOff);
|
|
if (mEditingState == newState) {
|
|
// No changes in editing mode.
|
|
return NS_OK;
|
|
}
|
|
|
|
if (newState == eOff) {
|
|
// Editing is being turned off.
|
|
nsAutoScriptBlocker scriptBlocker;
|
|
NotifyEditableStateChange(this, this);
|
|
return TurnEditingOff();
|
|
}
|
|
|
|
// Flush out style changes on our _parent_ document, if any, so that
|
|
// our check for a presshell won't get stale information.
|
|
if (mParentDocument) {
|
|
mParentDocument->FlushPendingNotifications(Flush_Style);
|
|
}
|
|
|
|
// get editing session, make sure this is a strong reference so the
|
|
// window can't get deleted during the rest of this call.
|
|
nsCOMPtr<nsPIDOMWindow> window = GetWindow();
|
|
if (!window)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
nsIDocShell *docshell = window->GetDocShell();
|
|
if (!docshell)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
nsresult rv;
|
|
nsCOMPtr<nsIEditingSession> editSession = do_GetInterface(docshell, &rv);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
nsCOMPtr<nsIEditor> existingEditor;
|
|
editSession->GetEditorForWindow(window, getter_AddRefs(existingEditor));
|
|
if (existingEditor) {
|
|
// We might already have an editor if it was set up for mail, let's see
|
|
// if this is actually the case.
|
|
nsCOMPtr<nsIHTMLEditor> htmlEditor = do_QueryInterface(existingEditor);
|
|
MOZ_ASSERT(htmlEditor, "If we have an editor, it must be an HTML editor");
|
|
uint32_t flags = 0;
|
|
existingEditor->GetFlags(&flags);
|
|
if (flags & nsIPlaintextEditor::eEditorMailMask) {
|
|
// We already have a mail editor, then we should not attempt to create
|
|
// another one.
|
|
return NS_OK;
|
|
}
|
|
}
|
|
|
|
if (!HasPresShell(window)) {
|
|
// We should not make the window editable or setup its editor.
|
|
// It's probably style=display:none.
|
|
return NS_OK;
|
|
}
|
|
|
|
bool makeWindowEditable = mEditingState == eOff;
|
|
bool updateState = false;
|
|
bool spellRecheckAll = false;
|
|
bool putOffToRemoveScriptBlockerUntilModifyingEditingState = false;
|
|
nsCOMPtr<nsIEditor> editor;
|
|
|
|
{
|
|
EditingState oldState = mEditingState;
|
|
nsAutoEditingState push(this, eSettingUp);
|
|
|
|
nsCOMPtr<nsIPresShell> presShell = GetShell();
|
|
NS_ENSURE_TRUE(presShell, NS_ERROR_FAILURE);
|
|
|
|
// Before making this window editable, we need to modify UA style sheet
|
|
// because new style may change whether focused element will be focusable
|
|
// or not.
|
|
nsCOMArray<nsIStyleSheet> agentSheets;
|
|
rv = presShell->GetAgentStyleSheets(agentSheets);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
CSSStyleSheet* contentEditableSheet =
|
|
nsLayoutStylesheetCache::ContentEditableSheet();
|
|
|
|
bool result;
|
|
|
|
if (!agentSheets.Contains(contentEditableSheet)) {
|
|
bool result = agentSheets.AppendObject(contentEditableSheet);
|
|
NS_ENSURE_TRUE(result, NS_ERROR_OUT_OF_MEMORY);
|
|
}
|
|
|
|
// Should we update the editable state of all the nodes in the document? We
|
|
// need to do this when the designMode value changes, as that overrides
|
|
// specific states on the elements.
|
|
if (designMode) {
|
|
// designMode is being turned on (overrides contentEditable).
|
|
CSSStyleSheet* designModeSheet =
|
|
nsLayoutStylesheetCache::DesignModeSheet();
|
|
if (!agentSheets.Contains(designModeSheet)) {
|
|
result = agentSheets.AppendObject(designModeSheet);
|
|
NS_ENSURE_TRUE(result, NS_ERROR_OUT_OF_MEMORY);
|
|
}
|
|
|
|
updateState = true;
|
|
spellRecheckAll = oldState == eContentEditable;
|
|
}
|
|
else if (oldState == eDesignMode) {
|
|
// designMode is being turned off (contentEditable is still on).
|
|
agentSheets.RemoveObject(nsLayoutStylesheetCache::DesignModeSheet());
|
|
updateState = true;
|
|
}
|
|
|
|
rv = presShell->SetAgentStyleSheets(agentSheets);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
presShell->ReconstructStyleData();
|
|
|
|
// Adjust focused element with new style but blur event shouldn't be fired
|
|
// until mEditingState is modified with newState.
|
|
nsAutoScriptBlocker scriptBlocker;
|
|
if (designMode) {
|
|
nsCOMPtr<nsPIDOMWindow> focusedWindow;
|
|
nsIContent* focusedContent =
|
|
nsFocusManager::GetFocusedDescendant(window, false,
|
|
getter_AddRefs(focusedWindow));
|
|
if (focusedContent) {
|
|
nsIFrame* focusedFrame = focusedContent->GetPrimaryFrame();
|
|
bool clearFocus = focusedFrame ? !focusedFrame->IsFocusable() :
|
|
!focusedContent->IsFocusable();
|
|
if (clearFocus) {
|
|
nsFocusManager* fm = nsFocusManager::GetFocusManager();
|
|
if (fm) {
|
|
fm->ClearFocus(window);
|
|
// If we need to dispatch blur event, we should put off after
|
|
// modifying mEditingState since blur event handler may change
|
|
// designMode state again.
|
|
putOffToRemoveScriptBlockerUntilModifyingEditingState = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (makeWindowEditable) {
|
|
// Editing is being turned on (through designMode or contentEditable)
|
|
// Turn on editor.
|
|
// XXX This can cause flushing which can change the editing state, so make
|
|
// sure to avoid recursing.
|
|
rv = editSession->MakeWindowEditable(window, "html", false, false,
|
|
true);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
|
|
// XXX Need to call TearDownEditorOnWindow for all failures.
|
|
docshell->GetEditor(getter_AddRefs(editor));
|
|
if (!editor)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
// If we're entering the design mode, put the selection at the beginning of
|
|
// the document for compatibility reasons.
|
|
if (designMode && oldState == eOff) {
|
|
editor->BeginningOfDocument();
|
|
}
|
|
|
|
if (putOffToRemoveScriptBlockerUntilModifyingEditingState) {
|
|
nsContentUtils::AddScriptBlocker();
|
|
}
|
|
}
|
|
|
|
mEditingState = newState;
|
|
if (putOffToRemoveScriptBlockerUntilModifyingEditingState) {
|
|
nsContentUtils::RemoveScriptBlocker();
|
|
// If mEditingState is overwritten by another call and already disabled
|
|
// the editing, we shouldn't keep making window editable.
|
|
if (mEditingState == eOff) {
|
|
return NS_OK;
|
|
}
|
|
}
|
|
|
|
if (makeWindowEditable) {
|
|
// Set the editor to not insert br's on return when in p
|
|
// elements by default.
|
|
// XXX Do we only want to do this for designMode?
|
|
bool unused;
|
|
rv = ExecCommand(NS_LITERAL_STRING("insertBrOnReturn"), false,
|
|
NS_LITERAL_STRING("false"), &unused);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
// Editor setup failed. Editing is not on after all.
|
|
// XXX Should we reset the editable flag on nodes?
|
|
editSession->TearDownEditorOnWindow(window);
|
|
mEditingState = eOff;
|
|
|
|
return rv;
|
|
}
|
|
}
|
|
|
|
if (updateState) {
|
|
nsAutoScriptBlocker scriptBlocker;
|
|
NotifyEditableStateChange(this, this);
|
|
}
|
|
|
|
// Resync the editor's spellcheck state.
|
|
if (spellRecheckAll) {
|
|
nsCOMPtr<nsISelectionController> selcon;
|
|
nsresult rv = editor->GetSelectionController(getter_AddRefs(selcon));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
nsCOMPtr<nsISelection> spellCheckSelection;
|
|
rv = selcon->GetSelection(nsISelectionController::SELECTION_SPELLCHECK,
|
|
getter_AddRefs(spellCheckSelection));
|
|
if (NS_SUCCEEDED(rv)) {
|
|
spellCheckSelection->RemoveAllRanges();
|
|
}
|
|
}
|
|
editor->SyncRealTimeSpell();
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::SetDesignMode(const nsAString & aDesignMode)
|
|
{
|
|
ErrorResult rv;
|
|
SetDesignMode(aDesignMode, rv);
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::SetDesignMode(const nsAString& aDesignMode, ErrorResult& rv)
|
|
{
|
|
if (!nsContentUtils::SubjectPrincipal()->Subsumes(NodePrincipal())) {
|
|
rv.Throw(NS_ERROR_DOM_PROP_ACCESS_DENIED);
|
|
return;
|
|
}
|
|
bool editableMode = HasFlag(NODE_IS_EDITABLE);
|
|
if (aDesignMode.LowerCaseEqualsASCII(editableMode ? "off" : "on")) {
|
|
SetEditableFlag(!editableMode);
|
|
|
|
rv = EditingStateChanged();
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLDocument::GetMidasCommandManager(nsICommandManager** aCmdMgr)
|
|
{
|
|
// initialize return value
|
|
NS_ENSURE_ARG_POINTER(aCmdMgr);
|
|
|
|
// check if we have it cached
|
|
if (mMidasCommandManager) {
|
|
NS_ADDREF(*aCmdMgr = mMidasCommandManager);
|
|
return NS_OK;
|
|
}
|
|
|
|
*aCmdMgr = nullptr;
|
|
|
|
nsPIDOMWindow *window = GetWindow();
|
|
if (!window)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
nsIDocShell *docshell = window->GetDocShell();
|
|
if (!docshell)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
mMidasCommandManager = do_GetInterface(docshell);
|
|
if (!mMidasCommandManager)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
NS_ADDREF(*aCmdMgr = mMidasCommandManager);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
struct MidasCommand {
|
|
const char* incomingCommandString;
|
|
const char* internalCommandString;
|
|
const char* internalParamString;
|
|
bool useNewParam;
|
|
bool convertToBoolean;
|
|
};
|
|
|
|
static const struct MidasCommand gMidasCommandTable[] = {
|
|
{ "bold", "cmd_bold", "", true, false },
|
|
{ "italic", "cmd_italic", "", true, false },
|
|
{ "underline", "cmd_underline", "", true, false },
|
|
{ "strikethrough", "cmd_strikethrough", "", true, false },
|
|
{ "subscript", "cmd_subscript", "", true, false },
|
|
{ "superscript", "cmd_superscript", "", true, false },
|
|
{ "cut", "cmd_cut", "", true, false },
|
|
{ "copy", "cmd_copy", "", true, false },
|
|
{ "paste", "cmd_paste", "", true, false },
|
|
{ "delete", "cmd_deleteCharBackward", "", true, false },
|
|
{ "forwarddelete", "cmd_deleteCharForward", "", true, false },
|
|
{ "selectall", "cmd_selectAll", "", true, false },
|
|
{ "undo", "cmd_undo", "", true, false },
|
|
{ "redo", "cmd_redo", "", true, false },
|
|
{ "indent", "cmd_indent", "", true, false },
|
|
{ "outdent", "cmd_outdent", "", true, false },
|
|
{ "backcolor", "cmd_highlight", "", false, false },
|
|
{ "forecolor", "cmd_fontColor", "", false, false },
|
|
{ "hilitecolor", "cmd_highlight", "", false, false },
|
|
{ "fontname", "cmd_fontFace", "", false, false },
|
|
{ "fontsize", "cmd_fontSize", "", false, false },
|
|
{ "increasefontsize", "cmd_increaseFont", "", false, false },
|
|
{ "decreasefontsize", "cmd_decreaseFont", "", false, false },
|
|
{ "inserthorizontalrule", "cmd_insertHR", "", true, false },
|
|
{ "createlink", "cmd_insertLinkNoUI", "", false, false },
|
|
{ "insertimage", "cmd_insertImageNoUI", "", false, false },
|
|
{ "inserthtml", "cmd_insertHTML", "", false, false },
|
|
{ "inserttext", "cmd_insertText", "", false, false },
|
|
{ "gethtml", "cmd_getContents", "", false, false },
|
|
{ "justifyleft", "cmd_align", "left", true, false },
|
|
{ "justifyright", "cmd_align", "right", true, false },
|
|
{ "justifycenter", "cmd_align", "center", true, false },
|
|
{ "justifyfull", "cmd_align", "justify", true, false },
|
|
{ "removeformat", "cmd_removeStyles", "", true, false },
|
|
{ "unlink", "cmd_removeLinks", "", true, false },
|
|
{ "insertorderedlist", "cmd_ol", "", true, false },
|
|
{ "insertunorderedlist", "cmd_ul", "", true, false },
|
|
{ "insertparagraph", "cmd_paragraphState", "p", true, false },
|
|
{ "formatblock", "cmd_paragraphState", "", false, false },
|
|
{ "heading", "cmd_paragraphState", "", false, false },
|
|
{ "styleWithCSS", "cmd_setDocumentUseCSS", "", false, true },
|
|
{ "contentReadOnly", "cmd_setDocumentReadOnly", "", false, true },
|
|
{ "insertBrOnReturn", "cmd_insertBrOnReturn", "", false, true },
|
|
{ "enableObjectResizing", "cmd_enableObjectResizing", "", false, true },
|
|
{ "enableInlineTableEditing", "cmd_enableInlineTableEditing", "", false, true },
|
|
#if 0
|
|
// no editor support to remove alignments right now
|
|
{ "justifynone", "cmd_align", "", true, false },
|
|
|
|
// the following will need special review before being turned on
|
|
{ "saveas", "cmd_saveAs", "", true, false },
|
|
{ "print", "cmd_print", "", true, false },
|
|
#endif
|
|
{ nullptr, nullptr, nullptr, false, false }
|
|
};
|
|
|
|
#define MidasCommandCount ((sizeof(gMidasCommandTable) / sizeof(struct MidasCommand)) - 1)
|
|
|
|
static const char* const gBlocks[] = {
|
|
"ADDRESS",
|
|
"BLOCKQUOTE",
|
|
"DD",
|
|
"DIV",
|
|
"DL",
|
|
"DT",
|
|
"H1",
|
|
"H2",
|
|
"H3",
|
|
"H4",
|
|
"H5",
|
|
"H6",
|
|
"P",
|
|
"PRE"
|
|
};
|
|
|
|
static bool
|
|
ConvertToMidasInternalCommandInner(const nsAString& inCommandID,
|
|
const nsAString& inParam,
|
|
nsACString& outCommandID,
|
|
nsACString& outParam,
|
|
bool& outIsBoolean,
|
|
bool& outBooleanValue,
|
|
bool aIgnoreParams)
|
|
{
|
|
NS_ConvertUTF16toUTF8 convertedCommandID(inCommandID);
|
|
|
|
// Hack to support old boolean commands that were backwards (see bug 301490).
|
|
bool invertBool = false;
|
|
if (convertedCommandID.LowerCaseEqualsLiteral("usecss")) {
|
|
convertedCommandID.AssignLiteral("styleWithCSS");
|
|
invertBool = true;
|
|
} else if (convertedCommandID.LowerCaseEqualsLiteral("readonly")) {
|
|
convertedCommandID.AssignLiteral("contentReadOnly");
|
|
invertBool = true;
|
|
}
|
|
|
|
uint32_t i;
|
|
bool found = false;
|
|
for (i = 0; i < MidasCommandCount; ++i) {
|
|
if (convertedCommandID.Equals(gMidasCommandTable[i].incomingCommandString,
|
|
nsCaseInsensitiveCStringComparator())) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found) {
|
|
// reset results if the command is not found in our table
|
|
outCommandID.SetLength(0);
|
|
outParam.SetLength(0);
|
|
outIsBoolean = false;
|
|
return false;
|
|
}
|
|
|
|
// set outCommandID (what we use internally)
|
|
outCommandID.Assign(gMidasCommandTable[i].internalCommandString);
|
|
|
|
// set outParam & outIsBoolean based on flags from the table
|
|
outIsBoolean = gMidasCommandTable[i].convertToBoolean;
|
|
|
|
if (aIgnoreParams) {
|
|
// No further work to do
|
|
return true;
|
|
}
|
|
|
|
if (gMidasCommandTable[i].useNewParam) {
|
|
// Just have to copy it, no checking
|
|
outParam.Assign(gMidasCommandTable[i].internalParamString);
|
|
return true;
|
|
}
|
|
|
|
// handle checking of param passed in
|
|
if (outIsBoolean) {
|
|
// If this is a boolean value and it's not explicitly false (e.g. no value)
|
|
// we default to "true". For old backwards commands we invert the check (see
|
|
// bug 301490).
|
|
if (invertBool) {
|
|
outBooleanValue = inParam.LowerCaseEqualsLiteral("false");
|
|
} else {
|
|
outBooleanValue = !inParam.LowerCaseEqualsLiteral("false");
|
|
}
|
|
outParam.Truncate();
|
|
|
|
return true;
|
|
}
|
|
|
|
// String parameter -- see if we need to convert it (necessary for
|
|
// cmd_paragraphState and cmd_fontSize)
|
|
if (outCommandID.EqualsLiteral("cmd_paragraphState")) {
|
|
const char16_t* start = inParam.BeginReading();
|
|
const char16_t* end = inParam.EndReading();
|
|
if (start != end && *start == '<' && *(end - 1) == '>') {
|
|
++start;
|
|
--end;
|
|
}
|
|
|
|
NS_ConvertUTF16toUTF8 convertedParam(Substring(start, end));
|
|
uint32_t j;
|
|
for (j = 0; j < ArrayLength(gBlocks); ++j) {
|
|
if (convertedParam.Equals(gBlocks[j],
|
|
nsCaseInsensitiveCStringComparator())) {
|
|
outParam.Assign(gBlocks[j]);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (j == ArrayLength(gBlocks)) {
|
|
outParam.Truncate();
|
|
}
|
|
} else if (outCommandID.EqualsLiteral("cmd_fontSize")) {
|
|
// Per editing spec as of April 23, 2012, we need to reject the value if
|
|
// it's not a valid floating-point number surrounded by optional whitespace.
|
|
// Otherwise, we parse it as a legacy font size. For now, we just parse as
|
|
// a legacy font size regardless (matching WebKit) -- bug 747879.
|
|
outParam.Truncate();
|
|
int32_t size = nsContentUtils::ParseLegacyFontSize(inParam);
|
|
if (size) {
|
|
outParam.AppendInt(size);
|
|
}
|
|
} else {
|
|
CopyUTF16toUTF8(inParam, outParam);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool
|
|
ConvertToMidasInternalCommand(const nsAString & inCommandID,
|
|
const nsAString & inParam,
|
|
nsACString& outCommandID,
|
|
nsACString& outParam,
|
|
bool& outIsBoolean,
|
|
bool& outBooleanValue)
|
|
{
|
|
return ConvertToMidasInternalCommandInner(inCommandID, inParam, outCommandID,
|
|
outParam, outIsBoolean,
|
|
outBooleanValue, false);
|
|
}
|
|
|
|
static bool
|
|
ConvertToMidasInternalCommand(const nsAString & inCommandID,
|
|
nsACString& outCommandID)
|
|
{
|
|
nsAutoCString dummyCString;
|
|
nsAutoString dummyString;
|
|
bool dummyBool;
|
|
return ConvertToMidasInternalCommandInner(inCommandID, dummyString,
|
|
outCommandID, dummyCString,
|
|
dummyBool, dummyBool, true);
|
|
}
|
|
|
|
/* TODO: don't let this call do anything if the page is not done loading */
|
|
/* boolean execCommand(in DOMString commandID, in boolean doShowUI,
|
|
in DOMString value); */
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::ExecCommand(const nsAString& commandID,
|
|
bool doShowUI,
|
|
const nsAString& value,
|
|
bool* _retval)
|
|
{
|
|
ErrorResult rv;
|
|
*_retval = ExecCommand(commandID, doShowUI, value, rv);
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
bool
|
|
nsHTMLDocument::ExecCommand(const nsAString& commandID,
|
|
bool doShowUI,
|
|
const nsAString& value,
|
|
ErrorResult& rv)
|
|
{
|
|
// for optional parameters see dom/src/base/nsHistory.cpp: HistoryImpl::Go()
|
|
// this might add some ugly JS dependencies?
|
|
|
|
nsAutoCString cmdToDispatch, paramStr;
|
|
bool isBool, boolVal;
|
|
if (!ConvertToMidasInternalCommand(commandID, value,
|
|
cmdToDispatch, paramStr,
|
|
isBool, boolVal)) {
|
|
return false;
|
|
}
|
|
|
|
bool isCutCopy = (commandID.LowerCaseEqualsLiteral("cut") ||
|
|
commandID.LowerCaseEqualsLiteral("copy"));
|
|
|
|
// if editing is not on, bail
|
|
if (!isCutCopy && !IsEditingOnAfterFlush()) {
|
|
return false;
|
|
}
|
|
|
|
// if they are requesting UI from us, let's fail since we have no UI
|
|
if (doShowUI) {
|
|
return false;
|
|
}
|
|
|
|
// special case for cut & copy
|
|
// cut & copy are allowed in non editable documents
|
|
if (isCutCopy) {
|
|
if (!nsContentUtils::IsCutCopyAllowed()) {
|
|
return false;
|
|
}
|
|
|
|
// For cut & copy commands, we need the behaviour from nsWindowRoot::GetControllers
|
|
// which is to look at the focused element, and defer to a focused textbox's controller
|
|
// The code past taken by other commands in ExecCommand always uses the window directly,
|
|
// rather than deferring to the textbox, which is desireable for most editor commands,
|
|
// but not 'cut' and 'copy' (as those should allow copying out of embedded editors).
|
|
// This behaviour is invoked if we call DoCommand directly on the docShell.
|
|
nsCOMPtr<nsIDocShell> docShell(mDocumentContainer);
|
|
if (docShell) {
|
|
nsresult res = docShell->DoCommand(cmdToDispatch.get());
|
|
return NS_SUCCEEDED(res);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
if (commandID.LowerCaseEqualsLiteral("gethtml")) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return false;
|
|
}
|
|
|
|
bool restricted = commandID.LowerCaseEqualsLiteral("paste");
|
|
if (restricted && !nsContentUtils::IsCallerChrome()) {
|
|
return false;
|
|
}
|
|
|
|
// get command manager and dispatch command to our window if it's acceptable
|
|
nsCOMPtr<nsICommandManager> cmdMgr;
|
|
GetMidasCommandManager(getter_AddRefs(cmdMgr));
|
|
if (!cmdMgr) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return false;
|
|
}
|
|
|
|
nsIDOMWindow* window = GetWindow();
|
|
if (!window) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return false;
|
|
}
|
|
|
|
if ((cmdToDispatch.EqualsLiteral("cmd_fontSize") ||
|
|
cmdToDispatch.EqualsLiteral("cmd_insertImageNoUI") ||
|
|
cmdToDispatch.EqualsLiteral("cmd_insertLinkNoUI") ||
|
|
cmdToDispatch.EqualsLiteral("cmd_paragraphState")) &&
|
|
paramStr.IsEmpty()) {
|
|
// Invalid value, return false
|
|
return false;
|
|
}
|
|
|
|
// Return false for disabled commands (bug 760052)
|
|
bool enabled = false;
|
|
cmdMgr->IsCommandEnabled(cmdToDispatch.get(), window, &enabled);
|
|
if (!enabled) {
|
|
return false;
|
|
}
|
|
|
|
if (!isBool && paramStr.IsEmpty()) {
|
|
rv = cmdMgr->DoCommand(cmdToDispatch.get(), nullptr, window);
|
|
} else {
|
|
// we have a command that requires a parameter, create params
|
|
nsCOMPtr<nsICommandParams> cmdParams = do_CreateInstance(
|
|
NS_COMMAND_PARAMS_CONTRACTID);
|
|
if (!cmdParams) {
|
|
rv.Throw(NS_ERROR_OUT_OF_MEMORY);
|
|
return false;
|
|
}
|
|
|
|
if (isBool) {
|
|
rv = cmdParams->SetBooleanValue("state_attribute", boolVal);
|
|
} else if (cmdToDispatch.EqualsLiteral("cmd_fontFace")) {
|
|
rv = cmdParams->SetStringValue("state_attribute", value);
|
|
} else if (cmdToDispatch.EqualsLiteral("cmd_insertHTML") ||
|
|
cmdToDispatch.EqualsLiteral("cmd_insertText")) {
|
|
rv = cmdParams->SetStringValue("state_data", value);
|
|
} else {
|
|
rv = cmdParams->SetCStringValue("state_attribute", paramStr.get());
|
|
}
|
|
if (rv.Failed()) {
|
|
return false;
|
|
}
|
|
rv = cmdMgr->DoCommand(cmdToDispatch.get(), cmdParams, window);
|
|
}
|
|
|
|
return !rv.Failed();
|
|
}
|
|
|
|
/* boolean queryCommandEnabled(in DOMString commandID); */
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::QueryCommandEnabled(const nsAString& commandID,
|
|
bool* _retval)
|
|
{
|
|
ErrorResult rv;
|
|
*_retval = QueryCommandEnabled(commandID, rv);
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
bool
|
|
nsHTMLDocument::QueryCommandEnabled(const nsAString& commandID, ErrorResult& rv)
|
|
{
|
|
nsAutoCString cmdToDispatch;
|
|
if (!ConvertToMidasInternalCommand(commandID, cmdToDispatch)) {
|
|
return false;
|
|
}
|
|
|
|
// cut & copy are always allowed
|
|
bool isCutCopy = commandID.LowerCaseEqualsLiteral("cut") ||
|
|
commandID.LowerCaseEqualsLiteral("copy");
|
|
if (isCutCopy) {
|
|
return nsContentUtils::IsCutCopyAllowed();
|
|
}
|
|
|
|
// Report false for restricted commands
|
|
bool restricted = commandID.LowerCaseEqualsLiteral("paste");
|
|
if (restricted && !nsContentUtils::IsCallerChrome()) {
|
|
return false;
|
|
}
|
|
|
|
// if editing is not on, bail
|
|
if (!IsEditingOnAfterFlush()) {
|
|
return false;
|
|
}
|
|
|
|
// get command manager and dispatch command to our window if it's acceptable
|
|
nsCOMPtr<nsICommandManager> cmdMgr;
|
|
GetMidasCommandManager(getter_AddRefs(cmdMgr));
|
|
if (!cmdMgr) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return false;
|
|
}
|
|
|
|
nsIDOMWindow* window = GetWindow();
|
|
if (!window) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return false;
|
|
}
|
|
|
|
bool retval;
|
|
rv = cmdMgr->IsCommandEnabled(cmdToDispatch.get(), window, &retval);
|
|
return retval;
|
|
}
|
|
|
|
/* boolean queryCommandIndeterm (in DOMString commandID); */
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::QueryCommandIndeterm(const nsAString & commandID,
|
|
bool *_retval)
|
|
{
|
|
ErrorResult rv;
|
|
*_retval = QueryCommandIndeterm(commandID, rv);
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
bool
|
|
nsHTMLDocument::QueryCommandIndeterm(const nsAString& commandID, ErrorResult& rv)
|
|
{
|
|
nsAutoCString cmdToDispatch;
|
|
if (!ConvertToMidasInternalCommand(commandID, cmdToDispatch)) {
|
|
return false;
|
|
}
|
|
|
|
// if editing is not on, bail
|
|
if (!IsEditingOnAfterFlush()) {
|
|
return false;
|
|
}
|
|
|
|
// get command manager and dispatch command to our window if it's acceptable
|
|
nsCOMPtr<nsICommandManager> cmdMgr;
|
|
GetMidasCommandManager(getter_AddRefs(cmdMgr));
|
|
if (!cmdMgr) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return false;
|
|
}
|
|
|
|
nsIDOMWindow* window = GetWindow();
|
|
if (!window) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return false;
|
|
}
|
|
|
|
nsresult res;
|
|
nsCOMPtr<nsICommandParams> cmdParams = do_CreateInstance(
|
|
NS_COMMAND_PARAMS_CONTRACTID, &res);
|
|
if (NS_FAILED(res)) {
|
|
rv.Throw(res);
|
|
return false;
|
|
}
|
|
|
|
rv = cmdMgr->GetCommandState(cmdToDispatch.get(), window, cmdParams);
|
|
if (rv.Failed()) {
|
|
return false;
|
|
}
|
|
|
|
// If command does not have a state_mixed value, this call fails and sets
|
|
// retval to false. This is fine -- we want to return false in that case
|
|
// anyway (bug 738385), so we just don't throw regardless.
|
|
bool retval = false;
|
|
cmdParams->GetBooleanValue("state_mixed", &retval);
|
|
return retval;
|
|
}
|
|
|
|
/* boolean queryCommandState(in DOMString commandID); */
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::QueryCommandState(const nsAString & commandID, bool *_retval)
|
|
{
|
|
ErrorResult rv;
|
|
*_retval = QueryCommandState(commandID, rv);
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
bool
|
|
nsHTMLDocument::QueryCommandState(const nsAString& commandID, ErrorResult& rv)
|
|
{
|
|
nsAutoCString cmdToDispatch, paramToCheck;
|
|
bool dummy, dummy2;
|
|
if (!ConvertToMidasInternalCommand(commandID, commandID,
|
|
cmdToDispatch, paramToCheck,
|
|
dummy, dummy2)) {
|
|
return false;
|
|
}
|
|
|
|
// if editing is not on, bail
|
|
if (!IsEditingOnAfterFlush()) {
|
|
return false;
|
|
}
|
|
|
|
// get command manager and dispatch command to our window if it's acceptable
|
|
nsCOMPtr<nsICommandManager> cmdMgr;
|
|
GetMidasCommandManager(getter_AddRefs(cmdMgr));
|
|
if (!cmdMgr) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return false;
|
|
}
|
|
|
|
nsIDOMWindow* window = GetWindow();
|
|
if (!window) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return false;
|
|
}
|
|
|
|
if (commandID.LowerCaseEqualsLiteral("usecss")) {
|
|
// Per spec, state is supported for styleWithCSS but not useCSS, so we just
|
|
// return false always.
|
|
return false;
|
|
}
|
|
|
|
nsCOMPtr<nsICommandParams> cmdParams = do_CreateInstance(
|
|
NS_COMMAND_PARAMS_CONTRACTID);
|
|
if (!cmdParams) {
|
|
rv.Throw(NS_ERROR_OUT_OF_MEMORY);
|
|
return false;
|
|
}
|
|
|
|
rv = cmdMgr->GetCommandState(cmdToDispatch.get(), window, cmdParams);
|
|
if (rv.Failed()) {
|
|
return false;
|
|
}
|
|
|
|
// handle alignment as a special case (possibly other commands too?)
|
|
// Alignment is special because the external api is individual
|
|
// commands but internally we use cmd_align with different
|
|
// parameters. When getting the state of this command, we need to
|
|
// return the boolean for this particular alignment rather than the
|
|
// string of 'which alignment is this?'
|
|
if (cmdToDispatch.EqualsLiteral("cmd_align")) {
|
|
char * actualAlignmentType = nullptr;
|
|
rv = cmdParams->GetCStringValue("state_attribute", &actualAlignmentType);
|
|
bool retval = false;
|
|
if (!rv.Failed() && actualAlignmentType && actualAlignmentType[0]) {
|
|
retval = paramToCheck.Equals(actualAlignmentType);
|
|
}
|
|
if (actualAlignmentType) {
|
|
free(actualAlignmentType);
|
|
}
|
|
return retval;
|
|
}
|
|
|
|
// If command does not have a state_all value, this call fails and sets
|
|
// retval to false. This is fine -- we want to return false in that case
|
|
// anyway (bug 738385), so we just succeed and return false regardless.
|
|
bool retval = false;
|
|
cmdParams->GetBooleanValue("state_all", &retval);
|
|
return retval;
|
|
}
|
|
|
|
/* boolean queryCommandSupported(in DOMString commandID); */
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::QueryCommandSupported(const nsAString & commandID,
|
|
bool *_retval)
|
|
{
|
|
*_retval = QueryCommandSupported(commandID);
|
|
return NS_OK;
|
|
}
|
|
|
|
bool
|
|
nsHTMLDocument::QueryCommandSupported(const nsAString& commandID)
|
|
{
|
|
// Gecko technically supports all the clipboard commands including
|
|
// cut/copy/paste, but non-privileged content will be unable to call
|
|
// paste, and depending on the pref "dom.allow_cut_copy", cut and copy
|
|
// may also be disallowed to be called from non-privileged content.
|
|
// For that reason, we report the support status of corresponding
|
|
// command accordingly.
|
|
if (!nsContentUtils::IsCallerChrome()) {
|
|
if (commandID.LowerCaseEqualsLiteral("paste")) {
|
|
return false;
|
|
}
|
|
if (nsContentUtils::IsCutCopyRestricted()) {
|
|
if (commandID.LowerCaseEqualsLiteral("cut") ||
|
|
commandID.LowerCaseEqualsLiteral("copy")) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// commandID is supported if it can be converted to a Midas command
|
|
nsAutoCString cmdToDispatch;
|
|
return ConvertToMidasInternalCommand(commandID, cmdToDispatch);
|
|
}
|
|
|
|
/* DOMString queryCommandValue(in DOMString commandID); */
|
|
NS_IMETHODIMP
|
|
nsHTMLDocument::QueryCommandValue(const nsAString & commandID,
|
|
nsAString &_retval)
|
|
{
|
|
ErrorResult rv;
|
|
QueryCommandValue(commandID, _retval, rv);
|
|
return rv.StealNSResult();
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::QueryCommandValue(const nsAString& commandID,
|
|
nsAString& aValue,
|
|
ErrorResult& rv)
|
|
{
|
|
aValue.Truncate();
|
|
|
|
nsAutoCString cmdToDispatch, paramStr;
|
|
if (!ConvertToMidasInternalCommand(commandID, cmdToDispatch)) {
|
|
// Return empty string
|
|
return;
|
|
}
|
|
|
|
// if editing is not on, bail
|
|
if (!IsEditingOnAfterFlush()) {
|
|
return;
|
|
}
|
|
|
|
// get command manager and dispatch command to our window if it's acceptable
|
|
nsCOMPtr<nsICommandManager> cmdMgr;
|
|
GetMidasCommandManager(getter_AddRefs(cmdMgr));
|
|
if (!cmdMgr) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return;
|
|
}
|
|
|
|
nsIDOMWindow* window = GetWindow();
|
|
if (!window) {
|
|
rv.Throw(NS_ERROR_FAILURE);
|
|
return;
|
|
}
|
|
|
|
// create params
|
|
nsCOMPtr<nsICommandParams> cmdParams = do_CreateInstance(
|
|
NS_COMMAND_PARAMS_CONTRACTID);
|
|
if (!cmdParams) {
|
|
rv.Throw(NS_ERROR_OUT_OF_MEMORY);
|
|
return;
|
|
}
|
|
|
|
// this is a special command since we are calling DoCommand rather than
|
|
// GetCommandState like the other commands
|
|
if (cmdToDispatch.EqualsLiteral("cmd_getContents")) {
|
|
rv = cmdParams->SetBooleanValue("selection_only", true);
|
|
if (rv.Failed()) {
|
|
return;
|
|
}
|
|
rv = cmdParams->SetCStringValue("format", "text/html");
|
|
if (rv.Failed()) {
|
|
return;
|
|
}
|
|
rv = cmdMgr->DoCommand(cmdToDispatch.get(), cmdParams, window);
|
|
if (rv.Failed()) {
|
|
return;
|
|
}
|
|
rv = cmdParams->GetStringValue("result", aValue);
|
|
return;
|
|
}
|
|
|
|
rv = cmdParams->SetCStringValue("state_attribute", paramStr.get());
|
|
if (rv.Failed()) {
|
|
return;
|
|
}
|
|
|
|
rv = cmdMgr->GetCommandState(cmdToDispatch.get(), window, cmdParams);
|
|
if (rv.Failed()) {
|
|
return;
|
|
}
|
|
|
|
// If command does not have a state_attribute value, this call fails, and
|
|
// aValue will wind up being the empty string. This is fine -- we want to
|
|
// return "" in that case anyway (bug 738385), so we just return NS_OK
|
|
// regardless.
|
|
nsXPIDLCString cStringResult;
|
|
cmdParams->GetCStringValue("state_attribute",
|
|
getter_Copies(cStringResult));
|
|
CopyUTF8toUTF16(cStringResult, aValue);
|
|
}
|
|
|
|
nsresult
|
|
nsHTMLDocument::Clone(mozilla::dom::NodeInfo *aNodeInfo, nsINode **aResult) const
|
|
{
|
|
NS_ASSERTION(aNodeInfo->NodeInfoManager() == mNodeInfoManager,
|
|
"Can't import this document into another document!");
|
|
|
|
nsRefPtr<nsHTMLDocument> clone = new nsHTMLDocument();
|
|
nsresult rv = CloneDocHelper(clone.get());
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// State from nsHTMLDocument
|
|
clone->mLoadFlags = mLoadFlags;
|
|
|
|
return CallQueryInterface(clone.get(), aResult);
|
|
}
|
|
|
|
bool
|
|
nsHTMLDocument::IsEditingOnAfterFlush()
|
|
{
|
|
nsIDocument* doc = GetParentDocument();
|
|
if (doc) {
|
|
// Make sure frames are up to date, since that can affect whether
|
|
// we're editable.
|
|
doc->FlushPendingNotifications(Flush_Frames);
|
|
}
|
|
|
|
return IsEditingOn();
|
|
}
|
|
|
|
void
|
|
nsHTMLDocument::RemovedFromDocShell()
|
|
{
|
|
mEditingState = eOff;
|
|
nsDocument::RemovedFromDocShell();
|
|
}
|
|
|
|
/* virtual */ void
|
|
nsHTMLDocument::DocAddSizeOfExcludingThis(nsWindowSizes* aWindowSizes) const
|
|
{
|
|
nsDocument::DocAddSizeOfExcludingThis(aWindowSizes);
|
|
|
|
// Measurement of the following members may be added later if DMD finds it is
|
|
// worthwhile:
|
|
// - mImages
|
|
// - mApplets
|
|
// - mEmbeds
|
|
// - mLinks
|
|
// - mAnchors
|
|
// - mScripts
|
|
// - mForms
|
|
// - mFormControls
|
|
// - mWyciwygChannel
|
|
// - mMidasCommandManager
|
|
}
|
|
|
|
bool
|
|
nsHTMLDocument::WillIgnoreCharsetOverride()
|
|
{
|
|
if (mType != eHTML) {
|
|
MOZ_ASSERT(mType == eXHTML);
|
|
return true;
|
|
}
|
|
if (mCharacterSetSource == kCharsetFromByteOrderMark) {
|
|
return true;
|
|
}
|
|
if (!EncodingUtils::IsAsciiCompatible(mCharacterSet)) {
|
|
return true;
|
|
}
|
|
nsCOMPtr<nsIWyciwygChannel> wyciwyg = do_QueryInterface(mChannel);
|
|
if (wyciwyg) {
|
|
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;
|
|
}
|