/* * Copyright (C) 2011 Google Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following disclaimer * in the documentation and/or other materials provided with the * distribution. * * Neither the name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "sky/engine/config.h" #include "sky/engine/core/dom/MutationRecord.h" #include "sky/engine/core/dom/Node.h" #include "sky/engine/core/dom/NodeList.h" #include "sky/engine/core/dom/QualifiedName.h" #include "sky/engine/core/dom/StaticNodeList.h" #include "sky/engine/wtf/StdLibExtras.h" namespace blink { namespace { class ChildListRecord : public MutationRecord { public: ChildListRecord(PassRefPtr target, PassRefPtr added, PassRefPtr removed, PassRefPtr previousSibling, PassRefPtr nextSibling) : m_target(target) , m_addedNodes(added) , m_removedNodes(removed) , m_previousSibling(previousSibling) , m_nextSibling(nextSibling) { } private: virtual const AtomicString& type() override; virtual Node* target() override { return m_target.get(); } virtual StaticNodeList* addedNodes() override { return m_addedNodes.get(); } virtual StaticNodeList* removedNodes() override { return m_removedNodes.get(); } virtual Node* previousSibling() override { return m_previousSibling.get(); } virtual Node* nextSibling() override { return m_nextSibling.get(); } RefPtr m_target; RefPtr m_addedNodes; RefPtr m_removedNodes; RefPtr m_previousSibling; RefPtr m_nextSibling; }; class RecordWithEmptyNodeLists : public MutationRecord { public: RecordWithEmptyNodeLists(PassRefPtr target, const String& oldValue) : m_target(target) , m_oldValue(oldValue) { } private: virtual Node* target() override { return m_target.get(); } virtual String oldValue() override { return m_oldValue; } virtual StaticNodeList* addedNodes() override { return lazilyInitializeEmptyNodeList(m_addedNodes); } virtual StaticNodeList* removedNodes() override { return lazilyInitializeEmptyNodeList(m_removedNodes); } static StaticNodeList* lazilyInitializeEmptyNodeList(RefPtr& nodeList) { if (!nodeList) nodeList = StaticNodeList::createEmpty(); return nodeList.get(); } RefPtr m_target; String m_oldValue; RefPtr m_addedNodes; RefPtr m_removedNodes; }; class AttributesRecord : public RecordWithEmptyNodeLists { public: AttributesRecord(PassRefPtr target, const QualifiedName& name, const AtomicString& oldValue) : RecordWithEmptyNodeLists(target, oldValue) , m_attributeName(name.localName()) { } private: virtual const AtomicString& type() override; virtual const AtomicString& attributeName() override { return m_attributeName; } AtomicString m_attributeName; }; class CharacterDataRecord : public RecordWithEmptyNodeLists { public: CharacterDataRecord(PassRefPtr target, const String& oldValue) : RecordWithEmptyNodeLists(target, oldValue) { } private: virtual const AtomicString& type() override; }; class MutationRecordWithNullOldValue : public MutationRecord { public: MutationRecordWithNullOldValue(PassRefPtr record) : m_record(record) { } private: virtual const AtomicString& type() override { return m_record->type(); } virtual Node* target() override { return m_record->target(); } virtual StaticNodeList* addedNodes() override { return m_record->addedNodes(); } virtual StaticNodeList* removedNodes() override { return m_record->removedNodes(); } virtual Node* previousSibling() override { return m_record->previousSibling(); } virtual Node* nextSibling() override { return m_record->nextSibling(); } virtual const AtomicString& attributeName() override { return m_record->attributeName(); } virtual String oldValue() override { return String(); } RefPtr m_record; }; const AtomicString& ChildListRecord::type() { DEFINE_STATIC_LOCAL(AtomicString, childList, ("childList", AtomicString::ConstructFromLiteral)); return childList; } const AtomicString& AttributesRecord::type() { DEFINE_STATIC_LOCAL(AtomicString, attributes, ("attributes", AtomicString::ConstructFromLiteral)); return attributes; } const AtomicString& CharacterDataRecord::type() { DEFINE_STATIC_LOCAL(AtomicString, characterData, ("characterData", AtomicString::ConstructFromLiteral)); return characterData; } } // namespace PassRefPtr MutationRecord::createChildList(PassRefPtr target, PassRefPtr added, PassRefPtr removed, PassRefPtr previousSibling, PassRefPtr nextSibling) { return adoptRef(new ChildListRecord(target, added, removed, previousSibling, nextSibling)); } PassRefPtr MutationRecord::createAttributes(PassRefPtr target, const QualifiedName& name, const AtomicString& oldValue) { return adoptRef(new AttributesRecord(target, name, oldValue)); } PassRefPtr MutationRecord::createCharacterData(PassRefPtr target, const String& oldValue) { return adoptRef(new CharacterDataRecord(target, oldValue)); } PassRefPtr MutationRecord::createWithNullOldValue(PassRefPtr record) { return adoptRef(new MutationRecordWithNullOldValue(record)); } MutationRecord::~MutationRecord() { } } // namespace blink