/* Copyright (C) 2007, 2008 Nikolas Zimmermann This library is free software; you can redistribute it and/or modify it under the terms of the GNU Library General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License for more details. You should have received a copy of the GNU Library General Public License along with this library; see the file COPYING.LIB. If not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "config.h" #if ENABLE(SVG) #include "SVGElementInstance.h" #include "ContainerNodeAlgorithms.h" #include "Event.h" #include "EventException.h" #include "EventListener.h" #include "EventNames.h" #include "FrameView.h" #include "SVGElementInstanceList.h" #include "SVGUseElement.h" #include #if USE(JSC) #include "GCController.h" #endif namespace WebCore { #ifndef NDEBUG static WTF::RefCountedLeakCounter instanceCounter("WebCoreSVGElementInstance"); #endif static EventTargetData& dummyEventTargetData() { DEFINE_STATIC_LOCAL(EventTargetData, dummyEventTargetData, ()); dummyEventTargetData.eventListenerMap.clear(); return dummyEventTargetData; } SVGElementInstance::SVGElementInstance(SVGUseElement* useElement, PassRefPtr originalElement) : m_needsUpdate(false) , m_useElement(useElement) , m_element(originalElement) , m_previousSibling(0) , m_nextSibling(0) , m_firstChild(0) , m_lastChild(0) { ASSERT(m_useElement); ASSERT(m_element); // Register as instance for passed element. m_element->mapInstanceToElement(this); #ifndef NDEBUG instanceCounter.increment(); #endif } SVGElementInstance::~SVGElementInstance() { #ifndef NDEBUG instanceCounter.decrement(); #endif // Deregister as instance for passed element. m_element->removeInstanceMapping(this); removeAllChildrenInContainer(this); } PassRefPtr SVGElementInstance::childNodes() { return SVGElementInstanceList::create(this); } void SVGElementInstance::setShadowTreeElement(SVGElement* element) { ASSERT(element); m_shadowTreeElement = element; } void SVGElementInstance::forgetWrapper() { #if USE(JSC) // FIXME: This is fragile, as discussed with Sam. Need to find a better solution. // Think about the case where JS explicitely holds "var root = useElement.instanceRoot;". // We still have to recreate this wrapper somehow. The gc collection below, won't catch it. // If the use shadow tree has been rebuilt, just the JSSVGElementInstance objects // are still holding RefPtrs of SVGElementInstance objects, which prevent us to // be deleted (and the shadow tree is not destructed as well). Force JS GC. gcController().garbageCollectNow(); #endif } void SVGElementInstance::appendChild(PassRefPtr child) { appendChildToContainer(child.get(), this); } void SVGElementInstance::invalidateAllInstancesOfElement(SVGElement* element) { if (!element) return; HashSet set = element->instancesForElement(); if (set.isEmpty()) return; // Find all use elements referencing the instances - ask them _once_ to rebuild. HashSet::const_iterator it = set.begin(); const HashSet::const_iterator end = set.end(); for (; it != end; ++it) (*it)->setNeedsUpdate(true); } void SVGElementInstance::setNeedsUpdate(bool value) { m_needsUpdate = value; if (m_needsUpdate) correspondingUseElement()->setNeedsStyleRecalc(); } ScriptExecutionContext* SVGElementInstance::scriptExecutionContext() const { if (SVGElement* element = correspondingElement()) return element->document(); return 0; } bool SVGElementInstance::addEventListener(const AtomicString& eventType, PassRefPtr listener, bool useCapture) { if (!correspondingElement()) return false; return correspondingElement()->addEventListener(eventType, listener, useCapture); } bool SVGElementInstance::removeEventListener(const AtomicString& eventType, EventListener* listener, bool useCapture) { if (!correspondingElement()) return false; return correspondingElement()->removeEventListener(eventType, listener, useCapture); } void SVGElementInstance::removeAllEventListeners() { if (!correspondingElement()) return; correspondingElement()->removeAllEventListeners(); } bool SVGElementInstance::dispatchEvent(PassRefPtr prpEvent) { RefPtr protect = this; RefPtr event = prpEvent; event->setTarget(this); SVGElement* element = shadowTreeElement(); if (!element) return false; RefPtr view = element->document()->view(); return element->dispatchGenericEvent(event.release()); } EventTargetData* SVGElementInstance::eventTargetData() { return correspondingElement() ? correspondingElement()->eventTargetData() : 0; } EventTargetData* SVGElementInstance::ensureEventTargetData() { return &dummyEventTargetData(); // return something, so we don't crash } } // namespace WebCore #endif // ENABLE(SVG)