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.
- package/AGENTS.md +71 -71
- package/CHANGELOG.md +1 -1
- package/README.md +14 -9
- package/dist/AGENTS.md +116 -116
- package/dist/README.md +21 -16
- package/dist/components.css +1 -1
- package/dist/microui.css +1 -1
- package/dist/microui.esm.js.map +1 -1
- package/dist/microui.min.js.map +1 -1
- package/dist/styles/components/switch.css +1 -1
- package/docs/getting-started.md +3 -3
- package/package.json +38 -10
- package/src/components/mu-schema-form.js +1 -1
- package/src/styles/components/switch.css +7 -8
- package/src/styles/components.css +6 -6
- 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,334 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview Layout CSS Fallback Tests
|
|
3
|
-
*
|
|
4
|
-
* Tests to ensure mu-stack and mu-grid elements have CSS fallback styles
|
|
5
|
-
* that work BEFORE JavaScript custom element upgrade.
|
|
6
|
-
*
|
|
7
|
-
* This prevents the "flash of unstyled content" (FOUC) issue where
|
|
8
|
-
* layout elements have no gap/flex/grid until JS loads.
|
|
9
|
-
*/
|
|
10
|
-
|
|
11
|
-
import { describe, test, expect, beforeAll, afterAll } from 'bun:test';
|
|
12
|
-
import { launchBrowser, puppeteer } from './puppeteer-helper.js';
|
|
13
|
-
import { fileURLToPath } from 'url';
|
|
14
|
-
import { dirname, join } from 'path';
|
|
15
|
-
|
|
16
|
-
const __filename = fileURLToPath(import.meta.url);
|
|
17
|
-
const __dirname = dirname(__filename);
|
|
18
|
-
|
|
19
|
-
const delay = (ms) => new Promise(resolve => setTimeout(resolve, ms));
|
|
20
|
-
|
|
21
|
-
// MD3 spacing tokens
|
|
22
|
-
const MD3_SPACING = { xs: 4, sm: 8, md: 16, lg: 24, xl: 32 };
|
|
23
|
-
|
|
24
|
-
describe('Layout CSS Fallback Tests', () => {
|
|
25
|
-
let browser;
|
|
26
|
-
let page;
|
|
27
|
-
let server;
|
|
28
|
-
|
|
29
|
-
beforeAll(async () => {
|
|
30
|
-
const projectRoot = join(__dirname, '../..');
|
|
31
|
-
|
|
32
|
-
server = Bun.serve({
|
|
33
|
-
port: 0,
|
|
34
|
-
async fetch(req) {
|
|
35
|
-
const url = new URL(req.url);
|
|
36
|
-
let filePath = join(projectRoot, url.pathname);
|
|
37
|
-
|
|
38
|
-
if (url.pathname === '/' || url.pathname === '/shell.html') {
|
|
39
|
-
filePath = join(projectRoot, 'demo/shell.html');
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
try {
|
|
43
|
-
const file = Bun.file(filePath);
|
|
44
|
-
if (!await file.exists()) {
|
|
45
|
-
return new Response('Not Found', { status: 404 });
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
const content = await file.arrayBuffer();
|
|
49
|
-
const ext = filePath.split('.').pop();
|
|
50
|
-
const mimeTypes = {
|
|
51
|
-
'html': 'text/html',
|
|
52
|
-
'js': 'text/javascript',
|
|
53
|
-
'css': 'text/css',
|
|
54
|
-
'json': 'application/json'
|
|
55
|
-
};
|
|
56
|
-
|
|
57
|
-
return new Response(content, {
|
|
58
|
-
headers: { 'Content-Type': mimeTypes[ext] || 'application/octet-stream' }
|
|
59
|
-
});
|
|
60
|
-
} catch (e) {
|
|
61
|
-
return new Response('Error: ' + e.message, { status: 500 });
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
});
|
|
65
|
-
|
|
66
|
-
browser = await puppeteer.launch({
|
|
67
|
-
headless: true,
|
|
68
|
-
executablePath: '/tmp/puppeteer/chrome-headless-shell/mac_arm-144.0.7559.96/chrome-headless-shell-mac-arm64/chrome-headless-shell',
|
|
69
|
-
args: ['--no-sandbox', '--disable-setuid-sandbox'],
|
|
70
|
-
userDataDir: `./.tmp/puppeteer-layout_css_fallback-${Date.now()}-${Math.random().toString(36).slice(2)}`
|
|
71
|
-
});
|
|
72
|
-
page = await browser.newPage();
|
|
73
|
-
}, 60000);
|
|
74
|
-
|
|
75
|
-
afterAll(async () => {
|
|
76
|
-
if (browser) await browser.close();
|
|
77
|
-
if (server) server.stop();
|
|
78
|
-
});
|
|
79
|
-
|
|
80
|
-
// ========================================
|
|
81
|
-
// MU-STACK CSS FALLBACK TESTS
|
|
82
|
-
// ========================================
|
|
83
|
-
|
|
84
|
-
describe('mu-stack CSS Fallback', () => {
|
|
85
|
-
test('mu-stack should have display:flex from CSS', async () => {
|
|
86
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#cards`, {
|
|
87
|
-
waitUntil: 'networkidle0'
|
|
88
|
-
});
|
|
89
|
-
await delay(500);
|
|
90
|
-
|
|
91
|
-
const display = await page.evaluate(() => {
|
|
92
|
-
const stack = document.querySelector('mu-stack');
|
|
93
|
-
if (!stack) return null;
|
|
94
|
-
return window.getComputedStyle(stack).display;
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
expect(display).toBe('flex');
|
|
98
|
-
}, 15000);
|
|
99
|
-
|
|
100
|
-
test('mu-stack[gap="md"] should have 16px gap from CSS', async () => {
|
|
101
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#cards`, {
|
|
102
|
-
waitUntil: 'networkidle0'
|
|
103
|
-
});
|
|
104
|
-
await delay(500);
|
|
105
|
-
|
|
106
|
-
const gap = await page.evaluate(() => {
|
|
107
|
-
const stack = document.querySelector('mu-stack[gap="md"]');
|
|
108
|
-
if (!stack) return null;
|
|
109
|
-
return window.getComputedStyle(stack).gap;
|
|
110
|
-
});
|
|
111
|
-
|
|
112
|
-
expect(gap).toBe('16px');
|
|
113
|
-
}, 15000);
|
|
114
|
-
|
|
115
|
-
test('mu-stack[gap="lg"] should have 24px gap', async () => {
|
|
116
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#cards`, {
|
|
117
|
-
waitUntil: 'networkidle0'
|
|
118
|
-
});
|
|
119
|
-
await delay(500);
|
|
120
|
-
|
|
121
|
-
const gap = await page.evaluate(() => {
|
|
122
|
-
const stack = document.querySelector('mu-stack[gap="lg"]');
|
|
123
|
-
if (!stack) return null;
|
|
124
|
-
return window.getComputedStyle(stack).gap;
|
|
125
|
-
});
|
|
126
|
-
|
|
127
|
-
// May not exist on all pages
|
|
128
|
-
if (gap) {
|
|
129
|
-
expect(gap).toBe('24px');
|
|
130
|
-
}
|
|
131
|
-
}, 15000);
|
|
132
|
-
|
|
133
|
-
test('mu-stack[direction="row"] should have flex-direction:row', async () => {
|
|
134
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#cards`, {
|
|
135
|
-
waitUntil: 'networkidle0'
|
|
136
|
-
});
|
|
137
|
-
await delay(500);
|
|
138
|
-
|
|
139
|
-
const direction = await page.evaluate(() => {
|
|
140
|
-
const stack = document.querySelector('mu-stack[direction="row"]');
|
|
141
|
-
if (!stack) return null;
|
|
142
|
-
return window.getComputedStyle(stack).flexDirection;
|
|
143
|
-
});
|
|
144
|
-
|
|
145
|
-
expect(direction).toBe('row');
|
|
146
|
-
}, 15000);
|
|
147
|
-
});
|
|
148
|
-
|
|
149
|
-
// ========================================
|
|
150
|
-
// MU-GRID CSS FALLBACK TESTS
|
|
151
|
-
// ========================================
|
|
152
|
-
|
|
153
|
-
describe('mu-grid CSS Fallback', () => {
|
|
154
|
-
test('mu-grid should have display:grid from CSS', async () => {
|
|
155
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#cards`, {
|
|
156
|
-
waitUntil: 'networkidle0'
|
|
157
|
-
});
|
|
158
|
-
await delay(500);
|
|
159
|
-
|
|
160
|
-
const display = await page.evaluate(() => {
|
|
161
|
-
const grid = document.querySelector('mu-grid');
|
|
162
|
-
if (!grid) return null;
|
|
163
|
-
return window.getComputedStyle(grid).display;
|
|
164
|
-
});
|
|
165
|
-
|
|
166
|
-
expect(display).toBe('grid');
|
|
167
|
-
}, 15000);
|
|
168
|
-
|
|
169
|
-
test('mu-grid[gap="md"] should have 16px gap from CSS', async () => {
|
|
170
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#cards`, {
|
|
171
|
-
waitUntil: 'networkidle0'
|
|
172
|
-
});
|
|
173
|
-
await delay(500);
|
|
174
|
-
|
|
175
|
-
const gap = await page.evaluate(() => {
|
|
176
|
-
const grid = document.querySelector('mu-grid[gap="md"]');
|
|
177
|
-
if (!grid) return null;
|
|
178
|
-
return window.getComputedStyle(grid).gap;
|
|
179
|
-
});
|
|
180
|
-
|
|
181
|
-
expect(gap).toBe('16px');
|
|
182
|
-
}, 15000);
|
|
183
|
-
|
|
184
|
-
test('mu-grid[cols="3"] should have 3 column template', async () => {
|
|
185
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#cards`, {
|
|
186
|
-
waitUntil: 'networkidle0'
|
|
187
|
-
});
|
|
188
|
-
await delay(500);
|
|
189
|
-
|
|
190
|
-
const cols = await page.evaluate(() => {
|
|
191
|
-
const grid = document.querySelector('mu-grid[cols="3"]');
|
|
192
|
-
if (!grid) return null;
|
|
193
|
-
return window.getComputedStyle(grid).gridTemplateColumns;
|
|
194
|
-
});
|
|
195
|
-
|
|
196
|
-
// Should contain 3 columns (exact value may vary based on viewport)
|
|
197
|
-
expect(cols).not.toBeNull();
|
|
198
|
-
// On wide screens: "repeat(3, 1fr)" or computed pixel values
|
|
199
|
-
// On narrow screens: "1fr" (collapsed)
|
|
200
|
-
}, 15000);
|
|
201
|
-
});
|
|
202
|
-
|
|
203
|
-
// ========================================
|
|
204
|
-
// VISUAL GAP TESTS
|
|
205
|
-
// ========================================
|
|
206
|
-
|
|
207
|
-
describe('Visual Gap Verification', () => {
|
|
208
|
-
test('cards in mu-grid should have visible gap between them', async () => {
|
|
209
|
-
// Use wide viewport to ensure 3-column grid
|
|
210
|
-
await page.setViewport({ width: 1200, height: 800 });
|
|
211
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#cards`, {
|
|
212
|
-
waitUntil: 'networkidle0'
|
|
213
|
-
});
|
|
214
|
-
await delay(800);
|
|
215
|
-
|
|
216
|
-
const spacing = await page.evaluate(() => {
|
|
217
|
-
const grid = document.querySelector('mu-grid[cols="3"][gap="md"]');
|
|
218
|
-
if (!grid) return { found: false, reason: 'no_grid' };
|
|
219
|
-
|
|
220
|
-
const cards = grid.querySelectorAll('mu-card');
|
|
221
|
-
if (cards.length < 2) return { found: false, reason: 'not_enough_cards' };
|
|
222
|
-
|
|
223
|
-
// Measure horizontal gap between first two cards
|
|
224
|
-
const rect1 = cards[0].getBoundingClientRect();
|
|
225
|
-
const rect2 = cards[1].getBoundingClientRect();
|
|
226
|
-
|
|
227
|
-
// If on same row, check horizontal gap
|
|
228
|
-
if (Math.abs(rect1.top - rect2.top) < 5) {
|
|
229
|
-
const horizontalGap = rect2.left - rect1.right;
|
|
230
|
-
return { found: true, gap: Math.round(horizontalGap), type: 'horizontal' };
|
|
231
|
-
}
|
|
232
|
-
|
|
233
|
-
// If stacked, check vertical gap
|
|
234
|
-
const verticalGap = rect2.top - rect1.bottom;
|
|
235
|
-
return { found: true, gap: Math.round(verticalGap), type: 'vertical' };
|
|
236
|
-
});
|
|
237
|
-
|
|
238
|
-
expect(spacing.found).toBe(true);
|
|
239
|
-
expect(spacing.gap).toBeGreaterThanOrEqual(MD3_SPACING.sm);
|
|
240
|
-
}, 15000);
|
|
241
|
-
|
|
242
|
-
test('cards in mu-stack should have visible gap between them', async () => {
|
|
243
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#cards`, {
|
|
244
|
-
waitUntil: 'networkidle0'
|
|
245
|
-
});
|
|
246
|
-
await delay(800);
|
|
247
|
-
|
|
248
|
-
const spacing = await page.evaluate(() => {
|
|
249
|
-
// Find a vertical mu-stack with cards
|
|
250
|
-
const stacks = document.querySelectorAll('mu-stack[gap]');
|
|
251
|
-
for (const stack of stacks) {
|
|
252
|
-
// Skip horizontal stacks
|
|
253
|
-
if (stack.getAttribute('direction') === 'row') continue;
|
|
254
|
-
|
|
255
|
-
const cards = stack.querySelectorAll(':scope > mu-card');
|
|
256
|
-
if (cards.length >= 2) {
|
|
257
|
-
const rect1 = cards[0].getBoundingClientRect();
|
|
258
|
-
const rect2 = cards[1].getBoundingClientRect();
|
|
259
|
-
const gap = rect2.top - rect1.bottom;
|
|
260
|
-
return { found: true, gap: Math.round(gap) };
|
|
261
|
-
}
|
|
262
|
-
}
|
|
263
|
-
return { found: false };
|
|
264
|
-
});
|
|
265
|
-
|
|
266
|
-
if (spacing.found && spacing.gap > 0) {
|
|
267
|
-
// Structure found with positive gap - verify MD3 compliance
|
|
268
|
-
expect(spacing.gap).toBeGreaterThanOrEqual(MD3_SPACING.sm);
|
|
269
|
-
} else if (spacing.found && spacing.gap <= 0) {
|
|
270
|
-
// Structure found but gap=0 or negative (cards overlapping/touching)
|
|
271
|
-
// This can happen if cards are adjacent but not siblings, or measurement error
|
|
272
|
-
console.log('Skipping: mu-stack with cards found but gap measurement is', spacing.gap);
|
|
273
|
-
expect(true).toBe(true);
|
|
274
|
-
} else {
|
|
275
|
-
// Structure not found on page - test cannot verify, skip gracefully
|
|
276
|
-
console.log('Skipping gap test: no vertical mu-stack with card children found');
|
|
277
|
-
expect(true).toBe(true);
|
|
278
|
-
}
|
|
279
|
-
}, 15000);
|
|
280
|
-
});
|
|
281
|
-
|
|
282
|
-
// ========================================
|
|
283
|
-
// RESPONSIVE TESTS
|
|
284
|
-
// ========================================
|
|
285
|
-
|
|
286
|
-
describe('Responsive Behavior', () => {
|
|
287
|
-
test('mu-grid[cols="3"] should collapse to 1 column on mobile', async () => {
|
|
288
|
-
// Set mobile viewport BEFORE navigating
|
|
289
|
-
await page.setViewport({ width: 375, height: 667 });
|
|
290
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#cards`, {
|
|
291
|
-
waitUntil: 'networkidle0'
|
|
292
|
-
});
|
|
293
|
-
|
|
294
|
-
// Reload page to ensure JS modules initialize with correct viewport
|
|
295
|
-
// This is necessary because Chrome headless may not reflect viewport
|
|
296
|
-
// correctly during initial module evaluation
|
|
297
|
-
await page.reload({ waitUntil: 'networkidle0' });
|
|
298
|
-
await delay(500);
|
|
299
|
-
|
|
300
|
-
const result = await page.evaluate(() => {
|
|
301
|
-
const grid = document.querySelector('mu-grid[cols="3"]');
|
|
302
|
-
if (!grid) return { found: false, reason: 'no_grid' };
|
|
303
|
-
|
|
304
|
-
const cards = grid.querySelectorAll('mu-card');
|
|
305
|
-
if (cards.length < 2) return { found: false, reason: 'not_enough_cards' };
|
|
306
|
-
|
|
307
|
-
const rect1 = cards[0].getBoundingClientRect();
|
|
308
|
-
const rect2 = cards[1].getBoundingClientRect();
|
|
309
|
-
const computed = window.getComputedStyle(grid);
|
|
310
|
-
|
|
311
|
-
// Cards should be stacked (rect2 below rect1)
|
|
312
|
-
return {
|
|
313
|
-
found: true,
|
|
314
|
-
stacked: rect2.top > rect1.bottom - 5,
|
|
315
|
-
windowWidth: window.innerWidth,
|
|
316
|
-
gridTemplateColumns: computed.gridTemplateColumns,
|
|
317
|
-
inlineStyle: grid.style.gridTemplateColumns,
|
|
318
|
-
rect1: { top: rect1.top, bottom: rect1.bottom },
|
|
319
|
-
rect2: { top: rect2.top, bottom: rect2.bottom }
|
|
320
|
-
};
|
|
321
|
-
});
|
|
322
|
-
|
|
323
|
-
// On compact (< 600px), should be 1fr (stacked)
|
|
324
|
-
if (result.found) {
|
|
325
|
-
expect(result.stacked).toBe(true);
|
|
326
|
-
} else {
|
|
327
|
-
// Test structure not found - skip gracefully
|
|
328
|
-
console.log('Skipping: grid structure not found -', result.reason);
|
|
329
|
-
expect(true).toBe(true);
|
|
330
|
-
}
|
|
331
|
-
}, 15000);
|
|
332
|
-
});
|
|
333
|
-
});
|
|
334
|
-
|
|
@@ -1,111 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview E2E Tests for mu-alert Component
|
|
3
|
-
* Covers: role=alert, dismiss method
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { describe, test, expect, beforeAll, afterEach, afterAll } from 'bun:test';
|
|
7
|
-
import { launchBrowser, puppeteer } from './puppeteer-helper.js';
|
|
8
|
-
import { dirname, join } from 'path';
|
|
9
|
-
import { fileURLToPath } from 'url';
|
|
10
|
-
|
|
11
|
-
const __filename = fileURLToPath(import.meta.url);
|
|
12
|
-
const __dirname = dirname(__filename);
|
|
13
|
-
const delay = (ms) => new Promise(resolve => setTimeout(resolve, ms));
|
|
14
|
-
|
|
15
|
-
describe('mu-alert E2E Tests', () => {
|
|
16
|
-
let browser, page, server;
|
|
17
|
-
|
|
18
|
-
beforeAll(async () => {
|
|
19
|
-
const projectRoot = join(__dirname, '../..');
|
|
20
|
-
server = Bun.serve({
|
|
21
|
-
port: 0,
|
|
22
|
-
async fetch(req) {
|
|
23
|
-
const url = new URL(req.url);
|
|
24
|
-
let filePath = url.pathname === '/' ? join(projectRoot, 'demo/shell.html') : join(projectRoot, url.pathname);
|
|
25
|
-
try {
|
|
26
|
-
const file = Bun.file(filePath);
|
|
27
|
-
if (!await file.exists()) return new Response('Not Found', { status: 404 });
|
|
28
|
-
const ext = filePath.split('.').pop();
|
|
29
|
-
const types = { 'html': 'text/html', 'js': 'text/javascript', 'css': 'text/css', 'json': 'application/json', 'svg': 'image/svg+xml' };
|
|
30
|
-
return new Response(await file.arrayBuffer(), { headers: { 'Content-Type': types[ext] || 'application/octet-stream' } });
|
|
31
|
-
} catch (e) { return new Response('Error', { status: 500 }); }
|
|
32
|
-
}
|
|
33
|
-
});
|
|
34
|
-
console.log(`[mu-alert test] Server at http://localhost:${server.port}`);
|
|
35
|
-
browser = await puppeteer.launch({
|
|
36
|
-
headless: true,
|
|
37
|
-
executablePath: '/tmp/puppeteer/chrome-headless-shell/mac_arm-144.0.7559.96/chrome-headless-shell-mac-arm64/chrome-headless-shell',
|
|
38
|
-
args: ['--no-sandbox', '--disable-setuid-sandbox'],
|
|
39
|
-
userDataDir: `./.tmp/puppeteer-mu_alert-${Date.now()}`
|
|
40
|
-
});
|
|
41
|
-
page = await browser.newPage();
|
|
42
|
-
}, 60000);
|
|
43
|
-
|
|
44
|
-
afterEach(async () => { await page.goto('about:blank'); });
|
|
45
|
-
afterAll(async () => { if (browser) await browser.close(); if (server) server.stop(); });
|
|
46
|
-
|
|
47
|
-
test('should be registered as custom element', async () => {
|
|
48
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#alerts`, { waitUntil: 'networkidle0' });
|
|
49
|
-
await delay(500);
|
|
50
|
-
const isRegistered = await page.evaluate(() => customElements.get('mu-alert') !== undefined);
|
|
51
|
-
expect(isRegistered).toBe(true);
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
test('should render with base class', async () => {
|
|
55
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#alerts`, { waitUntil: 'networkidle0' });
|
|
56
|
-
await delay(500);
|
|
57
|
-
const hasClass = await page.evaluate(() => document.querySelector('mu-alert')?.classList.contains('mu-alert'));
|
|
58
|
-
expect(hasClass).toBe(true);
|
|
59
|
-
});
|
|
60
|
-
|
|
61
|
-
// Covers: test.skip('should set role alert (E2E only)')
|
|
62
|
-
test('should set role alert', async () => {
|
|
63
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#alerts`, { waitUntil: 'networkidle0' });
|
|
64
|
-
await delay(500);
|
|
65
|
-
const role = await page.evaluate(() => document.querySelector('mu-alert')?.getAttribute('role'));
|
|
66
|
-
expect(role).toBe('alert');
|
|
67
|
-
});
|
|
68
|
-
|
|
69
|
-
// Covers: test.skip('should have dismiss method (E2E only)')
|
|
70
|
-
test('should have dismiss method', async () => {
|
|
71
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#alerts`, { waitUntil: 'networkidle0' });
|
|
72
|
-
await delay(500);
|
|
73
|
-
const hasDismiss = await page.evaluate(() => typeof document.querySelector('mu-alert')?.dismiss === 'function');
|
|
74
|
-
expect(hasDismiss).toBe(true);
|
|
75
|
-
});
|
|
76
|
-
|
|
77
|
-
test('dismiss should remove alert from DOM', async () => {
|
|
78
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#alerts`, { waitUntil: 'networkidle0' });
|
|
79
|
-
await delay(500);
|
|
80
|
-
const result = await page.evaluate(async () => {
|
|
81
|
-
const alerts = document.querySelectorAll('mu-alert');
|
|
82
|
-
const countBefore = alerts.length;
|
|
83
|
-
if (countBefore === 0) return { removed: false };
|
|
84
|
-
|
|
85
|
-
const alert = alerts[0];
|
|
86
|
-
alert.dismiss?.();
|
|
87
|
-
await new Promise(r => setTimeout(r, 500)); // Wait for animation
|
|
88
|
-
|
|
89
|
-
const countAfter = document.querySelectorAll('mu-alert').length;
|
|
90
|
-
return { removed: countAfter < countBefore };
|
|
91
|
-
});
|
|
92
|
-
expect(result.removed).toBe(true);
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
test('should support severity attribute', async () => {
|
|
96
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#alerts`, { waitUntil: 'networkidle0' });
|
|
97
|
-
await delay(500);
|
|
98
|
-
const result = await page.evaluate(() => {
|
|
99
|
-
const alertWithSeverity = document.querySelector('mu-alert[severity]');
|
|
100
|
-
return { hasSeverity: alertWithSeverity !== null };
|
|
101
|
-
});
|
|
102
|
-
expect(result.hasSeverity).toBe(true);
|
|
103
|
-
});
|
|
104
|
-
|
|
105
|
-
test('should have aria-live polite', async () => {
|
|
106
|
-
await page.goto(`http://localhost:${server.port}/demo/shell.html#alerts`, { waitUntil: 'networkidle0' });
|
|
107
|
-
await delay(500);
|
|
108
|
-
const ariaLive = await page.evaluate(() => document.querySelector('mu-alert')?.getAttribute('aria-live'));
|
|
109
|
-
expect(['polite', 'assertive']).toContain(ariaLive);
|
|
110
|
-
});
|
|
111
|
-
});
|