1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-07-24 22:17:42 +00:00

LibHTML: Implement matching for descendant selectors

The CSS engine now correctly matches selectors like "#foo #bar #baz".
This commit is contained in:
Andreas Kling 2019-10-06 15:34:42 +02:00
parent b587eb2f4d
commit 156b35742a
3 changed files with 60 additions and 5 deletions

View file

@ -15,12 +15,8 @@ StyleResolver::~StyleResolver()
{
}
static bool matches(const Selector& selector, const Element& element)
static bool matches(const Selector::Component& component, const Element& element)
{
// FIXME: Support compound selectors.
ASSERT(selector.components().size() == 1);
auto& component = selector.components().first();
switch (component.type) {
case Selector::Component::Type::Id:
return component.value == element.attribute("id");
@ -33,6 +29,32 @@ static bool matches(const Selector& selector, const Element& element)
}
}
static bool matches(const Selector& selector, int component_index, const Element& element)
{
auto& component = selector.components()[component_index];
if (!matches(component, element))
return false;
if (component.relation == Selector::Component::Relation::None)
return true;
if (component.relation == Selector::Component::Relation::Descendant) {
ASSERT(component_index != 0);
for (auto* ancestor = element.parent(); ancestor; ancestor = ancestor->parent()) {
if (!ancestor->is_element())
continue;
if (matches(selector, component_index - 1, static_cast<const Element&>(*ancestor)))
return true;
}
return false;
}
ASSERT_NOT_REACHED();
}
static bool matches(const Selector& selector, const Element& element)
{
ASSERT(!selector.components().is_empty());
return matches(selector, selector.components().size() - 1, element);
}
static StyleSheet& default_stylesheet()
{
static StyleSheet* sheet;