@effindomv2/fui-as 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 (137) hide show
  1. package/LICENSE.md +7 -0
  2. package/browser/src/common-harness/host-imports.ts +430 -0
  3. package/browser/src/common-harness/interop.ts +39 -0
  4. package/browser/src/common-harness/managed-harness-bitmap-host.ts +92 -0
  5. package/browser/src/common-harness/managed-harness-fetch-host.ts +201 -0
  6. package/browser/src/common-harness/managed-harness-file-host.ts +1101 -0
  7. package/browser/src/common-harness/managed-harness-file-payloads.ts +143 -0
  8. package/browser/src/common-harness/managed-harness-file-types.ts +106 -0
  9. package/browser/src/common-harness/managed-harness-session.ts +15 -0
  10. package/browser/src/common-harness/managed-harness.ts +1323 -0
  11. package/browser/src/common-harness/managed-history.ts +168 -0
  12. package/browser/src/common-harness/persisted-restore-policy.ts +50 -0
  13. package/browser/src/common-harness/persisted-ui-state-controller.ts +309 -0
  14. package/browser/src/common-harness/text-session-bridge.ts +452 -0
  15. package/browser/src/common-harness/types.ts +205 -0
  16. package/browser/src/common-harness/ui-chrome.ts +191 -0
  17. package/browser/src/common-harness/ui-imports.ts +529 -0
  18. package/browser/src/common-harness/wasm-module-cache.ts +47 -0
  19. package/browser/src/common-harness.ts +27 -0
  20. package/browser/src/file-processing-worker.ts +89 -0
  21. package/browser/src/host-events.ts +97 -0
  22. package/browser/src/host-services.ts +203 -0
  23. package/browser/src/index.ts +62 -0
  24. package/browser/src/persisted-ui-state.ts +206 -0
  25. package/browser/src/routed-harness.ts +198 -0
  26. package/browser/src/worker-bootstrap.ts +483 -0
  27. package/browser/src/worker-manager.ts +230 -0
  28. package/browser/src/worker-types.ts +50 -0
  29. package/package.json +89 -0
  30. package/scripts/build-demo-as.sh +91 -0
  31. package/scripts/build.sh +325 -0
  32. package/scripts/generate-host-events.ts +175 -0
  33. package/scripts/generate-host-services.ts +157 -0
  34. package/src/Fui.ts +205 -0
  35. package/src/FuiExports.ts +55 -0
  36. package/src/FuiPrimitives.ts +15 -0
  37. package/src/FuiWorker.ts +3 -0
  38. package/src/FuiWorkerExports.ts +6 -0
  39. package/src/bindings/ui.ts +531 -0
  40. package/src/color.ts +86 -0
  41. package/src/controls/AntiSelectionArea.ts +23 -0
  42. package/src/controls/Button.ts +750 -0
  43. package/src/controls/Checkbox.ts +181 -0
  44. package/src/controls/ContextMenu.ts +885 -0
  45. package/src/controls/ControlTemplateSet.ts +37 -0
  46. package/src/controls/Dialog.ts +355 -0
  47. package/src/controls/Dropdown.ts +856 -0
  48. package/src/controls/Form.ts +110 -0
  49. package/src/controls/NavLink.ts +211 -0
  50. package/src/controls/Popup.ts +129 -0
  51. package/src/controls/ProgressBar.ts +180 -0
  52. package/src/controls/RadioButton.ts +135 -0
  53. package/src/controls/RadioGroup.ts +244 -0
  54. package/src/controls/SelectionArea.ts +75 -0
  55. package/src/controls/Slider.ts +471 -0
  56. package/src/controls/Switch.ts +132 -0
  57. package/src/controls/TextArea.ts +20 -0
  58. package/src/controls/TextInput.ts +7 -0
  59. package/src/controls/index.ts +18 -0
  60. package/src/controls/internal/ButtonPresenter.ts +95 -0
  61. package/src/controls/internal/CheckboxIndicatorPresenter.ts +93 -0
  62. package/src/controls/internal/DropdownChevronPresenter.ts +67 -0
  63. package/src/controls/internal/DropdownFieldPresenter.ts +110 -0
  64. package/src/controls/internal/DropdownOptionRowPresenter.ts +82 -0
  65. package/src/controls/internal/PopupPresenter.ts +198 -0
  66. package/src/controls/internal/PressableIndicatorPresenter.ts +32 -0
  67. package/src/controls/internal/PressableLabeledControl.ts +221 -0
  68. package/src/controls/internal/RadioIndicatorPresenter.ts +73 -0
  69. package/src/controls/internal/SliderPresenter.ts +157 -0
  70. package/src/controls/internal/SwitchIndicatorPresenter.ts +72 -0
  71. package/src/controls/internal/TextInputCore.ts +695 -0
  72. package/src/controls/internal/TextInputPresenter.ts +72 -0
  73. package/src/controls/templating.ts +54 -0
  74. package/src/core/Action.ts +94 -0
  75. package/src/core/Actions.ts +37 -0
  76. package/src/core/Animation.ts +412 -0
  77. package/src/core/Application.ts +328 -0
  78. package/src/core/Assets.ts +264 -0
  79. package/src/core/AttachedProperties.ts +32 -0
  80. package/src/core/Bitmap.ts +70 -0
  81. package/src/core/BoundCallback.ts +104 -0
  82. package/src/core/Callbacks.ts +17 -0
  83. package/src/core/ContextMenuManager.ts +466 -0
  84. package/src/core/DebugApi.ts +30 -0
  85. package/src/core/Disposable.ts +10 -0
  86. package/src/core/DragDropManager.ts +179 -0
  87. package/src/core/DragGesture.ts +184 -0
  88. package/src/core/DynamicAssetIds.ts +24 -0
  89. package/src/core/Errors.ts +48 -0
  90. package/src/core/EventRouter.ts +408 -0
  91. package/src/core/ExternalDropManager.ts +122 -0
  92. package/src/core/Fetch.ts +264 -0
  93. package/src/core/FetchFfi.ts +15 -0
  94. package/src/core/File.ts +1002 -0
  95. package/src/core/FocusAdornerManager.ts +263 -0
  96. package/src/core/FocusVisibility.ts +36 -0
  97. package/src/core/FrameScheduler.ts +28 -0
  98. package/src/core/KeyboardScroll.ts +161 -0
  99. package/src/core/KeyboardScrollTracker.ts +386 -0
  100. package/src/core/Logger.ts +80 -0
  101. package/src/core/Navigation.ts +13 -0
  102. package/src/core/Node.ts +1708 -0
  103. package/src/core/PersistedState.ts +102 -0
  104. package/src/core/PersistedUiState.ts +142 -0
  105. package/src/core/Platform.ts +219 -0
  106. package/src/core/Signal.ts +89 -0
  107. package/src/core/Theme.ts +365 -0
  108. package/src/core/Timers.ts +129 -0
  109. package/src/core/ToolTip.ts +122 -0
  110. package/src/core/ToolTipManager.ts +459 -0
  111. package/src/core/Transitions.ts +34 -0
  112. package/src/core/Typography.ts +204 -0
  113. package/src/core/Worker.ts +196 -0
  114. package/src/core/bind.ts +37 -0
  115. package/src/core/event_exports.ts +596 -0
  116. package/src/core/ffi.ts +728 -0
  117. package/src/host-services/runtime.ts +25 -0
  118. package/src/nodes/FlexBox.ts +789 -0
  119. package/src/nodes/GradientStop.ts +9 -0
  120. package/src/nodes/Grid.ts +183 -0
  121. package/src/nodes/Image.ts +189 -0
  122. package/src/nodes/Portal.ts +14 -0
  123. package/src/nodes/RichText.ts +312 -0
  124. package/src/nodes/ScrollBar.ts +570 -0
  125. package/src/nodes/ScrollBox.ts +415 -0
  126. package/src/nodes/ScrollState.ts +10 -0
  127. package/src/nodes/ScrollView.ts +511 -0
  128. package/src/nodes/Svg.ts +142 -0
  129. package/src/nodes/Text.ts +145 -0
  130. package/src/nodes/TextCore.ts +558 -0
  131. package/src/nodes/VirtualList.ts +431 -0
  132. package/src/nodes/helpers.ts +25 -0
  133. package/src/nodes/index.ts +14 -0
  134. package/src/tsconfig.json +7 -0
  135. package/src/worker/Worker.ts +169 -0
  136. package/src/worker/WorkerJob.ts +65 -0
  137. package/src/worker/ffi.ts +23 -0
