This source file includes following definitions.
- hasNonASCIIOrUpper
- hasNonASCIIOrUpper
- createVector
- createVector
- containsAll
- add
- remove
- add
- remove
- sharedDataMap
- set
- create
- createUnique
- m_vector
#include "config.h"
#include "core/dom/SpaceSplitString.h"
#include "core/html/parser/HTMLParserIdioms.h"
#include "wtf/ASCIICType.h"
#include "wtf/HashMap.h"
#include "wtf/text/AtomicStringHash.h"
using namespace WTF;
namespace WebCore {
template <typename CharacterType>
static inline bool hasNonASCIIOrUpper(const CharacterType* characters, unsigned length)
{
bool hasUpper = false;
CharacterType ored = 0;
for (unsigned i = 0; i < length; i++) {
CharacterType c = characters[i];
hasUpper |= isASCIIUpper(c);
ored |= c;
}
return hasUpper || (ored & ~0x7F);
}
static inline bool hasNonASCIIOrUpper(const String& string)
{
unsigned length = string.length();
if (string.is8Bit())
return hasNonASCIIOrUpper(string.characters8(), length);
return hasNonASCIIOrUpper(string.characters16(), length);
}
template <typename CharacterType>
inline void SpaceSplitStringData::createVector(const CharacterType* characters, unsigned length)
{
unsigned start = 0;
while (true) {
while (start < length && isHTMLSpace<CharacterType>(characters[start]))
++start;
if (start >= length)
break;
unsigned end = start + 1;
while (end < length && isNotHTMLSpace<CharacterType>(characters[end]))
++end;
m_vector.append(AtomicString(characters + start, end - start));
start = end + 1;
}
}
void SpaceSplitStringData::createVector(const String& string)
{
unsigned length = string.length();
if (string.is8Bit()) {
createVector(string.characters8(), length);
return;
}
createVector(string.characters16(), length);
}
bool SpaceSplitStringData::containsAll(SpaceSplitStringData& other)
{
if (this == &other)
return true;
size_t thisSize = m_vector.size();
size_t otherSize = other.m_vector.size();
for (size_t i = 0; i < otherSize; ++i) {
const AtomicString& name = other.m_vector[i];
size_t j;
for (j = 0; j < thisSize; ++j) {
if (m_vector[j] == name)
break;
}
if (j == thisSize)
return false;
}
return true;
}
void SpaceSplitStringData::add(const AtomicString& string)
{
ASSERT(hasOneRef());
ASSERT(!contains(string));
m_vector.append(string);
}
void SpaceSplitStringData::remove(unsigned index)
{
ASSERT(hasOneRef());
m_vector.remove(index);
}
void SpaceSplitString::add(const AtomicString& string)
{
if (contains(string))
return;
ensureUnique();
if (m_data)
m_data->add(string);
}
bool SpaceSplitString::remove(const AtomicString& string)
{
if (!m_data)
return false;
unsigned i = 0;
bool changed = false;
while (i < m_data->size()) {
if ((*m_data)[i] == string) {
if (!changed)
ensureUnique();
m_data->remove(i);
changed = true;
continue;
}
++i;
}
return changed;
}
typedef HashMap<AtomicString, SpaceSplitStringData*> SpaceSplitStringDataMap;
static SpaceSplitStringDataMap& sharedDataMap()
{
DEFINE_STATIC_LOCAL(SpaceSplitStringDataMap, map, ());
return map;
}
void SpaceSplitString::set(const AtomicString& inputString, bool shouldFoldCase)
{
if (inputString.isNull()) {
clear();
return;
}
String string(inputString.string());
if (shouldFoldCase && hasNonASCIIOrUpper(string))
string = string.foldCase();
m_data = SpaceSplitStringData::create(AtomicString(string));
}
SpaceSplitStringData::~SpaceSplitStringData()
{
if (!m_keyString.isNull())
sharedDataMap().remove(m_keyString);
}
PassRefPtr<SpaceSplitStringData> SpaceSplitStringData::create(const AtomicString& string)
{
SpaceSplitStringData*& data = sharedDataMap().add(string, 0).storedValue->value;
if (!data) {
data = new SpaceSplitStringData(string);
return adoptRef(data);
}
return data;
}
PassRefPtr<SpaceSplitStringData> SpaceSplitStringData::createUnique(const SpaceSplitStringData& other)
{
return adoptRef(new SpaceSplitStringData(other));
}
SpaceSplitStringData::SpaceSplitStringData(const AtomicString& string)
: m_keyString(string)
{
ASSERT(!string.isNull());
createVector(string);
}
SpaceSplitStringData::SpaceSplitStringData(const SpaceSplitStringData& other)
: RefCounted<SpaceSplitStringData>()
, m_vector(other.m_vector)
{
}
}