feat(api): all getByTitle, getByAltText (#17724)

This commit is contained in:
Pavel Feldman 2022-09-29 20:45:44 -08:00 committed by GitHub
parent 7d3997e620
commit bfb8a43ece
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
11 changed files with 313 additions and 5 deletions

View File

@ -908,6 +908,16 @@ Attribute name to get the value for.
* since: v1.8 * since: v1.8
## method: Frame.getByAltText
* since: v1.27
- returns: <[Locator]>
%%-template-locator-get-by-alt-text-%%
### param: Frame.getByAltText.text = %%-locator-get-by-text-text-%%
### option: Frame.getByAltText.exact = %%-locator-get-by-text-exact-%%
## method: Frame.getByLabelText ## method: Frame.getByLabelText
* since: v1.27 * since: v1.27
- returns: <[Locator]> - returns: <[Locator]>
@ -960,6 +970,16 @@ Attribute name to get the value for.
### option: Frame.getByText.exact = %%-locator-get-by-text-exact-%% ### option: Frame.getByText.exact = %%-locator-get-by-text-exact-%%
## method: Frame.getByTitle
* since: v1.27
- returns: <[Locator]>
%%-template-locator-get-by-title-%%
### param: Frame.getByTitle.text = %%-locator-get-by-text-text-%%
### option: Frame.getByTitle.exact = %%-locator-get-by-text-exact-%%
## async method: Frame.goto ## async method: Frame.goto
* since: v1.8 * since: v1.8
* langs: * langs:

View File

@ -114,6 +114,16 @@ in that iframe.
* since: v1.17 * since: v1.17
## method: FrameLocator.getByAltText
* since: v1.27
- returns: <[Locator]>
%%-template-locator-get-by-alt-text-%%
### param: FrameLocator.getByAltText.text = %%-locator-get-by-text-text-%%
### option: FrameLocator.getByAltText.exact = %%-locator-get-by-text-exact-%%
## method: FrameLocator.getByLabelText ## method: FrameLocator.getByLabelText
* since: v1.27 * since: v1.27
- returns: <[Locator]> - returns: <[Locator]>
@ -165,6 +175,16 @@ in that iframe.
### option: FrameLocator.getByText.exact = %%-locator-get-by-text-exact-%% ### option: FrameLocator.getByText.exact = %%-locator-get-by-text-exact-%%
## method: FrameLocator.getByTitle
* since: v1.27
- returns: <[Locator]>
%%-template-locator-get-by-title-%%
### param: FrameLocator.getByTitle.text = %%-locator-get-by-text-text-%%
### option: FrameLocator.getByTitle.exact = %%-locator-get-by-text-exact-%%
## method: FrameLocator.last ## method: FrameLocator.last
* since: v1.17 * since: v1.17
- returns: <[FrameLocator]> - returns: <[FrameLocator]>

View File

@ -634,6 +634,16 @@ Attribute name to get the value for.
* since: v1.14 * since: v1.14
## method: Locator.getByAltText
* since: v1.27
- returns: <[Locator]>
%%-template-locator-get-by-alt-text-%%
### param: Locator.getByAltText.text = %%-locator-get-by-text-text-%%
### option: Locator.getByAltText.exact = %%-locator-get-by-text-exact-%%
## method: Locator.getByLabelText ## method: Locator.getByLabelText
* since: v1.27 * since: v1.27
- returns: <[Locator]> - returns: <[Locator]>
@ -685,6 +695,16 @@ Attribute name to get the value for.
### option: Locator.getByText.exact = %%-locator-get-by-text-exact-%% ### option: Locator.getByText.exact = %%-locator-get-by-text-exact-%%
## method: Locator.getByTitle
* since: v1.27
- returns: <[Locator]>
%%-template-locator-get-by-title-%%
### param: Locator.getByTitle.text = %%-locator-get-by-text-text-%%
### option: Locator.getByTitle.exact = %%-locator-get-by-text-exact-%%
## async method: Locator.highlight ## async method: Locator.highlight
* since: v1.20 * since: v1.20

View File

@ -2183,6 +2183,16 @@ Attribute name to get the value for.
* since: v1.8 * since: v1.8
## method: Page.getByAltText
* since: v1.27
- returns: <[Locator]>
%%-template-locator-get-by-alt-text-%%
### param: Page.getByAltText.text = %%-locator-get-by-text-text-%%
### option: Page.getByAltText.exact = %%-locator-get-by-text-exact-%%
## method: Page.getByLabelText ## method: Page.getByLabelText
* since: v1.27 * since: v1.27
- returns: <[Locator]> - returns: <[Locator]>
@ -2234,6 +2244,16 @@ Attribute name to get the value for.
### option: Page.getByText.exact = %%-locator-get-by-text-exact-%% ### option: Page.getByText.exact = %%-locator-get-by-text-exact-%%
## method: Page.getByTitle
* since: v1.27
- returns: <[Locator]>
%%-template-locator-get-by-title-%%
### param: Page.getByTitle.text = %%-locator-get-by-text-text-%%
### option: Page.getByTitle.exact = %%-locator-get-by-text-exact-%%
## async method: Page.goBack ## async method: Page.goBack
* since: v1.8 * since: v1.8
- returns: <[null]|[Response]> - returns: <[null]|[Response]>

View File

@ -1180,6 +1180,14 @@ Locate element by the test id. By default, the `data-testid` attribute is used a
Allows locating elements that contain given text. Allows locating elements that contain given text.
## template-locator-get-by-alt-text
Allows locating elements by their alt text. For example, this method will find the image by alt text "Castle":
```html
<img alt='Castle'>
```
## template-locator-get-by-label-text ## template-locator-get-by-label-text
Allows locating input elements by the text of the associated label. For example, this method will find the input by label text Password in the following DOM: Allows locating input elements by the text of the associated label. For example, this method will find the input by label text Password in the following DOM:
@ -1188,6 +1196,7 @@ Allows locating input elements by the text of the associated label. For example,
<label for="password-input">Password:</label> <label for="password-input">Password:</label>
<input id="password-input"> <input id="password-input">
``` ```
## template-locator-get-by-placeholder-text ## template-locator-get-by-placeholder-text
Allows locating input elements by the placeholder text. For example, this method will find the input by placeholder "Country": Allows locating input elements by the placeholder text. For example, this method will find the input by placeholder "Country":
@ -1202,3 +1211,10 @@ Allows locating elements by their [ARIA role](https://www.w3.org/TR/wai-aria-1.2
Note that many html elements have an implicitly [defined role](https://w3c.github.io/html-aam/#html-element-role-mappings) that is recognized by the role selector. You can find all the [supported roles here](https://www.w3.org/TR/wai-aria-1.2/#role_definitions). ARIA guidelines **do not recommend** duplicating implicit roles and attributes by setting `role` and/or `aria-*` attributes to default values. Note that many html elements have an implicitly [defined role](https://w3c.github.io/html-aam/#html-element-role-mappings) that is recognized by the role selector. You can find all the [supported roles here](https://www.w3.org/TR/wai-aria-1.2/#role_definitions). ARIA guidelines **do not recommend** duplicating implicit roles and attributes by setting `role` and/or `aria-*` attributes to default values.
## template-locator-get-by-title
Allows locating elements by their title. For example, this method will find the button by its title "Submit":
```html
<button title='Place the order'>Order Now</button>
```

View File

@ -303,6 +303,10 @@ export class Frame extends ChannelOwner<channels.FrameChannel> implements api.Fr
return this.locator(Locator.getByTestIdSelector(testId)); return this.locator(Locator.getByTestIdSelector(testId));
} }
getByAltText(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.locator(Locator.getByAltTextSelector(text, options));
}
getByLabelText(text: string | RegExp, options?: { exact?: boolean }): Locator { getByLabelText(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.locator(Locator.getByLabelTextSelector(text, options)); return this.locator(Locator.getByLabelTextSelector(text, options));
} }
@ -315,6 +319,10 @@ export class Frame extends ChannelOwner<channels.FrameChannel> implements api.Fr
return this.locator(Locator.getByTextSelector(text, options)); return this.locator(Locator.getByTextSelector(text, options));
} }
getByTitle(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.locator(Locator.getByTitleSelector(text, options));
}
getByRole(role: string, options: ByRoleOptions = {}): Locator { getByRole(role: string, options: ByRoleOptions = {}): Locator {
return this.locator(Locator.getByRoleSelector(role, options)); return this.locator(Locator.getByRoleSelector(role, options));
} }

