зеркало из https://github.com/mozilla/gecko-dev.git
140 строки
5.1 KiB
C++
140 строки
5.1 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/. */
|
|
|
|
/* State that is passed down to BindToTree. */
|
|
|
|
#ifndef mozilla_dom_BindContext_h__
|
|
#define mozilla_dom_BindContext_h__
|
|
|
|
#include "mozilla/Attributes.h"
|
|
#include "mozilla/AutoRestore.h"
|
|
#include "mozilla/dom/Document.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "mozilla/dom/ShadowRoot.h"
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
struct MOZ_STACK_CLASS BindContext final {
|
|
struct NestingLevel;
|
|
friend struct NestingLevel;
|
|
|
|
// The document that owns the tree we're getting bound to.
|
|
//
|
|
// This is mostly an optimization to avoid silly pointer-chases to get the
|
|
// OwnerDoc().
|
|
Document& OwnerDoc() const { return mDoc; }
|
|
|
|
// Whether we're getting connected.
|
|
//
|
|
// https://dom.spec.whatwg.org/#connected
|
|
bool InComposedDoc() const { return mInComposedDoc; }
|
|
|
|
// Whether we're getting bound to the document tree.
|
|
//
|
|
// https://dom.spec.whatwg.org/#in-a-document-tree
|
|
bool InUncomposedDoc() const { return mInUncomposedDoc; }
|
|
|
|
Document* GetComposedDoc() const { return mInComposedDoc ? &mDoc : nullptr; }
|
|
|
|
Document* GetUncomposedDoc() const {
|
|
return mInUncomposedDoc ? &mDoc : nullptr;
|
|
}
|
|
|
|
// Whether our subtree root is changing as a result of this operation.
|
|
bool SubtreeRootChanges() const { return mSubtreeRootChanges; }
|
|
|
|
// This constructor should be used for regular appends to content.
|
|
explicit BindContext(nsINode& aParent)
|
|
: mDoc(*aParent.OwnerDoc()),
|
|
mInComposedDoc(aParent.IsInComposedDoc()),
|
|
mInUncomposedDoc(aParent.IsInUncomposedDoc()),
|
|
mSubtreeRootChanges(true),
|
|
mCollectingDisplayedNodeDataDuringLoad(
|
|
ShouldCollectDisplayedNodeDataDuringLoad(mInComposedDoc, mDoc,
|
|
aParent)) {}
|
|
|
|
// When re-binding a shadow host into a tree, we re-bind all the shadow tree
|
|
// from the root. In that case, the shadow tree contents remain within the
|
|
// same subtree root. So children should avoid doing silly things like adding
|
|
// themselves to the ShadowRoot's id table twice or what not.
|
|
//
|
|
// This constructor is only meant to be used in that situation.
|
|
explicit BindContext(ShadowRoot& aShadowRoot)
|
|
: mDoc(*aShadowRoot.OwnerDoc()),
|
|
mInComposedDoc(aShadowRoot.IsInComposedDoc()),
|
|
mInUncomposedDoc(false),
|
|
mSubtreeRootChanges(false),
|
|
mCollectingDisplayedNodeDataDuringLoad(
|
|
ShouldCollectDisplayedNodeDataDuringLoad(mInComposedDoc, mDoc,
|
|
aShadowRoot)) {}
|
|
|
|
// This constructor is meant to be used when inserting native-anonymous
|
|
// children into a subtree.
|
|
enum ForNativeAnonymous { ForNativeAnonymous };
|
|
BindContext(Element& aParentElement, enum ForNativeAnonymous)
|
|
: mDoc(*aParentElement.OwnerDoc()),
|
|
mInComposedDoc(aParentElement.IsInComposedDoc()),
|
|
mInUncomposedDoc(aParentElement.IsInUncomposedDoc()),
|
|
mSubtreeRootChanges(true),
|
|
mCollectingDisplayedNodeDataDuringLoad(
|
|
ShouldCollectDisplayedNodeDataDuringLoad(mInComposedDoc, mDoc,
|
|
aParentElement)) {
|
|
MOZ_ASSERT(mInComposedDoc, "Binding NAC in a disconnected subtree?");
|
|
}
|
|
|
|
bool CollectingDisplayedNodeDataDuringLoad() const {
|
|
return mCollectingDisplayedNodeDataDuringLoad;
|
|
}
|
|
|
|
private:
|
|
static bool IsLikelyUndisplayed(const nsINode& aParent) {
|
|
return aParent.IsAnyOfHTMLElements(nsGkAtoms::style, nsGkAtoms::script);
|
|
}
|
|
|
|
static bool ShouldCollectDisplayedNodeDataDuringLoad(bool aConnected,
|
|
Document& aDoc,
|
|
nsINode& aParent) {
|
|
return aDoc.GetReadyStateEnum() == Document::READYSTATE_LOADING &&
|
|
aConnected && !IsLikelyUndisplayed(aParent);
|
|
}
|
|
|
|
Document& mDoc;
|
|
|
|
const bool mInComposedDoc;
|
|
const bool mInUncomposedDoc;
|
|
|
|
// Whether the bind operation will change the subtree root of the content
|
|
// we're binding.
|
|
const bool mSubtreeRootChanges;
|
|
|
|
// Whether it's likely that we're in an undisplayed part of the DOM.
|
|
//
|
|
// NOTE(emilio): We don't inherit this in BindContext's for Shadow DOM or XBL
|
|
// or such. This means that if you have a shadow tree inside an undisplayed
|
|
// element it will be incorrectly counted. But given our current definition
|
|
// of undisplayed element this is not likely to matter in practice.
|
|
bool mCollectingDisplayedNodeDataDuringLoad;
|
|
};
|
|
|
|
struct MOZ_STACK_CLASS BindContext::NestingLevel {
|
|
explicit NestingLevel(BindContext& aContext, const Element& aParent)
|
|
: mRestoreCollecting(aContext.mCollectingDisplayedNodeDataDuringLoad) {
|
|
if (aContext.mCollectingDisplayedNodeDataDuringLoad) {
|
|
aContext.mCollectingDisplayedNodeDataDuringLoad =
|
|
BindContext::IsLikelyUndisplayed(aParent);
|
|
}
|
|
}
|
|
|
|
private:
|
|
AutoRestore<bool> mRestoreCollecting;
|
|
};
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|
|
|
|
#endif
|