@akashjs/runtime 0.2.0 → 0.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (84) hide show
  1. package/dist/animate.d.cts +145 -0
  2. package/dist/animate.d.ts +145 -0
  3. package/dist/{chunk-V7VG23IO.cjs → chunk-3AL2DVPZ.cjs} +3 -3
  4. package/dist/{chunk-V7VG23IO.cjs.map → chunk-3AL2DVPZ.cjs.map} +1 -1
  5. package/dist/{chunk-CHF5LH56.cjs → chunk-6NX6JRSV.cjs} +3 -3
  6. package/dist/{chunk-CHF5LH56.cjs.map → chunk-6NX6JRSV.cjs.map} +1 -1
  7. package/dist/{chunk-DYJUCZXA.js → chunk-BT6HNBE7.js} +4 -4
  8. package/dist/chunk-BT6HNBE7.js.map +1 -0
  9. package/dist/{chunk-CGKMCVEZ.js → chunk-D6QQYZIC.js} +2 -2
  10. package/dist/{chunk-CGKMCVEZ.js.map → chunk-D6QQYZIC.js.map} +1 -1
  11. package/dist/{chunk-R3W5W647.cjs → chunk-H4SAK7A5.cjs} +2 -2
  12. package/dist/{chunk-R3W5W647.cjs.map → chunk-H4SAK7A5.cjs.map} +1 -1
  13. package/dist/chunk-IKZMC3T6.js +2 -0
  14. package/dist/chunk-IKZMC3T6.js.map +1 -0
  15. package/dist/{chunk-VFOAULHK.js → chunk-IM2VW4TK.js} +3 -3
  16. package/dist/{chunk-VFOAULHK.js.map → chunk-IM2VW4TK.js.map} +1 -1
  17. package/dist/chunk-NQVWTQ2I.cjs +5 -0
  18. package/dist/chunk-NQVWTQ2I.cjs.map +1 -0
  19. package/dist/{chunk-5EFX654I.cjs → chunk-NVZLEJXB.cjs} +2 -2
  20. package/dist/{chunk-5EFX654I.cjs.map → chunk-NVZLEJXB.cjs.map} +1 -1
  21. package/dist/{chunk-DPJ6RJ7A.js → chunk-ODDXU5DO.js} +2 -2
  22. package/dist/chunk-ODDXU5DO.js.map +1 -0
  23. package/dist/{chunk-EUKRTV4W.js → chunk-POLTPHUA.js} +3 -3
  24. package/dist/{chunk-EUKRTV4W.js.map → chunk-POLTPHUA.js.map} +1 -1
  25. package/dist/{chunk-4HAE7H7W.cjs → chunk-YIB4EKVI.cjs} +2 -2
  26. package/dist/chunk-YIB4EKVI.cjs.map +1 -0
  27. package/dist/chunk-YJQLXUTN.cjs +2 -0
  28. package/dist/chunk-YJQLXUTN.cjs.map +1 -0
  29. package/dist/{chunk-SRPWGLOQ.js → chunk-Z5LQV5ND.js} +2 -2
  30. package/dist/{chunk-SRPWGLOQ.js.map → chunk-Z5LQV5ND.js.map} +1 -1
  31. package/dist/component-C1WnFcRp.d.cts +59 -0
  32. package/dist/component-C1WnFcRp.d.ts +59 -0
  33. package/dist/context-2uQ6fuxu.d.cts +57 -0
  34. package/dist/context-2uQ6fuxu.d.ts +57 -0
  35. package/dist/core.cjs +1 -1
  36. package/dist/core.d.cts +65 -0
  37. package/dist/core.d.ts +65 -0
  38. package/dist/core.js +1 -1
  39. package/dist/index.cjs +8 -8
  40. package/dist/index.cjs.map +1 -1
  41. package/dist/index.d.cts +3218 -0
  42. package/dist/index.d.ts +3218 -0
  43. package/dist/index.js +8 -8
  44. package/dist/index.js.map +1 -1
  45. package/dist/machine.cjs +1 -1
  46. package/dist/machine.d.cts +87 -0
  47. package/dist/machine.d.ts +87 -0
  48. package/dist/machine.js +1 -1
  49. package/dist/offline.cjs +1 -1
  50. package/dist/offline.d.cts +73 -0
  51. package/dist/offline.d.ts +73 -0
  52. package/dist/offline.js +1 -1
  53. package/dist/pwa.cjs +1 -1
  54. package/dist/pwa.d.cts +80 -0
  55. package/dist/pwa.d.ts +80 -0
  56. package/dist/pwa.js +1 -1
  57. package/dist/signals-C7XfOHHR.d.cts +55 -0
  58. package/dist/signals-C7XfOHHR.d.ts +55 -0
  59. package/dist/ssr.cjs +1 -1
  60. package/dist/ssr.d.cts +78 -0
  61. package/dist/ssr.d.ts +78 -0
  62. package/dist/ssr.js +1 -1
  63. package/dist/store.cjs +1 -1
  64. package/dist/store.d.cts +84 -0
  65. package/dist/store.d.ts +84 -0
  66. package/dist/store.js +1 -1
  67. package/dist/sync.cjs +1 -1
  68. package/dist/sync.d.cts +128 -0
  69. package/dist/sync.d.ts +128 -0
  70. package/dist/sync.js +1 -1
  71. package/dist/test.cjs +2 -2
  72. package/dist/test.d.cts +109 -0
  73. package/dist/test.d.ts +109 -0
  74. package/dist/test.js +1 -1
  75. package/package.json +1 -1
  76. package/dist/chunk-4HAE7H7W.cjs.map +0 -1
  77. package/dist/chunk-DPJ6RJ7A.js.map +0 -1
  78. package/dist/chunk-DYJUCZXA.js.map +0 -1
  79. package/dist/chunk-P5GADKQS.cjs +0 -2
  80. package/dist/chunk-P5GADKQS.cjs.map +0 -1
  81. package/dist/chunk-Q5BER4ZB.js +0 -2
  82. package/dist/chunk-Q5BER4ZB.js.map +0 -1
  83. package/dist/chunk-QDCIW4YE.cjs +0 -5
  84. package/dist/chunk-QDCIW4YE.cjs.map +0 -1