View File

@ -52,7 +52,13 @@ export class Locator implements api.Locator {
} }
static getByTestIdSelector(testId: string): string { static getByTestIdSelector(testId: string): string {
return `attr=[${Locator._testIdAttributeName}=${JSON.stringify(testId)}]`; return Locator.getByAttributeTextSelector(this._testIdAttributeName, testId, { exact: true });
}
private static getByAttributeTextSelector(attrName: string, text: string | RegExp, options?: { exact?: boolean }): string {
if (!isString(text))
return `attr=[${attrName}=${text}]`;
return `attr=[${attrName}=${JSON.stringify(text)}${options?.exact ? 's' : 'i'}]`;
} }
static getByLabelTextSelector(text: string | RegExp, options?: { exact?: boolean }): string { static getByLabelTextSelector(text: string | RegExp, options?: { exact?: boolean }): string {
@ -63,10 +69,16 @@ export class Locator implements api.Locator {
return selector + ' >> control=resolve-label'; return selector + ' >> control=resolve-label';
} }
static getByAltTextSelector(text: string | RegExp, options?: { exact?: boolean }): string {
return Locator.getByAttributeTextSelector('alt', text, options);
}
static getByTitleSelector(text: string | RegExp, options?: { exact?: boolean }): string {
return Locator.getByAttributeTextSelector('title', text, options);
}
static getByPlaceholderTextSelector(text: string | RegExp, options?: { exact?: boolean }): string { static getByPlaceholderTextSelector(text: string | RegExp, options?: { exact?: boolean }): string {
if (!isString(text)) return Locator.getByAttributeTextSelector('placeholder', text, options);
return `attr=[placeholder=${text}]`;
return `attr=[placeholder=${JSON.stringify(text)}${options?.exact ? 's' : 'i'}]`;
} }
static getByTextSelector(text: string | RegExp, options?: { exact?: boolean }): string { static getByTextSelector(text: string | RegExp, options?: { exact?: boolean }): string {
@ -199,6 +211,10 @@ export class Locator implements api.Locator {
return this.locator(Locator.getByTestIdSelector(testId)); return this.locator(Locator.getByTestIdSelector(testId));
} }
getByAltText(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.locator(Locator.getByAltTextSelector(text, options));
}
getByLabelText(text: string | RegExp, options?: { exact?: boolean }): Locator { getByLabelText(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.locator(Locator.getByLabelTextSelector(text, options)); return this.locator(Locator.getByLabelTextSelector(text, options));
} }
@ -211,6 +227,10 @@ export class Locator implements api.Locator {
return this.locator(Locator.getByTextSelector(text, options)); return this.locator(Locator.getByTextSelector(text, options));
} }
getByTitle(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.locator(Locator.getByTitleSelector(text, options));
}
getByRole(role: string, options: ByRoleOptions = {}): Locator { getByRole(role: string, options: ByRoleOptions = {}): Locator {
return this.locator(Locator.getByRoleSelector(role, options)); return this.locator(Locator.getByRoleSelector(role, options));
} }
@ -393,6 +413,10 @@ export class FrameLocator implements api.FrameLocator {
return this.locator(Locator.getByTestIdSelector(testId)); return this.locator(Locator.getByTestIdSelector(testId));
} }
getByAltText(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.locator(Locator.getByAltTextSelector(text, options));
}
getByLabelText(text: string | RegExp, options?: { exact?: boolean }): Locator { getByLabelText(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.locator(Locator.getByLabelTextSelector(text, options)); return this.locator(Locator.getByLabelTextSelector(text, options));
} }
@ -405,6 +429,10 @@ export class FrameLocator implements api.FrameLocator {
return this.locator(Locator.getByTextSelector(text, options)); return this.locator(Locator.getByTextSelector(text, options));
} }
getByTitle(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.locator(Locator.getByTitleSelector(text, options));
}
getByRole(role: string, options: ByRoleOptions = {}): Locator { getByRole(role: string, options: ByRoleOptions = {}): Locator {
return this.locator(Locator.getByRoleSelector(role, options)); return this.locator(Locator.getByRoleSelector(role, options));
} }

