microui-wc 0.1.1 → 0.1.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AGENTS.md +71 -71
- package/CHANGELOG.md +1 -1
- package/README.md +14 -9
- package/dist/AGENTS.md +71 -71
- package/dist/README.md +14 -9
- package/dist/microui.css +1 -1
- package/dist/microui.esm.js.map +1 -1
- package/dist/microui.min.js.map +1 -1
- package/docs/getting-started.md +3 -3
- package/package.json +39 -11
- package/src/components/mu-schema-form.js +1 -1
- package/.github/ISSUE_TEMPLATE/bug_report.md +0 -40
- package/.github/ISSUE_TEMPLATE/feature_request.md +0 -33
- package/.github/PULL_REQUEST_TEMPLATE.md +0 -28
- package/.github/workflows/ci.yml +0 -42
- package/.github/workflows/deploy-pages.yml +0 -112
- package/CODE_OF_CONDUCT.md +0 -59
- package/CONTRIBUTING.md +0 -156
- package/SECURITY.md +0 -58
- package/app/.generated/routes/alerts.js +0 -8
- package/app/.generated/routes/avatars.js +0 -8
- package/app/.generated/routes/badges.js +0 -8
- package/app/.generated/routes/buttons.js +0 -10
- package/app/.generated/routes/cards.js +0 -10
- package/app/.generated/routes/checkboxes.js +0 -9
- package/app/.generated/routes/chips.js +0 -8
- package/app/.generated/routes/dropdowns.js +0 -9
- package/app/.generated/routes/home.js +0 -7
- package/app/.generated/routes/icons.js +0 -9
- package/app/.generated/routes/inputs.js +0 -10
- package/app/.generated/routes/installation.js +0 -7
- package/app/.generated/routes/layout.js +0 -9
- package/app/.generated/routes/modals.js +0 -9
- package/app/.generated/routes/navbar.js +0 -7
- package/app/.generated/routes/progress.js +0 -9
- package/app/.generated/routes/radios.js +0 -9
- package/app/.generated/routes/switches.js +0 -9
- package/app/.generated/routes/tabs.js +0 -8
- package/app/.generated/routes/toasts.js +0 -9
- package/app/index.html +0 -67
- package/app/pages/alerts.html +0 -23
- package/app/pages/avatars.html +0 -22
- package/app/pages/badges.html +0 -22
- package/app/pages/buttons.html +0 -71
- package/app/pages/cards.html +0 -54
- package/app/pages/checkboxes.html +0 -39
- package/app/pages/chips.html +0 -23
- package/app/pages/dropdowns.html +0 -41
- package/app/pages/home.html +0 -59
- package/app/pages/icons.html +0 -29
- package/app/pages/inputs.html +0 -66
- package/app/pages/installation.html +0 -34
- package/app/pages/layout.html +0 -30
- package/app/pages/modals.html +0 -21
- package/app/pages/navbar.html +0 -22
- package/app/pages/progress.html +0 -35
- package/app/pages/radios.html +0 -40
- package/app/pages/switches.html +0 -39
- package/app/pages/tabs.html +0 -30
- package/app/pages/toasts.html +0 -22
- package/app-dist/index.html +0 -67
- package/app-dist/pages/alerts.html +0 -23
- package/app-dist/pages/avatars.html +0 -22
- package/app-dist/pages/badges.html +0 -22
- package/app-dist/pages/buttons.html +0 -71
- package/app-dist/pages/cards.html +0 -54
- package/app-dist/pages/checkboxes.html +0 -39
- package/app-dist/pages/chips.html +0 -23
- package/app-dist/pages/dropdowns.html +0 -41
- package/app-dist/pages/home.html +0 -59
- package/app-dist/pages/icons.html +0 -29
- package/app-dist/pages/inputs.html +0 -66
- package/app-dist/pages/installation.html +0 -34
- package/app-dist/pages/layout.html +0 -30
- package/app-dist/pages/modals.html +0 -21
- package/app-dist/pages/navbar.html +0 -22
- package/app-dist/pages/progress.html +0 -35
- package/app-dist/pages/radios.html +0 -40
- package/app-dist/pages/switches.html +0 -39
- package/app-dist/pages/tabs.html +0 -30
- package/app-dist/pages/toasts.html +0 -22
- package/app-dist/pages.json +0 -217
- package/app-dist/routes/alerts.js +0 -5
- package/app-dist/routes/avatars.js +0 -1
- package/app-dist/routes/badges.js +0 -1
- package/app-dist/routes/buttons.js +0 -1
- package/app-dist/routes/cards.js +0 -1
- package/app-dist/routes/checkboxes.js +0 -9
- package/app-dist/routes/chips.js +0 -4
- package/app-dist/routes/chunk-019e5e2f.js +0 -5
- package/app-dist/routes/chunk-0m4j19yd.js +0 -2
- package/app-dist/routes/chunk-0tmmp5q0.js +0 -1
- package/app-dist/routes/chunk-10xn709r.js +0 -1
- package/app-dist/routes/chunk-15m2qcda.js +0 -2
- package/app-dist/routes/chunk-1bh8g23n.js +0 -1
- package/app-dist/routes/chunk-1vg0v937.js +0 -1
- package/app-dist/routes/chunk-1zvcgy3j.js +0 -1
- package/app-dist/routes/chunk-2afb0861.js +0 -1
- package/app-dist/routes/chunk-2c6ttpzt.js +0 -5
- package/app-dist/routes/chunk-3dy30fhs.js +0 -1
- package/app-dist/routes/chunk-426dnces.js +0 -13
- package/app-dist/routes/chunk-44kgxery.js +0 -1
- package/app-dist/routes/chunk-47fdnejd.js +0 -33
- package/app-dist/routes/chunk-49a6t2vq.js +0 -1
- package/app-dist/routes/chunk-4fe1rm5b.js +0 -1
- package/app-dist/routes/chunk-4ggmvkta.js +0 -33
- package/app-dist/routes/chunk-4vkz81q7.js +0 -33
- package/app-dist/routes/chunk-4w4tmj8f.js +0 -31
- package/app-dist/routes/chunk-532s62kr.js +0 -31
- package/app-dist/routes/chunk-5hm3bssy.js +0 -33
- package/app-dist/routes/chunk-5vrh24hc.js +0 -1
- package/app-dist/routes/chunk-61pcg25a.js +0 -1
- package/app-dist/routes/chunk-6nfhygvf.js +0 -1
- package/app-dist/routes/chunk-700e7je6.js +0 -33
- package/app-dist/routes/chunk-7fsn17kg.js +0 -1
- package/app-dist/routes/chunk-7k789b32.js +0 -1
- package/app-dist/routes/chunk-7r46q0ys.js +0 -36
- package/app-dist/routes/chunk-86fmc1fr.js +0 -5
- package/app-dist/routes/chunk-8qth37vw.js +0 -1
- package/app-dist/routes/chunk-924wv8n0.js +0 -1
- package/app-dist/routes/chunk-9mbhgxk9.js +0 -1
- package/app-dist/routes/chunk-a216hyd9.js +0 -1
- package/app-dist/routes/chunk-akzxykh9.js +0 -33
- package/app-dist/routes/chunk-b3dcvy8c.js +0 -1
- package/app-dist/routes/chunk-b74zahz5.js +0 -31
- package/app-dist/routes/chunk-bftj53p2.js +0 -5
- package/app-dist/routes/chunk-c01hnz3e.js +0 -1
- package/app-dist/routes/chunk-d8pvv5km.js +0 -1
- package/app-dist/routes/chunk-dev0aezr.js +0 -2
- package/app-dist/routes/chunk-dh6vnv0e.js +0 -1
- package/app-dist/routes/chunk-dn2cbpva.js +0 -36
- package/app-dist/routes/chunk-dvn0my90.js +0 -1
- package/app-dist/routes/chunk-dvq8mnve.js +0 -36
- package/app-dist/routes/chunk-e8c2gc4d.js +0 -5
- package/app-dist/routes/chunk-ejf9ak2x.js +0 -1
- package/app-dist/routes/chunk-f083m55s.js +0 -1
- package/app-dist/routes/chunk-fnrj28s1.js +0 -31
- package/app-dist/routes/chunk-fvg3yjdp.js +0 -31
- package/app-dist/routes/chunk-g7k381n1.js +0 -1
- package/app-dist/routes/chunk-h01kq2ae.js +0 -13
- package/app-dist/routes/chunk-h4dk761v.js +0 -5
- package/app-dist/routes/chunk-hmx91z2x.js +0 -5
- package/app-dist/routes/chunk-hxbg4m42.js +0 -36
- package/app-dist/routes/chunk-jbjnfp2b.js +0 -2
- package/app-dist/routes/chunk-jxtz5vv6.js +0 -36
- package/app-dist/routes/chunk-jxzcs0ey.js +0 -36
- package/app-dist/routes/chunk-kt7wwhcx.js +0 -1
- package/app-dist/routes/chunk-kzptszyc.js +0 -33
- package/app-dist/routes/chunk-mhgca4w4.js +0 -2
- package/app-dist/routes/chunk-mhswxa20.js +0 -1
- package/app-dist/routes/chunk-n8zfeex6.js +0 -1
- package/app-dist/routes/chunk-pee47b2r.js +0 -1
- package/app-dist/routes/chunk-pesmw829.js +0 -1
- package/app-dist/routes/chunk-pgc4c6f3.js +0 -36
- package/app-dist/routes/chunk-q8egegm1.js +0 -1
- package/app-dist/routes/chunk-q9mn2qyq.js +0 -36
- package/app-dist/routes/chunk-qh0rtaf3.js +0 -5
- package/app-dist/routes/chunk-qqhmk6ye.js +0 -2
- package/app-dist/routes/chunk-qrxygmf7.js +0 -33
- package/app-dist/routes/chunk-r46yzksx.js +0 -36
- package/app-dist/routes/chunk-rgpbw2w0.js +0 -5
- package/app-dist/routes/chunk-rnpzv3d8.js +0 -2
- package/app-dist/routes/chunk-s5v8cv05.js +0 -2
- package/app-dist/routes/chunk-sbwn5bpc.js +0 -1
- package/app-dist/routes/chunk-sqbg8jbt.js +0 -33
- package/app-dist/routes/chunk-sv8dqnf7.js +0 -1
- package/app-dist/routes/chunk-t67sw3za.js +0 -1
- package/app-dist/routes/chunk-tjdpqwdf.js +0 -31
- package/app-dist/routes/chunk-tq2mfghg.js +0 -1
- package/app-dist/routes/chunk-ttn10vt6.js +0 -1
- package/app-dist/routes/chunk-v2hzpjxr.js +0 -1
- package/app-dist/routes/chunk-wfjjkw9y.js +0 -1
- package/app-dist/routes/chunk-wt8cxzmf.js +0 -31
- package/app-dist/routes/chunk-x45d372k.js +0 -5
- package/app-dist/routes/chunk-y3wsazkt.js +0 -1
- package/app-dist/routes/chunk-y7pmgc7t.js +0 -33
- package/app-dist/routes/chunk-zefdt2q3.js +0 -31
- package/app-dist/routes/dropdowns.js +0 -6
- package/app-dist/routes/home.js +0 -1
- package/app-dist/routes/icons.js +0 -1
- package/app-dist/routes/inputs.js +0 -12
- package/app-dist/routes/installation.js +0 -1
- package/app-dist/routes/layout.js +0 -1
- package/app-dist/routes/modals.js +0 -7
- package/app-dist/routes/navbar.js +0 -1
- package/app-dist/routes/progress.js +0 -1
- package/app-dist/routes/radios.js +0 -6
- package/app-dist/routes/switches.js +0 -6
- package/app-dist/routes/tabs.js +0 -1
- package/app-dist/routes/toasts.js +0 -16
- package/assets/fonts/material-symbols-mini.woff2 +0 -0
- package/assets/fonts/material-symbols.woff2 +0 -0
- package/assets/fonts/roboto-400.woff2 +0 -0
- package/assets/fonts/roboto-500.woff2 +0 -0
- package/assets/fonts/roboto-700.woff2 +0 -0
- package/assets/logo-banner-400.jpg +0 -0
- package/assets/logo-banner-400.webp +0 -0
- package/assets/logo-banner-800.webp +0 -0
- package/assets/logo-banner.jpg +0 -0
- package/assets/logo-icon-64.jpg +0 -0
- package/assets/logo-icon-64.webp +0 -0
- package/assets/logo-icon.jpg +0 -0
- package/assets/logo-square.jpg +0 -0
- package/bun.lock +0 -312
- package/bunfig.toml +0 -4
- package/custom-elements.json +0 -1916
- package/demo/api/sample-data.json +0 -38
- package/demo/content/alerts.html +0 -115
- package/demo/content/avatars.html +0 -70
- package/demo/content/badges.html +0 -65
- package/demo/content/buttons.html +0 -188
- package/demo/content/callouts.html +0 -91
- package/demo/content/cards.html +0 -121
- package/demo/content/checkboxes.html +0 -178
- package/demo/content/chips.html +0 -67
- package/demo/content/codeblocks.html +0 -101
- package/demo/content/confirms.html +0 -115
- package/demo/content/datatables.html +0 -149
- package/demo/content/dividers.html +0 -119
- package/demo/content/dropdowns.html +0 -89
- package/demo/content/enterprise.html +0 -252
- package/demo/content/home.html +0 -149
- package/demo/content/icons.html +0 -89
- package/demo/content/inputs.html +0 -135
- package/demo/content/installation.html +0 -16
- package/demo/content/layout.html +0 -136
- package/demo/content/modals.html +0 -141
- package/demo/content/navbar.html +0 -70
- package/demo/content/progress.html +0 -119
- package/demo/content/radios.html +0 -88
- package/demo/content/skeletons.html +0 -109
- package/demo/content/spinners.html +0 -96
- package/demo/content/switches.html +0 -84
- package/demo/content/tables.html +0 -124
- package/demo/content/tabs.html +0 -85
- package/demo/content/toasts.html +0 -116
- package/demo/content/tooltips.html +0 -107
- package/demo/content/virtual-lists.html +0 -233
- package/demo/favicon.ico +0 -0
- package/demo/favicon.png +0 -0
- package/demo/full.html +0 -52
- package/demo/iife.html +0 -46
- package/demo/manifest.json +0 -34
- package/demo/pages/datatable-demo.html +0 -237
- package/demo/pages/prompt-ui-demo.html +0 -218
- package/demo/pages/responsive-demo.html +0 -122
- package/demo/pages/schema-form-demo.html +0 -270
- package/demo/robots.txt +0 -6
- package/demo/shell.html +0 -712
- package/demo/sw.js +0 -387
- package/lighthouse-audit.mjs +0 -113
- package/scripts/analyze-components.js +0 -105
- package/scripts/build-app.js +0 -193
- package/scripts/build-framework.js +0 -444
- package/scripts/build-utils.js +0 -101
- package/scripts/test-isolated.js +0 -151
- package/server.js +0 -256
- package/tests/agents/agent-integration.test.js +0 -76
- package/tests/benchmark.html +0 -296
- package/tests/build/scan-components.test.js +0 -173
- package/tests/components/all-components.test.js +0 -245
- package/tests/components/all-missing-components.test.js +0 -574
- package/tests/components/mu-alert.test.js +0 -113
- package/tests/components/mu-avatar.test.js +0 -148
- package/tests/components/mu-badge.test.js +0 -92
- package/tests/components/mu-button.test.js +0 -112
- package/tests/components/mu-card.test.js +0 -89
- package/tests/components/mu-checkbox.test.js +0 -158
- package/tests/components/mu-chip.test.js +0 -118
- package/tests/components/mu-container.test.js +0 -120
- package/tests/components/mu-divider.test.js +0 -98
- package/tests/components/mu-drawer-item.test.js +0 -199
- package/tests/components/mu-drawer.test.js +0 -96
- package/tests/components/mu-dropdown.test.js +0 -125
- package/tests/components/mu-form.test.js +0 -138
- package/tests/components/mu-grid.test.js +0 -135
- package/tests/components/mu-icon.test.js +0 -110
- package/tests/components/mu-input.test.js +0 -131
- package/tests/components/mu-lazy.test.js +0 -103
- package/tests/components/mu-modal.test.js +0 -275
- package/tests/components/mu-navbar.test.js +0 -101
- package/tests/components/mu-progress.test.js +0 -115
- package/tests/components/mu-radio.test.js +0 -114
- package/tests/components/mu-repeat.test.js +0 -106
- package/tests/components/mu-sidebar.test.js +0 -126
- package/tests/components/mu-skeleton.test.js +0 -162
- package/tests/components/mu-stack.test.js +0 -143
- package/tests/components/mu-switch.test.js +0 -292
- package/tests/components/mu-table.test.js +0 -124
- package/tests/components/mu-tabs.test.js +0 -104
- package/tests/components/mu-textarea.test.js +0 -115
- package/tests/components/mu-toast.test.js +0 -321
- package/tests/components/mu-tooltip.test.js +0 -133
- package/tests/components/mu-virtual-list.test.js +0 -109
- package/tests/core/MuElement.test.js +0 -120
- package/tests/core/agent-api.test.js +0 -125
- package/tests/core/all-core-modules.test.js +0 -442
- package/tests/core/bus.test.js +0 -364
- package/tests/core/component-schema.test.js +0 -160
- package/tests/core/feature-registry.test.js +0 -198
- package/tests/core/form-state.test.js +0 -167
- package/tests/core/http.test.js +0 -119
- package/tests/core/keyboard.test.js +0 -319
- package/tests/core/layers.test.js +0 -129
- package/tests/core/namespaced-stores.test.js +0 -114
- package/tests/core/render.test.js +0 -121
- package/tests/core/ripple.test.js +0 -131
- package/tests/core/router.test.js +0 -89
- package/tests/core/scheduler.test.js +0 -121
- package/tests/core/signals.test.js +0 -128
- package/tests/core/store.test.js +0 -171
- package/tests/core/transitions.test.js +0 -82
- package/tests/e2e/accessibility-harness.html +0 -58
- package/tests/e2e/accessibility.test.js +0 -401
- package/tests/e2e/agent-features.test.js +0 -372
- package/tests/e2e/card-spacing.test.js +0 -287
- package/tests/e2e/components.test.js +0 -439
- package/tests/e2e/demo-routes.test.js +0 -478
- package/tests/e2e/layout-css-fallback.test.js +0 -334
- package/tests/e2e/mu-alert.e2e.test.js +0 -111
- package/tests/e2e/mu-checkbox.test.js +0 -489
- package/tests/e2e/mu-chip.test.js +0 -347
- package/tests/e2e/mu-form.test.js +0 -499
- package/tests/e2e/mu-icon.test.js +0 -114
- package/tests/e2e/mu-radio.test.js +0 -113
- package/tests/e2e/mu-skeleton.test.js +0 -140
- package/tests/e2e/mu-switch.test.js +0 -415
- package/tests/e2e/mu-tabs.test.js +0 -494
- package/tests/e2e/mu-textarea.test.js +0 -242
- package/tests/e2e/mu-virtual-list.test.js +0 -427
- package/tests/e2e/perf-memory.test.js +0 -161
- package/tests/e2e/puppeteer-helper.js +0 -137
- package/tests/e2e/puppeteer.test.js +0 -226
- package/tests/e2e/pwa.test.js +0 -261
- package/tests/e2e/test-harness.html +0 -319
- package/tests/manual/test-components.html +0 -120
- package/tests/memory-test.html +0 -309
- package/tests/setup-dom.js +0 -93
- package/tests/visual-test.html +0 -301
|
@@ -1,125 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview Unit Tests for agent-api.js Module
|
|
3
|
-
* Target: 35% → 70% coverage
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { describe, test, expect, beforeAll, beforeEach } from 'bun:test';
|
|
7
|
-
import { parseHTML } from 'linkedom';
|
|
8
|
-
|
|
9
|
-
let getMuComponentTree, findByLabel, enableVisualMarkers, disableVisualMarkers, getMCPActions, executeAction;
|
|
10
|
-
|
|
11
|
-
describe('agent-api Module Unit Tests', () => {
|
|
12
|
-
|
|
13
|
-
beforeAll(async () => {
|
|
14
|
-
const dom = parseHTML('<!DOCTYPE html><html><body></body></html>');
|
|
15
|
-
globalThis.document = dom.document;
|
|
16
|
-
globalThis.window = dom.window;
|
|
17
|
-
globalThis.HTMLElement = dom.HTMLElement;
|
|
18
|
-
globalThis.customElements = dom.customElements;
|
|
19
|
-
globalThis.requestAnimationFrame = (cb) => { cb(Date.now()); return 0; };
|
|
20
|
-
|
|
21
|
-
// Mock window dimensions
|
|
22
|
-
globalThis.window.innerHeight = 768;
|
|
23
|
-
globalThis.window.innerWidth = 1024;
|
|
24
|
-
|
|
25
|
-
const module = await import('../../src/core/agent-api.js');
|
|
26
|
-
getMuComponentTree = module.getMuComponentTree;
|
|
27
|
-
findByLabel = module.findByLabel;
|
|
28
|
-
enableVisualMarkers = module.enableVisualMarkers;
|
|
29
|
-
disableVisualMarkers = module.disableVisualMarkers;
|
|
30
|
-
getMCPActions = module.getMCPActions;
|
|
31
|
-
executeAction = module.executeAction;
|
|
32
|
-
});
|
|
33
|
-
|
|
34
|
-
beforeEach(() => {
|
|
35
|
-
globalThis.document.body.innerHTML = '';
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
// GET MU COMPONENT TREE
|
|
39
|
-
test('getMuComponentTree should be a function', () => {
|
|
40
|
-
expect(typeof getMuComponentTree).toBe('function');
|
|
41
|
-
});
|
|
42
|
-
|
|
43
|
-
test('getMuComponentTree should return array', () => {
|
|
44
|
-
const result = getMuComponentTree();
|
|
45
|
-
expect(Array.isArray(result)).toBe(true);
|
|
46
|
-
});
|
|
47
|
-
|
|
48
|
-
test('getMuComponentTree should return empty array for empty body', () => {
|
|
49
|
-
const result = getMuComponentTree();
|
|
50
|
-
expect(result.length).toBe(0);
|
|
51
|
-
});
|
|
52
|
-
|
|
53
|
-
// FIND BY LABEL
|
|
54
|
-
test('findByLabel should be a function', () => {
|
|
55
|
-
expect(typeof findByLabel).toBe('function');
|
|
56
|
-
});
|
|
57
|
-
|
|
58
|
-
test('findByLabel should return empty array for no matches', () => {
|
|
59
|
-
const result = findByLabel('nonexistent');
|
|
60
|
-
expect(Array.isArray(result)).toBe(true);
|
|
61
|
-
expect(result.length).toBe(0);
|
|
62
|
-
});
|
|
63
|
-
|
|
64
|
-
// ENABLE VISUAL MARKERS
|
|
65
|
-
test('enableVisualMarkers should be a function', () => {
|
|
66
|
-
expect(typeof enableVisualMarkers).toBe('function');
|
|
67
|
-
});
|
|
68
|
-
|
|
69
|
-
// DISABLE VISUAL MARKERS
|
|
70
|
-
test('disableVisualMarkers should be a function', () => {
|
|
71
|
-
expect(typeof disableVisualMarkers).toBe('function');
|
|
72
|
-
});
|
|
73
|
-
|
|
74
|
-
// GET MCP ACTIONS
|
|
75
|
-
test('getMCPActions should be a function', () => {
|
|
76
|
-
expect(typeof getMCPActions).toBe('function');
|
|
77
|
-
});
|
|
78
|
-
|
|
79
|
-
test('getMCPActions should return object with name', () => {
|
|
80
|
-
const result = getMCPActions();
|
|
81
|
-
expect(result.name).toBe('microui');
|
|
82
|
-
});
|
|
83
|
-
|
|
84
|
-
test('getMCPActions should have actions array', () => {
|
|
85
|
-
const result = getMCPActions();
|
|
86
|
-
expect(Array.isArray(result.actions)).toBe(true);
|
|
87
|
-
});
|
|
88
|
-
|
|
89
|
-
test('getMCPActions should have click_button action', () => {
|
|
90
|
-
const result = getMCPActions();
|
|
91
|
-
const clickAction = result.actions.find(a => a.name === 'click_button');
|
|
92
|
-
expect(clickAction).toBeDefined();
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
test('getMCPActions should have fill_input action', () => {
|
|
96
|
-
const result = getMCPActions();
|
|
97
|
-
const fillAction = result.actions.find(a => a.name === 'fill_input');
|
|
98
|
-
expect(fillAction).toBeDefined();
|
|
99
|
-
});
|
|
100
|
-
|
|
101
|
-
test('getMCPActions should have toggle_checkbox action', () => {
|
|
102
|
-
const result = getMCPActions();
|
|
103
|
-
const toggleAction = result.actions.find(a => a.name === 'toggle_checkbox');
|
|
104
|
-
expect(toggleAction).toBeDefined();
|
|
105
|
-
});
|
|
106
|
-
|
|
107
|
-
// Note: executeAction is internal MCP handler, not a direct export
|
|
108
|
-
// MCP actions are executed through the getMCPActions interface
|
|
109
|
-
|
|
110
|
-
// OPTIONS
|
|
111
|
-
test('getMuComponentTree should accept visibleOnly option', () => {
|
|
112
|
-
const result = getMuComponentTree(globalThis.document.body, { visibleOnly: true });
|
|
113
|
-
expect(Array.isArray(result)).toBe(true);
|
|
114
|
-
});
|
|
115
|
-
|
|
116
|
-
test('getMuComponentTree should accept interactiveOnly option', () => {
|
|
117
|
-
const result = getMuComponentTree(globalThis.document.body, { interactiveOnly: true });
|
|
118
|
-
expect(Array.isArray(result)).toBe(true);
|
|
119
|
-
});
|
|
120
|
-
|
|
121
|
-
test('getMuComponentTree should accept types option', () => {
|
|
122
|
-
const result = getMuComponentTree(globalThis.document.body, { types: ['mu-button'] });
|
|
123
|
-
expect(Array.isArray(result)).toBe(true);
|
|
124
|
-
});
|
|
125
|
-
});
|
|
@@ -1,442 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview Unit Tests for microUI Core Modules
|
|
3
|
-
* Coverage: bus, signals, store, theme, router, http, scheduler, transitions, render, ripple
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { describe, test, expect, beforeAll, beforeEach } from 'bun:test';
|
|
7
|
-
|
|
8
|
-
// Setup linkedom for DOM testing
|
|
9
|
-
import { parseHTML } from 'linkedom';
|
|
10
|
-
const { document, customElements, HTMLElement, window } = parseHTML('<!DOCTYPE html><html><body></body></html>');
|
|
11
|
-
globalThis.document = document;
|
|
12
|
-
globalThis.customElements = customElements;
|
|
13
|
-
globalThis.HTMLElement = HTMLElement;
|
|
14
|
-
globalThis.window = window;
|
|
15
|
-
globalThis.window.location = { hash: '', href: '' };
|
|
16
|
-
globalThis.localStorage = {
|
|
17
|
-
_data: {},
|
|
18
|
-
getItem(k) { return this._data[k] || null; },
|
|
19
|
-
setItem(k, v) { this._data[k] = v; },
|
|
20
|
-
removeItem(k) { delete this._data[k]; },
|
|
21
|
-
clear() { this._data = {}; }
|
|
22
|
-
};
|
|
23
|
-
globalThis.CustomEvent = class CustomEvent extends Event {
|
|
24
|
-
constructor(type, options = {}) {
|
|
25
|
-
super(type, options);
|
|
26
|
-
this.detail = options.detail;
|
|
27
|
-
}
|
|
28
|
-
};
|
|
29
|
-
// Mock requestAnimationFrame
|
|
30
|
-
globalThis.requestAnimationFrame = (cb) => setTimeout(cb, 16);
|
|
31
|
-
globalThis.cancelAnimationFrame = (id) => clearTimeout(id);
|
|
32
|
-
|
|
33
|
-
// ============================================
|
|
34
|
-
// BUS MODULE (SignalBus Wrapper)
|
|
35
|
-
// ============================================
|
|
36
|
-
|
|
37
|
-
describe('bus Module', () => {
|
|
38
|
-
let bus, UIEvents;
|
|
39
|
-
|
|
40
|
-
beforeAll(async () => {
|
|
41
|
-
const module = await import('../../src/core/bus.js');
|
|
42
|
-
bus = module.bus;
|
|
43
|
-
UIEvents = module.UIEvents;
|
|
44
|
-
});
|
|
45
|
-
|
|
46
|
-
test('should export bus singleton', () => {
|
|
47
|
-
expect(bus).toBeDefined();
|
|
48
|
-
expect(typeof bus.on).toBe('function');
|
|
49
|
-
expect(typeof bus.emit).toBe('function');
|
|
50
|
-
});
|
|
51
|
-
|
|
52
|
-
test('should export UIEvents object', () => {
|
|
53
|
-
expect(UIEvents).toBeDefined();
|
|
54
|
-
expect(typeof UIEvents).toBe('object');
|
|
55
|
-
});
|
|
56
|
-
|
|
57
|
-
test('bus.on should return unsubscribe function', () => {
|
|
58
|
-
const unsub = bus.on('test:event', () => { });
|
|
59
|
-
expect(typeof unsub).toBe('function');
|
|
60
|
-
unsub(); // cleanup
|
|
61
|
-
});
|
|
62
|
-
|
|
63
|
-
test('bus.emit should work without errors', () => {
|
|
64
|
-
expect(() => bus.emit('test:event', { data: 123 })).not.toThrow();
|
|
65
|
-
});
|
|
66
|
-
|
|
67
|
-
test('should deliver events to subscribers', () => {
|
|
68
|
-
let received = null;
|
|
69
|
-
const unsub = bus.on('delivery:test', (data) => {
|
|
70
|
-
received = data;
|
|
71
|
-
});
|
|
72
|
-
bus.emit('delivery:test', { value: 42 });
|
|
73
|
-
expect(received).toEqual({ value: 42 });
|
|
74
|
-
unsub();
|
|
75
|
-
});
|
|
76
|
-
});
|
|
77
|
-
|
|
78
|
-
// ============================================
|
|
79
|
-
// SIGNALS MODULE (Reactive State)
|
|
80
|
-
// ============================================
|
|
81
|
-
|
|
82
|
-
describe('signals Module', () => {
|
|
83
|
-
let signal, computed, effect, batch;
|
|
84
|
-
|
|
85
|
-
beforeAll(async () => {
|
|
86
|
-
const module = await import('../../src/core/signals.js');
|
|
87
|
-
signal = module.signal;
|
|
88
|
-
computed = module.computed;
|
|
89
|
-
effect = module.effect;
|
|
90
|
-
batch = module.batch;
|
|
91
|
-
});
|
|
92
|
-
|
|
93
|
-
test('should export signal function', () => {
|
|
94
|
-
expect(typeof signal).toBe('function');
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
test('should export computed function', () => {
|
|
98
|
-
expect(typeof computed).toBe('function');
|
|
99
|
-
});
|
|
100
|
-
|
|
101
|
-
test('should export effect function', () => {
|
|
102
|
-
expect(typeof effect).toBe('function');
|
|
103
|
-
});
|
|
104
|
-
|
|
105
|
-
test('should export batch function', () => {
|
|
106
|
-
expect(typeof batch).toBe('function');
|
|
107
|
-
});
|
|
108
|
-
|
|
109
|
-
// SKIP: These tests fail in linkedom because:
|
|
110
|
-
// 1. Function.name is readonly in strict mode (signals.js line 59: read.name = id)
|
|
111
|
-
// 2. The signals work correctly in browser environment
|
|
112
|
-
test.skip('signal should create reactive value with functional accessor (linkedom limitation)', () => { });
|
|
113
|
-
test.skip('signal.set should update value (linkedom limitation)', () => { });
|
|
114
|
-
test.skip('signal.update should work with function (linkedom limitation)', () => { });
|
|
115
|
-
test.skip('signal.peek should return value without tracking (linkedom limitation)', () => { });
|
|
116
|
-
test.skip('computed should derive value (linkedom limitation)', () => { });
|
|
117
|
-
|
|
118
|
-
test('effect should run immediately', () => {
|
|
119
|
-
let ran = false;
|
|
120
|
-
const dispose = effect(() => {
|
|
121
|
-
ran = true;
|
|
122
|
-
});
|
|
123
|
-
expect(ran).toBe(true);
|
|
124
|
-
dispose();
|
|
125
|
-
});
|
|
126
|
-
|
|
127
|
-
test('effect should return dispose function', () => {
|
|
128
|
-
const dispose = effect(() => { });
|
|
129
|
-
expect(typeof dispose).toBe('function');
|
|
130
|
-
dispose();
|
|
131
|
-
});
|
|
132
|
-
});
|
|
133
|
-
|
|
134
|
-
// ============================================
|
|
135
|
-
// STORE MODULE (State Management)
|
|
136
|
-
// ============================================
|
|
137
|
-
|
|
138
|
-
describe('store Module', () => {
|
|
139
|
-
let createStore;
|
|
140
|
-
|
|
141
|
-
beforeAll(async () => {
|
|
142
|
-
const module = await import('../../src/core/store.js');
|
|
143
|
-
createStore = module.createStore;
|
|
144
|
-
});
|
|
145
|
-
|
|
146
|
-
test('should export createStore function', () => {
|
|
147
|
-
expect(typeof createStore).toBe('function');
|
|
148
|
-
});
|
|
149
|
-
|
|
150
|
-
test('createStore should return store object', () => {
|
|
151
|
-
const store = createStore({ count: 0, user: null });
|
|
152
|
-
expect(store).toBeDefined();
|
|
153
|
-
expect(typeof store.get).toBe('function');
|
|
154
|
-
expect(typeof store.set).toBe('function');
|
|
155
|
-
});
|
|
156
|
-
|
|
157
|
-
test('store.get should return entire state', () => {
|
|
158
|
-
const store = createStore({ count: 42, name: 'test' });
|
|
159
|
-
const state = store.get();
|
|
160
|
-
expect(state.count).toBe(42);
|
|
161
|
-
expect(state.name).toBe('test');
|
|
162
|
-
});
|
|
163
|
-
|
|
164
|
-
test('store.set should merge new state', () => {
|
|
165
|
-
const store = createStore({ count: 0, name: 'old' });
|
|
166
|
-
store.set({ count: 10 });
|
|
167
|
-
expect(store.get().count).toBe(10);
|
|
168
|
-
expect(store.get().name).toBe('old'); // preserved
|
|
169
|
-
});
|
|
170
|
-
|
|
171
|
-
test('store.update should update with function', () => {
|
|
172
|
-
const store = createStore({ count: 5 });
|
|
173
|
-
store.update(s => ({ count: s.count + 10 }));
|
|
174
|
-
expect(store.get().count).toBe(15);
|
|
175
|
-
});
|
|
176
|
-
|
|
177
|
-
test('store.subscribe should notify on change', () => {
|
|
178
|
-
const store = createStore({ count: 0 });
|
|
179
|
-
let notifiedState = null;
|
|
180
|
-
const unsub = store.subscribe((state) => {
|
|
181
|
-
notifiedState = state;
|
|
182
|
-
});
|
|
183
|
-
// subscribe calls immediately with current state
|
|
184
|
-
expect(notifiedState).toEqual({ count: 0 });
|
|
185
|
-
store.set({ count: 5 });
|
|
186
|
-
expect(notifiedState.count).toBe(5);
|
|
187
|
-
unsub();
|
|
188
|
-
});
|
|
189
|
-
|
|
190
|
-
test('store.reset should restore initial state', () => {
|
|
191
|
-
const store = createStore({ count: 0 });
|
|
192
|
-
store.set({ count: 100 });
|
|
193
|
-
store.reset();
|
|
194
|
-
expect(store.get().count).toBe(0);
|
|
195
|
-
});
|
|
196
|
-
});
|
|
197
|
-
|
|
198
|
-
// ============================================
|
|
199
|
-
// THEME MODULE (Dark/Light Mode)
|
|
200
|
-
// ============================================
|
|
201
|
-
|
|
202
|
-
describe('theme Module', () => {
|
|
203
|
-
// NOTE: Theme module auto-initializes on import and requires window.matchMedia
|
|
204
|
-
// which is not available in linkedom. These tests verify exports only.
|
|
205
|
-
|
|
206
|
-
test.skip('Theme module tests skipped (requires matchMedia - linkedom limitation)', () => { });
|
|
207
|
-
});
|
|
208
|
-
|
|
209
|
-
// ============================================
|
|
210
|
-
// ROUTER MODULE (SPA Navigation) - Singleton Export
|
|
211
|
-
// ============================================
|
|
212
|
-
|
|
213
|
-
describe('router Module', () => {
|
|
214
|
-
let Router;
|
|
215
|
-
|
|
216
|
-
beforeAll(async () => {
|
|
217
|
-
const module = await import('../../src/core/router.js');
|
|
218
|
-
Router = module.Router;
|
|
219
|
-
});
|
|
220
|
-
|
|
221
|
-
test('should export Router singleton', () => {
|
|
222
|
-
expect(Router).toBeDefined();
|
|
223
|
-
expect(typeof Router).toBe('object');
|
|
224
|
-
});
|
|
225
|
-
|
|
226
|
-
test('Router should have on method', () => {
|
|
227
|
-
expect(typeof Router.on).toBe('function');
|
|
228
|
-
});
|
|
229
|
-
|
|
230
|
-
test('Router should have navigate method', () => {
|
|
231
|
-
expect(typeof Router.navigate).toBe('function');
|
|
232
|
-
});
|
|
233
|
-
|
|
234
|
-
test('Router should have start method', () => {
|
|
235
|
-
expect(typeof Router.start).toBe('function');
|
|
236
|
-
});
|
|
237
|
-
|
|
238
|
-
test('Router should have current getter', () => {
|
|
239
|
-
expect(Router.current).toBeDefined();
|
|
240
|
-
});
|
|
241
|
-
});
|
|
242
|
-
|
|
243
|
-
// ============================================
|
|
244
|
-
// HTTP MODULE (Fetch Wrapper)
|
|
245
|
-
// ============================================
|
|
246
|
-
|
|
247
|
-
describe('http Module', () => {
|
|
248
|
-
let http, HttpError;
|
|
249
|
-
|
|
250
|
-
beforeAll(async () => {
|
|
251
|
-
const module = await import('../../src/core/http.js');
|
|
252
|
-
http = module.http;
|
|
253
|
-
HttpError = module.HttpError;
|
|
254
|
-
});
|
|
255
|
-
|
|
256
|
-
test('should export http object', () => {
|
|
257
|
-
expect(http).toBeDefined();
|
|
258
|
-
});
|
|
259
|
-
|
|
260
|
-
test('http should have get method', () => {
|
|
261
|
-
expect(typeof http.get).toBe('function');
|
|
262
|
-
});
|
|
263
|
-
|
|
264
|
-
test('http should have post method', () => {
|
|
265
|
-
expect(typeof http.post).toBe('function');
|
|
266
|
-
});
|
|
267
|
-
|
|
268
|
-
test('http should have put method', () => {
|
|
269
|
-
expect(typeof http.put).toBe('function');
|
|
270
|
-
});
|
|
271
|
-
|
|
272
|
-
test('http should have delete method', () => {
|
|
273
|
-
expect(typeof http.delete).toBe('function');
|
|
274
|
-
});
|
|
275
|
-
|
|
276
|
-
test('should export HttpError class', () => {
|
|
277
|
-
expect(HttpError).toBeDefined();
|
|
278
|
-
expect(typeof HttpError).toBe('function');
|
|
279
|
-
});
|
|
280
|
-
|
|
281
|
-
test('HttpError instance should have correct properties', () => {
|
|
282
|
-
const err = new HttpError(404, 'Not Found', 'Resource missing');
|
|
283
|
-
expect(err.status).toBe(404);
|
|
284
|
-
expect(err.statusText).toBe('Not Found');
|
|
285
|
-
expect(err.body).toBe('Resource missing');
|
|
286
|
-
expect(err.message).toBe('HTTP 404: Not Found');
|
|
287
|
-
});
|
|
288
|
-
});
|
|
289
|
-
|
|
290
|
-
// ============================================
|
|
291
|
-
// SCHEDULER MODULE (Task Scheduling)
|
|
292
|
-
// ============================================
|
|
293
|
-
|
|
294
|
-
describe('scheduler Module', () => {
|
|
295
|
-
let scheduleTask, yieldToMain, runBackground;
|
|
296
|
-
|
|
297
|
-
beforeAll(async () => {
|
|
298
|
-
const module = await import('../../src/core/scheduler.js');
|
|
299
|
-
scheduleTask = module.scheduleTask;
|
|
300
|
-
yieldToMain = module.yieldToMain;
|
|
301
|
-
runBackground = module.runBackground;
|
|
302
|
-
});
|
|
303
|
-
|
|
304
|
-
test('should export scheduleTask function', () => {
|
|
305
|
-
expect(typeof scheduleTask).toBe('function');
|
|
306
|
-
});
|
|
307
|
-
|
|
308
|
-
test('should export yieldToMain function', () => {
|
|
309
|
-
expect(typeof yieldToMain).toBe('function');
|
|
310
|
-
});
|
|
311
|
-
|
|
312
|
-
test('should export runBackground function', () => {
|
|
313
|
-
expect(typeof runBackground).toBe('function');
|
|
314
|
-
});
|
|
315
|
-
|
|
316
|
-
test('scheduleTask should return promise', async () => {
|
|
317
|
-
const result = scheduleTask(() => 42);
|
|
318
|
-
expect(result instanceof Promise).toBe(true);
|
|
319
|
-
});
|
|
320
|
-
|
|
321
|
-
test('yieldToMain should return promise', async () => {
|
|
322
|
-
const result = yieldToMain();
|
|
323
|
-
expect(result instanceof Promise).toBe(true);
|
|
324
|
-
});
|
|
325
|
-
});
|
|
326
|
-
|
|
327
|
-
// ============================================
|
|
328
|
-
// TRANSITIONS MODULE (View Transitions)
|
|
329
|
-
// ============================================
|
|
330
|
-
|
|
331
|
-
describe('transitions Module', () => {
|
|
332
|
-
let transition, supportsViewTransitions;
|
|
333
|
-
|
|
334
|
-
beforeAll(async () => {
|
|
335
|
-
const module = await import('../../src/core/transitions.js');
|
|
336
|
-
transition = module.transition;
|
|
337
|
-
supportsViewTransitions = module.supportsViewTransitions;
|
|
338
|
-
});
|
|
339
|
-
|
|
340
|
-
test('should export transition function', () => {
|
|
341
|
-
expect(typeof transition).toBe('function');
|
|
342
|
-
});
|
|
343
|
-
|
|
344
|
-
test('should export supportsViewTransitions', () => {
|
|
345
|
-
expect(typeof supportsViewTransitions).toBe('boolean');
|
|
346
|
-
});
|
|
347
|
-
|
|
348
|
-
test('transition should return promise', async () => {
|
|
349
|
-
const result = transition(() => { });
|
|
350
|
-
expect(result instanceof Promise).toBe(true);
|
|
351
|
-
});
|
|
352
|
-
|
|
353
|
-
test('transition should execute callback', async () => {
|
|
354
|
-
let executed = false;
|
|
355
|
-
await transition(() => {
|
|
356
|
-
executed = true;
|
|
357
|
-
});
|
|
358
|
-
expect(executed).toBe(true);
|
|
359
|
-
});
|
|
360
|
-
});
|
|
361
|
-
|
|
362
|
-
// ============================================
|
|
363
|
-
// RENDER MODULE (DOM Utilities)
|
|
364
|
-
// ============================================
|
|
365
|
-
|
|
366
|
-
describe('render Module', () => {
|
|
367
|
-
let memo, debounce, throttle;
|
|
368
|
-
|
|
369
|
-
beforeAll(async () => {
|
|
370
|
-
const module = await import('../../src/core/render.js');
|
|
371
|
-
memo = module.memo;
|
|
372
|
-
debounce = module.debounce;
|
|
373
|
-
throttle = module.throttle;
|
|
374
|
-
});
|
|
375
|
-
|
|
376
|
-
test('should export memo function', () => {
|
|
377
|
-
expect(typeof memo).toBe('function');
|
|
378
|
-
});
|
|
379
|
-
|
|
380
|
-
test('should export debounce function', () => {
|
|
381
|
-
expect(typeof debounce).toBe('function');
|
|
382
|
-
});
|
|
383
|
-
|
|
384
|
-
test('should export throttle function', () => {
|
|
385
|
-
expect(typeof throttle).toBe('function');
|
|
386
|
-
});
|
|
387
|
-
|
|
388
|
-
test('memo should cache results', () => {
|
|
389
|
-
let callCount = 0;
|
|
390
|
-
const expensive = memo((x) => {
|
|
391
|
-
callCount++;
|
|
392
|
-
return x * 2;
|
|
393
|
-
});
|
|
394
|
-
expect(expensive(5)).toBe(10);
|
|
395
|
-
expect(expensive(5)).toBe(10);
|
|
396
|
-
expect(callCount).toBe(1); // cached
|
|
397
|
-
});
|
|
398
|
-
|
|
399
|
-
test('debounce should return function', () => {
|
|
400
|
-
const debounced = debounce(() => { }, 100);
|
|
401
|
-
expect(typeof debounced).toBe('function');
|
|
402
|
-
});
|
|
403
|
-
|
|
404
|
-
test('throttle should return function', () => {
|
|
405
|
-
const throttled = throttle(() => { }, 100);
|
|
406
|
-
expect(typeof throttled).toBe('function');
|
|
407
|
-
});
|
|
408
|
-
});
|
|
409
|
-
|
|
410
|
-
// ============================================
|
|
411
|
-
// RIPPLE MODULE (MD3 Ripple Effect)
|
|
412
|
-
// ============================================
|
|
413
|
-
|
|
414
|
-
describe('ripple Module', () => {
|
|
415
|
-
let createRipple, attachRipple, RippleMixin;
|
|
416
|
-
|
|
417
|
-
beforeAll(async () => {
|
|
418
|
-
const module = await import('../../src/core/ripple.js');
|
|
419
|
-
createRipple = module.createRipple;
|
|
420
|
-
attachRipple = module.attachRipple;
|
|
421
|
-
RippleMixin = module.RippleMixin;
|
|
422
|
-
});
|
|
423
|
-
|
|
424
|
-
test('should export createRipple function', () => {
|
|
425
|
-
expect(typeof createRipple).toBe('function');
|
|
426
|
-
});
|
|
427
|
-
|
|
428
|
-
test('should export attachRipple function', () => {
|
|
429
|
-
expect(typeof attachRipple).toBe('function');
|
|
430
|
-
});
|
|
431
|
-
|
|
432
|
-
test('should export RippleMixin', () => {
|
|
433
|
-
expect(RippleMixin).toBeDefined();
|
|
434
|
-
});
|
|
435
|
-
|
|
436
|
-
test('attachRipple should return cleanup function', () => {
|
|
437
|
-
const el = document.createElement('div');
|
|
438
|
-
const cleanup = attachRipple(el);
|
|
439
|
-
expect(typeof cleanup).toBe('function');
|
|
440
|
-
cleanup();
|
|
441
|
-
});
|
|
442
|
-
});
|