@@ -0,0 +1,109 @@
1
+ import { I as InjectionKey } from './context-2uQ6fuxu.cjs';
2
+ export { f as flush } from './context-2uQ6fuxu.cjs';
3
+ import { S as Signal } from './signals-C7XfOHHR.cjs';
4
+ import { C as Component } from './component-C1WnFcRp.cjs';
5
+
6
+ /**
7
+ * @akashjs/runtime/test — Test utilities.
8
+ *
9
+ * Provides mount(), fireEvent, and query helpers for testing
10
+ * AkashJS components with Vitest (or any test runner).
11
+ *
12
+ * No TestBed, no module configuration, no compileComponents().
13
+ * Just mount a component and query the resulting DOM.
14
+ */
15
+
16
+ interface MountResult {
17
+ /** The root container element */
18
+ container: HTMLElement;
19
+ /** Unmount the component and clean up */
20
+ unmount(): void;
21
+ /** Find the first element whose text content contains the given string */
22
+ getByText(text: string): HTMLElement;
23
+ /** Find the first element with the given ARIA role */
24
+ getByRole(role: string): HTMLElement;
25
+ /** Find the first element with the given data-testid */
26
+ getByTestId(id: string): HTMLElement;
27
+ /** Query all elements matching a CSS selector */
28
+ queryAll(selector: string): HTMLElement[];
29
+ /** Query the first element matching a CSS selector, or null */
30
+ query(selector: string): HTMLElement | null;
31
+ }
32
+ interface MountOptions<P extends Record<string, unknown>> {
33
+ /** Props to pass to the component */
34
+ props?: P;
35
+ /** Context values to provide (Map of InjectionKey -> value) */
36
+ provide?: Map<InjectionKey<any>, any>;
37
+ }
38
+ /**
39
+ * Mount a component into a detached DOM element for testing.
40
+ *
41
+ * ```ts
42
+ * const { getByText, getByRole } = mount(Counter, { props: { initial: 5 } });
43
+ * expect(getByText('Count: 5')).toBeTruthy();
44
+ * ```
45
+ */
46
+ declare function mount<P extends Record<string, unknown> = Record<string, unknown>>(component: Component<P>, options?: MountOptions<P>): MountResult;
47
+ /**
48
+ * Fire DOM events on elements. Returns a promise that resolves
49
+ * after a microtask, giving effects time to flush.
50
+ *
51
+ * ```ts
52
+ * await fireEvent.click(button);
53
+ * await fireEvent.input(input, 'hello');
54
+ * ```
55
+ */
56
+ declare const fireEvent: {
57
+ click(el: HTMLElement): Promise<void>;
58
+ input(el: HTMLInputElement | HTMLTextAreaElement, value: string): Promise<void>;
59
+ submit(el: HTMLFormElement): Promise<void>;
60
+ focus(el: HTMLElement): Promise<void>;
61
+ blur(el: HTMLElement): Promise<void>;
62
+ keyDown(el: HTMLElement, key: string, options?: KeyboardEventInit): Promise<void>;
63
+ keyUp(el: HTMLElement, key: string, options?: KeyboardEventInit): Promise<void>;
64
+ };
65
+ interface WaitForOptions {
66
+ /** Timeout in ms (default: 1000) */
67
+ timeout?: number;
68
+ /** Poll interval in ms (default: 50) */
69
+ interval?: number;
70
+ }
71
+ /**
72
+ * Wait for an assertion to pass. Retries until it doesn't throw or times out.
73
+ *
74
+ * ```ts
75
+ * await waitFor(() => expect(getByText('loaded')).toBeTruthy());
76
+ * ```
77
+ */
78
+ declare function waitFor(assertion: () => void | Promise<void>, options?: WaitForOptions): Promise<void>;
79
+ /**
80
+ * Wait for an element matching a selector to appear in the container.
81
+ *
82
+ * ```ts
83
+ * const el = await waitForElement(container, '.loaded');
84
+ * ```
85
+ */
86
+ declare function waitForElement(container: HTMLElement, selector: string, options?: WaitForOptions): Promise<HTMLElement>;
87
+
88
+ interface TestSignal<T> extends Signal<T> {
89
+ /** History of all values set on this signal (including initial) */
90
+ history: T[];
91
+ /** Number of times set() or update() was called */
92
+ setCount: number;
93
+ /** Reset history and set count */
94
+ resetHistory(): void;
95
+ }
96
+ /**
97
+ * Create a signal with test inspection capabilities.
98
+ *
99
+ * ```ts
100
+ * const count = createTestSignal(0);
101
+ * count.set(1);
102
+ * count.set(2);
103
+ * expect(count.history).toEqual([0, 1, 2]);
104
+ * expect(count.setCount).toBe(2);
105
+ * ```
106
+ */
107
+ declare function createTestSignal<T>(initialValue: T): TestSignal<T>;
108
+
109
+ export { type MountOptions, type MountResult, type TestSignal, type WaitForOptions, createTestSignal, fireEvent, mount, waitFor, waitForElement };
package/dist/test.d.ts ADDED
@@ -0,0 +1,109 @@
1
+ import { I as InjectionKey } from './context-2uQ6fuxu.js';
2
+ export { f as flush } from './context-2uQ6fuxu.js';
3
+ import { S as Signal } from './signals-C7XfOHHR.js';
4
+ import { C as Component } from './component-C1WnFcRp.js';
5
+
6
+ /**
7
+ * @akashjs/runtime/test — Test utilities.
8
+ *
9
+ * Provides mount(), fireEvent, and query helpers for testing
10
+ * AkashJS components with Vitest (or any test runner).
11
+ *
12
+ * No TestBed, no module configuration, no compileComponents().
13
+ * Just mount a component and query the resulting DOM.
14
+ */
15
+
16
+ interface MountResult {
17
+ /** The root container element */
18
+ container: HTMLElement;
19
+ /** Unmount the component and clean up */
20
+ unmount(): void;
21
+ /** Find the first element whose text content contains the given string */
22
+ getByText(text: string): HTMLElement;
23
+ /** Find the first element with the given ARIA role */
24
+ getByRole(role: string): HTMLElement;
25
+ /** Find the first element with the given data-testid */
26
+ getByTestId(id: string): HTMLElement;
27
+ /** Query all elements matching a CSS selector */
28
+ queryAll(selector: string): HTMLElement[];
29
+ /** Query the first element matching a CSS selector, or null */
30
+ query(selector: string): HTMLElement | null;
31
+ }
32
+ interface MountOptions<P extends Record<string, unknown>> {
33
+ /** Props to pass to the component */
34
+ props?: P;
35
+ /** Context values to provide (Map of InjectionKey -> value) */
36
+ provide?: Map<InjectionKey<any>, any>;
37
+ }
38
+ /**
39
+ * Mount a component into a detached DOM element for testing.
40
+ *
41
+ * ```ts
42
+ * const { getByText, getByRole } = mount(Counter, { props: { initial: 5 } });
43
+ * expect(getByText('Count: 5')).toBeTruthy();
44
+ * ```
45
+ */
46
+ declare function mount<P extends Record<string, unknown> = Record<string, unknown>>(component: Component<P>, options?: MountOptions<P>): MountResult;
47
+ /**
48
+ * Fire DOM events on elements. Returns a promise that resolves
49
+ * after a microtask, giving effects time to flush.
50
+ *
51
+ * ```ts
52
+ * await fireEvent.click(button);
53
+ * await fireEvent.input(input, 'hello');
54
+ * ```
55
+ */
56
+ declare const fireEvent: {
57
+ click(el: HTMLElement): Promise<void>;
58
+ input(el: HTMLInputElement | HTMLTextAreaElement, value: string): Promise<void>;
59
+ submit(el: HTMLFormElement): Promise<void>;
60
+ focus(el: HTMLElement): Promise<void>;
61
+ blur(el: HTMLElement): Promise<void>;
62
+ keyDown(el: HTMLElement, key: string, options?: KeyboardEventInit): Promise<void>;
63
+ keyUp(el: HTMLElement, key: string, options?: KeyboardEventInit): Promise<void>;
64
+ };
65
+ interface WaitForOptions {
66
+ /** Timeout in ms (default: 1000) */
67
+ timeout?: number;
68
+ /** Poll interval in ms (default: 50) */
69
+ interval?: number;
70
+ }
71
+ /**
72
+ * Wait for an assertion to pass. Retries until it doesn't throw or times out.
73
+ *
74
+ * ```ts
75
+ * await waitFor(() => expect(getByText('loaded')).toBeTruthy());
76
+ * ```
77
+ */
78
+ declare function waitFor(assertion: () => void | Promise<void>, options?: WaitForOptions): Promise<void>;
79
+ /**
80
+ * Wait for an element matching a selector to appear in the container.
81
+ *
82
+ * ```ts
83
+ * const el = await waitForElement(container, '.loaded');
84
+ * ```
85
+ */
86
+ declare function waitForElement(container: HTMLElement, selector: string, options?: WaitForOptions): Promise<HTMLElement>;
87
+
88
+ interface TestSignal<T> extends Signal<T> {
89
+ /** History of all values set on this signal (including initial) */
90
+ history: T[];
91
+ /** Number of times set() or update() was called */
92
+ setCount: number;
93
+ /** Reset history and set count */
94
+ resetHistory(): void;
95
+ }
96
+ /**
97
+ * Create a signal with test inspection capabilities.
98
+ *
99
+ * ```ts
100
+ * const count = createTestSignal(0);
101
+ * count.set(1);
102
+ * count.set(2);
103
+ * expect(count.history).toEqual([0, 1, 2]);
104
+ * expect(count.setCount).toBe(2);
105
+ * ```
106
+ */
107
+ declare function createTestSignal<T>(initialValue: T): TestSignal<T>;
108
+
109
+ export { type MountOptions, type MountResult, type TestSignal, type WaitForOptions, createTestSignal, fireEvent, mount, waitFor, waitForElement };
package/dist/test.js CHANGED
@@ -1,4 +1,4 @@
1
- import {B,h}from'./chunk-DYJUCZXA.js';import {l}from'./chunk-EUKRTV4W.js';export{b as flush}from'./chunk-EUKRTV4W.js';function H(t,n){let e=document.createElement("div");e.setAttribute("data-akash-test-root","");let i=n?.props??{},r=n?.provide,s;return r&&r.size>0?s=B(()=>{for(let[l,d]of r)h(l,d);return ()=>t(i)})({}):s=t(i),e.appendChild(s),document.body.appendChild(e),{container:e,unmount(){e.remove();},getByText(o){let l=T(e,o);if(!l)throw new Error(`[AkashJS Test] Could not find element with text: "${o}"
1
+ import {B,h}from'./chunk-BT6HNBE7.js';import {l}from'./chunk-POLTPHUA.js';export{b as flush}from'./chunk-POLTPHUA.js';function H(t,n){let e=document.createElement("div");e.setAttribute("data-akash-test-root","");let i=n?.props??{},r=n?.provide,s;return r&&r.size>0?s=B(()=>{for(let[l,d]of r)h(l,d);return ()=>t(i)})({}):s=t(i),e.appendChild(s),document.body.appendChild(e),{container:e,unmount(){e.remove();},getByText(o){let l=T(e,o);if(!l)throw new Error(`[AkashJS Test] Could not find element with text: "${o}"
2
2
  Container HTML: ${e.innerHTML.slice(0,200)}`);return l},getByRole(o){let l=e.querySelector(`[role="${o}"]`)??f(e,o);if(!l)throw new Error(`[AkashJS Test] Could not find element with role: "${o}"
3
3
  Container HTML: ${e.innerHTML.slice(0,200)}`);return l},getByTestId(o){let l=e.querySelector(`[data-testid="${o}"]`);if(!l)throw new Error(`[AkashJS Test] Could not find element with data-testid: "${o}"
4
4
  Container HTML: ${e.innerHTML.slice(0,200)}`);return l},queryAll(o){return Array.from(e.querySelectorAll(o))},query(o){return e.querySelector(o)}}}var L={async click(t){t.dispatchEvent(new MouseEvent("click",{bubbles:true,cancelable:true})),await a();},async input(t,n){let e=Object.getOwnPropertyDescriptor(t instanceof HTMLTextAreaElement?HTMLTextAreaElement.prototype:HTMLInputElement.prototype,"value")?.set;e?e.call(t,n):t.value=n,t.dispatchEvent(new Event("input",{bubbles:true})),t.dispatchEvent(new Event("change",{bubbles:true})),await a();},async submit(t){t.dispatchEvent(new Event("submit",{bubbles:true,cancelable:true})),await a();},async focus(t){t.focus(),t.dispatchEvent(new FocusEvent("focus",{bubbles:true})),await a();},async blur(t){t.blur(),t.dispatchEvent(new FocusEvent("blur",{bubbles:true})),await a();},async keyDown(t,n,e){t.dispatchEvent(new KeyboardEvent("keydown",{key:n,bubbles:true,...e})),await a();},async keyUp(t,n,e){t.dispatchEvent(new KeyboardEvent("keyup",{key:n,bubbles:true,...e})),await a();}};function a(){return new Promise(t=>queueMicrotask(t))}function T(t,n){let e=null,i=document.createTreeWalker(t,NodeFilter.SHOW_ELEMENT),r=i.nextNode();for(;r;)r instanceof HTMLElement&&r.textContent?.includes(n)&&(e=r),r=i.nextNode();return !e&&t.textContent?.includes(n)&&(e=t),e}var E={button:["button"],a:["link"],input:["textbox","checkbox","radio","spinbutton","slider"],select:["combobox","listbox"],textarea:["textbox"],img:["img"],form:["form"],nav:["navigation"],main:["main"],header:["banner"],footer:["contentinfo"],aside:["complementary"],section:["region"],article:["article"],ul:["list"],ol:["list"],li:["listitem"],table:["table"],th:["columnheader"],td:["cell"],h1:["heading"],h2:["heading"],h3:["heading"],h4:["heading"],h5:["heading"],h6:["heading"]};function f(t,n){let e=[];for(let[r,s]of Object.entries(E))s.includes(n)&&e.push(r);if(e.length===0)return null;let i=e.join(", ");return t.querySelector(i)}async function y(t,n){let{timeout:e=1e3,interval:i=50}=n??{},r=Date.now();for(;;)try{await t();return}catch(s){if(Date.now()-r>=e)throw s;await new Promise(o=>setTimeout(o,i));}}async function w(t,n,e){let i=null;return await y(()=>{if(i=t.querySelector(n),!i)throw new Error(`Element "${n}" not found`)},e),i}function v(t){let n=l(t),e=[t],i=0,r=(()=>n());return r.set=s=>{n.set(s),e.push(s),i++;},r.update=s=>{let o=s(n.peek());r.set(o);},r.peek=()=>n.peek(),Object.defineProperty(r,"history",{get:()=>[...e]}),Object.defineProperty(r,"setCount",{get:()=>i}),r.resetHistory=()=>{e.length=0,e.push(n.peek()),i=0;},r}export{v as createTestSignal,L as fireEvent,H as mount,y as waitFor,w as waitForElement};//# sourceMappingURL=test.js.map
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@akashjs/runtime",
3
- "version": "0.2.0",
3
+ "version": "0.2.2",
4
4
  "description": "AkashJS core runtime — signals, components, DOM rendering",
5
5
  "type": "module",
6
6
  "main": "dist/index.cjs",
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/sync.ts"],"names":["LWWRegister","initialValue","peerId","value","ts","remote","createWebSocketTransport","options","ws","opHandlers","presenceHandlers","op","handler","removed","i","data","e","msg","h","createLocalTransport","peerIdCounter","createSync","roomId","initialState","transport","registers","stateSignals","key","signal","state","original","register","proxy","fn","peers","connected","presence","peerPresenceMap","unsubOps","unsubPresence","remotePeerId","map","next"],"mappings":"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","file":"chunk-4HAE7H7W.cjs","sourcesContent":["/**\n * Collaborative signals with CRDT.\n *\n * Make any signal multiplayer with one line. Multiple users can\n * edit the same state simultaneously with automatic conflict\n * resolution via Last-Writer-Wins Register and Operation-based CRDTs.\n *\n * ```ts\n * const doc = createSync('doc-123', {\n * title: '',\n * blocks: [],\n * cursor: { x: 0, y: 0 },\n * });\n *\n * doc.state.title.set('Hello'); // syncs to all peers\n * doc.peers(); // list of connected users\n * doc.presence.set({ cursor: { x: 10, y: 20 } });\n * ```\n */\n\nimport { signal, computed } from './signals.js';\nimport type { Signal, ReadonlySignal } from './signals.js';\n\n// =========================================================================\n// CRDT — Last-Writer-Wins Register\n// =========================================================================\n\nexport interface LWWEntry<T> {\n value: T;\n timestamp: number;\n peerId: string;\n}\n\n/**\n * Last-Writer-Wins Register — simplest CRDT for single values.\n * The write with the highest timestamp wins on conflict.\n */\nexport class LWWRegister<T> {\n private entry: LWWEntry<T>;\n\n constructor(initialValue: T, peerId: string) {\n this.entry = { value: initialValue, timestamp: Date.now(), peerId };\n }\n\n get value(): T {\n return this.entry.value;\n }\n\n get timestamp(): number {\n return this.entry.timestamp;\n }\n\n set(value: T, peerId: string): boolean {\n const ts = Date.now();\n // Local writes always succeed (same peer always advances its own state).\n // Cross-peer conflicts resolve by highest timestamp, then peerId tiebreak.\n if (\n peerId === this.entry.peerId ||\n ts > this.entry.timestamp ||\n (ts === this.entry.timestamp && peerId > this.entry.peerId)\n ) {\n this.entry = { value, timestamp: Math.max(ts, this.entry.timestamp + 1), peerId };\n return true;\n }\n return false;\n }\n\n merge(remote: LWWEntry<T>): boolean {\n if (\n remote.timestamp > this.entry.timestamp ||\n (remote.timestamp === this.entry.timestamp && remote.peerId > this.entry.peerId)\n ) {\n this.entry = remote;\n return true;\n }\n return false;\n }\n\n toEntry(): LWWEntry<T> {\n return { ...this.entry };\n }\n}\n\n// =========================================================================\n// Operation log for list CRDTs\n// =========================================================================\n\nexport type SyncOp =\n | { type: 'set'; key: string; value: unknown; timestamp: number; peerId: string }\n | { type: 'insert'; key: string; index: number; value: unknown; timestamp: number; peerId: string }\n | { type: 'delete'; key: string; index: number; timestamp: number; peerId: string };\n\n// =========================================================================\n// Sync transport interface\n// =========================================================================\n\nexport interface SyncTransport {\n /** Send an operation to peers */\n send(op: SyncOp): void;\n /** Listen for operations from peers */\n onReceive(handler: (op: SyncOp) => void): () => void;\n /** Listen for peer presence updates */\n onPresence?(handler: (peerId: string, data: unknown) => void): () => void;\n /** Send presence data */\n sendPresence?(data: unknown): void;\n /** Connect to the sync channel */\n connect(): void;\n /** Disconnect */\n disconnect(): void;\n}\n\n// =========================================================================\n// WebSocket transport\n// =========================================================================\n\nexport interface WebSocketTransportOptions {\n url: string;\n room: string;\n protocols?: string | string[];\n}\n\nexport function createWebSocketTransport(options: WebSocketTransportOptions): SyncTransport {\n let ws: WebSocket | null = null;\n const opHandlers: Array<(op: SyncOp) => void> = [];\n const presenceHandlers: Array<(peerId: string, data: unknown) => void> = [];\n\n return {\n send(op: SyncOp) {\n ws?.send(JSON.stringify({ type: 'op', room: options.room, ...op }));\n },\n onReceive(handler) {\n opHandlers.push(handler);\n let removed = false;\n return () => {\n if (removed) return;\n removed = true;\n const i = opHandlers.indexOf(handler);\n if (i !== -1) opHandlers.splice(i, 1);\n };\n },\n onPresence(handler) {\n presenceHandlers.push(handler);\n return () => {\n const i = presenceHandlers.indexOf(handler);\n if (i !== -1) presenceHandlers.splice(i, 1);\n };\n },\n sendPresence(data) {\n ws?.send(JSON.stringify({ type: 'presence', room: options.room, data }));\n },\n connect() {\n ws = new WebSocket(options.url, options.protocols);\n ws.onmessage = (e) => {\n try {\n const msg = JSON.parse(e.data);\n if (msg.type === 'op') {\n for (const h of opHandlers) h(msg);\n } else if (msg.type === 'presence') {\n for (const h of presenceHandlers) h(msg.peerId, msg.data);\n }\n } catch { /* ignore parse errors */ }\n };\n ws.onopen = () => {\n ws?.send(JSON.stringify({ type: 'join', room: options.room }));\n };\n },\n disconnect() {\n ws?.close();\n ws = null;\n },\n };\n}\n\n// =========================================================================\n// In-memory transport (for testing / single-tab)\n// =========================================================================\n\nexport function createLocalTransport(): SyncTransport {\n const opHandlers: Array<(op: SyncOp) => void> = [];\n return {\n send(op) {\n for (const h of opHandlers) h(op);\n },\n onReceive(handler) {\n opHandlers.push(handler);\n let removed = false;\n return () => {\n if (removed) return;\n removed = true;\n const i = opHandlers.indexOf(handler);\n if (i !== -1) opHandlers.splice(i, 1);\n };\n },\n connect() {},\n disconnect() {},\n };\n}\n\n// =========================================================================\n// createSync — the main API\n// =========================================================================\n\nexport interface SyncOptions {\n /** Transport for sending/receiving operations */\n transport?: SyncTransport;\n /** Unique peer ID (default: random) */\n peerId?: string;\n}\n\nexport interface SyncDoc<T extends Record<string, unknown>> {\n /** Reactive synced state — each key is a Signal */\n state: { [K in keyof T]: Signal<T[K]> };\n /** Connected peers (reactive) */\n peers: ReadonlySignal<PeerInfo[]>;\n /** Local presence data */\n presence: Signal<Record<string, unknown>>;\n /** Peer presence map (reactive) */\n peerPresence: ReadonlySignal<Map<string, unknown>>;\n /** This peer's ID */\n peerId: string;\n /** Whether connected */\n connected: ReadonlySignal<boolean>;\n /** Connect to the sync channel */\n connect(): void;\n /** Disconnect */\n disconnect(): void;\n /** Dispose the sync doc */\n dispose(): void;\n}\n\nexport interface PeerInfo {\n id: string;\n joinedAt: number;\n}\n\nlet peerIdCounter = 0;\n\n/**\n * Create a collaborative synced document.\n *\n * ```ts\n * const doc = createSync('room-1', { title: '', count: 0 }, {\n * transport: createWebSocketTransport({ url: 'wss://sync.example.com', room: 'room-1' }),\n * });\n *\n * doc.state.title.set('Hello'); // auto-syncs to all peers\n * doc.peers(); // connected users\n * ```\n */\nexport function createSync<T extends Record<string, unknown>>(\n roomId: string,\n initialState: T,\n options: SyncOptions = {},\n): SyncDoc<T> {\n const peerId = options.peerId ?? `peer-${++peerIdCounter}-${Date.now()}`;\n const transport = options.transport ?? createLocalTransport();\n\n // Create CRDT registers and signals for each state key\n const registers = new Map<string, LWWRegister<unknown>>();\n const stateSignals: Record<string, Signal<unknown>> = {};\n\n for (const [key, value] of Object.entries(initialState)) {\n registers.set(key, new LWWRegister(value, peerId));\n stateSignals[key] = signal(value);\n }\n\n // Intercept signal.set to broadcast operations\n const state = {} as { [K in keyof T]: Signal<T[K]> };\n for (const key of Object.keys(initialState)) {\n const original = stateSignals[key];\n const register = registers.get(key)!;\n\n const proxy: Signal<any> = (() => original()) as any;\n proxy.set = (value: any) => {\n register.set(value, peerId);\n original.set(value);\n transport.send({\n type: 'set',\n key,\n value,\n timestamp: register.timestamp,\n peerId,\n });\n };\n proxy.update = (fn: (prev: any) => any) => {\n proxy.set(fn(original()));\n };\n proxy.peek = () => original.peek();\n\n (state as any)[key] = proxy;\n }\n\n // Peers\n const peers = signal<PeerInfo[]>([]);\n const connected = signal(false);\n const presence = signal<Record<string, unknown>>({});\n const peerPresenceMap = signal(new Map<string, unknown>());\n\n // Listen for remote operations\n const unsubOps = transport.onReceive((op) => {\n if (op.type === 'set' && registers.has(op.key)) {\n const register = registers.get(op.key)!;\n const merged = register.merge({\n value: op.value,\n timestamp: op.timestamp,\n peerId: op.peerId,\n });\n if (merged) {\n stateSignals[op.key].set(op.value);\n }\n }\n });\n\n // Listen for presence\n const unsubPresence = transport.onPresence?.((remotePeerId, data) => {\n peerPresenceMap.update((map) => {\n const next = new Map(map);\n next.set(remotePeerId, data);\n return next;\n });\n });\n\n return {\n state,\n peers: () => peers(),\n presence,\n peerPresence: () => peerPresenceMap(),\n peerId,\n connected: () => connected(),\n connect() {\n transport.connect();\n connected.set(true);\n },\n disconnect() {\n transport.disconnect();\n connected.set(false);\n },\n dispose() {\n unsubOps();\n unsubPresence?.();\n transport.disconnect();\n },\n };\n}\n"]}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/sync.ts"],"names":["LWWRegister","initialValue","peerId","value","ts","remote","createWebSocketTransport","options","ws","opHandlers","presenceHandlers","op","handler","removed","i","data","e","msg","h","createLocalTransport","peerIdCounter","createSync","roomId","initialState","transport","registers","stateSignals","key","signal","state","original","register","proxy","fn","peers","connected","presence","peerPresenceMap","unsubOps","unsubPresence","remotePeerId","map","next"],"mappings":"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","file":"chunk-DPJ6RJ7A.js","sourcesContent":["/**\n * Collaborative signals with CRDT.\n *\n * Make any signal multiplayer with one line. Multiple users can\n * edit the same state simultaneously with automatic conflict\n * resolution via Last-Writer-Wins Register and Operation-based CRDTs.\n *\n * ```ts\n * const doc = createSync('doc-123', {\n * title: '',\n * blocks: [],\n * cursor: { x: 0, y: 0 },\n * });\n *\n * doc.state.title.set('Hello'); // syncs to all peers\n * doc.peers(); // list of connected users\n * doc.presence.set({ cursor: { x: 10, y: 20 } });\n * ```\n */\n\nimport { signal, computed } from './signals.js';\nimport type { Signal, ReadonlySignal } from './signals.js';\n\n// =========================================================================\n// CRDT — Last-Writer-Wins Register\n// =========================================================================\n\nexport interface LWWEntry<T> {\n value: T;\n timestamp: number;\n peerId: string;\n}\n\n/**\n * Last-Writer-Wins Register — simplest CRDT for single values.\n * The write with the highest timestamp wins on conflict.\n */\nexport class LWWRegister<T> {\n private entry: LWWEntry<T>;\n\n constructor(initialValue: T, peerId: string) {\n this.entry = { value: initialValue, timestamp: Date.now(), peerId };\n }\n\n get value(): T {\n return this.entry.value;\n }\n\n get timestamp(): number {\n return this.entry.timestamp;\n }\n\n set(value: T, peerId: string): boolean {\n const ts = Date.now();\n // Local writes always succeed (same peer always advances its own state).\n // Cross-peer conflicts resolve by highest timestamp, then peerId tiebreak.\n if (\n peerId === this.entry.peerId ||\n ts > this.entry.timestamp ||\n (ts === this.entry.timestamp && peerId > this.entry.peerId)\n ) {\n this.entry = { value, timestamp: Math.max(ts, this.entry.timestamp + 1), peerId };\n return true;\n }\n return false;\n }\n\n merge(remote: LWWEntry<T>): boolean {\n if (\n remote.timestamp > this.entry.timestamp ||\n (remote.timestamp === this.entry.timestamp && remote.peerId > this.entry.peerId)\n ) {\n this.entry = remote;\n return true;\n }\n return false;\n }\n\n toEntry(): LWWEntry<T> {\n return { ...this.entry };\n }\n}\n\n// =========================================================================\n// Operation log for list CRDTs\n// =========================================================================\n\nexport type SyncOp =\n | { type: 'set'; key: string; value: unknown; timestamp: number; peerId: string }\n | { type: 'insert'; key: string; index: number; value: unknown; timestamp: number; peerId: string }\n | { type: 'delete'; key: string; index: number; timestamp: number; peerId: string };\n\n// =========================================================================\n// Sync transport interface\n// =========================================================================\n\nexport interface SyncTransport {\n /** Send an operation to peers */\n send(op: SyncOp): void;\n /** Listen for operations from peers */\n onReceive(handler: (op: SyncOp) => void): () => void;\n /** Listen for peer presence updates */\n onPresence?(handler: (peerId: string, data: unknown) => void): () => void;\n /** Send presence data */\n sendPresence?(data: unknown): void;\n /** Connect to the sync channel */\n connect(): void;\n /** Disconnect */\n disconnect(): void;\n}\n\n// =========================================================================\n// WebSocket transport\n// =========================================================================\n\nexport interface WebSocketTransportOptions {\n url: string;\n room: string;\n protocols?: string | string[];\n}\n\nexport function createWebSocketTransport(options: WebSocketTransportOptions): SyncTransport {\n let ws: WebSocket | null = null;\n const opHandlers: Array<(op: SyncOp) => void> = [];\n const presenceHandlers: Array<(peerId: string, data: unknown) => void> = [];\n\n return {\n send(op: SyncOp) {\n ws?.send(JSON.stringify({ type: 'op', room: options.room, ...op }));\n },\n onReceive(handler) {\n opHandlers.push(handler);\n let removed = false;\n return () => {\n if (removed) return;\n removed = true;\n const i = opHandlers.indexOf(handler);\n if (i !== -1) opHandlers.splice(i, 1);\n };\n },\n onPresence(handler) {\n presenceHandlers.push(handler);\n return () => {\n const i = presenceHandlers.indexOf(handler);\n if (i !== -1) presenceHandlers.splice(i, 1);\n };\n },\n sendPresence(data) {\n ws?.send(JSON.stringify({ type: 'presence', room: options.room, data }));\n },\n connect() {\n ws = new WebSocket(options.url, options.protocols);\n ws.onmessage = (e) => {\n try {\n const msg = JSON.parse(e.data);\n if (msg.type === 'op') {\n for (const h of opHandlers) h(msg);\n } else if (msg.type === 'presence') {\n for (const h of presenceHandlers) h(msg.peerId, msg.data);\n }\n } catch { /* ignore parse errors */ }\n };\n ws.onopen = () => {\n ws?.send(JSON.stringify({ type: 'join', room: options.room }));\n };\n },\n disconnect() {\n ws?.close();\n ws = null;\n },\n };\n}\n\n// =========================================================================\n// In-memory transport (for testing / single-tab)\n// =========================================================================\n\nexport function createLocalTransport(): SyncTransport {\n const opHandlers: Array<(op: SyncOp) => void> = [];\n return {\n send(op) {\n for (const h of opHandlers) h(op);\n },\n onReceive(handler) {\n opHandlers.push(handler);\n let removed = false;\n return () => {\n if (removed) return;\n removed = true;\n const i = opHandlers.indexOf(handler);\n if (i !== -1) opHandlers.splice(i, 1);\n };\n },\n connect() {},\n disconnect() {},\n };\n}\n\n// =========================================================================\n// createSync — the main API\n// =========================================================================\n\nexport interface SyncOptions {\n /** Transport for sending/receiving operations */\n transport?: SyncTransport;\n /** Unique peer ID (default: random) */\n peerId?: string;\n}\n\nexport interface SyncDoc<T extends Record<string, unknown>> {\n /** Reactive synced state — each key is a Signal */\n state: { [K in keyof T]: Signal<T[K]> };\n /** Connected peers (reactive) */\n peers: ReadonlySignal<PeerInfo[]>;\n /** Local presence data */\n presence: Signal<Record<string, unknown>>;\n /** Peer presence map (reactive) */\n peerPresence: ReadonlySignal<Map<string, unknown>>;\n /** This peer's ID */\n peerId: string;\n /** Whether connected */\n connected: ReadonlySignal<boolean>;\n /** Connect to the sync channel */\n connect(): void;\n /** Disconnect */\n disconnect(): void;\n /** Dispose the sync doc */\n dispose(): void;\n}\n\nexport interface PeerInfo {\n id: string;\n joinedAt: number;\n}\n\nlet peerIdCounter = 0;\n\n/**\n * Create a collaborative synced document.\n *\n * ```ts\n * const doc = createSync('room-1', { title: '', count: 0 }, {\n * transport: createWebSocketTransport({ url: 'wss://sync.example.com', room: 'room-1' }),\n * });\n *\n * doc.state.title.set('Hello'); // auto-syncs to all peers\n * doc.peers(); // connected users\n * ```\n */\nexport function createSync<T extends Record<string, unknown>>(\n roomId: string,\n initialState: T,\n options: SyncOptions = {},\n): SyncDoc<T> {\n const peerId = options.peerId ?? `peer-${++peerIdCounter}-${Date.now()}`;\n const transport = options.transport ?? createLocalTransport();\n\n // Create CRDT registers and signals for each state key\n const registers = new Map<string, LWWRegister<unknown>>();\n const stateSignals: Record<string, Signal<unknown>> = {};\n\n for (const [key, value] of Object.entries(initialState)) {\n registers.set(key, new LWWRegister(value, peerId));\n stateSignals[key] = signal(value);\n }\n\n // Intercept signal.set to broadcast operations\n const state = {} as { [K in keyof T]: Signal<T[K]> };\n for (const key of Object.keys(initialState)) {\n const original = stateSignals[key];\n const register = registers.get(key)!;\n\n const proxy: Signal<any> = (() => original()) as any;\n proxy.set = (value: any) => {\n register.set(value, peerId);\n original.set(value);\n transport.send({\n type: 'set',\n key,\n value,\n timestamp: register.timestamp,\n peerId,\n });\n };\n proxy.update = (fn: (prev: any) => any) => {\n proxy.set(fn(original()));\n };\n proxy.peek = () => original.peek();\n\n (state as any)[key] = proxy;\n }\n\n // Peers\n const peers = signal<PeerInfo[]>([]);\n const connected = signal(false);\n const presence = signal<Record<string, unknown>>({});\n const peerPresenceMap = signal(new Map<string, unknown>());\n\n // Listen for remote operations\n const unsubOps = transport.onReceive((op) => {\n if (op.type === 'set' && registers.has(op.key)) {\n const register = registers.get(op.key)!;\n const merged = register.merge({\n value: op.value,\n timestamp: op.timestamp,\n peerId: op.peerId,\n });\n if (merged) {\n stateSignals[op.key].set(op.value);\n }\n }\n });\n\n // Listen for presence\n const unsubPresence = transport.onPresence?.((remotePeerId, data) => {\n peerPresenceMap.update((map) => {\n const next = new Map(map);\n next.set(remotePeerId, data);\n return next;\n });\n });\n\n return {\n state,\n peers: () => peers(),\n presence,\n peerPresence: () => peerPresenceMap(),\n peerId,\n connected: () => connected(),\n connect() {\n transport.connect();\n connected.set(true);\n },\n disconnect() {\n transport.disconnect();\n connected.set(false);\n },\n dispose() {\n unsubOps();\n unsubPresence?.();\n transport.disconnect();\n },\n };\n}\n"]}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/errors.ts","../src/context.ts","../src/dom.ts","../src/component.ts"],"names":["errors","DOC_BASE","formatError","code","context","def","msg","akashError","getErrorDef","getAllErrorCodes","CONTEXT_BRAND","currentScope","pushScope","parent","scope","popScope","getCurrentScope","runInScope","fn","prev","createContext","defaultValue","provide","key","value","inject","fallback","createElement","tag","attrs","el","setProperty","createText","cloneTemplate","html","template","event","bindText","node","effect","bindProperty","bindVisible","createAnchor","label","renderConditional","anchor","condition","trueBranch","falseBranch","current","dispose","branch","liveParent","fragment","nodes","renderList","items","keyFn","renderItem","currentItems","newData","newItems","oldMap","item","i","data","existing","firstNode","realParent","Show","props","container","getWhen","whenValue","nodeToDOM","For","getEach","index","child","insert","domNode","__getter","currentHooks","onMount","onUnmount","onError","ref","defineComponent","setup","component","rawProps","childrenProp","restProps","ctx","hooks","prevHooks","parentScope","renderFn","untrack","rendered","err","handler","mountFn","cleanup"],"mappings":"sCAqBA,IAAMA,EAAmC,CAEvC,MAAA,CAAQ,CACN,IAAA,CAAM,QAAA,CACN,QAAS,8CAAA,CACT,IAAA,CAAM,oDACR,CAAA,CACA,MAAA,CAAQ,CACN,IAAA,CAAM,QAAA,CACN,QAAS,6CAAA,CACT,IAAA,CAAM,mDACR,CAAA,CACA,MAAA,CAAQ,CACN,IAAA,CAAM,QAAA,CACN,QAAS,yCAAA,CACT,IAAA,CAAM,gEACR,CAAA,CAGA,MAAA,CAAQ,CACN,IAAA,CAAM,QAAA,CACN,QAAS,8CAAA,CACT,IAAA,CAAM,oDACR,CAAA,CACA,MAAA,CAAQ,CACN,IAAA,CAAM,QAAA,CACN,QAAS,gDAAA,CACT,IAAA,CAAM,sDACR,CAAA,CACA,OAAQ,CACN,IAAA,CAAM,SACN,OAAA,CAAS,8CAAA,CACT,KAAM,oDACR,CAAA,CAGA,OAAQ,CACN,IAAA,CAAM,SACN,OAAA,CAAS,kDAAA,CACT,KAAM,uEACR,CAAA,CACA,OAAQ,CACN,IAAA,CAAM,SACN,OAAA,CAAS,yCAAA,CACT,KAAM,gHACR,CAAA,CAGA,OAAQ,CACN,IAAA,CAAM,SACN,OAAA,CAAS,gDAAA,CACT,KAAM,uEACR,CAAA,CACA,OAAQ,CACN,IAAA,CAAM,SACN,OAAA,CAAS,2BAAA,CACT,KAAM,gEACR,CAAA,CAGA,MAAA,CAAQ,CACN,KAAM,QAAA,CACN,OAAA,CAAS,+CACT,IAAA,CAAM,gEACR,EACA,MAAA,CAAQ,CACN,KAAM,QAAA,CACN,OAAA,CAAS,oCACT,IAAA,CAAM,8DACR,EACA,MAAA,CAAQ,CACN,KAAM,QAAA,CACN,OAAA,CAAS,8BACT,IAAA,CAAM,oDACR,EAGA,MAAA,CAAQ,CACN,KAAM,QAAA,CACN,OAAA,CAAS,gCACT,IAAA,CAAM,mFACR,EACA,MAAA,CAAQ,CACN,KAAM,QAAA,CACN,OAAA,CAAS,6BACT,IAAA,CAAM,kGACR,EAGA,MAAA,CAAQ,CACN,KAAM,QAAA,CACN,OAAA,CAAS,uBACT,IAAA,CAAM,4DACR,EACA,MAAA,CAAQ,CACN,KAAM,QAAA,CACN,OAAA,CAAS,2CACT,IAAA,CAAM,wDACR,CACF,CAAA,CAIMC,CAAAA,CAAW,6BAKV,SAASC,CAAAA,CAAYC,EAAcC,CAAAA,CAA0B,CAClE,IAAMC,CAAAA,CAAML,CAAAA,CAAOG,CAAI,CAAA,CACvB,GAAI,CAACE,CAAAA,CACH,OAAO,YAAYF,CAAI,CAAA,qBAAA,CAAA,CAGzB,IAAIG,CAAAA,CAAM,CAAA,SAAA,EAAYH,CAAI,CAAA,EAAA,EAAKE,CAAAA,CAAI,OAAO,CAAA,CAAA,CAC1C,OAAID,IACFE,CAAAA,EAAO;AAAA,EAAA,EAAOF,CAAO,IAEvBE,CAAAA,EAAO;AAAA,EAAA,EAAOD,CAAAA,CAAI,IAAI,CAAA,CAAA,CACtBC,CAAAA,EAAO;AAAA,OAAA,EAAYL,CAAQ,CAAA,CAAA,EAAIE,CAAI,CAAA,CAAA,CAC5BG,CACT,CAKO,SAASC,CAAAA,CAAWJ,CAAAA,CAAcC,CAAAA,CAAyB,CAChE,OAAO,IAAI,KAAA,CAAMF,CAAAA,CAAYC,EAAMC,CAAO,CAAC,CAC7C,CAKO,SAASI,CAAAA,CAAYL,CAAAA,CAAoC,CAC9D,OAAOH,EAAOG,CAAI,CACpB,CAKO,SAASM,GAA6B,CAC3C,OAAO,MAAA,CAAO,IAAA,CAAKT,CAAM,CAC3B,CCxJA,IAAMU,CAAAA,CAAgB,OAAO,eAAe,CAAA,CAgBxCC,CAAAA,CAAoC,IAAA,CAGjC,SAASC,CAAAA,CAAUC,CAAAA,CAA8BF,CAAAA,CAA4B,CAClF,IAAMG,CAAAA,CAAsB,CAAE,MAAA,CAAQ,IAAI,IAAO,MAAA,CAAAD,CAAO,CAAA,CACxD,OAAAF,EAAeG,CAAAA,CACRA,CACT,CAGO,SAASC,EAASD,CAAAA,CAA2B,CAClDH,CAAAA,CAAeG,CAAAA,CAAM,OACvB,CAGO,SAASE,CAAAA,EAAuC,CACrD,OAAOL,CACT,CAOO,SAASM,CAAAA,CAAcH,EAAqBI,CAAAA,CAAgB,CACjE,IAAMC,CAAAA,CAAOR,EACbA,CAAAA,CAAeG,CAAAA,CACf,GAAI,CACF,OAAOI,CAAAA,EACT,CAAA,OAAE,CACAP,EAAeQ,EACjB,CACF,CAWO,SAASC,EAAiBC,CAAAA,CAAmC,CAClE,OAAO,CACL,CAACX,CAAa,EAAG,IAAA,CACjB,KAAA,CAAO,OACP,YAAA,CAAAW,CAAAA,CACA,EAAA,CAAI,MAAA,CAAO,SAAS,CACtB,CACF,CAQO,SAASC,EAAWC,CAAAA,CAAsBC,CAAAA,CAAgB,CAC/D,GAAI,CAACb,CAAAA,CACH,MAAMJ,CAAAA,CAAW,QAAQ,EAE3BI,CAAAA,CAAa,MAAA,CAAO,GAAA,CAAIY,CAAAA,CAAI,GAAIC,CAAK,EACvC,CASO,SAASC,EAAUF,CAAAA,CAAsBG,CAAAA,CAAiB,CAC/D,GAAI,CAACf,CAAAA,CACH,MAAMJ,CAAAA,CAAW,QAAQ,EAI3B,IAAIO,CAAAA,CAA6BH,CAAAA,CACjC,KAAOG,GAAO,CACZ,GAAIA,CAAAA,CAAM,MAAA,CAAO,IAAIS,CAAAA,CAAI,EAAE,CAAA,CACzB,OAAOT,EAAM,MAAA,CAAO,GAAA,CAAIS,CAAAA,CAAI,EAAE,EAEhCT,CAAAA,CAAQA,CAAAA,CAAM,OAChB,CAGA,GAAIY,CAAAA,GAAa,MAAA,CAAW,OAAOA,CAAAA,CACnC,GAAIH,CAAAA,CAAI,YAAA,GAAiB,MAAA,CAAW,OAAOA,EAAI,YAAA,CAE/C,MAAMhB,CAAAA,CAAW,QAAQ,CAC3B,CCvGO,SAASoB,CAAAA,CACdC,CAAAA,CACAC,EACa,CACb,IAAMC,CAAAA,CAAK,QAAA,CAAS,cAAcF,CAAG,CAAA,CACrC,GAAIC,CAAAA,CACF,OAAW,CAACN,CAAAA,CAAKC,CAAK,CAAA,GAAK,OAAO,OAAA,CAAQK,CAAK,CAAA,CAC7CE,CAAAA,CAAYD,EAAIP,CAAAA,CAAKC,CAAK,CAAA,CAG9B,OAAOM,CACT,CAGO,SAASE,CAAAA,CAAWR,CAAAA,CAAqB,CAC9C,OAAO,QAAA,CAAS,cAAA,CAAeA,CAAK,CACtC,CAGO,SAASS,CAAAA,CAAcC,CAAAA,CAAgC,CAC5D,IAAMC,CAAAA,CAAW,QAAA,CAAS,aAAA,CAAc,UAAU,CAAA,CAClD,OAAAA,CAAAA,CAAS,SAAA,CAAYD,EACdC,CAAAA,CAAS,OAAA,CAAQ,SAAA,CAAU,IAAI,CACxC,CAIO,SAASJ,CAAAA,CAAYD,CAAAA,CAAiBP,EAAaC,CAAAA,CAAsB,CAC9E,GAAID,CAAAA,GAAQ,SAAWA,CAAAA,GAAQ,WAAA,CAC7BO,CAAAA,CAAG,SAAA,CAAYN,UACND,CAAAA,GAAQ,OAAA,EAAW,OAAOC,CAAAA,EAAU,UAAYA,CAAAA,GAAU,IAAA,CACnE,MAAA,CAAO,MAAA,CAAOM,EAAG,KAAA,CAAON,CAAK,CAAA,CAAA,KAAA,GACpBD,CAAAA,GAAQ,YACjBO,CAAAA,CAAG,SAAA,CAAYN,CAAAA,CAAAA,KAAAA,GACND,CAAAA,GAAQ,MAEZ,GAAIA,CAAAA,CAAI,UAAA,CAAW,IAAI,EAAG,CAC/B,IAAMa,CAAAA,CAAQb,CAAAA,CAAI,MAAM,CAAC,CAAA,CAAE,WAAA,EAAY,CACvCO,EAAG,gBAAA,CAAiBM,CAAAA,CAAOZ,CAAsB,EACnD,MAAWD,CAAAA,IAAOO,CAAAA,CACfA,CAAAA,CAA+BP,CAAG,EAAIC,CAAAA,CAC9BA,CAAAA,GAAU,OAASA,CAAAA,EAAS,IAAA,CACrCM,EAAG,eAAA,CAAgBP,CAAG,CAAA,CAEtBO,CAAAA,CAAG,aAAaP,CAAAA,CAAKC,CAAAA,GAAU,IAAA,CAAO,EAAA,CAAK,OAAOA,CAAK,CAAC,EAE5D,CAKO,SAASa,CAAAA,CAASC,CAAAA,CAAYpB,GAAAA,CAA+B,CAClE,OAAOqB,CAAAA,CACL,IAAM,CACJD,CAAAA,CAAK,YAAc,MAAA,CAAOpB,GAAAA,EAAI,EAChC,EACA,CAAE,MAAA,CAAQ,IAAK,CACjB,CACF,CAGO,SAASsB,CAAAA,CACdV,CAAAA,CACAP,IACAL,CAAAA,CACY,CACZ,OAAOqB,CAAAA,CACL,IAAM,CACJR,CAAAA,CAAYD,CAAAA,CAAIP,GAAAA,CAAKL,GAAI,EAC3B,CAAA,CACA,CAAE,OAAQ,IAAK,CACjB,CACF,CAGO,SAASuB,CAAAA,CAAYX,CAAAA,CAAiBZ,GAAAA,CAA+B,CAC1E,OAAOqB,CAAAA,CACL,IAAM,CACJT,CAAAA,CAAG,MAAM,OAAA,CAAUZ,GAAAA,EAAG,CAAI,EAAA,CAAK,OACjC,CAAA,CACA,CAAE,MAAA,CAAQ,IAAK,CACjB,CACF,CAUA,SAASwB,CAAAA,CAAaC,EAAQ,EAAA,CAAY,CACxC,OAAO,QAAA,CAAS,cAAcA,CAAK,CACrC,CAyBO,SAASC,EACd/B,CAAAA,CACAgC,GAAAA,CACAC,CAAAA,CACAC,CAAAA,CACAC,EACY,CACZ,IAAIC,CAAAA,CAAmC,IAAA,CAGjCnC,EAAQE,CAAAA,EAAgB,CAExBkC,CAAAA,CAAUX,CAAAA,CACd,IAAM,CACJ,IAAMf,CAAAA,CAAQsB,CAAAA,GAGd,GAAIG,CAAAA,CAAS,CACX,IAAA,IAAWX,KAAQW,CAAAA,CAAQ,KAAA,CACzB,GAAI,CAAEX,EAAK,UAAA,EAAY,WAAA,CAAYA,CAAI,EAAG,MAAQ,CAAC,CAErDW,CAAAA,CAAQ,OAAA,KACRA,CAAAA,CAAU,KACZ,CAKA,IAAME,EAAS3B,CAAAA,CAAQuB,CAAAA,CAAaC,CAAAA,CAChCI,CAAAA,CAAaP,IAAO,UAAA,CAIxB,GAHI,CAACO,CAAAA,EAAcH,GAAS,KAAA,CAAM,CAAC,CAAA,EAAG,UAAA,GACpCG,EAAaH,CAAAA,CAAQ,KAAA,CAAM,CAAC,CAAA,CAAE,YAE5BE,CAAAA,EAAUC,CAAAA,CAAY,CACxB,IAAMC,EAAWvC,CAAAA,CAAQG,CAAAA,CAAWH,CAAAA,CAAOqC,CAAM,EAAIA,CAAAA,EAAO,CACtDG,CAAAA,CAAQD,CAAAA,YAAoB,iBAC9B,KAAA,CAAM,IAAA,CAAKA,CAAAA,CAAS,UAAU,EAC9B,CAACA,CAAQ,CAAA,CACb,IAAA,IAAWf,KAAQgB,CAAAA,CACjBF,CAAAA,CAAW,YAAA,CAAad,CAAAA,CAAMO,GAAM,CAAA,CAEtCI,CAAAA,CAAU,CAAE,KAAA,CAAAK,EAAO,OAAA,CAAS,IAAK,EACnC,CACF,EACA,CAAE,MAAA,CAAQ,IAAK,CACjB,EAEA,OAAO,IAAM,CAEX,GADAJ,GAAQ,CACJD,CAAAA,CAAS,CACX,IAAA,IAAWX,KAAQW,CAAAA,CAAQ,KAAA,CACzB,GAAI,CAAEX,EAAK,UAAA,EAAY,WAAA,CAAYA,CAAI,EAAG,MAAQ,CAAC,CAErDW,CAAAA,CAAQ,OAAA,KACV,CACF,CACF,CAeO,SAASM,EACd1C,CAAAA,CACAgC,GAAAA,CACAW,CAAAA,CACAC,CAAAA,CACAC,EACY,CACZ,IAAIC,CAAAA,CAA8B,GAG5B7C,CAAAA,CAAQE,CAAAA,EAAgB,CAExBkC,CAAAA,CAAUX,EACd,IAAM,CACJ,IAAMqB,CAAAA,CAAUJ,GAAM,CAChBK,CAAAA,CAA0B,EAAC,CAC3BC,EAAS,IAAI,GAAA,CAEnB,IAAA,IAAWC,CAAAA,IAAQJ,EACjBG,CAAAA,CAAO,GAAA,CAAIC,CAAAA,CAAK,GAAA,CAAKA,CAAI,CAAA,CAI3B,IAAA,IAASC,CAAAA,CAAI,CAAA,CAAGA,EAAIJ,CAAAA,CAAQ,MAAA,CAAQI,CAAAA,EAAAA,CAAK,CACvC,IAAMC,CAAAA,CAAOL,CAAAA,CAAQI,CAAC,CAAA,CAChBzC,EAAMkC,CAAAA,CAAMQ,CAAAA,CAAMD,CAAC,CAAA,CACnBE,EAAWJ,CAAAA,CAAO,GAAA,CAAIvC,CAAG,CAAA,CAE/B,GAAI2C,CAAAA,CAAU,CACZJ,CAAAA,CAAO,MAAA,CAAOvC,CAAG,CAAA,CAGjB,IAAM4C,CAAAA,CAAYD,CAAAA,CAAS,MAAM,CAAC,CAAA,CAClC,GAAIC,CAAAA,EAAa,CAACA,CAAAA,CAAU,UAAA,CAAY,CAEtC,IAAMd,EAAWvC,CAAAA,CAAQG,CAAAA,CAAWH,EAAO,IAAM4C,CAAAA,CAAWO,EAAMD,CAAC,CAAC,CAAA,CAAIN,CAAAA,CAAWO,EAAMD,CAAC,CAAA,CACpFV,CAAAA,CAAQD,CAAAA,YAAoB,iBAC9B,KAAA,CAAM,IAAA,CAAKA,CAAAA,CAAS,UAAU,EAC9B,CAACA,CAAQ,CAAA,CACbQ,CAAAA,CAAS,KAAK,CAAE,GAAA,CAAAtC,CAAAA,CAAK,KAAA,CAAO0C,EAAM,KAAA,CAAAX,CAAAA,CAAO,OAAA,CAAS,IAAK,CAAC,EAC1D,CAAA,KACEY,CAAAA,CAAS,KAAA,CAAQD,EACjBJ,CAAAA,CAAS,IAAA,CAAKK,CAAQ,EAE1B,MAAO,CACL,IAAMb,CAAAA,CAAWvC,CAAAA,CAAQG,EAAWH,CAAAA,CAAO,IAAM4C,CAAAA,CAAWO,CAAAA,CAAMD,CAAC,CAAC,CAAA,CAAIN,CAAAA,CAAWO,CAAAA,CAAMD,CAAC,CAAA,CACpFV,CAAAA,CAAQD,CAAAA,YAAoB,gBAAA,CAC9B,MAAM,IAAA,CAAKA,CAAAA,CAAS,UAAU,CAAA,CAC9B,CAACA,CAAQ,CAAA,CACbQ,CAAAA,CAAS,IAAA,CAAK,CAAE,GAAA,CAAAtC,CAAAA,CAAK,KAAA,CAAO0C,CAAAA,CAAM,MAAAX,CAAAA,CAAO,OAAA,CAAS,IAAK,CAAC,EAC1D,CACF,CAGA,IAAA,IAAWS,CAAAA,IAAQD,EAAO,MAAA,EAAO,CAAG,CAClC,IAAA,IAAWxB,KAAQyB,CAAAA,CAAK,KAAA,CAClBzB,CAAAA,CAAK,UAAA,EAAYA,EAAK,UAAA,CAAW,WAAA,CAAYA,CAAI,CAAA,CAEvDyB,EAAK,OAAA,KACP,CAGA,IAAMX,EAAaP,GAAAA,CAAO,UAAA,CAC1B,GAAIO,CAAAA,CAEF,QAAWW,CAAAA,IAAQF,CAAAA,CACjB,IAAA,IAAWvB,CAAAA,IAAQyB,EAAK,KAAA,CACtBX,CAAAA,CAAW,YAAA,CAAad,CAAAA,CAAMO,GAAM,CAAA,CAAA,KAGnC,CAGL,IAAIuB,CAAAA,CAA0B,KAC9B,IAAA,IAAWL,CAAAA,IAAQJ,CAAAA,CACjB,GAAII,EAAK,KAAA,CAAM,CAAC,CAAA,EAAG,UAAA,CAAY,CAAEK,CAAAA,CAAaL,CAAAA,CAAK,KAAA,CAAM,CAAC,EAAE,UAAA,CAAY,KAAO,CAEjF,GAAIK,EAGF,IAAA,IAAWL,CAAAA,IAAQF,CAAAA,CACjB,IAAA,IAAWvB,KAAQyB,CAAAA,CAAK,KAAA,CACtBK,CAAAA,CAAW,WAAA,CAAY9B,CAAI,EAInC,CAEAqB,CAAAA,CAAeE,EACjB,EACA,CAAE,MAAA,CAAQ,IAAK,CACjB,EAEA,OAAO,IAAM,CACXX,CAAAA,GACA,IAAA,IAAWa,CAAAA,IAAQJ,CAAAA,CAAc,CAC/B,QAAWrB,CAAAA,IAAQyB,CAAAA,CAAK,KAAA,CAClBzB,CAAAA,CAAK,YAAYA,CAAAA,CAAK,UAAA,CAAW,WAAA,CAAYA,CAAI,EAEvDyB,CAAAA,CAAK,OAAA,KACP,CACAJ,EAAe,GACjB,CACF,CAgBO,SAASU,CAAAA,CAAQC,CAAAA,CAA2B,CACjD,IAAMzB,EAASH,CAAAA,CAAa,MAAM,CAAA,CAC5B6B,CAAAA,CAAY,SAAS,sBAAA,EAAuB,CAClDA,CAAAA,CAAU,WAAA,CAAY1B,CAAM,CAAA,CAI5B,IAAM2B,CAAAA,CAAU,OAAOF,EAAM,IAAA,EAAS,UAAA,CAClCA,CAAAA,CAAM,IAAA,CACN,IAAMA,CAAAA,CAAM,IAAA,CAIZG,CAAAA,CACJ,OAAA7B,EACE2B,CAAAA,CACA1B,CAAAA,CACA,KAAQ4B,CAAAA,CAAYD,GAAQ,CAAU,CAAC,CAACC,CAAAA,CAAAA,CACxC,IAAMC,CAAAA,CAAUJ,CAAAA,CAAM,QAAA,CAASG,CAAc,CAAC,CAAA,CAC9CH,CAAAA,CAAM,QAAA,CAAW,IAAMI,EAAUJ,CAAAA,CAAM,QAAA,EAAW,CAAA,CAAI,MACxD,CAAA,CAEOC,CACT,CAaO,SAASI,GAAOL,CAAAA,CAA0B,CAC/C,IAAMzB,CAAAA,CAASH,EAAa,KAAK,CAAA,CAC3B6B,CAAAA,CAAY,QAAA,CAAS,wBAAuB,CAClDA,CAAAA,CAAU,WAAA,CAAY1B,CAAM,EAI5B,IAAM+B,CAAAA,CAAU,OAAON,CAAAA,CAAM,MAAS,UAAA,CAClCA,CAAAA,CAAM,IAAA,CACN,IAAMA,EAAM,IAAA,CAEhB,OAAAf,CAAAA,CACEgB,CAAAA,CACA1B,EACA+B,CAAAA,CACAN,CAAAA,CAAM,GAAA,CACN,CAACP,EAAMc,CAAAA,GAAUH,CAAAA,CAAUJ,CAAAA,CAAM,QAAA,CAASP,EAAMc,CAAK,CAAC,CACxD,CAAA,CAEON,CACT,CAKO,SAASG,CAAAA,CAAUpC,CAAAA,CAAuB,CAC/C,GAAIA,CAAAA,EAAQ,IAAA,EAAQ,OAAOA,GAAS,SAAA,CAClC,OAAO,QAAA,CAAS,cAAA,CAAe,EAAE,CAAA,CAEnC,GAAI,OAAOA,CAAAA,EAAS,QAAA,EAAY,OAAOA,CAAAA,EAAS,QAAA,CAC9C,OAAO,QAAA,CAAS,eAAe,MAAA,CAAOA,CAAI,CAAC,CAAA,CAE7C,GAAIA,CAAAA,YAAgB,IAAA,CAClB,OAAOA,CAAAA,CAET,GAAI,KAAA,CAAM,OAAA,CAAQA,CAAI,CAAA,CAAG,CACvB,IAAMe,CAAAA,CAAW,QAAA,CAAS,sBAAA,GAC1B,IAAA,IAAWyB,CAAAA,IAASxC,CAAAA,CAClBe,CAAAA,CAAS,YAAYqB,CAAAA,CAAUI,CAAK,CAAC,CAAA,CAEvC,OAAOzB,CACT,CACA,OAAO,QAAA,CAAS,eAAe,MAAA,CAAOf,CAAI,CAAC,CAC7C,CAGO,SAASyC,EAAAA,CAAOlE,CAAAA,CAAcyB,CAAAA,CAAiBO,EAAqB,CACzE,IAAMmC,CAAAA,CAAUN,CAAAA,CAAUpC,CAAI,CAAA,CAC1BO,CAAAA,CACFhC,CAAAA,CAAO,YAAA,CAAamE,EAASnC,CAAM,CAAA,CAEnChC,CAAAA,CAAO,WAAA,CAAYmE,CAAO,EAE9B,CC5YO,SAASC,EAAAA,CAAY/D,EAA+C,CACzE,OAACA,CAAAA,CAAW,UAAA,CAAa,KAClBA,CACT,CA0BA,IAAIgE,CAAAA,CAAsC,KAQnC,SAASC,EAAAA,CAAQjE,CAAAA,CAAqC,CAC3D,GAAI,CAACgE,CAAAA,CACH,MAAM3E,CAAAA,CAAW,QAAQ,CAAA,CAE3B2E,CAAAA,CAAa,KAAA,CAAM,IAAA,CAAKhE,CAAE,EAC5B,CAKO,SAASkE,EAAAA,CAAUlE,EAAsB,CAC9C,GAAI,CAACgE,CAAAA,CACH,MAAM3E,CAAAA,CAAW,QAAQ,CAAA,CAE3B2E,CAAAA,CAAa,QAAQ,IAAA,CAAKhE,CAAE,EAC9B,CAKO,SAASmE,EAAAA,CAAQnE,CAAAA,CAAkC,CACxD,GAAI,CAACgE,CAAAA,CACH,MAAM3E,CAAAA,CAAW,QAAQ,EAE3B2E,CAAAA,CAAa,KAAA,CAAM,IAAA,CAAKhE,CAAE,EAC5B,CAKO,SAASoE,EAAAA,EAA+B,CAC7C,OAAO,CAAE,OAAA,CAAS,MAAU,CAC9B,CAeO,SAASC,EAAAA,CACdC,CAAAA,CACc,CACd,IAAMC,CAAAA,CAAaC,CAAAA,EAAqE,CAEtF,GAAM,CAAE,QAAA,CAAUC,CAAAA,CAAc,GAAGC,CAAU,EAAIF,CAAAA,EAAY,EAAC,CAaxDG,CAAAA,CAA2B,CAC/B,KAAA,CARYD,CAAAA,CASZ,QAAA,CANA,OAAOD,GAAiB,UAAA,CACpBA,CAAAA,CACA,IAAMA,CAAAA,EAAgB,IAK5B,CAAA,CAGMG,CAAAA,CAAwB,CAAE,KAAA,CAAO,EAAC,CAAG,OAAA,CAAS,EAAC,CAAG,MAAO,EAAG,CAAA,CAC5DC,CAAAA,CAAYb,EAClBA,CAAAA,CAAeY,CAAAA,CAGf,IAAME,CAAAA,CAAchF,GAAgB,CAC9BF,CAAAA,CAAQF,CAAAA,CAAUoF,CAAW,EAE/BC,GAAAA,CACAjB,CAAAA,CAEJ,GAAI,CAMFA,EAAUkB,CAAAA,CAAQ,IAAM,CAEtB,GADAD,IAAWT,CAAAA,CAAMK,CAAG,CAAA,CAChB,OAAOI,KAAa,UAAA,CAAc,MAAM1F,CAAAA,CAAW,QAAQ,EAC/D,IAAM4F,CAAAA,CAAWF,GAAAA,EAAS,CAC1B,OAAOvB,CAAAA,CAAUyB,CAAQ,CAC3B,CAAC,EACH,CAAA,MAASC,CAAAA,CAAK,CAGZ,GAFArF,EAASD,CAAK,CAAA,CACdoE,CAAAA,CAAea,CAAAA,CACXD,EAAM,KAAA,CAAM,MAAA,CAAS,CAAA,CAAG,CAC1B,QAAWO,CAAAA,IAAWP,CAAAA,CAAM,KAAA,CAC1BO,CAAAA,CAAQD,aAAe,KAAA,CAAQA,CAAAA,CAAM,IAAI,KAAA,CAAM,OAAOA,CAAG,CAAC,CAAC,CAAA,CAE7D,OAAO,QAAA,CAAS,aAAA,CAAc,OAAO,CACvC,CACA,MAAMA,CACR,CAGA,OAAArF,EAASD,CAAK,CAAA,CACdoE,CAAAA,CAAea,CAAAA,CAGXD,EAAM,KAAA,CAAM,MAAA,CAAS,CAAA,EACvB,cAAA,CAAe,IAAM,CACnB,IAAA,IAAWQ,CAAAA,IAAWR,CAAAA,CAAM,MAC1B,GAAI,CACF,IAAMS,CAAAA,CAAUD,GAAQ,CACpB,OAAOC,CAAAA,EAAY,UAAA,EACrBT,EAAM,OAAA,CAAQ,IAAA,CAAKS,CAAO,EAE9B,OAASH,CAAAA,CAAK,CACZ,IAAA,IAAWC,CAAAA,IAAWP,EAAM,KAAA,CAC1BO,CAAAA,CAAQD,CAAAA,YAAe,KAAA,CAAQA,EAAM,IAAI,KAAA,CAAM,MAAA,CAAOA,CAAG,CAAC,CAAC,EAE/D,CAEJ,CAAC,EAGIpB,CACT,CAAA,CAEA,OAAAS,CAAAA,CAAU,MAAA,CAAS,KACZA,CACT","file":"chunk-DYJUCZXA.js","sourcesContent":["/**\n * Error message catalog.\n *\n * Every AkashJS runtime error has a unique code (AK0001, AK0002, etc.),\n * a one-line description, and a link to detailed documentation.\n *\n * Usage:\n * throw akashError('AK0010', 'provide()');\n */\n\n// --- Error registry ---\n\nexport interface ErrorDef {\n /** Error code */\n code: string;\n /** Short description */\n message: string;\n /** Extended explanation and fix suggestion */\n hint: string;\n}\n\nconst errors: Record<string, ErrorDef> = {\n // --- Context errors (AK001x) ---\n AK0010: {\n code: 'AK0010',\n message: 'provide() called outside of component setup.',\n hint: 'provide() must be called inside defineComponent().',\n },\n AK0012: {\n code: 'AK0012',\n message: 'inject() called outside of component setup.',\n hint: 'inject() must be called inside defineComponent().',\n },\n AK0013: {\n code: 'AK0013',\n message: 'No provider found for injected context.',\n hint: 'Make sure an ancestor component calls provide() with this key.',\n },\n\n // --- Lifecycle errors (AK002x) ---\n AK0020: {\n code: 'AK0020',\n message: 'onMount() called outside of component setup.',\n hint: 'onMount() must be called inside defineComponent().',\n },\n AK0021: {\n code: 'AK0021',\n message: 'onUnmount() called outside of component setup.',\n hint: 'onUnmount() must be called inside defineComponent().',\n },\n AK0022: {\n code: 'AK0022',\n message: 'onError() called outside of component setup.',\n hint: 'onError() must be called inside defineComponent().',\n },\n\n // --- Signal errors (AK003x) ---\n AK0030: {\n code: 'AK0030',\n message: 'Circular dependency detected in computed signal.',\n hint: 'A computed signal is reading itself, directly or via other computeds.',\n },\n AK0031: {\n code: 'AK0031',\n message: 'Signal set() called during computation.',\n hint: 'Do not set signals inside computed() or during effect execution. Use batch() or set signals in event handlers.',\n },\n\n // --- Component errors (AK004x) ---\n AK0040: {\n code: 'AK0040',\n message: 'Component setup must return a render function.',\n hint: 'The function passed to defineComponent() must return () => AkashNode.',\n },\n AK0041: {\n code: 'AK0041',\n message: 'Required prop is missing.',\n hint: 'Check that the parent component is passing all required props.',\n },\n\n // --- Router errors (AK005x) ---\n AK0050: {\n code: 'AK0050',\n message: 'useRoute() called outside of router context.',\n hint: 'Make sure your component is rendered inside a router provider.',\n },\n AK0051: {\n code: 'AK0051',\n message: 'No route matched the current URL.',\n hint: 'Add a catch-all route ([...rest]) to handle unmatched paths.',\n },\n AK0052: {\n code: 'AK0052',\n message: 'Route guard threw an error.',\n hint: 'Check the guard function for unhandled exceptions.',\n },\n\n // --- Form errors (AK006x) ---\n AK0060: {\n code: 'AK0060',\n message: 'Form submitted while invalid.',\n hint: 'The submit handler was not called because validation failed. Check form.errors().',\n },\n AK0061: {\n code: 'AK0061',\n message: 'Async validator timed out.',\n hint: 'The async validator did not resolve within the expected time. Check your async validation logic.',\n },\n\n // --- HTTP errors (AK007x) ---\n AK0070: {\n code: 'AK0070',\n message: 'HTTP request failed.',\n hint: 'Check the server response status and network connectivity.',\n },\n AK0071: {\n code: 'AK0071',\n message: 'createResource() fetcher threw an error.',\n hint: 'Check the fetcher function passed to createResource().',\n },\n};\n\n// --- Public API ---\n\nconst DOC_BASE = 'https://akashjs.dev/errors';\n\n/**\n * Format an AkashJS error with code, message, hint, and doc link.\n */\nexport function formatError(code: string, context?: string): string {\n const def = errors[code];\n if (!def) {\n return `[AkashJS ${code}] Unknown error code.`;\n }\n\n let msg = `[AkashJS ${code}] ${def.message}`;\n if (context) {\n msg += `\\n ${context}`;\n }\n msg += `\\n ${def.hint}`;\n msg += `\\n See: ${DOC_BASE}/${code}`;\n return msg;\n}\n\n/**\n * Create and throw an AkashJS error.\n */\nexport function akashError(code: string, context?: string): Error {\n return new Error(formatError(code, context));\n}\n\n/**\n * Get the error definition for a code.\n */\nexport function getErrorDef(code: string): ErrorDef | undefined {\n return errors[code];\n}\n\n/**\n * Get all registered error codes.\n */\nexport function getAllErrorCodes(): string[] {\n return Object.keys(errors);\n}\n","/**\n * Lightweight dependency injection via provide/inject.\n *\n * Context is scoped to the component tree — no injector hierarchy,\n * no decorators, no classes. Just createContext(), provide(), inject().\n */\n\n// --- Types ---\n\nimport { akashError } from './errors.js';\n\nconst CONTEXT_BRAND = Symbol('akash.context');\n\nexport interface InjectionKey<T> {\n readonly [CONTEXT_BRAND]: true;\n readonly _type: T; // phantom type — never used at runtime\n readonly defaultValue: T | undefined;\n readonly id: symbol;\n}\n\n// --- Context stack (managed by component system) ---\n\ninterface ContextScope {\n values: Map<symbol, unknown>;\n parent: ContextScope | null;\n}\n\nlet currentScope: ContextScope | null = null;\n\n/** @internal — called by defineComponent to push/pop context scopes */\nexport function pushScope(parent: ContextScope | null = currentScope): ContextScope {\n const scope: ContextScope = { values: new Map(), parent };\n currentScope = scope;\n return scope;\n}\n\n/** @internal */\nexport function popScope(scope: ContextScope): void {\n currentScope = scope.parent;\n}\n\n/** @internal */\nexport function getCurrentScope(): ContextScope | null {\n return currentScope;\n}\n\n/**\n * Run a function within a given scope so that provide/inject\n * work correctly for components created asynchronously\n * (e.g., lazy-loaded route components).\n */\nexport function runInScope<T>(scope: ContextScope, fn: () => T): T {\n const prev = currentScope;\n currentScope = scope;\n try {\n return fn();\n } finally {\n currentScope = prev;\n }\n}\n\n// --- Public API ---\n\n/**\n * Create a typed context key with an optional default value.\n *\n * ```ts\n * const ThemeContext = createContext<'light' | 'dark'>('light');\n * ```\n */\nexport function createContext<T>(defaultValue?: T): InjectionKey<T> {\n return {\n [CONTEXT_BRAND]: true,\n _type: undefined as T,\n defaultValue,\n id: Symbol('context'),\n };\n}\n\n/**\n * Provide a value for a context key in the current component scope.\n * All descendant components can inject() this value.\n *\n * Must be called inside defineComponent() setup.\n */\nexport function provide<T>(key: InjectionKey<T>, value: T): void {\n if (!currentScope) {\n throw akashError('AK0010');\n }\n currentScope.values.set(key.id, value);\n}\n\n/**\n * Inject a value from the nearest ancestor that provided it.\n *\n * Must be called inside defineComponent() setup.\n */\nexport function inject<T>(key: InjectionKey<T>): T;\nexport function inject<T>(key: InjectionKey<T>, fallback: T): T;\nexport function inject<T>(key: InjectionKey<T>, fallback?: T): T {\n if (!currentScope) {\n throw akashError('AK0012');\n }\n\n // Walk up the scope chain\n let scope: ContextScope | null = currentScope;\n while (scope) {\n if (scope.values.has(key.id)) {\n return scope.values.get(key.id) as T;\n }\n scope = scope.parent;\n }\n\n // Check fallback, then default\n if (fallback !== undefined) return fallback;\n if (key.defaultValue !== undefined) return key.defaultValue;\n\n throw akashError('AK0013');\n}\n","/**\n * Direct DOM rendering runtime.\n *\n * No virtual DOM. The compiler generates calls to these helpers.\n * Signal reads inside templates become fine-grained effects that\n * update only the specific DOM node that changed.\n */\n\nimport { effect } from './signals.js';\nimport { getCurrentScope, runInScope } from './context.js';\nimport type { AkashNode } from './types.js';\n\n// --- DOM creation helpers (used by compiler output) ---\n\n/** Create an element and optionally set static attributes */\nexport function createElement(\n tag: string,\n attrs?: Record<string, unknown>,\n): HTMLElement {\n const el = document.createElement(tag);\n if (attrs) {\n for (const [key, value] of Object.entries(attrs)) {\n setProperty(el, key, value);\n }\n }\n return el;\n}\n\n/** Create a text node */\nexport function createText(value: string): Text {\n return document.createTextNode(value);\n}\n\n/** Clone a template element for static structure */\nexport function cloneTemplate(html: string): DocumentFragment {\n const template = document.createElement('template');\n template.innerHTML = html;\n return template.content.cloneNode(true) as DocumentFragment;\n}\n\n// --- Property/attribute setting ---\n\nexport function setProperty(el: HTMLElement, key: string, value: unknown): void {\n if (key === 'class' || key === 'className') {\n el.className = value as string;\n } else if (key === 'style' && typeof value === 'object' && value !== null) {\n Object.assign(el.style, value);\n } else if (key === 'innerHTML') {\n el.innerHTML = value as string;\n } else if (key === 'ref') {\n // Handled separately by component system\n } else if (key.startsWith('on')) {\n const event = key.slice(2).toLowerCase();\n el.addEventListener(event, value as EventListener);\n } else if (key in el) {\n (el as Record<string, unknown>)[key] = value;\n } else if (value === false || value == null) {\n el.removeAttribute(key);\n } else {\n el.setAttribute(key, value === true ? '' : String(value));\n }\n}\n\n// --- Reactive binding (used by compiler for dynamic expressions) ---\n\n/** Bind a reactive expression to a text node's content */\nexport function bindText(node: Text, fn: () => unknown): () => void {\n return effect(\n () => {\n node.textContent = String(fn());\n },\n { render: true },\n );\n}\n\n/** Bind a reactive expression to an element's attribute/property */\nexport function bindProperty(\n el: HTMLElement,\n key: string,\n fn: () => unknown,\n): () => void {\n return effect(\n () => {\n setProperty(el, key, fn());\n },\n { render: true },\n );\n}\n\n/** Bind a reactive expression to an element's visibility (display) */\nexport function bindVisible(el: HTMLElement, fn: () => boolean): () => void {\n return effect(\n () => {\n el.style.display = fn() ? '' : 'none';\n },\n { render: true },\n );\n}\n\n// --- Conditional rendering ---\n\n/** Anchor node — comment for normal DOM, hidden element for table contexts */\ntype Anchor = Comment | HTMLElement;\n\n/** Table elements where comment nodes get relocated by the browser */\nconst TABLE_TAGS = new Set(['TABLE', 'THEAD', 'TBODY', 'TFOOT', 'TR']);\n\nfunction createAnchor(label = ''): Anchor {\n return document.createComment(label);\n}\n\n/**\n * Create a table-safe anchor. Browsers relocate comment nodes inside\n * table elements, breaking anchor-based insertion. Use a hidden element instead.\n */\nfunction createTableSafeAnchor(parent: Node): Anchor {\n if (parent instanceof HTMLElement && TABLE_TAGS.has(parent.tagName)) {\n const tag = parent.tagName === 'TR' ? 'td' : 'tr';\n const el = document.createElement(tag);\n el.style.display = 'none';\n return el;\n }\n return document.createComment('');\n}\n\ninterface ConditionalBlock {\n nodes: Node[];\n dispose: (() => void) | null;\n}\n\n/**\n * Render a conditional block. Swaps DOM fragments based on a reactive\n * condition. Used by the compiler for :if directives and <Show>.\n */\nexport function renderConditional(\n parent: Node,\n anchor: Node,\n condition: () => boolean,\n trueBranch: () => Node,\n falseBranch?: () => Node,\n): () => void {\n let current: ConditionalBlock | null = null;\n\n // Capture scope so children created in branches inherit provide/inject context\n const scope = getCurrentScope();\n\n const dispose = effect(\n () => {\n const value = condition();\n\n // Remove old nodes\n if (current) {\n for (const node of current.nodes) {\n try { node.parentNode?.removeChild(node); } catch {}\n }\n current.dispose?.();\n current = null;\n }\n\n // Insert new nodes — use anchor.parentNode since the anchor may have\n // moved from the initial DocumentFragment into the real DOM.\n // In table contexts, the browser may relocate comment anchors.\n const branch = value ? trueBranch : falseBranch;\n let liveParent = anchor.parentNode;\n if (!liveParent && current?.nodes[0]?.parentNode) {\n liveParent = current.nodes[0].parentNode;\n }\n if (branch && liveParent) {\n const fragment = scope ? runInScope(scope, branch) : branch();\n const nodes = fragment instanceof DocumentFragment\n ? Array.from(fragment.childNodes)\n : [fragment];\n for (const node of nodes) {\n liveParent.insertBefore(node, anchor);\n }\n current = { nodes, dispose: null };\n }\n },\n { render: true },\n );\n\n return () => {\n dispose();\n if (current) {\n for (const node of current.nodes) {\n try { node.parentNode?.removeChild(node); } catch {}\n }\n current.dispose?.();\n }\n };\n}\n\n// --- List rendering ---\n\ninterface ListItem<T> {\n key: unknown;\n value: T;\n nodes: Node[];\n dispose: (() => void) | null;\n}\n\n/**\n * Render a reactive list with keyed reconciliation.\n * Used by the compiler for :for directives and <For>.\n */\nexport function renderList<T>(\n parent: Node,\n anchor: Node,\n items: () => T[],\n keyFn: (item: T, index: number) => unknown,\n renderItem: (item: T, index: number) => Node,\n): () => void {\n let currentItems: ListItem<T>[] = [];\n\n // Capture scope so children created in renderItem inherit provide/inject context\n const scope = getCurrentScope();\n\n const dispose = effect(\n () => {\n const newData = items();\n const newItems: ListItem<T>[] = [];\n const oldMap = new Map<unknown, ListItem<T>>();\n\n for (const item of currentItems) {\n oldMap.set(item.key, item);\n }\n\n // Build new list, reuse existing DOM nodes when keys match\n for (let i = 0; i < newData.length; i++) {\n const data = newData[i];\n const key = keyFn(data, i);\n const existing = oldMap.get(key);\n\n if (existing) {\n oldMap.delete(key);\n // If the reused item's nodes were detached (e.g., parent removed them),\n // treat as new to avoid stale DOM references\n const firstNode = existing.nodes[0];\n if (firstNode && !firstNode.parentNode) {\n // Nodes were detached — create fresh\n const fragment = scope ? runInScope(scope, () => renderItem(data, i)) : renderItem(data, i);\n const nodes = fragment instanceof DocumentFragment\n ? Array.from(fragment.childNodes)\n : [fragment];\n newItems.push({ key, value: data, nodes, dispose: null });\n } else {\n existing.value = data;\n newItems.push(existing);\n }\n } else {\n const fragment = scope ? runInScope(scope, () => renderItem(data, i)) : renderItem(data, i);\n const nodes = fragment instanceof DocumentFragment\n ? Array.from(fragment.childNodes)\n : [fragment];\n newItems.push({ key, value: data, nodes, dispose: null });\n }\n }\n\n // Remove items that are no longer in the list\n for (const item of oldMap.values()) {\n for (const node of item.nodes) {\n if (node.parentNode) node.parentNode.removeChild(node);\n }\n item.dispose?.();\n }\n\n // Reconcile DOM order — insertBefore moves existing nodes, inserts new ones\n const liveParent = anchor.parentNode;\n if (liveParent) {\n // Normal path — anchor is in the DOM, insert before it\n for (const item of newItems) {\n for (const node of item.nodes) {\n liveParent.insertBefore(node, anchor);\n }\n }\n } else {\n // Table fallback — browser relocated the comment anchor.\n // Find the real parent from any node still in the DOM.\n let realParent: Node | null = null;\n for (const item of currentItems) {\n if (item.nodes[0]?.parentNode) { realParent = item.nodes[0].parentNode; break; }\n }\n if (realParent) {\n // Remove old nodes first (oldMap items already removed above)\n // Then append all new items\n for (const item of newItems) {\n for (const node of item.nodes) {\n realParent.appendChild(node);\n }\n }\n }\n }\n\n currentItems = newItems;\n },\n { render: true },\n );\n\n return () => {\n dispose();\n for (const item of currentItems) {\n for (const node of item.nodes) {\n if (node.parentNode) node.parentNode.removeChild(node);\n }\n item.dispose?.();\n }\n currentItems = [];\n };\n}\n\n// --- Built-in control flow components ---\n\n/** Props for the <Show> component */\nexport interface ShowProps<T> {\n when: (() => T | null | undefined | false) | T | null | undefined | false;\n fallback?: () => AkashNode;\n children: (value: T) => AkashNode;\n}\n\n/**\n * <Show> component — conditionally renders children with type narrowing.\n * The children callback receives the non-null/undefined value.\n * `when` can be a reactive getter or a static value.\n */\nexport function Show<T>(props: ShowProps<T>): Node {\n const anchor = createAnchor('show');\n const container = document.createDocumentFragment();\n container.appendChild(anchor);\n\n // Resolve when — if it's a function, always call it (supports both\n // compiler-generated __reactive getters and plain programmatic functions)\n const getWhen = typeof props.when === 'function'\n ? props.when as () => T | null | undefined | false\n : () => props.when as T | null | undefined | false;\n\n // Capture the when value once per evaluation so the condition check\n // and the children callback receive the same value (fixes nested For contexts)\n let whenValue: T | null | undefined | false;\n renderConditional(\n container,\n anchor,\n () => { whenValue = getWhen(); return !!whenValue; },\n () => nodeToDOM(props.children(whenValue as T)),\n props.fallback ? () => nodeToDOM(props.fallback!()) : undefined,\n );\n\n return container;\n}\n\n/** Props for the <For> component */\nexport interface ForProps<T> {\n each: (() => T[]) | T[];\n key: (item: T) => unknown;\n children: (item: T, index: number) => AkashNode;\n}\n\n/**\n * <For> component — renders a list with keyed reconciliation.\n * `each` can be a reactive getter or a static array.\n */\nexport function For<T>(props: ForProps<T>): Node {\n const anchor = createAnchor('for');\n const container = document.createDocumentFragment();\n container.appendChild(anchor);\n\n // Resolve each — if it's a function, always call it (supports both\n // compiler-generated __reactive getters and plain programmatic functions)\n const getEach = typeof props.each === 'function'\n ? props.each as () => T[]\n : () => props.each as T[];\n\n renderList(\n container,\n anchor,\n getEach,\n props.key,\n (item, index) => nodeToDOM(props.children(item, index)),\n );\n\n return container;\n}\n\n// --- Helpers ---\n\n/** Convert an AkashNode to a DOM Node */\nexport function nodeToDOM(node: AkashNode): Node {\n if (node == null || typeof node === 'boolean') {\n return document.createTextNode('');\n }\n if (typeof node === 'string' || typeof node === 'number') {\n return document.createTextNode(String(node));\n }\n if (node instanceof Node) {\n return node;\n }\n if (Array.isArray(node)) {\n const fragment = document.createDocumentFragment();\n for (const child of node) {\n fragment.appendChild(nodeToDOM(child));\n }\n return fragment;\n }\n return document.createTextNode(String(node));\n}\n\n/** Insert a node into a parent before an anchor */\nexport function insert(parent: Node, node: AkashNode, anchor?: Node): void {\n const domNode = nodeToDOM(node);\n if (anchor) {\n parent.insertBefore(domNode, anchor);\n } else {\n parent.appendChild(domNode);\n }\n}\n","/**\n * Component system.\n *\n * Components are functions. defineComponent() wraps a setup function\n * that runs once, establishes signals and effects, and returns a\n * render function that produces DOM nodes.\n */\n\nimport { effect, untrack } from './signals.js';\nimport { pushScope, popScope, getCurrentScope } from './context.js';\nimport { nodeToDOM } from './dom.js';\nimport { akashError } from './errors.js';\nimport type { AkashNode } from './types.js';\n\n// --- Reactive getter marker (used by compiler) ---\n\n/** Mark a function as a compiler-generated reactive getter */\nexport function __getter<T>(fn: () => T): (() => T) & { __reactive: true } {\n (fn as any).__reactive = true;\n return fn as any;\n}\n\n// --- Types ---\n\nexport interface Ref<T = HTMLElement> {\n current: T | undefined;\n}\n\nexport interface ComponentContext<P extends Record<string, unknown> = Record<string, unknown>> {\n props: Readonly<P>;\n children: () => AkashNode;\n}\n\nexport type Component<P extends Record<string, unknown> = Record<string, unknown>> = {\n (props: P & { children?: AkashNode | (() => AkashNode) }): Node;\n _akash: true;\n};\n\n// --- Lifecycle hook storage ---\n\ninterface LifecycleHooks {\n mount: Array<() => void | (() => void)>;\n unmount: Array<() => void>;\n error: Array<(error: Error) => void>;\n}\n\nlet currentHooks: LifecycleHooks | null = null;\n\n// --- Public lifecycle hooks ---\n\n/**\n * Register a callback to run after the component is mounted to the DOM.\n * If the callback returns a function, it will be called on unmount (cleanup).\n */\nexport function onMount(fn: () => void | (() => void)): void {\n if (!currentHooks) {\n throw akashError('AK0020');\n }\n currentHooks.mount.push(fn);\n}\n\n/**\n * Register a callback to run before the component is unmounted.\n */\nexport function onUnmount(fn: () => void): void {\n if (!currentHooks) {\n throw akashError('AK0021');\n }\n currentHooks.unmount.push(fn);\n}\n\n/**\n * Register an error handler for this component and its descendants.\n */\nexport function onError(fn: (error: Error) => void): void {\n if (!currentHooks) {\n throw akashError('AK0022');\n }\n currentHooks.error.push(fn);\n}\n\n/**\n * Create a ref for accessing a DOM element.\n */\nexport function ref<T = HTMLElement>(): Ref<T> {\n return { current: undefined };\n}\n\n// --- defineComponent ---\n\n/**\n * Define a component. The setup function runs once per instance.\n * It receives a context with typed props and must return a render function.\n *\n * ```ts\n * const Counter = defineComponent<{ initial: number }>((ctx) => {\n * const count = signal(ctx.props.initial);\n * return () => <div>{count()}</div>;\n * });\n * ```\n */\nexport function defineComponent<P extends Record<string, unknown> = Record<string, unknown>>(\n setup: (ctx: ComponentContext<P>) => () => AkashNode,\n): Component<P> {\n const component = (rawProps: P & { children?: AkashNode | (() => AkashNode) }): Node => {\n // Separate children from props, unwrap getter functions for reactivity\n const { children: childrenProp, ...restProps } = rawProps ?? {};\n // Props are passed through as-is — no auto-unwrapping.\n // Components use readProp() to unwrap reactive getters/signals inside\n // effects, which preserves dependency tracking. Auto-unwrapping in the\n // Proxy broke reactivity because it called the getter once and returned\n // the plain value, so effects never subscribed to the underlying signal.\n const props = restProps as unknown as P;\n\n const childrenFn: () => AkashNode =\n typeof childrenProp === 'function'\n ? childrenProp\n : () => childrenProp ?? null;\n\n const ctx: ComponentContext<P> = {\n props,\n children: childrenFn,\n };\n\n // Set up lifecycle hooks collector\n const hooks: LifecycleHooks = { mount: [], unmount: [], error: [] };\n const prevHooks = currentHooks;\n currentHooks = hooks;\n\n // Push context scope for provide/inject\n const parentScope = getCurrentScope();\n const scope = pushScope(parentScope);\n\n let renderFn: () => AkashNode;\n let domNode: Node;\n\n try {\n // Untrack component creation so that signals created inside the component\n // don't register with a parent render effect. This prevents parent effects\n // from re-running (and re-creating the entire component) when internal\n // component state changes. The component's own effects still track their\n // dependencies because effect() sets its own subscriber during execution.\n domNode = untrack(() => {\n renderFn = setup(ctx);\n if (typeof renderFn !== 'function') { throw akashError('AK0040'); }\n const rendered = renderFn();\n return nodeToDOM(rendered);\n });\n } catch (err) {\n popScope(scope);\n currentHooks = prevHooks;\n if (hooks.error.length > 0) {\n for (const handler of hooks.error) {\n handler(err instanceof Error ? err : new Error(String(err)));\n }\n return document.createComment('error');\n }\n throw err;\n }\n\n // Restore parent state\n popScope(scope);\n currentHooks = prevHooks;\n\n // Run mount callbacks (microtask to ensure DOM is attached)\n if (hooks.mount.length > 0) {\n queueMicrotask(() => {\n for (const mountFn of hooks.mount) {\n try {\n const cleanup = mountFn();\n if (typeof cleanup === 'function') {\n hooks.unmount.push(cleanup);\n }\n } catch (err) {\n for (const handler of hooks.error) {\n handler(err instanceof Error ? err : new Error(String(err)));\n }\n }\n }\n });\n }\n\n return domNode;\n };\n\n component._akash = true as const;\n return component as Component<P>;\n}\n"]}
@@ -1,2 +0,0 @@
1
- 'use strict';var chunkV7VG23IO_cjs=require('./chunk-V7VG23IO.cjs');var k=[];function v(s){k.push(...s.plugins);}var g=new Map;function b(s,a){return ()=>{if(g.has(s))return g.get(s);let r=a.state(),o=A(s,r,a);return g.set(s,o),o}}function A(s,a,r){let o={},c=Object.keys(a);for(let t of c)o[t]=chunkV7VG23IO_cjs.l(a[t]);let n={$id:s};for(let t of c)n[t]=o[t];if(r.getters)for(let[t,e]of Object.entries(r.getters))n[t]=chunkV7VG23IO_cjs.m(()=>e.call(n,o));let l=[...k,...r.plugins??[]];if(r.actions)for(let[t,e]of Object.entries(r.actions))n[t]=(...i)=>{for(let f of l)f.onAction?.(n,t,i);return e.apply(n,i)};n.$reset=()=>{let t=r.state();for(let e of c)o[e].set(t[e]);},n.$patch=t=>{if(typeof t=="function")t(o);else for(let[e,i]of Object.entries(t))e in o&&o[e].set(i);},n.$snapshot=()=>{let t={};for(let e of c)t[e]=o[e]();return t};let u=new Set,S=null;n.$subscribe=t=>{if(u.add(t),!S){let e=true;S=chunkV7VG23IO_cjs.n(()=>{for(let f of c)o[f]();if(e){e=false;return}let i=n.$snapshot();for(let f of u)f(i);});}return ()=>{u.delete(t),u.size===0&&S&&(S(),S=null);}};for(let t of l)t.init?.(n);return n}function x(){g.clear();}function R(){return Object.fromEntries(g)}exports.a=v;exports.b=b;exports.c=x;exports.d=R;//# sourceMappingURL=chunk-P5GADKQS.cjs.map
2
- //# sourceMappingURL=chunk-P5GADKQS.cjs.map
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/store.ts"],"names":["globalPlugins","configureStores","options","storeInstances","defineStore","id","definition","initialState","store","createStoreInstance","stateSignals","stateKeys","key","signal","getterFn","computed","plugins","actionFn","args","plugin","fresh","partialOrFn","value","snapshot","subscribers","subscribeEffect","callback","isInitial","effect","cb","clearStores","__getStoreInstances"],"mappings":"mEAoFA,IAAMA,EAA+B,EAAC,CAG/B,SAASC,CAAAA,CAAgBC,EAA2C,CACzEF,CAAAA,CAAc,IAAA,CAAK,GAAGE,EAAQ,OAAO,EACvC,CAIA,IAAMC,CAAAA,CAAiB,IAAI,GAAA,CAQpB,SAASC,CAAAA,CAKdC,CAAAA,CACAC,EACsB,CACtB,OAAO,IAAM,CAEX,GAAIH,CAAAA,CAAe,GAAA,CAAIE,CAAE,CAAA,CACvB,OAAOF,CAAAA,CAAe,GAAA,CAAIE,CAAE,CAAA,CAG9B,IAAME,EAAeD,CAAAA,CAAW,KAAA,EAAM,CAChCE,CAAAA,CAAQC,EAAoBJ,CAAAA,CAAIE,CAAAA,CAAcD,CAAU,CAAA,CAC9D,OAAAH,CAAAA,CAAe,GAAA,CAAIE,CAAAA,CAAIG,CAAK,EACrBA,CACT,CACF,CAEA,SAASC,CAAAA,CAKPJ,EACAE,CAAAA,CACAD,CAAAA,CACgB,CAEhB,IAAMI,EAAgD,EAAC,CACjDC,CAAAA,CAAY,MAAA,CAAO,KAAKJ,CAAY,CAAA,CAE1C,IAAA,IAAWK,CAAAA,IAAOD,EAChBD,CAAAA,CAAaE,CAAG,EAAIC,mBAAAA,CAAON,CAAAA,CAAaK,CAAG,CAAC,CAAA,CAI9C,IAAMJ,CAAAA,CAAa,CAAE,GAAA,CAAKH,CAAG,CAAA,CAG7B,IAAA,IAAWO,KAAOD,CAAAA,CAChBH,CAAAA,CAAMI,CAAG,CAAA,CAAIF,EAAaE,CAAG,CAAA,CAI/B,GAAIN,CAAAA,CAAW,OAAA,CACb,OAAW,CAACM,CAAAA,CAAKE,CAAQ,CAAA,GAAK,OAAO,OAAA,CAAQR,CAAAA,CAAW,OAAO,CAAA,CAC7DE,EAAMI,CAAG,CAAA,CAAIG,mBAAAA,CAAS,IACnBD,EAAsB,IAAA,CAAKN,CAAAA,CAAOE,CAAY,CACjD,CAAA,CAKJ,IAAMM,CAAAA,CAAU,CAAC,GAAGhB,CAAAA,CAAe,GAAIM,CAAAA,CAAW,OAAA,EAAW,EAAG,EAGhE,GAAIA,CAAAA,CAAW,OAAA,CACb,IAAA,GAAW,CAACM,CAAAA,CAAKK,CAAQ,IAAK,MAAA,CAAO,OAAA,CAAQX,EAAW,OAAO,CAAA,CAC7DE,CAAAA,CAAMI,CAAG,EAAI,CAAA,GAAIM,CAAAA,GAAoB,CACnC,IAAA,IAAWC,KAAUH,CAAAA,CAASG,CAAAA,CAAO,QAAA,GAAWX,CAAAA,CAAOI,EAAKM,CAAI,CAAA,CAChE,OAAQD,CAAAA,CAAsB,KAAA,CAAMT,EAAOU,CAAI,CACjD,CAAA,CAKJV,CAAAA,CAAM,OAAS,IAAM,CACnB,IAAMY,CAAAA,CAAQd,EAAW,KAAA,EAAM,CAC/B,IAAA,IAAWM,CAAAA,IAAOD,EAChBD,CAAAA,CAAaE,CAAG,EAAE,GAAA,CAAIQ,CAAAA,CAAMR,CAAc,CAAC,EAE/C,CAAA,CAGAJ,CAAAA,CAAM,OAAUa,CAAAA,EAAqD,CACnE,GAAI,OAAOA,GAAgB,UAAA,CACzBA,CAAAA,CAAYX,CAAY,CAAA,CAAA,YAEb,CAACE,CAAAA,CAAKU,CAAK,CAAA,GAAK,MAAA,CAAO,QAAQD,CAAW,CAAA,CAC/CT,CAAAA,IAAOF,CAAAA,EACTA,EAAaE,CAAG,CAAA,CAAE,GAAA,CAAIU,CAAK,EAInC,CAAA,CAGAd,CAAAA,CAAM,SAAA,CAAY,IAAS,CACzB,IAAMe,CAAAA,CAAoC,EAAC,CAC3C,IAAA,IAAWX,KAAOD,CAAAA,CAChBY,CAAAA,CAASX,CAAG,CAAA,CAAIF,EAAaE,CAAG,CAAA,EAAE,CAEpC,OAAOW,CACT,CAAA,CAGA,IAAMC,CAAAA,CAAc,IAAI,IACpBC,CAAAA,CAAuC,IAAA,CAC3CjB,EAAM,UAAA,CAAckB,CAAAA,EAA+C,CAGjE,GAFAF,CAAAA,CAAY,GAAA,CAAIE,CAAQ,EAEpB,CAACD,CAAAA,CAAiB,CACpB,IAAIE,EAAY,IAAA,CAChBF,CAAAA,CAAkBG,mBAAAA,CAAO,IAAM,CAE7B,IAAA,IAAWhB,CAAAA,IAAOD,EAChBD,CAAAA,CAAaE,CAAG,GAAE,CAGpB,GAAIe,CAAAA,CAAW,CACbA,EAAY,KAAA,CACZ,MACF,CACA,IAAMJ,EAAWf,CAAAA,CAAM,SAAA,EAAU,CACjC,IAAA,IAAWqB,KAAML,CAAAA,CACfK,CAAAA,CAAGN,CAAQ,EAEf,CAAC,EACH,CACA,OAAO,IAAM,CACXC,EAAY,MAAA,CAAOE,CAAQ,CAAA,CAEvBF,CAAAA,CAAY,OAAS,CAAA,EAAKC,CAAAA,GAC5BA,CAAAA,EAAgB,CAChBA,EAAkB,IAAA,EAEtB,CACF,EAGA,IAAA,IAAWN,CAAAA,IAAUH,EAASG,CAAAA,CAAO,IAAA,GAAOX,CAAK,CAAA,CAEjD,OAAOA,CACT,CAKO,SAASsB,CAAAA,EAAoB,CAClC3B,CAAAA,CAAe,KAAA,GACjB,CAGO,SAAS4B,CAAAA,EAA4D,CAC1E,OAAO,MAAA,CAAO,WAAA,CAAY5B,CAAc,CAC1C","file":"chunk-P5GADKQS.cjs","sourcesContent":["/**\n * Global state management via defineStore().\n *\n * Stores are singleton signal containers that persist across\n * components. They provide shared state, computed getters,\n * and actions — no external library needed.\n *\n * ```ts\n * const useCounterStore = defineStore('counter', {\n * state: () => ({ count: 0, name: 'Counter' }),\n * getters: {\n * doubled: (state) => state.count() * 2,\n * },\n * actions: {\n * increment() { this.count.update(c => c + 1); },\n * reset() { this.count.set(0); },\n * },\n * });\n *\n * // In any component:\n * const store = useCounterStore();\n * store.count(); // 0\n * store.doubled(); // 0\n * store.increment();\n * store.count(); // 1\n * ```\n */\n\nimport { signal, computed, effect } from './signals.js';\nimport type { Signal, ReadonlySignal } from './signals.js';\n\n// --- Types ---\n\ntype StateFactory<S> = () => S;\n\ntype Getters<S, G> = {\n [K in keyof G]: (state: SignalifiedState<S>) => G[K];\n};\n\ntype Actions<A> = {\n [K in keyof A]: A[K] extends (...args: infer P) => infer R\n ? (...args: P) => R\n : never;\n};\n\n/** Maps plain state values to signals */\ntype SignalifiedState<S> = {\n [K in keyof S]: Signal<S[K]>;\n};\n\n/** The store instance returned to consumers */\nexport type Store<S, G, A> = SignalifiedState<S> & {\n [K in keyof G]: ReadonlySignal<G[K]>;\n} & {\n [K in keyof A]: A[K] extends (...args: infer P) => infer R\n ? (...args: P) => R\n : never;\n} & {\n /** Reset all state to initial values */\n $reset(): void;\n /** Merge partial state into the store, or apply changes via callback */\n $patch(partialOrFn: Partial<S> | ((state: SignalifiedState<S>) => void)): void;\n /** Subscribe to all state changes */\n $subscribe(callback: (state: S) => void): () => void;\n /** Get a plain snapshot of current state */\n $snapshot(): S;\n /** Store ID */\n $id: string;\n};\n\nexport interface StoreDefinition<S, G, A> {\n state: StateFactory<S>;\n getters?: Getters<S, G>;\n actions?: A;\n plugins?: StorePlugin[];\n}\n\n// --- Plugin system ---\n\nexport interface StorePlugin {\n init?(store: Store<any, any, any>): void;\n onAction?(store: Store<any, any, any>, actionName: string, args: unknown[]): void;\n}\n\nconst globalPlugins: StorePlugin[] = [];\n\n/** Register global plugins that apply to all stores */\nexport function configureStores(options: { plugins: StorePlugin[] }): void {\n globalPlugins.push(...options.plugins);\n}\n\n// --- Store registry (singleton) ---\n\nconst storeInstances = new Map<string, Store<any, any, any>>();\n\n// --- defineStore ---\n\n/**\n * Define a global store. Returns a composable function that\n * always returns the same store instance (singleton).\n */\nexport function defineStore<\n S extends Record<string, unknown>,\n G extends Record<string, unknown> = {},\n A extends Record<string, (...args: any[]) => any> = {},\n>(\n id: string,\n definition: StoreDefinition<S, G, A>,\n): () => Store<S, G, A> {\n return () => {\n // Return existing instance if already created\n if (storeInstances.has(id)) {\n return storeInstances.get(id) as Store<S, G, A>;\n }\n\n const initialState = definition.state();\n const store = createStoreInstance(id, initialState, definition);\n storeInstances.set(id, store);\n return store;\n };\n}\n\nfunction createStoreInstance<\n S extends Record<string, unknown>,\n G extends Record<string, unknown>,\n A extends Record<string, (...args: any[]) => any>,\n>(\n id: string,\n initialState: S,\n definition: StoreDefinition<S, G, A>,\n): Store<S, G, A> {\n // Create signals for each state property\n const stateSignals: Record<string, Signal<unknown>> = {};\n const stateKeys = Object.keys(initialState);\n\n for (const key of stateKeys) {\n stateSignals[key] = signal(initialState[key]);\n }\n\n // Build the store object first so getters can reference other getters via `this`\n const store: any = { $id: id };\n\n // Add state signals\n for (const key of stateKeys) {\n store[key] = stateSignals[key];\n }\n\n // Create computed getters — bound to store so `this.otherGetter()` works\n if (definition.getters) {\n for (const [key, getterFn] of Object.entries(definition.getters)) {\n store[key] = computed(() =>\n (getterFn as Function).call(store, stateSignals),\n );\n }\n }\n\n // Collect all plugins (global + per-store)\n const plugins = [...globalPlugins, ...(definition.plugins ?? [])];\n\n // Bind actions with `this` pointing to the full store (state + getters + actions)\n if (definition.actions) {\n for (const [key, actionFn] of Object.entries(definition.actions)) {\n store[key] = (...args: unknown[]) => {\n for (const plugin of plugins) plugin.onAction?.(store, key, args);\n return (actionFn as Function).apply(store, args);\n };\n }\n }\n\n // $reset\n store.$reset = () => {\n const fresh = definition.state();\n for (const key of stateKeys) {\n stateSignals[key].set(fresh[key as keyof S]);\n }\n };\n\n // $patch — merge partial state or apply via callback\n store.$patch = (partialOrFn: Partial<S> | ((state: any) => void)) => {\n if (typeof partialOrFn === 'function') {\n partialOrFn(stateSignals);\n } else {\n for (const [key, value] of Object.entries(partialOrFn)) {\n if (key in stateSignals) {\n stateSignals[key].set(value);\n }\n }\n }\n };\n\n // $snapshot\n store.$snapshot = (): S => {\n const snapshot: Record<string, unknown> = {};\n for (const key of stateKeys) {\n snapshot[key] = stateSignals[key]();\n }\n return snapshot as S;\n };\n\n // $subscribe — watch all state signals and notify on change\n const subscribers = new Set<(state: S) => void>();\n let subscribeEffect: (() => void) | null = null;\n store.$subscribe = (callback: (state: S) => void): (() => void) => {\n subscribers.add(callback);\n // Start watching if this is the first subscriber\n if (!subscribeEffect) {\n let isInitial = true;\n subscribeEffect = effect(() => {\n // Read all state signals to track them\n for (const key of stateKeys) {\n stateSignals[key]();\n }\n // Skip initial run — only notify on actual changes\n if (isInitial) {\n isInitial = false;\n return;\n }\n const snapshot = store.$snapshot();\n for (const cb of subscribers) {\n cb(snapshot);\n }\n });\n }\n return () => {\n subscribers.delete(callback);\n // Dispose effect when no subscribers remain\n if (subscribers.size === 0 && subscribeEffect) {\n subscribeEffect();\n subscribeEffect = null;\n }\n };\n };\n\n // Initialize plugins\n for (const plugin of plugins) plugin.init?.(store);\n\n return store as Store<S, G, A>;\n}\n\n/**\n * Clear all store instances (useful for testing).\n */\nexport function clearStores(): void {\n storeInstances.clear();\n}\n\n/** @internal — exposes store registry for devtools */\nexport function __getStoreInstances(): Record<string, Store<any, any, any>> {\n return Object.fromEntries(storeInstances);\n}\n"]}
@@ -1,2 +0,0 @@
1
- import {l,m,n}from'./chunk-EUKRTV4W.js';var k=[];function v(s){k.push(...s.plugins);}var g=new Map;function b(s,a){return ()=>{if(g.has(s))return g.get(s);let r=a.state(),o=A(s,r,a);return g.set(s,o),o}}function A(s,a,r){let o={},c=Object.keys(a);for(let t of c)o[t]=l(a[t]);let n$1={$id:s};for(let t of c)n$1[t]=o[t];if(r.getters)for(let[t,e]of Object.entries(r.getters))n$1[t]=m(()=>e.call(n$1,o));let l$1=[...k,...r.plugins??[]];if(r.actions)for(let[t,e]of Object.entries(r.actions))n$1[t]=(...i)=>{for(let f of l$1)f.onAction?.(n$1,t,i);return e.apply(n$1,i)};n$1.$reset=()=>{let t=r.state();for(let e of c)o[e].set(t[e]);},n$1.$patch=t=>{if(typeof t=="function")t(o);else for(let[e,i]of Object.entries(t))e in o&&o[e].set(i);},n$1.$snapshot=()=>{let t={};for(let e of c)t[e]=o[e]();return t};let u=new Set,S=null;n$1.$subscribe=t=>{if(u.add(t),!S){let e=true;S=n(()=>{for(let f of c)o[f]();if(e){e=false;return}let i=n$1.$snapshot();for(let f of u)f(i);});}return ()=>{u.delete(t),u.size===0&&S&&(S(),S=null);}};for(let t of l$1)t.init?.(n$1);return n$1}function x(){g.clear();}function R(){return Object.fromEntries(g)}export{v as a,b,x as c,R as d};//# sourceMappingURL=chunk-Q5BER4ZB.js.map
2
- //# sourceMappingURL=chunk-Q5BER4ZB.js.map
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/store.ts"],"names":["globalPlugins","configureStores","options","storeInstances","defineStore","id","definition","initialState","store","createStoreInstance","stateSignals","stateKeys","key","signal","getterFn","computed","plugins","actionFn","args","plugin","fresh","partialOrFn","value","snapshot","subscribers","subscribeEffect","callback","isInitial","effect","cb","clearStores","__getStoreInstances"],"mappings":"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","file":"chunk-Q5BER4ZB.js","sourcesContent":["/**\n * Global state management via defineStore().\n *\n * Stores are singleton signal containers that persist across\n * components. They provide shared state, computed getters,\n * and actions — no external library needed.\n *\n * ```ts\n * const useCounterStore = defineStore('counter', {\n * state: () => ({ count: 0, name: 'Counter' }),\n * getters: {\n * doubled: (state) => state.count() * 2,\n * },\n * actions: {\n * increment() { this.count.update(c => c + 1); },\n * reset() { this.count.set(0); },\n * },\n * });\n *\n * // In any component:\n * const store = useCounterStore();\n * store.count(); // 0\n * store.doubled(); // 0\n * store.increment();\n * store.count(); // 1\n * ```\n */\n\nimport { signal, computed, effect } from './signals.js';\nimport type { Signal, ReadonlySignal } from './signals.js';\n\n// --- Types ---\n\ntype StateFactory<S> = () => S;\n\ntype Getters<S, G> = {\n [K in keyof G]: (state: SignalifiedState<S>) => G[K];\n};\n\ntype Actions<A> = {\n [K in keyof A]: A[K] extends (...args: infer P) => infer R\n ? (...args: P) => R\n : never;\n};\n\n/** Maps plain state values to signals */\ntype SignalifiedState<S> = {\n [K in keyof S]: Signal<S[K]>;\n};\n\n/** The store instance returned to consumers */\nexport type Store<S, G, A> = SignalifiedState<S> & {\n [K in keyof G]: ReadonlySignal<G[K]>;\n} & {\n [K in keyof A]: A[K] extends (...args: infer P) => infer R\n ? (...args: P) => R\n : never;\n} & {\n /** Reset all state to initial values */\n $reset(): void;\n /** Merge partial state into the store, or apply changes via callback */\n $patch(partialOrFn: Partial<S> | ((state: SignalifiedState<S>) => void)): void;\n /** Subscribe to all state changes */\n $subscribe(callback: (state: S) => void): () => void;\n /** Get a plain snapshot of current state */\n $snapshot(): S;\n /** Store ID */\n $id: string;\n};\n\nexport interface StoreDefinition<S, G, A> {\n state: StateFactory<S>;\n getters?: Getters<S, G>;\n actions?: A;\n plugins?: StorePlugin[];\n}\n\n// --- Plugin system ---\n\nexport interface StorePlugin {\n init?(store: Store<any, any, any>): void;\n onAction?(store: Store<any, any, any>, actionName: string, args: unknown[]): void;\n}\n\nconst globalPlugins: StorePlugin[] = [];\n\n/** Register global plugins that apply to all stores */\nexport function configureStores(options: { plugins: StorePlugin[] }): void {\n globalPlugins.push(...options.plugins);\n}\n\n// --- Store registry (singleton) ---\n\nconst storeInstances = new Map<string, Store<any, any, any>>();\n\n// --- defineStore ---\n\n/**\n * Define a global store. Returns a composable function that\n * always returns the same store instance (singleton).\n */\nexport function defineStore<\n S extends Record<string, unknown>,\n G extends Record<string, unknown> = {},\n A extends Record<string, (...args: any[]) => any> = {},\n>(\n id: string,\n definition: StoreDefinition<S, G, A>,\n): () => Store<S, G, A> {\n return () => {\n // Return existing instance if already created\n if (storeInstances.has(id)) {\n return storeInstances.get(id) as Store<S, G, A>;\n }\n\n const initialState = definition.state();\n const store = createStoreInstance(id, initialState, definition);\n storeInstances.set(id, store);\n return store;\n };\n}\n\nfunction createStoreInstance<\n S extends Record<string, unknown>,\n G extends Record<string, unknown>,\n A extends Record<string, (...args: any[]) => any>,\n>(\n id: string,\n initialState: S,\n definition: StoreDefinition<S, G, A>,\n): Store<S, G, A> {\n // Create signals for each state property\n const stateSignals: Record<string, Signal<unknown>> = {};\n const stateKeys = Object.keys(initialState);\n\n for (const key of stateKeys) {\n stateSignals[key] = signal(initialState[key]);\n }\n\n // Build the store object first so getters can reference other getters via `this`\n const store: any = { $id: id };\n\n // Add state signals\n for (const key of stateKeys) {\n store[key] = stateSignals[key];\n }\n\n // Create computed getters — bound to store so `this.otherGetter()` works\n if (definition.getters) {\n for (const [key, getterFn] of Object.entries(definition.getters)) {\n store[key] = computed(() =>\n (getterFn as Function).call(store, stateSignals),\n );\n }\n }\n\n // Collect all plugins (global + per-store)\n const plugins = [...globalPlugins, ...(definition.plugins ?? [])];\n\n // Bind actions with `this` pointing to the full store (state + getters + actions)\n if (definition.actions) {\n for (const [key, actionFn] of Object.entries(definition.actions)) {\n store[key] = (...args: unknown[]) => {\n for (const plugin of plugins) plugin.onAction?.(store, key, args);\n return (actionFn as Function).apply(store, args);\n };\n }\n }\n\n // $reset\n store.$reset = () => {\n const fresh = definition.state();\n for (const key of stateKeys) {\n stateSignals[key].set(fresh[key as keyof S]);\n }\n };\n\n // $patch — merge partial state or apply via callback\n store.$patch = (partialOrFn: Partial<S> | ((state: any) => void)) => {\n if (typeof partialOrFn === 'function') {\n partialOrFn(stateSignals);\n } else {\n for (const [key, value] of Object.entries(partialOrFn)) {\n if (key in stateSignals) {\n stateSignals[key].set(value);\n }\n }\n }\n };\n\n // $snapshot\n store.$snapshot = (): S => {\n const snapshot: Record<string, unknown> = {};\n for (const key of stateKeys) {\n snapshot[key] = stateSignals[key]();\n }\n return snapshot as S;\n };\n\n // $subscribe — watch all state signals and notify on change\n const subscribers = new Set<(state: S) => void>();\n let subscribeEffect: (() => void) | null = null;\n store.$subscribe = (callback: (state: S) => void): (() => void) => {\n subscribers.add(callback);\n // Start watching if this is the first subscriber\n if (!subscribeEffect) {\n let isInitial = true;\n subscribeEffect = effect(() => {\n // Read all state signals to track them\n for (const key of stateKeys) {\n stateSignals[key]();\n }\n // Skip initial run — only notify on actual changes\n if (isInitial) {\n isInitial = false;\n return;\n }\n const snapshot = store.$snapshot();\n for (const cb of subscribers) {\n cb(snapshot);\n }\n });\n }\n return () => {\n subscribers.delete(callback);\n // Dispose effect when no subscribers remain\n if (subscribers.size === 0 && subscribeEffect) {\n subscribeEffect();\n subscribeEffect = null;\n }\n };\n };\n\n // Initialize plugins\n for (const plugin of plugins) plugin.init?.(store);\n\n return store as Store<S, G, A>;\n}\n\n/**\n * Clear all store instances (useful for testing).\n */\nexport function clearStores(): void {\n storeInstances.clear();\n}\n\n/** @internal — exposes store registry for devtools */\nexport function __getStoreInstances(): Record<string, Store<any, any, any>> {\n return Object.fromEntries(storeInstances);\n}\n"]}
@@ -1,5 +0,0 @@
1
- 'use strict';var chunkV7VG23IO_cjs=require('./chunk-V7VG23IO.cjs');var w={AK0010:{code:"AK0010",message:"provide() called outside of component setup.",hint:"provide() must be called inside defineComponent()."},AK0012:{code:"AK0012",message:"inject() called outside of component setup.",hint:"inject() must be called inside defineComponent()."},AK0013:{code:"AK0013",message:"No provider found for injected context.",hint:"Make sure an ancestor component calls provide() with this key."},AK0020:{code:"AK0020",message:"onMount() called outside of component setup.",hint:"onMount() must be called inside defineComponent()."},AK0021:{code:"AK0021",message:"onUnmount() called outside of component setup.",hint:"onUnmount() must be called inside defineComponent()."},AK0022:{code:"AK0022",message:"onError() called outside of component setup.",hint:"onError() must be called inside defineComponent()."},AK0030:{code:"AK0030",message:"Circular dependency detected in computed signal.",hint:"A computed signal is reading itself, directly or via other computeds."},AK0031:{code:"AK0031",message:"Signal set() called during computation.",hint:"Do not set signals inside computed() or during effect execution. Use batch() or set signals in event handlers."},AK0040:{code:"AK0040",message:"Component setup must return a render function.",hint:"The function passed to defineComponent() must return () => AkashNode."},AK0041:{code:"AK0041",message:"Required prop is missing.",hint:"Check that the parent component is passing all required props."},AK0050:{code:"AK0050",message:"useRoute() called outside of router context.",hint:"Make sure your component is rendered inside a router provider."},AK0051:{code:"AK0051",message:"No route matched the current URL.",hint:"Add a catch-all route ([...rest]) to handle unmatched paths."},AK0052:{code:"AK0052",message:"Route guard threw an error.",hint:"Check the guard function for unhandled exceptions."},AK0060:{code:"AK0060",message:"Form submitted while invalid.",hint:"The submit handler was not called because validation failed. Check form.errors()."},AK0061:{code:"AK0061",message:"Async validator timed out.",hint:"The async validator did not resolve within the expected time. Check your async validation logic."},AK0070:{code:"AK0070",message:"HTTP request failed.",hint:"Check the server response status and network connectivity."},AK0071:{code:"AK0071",message:"createResource() fetcher threw an error.",hint:"Check the fetcher function passed to createResource()."}},P="https://akashjs.dev/errors";function R(e,n){let t=w[e];if(!t)return `[AkashJS ${e}] Unknown error code.`;let s=`[AkashJS ${e}] ${t.message}`;return n&&(s+=`
2
- ${n}`),s+=`
3
- ${t.hint}`,s+=`
4
- See: ${P}/${e}`,s}function T(e,n){return new Error(R(e,n))}function F(e){return w[e]}function _(){return Object.keys(w)}var D=Symbol("akash.context"),m=null;function S(e=m){let n={values:new Map,parent:e};return m=n,n}function K(e){m=e.parent;}function C(){return m}function v(e,n){let t=m;m=e;try{return n()}finally{m=t;}}function O(e){return {[D]:true,_type:void 0,defaultValue:e,id:Symbol("context")}}function $(e,n){if(!m)throw T("AK0010");m.values.set(e.id,n);}function U(e,n){if(!m)throw T("AK0012");let t=m;for(;t;){if(t.values.has(e.id))return t.values.get(e.id);t=t.parent;}if(n!==void 0)return n;if(e.defaultValue!==void 0)return e.defaultValue;throw T("AK0013")}function W(e,n){let t=document.createElement(e);if(n)for(let[s,a]of Object.entries(n))b(t,s,a);return t}function X(e){return document.createTextNode(e)}function G(e){let n=document.createElement("template");return n.innerHTML=e,n.content.cloneNode(true)}function b(e,n,t){if(n==="class"||n==="className")e.className=t;else if(n==="style"&&typeof t=="object"&&t!==null)Object.assign(e.style,t);else if(n==="innerHTML")e.innerHTML=t;else if(n!=="ref")if(n.startsWith("on")){let s=n.slice(2).toLowerCase();e.addEventListener(s,t);}else n in e?e[n]=t:t===false||t==null?e.removeAttribute(n):e.setAttribute(n,t===true?"":String(t));}function Y(e,n){return chunkV7VG23IO_cjs.n(()=>{e.textContent=String(n());},{render:true})}function z(e,n,t){return chunkV7VG23IO_cjs.n(()=>{b(e,n,t());},{render:true})}function Q(e,n){return chunkV7VG23IO_cjs.n(()=>{e.style.display=n()?"":"none";},{render:true})}function L(e=""){return document.createComment(e)}function j(e,n,t,s,a){let i=null,g=C(),y=chunkV7VG23IO_cjs.n(()=>{let c=t();if(i){for(let f of i.nodes)try{f.parentNode?.removeChild(f);}catch{}i.dispose?.(),i=null;}let d=c?s:a,p=n.parentNode;if(!p&&i?.nodes[0]?.parentNode&&(p=i.nodes[0].parentNode),d&&p){let f=g?v(g,d):d(),o=f instanceof DocumentFragment?Array.from(f.childNodes):[f];for(let r of o)p.insertBefore(r,n);i={nodes:o,dispose:null};}},{render:true});return ()=>{if(y(),i){for(let c of i.nodes)try{c.parentNode?.removeChild(c);}catch{}i.dispose?.();}}}function H(e,n,t,s,a){let i=[],g=C(),y=chunkV7VG23IO_cjs.n(()=>{let c=t(),d=[],p=new Map;for(let o of i)p.set(o.key,o);for(let o=0;o<c.length;o++){let r=c[o],u=s(r,o),l=p.get(u);if(l){p.delete(u);let A=l.nodes[0];if(A&&!A.parentNode){let k=g?v(g,()=>a(r,o)):a(r,o),M=k instanceof DocumentFragment?Array.from(k.childNodes):[k];d.push({key:u,value:r,nodes:M,dispose:null});}else l.value=r,d.push(l);}else {let A=g?v(g,()=>a(r,o)):a(r,o),k=A instanceof DocumentFragment?Array.from(A.childNodes):[A];d.push({key:u,value:r,nodes:k,dispose:null});}}for(let o of p.values()){for(let r of o.nodes)r.parentNode&&r.parentNode.removeChild(r);o.dispose?.();}let f=n.parentNode;if(f)for(let o of d)for(let r of o.nodes)f.insertBefore(r,n);else {let o=null;for(let r of i)if(r.nodes[0]?.parentNode){o=r.nodes[0].parentNode;break}if(o)for(let r of d)for(let u of r.nodes)o.appendChild(u);}i=d;},{render:true});return ()=>{y();for(let c of i){for(let d of c.nodes)d.parentNode&&d.parentNode.removeChild(d);c.dispose?.();}i=[];}}function Z(e){let n=L("show"),t=document.createDocumentFragment();t.appendChild(n);let s=typeof e.when=="function"?e.when:()=>e.when,a;return j(t,n,()=>(a=s(),!!a),()=>x(e.children(a)),e.fallback?()=>x(e.fallback()):void 0),t}function ee(e){let n=L("for"),t=document.createDocumentFragment();t.appendChild(n);let s=typeof e.each=="function"?e.each:()=>e.each;return H(t,n,s,e.key,(a,i)=>x(e.children(a,i))),t}function x(e){if(e==null||typeof e=="boolean")return document.createTextNode("");if(typeof e=="string"||typeof e=="number")return document.createTextNode(String(e));if(e instanceof Node)return e;if(Array.isArray(e)){let n=document.createDocumentFragment();for(let t of e)n.appendChild(x(t));return n}return document.createTextNode(String(e))}function ne(e,n,t){let s=x(n);t?e.insertBefore(s,t):e.appendChild(s);}function de(e){return e.__reactive=true,e}var h=null;function ae(e){if(!h)throw T("AK0020");h.mount.push(e);}function ue(e){if(!h)throw T("AK0021");h.unmount.push(e);}function le(e){if(!h)throw T("AK0022");h.error.push(e);}function fe(){return {current:void 0}}function pe(e){let n=t=>{let{children:s,...a}=t??{},y={props:a,children:typeof s=="function"?s:()=>s??null},c={mount:[],unmount:[],error:[]},d=h;h=c;let p=C(),f=S(p),o,r;try{r=chunkV7VG23IO_cjs.o(()=>{if(o=e(y),typeof o!="function")throw T("AK0040");let u=o();return x(u)});}catch(u){if(K(f),h=d,c.error.length>0){for(let l of c.error)l(u instanceof Error?u:new Error(String(u)));return document.createComment("error")}throw u}return K(f),h=d,c.mount.length>0&&queueMicrotask(()=>{for(let u of c.mount)try{let l=u();typeof l=="function"&&c.unmount.push(l);}catch(l){for(let A of c.error)A(l instanceof Error?l:new Error(String(l)));}}),r};return n._akash=true,n}exports.A=fe;exports.B=pe;exports.a=R;exports.b=T;exports.c=F;exports.d=_;exports.e=C;exports.f=v;exports.g=O;exports.h=$;exports.i=U;exports.j=W;exports.k=X;exports.l=G;exports.m=b;exports.n=Y;exports.o=z;exports.p=Q;exports.q=j;exports.r=H;exports.s=Z;exports.t=ee;exports.u=x;exports.v=ne;exports.w=de;exports.x=ae;exports.y=ue;exports.z=le;//# sourceMappingURL=chunk-QDCIW4YE.cjs.map
5
- //# sourceMappingURL=chunk-QDCIW4YE.cjs.map