2020-08-04 01:23:53 +03:00
|
|
|
/**
|
|
|
|
* Copyright 2018 Google Inc. All rights reserved.
|
|
|
|
* Modifications copyright (c) Microsoft Corporation.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
2020-08-20 07:32:12 +03:00
|
|
|
|
2020-09-26 20:59:27 +03:00
|
|
|
import { it, expect } from './fixtures';
|
2020-08-04 01:23:53 +03:00
|
|
|
|
2020-08-28 14:20:29 +03:00
|
|
|
it('should work for open shadow roots', async ({page, server}) => {
|
2020-08-04 01:23:53 +03:00
|
|
|
await page.goto(server.PREFIX + '/deep-shadow.html');
|
|
|
|
expect(await page.$eval(`css=span`, e => e.textContent)).toBe('Hello from root1');
|
|
|
|
expect(await page.$eval(`css=[attr="value\\ space"]`, e => e.textContent)).toBe('Hello from root3 #2');
|
|
|
|
expect(await page.$eval(`css=[attr='value\\ \\space']`, e => e.textContent)).toBe('Hello from root3 #2');
|
|
|
|
expect(await page.$eval(`css=div div span`, e => e.textContent)).toBe('Hello from root2');
|
|
|
|
expect(await page.$eval(`css=div span + span`, e => e.textContent)).toBe('Hello from root3 #2');
|
|
|
|
expect(await page.$eval(`css=span + [attr*="value"]`, e => e.textContent)).toBe('Hello from root3 #2');
|
|
|
|
expect(await page.$eval(`css=[data-testid="foo"] + [attr*="value"]`, e => e.textContent)).toBe('Hello from root3 #2');
|
|
|
|
expect(await page.$eval(`css=#target`, e => e.textContent)).toBe('Hello from root2');
|
|
|
|
expect(await page.$eval(`css=div #target`, e => e.textContent)).toBe('Hello from root2');
|
|
|
|
expect(await page.$eval(`css=div div #target`, e => e.textContent)).toBe('Hello from root2');
|
|
|
|
expect(await page.$(`css=div div div #target`)).toBe(null);
|
|
|
|
expect(await page.$eval(`css=section > div div span`, e => e.textContent)).toBe('Hello from root2');
|
|
|
|
expect(await page.$eval(`css=section > div div span:nth-child(2)`, e => e.textContent)).toBe('Hello from root3 #2');
|
|
|
|
expect(await page.$(`css=section div div div div`)).toBe(null);
|
|
|
|
|
|
|
|
const root2 = await page.$(`css=div div`);
|
|
|
|
expect(await root2.$eval(`css=#target`, e => e.textContent)).toBe('Hello from root2');
|
|
|
|
expect(await root2.$(`css:light=#target`)).toBe(null);
|
|
|
|
const root2Shadow = await root2.evaluateHandle(r => r.shadowRoot);
|
|
|
|
expect(await root2Shadow.$eval(`css:light=#target`, e => e.textContent)).toBe('Hello from root2');
|
|
|
|
const root3 = (await page.$$(`css=div div`))[1];
|
|
|
|
expect(await root3.$eval(`text=root3`, e => e.textContent)).toBe('Hello from root3');
|
|
|
|
expect(await root3.$eval(`css=[attr*="value"]`, e => e.textContent)).toBe('Hello from root3 #2');
|
|
|
|
expect(await root3.$(`css:light=[attr*="value"]`)).toBe(null);
|
|
|
|
});
|
|
|
|
|
2020-08-28 14:20:29 +03:00
|
|
|
it('should work with > combinator and spaces', async ({page, server}) => {
|
2020-08-04 01:23:53 +03:00
|
|
|
await page.setContent(`<div foo="bar" bar="baz"><span></span></div>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"] > span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"]> span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"] >span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"]>span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"] > span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"]> span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"] >span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"][bar="baz"] > span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"][bar="baz"]> span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"][bar="baz"] >span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"][bar="baz"]>span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"][bar="baz"] > span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"][bar="baz"]> span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
expect(await page.$eval(`div[foo="bar"][bar="baz"] >span`, e => e.outerHTML)).toBe(`<span></span>`);
|
|
|
|
});
|
|
|
|
|
2020-08-28 14:20:29 +03:00
|
|
|
it('should work with comma separated list', async ({page, server}) => {
|
2020-08-04 01:23:53 +03:00
|
|
|
await page.goto(server.PREFIX + '/deep-shadow.html');
|
|
|
|
expect(await page.$$eval(`css=span,section #root1`, els => els.length)).toBe(5);
|
|
|
|
expect(await page.$$eval(`css=section #root1, div span`, els => els.length)).toBe(5);
|
|
|
|
expect(await page.$eval(`css=doesnotexist , section #root1`, e => e.id)).toBe('root1');
|
|
|
|
expect(await page.$$eval(`css=doesnotexist ,section #root1`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=span,div span`, els => els.length)).toBe(4);
|
|
|
|
expect(await page.$$eval(`css=span,div span,div div span`, els => els.length)).toBe(4);
|
|
|
|
expect(await page.$$eval(`css=#target,[attr="value\\ space"]`, els => els.length)).toBe(2);
|
|
|
|
expect(await page.$$eval(`css=#target,[data-testid="foo"],[attr="value\\ space"]`, els => els.length)).toBe(4);
|
|
|
|
expect(await page.$$eval(`css=#target,[data-testid="foo"],[attr="value\\ space"],span`, els => els.length)).toBe(4);
|
|
|
|
});
|
|
|
|
|
2020-08-28 14:20:29 +03:00
|
|
|
it('should keep dom order with comma separated list', async ({page}) => {
|
2020-08-04 01:23:53 +03:00
|
|
|
await page.setContent(`<section><span><div><x></x><y></y></div></span></section>`);
|
|
|
|
expect(await page.$$eval(`css=span,div`, els => els.map(e => e.nodeName).join(','))).toBe('SPAN,DIV');
|
|
|
|
expect(await page.$$eval(`css=div,span`, els => els.map(e => e.nodeName).join(','))).toBe('SPAN,DIV');
|
|
|
|
expect(await page.$$eval(`css=span div, div`, els => els.map(e => e.nodeName).join(','))).toBe('DIV');
|
|
|
|
expect(await page.$$eval(`*css=section >> css=div,span`, els => els.map(e => e.nodeName).join(','))).toBe('SECTION');
|
|
|
|
expect(await page.$$eval(`css=section >> *css=div >> css=x,y`, els => els.map(e => e.nodeName).join(','))).toBe('DIV');
|
|
|
|
expect(await page.$$eval(`css=section >> *css=div,span >> css=x,y`, els => els.map(e => e.nodeName).join(','))).toBe('SPAN,DIV');
|
|
|
|
expect(await page.$$eval(`css=section >> *css=div,span >> css=y`, els => els.map(e => e.nodeName).join(','))).toBe('SPAN,DIV');
|
|
|
|
});
|
|
|
|
|
2020-08-28 14:20:29 +03:00
|
|
|
it('should work with comma inside text', async ({page}) => {
|
2020-08-04 01:23:53 +03:00
|
|
|
await page.setContent(`<span></span><div attr="hello,world!"></div>`);
|
|
|
|
expect(await page.$eval(`css=div[attr="hello,world!"]`, e => e.outerHTML)).toBe('<div attr="hello,world!"></div>');
|
|
|
|
expect(await page.$eval(`css=[attr="hello,world!"]`, e => e.outerHTML)).toBe('<div attr="hello,world!"></div>');
|
|
|
|
expect(await page.$eval(`css=div[attr='hello,world!']`, e => e.outerHTML)).toBe('<div attr="hello,world!"></div>');
|
|
|
|
expect(await page.$eval(`css=[attr='hello,world!']`, e => e.outerHTML)).toBe('<div attr="hello,world!"></div>');
|
|
|
|
expect(await page.$eval(`css=div[attr="hello,world!"],span`, e => e.outerHTML)).toBe('<span></span>');
|
|
|
|
});
|
|
|
|
|
2020-08-28 14:20:29 +03:00
|
|
|
it('should work with attribute selectors', async ({page}) => {
|
2020-08-04 01:23:53 +03:00
|
|
|
await page.setContent(`<div attr="hello world" attr2="hello-''>>foo=bar[]" attr3="] span"><span></span></div>`);
|
2020-08-12 01:50:53 +03:00
|
|
|
await page.evaluate(() => window['div'] = document.querySelector('div'));
|
2020-08-04 01:23:53 +03:00
|
|
|
const selectors = [
|
|
|
|
`[attr="hello world"]`,
|
|
|
|
`[attr = "hello world"]`,
|
|
|
|
`[attr ~= world]`,
|
|
|
|
`[attr ^=hello ]`,
|
|
|
|
`[attr $= world ]`,
|
|
|
|
`[attr *= "llo wor" ]`,
|
|
|
|
`[attr2 |= hello]`,
|
|
|
|
`[attr = "Hello World" i ]`,
|
|
|
|
`[attr *= "llo WOR"i]`,
|
|
|
|
`[attr $= woRLD i]`,
|
|
|
|
`[attr2 = "hello-''>>foo=bar[]"]`,
|
|
|
|
`[attr2 $="foo=bar[]"]`,
|
|
|
|
];
|
|
|
|
for (const selector of selectors)
|
2020-08-12 01:50:53 +03:00
|
|
|
expect(await page.$eval(selector, e => e === window['div'])).toBe(true);
|
|
|
|
expect(await page.$eval(`[attr*=hello] span`, e => e.parentNode === window['div'])).toBe(true);
|
|
|
|
expect(await page.$eval(`[attr*=hello] >> span`, e => e.parentNode === window['div'])).toBe(true);
|
|
|
|
expect(await page.$eval(`[attr3="] span"] >> span`, e => e.parentNode === window['div'])).toBe(true);
|
2020-08-04 01:23:53 +03:00
|
|
|
});
|
2020-09-03 02:00:15 +03:00
|
|
|
|
|
|
|
it('should not match root after >>', async ({page, server}) => {
|
|
|
|
await page.setContent('<section><div>test</div></section>');
|
|
|
|
const element = await page.$('css=section >> css=section');
|
|
|
|
expect(element).toBe(null);
|
|
|
|
});
|
2020-11-13 22:47:44 +03:00
|
|
|
|
|
|
|
it('should work with numerical id', async ({page, server}) => {
|
|
|
|
await page.setContent('<section id="123"></section>');
|
|
|
|
const element = await page.$('#\\31\\32\\33');
|
|
|
|
expect(element).toBeTruthy();
|
|
|
|
});
|
2020-12-01 03:31:40 +03:00
|
|
|
|
|
|
|
it('should work with :nth-child', async ({page, server}) => {
|
|
|
|
await page.goto(server.PREFIX + '/deep-shadow.html');
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(odd)`, els => els.length)).toBe(3);
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(even)`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(n+1)`, els => els.length)).toBe(4);
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(n+2)`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(2n)`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(2n+1)`, els => els.length)).toBe(3);
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(-n)`, els => els.length)).toBe(0);
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(-n+1)`, els => els.length)).toBe(3);
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(-n+2)`, els => els.length)).toBe(4);
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(23n+2)`, els => els.length)).toBe(1);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should work with :not', async ({page, server}) => {
|
|
|
|
await page.goto(server.PREFIX + '/deep-shadow.html');
|
|
|
|
expect(await page.$$eval(`css=div:not(#root1)`, els => els.length)).toBe(2);
|
|
|
|
expect(await page.$$eval(`css=body :not(span)`, els => els.length)).toBe(4);
|
|
|
|
expect(await page.$$eval(`css=div > :not(span):not(div)`, els => els.length)).toBe(0);
|
|
|
|
});
|
|
|
|
|
2020-12-03 01:46:19 +03:00
|
|
|
it('should work with ~', async ({page}) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<div id=div1></div>
|
|
|
|
<div id=div2></div>
|
|
|
|
<div id=div3></div>
|
|
|
|
<div id=div4></div>
|
|
|
|
<div id=div5></div>
|
|
|
|
<div id=div6></div>
|
|
|
|
`);
|
|
|
|
expect(await page.$$eval(`css=#div1 ~ div ~ #div6`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=#div1 ~ div ~ div`, els => els.length)).toBe(4);
|
|
|
|
expect(await page.$$eval(`css=#div3 ~ div ~ div`, els => els.length)).toBe(2);
|
|
|
|
expect(await page.$$eval(`css=#div4 ~ div ~ div`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=#div5 ~ div ~ div`, els => els.length)).toBe(0);
|
|
|
|
expect(await page.$$eval(`css=#div3 ~ #div2 ~ #div6`, els => els.length)).toBe(0);
|
|
|
|
expect(await page.$$eval(`css=#div3 ~ #div4 ~ #div5`, els => els.length)).toBe(1);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should work with +', async ({page}) => {
|
|
|
|
await page.setContent(`
|
|
|
|
<div id=div1></div>
|
|
|
|
<div id=div2></div>
|
|
|
|
<div id=div3></div>
|
|
|
|
<div id=div4></div>
|
|
|
|
<div id=div5></div>
|
|
|
|
<div id=div6></div>
|
|
|
|
`);
|
|
|
|
expect(await page.$$eval(`css=#div1 ~ div + #div6`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=#div1 ~ div + div`, els => els.length)).toBe(4);
|
|
|
|
expect(await page.$$eval(`css=#div3 + div + div`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=#div4 ~ #div5 + div`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=#div5 + div + div`, els => els.length)).toBe(0);
|
|
|
|
expect(await page.$$eval(`css=#div3 ~ #div2 + #div6`, els => els.length)).toBe(0);
|
|
|
|
expect(await page.$$eval(`css=#div3 + #div4 + #div5`, els => els.length)).toBe(1);
|
|
|
|
});
|
|
|
|
|
2020-12-01 03:31:40 +03:00
|
|
|
it('should work with spaces in :nth-child and :not', test => {
|
|
|
|
test.fixme('Our selector parser is broken');
|
|
|
|
}, async ({page, server}) => {
|
|
|
|
await page.goto(server.PREFIX + '/deep-shadow.html');
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(23n +2)`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=span:nth-child(23n+ 2)`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=span:nth-child( 23n + 2 )`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=span:not(#root1 #target)`, els => els.length)).toBe(3);
|
|
|
|
expect(await page.$$eval(`css=span:not(:not(#root1 #target))`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=span:not(span:not(#root1 #target))`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=div > :not(span)`, els => els.length)).toBe(2);
|
|
|
|
expect(await page.$$eval(`css=body :not(span, div)`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=span, section:not(span, div)`, els => els.length)).toBe(5);
|
|
|
|
});
|
2020-12-03 01:46:19 +03:00
|
|
|
|
|
|
|
it('should work with :is', test => {
|
|
|
|
test.skip('Needs a new selector evaluator');
|
|
|
|
}, async ({page, server}) => {
|
|
|
|
await page.goto(server.PREFIX + '/deep-shadow.html');
|
|
|
|
expect(await page.$$eval(`css=div:is(#root1)`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=div:is(#root1, #target)`, els => els.length)).toBe(1);
|
|
|
|
expect(await page.$$eval(`css=div:is(span, #target)`, els => els.length)).toBe(0);
|
|
|
|
expect(await page.$$eval(`css=div:is(span, #root1 > *)`, els => els.length)).toBe(2);
|
|
|
|
expect(await page.$$eval(`css=div:is(section div)`, els => els.length)).toBe(3);
|
|
|
|
expect(await page.$$eval(`css=:is(div, span)`, els => els.length)).toBe(7);
|
|
|
|
expect(await page.$$eval(`css=section:is(section) div:is(section div)`, els => els.length)).toBe(3);
|
|
|
|
expect(await page.$$eval(`css=:is(div, span) > *`, els => els.length)).toBe(6);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should work with :has', test => {
|
|
|
|
test.skip('Needs a new selector evaluator');
|
|
|
|
}, async ({page, server}) => {
|
|
|
|
await page.goto(server.PREFIX + '/deep-shadow.html');
|
|
|
|
expect(await page.$$eval(`css=div:has(#target)`, els => els.length)).toBe(2);
|
|
|
|
expect(await page.$$eval(`css=div:has([data-testid=foo])`, els => els.length)).toBe(3);
|
|
|
|
expect(await page.$$eval(`css=div:has([attr*=value])`, els => els.length)).toBe(2);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should work with :scope', test => {
|
|
|
|
test.skip('Needs a new selector evaluator');
|
|
|
|
}, async ({page, server}) => {
|
|
|
|
await page.goto(server.PREFIX + '/deep-shadow.html');
|
|
|
|
// 'is' does not change the scope, so it remains 'html'.
|
|
|
|
expect(await page.$$eval(`css=div:is(:scope#root1)`, els => els.length)).toBe(0);
|
|
|
|
expect(await page.$$eval(`css=div:is(:scope #root1)`, els => els.length)).toBe(1);
|
|
|
|
// 'has' does change the scope, so it becomes the 'div' we are querying.
|
|
|
|
expect(await page.$$eval(`css=div:has(:scope > #target)`, els => els.length)).toBe(1);
|
|
|
|
|
|
|
|
const handle = await page.$(`css=span`);
|
|
|
|
for (const scope of [page, handle]) {
|
|
|
|
expect(await scope.$$eval(`css=:scope`, els => els.length)).toBe(1);
|
|
|
|
expect(await scope.$$eval(`css=* :scope`, els => els.length)).toBe(0);
|
|
|
|
expect(await scope.$$eval(`css=* + :scope`, els => els.length)).toBe(0);
|
|
|
|
expect(await scope.$$eval(`css=* > :scope`, els => els.length)).toBe(0);
|
|
|
|
expect(await scope.$$eval(`css=* ~ :scope`, els => els.length)).toBe(0);
|
|
|
|
}
|
|
|
|
});
|