Selector: Stop relying on CSS.supports( "selector(...)" )

`CSS.supports( "selector(...)" )` has different semantics than selectors passed
to `querySelectorAll`. Apart from the fact that the former returns `false` for
unrecognized selectors and the latter throws, `qSA` is more forgiving and
accepts some invalid selectors, auto-correcting them where needed - for
example, mismatched brackers are auto-closed. This behavior difference is
breaking for many users.

To add to that, a recent CSSWG resolution made `:is()` & `:where()` the only
pseudos with forgiving parsing; browsers are in the process of making `:has()`
parsing unforgiving.

Taking all that into account, we go back to our previous try-catch approach
without relying on `CSS.supports( "selector(...)" )`. The only difference
is we detect forgiving parsing in `:has()` and mark the selector as buggy.

The PR also updates `playwright-webkit` so that we test against a version
of WebKit that already has non-forgiving `:has()`.

Fixes gh-5194
Closes gh-5207
Ref gh-5206
Ref gh-5098
Ref gh-5107
Ref w3c/csswg-drafts#7676
This commit is contained in:
Michał Gołębiowski-Owczarek 2023-02-14 11:42:29 +01:00 committed by GitHub
parent ac1c59a354
commit 63c3af481c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 148 additions and 183 deletions

View File

@ -55,7 +55,7 @@
"karma-webkit-launcher": "2.1.0", "karma-webkit-launcher": "2.1.0",
"load-grunt-tasks": "5.1.0", "load-grunt-tasks": "5.1.0",
"native-promise-only": "0.8.1", "native-promise-only": "0.8.1",
"playwright-webkit": "1.29.2", "playwright-webkit": "1.30.0",
"promises-aplus-tests": "2.1.2", "promises-aplus-tests": "2.1.2",
"q": "1.5.1", "q": "1.5.1",
"qunit": "2.9.2", "qunit": "2.9.2",

View File