View File

@ -568,6 +568,10 @@ export class Page extends ChannelOwner<channels.PageChannel> implements api.Page
return this.mainFrame().getByTestId(testId); return this.mainFrame().getByTestId(testId);
} }
getByAltText(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.mainFrame().getByAltText(text, options);
}
getByLabelText(text: string | RegExp, options?: { exact?: boolean }): Locator { getByLabelText(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.mainFrame().getByLabelText(text, options); return this.mainFrame().getByLabelText(text, options);
} }
@ -580,6 +584,10 @@ export class Page extends ChannelOwner<channels.PageChannel> implements api.Page
return this.mainFrame().getByText(text, options); return this.mainFrame().getByText(text, options);
} }
getByTitle(text: string | RegExp, options?: { exact?: boolean }): Locator {
return this.mainFrame().getByTitle(text, options);
}
getByRole(role: string, options: ByRoleOptions = {}): Locator { getByRole(role: string, options: ByRoleOptions = {}): Locator {
return this.mainFrame().getByRole(role, options); return this.mainFrame().getByRole(role, options);
} }

View File

@ -2456,6 +2456,23 @@ export interface Page {
timeout?: number; timeout?: number;
}): Promise<null|string>; }): Promise<null|string>;
/**
* Allows locating elements by their alt text. For example, this method will find the image by alt text "Castle":
*
* ```html
* <img alt='Castle'>
* ```
*
* @param text Text to locate the element for.
* @param options
*/
getByAltText(text: string|RegExp, options?: {
/**
* Whether to find an exact match: case-sensitive and whole-string. Default to false.
*/
exact?: boolean;
}): Locator;
/** /**
* Allows locating input elements by the text of the associated label. For example, this method will find the input by * Allows locating input elements by the text of the associated label. For example, this method will find the input by
* label text Password in the following DOM: * label text Password in the following DOM:
@ -2588,6 +2605,23 @@ export interface Page {
exact?: boolean; exact?: boolean;
}): Locator; }): Locator;
/**
* Allows locating elements by their title. For example, this method will find the button by its title "Submit":
*
* ```html
* <button title='Place the order'>Order Now</button>
* ```
*
* @param text Text to locate the element for.
* @param options
*/
getByTitle(text: string|RegExp, options?: {
/**
* Whether to find an exact match: case-sensitive and whole-string. Default to false.
*/
exact?: boolean;
}): Locator;
/** /**
* Returns the main resource response. In case of multiple redirects, the navigation will resolve with the response of the * Returns the main resource response. In case of multiple redirects, the navigation will resolve with the response of the
* last redirect. If can not go back, returns `null`. * last redirect. If can not go back, returns `null`.
@ -5508,6 +5542,23 @@ export interface Frame {
timeout?: number; timeout?: number;
}): Promise<null|string>; }): Promise<null|string>;
/**
* Allows locating elements by their alt text. For example, this method will find the image by alt text "Castle":
*
* ```html
* <img alt='Castle'>
* ```
*
* @param text Text to locate the element for.
* @param options
*/
getByAltText(text: string|RegExp, options?: {
/**
* Whether to find an exact match: case-sensitive and whole-string. Default to false.
*/
exact?: boolean;
}): Locator;
/** /**
* Allows locating input elements by the text of the associated label. For example, this method will find the input by * Allows locating input elements by the text of the associated label. For example, this method will find the input by
* label text Password in the following DOM: * label text Password in the following DOM:
@ -5640,6 +5691,23 @@ export interface Frame {
exact?: boolean; exact?: boolean;
}): Locator; }): Locator;
/**
* Allows locating elements by their title. For example, this method will find the button by its title "Submit":
*
* ```html
* <button title='Place the order'>Order Now</button>
* ```
*
* @param text Text to locate the element for.
* @param options
*/
getByTitle(text: string|RegExp, options?: {
/**
* Whether to find an exact match: case-sensitive and whole-string. Default to false.
*/
exact?: boolean;
}): Locator;
/** /**
* Returns the main resource response. In case of multiple redirects, the navigation will resolve with the response of the * Returns the main resource response. In case of multiple redirects, the navigation will resolve with the response of the
* last redirect. * last redirect.
@ -9908,6 +9976,23 @@ export interface Locator {
timeout?: number; timeout?: number;
}): Promise<null|string>; }): Promise<null|string>;
/**
* Allows locating elements by their alt text. For example, this method will find the image by alt text "Castle":
*
* ```html
* <img alt='Castle'>
* ```
*
* @param text Text to locate the element for.
* @param options
*/
getByAltText(text: string|RegExp, options?: {
/**
* Whether to find an exact match: case-sensitive and whole-string. Default to false.
*/
exact?: boolean;
}): Locator;
/** /**
* Allows locating input elements by the text of the associated label. For example, this method will find the input by * Allows locating input elements by the text of the associated label. For example, this method will find the input by
* label text Password in the following DOM: * label text Password in the following DOM:
@ -10040,6 +10125,23 @@ export interface Locator {
exact?: boolean; exact?: boolean;
}): Locator; }): Locator;
/**
* Allows locating elements by their title. For example, this method will find the button by its title "Submit":
*
* ```html
* <button title='Place the order'>Order Now</button>
* ```
*
* @param text Text to locate the element for.
* @param options
*/
getByTitle(text: string|RegExp, options?: {
/**
* Whether to find an exact match: case-sensitive and whole-string. Default to false.
*/
exact?: boolean;
}): Locator;
/** /**
* Highlight the corresponding element(s) on the screen. Useful for debugging, don't commit the code that uses * Highlight the corresponding element(s) on the screen. Useful for debugging, don't commit the code that uses
* [locator.highlight()](https://playwright.dev/docs/api/class-locator#locator-highlight). * [locator.highlight()](https://playwright.dev/docs/api/class-locator#locator-highlight).
@ -15129,6 +15231,23 @@ export interface FrameLocator {
*/ */
frameLocator(selector: string): FrameLocator; frameLocator(selector: string): FrameLocator;
/**
* Allows locating elements by their alt text. For example, this method will find the image by alt text "Castle":
*
* ```html
* <img alt='Castle'>
* ```
*
* @param text Text to locate the element for.
* @param options
*/
getByAltText(text: string|RegExp, options?: {
/**
* Whether to find an exact match: case-sensitive and whole-string. Default to false.
*/
exact?: boolean;
}): Locator;
/** /**
* Allows locating input elements by the text of the associated label. For example, this method will find the input by * Allows locating input elements by the text of the associated label. For example, this method will find the input by
* label text Password in the following DOM: * label text Password in the following DOM:
@ -15261,6 +15380,23 @@ export interface FrameLocator {
exact?: boolean; exact?: boolean;
}): Locator; }): Locator;
/**
* Allows locating elements by their title. For example, this method will find the button by its title "Submit":
*
* ```html
* <button title='Place the order'>Order Now</button>
* ```
*
* @param text Text to locate the element for.
* @param options
*/
getByTitle(text: string|RegExp, options?: {
/**
* Whether to find an exact match: case-sensitive and whole-string. Default to false.
*/
exact?: boolean;
}): Locator;
/** /**
* Returns locator to the last matching frame. * Returns locator to the last matching frame.
*/ */

