2007-03-22 10:30:00 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is Mozilla.org code.
|
|
|
|
*
|
2009-12-21 13:50:30 -08:00
|
|
|
* The Initial Developer of the Original Code is Mozilla Foundation.
|
2007-03-22 10:30:00 -07:00
|
|
|
* Portions created by the Initial Developer are Copyright (C) 2006
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
* Jonas Sicking <jonas@sicking.cc> (Original Author)
|
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
|
|
* the provisions above, a recipient may use your version of this file under
|
|
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
|
|
*
|
|
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
|
|
|
|
#include "nsNodeUtils.h"
|
|
|
|
#include "nsContentUtils.h"
|
|
|
|
#include "nsINode.h"
|
|
|
|
#include "nsIContent.h"
|
2010-05-05 11:18:05 -07:00
|
|
|
#include "mozilla/dom/Element.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
#include "nsIMutationObserver.h"
|
|
|
|
#include "nsIDocument.h"
|
|
|
|
#include "nsIDOMUserDataHandler.h"
|
|
|
|
#include "nsIEventListenerManager.h"
|
|
|
|
#include "nsIAttribute.h"
|
|
|
|
#include "nsIXPConnect.h"
|
|
|
|
#include "nsGenericElement.h"
|
|
|
|
#include "pldhash.h"
|
|
|
|
#include "nsIDOMAttr.h"
|
|
|
|
#include "nsCOMArray.h"
|
|
|
|
#include "nsPIDOMWindow.h"
|
2009-09-10 06:23:40 -07:00
|
|
|
#include "nsDocument.h"
|
2007-03-22 10:30:00 -07:00
|
|
|
#ifdef MOZ_XUL
|
|
|
|
#include "nsXULElement.h"
|
|
|
|
#endif
|
2008-09-05 08:45:03 -07:00
|
|
|
#include "nsBindingManager.h"
|
2009-09-30 15:56:50 -07:00
|
|
|
#include "nsGenericHTMLElement.h"
|
2009-10-29 16:55:05 -07:00
|
|
|
#ifdef MOZ_MEDIA
|
2009-10-01 07:10:13 -07:00
|
|
|
#include "nsHTMLMediaElement.h"
|
2009-10-29 16:55:05 -07:00
|
|
|
#endif // MOZ_MEDIA
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-04-30 06:12:05 -07:00
|
|
|
using namespace mozilla::dom;
|
|
|
|
|
2007-12-04 10:37:54 -08:00
|
|
|
// This macro expects the ownerDocument of content_ to be in scope as
|
|
|
|
// |nsIDocument* doc|
|
2007-03-22 10:30:00 -07:00
|
|
|
#define IMPL_MUTATION_NOTIFICATION(func_, content_, params_) \
|
|
|
|
PR_BEGIN_MACRO \
|
|
|
|
nsINode* node = content_; \
|
2007-12-04 10:37:54 -08:00
|
|
|
NS_ASSERTION(node->GetOwnerDoc() == doc, "Bogus document"); \
|
|
|
|
if (doc) { \
|
|
|
|
static_cast<nsIMutationObserver*>(doc->BindingManager())-> \
|
|
|
|
func_ params_; \
|
|
|
|
} \
|
2007-03-22 10:30:00 -07:00
|
|
|
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_); \
|
|
|
|
} \
|
|
|
|
node = node->GetNodeParent(); \
|
|
|
|
} while (node); \
|
|
|
|
PR_END_MACRO
|
|
|
|
|
2007-09-05 01:22:17 -07:00
|
|
|
void
|
|
|
|
nsNodeUtils::CharacterDataWillChange(nsIContent* aContent,
|
|
|
|
CharacterDataChangeInfo* aInfo)
|
|
|
|
{
|
|
|
|
nsIDocument* doc = aContent->GetOwnerDoc();
|
2010-07-21 08:33:32 -07:00
|
|
|
IMPL_MUTATION_NOTIFICATION(CharacterDataWillChange, aContent,
|
2007-09-05 01:22:17 -07:00
|
|
|
(doc, aContent, aInfo));
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
void
|
|
|
|
nsNodeUtils::CharacterDataChanged(nsIContent* aContent,
|
|
|
|
CharacterDataChangeInfo* aInfo)
|
|
|
|
{
|
|
|
|
nsIDocument* doc = aContent->GetOwnerDoc();
|
2010-07-21 08:33:32 -07:00
|
|
|
IMPL_MUTATION_NOTIFICATION(CharacterDataChanged, aContent,
|
2007-03-22 10:30:00 -07:00
|
|
|
(doc, aContent, aInfo));
|
|
|
|
}
|
|
|
|
|
2009-06-29 11:36:25 -07:00
|
|
|
void
|
2010-08-24 00:06:20 -07:00
|
|
|
nsNodeUtils::AttributeWillChange(Element* aElement,
|
2009-06-29 11:36:25 -07:00
|
|
|
PRInt32 aNameSpaceID,
|
|
|
|
nsIAtom* aAttribute,
|
|
|
|
PRInt32 aModType)
|
|
|
|
{
|
2010-08-24 00:06:20 -07:00
|
|
|
nsIDocument* doc = aElement->GetOwnerDoc();
|
|
|
|
IMPL_MUTATION_NOTIFICATION(AttributeWillChange, aElement,
|
|
|
|
(doc, aElement, aNameSpaceID, aAttribute,
|
2009-06-29 11:36:25 -07:00
|
|
|
aModType));
|
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
void
|
2010-08-24 00:06:07 -07:00
|
|
|
nsNodeUtils::AttributeChanged(Element* aElement,
|
2007-03-22 10:30:00 -07:00
|
|
|
PRInt32 aNameSpaceID,
|
|
|
|
nsIAtom* aAttribute,
|
2009-12-10 14:36:04 -08:00
|
|
|
PRInt32 aModType)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
2010-08-24 00:05:56 -07:00
|
|
|
nsIDocument* doc = aElement->GetOwnerDoc();
|
|
|
|
IMPL_MUTATION_NOTIFICATION(AttributeChanged, aElement,
|
|
|
|
(doc, aElement, aNameSpaceID, aAttribute,
|
2009-12-10 14:36:04 -08:00
|
|
|
aModType));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsNodeUtils::ContentAppended(nsIContent* aContainer,
|
2010-05-10 18:12:34 -07:00
|
|
|
nsIContent* aFirstNewContent,
|
2007-03-22 10:30:00 -07:00
|
|
|
PRInt32 aNewIndexInContainer)
|
|
|
|
{
|
2007-12-04 10:37:54 -08:00
|
|
|
nsIDocument* doc = aContainer->GetOwnerDoc();
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2010-07-21 08:33:32 -07:00
|
|
|
IMPL_MUTATION_NOTIFICATION(ContentAppended, aContainer,
|
2010-05-10 18:12:34 -07:00
|
|
|
(doc, aContainer, aFirstNewContent,
|
|
|
|
aNewIndexInContainer));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsNodeUtils::ContentInserted(nsINode* aContainer,
|
|
|
|
nsIContent* aChild,
|
|
|
|
PRInt32 aIndexInContainer)
|
|
|
|
{
|
|
|
|
NS_PRECONDITION(aContainer->IsNodeOfType(nsINode::eCONTENT) ||
|
|
|
|
aContainer->IsNodeOfType(nsINode::eDOCUMENT),
|
|
|
|
"container must be an nsIContent or an nsIDocument");
|
|
|
|
nsIContent* container;
|
2007-12-04 10:37:54 -08:00
|
|
|
nsIDocument* doc = aContainer->GetOwnerDoc();
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIDocument* document;
|
|
|
|
if (aContainer->IsNodeOfType(nsINode::eCONTENT)) {
|
2007-07-08 00:08:04 -07:00
|
|
|
container = static_cast<nsIContent*>(aContainer);
|
2007-12-04 10:37:54 -08:00
|
|
|
document = doc;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
container = nsnull;
|
2007-07-08 00:08:04 -07:00
|
|
|
document = static_cast<nsIDocument*>(aContainer);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-07-21 08:33:32 -07:00
|
|
|
IMPL_MUTATION_NOTIFICATION(ContentInserted, aContainer,
|
2007-03-22 10:30:00 -07:00
|
|
|
(document, container, aChild, aIndexInContainer));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsNodeUtils::ContentRemoved(nsINode* aContainer,
|
|
|
|
nsIContent* aChild,
|
2010-07-21 15:05:17 -07:00
|
|
|
PRInt32 aIndexInContainer,
|
|
|
|
nsIContent* aPreviousSibling)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
NS_PRECONDITION(aContainer->IsNodeOfType(nsINode::eCONTENT) ||
|
|
|
|
aContainer->IsNodeOfType(nsINode::eDOCUMENT),
|
|
|
|
"container must be an nsIContent or an nsIDocument");
|
|
|
|
nsIContent* container;
|
2007-12-04 10:37:54 -08:00
|
|
|
nsIDocument* doc = aContainer->GetOwnerDoc();
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIDocument* document;
|
|
|
|
if (aContainer->IsNodeOfType(nsINode::eCONTENT)) {
|
2007-07-08 00:08:04 -07:00
|
|
|
container = static_cast<nsIContent*>(aContainer);
|
2007-12-04 10:37:54 -08:00
|
|
|
document = doc;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
container = nsnull;
|
2007-07-08 00:08:04 -07:00
|
|
|
document = static_cast<nsIDocument*>(aContainer);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2010-07-21 08:37:41 -07:00
|
|
|
IMPL_MUTATION_NOTIFICATION(ContentRemoved, aContainer,
|
2010-07-21 15:05:17 -07:00
|
|
|
(document, container, aChild, aIndexInContainer,
|
|
|
|
aPreviousSibling));
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nsNodeUtils::ParentChainChanged(nsIContent *aContent)
|
|
|
|
{
|
|
|
|
// No need to notify observers on the parents since their parent
|
|
|
|
// chain must have been changed too and so their observers were
|
|
|
|
// notified at that time.
|
|
|
|
|
|
|
|
nsINode::nsSlots* slots = aContent->GetExistingSlots();
|
|
|
|
if (slots && !slots->mMutationObservers.IsEmpty()) {
|
2010-07-21 08:33:31 -07:00
|
|
|
NS_OBSERVER_ARRAY_NOTIFY_OBSERVERS(
|
2007-03-22 10:30:00 -07:00
|
|
|
slots->mMutationObservers,
|
|
|
|
nsIMutationObserver,
|
|
|
|
ParentChainChanged,
|
|
|
|
(aContent));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2007-05-12 08:36:28 -07:00
|
|
|
nsNodeUtils::LastRelease(nsINode* aNode)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
nsINode::nsSlots* slots = aNode->GetExistingSlots();
|
|
|
|
if (slots) {
|
|
|
|
if (!slots->mMutationObservers.IsEmpty()) {
|
2010-07-21 08:33:32 -07:00
|
|
|
NS_OBSERVER_ARRAY_NOTIFY_OBSERVERS(slots->mMutationObservers,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsIMutationObserver,
|
|
|
|
NodeWillBeDestroyed, (aNode));
|
|
|
|
}
|
|
|
|
|
|
|
|
PtrBits flags = slots->mFlags | NODE_DOESNT_HAVE_SLOTS;
|
2008-02-02 15:41:24 -08:00
|
|
|
delete slots;
|
2007-03-22 10:30:00 -07:00
|
|
|
aNode->mFlagsOrSlots = flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Kill properties first since that may run external code, so we want to
|
|
|
|
// be in as complete state as possible at that time.
|
2007-05-12 08:36:28 -07:00
|
|
|
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.
|
2010-04-22 19:41:38 -07:00
|
|
|
static_cast<nsIDocument*>(aNode)->DeleteAllProperties();
|
2007-05-12 08:36:28 -07:00
|
|
|
}
|
2009-09-30 15:56:50 -07:00
|
|
|
else {
|
|
|
|
if (aNode->HasProperties()) {
|
|
|
|
// Strong reference to the document so that deleting properties can't
|
|
|
|
// delete the document.
|
|
|
|
nsCOMPtr<nsIDocument> document = aNode->GetOwnerDoc();
|
|
|
|
if (document) {
|
2010-04-22 19:41:38 -07:00
|
|
|
document->DeleteAllPropertiesFor(aNode);
|
2009-09-30 15:56:50 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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(PR_TRUE,
|
|
|
|
PR_FALSE);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
2007-05-12 08:36:28 -07:00
|
|
|
aNode->UnsetFlags(NODE_HAS_PROPERTIES);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (aNode->HasFlag(NODE_HAS_LISTENERMANAGER)) {
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (nsContentUtils::IsInitialized()) {
|
2009-06-23 04:23:52 -07:00
|
|
|
nsIEventListenerManager* manager =
|
|
|
|
nsContentUtils::GetListenerManager(aNode, PR_FALSE);
|
2007-03-22 10:30:00 -07:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2010-04-30 06:12:05 -07:00
|
|
|
if (aNode->IsElement()) {
|
2008-02-14 12:45:07 -08:00
|
|
|
nsIDocument* ownerDoc = aNode->GetOwnerDoc();
|
2010-05-26 15:39:13 -07:00
|
|
|
Element* elem = aNode->AsElement();
|
2008-02-14 12:45:07 -08:00
|
|
|
if (ownerDoc) {
|
2010-05-26 15:39:13 -07:00
|
|
|
ownerDoc->ClearBoxObjectFor(elem);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_ASSERTION(aNode->HasFlag(NODE_FORCE_XBL_BINDINGS) ||
|
|
|
|
!ownerDoc ||
|
|
|
|
!ownerDoc->BindingManager() ||
|
2010-05-26 15:59:23 -07:00
|
|
|
!ownerDoc->BindingManager()->GetBinding(elem),
|
2010-05-26 15:39:13 -07:00
|
|
|
"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 && ownerDoc->BindingManager()) {
|
2010-06-03 18:09:08 -07:00
|
|
|
ownerDoc->BindingManager()->RemovedFromDocument(elem, ownerDoc);
|
2008-02-14 12:45:07 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-12 13:20:42 -07:00
|
|
|
nsContentUtils::ReleaseWrapper(aNode, aNode);
|
|
|
|
|
2008-02-02 15:41:24 -08:00
|
|
|
delete aNode;
|
2007-05-12 08:36:28 -07:00
|
|
|
}
|
|
|
|
|
2008-06-30 18:03:50 -07:00
|
|
|
struct NS_STACK_CLASS nsHandlerData
|
2007-05-12 08:36:28 -07:00
|
|
|
{
|
|
|
|
PRUint16 mOperation;
|
2008-06-30 18:03:50 -07:00
|
|
|
nsCOMPtr<nsIDOMNode> mSource;
|
|
|
|
nsCOMPtr<nsIDOMNode> mDest;
|
2010-04-27 02:51:28 -07:00
|
|
|
nsCxPusher mPusher;
|
2007-05-12 08:36:28 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
CallHandler(void *aObject, nsIAtom *aKey, void *aHandler, void *aData)
|
|
|
|
{
|
2007-07-08 00:08:04 -07:00
|
|
|
nsHandlerData *handlerData = static_cast<nsHandlerData*>(aData);
|
2007-05-12 08:36:28 -07:00
|
|
|
nsCOMPtr<nsIDOMUserDataHandler> handler =
|
2007-07-08 00:08:04 -07:00
|
|
|
static_cast<nsIDOMUserDataHandler*>(aHandler);
|
|
|
|
nsINode *node = static_cast<nsINode*>(aObject);
|
2007-05-12 08:36:28 -07:00
|
|
|
nsCOMPtr<nsIVariant> data =
|
2007-07-08 00:08:04 -07:00
|
|
|
static_cast<nsIVariant*>(node->GetProperty(DOM_USER_DATA, aKey));
|
2007-05-12 08:36:28 -07:00
|
|
|
NS_ASSERTION(data, "Handler without data?");
|
|
|
|
|
2010-04-27 02:51:28 -07:00
|
|
|
if (!handlerData->mPusher.RePush(node)) {
|
|
|
|
return;
|
|
|
|
}
|
2007-05-12 08:36:28 -07:00
|
|
|
nsAutoString key;
|
|
|
|
aKey->ToString(key);
|
|
|
|
handler->Handle(handlerData->mOperation, key, data, handlerData->mSource,
|
|
|
|
handlerData->mDest);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* static */
|
|
|
|
nsresult
|
|
|
|
nsNodeUtils::CallUserDataHandlers(nsCOMArray<nsINode> &aNodesWithProperties,
|
|
|
|
nsIDocument *aOwnerDocument,
|
|
|
|
PRUint16 aOperation, PRBool aCloned)
|
|
|
|
{
|
|
|
|
NS_PRECONDITION(!aCloned || (aNodesWithProperties.Count() % 2 == 0),
|
|
|
|
"Expected aNodesWithProperties to contain original and "
|
|
|
|
"cloned nodes.");
|
|
|
|
|
2010-04-22 19:41:38 -07:00
|
|
|
nsPropertyTable *table = aOwnerDocument->PropertyTable(DOM_USER_DATA_HANDLER);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// Keep the document alive, just in case one of the handlers causes it to go
|
|
|
|
// away.
|
|
|
|
nsCOMPtr<nsIDocument> ownerDoc = aOwnerDocument;
|
|
|
|
|
|
|
|
nsHandlerData handlerData;
|
|
|
|
handlerData.mOperation = aOperation;
|
|
|
|
|
|
|
|
PRUint32 i, count = aNodesWithProperties.Count();
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
nsINode *nodeWithProperties = aNodesWithProperties[i];
|
|
|
|
|
|
|
|
nsresult rv;
|
|
|
|
handlerData.mSource = do_QueryInterface(nodeWithProperties, &rv);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
if (aCloned) {
|
|
|
|
handlerData.mDest = do_QueryInterface(aNodesWithProperties[++i], &rv);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
}
|
|
|
|
|
2010-04-22 19:41:38 -07:00
|
|
|
table->Enumerate(nodeWithProperties, CallHandler, &handlerData);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2007-05-12 08:36:28 -07:00
|
|
|
static void
|
|
|
|
NoteUserData(void *aObject, nsIAtom *aKey, void *aXPCOMChild, void *aData)
|
|
|
|
{
|
|
|
|
nsCycleCollectionTraversalCallback* cb =
|
2007-07-08 00:08:04 -07:00
|
|
|
static_cast<nsCycleCollectionTraversalCallback*>(aData);
|
2008-03-17 16:11:08 -07:00
|
|
|
NS_CYCLE_COLLECTION_NOTE_EDGE_NAME(*cb, "[user data (or handler)]");
|
2007-07-08 00:08:04 -07:00
|
|
|
cb->NoteXPCOMChild(static_cast<nsISupports*>(aXPCOMChild));
|
2007-05-12 08:36:28 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/* static */
|
|
|
|
void
|
|
|
|
nsNodeUtils::TraverseUserData(nsINode* aNode,
|
|
|
|
nsCycleCollectionTraversalCallback &aCb)
|
|
|
|
{
|
|
|
|
nsIDocument* ownerDoc = aNode->GetOwnerDoc();
|
|
|
|
if (!ownerDoc) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-04-22 19:41:38 -07:00
|
|
|
ownerDoc->PropertyTable(DOM_USER_DATA)->Enumerate(aNode, NoteUserData, &aCb);
|
|
|
|
ownerDoc->PropertyTable(DOM_USER_DATA_HANDLER)->Enumerate(aNode, NoteUserData, &aCb);
|
2007-05-12 08:36:28 -07:00
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
/* static */
|
|
|
|
nsresult
|
|
|
|
nsNodeUtils::CloneNodeImpl(nsINode *aNode, PRBool aDeep, nsIDOMNode **aResult)
|
|
|
|
{
|
|
|
|
*aResult = nsnull;
|
|
|
|
|
|
|
|
nsCOMPtr<nsIDOMNode> newNode;
|
|
|
|
nsCOMArray<nsINode> nodesWithProperties;
|
|
|
|
nsresult rv = Clone(aNode, aDeep, nsnull, nodesWithProperties,
|
|
|
|
getter_AddRefs(newNode));
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
nsIDocument *ownerDoc = aNode->GetOwnerDoc();
|
|
|
|
if (ownerDoc) {
|
|
|
|
rv = CallUserDataHandlers(nodesWithProperties, ownerDoc,
|
|
|
|
nsIDOMUserDataHandler::NODE_CLONED, PR_TRUE);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
}
|
|
|
|
|
|
|
|
newNode.swap(*aResult);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* static */
|
|
|
|
nsresult
|
|
|
|
nsNodeUtils::CloneAndAdopt(nsINode *aNode, PRBool aClone, PRBool aDeep,
|
|
|
|
nsNodeInfoManager *aNewNodeInfoManager,
|
|
|
|
JSContext *aCx, JSObject *aOldScope,
|
|
|
|
JSObject *aNewScope,
|
|
|
|
nsCOMArray<nsINode> &aNodesWithProperties,
|
2009-09-28 13:33:29 -07:00
|
|
|
nsINode *aParent, nsINode **aResult)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
NS_PRECONDITION((!aClone && aNewNodeInfoManager) || !aCx,
|
|
|
|
"If cloning or not getting a new nodeinfo we shouldn't "
|
|
|
|
"rewrap");
|
|
|
|
NS_PRECONDITION(!aCx || (aOldScope && aNewScope), "Must have scopes");
|
2009-09-28 13:33:29 -07:00
|
|
|
NS_PRECONDITION(!aParent || aNode->IsNodeOfType(nsINode::eCONTENT),
|
|
|
|
"Can't insert document or attribute nodes into a parent");
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
*aResult = nsnull;
|
|
|
|
|
|
|
|
// First deal with aNode and walk its attributes (and their children). Then,
|
|
|
|
// if aDeep is PR_TRUE, deal with aNode's children (and recurse into their
|
|
|
|
// attributes and children).
|
|
|
|
|
|
|
|
nsresult rv;
|
|
|
|
nsNodeInfoManager *nodeInfoManager = aNewNodeInfoManager;
|
|
|
|
|
|
|
|
// aNode.
|
|
|
|
nsINodeInfo *nodeInfo = aNode->mNodeInfo;
|
|
|
|
nsCOMPtr<nsINodeInfo> newNodeInfo;
|
|
|
|
if (nodeInfoManager) {
|
2008-10-13 09:12:26 -07:00
|
|
|
|
|
|
|
// Don't allow importing/adopting nodes from non-privileged "scriptable"
|
|
|
|
// documents to "non-scriptable" documents.
|
|
|
|
nsIDocument* newDoc = nodeInfoManager->GetDocument();
|
2008-12-03 02:25:21 -08:00
|
|
|
NS_ENSURE_STATE(newDoc);
|
2008-10-13 09:12:26 -07:00
|
|
|
PRBool hasHadScriptHandlingObject = PR_FALSE;
|
|
|
|
if (!newDoc->GetScriptHandlingObject(hasHadScriptHandlingObject) &&
|
|
|
|
!hasHadScriptHandlingObject) {
|
2008-12-03 02:25:21 -08:00
|
|
|
nsIDocument* currentDoc = aNode->GetOwnerDoc();
|
|
|
|
NS_ENSURE_STATE(currentDoc &&
|
|
|
|
(nsContentUtils::IsChromeDoc(currentDoc) ||
|
|
|
|
(!currentDoc->GetScriptHandlingObject(hasHadScriptHandlingObject) &&
|
|
|
|
!hasHadScriptHandlingObject)));
|
2008-10-13 09:12:26 -07:00
|
|
|
}
|
|
|
|
|
2008-09-12 15:32:18 -07:00
|
|
|
newNodeInfo = nodeInfoManager->GetNodeInfo(nodeInfo->NameAtom(),
|
|
|
|
nodeInfo->GetPrefixAtom(),
|
|
|
|
nodeInfo->NamespaceID());
|
2008-09-25 15:46:52 -07:00
|
|
|
NS_ENSURE_TRUE(newNodeInfo, NS_ERROR_OUT_OF_MEMORY);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
nodeInfo = newNodeInfo;
|
|
|
|
}
|
|
|
|
|
2010-04-30 06:12:05 -07:00
|
|
|
nsGenericElement *elem = aNode->IsElement() ?
|
2007-07-08 00:08:04 -07:00
|
|
|
static_cast<nsGenericElement*>(aNode) :
|
2007-03-22 10:30:00 -07:00
|
|
|
nsnull;
|
|
|
|
|
|
|
|
nsCOMPtr<nsINode> clone;
|
2009-09-10 06:23:40 -07:00
|
|
|
PRBool isDeepDocumentClone = PR_FALSE;
|
2007-03-22 10:30:00 -07:00
|
|
|
if (aClone) {
|
|
|
|
rv = aNode->Clone(nodeInfo, getter_AddRefs(clone));
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
if (aParent) {
|
|
|
|
// If we're cloning we need to insert the cloned children into the cloned
|
|
|
|
// parent.
|
2009-09-28 13:33:29 -07:00
|
|
|
rv = aParent->AppendChildTo(static_cast<nsIContent*>(clone.get()),
|
|
|
|
PR_FALSE);
|
2007-03-22 10:30:00 -07:00
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
}
|
|
|
|
else if (aDeep && clone->IsNodeOfType(nsINode::eDOCUMENT)) {
|
2009-09-10 06:23:40 -07:00
|
|
|
isDeepDocumentClone = PR_TRUE;
|
2007-03-22 10:30:00 -07:00
|
|
|
// 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->GetOwnerDoc();
|
2009-05-07 18:32:32 -07:00
|
|
|
PRBool wasRegistered = PR_FALSE;
|
2010-04-30 06:12:05 -07:00
|
|
|
if (oldDoc && aNode->IsElement()) {
|
|
|
|
Element* element = aNode->AsElement();
|
|
|
|
oldDoc->ClearBoxObjectFor(element);
|
|
|
|
wasRegistered = oldDoc->UnregisterFreezableElement(element);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
aNode->mNodeInfo.swap(newNodeInfo);
|
|
|
|
|
|
|
|
nsIDocument* newDoc = aNode->GetOwnerDoc();
|
|
|
|
if (newDoc) {
|
2009-10-01 07:10:13 -07:00
|
|
|
// XXX what if oldDoc is null, we don't know if this should be
|
|
|
|
// registered or not! Can that really happen?
|
2009-05-07 18:32:32 -07:00
|
|
|
if (wasRegistered) {
|
2010-04-30 06:12:05 -07:00
|
|
|
newDoc->RegisterFreezableElement(aNode->AsElement());
|
2009-05-07 18:32:32 -07:00
|
|
|
}
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
nsPIDOMWindow* window = newDoc->GetInnerWindow();
|
|
|
|
if (window) {
|
2009-06-23 04:23:52 -07:00
|
|
|
nsIEventListenerManager* elm = aNode->GetListenerManager(PR_FALSE);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (elm) {
|
|
|
|
window->SetMutationListeners(elm->MutationListenerBits());
|
2008-10-15 14:06:32 -07:00
|
|
|
if (elm->MayHavePaintEventListener()) {
|
|
|
|
window->SetHasPaintEventListeners();
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-01 07:10:13 -07:00
|
|
|
#ifdef MOZ_MEDIA
|
|
|
|
if (wasRegistered && oldDoc != newDoc) {
|
|
|
|
nsCOMPtr<nsIDOMHTMLMediaElement> domMediaElem(do_QueryInterface(aNode));
|
|
|
|
if (domMediaElem) {
|
|
|
|
nsHTMLMediaElement* mediaElem = static_cast<nsHTMLMediaElement*>(aNode);
|
|
|
|
mediaElem->NotifyOwnerDocumentActivityChanged();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2007-03-22 10:30:00 -07:00
|
|
|
if (elem) {
|
|
|
|
elem->RecompileScriptEventListeners();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aCx) {
|
|
|
|
nsIXPConnect *xpc = nsContentUtils::XPConnect();
|
|
|
|
if (xpc) {
|
|
|
|
nsCOMPtr<nsIXPConnectJSObjectHolder> oldWrapper;
|
|
|
|
rv = xpc->ReparentWrappedNativeIfFound(aCx, aOldScope, aNewScope, aNode,
|
|
|
|
getter_AddRefs(oldWrapper));
|
|
|
|
if (NS_FAILED(rv)) {
|
|
|
|
aNode->mNodeInfo.swap(nodeInfo);
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-05-03 06:23:36 -07:00
|
|
|
// XXX If there are any attribute nodes on this element with UserDataHandlers
|
|
|
|
// we should technically adopt/clone/import such attribute nodes and notify
|
|
|
|
// those handlers. However we currently don't have code to do so without
|
|
|
|
// also notifying when it's not safe so we're not doing that at this time.
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
// The DOM spec says to always adopt/clone/import the children of attribute
|
|
|
|
// nodes.
|
|
|
|
// XXX The following block is here because our implementation of attribute
|
|
|
|
// nodes is broken when it comes to inserting children. Instead of cloning
|
|
|
|
// their children we force creation of the only child by calling
|
|
|
|
// GetChildAt(0). We can remove this when
|
|
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=56758 is fixed.
|
|
|
|
if (aClone && aNode->IsNodeOfType(nsINode::eATTRIBUTE)) {
|
|
|
|
nsCOMPtr<nsINode> attrChildNode = aNode->GetChildAt(0);
|
|
|
|
// We only need to do this if the child node has properties (because we
|
|
|
|
// might need to call a userdata handler).
|
|
|
|
if (attrChildNode && attrChildNode->HasProperties()) {
|
|
|
|
nsCOMPtr<nsINode> clonedAttrChildNode = clone->GetChildAt(0);
|
|
|
|
if (clonedAttrChildNode) {
|
|
|
|
PRBool ok = aNodesWithProperties.AppendObject(attrChildNode) &&
|
|
|
|
aNodesWithProperties.AppendObject(clonedAttrChildNode);
|
|
|
|
NS_ENSURE_TRUE(ok, NS_ERROR_OUT_OF_MEMORY);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// XXX End of workaround for broken attribute nodes.
|
|
|
|
else if (aDeep || aNode->IsNodeOfType(nsINode::eATTRIBUTE)) {
|
|
|
|
// aNode's children.
|
|
|
|
PRUint32 i, length = aNode->GetChildCount();
|
|
|
|
for (i = 0; i < length; ++i) {
|
2009-09-28 13:33:29 -07:00
|
|
|
nsCOMPtr<nsINode> child;
|
2007-03-22 10:30:00 -07:00
|
|
|
rv = CloneAndAdopt(aNode->GetChildAt(i), aClone, PR_TRUE, nodeInfoManager,
|
|
|
|
aCx, aOldScope, aNewScope, aNodesWithProperties,
|
|
|
|
clone, 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
|
2010-04-30 06:12:05 -07:00
|
|
|
if (aClone && !aParent && aNode->IsElement() &&
|
|
|
|
aNode->AsElement()->IsXUL()) {
|
2007-07-08 00:08:04 -07:00
|
|
|
nsXULElement *xulElem = static_cast<nsXULElement*>(elem);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (!xulElem->mPrototype || xulElem->IsInDoc()) {
|
2007-05-15 18:13:47 -07:00
|
|
|
clone->SetFlags(NODE_FORCE_XBL_BINDINGS);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (aNode->HasProperties()) {
|
|
|
|
PRBool ok = aNodesWithProperties.AppendObject(aNode);
|
|
|
|
if (aClone) {
|
|
|
|
ok = ok && aNodesWithProperties.AppendObject(clone);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_ENSURE_TRUE(ok, NS_ERROR_OUT_OF_MEMORY);
|
|
|
|
}
|
|
|
|
|
2009-09-28 13:33:29 -07:00
|
|
|
clone.forget(aResult);
|
|
|
|
|
|
|
|
return NS_OK;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
2007-05-12 08:36:28 -07:00
|
|
|
|
|
|
|
|
|
|
|
/* 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->GetOwnerDoc();
|
|
|
|
if (document) {
|
2010-04-22 19:41:38 -07:00
|
|
|
document->PropertyTable(DOM_USER_DATA)->DeleteAllPropertiesFor(aNode);
|
|
|
|
document->PropertyTable(DOM_USER_DATA_HANDLER)->DeleteAllPropertiesFor(aNode);
|
2007-05-12 08:36:28 -07:00
|
|
|
}
|
|
|
|
}
|