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,439 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview E2E Tests for All Components requiring real browser
|
|
3
|
-
* Covers: mu-button, mu-toast, mu-input, mu-dropdown, theme-toggle, layout, sidebar
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { describe, test, expect, beforeAll, afterAll } from 'bun:test';
|
|
7
|
-
import { launchBrowser, puppeteer } from './puppeteer-helper.js';
|
|
8
|
-
import { fileURLToPath } from 'url';
|
|
9
|
-
import { dirname, join } from 'path';
|
|
10
|
-
import { mkdirSync, rmSync } from 'fs';
|
|
11
|
-
|
|
12
|
-
const __filename = fileURLToPath(import.meta.url);
|
|
13
|
-
const __dirname = dirname(__filename);
|
|
14
|
-
|
|
15
|
-
describe('Components E2E Tests', () => {
|
|
16
|
-
let browser;
|
|
17
|
-
let page;
|
|
18
|
-
let server;
|
|
19
|
-
const userDataDir = `./.tmp/puppeteer-components-${Date.now()}-${Math.random().toString(36).slice(2)}`;
|
|
20
|
-
|
|
21
|
-
beforeAll(async () => {
|
|
22
|
-
mkdirSync(userDataDir, { recursive: true });
|
|
23
|
-
|
|
24
|
-
const projectRoot = join(__dirname, '../..');
|
|
25
|
-
server = Bun.serve({
|
|
26
|
-
port: 0,
|
|
27
|
-
async fetch(req) {
|
|
28
|
-
const url = new URL(req.url);
|
|
29
|
-
let filePath = join(projectRoot, url.pathname);
|
|
30
|
-
|
|
31
|
-
if (url.pathname === '/') {
|
|
32
|
-
return new Response(`<!DOCTYPE html>
|
|
33
|
-
<html data-theme="light">
|
|
34
|
-
<head>
|
|
35
|
-
<title>Component Test</title>
|
|
36
|
-
<link rel="stylesheet" href="/dist/microui.min.css">
|
|
37
|
-
<style>
|
|
38
|
-
:root {
|
|
39
|
-
--mu-color-primary: #6750a4;
|
|
40
|
-
--mu-color-surface: #ffffff;
|
|
41
|
-
}
|
|
42
|
-
[data-theme="dark"] {
|
|
43
|
-
--mu-color-primary: #cfbcff;
|
|
44
|
-
--mu-color-surface: #1c1b1f;
|
|
45
|
-
}
|
|
46
|
-
.app-main { min-height: 100vh; }
|
|
47
|
-
.sidebar { position: sticky; top: 0; max-height: 100vh; overflow-y: auto; }
|
|
48
|
-
.page { display: none; }
|
|
49
|
-
.page.active { display: block; }
|
|
50
|
-
</style>
|
|
51
|
-
</head>
|
|
52
|
-
<body>
|
|
53
|
-
<div class="app-main">
|
|
54
|
-
<nav class="sidebar">
|
|
55
|
-
<a href="#home" data-page="home" class="sidebar-link active">Home</a>
|
|
56
|
-
<a href="#about" data-page="about" class="sidebar-link">About</a>
|
|
57
|
-
</nav>
|
|
58
|
-
<main>
|
|
59
|
-
<div class="page active" id="home">Home Page</div>
|
|
60
|
-
<div class="page" id="about">About Page</div>
|
|
61
|
-
</main>
|
|
62
|
-
</div>
|
|
63
|
-
|
|
64
|
-
<mu-button id="btn1" variant="filled" size="md">Primary Button</mu-button>
|
|
65
|
-
<mu-button id="btn2" variant="text" size="lg">Text Button</mu-button>
|
|
66
|
-
<mu-button id="btn-disabled" disabled>Disabled</mu-button>
|
|
67
|
-
<mu-input id="input1" label="Email" placeholder="Enter email"></mu-input>
|
|
68
|
-
<mu-dropdown id="dropdown1">
|
|
69
|
-
<span slot="trigger">Select</span>
|
|
70
|
-
<mu-option value="1">Option 1</mu-option>
|
|
71
|
-
<mu-option value="2">Option 2</mu-option>
|
|
72
|
-
</mu-dropdown>
|
|
73
|
-
<mu-toast id="toast1" severity="info">Test Toast</mu-toast>
|
|
74
|
-
<mu-checkbox id="check1">Accept Terms</mu-checkbox>
|
|
75
|
-
<mu-switch id="switch1">Enable Feature</mu-switch>
|
|
76
|
-
<mu-theme-toggle id="theme-toggle"></mu-theme-toggle>
|
|
77
|
-
|
|
78
|
-
<script src="/dist/microui.min.js"></script>
|
|
79
|
-
<script>
|
|
80
|
-
function showPage(pageId) {
|
|
81
|
-
window.scrollTo(0, 0);
|
|
82
|
-
document.querySelectorAll('.page').forEach(p => p.classList.remove('active'));
|
|
83
|
-
document.getElementById(pageId)?.classList.add('active');
|
|
84
|
-
}
|
|
85
|
-
</script>
|
|
86
|
-
</body>
|
|
87
|
-
</html>`, { headers: { 'Content-Type': 'text/html' } });
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
try {
|
|
91
|
-
const file = Bun.file(filePath);
|
|
92
|
-
if (!await file.exists()) {
|
|
93
|
-
return new Response('Not Found', { status: 404 });
|
|
94
|
-
}
|
|
95
|
-
const content = await file.text();
|
|
96
|
-
const ext = filePath.split('.').pop();
|
|
97
|
-
const mimeTypes = { 'html': 'text/html', 'js': 'text/javascript', 'css': 'text/css' };
|
|
98
|
-
return new Response(content, { headers: { 'Content-Type': mimeTypes[ext] || 'text/plain' } });
|
|
99
|
-
} catch (e) {
|
|
100
|
-
return new Response('Error: ' + e.message, { status: 500 });
|
|
101
|
-
}
|
|
102
|
-
}
|
|
103
|
-
});
|
|
104
|
-
|
|
105
|
-
browser = await launchBrowser({ testName: 'e2e' });
|
|
106
|
-
page = await browser.newPage();
|
|
107
|
-
await page.goto(`http://localhost:${server.port}`, { waitUntil: 'networkidle0', timeout: 30000 });
|
|
108
|
-
await page.waitForFunction(() => typeof window.microUI !== 'undefined', { timeout: 15000 });
|
|
109
|
-
}, 60000);
|
|
110
|
-
|
|
111
|
-
afterAll(async () => {
|
|
112
|
-
await browser?.close();
|
|
113
|
-
server?.stop();
|
|
114
|
-
try { rmSync(userDataDir, { recursive: true, force: true }); } catch { }
|
|
115
|
-
});
|
|
116
|
-
|
|
117
|
-
// ========================================
|
|
118
|
-
// MU-BUTTON COMPONENT
|
|
119
|
-
// ========================================
|
|
120
|
-
|
|
121
|
-
describe('mu-button Component', () => {
|
|
122
|
-
test('should be registered as custom element', async () => {
|
|
123
|
-
const result = await page.evaluate(() => customElements.get('mu-button') !== undefined);
|
|
124
|
-
expect(result).toBe(true);
|
|
125
|
-
});
|
|
126
|
-
|
|
127
|
-
test('should render with base class', async () => {
|
|
128
|
-
const result = await page.evaluate(() => {
|
|
129
|
-
const btn = document.querySelector('mu-button');
|
|
130
|
-
return btn && btn.classList.contains('mu-button');
|
|
131
|
-
});
|
|
132
|
-
expect(result).toBe(true);
|
|
133
|
-
});
|
|
134
|
-
|
|
135
|
-
test('should apply variant attribute', async () => {
|
|
136
|
-
const result = await page.evaluate(() => {
|
|
137
|
-
const btn = document.querySelector('mu-button[variant="filled"]');
|
|
138
|
-
return btn && btn.classList.contains('mu-button--filled');
|
|
139
|
-
});
|
|
140
|
-
expect(result).toBe(true);
|
|
141
|
-
});
|
|
142
|
-
|
|
143
|
-
test('should apply size attribute', async () => {
|
|
144
|
-
const result = await page.evaluate(() => {
|
|
145
|
-
const btn = document.querySelector('mu-button[size="lg"]');
|
|
146
|
-
return btn && btn.classList.contains('mu-button--lg');
|
|
147
|
-
});
|
|
148
|
-
expect(result).toBe(true);
|
|
149
|
-
});
|
|
150
|
-
|
|
151
|
-
test('should have role=button', async () => {
|
|
152
|
-
const result = await page.evaluate(() => {
|
|
153
|
-
const btn = document.querySelector('mu-button');
|
|
154
|
-
return btn && btn.getAttribute('role') === 'button';
|
|
155
|
-
});
|
|
156
|
-
expect(result).toBe(true);
|
|
157
|
-
});
|
|
158
|
-
|
|
159
|
-
test('should have tabindex=0 by default', async () => {
|
|
160
|
-
const result = await page.evaluate(() => {
|
|
161
|
-
const btn = document.querySelector('mu-button:not([disabled])');
|
|
162
|
-
return btn && btn.getAttribute('tabindex') === '0';
|
|
163
|
-
});
|
|
164
|
-
expect(result).toBe(true);
|
|
165
|
-
});
|
|
166
|
-
|
|
167
|
-
test('should have tabindex=-1 when disabled', async () => {
|
|
168
|
-
const result = await page.evaluate(() => {
|
|
169
|
-
const btn = document.querySelector('mu-button[disabled]');
|
|
170
|
-
return btn && btn.getAttribute('tabindex') === '-1';
|
|
171
|
-
});
|
|
172
|
-
expect(result).toBe(true);
|
|
173
|
-
});
|
|
174
|
-
|
|
175
|
-
test('should add is-disabled class when disabled', async () => {
|
|
176
|
-
const result = await page.evaluate(() => {
|
|
177
|
-
const btn = document.querySelector('mu-button[disabled]');
|
|
178
|
-
return btn && btn.classList.contains('is-disabled');
|
|
179
|
-
});
|
|
180
|
-
expect(result).toBe(true);
|
|
181
|
-
});
|
|
182
|
-
});
|
|
183
|
-
|
|
184
|
-
// ========================================
|
|
185
|
-
// MU-INPUT COMPONENT
|
|
186
|
-
// ========================================
|
|
187
|
-
|
|
188
|
-
describe('mu-input Component', () => {
|
|
189
|
-
test('should be registered', async () => {
|
|
190
|
-
const result = await page.evaluate(() => customElements.get('mu-input') !== undefined);
|
|
191
|
-
expect(result).toBe(true);
|
|
192
|
-
});
|
|
193
|
-
|
|
194
|
-
test('should render input field', async () => {
|
|
195
|
-
const result = await page.evaluate(() => {
|
|
196
|
-
const input = document.querySelector('mu-input');
|
|
197
|
-
return input && input.querySelector('input') !== null;
|
|
198
|
-
});
|
|
199
|
-
expect(result).toBe(true);
|
|
200
|
-
});
|
|
201
|
-
|
|
202
|
-
test('should have label', async () => {
|
|
203
|
-
const result = await page.evaluate(() => {
|
|
204
|
-
const input = document.querySelector('mu-input[label]');
|
|
205
|
-
return input && (input.querySelector('label') !== null || input.textContent.includes('Email'));
|
|
206
|
-
});
|
|
207
|
-
expect(result).toBe(true);
|
|
208
|
-
});
|
|
209
|
-
});
|
|
210
|
-
|
|
211
|
-
// ========================================
|
|
212
|
-
// MU-DROPDOWN COMPONENT
|
|
213
|
-
// ========================================
|
|
214
|
-
|
|
215
|
-
describe('mu-dropdown Component', () => {
|
|
216
|
-
test('should be registered', async () => {
|
|
217
|
-
const result = await page.evaluate(() => customElements.get('mu-dropdown') !== undefined);
|
|
218
|
-
expect(result).toBe(true);
|
|
219
|
-
});
|
|
220
|
-
|
|
221
|
-
test('should have trigger element', async () => {
|
|
222
|
-
const result = await page.evaluate(() => {
|
|
223
|
-
const dropdown = document.querySelector('mu-dropdown');
|
|
224
|
-
// Check for slotted trigger or internal button
|
|
225
|
-
return dropdown && (
|
|
226
|
-
dropdown.querySelector('[slot="trigger"]') !== null ||
|
|
227
|
-
dropdown.querySelector('button') !== null ||
|
|
228
|
-
dropdown.shadowRoot?.querySelector('button') !== null
|
|
229
|
-
);
|
|
230
|
-
});
|
|
231
|
-
expect(result).toBe(true);
|
|
232
|
-
});
|
|
233
|
-
|
|
234
|
-
test('should have open/close/toggle methods', async () => {
|
|
235
|
-
const result = await page.evaluate(() => {
|
|
236
|
-
const dropdown = document.querySelector('mu-dropdown');
|
|
237
|
-
return dropdown &&
|
|
238
|
-
typeof dropdown.open === 'function' &&
|
|
239
|
-
typeof dropdown.close === 'function' &&
|
|
240
|
-
typeof dropdown.toggle === 'function';
|
|
241
|
-
});
|
|
242
|
-
expect(result).toBe(true);
|
|
243
|
-
});
|
|
244
|
-
});
|
|
245
|
-
|
|
246
|
-
// ========================================
|
|
247
|
-
// MU-TOAST COMPONENT
|
|
248
|
-
// ========================================
|
|
249
|
-
|
|
250
|
-
describe('mu-toast Component', () => {
|
|
251
|
-
test('should be registered', async () => {
|
|
252
|
-
const result = await page.evaluate(() => customElements.get('mu-toast') !== undefined);
|
|
253
|
-
expect(result).toBe(true);
|
|
254
|
-
});
|
|
255
|
-
|
|
256
|
-
test('should have severity attribute', async () => {
|
|
257
|
-
const result = await page.evaluate(() => {
|
|
258
|
-
const toast = document.querySelector('mu-toast');
|
|
259
|
-
return toast && toast.hasAttribute('severity');
|
|
260
|
-
});
|
|
261
|
-
expect(result).toBe(true);
|
|
262
|
-
});
|
|
263
|
-
});
|
|
264
|
-
|
|
265
|
-
// ========================================
|
|
266
|
-
// THEME TOGGLE
|
|
267
|
-
// ========================================
|
|
268
|
-
|
|
269
|
-
describe('Theme Toggle', () => {
|
|
270
|
-
test('should have CSS variables in :root', async () => {
|
|
271
|
-
const result = await page.evaluate(() => {
|
|
272
|
-
const style = getComputedStyle(document.documentElement);
|
|
273
|
-
return style.getPropertyValue('--mu-color-primary').trim().length > 0;
|
|
274
|
-
});
|
|
275
|
-
expect(result).toBe(true);
|
|
276
|
-
});
|
|
277
|
-
|
|
278
|
-
test('mu-theme-toggle should be registered', async () => {
|
|
279
|
-
const result = await page.evaluate(() => customElements.get('mu-theme-toggle') !== undefined);
|
|
280
|
-
expect(result).toBe(true);
|
|
281
|
-
});
|
|
282
|
-
|
|
283
|
-
test('should be able to set data-theme attribute', async () => {
|
|
284
|
-
const result = await page.evaluate(() => {
|
|
285
|
-
document.documentElement.setAttribute('data-theme', 'dark');
|
|
286
|
-
const isDark = document.documentElement.getAttribute('data-theme') === 'dark';
|
|
287
|
-
document.documentElement.setAttribute('data-theme', 'light');
|
|
288
|
-
return isDark;
|
|
289
|
-
});
|
|
290
|
-
expect(result).toBe(true);
|
|
291
|
-
});
|
|
292
|
-
|
|
293
|
-
test('should toggle between light and dark', async () => {
|
|
294
|
-
const result = await page.evaluate(() => {
|
|
295
|
-
document.documentElement.setAttribute('data-theme', 'light');
|
|
296
|
-
const wasLight = document.documentElement.getAttribute('data-theme') === 'light';
|
|
297
|
-
document.documentElement.setAttribute('data-theme', 'dark');
|
|
298
|
-
const nowDark = document.documentElement.getAttribute('data-theme') === 'dark';
|
|
299
|
-
document.documentElement.setAttribute('data-theme', 'light');
|
|
300
|
-
return wasLight && nowDark;
|
|
301
|
-
});
|
|
302
|
-
expect(result).toBe(true);
|
|
303
|
-
});
|
|
304
|
-
});
|
|
305
|
-
|
|
306
|
-
// ========================================
|
|
307
|
-
// LAYOUT STABILITY
|
|
308
|
-
// ========================================
|
|
309
|
-
|
|
310
|
-
describe('Layout Stability', () => {
|
|
311
|
-
test('should have min-height on .app-main', async () => {
|
|
312
|
-
const result = await page.evaluate(() => {
|
|
313
|
-
const main = document.querySelector('.app-main');
|
|
314
|
-
if (!main) return false;
|
|
315
|
-
const style = getComputedStyle(main);
|
|
316
|
-
return parseInt(style.minHeight) > 0;
|
|
317
|
-
});
|
|
318
|
-
expect(result).toBe(true);
|
|
319
|
-
});
|
|
320
|
-
|
|
321
|
-
test('should have sticky positioned sidebar', async () => {
|
|
322
|
-
const result = await page.evaluate(() => {
|
|
323
|
-
const sidebar = document.querySelector('.sidebar');
|
|
324
|
-
if (!sidebar) return false;
|
|
325
|
-
const style = getComputedStyle(sidebar);
|
|
326
|
-
return style.position === 'sticky';
|
|
327
|
-
});
|
|
328
|
-
expect(result).toBe(true);
|
|
329
|
-
});
|
|
330
|
-
|
|
331
|
-
test('should have page elements with display:none by default', async () => {
|
|
332
|
-
const result = await page.evaluate(() => {
|
|
333
|
-
const inactivePage = document.querySelector('.page:not(.active)');
|
|
334
|
-
if (!inactivePage) return true; // If no inactive pages, test passes
|
|
335
|
-
const style = getComputedStyle(inactivePage);
|
|
336
|
-
return style.display === 'none';
|
|
337
|
-
});
|
|
338
|
-
expect(result).toBe(true);
|
|
339
|
-
});
|
|
340
|
-
|
|
341
|
-
test('should have .page.active with display:block', async () => {
|
|
342
|
-
const result = await page.evaluate(() => {
|
|
343
|
-
const activePage = document.querySelector('.page.active');
|
|
344
|
-
if (!activePage) return false;
|
|
345
|
-
const style = getComputedStyle(activePage);
|
|
346
|
-
return style.display === 'block';
|
|
347
|
-
});
|
|
348
|
-
expect(result).toBe(true);
|
|
349
|
-
});
|
|
350
|
-
});
|
|
351
|
-
|
|
352
|
-
// ========================================
|
|
353
|
-
// SIDEBAR NAVIGATION
|
|
354
|
-
// ========================================
|
|
355
|
-
|
|
356
|
-
describe('Sidebar Navigation', () => {
|
|
357
|
-
test('should have sidebar element', async () => {
|
|
358
|
-
const result = await page.evaluate(() => document.querySelector('.sidebar') !== null);
|
|
359
|
-
expect(result).toBe(true);
|
|
360
|
-
});
|
|
361
|
-
|
|
362
|
-
test('should have sidebar links with data-page attributes', async () => {
|
|
363
|
-
const result = await page.evaluate(() => {
|
|
364
|
-
const links = document.querySelectorAll('.sidebar-link[data-page]');
|
|
365
|
-
return links.length > 0;
|
|
366
|
-
});
|
|
367
|
-
expect(result).toBe(true);
|
|
368
|
-
});
|
|
369
|
-
|
|
370
|
-
test('should have one active sidebar link by default', async () => {
|
|
371
|
-
const result = await page.evaluate(() => {
|
|
372
|
-
const activeLinks = document.querySelectorAll('.sidebar-link.active');
|
|
373
|
-
return activeLinks.length === 1;
|
|
374
|
-
});
|
|
375
|
-
expect(result).toBe(true);
|
|
376
|
-
});
|
|
377
|
-
|
|
378
|
-
test('should have matching page elements for sidebar links', async () => {
|
|
379
|
-
const result = await page.evaluate(() => {
|
|
380
|
-
const links = document.querySelectorAll('.sidebar-link[data-page]');
|
|
381
|
-
for (const link of links) {
|
|
382
|
-
const pageId = link.getAttribute('data-page');
|
|
383
|
-
if (!document.getElementById(pageId)) return false;
|
|
384
|
-
}
|
|
385
|
-
return true;
|
|
386
|
-
});
|
|
387
|
-
expect(result).toBe(true);
|
|
388
|
-
});
|
|
389
|
-
|
|
390
|
-
test('should have exactly one active page by default', async () => {
|
|
391
|
-
const result = await page.evaluate(() => {
|
|
392
|
-
const activePages = document.querySelectorAll('.page.active');
|
|
393
|
-
return activePages.length === 1;
|
|
394
|
-
});
|
|
395
|
-
expect(result).toBe(true);
|
|
396
|
-
});
|
|
397
|
-
});
|
|
398
|
-
|
|
399
|
-
// ========================================
|
|
400
|
-
// NAVIGATION SCROLL RESET
|
|
401
|
-
// ========================================
|
|
402
|
-
|
|
403
|
-
describe('Navigation Scroll Reset', () => {
|
|
404
|
-
test('showPage function should exist', async () => {
|
|
405
|
-
const result = await page.evaluate(() => typeof window.showPage === 'function');
|
|
406
|
-
expect(result).toBe(true);
|
|
407
|
-
});
|
|
408
|
-
|
|
409
|
-
test('showPage should scroll to top', async () => {
|
|
410
|
-
const result = await page.evaluate(() => {
|
|
411
|
-
window.scrollTo(0, 100);
|
|
412
|
-
window.showPage('about');
|
|
413
|
-
return window.scrollY === 0;
|
|
414
|
-
});
|
|
415
|
-
expect(result).toBe(true);
|
|
416
|
-
});
|
|
417
|
-
});
|
|
418
|
-
|
|
419
|
-
// ========================================
|
|
420
|
-
// OTHER COMPONENTS
|
|
421
|
-
// ========================================
|
|
422
|
-
|
|
423
|
-
describe('Other Components', () => {
|
|
424
|
-
test('mu-checkbox should be registered', async () => {
|
|
425
|
-
const result = await page.evaluate(() => customElements.get('mu-checkbox') !== undefined);
|
|
426
|
-
expect(result).toBe(true);
|
|
427
|
-
});
|
|
428
|
-
|
|
429
|
-
test('mu-switch should be registered', async () => {
|
|
430
|
-
const result = await page.evaluate(() => customElements.get('mu-switch') !== undefined);
|
|
431
|
-
expect(result).toBe(true);
|
|
432
|
-
});
|
|
433
|
-
|
|
434
|
-
test('mu-option should be registered', async () => {
|
|
435
|
-
const result = await page.evaluate(() => customElements.get('mu-option') !== undefined);
|
|
436
|
-
expect(result).toBe(true);
|
|
437
|
-
});
|
|
438
|
-
});
|
|
439
|
-
});
|