/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ // vim:cindent:tabstop=2:expandtab:shiftwidth=2: /* 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 a CSS style sheet */ #ifndef mozilla_CSSStyleSheet_h #define mozilla_CSSStyleSheet_h #include "mozilla/Attributes.h" #include "mozilla/IncrementalClearCOMRuleArray.h" #include "mozilla/MemoryReporting.h" #include "mozilla/StyleSheet.h" #include "mozilla/StyleSheetInfo.h" #include "mozilla/css/SheetParsingMode.h" #include "mozilla/dom/Element.h" #include "mozilla/dom/CSSStyleSheetBinding.h" #include "nscore.h" #include "nsCOMPtr.h" #include "nsAutoPtr.h" #include "nsIDOMCSSStyleSheet.h" #include "nsICSSLoaderObserver.h" #include "nsTArrayForwardDeclare.h" #include "nsString.h" #include "mozilla/CORSMode.h" #include "nsCycleCollectionParticipant.h" #include "nsWrapperCache.h" #include "mozilla/net/ReferrerPolicy.h" #include "mozilla/dom/SRIMetadata.h" class CSSRuleListImpl; class nsCSSRuleProcessor; class nsIPrincipal; class nsIURI; class nsMediaList; class nsMediaQueryResultCacheKey; class nsStyleSet; class nsPresContext; class nsXMLNameSpaceMap; namespace mozilla { struct ChildSheetListBuilder; class CSSStyleSheet; namespace css { class Rule; class GroupRule; class ImportRule; } // namespace css namespace dom { class CSSRuleList; } // namespace dom // ------------------------------- // CSS Style Sheet Inner Data Container // class CSSStyleSheetInner : public StyleSheetInfo { public: friend class mozilla::CSSStyleSheet; friend class ::nsCSSRuleProcessor; private: CSSStyleSheetInner(CSSStyleSheet* aPrimarySheet, CORSMode aCORSMode, ReferrerPolicy aReferrerPolicy, const dom::SRIMetadata& aIntegrity); CSSStyleSheetInner(CSSStyleSheetInner& aCopy, CSSStyleSheet* aPrimarySheet); ~CSSStyleSheetInner(); CSSStyleSheetInner* CloneFor(CSSStyleSheet* aPrimarySheet); void AddSheet(CSSStyleSheet* aSheet); void RemoveSheet(CSSStyleSheet* aSheet); void RebuildNameSpaces(); // Create a new namespace map nsresult CreateNamespaceMap(); size_t SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const; AutoTArray mSheets; IncrementalClearCOMRuleArray mOrderedRules; nsAutoPtr mNameSpaceMap; // Linked list of child sheets. This is al fundamentally broken, because // each of the child sheets has a unique parent... We can only hope (and // currently this is the case) that any time page JS can get ts hands on a // child sheet that means we've already ensured unique inners throughout its // parent chain and things are good. RefPtr mFirstChild; }; // ------------------------------- // CSS Style Sheet // // CID for the CSSStyleSheet class // 7985c7ac-9ddc-444d-9899-0c86ec122f54 #define NS_CSS_STYLE_SHEET_IMPL_CID \ { 0x7985c7ac, 0x9ddc, 0x444d, \ { 0x98, 0x99, 0x0c, 0x86, 0xec, 0x12, 0x2f, 0x54 } } class CSSStyleSheet final : public nsIDOMCSSStyleSheet, public nsICSSLoaderObserver, public nsWrapperCache, public StyleSheet { public: typedef net::ReferrerPolicy ReferrerPolicy; CSSStyleSheet(CORSMode aCORSMode, ReferrerPolicy aReferrerPolicy); CSSStyleSheet(CORSMode aCORSMode, ReferrerPolicy aReferrerPolicy, const dom::SRIMetadata& aIntegrity); NS_DECL_CYCLE_COLLECTING_ISUPPORTS NS_DECL_CYCLE_COLLECTION_SCRIPT_HOLDER_CLASS_AMBIGUOUS(CSSStyleSheet, nsIDOMCSSStyleSheet) NS_DECLARE_STATIC_IID_ACCESSOR(NS_CSS_STYLE_SHEET_IMPL_CID) nsIURI* GetSheetURI() const; nsIURI* GetBaseURI() const; void GetTitle(nsString& aTitle) const; void GetType(nsString& aType) const; bool HasRules() const; /** * Whether the sheet is applicable. A sheet that is not applicable * should never be inserted into a style set. A sheet may not be * applicable for a variety of reasons including being disabled and * being incomplete. */ bool IsApplicable() const; /** * Set the stylesheet to be enabled. This may or may not make it * applicable. Note that this WILL inform the sheet's document of * its new applicable state if the state changes but WILL NOT call * BeginUpdate() or EndUpdate() on the document -- calling those is * the caller's responsibility. This allows use of SetEnabled when * batched updates are desired. If you want updates handled for * you, see nsIDOMStyleSheet::SetDisabled(). */ void SetEnabled(bool aEnabled); // style sheet owner info CSSStyleSheet* GetParentSheet() const; // may be null nsIDocument* GetOwningDocument() const; // may be null void SetOwningDocument(nsIDocument* aDocument); // Find the ID of the owner inner window. uint64_t FindOwningWindowInnerID() const; #ifdef DEBUG void List(FILE* out = stdout, int32_t aIndent = 0) const; #endif void AppendStyleSheet(CSSStyleSheet* aSheet); // XXX do these belong here or are they generic? void AppendStyleRule(css::Rule* aRule); int32_t StyleRuleCount() const; css::Rule* GetStyleRuleAt(int32_t aIndex) const; nsresult DeleteRuleFromGroup(css::GroupRule* aGroup, uint32_t aIndex); nsresult InsertRuleIntoGroup(const nsAString& aRule, css::GroupRule* aGroup, uint32_t aIndex, uint32_t* _retval); /** * SetURIs must be called on all sheets before parsing into them. * SetURIs may only be called while the sheet is 1) incomplete and 2) * has no rules in it */ void SetURIs(nsIURI* aSheetURI, nsIURI* aOriginalSheetURI, nsIURI* aBaseURI); /** * SetPrincipal should be called on all sheets before parsing into them. * This can only be called once with a non-null principal. Calling this with * a null pointer is allowed and is treated as a no-op. */ void SetPrincipal(nsIPrincipal* aPrincipal); // Principal() never returns a null pointer. nsIPrincipal* Principal() const { return mInner->mPrincipal; } void SetTitle(const nsAString& aTitle) { mTitle = aTitle; } void SetMedia(nsMediaList* aMedia); void SetOwnerRule(css::ImportRule* aOwnerRule) { mOwnerRule = aOwnerRule; /* Not ref counted */ } css::ImportRule* GetOwnerRule() const { return mOwnerRule; } nsXMLNameSpaceMap* GetNameSpaceMap() const { return mInner->mNameSpaceMap; } already_AddRefed Clone(CSSStyleSheet* aCloneParent, css::ImportRule* aCloneOwnerRule, nsIDocument* aCloneDocument, nsINode* aCloneOwningNode) const; bool IsModified() const { return mDirty; } void SetModifiedByChildRule() { NS_ASSERTION(mDirty, "sheet must be marked dirty before handing out child rules"); DidDirty(); } nsresult AddRuleProcessor(nsCSSRuleProcessor* aProcessor); nsresult DropRuleProcessor(nsCSSRuleProcessor* aProcessor); void AddStyleSet(nsStyleSet* aStyleSet); void DropStyleSet(nsStyleSet* aStyleSet); /** * Like the DOM insertRule() method, but doesn't do any security checks */ nsresult InsertRuleInternal(const nsAString& aRule, uint32_t aIndex, uint32_t* aReturn); /* Get the URI this sheet was originally loaded from, if any. Can return null */ nsIURI* GetOriginalURI() const { return mInner->mOriginalSheetURI; } // Whether the sheet is for an inline