blob: 766a0beb0d3d5b0a744a07fa706037f27d30dc88 [file] [log] [blame]
/*
* Copyright (C) 2003 Lars Knoll (knoll@kde.org)
* Copyright (C) 2004, 2005, 2006, 2008, 2009, 2010, 2014 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., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#ifndef CSSParserValues_h
#define CSSParserValues_h
#include "CSSSelector.h"
#include "CSSValueKeywords.h"
#include "CSSValueList.h"
#include <wtf/text/AtomicString.h>
#include <wtf/text/AtomicStringHash.h>
#include <wtf/text/WTFString.h>
namespace WebCore {
class CSSValue;
class QualifiedName;
// This can't be a StringView for 2 reasons:
// 1. lower() clobbers the data we point to.
// 2. We are an element of a union (in CSSParserValue) so we need to have a trivial destructor.
struct CSSParserString {
void init(LChar* characters, unsigned length)
{
m_data.characters8 = characters;
m_length = length;
m_is8Bit = true;
}
void init(UChar* characters, unsigned length)
{
m_data.characters16 = characters;
m_length = length;
m_is8Bit = false;
}
void init(const String& string)
{
m_length = string.length();
if (!m_length || string.is8Bit()) {
m_data.characters8 = const_cast<LChar*>(string.characters8());
m_is8Bit = true;
} else {
m_data.characters16 = const_cast<UChar*>(string.characters16());
m_is8Bit = false;
}
}
void clear()
{
m_data.characters8 = 0;
m_length = 0;
m_is8Bit = true;
}
bool is8Bit() const { return m_is8Bit; }
LChar* characters8() const { ASSERT(is8Bit()); return m_data.characters8; }
UChar* characters16() const { ASSERT(!is8Bit()); return m_data.characters16; }
template <typename CharacterType>
CharacterType* characters() const;
unsigned length() const { return m_length; }
void setLength(unsigned length) { m_length = length; }
void lower();
UChar operator[](unsigned i) const
{
ASSERT_WITH_SECURITY_IMPLICATION(i < m_length);
if (is8Bit())
return m_data.characters8[i];
return m_data.characters16[i];
}
bool equalIgnoringCase(const char* str) const
{
if (is8Bit())
return WTF::equalIgnoringCase(str, characters8(), length());
return WTF::equalIgnoringCase(str, characters16(), length());
}
operator String() const { return is8Bit() ? String(m_data.characters8, m_length) : String(m_data.characters16, m_length); }
operator AtomicString() const { return is8Bit() ? AtomicString(m_data.characters8, m_length) : AtomicString(m_data.characters16, m_length); }
union {
LChar* characters8;
UChar* characters16;
} m_data;
unsigned m_length;
bool m_is8Bit;
};
struct CSSParserFunction;
struct CSSParserVariable;
struct CSSParserValue {
CSSValueID id;
bool isInt;
union {
double fValue;
int iValue;
CSSParserString string;
CSSParserFunction* function;
CSSParserVariable* variable;
CSSParserValueList* valueList;
};
enum {
Operator = 0x100000,
Function = 0x100001,
ValueList = 0x100002,
Q_EMS = 0x100003,
Variable = 0x100004
};
int unit;
void setFromValueList(std::unique_ptr<CSSParserValueList>);
PassRefPtr<CSSValue> createCSSValue();
};
void destroy(const CSSParserValue&);
class CSSParserValueList {
WTF_MAKE_FAST_ALLOCATED;
public:
CSSParserValueList()
: m_current(0)
{
}
~CSSParserValueList();
void addValue(const CSSParserValue&);
void insertValueAt(unsigned, const CSSParserValue&);
void deleteValueAt(unsigned);
void extend(CSSParserValueList&);
unsigned size() const { return m_values.size(); }
unsigned currentIndex() { return m_current; }
CSSParserValue* current() { return m_current < m_values.size() ? &m_values[m_current] : 0; }
CSSParserValue* next() { ++m_current; return current(); }
CSSParserValue* previous()
{
if (!m_current)
return 0;
--m_current;
return current();
}
void setCurrentIndex(unsigned index)
{
ASSERT(index < m_values.size());
if (index < m_values.size())
m_current = index;
}
CSSParserValue* valueAt(unsigned i) { return i < m_values.size() ? &m_values[i] : 0; }
void clear() { m_values.clear(); }
String toString();
bool containsVariables() const;
private:
unsigned m_current;
Vector<CSSParserValue, 4> m_values;
};
struct CSSParserFunction {
WTF_MAKE_FAST_ALLOCATED;
public:
CSSParserString name;
std::unique_ptr<CSSParserValueList> args;
};
struct CSSParserVariable {
WTF_MAKE_FAST_ALLOCATED;
public:
CSSParserString name; // The custom property name
std::unique_ptr<CSSParserValueList> args; // The fallback args
};
enum class CSSParserSelectorCombinator {
Child,
DescendantSpace,
#if ENABLE(CSS_SELECTORS_LEVEL4)
DescendantDoubleChild,
#endif
DirectAdjacent,
IndirectAdjacent
};
class CSSParserSelector {
WTF_MAKE_FAST_ALLOCATED;
public:
static CSSParserSelector* parsePagePseudoSelector(const CSSParserString& pseudoTypeString);
static CSSParserSelector* parsePseudoElementSelector(CSSParserString& pseudoTypeString);
static CSSParserSelector* parsePseudoElementCueFunctionSelector(const CSSParserString& functionIdentifier, Vector<std::unique_ptr<CSSParserSelector>>* selectorVector);
static CSSParserSelector* parsePseudoClassAndCompatibilityElementSelector(CSSParserString& pseudoTypeString);
CSSParserSelector();
explicit CSSParserSelector(const QualifiedName&);
~CSSParserSelector();
std::unique_ptr<CSSSelector> releaseSelector() { return WTF::move(m_selector); }
void setValue(const AtomicString& value) { m_selector->setValue(value); }
void setAttribute(const QualifiedName& value, bool isCaseInsensitive) { m_selector->setAttribute(value, isCaseInsensitive); }
void setArgument(const AtomicString& value) { m_selector->setArgument(value); }
void setAttributeValueMatchingIsCaseInsensitive(bool isCaseInsensitive) { m_selector->setAttributeValueMatchingIsCaseInsensitive(isCaseInsensitive); }
void setMatch(CSSSelector::Match value) { m_selector->setMatch(value); }
void setRelation(CSSSelector::Relation value) { m_selector->setRelation(value); }
void setForPage() { m_selector->setForPage(); }
void adoptSelectorVector(Vector<std::unique_ptr<CSSParserSelector>>& selectorVector);
void setLangArgumentList(const Vector<CSSParserString>& stringVector);
void setPseudoClassValue(const CSSParserString& pseudoClassString);
CSSSelector::PseudoClassType pseudoClassType() const { return m_selector->pseudoClassType(); }
bool isCustomPseudoElement() const { return m_selector->isCustomPseudoElement(); }
bool isPseudoElementCueFunction() const
{
#if ENABLE(VIDEO_TRACK)
return m_selector->match() == CSSSelector::PseudoElement && m_selector->pseudoElementType() == CSSSelector::PseudoElementCue;
#else
return false;
#endif
}
bool hasShadowDescendant() const;
bool matchesPseudoElement() const;
CSSParserSelector* tagHistory() const { return m_tagHistory.get(); }
void setTagHistory(std::unique_ptr<CSSParserSelector> selector) { m_tagHistory = WTF::move(selector); }
void clearTagHistory() { m_tagHistory.reset(); }
void insertTagHistory(CSSSelector::Relation before, std::unique_ptr<CSSParserSelector>, CSSSelector::Relation after);
void appendTagHistory(CSSSelector::Relation, std::unique_ptr<CSSParserSelector>);
void appendTagHistory(CSSParserSelectorCombinator, std::unique_ptr<CSSParserSelector>);
void prependTagSelector(const QualifiedName&, bool tagIsForNamespaceRule = false);
private:
#if ENABLE(CSS_SELECTORS_LEVEL4)
void setDescendantUseDoubleChildSyntax() { m_selector->setDescendantUseDoubleChildSyntax(); }
#endif
std::unique_ptr<CSSSelector> m_selector;
std::unique_ptr<CSSParserSelector> m_tagHistory;
};
inline bool CSSParserSelector::hasShadowDescendant() const
{
return m_selector->relation() == CSSSelector::ShadowDescendant;
}
inline void CSSParserValue::setFromValueList(std::unique_ptr<CSSParserValueList> valueList)
{
id = CSSValueInvalid;
this->valueList = valueList.release();
unit = ValueList;
}
}
#endif