/* -*- 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/. */ #ifndef nsINode_h___ #define nsINode_h___ #include "mozilla/Likely.h" #include "mozilla/UniquePtr.h" #include "nsCOMPtr.h" // for member, local #include "nsGkAtoms.h" // for nsGkAtoms::baseURIProperty #include "mozilla/dom/NodeInfo.h" // member (in nsCOMPtr) #include "nsIWeakReference.h" #include "nsNodeInfoManager.h" // for use in NodePrincipal() #include "nsPropertyTable.h" // for typedefs #include "nsTObserverArray.h" // for member #include "mozilla/ErrorResult.h" #include "mozilla/LinkedList.h" #include "mozilla/MemoryReporting.h" #include "mozilla/dom/EventTarget.h" // for base class #include "js/TypeDecls.h" // for Handle, Value, JSObject, JSContext #include "mozilla/dom/DOMString.h" #include "mozilla/dom/BindingDeclarations.h" #include "mozilla/dom/NodeBinding.h" #include "nsTHashtable.h" #include // Including 'windows.h' will #define GetClassInfo to something else. #ifdef XP_WIN # ifdef GetClassInfo # undef GetClassInfo # endif #endif class AttrArray; class nsAttrChildContentList; template class nsCOMArray; class nsDOMAttributeMap; class nsIAnimationObserver; class nsIContent; class nsIContentSecurityPolicy; class nsIFrame; class nsIHTMLCollection; class nsIMutationObserver; class nsINode; class nsINodeList; class nsIPrincipal; class nsIURI; class nsNodeSupportsWeakRefTearoff; class nsDOMMutationObserver; class nsRange; class nsWindowSizes; struct RawServoSelectorList; namespace mozilla { class EventListenerManager; class PresShell; class TextEditor; namespace dom { /** * @return true if aChar is what the WHATWG defines as a 'ascii whitespace'. * https://infra.spec.whatwg.org/#ascii-whitespace */ inline bool IsSpaceCharacter(char16_t aChar) { return aChar == ' ' || aChar == '\t' || aChar == '\n' || aChar == '\r' || aChar == '\f'; } inline bool IsSpaceCharacter(char aChar) { return aChar == ' ' || aChar == '\t' || aChar == '\n' || aChar == '\r' || aChar == '\f'; } class AccessibleNode; template class AncestorsOfTypeIterator; struct BoxQuadOptions; struct ConvertCoordinateOptions; class DocGroup; class Document; class DocumentFragment; class DocumentOrShadowRoot; class DOMPoint; class DOMQuad; class DOMRectReadOnly; class Element; class EventHandlerNonNull; template class FlatTreeAncestorsOfTypeIterator; template class InclusiveAncestorsOfTypeIterator; template class InclusiveFlatTreeAncestorsOfTypeIterator; class LinkStyle; class MutationObservers; template class Optional; class OwningNodeOrString; template class Sequence; class ShadowRoot; class SVGUseElement; class Text; class TextOrElementOrDocument; struct DOMPointInit; struct GetRootNodeOptions; enum class CallerType : uint32_t; } // namespace dom } // namespace mozilla #define NODE_FLAG_BIT(n_) \ (nsWrapperCache::FlagsType(1U) << (WRAPPER_CACHE_FLAGS_BITS_USED + (n_))) enum { // This bit will be set if the node has a listener manager. NODE_HAS_LISTENERMANAGER = NODE_FLAG_BIT(0), // Whether this node has had any properties set on it NODE_HAS_PROPERTIES = NODE_FLAG_BIT(1), // Whether the node has some ancestor, possibly itself, that is native // anonymous. This includes ancestors crossing XBL scopes, in cases when an // XBL binding is attached to an element which has a native anonymous // ancestor. This flag is set-once: once a node has it, it must not be // removed. // NOTE: Should only be used on nsIContent nodes NODE_IS_IN_NATIVE_ANONYMOUS_SUBTREE = NODE_FLAG_BIT(2), // Whether this node is the root of a native anonymous (from the perspective // of its parent) subtree. This flag is set-once: once a node has it, it // must not be removed. // NOTE: Should only be used on nsIContent nodes NODE_IS_NATIVE_ANONYMOUS_ROOT = NODE_FLAG_BIT(3), NODE_IS_EDITABLE = NODE_FLAG_BIT(4), // Whether the node participates in a shadow tree. NODE_IS_IN_SHADOW_TREE = NODE_FLAG_BIT(5), // Node has an :empty or :-moz-only-whitespace selector NODE_HAS_EMPTY_SELECTOR = NODE_FLAG_BIT(6), // A child of the node has a selector such that any insertion, // removal, or appending of children requires restyling the parent. NODE_HAS_SLOW_SELECTOR = NODE_FLAG_BIT(7), // A child of the node has a :first-child, :-moz-first-node, // :only-child, :last-child or :-moz-last-node selector. NODE_HAS_EDGE_CHILD_SELECTOR = NODE_FLAG_BIT(8), // A child of the node has a selector such that any insertion or // removal of children requires restyling later siblings of that // element. Additionally (in this manner it is stronger than // NODE_HAS_SLOW_SELECTOR), if a child's style changes due to any // other content tree changes (e.g., the child changes to or from // matching :empty due to a grandchild insertion or removal), the // child's later siblings must also be restyled. NODE_HAS_SLOW_SELECTOR_LATER_SIBLINGS = NODE_FLAG_BIT(9), NODE_ALL_SELECTOR_FLAGS = NODE_HAS_EMPTY_SELECTOR | NODE_HAS_SLOW_SELECTOR | NODE_HAS_EDGE_CHILD_SELECTOR | NODE_HAS_SLOW_SELECTOR_LATER_SIBLINGS, // This node needs to go through frame construction to get a frame (or // undisplayed entry). NODE_NEEDS_FRAME = NODE_FLAG_BIT(10), // At least one descendant in the flattened tree has NODE_NEEDS_FRAME set. // This should be set on every node on the flattened tree path between the // node(s) with NODE_NEEDS_FRAME and the root content. NODE_DESCENDANTS_NEED_FRAMES = NODE_FLAG_BIT(11), // Set if the node has the accesskey attribute set. NODE_HAS_ACCESSKEY = NODE_FLAG_BIT(12), // Set if the node has right-to-left directionality NODE_HAS_DIRECTION_RTL = NODE_FLAG_BIT(13), // Set if the node has left-to-right directionality NODE_HAS_DIRECTION_LTR = NODE_FLAG_BIT(14), NODE_ALL_DIRECTION_FLAGS = NODE_HAS_DIRECTION_LTR | NODE_HAS_DIRECTION_RTL, NODE_HAS_BEEN_IN_UA_WIDGET = NODE_FLAG_BIT(15), // Set if the node has a nonce value and a header delivered CSP. NODE_HAS_NONCE_AND_HEADER_CSP = NODE_FLAG_BIT(16), NODE_KEEPS_DOMARENA = NODE_FLAG_BIT(17), // Remaining bits are node type specific. NODE_TYPE_SPECIFIC_BITS_OFFSET = 18 }; // Make sure we have space for our bits #define ASSERT_NODE_FLAGS_SPACE(n) \ static_assert(WRAPPER_CACHE_FLAGS_BITS_USED + (n) <= \ sizeof(nsWrapperCache::FlagsType) * 8, \ "Not enough space for our bits") ASSERT_NODE_FLAGS_SPACE(NODE_TYPE_SPECIFIC_BITS_OFFSET); /** * Class used to detect unexpected mutations. To use the class create an * nsMutationGuard on the stack before unexpected mutations could occur. * You can then at any time call Mutated to check if any unexpected mutations * have occurred. */ class nsMutationGuard { public: nsMutationGuard() { mStartingGeneration = sGeneration; } /** * Returns true if any unexpected mutations have occurred. You can pass in * an 8-bit ignore count to ignore a number of expected mutations. * * We don't need to care about overflow because subtraction of uint64_t's is * finding the difference between two elements of the group Z < 2^64. Once * we know the difference between two elements we only need to check that is * less than the given number of mutations to know less than that many * mutations occured. Assuming constant 1ns mutations it would take 584 * years for sGeneration to fully wrap around so we can ignore a guard living * through a full wrap around. */ bool Mutated(uint8_t aIgnoreCount) { return (sGeneration - mStartingGeneration) > aIgnoreCount; } // This function should be called whenever a mutation that we want to keep // track of happen. For now this is only done when children are added or // removed, but we might do it for attribute changes too in the future. static void DidMutate() { sGeneration++; } private: // This is the value sGeneration had when the guard was constructed. uint64_t mStartingGeneration; // This value is incremented on every mutation, for the life of the process. static uint64_t sGeneration; }; /** * A class that implements nsIWeakReference */ class nsNodeWeakReference final : public nsIWeakReference { public: explicit nsNodeWeakReference(nsINode* aNode); // nsISupports NS_DECL_ISUPPORTS // nsIWeakReference NS_DECL_NSIWEAKREFERENCE size_t SizeOfOnlyThis(mozilla::MallocSizeOf aMallocSizeOf) const override; void NoticeNodeDestruction() { mObject = nullptr; } private: ~nsNodeWeakReference(); }; // This should be used for any nsINode sub-class that has fields of its own // that it needs to measure; any sub-class that doesn't use it will inherit // AddSizeOfExcludingThis from its super-class. AddSizeOfIncludingThis() need // not be defined, it is inherited from nsINode. #define NS_DECL_ADDSIZEOFEXCLUDINGTHIS \ virtual void AddSizeOfExcludingThis(nsWindowSizes& aSizes, \ size_t* aNodeSize) const override; // IID for the nsINode interface // Must be kept in sync with xpcom/rust/xpcom/src/interfaces/nonidl.rs #define NS_INODE_IID \ { \ 0x70ba4547, 0x7699, 0x44fc, { \ 0xb3, 0x20, 0x52, 0xdb, 0xe3, 0xd1, 0xf9, 0x0a \ } \ } /** * An internal interface that abstracts some DOMNode-related parts that both * nsIContent and Document share. An instance of this interface has a list * of nsIContent children and provides access to them. */ class nsINode : public mozilla::dom::EventTarget { #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED void AssertInvariantsOnNodeInfoChange(); #endif public: typedef mozilla::dom::BoxQuadOptions BoxQuadOptions; typedef mozilla::dom::ConvertCoordinateOptions ConvertCoordinateOptions; typedef mozilla::dom::DocGroup DocGroup; typedef mozilla::dom::Document Document; typedef mozilla::dom::DOMPoint DOMPoint; typedef mozilla::dom::DOMPointInit DOMPointInit; typedef mozilla::dom::DOMQuad DOMQuad; typedef mozilla::dom::DOMRectReadOnly DOMRectReadOnly; typedef mozilla::dom::OwningNodeOrString OwningNodeOrString; typedef mozilla::dom::TextOrElementOrDocument TextOrElementOrDocument; typedef mozilla::dom::CallerType CallerType; typedef mozilla::ErrorResult ErrorResult; // XXXbz Maybe we should codegen a class holding these constants and // inherit from it... static const auto ELEMENT_NODE = mozilla::dom::Node_Binding::ELEMENT_NODE; static const auto ATTRIBUTE_NODE = mozilla::dom::Node_Binding::ATTRIBUTE_NODE; static const auto TEXT_NODE = mozilla::dom::Node_Binding::TEXT_NODE; static const auto CDATA_SECTION_NODE = mozilla::dom::Node_Binding::CDATA_SECTION_NODE; static const auto ENTITY_REFERENCE_NODE = mozilla::dom::Node_Binding::ENTITY_REFERENCE_NODE; static const auto ENTITY_NODE = mozilla::dom::Node_Binding::ENTITY_NODE; static const auto PROCESSING_INSTRUCTION_NODE = mozilla::dom::Node_Binding::PROCESSING_INSTRUCTION_NODE; static const auto COMMENT_NODE = mozilla::dom::Node_Binding::COMMENT_NODE; static const auto DOCUMENT_NODE = mozilla::dom::Node_Binding::DOCUMENT_NODE; static const auto DOCUMENT_TYPE_NODE = mozilla::dom::Node_Binding::DOCUMENT_TYPE_NODE; static const auto DOCUMENT_FRAGMENT_NODE = mozilla::dom::Node_Binding::DOCUMENT_FRAGMENT_NODE; static const auto NOTATION_NODE = mozilla::dom::Node_Binding::NOTATION_NODE; static const auto MAX_NODE_TYPE = NOTATION_NODE; void* operator new(size_t aSize, nsNodeInfoManager* aManager); void* operator new(size_t aSize) = delete; void operator delete(void* aPtr); template using Sequence = mozilla::dom::Sequence; NS_DECLARE_STATIC_IID_ACCESSOR(NS_INODE_IID) // The |aNodeSize| outparam on this function is where the actual node size // value is put. It gets added to the appropriate value within |aSizes| by // AddSizeOfNodeTree(). // // Among the sub-classes that inherit (directly or indirectly) from nsINode, // measurement of the following members may be added later if DMD finds it is // worthwhile: // - nsGenericHTMLElement: mForm, mFieldSet // - nsGenericHTMLFrameElement: mFrameLoader (bug 672539) // - HTMLBodyElement: mContentStyleRule // - HTMLDataListElement: mOptions // - HTMLFieldSetElement: mElements, mDependentElements, mFirstLegend // - HTMLFormElement: many! // - HTMLFrameSetElement: mRowSpecs, mColSpecs // - HTMLInputElement: mInputData, mFiles, mFileList, mStaticDocfileList // - nsHTMLMapElement: mAreas // - HTMLMediaElement: many! // - nsHTMLOutputElement: mDefaultValue, mTokenList // - nsHTMLRowElement: mCells // - nsHTMLSelectElement: mOptions, mRestoreState // - nsHTMLTableElement: mTBodies, mRows, mTableInheritedAttributes // - nsHTMLTableSectionElement: mRows // - nsHTMLTextAreaElement: mControllers, mState // // The following members don't need to be measured: // - nsIContent: mPrimaryFrame, because it's non-owning and measured elsewhere // virtual void AddSizeOfExcludingThis(nsWindowSizes& aSizes, size_t* aNodeSize) const; // SizeOfIncludingThis doesn't need to be overridden by sub-classes because // sub-classes of nsINode are guaranteed to be laid out in memory in such a // way that |this| points to the start of the allocated object, even in // methods of nsINode's sub-classes, so aSizes.mState.mMallocSizeOf(this) is // always safe to call no matter which object it was invoked on. void AddSizeOfIncludingThis(nsWindowSizes& aSizes, size_t* aNodeSize) const; friend class nsNodeWeakReference; friend class nsNodeSupportsWeakRefTearoff; friend class AttrArray; #ifdef MOZILLA_INTERNAL_API explicit nsINode(already_AddRefed&& aNodeInfo); #endif virtual ~nsINode(); /** * Bit-flags to pass (or'ed together) to IsNodeOfType() */ enum { /** form control elements */ eHTML_FORM_CONTROL = 1 << 6, /** SVG use targets */ eUSE_TARGET = 1 << 9, /** SVG shapes such as lines and polygons, but not images */ eSHAPE = 1 << 12 }; /** * API for doing a quick check if a content is of a given * type, such as Text, Document, Comment ... Use this when you can instead of * checking the tag. * * @param aFlags what types you want to test for (see above) * @return whether the content matches ALL flags passed in */ virtual bool IsNodeOfType(uint32_t aFlags) const = 0; bool IsContainerNode() const { return IsElement() || IsDocument() || IsDocumentFragment(); } /** * Returns true if the node is a HTMLTemplate element. */ bool IsTemplateElement() const { return IsHTMLElement(nsGkAtoms::_template); } bool IsSlotable() const { return IsElement() || IsText(); } /** * Returns true if this is a document node. */ bool IsDocument() const { // One less pointer-chase than checking NodeType(). return !GetParentNode() && IsInUncomposedDoc(); } /** * Return this node as a document. Asserts IsDocument(). * * This is defined inline in Document.h. */ inline Document* AsDocument(); inline const Document* AsDocument() const; /** * Returns true if this is a document fragment node. */ bool IsDocumentFragment() const { return NodeType() == DOCUMENT_FRAGMENT_NODE; } /** * https://dom.spec.whatwg.org/#concept-tree-inclusive-descendant * * @param aNode must not be nullptr. */ bool IsInclusiveDescendantOf(const nsINode* aNode) const; /** * https://dom.spec.whatwg.org/#concept-shadow-including-inclusive-descendant * * @param aNode must not be nullptr. */ bool IsShadowIncludingInclusiveDescendantOf(const nsINode* aNode) const; /** * Return this node as a document fragment. Asserts IsDocumentFragment(). * * This is defined inline in DocumentFragment.h. */ inline mozilla::dom::DocumentFragment* AsDocumentFragment(); inline const mozilla::dom::DocumentFragment* AsDocumentFragment() const; JSObject* WrapObject(JSContext*, JS::Handle aGivenProto) final; /** * Hook for constructing JS::ubi::Concrete specializations for memory * reporting. Specializations are defined in NodeUbiReporting.h. */ virtual void ConstructUbiNode(void* storage) = 0; /** * returns true if we are in priviliged code or * layout.css.getBoxQuads.enabled == true. */ static bool HasBoxQuadsSupport(JSContext* aCx, JSObject* /* unused */); protected: /** * WrapNode is called from WrapObject to actually wrap this node, WrapObject * does some additional checks and fix-up that's common to all nodes. WrapNode * should just call the DOM binding's Wrap function. * * aGivenProto is the prototype to use (or null if the default one should be * used) and should just be passed directly on to the DOM binding's Wrap * function. */ virtual JSObject* WrapNode(JSContext* aCx, JS::Handle aGivenProto) = 0; public: mozilla::dom::ParentObject GetParentObject() const; // Implemented in Document.h /** * Returns the first child of a node or the first child of * a template element's content if the provided node is a * template element. */ nsIContent* GetFirstChildOfTemplateOrNode(); /** * Return the scope chain parent for this node, for use in things * like event handler compilation. Returning null means to use the * global object as the scope chain parent. */ virtual nsINode* GetScopeChainParent() const; MOZ_CAN_RUN_SCRIPT mozilla::dom::Element* GetParentFlexElement(); /** * Return whether the node is an Element node. Faster than using `NodeType()`. */ bool IsElement() const { return GetBoolFlag(NodeIsElement); } virtual bool IsTextControlElement() const { return false; } // Returns non-null if this element subclasses `LinkStyle`. virtual const mozilla::dom::LinkStyle* AsLinkStyle() const { return nullptr; } mozilla::dom::LinkStyle* AsLinkStyle() { return const_cast( static_cast(this)->AsLinkStyle()); } /** * Return this node as an Element. Should only be used for nodes * for which IsElement() is true. This is defined inline in Element.h. */ inline mozilla::dom::Element* AsElement(); inline const mozilla::dom::Element* AsElement() const; /** * Return whether the node is an nsStyledElement instance or not. */ virtual bool IsStyledElement() const { return false; } /** * Return this node as nsIContent. Should only be used for nodes for which * IsContent() is true. * * The assertion in nsIContent's constructor makes this safe. */ nsIContent* AsContent() { MOZ_ASSERT(IsContent()); return reinterpret_cast(this); } const nsIContent* AsContent() const { MOZ_ASSERT(IsContent()); return reinterpret_cast(this); } /* * Return whether the node is a Text node (which might be an actual * textnode, or might be a CDATA section). */ bool IsText() const { uint32_t nodeType = NodeType(); return nodeType == TEXT_NODE || nodeType == CDATA_SECTION_NODE; } /** * Return this node as Text if it is one, otherwise null. This is defined * inline in Text.h. */ inline mozilla::dom::Text* GetAsText(); inline const mozilla::dom::Text* GetAsText() const; /** * Return this node as Text. Asserts IsText(). This is defined inline in * Text.h. */ inline mozilla::dom::Text* AsText(); inline const mozilla::dom::Text* AsText() const; /* * Return whether the node is a ProcessingInstruction node. */ bool IsProcessingInstruction() const { return NodeType() == PROCESSING_INSTRUCTION_NODE; } /* * Return whether the node is a CharacterData node (text, cdata, * comment, processing instruction) */ bool IsCharacterData() const { uint32_t nodeType = NodeType(); return nodeType == TEXT_NODE || nodeType == CDATA_SECTION_NODE || nodeType == PROCESSING_INSTRUCTION_NODE || nodeType == COMMENT_NODE; } /** * Return whether the node is a Comment node. */ bool IsComment() const { return NodeType() == COMMENT_NODE; } /** * Return whether the node is an Attr node. */ bool IsAttr() const { return NodeType() == ATTRIBUTE_NODE; } /** * Return if this node has any children. */ bool HasChildren() const { return !!mFirstChild; } /** * Get the number of children * @return the number of children */ uint32_t GetChildCount() const { return mChildCount; } /** * NOTE: this function is going to be removed soon (hopefully!) Don't use it * in new code. * * Get a child by index * @param aIndex the index of the child to get * @return the child, or null if index out of bounds */ nsIContent* GetChildAt_Deprecated(uint32_t aIndex) const; /** * Get the index of a child within this content. * * @param aPossibleChild the child to get the index of. * @return the index of the child, or -1 if not a child. Be aware that * anonymous children (e.g. a
child of an element) will * result in -1. * * If the return value is not -1, then calling GetChildAt_Deprecated() with * that value will return aPossibleChild. */ virtual int32_t ComputeIndexOf(const nsINode* aPossibleChild) const; /** * Returns the "node document" of this node. * * https://dom.spec.whatwg.org/#concept-node-document * * Note that in the case that this node is a document node this method * will return |this|. That is different to the Node.ownerDocument DOM * attribute (implemented by nsINode::GetOwnerDocument) which is specified to * be null in that case: * * https://dom.spec.whatwg.org/#dom-node-ownerdocument * * For all other cases OwnerDoc and GetOwnerDocument behave identically. */ Document* OwnerDoc() const MOZ_NONNULL_RETURN { return mNodeInfo->GetDocument(); } /** * Return the "owner document" of this node as an nsINode*. Implemented * in Document.h. */ inline nsINode* OwnerDocAsNode() const MOZ_NONNULL_RETURN; /** * Returns true if the content has an ancestor that is a document. * * @return whether this content is in a document tree */ bool IsInUncomposedDoc() const { return GetBoolFlag(IsInDocument); } /** * Get the document that this content is currently in, if any. This will be * null if the content has no ancestor that is a document. * * @return the current document */ Document* GetUncomposedDoc() const { return IsInUncomposedDoc() ? OwnerDoc() : nullptr; } /** * Returns true if we're connected, and thus GetComposedDoc() would return a * non-null value. */ bool IsInComposedDoc() const { return GetBoolFlag(IsConnected); } /** * This method returns the owner document if the node is connected to it * (as defined in the DOM spec), otherwise it returns null. * In other words, returns non-null even in the case the node is in * Shadow DOM, if there is a possibly shadow boundary crossing path from * the node to its owner document. */ Document* GetComposedDoc() const { return IsInComposedDoc() ? OwnerDoc() : nullptr; } /** * Returns OwnerDoc() if the node is in uncomposed document and ShadowRoot if * the node is in Shadow DOM and is in composed document. */ mozilla::dom::DocumentOrShadowRoot* GetUncomposedDocOrConnectedShadowRoot() const; /** * To be called when reference count of the node drops to zero. */ void LastRelease(); /** * The values returned by this function are the ones defined for * Node.nodeType */ uint16_t NodeType() const { return mNodeInfo->NodeType(); } const nsString& NodeName() const { return mNodeInfo->NodeName(); } const nsString& LocalName() const { return mNodeInfo->LocalName(); } /** * Get the NodeInfo for this element * @return the nodes node info */ inline mozilla::dom::NodeInfo* NodeInfo() const { return mNodeInfo; } /** * Called when we have been adopted, and the information of the * node has been changed. * * The new document can be reached via OwnerDoc(). * * If you override this method, * please call up to the parent NodeInfoChanged. * * If you change this, change also the similar method in Link. */ virtual void NodeInfoChanged(Document* aOldDoc) { #ifdef MOZ_DIAGNOSTIC_ASSERT_ENABLED AssertInvariantsOnNodeInfoChange(); #endif } inline bool IsInNamespace(int32_t aNamespace) const { return mNodeInfo->NamespaceID() == aNamespace; } /** * Returns the DocGroup of the "node document" of this node. */ DocGroup* GetDocGroup() const; /** * Print a debugger friendly descriptor of this element. This will describe * the position of this element in the document. */ friend std::ostream& operator<<(std::ostream& aStream, const nsINode& aNode); protected: // These 2 methods are useful for the recursive templates IsHTMLElement, // IsSVGElement, etc. inline bool IsNodeInternal() const { return false; } template inline bool IsNodeInternal(First aFirst, Args... aArgs) const { return mNodeInfo->Equals(aFirst) || IsNodeInternal(aArgs...); } public: inline bool IsHTMLElement() const { return IsElement() && IsInNamespace(kNameSpaceID_XHTML); } inline bool IsHTMLElement(const nsAtom* aTag) const { return IsElement() && mNodeInfo->Equals(aTag, kNameSpaceID_XHTML); } template inline bool IsAnyOfHTMLElements(First aFirst, Args... aArgs) const { return IsHTMLElement() && IsNodeInternal(aFirst, aArgs...); } inline bool IsSVGElement() const { return IsElement() && IsInNamespace(kNameSpaceID_SVG); } inline bool IsSVGElement(const nsAtom* aTag) const { return IsElement() && mNodeInfo->Equals(aTag, kNameSpaceID_SVG); } template inline bool IsAnyOfSVGElements(First aFirst, Args... aArgs) const { return IsSVGElement() && IsNodeInternal(aFirst, aArgs...); } inline bool IsXULElement() const { return IsElement() && IsInNamespace(kNameSpaceID_XUL); } inline bool IsXULElement(const nsAtom* aTag) const { return IsElement() && mNodeInfo->Equals(aTag, kNameSpaceID_XUL); } template inline bool IsAnyOfXULElements(First aFirst, Args... aArgs) const { return IsXULElement() && IsNodeInternal(aFirst, aArgs...); } inline bool IsMathMLElement() const { return IsElement() && IsInNamespace(kNameSpaceID_MathML); } inline bool IsMathMLElement(const nsAtom* aTag) const { return IsElement() && mNodeInfo->Equals(aTag, kNameSpaceID_MathML); } template inline bool IsAnyOfMathMLElements(First aFirst, Args... aArgs) const { return IsMathMLElement() && IsNodeInternal(aFirst, aArgs...); } bool IsShadowRoot() const { const bool isShadowRoot = IsInShadowTree() && !GetParentNode(); MOZ_ASSERT_IF(isShadowRoot, IsDocumentFragment()); return isShadowRoot; } bool IsHTMLHeadingElement() const { return IsAnyOfHTMLElements(nsGkAtoms::h1, nsGkAtoms::h2, nsGkAtoms::h3, nsGkAtoms::h4, nsGkAtoms::h5, nsGkAtoms::h6); } /** * Insert a content node before another or at the end. * This method handles calling BindToTree on the child appropriately. * * @param aKid the content to insert * @param aBeforeThis an existing node. Use nullptr if you want to * add aKid at the end. * @param aNotify whether to notify the document (current document for * nsIContent, and |this| for Document) that the insert has occurred * * @throws NS_ERROR_DOM_HIERARCHY_REQUEST_ERR if one attempts to have more * than one element node as a child of a document. Doing this will also * assert -- you shouldn't be doing it! Check with Document::GetRootElement() * first if you're not sure. Apart from this one constraint, this doesn't do * any checking on whether aKid is a valid child of |this|. * * @throws NS_ERROR_OUT_OF_MEMORY in some cases (from BindToTree). */ virtual nsresult InsertChildBefore(nsIContent* aKid, nsIContent* aBeforeThis, bool aNotify); /** * Append a content node to the end of the child list. This method handles * calling BindToTree on the child appropriately. * * @param aKid the content to append * @param aNotify whether to notify the document (current document for * nsIContent, and |this| for Document) that the append has occurred * * @throws NS_ERROR_DOM_HIERARCHY_REQUEST_ERR if one attempts to have more * than one element node as a child of a document. Doing this will also * assert -- you shouldn't be doing it! Check with Document::GetRootElement() * first if you're not sure. Apart from this one constraint, this doesn't do * any checking on whether aKid is a valid child of |this|. * * @throws NS_ERROR_OUT_OF_MEMORY in some cases (from BindToTree). */ nsresult AppendChildTo(nsIContent* aKid, bool aNotify) { return InsertChildBefore(aKid, nullptr, aNotify); } /** * Remove a child from this node. This method handles calling UnbindFromTree * on the child appropriately. * * @param aKid the content to remove * @param aNotify whether to notify the document (current document for * nsIContent, and |this| for Document) that the remove has occurred */ virtual void RemoveChildNode(nsIContent* aKid, bool aNotify); /** * Get a property associated with this node. * * @param aPropertyName name of property to get. * @param aStatus out parameter for storing resulting status. * Set to NS_PROPTABLE_PROP_NOT_THERE if the property * is not set. * @return the property. Null if the property is not set * (though a null return value does not imply the * property was not set, i.e. it can be set to null). */ void* GetProperty(const nsAtom* aPropertyName, nsresult* aStatus = nullptr) const; /** * Set a property to be associated with this node. This will overwrite an * existing value if one exists. The existing value is destroyed using the * destructor function given when that value was set. * * @param aPropertyName name of property to set. * @param aValue new value of property. * @param aDtor destructor function to be used when this property * is destroyed. * @param aTransfer if true the property will not be deleted when the * ownerDocument of the node changes, if false it * will be deleted. * * @return NS_PROPTABLE_PROP_OVERWRITTEN (success value) if the property * was already set * @throws NS_ERROR_OUT_OF_MEMORY if that occurs */ nsresult SetProperty(nsAtom* aPropertyName, void* aValue, NSPropertyDtorFunc aDtor = nullptr, bool aTransfer = false); /** * A generic destructor for property values allocated with new. */ template static void DeleteProperty(void*, nsAtom*, void* aPropertyValue, void*) { delete static_cast(aPropertyValue); } /** * Removes a property associated with this node. The value is destroyed using * the destruction function given when that value was set. * * @param aPropertyName name of property to destroy. */ void RemoveProperty(const nsAtom* aPropertyName); /** * Take a property associated with this node. The value will not be destroyed * but rather returned. It is the caller's responsibility to destroy the value * after that point. * * @param aPropertyName name of property to unset. * @param aStatus out parameter for storing resulting status. * Set to NS_PROPTABLE_PROP_NOT_THERE if the property * is not set. * @return the property. Null if the property is not set * (though a null return value does not imply the * property was not set, i.e. it can be set to null). */ void* TakeProperty(const nsAtom* aPropertyName, nsresult* aStatus = nullptr); bool HasProperties() const { return HasFlag(NODE_HAS_PROPERTIES); } /** * Return the principal of this node. This is guaranteed to never be a null * pointer. */ nsIPrincipal* NodePrincipal() const { return mNodeInfo->NodeInfoManager()->DocumentPrincipal(); } /** * Return the CSP of this node's document, if any. */ nsIContentSecurityPolicy* GetCsp() const; /** * Get the parent nsIContent for this node. * @return the parent, or null if no parent or the parent is not an nsIContent */ nsIContent* GetParent() const { return MOZ_LIKELY(GetBoolFlag(ParentIsContent)) ? mParent->AsContent() : nullptr; } /** * Get the parent nsINode for this node. This can be either an nsIContent, a * Document or an Attr. * @return the parent node */ nsINode* GetParentNode() const { return mParent; } nsINode* GetParentOrShadowHostNode() const; enum FlattenedParentType { eNotForStyle, eForStyle }; /** * Returns the node that is the parent of this node in the flattened * tree. This differs from the normal parent if the node is filtered * into an insertion point, or if the node is a direct child of a * shadow root. * * @return the flattened tree parent */ inline nsINode* GetFlattenedTreeParentNode() const; nsINode* GetFlattenedTreeParentNodeNonInline() const; /** * Like GetFlattenedTreeParentNode, but returns the document for any native * anonymous content that was generated for ancestor frames of the document * element's primary frame, such as scrollbar elements created by the root * scroll frame. */ inline nsINode* GetFlattenedTreeParentNodeForStyle() const; inline mozilla::dom::Element* GetFlattenedTreeParentElement() const; inline mozilla::dom::Element* GetFlattenedTreeParentElementForStyle() const; /** * Get the parent nsINode for this node if it is an Element. * * Defined inline in Element.h * * @return the parent node */ inline mozilla::dom::Element* GetParentElement() const; /** * Get the parent Element of this node, traversing over a ShadowRoot * to its host if necessary. */ mozilla::dom::Element* GetParentElementCrossingShadowRoot() const; /** * Get closest element node for the node. Meaning that if the node is an * element node, returns itself. Otherwise, returns parent element or null. */ inline mozilla::dom::Element* GetAsElementOrParentElement() const; /** * Get the root of the subtree this node belongs to. This never returns * null. It may return 'this' (e.g. for document nodes, and nodes that * are the roots of disconnected subtrees). */ nsINode* SubtreeRoot() const; /* * Get context object's shadow-including root if options's composed is true, * and context object's root otherwise. */ nsINode* GetRootNode(const mozilla::dom::GetRootNodeOptions& aOptions); virtual mozilla::EventListenerManager* GetExistingListenerManager() const override; virtual mozilla::EventListenerManager* GetOrCreateListenerManager() override; mozilla::Maybe GetDebuggerNotificationType() const override; bool ComputeDefaultWantsUntrusted(mozilla::ErrorResult& aRv) final; virtual bool IsApzAware() const override; virtual nsPIDOMWindowOuter* GetOwnerGlobalForBindingsInternal() override; virtual nsIGlobalObject* GetOwnerGlobal() const override; using mozilla::dom::EventTarget::DispatchEvent; bool DispatchEvent(mozilla::dom::Event& aEvent, mozilla::dom::CallerType aCallerType, mozilla::ErrorResult& aRv) override; MOZ_CAN_RUN_SCRIPT nsresult PostHandleEvent(mozilla::EventChainPostVisitor& aVisitor) override; /** * Adds a mutation observer to be notified when this node, or any of its * descendants, are modified. The node will hold a weak reference to the * observer, which means that it is the responsibility of the observer to * remove itself in case it dies before the node. If an observer is added * while observers are being notified, it may also be notified. In general, * adding observers while inside a notification is not a good idea. An * observer that is already observing the node must not be added without * being removed first. * * For mutation observers that implement nsIAnimationObserver, use * AddAnimationObserver instead. */ void AddMutationObserver(nsIMutationObserver* aMutationObserver) { nsSlots* s = Slots(); NS_ASSERTION(s->mMutationObservers.IndexOf(aMutationObserver) == nsTArray::NoIndex, "Observer already in the list"); s->mMutationObservers.AppendElement(aMutationObserver); } /** * Same as above, but only adds the observer if its not observing * the node already. * * For mutation observers that implement nsIAnimationObserver, use * AddAnimationObserverUnlessExists instead. */ void AddMutationObserverUnlessExists(nsIMutationObserver* aMutationObserver) { nsSlots* s = Slots(); s->mMutationObservers.AppendElementUnlessExists(aMutationObserver); } /** * Same as AddMutationObserver, but for nsIAnimationObservers. This * additionally records on the document that animation observers have * been registered, which is used to determine whether notifications * must be fired when animations are added, removed or changed. */ void AddAnimationObserver(nsIAnimationObserver* aAnimationObserver); /** * Same as above, but only adds the observer if its not observing * the node already. */ void AddAnimationObserverUnlessExists( nsIAnimationObserver* aAnimationObserver); /** * Removes a mutation observer. */ void RemoveMutationObserver(nsIMutationObserver* aMutationObserver) { nsSlots* s = GetExistingSlots(); if (s) { s->mMutationObservers.RemoveElement(aMutationObserver); } } nsAutoTObserverArray* GetMutationObservers() { return HasSlots() ? &GetExistingSlots()->mMutationObservers : nullptr; } /** * Helper methods to access ancestor node(s) of type T. * The implementations of the methods are in mozilla/dom/AncestorIterator.h. */ template inline mozilla::dom::AncestorsOfTypeIterator AncestorsOfType() const; template inline mozilla::dom::InclusiveAncestorsOfTypeIterator InclusiveAncestorsOfType() const; template inline mozilla::dom::FlatTreeAncestorsOfTypeIterator FlatTreeAncestorsOfType() const; template inline mozilla::dom::InclusiveFlatTreeAncestorsOfTypeIterator InclusiveFlatTreeAncestorsOfType() const; template T* FirstAncestorOfType() const; private: /** * Walks aNode, its attributes and, if aDeep is true, its descendant nodes. * If aClone is true the nodes will be cloned. If aNewNodeInfoManager is * not null, it is used to create new nodeinfos for the nodes. Also reparents * the XPConnect wrappers for the nodes into aReparentScope if non-null. * * @param aNode Node to adopt/clone. * @param aClone If true the node will be cloned and the cloned node will * be returned. * @param aDeep If true the function will be called recursively on * descendants of the node * @param aNewNodeInfoManager The nodeinfo manager to use to create new * nodeinfos for aNode and its attributes and * descendants. May be null if the nodeinfos * shouldn't be changed. * @param aReparentScope Scope into which wrappers should be reparented, or * null if no reparenting should be done. * @param aParent If aClone is true the cloned node will be appended to * aParent's children. May be null. If not null then aNode * must be an nsIContent. * @param aError The error, if any. * * @return If aClone is true then the cloned node will be returned, * unless an error occurred. In error conditions, null * will be returned. */ static already_AddRefed CloneAndAdopt( nsINode* aNode, bool aClone, bool aDeep, nsNodeInfoManager* aNewNodeInfoManager, JS::Handle aReparentScope, nsINode* aParent, mozilla::ErrorResult& aError); public: /** * Walks the node, its attributes and descendant nodes. If aNewNodeInfoManager * is not null, it is used to create new nodeinfos for the nodes. Also * reparents the XPConnect wrappers for the nodes into aReparentScope if * non-null. * * @param aNewNodeInfoManager The nodeinfo manager to use to create new * nodeinfos for the node and its attributes and * descendants. May be null if the nodeinfos * shouldn't be changed. * @param aReparentScope New scope for the wrappers, or null if no reparenting * should be done. * @param aError The error, if any. */ void Adopt(nsNodeInfoManager* aNewNodeInfoManager, JS::Handle aReparentScope, mozilla::ErrorResult& aError); /** * Clones the node, its attributes and, if aDeep is true, its descendant nodes * If aNewNodeInfoManager is not null, it is used to create new nodeinfos for * the clones. * * @param aDeep If true the function will be called recursively on * descendants of the node * @param aNewNodeInfoManager The nodeinfo manager to use to create new * nodeinfos for the node and its attributes and * descendants. May be null if the nodeinfos * shouldn't be changed. * @param aError The error, if any. * * @return The newly created node. Null in error conditions. */ already_AddRefed Clone(bool aDeep, nsNodeInfoManager* aNewNodeInfoManager, mozilla::ErrorResult& aError); /** * Clones this node. This needs to be overriden by all node classes. aNodeInfo * should be identical to this node's nodeInfo, except for the document which * may be different. When cloning an element, all attributes of the element * will be cloned. The children of the node will not be cloned. * * @param aNodeInfo the nodeinfo to use for the clone * @param aResult the clone */ virtual nsresult Clone(mozilla::dom::NodeInfo*, nsINode** aResult) const = 0; // This class can be extended by subclasses that wish to store more // information in the slots. class nsSlots { public: nsSlots(); // If needed we could remove the vtable pointer this dtor causes by // putting a DestroySlots function on nsINode virtual ~nsSlots(); virtual void Traverse(nsCycleCollectionTraversalCallback&); virtual void Unlink(); /** * A list of mutation observers */ nsAutoTObserverArray mMutationObservers; /** * An object implementing NodeList for this content (childNodes) * @see NodeList * @see nsGenericHTMLElement::GetChildNodes */ RefPtr mChildNodes; /** * Weak reference to this node. This is cleared by the destructor of * nsNodeWeakReference. */ nsNodeWeakReference* MOZ_NON_OWNING_REF mWeakReference; /** * A set of ranges which are in the selection and which have this node as * their endpoints' closest common inclusive ancestor * (https://dom.spec.whatwg.org/#concept-tree-inclusive-ancestor). This is * a UniquePtr instead of just a LinkedList, because that prevents us from * pushing DOMSlots up to the next allocation bucket size, at the cost of * some complexity. */ mozilla::UniquePtr> mClosestCommonInclusiveAncestorRanges; }; /** * Functions for managing flags and slots */ #ifdef DEBUG nsSlots* DebugGetSlots() { return Slots(); } #endif void SetFlags(FlagsType aFlagsToSet) { NS_ASSERTION( !(aFlagsToSet & (NODE_IS_NATIVE_ANONYMOUS_ROOT | NODE_IS_IN_NATIVE_ANONYMOUS_SUBTREE | NODE_DESCENDANTS_NEED_FRAMES | NODE_NEEDS_FRAME | NODE_HAS_BEEN_IN_UA_WIDGET)) || IsContent(), "Flag only permitted on nsIContent nodes"); nsWrapperCache::SetFlags(aFlagsToSet); } void UnsetFlags(FlagsType aFlagsToUnset) { NS_ASSERTION(!(aFlagsToUnset & (NODE_HAS_BEEN_IN_UA_WIDGET | NODE_IS_NATIVE_ANONYMOUS_ROOT)), "Trying to unset write-only flags"); nsWrapperCache::UnsetFlags(aFlagsToUnset); } void SetEditableFlag(bool aEditable) { if (aEditable) { SetFlags(NODE_IS_EDITABLE); } else { UnsetFlags(NODE_IS_EDITABLE); } } inline bool IsEditable() const; /** * Returns true if |this| or any of its ancestors is native anonymous. */ bool IsInNativeAnonymousSubtree() const { return HasFlag(NODE_IS_IN_NATIVE_ANONYMOUS_SUBTREE); } /** * If |this| or any ancestor is native anonymous, return the root of the * native anonymous subtree. Note that in case of nested native anonymous * content, this returns the innermost root, not the outermost. */ nsIContent* GetClosestNativeAnonymousSubtreeRoot() const { if (!IsInNativeAnonymousSubtree()) { return nullptr; } MOZ_ASSERT(IsContent(), "How did non-content end up in NAC?"); for (const nsINode* node = this; node; node = node->GetParentNode()) { if (node->IsRootOfNativeAnonymousSubtree()) { return const_cast(node)->AsContent(); } } // FIXME(emilio): This should not happen, usually, but editor removes nodes // in native anonymous subtrees, and we don't clean nodes from the current // event content stack from ContentRemoved, so it can actually happen, see // bug 1510208. NS_WARNING("GetClosestNativeAnonymousSubtreeRoot on disconnected NAC!"); return nullptr; } /** * If |this| or any ancestor is native anonymous, return the parent of the * native anonymous subtree. Note that in case of nested native anonymous * content, this returns the parent of the innermost root, not the outermost. */ nsIContent* GetClosestNativeAnonymousSubtreeRootParent() const { const nsIContent* root = GetClosestNativeAnonymousSubtreeRoot(); if (!root) { return nullptr; } // We could put this in nsIContentInlines.h or such to avoid this // reinterpret_cast, but it doesn't seem worth it. return reinterpret_cast(root)->GetParent(); } /** * Gets the root of the node tree for this content if it is in a shadow tree. */ mozilla::dom::ShadowRoot* GetContainingShadow() const; /** * Gets the shadow host if this content is in a shadow tree. That is, the host * of |GetContainingShadow|, if its not null. * * @return The shadow host, if this is in shadow tree, or null. */ nsIContent* GetContainingShadowHost() const; bool IsInSVGUseShadowTree() const { return !!GetContainingSVGUseShadowHost(); } mozilla::dom::SVGUseElement* GetContainingSVGUseShadowHost() const { if (!IsInShadowTree()) { return nullptr; } return DoGetContainingSVGUseShadowHost(); } // Whether this node has ever been part of a UA widget shadow tree. bool HasBeenInUAWidget() const { return HasFlag(NODE_HAS_BEEN_IN_UA_WIDGET); } // True for native anonymous content and for content in UA widgets. // Only nsIContent can fulfill this condition. bool ChromeOnlyAccess() const { return HasFlag(NODE_IS_IN_NATIVE_ANONYMOUS_SUBTREE | NODE_HAS_BEEN_IN_UA_WIDGET); } const nsIContent* GetChromeOnlyAccessSubtreeRootParent() const { if (!ChromeOnlyAccess()) { return nullptr; } // We can have NAC in UA widgets, but not the other way around. if (IsInNativeAnonymousSubtree()) { return GetClosestNativeAnonymousSubtreeRootParent(); } return GetContainingShadowHost(); } bool IsInShadowTree() const { return HasFlag(NODE_IS_IN_SHADOW_TREE); } /** * Get whether this node is C++-generated anonymous content * @see nsIAnonymousContentCreator * @return whether this content is anonymous */ bool IsRootOfNativeAnonymousSubtree() const { NS_ASSERTION( !HasFlag(NODE_IS_NATIVE_ANONYMOUS_ROOT) || IsInNativeAnonymousSubtree(), "Some flags seem to be missing!"); return HasFlag(NODE_IS_NATIVE_ANONYMOUS_ROOT); } // Whether this node is a UA Widget ShadowRoot. inline bool IsUAWidget() const; // Whether this node is currently in a UA Widget Shadow tree. inline bool IsInUAWidget() const; // Whether this node is the root of a ChromeOnlyAccess DOM subtree. inline bool IsRootOfChromeAccessOnlySubtree() const; /** * Returns true if |this| node is the closest common inclusive ancestor * (https://dom.spec.whatwg.org/#concept-tree-inclusive-ancestor) of the * start/end nodes of a Range in a Selection or a descendant of such a common * ancestor. This node is definitely not selected when |false| is returned, * but it may or may not be selected when |true| is returned. */ bool IsMaybeSelected() const { return IsDescendantOfClosestCommonInclusiveAncestorForRangeInSelection() || IsClosestCommonInclusiveAncestorForRangeInSelection(); } /** * Return true if any part of (this, aStartOffset) .. (this, aEndOffset) * overlaps any nsRange in * GetClosestCommonInclusiveAncestorForRangeInSelection ranges (i.e. * where this is a descendant of a range's common inclusive ancestor node). * If a nsRange starts in (this, aEndOffset) or if it ends in * (this, aStartOffset) then it is non-overlapping and the result is false * for that nsRange. Collapsed ranges always counts as non-overlapping. * * @param aStartOffset has to be less or equal to aEndOffset. */ bool IsSelected(uint32_t aStartOffset, uint32_t aEndOffset) const; /** * Get the root element of the text editor associated with this node or the * root element of the text editor of the ancestor 'TextControlElement' if * this is in its native anonymous subtree. I.e., this returns anonymous * `
` element of a `TextEditor`. Note that this can be used only for * getting root content of `` or `