Add component hydration in .md E2E tests, refactor (#3610)
Co-authored-by: Tony Sullivan <tony.f.sullivan@outlook.com> Co-authored-by: Nate Moore <natemoo-re@users.noreply.github.com>
This commit is contained in:
parent
56937563c3
commit
6ab749be5c
19 changed files with 399 additions and 549 deletions
5
.changeset/hungry-ravens-cheat.md
Normal file
5
.changeset/hungry-ravens-cheat.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
'astro': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Add component hydration in Markdown E2E tests
|
|
@ -1,5 +1,5 @@
|
||||||
import { h } from 'preact';
|
import { h } from 'preact';
|
||||||
|
|
||||||
export default function({ id }) {
|
export default function({ id }) {
|
||||||
return <div id={id}>Preact client:only component</div>
|
return <div id={id}>Framework client:only component</div>
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,31 @@
|
||||||
|
---
|
||||||
|
setup: |
|
||||||
|
import Counter from '../components/Counter.jsx';
|
||||||
|
import PreactComponent from '../components/JSXComponent.jsx';
|
||||||
|
|
||||||
|
const someProps = {
|
||||||
|
count: 0,
|
||||||
|
};
|
||||||
|
---
|
||||||
|
|
||||||
|
<Counter id="server-only" {...someProps}>
|
||||||
|
# Hello, server!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-idle" {...someProps} client:idle>
|
||||||
|
# Hello, client:idle!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-load" {...someProps} client:load>
|
||||||
|
# Hello, client:load!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-visible" {...someProps} client:visible>
|
||||||
|
# Hello, client:visible!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-media" {...someProps} client:media="(max-width: 50em)">
|
||||||
|
# Hello, client:media!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<PreactComponent id="client-only" client:only="preact" />
|
|
@ -1,5 +1,5 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
export default function({ id }) {
|
export default function({ id }) {
|
||||||
return <div id={id}>React client:only component</div>
|
return <div id={id}>Framework client:only component</div>
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,31 @@
|
||||||
|
---
|
||||||
|
setup: |
|
||||||
|
import Counter from '../components/Counter.jsx';
|
||||||
|
import ReactComponent from '../components/JSXComponent.jsx';
|
||||||
|
|
||||||
|
const someProps = {
|
||||||
|
count: 0,
|
||||||
|
};
|
||||||
|
---
|
||||||
|
|
||||||
|
<Counter id="server-only" {...someProps}>
|
||||||
|
# Hello, server!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-idle" {...someProps} client:idle>
|
||||||
|
# Hello, client:idle!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-load" {...someProps} client:load>
|
||||||
|
# Hello, client:load!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-visible" {...someProps} client:visible>
|
||||||
|
# Hello, client:visible!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-media" {...someProps} client:media="(max-width: 50em)">
|
||||||
|
# Hello, client:media!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<ReactComponent id="client-only" client:only="react" />
|
|
@ -0,0 +1,5 @@
|
||||||
|
export default function SolidComponent({ id }) {
|
||||||
|
return (
|
||||||
|
<div id={id}>Framework client:only component</div>
|
||||||
|
);
|
||||||
|
}
|
|
@ -1,5 +1,6 @@
|
||||||
---
|
---
|
||||||
import Counter from '../components/Counter.jsx';
|
import Counter from '../components/Counter.jsx';
|
||||||
|
import SolidComponent from '../components/SolidComponent.jsx';
|
||||||
|
|
||||||
const someProps = {
|
const someProps = {
|
||||||
count: 0,
|
count: 0,
|
||||||
|
@ -30,5 +31,7 @@ const someProps = {
|
||||||
<Counter id="client-media" {...someProps} client:media="(max-width: 50em)">
|
<Counter id="client-media" {...someProps} client:media="(max-width: 50em)">
|
||||||
<h1>Hello, client:media!</h1>
|
<h1>Hello, client:media!</h1>
|
||||||
</Counter>
|
</Counter>
|
||||||
|
|
||||||
|
<SolidComponent id="client-only" client:only="solid" />
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|
|
@ -0,0 +1,31 @@
|
||||||
|
---
|
||||||
|
setup: |
|
||||||
|
import Counter from '../components/Counter.jsx';
|
||||||
|
import SolidComponent from '../components/SolidComponent.jsx';
|
||||||
|
|
||||||
|
const someProps = {
|
||||||
|
count: 0,
|
||||||
|
};
|
||||||
|
---
|
||||||
|
|
||||||
|
<Counter id="server-only" {...someProps}>
|
||||||
|
# Hello, server!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-idle" {...someProps} client:idle>
|
||||||
|
# Hello, client:idle!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-load" {...someProps} client:load>
|
||||||
|
# Hello, client:load!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-visible" {...someProps} client:visible>
|
||||||
|
# Hello, client:visible!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-media" {...someProps} client:media="(max-width: 50em)">
|
||||||
|
# Hello, client:media!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<SolidComponent id="client-only" client:only="solid" />
|
|
@ -1,7 +1,6 @@
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
export let id: string;
|
export let id: string;
|
||||||
|
export let count: number = 0;
|
||||||
let count = 0;
|
|
||||||
|
|
||||||
function add() {
|
function add() {
|
||||||
count += 1;
|
count += 1;
|
||||||
|
|
|
@ -0,0 +1,5 @@
|
||||||
|
<script lang="ts">
|
||||||
|
export let id: string;
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div {id}>Framework client:only component</div>
|
|
@ -1,5 +1,6 @@
|
||||||
---
|
---
|
||||||
import Counter from '../components/Counter.svelte';
|
import Counter from '../components/Counter.svelte';
|
||||||
|
import SvelteComponent from '../components/SvelteComponent.svelte';
|
||||||
|
|
||||||
const someProps = {
|
const someProps = {
|
||||||
count: 0,
|
count: 0,
|
||||||
|
@ -11,24 +12,26 @@ const someProps = {
|
||||||
<!-- Head Stuff -->
|
<!-- Head Stuff -->
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<Counter id="server-only">
|
<Counter id="server-only" {...someProps}>
|
||||||
<h1>Hello, server!</h1>
|
<h1>Hello, server!</h1>
|
||||||
</Counter>
|
</Counter>
|
||||||
|
|
||||||
<Counter id="client-idle" client:idle>
|
<Counter id="client-idle" {...someProps} client:idle>
|
||||||
<h1>Hello, client:idle!</h1>
|
<h1>Hello, client:idle!</h1>
|
||||||
</Counter>
|
</Counter>
|
||||||
|
|
||||||
<Counter id="client-load" client:load>
|
<Counter id="client-load" {...someProps} client:load>
|
||||||
<h1>Hello, client:load!</h1>
|
<h1>Hello, client:load!</h1>
|
||||||
</Counter>
|
</Counter>
|
||||||
|
|
||||||
<Counter id="client-visible" client:visible>
|
<Counter id="client-visible" {...someProps} client:visible>
|
||||||
<h1>Hello, client:visible!</h1>
|
<h1>Hello, client:visible!</h1>
|
||||||
</Counter>
|
</Counter>
|
||||||
|
|
||||||
<Counter id="client-media" client:media="(max-width: 50rem)">
|
<Counter id="client-media" {...someProps} client:media="(max-width: 50em)">
|
||||||
<h1>Hello, client:media!</h1>
|
<h1>Hello, client:media!</h1>
|
||||||
</Counter>
|
</Counter>
|
||||||
|
|
||||||
|
<SvelteComponent id="client-only" client:only="svelte" />
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|
|
@ -0,0 +1,31 @@
|
||||||
|
---
|
||||||
|
setup: |
|
||||||
|
import Counter from '../components/Counter.svelte';
|
||||||
|
import SvelteComponent from '../components/SvelteComponent.svelte';
|
||||||
|
|
||||||
|
const someProps = {
|
||||||
|
count: 0,
|
||||||
|
};
|
||||||
|
---
|
||||||
|
|
||||||
|
<Counter id="server-only" {...someProps}>
|
||||||
|
# Hello, server!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-idle" {...someProps} client:idle>
|
||||||
|
# Hello, client:idle!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-load" {...someProps} client:load>
|
||||||
|
# Hello, client:load!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-visible" {...someProps} client:visible>
|
||||||
|
# Hello, client:visible!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<Counter id="client-media" {...someProps} client:media="(max-width: 50em)">
|
||||||
|
# Hello, client:media!
|
||||||
|
</Counter>
|
||||||
|
|
||||||
|
<SvelteComponent id="client-only" client:only="svelte" />
|
|
@ -1,145 +1,19 @@
|
||||||
import { test as base, expect } from '@playwright/test';
|
import { prepareTestFactory } from './shared-component-tests.js';
|
||||||
import { loadFixture } from './test-utils.js';
|
|
||||||
|
|
||||||
const test = base.extend({
|
const { test, createTests } = prepareTestFactory({ root: './fixtures/preact-component/' });
|
||||||
astro: async ({}, use) => {
|
|
||||||
const fixture = await loadFixture({ root: './fixtures/preact-component/' });
|
|
||||||
await use(fixture);
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
let devServer;
|
test.describe('Preact components in Astro files', () => {
|
||||||
|
createTests({
|
||||||
test.beforeEach(async ({ astro }) => {
|
pageUrl: '/',
|
||||||
devServer = await astro.startDevServer();
|
pageSourceFilePath: './src/pages/index.astro',
|
||||||
});
|
componentFilePath: './src/components/JSXComponent.jsx',
|
||||||
|
});
|
||||||
test.afterEach(async () => {
|
});
|
||||||
await devServer.stop();
|
|
||||||
});
|
test.describe('Preact components in Markdown files', () => {
|
||||||
|
createTests({
|
||||||
test.describe('Preact components', () => {
|
pageUrl: '/markdown/',
|
||||||
test('server only', async ({ page, astro }) => {
|
pageSourceFilePath: './src/pages/markdown.md',
|
||||||
await page.goto(astro.resolveUrl('/'));
|
componentFilePath: './src/components/JSXComponent.jsx',
|
||||||
|
|
||||||
const counter = page.locator('#server-only');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('0');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:idle', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-idle');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:load', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-load');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:visible', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
// Make sure the component is on screen to trigger hydration
|
|
||||||
const counter = page.locator('#client-visible');
|
|
||||||
await counter.scrollIntoViewIfNeeded();
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:media', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-media');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
await expect(count, 'component not hydrated yet').toHaveText('0');
|
|
||||||
|
|
||||||
// Reset the viewport to hydrate the component (max-width: 50rem)
|
|
||||||
await page.setViewportSize({ width: 414, height: 1124 });
|
|
||||||
await inc.click();
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:only', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const label = page.locator('#client-only');
|
|
||||||
await expect(label, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
await expect(label, 'slot text is visible').toHaveText('Preact client:only component');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('HMR', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-idle');
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
// Edit the component's initial count prop
|
|
||||||
await astro.editFile('./src/pages/index.astro', (original) =>
|
|
||||||
original.replace('id="client-idle" {...someProps}', 'id="client-idle" count={5}')
|
|
||||||
);
|
|
||||||
|
|
||||||
await expect(count, 'count prop updated').toHaveText('5');
|
|
||||||
await expect(counter, 'component styles persisted').toHaveCSS('display', 'grid');
|
|
||||||
|
|
||||||
// Edit the component's slot text
|
|
||||||
await astro.editFile('./src/components/JSXComponent.jsx', (original) =>
|
|
||||||
original.replace('Preact client:only component', 'Updated preact client:only component')
|
|
||||||
);
|
|
||||||
|
|
||||||
const label = page.locator('#client-only');
|
|
||||||
await expect(label, 'client:only component is visible').toBeVisible();
|
|
||||||
await expect(label, 'client:only slot text is visible').toHaveText(
|
|
||||||
'Updated preact client:only component'
|
|
||||||
);
|
|
||||||
|
|
||||||
// Edit the imported CSS file
|
|
||||||
await astro.editFile('./src/components/Counter.css', (original) =>
|
|
||||||
original.replace('font-size: 2em;', 'font-size: 24px;')
|
|
||||||
);
|
|
||||||
|
|
||||||
await expect(count, 'imported styles updated').toHaveCSS('font-size', '24px');
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,145 +1,19 @@
|
||||||
import { test as base, expect } from '@playwright/test';
|
import { prepareTestFactory } from './shared-component-tests.js';
|
||||||
import { loadFixture } from './test-utils.js';
|
|
||||||
|
|
||||||
const test = base.extend({
|
const { test, createTests } = prepareTestFactory({ root: './fixtures/react-component/' });
|
||||||
astro: async ({}, use) => {
|
|
||||||
const fixture = await loadFixture({ root: './fixtures/react-component/' });
|
|
||||||
await use(fixture);
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
let devServer;
|
test.describe('React components in Astro files', () => {
|
||||||
|
createTests({
|
||||||
test.beforeEach(async ({ astro }) => {
|
pageUrl: '/',
|
||||||
devServer = await astro.startDevServer();
|
pageSourceFilePath: './src/pages/index.astro',
|
||||||
});
|
componentFilePath: './src/components/JSXComponent.jsx',
|
||||||
|
});
|
||||||
test.afterEach(async () => {
|
});
|
||||||
await devServer.stop();
|
|
||||||
});
|
test.describe('React components in Markdown files', () => {
|
||||||
|
createTests({
|
||||||
test.describe('React components', () => {
|
pageUrl: '/markdown/',
|
||||||
test('server only', async ({ page, astro }) => {
|
pageSourceFilePath: './src/pages/markdown.md',
|
||||||
await page.goto(astro.resolveUrl('/'));
|
componentFilePath: './src/components/JSXComponent.jsx',
|
||||||
|
|
||||||
const counter = page.locator('#server-only');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'component not hydrated').toHaveText('0');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:idle', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-idle');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:load', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-load');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:visible', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
// Make sure the component is on screen to trigger hydration
|
|
||||||
const counter = page.locator('#client-visible');
|
|
||||||
await counter.scrollIntoViewIfNeeded();
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:media', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-media');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
await expect(count, 'component not hydrated yet').toHaveText('0');
|
|
||||||
|
|
||||||
// Reset the viewport to hydrate the component (max-width: 50rem)
|
|
||||||
await page.setViewportSize({ width: 414, height: 1124 });
|
|
||||||
await inc.click();
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:only', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const label = page.locator('#client-only');
|
|
||||||
await expect(label, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
await expect(label, 'slot text is visible').toHaveText('React client:only component');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('HMR', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-idle');
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
// Edit the component's initial count prop
|
|
||||||
await astro.editFile('./src/pages/index.astro', (original) =>
|
|
||||||
original.replace('id="client-idle" {...someProps}', 'id="client-idle" count={5}')
|
|
||||||
);
|
|
||||||
|
|
||||||
await expect(count, 'count prop updated').toHaveText('5');
|
|
||||||
await expect(counter, 'component styles persisted').toHaveCSS('display', 'grid');
|
|
||||||
|
|
||||||
// Edit the component's slot text
|
|
||||||
await astro.editFile('./src/components/JSXComponent.jsx', (original) =>
|
|
||||||
original.replace('React client:only component', 'Updated react client:only component')
|
|
||||||
);
|
|
||||||
|
|
||||||
const label = page.locator('#client-only');
|
|
||||||
await expect(label, 'client:only component is visible').toBeVisible();
|
|
||||||
await expect(label, 'client:only slot text is visible').toHaveText(
|
|
||||||
'Updated react client:only component'
|
|
||||||
);
|
|
||||||
|
|
||||||
// Edit the imported CSS file
|
|
||||||
await astro.editFile('./src/components/Counter.css', (original) =>
|
|
||||||
original.replace('font-size: 2em;', 'font-size: 24px;')
|
|
||||||
);
|
|
||||||
|
|
||||||
await expect(count, 'imported CSS updated').toHaveCSS('font-size', '24px');
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
160
packages/astro/e2e/shared-component-tests.js
Normal file
160
packages/astro/e2e/shared-component-tests.js
Normal file
|
@ -0,0 +1,160 @@
|
||||||
|
import { test as base, expect } from '@playwright/test';
|
||||||
|
import { loadFixture } from './test-utils.js';
|
||||||
|
|
||||||
|
export function prepareTestFactory({ root }) {
|
||||||
|
const test = base.extend({
|
||||||
|
astro: async ({}, use) => {
|
||||||
|
const fixture = await loadFixture({ root });
|
||||||
|
await use(fixture);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
let devServer;
|
||||||
|
|
||||||
|
test.beforeEach(async ({ astro }) => {
|
||||||
|
devServer = await astro.startDevServer();
|
||||||
|
});
|
||||||
|
|
||||||
|
test.afterEach(async () => {
|
||||||
|
await devServer.stop();
|
||||||
|
});
|
||||||
|
|
||||||
|
const createTests = ({
|
||||||
|
pageUrl,
|
||||||
|
pageSourceFilePath,
|
||||||
|
componentFilePath,
|
||||||
|
counterCssFilePath,
|
||||||
|
}) => {
|
||||||
|
test('server only', async ({ page, astro }) => {
|
||||||
|
await page.goto(astro.resolveUrl(pageUrl));
|
||||||
|
|
||||||
|
const counter = page.locator('#server-only');
|
||||||
|
await expect(counter, 'component is visible').toBeVisible();
|
||||||
|
|
||||||
|
const count = counter.locator('pre');
|
||||||
|
await expect(count, 'initial count is 0').toHaveText('0');
|
||||||
|
|
||||||
|
const inc = counter.locator('.increment');
|
||||||
|
await inc.click();
|
||||||
|
|
||||||
|
await expect(count, 'component not hydrated').toHaveText('0');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('client:idle', async ({ page, astro }) => {
|
||||||
|
await page.goto(astro.resolveUrl(pageUrl));
|
||||||
|
|
||||||
|
const counter = page.locator('#client-idle');
|
||||||
|
await expect(counter, 'component is visible').toBeVisible();
|
||||||
|
|
||||||
|
const count = counter.locator('pre');
|
||||||
|
await expect(count, 'initial count is 0').toHaveText('0');
|
||||||
|
|
||||||
|
const inc = counter.locator('.increment');
|
||||||
|
await inc.click();
|
||||||
|
|
||||||
|
await expect(count, 'count incremented by 1').toHaveText('1');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('client:load', async ({ page, astro }) => {
|
||||||
|
await page.goto(astro.resolveUrl(pageUrl));
|
||||||
|
|
||||||
|
const counter = page.locator('#client-load');
|
||||||
|
await expect(counter, 'component is visible').toBeVisible();
|
||||||
|
|
||||||
|
const count = counter.locator('pre');
|
||||||
|
await expect(count, 'initial count is 0').toHaveText('0');
|
||||||
|
|
||||||
|
const inc = counter.locator('.increment');
|
||||||
|
await inc.click();
|
||||||
|
|
||||||
|
await expect(count, 'count incremented by 1').toHaveText('1');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('client:visible', async ({ page, astro }) => {
|
||||||
|
await page.goto(astro.resolveUrl(pageUrl));
|
||||||
|
|
||||||
|
// Make sure the component is on screen to trigger hydration
|
||||||
|
const counter = page.locator('#client-visible');
|
||||||
|
await counter.scrollIntoViewIfNeeded();
|
||||||
|
await expect(counter, 'component is visible').toBeVisible();
|
||||||
|
|
||||||
|
const count = counter.locator('pre');
|
||||||
|
await expect(count, 'initial count is 0').toHaveText('0');
|
||||||
|
|
||||||
|
const inc = counter.locator('.increment');
|
||||||
|
await inc.click();
|
||||||
|
|
||||||
|
await expect(count, 'count incremented by 1').toHaveText('1');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('client:media', async ({ page, astro }) => {
|
||||||
|
await page.goto(astro.resolveUrl(pageUrl));
|
||||||
|
|
||||||
|
const counter = page.locator('#client-media');
|
||||||
|
await expect(counter, 'component is visible').toBeVisible();
|
||||||
|
|
||||||
|
const count = counter.locator('pre');
|
||||||
|
await expect(count, 'initial count is 0').toHaveText('0');
|
||||||
|
|
||||||
|
const inc = counter.locator('.increment');
|
||||||
|
await inc.click();
|
||||||
|
await expect(count, 'component not hydrated yet').toHaveText('0');
|
||||||
|
|
||||||
|
// Reset the viewport to hydrate the component (max-width: 50rem)
|
||||||
|
await page.setViewportSize({ width: 414, height: 1124 });
|
||||||
|
await inc.click();
|
||||||
|
await expect(count, 'count incremented by 1').toHaveText('1');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('client:only', async ({ page, astro }) => {
|
||||||
|
await page.goto(astro.resolveUrl(pageUrl));
|
||||||
|
|
||||||
|
const label = page.locator('#client-only');
|
||||||
|
await expect(label, 'component is visible').toBeVisible();
|
||||||
|
|
||||||
|
await expect(label, 'slot text is visible').toHaveText('Framework client:only component');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('HMR', async ({ page, astro }) => {
|
||||||
|
await page.goto(astro.resolveUrl(pageUrl));
|
||||||
|
|
||||||
|
const counter = page.locator('#client-idle');
|
||||||
|
const count = counter.locator('pre');
|
||||||
|
await expect(count, 'initial count is 0').toHaveText('0');
|
||||||
|
|
||||||
|
// Edit the component's initial count prop
|
||||||
|
await astro.editFile(pageSourceFilePath, (original) =>
|
||||||
|
original.replace('id="client-idle" {...someProps}', 'id="client-idle" count={5}')
|
||||||
|
);
|
||||||
|
|
||||||
|
await expect(count, 'count prop updated').toHaveText('5');
|
||||||
|
await expect(counter, 'component styles persisted').toHaveCSS('display', 'grid');
|
||||||
|
|
||||||
|
// Edit the client:only component's slot text
|
||||||
|
await astro.editFile(componentFilePath, (original) =>
|
||||||
|
original.replace('Framework client:only component', 'Updated framework client:only component')
|
||||||
|
);
|
||||||
|
|
||||||
|
const label = page.locator('#client-only');
|
||||||
|
await expect(label, 'client:only component is visible').toBeVisible();
|
||||||
|
await expect(label, 'client:only slot text is visible').toHaveText(
|
||||||
|
'Updated framework client:only component'
|
||||||
|
);
|
||||||
|
|
||||||
|
// Edit the imported CSS file
|
||||||
|
await astro.editFile(counterCssFilePath || './src/components/Counter.css', (original) =>
|
||||||
|
original.replace('font-size: 2em;', 'font-size: 24px;')
|
||||||
|
);
|
||||||
|
|
||||||
|
await expect(count, 'imported CSS updated').toHaveCSS('font-size', '24px');
|
||||||
|
|
||||||
|
// Revert our edits
|
||||||
|
astro.resetAllFiles();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
test,
|
||||||
|
createTests,
|
||||||
|
};
|
||||||
|
}
|
|
@ -1,125 +1,19 @@
|
||||||
import { test as base, expect } from '@playwright/test';
|
import { prepareTestFactory } from './shared-component-tests.js';
|
||||||
import { loadFixture } from './test-utils.js';
|
|
||||||
|
|
||||||
const test = base.extend({
|
const { test, createTests } = prepareTestFactory({ root: './fixtures/solid-component/' });
|
||||||
astro: async ({}, use) => {
|
|
||||||
const fixture = await loadFixture({ root: './fixtures/solid-component/' });
|
|
||||||
await use(fixture);
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
let devServer;
|
test.describe('Solid components in Astro files', () => {
|
||||||
|
createTests({
|
||||||
test.beforeEach(async ({ astro }) => {
|
pageUrl: '/',
|
||||||
devServer = await astro.startDevServer();
|
pageSourceFilePath: './src/pages/index.astro',
|
||||||
});
|
componentFilePath: './src/components/SolidComponent.jsx',
|
||||||
|
});
|
||||||
test.afterEach(async () => {
|
});
|
||||||
await devServer.stop();
|
|
||||||
});
|
test.describe('Solid components in Markdown files', () => {
|
||||||
|
createTests({
|
||||||
test.describe('Solid components', () => {
|
pageUrl: '/markdown/',
|
||||||
test('server only', async ({ page, astro }) => {
|
pageSourceFilePath: './src/pages/markdown.md',
|
||||||
await page.goto(astro.resolveUrl('/'));
|
componentFilePath: './src/components/SolidComponent.jsx',
|
||||||
|
|
||||||
const counter = page.locator('#server-only');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'component not hydrated').toHaveText('0');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:idle', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-idle');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:load', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-load');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:visible', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
// Make sure the component is on screen to trigger hydration
|
|
||||||
const counter = page.locator('#client-visible');
|
|
||||||
await counter.scrollIntoViewIfNeeded();
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:media', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-media');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
await expect(count, 'component not hydrated yet').toHaveText('0');
|
|
||||||
|
|
||||||
// Reset the viewport to hydrate the component (max-width: 50rem)
|
|
||||||
await page.setViewportSize({ width: 414, height: 1124 });
|
|
||||||
await inc.click();
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('HMR', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-idle');
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
// Edit the component's initial count prop
|
|
||||||
await astro.editFile('./src/pages/index.astro', (original) =>
|
|
||||||
original.replace('id="client-idle" {...someProps}', 'id="client-idle" count={5}')
|
|
||||||
);
|
|
||||||
|
|
||||||
await expect(counter, 'component styles persisted').toHaveCSS('display', 'grid');
|
|
||||||
await expect(count, 'count prop updated').toHaveText('5');
|
|
||||||
|
|
||||||
// Edit the imported CSS
|
|
||||||
await astro.editFile('./src/components/Counter.css', (original) =>
|
|
||||||
original.replace('font-size: 2em;', 'font-size: 24px;')
|
|
||||||
);
|
|
||||||
|
|
||||||
await expect(count, 'imported CSS updated').toHaveCSS('font-size', '24px');
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -1,117 +1,21 @@
|
||||||
import { test as base, expect } from '@playwright/test';
|
import { prepareTestFactory } from './shared-component-tests.js';
|
||||||
import { loadFixture } from './test-utils.js';
|
|
||||||
|
|
||||||
const test = base.extend({
|
const { test, createTests } = prepareTestFactory({ root: './fixtures/svelte-component/' });
|
||||||
astro: async ({}, use) => {
|
|
||||||
const fixture = await loadFixture({ root: './fixtures/svelte-component/' });
|
|
||||||
await use(fixture);
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
let devServer;
|
test.describe('Svelte components in Astro files', () => {
|
||||||
|
createTests({
|
||||||
test.beforeEach(async ({ astro }) => {
|
pageUrl: '/',
|
||||||
devServer = await astro.startDevServer();
|
pageSourceFilePath: './src/pages/index.astro',
|
||||||
});
|
componentFilePath: './src/components/SvelteComponent.svelte',
|
||||||
|
counterCssFilePath: './src/components/Counter.svelte',
|
||||||
test.afterEach(async () => {
|
});
|
||||||
await devServer.stop();
|
});
|
||||||
});
|
|
||||||
|
test.describe('Svelte components in Markdown files', () => {
|
||||||
test.describe('Svelte components', () => {
|
createTests({
|
||||||
test('server only', async ({ page, astro }) => {
|
pageUrl: '/markdown/',
|
||||||
await page.goto(astro.resolveUrl('/'));
|
pageSourceFilePath: './src/pages/markdown.md',
|
||||||
|
componentFilePath: './src/components/SvelteComponent.svelte',
|
||||||
const counter = page.locator('#server-only');
|
counterCssFilePath: './src/components/Counter.svelte',
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'component not hydrated').toHaveText('0');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:idle', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-idle');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:load', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-load');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:visible', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
// Make sure the component is on screen to trigger hydration
|
|
||||||
const counter = page.locator('#client-visible');
|
|
||||||
await counter.scrollIntoViewIfNeeded();
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('client:media', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
const counter = page.locator('#client-media');
|
|
||||||
await expect(counter, 'component is visible').toBeVisible();
|
|
||||||
|
|
||||||
const count = counter.locator('pre');
|
|
||||||
await expect(count, 'initial count is 0').toHaveText('0');
|
|
||||||
|
|
||||||
const inc = counter.locator('.increment');
|
|
||||||
await inc.click();
|
|
||||||
await expect(count, 'component not hydrated yet').toHaveText('0');
|
|
||||||
|
|
||||||
// Reset the viewport to hydrate the component (max-width: 50rem)
|
|
||||||
await page.setViewportSize({ width: 414, height: 1124 });
|
|
||||||
await inc.click();
|
|
||||||
await expect(count, 'count incremented by 1').toHaveText('1');
|
|
||||||
});
|
|
||||||
|
|
||||||
test('HMR', async ({ page, astro }) => {
|
|
||||||
await page.goto(astro.resolveUrl('/'));
|
|
||||||
|
|
||||||
// Edit the component's slot text
|
|
||||||
await astro.editFile('./src/pages/index.astro', (original) =>
|
|
||||||
original.replace('Hello, client:idle!', 'Hello, updated client:idle!')
|
|
||||||
);
|
|
||||||
|
|
||||||
const counter = page.locator('#client-idle');
|
|
||||||
const label = page.locator('#client-idle-message');
|
|
||||||
|
|
||||||
await expect(label, 'slot text updated').toHaveText('Hello, updated client:idle!');
|
|
||||||
await expect(counter, 'component styles persisted').toHaveCSS('display', 'grid');
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in a new issue