/* -*- 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/. */ /* representation of CSSRuleList for stylo */ #include "mozilla/ServoCSSRuleList.h" #include "mozilla/IntegerRange.h" #include "mozilla/ServoBindings.h" #include "mozilla/ServoDocumentRule.h" #include "mozilla/ServoImportRule.h" #include "mozilla/ServoFontFeatureValuesRule.h" #include "mozilla/ServoKeyframesRule.h" #include "mozilla/ServoMediaRule.h" #include "mozilla/ServoNamespaceRule.h" #include "mozilla/ServoPageRule.h" #include "mozilla/ServoStyleRule.h" #include "mozilla/ServoStyleSheet.h" #include "mozilla/ServoSupportsRule.h" #include "nsCSSCounterStyleRule.h" #include "nsCSSFontFaceRule.h" namespace mozilla { ServoCSSRuleList::ServoCSSRuleList(already_AddRefed aRawRules, ServoStyleSheet* aDirectOwnerStyleSheet) : mStyleSheet(aDirectOwnerStyleSheet) , mRawRules(aRawRules) { Servo_CssRules_ListTypes(mRawRules, &mRules); } // QueryInterface implementation for ServoCSSRuleList NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(ServoCSSRuleList) NS_INTERFACE_MAP_END_INHERITING(dom::CSSRuleList) NS_IMPL_ADDREF_INHERITED(ServoCSSRuleList, dom::CSSRuleList) NS_IMPL_RELEASE_INHERITED(ServoCSSRuleList, dom::CSSRuleList) NS_IMPL_CYCLE_COLLECTION_CLASS(ServoCSSRuleList) NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(ServoCSSRuleList) tmp->DropAllRules(); NS_IMPL_CYCLE_COLLECTION_UNLINK_END_INHERITED(dom::CSSRuleList) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(ServoCSSRuleList, dom::CSSRuleList) tmp->EnumerateInstantiatedRules([&](css::Rule* aRule) { if (!aRule->IsCCLeaf()) { NS_CYCLE_COLLECTION_NOTE_EDGE_NAME(cb, "mRules[i]"); cb.NoteXPCOMChild(aRule); // Note about @font-face and @counter-style rule again, since // there is an indirect owning edge through Servo's struct that // FontFaceRule / CounterStyleRule in Servo owns a Gecko // nsCSSFontFaceRule / nsCSSCounterStyleRule object. auto type = aRule->Type(); if (type == nsIDOMCSSRule::FONT_FACE_RULE || type == nsIDOMCSSRule::COUNTER_STYLE_RULE) { NS_CYCLE_COLLECTION_NOTE_EDGE_NAME(cb, "mRawRules[i]"); cb.NoteXPCOMChild(aRule); } } }); NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END void ServoCSSRuleList::SetParentRule(css::GroupRule* aParentRule) { mParentRule = aParentRule; EnumerateInstantiatedRules([aParentRule](css::Rule* rule) { rule->SetParentRule(aParentRule); }); } void ServoCSSRuleList::SetStyleSheet(StyleSheet* aStyleSheet) { mStyleSheet = aStyleSheet ? aStyleSheet->AsServo() : nullptr; EnumerateInstantiatedRules([this](css::Rule* rule) { rule->SetStyleSheet(mStyleSheet); }); } css::Rule* ServoCSSRuleList::GetRule(uint32_t aIndex) { uintptr_t rule = mRules[aIndex]; if (rule <= kMaxRuleType) { RefPtr ruleObj = nullptr; switch (rule) { #define CASE_RULE(const_, name_) \ case nsIDOMCSSRule::const_##_RULE: { \ uint32_t line = 0, column = 0; \ RefPtr rule = \ Servo_CssRules_Get##name_##RuleAt( \ mRawRules, aIndex, &line, &column \ ).Consume(); \ MOZ_ASSERT(rule); \ ruleObj = new Servo##name_##Rule(rule.forget(), line, column); \ break; \ } CASE_RULE(STYLE, Style) CASE_RULE(KEYFRAMES, Keyframes) CASE_RULE(MEDIA, Media) CASE_RULE(NAMESPACE, Namespace) CASE_RULE(PAGE, Page) CASE_RULE(SUPPORTS, Supports) CASE_RULE(DOCUMENT, Document) CASE_RULE(IMPORT, Import) CASE_RULE(FONT_FEATURE_VALUES, FontFeatureValues) #undef CASE_RULE // For @font-face and @counter-style rules, the function returns // a borrowed Gecko rule object directly, so we don't need to // create anything here. But we still need to have the style sheet // and parent rule set properly. case nsIDOMCSSRule::FONT_FACE_RULE: { ruleObj = Servo_CssRules_GetFontFaceRuleAt(mRawRules, aIndex); break; } case nsIDOMCSSRule::COUNTER_STYLE_RULE: { ruleObj = Servo_CssRules_GetCounterStyleRuleAt(mRawRules, aIndex); break; } case nsIDOMCSSRule::KEYFRAME_RULE: MOZ_ASSERT_UNREACHABLE("keyframe rule cannot be here"); return nullptr; default: NS_WARNING("stylo: not implemented yet"); return nullptr; } ruleObj->SetStyleSheet(mStyleSheet); ruleObj->SetParentRule(mParentRule); rule = CastToUint(ruleObj.forget().take()); mRules[aIndex] = rule; } return CastToPtr(rule); } css::Rule* ServoCSSRuleList::IndexedGetter(uint32_t aIndex, bool& aFound) { if (aIndex >= mRules.Length()) { aFound = false; return nullptr; } aFound = true; return GetRule(aIndex); } template void ServoCSSRuleList::EnumerateInstantiatedRules(Func aCallback) { for (uintptr_t rule : mRules) { if (rule > kMaxRuleType) { aCallback(CastToPtr(rule)); } } } static void DropRule(already_AddRefed aRule) { RefPtr rule = aRule; rule->SetStyleSheet(nullptr); rule->SetParentRule(nullptr); } void ServoCSSRuleList::DropAllRules() { EnumerateInstantiatedRules([](css::Rule* rule) { DropRule(already_AddRefed(rule)); }); mRules.Clear(); mRawRules = nullptr; } void ServoCSSRuleList::DropReference() { mStyleSheet = nullptr; mParentRule = nullptr; DropAllRules(); } nsresult ServoCSSRuleList::InsertRule(const nsAString& aRule, uint32_t aIndex) { MOZ_ASSERT(mStyleSheet, "Caller must ensure that " "the list is not unlinked from stylesheet"); NS_ConvertUTF16toUTF8 rule(aRule); bool nested = !!mParentRule; css::Loader* loader = nullptr; if (nsIDocument* doc = mStyleSheet->GetAssociatedDocument()) { loader = doc->CSSLoader(); } uint16_t type; nsresult rv = Servo_CssRules_InsertRule(mRawRules, mStyleSheet->RawContents(), &rule, aIndex, nested, loader, mStyleSheet, &type); if (NS_FAILED(rv)) { return rv; } mRules.InsertElementAt(aIndex, type); return rv; } nsresult ServoCSSRuleList::DeleteRule(uint32_t aIndex) { nsresult rv = Servo_CssRules_DeleteRule(mRawRules, aIndex); if (!NS_FAILED(rv)) { uintptr_t rule = mRules[aIndex]; if (rule > kMaxRuleType) { DropRule(already_AddRefed(CastToPtr(rule))); } mRules.RemoveElementAt(aIndex); } return rv; } uint16_t ServoCSSRuleList::GetDOMCSSRuleType(uint32_t aIndex) const { uintptr_t rule = mRules[aIndex]; if (rule <= kMaxRuleType) { return rule; } return CastToPtr(rule)->Type(); } ServoCSSRuleList::~ServoCSSRuleList() { MOZ_ASSERT(!mStyleSheet, "Backpointer should have been cleared"); MOZ_ASSERT(!mParentRule, "Backpointer should have been cleared"); DropAllRules(); } } // namespace mozilla