/* * Copyright (C) 2010 Google Inc. All rights reserved. * Copyright (C) 2015, 2016 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #include "DOMTokenList.h" #include "ExceptionCode.h" #include "HTMLParserIdioms.h" #include "SpaceSplitString.h" #include #include #include namespace WebCore { bool DOMTokenList::validateToken(const String& token, ExceptionCode& ec) { if (token.isEmpty()) { ec = SYNTAX_ERR; return false; } unsigned length = token.length(); for (unsigned i = 0; i < length; ++i) { if (isHTMLSpace(token[i])) { ec = INVALID_CHARACTER_ERR; return false; } } return true; } bool DOMTokenList::validateTokens(const String* tokens, size_t length, ExceptionCode& ec) { for (size_t i = 0; i < length; ++i) { if (!validateToken(tokens[i], ec)) return false; } return true; } bool DOMTokenList::contains(const AtomicString& token, ExceptionCode& ec) const { if (!validateToken(token, ec)) return false; return m_tokens.contains(token); } inline void DOMTokenList::addInternal(const String* tokens, size_t length, ExceptionCode& ec) { // This is usually called with a single token. Vector uniqueTokens; uniqueTokens.reserveInitialCapacity(length); for (size_t i = 0; i < length; ++i) { if (!validateToken(tokens[i], ec)) return; if (!m_tokens.contains(tokens[i]) && !uniqueTokens.contains(tokens[i])) uniqueTokens.uncheckedAppend(tokens[i]); } if (!uniqueTokens.isEmpty()) m_tokens.appendVector(uniqueTokens); updateAfterTokenChange(); } void DOMTokenList::add(const Vector& tokens, ExceptionCode& ec) { addInternal(tokens.data(), tokens.size(), ec); } void DOMTokenList::add(const WTF::AtomicString& token, ExceptionCode& ec) { addInternal(&token.string(), 1, ec); } inline void DOMTokenList::removeInternal(const String* tokens, size_t length, ExceptionCode& ec) { if (!validateTokens(tokens, length, ec)) return; for (size_t i = 0; i < length; ++i) m_tokens.removeFirst(tokens[i]); updateAfterTokenChange(); } void DOMTokenList::remove(const Vector& tokens, ExceptionCode& ec) { removeInternal(tokens.data(), tokens.size(), ec); } void DOMTokenList::remove(const WTF::AtomicString& token, ExceptionCode& ec) { removeInternal(&token.string(), 1, ec); } bool DOMTokenList::toggle(const AtomicString& token, Optional force, ExceptionCode& ec) { if (!validateToken(token, ec)) return false; if (m_tokens.contains(token)) { if (!force.valueOr(false)) { m_tokens.removeFirst(token); updateAfterTokenChange(); return false; } return true; } if (force && !force.value()) return false; m_tokens.append(token); updateAfterTokenChange(); return true; } const AtomicString& DOMTokenList::value() const { if (m_cachedValue.isNull()) { // https://dom.spec.whatwg.org/#concept-ordered-set-serializer StringBuilder builder; for (auto& token : m_tokens) { if (!builder.isEmpty()) builder.append(' '); builder.append(token); } m_cachedValue = builder.toAtomicString(); ASSERT(!m_cachedValue.isNull()); } return m_cachedValue; } void DOMTokenList::setValue(const String& value) { setValueInternal(value); updateAfterTokenChange(); } void DOMTokenList::setValueInternal(const WTF::String& value) { // Clear tokens but not capacity. m_tokens.shrink(0); HashSet addedTokens; // https://dom.spec.whatwg.org/#ordered%20sets for (unsigned start = 0; ; ) { while (start < value.length() && isHTMLSpace(value[start])) ++start; if (start >= value.length()) break; unsigned end = start + 1; while (end < value.length() && !isHTMLSpace(value[end])) ++end; AtomicString token = value.substring(start, end - start); if (!addedTokens.contains(token)) { m_tokens.append(token); addedTokens.add(token); } start = end + 1; } m_tokens.shrinkToFit(); m_cachedValue = nullAtom; } } // namespace WebCore