lastriko 0.1.0

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 (101) hide show
  1. package/CHANGELOG.md +7 -0
  2. package/README.md +58 -0
  3. package/dist/__tests__/integration/ws-flow.integration.test.d.ts +1 -0
  4. package/dist/__tests__/integration/ws-flow.integration.test.js +249 -0
  5. package/dist/__tests__/integration/ws-flow.integration.test.js.map +1 -0
  6. package/dist/__tests__/integration/ws-flow.test.d.ts +1 -0
  7. package/dist/__tests__/integration/ws-flow.test.js +70 -0
  8. package/dist/__tests__/integration/ws-flow.test.js.map +1 -0
  9. package/dist/client/events.d.ts +6 -0
  10. package/dist/client/events.js +114 -0
  11. package/dist/client/events.js.map +1 -0
  12. package/dist/client/index.d.ts +2 -0
  13. package/dist/client/index.js +5 -0
  14. package/dist/client/index.js.map +1 -0
  15. package/dist/client/swap.d.ts +4 -0
  16. package/dist/client/swap.js +47 -0
  17. package/dist/client/swap.js.map +1 -0
  18. package/dist/client/ws.d.ts +11 -0
  19. package/dist/client/ws.js +111 -0
  20. package/dist/client/ws.js.map +1 -0
  21. package/dist/components/context.d.ts +51 -0
  22. package/dist/components/context.js +595 -0
  23. package/dist/components/context.js.map +1 -0
  24. package/dist/components/context.test.d.ts +1 -0
  25. package/dist/components/context.test.js +69 -0
  26. package/dist/components/context.test.js.map +1 -0
  27. package/dist/components/id.d.ts +3 -0
  28. package/dist/components/id.js +9 -0
  29. package/dist/components/id.js.map +1 -0
  30. package/dist/components/registry.d.ts +2 -0
  31. package/dist/components/registry.js +97 -0
  32. package/dist/components/registry.js.map +1 -0
  33. package/dist/components/types.d.ts +390 -0
  34. package/dist/components/types.js +2 -0
  35. package/dist/components/types.js.map +1 -0
  36. package/dist/engine/executor.d.ts +20 -0
  37. package/dist/engine/executor.js +84 -0
  38. package/dist/engine/executor.js.map +1 -0
  39. package/dist/engine/executor.test.d.ts +1 -0
  40. package/dist/engine/executor.test.js +79 -0
  41. package/dist/engine/executor.test.js.map +1 -0
  42. package/dist/engine/index.d.ts +6 -0
  43. package/dist/engine/index.js +7 -0
  44. package/dist/engine/index.js.map +1 -0
  45. package/dist/engine/lifecycle.d.ts +16 -0
  46. package/dist/engine/lifecycle.js +44 -0
  47. package/dist/engine/lifecycle.js.map +1 -0
  48. package/dist/engine/lifecycle.test.d.ts +1 -0
  49. package/dist/engine/lifecycle.test.js +15 -0
  50. package/dist/engine/lifecycle.test.js.map +1 -0
  51. package/dist/engine/messages.d.ts +95 -0
  52. package/dist/engine/messages.js +21 -0
  53. package/dist/engine/messages.js.map +1 -0
  54. package/dist/engine/renderer.components.test.d.ts +1 -0
  55. package/dist/engine/renderer.components.test.js +302 -0
  56. package/dist/engine/renderer.components.test.js.map +1 -0
  57. package/dist/engine/renderer.d.ts +4 -0
  58. package/dist/engine/renderer.js +408 -0
  59. package/dist/engine/renderer.js.map +1 -0
  60. package/dist/engine/renderer.test.d.ts +1 -0
  61. package/dist/engine/renderer.test.js +80 -0
  62. package/dist/engine/renderer.test.js.map +1 -0
  63. package/dist/engine/server.d.ts +43 -0
  64. package/dist/engine/server.js +402 -0
  65. package/dist/engine/server.js.map +1 -0
  66. package/dist/engine/server.test.d.ts +1 -0
  67. package/dist/engine/server.test.js +149 -0
  68. package/dist/engine/server.test.js.map +1 -0
  69. package/dist/engine/shell.d.ts +7 -0
  70. package/dist/engine/shell.js +25 -0
  71. package/dist/engine/shell.js.map +1 -0
  72. package/dist/engine/theme-path.d.ts +11 -0
  73. package/dist/engine/theme-path.js +32 -0
  74. package/dist/engine/theme-path.js.map +1 -0
  75. package/dist/engine/theme-path.test.d.ts +1 -0
  76. package/dist/engine/theme-path.test.js +52 -0
  77. package/dist/engine/theme-path.test.js.map +1 -0
  78. package/dist/engine/watcher.d.ts +8 -0
  79. package/dist/engine/watcher.js +27 -0
  80. package/dist/engine/watcher.js.map +1 -0
  81. package/dist/engine/websocket.d.ts +24 -0
  82. package/dist/engine/websocket.hub.test.d.ts +1 -0
  83. package/dist/engine/websocket.hub.test.js +75 -0
  84. package/dist/engine/websocket.hub.test.js.map +1 -0
  85. package/dist/engine/websocket.js +119 -0
  86. package/dist/engine/websocket.js.map +1 -0
  87. package/dist/index.d.ts +21 -0
  88. package/dist/index.js +26 -0
  89. package/dist/index.js.map +1 -0
  90. package/dist/plugins/registry.d.ts +11 -0
  91. package/dist/plugins/registry.js +61 -0
  92. package/dist/plugins/registry.js.map +1 -0
  93. package/dist/plugins/registry.test.d.ts +1 -0
  94. package/dist/plugins/registry.test.js +44 -0
  95. package/dist/plugins/registry.test.js.map +1 -0
  96. package/dist/plugins/types.d.ts +30 -0
  97. package/dist/plugins/types.js +2 -0
  98. package/dist/plugins/types.js.map +1 -0
  99. package/dist/style.css +134 -0
  100. package/dist/theme/lastriko.css +134 -0
  101. package/package.json +60 -0
