@arcadialdev/arcality 2.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.agents/skills/e2e-testing-expert/SKILL.md +28 -0
- package/.agents/skills/frontend-design/LICENSE.txt +177 -0
- package/.agents/skills/frontend-design/SKILL.md +42 -0
- package/.agents/skills/nodejs-backend-patterns/SKILL.md +639 -0
- package/.agents/skills/nodejs-backend-patterns/references/advanced-patterns.md +430 -0
- package/.agents/skills/playwright-best-practices/LICENSE.md +7 -0
- package/.agents/skills/playwright-best-practices/README.md +147 -0
- package/.agents/skills/playwright-best-practices/SKILL.md +303 -0
- package/.agents/skills/playwright-best-practices/advanced/authentication-flows.md +360 -0
- package/.agents/skills/playwright-best-practices/advanced/authentication.md +871 -0
- package/.agents/skills/playwright-best-practices/advanced/clock-mocking.md +364 -0
- package/.agents/skills/playwright-best-practices/advanced/mobile-testing.md +409 -0
- package/.agents/skills/playwright-best-practices/advanced/multi-context.md +288 -0
- package/.agents/skills/playwright-best-practices/advanced/multi-user.md +393 -0
- package/.agents/skills/playwright-best-practices/advanced/network-advanced.md +452 -0
- package/.agents/skills/playwright-best-practices/advanced/third-party.md +464 -0
- package/.agents/skills/playwright-best-practices/architecture/pom-vs-fixtures.md +363 -0
- package/.agents/skills/playwright-best-practices/architecture/test-architecture.md +369 -0
- package/.agents/skills/playwright-best-practices/architecture/when-to-mock.md +383 -0
- package/.agents/skills/playwright-best-practices/browser-apis/browser-apis.md +391 -0
- package/.agents/skills/playwright-best-practices/browser-apis/iframes.md +403 -0
- package/.agents/skills/playwright-best-practices/browser-apis/service-workers.md +504 -0
- package/.agents/skills/playwright-best-practices/browser-apis/websockets.md +403 -0
- package/.agents/skills/playwright-best-practices/core/annotations.md +424 -0
- package/.agents/skills/playwright-best-practices/core/assertions-waiting.md +361 -0
- package/.agents/skills/playwright-best-practices/core/configuration.md +452 -0
- package/.agents/skills/playwright-best-practices/core/fixtures-hooks.md +417 -0
- package/.agents/skills/playwright-best-practices/core/global-setup.md +434 -0
- package/.agents/skills/playwright-best-practices/core/locators.md +242 -0
- package/.agents/skills/playwright-best-practices/core/page-object-model.md +315 -0
- package/.agents/skills/playwright-best-practices/core/projects-dependencies.md +453 -0
- package/.agents/skills/playwright-best-practices/core/test-data.md +492 -0
- package/.agents/skills/playwright-best-practices/core/test-suite-structure.md +361 -0
- package/.agents/skills/playwright-best-practices/core/test-tags.md +298 -0
- package/.agents/skills/playwright-best-practices/debugging/console-errors.md +420 -0
- package/.agents/skills/playwright-best-practices/debugging/debugging.md +504 -0
- package/.agents/skills/playwright-best-practices/debugging/error-testing.md +360 -0
- package/.agents/skills/playwright-best-practices/debugging/flaky-tests.md +496 -0
- package/.agents/skills/playwright-best-practices/frameworks/angular.md +530 -0
- package/.agents/skills/playwright-best-practices/frameworks/nextjs.md +469 -0
- package/.agents/skills/playwright-best-practices/frameworks/react.md +531 -0
- package/.agents/skills/playwright-best-practices/frameworks/vue.md +574 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/ci-cd.md +468 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/docker.md +283 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/github-actions.md +546 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/gitlab.md +397 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/other-providers.md +521 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/parallel-sharding.md +371 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/performance.md +453 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/reporting.md +424 -0
- package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/test-coverage.md +497 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/accessibility.md +359 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/api-testing.md +719 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/browser-extensions.md +506 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/canvas-webgl.md +493 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/component-testing.md +500 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/drag-drop.md +576 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/electron.md +509 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/file-operations.md +377 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/file-upload-download.md +562 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/forms-validation.md +561 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/graphql-testing.md +331 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/i18n.md +508 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/performance-testing.md +476 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/security-testing.md +430 -0
- package/.agents/skills/playwright-best-practices/testing-patterns/visual-regression.md +634 -0
- package/.env.example +21 -0
- package/README.md +30 -0
- package/bin/arcality.mjs +86 -0
- package/package.json +66 -0
- package/playwright.config.ts +12 -0
- package/scripts/cleanup-qmsdev.mjs +63 -0
- package/scripts/discover-view.mjs +52 -0
- package/scripts/extract-view.mjs +64 -0
- package/scripts/gen-and-run.mjs +838 -0
- package/scripts/init.mjs +290 -0
- package/scripts/migrate-to-central-out.mjs +157 -0
- package/scripts/postinstall.mjs +63 -0
- package/scripts/rebrand-report.mjs +241 -0
- package/scripts/setup.mjs +166 -0
- package/src/KnowledgeService.ts +239 -0
- package/src/arcalityClient.mjs +266 -0
- package/src/configLoader.mjs +179 -0
- package/src/configManager.mjs +172 -0
- package/src/consoleBanner.ts +32 -0
- package/src/envSetup.ts +205 -0
- package/src/index.ts +25 -0
- package/src/projectInspector.ts +42 -0
- package/src/services/collectiveMemoryService.ts +178 -0
- package/src/testRunner.ts +201 -0
- package/tests/_helpers/ArcalityReporter.ts +490 -0
- package/tests/_helpers/agentic-runner.spec.ts +741 -0
- package/tests/_helpers/ai-agent-helper.ts +1573 -0
- package/tests/_helpers/discover-view.spec.ts +238 -0
- package/tests/_helpers/extract-view.spec.ts +118 -0
- package/tests/_helpers/qa-tools.ts +333 -0
- package/tests/_helpers/smart-action.spec.ts +1458 -0
|
@@ -0,0 +1,530 @@
|
|
|
1
|
+
# Angular Testing with Playwright
|
|
2
|
+
|
|
3
|
+
## Table of Contents
|
|
4
|
+
|
|
5
|
+
1. [Configuration](#configuration)
|
|
6
|
+
2. [Locator Strategies](#locator-strategies)
|
|
7
|
+
3. [Reactive Forms](#reactive-forms)
|
|
8
|
+
4. [Angular Material Components](#angular-material-components)
|
|
9
|
+
5. [Router Navigation](#router-navigation)
|
|
10
|
+
6. [Lazy-Loaded Modules](#lazy-loaded-modules)
|
|
11
|
+
7. [Signals and Observables](#signals-and-observables)
|
|
12
|
+
8. [Zone.js and Change Detection](#zonejs-and-change-detection)
|
|
13
|
+
9. [SSR Testing](#ssr-testing)
|
|
14
|
+
10. [Protractor Migration Reference](#protractor-migration-reference)
|
|
15
|
+
11. [Build Configurations](#build-configurations)
|
|
16
|
+
12. [CDK Overlay Container](#cdk-overlay-container)
|
|
17
|
+
13. [Anti-Patterns](#anti-patterns)
|
|
18
|
+
14. [Related](#related)
|
|
19
|
+
|
|
20
|
+
> **When to use**: Testing Angular applications with reactive forms, Angular Material components, Router navigation, lazy-loaded modules, signals, observables, and Zone.js change detection.
|
|
21
|
+
> **Prerequisites**: [core/configuration.md](../core/configuration.md), [core/locators.md](../core/locators.md)
|
|
22
|
+
|
|
23
|
+
## Configuration
|
|
24
|
+
|
|
25
|
+
### Playwright Config
|
|
26
|
+
|
|
27
|
+
```typescript
|
|
28
|
+
import { defineConfig, devices } from '@playwright/test';
|
|
29
|
+
|
|
30
|
+
export default defineConfig({
|
|
31
|
+
testDir: './e2e',
|
|
32
|
+
testMatch: '**/*.spec.ts',
|
|
33
|
+
fullyParallel: true,
|
|
34
|
+
forbidOnly: !!process.env.CI,
|
|
35
|
+
retries: process.env.CI ? 2 : 0,
|
|
36
|
+
workers: process.env.CI ? '50%' : undefined,
|
|
37
|
+
|
|
38
|
+
use: {
|
|
39
|
+
baseURL: 'http://localhost:4200',
|
|
40
|
+
trace: 'on-first-retry',
|
|
41
|
+
screenshot: 'only-on-failure',
|
|
42
|
+
},
|
|
43
|
+
|
|
44
|
+
projects: [
|
|
45
|
+
{ name: 'chromium', use: { ...devices['Desktop Chrome'] } },
|
|
46
|
+
{ name: 'firefox', use: { ...devices['Desktop Firefox'] } },
|
|
47
|
+
{ name: 'mobile', use: { ...devices['iPhone 14'] } },
|
|
48
|
+
],
|
|
49
|
+
|
|
50
|
+
webServer: {
|
|
51
|
+
command: process.env.CI
|
|
52
|
+
? 'npx ng build && npx http-server dist/my-app/browser -p 4200 -s'
|
|
53
|
+
: 'npx ng serve',
|
|
54
|
+
url: 'http://localhost:4200',
|
|
55
|
+
reuseExistingServer: !process.env.CI,
|
|
56
|
+
timeout: 120_000,
|
|
57
|
+
},
|
|
58
|
+
});
|
|
59
|
+
```
|
|
60
|
+
|
|
61
|
+
### Project Structure
|
|
62
|
+
|
|
63
|
+
```text
|
|
64
|
+
my-angular-app/
|
|
65
|
+
src/
|
|
66
|
+
e2e/
|
|
67
|
+
tests/
|
|
68
|
+
dashboard.spec.ts
|
|
69
|
+
login.spec.ts
|
|
70
|
+
fixtures/
|
|
71
|
+
auth.fixture.ts
|
|
72
|
+
playwright.config.ts
|
|
73
|
+
angular.json
|
|
74
|
+
```
|
|
75
|
+
|
|
76
|
+
### Package Scripts
|
|
77
|
+
|
|
78
|
+
```json
|
|
79
|
+
{
|
|
80
|
+
"scripts": {
|
|
81
|
+
"e2e": "playwright test",
|
|
82
|
+
"e2e:headed": "playwright test --headed",
|
|
83
|
+
"e2e:debug": "playwright test --debug",
|
|
84
|
+
"e2e:report": "playwright show-report"
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
```
|
|
88
|
+
|
|
89
|
+
## Locator Strategies
|
|
90
|
+
|
|
91
|
+
Angular generates internal attributes (`_ngcontent-*`, `_nghost-*`, `ng-reflect-*`) that change every build. Always use semantic locators.
|
|
92
|
+
|
|
93
|
+
```typescript
|
|
94
|
+
test('use semantic locators for Angular apps', async ({ page }) => {
|
|
95
|
+
await page.goto('/projects');
|
|
96
|
+
|
|
97
|
+
// Role-based locators work with Angular Material and native HTML
|
|
98
|
+
await page.getByRole('button', { name: 'New project' }).click();
|
|
99
|
+
await expect(page.getByRole('heading', { name: 'Create Project' })).toBeVisible();
|
|
100
|
+
|
|
101
|
+
// Label-based for form fields
|
|
102
|
+
await page.getByLabel('Project title').fill('Alpha');
|
|
103
|
+
|
|
104
|
+
// Test IDs for complex components without semantic roles
|
|
105
|
+
const chart = page.getByTestId('metrics-chart');
|
|
106
|
+
await expect(chart).toBeVisible();
|
|
107
|
+
|
|
108
|
+
// Scope locators within component boundaries
|
|
109
|
+
const projectTable = page.getByRole('table', { name: 'Projects' });
|
|
110
|
+
const activeRow = projectTable.getByRole('row').filter({
|
|
111
|
+
has: page.getByRole('cell', { name: 'Active' }),
|
|
112
|
+
});
|
|
113
|
+
await activeRow.getByRole('button', { name: 'Edit' }).click();
|
|
114
|
+
});
|
|
115
|
+
```
|
|
116
|
+
|
|
117
|
+
## Reactive Forms
|
|
118
|
+
|
|
119
|
+
Playwright interacts with the rendered DOM, so reactive forms (`FormGroup`, `FormControl`, `FormArray`) are transparent.
|
|
120
|
+
|
|
121
|
+
```typescript
|
|
122
|
+
test.describe('form validation', () => {
|
|
123
|
+
test.beforeEach(async ({ page }) => {
|
|
124
|
+
await page.goto('/signup');
|
|
125
|
+
});
|
|
126
|
+
|
|
127
|
+
test('displays validation errors on blur', async ({ page }) => {
|
|
128
|
+
const emailField = page.getByLabel('Email');
|
|
129
|
+
await emailField.click();
|
|
130
|
+
await emailField.blur();
|
|
131
|
+
await expect(page.getByText('Email is required')).toBeVisible();
|
|
132
|
+
|
|
133
|
+
await emailField.fill('invalid');
|
|
134
|
+
await emailField.blur();
|
|
135
|
+
await expect(page.getByText('Invalid email format')).toBeVisible();
|
|
136
|
+
});
|
|
137
|
+
|
|
138
|
+
test('validates password confirmation', async ({ page }) => {
|
|
139
|
+
await page.getByLabel('Password', { exact: true }).fill('Secret123!');
|
|
140
|
+
await page.getByLabel('Confirm password').fill('Mismatch');
|
|
141
|
+
await page.getByLabel('Confirm password').blur();
|
|
142
|
+
|
|
143
|
+
await expect(page.getByText('Passwords must match')).toBeVisible();
|
|
144
|
+
|
|
145
|
+
await page.getByLabel('Confirm password').fill('Secret123!');
|
|
146
|
+
await expect(page.getByText('Passwords must match')).toBeHidden();
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
test('handles FormArray add/remove', async ({ page }) => {
|
|
150
|
+
await page.goto('/contacts/edit');
|
|
151
|
+
|
|
152
|
+
await page.getByRole('button', { name: 'Add email' }).click();
|
|
153
|
+
const emailInputs = page.getByLabel(/Email address/);
|
|
154
|
+
await expect(emailInputs).toHaveCount(2);
|
|
155
|
+
|
|
156
|
+
await emailInputs.nth(1).fill('backup@example.com');
|
|
157
|
+
await page.getByRole('button', { name: 'Remove email 1' }).click();
|
|
158
|
+
|
|
159
|
+
await expect(emailInputs).toHaveCount(1);
|
|
160
|
+
await expect(emailInputs.first()).toHaveValue('backup@example.com');
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
test('disables submit until form is valid', async ({ page }) => {
|
|
164
|
+
const submitBtn = page.getByRole('button', { name: 'Register' });
|
|
165
|
+
await expect(submitBtn).toBeDisabled();
|
|
166
|
+
|
|
167
|
+
await page.getByLabel('Name').fill('Alice');
|
|
168
|
+
await page.getByLabel('Email').fill('alice@test.com');
|
|
169
|
+
await page.getByLabel('Password', { exact: true }).fill('Secret123!');
|
|
170
|
+
await page.getByLabel('Confirm password').fill('Secret123!');
|
|
171
|
+
await page.getByLabel('Accept terms').check();
|
|
172
|
+
|
|
173
|
+
await expect(submitBtn).toBeEnabled();
|
|
174
|
+
});
|
|
175
|
+
|
|
176
|
+
test('shows async validator loading state', async ({ page }) => {
|
|
177
|
+
await page.route('**/api/username-check*', async (route) => {
|
|
178
|
+
await new Promise((r) => setTimeout(r, 800));
|
|
179
|
+
await route.fulfill({ json: { available: true } });
|
|
180
|
+
});
|
|
181
|
+
|
|
182
|
+
await page.getByLabel('Username').fill('alice');
|
|
183
|
+
await page.getByLabel('Username').blur();
|
|
184
|
+
|
|
185
|
+
await expect(page.getByTestId('username-loading')).toBeVisible();
|
|
186
|
+
await expect(page.getByTestId('username-loading')).toBeHidden();
|
|
187
|
+
await expect(page.getByText('Username available')).toBeVisible();
|
|
188
|
+
});
|
|
189
|
+
});
|
|
190
|
+
```
|
|
191
|
+
|
|
192
|
+
## Angular Material Components
|
|
193
|
+
|
|
194
|
+
Angular Material uses proper ARIA attributes. Use role-based locators instead of CSS classes like `.mat-mdc-button`.
|
|
195
|
+
|
|
196
|
+
```typescript
|
|
197
|
+
test.describe('Material components', () => {
|
|
198
|
+
test('mat-select dropdown', async ({ page }) => {
|
|
199
|
+
await page.goto('/preferences');
|
|
200
|
+
|
|
201
|
+
await page.getByRole('combobox', { name: 'Language' }).click();
|
|
202
|
+
await page.getByRole('option', { name: 'Spanish' }).click();
|
|
203
|
+
|
|
204
|
+
await expect(page.getByRole('combobox', { name: 'Language' })).toContainText('Spanish');
|
|
205
|
+
});
|
|
206
|
+
|
|
207
|
+
test('mat-autocomplete suggestions', async ({ page }) => {
|
|
208
|
+
await page.goto('/members/add');
|
|
209
|
+
|
|
210
|
+
const roleField = page.getByRole('combobox', { name: 'Role' });
|
|
211
|
+
await roleField.fill('dev');
|
|
212
|
+
|
|
213
|
+
await expect(page.getByRole('option', { name: 'Developer' })).toBeVisible();
|
|
214
|
+
await expect(page.getByRole('option', { name: 'DevOps' })).toBeVisible();
|
|
215
|
+
|
|
216
|
+
await page.getByRole('option', { name: 'Developer' }).click();
|
|
217
|
+
await expect(roleField).toHaveValue('Developer');
|
|
218
|
+
});
|
|
219
|
+
|
|
220
|
+
test('mat-dialog interaction', async ({ page }) => {
|
|
221
|
+
await page.goto('/items');
|
|
222
|
+
|
|
223
|
+
await page.getByRole('button', { name: 'Remove item' }).first().click();
|
|
224
|
+
|
|
225
|
+
const dialog = page.getByRole('dialog');
|
|
226
|
+
await expect(dialog).toBeVisible();
|
|
227
|
+
await expect(dialog.getByText('Confirm deletion?')).toBeVisible();
|
|
228
|
+
|
|
229
|
+
await dialog.getByRole('button', { name: 'Cancel' }).click();
|
|
230
|
+
await expect(dialog).toBeHidden();
|
|
231
|
+
});
|
|
232
|
+
|
|
233
|
+
test('mat-table sorting', async ({ page }) => {
|
|
234
|
+
await page.goto('/members');
|
|
235
|
+
|
|
236
|
+
await page.getByRole('columnheader', { name: 'Name' }).click();
|
|
237
|
+
const header = page.getByRole('columnheader', { name: 'Name' });
|
|
238
|
+
await expect(header).toHaveAttribute('aria-sort', 'ascending');
|
|
239
|
+
|
|
240
|
+
await page.getByRole('columnheader', { name: 'Name' }).click();
|
|
241
|
+
await expect(header).toHaveAttribute('aria-sort', 'descending');
|
|
242
|
+
});
|
|
243
|
+
|
|
244
|
+
test('mat-paginator navigation', async ({ page }) => {
|
|
245
|
+
await page.goto('/members');
|
|
246
|
+
|
|
247
|
+
await expect(page.getByText('1 - 10 of 100')).toBeVisible();
|
|
248
|
+
|
|
249
|
+
await page.getByRole('button', { name: 'Next page' }).click();
|
|
250
|
+
await expect(page.getByText('11 - 20 of 100')).toBeVisible();
|
|
251
|
+
|
|
252
|
+
await page.getByRole('combobox', { name: 'Items per page' }).click();
|
|
253
|
+
await page.getByRole('option', { name: '50' }).click();
|
|
254
|
+
await expect(page.getByText('1 - 50 of 100')).toBeVisible();
|
|
255
|
+
});
|
|
256
|
+
|
|
257
|
+
test('mat-snack-bar notification', async ({ page }) => {
|
|
258
|
+
await page.goto('/preferences');
|
|
259
|
+
|
|
260
|
+
await page.getByRole('button', { name: 'Save' }).click();
|
|
261
|
+
await expect(page.getByText('Changes saved')).toBeVisible();
|
|
262
|
+
|
|
263
|
+
await page.getByRole('button', { name: 'Close' }).click();
|
|
264
|
+
await expect(page.getByText('Changes saved')).toBeHidden();
|
|
265
|
+
});
|
|
266
|
+
|
|
267
|
+
test('mat-stepper wizard', async ({ page }) => {
|
|
268
|
+
await page.goto('/wizard');
|
|
269
|
+
|
|
270
|
+
await expect(page.getByText('Step 1 of 3')).toBeVisible();
|
|
271
|
+
await page.getByLabel('Name').fill('Bob');
|
|
272
|
+
await page.getByRole('button', { name: 'Next' }).click();
|
|
273
|
+
|
|
274
|
+
await expect(page.getByText('Step 2 of 3')).toBeVisible();
|
|
275
|
+
await page.getByLabel('Organization').fill('Acme');
|
|
276
|
+
await page.getByRole('button', { name: 'Next' }).click();
|
|
277
|
+
|
|
278
|
+
await expect(page.getByText('Step 3 of 3')).toBeVisible();
|
|
279
|
+
await expect(page.getByText('Bob')).toBeVisible();
|
|
280
|
+
await expect(page.getByText('Acme')).toBeVisible();
|
|
281
|
+
});
|
|
282
|
+
});
|
|
283
|
+
```
|
|
284
|
+
|
|
285
|
+
## Router Navigation
|
|
286
|
+
|
|
287
|
+
```typescript
|
|
288
|
+
test.describe('Angular Router', () => {
|
|
289
|
+
test('lazy-loaded module loads on navigation', async ({ page }) => {
|
|
290
|
+
await page.goto('/');
|
|
291
|
+
|
|
292
|
+
await page.getByRole('link', { name: 'Reports' }).click();
|
|
293
|
+
await page.waitForURL('/reports');
|
|
294
|
+
|
|
295
|
+
await expect(page.getByRole('heading', { name: 'Reports Dashboard' })).toBeVisible();
|
|
296
|
+
});
|
|
297
|
+
|
|
298
|
+
test('route guard redirects unauthorized users', async ({ page }) => {
|
|
299
|
+
await page.goto('/admin/settings');
|
|
300
|
+
|
|
301
|
+
await expect(page).toHaveURL(/\/login/);
|
|
302
|
+
await expect(page.getByRole('heading', { name: 'Sign in' })).toBeVisible();
|
|
303
|
+
});
|
|
304
|
+
|
|
305
|
+
test('resolver prefetches data', async ({ page }) => {
|
|
306
|
+
const resolverPromise = page.waitForResponse('**/api/items/*');
|
|
307
|
+
await page.goto('/items/42');
|
|
308
|
+
await resolverPromise;
|
|
309
|
+
|
|
310
|
+
await expect(page.getByRole('heading', { level: 1 })).toContainText('Item');
|
|
311
|
+
});
|
|
312
|
+
|
|
313
|
+
test('nested router-outlet renders children', async ({ page }) => {
|
|
314
|
+
await page.goto('/account/profile');
|
|
315
|
+
|
|
316
|
+
await expect(page.getByRole('heading', { name: 'Account' })).toBeVisible();
|
|
317
|
+
await expect(page.getByRole('heading', { name: 'Profile', level: 2 })).toBeVisible();
|
|
318
|
+
|
|
319
|
+
await page.getByRole('link', { name: 'Security' }).click();
|
|
320
|
+
await page.waitForURL('/account/security');
|
|
321
|
+
|
|
322
|
+
await expect(page.getByRole('heading', { name: 'Account' })).toBeVisible();
|
|
323
|
+
await expect(page.getByRole('heading', { name: 'Security', level: 2 })).toBeVisible();
|
|
324
|
+
});
|
|
325
|
+
|
|
326
|
+
test('query parameters drive filters', async ({ page }) => {
|
|
327
|
+
await page.goto('/products?type=hardware&page=3');
|
|
328
|
+
|
|
329
|
+
await expect(page.getByRole('heading', { name: 'Hardware' })).toBeVisible();
|
|
330
|
+
await expect(page.getByText('Page 3')).toBeVisible();
|
|
331
|
+
});
|
|
332
|
+
|
|
333
|
+
test('browser back navigates history', async ({ page }) => {
|
|
334
|
+
await page.goto('/');
|
|
335
|
+
await page.getByRole('link', { name: 'Products' }).click();
|
|
336
|
+
await page.waitForURL('/products');
|
|
337
|
+
await page.getByRole('link', { name: 'About' }).click();
|
|
338
|
+
await page.waitForURL('/about');
|
|
339
|
+
|
|
340
|
+
await page.goBack();
|
|
341
|
+
await expect(page).toHaveURL(/\/products/);
|
|
342
|
+
|
|
343
|
+
await page.goBack();
|
|
344
|
+
await expect(page).toHaveURL(/\/$/);
|
|
345
|
+
});
|
|
346
|
+
});
|
|
347
|
+
```
|
|
348
|
+
|
|
349
|
+
## Lazy-Loaded Modules
|
|
350
|
+
|
|
351
|
+
```typescript
|
|
352
|
+
test('lazy module loads without chunk errors', async ({ page }) => {
|
|
353
|
+
const consoleErrors: string[] = [];
|
|
354
|
+
page.on('console', (msg) => {
|
|
355
|
+
if (msg.type() === 'error') consoleErrors.push(msg.text());
|
|
356
|
+
});
|
|
357
|
+
|
|
358
|
+
await page.goto('/');
|
|
359
|
+
|
|
360
|
+
const chunkRequest = page.waitForResponse((r) =>
|
|
361
|
+
r.url().includes('.js') && r.status() === 200
|
|
362
|
+
);
|
|
363
|
+
await page.getByRole('link', { name: 'Analytics' }).click();
|
|
364
|
+
await chunkRequest;
|
|
365
|
+
|
|
366
|
+
await page.waitForURL('/analytics');
|
|
367
|
+
await expect(page.getByRole('heading', { name: 'Analytics' })).toBeVisible();
|
|
368
|
+
|
|
369
|
+
const chunkErrors = consoleErrors.filter(
|
|
370
|
+
(e) => e.includes('ChunkLoadError') || e.includes('Loading chunk')
|
|
371
|
+
);
|
|
372
|
+
expect(chunkErrors).toEqual([]);
|
|
373
|
+
});
|
|
374
|
+
```
|
|
375
|
+
|
|
376
|
+
## Signals and Observables
|
|
377
|
+
|
|
378
|
+
Playwright cannot subscribe to observables or read signals directly. Test through the rendered output.
|
|
379
|
+
|
|
380
|
+
```typescript
|
|
381
|
+
test.describe('signals through UI', () => {
|
|
382
|
+
test('signal-based counter updates DOM', async ({ page }) => {
|
|
383
|
+
await page.goto('/counter');
|
|
384
|
+
|
|
385
|
+
await expect(page.getByTestId('value')).toHaveText('0');
|
|
386
|
+
|
|
387
|
+
await page.getByRole('button', { name: 'Increment' }).click();
|
|
388
|
+
await expect(page.getByTestId('value')).toHaveText('1');
|
|
389
|
+
|
|
390
|
+
await page.getByRole('button', { name: 'Reset' }).click();
|
|
391
|
+
await expect(page.getByTestId('value')).toHaveText('0');
|
|
392
|
+
});
|
|
393
|
+
|
|
394
|
+
test('computed signal updates derived values', async ({ page }) => {
|
|
395
|
+
await page.goto('/cart');
|
|
396
|
+
await expect(page.getByTestId('total')).toHaveText('$0.00');
|
|
397
|
+
|
|
398
|
+
await page.goto('/catalog');
|
|
399
|
+
await page.getByRole('listitem')
|
|
400
|
+
.filter({ hasText: '$19.99' })
|
|
401
|
+
.getByRole('button', { name: 'Add' })
|
|
402
|
+
.click();
|
|
403
|
+
|
|
404
|
+
await page.getByRole('link', { name: 'Cart' }).click();
|
|
405
|
+
await expect(page.getByTestId('total')).toHaveText('$19.99');
|
|
406
|
+
});
|
|
407
|
+
});
|
|
408
|
+
|
|
409
|
+
test.describe('observables through UI', () => {
|
|
410
|
+
test('debounced search batches API calls', async ({ page }) => {
|
|
411
|
+
await page.goto('/search');
|
|
412
|
+
|
|
413
|
+
const apiCalls: string[] = [];
|
|
414
|
+
await page.route('**/api/search*', async (route) => {
|
|
415
|
+
apiCalls.push(route.request().url());
|
|
416
|
+
await route.continue();
|
|
417
|
+
});
|
|
418
|
+
|
|
419
|
+
await page.getByRole('textbox', { name: 'Search' }).pressSequentially('playwright', {
|
|
420
|
+
delay: 50,
|
|
421
|
+
});
|
|
422
|
+
|
|
423
|
+
await expect(page.getByRole('listitem')).toHaveCount(5);
|
|
424
|
+
expect(apiCalls.length).toBeLessThanOrEqual(2);
|
|
425
|
+
});
|
|
426
|
+
|
|
427
|
+
test('switchMap cancels stale requests', async ({ page }) => {
|
|
428
|
+
await page.goto('/search');
|
|
429
|
+
|
|
430
|
+
await page.getByRole('textbox', { name: 'Search' }).fill('initial');
|
|
431
|
+
await page.getByRole('textbox', { name: 'Search' }).fill('final');
|
|
432
|
+
|
|
433
|
+
await expect(page.getByRole('listitem').first()).toContainText(/final/i);
|
|
434
|
+
});
|
|
435
|
+
});
|
|
436
|
+
```
|
|
437
|
+
|
|
438
|
+
## Zone.js and Change Detection
|
|
439
|
+
|
|
440
|
+
Angular uses Zone.js for change detection. Playwright does not depend on Zone.js and interacts with the DOM directly.
|
|
441
|
+
|
|
442
|
+
- **Change detection timing**: After interactions, Angular schedules change detection via Zone.js. Playwright's auto-waiting handles this.
|
|
443
|
+
- **Zoneless Angular**: Angular 17+ supports zoneless change detection. Tests work identically since Playwright waits for DOM changes.
|
|
444
|
+
- **Long-running async**: `setInterval` or long-running observables keep Angular "not stable." This does not affect Playwright (unlike Protractor).
|
|
445
|
+
|
|
446
|
+
## SSR Testing
|
|
447
|
+
|
|
448
|
+
```typescript
|
|
449
|
+
// playwright.config.ts for SSR
|
|
450
|
+
webServer: {
|
|
451
|
+
command: process.env.CI
|
|
452
|
+
? 'npx ng build --ssr && node dist/my-app/server/server.mjs'
|
|
453
|
+
: 'npx ng serve --ssr',
|
|
454
|
+
url: 'http://localhost:4200',
|
|
455
|
+
reuseExistingServer: !process.env.CI,
|
|
456
|
+
timeout: 180_000,
|
|
457
|
+
},
|
|
458
|
+
```
|
|
459
|
+
|
|
460
|
+
```typescript
|
|
461
|
+
test('no hydration errors', async ({ page }) => {
|
|
462
|
+
const errors: string[] = [];
|
|
463
|
+
page.on('console', (msg) => {
|
|
464
|
+
if (msg.type() === 'error' && msg.text().includes('hydration')) {
|
|
465
|
+
errors.push(msg.text());
|
|
466
|
+
}
|
|
467
|
+
});
|
|
468
|
+
|
|
469
|
+
await page.goto('/');
|
|
470
|
+
await page.getByRole('button', { name: 'Get started' }).click();
|
|
471
|
+
|
|
472
|
+
expect(errors).toEqual([]);
|
|
473
|
+
});
|
|
474
|
+
```
|
|
475
|
+
|
|
476
|
+
## Protractor Migration Reference
|
|
477
|
+
|
|
478
|
+
| Protractor | Playwright |
|
|
479
|
+
|---|---|
|
|
480
|
+
| `element(by.css('.btn'))` | `page.getByRole('button', { name: '...' })` |
|
|
481
|
+
| `element(by.id('login'))` | `page.getByTestId('login')` |
|
|
482
|
+
| `element(by.buttonText('Submit'))` | `page.getByRole('button', { name: 'Submit' })` |
|
|
483
|
+
| `element(by.model('user.name'))` | `page.getByLabel('Name')` |
|
|
484
|
+
| `element(by.binding('user.name'))` | `page.getByText(expectedValue)` |
|
|
485
|
+
| `element(by.repeater('item in items'))` | `page.getByRole('listitem')` |
|
|
486
|
+
| `browser.waitForAngular()` | Not needed — Playwright auto-waits |
|
|
487
|
+
| `browser.sleep(3000)` | `await expect(locator).toBeVisible()` |
|
|
488
|
+
| `browser.get('/path')` | `await page.goto('/path')` |
|
|
489
|
+
| `protractor.ExpectedConditions` | `await expect(locator).toBeVisible()` |
|
|
490
|
+
|
|
491
|
+
## Build Configurations
|
|
492
|
+
|
|
493
|
+
| Scenario | Command | Notes |
|
|
494
|
+
|---|---|---|
|
|
495
|
+
| Local dev | `npx ng serve` | Fast rebuild, source maps |
|
|
496
|
+
| CI production | `npx ng build && npx http-server dist/app/browser -p 4200 -s` | Tests production bundle |
|
|
497
|
+
| CI SSR | `npx ng build --ssr && node dist/app/server/server.mjs` | Tests server-side rendering |
|
|
498
|
+
| Staging | No `webServer` | Point `baseURL` to staging URL |
|
|
499
|
+
|
|
500
|
+
The `-s` flag on `http-server` enables SPA fallback for Angular Router.
|
|
501
|
+
|
|
502
|
+
## CDK Overlay Container
|
|
503
|
+
|
|
504
|
+
Angular Material and CDK render overlays (dialogs, menus, selects) in a special container outside the component tree. Playwright sees these as regular DOM elements:
|
|
505
|
+
|
|
506
|
+
```typescript
|
|
507
|
+
const dialog = page.getByRole('dialog');
|
|
508
|
+
const menu = page.getByRole('menu');
|
|
509
|
+
const listbox = page.getByRole('listbox');
|
|
510
|
+
```
|
|
511
|
+
|
|
512
|
+
## Anti-Patterns
|
|
513
|
+
|
|
514
|
+
| Anti-Pattern | Problem | Solution |
|
|
515
|
+
|---|---|---|
|
|
516
|
+
| `page.locator('[_ngcontent-xyz]')` | Scoped style attributes change every build | Use `getByRole`, `getByLabel`, `getByTestId` |
|
|
517
|
+
| `page.locator('[ng-reflect-model]')` | Only exists in dev mode | Test rendered value: `expect(input).toHaveValue()` |
|
|
518
|
+
| `page.locator('app-my-component')` | Component selectors are implementation details | Target rendered content with semantic locators |
|
|
519
|
+
| `page.locator('.mat-mdc-button')` | Material classes change between versions | `page.getByRole('button', { name: '...' })` |
|
|
520
|
+
| `page.evaluate(() => window.ng)` | Not available in production builds | Test through the DOM |
|
|
521
|
+
| `await page.waitForTimeout(500)` | Zone.js timing varies | Use auto-retrying assertions |
|
|
522
|
+
| `browser.waitForAngular()` | Does not exist in Playwright | Remove entirely |
|
|
523
|
+
| `ng serve` in CI | Slower, includes debug code | Use `ng build && http-server` |
|
|
524
|
+
|
|
525
|
+
## Related
|
|
526
|
+
|
|
527
|
+
- [core/locators.md](../core/locators.md) — locator strategies for Angular Material
|
|
528
|
+
- [core/assertions-waiting.md](../core/assertions-waiting.md) — auto-waiting assertions
|
|
529
|
+
- [core/forms-validation.md](../testing-patterns/forms-validation.md) — form testing patterns
|
|
530
|
+
- [architecture/test-architecture.md](../architecture/test-architecture.md) — E2E vs unit tests with TestBed
|