microui-wc 0.1.1 → 0.1.2
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.md +71 -71
- package/CHANGELOG.md +1 -1
- package/README.md +14 -9
- package/dist/AGENTS.md +71 -71
- package/dist/README.md +14 -9
- package/dist/microui.css +1 -1
- package/dist/microui.esm.js.map +1 -1
- package/dist/microui.min.js.map +1 -1
- package/docs/getting-started.md +3 -3
- package/package.json +39 -11
- package/src/components/mu-schema-form.js +1 -1
- package/.github/ISSUE_TEMPLATE/bug_report.md +0 -40
- package/.github/ISSUE_TEMPLATE/feature_request.md +0 -33
- package/.github/PULL_REQUEST_TEMPLATE.md +0 -28
- package/.github/workflows/ci.yml +0 -42
- package/.github/workflows/deploy-pages.yml +0 -112
- package/CODE_OF_CONDUCT.md +0 -59
- package/CONTRIBUTING.md +0 -156
- package/SECURITY.md +0 -58
- package/app/.generated/routes/alerts.js +0 -8
- package/app/.generated/routes/avatars.js +0 -8
- package/app/.generated/routes/badges.js +0 -8
- package/app/.generated/routes/buttons.js +0 -10
- package/app/.generated/routes/cards.js +0 -10
- package/app/.generated/routes/checkboxes.js +0 -9
- package/app/.generated/routes/chips.js +0 -8
- package/app/.generated/routes/dropdowns.js +0 -9
- package/app/.generated/routes/home.js +0 -7
- package/app/.generated/routes/icons.js +0 -9
- package/app/.generated/routes/inputs.js +0 -10
- package/app/.generated/routes/installation.js +0 -7
- package/app/.generated/routes/layout.js +0 -9
- package/app/.generated/routes/modals.js +0 -9
- package/app/.generated/routes/navbar.js +0 -7
- package/app/.generated/routes/progress.js +0 -9
- package/app/.generated/routes/radios.js +0 -9
- package/app/.generated/routes/switches.js +0 -9
- package/app/.generated/routes/tabs.js +0 -8
- package/app/.generated/routes/toasts.js +0 -9
- package/app/index.html +0 -67
- package/app/pages/alerts.html +0 -23
- package/app/pages/avatars.html +0 -22
- package/app/pages/badges.html +0 -22
- package/app/pages/buttons.html +0 -71
- package/app/pages/cards.html +0 -54
- package/app/pages/checkboxes.html +0 -39
- package/app/pages/chips.html +0 -23
- package/app/pages/dropdowns.html +0 -41
- package/app/pages/home.html +0 -59
- package/app/pages/icons.html +0 -29
- package/app/pages/inputs.html +0 -66
- package/app/pages/installation.html +0 -34
- package/app/pages/layout.html +0 -30
- package/app/pages/modals.html +0 -21
- package/app/pages/navbar.html +0 -22
- package/app/pages/progress.html +0 -35
- package/app/pages/radios.html +0 -40
- package/app/pages/switches.html +0 -39
- package/app/pages/tabs.html +0 -30
- package/app/pages/toasts.html +0 -22
- package/app-dist/index.html +0 -67
- package/app-dist/pages/alerts.html +0 -23
- package/app-dist/pages/avatars.html +0 -22
- package/app-dist/pages/badges.html +0 -22
- package/app-dist/pages/buttons.html +0 -71
- package/app-dist/pages/cards.html +0 -54
- package/app-dist/pages/checkboxes.html +0 -39
- package/app-dist/pages/chips.html +0 -23
- package/app-dist/pages/dropdowns.html +0 -41
- package/app-dist/pages/home.html +0 -59
- package/app-dist/pages/icons.html +0 -29
- package/app-dist/pages/inputs.html +0 -66
- package/app-dist/pages/installation.html +0 -34
- package/app-dist/pages/layout.html +0 -30
- package/app-dist/pages/modals.html +0 -21
- package/app-dist/pages/navbar.html +0 -22
- package/app-dist/pages/progress.html +0 -35
- package/app-dist/pages/radios.html +0 -40
- package/app-dist/pages/switches.html +0 -39
- package/app-dist/pages/tabs.html +0 -30
- package/app-dist/pages/toasts.html +0 -22
- package/app-dist/pages.json +0 -217
- package/app-dist/routes/alerts.js +0 -5
- package/app-dist/routes/avatars.js +0 -1
- package/app-dist/routes/badges.js +0 -1
- package/app-dist/routes/buttons.js +0 -1
- package/app-dist/routes/cards.js +0 -1
- package/app-dist/routes/checkboxes.js +0 -9
- package/app-dist/routes/chips.js +0 -4
- package/app-dist/routes/chunk-019e5e2f.js +0 -5
- package/app-dist/routes/chunk-0m4j19yd.js +0 -2
- package/app-dist/routes/chunk-0tmmp5q0.js +0 -1
- package/app-dist/routes/chunk-10xn709r.js +0 -1
- package/app-dist/routes/chunk-15m2qcda.js +0 -2
- package/app-dist/routes/chunk-1bh8g23n.js +0 -1
- package/app-dist/routes/chunk-1vg0v937.js +0 -1
- package/app-dist/routes/chunk-1zvcgy3j.js +0 -1
- package/app-dist/routes/chunk-2afb0861.js +0 -1
- package/app-dist/routes/chunk-2c6ttpzt.js +0 -5
- package/app-dist/routes/chunk-3dy30fhs.js +0 -1
- package/app-dist/routes/chunk-426dnces.js +0 -13
- package/app-dist/routes/chunk-44kgxery.js +0 -1
- package/app-dist/routes/chunk-47fdnejd.js +0 -33
- package/app-dist/routes/chunk-49a6t2vq.js +0 -1
- package/app-dist/routes/chunk-4fe1rm5b.js +0 -1
- package/app-dist/routes/chunk-4ggmvkta.js +0 -33
- package/app-dist/routes/chunk-4vkz81q7.js +0 -33
- package/app-dist/routes/chunk-4w4tmj8f.js +0 -31
- package/app-dist/routes/chunk-532s62kr.js +0 -31
- package/app-dist/routes/chunk-5hm3bssy.js +0 -33
- package/app-dist/routes/chunk-5vrh24hc.js +0 -1
- package/app-dist/routes/chunk-61pcg25a.js +0 -1
- package/app-dist/routes/chunk-6nfhygvf.js +0 -1
- package/app-dist/routes/chunk-700e7je6.js +0 -33
- package/app-dist/routes/chunk-7fsn17kg.js +0 -1
- package/app-dist/routes/chunk-7k789b32.js +0 -1
- package/app-dist/routes/chunk-7r46q0ys.js +0 -36
- package/app-dist/routes/chunk-86fmc1fr.js +0 -5
- package/app-dist/routes/chunk-8qth37vw.js +0 -1
- package/app-dist/routes/chunk-924wv8n0.js +0 -1
- package/app-dist/routes/chunk-9mbhgxk9.js +0 -1
- package/app-dist/routes/chunk-a216hyd9.js +0 -1
- package/app-dist/routes/chunk-akzxykh9.js +0 -33
- package/app-dist/routes/chunk-b3dcvy8c.js +0 -1
- package/app-dist/routes/chunk-b74zahz5.js +0 -31
- package/app-dist/routes/chunk-bftj53p2.js +0 -5
- package/app-dist/routes/chunk-c01hnz3e.js +0 -1
- package/app-dist/routes/chunk-d8pvv5km.js +0 -1
- package/app-dist/routes/chunk-dev0aezr.js +0 -2
- package/app-dist/routes/chunk-dh6vnv0e.js +0 -1
- package/app-dist/routes/chunk-dn2cbpva.js +0 -36
- package/app-dist/routes/chunk-dvn0my90.js +0 -1
- package/app-dist/routes/chunk-dvq8mnve.js +0 -36
- package/app-dist/routes/chunk-e8c2gc4d.js +0 -5
- package/app-dist/routes/chunk-ejf9ak2x.js +0 -1
- package/app-dist/routes/chunk-f083m55s.js +0 -1
- package/app-dist/routes/chunk-fnrj28s1.js +0 -31
- package/app-dist/routes/chunk-fvg3yjdp.js +0 -31
- package/app-dist/routes/chunk-g7k381n1.js +0 -1
- package/app-dist/routes/chunk-h01kq2ae.js +0 -13
- package/app-dist/routes/chunk-h4dk761v.js +0 -5
- package/app-dist/routes/chunk-hmx91z2x.js +0 -5
- package/app-dist/routes/chunk-hxbg4m42.js +0 -36
- package/app-dist/routes/chunk-jbjnfp2b.js +0 -2
- package/app-dist/routes/chunk-jxtz5vv6.js +0 -36
- package/app-dist/routes/chunk-jxzcs0ey.js +0 -36
- package/app-dist/routes/chunk-kt7wwhcx.js +0 -1
- package/app-dist/routes/chunk-kzptszyc.js +0 -33
- package/app-dist/routes/chunk-mhgca4w4.js +0 -2
- package/app-dist/routes/chunk-mhswxa20.js +0 -1
- package/app-dist/routes/chunk-n8zfeex6.js +0 -1
- package/app-dist/routes/chunk-pee47b2r.js +0 -1
- package/app-dist/routes/chunk-pesmw829.js +0 -1
- package/app-dist/routes/chunk-pgc4c6f3.js +0 -36
- package/app-dist/routes/chunk-q8egegm1.js +0 -1
- package/app-dist/routes/chunk-q9mn2qyq.js +0 -36
- package/app-dist/routes/chunk-qh0rtaf3.js +0 -5
- package/app-dist/routes/chunk-qqhmk6ye.js +0 -2
- package/app-dist/routes/chunk-qrxygmf7.js +0 -33
- package/app-dist/routes/chunk-r46yzksx.js +0 -36
- package/app-dist/routes/chunk-rgpbw2w0.js +0 -5
- package/app-dist/routes/chunk-rnpzv3d8.js +0 -2
- package/app-dist/routes/chunk-s5v8cv05.js +0 -2
- package/app-dist/routes/chunk-sbwn5bpc.js +0 -1
- package/app-dist/routes/chunk-sqbg8jbt.js +0 -33
- package/app-dist/routes/chunk-sv8dqnf7.js +0 -1
- package/app-dist/routes/chunk-t67sw3za.js +0 -1
- package/app-dist/routes/chunk-tjdpqwdf.js +0 -31
- package/app-dist/routes/chunk-tq2mfghg.js +0 -1
- package/app-dist/routes/chunk-ttn10vt6.js +0 -1
- package/app-dist/routes/chunk-v2hzpjxr.js +0 -1
- package/app-dist/routes/chunk-wfjjkw9y.js +0 -1
- package/app-dist/routes/chunk-wt8cxzmf.js +0 -31
- package/app-dist/routes/chunk-x45d372k.js +0 -5
- package/app-dist/routes/chunk-y3wsazkt.js +0 -1
- package/app-dist/routes/chunk-y7pmgc7t.js +0 -33
- package/app-dist/routes/chunk-zefdt2q3.js +0 -31
- package/app-dist/routes/dropdowns.js +0 -6
- package/app-dist/routes/home.js +0 -1
- package/app-dist/routes/icons.js +0 -1
- package/app-dist/routes/inputs.js +0 -12
- package/app-dist/routes/installation.js +0 -1
- package/app-dist/routes/layout.js +0 -1
- package/app-dist/routes/modals.js +0 -7
- package/app-dist/routes/navbar.js +0 -1
- package/app-dist/routes/progress.js +0 -1
- package/app-dist/routes/radios.js +0 -6
- package/app-dist/routes/switches.js +0 -6
- package/app-dist/routes/tabs.js +0 -1
- package/app-dist/routes/toasts.js +0 -16
- package/assets/fonts/material-symbols-mini.woff2 +0 -0
- package/assets/fonts/material-symbols.woff2 +0 -0
- package/assets/fonts/roboto-400.woff2 +0 -0
- package/assets/fonts/roboto-500.woff2 +0 -0
- package/assets/fonts/roboto-700.woff2 +0 -0
- package/assets/logo-banner-400.jpg +0 -0
- package/assets/logo-banner-400.webp +0 -0
- package/assets/logo-banner-800.webp +0 -0
- package/assets/logo-banner.jpg +0 -0
- package/assets/logo-icon-64.jpg +0 -0
- package/assets/logo-icon-64.webp +0 -0
- package/assets/logo-icon.jpg +0 -0
- package/assets/logo-square.jpg +0 -0
- package/bun.lock +0 -312
- package/bunfig.toml +0 -4
- package/custom-elements.json +0 -1916
- package/demo/api/sample-data.json +0 -38
- package/demo/content/alerts.html +0 -115
- package/demo/content/avatars.html +0 -70
- package/demo/content/badges.html +0 -65
- package/demo/content/buttons.html +0 -188
- package/demo/content/callouts.html +0 -91
- package/demo/content/cards.html +0 -121
- package/demo/content/checkboxes.html +0 -178
- package/demo/content/chips.html +0 -67
- package/demo/content/codeblocks.html +0 -101
- package/demo/content/confirms.html +0 -115
- package/demo/content/datatables.html +0 -149
- package/demo/content/dividers.html +0 -119
- package/demo/content/dropdowns.html +0 -89
- package/demo/content/enterprise.html +0 -252
- package/demo/content/home.html +0 -149
- package/demo/content/icons.html +0 -89
- package/demo/content/inputs.html +0 -135
- package/demo/content/installation.html +0 -16
- package/demo/content/layout.html +0 -136
- package/demo/content/modals.html +0 -141
- package/demo/content/navbar.html +0 -70
- package/demo/content/progress.html +0 -119
- package/demo/content/radios.html +0 -88
- package/demo/content/skeletons.html +0 -109
- package/demo/content/spinners.html +0 -96
- package/demo/content/switches.html +0 -84
- package/demo/content/tables.html +0 -124
- package/demo/content/tabs.html +0 -85
- package/demo/content/toasts.html +0 -116
- package/demo/content/tooltips.html +0 -107
- package/demo/content/virtual-lists.html +0 -233
- package/demo/favicon.ico +0 -0
- package/demo/favicon.png +0 -0
- package/demo/full.html +0 -52
- package/demo/iife.html +0 -46
- package/demo/manifest.json +0 -34
- package/demo/pages/datatable-demo.html +0 -237
- package/demo/pages/prompt-ui-demo.html +0 -218
- package/demo/pages/responsive-demo.html +0 -122
- package/demo/pages/schema-form-demo.html +0 -270
- package/demo/robots.txt +0 -6
- package/demo/shell.html +0 -712
- package/demo/sw.js +0 -387
- package/lighthouse-audit.mjs +0 -113
- package/scripts/analyze-components.js +0 -105
- package/scripts/build-app.js +0 -193
- package/scripts/build-framework.js +0 -444
- package/scripts/build-utils.js +0 -101
- package/scripts/test-isolated.js +0 -151
- package/server.js +0 -256
- package/tests/agents/agent-integration.test.js +0 -76
- package/tests/benchmark.html +0 -296
- package/tests/build/scan-components.test.js +0 -173
- package/tests/components/all-components.test.js +0 -245
- package/tests/components/all-missing-components.test.js +0 -574
- package/tests/components/mu-alert.test.js +0 -113
- package/tests/components/mu-avatar.test.js +0 -148
- package/tests/components/mu-badge.test.js +0 -92
- package/tests/components/mu-button.test.js +0 -112
- package/tests/components/mu-card.test.js +0 -89
- package/tests/components/mu-checkbox.test.js +0 -158
- package/tests/components/mu-chip.test.js +0 -118
- package/tests/components/mu-container.test.js +0 -120
- package/tests/components/mu-divider.test.js +0 -98
- package/tests/components/mu-drawer-item.test.js +0 -199
- package/tests/components/mu-drawer.test.js +0 -96
- package/tests/components/mu-dropdown.test.js +0 -125
- package/tests/components/mu-form.test.js +0 -138
- package/tests/components/mu-grid.test.js +0 -135
- package/tests/components/mu-icon.test.js +0 -110
- package/tests/components/mu-input.test.js +0 -131
- package/tests/components/mu-lazy.test.js +0 -103
- package/tests/components/mu-modal.test.js +0 -275
- package/tests/components/mu-navbar.test.js +0 -101
- package/tests/components/mu-progress.test.js +0 -115
- package/tests/components/mu-radio.test.js +0 -114
- package/tests/components/mu-repeat.test.js +0 -106
- package/tests/components/mu-sidebar.test.js +0 -126
- package/tests/components/mu-skeleton.test.js +0 -162
- package/tests/components/mu-stack.test.js +0 -143
- package/tests/components/mu-switch.test.js +0 -292
- package/tests/components/mu-table.test.js +0 -124
- package/tests/components/mu-tabs.test.js +0 -104
- package/tests/components/mu-textarea.test.js +0 -115
- package/tests/components/mu-toast.test.js +0 -321
- package/tests/components/mu-tooltip.test.js +0 -133
- package/tests/components/mu-virtual-list.test.js +0 -109
- package/tests/core/MuElement.test.js +0 -120
- package/tests/core/agent-api.test.js +0 -125
- package/tests/core/all-core-modules.test.js +0 -442
- package/tests/core/bus.test.js +0 -364
- package/tests/core/component-schema.test.js +0 -160
- package/tests/core/feature-registry.test.js +0 -198
- package/tests/core/form-state.test.js +0 -167
- package/tests/core/http.test.js +0 -119
- package/tests/core/keyboard.test.js +0 -319
- package/tests/core/layers.test.js +0 -129
- package/tests/core/namespaced-stores.test.js +0 -114
- package/tests/core/render.test.js +0 -121
- package/tests/core/ripple.test.js +0 -131
- package/tests/core/router.test.js +0 -89
- package/tests/core/scheduler.test.js +0 -121
- package/tests/core/signals.test.js +0 -128
- package/tests/core/store.test.js +0 -171
- package/tests/core/transitions.test.js +0 -82
- package/tests/e2e/accessibility-harness.html +0 -58
- package/tests/e2e/accessibility.test.js +0 -401
- package/tests/e2e/agent-features.test.js +0 -372
- package/tests/e2e/card-spacing.test.js +0 -287
- package/tests/e2e/components.test.js +0 -439
- package/tests/e2e/demo-routes.test.js +0 -478
- package/tests/e2e/layout-css-fallback.test.js +0 -334
- package/tests/e2e/mu-alert.e2e.test.js +0 -111
- package/tests/e2e/mu-checkbox.test.js +0 -489
- package/tests/e2e/mu-chip.test.js +0 -347
- package/tests/e2e/mu-form.test.js +0 -499
- package/tests/e2e/mu-icon.test.js +0 -114
- package/tests/e2e/mu-radio.test.js +0 -113
- package/tests/e2e/mu-skeleton.test.js +0 -140
- package/tests/e2e/mu-switch.test.js +0 -415
- package/tests/e2e/mu-tabs.test.js +0 -494
- package/tests/e2e/mu-textarea.test.js +0 -242
- package/tests/e2e/mu-virtual-list.test.js +0 -427
- package/tests/e2e/perf-memory.test.js +0 -161
- package/tests/e2e/puppeteer-helper.js +0 -137
- package/tests/e2e/puppeteer.test.js +0 -226
- package/tests/e2e/pwa.test.js +0 -261
- package/tests/e2e/test-harness.html +0 -319
- package/tests/manual/test-components.html +0 -120
- package/tests/memory-test.html +0 -309
- package/tests/setup-dom.js +0 -93
- package/tests/visual-test.html +0 -301
|
@@ -1,478 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview E2E Tests for index-ultra.html demo routes
|
|
3
|
-
*
|
|
4
|
-
* Tests all 20 routes to verify:
|
|
5
|
-
* 1. Route loads without errors
|
|
6
|
-
* 2. Components render correctly
|
|
7
|
-
* 3. Navigation works
|
|
8
|
-
* 4. No console errors
|
|
9
|
-
*/
|
|
10
|
-
|
|
11
|
-
import { describe, test, expect, beforeAll, afterAll } from 'bun:test';
|
|
12
|
-
// Set cache dir BEFORE importing puppeteer (it reads env at import time)
|
|
13
|
-
process.env.PUPPETEER_CACHE_DIR = '/tmp/puppeteer';
|
|
14
|
-
import { launchBrowser, puppeteer } from './puppeteer-helper.js';
|
|
15
|
-
import { fileURLToPath } from 'url';
|
|
16
|
-
import { dirname, join } from 'path';
|
|
17
|
-
|
|
18
|
-
const __filename = fileURLToPath(import.meta.url);
|
|
19
|
-
const __dirname = dirname(__filename);
|
|
20
|
-
|
|
21
|
-
// Helper to replace deprecated page.waitForTimeout
|
|
22
|
-
const delay = (ms) => new Promise(resolve => setTimeout(resolve, ms));
|
|
23
|
-
|
|
24
|
-
// All routes from index-ultra.html
|
|
25
|
-
const DEMO_ROUTES = [
|
|
26
|
-
{ hash: 'home', title: 'Home', component: null },
|
|
27
|
-
{ hash: 'installation', title: 'Installation', component: null },
|
|
28
|
-
{ hash: 'buttons', title: 'Buttons', component: 'mu-button' },
|
|
29
|
-
{ hash: 'inputs', title: 'Inputs', component: 'mu-input' },
|
|
30
|
-
{ hash: 'cards', title: 'Cards', component: 'mu-card' },
|
|
31
|
-
{ hash: 'checkboxes', title: 'Checkboxes', component: 'mu-checkbox' },
|
|
32
|
-
{ hash: 'switches', title: 'Switches', component: 'mu-switch' },
|
|
33
|
-
{ hash: 'radios', title: 'Radios', component: 'mu-radio' },
|
|
34
|
-
{ hash: 'dropdowns', title: 'Dropdowns', component: 'mu-dropdown' },
|
|
35
|
-
{ hash: 'tabs', title: 'Tabs', component: 'mu-tabs' },
|
|
36
|
-
{ hash: 'modals', title: 'Modals', component: 'mu-modal' },
|
|
37
|
-
{ hash: 'alerts', title: 'Alerts', component: 'mu-alert' },
|
|
38
|
-
{ hash: 'toasts', title: 'Toasts', component: 'mu-button' }, // Toast page shows buttons to trigger toasts
|
|
39
|
-
{ hash: 'progress', title: 'Progress', component: 'mu-progress' },
|
|
40
|
-
{ hash: 'layout', title: 'Layout', component: null }, // Layout page uses CSS grid utilities
|
|
41
|
-
{ hash: 'navbar', title: 'Navbar', component: 'mu-navbar' },
|
|
42
|
-
{ hash: 'avatars', title: 'Avatars', component: 'mu-avatar' },
|
|
43
|
-
{ hash: 'badges', title: 'Badges', component: 'mu-badge' },
|
|
44
|
-
{ hash: 'chips', title: 'Chips', component: 'mu-chip' },
|
|
45
|
-
{ hash: 'icons', title: 'Icons', component: 'mu-icon' }
|
|
46
|
-
];
|
|
47
|
-
|
|
48
|
-
describe('Demo Routes E2E Tests', () => {
|
|
49
|
-
let browser;
|
|
50
|
-
let page;
|
|
51
|
-
let server;
|
|
52
|
-
const consoleErrors = [];
|
|
53
|
-
|
|
54
|
-
beforeAll(async () => {
|
|
55
|
-
const projectRoot = join(__dirname, '../..');
|
|
56
|
-
|
|
57
|
-
// Start HTTP server with proper MIME types
|
|
58
|
-
server = Bun.serve({
|
|
59
|
-
port: 0,
|
|
60
|
-
async fetch(req) {
|
|
61
|
-
const url = new URL(req.url);
|
|
62
|
-
let filePath = join(projectRoot, url.pathname);
|
|
63
|
-
|
|
64
|
-
// Default to index-ultra.html
|
|
65
|
-
if (url.pathname === '/' || url.pathname === '/index-ultra.html') {
|
|
66
|
-
filePath = join(projectRoot, 'index-ultra.html');
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
try {
|
|
70
|
-
const file = Bun.file(filePath);
|
|
71
|
-
const exists = await file.exists();
|
|
72
|
-
if (!exists) {
|
|
73
|
-
return new Response('Not Found', { status: 404 });
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
const content = await file.arrayBuffer();
|
|
77
|
-
const ext = filePath.split('.').pop();
|
|
78
|
-
const mimeTypes = {
|
|
79
|
-
'html': 'text/html',
|
|
80
|
-
'js': 'text/javascript',
|
|
81
|
-
'css': 'text/css',
|
|
82
|
-
'json': 'application/json',
|
|
83
|
-
'svg': 'image/svg+xml',
|
|
84
|
-
'png': 'image/png',
|
|
85
|
-
'ico': 'image/x-icon'
|
|
86
|
-
};
|
|
87
|
-
|
|
88
|
-
return new Response(content, {
|
|
89
|
-
headers: { 'Content-Type': mimeTypes[ext] || 'application/octet-stream' }
|
|
90
|
-
});
|
|
91
|
-
} catch (e) {
|
|
92
|
-
return new Response('Error: ' + e.message, { status: 500 });
|
|
93
|
-
}
|
|
94
|
-
}
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
console.log(`Test server started at http://localhost:${server.port}`);
|
|
98
|
-
|
|
99
|
-
browser = await puppeteer.launch({
|
|
100
|
-
headless: 'shell',
|
|
101
|
-
executablePath: '/tmp/puppeteer/chrome-headless-shell/mac_arm-144.0.7559.96/chrome-headless-shell-mac-arm64/chrome-headless-shell',
|
|
102
|
-
args: ['--no-sandbox', '--disable-setuid-sandbox'],
|
|
103
|
-
userDataDir: `/tmp/puppeteer-demo_routes-${Date.now()}-${Math.random().toString(36).slice(2)}`
|
|
104
|
-
});
|
|
105
|
-
page = await browser.newPage();
|
|
106
|
-
|
|
107
|
-
// Capture console errors
|
|
108
|
-
page.on('pageerror', err => {
|
|
109
|
-
consoleErrors.push(err.message);
|
|
110
|
-
});
|
|
111
|
-
|
|
112
|
-
// Navigate to demo page
|
|
113
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html`, {
|
|
114
|
-
waitUntil: 'networkidle0',
|
|
115
|
-
timeout: 30000
|
|
116
|
-
});
|
|
117
|
-
}, 60000);
|
|
118
|
-
|
|
119
|
-
afterAll(async () => {
|
|
120
|
-
if (browser) await browser.close();
|
|
121
|
-
if (server) server.stop();
|
|
122
|
-
});
|
|
123
|
-
|
|
124
|
-
// ========================================
|
|
125
|
-
// ROUTE LOADING TESTS
|
|
126
|
-
// ========================================
|
|
127
|
-
|
|
128
|
-
describe('Route Loading', () => {
|
|
129
|
-
for (const route of DEMO_ROUTES) {
|
|
130
|
-
test(`#${route.hash} should load successfully`, async () => {
|
|
131
|
-
// Navigate to route
|
|
132
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#${route.hash}`, {
|
|
133
|
-
waitUntil: 'networkidle0'
|
|
134
|
-
});
|
|
135
|
-
|
|
136
|
-
// Wait for router to process
|
|
137
|
-
await delay(300);
|
|
138
|
-
|
|
139
|
-
// Check that content area has content
|
|
140
|
-
const hasContent = await page.evaluate(() => {
|
|
141
|
-
const main = document.querySelector('.main-content, main, [role="main"]');
|
|
142
|
-
return main ? main.innerHTML.length > 50 : false;
|
|
143
|
-
});
|
|
144
|
-
|
|
145
|
-
expect(hasContent).toBe(true);
|
|
146
|
-
}, 15000);
|
|
147
|
-
}
|
|
148
|
-
});
|
|
149
|
-
|
|
150
|
-
// ========================================
|
|
151
|
-
// COMPONENT RENDERING TESTS
|
|
152
|
-
// ========================================
|
|
153
|
-
|
|
154
|
-
describe('Component Rendering', () => {
|
|
155
|
-
for (const route of DEMO_ROUTES.filter(r => r.component)) {
|
|
156
|
-
test(`#${route.hash} should render ${route.component}`, async () => {
|
|
157
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#${route.hash}`, {
|
|
158
|
-
waitUntil: 'networkidle0'
|
|
159
|
-
});
|
|
160
|
-
|
|
161
|
-
await delay(500);
|
|
162
|
-
|
|
163
|
-
// Check component exists
|
|
164
|
-
const componentExists = await page.evaluate((tag) => {
|
|
165
|
-
return document.querySelectorAll(tag).length > 0;
|
|
166
|
-
}, route.component);
|
|
167
|
-
|
|
168
|
-
expect(componentExists).toBe(true);
|
|
169
|
-
}, 15000);
|
|
170
|
-
}
|
|
171
|
-
});
|
|
172
|
-
|
|
173
|
-
// ========================================
|
|
174
|
-
// NAVIGATION TESTS
|
|
175
|
-
// ========================================
|
|
176
|
-
|
|
177
|
-
describe('Navigation', () => {
|
|
178
|
-
test('sidebar navigation should work', async () => {
|
|
179
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#home`, {
|
|
180
|
-
waitUntil: 'networkidle0'
|
|
181
|
-
});
|
|
182
|
-
|
|
183
|
-
// Click on Buttons nav item
|
|
184
|
-
const clicked = await page.evaluate(() => {
|
|
185
|
-
const navItems = document.querySelectorAll('.sidebar a, mu-nav-item, [href="#buttons"]');
|
|
186
|
-
for (const item of navItems) {
|
|
187
|
-
if (item.textContent.includes('Buttons') || item.href?.includes('#buttons')) {
|
|
188
|
-
item.click();
|
|
189
|
-
return true;
|
|
190
|
-
}
|
|
191
|
-
}
|
|
192
|
-
return false;
|
|
193
|
-
});
|
|
194
|
-
|
|
195
|
-
if (clicked) {
|
|
196
|
-
await delay(500);
|
|
197
|
-
const hash = await page.evaluate(() => window.location.hash);
|
|
198
|
-
expect(hash).toBe('#buttons');
|
|
199
|
-
}
|
|
200
|
-
}, 15000);
|
|
201
|
-
|
|
202
|
-
test('back/forward navigation should work', async () => {
|
|
203
|
-
// Navigate to two pages
|
|
204
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#buttons`, {
|
|
205
|
-
waitUntil: 'networkidle0'
|
|
206
|
-
});
|
|
207
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#cards`, {
|
|
208
|
-
waitUntil: 'networkidle0'
|
|
209
|
-
});
|
|
210
|
-
|
|
211
|
-
// Go back
|
|
212
|
-
await page.goBack();
|
|
213
|
-
await delay(300);
|
|
214
|
-
|
|
215
|
-
const hash = await page.evaluate(() => window.location.hash);
|
|
216
|
-
expect(hash).toBe('#buttons');
|
|
217
|
-
}, 15000);
|
|
218
|
-
|
|
219
|
-
// REGRESSION TEST: Direct URL hash should select matching drawer item
|
|
220
|
-
// Bug: Opening http://localhost:5001/#buttons showed "Overview" selected instead of "Buttons"
|
|
221
|
-
test('direct hash URL should select correct drawer item', async () => {
|
|
222
|
-
// Open directly with hash (simulating user sharing a link)
|
|
223
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#buttons`, {
|
|
224
|
-
waitUntil: 'networkidle0'
|
|
225
|
-
});
|
|
226
|
-
await delay(500);
|
|
227
|
-
|
|
228
|
-
// Verify the correct drawer item has the active attribute
|
|
229
|
-
const activeItems = await page.evaluate(() => {
|
|
230
|
-
const items = document.querySelectorAll('mu-drawer-item');
|
|
231
|
-
return Array.from(items)
|
|
232
|
-
.filter(item => item.hasAttribute('active'))
|
|
233
|
-
.map(item => ({
|
|
234
|
-
dataPage: item.getAttribute('data-page'),
|
|
235
|
-
label: item.getAttribute('label')
|
|
236
|
-
}));
|
|
237
|
-
});
|
|
238
|
-
|
|
239
|
-
// Should have exactly one active item
|
|
240
|
-
expect(activeItems.length).toBe(1);
|
|
241
|
-
// The active item should be "buttons", not "home" (Overview)
|
|
242
|
-
expect(activeItems[0].dataPage).toBe('buttons');
|
|
243
|
-
}, 15000);
|
|
244
|
-
|
|
245
|
-
test('sidebar should update on hash change', async () => {
|
|
246
|
-
// Start on home
|
|
247
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#home`, {
|
|
248
|
-
waitUntil: 'networkidle0'
|
|
249
|
-
});
|
|
250
|
-
await delay(300);
|
|
251
|
-
|
|
252
|
-
// Navigate to a different page via hash
|
|
253
|
-
await page.evaluate(() => { window.location.hash = 'cards'; });
|
|
254
|
-
await delay(500);
|
|
255
|
-
|
|
256
|
-
// Verify drawer item updated
|
|
257
|
-
const activeItems = await page.evaluate(() => {
|
|
258
|
-
return Array.from(document.querySelectorAll('mu-drawer-item'))
|
|
259
|
-
.filter(item => item.hasAttribute('active'))
|
|
260
|
-
.map(item => item.getAttribute('data-page'));
|
|
261
|
-
});
|
|
262
|
-
|
|
263
|
-
expect(activeItems).toContain('cards');
|
|
264
|
-
expect(activeItems).not.toContain('home');
|
|
265
|
-
}, 15000);
|
|
266
|
-
});
|
|
267
|
-
|
|
268
|
-
// ========================================
|
|
269
|
-
// COMPONENT INTERACTION TESTS
|
|
270
|
-
// ========================================
|
|
271
|
-
|
|
272
|
-
describe('Component Interactions', () => {
|
|
273
|
-
test('mu-button click should work', async () => {
|
|
274
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#buttons`, {
|
|
275
|
-
waitUntil: 'networkidle0'
|
|
276
|
-
});
|
|
277
|
-
await delay(500);
|
|
278
|
-
|
|
279
|
-
const clicked = await page.evaluate(() => {
|
|
280
|
-
const btn = document.querySelector('mu-button');
|
|
281
|
-
if (btn) {
|
|
282
|
-
btn.click();
|
|
283
|
-
return true;
|
|
284
|
-
}
|
|
285
|
-
return false;
|
|
286
|
-
});
|
|
287
|
-
|
|
288
|
-
expect(clicked).toBe(true);
|
|
289
|
-
}, 15000);
|
|
290
|
-
|
|
291
|
-
test('mu-checkbox should toggle', async () => {
|
|
292
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#checkboxes`, {
|
|
293
|
-
waitUntil: 'networkidle0'
|
|
294
|
-
});
|
|
295
|
-
await delay(500);
|
|
296
|
-
|
|
297
|
-
const toggled = await page.evaluate(() => {
|
|
298
|
-
const cb = document.querySelector('mu-checkbox');
|
|
299
|
-
if (cb) {
|
|
300
|
-
const before = cb.hasAttribute('checked');
|
|
301
|
-
cb.click();
|
|
302
|
-
const after = cb.hasAttribute('checked');
|
|
303
|
-
return before !== after;
|
|
304
|
-
}
|
|
305
|
-
return false;
|
|
306
|
-
});
|
|
307
|
-
|
|
308
|
-
expect(toggled).toBe(true);
|
|
309
|
-
}, 15000);
|
|
310
|
-
|
|
311
|
-
test('mu-switch should toggle', async () => {
|
|
312
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#switches`, {
|
|
313
|
-
waitUntil: 'networkidle0'
|
|
314
|
-
});
|
|
315
|
-
await delay(500);
|
|
316
|
-
|
|
317
|
-
const toggled = await page.evaluate(() => {
|
|
318
|
-
const sw = document.querySelector('mu-switch');
|
|
319
|
-
if (sw) {
|
|
320
|
-
const before = sw.hasAttribute('checked');
|
|
321
|
-
sw.click();
|
|
322
|
-
const after = sw.hasAttribute('checked');
|
|
323
|
-
return before !== after;
|
|
324
|
-
}
|
|
325
|
-
return false;
|
|
326
|
-
});
|
|
327
|
-
|
|
328
|
-
expect(toggled).toBe(true);
|
|
329
|
-
}, 15000);
|
|
330
|
-
|
|
331
|
-
test('mu-modal should open and close', async () => {
|
|
332
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#modals`, {
|
|
333
|
-
waitUntil: 'networkidle0'
|
|
334
|
-
});
|
|
335
|
-
await delay(500);
|
|
336
|
-
|
|
337
|
-
// Note: modal.close() is async with ~200ms animation delay
|
|
338
|
-
const result = await page.evaluate(async () => {
|
|
339
|
-
const modal = document.querySelector('mu-modal');
|
|
340
|
-
if (modal && typeof modal.open === 'function') {
|
|
341
|
-
modal.open();
|
|
342
|
-
const wasOpen = modal.hasAttribute('open');
|
|
343
|
-
modal.close();
|
|
344
|
-
// Wait for close animation (200ms internal timeout)
|
|
345
|
-
await new Promise(r => setTimeout(r, 300));
|
|
346
|
-
const wasClosed = !modal.hasAttribute('open');
|
|
347
|
-
return { opened: wasOpen, closed: wasClosed };
|
|
348
|
-
}
|
|
349
|
-
return { opened: false, closed: false };
|
|
350
|
-
});
|
|
351
|
-
|
|
352
|
-
expect(result.opened).toBe(true);
|
|
353
|
-
expect(result.closed).toBe(true);
|
|
354
|
-
}, 15000);
|
|
355
|
-
|
|
356
|
-
test('mu-tabs should switch panels', async () => {
|
|
357
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#tabs`, {
|
|
358
|
-
waitUntil: 'networkidle0'
|
|
359
|
-
});
|
|
360
|
-
await delay(500);
|
|
361
|
-
|
|
362
|
-
const switched = await page.evaluate(() => {
|
|
363
|
-
const tabs = document.querySelector('mu-tabs');
|
|
364
|
-
const tabItems = document.querySelectorAll('mu-tab');
|
|
365
|
-
if (tabs && tabItems.length > 1) {
|
|
366
|
-
tabItems[1].click();
|
|
367
|
-
return true;
|
|
368
|
-
}
|
|
369
|
-
return false;
|
|
370
|
-
});
|
|
371
|
-
|
|
372
|
-
expect(switched).toBe(true);
|
|
373
|
-
}, 15000);
|
|
374
|
-
});
|
|
375
|
-
|
|
376
|
-
// ========================================
|
|
377
|
-
// ERROR CHECKING
|
|
378
|
-
// ========================================
|
|
379
|
-
|
|
380
|
-
describe('Error Checking', () => {
|
|
381
|
-
test('no critical console errors during all route navigation', async () => {
|
|
382
|
-
const errors = [];
|
|
383
|
-
|
|
384
|
-
page.on('pageerror', err => errors.push(err.message));
|
|
385
|
-
|
|
386
|
-
// Navigate through all routes
|
|
387
|
-
for (const route of DEMO_ROUTES) {
|
|
388
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#${route.hash}`, {
|
|
389
|
-
waitUntil: 'networkidle0'
|
|
390
|
-
});
|
|
391
|
-
await delay(200);
|
|
392
|
-
}
|
|
393
|
-
|
|
394
|
-
// Filter out known non-critical errors
|
|
395
|
-
const criticalErrors = errors.filter(e =>
|
|
396
|
-
!e.includes('ResizeObserver') &&
|
|
397
|
-
!e.includes('Script error')
|
|
398
|
-
);
|
|
399
|
-
|
|
400
|
-
expect(criticalErrors.length).toBe(0);
|
|
401
|
-
}, 60000);
|
|
402
|
-
});
|
|
403
|
-
|
|
404
|
-
// ========================================
|
|
405
|
-
// ACCESSIBILITY QUICK CHECKS
|
|
406
|
-
// ========================================
|
|
407
|
-
|
|
408
|
-
describe('Accessibility', () => {
|
|
409
|
-
test('interactive components should have proper roles', async () => {
|
|
410
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#buttons`, {
|
|
411
|
-
waitUntil: 'networkidle0'
|
|
412
|
-
});
|
|
413
|
-
await delay(500);
|
|
414
|
-
|
|
415
|
-
const hasRoles = await page.evaluate(() => {
|
|
416
|
-
const btn = document.querySelector('mu-button');
|
|
417
|
-
return btn ? btn.getAttribute('role') === 'button' : false;
|
|
418
|
-
});
|
|
419
|
-
|
|
420
|
-
expect(hasRoles).toBe(true);
|
|
421
|
-
}, 15000);
|
|
422
|
-
|
|
423
|
-
test('checkboxes should have correct aria-checked', async () => {
|
|
424
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#checkboxes`, {
|
|
425
|
-
waitUntil: 'networkidle0'
|
|
426
|
-
});
|
|
427
|
-
await delay(500);
|
|
428
|
-
|
|
429
|
-
const hasAria = await page.evaluate(() => {
|
|
430
|
-
const cb = document.querySelector('mu-checkbox');
|
|
431
|
-
if (cb) {
|
|
432
|
-
const ariaChecked = cb.getAttribute('aria-checked');
|
|
433
|
-
return ariaChecked === 'true' || ariaChecked === 'false';
|
|
434
|
-
}
|
|
435
|
-
return false;
|
|
436
|
-
});
|
|
437
|
-
|
|
438
|
-
expect(hasAria).toBe(true);
|
|
439
|
-
}, 15000);
|
|
440
|
-
});
|
|
441
|
-
|
|
442
|
-
// ========================================
|
|
443
|
-
// PERFORMANCE CHECKS
|
|
444
|
-
// ========================================
|
|
445
|
-
|
|
446
|
-
describe('Performance', () => {
|
|
447
|
-
test('route transitions should be fast (<500ms)', async () => {
|
|
448
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#home`, {
|
|
449
|
-
waitUntil: 'networkidle0'
|
|
450
|
-
});
|
|
451
|
-
|
|
452
|
-
const start = Date.now();
|
|
453
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#buttons`, {
|
|
454
|
-
waitUntil: 'networkidle0'
|
|
455
|
-
});
|
|
456
|
-
const duration = Date.now() - start;
|
|
457
|
-
|
|
458
|
-
expect(duration).toBeLessThan(2000); // Allow 2s for network
|
|
459
|
-
}, 15000);
|
|
460
|
-
|
|
461
|
-
test('all routes should load within time budget', async () => {
|
|
462
|
-
const times = [];
|
|
463
|
-
|
|
464
|
-
for (const route of DEMO_ROUTES.slice(0, 5)) { // Test first 5 routes
|
|
465
|
-
const start = Date.now();
|
|
466
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#${route.hash}`, {
|
|
467
|
-
waitUntil: 'networkidle0'
|
|
468
|
-
});
|
|
469
|
-
times.push(Date.now() - start);
|
|
470
|
-
}
|
|
471
|
-
|
|
472
|
-
const avgTime = times.reduce((a, b) => a + b, 0) / times.length;
|
|
473
|
-
console.log(`Average route load time: ${avgTime.toFixed(0)}ms`);
|
|
474
|
-
|
|
475
|
-
expect(avgTime).toBeLessThan(3000); // Average under 3s
|
|
476
|
-
}, 45000);
|
|
477
|
-
});
|
|
478
|
-
});
|