package/CHANGELOG.md ADDED
@@ -0,0 +1,7 @@
1
+ # Changelog
2
+
3
+ All notable changes to the **`lastriko`** npm package are documented here. The monorepo roadmap and design history live in the repository root [MANIFEST.md](https://github.com/aixpose/lastriko/blob/main/MANIFEST.md).
4
+
5
+ ## 0.1.0 — 2026-04-08
6
+
7
+ Initial published MVP (Phase 2 scope): `app()`, WebSocket + HTML fragments, MVP component set, `lastriko/style.css`, Node.js 22+.
package/README.md ADDED
@@ -0,0 +1,58 @@
1
+ # lastriko
2
+
3
+ Server-driven UI for AI demos and rapid prototypes: your TypeScript `app()` builds handles, the engine renders HTML fragments, and the browser swaps `outerHTML` over a WebSocket.
4
+
5
+ ## Install
6
+
7
+ ```bash
8
+ npm install lastriko
9
+ ```
10
+
11
+ Requires **Node.js 22+**. [Bun](https://bun.sh) is also supported.
12
+
13
+ ## Quick start
14
+
15
+ ```typescript
16
+ import { app } from 'lastriko';
17
+
18
+ await app('Hello', (ui) => {
19
+ const name = ui.textInput('Name');
20
+ const out = ui.text('Hi!');
21
+ ui.button('Greet', () => {
22
+ out.update(`Hello, ${name.value || 'stranger'}!`);
23
+ });
24
+ });
25
+ ```
26
+
27
+ Run with `tsx` or `bun`; open the URL printed in the console (default port **3500**).
28
+
29
+ ## Exports
30
+
31
+ - **JS API** — `import { app, defineConfig } from 'lastriko'`
32
+ - **Styles** — `import 'lastriko/style.css'` (or load `/style.css` from the dev server)
33
+
34
+ ## Docs
35
+
36
+ - [Project manifest & roadmap](https://github.com/aixpose/lastriko/blob/main/MANIFEST.md)
37
+ - [API reference](https://github.com/aixpose/lastriko/blob/main/docs/API-REFERENCE.md)
38
+ - [Phase 2 (MVP) scope](https://github.com/aixpose/lastriko/blob/main/docs/phases/PHASE-2.md)
39
+
40
+ ## Examples in this repo
41
+
42
+ - `examples/component-gallery` — all MVP components on one page
43
+ - `examples/experiment-monitor`, `examples/image-viewer` — larger demos
44
+
45
+ ## Publishing (maintainers)
46
+
47
+ From the monorepo root, after tests pass:
48
+
49
+ ```bash
50
+ npm run build -w lastriko
51
+ npm publish -w lastriko --access public
52
+ ```
53
+
54
+ `prepack` runs `build` automatically on `npm publish`.
55
+
56
+ ## License
57
+
58
+ MIT — see [LICENSE](https://github.com/aixpose/lastriko/blob/main/LICENSE) in the repository root.
@@ -0,0 +1,249 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { createWebSocketHub } from '../../engine/websocket';
3
+ function createSocket() {
4
+ const sent = [];
5
+ return {
6
+ socket: {
7
+ send(data) {
8
+ sent.push(data);
9
+ },
10
+ },
11
+ sent,
12
+ };
13
+ }
14
+ function byType(sent, type) {
15
+ return sent
16
+ .map((item) => JSON.parse(item))
17
+ .filter((msg) => msg.type === type);
18
+ }
19
+ describe('websocket integration flow', () => {
20
+ it('ready emits RENDER and button click emits FRAGMENT', () => {
21
+ const { socket, sent } = createSocket();
22
+ const runtime = createWebSocketHub({
23
+ title: 'Phase2',
24
+ callback: (ui) => {
25
+ const text = ui.text('before');
26
+ ui.button('go', () => {
27
+ text.update('after');
28
+ });
29
+ },
30
+ });
31
+ runtime.addConnection(socket);
32
+ runtime.handleRawMessage(socket, JSON.stringify({
33
+ type: 'READY',
34
+ payload: { viewport: { width: 100, height: 100 }, theme: 'dark' },
35
+ }));
36
+ const render = byType(sent, 'RENDER')[0];
37
+ expect(render).toBeDefined();
38
+ expect(render?.payload.html).toContain('before');
39
+ const buttonId = render?.payload.html.match(/data-lk-id="(button-[^"]+)"/)?.[1];
40
+ expect(buttonId).toBeDefined();
41
+ runtime.handleRawMessage(socket, JSON.stringify({
42
+ type: 'EVENT',
43
+ payload: { id: buttonId, event: 'click' },
44
+ }));
45
+ const fragment = byType(sent, 'FRAGMENT').at(-1);
46
+ expect(fragment?.payload.html).toContain('after');
47
+ });
48
+ it('change EVENT updates input value and pushes FRAGMENT', () => {
49
+ const { socket, sent } = createSocket();
50
+ const runtime = createWebSocketHub({
51
+ title: 'Inputs',
52
+ callback: (ui) => {
53
+ ui.textInput('Prompt', { default: 'a' });
54
+ },
55
+ });
56
+ runtime.addConnection(socket);
57
+ runtime.handleRawMessage(socket, JSON.stringify({
58
+ type: 'READY',
59
+ payload: { viewport: { width: 100, height: 100 }, theme: null },
60
+ }));
61
+ const render = byType(sent, 'RENDER')[0];
62
+ const inputId = render.payload.html.match(/data-lk-id="(textInput-[^"]+)"/)?.[1];
63
+ expect(inputId).toBeDefined();
64
+ runtime.handleRawMessage(socket, JSON.stringify({
65
+ type: 'EVENT',
66
+ payload: { id: inputId, event: 'change', value: 'updated' },
67
+ }));
68
+ const fragment = byType(sent, 'FRAGMENT').at(-1);
69
+ expect(fragment?.payload.id).toBe(inputId);
70
+ expect(fragment?.payload.html).toContain('updated');
71
+ });
72
+ it('streamText append emits STREAM_CHUNK messages in order', () => {
73
+ const { socket, sent } = createSocket();
74
+ const runtime = createWebSocketHub({
75
+ title: 'Streaming',
76
+ callback: (ui) => {
77
+ const stream = ui.streamText({ format: 'plain' });
78
+ ui.button('go', () => {
79
+ stream.append('a');
80
+ stream.append('b');
81
+ stream.done();
82
+ });
83
+ },
84
+ });
85
+ runtime.addConnection(socket);
86
+ runtime.handleRawMessage(socket, JSON.stringify({
87
+ type: 'READY',
88
+ payload: { viewport: { width: 100, height: 100 }, theme: null },
89
+ }));
90
+ const render = byType(sent, 'RENDER')[0];
91
+ const buttonId = render.payload.html.match(/data-lk-id="(button-[^"]+)"/)?.[1];
92
+ runtime.handleRawMessage(socket, JSON.stringify({
93
+ type: 'EVENT',
94
+ payload: { id: buttonId, event: 'click' },
95
+ }));
96
+ const chunks = byType(sent, 'STREAM_CHUNK');
97
+ expect(chunks).toHaveLength(3);
98
+ expect(chunks[0]?.payload.chunk).toBe('a');
99
+ expect(chunks[1]?.payload.chunk).toBe('b');
100
+ expect(chunks[2]?.payload.done).toBe(true);
101
+ });
102
+ it('table row click EVENT dispatches onRowClick handler', () => {
103
+ const { socket, sent } = createSocket();
104
+ const clicks = [];
105
+ const runtime = createWebSocketHub({
106
+ title: 'Table',
107
+ callback: (ui) => {
108
+ const table = ui.table([{ name: 'row-1' }], { columns: ['name'] });
109
+ table.onRowClick((row) => {
110
+ clicks.push(String(row.name));
111
+ });
112
+ },
113
+ });
114
+ runtime.addConnection(socket);
115
+ runtime.handleRawMessage(socket, JSON.stringify({
116
+ type: 'READY',
117
+ payload: { viewport: { width: 120, height: 80 }, theme: null },
118
+ }));
119
+ const render = byType(sent, 'RENDER')[0];
120
+ const tableId = render.payload.html.match(/data-lk-id="(table-[^"]+)"/)?.[1];
121
+ const rowId = render.payload.html.match(/data-lk-table-row-id="([^"]+)"/)?.[1];
122
+ expect(tableId).toBeDefined();
123
+ expect(rowId).toBeDefined();
124
+ runtime.handleRawMessage(socket, JSON.stringify({
125
+ type: 'EVENT',
126
+ payload: { id: tableId, event: 'click', value: rowId },
127
+ }));
128
+ expect(clicks).toEqual(['row-1']);
129
+ });
130
+ it('tracks READY viewport/theme and handles THEME_CHANGE', () => {
131
+ const { socket, sent } = createSocket();
132
+ const runtime = createWebSocketHub({
133
+ title: 'Scope',
134
+ callback: (ui) => {
135
+ ui.text(`viewport-${ui.viewport.width}x${ui.viewport.height}-${ui.theme}`);
136
+ },
137
+ });
138
+ const scope = runtime.addConnection(socket);
139
+ runtime.handleRawMessage(socket, JSON.stringify({
140
+ type: 'READY',
141
+ payload: { viewport: { width: 10, height: 20 }, theme: 'dark' },
142
+ }));
143
+ expect(scope.viewport).toEqual({ width: 10, height: 20 });
144
+ expect(scope.theme).toBe('dark');
145
+ runtime.handleRawMessage(socket, JSON.stringify({
146
+ type: 'THEME_CHANGE',
147
+ payload: { mode: 'light' },
148
+ }));
149
+ const themeMsg = byType(sent, 'THEME').at(-1);
150
+ expect(themeMsg?.payload.mode).toBe('light');
151
+ });
152
+ it('numberInput and toggle change EVENT update values and emit FRAGMENT', () => {
153
+ const { socket, sent } = createSocket();
154
+ const runtime = createWebSocketHub({
155
+ title: 'More inputs',
156
+ callback: (ui) => {
157
+ ui.numberInput('N', { default: 1 });
158
+ ui.toggle('T', { default: false });
159
+ },
160
+ });
161
+ runtime.addConnection(socket);
162
+ runtime.handleRawMessage(socket, JSON.stringify({
163
+ type: 'READY',
164
+ payload: { viewport: { width: 100, height: 100 }, theme: null },
165
+ }));
166
+ const render = byType(sent, 'RENDER')[0];
167
+ const numId = render.payload.html.match(/data-lk-id="(numberInput-[^"]+)"/)?.[1];
168
+ const toggleId = render.payload.html.match(/data-lk-id="(toggle-[^"]+)"/)?.[1];
169
+ expect(numId).toBeDefined();
170
+ expect(toggleId).toBeDefined();
171
+ runtime.handleRawMessage(socket, JSON.stringify({
172
+ type: 'EVENT',
173
+ payload: { id: numId, event: 'change', value: 42 },
174
+ }));
175
+ const fragNum = byType(sent, 'FRAGMENT').at(-1);
176
+ expect(fragNum?.payload.id).toBe(numId);
177
+ expect(fragNum?.payload.html).toContain('value="42"');
178
+ runtime.handleRawMessage(socket, JSON.stringify({
179
+ type: 'EVENT',
180
+ payload: { id: toggleId, event: 'change', value: true },
181
+ }));
182
+ const fragToggle = byType(sent, 'FRAGMENT').at(-1);
183
+ expect(fragToggle?.payload.id).toBe(toggleId);
184
+ expect(fragToggle?.payload.html).toContain('checked');
185
+ });
186
+ it('cross-region: sidebar button updates header metric via shared scope', () => {
187
+ const { socket, sent } = createSocket();
188
+ const runtime = createWebSocketHub({
189
+ title: 'Shell',
190
+ callback: (ui) => {
191
+ let metric;
192
+ ui.shell({
193
+ header: (h) => {
194
+ metric = h.metric('Count', '0');
195
+ },
196
+ main: () => { },
197
+ sidebar: (s) => {
198
+ s.button('inc', () => {
199
+ metric?.update('1');
200
+ });
201
+ },
202
+ });
203
+ },
204
+ });
205
+ runtime.addConnection(socket);
206
+ runtime.handleRawMessage(socket, JSON.stringify({
207
+ type: 'READY',
208
+ payload: { viewport: { width: 200, height: 200 }, theme: null },
209
+ }));
210
+ const render = byType(sent, 'RENDER')[0];
211
+ const buttonId = render.payload.html.match(/data-lk-id="(button-[^"]+)"/)?.[1];
212
+ expect(buttonId).toBeDefined();
213
+ expect(render.payload.html).toContain('lk-metric-value');
214
+ runtime.handleRawMessage(socket, JSON.stringify({
215
+ type: 'EVENT',
216
+ payload: { id: buttonId, event: 'click' },
217
+ }));
218
+ const lastMetricFrag = [...byType(sent, 'FRAGMENT')]
219
+ .filter((f) => f.payload.html.includes('lk-metric'))
220
+ .at(-1);
221
+ expect(lastMetricFrag?.payload.html).toContain('1');
222
+ });
223
+ it('keeps scopes isolated across connections', () => {
224
+ const a = createSocket();
225
+ const b = createSocket();
226
+ const runtime = createWebSocketHub({
227
+ title: 'Isolation',
228
+ callback: (ui) => {
229
+ ui.text(`scope-${ui.scope.id}`);
230
+ },
231
+ });
232
+ const scopeA = runtime.addConnection(a.socket);
233
+ const scopeB = runtime.addConnection(b.socket);
234
+ runtime.handleRawMessage(a.socket, JSON.stringify({
235
+ type: 'READY',
236
+ payload: { viewport: { width: 10, height: 10 }, theme: null },
237
+ }));
238
+ runtime.handleRawMessage(b.socket, JSON.stringify({
239
+ type: 'READY',
240
+ payload: { viewport: { width: 20, height: 20 }, theme: null },
241
+ }));
242
+ const aRender = byType(a.sent, 'RENDER')[0];
243
+ const bRender = byType(b.sent, 'RENDER')[0];
244
+ expect(aRender.payload.html).toContain(`scope-${scopeA.id}`);
245
+ expect(bRender.payload.html).toContain(`scope-${scopeB.id}`);
246
+ expect(scopeA.id).not.toBe(scopeB.id);
247
+ });
248
+ });
249
+ //# sourceMappingURL=ws-flow.integration.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws-flow.integration.test.js","sourceRoot":"","sources":["../../../src/__tests__/integration/ws-flow.integration.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,70 @@
1
+ import { describe, expect, it } from 'vitest';
2
+ import { createWebSocketHub } from '../../engine/websocket';
3
+ function createSocket() {
4
+ const sent = [];
5
+ return {
6
+ socket: {
7
+ send(data) {
8
+ sent.push(data);
9
+ },
10
+ },
11
+ sent,
12
+ };
13
+ }
14
+ describe('websocket integration flow', () => {
15
+ it('rEADY emits RENDER and click EVENT emits FRAGMENT', async () => {
16
+ const { socket, sent } = createSocket();
17
+ const runtime = createWebSocketHub({
18
+ title: 'Phase1',
19
+ callback: (ui) => {
20
+ const text = ui.text('before');
21
+ ui.button('go', () => {
22
+ text.update('after');
23
+ });
24
+ },
25
+ });
26
+ runtime.addConnection(socket);
27
+ runtime.handleRawMessage(socket, JSON.stringify({
28
+ type: 'READY',
29
+ payload: { viewport: { width: 100, height: 100 }, theme: null },
30
+ }));
31
+ const renderRaw = sent.find((item) => JSON.parse(item).type === 'RENDER');
32
+ expect(renderRaw).toBeDefined();
33
+ const render = JSON.parse(renderRaw ?? '{}');
34
+ expect(render.payload.html).toContain('before');
35
+ const buttonId = render.payload.html.match(/data-lk-id="(button-[^"]+)"/)?.[1];
36
+ runtime.handleRawMessage(socket, JSON.stringify({
37
+ type: 'EVENT',
38
+ payload: { id: buttonId, event: 'click' },
39
+ }));
40
+ const fragmentRaw = sent.find((item) => JSON.parse(item).type === 'FRAGMENT');
41
+ expect(fragmentRaw).toBeDefined();
42
+ const fragment = JSON.parse(fragmentRaw ?? '{}');
43
+ expect(fragment.payload.html).toContain('after');
44
+ });
45
+ it('two connections render independently', () => {
46
+ const a = createSocket();
47
+ const b = createSocket();
48
+ const runtime = createWebSocketHub({
49
+ title: 'Scope',
50
+ callback: (ui) => {
51
+ ui.text(`scope-${ui.scope.id}`);
52
+ },
53
+ });
54
+ const scopeA = runtime.addConnection(a.socket);
55
+ const scopeB = runtime.addConnection(b.socket);
56
+ runtime.handleRawMessage(a.socket, JSON.stringify({
57
+ type: 'READY',
58
+ payload: { viewport: { width: 10, height: 10 }, theme: null },
59
+ }));
60
+ runtime.handleRawMessage(b.socket, JSON.stringify({
61
+ type: 'READY',
62
+ payload: { viewport: { width: 10, height: 10 }, theme: null },
63
+ }));
64
+ const aRender = JSON.parse(a.sent.find((item) => JSON.parse(item).type === 'RENDER') ?? '{}');
65
+ const bRender = JSON.parse(b.sent.find((item) => JSON.parse(item).type === 'RENDER') ?? '{}');
66
+ expect(aRender.payload.html).toContain(`scope-${scopeA.id}`);
67
+ expect(bRender.payload.html).toContain(`scope-${scopeB.id}`);
68
+ });
69
+ });
70
+ //# sourceMappingURL=ws-flow.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws-flow.test.js","sourceRoot":"","sources":["../../../src/__tests__/integration/ws-flow.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,6 @@
1
+ import type { ClientToServerMessage } from '../engine/messages';
2
+ export interface EventChannel {
3
+ send: (message: ClientToServerMessage) => void;
4
+ }
5
+ export declare function bindEventDelegation(root: Document, channel: EventChannel): void;
6
+ export declare function bindThemeToggle(root: Document, channel: EventChannel): void;
@@ -0,0 +1,114 @@
1
+ function coerceValue(target) {
2
+ if (!(target instanceof HTMLElement))
3
+ return undefined;
4
+ if (target instanceof HTMLInputElement) {
5
+ if (target.type === 'checkbox')
6
+ return target.checked;
7
+ if (target.type === 'number' || target.type === 'range')
8
+ return Number(target.value);
9
+ return target.value;
10
+ }
11
+ if (target instanceof HTMLSelectElement || target instanceof HTMLTextAreaElement) {
12
+ return target.value;
13
+ }
14
+ return undefined;
15
+ }
16
+ function resolveComponentId(target) {
17
+ if (!(target instanceof Element))
18
+ return null;
19
+ const holder = target.closest('[data-lk-id]');
20
+ return holder?.dataset.lkId ?? null;
21
+ }
22
+ export function bindEventDelegation(root, channel) {
23
+ root.addEventListener('click', (event) => {
24
+ if (!(event.target instanceof Element))
25
+ return;
26
+ const clickable = event.target.closest('[data-lk-event="click"]');
27
+ if (!clickable)
28
+ return;
29
+ const id = resolveComponentId(clickable);
30
+ if (!id)
31
+ return;
32
+ channel.send({ type: 'EVENT', payload: { id, event: 'click' } });
33
+ });
34
+ root.addEventListener('click', (event) => {
35
+ if (!(event.target instanceof Element))
36
+ return;
37
+ const tableRow = event.target.closest('[data-lk-table-row-id]');
38
+ if (!tableRow)
39
+ return;
40
+ const table = tableRow.closest('[data-lk-id][data-lk-kind="table"]');
41
+ const id = table?.dataset.lkId;
42
+ const rowId = tableRow.dataset.lkTableRowId;
43
+ if (!id || !rowId)
44
+ return;
45
+ channel.send({ type: 'EVENT', payload: { id, event: 'click', value: rowId } });
46
+ });
47
+ root.addEventListener('change', (event) => {
48
+ if (!(event.target instanceof Element))
49
+ return;
50
+ const changeable = event.target.closest('[data-lk-event="change"]');
51
+ if (!changeable)
52
+ return;
53
+ const id = resolveComponentId(changeable);
54
+ if (!id)
55
+ return;
56
+ channel.send({
57
+ type: 'EVENT',
58
+ payload: { id, event: 'change', value: coerceValue(event.target) },
59
+ });
60
+ });
61
+ root.addEventListener('change', async (event) => {
62
+ if (!(event.target instanceof HTMLInputElement))
63
+ return;
64
+ if (event.target.type !== 'file')
65
+ return;
66
+ const holder = event.target.closest('[data-lk-id][data-lk-kind="fileUpload"]');
67
+ const id = holder?.dataset.lkId;
68
+ if (!id)
69
+ return;
70
+ const files = event.target.files ? Array.from(event.target.files) : [];
71
+ if (files.length === 0) {
72
+ channel.send({ type: 'EVENT', payload: { id, event: 'change', value: null } });
73
+ return;
74
+ }
75
+ const formData = new FormData();
76
+ for (const file of files) {
77
+ formData.append('file', file, file.name);
78
+ }
79
+ try {
80
+ const connectionId = String(window.localStorage.getItem('lk-connection-id') ?? '');
81
+ const search = connectionId ? `?connectionId=${encodeURIComponent(connectionId)}` : '';
82
+ const response = await fetch(`/upload${search}`, { method: 'POST', body: formData });
83
+ if (!response.ok) {
84
+ return;
85
+ }
86
+ const payload = await response.json();
87
+ const uploaded = Array.isArray(payload)
88
+ ? payload
89
+ : payload.file;
90
+ channel.send({
91
+ type: 'EVENT',
92
+ payload: {
93
+ id,
94
+ event: 'change',
95
+ value: uploaded ?? null,
96
+ },
97
+ });
98
+ }
99
+ catch {
100
+ // best effort
101
+ }
102
+ });
103
+ }
104
+ export function bindThemeToggle(root, channel) {
105
+ const toggle = root.getElementById('lk-theme-toggle');
106
+ if (!toggle)
107
+ return;
108
+ toggle.addEventListener('click', () => {
109
+ const current = root.documentElement.getAttribute('data-theme') === 'dark' ? 'dark' : 'light';
110
+ const mode = current === 'dark' ? 'light' : 'dark';
111
+ channel.send({ type: 'THEME_CHANGE', payload: { mode } });
112
+ });
113
+ }
114
+ //# sourceMappingURL=events.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"events.js","sourceRoot":"","sources":["../../src/client/events.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import { createWSManager } from './ws';
2
+ export { createWSManager };
@@ -0,0 +1,5 @@
1
+ import { createWSManager } from './ws';
2
+ const manager = createWSManager();
3
+ manager.connect();
4
+ export { createWSManager };
5
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/client/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,MAAM,CAAC;AAEvC,MAAM,OAAO,GAAG,eAAe,EAAE,CAAC;AAClC,OAAO,CAAC,OAAO,EAAE,CAAC;AAElB,OAAO,EAAE,eAAe,EAAE,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { RenderMessage } from '../engine/messages';
2
+ export declare function applyRender(payload: RenderMessage['payload']): void;
3
+ export declare function applyFragmentSwap(id: string, html: string): void;
4
+ export declare function applyStreamChunk(id: string, chunk: string, done: boolean, format: 'plain' | 'markdown'): void;
@@ -0,0 +1,47 @@
1
+ export function applyRender(payload) {
2
+ const root = document.getElementById('lk-root');
3
+ if (!root) {
4
+ return;
5
+ }
6
+ root.innerHTML = payload.html;
7
+ document.title = payload.title;
8
+ document.documentElement.setAttribute('data-theme', payload.theme);
9
+ }
10
+ export function applyFragmentSwap(id, html) {
11
+ const target = document.querySelector(`[data-lk-id="${id}"]`);
12
+ if (!target) {
13
+ return;
14
+ }
15
+ target.outerHTML = html;
16
+ }
17
+ function markdownToHtml(input) {
18
+ const escaped = input
19
+ .replaceAll('&', '&')
20
+ .replaceAll('<', '&lt;')
21
+ .replaceAll('>', '&gt;');
22
+ return escaped.replaceAll('\n', '<br />');
23
+ }
24
+ export function applyStreamChunk(id, chunk, done, format) {
25
+ const container = document.querySelector(`[data-lk-id="${id}"]`);
26
+ if (!container) {
27
+ return;
28
+ }
29
+ const body = container.querySelector('[data-lk-stream-body]');
30
+ if (!body) {
31
+ return;
32
+ }
33
+ const current = body.dataset.lkText ?? '';
34
+ const next = `${current}${chunk}`;
35
+ body.dataset.lkText = next;
36
+ if (format === 'markdown') {
37
+ body.innerHTML = markdownToHtml(next);
38
+ }
39
+ else {
40
+ body.textContent = next;
41
+ }
42
+ const cursor = container.querySelector('[data-lk-stream-cursor]');
43
+ if (cursor) {
44
+ cursor.style.display = done ? 'none' : '';
45
+ }
46
+ }
47
+ //# sourceMappingURL=swap.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"swap.js","sourceRoot":"","sources":["../../src/client/swap.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,WAAW,CAAC,OAAiC;IAC3D,MAAM,IAAI,GAAG,QAAQ,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC;IAChD,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,OAAO;IACT,CAAC;IACD,IAAI,CAAC,SAAS,GAAG,OAAO,CAAC,IAAI,CAAC;IAC9B,QAAQ,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC;IAC/B,QAAQ,CAAC,eAAe,CAAC,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC,KAAK,CAAC,CAAC;AACrE,CAAC;AAED,MAAM,UAAU,iBAAiB,CAAC,EAAU,EAAE,IAAY;IACxD,MAAM,MAAM,GAAG,QAAQ,CAAC,aAAa,CAAc,gBAAgB,EAAE,IAAI,CAAC,CAAC;IAC3E,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,OAAO;IACT,CAAC;IACD,MAAM,CAAC,SAAS,GAAG,IAAI,CAAC;AAC1B,CAAC;AAED,SAAS,cAAc,CAAC,KAAa;IACnC,MAAM,OAAO,GAAG,KAAK;SAClB,UAAU,CAAC,GAAG,EAAE,OAAO,CAAC;SACxB,UAAU,CAAC,GAAG,EAAE,MAAM,CAAC;SACvB,UAAU,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;IAC3B,OAAO,OAAO,CAAC,UAAU,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;AAC5C,CAAC;AAED,MAAM,UAAU,gBAAgB,CAC9B,EAAU,EACV,KAAa,EACb,IAAa,EACb,MAA4B;IAE5B,MAAM,SAAS,GAAG,QAAQ,CAAC,aAAa,CAAc,gBAAgB,EAAE,IAAI,CAAC,CAAC;IAC9E,IAAI,CAAC,SAAS,EAAE,CAAC;QACf,OAAO;IACT,CAAC;IACD,MAAM,IAAI,GAAG,SAAS,CAAC,aAAa,CAAc,uBAAuB,CAAC,CAAC;IAC3E,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,OAAO;IACT,CAAC;IAED,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,IAAI,EAAE,CAAC;IAC1C,MAAM,IAAI,GAAG,GAAG,OAAO,GAAG,KAAK,EAAE,CAAC;IAClC,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;IAC3B,IAAI,MAAM,KAAK,UAAU,EAAE,CAAC;QAC1B,IAAI,CAAC,SAAS,GAAG,cAAc,CAAC,IAAI,CAAC,CAAC;IACxC,CAAC;SAAM,CAAC;QACN,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;IAC1B,CAAC;IAED,MAAM,MAAM,GAAG,SAAS,CAAC,aAAa,CAAc,yBAAyB,CAAC,CAAC;IAC/E,IAAI,MAAM,EAAE,CAAC;QACX,MAAM,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC;IAC5C,CAAC;AACH,CAAC"}
@@ -0,0 +1,11 @@
1
+ export interface WSManagerOptions {
2
+ wsFactory?: (url: string) => WebSocket;
3
+ url?: string;
4
+ maxRetries?: number;
5
+ initialDelayMs?: number;
6
+ }
7
+ export interface WSManager {
8
+ connect(): void;
9
+ disconnect(): void;
10
+ }
11
+ export declare function createWSManager(opts?: WSManagerOptions): WSManager;