microui-wc 0.1.0 → 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.
Files changed (343) hide show
  1. package/AGENTS.md +71 -71
  2. package/CHANGELOG.md +1 -1
  3. package/README.md +14 -9
  4. package/dist/AGENTS.md +116 -116
  5. package/dist/README.md +21 -16
  6. package/dist/components.css +1 -1
  7. package/dist/microui.css +1 -1
  8. package/dist/microui.esm.js.map +1 -1
  9. package/dist/microui.min.js.map +1 -1
  10. package/dist/styles/components/switch.css +1 -1
  11. package/docs/getting-started.md +3 -3
  12. package/package.json +38 -10
  13. package/src/components/mu-schema-form.js +1 -1
  14. package/src/styles/components/switch.css +7 -8
  15. package/src/styles/components.css +6 -6
  16. package/.github/ISSUE_TEMPLATE/bug_report.md +0 -40
  17. package/.github/ISSUE_TEMPLATE/feature_request.md +0 -33
  18. package/.github/PULL_REQUEST_TEMPLATE.md +0 -28
  19. package/.github/workflows/ci.yml +0 -42
  20. package/.github/workflows/deploy-pages.yml +0 -112
  21. package/CODE_OF_CONDUCT.md +0 -59
  22. package/CONTRIBUTING.md +0 -156
  23. package/SECURITY.md +0 -58
  24. package/app/.generated/routes/alerts.js +0 -8
  25. package/app/.generated/routes/avatars.js +0 -8
  26. package/app/.generated/routes/badges.js +0 -8
  27. package/app/.generated/routes/buttons.js +0 -10
  28. package/app/.generated/routes/cards.js +0 -10
  29. package/app/.generated/routes/checkboxes.js +0 -9
  30. package/app/.generated/routes/chips.js +0 -8
  31. package/app/.generated/routes/dropdowns.js +0 -9
  32. package/app/.generated/routes/home.js +0 -7
  33. package/app/.generated/routes/icons.js +0 -9
  34. package/app/.generated/routes/inputs.js +0 -10
  35. package/app/.generated/routes/installation.js +0 -7
  36. package/app/.generated/routes/layout.js +0 -9
  37. package/app/.generated/routes/modals.js +0 -9
  38. package/app/.generated/routes/navbar.js +0 -7
  39. package/app/.generated/routes/progress.js +0 -9
  40. package/app/.generated/routes/radios.js +0 -9
  41. package/app/.generated/routes/switches.js +0 -9
  42. package/app/.generated/routes/tabs.js +0 -8
  43. package/app/.generated/routes/toasts.js +0 -9
  44. package/app/index.html +0 -67
  45. package/app/pages/alerts.html +0 -23
  46. package/app/pages/avatars.html +0 -22
  47. package/app/pages/badges.html +0 -22
  48. package/app/pages/buttons.html +0 -71
  49. package/app/pages/cards.html +0 -54
  50. package/app/pages/checkboxes.html +0 -39
  51. package/app/pages/chips.html +0 -23
  52. package/app/pages/dropdowns.html +0 -41
  53. package/app/pages/home.html +0 -59
  54. package/app/pages/icons.html +0 -29
  55. package/app/pages/inputs.html +0 -66
  56. package/app/pages/installation.html +0 -34
  57. package/app/pages/layout.html +0 -30
  58. package/app/pages/modals.html +0 -21
  59. package/app/pages/navbar.html +0 -22
  60. package/app/pages/progress.html +0 -35
  61. package/app/pages/radios.html +0 -40
  62. package/app/pages/switches.html +0 -39
  63. package/app/pages/tabs.html +0 -30
  64. package/app/pages/toasts.html +0 -22
  65. package/app-dist/index.html +0 -67
  66. package/app-dist/pages/alerts.html +0 -23
  67. package/app-dist/pages/avatars.html +0 -22
  68. package/app-dist/pages/badges.html +0 -22
  69. package/app-dist/pages/buttons.html +0 -71
  70. package/app-dist/pages/cards.html +0 -54
  71. package/app-dist/pages/checkboxes.html +0 -39
  72. package/app-dist/pages/chips.html +0 -23
  73. package/app-dist/pages/dropdowns.html +0 -41
  74. package/app-dist/pages/home.html +0 -59
  75. package/app-dist/pages/icons.html +0 -29
  76. package/app-dist/pages/inputs.html +0 -66
  77. package/app-dist/pages/installation.html +0 -34
  78. package/app-dist/pages/layout.html +0 -30
  79. package/app-dist/pages/modals.html +0 -21
  80. package/app-dist/pages/navbar.html +0 -22
  81. package/app-dist/pages/progress.html +0 -35
  82. package/app-dist/pages/radios.html +0 -40
  83. package/app-dist/pages/switches.html +0 -39
  84. package/app-dist/pages/tabs.html +0 -30
  85. package/app-dist/pages/toasts.html +0 -22
  86. package/app-dist/pages.json +0 -217
  87. package/app-dist/routes/alerts.js +0 -5
  88. package/app-dist/routes/avatars.js +0 -1
  89. package/app-dist/routes/badges.js +0 -1
  90. package/app-dist/routes/buttons.js +0 -1
  91. package/app-dist/routes/cards.js +0 -1
  92. package/app-dist/routes/checkboxes.js +0 -9
  93. package/app-dist/routes/chips.js +0 -4
  94. package/app-dist/routes/chunk-019e5e2f.js +0 -5
  95. package/app-dist/routes/chunk-0m4j19yd.js +0 -2
  96. package/app-dist/routes/chunk-0tmmp5q0.js +0 -1
  97. package/app-dist/routes/chunk-10xn709r.js +0 -1
  98. package/app-dist/routes/chunk-15m2qcda.js +0 -2
  99. package/app-dist/routes/chunk-1bh8g23n.js +0 -1
  100. package/app-dist/routes/chunk-1vg0v937.js +0 -1
  101. package/app-dist/routes/chunk-1zvcgy3j.js +0 -1
  102. package/app-dist/routes/chunk-2afb0861.js +0 -1
  103. package/app-dist/routes/chunk-2c6ttpzt.js +0 -5
  104. package/app-dist/routes/chunk-3dy30fhs.js +0 -1
  105. package/app-dist/routes/chunk-426dnces.js +0 -13
  106. package/app-dist/routes/chunk-44kgxery.js +0 -1
  107. package/app-dist/routes/chunk-47fdnejd.js +0 -33
  108. package/app-dist/routes/chunk-49a6t2vq.js +0 -1
  109. package/app-dist/routes/chunk-4fe1rm5b.js +0 -1
  110. package/app-dist/routes/chunk-4ggmvkta.js +0 -33
  111. package/app-dist/routes/chunk-4vkz81q7.js +0 -33
  112. package/app-dist/routes/chunk-4w4tmj8f.js +0 -31
  113. package/app-dist/routes/chunk-532s62kr.js +0 -31
  114. package/app-dist/routes/chunk-5hm3bssy.js +0 -33
  115. package/app-dist/routes/chunk-5vrh24hc.js +0 -1
  116. package/app-dist/routes/chunk-61pcg25a.js +0 -1
  117. package/app-dist/routes/chunk-6nfhygvf.js +0 -1
  118. package/app-dist/routes/chunk-700e7je6.js +0 -33
  119. package/app-dist/routes/chunk-7fsn17kg.js +0 -1
  120. package/app-dist/routes/chunk-7k789b32.js +0 -1
  121. package/app-dist/routes/chunk-7r46q0ys.js +0 -36
  122. package/app-dist/routes/chunk-86fmc1fr.js +0 -5
  123. package/app-dist/routes/chunk-8qth37vw.js +0 -1
  124. package/app-dist/routes/chunk-924wv8n0.js +0 -1
  125. package/app-dist/routes/chunk-9mbhgxk9.js +0 -1
  126. package/app-dist/routes/chunk-a216hyd9.js +0 -1
  127. package/app-dist/routes/chunk-akzxykh9.js +0 -33
  128. package/app-dist/routes/chunk-b3dcvy8c.js +0 -1
  129. package/app-dist/routes/chunk-b74zahz5.js +0 -31
  130. package/app-dist/routes/chunk-bftj53p2.js +0 -5
  131. package/app-dist/routes/chunk-c01hnz3e.js +0 -1
  132. package/app-dist/routes/chunk-d8pvv5km.js +0 -1
  133. package/app-dist/routes/chunk-dev0aezr.js +0 -2
  134. package/app-dist/routes/chunk-dh6vnv0e.js +0 -1
  135. package/app-dist/routes/chunk-dn2cbpva.js +0 -36
  136. package/app-dist/routes/chunk-dvn0my90.js +0 -1
  137. package/app-dist/routes/chunk-dvq8mnve.js +0 -36
  138. package/app-dist/routes/chunk-e8c2gc4d.js +0 -5
  139. package/app-dist/routes/chunk-ejf9ak2x.js +0 -1
  140. package/app-dist/routes/chunk-f083m55s.js +0 -1
  141. package/app-dist/routes/chunk-fnrj28s1.js +0 -31
  142. package/app-dist/routes/chunk-fvg3yjdp.js +0 -31
  143. package/app-dist/routes/chunk-g7k381n1.js +0 -1
  144. package/app-dist/routes/chunk-h01kq2ae.js +0 -13
  145. package/app-dist/routes/chunk-h4dk761v.js +0 -5
  146. package/app-dist/routes/chunk-hmx91z2x.js +0 -5
  147. package/app-dist/routes/chunk-hxbg4m42.js +0 -36
  148. package/app-dist/routes/chunk-jbjnfp2b.js +0 -2
  149. package/app-dist/routes/chunk-jxtz5vv6.js +0 -36
  150. package/app-dist/routes/chunk-jxzcs0ey.js +0 -36
  151. package/app-dist/routes/chunk-kt7wwhcx.js +0 -1
  152. package/app-dist/routes/chunk-kzptszyc.js +0 -33
  153. package/app-dist/routes/chunk-mhgca4w4.js +0 -2
  154. package/app-dist/routes/chunk-mhswxa20.js +0 -1
  155. package/app-dist/routes/chunk-n8zfeex6.js +0 -1
  156. package/app-dist/routes/chunk-pee47b2r.js +0 -1
  157. package/app-dist/routes/chunk-pesmw829.js +0 -1
  158. package/app-dist/routes/chunk-pgc4c6f3.js +0 -36
  159. package/app-dist/routes/chunk-q8egegm1.js +0 -1
  160. package/app-dist/routes/chunk-q9mn2qyq.js +0 -36
  161. package/app-dist/routes/chunk-qh0rtaf3.js +0 -5
  162. package/app-dist/routes/chunk-qqhmk6ye.js +0 -2
  163. package/app-dist/routes/chunk-qrxygmf7.js +0 -33
  164. package/app-dist/routes/chunk-r46yzksx.js +0 -36
  165. package/app-dist/routes/chunk-rgpbw2w0.js +0 -5
  166. package/app-dist/routes/chunk-rnpzv3d8.js +0 -2
  167. package/app-dist/routes/chunk-s5v8cv05.js +0 -2
  168. package/app-dist/routes/chunk-sbwn5bpc.js +0 -1
  169. package/app-dist/routes/chunk-sqbg8jbt.js +0 -33
  170. package/app-dist/routes/chunk-sv8dqnf7.js +0 -1
  171. package/app-dist/routes/chunk-t67sw3za.js +0 -1
  172. package/app-dist/routes/chunk-tjdpqwdf.js +0 -31
  173. package/app-dist/routes/chunk-tq2mfghg.js +0 -1
  174. package/app-dist/routes/chunk-ttn10vt6.js +0 -1
  175. package/app-dist/routes/chunk-v2hzpjxr.js +0 -1
  176. package/app-dist/routes/chunk-wfjjkw9y.js +0 -1
  177. package/app-dist/routes/chunk-wt8cxzmf.js +0 -31
  178. package/app-dist/routes/chunk-x45d372k.js +0 -5
  179. package/app-dist/routes/chunk-y3wsazkt.js +0 -1
  180. package/app-dist/routes/chunk-y7pmgc7t.js +0 -33
  181. package/app-dist/routes/chunk-zefdt2q3.js +0 -31
  182. package/app-dist/routes/dropdowns.js +0 -6
  183. package/app-dist/routes/home.js +0 -1
  184. package/app-dist/routes/icons.js +0 -1
  185. package/app-dist/routes/inputs.js +0 -12
  186. package/app-dist/routes/installation.js +0 -1
  187. package/app-dist/routes/layout.js +0 -1
  188. package/app-dist/routes/modals.js +0 -7
  189. package/app-dist/routes/navbar.js +0 -1
  190. package/app-dist/routes/progress.js +0 -1
  191. package/app-dist/routes/radios.js +0 -6
  192. package/app-dist/routes/switches.js +0 -6
  193. package/app-dist/routes/tabs.js +0 -1
  194. package/app-dist/routes/toasts.js +0 -16
  195. package/assets/fonts/material-symbols-mini.woff2 +0 -0
  196. package/assets/fonts/material-symbols.woff2 +0 -0
  197. package/assets/fonts/roboto-400.woff2 +0 -0
  198. package/assets/fonts/roboto-500.woff2 +0 -0
  199. package/assets/fonts/roboto-700.woff2 +0 -0
  200. package/assets/logo-banner-400.jpg +0 -0
  201. package/assets/logo-banner-400.webp +0 -0
  202. package/assets/logo-banner-800.webp +0 -0
  203. package/assets/logo-banner.jpg +0 -0
  204. package/assets/logo-icon-64.jpg +0 -0
  205. package/assets/logo-icon-64.webp +0 -0
  206. package/assets/logo-icon.jpg +0 -0
  207. package/assets/logo-square.jpg +0 -0
  208. package/bun.lock +0 -312
  209. package/bunfig.toml +0 -4
  210. package/custom-elements.json +0 -1916
  211. package/demo/api/sample-data.json +0 -38
  212. package/demo/content/alerts.html +0 -115
  213. package/demo/content/avatars.html +0 -70
  214. package/demo/content/badges.html +0 -65
  215. package/demo/content/buttons.html +0 -188
  216. package/demo/content/callouts.html +0 -91
  217. package/demo/content/cards.html +0 -121
  218. package/demo/content/checkboxes.html +0 -178
  219. package/demo/content/chips.html +0 -67
  220. package/demo/content/codeblocks.html +0 -101
  221. package/demo/content/confirms.html +0 -115
  222. package/demo/content/datatables.html +0 -149
  223. package/demo/content/dividers.html +0 -119
  224. package/demo/content/dropdowns.html +0 -89
  225. package/demo/content/enterprise.html +0 -252
  226. package/demo/content/home.html +0 -149
  227. package/demo/content/icons.html +0 -89
  228. package/demo/content/inputs.html +0 -135
  229. package/demo/content/installation.html +0 -16
  230. package/demo/content/layout.html +0 -136
  231. package/demo/content/modals.html +0 -141
  232. package/demo/content/navbar.html +0 -70
  233. package/demo/content/progress.html +0 -119
  234. package/demo/content/radios.html +0 -88
  235. package/demo/content/skeletons.html +0 -109
  236. package/demo/content/spinners.html +0 -96
  237. package/demo/content/switches.html +0 -84
  238. package/demo/content/tables.html +0 -124
  239. package/demo/content/tabs.html +0 -85
  240. package/demo/content/toasts.html +0 -116
  241. package/demo/content/tooltips.html +0 -107
  242. package/demo/content/virtual-lists.html +0 -233
  243. package/demo/favicon.ico +0 -0
  244. package/demo/favicon.png +0 -0
  245. package/demo/full.html +0 -52
  246. package/demo/iife.html +0 -46
  247. package/demo/manifest.json +0 -34
  248. package/demo/pages/datatable-demo.html +0 -237
  249. package/demo/pages/prompt-ui-demo.html +0 -218
  250. package/demo/pages/responsive-demo.html +0 -122
  251. package/demo/pages/schema-form-demo.html +0 -270
  252. package/demo/robots.txt +0 -6
  253. package/demo/shell.html +0 -712
  254. package/demo/sw.js +0 -387
  255. package/lighthouse-audit.mjs +0 -113
  256. package/scripts/analyze-components.js +0 -105
  257. package/scripts/build-app.js +0 -193
  258. package/scripts/build-framework.js +0 -444
  259. package/scripts/build-utils.js +0 -101
  260. package/scripts/test-isolated.js +0 -151
  261. package/server.js +0 -256
  262. package/tests/agents/agent-integration.test.js +0 -76
  263. package/tests/benchmark.html +0 -296
  264. package/tests/build/scan-components.test.js +0 -173
  265. package/tests/components/all-components.test.js +0 -245
  266. package/tests/components/all-missing-components.test.js +0 -574
  267. package/tests/components/mu-alert.test.js +0 -113
  268. package/tests/components/mu-avatar.test.js +0 -148
  269. package/tests/components/mu-badge.test.js +0 -92
  270. package/tests/components/mu-button.test.js +0 -112
  271. package/tests/components/mu-card.test.js +0 -89
  272. package/tests/components/mu-checkbox.test.js +0 -158
  273. package/tests/components/mu-chip.test.js +0 -118
  274. package/tests/components/mu-container.test.js +0 -120
  275. package/tests/components/mu-divider.test.js +0 -98
  276. package/tests/components/mu-drawer-item.test.js +0 -199
  277. package/tests/components/mu-drawer.test.js +0 -96
  278. package/tests/components/mu-dropdown.test.js +0 -125
  279. package/tests/components/mu-form.test.js +0 -138
  280. package/tests/components/mu-grid.test.js +0 -135
  281. package/tests/components/mu-icon.test.js +0 -110
  282. package/tests/components/mu-input.test.js +0 -131
  283. package/tests/components/mu-lazy.test.js +0 -103
  284. package/tests/components/mu-modal.test.js +0 -275
  285. package/tests/components/mu-navbar.test.js +0 -101
  286. package/tests/components/mu-progress.test.js +0 -115
  287. package/tests/components/mu-radio.test.js +0 -114
  288. package/tests/components/mu-repeat.test.js +0 -106
  289. package/tests/components/mu-sidebar.test.js +0 -126
  290. package/tests/components/mu-skeleton.test.js +0 -162
  291. package/tests/components/mu-stack.test.js +0 -143
  292. package/tests/components/mu-switch.test.js +0 -292
  293. package/tests/components/mu-table.test.js +0 -124
  294. package/tests/components/mu-tabs.test.js +0 -104
  295. package/tests/components/mu-textarea.test.js +0 -115
  296. package/tests/components/mu-toast.test.js +0 -321
  297. package/tests/components/mu-tooltip.test.js +0 -133
  298. package/tests/components/mu-virtual-list.test.js +0 -109
  299. package/tests/core/MuElement.test.js +0 -120
  300. package/tests/core/agent-api.test.js +0 -125
  301. package/tests/core/all-core-modules.test.js +0 -442
  302. package/tests/core/bus.test.js +0 -364
  303. package/tests/core/component-schema.test.js +0 -160
  304. package/tests/core/feature-registry.test.js +0 -198
  305. package/tests/core/form-state.test.js +0 -167
  306. package/tests/core/http.test.js +0 -119
  307. package/tests/core/keyboard.test.js +0 -319
  308. package/tests/core/layers.test.js +0 -129
  309. package/tests/core/namespaced-stores.test.js +0 -114
  310. package/tests/core/render.test.js +0 -121
  311. package/tests/core/ripple.test.js +0 -131
  312. package/tests/core/router.test.js +0 -89
  313. package/tests/core/scheduler.test.js +0 -121
  314. package/tests/core/signals.test.js +0 -128
  315. package/tests/core/store.test.js +0 -171
  316. package/tests/core/transitions.test.js +0 -82
  317. package/tests/e2e/accessibility-harness.html +0 -58
  318. package/tests/e2e/accessibility.test.js +0 -401
  319. package/tests/e2e/agent-features.test.js +0 -372
  320. package/tests/e2e/card-spacing.test.js +0 -287
  321. package/tests/e2e/components.test.js +0 -439
  322. package/tests/e2e/demo-routes.test.js +0 -478
  323. package/tests/e2e/layout-css-fallback.test.js +0 -334
  324. package/tests/e2e/mu-alert.e2e.test.js +0 -111
  325. package/tests/e2e/mu-checkbox.test.js +0 -489
  326. package/tests/e2e/mu-chip.test.js +0 -347
  327. package/tests/e2e/mu-form.test.js +0 -499
  328. package/tests/e2e/mu-icon.test.js +0 -114
  329. package/tests/e2e/mu-radio.test.js +0 -113
  330. package/tests/e2e/mu-skeleton.test.js +0 -140
  331. package/tests/e2e/mu-switch.test.js +0 -415
  332. package/tests/e2e/mu-tabs.test.js +0 -494
  333. package/tests/e2e/mu-textarea.test.js +0 -242
  334. package/tests/e2e/mu-virtual-list.test.js +0 -427
  335. package/tests/e2e/perf-memory.test.js +0 -161
  336. package/tests/e2e/puppeteer-helper.js +0 -137
  337. package/tests/e2e/puppeteer.test.js +0 -226
  338. package/tests/e2e/pwa.test.js +0 -261
  339. package/tests/e2e/test-harness.html +0 -319
  340. package/tests/manual/test-components.html +0 -120
  341. package/tests/memory-test.html +0 -309
  342. package/tests/setup-dom.js +0 -93
  343. 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
- });