#pragma once
#include "SVGAnimatedString.h"
#include "SVGAttributeOwnerProxy.h"
#include "SVGLangSpace.h"
#include "SVGLocatable.h"
#include "SVGNames.h"
#include "SVGParsingError.h"
#include "StyledElement.h"
#include <wtf/HashMap.h>
#include <wtf/HashSet.h>
#include <wtf/WeakPtr.h>
namespace WebCore {
class AffineTransform;
class CSSStyleDeclaration;
class DeprecatedCSSOMValue;
class Document;
class SVGDocumentExtensions;
class SVGElementRareData;
class SVGSVGElement;
class SVGUseElement;
void mapAttributeToCSSProperty(HashMap<AtomicStringImpl*, CSSPropertyID>* propertyNameToIdMap, const QualifiedName& attrName);
class SVGElement : public StyledElement, public SVGLangSpace, public CanMakeWeakPtr<SVGElement> {
WTF_MAKE_ISO_ALLOCATED(SVGElement);
public:
bool isOutermostSVGSVGElement() const;
SVGSVGElement* ownerSVGElement() const;
SVGElement* viewportElement() const;
String title() const override;
static bool isAnimatableCSSProperty(const QualifiedName&);
bool isPresentationAttributeWithSVGDOM(const QualifiedName&);
RefPtr<DeprecatedCSSOMValue> getPresentationAttribute(const String& name);
virtual bool supportsMarkers() const { return false; }
bool hasRelativeLengths() const { return !m_elementsWithRelativeLengths.isEmpty(); }
virtual bool needsPendingResourceHandling() const { return true; }
bool instanceUpdatesBlocked() const;
void setInstanceUpdatesBlocked(bool);
virtual AffineTransform localCoordinateSpaceTransform(SVGLocatable::CTMScope) const;
virtual bool isSVGGraphicsElement() const { return false; }
virtual bool isSVGGeometryElement() const { return false; }
virtual bool isFilterEffect() const { return false; }
virtual bool isGradientStop() const { return false; }
virtual bool isTextContent() const { return false; }
virtual bool isSMILElement() const { return false; }
virtual bool isValid() const { return true; }
virtual void svgAttributeChanged(const QualifiedName&);
Vector<AnimatedPropertyType> animatedPropertyTypesForAttribute(const QualifiedName&);
void sendSVGLoadEventIfPossible(bool sendParentLoadEvents = false);
void sendSVGLoadEventIfPossibleAsynchronously();
void svgLoadEventTimerFired();
virtual Timer* svgLoadEventTimer();
virtual AffineTransform* supplementalTransform() { return nullptr; }
void invalidateSVGAttributes() { ensureUniqueElementData().setAnimatedSVGAttributesAreDirty(true); }
void invalidateSVGPresentationAttributeStyle()
{
ensureUniqueElementData().setPresentationAttributeStyleIsDirty(true);
invalidateStyle();
}
const HashSet<SVGElement*>& instances() const;
bool getBoundingBox(FloatRect&, SVGLocatable::StyleUpdateStrategy = SVGLocatable::AllowStyleUpdate);
SVGElement* correspondingElement() const;
RefPtr<SVGUseElement> correspondingUseElement() const;
void setCorrespondingElement(SVGElement*);
void synchronizeAnimatedSVGAttribute(const QualifiedName&) const;
static void synchronizeAllAnimatedSVGAttribute(SVGElement*);
Optional<ElementStyle> resolveCustomStyle(const RenderStyle& parentStyle, const RenderStyle* shadowHostStyle) override;
static QualifiedName animatableAttributeForName(const AtomicString&);
#ifndef NDEBUG
bool isAnimatableAttribute(const QualifiedName&) const;
#endif
MutableStyleProperties* animatedSMILStyleProperties() const;
MutableStyleProperties& ensureAnimatedSMILStyleProperties();
void setUseOverrideComputedStyle(bool);
virtual bool haveLoadedRequiredResources();
bool addEventListener(const AtomicString& eventType, Ref<EventListener>&&, const AddEventListenerOptions&) override;
bool removeEventListener(const AtomicString& eventType, EventListener&, const ListenerOptions&) override;
bool hasFocusEventListeners() const;
bool hasTagName(const SVGQualifiedName& name) const { return hasLocalName(name.localName()); }
int tabIndex() const override;
void callClearTarget() { clearTarget(); }
class InstanceUpdateBlocker;
class InstanceInvalidationGuard;
using AttributeOwnerProxy = SVGAttributeOwnerProxyImpl<SVGElement, SVGLangSpace>;
static auto& attributeRegistry() { return AttributeOwnerProxy::attributeRegistry(); }
static bool isKnownAttribute(const QualifiedName& attributeName) { return AttributeOwnerProxy::isKnownAttribute(attributeName); }
static void registerAttributes();
virtual const SVGAttributeOwnerProxy& attributeOwnerProxy() const { return m_attributeOwnerProxy; }
void synchronizeAttribute(const QualifiedName& name) { attributeOwnerProxy().synchronizeAttribute(name); }
void synchronizeAttributes() { attributeOwnerProxy().synchronizeAttributes(); }
Vector<AnimatedPropertyType> animatedTypes(const QualifiedName& attributeName) const { return attributeOwnerProxy().animatedTypes(attributeName); }
RefPtr<SVGAnimatedProperty> lookupAnimatedProperty(const SVGAttribute& attribute) const { return attributeOwnerProxy().lookupAnimatedProperty(attribute); }
RefPtr<SVGAnimatedProperty> lookupOrCreateAnimatedProperty(const SVGAttribute& attribute) { return attributeOwnerProxy().lookupOrCreateAnimatedProperty(attribute); }
Vector<RefPtr<SVGAnimatedProperty>> lookupOrCreateAnimatedProperties(const QualifiedName& name) { return attributeOwnerProxy().lookupOrCreateAnimatedProperties(name); }
const auto& className() const { return m_className.currentValue(attributeOwnerProxy()); }
auto classNameAnimated() { return m_className.animatedProperty(attributeOwnerProxy()); }
protected:
SVGElement(const QualifiedName&, Document&);
virtual ~SVGElement();
bool isMouseFocusable() const override;
bool supportsFocus() const override { return false; }
bool rendererIsNeeded(const RenderStyle&) override;
void parseAttribute(const QualifiedName&, const AtomicString&) override;
void finishParsingChildren() override;
void attributeChanged(const QualifiedName&, const AtomicString& oldValue, const AtomicString& newValue, AttributeModificationReason = ModifiedDirectly) override;
bool childShouldCreateRenderer(const Node&) const override;
SVGElementRareData& ensureSVGRareData();
void reportAttributeParsingError(SVGParsingError, const QualifiedName&, const AtomicString&);
static CSSPropertyID cssPropertyIdForSVGAttributeName(const QualifiedName&);
bool isPresentationAttribute(const QualifiedName&) const override;
void collectStyleForPresentationAttribute(const QualifiedName&, const AtomicString&, MutableStyleProperties&) override;
InsertedIntoAncestorResult insertedIntoAncestor(InsertionType, ContainerNode&) override;
void removedFromAncestor(RemovalType, ContainerNode&) override;
void childrenChanged(const ChildChange&) override;
virtual bool selfHasRelativeLengths() const { return false; }
void updateRelativeLengthsInformation() { updateRelativeLengthsInformation(selfHasRelativeLengths(), this); }
void updateRelativeLengthsInformation(bool hasRelativeLengths, SVGElement*);
void willRecalcStyle(Style::Change) override;
private:
const RenderStyle* computedStyle(PseudoId = PseudoId::None) final;
virtual void clearTarget() { }
void buildPendingResourcesIfNeeded();
void accessKeyAction(bool sendMouseEvents) override;
#ifndef NDEBUG
virtual bool filterOutAnimatableAttribute(const QualifiedName&) const;
#endif
void invalidateInstances();
std::unique_ptr<SVGElementRareData> m_svgRareData;
HashSet<SVGElement*> m_elementsWithRelativeLengths;
AttributeOwnerProxy m_attributeOwnerProxy { *this };
SVGAnimatedStringAttribute m_className;
};
class SVGElement::InstanceInvalidationGuard {
public:
InstanceInvalidationGuard(SVGElement&);
~InstanceInvalidationGuard();
private:
SVGElement& m_element;
};
class SVGElement::InstanceUpdateBlocker {
public:
InstanceUpdateBlocker(SVGElement&);
~InstanceUpdateBlocker();
private:
SVGElement& m_element;
};
struct SVGAttributeHashTranslator {
static unsigned hash(const QualifiedName& key)
{
if (key.hasPrefix()) {
QualifiedNameComponents components = { nullAtom().impl(), key.localName().impl(), key.namespaceURI().impl() };
return hashComponents(components);
}
return DefaultHash<QualifiedName>::Hash::hash(key);
}
static bool equal(const QualifiedName& a, const QualifiedName& b) { return a.matches(b); }
};
inline SVGElement::InstanceInvalidationGuard::InstanceInvalidationGuard(SVGElement& element)
: m_element(element)
{
}
inline SVGElement::InstanceInvalidationGuard::~InstanceInvalidationGuard()
{
m_element.invalidateInstances();
}
inline SVGElement::InstanceUpdateBlocker::InstanceUpdateBlocker(SVGElement& element)
: m_element(element)
{
m_element.setInstanceUpdatesBlocked(true);
}
inline SVGElement::InstanceUpdateBlocker::~InstanceUpdateBlocker()
{
m_element.setInstanceUpdatesBlocked(false);
}
inline bool Node::hasTagName(const SVGQualifiedName& name) const
{
return isSVGElement() && downcast<SVGElement>(*this).hasTagName(name);
}
}
SPECIALIZE_TYPE_TRAITS_BEGIN(WebCore::SVGElement)
static bool isType(const WebCore::Node& node) { return node.isSVGElement(); }
SPECIALIZE_TYPE_TRAITS_END()
#include "SVGElementTypeHelpers.h"