@ -303,32 +303,6 @@ function find( selector, context, results, seed ) {
} }
try { try {
// `qSA` may not throw for unrecognized parts using forgiving parsing:
// https://drafts.csswg.org/selectors/#forgiving-selector
// like the `:is()` pseudo-class:
// https://drafts.csswg.org/selectors/#matches
// `CSS.supports` is still expected to return `false` then:
// https://drafts.csswg.org/css-conditional-4/#typedef-supports-selector-fn
// https://drafts.csswg.org/css-conditional-4/#dfn-support-selector
if ( support.cssSupportsSelector &&
// `CSS.supports( "selector(...)" )` requires the argument to the
// `selector` function to be a `<complex-selector>`, not
// a `<complex-selector-list>` which our selector may be. Wrapping with
// `:is` works around the issue and is supported by all browsers
// we support except for IE which will fail the support test anyway.
// eslint-disable-next-line no-undef
!CSS.supports( "selector(:is(" + newSelector + "))" ) ) {
// Support: IE 9 - 11+
// Throw to get to the same code path as an error directly in qSA.
// Note: once we only support browser supporting
// `CSS.supports('selector(...)')`, we can most likely drop
// the `try-catch`. IE doesn't implement the API.
throw new Error();
}
push.apply( results, push.apply( results,
newContext.querySelectorAll( newSelector ) newContext.querySelectorAll( newSelector )
); );
@ -575,33 +549,22 @@ function setDocument( node ) {
return document.querySelectorAll( ":scope" ); return document.querySelectorAll( ":scope" );
} ); } );
// Support: IE 9 - 11+ // Support: Chrome 105 - 110+, Safari 15.4 - 16.3+
// IE doesn't support `CSS.supports( "selector(...)" )`; it will throw // Make sure the the `:has()` argument is parsed unforgivingly.
// in this support test. // We include `*` in the test to detect buggy implementations that are
// // _selectively_ forgiving (specifically when the list includes at least
// Support: Chrome 105+, Firefox <106, Safari 15.4+ // one valid selector).
// Make sure forgiving mode is not used in `CSS.supports( "selector(...)" )`. // Note that we treat complete lack of support for `:has()` as if it were
// // spec-compliant support, which is fine because use of `:has()` in such
// `:is()` uses a forgiving selector list as an argument and is widely // environments will fail in the qSA path and fall back to jQuery traversal
// implemented, so it's a good one to test against. // anyway.
support.cssSupportsSelector = assert( function() { support.cssHas = assert( function() {
/* eslint-disable no-undef */ try {
document.querySelector( ":has(*,:jqfake)" );
return CSS.supports( "selector(*)" ) && return false;
} catch ( e ) {
// Support: Firefox 78-81 only return true;
// In old Firefox, `:is()` didn't use forgiving parsing. In that case, }
// fail this test as there's no selector to test against that.
// `CSS.supports` uses unforgiving parsing
document.querySelectorAll( ":is(:jqfake)" ) &&
// `*` is needed as Safari & newer Chrome implemented something in between
// for `:has()` - it throws in `qSA` if it only contains an unsupported
// argument but multiple ones, one of which is supported, are fine.
// We want to play safe in case `:is()` gets the same treatment.
!CSS.supports( "selector(:is(*,:jqfake))" );
/* eslint-enable */
} ); } );
// ID filter and find // ID filter and find
@ -752,14 +715,14 @@ function setDocument( node ) {
} }
} ); } );
if ( !support.cssSupportsSelector ) { if ( !support.cssHas ) {
// Support: Chrome 105+, Safari 15.4+ // Support: Chrome 105 - 110+, Safari 15.4 - 16.3+
// `:has()` uses a forgiving selector list as an argument so our regular // Our regular `try-catch` mechanism fails to detect natively-unsupported
// `try-catch` mechanism fails to catch `:has()` with arguments not supported // pseudo-classes inside `:has()` (such as `:has(:contains("Foo"))`)
// natively like `:has(:contains("Foo"))`. Where supported & spec-compliant, // in browsers that parse the `:has()` argument as a forgiving selector list.
// we now use `CSS.supports("selector(:is(SELECTOR_TO_BE_TESTED))")`, but // https://drafts.csswg.org/selectors/#relational now requires the argument
// outside that we mark `:has` as buggy. // to be parsed unforgivingly, but browsers have not yet fully adjusted.
rbuggyQSA.push( ":has" ); rbuggyQSA.push( ":has" );
} }

View File

