summaryrefslogtreecommitdiffstats
path: root/layout/style/CSSImportRule.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'layout/style/CSSImportRule.cpp')
-rw-r--r--layout/style/CSSImportRule.cpp147
1 files changed, 147 insertions, 0 deletions
diff --git a/layout/style/CSSImportRule.cpp b/layout/style/CSSImportRule.cpp
new file mode 100644
index 0000000000..efdb773b08
--- /dev/null
+++ b/layout/style/CSSImportRule.cpp
@@ -0,0 +1,147 @@
+/* -*- 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/CSSImportRule.h"
+
+#include "mozilla/dom/CSSImportRuleBinding.h"
+#include "mozilla/dom/MediaList.h"
+#include "mozilla/ServoBindings.h"
+#include "mozilla/StyleSheet.h"
+
+namespace mozilla {
+namespace dom {
+
+CSSImportRule::CSSImportRule(RefPtr<RawServoImportRule> aRawRule,
+ StyleSheet* aSheet, css::Rule* aParentRule,
+ uint32_t aLine, uint32_t aColumn)
+ : css::Rule(aSheet, aParentRule, aLine, aColumn),
+ mRawRule(std::move(aRawRule)) {
+ const auto* sheet = Servo_ImportRule_GetSheet(mRawRule.get());
+ MOZ_ASSERT(sheet);
+ mChildSheet = const_cast<StyleSheet*>(sheet);
+ mChildSheet->AddReferencingRule(*this);
+}
+
+CSSImportRule::~CSSImportRule() {
+ if (mChildSheet) {
+ mChildSheet->RemoveReferencingRule(*this);
+ }
+}
+
+// QueryInterface implementation for CSSImportRule
+NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(CSSImportRule)
+NS_INTERFACE_MAP_END_INHERITING(css::Rule)
+
+NS_IMPL_CYCLE_COLLECTION_CLASS(CSSImportRule)
+
+NS_IMPL_ADDREF_INHERITED(CSSImportRule, css::Rule)
+NS_IMPL_RELEASE_INHERITED(CSSImportRule, css::Rule)
+
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(CSSImportRule, css::Rule)
+ NS_CYCLE_COLLECTION_NOTE_EDGE_NAME(cb, "mChildSheet");
+ cb.NoteXPCOMChild(tmp->mChildSheet);
+ MOZ_ASSERT_IF(tmp->mRawRule,
+ Servo_ImportRule_GetSheet(tmp->mRawRule) == tmp->mChildSheet);
+ // Note the child sheet twice, since the Servo rule also holds a strong
+ // reference to it, but only if we're the "primary" rule reference.
+ if (tmp->mChildSheet && tmp->mChildSheet->GetOwnerRule() == tmp) {
+ NS_CYCLE_COLLECTION_NOTE_EDGE_NAME(cb, "mRawRule.stylesheet");
+ cb.NoteXPCOMChild(tmp->mChildSheet);
+ }
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
+
+NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(CSSImportRule)
+ if (tmp->mChildSheet) {
+ tmp->mChildSheet->RemoveReferencingRule(*tmp);
+ tmp->mChildSheet = nullptr;
+ }
+ tmp->mRawRule = nullptr;
+NS_IMPL_CYCLE_COLLECTION_UNLINK_END_INHERITED(css::Rule)
+
+#ifdef DEBUG
+/* virtual */
+void CSSImportRule::List(FILE* out, int32_t aIndent) const {
+ nsAutoCString str;
+ for (int32_t i = 0; i < aIndent; i++) {
+ str.AppendLiteral(" ");
+ }
+ Servo_ImportRule_Debug(mRawRule, &str);
+ fprintf_stderr(out, "%s\n", str.get());
+}
+#endif
+
+StyleCssRuleType CSSImportRule::Type() const {
+ return StyleCssRuleType::Import;
+}
+
+void CSSImportRule::SetRawAfterClone(RefPtr<RawServoImportRule> aRaw) {
+ mRawRule = std::move(aRaw);
+ if (mChildSheet) {
+ mChildSheet->RemoveReferencingRule(*this);
+ }
+ mChildSheet =
+ const_cast<StyleSheet*>(Servo_ImportRule_GetSheet(mRawRule.get()));
+ mChildSheet->AddReferencingRule(*this);
+}
+
+StyleSheet* CSSImportRule::GetStyleSheetForBindings() {
+ // FIXME(emilio): This is needed to make sure we don't expose shared sheets to
+ // the OM (see wpt /css/cssom/cssimportrule-sheet-identity.html for example).
+ //
+ // Perhaps instead we could create a clone of the stylesheet and keep it in
+ // mChildSheet, without calling EnsureUniqueInner(), or something like that?
+ if (StyleSheet* parent = GetParentStyleSheet()) {
+ parent->EnsureUniqueInner();
+ }
+ return mChildSheet;
+}
+
+dom::MediaList* CSSImportRule::GetMedia() {
+ auto* sheet = GetStyleSheetForBindings();
+ // When Bug 1326509 is fixed, we can assert mChildSheet instead.
+ return sheet ? sheet->Media() : nullptr;
+}
+
+void CSSImportRule::DropSheetReference() {
+ if (mChildSheet) {
+ mChildSheet->RemoveFromParent();
+ }
+ Rule::DropSheetReference();
+}
+
+void CSSImportRule::GetHref(nsAString& aHref) const {
+ Servo_ImportRule_GetHref(mRawRule, &aHref);
+}
+
+void CSSImportRule::GetLayerName(nsACString& aName) const {
+ Servo_ImportRule_GetLayerName(mRawRule, &aName);
+}
+
+/* virtual */
+void CSSImportRule::GetCssText(nsACString& aCssText) const {
+ Servo_ImportRule_GetCssText(mRawRule, &aCssText);
+}
+
+/* virtual */
+size_t CSSImportRule::SizeOfIncludingThis(
+ mozilla::MallocSizeOf aMallocSizeOf) const {
+ // TODO Implement this!
+ return aMallocSizeOf(this);
+}
+
+bool CSSImportRule::IsCCLeaf() const {
+ // We're not a leaf.
+ return false;
+}
+
+/* virtual */
+JSObject* CSSImportRule::WrapObject(JSContext* aCx,
+ JS::Handle<JSObject*> aGivenProto) {
+ return CSSImportRule_Binding::Wrap(aCx, this, aGivenProto);
+}
+
+} // namespace dom
+} // namespace mozilla