зеркало из https://github.com/mozilla/gecko-dev.git
494 строки
15 KiB
C++
494 строки
15 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/ServoStyleSheet.h"
|
|
|
|
#include "mozilla/css/Rule.h"
|
|
#include "mozilla/StyleBackendType.h"
|
|
#include "mozilla/ServoBindings.h"
|
|
#include "mozilla/ServoImportRule.h"
|
|
#include "mozilla/ServoMediaList.h"
|
|
#include "mozilla/ServoCSSRuleList.h"
|
|
#include "mozilla/css/GroupRule.h"
|
|
#include "mozilla/dom/CSSRuleList.h"
|
|
#include "mozilla/dom/MediaList.h"
|
|
#include "nsIStyleSheetLinkingElement.h"
|
|
#include "Loader.h"
|
|
|
|
|
|
#include "mozAutoDocUpdate.h"
|
|
#include "nsIDOMCSSStyleSheet.h"
|
|
|
|
using namespace mozilla::dom;
|
|
|
|
namespace mozilla {
|
|
|
|
// -------------------------------
|
|
// CSS Style Sheet Inner Data Container
|
|
//
|
|
|
|
ServoStyleSheetInner::ServoStyleSheetInner(CORSMode aCORSMode,
|
|
ReferrerPolicy aReferrerPolicy,
|
|
const SRIMetadata& aIntegrity,
|
|
css::SheetParsingMode aParsingMode)
|
|
: StyleSheetInfo(aCORSMode, aReferrerPolicy, aIntegrity)
|
|
{
|
|
mContents = Servo_StyleSheet_Empty(aParsingMode).Consume();
|
|
mURLData = URLExtraData::Dummy();
|
|
MOZ_COUNT_CTOR(ServoStyleSheetInner);
|
|
}
|
|
|
|
ServoStyleSheetInner::ServoStyleSheetInner(ServoStyleSheetInner& aCopy,
|
|
ServoStyleSheet* aPrimarySheet)
|
|
: StyleSheetInfo(aCopy, aPrimarySheet)
|
|
, mURLData(aCopy.mURLData)
|
|
{
|
|
MOZ_COUNT_CTOR(ServoStyleSheetInner);
|
|
|
|
// Actually clone aCopy's mContents and use that as ours.
|
|
mContents = Servo_StyleSheet_Clone(
|
|
aCopy.mContents.get(), aPrimarySheet).Consume();
|
|
|
|
// Our child list is fixed up by our parent.
|
|
}
|
|
|
|
void
|
|
ServoStyleSheet::BuildChildListAfterInnerClone()
|
|
{
|
|
MOZ_ASSERT(Inner()->mSheets.Length() == 1, "Should've just cloned");
|
|
MOZ_ASSERT(Inner()->mSheets[0] == this);
|
|
MOZ_ASSERT(!Inner()->mFirstChild);
|
|
|
|
auto* contents = Inner()->mContents.get();
|
|
RefPtr<ServoCssRules> rules =
|
|
Servo_StyleSheet_GetRules(contents).Consume();
|
|
|
|
uint32_t index = 0;
|
|
while (true) {
|
|
uint32_t line, column; // Actually unused.
|
|
RefPtr<RawServoImportRule> import =
|
|
Servo_CssRules_GetImportRuleAt(rules, index, &line, &column).Consume();
|
|
if (!import) {
|
|
// Note that only @charset rules come before @import rules, and @charset
|
|
// rules are parsed but skipped, so we can stop iterating as soon as we
|
|
// find something that isn't an @import rule.
|
|
break;
|
|
}
|
|
auto* sheet =
|
|
const_cast<ServoStyleSheet*>(Servo_ImportRule_GetSheet(import));
|
|
MOZ_ASSERT(sheet);
|
|
PrependStyleSheetSilently(sheet);
|
|
index++;
|
|
}
|
|
}
|
|
|
|
already_AddRefed<ServoStyleSheet>
|
|
ServoStyleSheet::CreateEmptyChildSheet(
|
|
already_AddRefed<dom::MediaList> aMediaList) const
|
|
{
|
|
RefPtr<ServoStyleSheet> child =
|
|
new ServoStyleSheet(
|
|
ParsingMode(),
|
|
CORSMode::CORS_NONE,
|
|
GetReferrerPolicy(),
|
|
SRIMetadata());
|
|
|
|
child->mMedia = aMediaList;
|
|
return child.forget();
|
|
}
|
|
|
|
|
|
ServoStyleSheetInner::~ServoStyleSheetInner()
|
|
{
|
|
MOZ_COUNT_DTOR(ServoStyleSheetInner);
|
|
}
|
|
|
|
StyleSheetInfo*
|
|
ServoStyleSheetInner::CloneFor(StyleSheet* aPrimarySheet)
|
|
{
|
|
return new ServoStyleSheetInner(*this,
|
|
static_cast<ServoStyleSheet*>(aPrimarySheet));
|
|
}
|
|
|
|
MOZ_DEFINE_MALLOC_SIZE_OF(ServoStyleSheetMallocSizeOf)
|
|
MOZ_DEFINE_MALLOC_ENCLOSING_SIZE_OF(ServoStyleSheetMallocEnclosingSizeOf)
|
|
|
|
size_t
|
|
ServoStyleSheetInner::SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const
|
|
{
|
|
size_t n = aMallocSizeOf(this);
|
|
n += Servo_StyleSheet_SizeOfIncludingThis(
|
|
ServoStyleSheetMallocSizeOf,
|
|
ServoStyleSheetMallocEnclosingSizeOf,
|
|
mContents);
|
|
return n;
|
|
}
|
|
|
|
ServoStyleSheet::ServoStyleSheet(css::SheetParsingMode aParsingMode,
|
|
CORSMode aCORSMode,
|
|
net::ReferrerPolicy aReferrerPolicy,
|
|
const dom::SRIMetadata& aIntegrity)
|
|
: StyleSheet(StyleBackendType::Servo, aParsingMode)
|
|
{
|
|
mInner = new ServoStyleSheetInner(
|
|
aCORSMode, aReferrerPolicy, aIntegrity, aParsingMode);
|
|
mInner->AddSheet(this);
|
|
}
|
|
|
|
ServoStyleSheet::ServoStyleSheet(const ServoStyleSheet& aCopy,
|
|
ServoStyleSheet* aParentToUse,
|
|
dom::CSSImportRule* aOwnerRuleToUse,
|
|
nsIDocument* aDocumentToUse,
|
|
nsINode* aOwningNodeToUse)
|
|
: StyleSheet(aCopy,
|
|
aParentToUse,
|
|
aOwnerRuleToUse,
|
|
aDocumentToUse,
|
|
aOwningNodeToUse)
|
|
{
|
|
if (mDirty) { // CSSOM's been there, force full copy now
|
|
NS_ASSERTION(mInner->mComplete,
|
|
"Why have rules been accessed on an incomplete sheet?");
|
|
// FIXME: handle failure?
|
|
//
|
|
// NOTE: It's important to call this from the subclass, since this could
|
|
// access uninitialized members otherwise.
|
|
EnsureUniqueInner();
|
|
}
|
|
}
|
|
|
|
ServoStyleSheet::~ServoStyleSheet()
|
|
{
|
|
}
|
|
|
|
void
|
|
ServoStyleSheet::LastRelease()
|
|
{
|
|
DropRuleList();
|
|
}
|
|
|
|
// QueryInterface implementation for ServoStyleSheet
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(ServoStyleSheet)
|
|
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIDOMCSSStyleSheet)
|
|
if (aIID.Equals(NS_GET_IID(ServoStyleSheet)))
|
|
foundInterface = reinterpret_cast<nsISupports*>(this);
|
|
else
|
|
NS_INTERFACE_MAP_END_INHERITING(StyleSheet)
|
|
|
|
NS_IMPL_ADDREF_INHERITED(ServoStyleSheet, StyleSheet)
|
|
NS_IMPL_RELEASE_INHERITED(ServoStyleSheet, StyleSheet)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(ServoStyleSheet)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(ServoStyleSheet)
|
|
tmp->DropRuleList();
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END_INHERITED(StyleSheet)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(ServoStyleSheet, StyleSheet)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRuleList)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
bool
|
|
ServoStyleSheet::HasRules() const
|
|
{
|
|
return Servo_StyleSheet_HasRules(Inner()->mContents);
|
|
}
|
|
|
|
nsresult
|
|
ServoStyleSheet::ParseSheet(css::Loader* aLoader,
|
|
Span<const uint8_t> aInput,
|
|
nsIURI* aSheetURI,
|
|
nsIURI* aBaseURI,
|
|
nsIPrincipal* aSheetPrincipal,
|
|
uint32_t aLineNumber,
|
|
nsCompatibility aCompatMode,
|
|
css::LoaderReusableStyleSheets* aReusableSheets)
|
|
{
|
|
MOZ_ASSERT(!mMedia || mMedia->IsServo());
|
|
RefPtr<URLExtraData> extraData =
|
|
new URLExtraData(aBaseURI, aSheetURI, aSheetPrincipal);
|
|
|
|
Inner()->mContents = Servo_StyleSheet_FromUTF8Bytes(aLoader,
|
|
this,
|
|
aInput.Elements(),
|
|
aInput.Length(),
|
|
mParsingMode,
|
|
extraData,
|
|
aLineNumber,
|
|
aCompatMode,
|
|
aReusableSheets)
|
|
.Consume();
|
|
|
|
nsString sourceMapURL;
|
|
Servo_StyleSheet_GetSourceMapURL(Inner()->mContents, &sourceMapURL);
|
|
SetSourceMapURLFromComment(sourceMapURL);
|
|
|
|
nsString sourceURL;
|
|
Servo_StyleSheet_GetSourceURL(Inner()->mContents, &sourceURL);
|
|
SetSourceURL(sourceURL);
|
|
|
|
Inner()->mURLData = extraData.forget();
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
ServoStyleSheet::ReparseSheet(const nsAString& aInput)
|
|
{
|
|
if (!mInner->mComplete) {
|
|
return NS_ERROR_DOM_INVALID_ACCESS_ERR;
|
|
}
|
|
|
|
// Hold strong ref to the CSSLoader in case the document update
|
|
// kills the document
|
|
RefPtr<css::Loader> loader;
|
|
if (mDocument) {
|
|
loader = mDocument->CSSLoader();
|
|
NS_ASSERTION(loader, "Document with no CSS loader!");
|
|
} else {
|
|
loader = new css::Loader(StyleBackendType::Servo, nullptr);
|
|
}
|
|
|
|
mozAutoDocUpdate updateBatch(mDocument, UPDATE_STYLE, true);
|
|
|
|
WillDirty();
|
|
|
|
// cache child sheets to reuse
|
|
css::LoaderReusableStyleSheets reusableSheets;
|
|
for (StyleSheet* child = GetFirstChild(); child; child = child->mNext) {
|
|
if (child->GetOriginalURI()) {
|
|
reusableSheets.AddReusableSheet(child);
|
|
}
|
|
}
|
|
|
|
// clean up child sheets list
|
|
for (StyleSheet* child = GetFirstChild(); child; ) {
|
|
StyleSheet* next = child->mNext;
|
|
child->mParent = nullptr;
|
|
child->SetAssociatedDocument(nullptr, NotOwnedByDocument);
|
|
child->mNext = nullptr;
|
|
child = next;
|
|
}
|
|
Inner()->mFirstChild = nullptr;
|
|
|
|
uint32_t lineNumber = 1;
|
|
if (mOwningNode) {
|
|
nsCOMPtr<nsIStyleSheetLinkingElement> link = do_QueryInterface(mOwningNode);
|
|
if (link) {
|
|
lineNumber = link->GetLineNumber();
|
|
}
|
|
}
|
|
|
|
// Notify mDocument that all our rules are removed.
|
|
if (mDocument) {
|
|
// Get the rule list.
|
|
ServoCSSRuleList* ruleList = GetCssRulesInternal();
|
|
MOZ_ASSERT(ruleList);
|
|
|
|
uint32_t ruleCount = ruleList->Length();
|
|
for (uint32_t i = 0; i < ruleCount; ++i) {
|
|
css::Rule* rule = ruleList->GetRule(i);
|
|
MOZ_ASSERT(rule);
|
|
if (rule->GetType() == css::Rule::IMPORT_RULE &&
|
|
RuleHasPendingChildSheet(rule)) {
|
|
continue; // notify when loaded (see StyleSheetLoaded)
|
|
}
|
|
mDocument->StyleRuleRemoved(this, rule);
|
|
|
|
// Document observers could possibly detach document from this sheet.
|
|
if (!mDocument) {
|
|
// If detached, don't process any more rules.
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
DropRuleList();
|
|
|
|
nsresult rv = ParseSheet(loader,
|
|
NS_ConvertUTF16toUTF8(aInput),
|
|
mInner->mSheetURI,
|
|
mInner->mBaseURI,
|
|
mInner->mPrincipal,
|
|
lineNumber,
|
|
eCompatibility_FullStandards,
|
|
&reusableSheets);
|
|
DidDirty();
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// Notify mDocument that all our new rules are added.
|
|
if (mDocument) {
|
|
// Get the rule list (which will need to be regenerated after ParseSheet).
|
|
ServoCSSRuleList* ruleList = GetCssRulesInternal();
|
|
MOZ_ASSERT(ruleList);
|
|
|
|
uint32_t ruleCount = ruleList->Length();
|
|
for (uint32_t i = 0; i < ruleCount; ++i) {
|
|
css::Rule* rule = ruleList->GetRule(i);
|
|
MOZ_ASSERT(rule);
|
|
if (rule->GetType() == css::Rule::IMPORT_RULE &&
|
|
RuleHasPendingChildSheet(rule)) {
|
|
continue; // notify when loaded (see StyleSheetLoaded)
|
|
}
|
|
|
|
mDocument->StyleRuleAdded(this, rule);
|
|
|
|
// Document observers could possibly detach document from this sheet.
|
|
if (!mDocument) {
|
|
// If detached, don't process any more rules.
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
// nsICSSLoaderObserver implementation
|
|
NS_IMETHODIMP
|
|
ServoStyleSheet::StyleSheetLoaded(StyleSheet* aSheet,
|
|
bool aWasAlternate,
|
|
nsresult aStatus)
|
|
{
|
|
MOZ_ASSERT(aSheet->IsServo(),
|
|
"why we were called back with a CSSStyleSheet?");
|
|
|
|
ServoStyleSheet* sheet = aSheet->AsServo();
|
|
if (sheet->GetParentSheet() == nullptr) {
|
|
return NS_OK; // ignore if sheet has been detached already
|
|
}
|
|
NS_ASSERTION(this == sheet->GetParentSheet(),
|
|
"We are being notified of a sheet load for a sheet that is not our child!");
|
|
|
|
if (mDocument && NS_SUCCEEDED(aStatus)) {
|
|
mozAutoDocUpdate updateBatch(mDocument, UPDATE_STYLE, true);
|
|
mDocument->StyleRuleAdded(this, sheet->GetOwnerRule());
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
ServoStyleSheet::DropRuleList()
|
|
{
|
|
if (mRuleList) {
|
|
mRuleList->DropReference();
|
|
mRuleList = nullptr;
|
|
}
|
|
}
|
|
|
|
already_AddRefed<StyleSheet>
|
|
ServoStyleSheet::Clone(StyleSheet* aCloneParent,
|
|
dom::CSSImportRule* aCloneOwnerRule,
|
|
nsIDocument* aCloneDocument,
|
|
nsINode* aCloneOwningNode) const
|
|
{
|
|
RefPtr<StyleSheet> clone = new ServoStyleSheet(*this,
|
|
static_cast<ServoStyleSheet*>(aCloneParent),
|
|
aCloneOwnerRule,
|
|
aCloneDocument,
|
|
aCloneOwningNode);
|
|
return clone.forget();
|
|
}
|
|
|
|
ServoCSSRuleList*
|
|
ServoStyleSheet::GetCssRulesInternal()
|
|
{
|
|
if (!mRuleList) {
|
|
EnsureUniqueInner();
|
|
|
|
RefPtr<ServoCssRules> rawRules =
|
|
Servo_StyleSheet_GetRules(Inner()->mContents).Consume();
|
|
MOZ_ASSERT(rawRules);
|
|
mRuleList = new ServoCSSRuleList(rawRules.forget(), this);
|
|
}
|
|
return mRuleList;
|
|
}
|
|
|
|
uint32_t
|
|
ServoStyleSheet::InsertRuleInternal(const nsAString& aRule,
|
|
uint32_t aIndex, ErrorResult& aRv)
|
|
{
|
|
// Ensure mRuleList is constructed.
|
|
GetCssRulesInternal();
|
|
|
|
mozAutoDocUpdate updateBatch(mDocument, UPDATE_STYLE, true);
|
|
aRv = mRuleList->InsertRule(aRule, aIndex);
|
|
if (aRv.Failed()) {
|
|
return 0;
|
|
}
|
|
if (mDocument) {
|
|
if (mRuleList->GetDOMCSSRuleType(aIndex) != nsIDOMCSSRule::IMPORT_RULE ||
|
|
!RuleHasPendingChildSheet(mRuleList->GetRule(aIndex))) {
|
|
// XXX We may not want to get the rule when stylesheet change event
|
|
// is not enabled.
|
|
mDocument->StyleRuleAdded(this, mRuleList->GetRule(aIndex));
|
|
}
|
|
}
|
|
return aIndex;
|
|
}
|
|
|
|
void
|
|
ServoStyleSheet::DeleteRuleInternal(uint32_t aIndex, ErrorResult& aRv)
|
|
{
|
|
// Ensure mRuleList is constructed.
|
|
GetCssRulesInternal();
|
|
if (aIndex >= mRuleList->Length()) {
|
|
aRv.Throw(NS_ERROR_DOM_INDEX_SIZE_ERR);
|
|
return;
|
|
}
|
|
|
|
mozAutoDocUpdate updateBatch(mDocument, UPDATE_STYLE, true);
|
|
// Hold a strong ref to the rule so it doesn't die when we remove it
|
|
// from the list. XXX We may not want to hold it if stylesheet change
|
|
// event is not enabled.
|
|
RefPtr<css::Rule> rule = mRuleList->GetRule(aIndex);
|
|
aRv = mRuleList->DeleteRule(aIndex);
|
|
MOZ_ASSERT(!aRv.ErrorCodeIs(NS_ERROR_DOM_INDEX_SIZE_ERR),
|
|
"IndexSizeError should have been handled earlier");
|
|
if (!aRv.Failed() && mDocument) {
|
|
mDocument->StyleRuleRemoved(this, rule);
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
ServoStyleSheet::InsertRuleIntoGroupInternal(const nsAString& aRule,
|
|
css::GroupRule* aGroup,
|
|
uint32_t aIndex)
|
|
{
|
|
auto rules = static_cast<ServoCSSRuleList*>(aGroup->CssRules());
|
|
MOZ_ASSERT(rules->GetParentRule() == aGroup);
|
|
return rules->InsertRule(aRule, aIndex);
|
|
}
|
|
|
|
size_t
|
|
ServoStyleSheet::SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const
|
|
{
|
|
size_t n = StyleSheet::SizeOfIncludingThis(aMallocSizeOf);
|
|
const ServoStyleSheet* s = this;
|
|
while (s) {
|
|
// See the comment in CSSStyleSheet::SizeOfIncludingThis() for an
|
|
// explanation of this.
|
|
if (s->Inner()->mSheets.LastElement() == s) {
|
|
n += s->Inner()->SizeOfIncludingThis(aMallocSizeOf);
|
|
}
|
|
|
|
// Measurement of the following members may be added later if DMD finds it
|
|
// is worthwhile:
|
|
// - s->mRuleList
|
|
|
|
s = s->mNext ? s->mNext->AsServo() : nullptr;
|
|
}
|
|
return n;
|
|
}
|
|
|
|
OriginFlags
|
|
ServoStyleSheet::GetOrigin()
|
|
{
|
|
return static_cast<OriginFlags>(
|
|
Servo_StyleSheet_GetOrigin(Inner()->mContents));
|
|
}
|
|
|
|
} // namespace mozilla
|