feat(ct): vue2 plugins (#18596)
This commit is contained in:
parent
76a7bd8c35
commit
f540ce08f2
11
packages/playwright-ct-vue2/hooks.d.ts
vendored
11
packages/playwright-ct-vue2/hooks.d.ts
vendored
|
|
@ -14,14 +14,19 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { CombinedVueInstance, Vue } from 'vue/types/vue';
|
import { ComponentOptions } from 'vue';
|
||||||
|
import { CombinedVueInstance, Vue, VueConstructor } from 'vue/types/vue';
|
||||||
|
|
||||||
type JsonPrimitive = string | number | boolean | null;
|
type JsonPrimitive = string | number | boolean | null;
|
||||||
type JsonValue = JsonPrimitive | JsonObject | JsonArray;
|
type JsonValue = JsonPrimitive | JsonObject | JsonArray;
|
||||||
type JsonArray = JsonValue[];
|
type JsonArray = JsonValue[];
|
||||||
type JsonObject = { [Key in string]?: JsonValue };
|
type JsonObject = { [Key in string]?: JsonValue };
|
||||||
|
|
||||||
export declare function beforeMount<HooksConfig extends JsonObject>(
|
export declare function beforeMount<HooksConfig extends JsonObject>(
|
||||||
callback: (params: { hooksConfig: HooksConfig }) => Promise<void>
|
callback: (params: {
|
||||||
|
hooksConfig: HooksConfig,
|
||||||
|
Vue: VueConstructor<Vue>,
|
||||||
|
}) => Promise<void | ComponentOptions<Vue> & Record<string, unknown>>
|
||||||
): void;
|
): void;
|
||||||
export declare function afterMount<HooksConfig extends JsonObject>(
|
export declare function afterMount<HooksConfig extends JsonObject>(
|
||||||
callback: (params: {
|
callback: (params: {
|
||||||
|
|
|
||||||
15
packages/playwright-ct-vue2/index.d.ts
vendored
15
packages/playwright-ct-vue2/index.d.ts
vendored
|
|
@ -48,17 +48,14 @@ export interface MountOptions<
|
||||||
props?: Props;
|
props?: Props;
|
||||||
slots?: Record<string, Slot> & { default?: Slot };
|
slots?: Record<string, Slot> & { default?: Slot };
|
||||||
on?: Record<string, Function>;
|
on?: Record<string, Function>;
|
||||||
hooksConfig?: JsonObject;
|
hooksConfig?: HooksConfig;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface MountResult<
|
interface MountResult<
|
||||||
HooksConfig extends JsonObject,
|
|
||||||
Props extends Record<string, unknown>
|
Props extends Record<string, unknown>
|
||||||
> extends Locator {
|
> extends Locator {
|
||||||
unmount(): Promise<void>;
|
unmount(): Promise<void>;
|
||||||
update(
|
update(options: Omit<MountOptions<never, Props>, 'hooksConfig'>): Promise<void>;
|
||||||
options: Omit<MountOptions<HooksConfig, Props>, 'hooksConfig'>
|
|
||||||
): Promise<void>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
interface MountResultJsx extends Locator {
|
interface MountResultJsx extends Locator {
|
||||||
|
|
@ -70,15 +67,15 @@ export interface ComponentFixtures {
|
||||||
mount(component: JSX.Element): Promise<MountResultJsx>;
|
mount(component: JSX.Element): Promise<MountResultJsx>;
|
||||||
mount<HooksConfig extends JsonObject>(
|
mount<HooksConfig extends JsonObject>(
|
||||||
component: any,
|
component: any,
|
||||||
options?: MountOptions<HooksConfig, any>
|
options?: MountOptions<HooksConfig, Record<string, unknown>>
|
||||||
): Promise<MountResult<HooksConfig, any>>;
|
): Promise<MountResult<Record<string, unknown>>>;
|
||||||
mount<
|
mount<
|
||||||
HooksConfig extends JsonObject,
|
HooksConfig extends JsonObject,
|
||||||
Props extends Record<string, unknown> = Record<string, unknown>
|
Props extends Record<string, unknown> = Record<string, unknown>
|
||||||
>(
|
>(
|
||||||
component: any,
|
component: any,
|
||||||
options: MountOptions<HooksConfig, any> & { props: Props }
|
options: MountOptions<HooksConfig, never> & { props: Props }
|
||||||
): Promise<MountResult<HooksConfig, Props>>;
|
): Promise<MountResult<Props>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const test: TestType<
|
export const test: TestType<
|
||||||
|
|
|
||||||
|
|
@ -160,10 +160,12 @@ const instanceKey = Symbol('instanceKey');
|
||||||
const wrapperKey = Symbol('wrapperKey');
|
const wrapperKey = Symbol('wrapperKey');
|
||||||
|
|
||||||
window.playwrightMount = async (component, rootElement, hooksConfig) => {
|
window.playwrightMount = async (component, rootElement, hooksConfig) => {
|
||||||
|
let options = {};
|
||||||
for (const hook of /** @type {any} */(window).__pw_hooks_before_mount || [])
|
for (const hook of /** @type {any} */(window).__pw_hooks_before_mount || [])
|
||||||
await hook({ hooksConfig });
|
options = await hook({ hooksConfig, Vue });
|
||||||
|
|
||||||
const instance = new Vue({
|
const instance = new Vue({
|
||||||
|
...options,
|
||||||
render: h => {
|
render: h => {
|
||||||
const wrapper = createWrapper(component, h);
|
const wrapper = createWrapper(component, h);
|
||||||
/** @type {any} */ (rootElement)[wrapperKey] = wrapper;
|
/** @type {any} */ (rootElement)[wrapperKey] = wrapper;
|
||||||
|
|
|
||||||
|
|
@ -22,7 +22,6 @@ const config: PlaywrightTestConfig = {
|
||||||
retries: process.env.CI ? 2 : 0,
|
retries: process.env.CI ? 2 : 0,
|
||||||
reporter: 'html',
|
reporter: 'html',
|
||||||
use: {
|
use: {
|
||||||
ctTemplateDir: 'playwright',
|
|
||||||
trace: 'on-first-retry'
|
trace: 'on-first-retry'
|
||||||
},
|
},
|
||||||
projects: [
|
projects: [
|
||||||
|
|
|
||||||
|
|
@ -10,7 +10,8 @@
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"core-js": "^3.8.3",
|
"core-js": "^3.8.3",
|
||||||
"vue": "^2.7.13"
|
"vue": "^2.7.13",
|
||||||
|
"vue-router": "^3.6.5"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "^7.12.16",
|
"@babel/core": "^7.12.16",
|
||||||
|
|
|
||||||
|
|
@ -1,14 +1,18 @@
|
||||||
import '../src/assets/index.css';
|
|
||||||
import { beforeMount, afterMount } from '@playwright/experimental-ct-vue2/hooks';
|
import { beforeMount, afterMount } from '@playwright/experimental-ct-vue2/hooks';
|
||||||
|
import Router from 'vue-router';
|
||||||
|
import { router } from '../src/router';
|
||||||
|
import '../src/assets/index.css';
|
||||||
|
|
||||||
export type hooksConfig = {
|
export type HooksConfig = {
|
||||||
route: string;
|
route: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
beforeMount(async ({ hooksConfig }) => {
|
beforeMount<HooksConfig>(async ({ Vue, hooksConfig }) => {
|
||||||
console.log(`Before mount: ${JSON.stringify(hooksConfig)}`);
|
console.log(`Before mount: ${JSON.stringify(hooksConfig)}`);
|
||||||
|
Vue.use(Router as any); // TODO: remove any and fix the various installed conflicting Vue versions
|
||||||
|
return { router }
|
||||||
});
|
});
|
||||||
|
|
||||||
afterMount(async ({ instance }) => {
|
afterMount<HooksConfig>(async ({ instance }) => {
|
||||||
console.log(`After mount el: ${instance.$el.constructor.name}`);
|
console.log(`After mount el: ${instance.$el.constructor.name}`);
|
||||||
});
|
});
|
||||||
|
|
|
||||||
|
|
@ -1,28 +1,10 @@
|
||||||
<template>
|
<template>
|
||||||
<div id="app">
|
<div id="app">
|
||||||
<img alt="Vue logo" src="./assets/logo.png">
|
<header>
|
||||||
<HelloWorld msg="Welcome to Your Vue.js App"/>
|
<img alt="Vue logo" src="./assets/logo.png" width="60" height="60">
|
||||||
|
<router-link to="/">Login</router-link>
|
||||||
|
<router-link to="/dashboard">Dashboard</router-link>
|
||||||
|
</header>
|
||||||
|
<router-view />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
|
||||||
import HelloWorld from './components/HelloWorld.vue'
|
|
||||||
|
|
||||||
export default {
|
|
||||||
name: 'App',
|
|
||||||
components: {
|
|
||||||
HelloWorld
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
#app {
|
|
||||||
font-family: Avenir, Helvetica, Arial, sans-serif;
|
|
||||||
-webkit-font-smoothing: antialiased;
|
|
||||||
-moz-osx-font-smoothing: grayscale;
|
|
||||||
text-align: center;
|
|
||||||
color: #2c3e50;
|
|
||||||
margin-top: 60px;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|
@ -1,58 +0,0 @@
|
||||||
<template>
|
|
||||||
<div class="hello">
|
|
||||||
<h1>{{ msg }}</h1>
|
|
||||||
<p>
|
|
||||||
For a guide and recipes on how to configure / customize this project,<br>
|
|
||||||
check out the
|
|
||||||
<a href="https://cli.vuejs.org" target="_blank" rel="noopener">vue-cli documentation</a>.
|
|
||||||
</p>
|
|
||||||
<h3>Installed CLI Plugins</h3>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://github.com/vuejs/vue-cli/tree/dev/packages/%40vue/cli-plugin-babel" target="_blank" rel="noopener">babel</a></li>
|
|
||||||
<li><a href="https://github.com/vuejs/vue-cli/tree/dev/packages/%40vue/cli-plugin-eslint" target="_blank" rel="noopener">eslint</a></li>
|
|
||||||
</ul>
|
|
||||||
<h3>Essential Links</h3>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://vuejs.org" target="_blank" rel="noopener">Core Docs</a></li>
|
|
||||||
<li><a href="https://forum.vuejs.org" target="_blank" rel="noopener">Forum</a></li>
|
|
||||||
<li><a href="https://chat.vuejs.org" target="_blank" rel="noopener">Community Chat</a></li>
|
|
||||||
<li><a href="https://twitter.com/vuejs" target="_blank" rel="noopener">Twitter</a></li>
|
|
||||||
<li><a href="https://news.vuejs.org" target="_blank" rel="noopener">News</a></li>
|
|
||||||
</ul>
|
|
||||||
<h3>Ecosystem</h3>
|
|
||||||
<ul>
|
|
||||||
<li><a href="https://router.vuejs.org" target="_blank" rel="noopener">vue-router</a></li>
|
|
||||||
<li><a href="https://vuex.vuejs.org" target="_blank" rel="noopener">vuex</a></li>
|
|
||||||
<li><a href="https://github.com/vuejs/vue-devtools#vue-devtools" target="_blank" rel="noopener">vue-devtools</a></li>
|
|
||||||
<li><a href="https://vue-loader.vuejs.org" target="_blank" rel="noopener">vue-loader</a></li>
|
|
||||||
<li><a href="https://github.com/vuejs/awesome-vue" target="_blank" rel="noopener">awesome-vue</a></li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script lang="ts">
|
|
||||||
export default {
|
|
||||||
name: 'HelloWorld',
|
|
||||||
props: {
|
|
||||||
msg: String
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<!-- Add "scoped" attribute to limit CSS to this component only -->
|
|
||||||
<style scoped>
|
|
||||||
h3 {
|
|
||||||
margin: 40px 0 0;
|
|
||||||
}
|
|
||||||
ul {
|
|
||||||
list-style-type: none;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
li {
|
|
||||||
display: inline-block;
|
|
||||||
margin: 0 10px;
|
|
||||||
}
|
|
||||||
a {
|
|
||||||
color: #42b983;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|
@ -1,9 +1,14 @@
|
||||||
import Vue from 'vue'
|
import Vue from 'vue';
|
||||||
import App from './App.vue'
|
import Router from 'vue-router';
|
||||||
|
import App from './App.vue';
|
||||||
|
import { router } from './router';
|
||||||
import './assets/index.css';
|
import './assets/index.css';
|
||||||
|
|
||||||
Vue.config.productionTip = false
|
Vue.config.productionTip = false;
|
||||||
|
|
||||||
|
Vue.use(Router);
|
||||||
|
|
||||||
new Vue({
|
new Vue({
|
||||||
|
router,
|
||||||
render: h => h(App),
|
render: h => h(App),
|
||||||
}).$mount('#app')
|
}).$mount('#app');
|
||||||
|
|
|
||||||
|
|
@ -1,10 +1,11 @@
|
||||||
import { test, expect } from '@playwright/experimental-ct-vue2'
|
import { test, expect } from '@playwright/experimental-ct-vue2'
|
||||||
import Button from './components/Button.vue'
|
import App from './App.vue';
|
||||||
import Counter from './components/Counter.vue'
|
import Button from './components/Button.vue';
|
||||||
import DefaultSlot from './components/DefaultSlot.vue'
|
import Counter from './components/Counter.vue';
|
||||||
import NamedSlots from './components/NamedSlots.vue'
|
import DefaultSlot from './components/DefaultSlot.vue';
|
||||||
import EmptyTemplate from './components/EmptyTemplate.vue'
|
import NamedSlots from './components/NamedSlots.vue';
|
||||||
import type { hooksConfig } from '../playwright'
|
import EmptyTemplate from './components/EmptyTemplate.vue';
|
||||||
|
import type { HooksConfig } from '../playwright';
|
||||||
|
|
||||||
test.use({ viewport: { width: 500, height: 500 } })
|
test.use({ viewport: { width: 500, height: 500 } })
|
||||||
|
|
||||||
|
|
@ -121,7 +122,7 @@ test('emit a event when a slot is clicked', async ({ mount }) => {
|
||||||
test('run hooks', async ({ page, mount }) => {
|
test('run hooks', async ({ page, mount }) => {
|
||||||
const messages: string[] = []
|
const messages: string[] = []
|
||||||
page.on('console', m => messages.push(m.text()))
|
page.on('console', m => messages.push(m.text()))
|
||||||
await mount<hooksConfig>(<Button title="Submit" />, {
|
await mount<HooksConfig>(<Button title="Submit" />, {
|
||||||
hooksConfig: { route: 'A' }
|
hooksConfig: { route: 'A' }
|
||||||
})
|
})
|
||||||
expect(messages).toEqual(['Before mount: {\"route\":\"A\"}', 'After mount el: HTMLButtonElement'])
|
expect(messages).toEqual(['Before mount: {\"route\":\"A\"}', 'After mount el: HTMLButtonElement'])
|
||||||
|
|
@ -140,3 +141,12 @@ test('get textContent of the empty template', async ({ mount }) => {
|
||||||
expect(await component.textContent()).toBe('');
|
expect(await component.textContent()).toBe('');
|
||||||
await expect(component).toHaveText('');
|
await expect(component).toHaveText('');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('navigate to a page by clicking a link', async ({ page, mount }) => {
|
||||||
|
const component = await mount(<App />);
|
||||||
|
await expect(component.getByRole('main')).toHaveText('Login');
|
||||||
|
await expect(page).toHaveURL('/');
|
||||||
|
await component.getByRole('link', { name: 'Dashboard' }).click();
|
||||||
|
await expect(component.getByRole('main')).toHaveText('Dashboard');
|
||||||
|
await expect(page).toHaveURL('/dashboard');
|
||||||
|
});
|
||||||
|
|
|
||||||
|
|
@ -1,15 +1,16 @@
|
||||||
import { test, expect } from '@playwright/experimental-ct-vue2'
|
import { test, expect } from '@playwright/experimental-ct-vue2';
|
||||||
import Button from './components/Button.vue'
|
import App from './App.vue';
|
||||||
import Counter from './components/Counter.vue'
|
import Button from './components/Button.vue';
|
||||||
import DefaultSlot from './components/DefaultSlot.vue'
|
import Counter from './components/Counter.vue';
|
||||||
import NamedSlots from './components/NamedSlots.vue'
|
import DefaultSlot from './components/DefaultSlot.vue';
|
||||||
import Component from './components/Component.vue'
|
import NamedSlots from './components/NamedSlots.vue';
|
||||||
import EmptyTemplate from './components/EmptyTemplate.vue'
|
import Component from './components/Component.vue';
|
||||||
import type { hooksConfig } from '../playwright'
|
import EmptyTemplate from './components/EmptyTemplate.vue';
|
||||||
|
import type { HooksConfig } from '../playwright';
|
||||||
|
|
||||||
test.use({ viewport: { width: 500, height: 500 } })
|
test.use({ viewport: { width: 500, height: 500 } })
|
||||||
|
|
||||||
test('render props', async ({ mount }) => {
|
test('render props', async ({ page, mount }) => {
|
||||||
const component = await mount(Button, {
|
const component = await mount(Button, {
|
||||||
props: {
|
props: {
|
||||||
title: 'Submit'
|
title: 'Submit'
|
||||||
|
|
@ -123,7 +124,7 @@ test('render a component without options', async ({ mount }) => {
|
||||||
test('run hooks', async ({ page, mount }) => {
|
test('run hooks', async ({ page, mount }) => {
|
||||||
const messages: string[] = []
|
const messages: string[] = []
|
||||||
page.on('console', m => messages.push(m.text()))
|
page.on('console', m => messages.push(m.text()))
|
||||||
await mount<hooksConfig>(Button, {
|
await mount<HooksConfig>(Button, {
|
||||||
props: {
|
props: {
|
||||||
title: 'Submit'
|
title: 'Submit'
|
||||||
},
|
},
|
||||||
|
|
@ -149,3 +150,12 @@ test('get textContent of the empty template', async ({ mount }) => {
|
||||||
expect(await component.textContent()).toBe('');
|
expect(await component.textContent()).toBe('');
|
||||||
await expect(component).toHaveText('');
|
await expect(component).toHaveText('');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('navigate to a page by clicking a link', async ({ page, mount }) => {
|
||||||
|
const component = await mount(App);
|
||||||
|
await expect(component.getByRole('main')).toHaveText('Login');
|
||||||
|
await expect(page).toHaveURL('/');
|
||||||
|
await component.getByRole('link', { name: 'Dashboard' }).click();
|
||||||
|
await expect(component.getByRole('main')).toHaveText('Dashboard');
|
||||||
|
await expect(page).toHaveURL('/dashboard');
|
||||||
|
});
|
||||||
|
|
|
||||||
3
tests/components/ct-vue2-cli/src/pages/DashboardPage.vue
Normal file
3
tests/components/ct-vue2-cli/src/pages/DashboardPage.vue
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
<template>
|
||||||
|
<main>Dashboard</main>
|
||||||
|
</template>
|
||||||
3
tests/components/ct-vue2-cli/src/pages/LoginPage.vue
Normal file
3
tests/components/ct-vue2-cli/src/pages/LoginPage.vue
Normal file
|
|
@ -0,0 +1,3 @@
|
||||||
|
<template>
|
||||||
|
<main>Login</main>
|
||||||
|
</template>
|
||||||
12
tests/components/ct-vue2-cli/src/router/index.js
Normal file
12
tests/components/ct-vue2-cli/src/router/index.js
Normal file
|
|
@ -0,0 +1,12 @@
|
||||||
|
import Router from 'vue-router';
|
||||||
|
import LoginPage from '../pages/LoginPage.vue';
|
||||||
|
import DashboardPage from '../pages/DashboardPage.vue';
|
||||||
|
|
||||||
|
export const router = new Router({
|
||||||
|
mode: 'history',
|
||||||
|
base: '/',
|
||||||
|
routes: [
|
||||||
|
{ path: '/', component: LoginPage },
|
||||||
|
{ path: '/dashboard', component: DashboardPage }
|
||||||
|
]
|
||||||
|
});
|
||||||
|
|
@ -1,8 +1,8 @@
|
||||||
{
|
{
|
||||||
"extends": "./tsconfig.app.json",
|
"extends": "./tsconfig.app.json",
|
||||||
"include": ["playwright"],
|
"include": ["playwright", "src/**/*"],
|
||||||
"exclude": [],
|
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
|
"allowJs": true,
|
||||||
"composite": true,
|
"composite": true,
|
||||||
"lib": [],
|
"lib": [],
|
||||||
"types": ["node"]
|
"types": ["node"]
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue