forked from mirrors/gecko-dev
This change captures the subject principal when a scripted caller sets the textContent or innerHTML property of a <style> node, and uses it as the triggering principal for the resulting stylesheet. If the node contents are modified in any way other than through textContent or innerHTML, the triggering principal is forgotten (which is an intentional design feature). MozReview-Commit-ID: GacZFIB5BzS --HG-- extra : rebase_source : 04926f30b8e2831d18d3fb64b850f670f006eb85
255 lines
7.5 KiB
C++
255 lines
7.5 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 "mozilla/dom/HTMLStyleElement.h"
|
|
#include "mozilla/dom/HTMLStyleElementBinding.h"
|
|
#include "nsGkAtoms.h"
|
|
#include "nsStyleConsts.h"
|
|
#include "nsIDOMStyleSheet.h"
|
|
#include "nsIDocument.h"
|
|
#include "nsUnicharUtils.h"
|
|
#include "nsThreadUtils.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsStubMutationObserver.h"
|
|
|
|
NS_IMPL_NS_NEW_HTML_ELEMENT(Style)
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
HTMLStyleElement::HTMLStyleElement(already_AddRefed<mozilla::dom::NodeInfo>& aNodeInfo)
|
|
: nsGenericHTMLElement(aNodeInfo)
|
|
{
|
|
AddMutationObserver(this);
|
|
}
|
|
|
|
HTMLStyleElement::~HTMLStyleElement()
|
|
{
|
|
}
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(HTMLStyleElement)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(HTMLStyleElement,
|
|
nsGenericHTMLElement)
|
|
tmp->nsStyleLinkElement::Traverse(cb);
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(HTMLStyleElement,
|
|
nsGenericHTMLElement)
|
|
tmp->nsStyleLinkElement::Unlink();
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
|
|
|
NS_IMPL_ISUPPORTS_CYCLE_COLLECTION_INHERITED(HTMLStyleElement,
|
|
nsGenericHTMLElement,
|
|
nsIStyleSheetLinkingElement,
|
|
nsIMutationObserver)
|
|
|
|
NS_IMPL_ELEMENT_CLONE(HTMLStyleElement)
|
|
|
|
|
|
bool
|
|
HTMLStyleElement::Disabled()
|
|
{
|
|
StyleSheet* ss = GetSheet();
|
|
return ss && ss->Disabled();
|
|
}
|
|
|
|
void
|
|
HTMLStyleElement::SetDisabled(bool aDisabled)
|
|
{
|
|
if (StyleSheet* ss = GetSheet()) {
|
|
ss->SetDisabled(aDisabled);
|
|
}
|
|
}
|
|
|
|
void
|
|
HTMLStyleElement::CharacterDataChanged(nsIDocument* aDocument,
|
|
nsIContent* aContent,
|
|
CharacterDataChangeInfo* aInfo)
|
|
{
|
|
ContentChanged(aContent);
|
|
}
|
|
|
|
void
|
|
HTMLStyleElement::ContentAppended(nsIDocument* aDocument,
|
|
nsIContent* aContainer,
|
|
nsIContent* aFirstNewContent)
|
|
{
|
|
ContentChanged(aContainer);
|
|
}
|
|
|
|
void
|
|
HTMLStyleElement::ContentInserted(nsIDocument* aDocument,
|
|
nsIContent* aContainer,
|
|
nsIContent* aChild)
|
|
{
|
|
ContentChanged(aChild);
|
|
}
|
|
|
|
void
|
|
HTMLStyleElement::ContentRemoved(nsIDocument* aDocument,
|
|
nsIContent* aContainer,
|
|
nsIContent* aChild,
|
|
nsIContent* aPreviousSibling)
|
|
{
|
|
ContentChanged(aChild);
|
|
}
|
|
|
|
void
|
|
HTMLStyleElement::ContentChanged(nsIContent* aContent)
|
|
{
|
|
mTriggeringPrincipal = nullptr;
|
|
if (nsContentUtils::IsInSameAnonymousTree(this, aContent)) {
|
|
UpdateStyleSheetInternal(nullptr, nullptr);
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
HTMLStyleElement::BindToTree(nsIDocument* aDocument, nsIContent* aParent,
|
|
nsIContent* aBindingParent,
|
|
bool aCompileEventHandlers)
|
|
{
|
|
nsresult rv = nsGenericHTMLElement::BindToTree(aDocument, aParent,
|
|
aBindingParent,
|
|
aCompileEventHandlers);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
void (HTMLStyleElement::*update)() = &HTMLStyleElement::UpdateStyleSheetInternal;
|
|
nsContentUtils::AddScriptRunner(
|
|
NewRunnableMethod("dom::HTMLStyleElement::BindToTree", this, update));
|
|
|
|
return rv;
|
|
}
|
|
|
|
void
|
|
HTMLStyleElement::UnbindFromTree(bool aDeep, bool aNullParent)
|
|
{
|
|
nsCOMPtr<nsIDocument> oldDoc = GetUncomposedDoc();
|
|
ShadowRoot* oldShadow = GetContainingShadow();
|
|
|
|
nsGenericHTMLElement::UnbindFromTree(aDeep, aNullParent);
|
|
|
|
if (oldShadow && GetContainingShadow()) {
|
|
// The style is in a shadow tree and is still in the
|
|
// shadow tree. Thus the sheets in the shadow DOM
|
|
// do not need to be updated.
|
|
return;
|
|
}
|
|
|
|
UpdateStyleSheetInternal(oldDoc, oldShadow);
|
|
}
|
|
|
|
nsresult
|
|
HTMLStyleElement::AfterSetAttr(int32_t aNameSpaceID, nsAtom* aName,
|
|
const nsAttrValue* aValue,
|
|
const nsAttrValue* aOldValue,
|
|
nsIPrincipal* aSubjectPrincipal,
|
|
bool aNotify)
|
|
{
|
|
if (aNameSpaceID == kNameSpaceID_None) {
|
|
if (aName == nsGkAtoms::title ||
|
|
aName == nsGkAtoms::media ||
|
|
aName == nsGkAtoms::type) {
|
|
UpdateStyleSheetInternal(nullptr, nullptr, true);
|
|
} else if (aName == nsGkAtoms::scoped &&
|
|
OwnerDoc()->IsScopedStyleEnabled()) {
|
|
bool isScoped = aValue;
|
|
UpdateStyleSheetScopedness(isScoped);
|
|
}
|
|
}
|
|
|
|
return nsGenericHTMLElement::AfterSetAttr(aNameSpaceID, aName, aValue,
|
|
aOldValue, aSubjectPrincipal, aNotify);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
HTMLStyleElement::GetInnerHTML(nsAString& aInnerHTML)
|
|
{
|
|
if (!nsContentUtils::GetNodeTextContent(this, false, aInnerHTML, fallible)) {
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
HTMLStyleElement::SetInnerHTML(const nsAString& aInnerHTML,
|
|
nsIPrincipal& aScriptedPrincipal,
|
|
ErrorResult& aError)
|
|
{
|
|
SetTextContentInternal(aInnerHTML, &aScriptedPrincipal, aError);
|
|
}
|
|
|
|
void
|
|
HTMLStyleElement::SetTextContentInternal(const nsAString& aTextContent,
|
|
nsIPrincipal* aScriptedPrincipal,
|
|
ErrorResult& aError)
|
|
{
|
|
SetEnableUpdates(false);
|
|
|
|
aError = nsContentUtils::SetNodeTextContent(this, aTextContent, true);
|
|
|
|
SetEnableUpdates(true);
|
|
|
|
mTriggeringPrincipal = aScriptedPrincipal;
|
|
|
|
UpdateStyleSheetInternal(nullptr, nullptr);
|
|
}
|
|
|
|
already_AddRefed<nsIURI>
|
|
HTMLStyleElement::GetStyleSheetURL(bool* aIsInline, nsIPrincipal** aTriggeringPrincipal)
|
|
{
|
|
*aIsInline = true;
|
|
*aTriggeringPrincipal = do_AddRef(mTriggeringPrincipal).take();
|
|
return nullptr;
|
|
}
|
|
|
|
void
|
|
HTMLStyleElement::GetStyleSheetInfo(nsAString& aTitle,
|
|
nsAString& aType,
|
|
nsAString& aMedia,
|
|
bool* aIsScoped,
|
|
bool* aIsAlternate)
|
|
{
|
|
aTitle.Truncate();
|
|
aType.Truncate();
|
|
aMedia.Truncate();
|
|
*aIsAlternate = false;
|
|
|
|
nsAutoString title;
|
|
GetAttr(kNameSpaceID_None, nsGkAtoms::title, title);
|
|
title.CompressWhitespace();
|
|
aTitle.Assign(title);
|
|
|
|
GetAttr(kNameSpaceID_None, nsGkAtoms::media, aMedia);
|
|
// The HTML5 spec is formulated in terms of the CSSOM spec, which specifies
|
|
// that media queries should be ASCII lowercased during serialization.
|
|
nsContentUtils::ASCIIToLower(aMedia);
|
|
|
|
GetAttr(kNameSpaceID_None, nsGkAtoms::type, aType);
|
|
|
|
*aIsScoped = HasAttr(kNameSpaceID_None, nsGkAtoms::scoped) &&
|
|
OwnerDoc()->IsScopedStyleEnabled();
|
|
|
|
nsAutoString mimeType;
|
|
nsAutoString notUsed;
|
|
nsContentUtils::SplitMimeType(aType, mimeType, notUsed);
|
|
if (!mimeType.IsEmpty() && !mimeType.LowerCaseEqualsLiteral("text/css")) {
|
|
return;
|
|
}
|
|
|
|
// If we get here we assume that we're loading a css file, so set the
|
|
// type to 'text/css'
|
|
aType.AssignLiteral("text/css");
|
|
}
|
|
|
|
JSObject*
|
|
HTMLStyleElement::WrapNode(JSContext *aCx, JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
return HTMLStyleElementBinding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|
|
|