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,198 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview Tests for Feature Registry (Enterprise Features)
|
|
3
|
-
* Tests createFeature, getFeatures, getFeature, getFeatureComponents, destroyFeature
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { describe, test, expect, beforeEach, beforeAll } from 'bun:test';
|
|
7
|
-
|
|
8
|
-
let createFeature, getFeatures, getFeature, getFeatureComponents;
|
|
9
|
-
let getFeatureRoutes, getFeatureByRoute, getFeatureByComponent;
|
|
10
|
-
let destroyFeature, getFeatureSummary, clearFeatures, FeatureRegistry;
|
|
11
|
-
|
|
12
|
-
describe('Feature Registry', () => {
|
|
13
|
-
|
|
14
|
-
beforeAll(async () => {
|
|
15
|
-
const module = await import('../../src/core/feature-registry.js');
|
|
16
|
-
createFeature = module.createFeature;
|
|
17
|
-
getFeatures = module.getFeatures;
|
|
18
|
-
getFeature = module.getFeature;
|
|
19
|
-
getFeatureComponents = module.getFeatureComponents;
|
|
20
|
-
getFeatureRoutes = module.getFeatureRoutes;
|
|
21
|
-
getFeatureByRoute = module.getFeatureByRoute;
|
|
22
|
-
getFeatureByComponent = module.getFeatureByComponent;
|
|
23
|
-
destroyFeature = module.destroyFeature;
|
|
24
|
-
getFeatureSummary = module.getFeatureSummary;
|
|
25
|
-
clearFeatures = module.clearFeatures;
|
|
26
|
-
FeatureRegistry = module.FeatureRegistry;
|
|
27
|
-
});
|
|
28
|
-
|
|
29
|
-
beforeEach(() => { clearFeatures(); });
|
|
30
|
-
|
|
31
|
-
// ========================================
|
|
32
|
-
// createFeature
|
|
33
|
-
// ========================================
|
|
34
|
-
|
|
35
|
-
test('createFeature creates feature with name', () => {
|
|
36
|
-
const f = createFeature('user', { routes: ['/user'] });
|
|
37
|
-
expect(f.name).toBe('user');
|
|
38
|
-
});
|
|
39
|
-
|
|
40
|
-
test('createFeature returns existing for same name', () => {
|
|
41
|
-
const f1 = createFeature('dup', {});
|
|
42
|
-
const f2 = createFeature('dup', { routes: ['/x'] });
|
|
43
|
-
expect(f1).toBe(f2);
|
|
44
|
-
});
|
|
45
|
-
|
|
46
|
-
test('createFeature with init calls init', () => {
|
|
47
|
-
let called = false;
|
|
48
|
-
createFeature('init-test', { init: () => { called = true; } });
|
|
49
|
-
expect(called).toBe(true);
|
|
50
|
-
});
|
|
51
|
-
|
|
52
|
-
test('createFeature with init sets initialized', () => {
|
|
53
|
-
const f = createFeature('init-flag', { init: () => { } });
|
|
54
|
-
expect(f.initialized).toBe(true);
|
|
55
|
-
});
|
|
56
|
-
|
|
57
|
-
test('createFeature sets registeredAt', () => {
|
|
58
|
-
const f = createFeature('time', {});
|
|
59
|
-
expect(typeof f.registeredAt).toBe('number');
|
|
60
|
-
});
|
|
61
|
-
|
|
62
|
-
test('createFeature handles init error', () => {
|
|
63
|
-
const f = createFeature('err', { init: () => { throw new Error('fail'); } });
|
|
64
|
-
expect(f.initialized).toBe(false);
|
|
65
|
-
});
|
|
66
|
-
|
|
67
|
-
// ========================================
|
|
68
|
-
// getFeatures / getFeature
|
|
69
|
-
// ========================================
|
|
70
|
-
|
|
71
|
-
test('getFeatures returns all features', () => {
|
|
72
|
-
createFeature('a', {});
|
|
73
|
-
createFeature('b', {});
|
|
74
|
-
const all = getFeatures();
|
|
75
|
-
expect(all.a).toBeDefined();
|
|
76
|
-
expect(all.b).toBeDefined();
|
|
77
|
-
});
|
|
78
|
-
|
|
79
|
-
test('getFeature returns feature by name', () => {
|
|
80
|
-
createFeature('single', { meta: { v: 1 } });
|
|
81
|
-
const f = getFeature('single');
|
|
82
|
-
expect(f.config.meta.v).toBe(1);
|
|
83
|
-
});
|
|
84
|
-
|
|
85
|
-
test('getFeature returns undefined for unknown', () => {
|
|
86
|
-
expect(getFeature('unknown')).toBeUndefined();
|
|
87
|
-
});
|
|
88
|
-
|
|
89
|
-
// ========================================
|
|
90
|
-
// getFeatureComponents / getFeatureRoutes
|
|
91
|
-
// ========================================
|
|
92
|
-
|
|
93
|
-
test('getFeatureComponents returns components', () => {
|
|
94
|
-
createFeature('comp', { components: ['mu-card', 'mu-form'] });
|
|
95
|
-
expect(getFeatureComponents('comp')).toEqual(['mu-card', 'mu-form']);
|
|
96
|
-
});
|
|
97
|
-
|
|
98
|
-
test('getFeatureRoutes returns routes', () => {
|
|
99
|
-
createFeature('route', { routes: ['/a', '/b'] });
|
|
100
|
-
expect(getFeatureRoutes('route')).toEqual(['/a', '/b']);
|
|
101
|
-
});
|
|
102
|
-
|
|
103
|
-
test('getFeatureComponents returns empty for unknown', () => {
|
|
104
|
-
expect(getFeatureComponents('nope')).toEqual([]);
|
|
105
|
-
});
|
|
106
|
-
|
|
107
|
-
test('getFeatureRoutes returns empty for unknown', () => {
|
|
108
|
-
expect(getFeatureRoutes('nope')).toEqual([]);
|
|
109
|
-
});
|
|
110
|
-
|
|
111
|
-
// ========================================
|
|
112
|
-
// getFeatureByRoute / getFeatureByComponent
|
|
113
|
-
// ========================================
|
|
114
|
-
|
|
115
|
-
test('getFeatureByRoute finds by route prefix', () => {
|
|
116
|
-
createFeature('admin', { routes: ['/admin'] });
|
|
117
|
-
const f = getFeatureByRoute('/admin/users');
|
|
118
|
-
expect(f.name).toBe('admin');
|
|
119
|
-
});
|
|
120
|
-
|
|
121
|
-
test('getFeatureByRoute returns null if no match', () => {
|
|
122
|
-
expect(getFeatureByRoute('/nonexistent')).toBeNull();
|
|
123
|
-
});
|
|
124
|
-
|
|
125
|
-
test('getFeatureByComponent finds by component tag', () => {
|
|
126
|
-
createFeature('widgets', { components: ['mu-widget'] });
|
|
127
|
-
const f = getFeatureByComponent('mu-widget');
|
|
128
|
-
expect(f.name).toBe('widgets');
|
|
129
|
-
});
|
|
130
|
-
|
|
131
|
-
test('getFeatureByComponent returns null if no match', () => {
|
|
132
|
-
expect(getFeatureByComponent('mu-unknown')).toBeNull();
|
|
133
|
-
});
|
|
134
|
-
|
|
135
|
-
// ========================================
|
|
136
|
-
// destroyFeature
|
|
137
|
-
// ========================================
|
|
138
|
-
|
|
139
|
-
test('destroyFeature removes feature', () => {
|
|
140
|
-
createFeature('destroy-test', {});
|
|
141
|
-
destroyFeature('destroy-test');
|
|
142
|
-
expect(getFeature('destroy-test')).toBeUndefined();
|
|
143
|
-
});
|
|
144
|
-
|
|
145
|
-
test('destroyFeature calls destroy callback', () => {
|
|
146
|
-
let called = false;
|
|
147
|
-
createFeature('destroy-cb', { destroy: () => { called = true; } });
|
|
148
|
-
destroyFeature('destroy-cb');
|
|
149
|
-
expect(called).toBe(true);
|
|
150
|
-
});
|
|
151
|
-
|
|
152
|
-
test('destroyFeature handles non-existent', () => {
|
|
153
|
-
destroyFeature('nope');
|
|
154
|
-
expect(true).toBe(true);
|
|
155
|
-
});
|
|
156
|
-
|
|
157
|
-
test('destroyFeature handles destroy error', () => {
|
|
158
|
-
createFeature('destroy-err', { destroy: () => { throw new Error('fail'); } });
|
|
159
|
-
destroyFeature('destroy-err');
|
|
160
|
-
expect(getFeature('destroy-err')).toBeUndefined();
|
|
161
|
-
});
|
|
162
|
-
|
|
163
|
-
// ========================================
|
|
164
|
-
// getFeatureSummary
|
|
165
|
-
// ========================================
|
|
166
|
-
|
|
167
|
-
test('getFeatureSummary returns summary object', () => {
|
|
168
|
-
createFeature('s1', { routes: ['/x'], components: ['a'], store: {} });
|
|
169
|
-
const summary = getFeatureSummary();
|
|
170
|
-
expect(summary.totalFeatures).toBe(1);
|
|
171
|
-
expect(summary.totalRoutes).toBe(1);
|
|
172
|
-
expect(summary.totalComponents).toBe(1);
|
|
173
|
-
expect(summary.features[0].hasStore).toBe(true);
|
|
174
|
-
});
|
|
175
|
-
|
|
176
|
-
// ========================================
|
|
177
|
-
// clearFeatures
|
|
178
|
-
// ========================================
|
|
179
|
-
|
|
180
|
-
test('clearFeatures removes all features', () => {
|
|
181
|
-
createFeature('a', {});
|
|
182
|
-
createFeature('b', {});
|
|
183
|
-
clearFeatures();
|
|
184
|
-
expect(Object.keys(getFeatures()).length).toBe(0);
|
|
185
|
-
});
|
|
186
|
-
|
|
187
|
-
// ========================================
|
|
188
|
-
// FeatureRegistry export
|
|
189
|
-
// ========================================
|
|
190
|
-
|
|
191
|
-
test('FeatureRegistry has all methods', () => {
|
|
192
|
-
expect(FeatureRegistry.createFeature).toBe(createFeature);
|
|
193
|
-
expect(FeatureRegistry.getFeatures).toBe(getFeatures);
|
|
194
|
-
expect(FeatureRegistry.clearFeatures).toBe(clearFeatures);
|
|
195
|
-
expect(FeatureRegistry.destroyFeature).toBe(destroyFeature);
|
|
196
|
-
expect(FeatureRegistry.getFeatureSummary).toBe(getFeatureSummary);
|
|
197
|
-
});
|
|
198
|
-
});
|
|
@@ -1,167 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview Unit Tests for form-state.js Module
|
|
3
|
-
* Target: 84% → 95% coverage
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { describe, test, expect, beforeAll, beforeEach } from 'bun:test';
|
|
7
|
-
import { parseHTML } from 'linkedom';
|
|
8
|
-
|
|
9
|
-
let createFormState;
|
|
10
|
-
|
|
11
|
-
describe('form-state 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.MutationObserver = class { observe() { } disconnect() { } };
|
|
19
|
-
|
|
20
|
-
const module = await import('../../src/core/form-state.js');
|
|
21
|
-
createFormState = module.createFormState;
|
|
22
|
-
});
|
|
23
|
-
|
|
24
|
-
beforeEach(() => {
|
|
25
|
-
globalThis.document.body.innerHTML = '';
|
|
26
|
-
});
|
|
27
|
-
|
|
28
|
-
// CREATE FORM STATE
|
|
29
|
-
test('createFormState should be a function', () => {
|
|
30
|
-
expect(typeof createFormState).toBe('function');
|
|
31
|
-
});
|
|
32
|
-
|
|
33
|
-
test('createFormState should return object', () => {
|
|
34
|
-
const form = globalThis.document.createElement('form');
|
|
35
|
-
const state = createFormState(form, {});
|
|
36
|
-
expect(typeof state).toBe('object');
|
|
37
|
-
});
|
|
38
|
-
|
|
39
|
-
test('createFormState should have fields property', () => {
|
|
40
|
-
const form = globalThis.document.createElement('form');
|
|
41
|
-
const state = createFormState(form, {});
|
|
42
|
-
expect(state.fields).toBeDefined();
|
|
43
|
-
});
|
|
44
|
-
|
|
45
|
-
test('createFormState should have isValid property', () => {
|
|
46
|
-
const form = globalThis.document.createElement('form');
|
|
47
|
-
const state = createFormState(form, {});
|
|
48
|
-
expect(typeof state.isValid).toBe('boolean');
|
|
49
|
-
});
|
|
50
|
-
|
|
51
|
-
test('createFormState should have isDirty property', () => {
|
|
52
|
-
const form = globalThis.document.createElement('form');
|
|
53
|
-
const state = createFormState(form, {});
|
|
54
|
-
expect(typeof state.isDirty).toBe('boolean');
|
|
55
|
-
});
|
|
56
|
-
|
|
57
|
-
test('createFormState should have isSubmitting property', () => {
|
|
58
|
-
const form = globalThis.document.createElement('form');
|
|
59
|
-
const state = createFormState(form, {});
|
|
60
|
-
expect(typeof state.isSubmitting).toBe('boolean');
|
|
61
|
-
});
|
|
62
|
-
|
|
63
|
-
test('createFormState should have getValues method', () => {
|
|
64
|
-
const form = globalThis.document.createElement('form');
|
|
65
|
-
const state = createFormState(form, {});
|
|
66
|
-
expect(typeof state.getValues).toBe('function');
|
|
67
|
-
});
|
|
68
|
-
|
|
69
|
-
test('createFormState should have getErrors method', () => {
|
|
70
|
-
const form = globalThis.document.createElement('form');
|
|
71
|
-
const state = createFormState(form, {});
|
|
72
|
-
expect(typeof state.getErrors).toBe('function');
|
|
73
|
-
});
|
|
74
|
-
|
|
75
|
-
test('createFormState should have reset method', () => {
|
|
76
|
-
const form = globalThis.document.createElement('form');
|
|
77
|
-
const state = createFormState(form, {});
|
|
78
|
-
expect(typeof state.reset).toBe('function');
|
|
79
|
-
});
|
|
80
|
-
|
|
81
|
-
test('createFormState should have validate method', () => {
|
|
82
|
-
const form = globalThis.document.createElement('form');
|
|
83
|
-
const state = createFormState(form, {});
|
|
84
|
-
expect(typeof state.validate).toBe('function');
|
|
85
|
-
});
|
|
86
|
-
|
|
87
|
-
// WITH SCHEMA
|
|
88
|
-
test('should create field from schema', () => {
|
|
89
|
-
const form = globalThis.document.createElement('form');
|
|
90
|
-
form.innerHTML = '<input name="email" value="">';
|
|
91
|
-
const state = createFormState(form, { email: { required: true } });
|
|
92
|
-
expect(state.fields.email).toBeDefined();
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
test('field should have value property', () => {
|
|
96
|
-
const form = globalThis.document.createElement('form');
|
|
97
|
-
form.innerHTML = '<input name="email" value="test@example.com">';
|
|
98
|
-
const state = createFormState(form, { email: {} });
|
|
99
|
-
expect(state.fields.email.value).toBe('test@example.com');
|
|
100
|
-
});
|
|
101
|
-
|
|
102
|
-
test('field should have error property', () => {
|
|
103
|
-
const form = globalThis.document.createElement('form');
|
|
104
|
-
form.innerHTML = '<input name="email" value="">';
|
|
105
|
-
const state = createFormState(form, { email: { required: true } });
|
|
106
|
-
expect(state.fields.email.error).toBeDefined();
|
|
107
|
-
});
|
|
108
|
-
|
|
109
|
-
test('field should have touched property', () => {
|
|
110
|
-
const form = globalThis.document.createElement('form');
|
|
111
|
-
form.innerHTML = '<input name="email" value="">';
|
|
112
|
-
const state = createFormState(form, { email: {} });
|
|
113
|
-
expect(state.fields.email.touched).toBe(false);
|
|
114
|
-
});
|
|
115
|
-
|
|
116
|
-
test('field should have dirty property', () => {
|
|
117
|
-
const form = globalThis.document.createElement('form');
|
|
118
|
-
form.innerHTML = '<input name="email" value="">';
|
|
119
|
-
const state = createFormState(form, { email: {} });
|
|
120
|
-
expect(state.fields.email.dirty).toBe(false);
|
|
121
|
-
});
|
|
122
|
-
|
|
123
|
-
// VALIDATION
|
|
124
|
-
test('isValid should be true for valid form', () => {
|
|
125
|
-
const form = globalThis.document.createElement('form');
|
|
126
|
-
form.innerHTML = '<input name="email" value="test@example.com">';
|
|
127
|
-
const state = createFormState(form, { email: { type: 'email' } });
|
|
128
|
-
expect(state.isValid).toBe(true);
|
|
129
|
-
});
|
|
130
|
-
|
|
131
|
-
test('isValid should be false for invalid form', () => {
|
|
132
|
-
const form = globalThis.document.createElement('form');
|
|
133
|
-
form.innerHTML = '<input name="email" value="">';
|
|
134
|
-
const state = createFormState(form, { email: { required: true } });
|
|
135
|
-
expect(state.isValid).toBe(false);
|
|
136
|
-
});
|
|
137
|
-
|
|
138
|
-
// METHODS
|
|
139
|
-
test('getValues should return all values', () => {
|
|
140
|
-
const form = globalThis.document.createElement('form');
|
|
141
|
-
form.innerHTML = '<input name="email" value="test@example.com">';
|
|
142
|
-
const state = createFormState(form, { email: {} });
|
|
143
|
-
expect(state.getValues().email).toBe('test@example.com');
|
|
144
|
-
});
|
|
145
|
-
|
|
146
|
-
test('getErrors should return all errors', () => {
|
|
147
|
-
const form = globalThis.document.createElement('form');
|
|
148
|
-
form.innerHTML = '<input name="email" value="">';
|
|
149
|
-
const state = createFormState(form, { email: { required: true } });
|
|
150
|
-
expect(state.getErrors().email).toBeDefined();
|
|
151
|
-
});
|
|
152
|
-
|
|
153
|
-
test('reset should clear values', () => {
|
|
154
|
-
const form = globalThis.document.createElement('form');
|
|
155
|
-
form.innerHTML = '<input name="email" value="test@example.com">';
|
|
156
|
-
const state = createFormState(form, { email: {} });
|
|
157
|
-
state.reset();
|
|
158
|
-
expect(state.isDirty).toBe(false);
|
|
159
|
-
});
|
|
160
|
-
|
|
161
|
-
test('validate should return boolean', () => {
|
|
162
|
-
const form = globalThis.document.createElement('form');
|
|
163
|
-
form.innerHTML = '<input name="email" value="test@example.com">';
|
|
164
|
-
const state = createFormState(form, { email: {} });
|
|
165
|
-
expect(typeof state.validate()).toBe('boolean');
|
|
166
|
-
});
|
|
167
|
-
});
|
package/tests/core/http.test.js
DELETED
|
@@ -1,119 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @fileoverview Unit Tests for http.js Module
|
|
3
|
-
* Target: 21% → 80% coverage
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { describe, test, expect, beforeAll, beforeEach, mock } from 'bun:test';
|
|
7
|
-
import { http, HttpError } from '../../src/core/http.js';
|
|
8
|
-
|
|
9
|
-
describe('http Module Unit Tests', () => {
|
|
10
|
-
|
|
11
|
-
beforeEach(() => {
|
|
12
|
-
// Reset state
|
|
13
|
-
http.baseURL = '';
|
|
14
|
-
http.headers = { 'Content-Type': 'application/json' };
|
|
15
|
-
});
|
|
16
|
-
|
|
17
|
-
// HTTP OBJECT
|
|
18
|
-
test('http should exist', () => {
|
|
19
|
-
expect(http).toBeDefined();
|
|
20
|
-
});
|
|
21
|
-
|
|
22
|
-
test('http should have baseURL property', () => {
|
|
23
|
-
expect(http.baseURL).toBe('');
|
|
24
|
-
});
|
|
25
|
-
|
|
26
|
-
test('http should have headers property', () => {
|
|
27
|
-
expect(http.headers['Content-Type']).toBe('application/json');
|
|
28
|
-
});
|
|
29
|
-
|
|
30
|
-
// SET BASE URL
|
|
31
|
-
test('setBaseURL should set baseURL', () => {
|
|
32
|
-
http.setBaseURL('https://api.example.com');
|
|
33
|
-
expect(http.baseURL).toBe('https://api.example.com');
|
|
34
|
-
});
|
|
35
|
-
|
|
36
|
-
// SET HEADER
|
|
37
|
-
test('setHeader should add header', () => {
|
|
38
|
-
http.setHeader('Authorization', 'Bearer token123');
|
|
39
|
-
expect(http.headers['Authorization']).toBe('Bearer token123');
|
|
40
|
-
});
|
|
41
|
-
|
|
42
|
-
test('setHeader should overwrite existing header', () => {
|
|
43
|
-
http.setHeader('Content-Type', 'text/plain');
|
|
44
|
-
expect(http.headers['Content-Type']).toBe('text/plain');
|
|
45
|
-
});
|
|
46
|
-
|
|
47
|
-
// HTTP METHODS
|
|
48
|
-
test('get should be a function', () => {
|
|
49
|
-
expect(typeof http.get).toBe('function');
|
|
50
|
-
});
|
|
51
|
-
|
|
52
|
-
test('post should be a function', () => {
|
|
53
|
-
expect(typeof http.post).toBe('function');
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
test('put should be a function', () => {
|
|
57
|
-
expect(typeof http.put).toBe('function');
|
|
58
|
-
});
|
|
59
|
-
|
|
60
|
-
test('delete should be a function', () => {
|
|
61
|
-
expect(typeof http.delete).toBe('function');
|
|
62
|
-
});
|
|
63
|
-
|
|
64
|
-
test('request should be a function', () => {
|
|
65
|
-
expect(typeof http.request).toBe('function');
|
|
66
|
-
});
|
|
67
|
-
|
|
68
|
-
// HTTP ERROR CLASS
|
|
69
|
-
test('HttpError should be a class', () => {
|
|
70
|
-
expect(HttpError).toBeDefined();
|
|
71
|
-
});
|
|
72
|
-
|
|
73
|
-
test('HttpError should extend Error', () => {
|
|
74
|
-
const err = new HttpError(404, 'Not Found', 'Resource not found');
|
|
75
|
-
expect(err instanceof Error).toBe(true);
|
|
76
|
-
});
|
|
77
|
-
|
|
78
|
-
test('HttpError should have status property', () => {
|
|
79
|
-
const err = new HttpError(500, 'Internal Server Error', 'body');
|
|
80
|
-
expect(err.status).toBe(500);
|
|
81
|
-
});
|
|
82
|
-
|
|
83
|
-
test('HttpError should have statusText property', () => {
|
|
84
|
-
const err = new HttpError(403, 'Forbidden', 'body');
|
|
85
|
-
expect(err.statusText).toBe('Forbidden');
|
|
86
|
-
});
|
|
87
|
-
|
|
88
|
-
test('HttpError should have body property', () => {
|
|
89
|
-
const err = new HttpError(400, 'Bad Request', 'Invalid data');
|
|
90
|
-
expect(err.body).toBe('Invalid data');
|
|
91
|
-
});
|
|
92
|
-
|
|
93
|
-
test('HttpError message should be formatted', () => {
|
|
94
|
-
const err = new HttpError(401, 'Unauthorized', 'body');
|
|
95
|
-
expect(err.message).toBe('HTTP 401: Unauthorized');
|
|
96
|
-
});
|
|
97
|
-
|
|
98
|
-
test('HttpError should be throwable', () => {
|
|
99
|
-
expect(() => { throw new HttpError(500, 'Error', 'body'); }).toThrow();
|
|
100
|
-
});
|
|
101
|
-
|
|
102
|
-
// REQUEST METHOD SIGNATURES
|
|
103
|
-
test('get should accept url and options', () => {
|
|
104
|
-
// Just verify it doesn't throw on call signature
|
|
105
|
-
expect(() => http.get).not.toThrow();
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
test('post should accept url, body and options', () => {
|
|
109
|
-
expect(() => http.post).not.toThrow();
|
|
110
|
-
});
|
|
111
|
-
|
|
112
|
-
test('put should accept url, body and options', () => {
|
|
113
|
-
expect(() => http.put).not.toThrow();
|
|
114
|
-
});
|
|
115
|
-
|
|
116
|
-
test('delete should accept url and options', () => {
|
|
117
|
-
expect(() => http.delete).not.toThrow();
|
|
118
|
-
});
|
|
119
|
-
});
|