mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:18:48 +00:00
e39f9f88f7
- Bug 1152171 part 2 - Rename AnimationTimeline to DocumentTimeline; r=smaug (26c118319) - Bug 1152171 part 3 - Update web-platform-tests expectations; r=jgraham (b7b4032aa) - Bug 1153734 part 1 - Remove AnimationEffect; r=smaug (9cf67a02e) - Bug 1153734 part 2 - Rename Animation to KeyframeEffectReadonly; r=smaug (b69556ee6) - Bug 1153734 part 3 - Rename AnimationPlayer.source to AnimationPlayer.effect; r=smaug (50d3130ee) - Bug 1153734 part 4 - Rename other uses of 'source' and 'source content'; r=jwatt (b02c4ba36) - Bug 1153734 part 5 - Add AnimationEffectReadonly as a superinterface of KeyframeEffectReadonly; r=smaug (c3395d3f5) - Bug 1149990 - Support replaying of finished CSS transitions by supporting setting of currentTime/startTime. r=birtles (3fb2cb401) - Bug 1154615 part 1 - Rename AnimationPlayer to Animation in WebIDL; r=smaug (6c2125b49) - Bug 1154615 part 2 - Rename PendingPlayerTracker to PendingAnimationTracker; r=jwatt (8d6804def) - Bug 1154615 part 3 - Rename internal members of PendingAnimationTracker; r=jwatt (f348f6355) - Bug 1154615 part 4 - Rename references to players in dom/animation; r=jwatt (0250572e8) - Bug 1117603 part 1 - Don't assume style rules have been refreshed in GetAnimationRule; r=dbaron (a5d340d0f) - remove kungFuDeathGrip (49df758e6) - Bug 1117603 part 2 - Don't unregister from the refresh driver unless we are also queueing events; r=dbaron (715c9caa1) - Bug 1154615 part 5 - Rename AnimationPlayerCollection to AnimationCollection; r=jwatt (4c596f089) - Bug 1154615 part 6 - Rename references to players within layout/; r=jwatt (42405f3fc) - Bug 1154615 part 7 - Rename CSSAnimationPlayer and CSSTransitionPlayer; r=jwatt (49ab272ed) - Bug 1154615 part 8 - Rename references to players in animation observers; r=jwatt (c3fa26d7a) - Bug 1154615 part 9 - Rename test files; r=jwatt (9d9f03e7b) - Bug 1145439 (Part 1) - Throttle requestAnimationFrame for non-visible iframes. r=mstange,mchang (be7d183d6) - Bug 1145439 (Part 2) - Make test_scroll_event_ordering.html wait for rAF to unthrottle. r=roc (9ac8317c9) - Bug 1144324 - Try to register for, and handle, touch events when APZ is enabled. r=dvander,jimm (fb75d1665) - Bug 1144324 - Remove the codepaths that conditionally enable touch events based on touch the presence of touch listeners. r=smaug,jimm (710617e6b) - Bug 1003991 - Disable https:// only load for ServiceWorkers when Developer Tools are open. r=nsm, r=miker (9d6669814) - Bug 1153267 - part 1 - use smart-pointer .forget() instead of NS_ADDREF+assign; r=ehsan (e4555c90c) - Bug 1153267 - part 2 - use smart pointers instead of manual NS_ADDREF'ing outparams; r=ehsan (ae8b60d5a) - Bug 1153267 - fix typo that broke OS X builds on a CLOSED TREE; r=bustage (08fdb3c4f) - Bug 1146843 - Revert part of cset 33c30e283fa8 because the code is used in Fennec. r=snorp (407248257) - Bug 1151940 part 1. Make some readonly properties defined on Window by CSSOM-view replaceable. r=smaug (5cb9b91f0) - Bug 1151940 part 2. Add a convenience function in nsGlobalWindow for replacing a property on the window with a new value. r=smaug (2ba39331c) - Bug 1151940 part 3. Make some writable cssom-view attributes that we only allow setting from chrome act the way readonly replaceables would when called from content. r=smaug (b485e1b44) - Goanna -> Gecko (2c539d7be) - Goanna -> Gecko (25d34e213) - Bug 1148962 - Use TakeOwnershipOfErrorReporting in CPOW code (r=bholley) (96c997639) - pointer style (a07fbffaa) - Bug 1152577: Add 'aReason' argument to AutoEntryScript constructor, and provide plausible names for its instantiations. r=bholley (512fa27e2) - bug 1155691 - Expose WindowRoot to chrome from window in webidl. r=smaug (235281924) - Bug 404828 - No need to assert that the top window isn't reachable. r=smaug (d73154fa0) - Bug 404828 - Followup: remove assertion expectations on a CLOSED TREE. a=tomcat (a5dabe1b7) - Bug 1156102 - Mark nsGlobalWindowObserver::mWindow as MOZ_NON_OWNING_REF; r=baku (c0d4208b7) - Bug 1107801 - Improve gamepad support on MacOS. r=ted (c591bd5ac) - Goanna -> Gecko (d9b81bc9e) - Bug 852944 - Gamepad API IPC; r=ted, r=baku (521892538) - Bug 1143529 part 1. Stop manually calling WrapObject in DataStoreService::GetDataStoresResolve. r=baku (056ad6bfe) - Bug 1143529 part 2. Tighten up the assert in binding Wrap methods. r=peterv (765a13325) - Bug 1152169 - DataStoreService should check if the first revision exists, r=bent (ee371cc5d) - Bug 1152169 followup: Mark FirstRevisionIdCallback methods Run() and HandleEvent() as 'override'. rs=ehsan (8186c4168) - Bug 1143651 - don't use CallQueryInterface when the compiler can do the cast for us; r=ehsan (a50f0a54b) - Bug 1144322 - Handle tabindex in overridden IsInteractiveHTMLContent methods. r=smaug (fd4b9beed) - Bug 1086684 - Stash the full path for file inputs to avoid doing IPC at inopportune times. r=ehsan/bent/gps (b843b1efc) - Bug 1143934 - Disallow mozSetFileNameArray in content processes. r=ehsan (42e5c8c6d) - Bug 1143934 - Fix assorted forms mochitests for e10s-compatibility. r=smaug (7a3babfed) - Bug 1143934 - Work around SessionStore dependency on current brokenness. r=ttaubert (5b0fcb5ce) - Bug 956530 - Clear the delayed caret data when clicking on a selected part of a text control if the focus event handler selects the control; r=roc (2859f07b4) - Bug 956530 follow-up: Fix the test failure on Windows 8 caused by the text box having a glowing outline as a result of being clicked on (d34e8da1a) - Bug 1157898 part 1. Make code of the form "return rv.ErrorCode();" where rv is an ErrorResult use StealNSResult instead. r=peterv (800da50e2) - Bug 1157898 part 2. Make code of the form "NS_ENSURE_SUCCESS(rv.ErrorCode(), rv.ErrorCode());" use Failed and StealNSResult instead. r=peterv (472432a83) - Bug 1157898 part 3. Fix the remaining consumers of rv.ErrorCode() in NS_ENSURE_* expressions to not do that. r=peterv (d452807e7) - Bug 1122238 part 1. Switch to using the new stackframe APIs in JSStackFrame. r=bholley (9d87b261a) - Bug 1122238 part 2. Stop caching things in JSStackFrame when we're called over Xrays. r=bholley (83eda7275) - Bug 1122238 part 3. Drop all the DOMException-cloning and sanitization gunk we added in bug 1107592 and bug 1107953 and bug 1117242 . r=bholley (f237aa948) - add support for NetBSD/SPARC64 (065783b70) - Bug 1153484 - Fetch should ignore invalid headers, but still process later headers. r=nsm (8925ddd77) - Bug 1157754 part 2. Convert consumers of ErrorResult::ClearMessage() to the new better APIs we have for suppressing exceptions on ErrorResult. r=bkelly (6519fbd5e) - Bug 1157754 part 3. Make ClearMessage private on ErrorResult. r=peterv (3fb218692) - Bug 1157898 part 4. Add ErrorResult::ErrorCodeIs() and use it in various places to get rid of ErrorCode(). r=peterv (bed7bfb4c) - Bug 1130686 - Refactor PromiseHolder in the service worker clients code. r=nsm (b3dbdcbfe) - Bug 1130686 - Implement client.focus. r=baku (5dee6d850) - Bug 1149163 part 1 - Clean up nsHTMLEditRules::GetInnerContent; r=froydnj (cc8f65b54) - Bug 1149163 part 2 - Make nsDOMIterator infallible; r=froydnj (d975f6c62) - Bug 1149163 part 3 - Clean up nsHTMLEditRules::BustUpInlinesAtBRs; r=froydnj (58155adad) - Bug 1149163 part 4 - Allow use of temporary nsBoolDomIterFunctor; r=froydnj (dbafec00f) - Bug 1149163 part 5 - Clean up nsHTMLEditRules::GetNodesForOperation; r=froydnj (41179d810) - Bug 1149163 part 6 - Clean up nsHTMLEditRules::LookInsideDivBQandList; r=froydnj (0b757bf14) - Bug 1149163 part 7 - Clean up nsHTMLEditRules::PromoteRange; r=froydnj (c49c714b1) - Bug 1149163 part 8 - Clean up nsHTMLEditRules::GetPromotedRanges; r=froydnj (5163a0026) - Bug 1148228 - Stop checking ul twice (43a22088c) - Bug 1141017 - resurrect serif and monospace. r=ehsan (95a1b6fcf) - Bug 1147412 part 1 - Make methods take nsINode*, not just nsIContent*; r=ehsan (7f762cdbe) - Bug 1147412 part 2 - Clean up nsHTMLEditor::SetInlinePropertyOnTextNode; r=ehsan (faf805587) - Bug 1147412 part 3 - Fix completely broken nsHTMLCSSUtils::IsCSSEquivalentToHTMLInlineStyleSet implementation; r=ehsan (73fea67c1) - Bug 1147412 part 4 - Clean up nsHTMLEditor::GetInlinePropertyBase; r=ehsan (3265bfbce) - Bug 1147412 part 5 - Clean up nsHTMLEditor::RemoveInlinePropertyImpl; r=ehsan (0f402bd7e) - Bug 1147412 part 6 - Remove nsHTMLCSSUtils::IsCSSEditableProperty(nsIDOMNode*,...); r=ehsan (100e4038a) - Bug 1147412 part 7 - Remove nsHTMLCSSUtils::GetComputedStyle(nsIDOMElement*); r=ehsan (6c51103bc) - Bug 1147412 part 8 - Clean up nsHTMLCSSUtils::IsCSSInvertible; r=ehsan (01e60c446) - Bug 1147412 part 9 - Convert some nsHTMLEditor members to Element; r=ehsan (e7efb1ac4) - Bug 1147412 part 10 - Clean up nsHTMLCSSUtils::Get*Property, GetCSSInlinePropertyBase; r=ehsan (54154143d) - Bug 1149163 part 9 - Clean up nsHTMLEditRules::GetNodesFromSelection; r=froydnj (5186308b9) - Bug 1154701 part 1 - Clean up nsHTMLEditor::CreateListOfNodesToPaste; r=ehsan (ea95238d5) - Bug 1153629 part 1 - Clean up nsHTMLEditRules::GetListActionNodes; r=ehsan (51f3b3e95) - Bug 1153629 part 2 - Clean up nsHTMLEditRules::GetParagraphFormatNodes; r=ehsan (a27bd7751) - Bug 1153629 part 3 - Clean up nsHTMLEditRules::GetNodesFromPoint; r=ehsan (edc7e4561) - Bug 1153629 part 4 - Clean up nsHTMLEditRules::ListIsEmptyLine; r=ehsan (ce3289bc7) - Bug 1153629 part 5 - Clean up nsHTMLEditRules::GetChildNodesForOperation; r=ehsan (b3a509dbf) - Bug 1153629 part 6 - Clean up nsHTMLEditRules::MakeBlockquote; r=ehsan (cb3808182) - Bug 1153629 part 7 - Clean up nsHTMLEditRules::RemoveBlockStyle, RemovePartOfBlock; r=ehsan (660b9f76e) - Bug 1153629 part 8 - Clean up nsHTMLEditRules::ApplyBlockStyle; r=ehsan (f54f9538c) - Bug 1153629 part 9 - Clean up nsHTMLEditRules::MakeTransitionList; r=ehsan (fb63cf6d8) - Bug 1153629 part 10 - Clean up nsHTMLEditRules::AlignInnerBlocks; r=ehsan (752d2df7a) - Bug 1153629 part 11 - Clean up nsHTMLEditRules::AdjustSpecialBreaks; r=ehsan (16ef0416b) - Bug 1153629 part 12 - Clean up nsHTMLEditRules::RemoveEmptyNodes; r=ehsan (d528e70e6) - Bug 1154701 part 2 - Use more OwningNonNull in editor; r=ehsan (85b1929e6) - Bug 1154701 part 3 - Clean up nsHTMLEditor::GetListAndTableParents, DiscoverPartialListsAndTables, ScanForListAndTableStructure, ReplaceOrphanedStructure; r=ehsan (7fe31f058) - Bug 1154701 part 4 - Switch nsHTMLEditor::mContentFilters to nsTArray; r=ehsan (64e6dd160) - Bug 1154701 part 5 - Switch nsHTMLEditor::objectResizeEventListeners to nsTArray; r=ehsan (036bc65fe) - Bug 1154701 part 6 - Clean up nsHTMLEditor::SetInlinePropertyOnNodeImpl; r=ehsan (2d619ca16) - Bug 1154701 part 7 - Clean up nsHTMLEditor::SetInlineProperty; r=ehsan (7a367d31b) - Bug 1154701 part 8 - Clean up nsHTMLEditor::SetInlinePropertyOnNode; r=ehsan (707c07d93) - Bug 1154701 part 9 - Clean up nsHTMLEditor::RelativeFontChange; r=ehsan (273ae9c64) - Bug 1154701 part 10 - Switch nsEditor::mActionListeners to nsTArray; r=ehsan (d2b5732fe) - Bug 1154701 part 11 - Switch nsEditor::mEditorObservers to nsTArray; r=ehsan (25a5af12e) - Bug 1154701 part 12 - Switch nsEditor::mDocStateListeners to nsTArray; r=ehsan (665af0792) - Bug 1154701 part 13 - Clean up nsHTMLEditor::SetCSSBackgroundColor; r=ehsan (ba424ade8) - Bug 1154701 part 14 - Remove unused nsCOMArray cruft; r=ehsan (3a8679a67) - Bug 1101651 - Part 1: xpcomrt version of dom media library need for standalone webrtcs. r=jesup (ae37b5464) - Bug 1137447 - New app update telemetry for patch type (complete or partial), extended error codes, and general cleanup. r=bbondy (c736ae502)
683 lines
24 KiB
C++
683 lines
24 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=2 sw=2 et tw=99: */
|
|
/* 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 "nsNodeUtils.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsINode.h"
|
|
#include "nsIContent.h"
|
|
#include "mozilla/dom/Element.h"
|
|
#include "nsIMutationObserver.h"
|
|
#include "nsIDocument.h"
|
|
#include "mozilla/EventListenerManager.h"
|
|
#include "nsIXPConnect.h"
|
|
#include "pldhash.h"
|
|
#include "nsIDOMAttr.h"
|
|
#include "nsCOMArray.h"
|
|
#include "nsPIDOMWindow.h"
|
|
#include "nsDocument.h"
|
|
#ifdef MOZ_XUL
|
|
#include "nsXULElement.h"
|
|
#endif
|
|
#include "nsBindingManager.h"
|
|
#include "nsGenericHTMLElement.h"
|
|
#include "mozilla/Assertions.h"
|
|
#include "mozilla/dom/Animation.h"
|
|
#include "mozilla/dom/HTMLImageElement.h"
|
|
#include "mozilla/dom/HTMLMediaElement.h"
|
|
#include "mozilla/dom/KeyframeEffect.h"
|
|
#include "nsWrapperCacheInlines.h"
|
|
#include "nsObjectLoadingContent.h"
|
|
#include "nsDOMMutationObserver.h"
|
|
#include "mozilla/dom/BindingUtils.h"
|
|
#include "mozilla/dom/HTMLTemplateElement.h"
|
|
#include "mozilla/dom/ShadowRoot.h"
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::dom;
|
|
using mozilla::AutoJSContext;
|
|
|
|
// This macro expects the ownerDocument of content_ to be in scope as
|
|
// |nsIDocument* doc|
|
|
#define IMPL_MUTATION_NOTIFICATION(func_, content_, params_) \
|
|
PR_BEGIN_MACRO \
|
|
bool needsEnterLeave = doc->MayHaveDOMMutationObservers(); \
|
|
if (needsEnterLeave) { \
|
|
nsDOMMutationObserver::EnterMutationHandling(); \
|
|
} \
|
|
nsINode* node = content_; \
|
|
NS_ASSERTION(node->OwnerDoc() == doc, "Bogus document"); \
|
|
if (doc) { \
|
|
doc->BindingManager()->func_ params_; \
|
|
} \
|
|
do { \
|
|
nsINode::nsSlots* slots = node->GetExistingSlots(); \
|
|
if (slots && !slots->mMutationObservers.IsEmpty()) { \
|
|
/* No need to explicitly notify the first observer first \
|
|
since that'll happen anyway. */ \
|
|
NS_OBSERVER_ARRAY_NOTIFY_OBSERVERS( \
|
|
slots->mMutationObservers, nsIMutationObserver, \
|
|
func_, params_); \
|
|
} \
|
|
ShadowRoot* shadow = ShadowRoot::FromNode(node); \
|
|
if (shadow) { \
|
|
node = shadow->GetPoolHost(); \
|
|
} else { \
|
|
node = node->GetParentNode(); \
|
|
} \
|
|
} while (node); \
|
|
if (needsEnterLeave) { \
|
|
nsDOMMutationObserver::LeaveMutationHandling(); \
|
|
} \
|
|
PR_END_MACRO
|
|
|
|
#define IMPL_ANIMATION_NOTIFICATION(func_, content_, params_) \
|
|
PR_BEGIN_MACRO \
|
|
bool needsEnterLeave = doc->MayHaveDOMMutationObservers(); \
|
|
if (needsEnterLeave) { \
|
|
nsDOMMutationObserver::EnterMutationHandling(); \
|
|
} \
|
|
nsINode* node = content_; \
|
|
do { \
|
|
nsINode::nsSlots* slots = node->GetExistingSlots(); \
|
|
if (slots && !slots->mMutationObservers.IsEmpty()) { \
|
|
/* No need to explicitly notify the first observer first \
|
|
since that'll happen anyway. */ \
|
|
NS_OBSERVER_ARRAY_NOTIFY_OBSERVERS_WITH_QI( \
|
|
slots->mMutationObservers, nsIMutationObserver, \
|
|
nsIAnimationObserver, func_, params_); \
|
|
} \
|
|
ShadowRoot* shadow = ShadowRoot::FromNode(node); \
|
|
if (shadow) { \
|
|
node = shadow->GetPoolHost(); \
|
|
} else { \
|
|
node = node->GetParentNode(); \
|
|
} \
|
|
} while (node); \
|
|
if (needsEnterLeave) { \
|
|
nsDOMMutationObserver::LeaveMutationHandling(); \
|
|
} \
|
|
PR_END_MACRO
|
|
|
|
void
|
|
nsNodeUtils::CharacterDataWillChange(nsIContent* aContent,
|
|
CharacterDataChangeInfo* aInfo)
|
|
{
|
|
nsIDocument* doc = aContent->OwnerDoc();
|
|
IMPL_MUTATION_NOTIFICATION(CharacterDataWillChange, aContent,
|
|
(doc, aContent, aInfo));
|
|
}
|
|
|
|
void
|
|
nsNodeUtils::CharacterDataChanged(nsIContent* aContent,
|
|
CharacterDataChangeInfo* aInfo)
|
|
{
|
|
nsIDocument* doc = aContent->OwnerDoc();
|
|
IMPL_MUTATION_NOTIFICATION(CharacterDataChanged, aContent,
|
|
(doc, aContent, aInfo));
|
|
}
|
|
|
|
void
|
|
nsNodeUtils::AttributeWillChange(Element* aElement,
|
|
int32_t aNameSpaceID,
|
|
nsIAtom* aAttribute,
|
|
int32_t aModType)
|
|
{
|
|
nsIDocument* doc = aElement->OwnerDoc();
|
|
IMPL_MUTATION_NOTIFICATION(AttributeWillChange, aElement,
|
|
(doc, aElement, aNameSpaceID, aAttribute,
|
|
aModType));
|
|
}
|
|
|
|
void
|
|
nsNodeUtils::AttributeChanged(Element* aElement,
|
|
int32_t aNameSpaceID,
|
|
nsIAtom* aAttribute,
|
|
int32_t aModType)
|
|
{
|
|
nsIDocument* doc = aElement->OwnerDoc();
|
|
IMPL_MUTATION_NOTIFICATION(AttributeChanged, aElement,
|
|
(doc, aElement, aNameSpaceID, aAttribute,
|
|
aModType));
|
|
}
|
|
|
|
void
|
|
nsNodeUtils::AttributeSetToCurrentValue(Element* aElement,
|
|
int32_t aNameSpaceID,
|
|
nsIAtom* aAttribute)
|
|
{
|
|
nsIDocument* doc = aElement->OwnerDoc();
|
|
IMPL_MUTATION_NOTIFICATION(AttributeSetToCurrentValue, aElement,
|
|
(doc, aElement, aNameSpaceID, aAttribute));
|
|
}
|
|
|
|
void
|
|
nsNodeUtils::ContentAppended(nsIContent* aContainer,
|
|
nsIContent* aFirstNewContent,
|
|
int32_t aNewIndexInContainer)
|
|
{
|
|
nsIDocument* doc = aContainer->OwnerDoc();
|
|
|
|
IMPL_MUTATION_NOTIFICATION(ContentAppended, aContainer,
|
|
(doc, aContainer, aFirstNewContent,
|
|
aNewIndexInContainer));
|
|
}
|
|
|
|
void
|
|
nsNodeUtils::ContentInserted(nsINode* aContainer,
|
|
nsIContent* aChild,
|
|
int32_t aIndexInContainer)
|
|
{
|
|
NS_PRECONDITION(aContainer->IsNodeOfType(nsINode::eCONTENT) ||
|
|
aContainer->IsNodeOfType(nsINode::eDOCUMENT),
|
|
"container must be an nsIContent or an nsIDocument");
|
|
nsIContent* container;
|
|
nsIDocument* doc = aContainer->OwnerDoc();
|
|
nsIDocument* document;
|
|
if (aContainer->IsNodeOfType(nsINode::eCONTENT)) {
|
|
container = static_cast<nsIContent*>(aContainer);
|
|
document = doc;
|
|
}
|
|
else {
|
|
container = nullptr;
|
|
document = static_cast<nsIDocument*>(aContainer);
|
|
}
|
|
|
|
IMPL_MUTATION_NOTIFICATION(ContentInserted, aContainer,
|
|
(document, container, aChild, aIndexInContainer));
|
|
}
|
|
|
|
void
|
|
nsNodeUtils::ContentRemoved(nsINode* aContainer,
|
|
nsIContent* aChild,
|
|
int32_t aIndexInContainer,
|
|
nsIContent* aPreviousSibling)
|
|
{
|
|
NS_PRECONDITION(aContainer->IsNodeOfType(nsINode::eCONTENT) ||
|
|
aContainer->IsNodeOfType(nsINode::eDOCUMENT),
|
|
"container must be an nsIContent or an nsIDocument");
|
|
nsIContent* container;
|
|
nsIDocument* doc = aContainer->OwnerDoc();
|
|
nsIDocument* document;
|
|
if (aContainer->IsNodeOfType(nsINode::eCONTENT)) {
|
|
container = static_cast<nsIContent*>(aContainer);
|
|
document = doc;
|
|
}
|
|
else {
|
|
container = nullptr;
|
|
document = static_cast<nsIDocument*>(aContainer);
|
|
}
|
|
|
|
IMPL_MUTATION_NOTIFICATION(ContentRemoved, aContainer,
|
|
(document, container, aChild, aIndexInContainer,
|
|
aPreviousSibling));
|
|
}
|
|
|
|
static inline Element*
|
|
GetTarget(Animation* aAnimation)
|
|
{
|
|
KeyframeEffectReadonly* effect = aAnimation->GetEffect();
|
|
if (!effect) {
|
|
return nullptr;
|
|
}
|
|
|
|
Element* target;
|
|
nsCSSPseudoElements::Type pseudoType;
|
|
effect->GetTarget(target, pseudoType);
|
|
|
|
// If the animation targets a pseudo-element, we don't dispatch
|
|
// notifications for it. (In the future we will have PseudoElement
|
|
// objects we can use as the target of the notifications.)
|
|
if (pseudoType != nsCSSPseudoElements::ePseudo_NotPseudoElement) {
|
|
return nullptr;
|
|
}
|
|
|
|
return effect->GetTarget();
|
|
}
|
|
|
|
void
|
|
nsNodeUtils::AnimationAdded(Animation* aAnimation)
|
|
{
|
|
Element* target = GetTarget(aAnimation);
|
|
if (!target) {
|
|
return;
|
|
}
|
|
nsIDocument* doc = target->OwnerDoc();
|
|
|
|
if (doc->MayHaveAnimationObservers()) {
|
|
IMPL_ANIMATION_NOTIFICATION(AnimationAdded, target, (aAnimation));
|
|
}
|
|
}
|
|
|
|
void
|
|
nsNodeUtils::AnimationChanged(Animation* aAnimation)
|
|
{
|
|
Element* target = GetTarget(aAnimation);
|
|
if (!target) {
|
|
return;
|
|
}
|
|
nsIDocument* doc = target->OwnerDoc();
|
|
|
|
if (doc->MayHaveAnimationObservers()) {
|
|
IMPL_ANIMATION_NOTIFICATION(AnimationChanged, target, (aAnimation));
|
|
}
|
|
}
|
|
|
|
void
|
|
nsNodeUtils::AnimationRemoved(Animation* aAnimation)
|
|
{
|
|
Element* target = GetTarget(aAnimation);
|
|
if (!target) {
|
|
return;
|
|
}
|
|
nsIDocument* doc = target->OwnerDoc();
|
|
|
|
if (doc->MayHaveAnimationObservers()) {
|
|
IMPL_ANIMATION_NOTIFICATION(AnimationRemoved, target, (aAnimation));
|
|
}
|
|
}
|
|
|
|
void
|
|
nsNodeUtils::LastRelease(nsINode* aNode)
|
|
{
|
|
nsINode::nsSlots* slots = aNode->GetExistingSlots();
|
|
if (slots) {
|
|
if (!slots->mMutationObservers.IsEmpty()) {
|
|
NS_OBSERVER_ARRAY_NOTIFY_OBSERVERS(slots->mMutationObservers,
|
|
nsIMutationObserver,
|
|
NodeWillBeDestroyed, (aNode));
|
|
}
|
|
|
|
delete slots;
|
|
aNode->mSlots = nullptr;
|
|
}
|
|
|
|
// Kill properties first since that may run external code, so we want to
|
|
// be in as complete state as possible at that time.
|
|
if (aNode->IsNodeOfType(nsINode::eDOCUMENT)) {
|
|
// Delete all properties before tearing down the document. Some of the
|
|
// properties are bound to nsINode objects and the destructor functions of
|
|
// the properties may want to use the owner document of the nsINode.
|
|
static_cast<nsIDocument*>(aNode)->DeleteAllProperties();
|
|
}
|
|
else {
|
|
if (aNode->HasProperties()) {
|
|
// Strong reference to the document so that deleting properties can't
|
|
// delete the document.
|
|
nsCOMPtr<nsIDocument> document = aNode->OwnerDoc();
|
|
document->DeleteAllPropertiesFor(aNode);
|
|
}
|
|
|
|
// I wonder whether it's faster to do the HasFlag check first....
|
|
if (aNode->IsNodeOfType(nsINode::eHTML_FORM_CONTROL) &&
|
|
aNode->HasFlag(ADDED_TO_FORM)) {
|
|
// Tell the form (if any) this node is going away. Don't
|
|
// notify, since we're being destroyed in any case.
|
|
static_cast<nsGenericHTMLFormElement*>(aNode)->ClearForm(true);
|
|
}
|
|
|
|
if (aNode->IsHTMLElement(nsGkAtoms::img) &&
|
|
aNode->HasFlag(ADDED_TO_FORM)) {
|
|
HTMLImageElement* imageElem = static_cast<HTMLImageElement*>(aNode);
|
|
imageElem->ClearForm(true);
|
|
}
|
|
}
|
|
aNode->UnsetFlags(NODE_HAS_PROPERTIES);
|
|
|
|
if (aNode->NodeType() != nsIDOMNode::DOCUMENT_NODE &&
|
|
aNode->HasFlag(NODE_HAS_LISTENERMANAGER)) {
|
|
#ifdef DEBUG
|
|
if (nsContentUtils::IsInitialized()) {
|
|
EventListenerManager* manager =
|
|
nsContentUtils::GetExistingListenerManagerForNode(aNode);
|
|
if (!manager) {
|
|
NS_ERROR("Huh, our bit says we have a listener manager list, "
|
|
"but there's nothing in the hash!?!!");
|
|
}
|
|
}
|
|
#endif
|
|
|
|
nsContentUtils::RemoveListenerManager(aNode);
|
|
aNode->UnsetFlags(NODE_HAS_LISTENERMANAGER);
|
|
}
|
|
|
|
if (aNode->IsElement()) {
|
|
nsIDocument* ownerDoc = aNode->OwnerDoc();
|
|
Element* elem = aNode->AsElement();
|
|
ownerDoc->ClearBoxObjectFor(elem);
|
|
|
|
NS_ASSERTION(aNode->HasFlag(NODE_FORCE_XBL_BINDINGS) ||
|
|
!elem->GetXBLBinding(),
|
|
"Non-forced node has binding on destruction");
|
|
|
|
// if NODE_FORCE_XBL_BINDINGS is set, the node might still have a binding
|
|
// attached
|
|
if (aNode->HasFlag(NODE_FORCE_XBL_BINDINGS) &&
|
|
ownerDoc->BindingManager()) {
|
|
ownerDoc->BindingManager()->RemovedFromDocument(elem, ownerDoc);
|
|
}
|
|
}
|
|
|
|
aNode->ReleaseWrapper(aNode);
|
|
|
|
FragmentOrElement::RemoveBlackMarkedNode(aNode);
|
|
}
|
|
|
|
static void
|
|
NoteUserData(void *aObject, nsIAtom *aKey, void *aXPCOMChild, void *aData)
|
|
{
|
|
nsCycleCollectionTraversalCallback* cb =
|
|
static_cast<nsCycleCollectionTraversalCallback*>(aData);
|
|
NS_CYCLE_COLLECTION_NOTE_EDGE_NAME(*cb, "[user data]");
|
|
cb->NoteXPCOMChild(static_cast<nsISupports*>(aXPCOMChild));
|
|
}
|
|
|
|
/* static */
|
|
void
|
|
nsNodeUtils::TraverseUserData(nsINode* aNode,
|
|
nsCycleCollectionTraversalCallback &aCb)
|
|
{
|
|
nsIDocument* ownerDoc = aNode->OwnerDoc();
|
|
ownerDoc->PropertyTable(DOM_USER_DATA)->Enumerate(aNode, NoteUserData, &aCb);
|
|
}
|
|
|
|
/* static */
|
|
nsresult
|
|
nsNodeUtils::CloneNodeImpl(nsINode *aNode, bool aDeep, nsINode **aResult)
|
|
{
|
|
*aResult = nullptr;
|
|
|
|
nsCOMPtr<nsINode> newNode;
|
|
nsCOMArray<nsINode> nodesWithProperties;
|
|
nsresult rv = Clone(aNode, aDeep, nullptr, nodesWithProperties,
|
|
getter_AddRefs(newNode));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
newNode.swap(*aResult);
|
|
return NS_OK;
|
|
}
|
|
|
|
/* static */
|
|
nsresult
|
|
nsNodeUtils::CloneAndAdopt(nsINode *aNode, bool aClone, bool aDeep,
|
|
nsNodeInfoManager *aNewNodeInfoManager,
|
|
JS::Handle<JSObject*> aReparentScope,
|
|
nsCOMArray<nsINode> &aNodesWithProperties,
|
|
nsINode *aParent, nsINode **aResult)
|
|
{
|
|
NS_PRECONDITION((!aClone && aNewNodeInfoManager) || !aReparentScope,
|
|
"If cloning or not getting a new nodeinfo we shouldn't "
|
|
"rewrap");
|
|
NS_PRECONDITION(!aParent || aNode->IsNodeOfType(nsINode::eCONTENT),
|
|
"Can't insert document or attribute nodes into a parent");
|
|
|
|
*aResult = nullptr;
|
|
|
|
// First deal with aNode and walk its attributes (and their children). Then,
|
|
// if aDeep is true, deal with aNode's children (and recurse into their
|
|
// attributes and children).
|
|
|
|
nsAutoScriptBlocker scriptBlocker;
|
|
AutoJSContext cx;
|
|
nsresult rv;
|
|
|
|
nsNodeInfoManager *nodeInfoManager = aNewNodeInfoManager;
|
|
|
|
// aNode.
|
|
NodeInfo *nodeInfo = aNode->mNodeInfo;
|
|
nsRefPtr<NodeInfo> newNodeInfo;
|
|
if (nodeInfoManager) {
|
|
|
|
// Don't allow importing/adopting nodes from non-privileged "scriptable"
|
|
// documents to "non-scriptable" documents.
|
|
nsIDocument* newDoc = nodeInfoManager->GetDocument();
|
|
NS_ENSURE_STATE(newDoc);
|
|
bool hasHadScriptHandlingObject = false;
|
|
if (!newDoc->GetScriptHandlingObject(hasHadScriptHandlingObject) &&
|
|
!hasHadScriptHandlingObject) {
|
|
nsIDocument* currentDoc = aNode->OwnerDoc();
|
|
NS_ENSURE_STATE((nsContentUtils::IsChromeDoc(currentDoc) ||
|
|
(!currentDoc->GetScriptHandlingObject(hasHadScriptHandlingObject) &&
|
|
!hasHadScriptHandlingObject)));
|
|
}
|
|
|
|
newNodeInfo = nodeInfoManager->GetNodeInfo(nodeInfo->NameAtom(),
|
|
nodeInfo->GetPrefixAtom(),
|
|
nodeInfo->NamespaceID(),
|
|
nodeInfo->NodeType(),
|
|
nodeInfo->GetExtraName());
|
|
|
|
nodeInfo = newNodeInfo;
|
|
}
|
|
|
|
Element *elem = aNode->IsElement() ? aNode->AsElement() : nullptr;
|
|
|
|
nsCOMPtr<nsINode> clone;
|
|
if (aClone) {
|
|
rv = aNode->Clone(nodeInfo, getter_AddRefs(clone));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
if (clone->IsElement()) {
|
|
// The cloned node may be a custom element that may require
|
|
// enqueing created callback and prototype swizzling.
|
|
Element* elem = clone->AsElement();
|
|
if (nsContentUtils::IsCustomElementName(nodeInfo->NameAtom())) {
|
|
elem->OwnerDoc()->SetupCustomElement(elem, nodeInfo->NamespaceID());
|
|
} else {
|
|
// Check if node may be custom element by type extension.
|
|
// ex. <button is="x-button">
|
|
nsAutoString extension;
|
|
if (elem->GetAttr(kNameSpaceID_None, nsGkAtoms::is, extension) &&
|
|
!extension.IsEmpty()) {
|
|
elem->OwnerDoc()->SetupCustomElement(elem, nodeInfo->NamespaceID(),
|
|
&extension);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (aParent) {
|
|
// If we're cloning we need to insert the cloned children into the cloned
|
|
// parent.
|
|
rv = aParent->AppendChildTo(static_cast<nsIContent*>(clone.get()),
|
|
false);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
else if (aDeep && clone->IsNodeOfType(nsINode::eDOCUMENT)) {
|
|
// After cloning the document itself, we want to clone the children into
|
|
// the cloned document (somewhat like cloning and importing them into the
|
|
// cloned document).
|
|
nodeInfoManager = clone->mNodeInfo->NodeInfoManager();
|
|
}
|
|
}
|
|
else if (nodeInfoManager) {
|
|
nsIDocument* oldDoc = aNode->OwnerDoc();
|
|
bool wasRegistered = false;
|
|
if (aNode->IsElement()) {
|
|
Element* element = aNode->AsElement();
|
|
oldDoc->ClearBoxObjectFor(element);
|
|
wasRegistered = oldDoc->UnregisterActivityObserver(element);
|
|
}
|
|
|
|
aNode->mNodeInfo.swap(newNodeInfo);
|
|
if (elem) {
|
|
elem->NodeInfoChanged(newNodeInfo);
|
|
}
|
|
|
|
nsIDocument* newDoc = aNode->OwnerDoc();
|
|
if (newDoc) {
|
|
// XXX what if oldDoc is null, we don't know if this should be
|
|
// registered or not! Can that really happen?
|
|
if (wasRegistered) {
|
|
newDoc->RegisterActivityObserver(aNode->AsElement());
|
|
}
|
|
|
|
nsPIDOMWindow* window = newDoc->GetInnerWindow();
|
|
if (window) {
|
|
EventListenerManager* elm = aNode->GetExistingListenerManager();
|
|
if (elm) {
|
|
window->SetMutationListeners(elm->MutationListenerBits());
|
|
if (elm->MayHavePaintEventListener()) {
|
|
window->SetHasPaintEventListeners();
|
|
}
|
|
if (elm->MayHaveTouchEventListener()) {
|
|
window->SetHasTouchEventListeners();
|
|
}
|
|
if (elm->MayHaveMouseEnterLeaveEventListener()) {
|
|
window->SetHasMouseEnterLeaveEventListeners();
|
|
}
|
|
if (elm->MayHavePointerEnterLeaveEventListener()) {
|
|
window->SetHasPointerEnterLeaveEventListeners();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (wasRegistered && oldDoc != newDoc) {
|
|
nsCOMPtr<nsIDOMHTMLMediaElement> domMediaElem(do_QueryInterface(aNode));
|
|
if (domMediaElem) {
|
|
HTMLMediaElement* mediaElem = static_cast<HTMLMediaElement*>(aNode);
|
|
mediaElem->NotifyOwnerDocumentActivityChanged();
|
|
}
|
|
nsCOMPtr<nsIObjectLoadingContent> objectLoadingContent(do_QueryInterface(aNode));
|
|
if (objectLoadingContent) {
|
|
nsObjectLoadingContent* olc = static_cast<nsObjectLoadingContent*>(objectLoadingContent.get());
|
|
olc->NotifyOwnerDocumentActivityChanged();
|
|
}
|
|
}
|
|
|
|
if (oldDoc != newDoc && oldDoc->MayHaveDOMMutationObservers()) {
|
|
newDoc->SetMayHaveDOMMutationObservers();
|
|
}
|
|
|
|
if (oldDoc != newDoc && oldDoc->MayHaveAnimationObservers()) {
|
|
newDoc->SetMayHaveAnimationObservers();
|
|
}
|
|
|
|
if (elem) {
|
|
elem->RecompileScriptEventListeners();
|
|
}
|
|
|
|
if (aReparentScope) {
|
|
JS::Rooted<JSObject*> wrapper(cx);
|
|
if ((wrapper = aNode->GetWrapper())) {
|
|
MOZ_ASSERT(IsDOMObject(wrapper));
|
|
JSAutoCompartment ac(cx, wrapper);
|
|
rv = ReparentWrapper(cx, wrapper);
|
|
if (NS_FAILED(rv)) {
|
|
if (wasRegistered) {
|
|
aNode->OwnerDoc()->UnregisterActivityObserver(aNode->AsElement());
|
|
}
|
|
aNode->mNodeInfo.swap(newNodeInfo);
|
|
|
|
if (wasRegistered) {
|
|
aNode->OwnerDoc()->RegisterActivityObserver(aNode->AsElement());
|
|
}
|
|
return rv;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (aNode->HasProperties()) {
|
|
bool ok = aNodesWithProperties.AppendObject(aNode);
|
|
MOZ_RELEASE_ASSERT(ok, "Out of memory");
|
|
if (aClone) {
|
|
ok = aNodesWithProperties.AppendObject(clone);
|
|
MOZ_RELEASE_ASSERT(ok, "Out of memory");
|
|
}
|
|
}
|
|
|
|
if (aDeep && (!aClone || !aNode->IsNodeOfType(nsINode::eATTRIBUTE))) {
|
|
// aNode's children.
|
|
for (nsIContent* cloneChild = aNode->GetFirstChild();
|
|
cloneChild;
|
|
cloneChild = cloneChild->GetNextSibling()) {
|
|
nsCOMPtr<nsINode> child;
|
|
rv = CloneAndAdopt(cloneChild, aClone, true, nodeInfoManager,
|
|
aReparentScope, aNodesWithProperties, clone,
|
|
getter_AddRefs(child));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
}
|
|
|
|
// Cloning template element.
|
|
if (aDeep && aClone && IsTemplateElement(aNode)) {
|
|
DocumentFragment* origContent =
|
|
static_cast<HTMLTemplateElement*>(aNode)->Content();
|
|
DocumentFragment* cloneContent =
|
|
static_cast<HTMLTemplateElement*>(clone.get())->Content();
|
|
|
|
// Clone the children into the clone's template content owner
|
|
// document's nodeinfo manager.
|
|
nsNodeInfoManager* ownerNodeInfoManager =
|
|
cloneContent->mNodeInfo->NodeInfoManager();
|
|
|
|
for (nsIContent* cloneChild = origContent->GetFirstChild();
|
|
cloneChild;
|
|
cloneChild = cloneChild->GetNextSibling()) {
|
|
nsCOMPtr<nsINode> child;
|
|
rv = CloneAndAdopt(cloneChild, aClone, aDeep, ownerNodeInfoManager,
|
|
aReparentScope, aNodesWithProperties, cloneContent,
|
|
getter_AddRefs(child));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
}
|
|
|
|
// XXX setting document on some nodes not in a document so XBL will bind
|
|
// and chrome won't break. Make XBL bind to document-less nodes!
|
|
// XXXbz Once this is fixed, fix up the asserts in all implementations of
|
|
// BindToTree to assert what they would like to assert, and fix the
|
|
// ChangeDocumentFor() call in nsXULElement::BindToTree as well. Also,
|
|
// remove the UnbindFromTree call in ~nsXULElement, and add back in the
|
|
// precondition in nsXULElement::UnbindFromTree and remove the line in
|
|
// nsXULElement.h that makes nsNodeUtils a friend of nsXULElement.
|
|
// Note: Make sure to do this witchery _after_ we've done any deep
|
|
// cloning, so kids of the new node aren't confused about whether they're
|
|
// in a document.
|
|
#ifdef MOZ_XUL
|
|
if (aClone && !aParent && aNode->IsXULElement()) {
|
|
if (!aNode->OwnerDoc()->IsLoadedAsInteractiveData()) {
|
|
clone->SetFlags(NODE_FORCE_XBL_BINDINGS);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
clone.forget(aResult);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
/* static */
|
|
void
|
|
nsNodeUtils::UnlinkUserData(nsINode *aNode)
|
|
{
|
|
NS_ASSERTION(aNode->HasProperties(), "Call to UnlinkUserData not needed.");
|
|
|
|
// Strong reference to the document so that deleting properties can't
|
|
// delete the document.
|
|
nsCOMPtr<nsIDocument> document = aNode->OwnerDoc();
|
|
document->PropertyTable(DOM_USER_DATA)->DeleteAllPropertiesFor(aNode);
|
|
}
|
|
|
|
bool
|
|
nsNodeUtils::IsTemplateElement(const nsINode *aNode)
|
|
{
|
|
return aNode->IsHTMLElement(nsGkAtoms::_template);
|
|
}
|
|
|
|
nsIContent*
|
|
nsNodeUtils::GetFirstChildOfTemplateOrNode(nsINode* aNode)
|
|
{
|
|
if (nsNodeUtils::IsTemplateElement(aNode)) {
|
|
DocumentFragment* frag =
|
|
static_cast<HTMLTemplateElement*>(aNode)->Content();
|
|
return frag->GetFirstChild();
|
|
}
|
|
|
|
return aNode->GetFirstChild();
|
|
}
|
|
|