:is and :where css selector basic support

This commit is contained in:
Adam D. Ruppe 2020-06-12 13:53:55 -04:00
parent dbe4eb45d5
commit e1aa0b2bf9
1 changed files with 41 additions and 7 deletions

48
dom.d
View File

@ -5,8 +5,6 @@
// FIXME: the scriptable list is quite arbitrary // FIXME: the scriptable list is quite arbitrary
// FIXME: https://developer.mozilla.org/en-US/docs/Web/CSS/:is
// xml entity references?! // xml entity references?!
@ -6940,6 +6938,9 @@ int intFromHex(string hex) {
string[] hasSelectors; /// :has(this) string[] hasSelectors; /// :has(this)
string[] notSelectors; /// :not(this) string[] notSelectors; /// :not(this)
string[] isSelectors; /// :is(this)
string[] whereSelectors; /// :where(this)
ParsedNth[] nthOfType; /// . ParsedNth[] nthOfType; /// .
ParsedNth[] nthLastOfType; /// . ParsedNth[] nthLastOfType; /// .
ParsedNth[] nthChild; /// . ParsedNth[] nthChild; /// .
@ -6990,6 +6991,9 @@ int intFromHex(string hex) {
foreach(a; notSelectors) ret ~= ":not(" ~ a ~ ")"; foreach(a; notSelectors) ret ~= ":not(" ~ a ~ ")";
foreach(a; hasSelectors) ret ~= ":has(" ~ a ~ ")"; foreach(a; hasSelectors) ret ~= ":has(" ~ a ~ ")";
foreach(a; isSelectors) ret ~= ":is(" ~ a ~ ")";
foreach(a; whereSelectors) ret ~= ":where(" ~ a ~ ")";
foreach(a; nthChild) ret ~= ":nth-child(" ~ a.toString ~ ")"; foreach(a; nthChild) ret ~= ":nth-child(" ~ a.toString ~ ")";
foreach(a; nthOfType) ret ~= ":nth-of-type(" ~ a.toString ~ ")"; foreach(a; nthOfType) ret ~= ":nth-of-type(" ~ a.toString ~ ")";
foreach(a; nthLastOfType) ret ~= ":nth-last-of-type(" ~ a.toString ~ ")"; foreach(a; nthLastOfType) ret ~= ":nth-last-of-type(" ~ a.toString ~ ")";
@ -7128,6 +7132,16 @@ int intFromHex(string hex) {
if(sel.matchesElement(e)) if(sel.matchesElement(e))
return false; return false;
} }
foreach(a; isSelectors) {
auto sel = Selector(a);
if(!sel.matchesElement(e))
return false;
}
foreach(a; whereSelectors) {
auto sel = Selector(a);
if(!sel.matchesElement(e))
return false;
}
foreach(a; nthChild) { foreach(a; nthChild) {
if(e.parentNode is null) if(e.parentNode is null)
@ -7548,9 +7562,14 @@ int intFromHex(string hex) {
SelectorComponent[] ret; SelectorComponent[] ret;
auto tokens = lexSelector(selector); // this will parse commas too auto tokens = lexSelector(selector); // this will parse commas too
// and now do comma-separated slices (i haz phobosophobia!) // and now do comma-separated slices (i haz phobosophobia!)
int parensCount = 0;
while (tokens.length > 0) { while (tokens.length > 0) {
size_t end = 0; size_t end = 0;
while (end < tokens.length && tokens[end] != ",") ++end; while (end < tokens.length && (parensCount > 0 || tokens[end] != ",")) {
if(tokens[end] == "(") parensCount++;
if(tokens[end] == ")") parensCount--;
++end;
}
if (end > 0) ret ~= parseSelector(tokens[0..end], caseSensitiveTags); if (end > 0) ret ~= parseSelector(tokens[0..end], caseSensitiveTags);
if (tokens.length-end < 2) break; if (tokens.length-end < 2) break;
tokens = tokens[end+1..$]; tokens = tokens[end+1..$];
@ -7744,6 +7763,14 @@ int intFromHex(string hex) {
current.nthLastOfType ~= ParsedNth(readFunctionalSelector()); current.nthLastOfType ~= ParsedNth(readFunctionalSelector());
state = State.SkippingFunctionalSelector; state = State.SkippingFunctionalSelector;
continue; continue;
case "is":
state = State.SkippingFunctionalSelector;
current.isSelectors ~= readFunctionalSelector();
continue; // now the rest of the parser skips past the parens we just handled
case "where":
state = State.SkippingFunctionalSelector;
current.whereSelectors ~= readFunctionalSelector();
continue; // now the rest of the parser skips past the parens we just handled
case "not": case "not":
state = State.SkippingFunctionalSelector; state = State.SkippingFunctionalSelector;
current.notSelectors ~= readFunctionalSelector(); current.notSelectors ~= readFunctionalSelector();
@ -7763,10 +7790,6 @@ int intFromHex(string hex) {
case "visited", "active", "hover", "target", "focus", "selected": case "visited", "active", "hover", "target", "focus", "selected":
current.attributesPresent ~= "nothing"; current.attributesPresent ~= "nothing";
// FIXME // FIXME
/*
// defined in the standard, but I don't implement it
case "not":
*/
/+ /+
// extensions not implemented // extensions not implemented
//case "text": // takes the text in the element and wraps it in an element, returning it //case "text": // takes the text in the element and wraps it in an element, returning it
@ -8768,6 +8791,17 @@ unittest {
assert(el.closest("p, div") is el); assert(el.closest("p, div") is el);
} }
unittest {
// https://developer.mozilla.org/en-US/docs/Web/CSS/:is
auto document = new Document(`<test>
<div class="foo"><p>cool</p><span>bar</span></div>
<main><p>two</p></main>
</test>`);
assert(document.querySelectorAll(":is(.foo, main) p").length == 2);
assert(document.querySelector("div:where(.foo)") !is null);
}
/* /*
Copyright: Adam D. Ruppe, 2010 - 2020 Copyright: Adam D. Ruppe, 2010 - 2020
License: <a href="http://www.boost.org/LICENSE_1_0.txt">Boost License 1.0</a>. License: <a href="http://www.boost.org/LICENSE_1_0.txt">Boost License 1.0</a>.