2020-08-05 01:09:24 +03:00
|
|
|
/**
|
|
|
|
* Copyright 2017 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
|
|
|
|
2021-05-06 17:08:22 +03:00
|
|
|
import { test as it, expect } from './pageTest';
|
2021-01-19 22:27:05 +03:00
|
|
|
|
|
|
|
async function giveItAChanceToResolve(page) {
|
|
|
|
for (let i = 0; i < 5; i++)
|
|
|
|
await page.evaluate(() => new Promise(f => requestAnimationFrame(() => requestAnimationFrame(f))));
|
|
|
|
}
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should select single option', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
await page.selectOption('select', 'blue');
|
2020-08-12 01:50:53 +03:00
|
|
|
expect(await page.evaluate(() => window['result'].onInput)).toEqual(['blue']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onChange)).toEqual(['blue']);
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should select single option by value', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
await page.selectOption('select', { value: 'blue' });
|
2020-08-12 01:50:53 +03:00
|
|
|
expect(await page.evaluate(() => window['result'].onInput)).toEqual(['blue']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onChange)).toEqual(['blue']);
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should select single option by label', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
await page.selectOption('select', { label: 'Indigo' });
|
2020-08-12 01:50:53 +03:00
|
|
|
expect(await page.evaluate(() => window['result'].onInput)).toEqual(['indigo']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onChange)).toEqual(['indigo']);
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should select single option by handle', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
await page.selectOption('select', await page.$('[id=whiteOption]'));
|
2020-08-12 01:50:53 +03:00
|
|
|
expect(await page.evaluate(() => window['result'].onInput)).toEqual(['white']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onChange)).toEqual(['white']);
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should select single option by index', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
await page.selectOption('select', { index: 2 });
|
2020-08-12 01:50:53 +03:00
|
|
|
expect(await page.evaluate(() => window['result'].onInput)).toEqual(['brown']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onChange)).toEqual(['brown']);
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should select single option by multiple attributes', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
await page.selectOption('select', { value: 'green', label: 'Green' });
|
2020-08-12 01:50:53 +03:00
|
|
|
expect(await page.evaluate(() => window['result'].onInput)).toEqual(['green']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onChange)).toEqual(['green']);
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should not select single option when some attributes do not match', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
2021-01-19 22:27:05 +03:00
|
|
|
await page.$eval('select', s => s.value = undefined);
|
|
|
|
try {
|
2021-09-27 19:58:08 +03:00
|
|
|
await page.selectOption('select', { value: 'green', label: 'Brown' }, { timeout: 300 });
|
2021-01-19 22:27:05 +03:00
|
|
|
} catch (e) {
|
|
|
|
expect(e.message).toContain('Timeout');
|
|
|
|
}
|
2020-08-05 01:09:24 +03:00
|
|
|
expect(await page.evaluate(() => document.querySelector('select').value)).toEqual('');
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should select only first option', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
2020-08-12 01:50:53 +03:00
|
|
|
await page.selectOption('select', ['blue', 'green', 'red']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onInput)).toEqual(['blue']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onChange)).toEqual(['blue']);
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should not throw when select causes navigation', async ({ page, server }) => {
|
2020-08-28 14:20:29 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
2020-08-12 01:50:53 +03:00
|
|
|
await page.$eval('select', select => select.addEventListener('input', () => window.location.href = '/empty.html'));
|
2020-08-05 01:09:24 +03:00
|
|
|
await Promise.all([
|
|
|
|
page.selectOption('select', 'blue'),
|
|
|
|
page.waitForNavigation(),
|
|
|
|
]);
|
|
|
|
expect(page.url()).toContain('empty.html');
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should select multiple options', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
2020-08-12 01:50:53 +03:00
|
|
|
await page.evaluate(() => window['makeMultiple']());
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.selectOption('select', ['blue', 'green', 'red']);
|
2020-08-12 01:50:53 +03:00
|
|
|
expect(await page.evaluate(() => window['result'].onInput)).toEqual(['blue', 'green', 'red']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onChange)).toEqual(['blue', 'green', 'red']);
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should select multiple options with attributes', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
2020-08-12 01:50:53 +03:00
|
|
|
await page.evaluate(() => window['makeMultiple']());
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.selectOption('select', [{ value: 'blue' }, { label: 'Green' }, { index: 4 }]);
|
2020-08-12 01:50:53 +03:00
|
|
|
expect(await page.evaluate(() => window['result'].onInput)).toEqual(['blue', 'gray', 'green']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onChange)).toEqual(['blue', 'gray', 'green']);
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should select options with sibling label', async ({ page, server }) => {
|
2020-11-12 02:33:23 +03:00
|
|
|
await page.setContent(`<label for=pet-select>Choose a pet</label>
|
|
|
|
<select id='pet-select'>
|
|
|
|
<option value='dog'>Dog</option>
|
|
|
|
<option value='cat'>Cat</option>
|
|
|
|
</select>`);
|
|
|
|
await page.selectOption('text=Choose a pet', 'cat');
|
|
|
|
expect(await page.$eval('select', select => select.options[select.selectedIndex].text)).toEqual('Cat');
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should select options with outer label', async ({ page, server }) => {
|
2020-11-12 02:33:23 +03:00
|
|
|
await page.setContent(`<label for=pet-select>Choose a pet
|
|
|
|
<select id='pet-select'>
|
|
|
|
<option value='dog'>Dog</option>
|
|
|
|
<option value='cat'>Cat</option>
|
|
|
|
</select></label>`);
|
|
|
|
await page.selectOption('text=Choose a pet', 'cat');
|
|
|
|
expect(await page.$eval('select', select => select.options[select.selectedIndex].text)).toEqual('Cat');
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should respect event bubbling', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
await page.selectOption('select', 'blue');
|
2020-08-12 01:50:53 +03:00
|
|
|
expect(await page.evaluate(() => window['result'].onBubblingInput)).toEqual(['blue']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onBubblingChange)).toEqual(['blue']);
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should throw when element is not a <select>', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
let error = null;
|
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
await page.selectOption('body', '').catch(e => error = e);
|
2021-09-25 06:51:09 +03:00
|
|
|
expect(error.message).toContain('Element is not a <select> element');
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should return [] on no matched values', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
2021-01-19 22:27:05 +03:00
|
|
|
const result = await page.selectOption('select', []);
|
2020-08-05 01:09:24 +03:00
|
|
|
expect(result).toEqual([]);
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should return an array of matched values', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
2020-08-12 01:50:53 +03:00
|
|
|
await page.evaluate(() => window['makeMultiple']());
|
2020-08-05 01:09:24 +03:00
|
|
|
const result = await page.selectOption('select', ['blue','black','magenta']);
|
|
|
|
expect(result.reduce((accumulator,current) => ['blue', 'black', 'magenta'].includes(current) && accumulator, true)).toEqual(true);
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should return an array of one element when multiple is not set', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
const result = await page.selectOption('select',['42','blue','black','magenta']);
|
|
|
|
expect(result.length).toEqual(1);
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should return [] on no values',async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
const result = await page.selectOption('select', []);
|
|
|
|
expect(result).toEqual([]);
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should not allow null items',async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
2020-08-12 01:50:53 +03:00
|
|
|
await page.evaluate(() => window['makeMultiple']());
|
2020-08-28 14:20:29 +03:00
|
|
|
let error = null;
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.selectOption('select', ['blue', null, 'black','magenta']).catch(e => error = e);
|
|
|
|
expect(error.message).toContain('options[1]: expected object, got null');
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should unselect with null',async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
2020-08-12 01:50:53 +03:00
|
|
|
await page.evaluate(() => window['makeMultiple']());
|
2020-08-05 01:09:24 +03:00
|
|
|
const result = await page.selectOption('select', ['blue', 'black','magenta']);
|
|
|
|
expect(result.reduce((accumulator,current) => ['blue', 'black', 'magenta'].includes(current) && accumulator, true)).toEqual(true);
|
|
|
|
await page.selectOption('select', null);
|
|
|
|
expect(await page.$eval('select', select => Array.from(select.options).every(option => !option.selected))).toEqual(true);
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should deselect all options when passed no values for a multiple select',async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
2020-08-12 01:50:53 +03:00
|
|
|
await page.evaluate(() => window['makeMultiple']());
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.selectOption('select', ['blue','black','magenta']);
|
|
|
|
await page.selectOption('select', []);
|
|
|
|
expect(await page.$eval('select', select => Array.from(select.options).every(option => !option.selected))).toEqual(true);
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should deselect all options when passed no values for a select without multiple',async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
await page.selectOption('select', ['blue','black','magenta']);
|
|
|
|
await page.selectOption('select', []);
|
|
|
|
expect(await page.$eval('select', select => Array.from(select.options).every(option => !option.selected))).toEqual(true);
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should throw if passed wrong types', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
let error;
|
|
|
|
await page.setContent('<select><option value="12"/></select>');
|
|
|
|
|
|
|
|
error = null;
|
|
|
|
try {
|
2020-09-09 13:06:52 +03:00
|
|
|
// @ts-expect-error cannot select numbers
|
|
|
|
await page.selectOption('select', 12);
|
2020-08-05 01:09:24 +03:00
|
|
|
} catch (e) {
|
|
|
|
error = e;
|
|
|
|
}
|
|
|
|
expect(error.message).toContain('options[0]: expected object, got number');
|
|
|
|
|
|
|
|
error = null;
|
|
|
|
try {
|
2020-09-09 13:06:52 +03:00
|
|
|
// @ts-expect-error cannot select numbers
|
|
|
|
await page.selectOption('select', { value: 12 });
|
2020-08-05 01:09:24 +03:00
|
|
|
} catch (e) {
|
|
|
|
error = e;
|
|
|
|
}
|
|
|
|
expect(error.message).toContain('options[0].value: expected string, got number');
|
|
|
|
|
|
|
|
error = null;
|
|
|
|
try {
|
2020-09-09 13:06:52 +03:00
|
|
|
// @ts-expect-error cannot select numbers
|
|
|
|
await page.selectOption('select', { label: 12 });
|
2020-08-05 01:09:24 +03:00
|
|
|
} catch (e) {
|
|
|
|
error = e;
|
|
|
|
}
|
|
|
|
expect(error.message).toContain('options[0].label: expected string, got number');
|
|
|
|
|
|
|
|
error = null;
|
|
|
|
try {
|
2020-09-09 13:06:52 +03:00
|
|
|
// @ts-expect-error cannot select string indices
|
|
|
|
await page.selectOption('select', { index: '12' });
|
2020-08-05 01:09:24 +03:00
|
|
|
} catch (e) {
|
|
|
|
error = e;
|
|
|
|
}
|
|
|
|
expect(error.message).toContain('options[0].index: expected number, got string');
|
|
|
|
});
|
|
|
|
// @see https://github.com/GoogleChrome/puppeteer/issues/3327
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should work when re-defining top-level Event class', async ({ page, server }) => {
|
2020-08-05 01:09:24 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
await page.evaluate(() => window.Event = null);
|
|
|
|
await page.selectOption('select', 'blue');
|
2020-08-12 01:50:53 +03:00
|
|
|
expect(await page.evaluate(() => window['result'].onInput)).toEqual(['blue']);
|
|
|
|
expect(await page.evaluate(() => window['result'].onChange)).toEqual(['blue']);
|
2020-08-05 01:09:24 +03:00
|
|
|
});
|
2021-01-19 22:27:05 +03:00
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should wait for option to be present',async ({ page, server }) => {
|
2021-01-19 22:27:05 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
const selectPromise = page.selectOption('select', 'scarlet');
|
|
|
|
let didSelect = false;
|
|
|
|
selectPromise.then(() => didSelect = true);
|
|
|
|
await giveItAChanceToResolve(page);
|
|
|
|
expect(didSelect).toBe(false);
|
|
|
|
await page.$eval('select', select => {
|
|
|
|
const option = document.createElement('option');
|
|
|
|
option.value = 'scarlet';
|
|
|
|
option.textContent = 'Scarlet';
|
|
|
|
select.appendChild(option);
|
|
|
|
});
|
|
|
|
const items = await selectPromise;
|
|
|
|
expect(items).toStrictEqual(['scarlet']);
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should wait for option index to be present',async ({ page, server }) => {
|
2021-01-19 22:27:05 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
const len = await page.$eval('select', select => select.options.length);
|
2021-09-27 19:58:08 +03:00
|
|
|
const selectPromise = page.selectOption('select', { index: len });
|
2021-01-19 22:27:05 +03:00
|
|
|
let didSelect = false;
|
|
|
|
selectPromise.then(() => didSelect = true);
|
|
|
|
await giveItAChanceToResolve(page);
|
|
|
|
expect(didSelect).toBe(false);
|
|
|
|
await page.$eval('select', select => {
|
|
|
|
const option = document.createElement('option');
|
|
|
|
option.value = 'scarlet';
|
|
|
|
option.textContent = 'Scarlet';
|
|
|
|
select.appendChild(option);
|
|
|
|
});
|
|
|
|
const items = await selectPromise;
|
|
|
|
expect(items).toStrictEqual(['scarlet']);
|
|
|
|
});
|
|
|
|
|
2021-09-27 19:58:08 +03:00
|
|
|
it('should wait for multiple options to be present',async ({ page, server }) => {
|
2021-01-19 22:27:05 +03:00
|
|
|
await page.goto(server.PREFIX + '/input/select.html');
|
|
|
|
await page.evaluate(() => window['makeMultiple']());
|
|
|
|
const selectPromise = page.selectOption('select', ['green', 'scarlet']);
|
|
|
|
let didSelect = false;
|
|
|
|
selectPromise.then(() => didSelect = true);
|
|
|
|
await giveItAChanceToResolve(page);
|
|
|
|
expect(didSelect).toBe(false);
|
|
|
|
await page.$eval('select', select => {
|
|
|
|
const option = document.createElement('option');
|
|
|
|
option.value = 'scarlet';
|
|
|
|
option.textContent = 'Scarlet';
|
|
|
|
select.appendChild(option);
|
|
|
|
});
|
|
|
|
const items = await selectPromise;
|
|
|
|
expect(items).toStrictEqual(['green', 'scarlet']);
|
|
|
|
});
|