This source file includes following definitions.
- adopt
 
- adoptSelectorVector
 
- length
 
- deleteSelectors
 
- selectorsText
 
- forEachTagSelector
 
- forEachSelector
 
- selectorsNeedNamespaceResolution
 
- hasShadowDistributedAt
 
- selectorCrossesTreeScopes
 
#include "config.h"
#include "core/css/CSSSelectorList.h"
#include "core/css/CSSParserValues.h"
#include "wtf/text/StringBuilder.h"
namespace WebCore {
CSSSelectorList::~CSSSelectorList()
{
    deleteSelectors();
}
CSSSelectorList::CSSSelectorList(const CSSSelectorList& other)
{
    unsigned otherLength = other.length();
    m_selectorArray = reinterpret_cast<CSSSelector*>(fastMalloc(sizeof(CSSSelector) * otherLength));
    for (unsigned i = 0; i < otherLength; ++i)
        new (&m_selectorArray[i]) CSSSelector(other.m_selectorArray[i]);
}
void CSSSelectorList::adopt(CSSSelectorList& list)
{
    deleteSelectors();
    m_selectorArray = list.m_selectorArray;
    list.m_selectorArray = 0;
}
void CSSSelectorList::adoptSelectorVector(Vector<OwnPtr<CSSParserSelector> >& selectorVector)
{
    deleteSelectors();
    size_t flattenedSize = 0;
    for (size_t i = 0; i < selectorVector.size(); ++i) {
        for (CSSParserSelector* selector = selectorVector[i].get(); selector; selector = selector->tagHistory())
            ++flattenedSize;
    }
    ASSERT(flattenedSize);
    m_selectorArray = reinterpret_cast<CSSSelector*>(fastMalloc(sizeof(CSSSelector) * flattenedSize));
    size_t arrayIndex = 0;
    for (size_t i = 0; i < selectorVector.size(); ++i) {
        CSSParserSelector* current = selectorVector[i].get();
        while (current) {
            
            CSSSelector* currentSelector = current->releaseSelector().leakPtr();
            memcpy(&m_selectorArray[arrayIndex], currentSelector, sizeof(CSSSelector));
            fastFree(currentSelector);
            current = current->tagHistory();
            ASSERT(!m_selectorArray[arrayIndex].isLastInSelectorList());
            if (current)
                m_selectorArray[arrayIndex].setNotLastInTagHistory();
            ++arrayIndex;
        }
        ASSERT(m_selectorArray[arrayIndex - 1].isLastInTagHistory());
    }
    ASSERT(flattenedSize == arrayIndex);
    m_selectorArray[arrayIndex - 1].setLastInSelectorList();
    selectorVector.clear();
}
unsigned CSSSelectorList::length() const
{
    if (!m_selectorArray)
        return 0;
    CSSSelector* current = m_selectorArray;
    while (!current->isLastInSelectorList())
        ++current;
    return (current - m_selectorArray) + 1;
}
void CSSSelectorList::deleteSelectors()
{
    if (!m_selectorArray)
        return;
    bool finished = false;
    for (CSSSelector* s = m_selectorArray; !finished; ++s) {
        finished = s->isLastInSelectorList();
        s->~CSSSelector();
    }
    fastFree(m_selectorArray);
}
String CSSSelectorList::selectorsText() const
{
    StringBuilder result;
    for (const CSSSelector* s = first(); s; s = next(*s)) {
        if (s != first())
            result.append(", ");
        result.append(s->selectorText());
    }
    return result.toString();
}
template <typename Functor>
static bool forEachTagSelector(Functor& functor, const CSSSelector& selector)
{
    const CSSSelector* current = &selector;
    do {
        if (functor(*current))
            return true;
        if (const CSSSelectorList* selectorList = current->selectorList()) {
            for (const CSSSelector* subSelector = selectorList->first(); subSelector; subSelector = CSSSelectorList::next(*subSelector)) {
                if (forEachTagSelector(functor, *subSelector))
                    return true;
            }
        }
    } while ((current = current->tagHistory()));
    return false;
}
template <typename Functor>
static bool forEachSelector(Functor& functor, const CSSSelectorList* selectorList)
{
    for (const CSSSelector* selector = selectorList->first(); selector; selector = CSSSelectorList::next(*selector)) {
        if (forEachTagSelector(functor, *selector))
            return true;
    }
    return false;
}
class SelectorNeedsNamespaceResolutionFunctor {
public:
    bool operator()(const CSSSelector& selector)
    {
        if (selector.m_match == CSSSelector::Tag && selector.tagQName().prefix() != nullAtom && selector.tagQName().prefix() != starAtom)
            return true;
        if (selector.isAttributeSelector() && selector.attribute().prefix() != nullAtom && selector.attribute().prefix() != starAtom)
            return true;
        return false;
    }
};
bool CSSSelectorList::selectorsNeedNamespaceResolution()
{
    SelectorNeedsNamespaceResolutionFunctor functor;
    return forEachSelector(functor, this);
}
class SelectorHasShadowDistributed {
public:
    bool operator()(const CSSSelector& selector)
    {
        return selector.relationIsAffectedByPseudoContent();
    }
};
bool CSSSelectorList::hasShadowDistributedAt(size_t index) const
{
    SelectorHasShadowDistributed functor;
    return forEachTagSelector(functor, selectorAt(index));
}
class SelectorCrossesTreeScopes {
public:
    bool operator()(const CSSSelector& selector)
    {
        return selector.relation() == CSSSelector::ShadowDeep || selector.isShadowPseudoElement();
    }
};
bool CSSSelectorList::selectorCrossesTreeScopes(size_t index) const
{
    SelectorCrossesTreeScopes functor;
    return forEachTagSelector(functor, selectorAt(index));
}
}