blob: 66abcbc45d5d37f2003c950c5391b0375c277714 [file] [log] [blame]
/*
* Copyright (C) 2007 David Smith (catfish.man@gmail.com)
* Copyright (C) 2007, 2008, 2011, 2012 Apple Inc. All rights reserved.
*
* 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., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
#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 blink {
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 SpaceSplitString::Data::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 SpaceSplitString::Data::createVector(const String& string) {
unsigned length = string.length();
if (string.is8Bit()) {
createVector(string.characters8(), length);
return;
}
createVector(string.characters16(), length);
}
bool SpaceSplitString::Data::containsAll(Data& 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 SpaceSplitString::Data::add(const AtomicString& string) {
DCHECK(hasOneRef());
DCHECK(!contains(string));
m_vector.append(string);
}
void SpaceSplitString::Data::remove(unsigned index) {
DCHECK(hasOneRef());
m_vector.remove(index);
}
void SpaceSplitString::add(const AtomicString& string) {
// FIXME: add() does not allow duplicates but createVector() does.
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;
}
SpaceSplitString::DataMap& SpaceSplitString::sharedDataMap() {
DEFINE_STATIC_LOCAL(DataMap, map, ());
return map;
}
void SpaceSplitString::set(const AtomicString& inputString,
CaseFolding caseFolding) {
if (inputString.isNull()) {
clear();
return;
}
if (caseFolding == ShouldFoldCase &&
hasNonASCIIOrUpper(inputString.getString())) {
String string(inputString.getString());
string = string.foldCase();
m_data = Data::create(AtomicString(string));
} else {
m_data = Data::create(inputString);
}
}
SpaceSplitString::Data::~Data() {
if (!m_keyString.isNull())
sharedDataMap().remove(m_keyString);
}
PassRefPtr<SpaceSplitString::Data> SpaceSplitString::Data::create(
const AtomicString& string) {
Data*& data = sharedDataMap().add(string, nullptr).storedValue->value;
if (!data) {
data = new Data(string);
return adoptRef(data);
}
return data;
}
PassRefPtr<SpaceSplitString::Data> SpaceSplitString::Data::createUnique(
const Data& other) {
return adoptRef(new SpaceSplitString::Data(other));
}
SpaceSplitString::Data::Data(const AtomicString& string) : m_keyString(string) {
DCHECK(!string.isNull());
createVector(string);
}
SpaceSplitString::Data::Data(const SpaceSplitString::Data& other)
: RefCounted<Data>(), m_vector(other.m_vector) {
// Note that we don't copy m_keyString to indicate to the destructor that
// there's nothing to be removed from the sharedDataMap().
}
} // namespace blink