@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.
Files changed (97) hide show
  1. package/.agents/skills/e2e-testing-expert/SKILL.md +28 -0
  2. package/.agents/skills/frontend-design/LICENSE.txt +177 -0
  3. package/.agents/skills/frontend-design/SKILL.md +42 -0
  4. package/.agents/skills/nodejs-backend-patterns/SKILL.md +639 -0
  5. package/.agents/skills/nodejs-backend-patterns/references/advanced-patterns.md +430 -0
  6. package/.agents/skills/playwright-best-practices/LICENSE.md +7 -0
  7. package/.agents/skills/playwright-best-practices/README.md +147 -0
  8. package/.agents/skills/playwright-best-practices/SKILL.md +303 -0
  9. package/.agents/skills/playwright-best-practices/advanced/authentication-flows.md +360 -0
  10. package/.agents/skills/playwright-best-practices/advanced/authentication.md +871 -0
  11. package/.agents/skills/playwright-best-practices/advanced/clock-mocking.md +364 -0
  12. package/.agents/skills/playwright-best-practices/advanced/mobile-testing.md +409 -0
  13. package/.agents/skills/playwright-best-practices/advanced/multi-context.md +288 -0
  14. package/.agents/skills/playwright-best-practices/advanced/multi-user.md +393 -0
  15. package/.agents/skills/playwright-best-practices/advanced/network-advanced.md +452 -0
  16. package/.agents/skills/playwright-best-practices/advanced/third-party.md +464 -0
  17. package/.agents/skills/playwright-best-practices/architecture/pom-vs-fixtures.md +363 -0
  18. package/.agents/skills/playwright-best-practices/architecture/test-architecture.md +369 -0
  19. package/.agents/skills/playwright-best-practices/architecture/when-to-mock.md +383 -0
  20. package/.agents/skills/playwright-best-practices/browser-apis/browser-apis.md +391 -0
  21. package/.agents/skills/playwright-best-practices/browser-apis/iframes.md +403 -0
  22. package/.agents/skills/playwright-best-practices/browser-apis/service-workers.md +504 -0
  23. package/.agents/skills/playwright-best-practices/browser-apis/websockets.md +403 -0
  24. package/.agents/skills/playwright-best-practices/core/annotations.md +424 -0
  25. package/.agents/skills/playwright-best-practices/core/assertions-waiting.md +361 -0
  26. package/.agents/skills/playwright-best-practices/core/configuration.md +452 -0
  27. package/.agents/skills/playwright-best-practices/core/fixtures-hooks.md +417 -0
  28. package/.agents/skills/playwright-best-practices/core/global-setup.md +434 -0
  29. package/.agents/skills/playwright-best-practices/core/locators.md +242 -0
  30. package/.agents/skills/playwright-best-practices/core/page-object-model.md +315 -0
  31. package/.agents/skills/playwright-best-practices/core/projects-dependencies.md +453 -0
  32. package/.agents/skills/playwright-best-practices/core/test-data.md +492 -0
  33. package/.agents/skills/playwright-best-practices/core/test-suite-structure.md +361 -0
  34. package/.agents/skills/playwright-best-practices/core/test-tags.md +298 -0
  35. package/.agents/skills/playwright-best-practices/debugging/console-errors.md +420 -0
  36. package/.agents/skills/playwright-best-practices/debugging/debugging.md +504 -0
  37. package/.agents/skills/playwright-best-practices/debugging/error-testing.md +360 -0
  38. package/.agents/skills/playwright-best-practices/debugging/flaky-tests.md +496 -0
  39. package/.agents/skills/playwright-best-practices/frameworks/angular.md +530 -0
  40. package/.agents/skills/playwright-best-practices/frameworks/nextjs.md +469 -0
  41. package/.agents/skills/playwright-best-practices/frameworks/react.md +531 -0
  42. package/.agents/skills/playwright-best-practices/frameworks/vue.md +574 -0
  43. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/ci-cd.md +468 -0
  44. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/docker.md +283 -0
  45. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/github-actions.md +546 -0
  46. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/gitlab.md +397 -0
  47. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/other-providers.md +521 -0
  48. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/parallel-sharding.md +371 -0
  49. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/performance.md +453 -0
  50. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/reporting.md +424 -0
  51. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/test-coverage.md +497 -0
  52. package/.agents/skills/playwright-best-practices/testing-patterns/accessibility.md +359 -0
  53. package/.agents/skills/playwright-best-practices/testing-patterns/api-testing.md +719 -0
  54. package/.agents/skills/playwright-best-practices/testing-patterns/browser-extensions.md +506 -0
  55. package/.agents/skills/playwright-best-practices/testing-patterns/canvas-webgl.md +493 -0
  56. package/.agents/skills/playwright-best-practices/testing-patterns/component-testing.md +500 -0
  57. package/.agents/skills/playwright-best-practices/testing-patterns/drag-drop.md +576 -0
  58. package/.agents/skills/playwright-best-practices/testing-patterns/electron.md +509 -0
  59. package/.agents/skills/playwright-best-practices/testing-patterns/file-operations.md +377 -0
  60. package/.agents/skills/playwright-best-practices/testing-patterns/file-upload-download.md +562 -0
  61. package/.agents/skills/playwright-best-practices/testing-patterns/forms-validation.md +561 -0
  62. package/.agents/skills/playwright-best-practices/testing-patterns/graphql-testing.md +331 -0
  63. package/.agents/skills/playwright-best-practices/testing-patterns/i18n.md +508 -0
  64. package/.agents/skills/playwright-best-practices/testing-patterns/performance-testing.md +476 -0
  65. package/.agents/skills/playwright-best-practices/testing-patterns/security-testing.md +430 -0
  66. package/.agents/skills/playwright-best-practices/testing-patterns/visual-regression.md +634 -0
  67. package/.env.example +21 -0
  68. package/README.md +30 -0
  69. package/bin/arcality.mjs +86 -0
  70. package/package.json +66 -0
  71. package/playwright.config.ts +12 -0
  72. package/scripts/cleanup-qmsdev.mjs +63 -0
  73. package/scripts/discover-view.mjs +52 -0
  74. package/scripts/extract-view.mjs +64 -0
  75. package/scripts/gen-and-run.mjs +838 -0
  76. package/scripts/init.mjs +290 -0
  77. package/scripts/migrate-to-central-out.mjs +157 -0
  78. package/scripts/postinstall.mjs +63 -0
  79. package/scripts/rebrand-report.mjs +241 -0
  80. package/scripts/setup.mjs +166 -0
  81. package/src/KnowledgeService.ts +239 -0
  82. package/src/arcalityClient.mjs +266 -0
  83. package/src/configLoader.mjs +179 -0
  84. package/src/configManager.mjs +172 -0
  85. package/src/consoleBanner.ts +32 -0
  86. package/src/envSetup.ts +205 -0
  87. package/src/index.ts +25 -0
  88. package/src/projectInspector.ts +42 -0
  89. package/src/services/collectiveMemoryService.ts +178 -0
  90. package/src/testRunner.ts +201 -0
  91. package/tests/_helpers/ArcalityReporter.ts +490 -0
  92. package/tests/_helpers/agentic-runner.spec.ts +741 -0
  93. package/tests/_helpers/ai-agent-helper.ts +1573 -0
  94. package/tests/_helpers/discover-view.spec.ts +238 -0
  95. package/tests/_helpers/extract-view.spec.ts +118 -0
  96. package/tests/_helpers/qa-tools.ts +333 -0
  97. 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