This source file includes following definitions.
- sizeForShareableElementDataWithAttributeCount
- m_animatedSVGAttributesAreDirty
- m_animatedSVGAttributesAreDirty
- m_idForStyleResolution
- destroy
- makeUniqueCopy
- isEquivalent
- getAttrIndex
- getAttributeItemIndexSlowCase
- createWithAttributes
- m_attributeVector
- create
- makeShareableCopy
- getAttributeItem
#include "config.h"
#include "core/dom/ElementData.h"
#include "core/css/StylePropertySet.h"
#include "core/dom/Attr.h"
#include "core/dom/QualifiedName.h"
#include "wtf/Vector.h"
namespace WebCore {
struct SameSizeAsElementData : public RefCounted<SameSizeAsElementData> {
unsigned bitfield;
void* refPtrs[3];
};
COMPILE_ASSERT(sizeof(ElementData) == sizeof(SameSizeAsElementData), element_attribute_data_should_stay_small);
static size_t sizeForShareableElementDataWithAttributeCount(unsigned count)
{
return sizeof(ShareableElementData) + sizeof(Attribute) * count;
}
ElementData::ElementData()
: m_isUnique(true)
, m_arraySize(0)
, m_presentationAttributeStyleIsDirty(false)
, m_styleAttributeIsDirty(false)
, m_animatedSVGAttributesAreDirty(false)
{
}
ElementData::ElementData(unsigned arraySize)
: m_isUnique(false)
, m_arraySize(arraySize)
, m_presentationAttributeStyleIsDirty(false)
, m_styleAttributeIsDirty(false)
, m_animatedSVGAttributesAreDirty(false)
{
}
ElementData::ElementData(const ElementData& other, bool isUnique)
: m_isUnique(isUnique)
, m_arraySize(isUnique ? 0 : other.length())
, m_presentationAttributeStyleIsDirty(other.m_presentationAttributeStyleIsDirty)
, m_styleAttributeIsDirty(other.m_styleAttributeIsDirty)
, m_animatedSVGAttributesAreDirty(other.m_animatedSVGAttributesAreDirty)
, m_classNames(other.m_classNames)
, m_idForStyleResolution(other.m_idForStyleResolution)
{
}
void ElementData::destroy()
{
if (m_isUnique)
delete static_cast<UniqueElementData*>(this);
else
delete static_cast<ShareableElementData*>(this);
}
PassRefPtr<UniqueElementData> ElementData::makeUniqueCopy() const
{
if (isUnique())
return adoptRef(new UniqueElementData(static_cast<const UniqueElementData&>(*this)));
return adoptRef(new UniqueElementData(static_cast<const ShareableElementData&>(*this)));
}
bool ElementData::isEquivalent(const ElementData* other) const
{
if (!other)
return isEmpty();
unsigned length = this->length();
if (length != other->length())
return false;
for (unsigned i = 0; i < length; ++i) {
const Attribute& attribute = attributeItem(i);
const Attribute* otherAttr = other->getAttributeItem(attribute.name());
if (!otherAttr || attribute.value() != otherAttr->value())
return false;
}
return true;
}
size_t ElementData::getAttrIndex(Attr* attr) const
{
unsigned length = this->length();
for (unsigned i = 0; i < length; ++i) {
if (attributeItem(i).name() == attr->qualifiedName())
return i;
}
return kNotFound;
}
size_t ElementData::getAttributeItemIndexSlowCase(const AtomicString& name, bool shouldIgnoreAttributeCase) const
{
unsigned length = this->length();
for (unsigned i = 0; i < length; ++i) {
const Attribute& attribute = attributeItem(i);
if (!attribute.name().hasPrefix()) {
if (shouldIgnoreAttributeCase && equalIgnoringCase(name, attribute.localName()))
return i;
} else {
if (equalPossiblyIgnoringCase(name, attribute.name().toString(), shouldIgnoreAttributeCase))
return i;
}
}
return kNotFound;
}
ShareableElementData::ShareableElementData(const Vector<Attribute>& attributes)
: ElementData(attributes.size())
{
for (unsigned i = 0; i < m_arraySize; ++i)
new (&m_attributeArray[i]) Attribute(attributes[i]);
}
ShareableElementData::~ShareableElementData()
{
for (unsigned i = 0; i < m_arraySize; ++i)
m_attributeArray[i].~Attribute();
}
ShareableElementData::ShareableElementData(const UniqueElementData& other)
: ElementData(other, false)
{
ASSERT(!other.m_presentationAttributeStyle);
if (other.m_inlineStyle) {
m_inlineStyle = other.m_inlineStyle->immutableCopyIfNeeded();
}
for (unsigned i = 0; i < m_arraySize; ++i)
new (&m_attributeArray[i]) Attribute(other.m_attributeVector.at(i));
}
PassRefPtr<ShareableElementData> ShareableElementData::createWithAttributes(const Vector<Attribute>& attributes)
{
void* slot = WTF::fastMalloc(sizeForShareableElementDataWithAttributeCount(attributes.size()));
return adoptRef(new (slot) ShareableElementData(attributes));
}
UniqueElementData::UniqueElementData()
{
}
UniqueElementData::UniqueElementData(const UniqueElementData& other)
: ElementData(other, true)
, m_presentationAttributeStyle(other.m_presentationAttributeStyle)
, m_attributeVector(other.m_attributeVector)
{
m_inlineStyle = other.m_inlineStyle ? other.m_inlineStyle->mutableCopy() : nullptr;
}
UniqueElementData::UniqueElementData(const ShareableElementData& other)
: ElementData(other, true)
{
ASSERT(!other.m_inlineStyle || !other.m_inlineStyle->isMutable());
m_inlineStyle = other.m_inlineStyle;
unsigned length = other.length();
m_attributeVector.reserveCapacity(length);
for (unsigned i = 0; i < length; ++i)
m_attributeVector.uncheckedAppend(other.m_attributeArray[i]);
}
PassRefPtr<UniqueElementData> UniqueElementData::create()
{
return adoptRef(new UniqueElementData);
}
PassRefPtr<ShareableElementData> UniqueElementData::makeShareableCopy() const
{
void* slot = WTF::fastMalloc(sizeForShareableElementDataWithAttributeCount(m_attributeVector.size()));
return adoptRef(new (slot) ShareableElementData(*this));
}
Attribute* UniqueElementData::getAttributeItem(const QualifiedName& name)
{
unsigned length = this->length();
for (unsigned i = 0; i < length; ++i) {
if (m_attributeVector.at(i).name().matches(name))
return &m_attributeVector.at(i);
}
return 0;
}
}