View File

@ -35,7 +35,7 @@ async function routeIframe(page: Page) {
</div> </div>
<span>1</span> <span>1</span>
<span>2</span> <span>2</span>
<label for=target>Name</label><input id=target type=text placeholder=Placeholder> <label for=target>Name</label><input id=target type=text placeholder=Placeholder title=Title alt=Alternative>
</html>`, </html>`,
contentType: 'text/html' contentType: 'text/html'
}).catch(() => {}); }).catch(() => {});
@ -252,4 +252,8 @@ it('getBy coverage', async ({ page, server }) => {
await expect(input1).toHaveValue(''); await expect(input1).toHaveValue('');
const input2 = page.frameLocator('iframe').getByPlaceholderText('Placeholder'); const input2 = page.frameLocator('iframe').getByPlaceholderText('Placeholder');
await expect(input2).toHaveValue(''); await expect(input2).toHaveValue('');
const input3 = page.frameLocator('iframe').getByAltText('Alternative');
await expect(input3).toHaveValue('');
const input4 = page.frameLocator('iframe').getByTitle('Title');
await expect(input4).toHaveValue('');
}); });

View File

@ -58,3 +58,31 @@ it('getByPlaceholderText should work', async ({ page }) => {
await expect(page.mainFrame().getByPlaceholderText('hello')).toHaveCount(2); await expect(page.mainFrame().getByPlaceholderText('hello')).toHaveCount(2);
await expect(page.locator('div').getByPlaceholderText('hello')).toHaveCount(2); await expect(page.locator('div').getByPlaceholderText('hello')).toHaveCount(2);
}); });
it('getByAltText should work', async ({ page }) => {
await page.setContent(`<div>
<input alt='Hello'>
<input alt='Hello World'>
</div>`);
await expect(page.getByAltText('hello')).toHaveCount(2);
await expect(page.getByAltText('Hello', { exact: true })).toHaveCount(1);
await expect(page.getByAltText(/wor/i)).toHaveCount(1);
// Coverage
await expect(page.mainFrame().getByAltText('hello')).toHaveCount(2);
await expect(page.locator('div').getByAltText('hello')).toHaveCount(2);
});
it('getByTitle should work', async ({ page }) => {
await page.setContent(`<div>
<input title='Hello'>
<input title='Hello World'>
</div>`);
await expect(page.getByTitle('hello')).toHaveCount(2);
await expect(page.getByTitle('Hello', { exact: true })).toHaveCount(1);
await expect(page.getByTitle(/wor/i)).toHaveCount(1);
// Coverage
await expect(page.mainFrame().getByTitle('hello')).toHaveCount(2);
await expect(page.locator('div').getByTitle('hello')).toHaveCount(2);
});