package/LICENSE.md ADDED
@@ -0,0 +1,7 @@
1
+ `@effindomv2/fui-as` is licensed under Business Source License 1.1 (BSL-1.1),
2
+ with a Change License to MIT after 5 years for each released version.
3
+
4
+ See:
5
+
6
+ - `../../LICENSE.md`
7
+ - `../../LICENSES/BSL-1.1.md`
@@ -0,0 +1,430 @@
1
+ import type { BridgeRuntime } from '@effindomv2/runtime';
2
+
3
+ import type { WorkerManager } from '../worker-manager';
4
+ import type { HarnessExports } from './types';
5
+ import type { PersistedUiStateController } from './persisted-ui-state-controller';
6
+ import type { TextSessionBridge } from './text-session-bridge';
7
+ import type { HarnessUiChrome } from './ui-chrome';
8
+ import { toBigIntHandle, type AppHandleLike } from './interop';
9
+
10
+ interface HostImportSessionLike {
11
+ readonly exports: HarnessExports;
12
+ readonly memory: WebAssembly.Memory;
13
+ readonly textBufferPtr: number;
14
+ readonly textBufferSize: number;
15
+ }
16
+
17
+ export interface HostImportDeps {
18
+ getRuntime(): BridgeRuntime;
19
+ getCurrentSession(): HostImportSessionLike;
20
+ getCurrentSessionOrNull(): HostImportSessionLike | null;
21
+ setAppFlushRequested(value: boolean): void;
22
+ queueHarnessFrame(): void;
23
+ uiChrome: HarnessUiChrome;
24
+ readAppUtf8(ptr: number, len: number): string;
25
+ writeAppFloat32(ptr: number, value: number): void;
26
+ writeAppUint32(ptr: number, value: number): void;
27
+ writeAppUtf8(ptr: number, capacity: number, text: string, context: string): number;
28
+ textBridge: TextSessionBridge;
29
+ persistedUiStateController: PersistedUiStateController;
30
+ navigateWithinDocument(target: string, openInNewTab: boolean): void;
31
+ canBrowserNavigateBack(): boolean;
32
+ canBrowserNavigateForward(): boolean;
33
+ navigateBrowserBack(): void;
34
+ navigateBrowserForward(): void;
35
+ cancelHostTimer(timerId: number): void;
36
+ getHostTimer(timerId: number): number | undefined;
37
+ setHostTimer(timerId: number, timeoutId: number): void;
38
+ deleteHostTimer(timerId: number): void;
39
+ workerManager: WorkerManager;
40
+ debugLogsEnabled: boolean;
41
+ notifySvgLoaded(session: HostImportSessionLike | null, svgId: number, width: number, height: number): void;
42
+ notifySvgFailed(session: HostImportSessionLike | null, svgId: number, error: string): void;
43
+ notifyTextureLoaded(session: HostImportSessionLike | null, textureId: number, width: number, height: number): void;
44
+ notifyTextureFailed(session: HostImportSessionLike | null, textureId: number, error: string): void;
45
+ }
46
+
47
+ export function createHostImportModule(deps: HostImportDeps) {
48
+ return {
49
+ request_render(): void {
50
+ const runtime = deps.getRuntime();
51
+ deps.setAppFlushRequested(true);
52
+ runtime.requestFrame();
53
+ deps.queueHarnessFrame();
54
+ },
55
+ get_viewport_width(): number {
56
+ const runtime = deps.getRuntime();
57
+ const sizeSource = deps.uiChrome.getCanvasSizeSource(runtime.canvas);
58
+ const rect = sizeSource.getBoundingClientRect();
59
+ return sizeSource.clientWidth > 0 ? sizeSource.clientWidth : (rect.width > 0 ? rect.width : runtime.canvas.width);
60
+ },
61
+ get_viewport_height(): number {
62
+ const runtime = deps.getRuntime();
63
+ const sizeSource = deps.uiChrome.getCanvasSizeSource(runtime.canvas);
64
+ const rect = sizeSource.getBoundingClientRect();
65
+ return sizeSource.clientHeight > 0 ? sizeSource.clientHeight : (rect.height > 0 ? rect.height : runtime.canvas.height);
66
+ },
67
+ fui_set_pointer_capture(handle: AppHandleLike): void {
68
+ deps.getRuntime().setCapturedPointerHandle(toBigIntHandle(handle));
69
+ },
70
+ fui_release_pointer_capture(): void {
71
+ deps.getRuntime().setCapturedPointerHandle(null);
72
+ },
73
+ fui_reload_page(): void {
74
+ window.location.reload();
75
+ },
76
+ fui_can_navigate_back(): number {
77
+ return deps.canBrowserNavigateBack() ? 1 : 0;
78
+ },
79
+ fui_can_navigate_forward(): number {
80
+ return deps.canBrowserNavigateForward() ? 1 : 0;
81
+ },
82
+ fui_navigate_back(): void {
83
+ deps.navigateBrowserBack();
84
+ },
85
+ fui_navigate_forward(): void {
86
+ deps.navigateBrowserForward();
87
+ },
88
+ fui_copy_text(ptr: number, len: number): void {
89
+ const text = deps.readAppUtf8(ptr, len);
90
+ window.__effindomCallbacks?.onClipboardWrite?.({ plainText: text });
91
+ },
92
+ fui_has_text_selection_snapshot(handle: AppHandleLike): number {
93
+ return deps.textBridge.resolveFrozenOrLiveTextSelection(handle) !== null ? 1 : 0;
94
+ },
95
+ fui_freeze_text_selection_snapshot(handle: AppHandleLike): void {
96
+ deps.textBridge.freezeTextSelectionSnapshot(handle);
97
+ },
98
+ fui_copy_text_selection_snapshot(handle: AppHandleLike): number {
99
+ const snapshot = deps.textBridge.resolveFrozenOrLiveTextSelection(handle);
100
+ if (snapshot === null) {
101
+ return 0;
102
+ }
103
+ window.__effindomCallbacks?.onClipboardWrite?.({
104
+ plainText: snapshot.text.slice(snapshot.start, snapshot.end),
105
+ });
106
+ return 1;
107
+ },
108
+ fui_cut_focused_text_selection(): number {
109
+ const editor = deps.textBridge.getHiddenTextEditor();
110
+ if (editor === null) {
111
+ return 0;
112
+ }
113
+ const selectionStart = editor.selectionStart ?? 0;
114
+ const selectionEnd = editor.selectionEnd ?? 0;
115
+ if (selectionStart === selectionEnd) {
116
+ return 0;
117
+ }
118
+ const start = Math.min(selectionStart, selectionEnd);
119
+ const end = Math.max(selectionStart, selectionEnd);
120
+ window.__effindomCallbacks?.onClipboardWrite?.({
121
+ plainText: editor.value.slice(start, end),
122
+ });
123
+ editor.focus({ preventScroll: true });
124
+ editor.setRangeText('', start, end, 'start');
125
+ editor.dispatchEvent(new InputEvent('input', { bubbles: true, inputType: 'deleteByCut', data: null }));
126
+ return 1;
127
+ },
128
+ fui_cut_text_selection_snapshot(handle: AppHandleLike): number {
129
+ const runtime = deps.getRuntime();
130
+ const snapshot = deps.textBridge.resolveFrozenOrLiveTextSelection(handle);
131
+ if (snapshot === null) {
132
+ return 0;
133
+ }
134
+ const { handleKey, text, start, end } = snapshot;
135
+ window.__effindomCallbacks?.onClipboardWrite?.({
136
+ plainText: text.slice(start, end),
137
+ });
138
+ const updatedText = text.slice(0, start) + text.slice(end);
139
+ const editor = deps.textBridge.getHiddenTextEditor();
140
+ if (editor !== null) {
141
+ editor.focus({ preventScroll: true });
142
+ editor.value = updatedText;
143
+ editor.setSelectionRange(start, start, 'none');
144
+ }
145
+ window.setTimeout(() => {
146
+ runtime.ui._ui_request_focus(toBigIntHandle(handle));
147
+ deps.textBridge.withUiUtf8('', (uiPtr, uiLen) => {
148
+ runtime.ui._ui_replace_text_range(toBigIntHandle(handle), start, end, uiPtr, uiLen, start);
149
+ });
150
+ runtime.commitFrame();
151
+ deps.queueHarnessFrame();
152
+ deps.textBridge.updateLiveTextAfterCut(handleKey, updatedText, start);
153
+ const activeEditor = deps.textBridge.getHiddenTextEditor();
154
+ if (activeEditor !== null) {
155
+ activeEditor.focus({ preventScroll: true });
156
+ activeEditor.setSelectionRange(start, start, 'none');
157
+ }
158
+ }, 0);
159
+ deps.textBridge.clearFrozenTextSelectionSnapshot();
160
+ return 1;
161
+ },
162
+ fui_cut_text_range_snapshot(handle: AppHandleLike, start: number, end: number): number {
163
+ const textSnapshot = deps.textBridge.resolveFrozenOrLiveTextSelection(handle);
164
+ const handleKey = toBigIntHandle(handle).toString();
165
+ const text = textSnapshot?.handleKey === handleKey
166
+ ? textSnapshot.text
167
+ : deps.textBridge.getLatestText(handle);
168
+ const resolvedText = text.length > 0 ? text : '';
169
+ if (resolvedText.length === 0) {
170
+ return 0;
171
+ }
172
+ const rangeStart = Math.max(0, Math.min(start, end));
173
+ const rangeEnd = Math.max(rangeStart, Math.min(resolvedText.length, Math.max(start, end)));
174
+ if (rangeStart === rangeEnd) {
175
+ return 0;
176
+ }
177
+ window.__effindomCallbacks?.onClipboardWrite?.({
178
+ plainText: resolvedText.slice(rangeStart, rangeEnd),
179
+ });
180
+ const updatedText = resolvedText.slice(0, rangeStart) + resolvedText.slice(rangeEnd);
181
+ const editor = deps.textBridge.getHiddenTextEditor();
182
+ if (editor !== null) {
183
+ editor.focus({ preventScroll: true });
184
+ editor.value = updatedText;
185
+ editor.setSelectionRange(rangeStart, rangeStart, 'none');
186
+ }
187
+ window.setTimeout(() => {
188
+ deps.getRuntime().ui._ui_request_focus(toBigIntHandle(handle));
189
+ deps.textBridge.syncEditableTextToRuntime(handle, updatedText, rangeStart);
190
+ deps.textBridge.updateLiveTextAfterCut(handleKey, updatedText, rangeStart);
191
+ const activeEditor = deps.textBridge.getHiddenTextEditor();
192
+ if (activeEditor !== null) {
193
+ activeEditor.focus({ preventScroll: true });
194
+ activeEditor.setSelectionRange(rangeStart, rangeStart, 'none');
195
+ }
196
+ }, 0);
197
+ return 1;
198
+ },
199
+ fui_delete_focused_text_range(start: number, end: number): number {
200
+ const editor = deps.textBridge.getHiddenTextEditor();
201
+ if (editor === null) {
202
+ return 0;
203
+ }
204
+ const rangeStart = Math.max(0, Math.min(start, end));
205
+ const rangeEnd = Math.max(rangeStart, Math.max(start, end));
206
+ editor.focus({ preventScroll: true });
207
+ editor.setSelectionRange(rangeStart, rangeEnd);
208
+ editor.setRangeText('', rangeStart, rangeEnd, 'start');
209
+ editor.dispatchEvent(new InputEvent('input', { bubbles: true, inputType: 'deleteByCut', data: null }));
210
+ return 1;
211
+ },
212
+ fui_commit_text_action_focus(handle: AppHandleLike): void {
213
+ const runtime = deps.getRuntime();
214
+ window.setTimeout(() => {
215
+ runtime.ui._ui_request_focus(toBigIntHandle(handle));
216
+ runtime.commitFrame();
217
+ deps.queueHarnessFrame();
218
+ const editor = deps.textBridge.getHiddenTextEditor();
219
+ if (editor !== null) {
220
+ editor.focus({ preventScroll: true });
221
+ }
222
+ }, 0);
223
+ },
224
+ fui_load_svg(svgId: number, ptr: number, len: number): void {
225
+ const runtime = deps.getRuntime();
226
+ const session = deps.getCurrentSessionOrNull();
227
+ const url = deps.readAppUtf8(ptr, len);
228
+ void runtime.loadSvg(svgId, url).then((result) => {
229
+ if (deps.getCurrentSessionOrNull() !== session) {
230
+ return;
231
+ }
232
+ deps.notifySvgLoaded(session, svgId, result.width, result.height);
233
+ }).catch((error: unknown) => {
234
+ const message = error instanceof Error ? error.message : String(error);
235
+ console.error(`[fui_host] SVG ${String(svgId)} failed to load from ${url}: ${message}`);
236
+ if (deps.getCurrentSessionOrNull() !== session) {
237
+ return;
238
+ }
239
+ deps.notifySvgFailed(session, svgId, message);
240
+ });
241
+ },
242
+ fui_load_texture(textureId: number, ptr: number, len: number): void {
243
+ const runtime = deps.getRuntime();
244
+ const session = deps.getCurrentSessionOrNull();
245
+ const url = deps.readAppUtf8(ptr, len);
246
+ void runtime.loadTexture(textureId, url).then((result) => {
247
+ if (deps.getCurrentSessionOrNull() !== session) {
248
+ return;
249
+ }
250
+ deps.notifyTextureLoaded(session, textureId, result.width, result.height);
251
+ }).catch((error: unknown) => {
252
+ const message = error instanceof Error ? error.message : String(error);
253
+ console.error(`[fui_host] texture ${String(textureId)} failed to load from ${url}: ${message}`);
254
+ if (deps.getCurrentSessionOrNull() !== session) {
255
+ return;
256
+ }
257
+ deps.notifyTextureFailed(session, textureId, message);
258
+ });
259
+ },
260
+ fui_release_svg(svgId: number): void {
261
+ deps.getRuntime().releaseSvg(svgId);
262
+ },
263
+ fui_release_texture(textureId: number): void {
264
+ deps.getRuntime().releaseTexture(textureId);
265
+ },
266
+ fui_load_font(fontId: number, ptr: number, len: number): void {
267
+ const url = deps.readAppUtf8(ptr, len);
268
+ deps.getRuntime().registerLazyFont(fontId, url);
269
+ },
270
+ fui_start_timer(timerId: number, delayMs: number): void {
271
+ deps.cancelHostTimer(timerId);
272
+ const session = deps.getCurrentSessionOrNull();
273
+ const clampedDelayMs = Math.max(0, Math.ceil(delayMs));
274
+ const timeoutId = window.setTimeout(() => {
275
+ if (deps.getHostTimer(timerId) !== timeoutId) {
276
+ return;
277
+ }
278
+ deps.deleteHostTimer(timerId);
279
+ if (session === null || deps.getCurrentSessionOrNull() !== session) {
280
+ return;
281
+ }
282
+ session.exports.__fui_on_timer(timerId);
283
+ }, clampedDelayMs);
284
+ deps.setHostTimer(timerId, timeoutId);
285
+ },
286
+ fui_cancel_timer(timerId: number): void {
287
+ deps.cancelHostTimer(timerId);
288
+ },
289
+ fui_now_ms(): number {
290
+ return performance.now();
291
+ },
292
+ fui_worker_start_string(workerId: number, entryPtr: number, entryLen: number, inputPtr: number, inputLen: number): void {
293
+ deps.workerManager.startString(workerId, deps.readAppUtf8(entryPtr, entryLen), deps.readAppUtf8(inputPtr, inputLen));
294
+ },
295
+ fui_worker_cancel(workerId: number): void {
296
+ deps.workerManager.cancel(workerId);
297
+ },
298
+ fui_set_cursor(style: number): void {
299
+ if (deps.uiChrome.detectCoarsePointer()) {
300
+ return;
301
+ }
302
+ const cursor =
303
+ style === 1 ? 'pointer' :
304
+ style === 2 ? 'text' :
305
+ style === 3 ? 'move' :
306
+ style === 4 ? 'grab' :
307
+ style === 5 ? 'grabbing' :
308
+ style === 6 ? 'ns-resize' :
309
+ style === 7 ? 'ew-resize' :
310
+ 'default';
311
+ deps.getRuntime().canvas.style.cursor = cursor;
312
+ },
313
+ fui_is_dark_mode(): number {
314
+ return window.matchMedia('(prefers-color-scheme: dark)').matches ? 1 : 0;
315
+ },
316
+ fui_get_accent_color(): number {
317
+ return deps.uiChrome.readHostAccentColor();
318
+ },
319
+ fui_get_platform_family(): number {
320
+ return deps.uiChrome.detectPlatformFamily();
321
+ },
322
+ fui_is_coarse_pointer(): number {
323
+ return deps.uiChrome.detectCoarsePointer() ? 1 : 0;
324
+ },
325
+ fui_show_url_preview(ptr: number, len: number): void {
326
+ const rawTarget = deps.readAppUtf8(ptr, len);
327
+ try {
328
+ const resolvedTarget = new URL(rawTarget, window.location.href);
329
+ deps.uiChrome.setUrlPreviewText(resolvedTarget.href);
330
+ } catch {
331
+ deps.uiChrome.setUrlPreviewText(rawTarget);
332
+ }
333
+ },
334
+ fui_hide_url_preview(): void {
335
+ deps.uiChrome.setUrlPreviewText('');
336
+ },
337
+ fui_navigate_to(ptr: number, len: number, openInNewTab: number): void {
338
+ deps.navigateWithinDocument(deps.readAppUtf8(ptr, len), openInNewTab !== 0);
339
+ },
340
+ fui_set_persisted_scroll_offset(nodeIdPtr: number, nodeIdLen: number, x: number, y: number): void {
341
+ const nodeId = deps.readAppUtf8(nodeIdPtr, nodeIdLen);
342
+ if (nodeId.length === 0) {
343
+ return;
344
+ }
345
+ deps.persistedUiStateController.setCurrentPersistedScrollEntry(nodeId, x, y);
346
+ },
347
+ fui_try_get_persisted_scroll_offset(nodeIdPtr: number, nodeIdLen: number, outX: number, outY: number): number {
348
+ const nodeId = deps.readAppUtf8(nodeIdPtr, nodeIdLen);
349
+ if (nodeId.length === 0) {
350
+ return 0;
351
+ }
352
+ const payload = deps.persistedUiStateController.getCurrentPersistedScrollEntry(nodeId);
353
+ if (payload === null) {
354
+ return 0;
355
+ }
356
+ deps.writeAppFloat32(outX, payload.x);
357
+ deps.writeAppFloat32(outY, payload.y);
358
+ return 1;
359
+ },
360
+ fui_set_persisted_state(
361
+ nodeIdPtr: number,
362
+ nodeIdLen: number,
363
+ kindPtr: number,
364
+ kindLen: number,
365
+ version: number,
366
+ payloadPtr: number,
367
+ payloadLen: number,
368
+ ): void {
369
+ const nodeId = deps.readAppUtf8(nodeIdPtr, nodeIdLen);
370
+ const kind = deps.readAppUtf8(kindPtr, kindLen);
371
+ if (nodeId.length === 0 || kind.length === 0) {
372
+ return;
373
+ }
374
+ deps.persistedUiStateController.setCurrentPersistedTextEntry(
375
+ nodeId,
376
+ kind,
377
+ version >>> 0,
378
+ deps.readAppUtf8(payloadPtr, payloadLen),
379
+ );
380
+ },
381
+ fui_copy_persisted_state(
382
+ nodeIdPtr: number,
383
+ nodeIdLen: number,
384
+ kindPtr: number,
385
+ kindLen: number,
386
+ outVersionPtr: number,
387
+ payloadPtr: number,
388
+ payloadCapacity: number,
389
+ ): number {
390
+ const nodeId = deps.readAppUtf8(nodeIdPtr, nodeIdLen);
391
+ const kind = deps.readAppUtf8(kindPtr, kindLen);
392
+ if (nodeId.length === 0 || kind.length === 0) {
393
+ return -1;
394
+ }
395
+ const entry = deps.persistedUiStateController.getCurrentPersistedTextEntry(nodeId, kind);
396
+ if (entry === null) {
397
+ return -1;
398
+ }
399
+ deps.writeAppUint32(outVersionPtr, entry.version >>> 0);
400
+ const payloadBytes = new TextEncoder().encode(entry.payload);
401
+ if (payloadBytes.length > payloadCapacity) {
402
+ return payloadBytes.length;
403
+ }
404
+ deps.writeAppUtf8(payloadPtr, payloadCapacity, entry.payload, `Persisted state ${kind}`);
405
+ return payloadBytes.length;
406
+ },
407
+ fui_log(catPtr: number, catLen: number, msgPtr: number, msgLen: number): void {
408
+ if (deps.getCurrentSessionOrNull() === null) {
409
+ return;
410
+ }
411
+ const category = deps.readAppUtf8(catPtr, catLen);
412
+ const message = deps.readAppUtf8(msgPtr, msgLen);
413
+ const formatted = `[fui:${category}] ${message}`;
414
+ if (category.startsWith('Warning/')) {
415
+ console.warn(formatted);
416
+ return;
417
+ }
418
+ if (category.startsWith('Error/')) {
419
+ console.error(formatted);
420
+ return;
421
+ }
422
+ if (deps.debugLogsEnabled) {
423
+ console.debug(formatted);
424
+ }
425
+ },
426
+ fui_logs_enabled(): number {
427
+ return deps.debugLogsEnabled ? 1 : 0;
428
+ },
429
+ };
430
+ }
@@ -0,0 +1,39 @@
1
+ import type { BridgeRuntime, WasmHandleLike } from '@effindomv2/runtime';
2
+ import {
3
+ handleToBigInt,
4
+ normalizePointerForWasm,
5
+ pointerToHeapOffset,
6
+ } from '@effindomv2/runtime';
7
+
8
+ export type AppHandleLike = number | bigint;
9
+
10
+ export function toBigIntHandle(handle: WasmHandleLike | AppHandleLike): bigint {
11
+ return handleToBigInt(handle);
12
+ }
13
+
14
+ export function toNumberHandle(handle: WasmHandleLike | AppHandleLike): number {
15
+ return pointerToHeapOffset(handle);
16
+ }
17
+
18
+ export function zeroPointer(runtime: BridgeRuntime): WasmHandleLike | number {
19
+ return normalizePointerForWasm(runtime.ui, 0);
20
+ }
21
+
22
+ export function normalizePointer(runtime: BridgeRuntime, ptr: WasmHandleLike | number): WasmHandleLike | number {
23
+ return normalizePointerForWasm(runtime.ui, ptr);
24
+ }
25
+
26
+ export function addUiPointer(
27
+ runtime: BridgeRuntime,
28
+ ptr: WasmHandleLike | number,
29
+ byteOffset: number,
30
+ ): WasmHandleLike | number {
31
+ if (runtime.ui.usesMemory64 === true) {
32
+ return toBigIntHandle(ptr) + BigInt(byteOffset);
33
+ }
34
+ return toNumberHandle(ptr) + byteOffset;
35
+ }
36
+
37
+ export function currentInteractionTimeMs(): bigint {
38
+ return BigInt(Math.floor(performance.now()));
39
+ }
@@ -0,0 +1,92 @@
1
+ import type { BridgeRuntime } from '@effindomv2/runtime';
2
+ import { writeBytesToHeap } from '@effindomv2/runtime';
3
+
4
+ interface CustomBitmapRecord {
5
+ readonly width: number;
6
+ readonly height: number;
7
+ readonly bytes: Uint8Array;
8
+ }
9
+
10
+ interface ManagedHarnessBitmapHostDependencies {
11
+ getRuntime(): BridgeRuntime;
12
+ readAppBytes(ptr: number, len: number): Uint8Array;
13
+ notifyBitmapChanged(): void;
14
+ }
15
+
16
+ export function createManagedHarnessBitmapHost(dependencies: ManagedHarnessBitmapHostDependencies) {
17
+ const customBitmapTextures = new Map<number, CustomBitmapRecord>();
18
+ const customBitmapReplayRuntimes = new WeakSet<BridgeRuntime>();
19
+
20
+ function uploadCustomBitmap(targetRuntime: BridgeRuntime, textureId: number, record: CustomBitmapRecord): void {
21
+ const textureBytes = writeBytesToHeap(targetRuntime.core, record.bytes);
22
+ try {
23
+ targetRuntime.core._ed_register_texture_rgba(
24
+ textureId,
25
+ textureBytes.ptr,
26
+ record.width,
27
+ record.height,
28
+ textureBytes.len,
29
+ );
30
+ } finally {
31
+ textureBytes.dispose();
32
+ }
33
+ }
34
+
35
+ function installReplay(targetRuntime: BridgeRuntime): void {
36
+ if (customBitmapReplayRuntimes.has(targetRuntime)) {
37
+ return;
38
+ }
39
+ const replayLoadedAssets = targetRuntime.replayLoadedAssets.bind(targetRuntime);
40
+ targetRuntime.replayLoadedAssets = async (): Promise<void> => {
41
+ await replayLoadedAssets();
42
+ for (const [textureId, record] of customBitmapTextures.entries()) {
43
+ uploadCustomBitmap(targetRuntime, textureId, record);
44
+ }
45
+ };
46
+ customBitmapReplayRuntimes.add(targetRuntime);
47
+ }
48
+
49
+ function clearTextures(targetRuntime: BridgeRuntime): void {
50
+ for (const textureId of customBitmapTextures.keys()) {
51
+ targetRuntime.core._ed_unregister_texture(textureId);
52
+ }
53
+ customBitmapTextures.clear();
54
+ }
55
+
56
+ return {
57
+ installReplay,
58
+ clearTextures,
59
+ imports: {
60
+ fui_bitmap_commit(textureId: number, ptr: number, len: number, width: number, height: number): void {
61
+ if (!Number.isInteger(textureId) || textureId <= 0) {
62
+ throw new Error('Bitmap commit requires a non-zero texture ID.');
63
+ }
64
+ if (!Number.isInteger(width) || !Number.isInteger(height) || width <= 0 || height <= 0) {
65
+ throw new Error('Bitmap commit requires positive integer dimensions.');
66
+ }
67
+ const expectedLength = width * height * 4;
68
+ if (len !== expectedLength) {
69
+ throw new Error(
70
+ `Bitmap commit byte length mismatch: expected ${String(expectedLength)} bytes for ${String(width)}x${String(height)}, received ${String(len)}.`,
71
+ );
72
+ }
73
+ const record: CustomBitmapRecord = {
74
+ width,
75
+ height,
76
+ bytes: dependencies.readAppBytes(ptr, len),
77
+ };
78
+ customBitmapTextures.set(textureId, record);
79
+ uploadCustomBitmap(dependencies.getRuntime(), textureId, record);
80
+ dependencies.notifyBitmapChanged();
81
+ },
82
+ fui_bitmap_release(textureId: number): void {
83
+ if (!Number.isInteger(textureId) || textureId <= 0) {
84
+ return;
85
+ }
86
+ customBitmapTextures.delete(textureId);
87
+ dependencies.getRuntime().core._ed_unregister_texture(textureId);
88
+ dependencies.notifyBitmapChanged();
89
+ },
90
+ },
91
+ };
92
+ }