playwright/docs/src/test-components-js.md

243 lines
6.6 KiB
Markdown
Raw Normal View History

---
id: test-components
title: "Experimental: components"
---
Playwright Test can now test your components.
<div className="embed-youtube">
<iframe src="https://www.youtube.com/embed/y3YxX4sFJbM" title="YouTube video player" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture" width="750" height="563" allowfullscreen></iframe>
</div>
## Example
Here is what a typical component test looks like:
```js
test('event should work', async ({ mount }) => {
let clicked = false;
// Mount a component. Returns locator pointing to the component.
2022-05-05 07:35:01 -07:00
const component = await mount(<Button title='Submit'
onClick={() => clicked = true}>
</Button>);
// As with any Playwright test, assert locator text.
await expect(component).toContainText('Submit');
// Perform locator click. This will trigger the event.
await component.click();
// Assert that respective events have been fired.
expect(clicked).toBeTruthy();
2022-05-05 16:59:37 -08:00
});
```
## How to get started
2022-09-27 22:26:29 +02:00
Adding Playwright Test to an existing React, Vue, Svelte or Solid project is easy. Below are the steps to enable Playwright Test for a sample create-react-app with TypeScript template.
### Step 1: Install Playwright Test for components for your respective framework
<Tabs
defaultValue="npm"
values={[
{label: 'npm', value: 'npm'},
{label: 'yarn', value: 'yarn'},
]
}>
<TabItem value="npm">
```bash
2022-05-11 12:24:04 -07:00
npm init playwright@latest -- --ct
```
</TabItem>
<TabItem value="yarn">
```bash
yarn create playwright --ct
```
</TabItem>
</Tabs>
2022-05-05 16:59:37 -08:00
This step creates several files in your workspace:
#### `playwright/index.html`
This file defines an html file that will be used to render components during testing.
It must contain element with `id="root"`, that's where components are mounted. It must
also link the script called `playwright/index.[tj]s`.
```html
<html lang="en">
<body>
<div id="root"></div>
<script type="module" src="/playwright/index.ts"></script>
</body>
</html>
```
2022-05-05 16:59:37 -08:00
#### `playwright/index.ts`
You can include stylesheets, apply theme and inject code into the page where
component is mounted using this script. It can be either a `.js` or `.ts` file.
2022-05-05 16:59:37 -08:00
```js
// Apply theme here, add anything your component needs at runtime here.
```
### Step 2. Create a test file `src/App.spec.{ts,tsx}`
<Tabs
defaultValue="react"
values={[
{label: 'React', value: 'react'},
{label: 'Vue', value: 'vue'},
{label: 'Svelte', value: 'svelte'},
]
}>
<TabItem value="react">
```js
import { test, expect } from '@playwright/experimental-ct-react';
import App from './App';
test.use({ viewport: { width: 500, height: 500 } });
test('should work', async ({ mount }) => {
const component = await mount(<App></App>);
await expect(component).toContainText('Learn React');
});
```
</TabItem>
<TabItem value="vue">
```js
import { test, expect } from '@playwright/experimental-ct-vue';
import App from './App.vue';
test.use({ viewport: { width: 500, height: 500 } });
test('should work', async ({ mount }) => {
const component = await mount(<App></App>);
await expect(component).toContainText('Vite + Vue');
});
```
If using TypeScript and Vue make sure to add a `vue.d.ts` file to your project:
```ts
declare module '*.vue';
```
</TabItem>
<TabItem value="svelte">
```js
import { test, expect } from '@playwright/experimental-ct-svelte';
import App from './App.svelte';
test.use({ viewport: { width: 500, height: 500 } });
test('should work', async ({ mount }) => {
const component = await mount(App);
await expect(component).toContainText('Vite + Svelte');
});
```
</TabItem>
</Tabs>
2022-05-05 16:59:37 -08:00
### Step 3. Run the tests
You can run tests using the [VS Code extension](./getting-started-vscode.md) or the command line.
```sh
npm run test-ct
```
### Further reading: configure reporting, browsers, tracing
Refer to [Playwright config](./test-configuration.md) for configuring your project.
## Hooks
You can use `beforeMount` and `afterMount` hooks to configure your app. This lets you setup things like your app router, fake server etc. giving you the flexibility you need. You can also pass custom configuration from the `mount` call from a test, which is accessible from the `hooksConfig` fixture.
#### `playwright/index.ts`
This includes any config that needs to be run before/after mounting the component. Here's an example of how to setup `miragejs` mocking library:
```js
import { beforeMount } from '@playwright/experimental-ct-react/hooks';
import { createServer } from "miragejs"
beforeMount(async ({ hooksConfig }) => {
// Setting default values if custom config is not provided
const users = hooksConfig.users ?? [
{ id: "1", name: "Luke" },
{ id: "2", name: "Leia" },
{ id: "3", name: "Han" },
];
createServer({
routes() {
this.get("/api/users", () => users)
},
});
});
```
#### In your test file:
```js
// src/Users.spec.tsx
import { test, expect } from "@playwright/experimental-ct-react";
import React from "react";
import { Users } from "./Users";
test("should work", async ({ mount }) => {
const component = await mount(<Users />, {
hooksConfig: {
users: [
{ id: "4", name: "Anakin" },
{ id: "5", name: "Padme" },
]
}
});
await expect(component.locator("li")).toContainText([
"Anakin",
"Padme",
]);
});
```
## Under the hood
When Playwright Test is used to test web components, tests run in Node.js, while components run in the real browser. This brings together the best of both worlds: components run in the real browser environment, real clicks are triggered, real layout is executed, visual regression is possible. At the same time, test can use all the powers of Node.js as well as all the Playwright Test features. As a result, the same parallel, parametrized tests with the same post-mortem Tracing story are available during component testing.
Here is how this is achieved:
- Once the tests are executed, Playwright creates a list of components that the tests need.
- It then compiles a bundle that includes these components and serves it using a local static web server.
- Upon the `mount` call within the test, Playwright navigates to the facade page `/playwright/index.html` of this bundle and tells it to render the component.
- Events are marshalled back to the Node.js environment to allow verification.
Playwright is using [Vite](https://vitejs.dev/) to create the components bundle and serve it.
2022-05-21 11:28:03 -07:00
## Known issues and limitations
Please refer to [this issue](https://github.com/microsoft/playwright/issues/14298) for the list of known issues and limitations.
## Planned work
- Watch mode: watch mode is highly anticipated and should be relatively straightforward in implementation.