@ -64,7 +64,7 @@ testIframe(
checkClone: true, checkClone: true,
checkOn: true, checkOn: true,
clearCloneStyle: true, clearCloneStyle: true,
cssSupportsSelector: false, cssHas: true,
cors: true, cors: true,
createHTMLDocument: true, createHTMLDocument: true,
disconnectedMatch: true, disconnectedMatch: true,
@ -83,38 +83,13 @@ testIframe(
sortDetached: true, sortDetached: true,
sortStable: true sortStable: true
}, },
ie_10_11: {
ajax: true,
boxSizingReliable: false,
checkClone: true,
checkOn: true,
clearCloneStyle: false,
cssSupportsSelector: false,
cors: true,
createHTMLDocument: true,
disconnectedMatch: true,
focusin: true,
getById: true,
noCloneChecked: false,
option: true,
optSelected: false,
pixelBoxStyles: true,
pixelPosition: true,
radioValue: false,
reliableMarginLeft: true,
reliableTrDimensions: false,
scope: false,
scrollboxSize: true,
sortDetached: true,
sortStable: true
},
ie_9: { ie_9: {
ajax: true, ajax: true,
boxSizingReliable: false, boxSizingReliable: false,
checkClone: true, checkClone: true,
checkOn: true, checkOn: true,
clearCloneStyle: false, clearCloneStyle: false,
cssSupportsSelector: false, cssHas: true,
cors: false, cors: false,
createHTMLDocument: true, createHTMLDocument: true,
disconnectedMatch: false, disconnectedMatch: false,
@ -133,13 +108,38 @@ testIframe(
sortDetached: true, sortDetached: true,
sortStable: true sortStable: true
}, },
ie_10_11: {
ajax: true,
boxSizingReliable: false,
checkClone: true,
checkOn: true,
clearCloneStyle: false,
cssHas: true,
cors: true,
createHTMLDocument: true,
disconnectedMatch: true,
focusin: true,
getById: true,
noCloneChecked: false,
option: true,
optSelected: false,
pixelBoxStyles: true,
pixelPosition: true,
radioValue: false,
reliableMarginLeft: true,
reliableTrDimensions: false,
scope: false,
scrollboxSize: true,
sortDetached: true,
sortStable: true
},
chrome: { chrome: {
ajax: true, ajax: true,
boxSizingReliable: true, boxSizingReliable: true,
checkClone: true, checkClone: true,
checkOn: true, checkOn: true,
clearCloneStyle: true, clearCloneStyle: true,
cssSupportsSelector: false, cssHas: false,
cors: true, cors: true,
createHTMLDocument: true, createHTMLDocument: true,
disconnectedMatch: true, disconnectedMatch: true,
@ -164,7 +164,7 @@ testIframe(
checkClone: true, checkClone: true,
checkOn: true, checkOn: true,
clearCloneStyle: true, clearCloneStyle: true,
cssSupportsSelector: false, cssHas: false,
cors: true, cors: true,
createHTMLDocument: true, createHTMLDocument: true,
disconnectedMatch: true, disconnectedMatch: true,
@ -189,7 +189,7 @@ testIframe(
checkClone: true, checkClone: true,
checkOn: true, checkOn: true,
clearCloneStyle: true, clearCloneStyle: true,
cssSupportsSelector: true, cssHas: true,
cors: true, cors: true,
createHTMLDocument: true, createHTMLDocument: true,
disconnectedMatch: true, disconnectedMatch: true,
@ -208,88 +208,13 @@ testIframe(
sortDetached: true, sortDetached: true,
sortStable: true sortStable: true
}, },
safari_9_10: {
ajax: true,
boxSizingReliable: true,
checkClone: true,
checkOn: true,
clearCloneStyle: true,
cssSupportsSelector: false,
cors: true,
createHTMLDocument: true,
disconnectedMatch: true,
focusin: false,
getById: true,
noCloneChecked: true,
option: true,
optSelected: true,
pixelBoxStyles: false,
pixelPosition: false,
radioValue: true,
reliableMarginLeft: true,
reliableTrDimensions: true,
scope: true,
scrollboxSize: true,
sortDetached: true,
sortStable: true
},
firefox: {
ajax: true,
boxSizingReliable: true,
checkClone: true,
checkOn: true,
clearCloneStyle: true,
cssSupportsSelector: true,
cors: true,
createHTMLDocument: true,
disconnectedMatch: true,
focusin: false,
getById: true,
noCloneChecked: true,
option: true,
optSelected: true,
pixelBoxStyles: true,
pixelPosition: true,
radioValue: true,
reliableMarginLeft: true,
reliableTrDimensions: false,
scope: true,
scrollboxSize: true,
sortDetached: true,
sortStable: true
},
firefox_102: {
ajax: true,
boxSizingReliable: true,
checkClone: true,
checkOn: true,
clearCloneStyle: true,
cssSupportsSelector: false,
cors: true,
createHTMLDocument: true,
disconnectedMatch: true,
focusin: false,
getById: true,
noCloneChecked: true,
option: true,
optSelected: true,
pixelBoxStyles: true,
pixelPosition: true,
radioValue: true,
reliableMarginLeft: true,
reliableTrDimensions: false,
scope: true,
scrollboxSize: true,
sortDetached: true,
sortStable: true
},
firefox_60: { firefox_60: {
ajax: true, ajax: true,
boxSizingReliable: true, boxSizingReliable: true,
checkClone: true, checkClone: true,
checkOn: true, checkOn: true,
clearCloneStyle: true, clearCloneStyle: true,
cssSupportsSelector: false, cssHas: true,
cors: true, cors: true,
createHTMLDocument: true, createHTMLDocument: true,
disconnectedMatch: true, disconnectedMatch: true,
@ -308,13 +233,13 @@ testIframe(
sortDetached: true, sortDetached: true,
sortStable: true sortStable: true
}, },
ios: { firefox_102: {
ajax: true, ajax: true,
boxSizingReliable: true, boxSizingReliable: true,
checkClone: true, checkClone: true,
checkOn: true, checkOn: true,
clearCloneStyle: true, clearCloneStyle: true,
cssSupportsSelector: false, cssHas: true,
cors: true, cors: true,
createHTMLDocument: true, createHTMLDocument: true,
disconnectedMatch: true, disconnectedMatch: true,
@ -327,19 +252,44 @@ testIframe(
pixelPosition: true, pixelPosition: true,
radioValue: true, radioValue: true,
reliableMarginLeft: true, reliableMarginLeft: true,
reliableTrDimensions: true, reliableTrDimensions: false,
scope: true, scope: true,
scrollboxSize: true, scrollboxSize: true,
sortDetached: true, sortDetached: true,
sortStable: true sortStable: true
}, },
ios_9_10: { firefox: {
ajax: true, ajax: true,
boxSizingReliable: true, boxSizingReliable: true,
checkClone: true, checkClone: true,
checkOn: true, checkOn: true,
clearCloneStyle: true, clearCloneStyle: true,
cssSupportsSelector: false, cssHas: true,
cors: true,
createHTMLDocument: true,
disconnectedMatch: true,
focusin: false,
getById: true,
noCloneChecked: true,
option: true,
optSelected: true,
pixelBoxStyles: true,
pixelPosition: true,
radioValue: true,
reliableMarginLeft: true,
reliableTrDimensions: false,
scope: true,
scrollboxSize: true,
sortDetached: true,
sortStable: true
},
ios_7: {
ajax: true,
boxSizingReliable: true,
checkClone: true,
checkOn: true,
clearCloneStyle: true,
cssHas: true,
cors: true, cors: true,
createHTMLDocument: true, createHTMLDocument: true,
disconnectedMatch: true, disconnectedMatch: true,
@ -364,7 +314,7 @@ testIframe(
checkClone: true, checkClone: true,
checkOn: true, checkOn: true,
clearCloneStyle: true, clearCloneStyle: true,
cssSupportsSelector: false, cssHas: true,
cors: true, cors: true,
createHTMLDocument: false, createHTMLDocument: false,
disconnectedMatch: true, disconnectedMatch: true,
@ -383,13 +333,13 @@ testIframe(
sortDetached: true, sortDetached: true,
sortStable: true sortStable: true
}, },
ios_7: { ios_9_10: {
ajax: true, ajax: true,
boxSizingReliable: true, boxSizingReliable: true,
checkClone: true, checkClone: true,
checkOn: true, checkOn: true,
clearCloneStyle: true, clearCloneStyle: true,
cssSupportsSelector: false, cssHas: true,
cors: true, cors: true,
createHTMLDocument: true, createHTMLDocument: true,
disconnectedMatch: true, disconnectedMatch: true,
@ -408,13 +358,63 @@ testIframe(
sortDetached: true, sortDetached: true,
sortStable: true sortStable: true
}, },
ios_11_15_3: {
ajax: true,
boxSizingReliable: true,
checkClone: true,
checkOn: true,
clearCloneStyle: true,
cssHas: true,
cors: true,
createHTMLDocument: true,
disconnectedMatch: true,
focusin: false,
getById: true,
noCloneChecked: true,
option: true,
optSelected: true,
pixelBoxStyles: true,
pixelPosition: true,
radioValue: true,
reliableMarginLeft: true,
reliableTrDimensions: true,
scope: true,
scrollboxSize: true,
sortDetached: true,
sortStable: true
},
ios: {
ajax: true,
boxSizingReliable: true,
checkClone: true,
checkOn: true,
clearCloneStyle: true,
cssHas: false,
cors: true,
createHTMLDocument: true,
disconnectedMatch: true,
focusin: false,
getById: true,
noCloneChecked: true,
option: true,
optSelected: true,
pixelBoxStyles: true,
pixelPosition: true,
radioValue: true,
reliableMarginLeft: true,
reliableTrDimensions: true,
scope: true,
scrollboxSize: true,
sortDetached: true,
sortStable: true
},
android: { android: {
ajax: true, ajax: true,
boxSizingReliable: true, boxSizingReliable: true,
checkClone: false, checkClone: false,
checkOn: false, checkOn: false,
clearCloneStyle: true, clearCloneStyle: true,
cssSupportsSelector: false, cssHas: true,
cors: true, cors: true,
createHTMLDocument: true, createHTMLDocument: true,
disconnectedMatch: true, disconnectedMatch: true,
@ -446,7 +446,7 @@ testIframe(
// Make the selector-native build pass tests. // Make the selector-native build pass tests.
for ( browserKey in expectedMap ) { for ( browserKey in expectedMap ) {
if ( !includesModule( "selector" ) ) { if ( !includesModule( "selector" ) ) {
delete expectedMap[ browserKey ].cssSupportsSelector; delete expectedMap[ browserKey ].cssHas;
delete expectedMap[ browserKey ].disconnectedMatch; delete expectedMap[ browserKey ].disconnectedMatch;
delete expectedMap[ browserKey ].getById; delete expectedMap[ browserKey ].getById;
delete expectedMap[ browserKey ].scope; delete expectedMap[ browserKey ].scope;
@ -457,10 +457,10 @@ testIframe(
if ( /edge\//i.test( userAgent ) ) { if ( /edge\//i.test( userAgent ) ) {
expected = expectedMap.edge; expected = expectedMap.edge;
} else if ( /(msie 10\.0|trident\/7\.0)/i.test( userAgent ) ) {
expected = expectedMap.ie_10_11;
} else if ( /msie 9\.0/i.test( userAgent ) ) { } else if ( /msie 9\.0/i.test( userAgent ) ) {
expected = expectedMap.ie_9; expected = expectedMap.ie_9;
} else if ( /(msie 10\.0|trident\/7\.0)/i.test( userAgent ) ) {
expected = expectedMap.ie_10_11;
} else if ( /chrome/i.test( userAgent ) ) { } else if ( /chrome/i.test( userAgent ) ) {
// Catches Chrome on Android as well (i.e. the default // Catches Chrome on Android as well (i.e. the default
@ -476,12 +476,14 @@ testIframe(
expected = expectedMap.firefox; expected = expectedMap.firefox;
} else if ( /android 4\.[0-3]/i.test( userAgent ) ) { } else if ( /android 4\.[0-3]/i.test( userAgent ) ) {
expected = expectedMap.android; expected = expectedMap.android;
} else if ( /iphone os (?:9|10)_/i.test( userAgent ) ) {
expected = expectedMap.ios_9_10;
} else if ( /iphone os 8_/i.test( userAgent ) ) {
expected = expectedMap.ios_8;
} else if ( /iphone os 7_/i.test( userAgent ) ) { } else if ( /iphone os 7_/i.test( userAgent ) ) {
expected = expectedMap.ios_7; expected = expectedMap.ios_7;
} else if ( /iphone os 8_/i.test( userAgent ) ) {
expected = expectedMap.ios_8;
} else if ( /iphone os (?:9|10)_/i.test( userAgent ) ) {
expected = expectedMap.ios_9_10;
} else if ( /iphone os (?:1[1234]_|15_[0123])/i.test( userAgent ) ) {
expected = expectedMap.ios_11_15_3;
} else if ( /(?:iphone|ipad);.*(?:iphone)? os \d+_/i.test( userAgent ) ) { } else if ( /(?:iphone|ipad);.*(?:iphone)? os \d+_/i.test( userAgent ) ) {
expected = expectedMap.ios; expected = expectedMap.ios;
} else if ( typeof URLSearchParams !== "undefined" && } else if ( typeof URLSearchParams !== "undefined" &&