@opentui/core 0.1.23 → 0.1.25
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/3d.js +1 -1
- package/3d.js.map +1 -1
- package/Renderable.d.ts +32 -23
- package/ansi.d.ts +2 -17
- package/buffer.d.ts +3 -1
- package/{index-a6ydv6yb.js → index-6kvgbzah.js} +1279 -618
- package/index-6kvgbzah.js.map +39 -0
- package/index.d.ts +1 -0
- package/index.js +376 -151
- package/index.js.map +11 -11
- package/lib/KeyHandler.d.ts +10 -3
- package/lib/env.d.ts +41 -0
- package/lib/index.d.ts +1 -1
- package/lib/parse.keypress-kitty.d.ts +2 -0
- package/lib/parse.keypress.d.ts +11 -1
- package/lib/renderable.validations.d.ts +12 -0
- package/package.json +14 -12
- package/renderables/ASCIIFont.d.ts +1 -1
- package/renderables/Input.d.ts +2 -2
- package/renderables/ScrollBox.d.ts +3 -1
- package/renderables/Slider.d.ts +24 -11
- package/renderables/Text.d.ts +18 -3
- package/renderables/TextNode.d.ts +13 -4
- package/renderer.d.ts +17 -2
- package/testing/mock-keys.d.ts +1 -0
- package/testing/mock-mouse.d.ts +38 -0
- package/testing/test-renderer.d.ts +16 -3
- package/testing.d.ts +3 -0
- package/testing.js +353 -0
- package/testing.js.map +12 -0
- package/text-buffer.d.ts +8 -9
- package/types.d.ts +5 -0
- package/utils.d.ts +2 -0
- package/zig.d.ts +16 -10
- package/index-a6ydv6yb.js.map +0 -37
- package/lib/TrackedNode.d.ts +0 -36
- /package/{singleton.d.ts → lib/singleton.d.ts} +0 -0
package/testing.js
ADDED
|
@@ -0,0 +1,353 @@
|
|
|
1
|
+
// @bun
|
|
2
|
+
import {
|
|
3
|
+
ANSI,
|
|
4
|
+
CliRenderer,
|
|
5
|
+
resolveRenderLib
|
|
6
|
+
} from "./index-6kvgbzah.js";
|
|
7
|
+
|
|
8
|
+
// src/testing/mock-keys.ts
|
|
9
|
+
var KeyCodes = {
|
|
10
|
+
ENTER: "\r",
|
|
11
|
+
TAB: "\t",
|
|
12
|
+
BACKSPACE: "\b",
|
|
13
|
+
DELETE: "\x1B[3~",
|
|
14
|
+
HOME: "\x1B[H",
|
|
15
|
+
END: "\x1B[F",
|
|
16
|
+
ESCAPE: "\x1B",
|
|
17
|
+
ARROW_UP: "\x1B[A",
|
|
18
|
+
ARROW_DOWN: "\x1B[B",
|
|
19
|
+
ARROW_RIGHT: "\x1B[C",
|
|
20
|
+
ARROW_LEFT: "\x1B[D",
|
|
21
|
+
F1: "\x1BOP",
|
|
22
|
+
F2: "\x1BOQ",
|
|
23
|
+
F3: "\x1BOR",
|
|
24
|
+
F4: "\x1BOS",
|
|
25
|
+
F5: "\x1B[15~",
|
|
26
|
+
F6: "\x1B[17~",
|
|
27
|
+
F7: "\x1B[18~",
|
|
28
|
+
F8: "\x1B[19~",
|
|
29
|
+
F9: "\x1B[20~",
|
|
30
|
+
F10: "\x1B[21~",
|
|
31
|
+
F11: "\x1B[23~",
|
|
32
|
+
F12: "\x1B[24~",
|
|
33
|
+
CTRL_A: "\x01",
|
|
34
|
+
CTRL_B: "\x02",
|
|
35
|
+
CTRL_C: "\x03",
|
|
36
|
+
CTRL_D: "\x04",
|
|
37
|
+
CTRL_E: "\x05",
|
|
38
|
+
CTRL_F: "\x06",
|
|
39
|
+
CTRL_G: "\x07",
|
|
40
|
+
CTRL_H: "\b",
|
|
41
|
+
CTRL_I: "\t",
|
|
42
|
+
CTRL_J: `
|
|
43
|
+
`,
|
|
44
|
+
CTRL_K: "\v",
|
|
45
|
+
CTRL_L: "\f",
|
|
46
|
+
CTRL_M: "\r",
|
|
47
|
+
CTRL_N: "\x0E",
|
|
48
|
+
CTRL_O: "\x0F",
|
|
49
|
+
CTRL_P: "\x10",
|
|
50
|
+
CTRL_Q: "\x11",
|
|
51
|
+
CTRL_R: "\x12",
|
|
52
|
+
CTRL_S: "\x13",
|
|
53
|
+
CTRL_T: "\x14",
|
|
54
|
+
CTRL_U: "\x15",
|
|
55
|
+
CTRL_V: "\x16",
|
|
56
|
+
CTRL_W: "\x17",
|
|
57
|
+
CTRL_X: "\x18",
|
|
58
|
+
CTRL_Y: "\x19",
|
|
59
|
+
CTRL_Z: "\x1A",
|
|
60
|
+
ALT_A: "\x1Ba",
|
|
61
|
+
ALT_B: "\x1Bb",
|
|
62
|
+
ALT_C: "\x1Bc"
|
|
63
|
+
};
|
|
64
|
+
function createMockKeys(renderer) {
|
|
65
|
+
const pressKeys = async (keys, delayMs = 0) => {
|
|
66
|
+
for (const key of keys) {
|
|
67
|
+
let keyCode;
|
|
68
|
+
if (typeof key === "string") {
|
|
69
|
+
if (key in KeyCodes) {
|
|
70
|
+
keyCode = KeyCodes[key];
|
|
71
|
+
} else {
|
|
72
|
+
keyCode = key;
|
|
73
|
+
}
|
|
74
|
+
} else {
|
|
75
|
+
keyCode = KeyCodes[key];
|
|
76
|
+
if (!keyCode) {
|
|
77
|
+
throw new Error(`Unknown key: ${key}`);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
renderer.stdin.emit("data", Buffer.from(keyCode));
|
|
81
|
+
if (delayMs > 0) {
|
|
82
|
+
await new Promise((resolve) => setTimeout(resolve, delayMs));
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
};
|
|
86
|
+
const pressKey = (key) => {
|
|
87
|
+
let keyCode;
|
|
88
|
+
if (typeof key === "string") {
|
|
89
|
+
if (key in KeyCodes) {
|
|
90
|
+
keyCode = KeyCodes[key];
|
|
91
|
+
} else {
|
|
92
|
+
keyCode = key;
|
|
93
|
+
}
|
|
94
|
+
} else {
|
|
95
|
+
keyCode = KeyCodes[key];
|
|
96
|
+
if (!keyCode) {
|
|
97
|
+
throw new Error(`Unknown key: ${key}`);
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
renderer.stdin.emit("data", Buffer.from(keyCode));
|
|
101
|
+
};
|
|
102
|
+
const typeText = async (text, delayMs = 0) => {
|
|
103
|
+
const keys = text.split("");
|
|
104
|
+
await pressKeys(keys, delayMs);
|
|
105
|
+
};
|
|
106
|
+
const pressEnter = () => {
|
|
107
|
+
pressKey(KeyCodes.ENTER);
|
|
108
|
+
};
|
|
109
|
+
const pressEscape = () => {
|
|
110
|
+
pressKey(KeyCodes.ESCAPE);
|
|
111
|
+
};
|
|
112
|
+
const pressTab = () => {
|
|
113
|
+
pressKey(KeyCodes.TAB);
|
|
114
|
+
};
|
|
115
|
+
const pressBackspace = () => {
|
|
116
|
+
pressKey(KeyCodes.BACKSPACE);
|
|
117
|
+
};
|
|
118
|
+
const pressArrow = (direction) => {
|
|
119
|
+
const keyMap = {
|
|
120
|
+
up: KeyCodes.ARROW_UP,
|
|
121
|
+
down: KeyCodes.ARROW_DOWN,
|
|
122
|
+
left: KeyCodes.ARROW_LEFT,
|
|
123
|
+
right: KeyCodes.ARROW_RIGHT
|
|
124
|
+
};
|
|
125
|
+
pressKey(keyMap[direction]);
|
|
126
|
+
};
|
|
127
|
+
const pressCtrlC = () => {
|
|
128
|
+
pressKey(KeyCodes.CTRL_C);
|
|
129
|
+
};
|
|
130
|
+
const pasteBracketedText = (text) => {
|
|
131
|
+
return pressKeys([ANSI.bracketedPasteStart, text, ANSI.bracketedPasteEnd]);
|
|
132
|
+
};
|
|
133
|
+
return {
|
|
134
|
+
pressKeys,
|
|
135
|
+
pressKey,
|
|
136
|
+
typeText,
|
|
137
|
+
pressEnter,
|
|
138
|
+
pressEscape,
|
|
139
|
+
pressTab,
|
|
140
|
+
pressBackspace,
|
|
141
|
+
pressArrow,
|
|
142
|
+
pressCtrlC,
|
|
143
|
+
pasteBracketedText
|
|
144
|
+
};
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
// src/testing/mock-mouse.ts
|
|
148
|
+
var MouseButtons = {
|
|
149
|
+
LEFT: 0,
|
|
150
|
+
MIDDLE: 1,
|
|
151
|
+
RIGHT: 2,
|
|
152
|
+
WHEEL_UP: 64,
|
|
153
|
+
WHEEL_DOWN: 65,
|
|
154
|
+
WHEEL_LEFT: 66,
|
|
155
|
+
WHEEL_RIGHT: 67
|
|
156
|
+
};
|
|
157
|
+
function createMockMouse(renderer) {
|
|
158
|
+
let currentPosition = { x: 0, y: 0 };
|
|
159
|
+
let buttonsPressed = new Set;
|
|
160
|
+
const generateMouseEvent = (type, x, y, button = MouseButtons.LEFT, modifiers = {}) => {
|
|
161
|
+
let buttonCode = button;
|
|
162
|
+
if (modifiers.shift)
|
|
163
|
+
buttonCode |= 4;
|
|
164
|
+
if (modifiers.alt)
|
|
165
|
+
buttonCode |= 8;
|
|
166
|
+
if (modifiers.ctrl)
|
|
167
|
+
buttonCode |= 16;
|
|
168
|
+
switch (type) {
|
|
169
|
+
case "move":
|
|
170
|
+
buttonCode = 32 | 3;
|
|
171
|
+
if (modifiers.shift)
|
|
172
|
+
buttonCode |= 4;
|
|
173
|
+
if (modifiers.alt)
|
|
174
|
+
buttonCode |= 8;
|
|
175
|
+
if (modifiers.ctrl)
|
|
176
|
+
buttonCode |= 16;
|
|
177
|
+
break;
|
|
178
|
+
case "drag":
|
|
179
|
+
buttonCode = (buttonsPressed.size > 0 ? Array.from(buttonsPressed)[0] : button) | 32;
|
|
180
|
+
if (modifiers.shift)
|
|
181
|
+
buttonCode |= 4;
|
|
182
|
+
if (modifiers.alt)
|
|
183
|
+
buttonCode |= 8;
|
|
184
|
+
if (modifiers.ctrl)
|
|
185
|
+
buttonCode |= 16;
|
|
186
|
+
break;
|
|
187
|
+
case "scroll":
|
|
188
|
+
break;
|
|
189
|
+
}
|
|
190
|
+
const ansiX = x + 1;
|
|
191
|
+
const ansiY = y + 1;
|
|
192
|
+
let pressRelease = "M";
|
|
193
|
+
if (type === "up" || type === "move" || type === "drag") {
|
|
194
|
+
pressRelease = "m";
|
|
195
|
+
}
|
|
196
|
+
return `\x1B[<${buttonCode};${ansiX};${ansiY}${pressRelease}`;
|
|
197
|
+
};
|
|
198
|
+
const emitMouseEvent = async (type, x, y, button = MouseButtons.LEFT, options = {}) => {
|
|
199
|
+
const { modifiers = {}, delayMs = 0 } = options;
|
|
200
|
+
const eventSequence = generateMouseEvent(type, x, y, button, modifiers);
|
|
201
|
+
renderer.stdin.emit("data", Buffer.from(eventSequence));
|
|
202
|
+
currentPosition = { x, y };
|
|
203
|
+
if (type === "down" && button < 64) {
|
|
204
|
+
buttonsPressed.add(button);
|
|
205
|
+
} else if (type === "up") {
|
|
206
|
+
buttonsPressed.delete(button);
|
|
207
|
+
}
|
|
208
|
+
if (delayMs > 0) {
|
|
209
|
+
await new Promise((resolve) => setTimeout(resolve, delayMs));
|
|
210
|
+
}
|
|
211
|
+
};
|
|
212
|
+
const moveTo = async (x, y, options = {}) => {
|
|
213
|
+
const { button = MouseButtons.LEFT, delayMs = 0, modifiers = {} } = options;
|
|
214
|
+
if (buttonsPressed.size > 0) {
|
|
215
|
+
await emitMouseEvent("drag", x, y, Array.from(buttonsPressed)[0], { modifiers, delayMs });
|
|
216
|
+
} else {
|
|
217
|
+
await emitMouseEvent("move", x, y, button, { modifiers, delayMs });
|
|
218
|
+
}
|
|
219
|
+
currentPosition = { x, y };
|
|
220
|
+
};
|
|
221
|
+
const click = async (x, y, button = MouseButtons.LEFT, options = {}) => {
|
|
222
|
+
const { delayMs = 10, modifiers = {} } = options;
|
|
223
|
+
await emitMouseEvent("down", x, y, button, { modifiers, delayMs });
|
|
224
|
+
await new Promise((resolve) => setTimeout(resolve, delayMs));
|
|
225
|
+
await emitMouseEvent("up", x, y, button, { modifiers, delayMs });
|
|
226
|
+
};
|
|
227
|
+
const doubleClick = async (x, y, button = MouseButtons.LEFT, options = {}) => {
|
|
228
|
+
const { delayMs = 10, modifiers = {} } = options;
|
|
229
|
+
await click(x, y, button, { modifiers, delayMs });
|
|
230
|
+
await new Promise((resolve) => setTimeout(resolve, delayMs));
|
|
231
|
+
await click(x, y, button, { modifiers, delayMs });
|
|
232
|
+
};
|
|
233
|
+
const pressDown = async (x, y, button = MouseButtons.LEFT, options = {}) => {
|
|
234
|
+
const { modifiers = {}, delayMs = 0 } = options;
|
|
235
|
+
await emitMouseEvent("down", x, y, button, { modifiers, delayMs });
|
|
236
|
+
};
|
|
237
|
+
const release = async (x, y, button = MouseButtons.LEFT, options = {}) => {
|
|
238
|
+
const { modifiers = {}, delayMs = 0 } = options;
|
|
239
|
+
await emitMouseEvent("up", x, y, button, { modifiers, delayMs });
|
|
240
|
+
};
|
|
241
|
+
const drag = async (startX, startY, endX, endY, button = MouseButtons.LEFT, options = {}) => {
|
|
242
|
+
const { delayMs = 10, modifiers = {} } = options;
|
|
243
|
+
await pressDown(startX, startY, button, { modifiers });
|
|
244
|
+
const steps = 5;
|
|
245
|
+
const dx = (endX - startX) / steps;
|
|
246
|
+
const dy = (endY - startY) / steps;
|
|
247
|
+
for (let i = 1;i <= steps; i++) {
|
|
248
|
+
const currentX = Math.round(startX + dx * i);
|
|
249
|
+
const currentY = Math.round(startY + dy * i);
|
|
250
|
+
await emitMouseEvent("drag", currentX, currentY, button, { modifiers, delayMs });
|
|
251
|
+
}
|
|
252
|
+
await release(endX, endY, button, { modifiers });
|
|
253
|
+
};
|
|
254
|
+
const scroll = async (x, y, direction, options = {}) => {
|
|
255
|
+
const { modifiers = {}, delayMs = 0 } = options;
|
|
256
|
+
let button;
|
|
257
|
+
switch (direction) {
|
|
258
|
+
case "up":
|
|
259
|
+
button = MouseButtons.WHEEL_UP;
|
|
260
|
+
break;
|
|
261
|
+
case "down":
|
|
262
|
+
button = MouseButtons.WHEEL_DOWN;
|
|
263
|
+
break;
|
|
264
|
+
case "left":
|
|
265
|
+
button = MouseButtons.WHEEL_LEFT;
|
|
266
|
+
break;
|
|
267
|
+
case "right":
|
|
268
|
+
button = MouseButtons.WHEEL_RIGHT;
|
|
269
|
+
break;
|
|
270
|
+
}
|
|
271
|
+
await emitMouseEvent("scroll", x, y, button, { modifiers, delayMs });
|
|
272
|
+
};
|
|
273
|
+
const getCurrentPosition = () => {
|
|
274
|
+
return { ...currentPosition };
|
|
275
|
+
};
|
|
276
|
+
const getPressedButtons = () => {
|
|
277
|
+
return Array.from(buttonsPressed);
|
|
278
|
+
};
|
|
279
|
+
return {
|
|
280
|
+
moveTo,
|
|
281
|
+
click,
|
|
282
|
+
doubleClick,
|
|
283
|
+
pressDown,
|
|
284
|
+
release,
|
|
285
|
+
drag,
|
|
286
|
+
scroll,
|
|
287
|
+
getCurrentPosition,
|
|
288
|
+
getPressedButtons,
|
|
289
|
+
emitMouseEvent
|
|
290
|
+
};
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
// src/testing/test-renderer.ts
|
|
294
|
+
var decoder = new TextDecoder;
|
|
295
|
+
async function createTestRenderer(options) {
|
|
296
|
+
process.env.OTUI_USE_CONSOLE = "false";
|
|
297
|
+
const renderer = await setupTestRenderer({
|
|
298
|
+
...options,
|
|
299
|
+
useAlternateScreen: false,
|
|
300
|
+
useConsole: false
|
|
301
|
+
});
|
|
302
|
+
renderer.disableStdoutInterception();
|
|
303
|
+
const mockInput = createMockKeys(renderer);
|
|
304
|
+
const mockMouse = createMockMouse(renderer);
|
|
305
|
+
const renderOnce = async () => {
|
|
306
|
+
await renderer.loop();
|
|
307
|
+
};
|
|
308
|
+
return {
|
|
309
|
+
renderer,
|
|
310
|
+
mockInput,
|
|
311
|
+
mockMouse,
|
|
312
|
+
renderOnce,
|
|
313
|
+
captureCharFrame: () => {
|
|
314
|
+
const currentBuffer = renderer.currentRenderBuffer;
|
|
315
|
+
const frameBytes = currentBuffer.getRealCharBytes(true);
|
|
316
|
+
return decoder.decode(frameBytes);
|
|
317
|
+
},
|
|
318
|
+
resize: (width, height) => {
|
|
319
|
+
renderer.processResize(width, height);
|
|
320
|
+
}
|
|
321
|
+
};
|
|
322
|
+
}
|
|
323
|
+
async function setupTestRenderer(config) {
|
|
324
|
+
const stdin = config.stdin || process.stdin;
|
|
325
|
+
const stdout = config.stdout || process.stdout;
|
|
326
|
+
const width = config.width || stdout.columns || 80;
|
|
327
|
+
const height = config.height || stdout.rows || 24;
|
|
328
|
+
const renderHeight = config.experimental_splitHeight && config.experimental_splitHeight > 0 ? config.experimental_splitHeight : height;
|
|
329
|
+
const ziglib = resolveRenderLib();
|
|
330
|
+
const rendererPtr = ziglib.createRenderer(width, renderHeight, { testing: true });
|
|
331
|
+
if (!rendererPtr) {
|
|
332
|
+
throw new Error("Failed to create test renderer");
|
|
333
|
+
}
|
|
334
|
+
if (config.useThread === undefined) {
|
|
335
|
+
config.useThread = true;
|
|
336
|
+
}
|
|
337
|
+
if (process.platform === "linux") {
|
|
338
|
+
config.useThread = false;
|
|
339
|
+
}
|
|
340
|
+
ziglib.setUseThread(rendererPtr, config.useThread);
|
|
341
|
+
const renderer = new CliRenderer(ziglib, rendererPtr, stdin, stdout, width, height, config);
|
|
342
|
+
return renderer;
|
|
343
|
+
}
|
|
344
|
+
export {
|
|
345
|
+
createTestRenderer,
|
|
346
|
+
createMockMouse,
|
|
347
|
+
createMockKeys,
|
|
348
|
+
MouseButtons,
|
|
349
|
+
KeyCodes
|
|
350
|
+
};
|
|
351
|
+
|
|
352
|
+
//# debugId=DDD6AB1840C0574664756E2164756E21
|
|
353
|
+
//# sourceMappingURL=testing.js.map
|
package/testing.js.map
ADDED
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../src/testing/mock-keys.ts", "../src/testing/mock-mouse.ts", "../src/testing/test-renderer.ts"],
|
|
4
|
+
"sourcesContent": [
|
|
5
|
+
"import type { CliRenderer } from \"../renderer\"\nimport { ANSI } from \"../ansi\"\n\nexport const KeyCodes = {\n // Control keys\n ENTER: \"\\r\",\n TAB: \"\\t\",\n BACKSPACE: \"\\b\",\n // NOTE: This may depend on the platform and terminals\n DELETE: \"\\x1b[3~\",\n HOME: \"\\x1b[H\",\n END: \"\\x1b[F\",\n ESCAPE: \"\\x1b\",\n\n // Arrow keys\n ARROW_UP: \"\\x1b[A\",\n ARROW_DOWN: \"\\x1b[B\",\n ARROW_RIGHT: \"\\x1b[C\",\n ARROW_LEFT: \"\\x1b[D\",\n\n // Function keys\n F1: \"\\x1bOP\",\n F2: \"\\x1bOQ\",\n F3: \"\\x1bOR\",\n F4: \"\\x1bOS\",\n F5: \"\\x1b[15~\",\n F6: \"\\x1b[17~\",\n F7: \"\\x1b[18~\",\n F8: \"\\x1b[19~\",\n F9: \"\\x1b[20~\",\n F10: \"\\x1b[21~\",\n F11: \"\\x1b[23~\",\n F12: \"\\x1b[24~\",\n\n // Control combinations\n CTRL_A: \"\\x01\",\n CTRL_B: \"\\x02\",\n CTRL_C: \"\\x03\",\n CTRL_D: \"\\x04\",\n CTRL_E: \"\\x05\",\n CTRL_F: \"\\x06\",\n CTRL_G: \"\\x07\",\n CTRL_H: \"\\x08\",\n CTRL_I: \"\\t\",\n CTRL_J: \"\\n\",\n CTRL_K: \"\\x0b\",\n CTRL_L: \"\\x0c\",\n CTRL_M: \"\\r\",\n CTRL_N: \"\\x0e\",\n CTRL_O: \"\\x0f\",\n CTRL_P: \"\\x10\",\n CTRL_Q: \"\\x11\",\n CTRL_R: \"\\x12\",\n CTRL_S: \"\\x13\",\n CTRL_T: \"\\x14\",\n CTRL_U: \"\\x15\",\n CTRL_V: \"\\x16\",\n CTRL_W: \"\\x17\",\n CTRL_X: \"\\x18\",\n CTRL_Y: \"\\x19\",\n CTRL_Z: \"\\x1a\",\n\n // Alt combinations\n ALT_A: \"\\x1ba\",\n ALT_B: \"\\x1bb\",\n ALT_C: \"\\x1bc\",\n // ... add more as needed\n} as const\n\nexport type KeyInput = string | keyof typeof KeyCodes\n\nexport function createMockKeys(renderer: CliRenderer) {\n const pressKeys = async (keys: KeyInput[], delayMs: number = 0): Promise<void> => {\n for (const key of keys) {\n let keyCode: string\n if (typeof key === \"string\") {\n // If it's a string but also exists in KeyCodes, use the KeyCodes value\n if (key in KeyCodes) {\n keyCode = KeyCodes[key as keyof typeof KeyCodes]\n } else {\n keyCode = key\n }\n } else {\n // It's a KeyCode enum value\n keyCode = KeyCodes[key]\n if (!keyCode) {\n throw new Error(`Unknown key: ${key}`)\n }\n }\n\n renderer.stdin.emit(\"data\", Buffer.from(keyCode))\n\n if (delayMs > 0) {\n await new Promise((resolve) => setTimeout(resolve, delayMs))\n }\n }\n }\n\n const pressKey = (key: KeyInput): void => {\n let keyCode: string\n if (typeof key === \"string\") {\n // If it's a string but also exists in KeyCodes, use the KeyCodes value\n if (key in KeyCodes) {\n keyCode = KeyCodes[key as keyof typeof KeyCodes]\n } else {\n keyCode = key\n }\n } else {\n // This branch handles KeyCode enum values (though they're strings at runtime)\n keyCode = KeyCodes[key]\n if (!keyCode) {\n throw new Error(`Unknown key: ${key}`)\n }\n }\n\n renderer.stdin.emit(\"data\", Buffer.from(keyCode))\n }\n\n const typeText = async (text: string, delayMs: number = 0): Promise<void> => {\n const keys = text.split(\"\")\n await pressKeys(keys, delayMs)\n }\n\n const pressEnter = (): void => {\n pressKey(KeyCodes.ENTER)\n }\n\n const pressEscape = (): void => {\n pressKey(KeyCodes.ESCAPE)\n }\n\n const pressTab = (): void => {\n pressKey(KeyCodes.TAB)\n }\n\n const pressBackspace = (): void => {\n pressKey(KeyCodes.BACKSPACE)\n }\n\n const pressArrow = (direction: \"up\" | \"down\" | \"left\" | \"right\"): void => {\n const keyMap = {\n up: KeyCodes.ARROW_UP,\n down: KeyCodes.ARROW_DOWN,\n left: KeyCodes.ARROW_LEFT,\n right: KeyCodes.ARROW_RIGHT,\n }\n pressKey(keyMap[direction])\n }\n\n const pressCtrlC = (): void => {\n pressKey(KeyCodes.CTRL_C)\n }\n\n const pasteBracketedText = (text: string): Promise<void> => {\n return pressKeys([ANSI.bracketedPasteStart, text, ANSI.bracketedPasteEnd])\n }\n\n return {\n pressKeys,\n pressKey,\n typeText,\n pressEnter,\n pressEscape,\n pressTab,\n pressBackspace,\n pressArrow,\n pressCtrlC,\n pasteBracketedText,\n }\n}\n",
|
|
6
|
+
"import type { CliRenderer } from \"../renderer\"\n\nexport const MouseButtons = {\n LEFT: 0,\n MIDDLE: 1,\n RIGHT: 2,\n\n WHEEL_UP: 64, // 64 = scroll flag + 0\n WHEEL_DOWN: 65, // 64 + 1\n WHEEL_LEFT: 66, // 64 + 2\n WHEEL_RIGHT: 67, // 64 + 3\n} as const\n\nexport type MouseButton = (typeof MouseButtons)[keyof typeof MouseButtons]\n\nexport interface MousePosition {\n x: number\n y: number\n}\n\nexport interface MouseModifiers {\n shift?: boolean\n alt?: boolean\n ctrl?: boolean\n}\n\nexport type MouseEventType = \"down\" | \"up\" | \"move\" | \"drag\" | \"scroll\"\n\nexport interface MouseEventOptions {\n button?: MouseButton\n modifiers?: MouseModifiers\n delayMs?: number\n}\n\nexport function createMockMouse(renderer: CliRenderer) {\n let currentPosition: MousePosition = { x: 0, y: 0 }\n let buttonsPressed = new Set<MouseButton>()\n\n // Generate SGR mouse event sequence\n const generateMouseEvent = (\n type: MouseEventType,\n x: number,\n y: number,\n button: MouseButton = MouseButtons.LEFT,\n modifiers: MouseModifiers = {},\n ): string => {\n // SGR format: \\x1b[<b;x;yM or \\x1b[<b;x;ym\n // where b = button code + modifier flags + motion/scroll flags\n\n let buttonCode: number = button\n\n // Add modifier flags\n if (modifiers.shift) buttonCode |= 4\n if (modifiers.alt) buttonCode |= 8\n if (modifiers.ctrl) buttonCode |= 16\n\n switch (type) {\n case \"move\":\n buttonCode = 32 | 3 // motion flag (32) + button 3 for motion without button press\n if (modifiers.shift) buttonCode |= 4\n if (modifiers.alt) buttonCode |= 8\n if (modifiers.ctrl) buttonCode |= 16\n break\n case \"drag\":\n buttonCode = (buttonsPressed.size > 0 ? Array.from(buttonsPressed)[0] : button) | 32\n if (modifiers.shift) buttonCode |= 4\n if (modifiers.alt) buttonCode |= 8\n if (modifiers.ctrl) buttonCode |= 16\n break\n case \"scroll\":\n // Scroll events already have the scroll flag set in the button code\n break\n }\n\n // Convert to 1-based coordinates for ANSI\n const ansiX = x + 1\n const ansiY = y + 1\n\n let pressRelease = \"M\" // Default to press\n if (type === \"up\" || type === \"move\" || type === \"drag\") {\n pressRelease = \"m\"\n }\n\n return `\\x1b[<${buttonCode};${ansiX};${ansiY}${pressRelease}`\n }\n\n const emitMouseEvent = async (\n type: MouseEventType,\n x: number,\n y: number,\n button: MouseButton = MouseButtons.LEFT,\n options: Omit<MouseEventOptions, \"button\"> = {},\n ): Promise<void> => {\n const { modifiers = {}, delayMs = 0 } = options\n\n const eventSequence = generateMouseEvent(type, x, y, button, modifiers)\n renderer.stdin.emit(\"data\", Buffer.from(eventSequence))\n\n currentPosition = { x, y }\n\n if (type === \"down\" && button < 64) {\n buttonsPressed.add(button)\n } else if (type === \"up\") {\n buttonsPressed.delete(button)\n }\n\n if (delayMs > 0) {\n await new Promise((resolve) => setTimeout(resolve, delayMs))\n }\n }\n\n const moveTo = async (x: number, y: number, options: MouseEventOptions = {}): Promise<void> => {\n const { button = MouseButtons.LEFT, delayMs = 0, modifiers = {} } = options\n\n if (buttonsPressed.size > 0) {\n await emitMouseEvent(\"drag\", x, y, Array.from(buttonsPressed)[0], { modifiers, delayMs })\n } else {\n await emitMouseEvent(\"move\", x, y, button, { modifiers, delayMs })\n }\n\n currentPosition = { x, y }\n }\n\n const click = async (\n x: number,\n y: number,\n button: MouseButton = MouseButtons.LEFT,\n options: MouseEventOptions = {},\n ): Promise<void> => {\n const { delayMs = 10, modifiers = {} } = options\n\n await emitMouseEvent(\"down\", x, y, button, { modifiers, delayMs })\n await new Promise((resolve) => setTimeout(resolve, delayMs))\n await emitMouseEvent(\"up\", x, y, button, { modifiers, delayMs })\n }\n\n const doubleClick = async (\n x: number,\n y: number,\n button: MouseButton = MouseButtons.LEFT,\n options: MouseEventOptions = {},\n ): Promise<void> => {\n const { delayMs = 10, modifiers = {} } = options\n\n await click(x, y, button, { modifiers, delayMs })\n await new Promise((resolve) => setTimeout(resolve, delayMs))\n await click(x, y, button, { modifiers, delayMs })\n }\n\n const pressDown = async (\n x: number,\n y: number,\n button: MouseButton = MouseButtons.LEFT,\n options: MouseEventOptions = {},\n ): Promise<void> => {\n const { modifiers = {}, delayMs = 0 } = options\n await emitMouseEvent(\"down\", x, y, button, { modifiers, delayMs })\n }\n\n const release = async (\n x: number,\n y: number,\n button: MouseButton = MouseButtons.LEFT,\n options: MouseEventOptions = {},\n ): Promise<void> => {\n const { modifiers = {}, delayMs = 0 } = options\n await emitMouseEvent(\"up\", x, y, button, { modifiers, delayMs })\n }\n\n const drag = async (\n startX: number,\n startY: number,\n endX: number,\n endY: number,\n button: MouseButton = MouseButtons.LEFT,\n options: MouseEventOptions = {},\n ): Promise<void> => {\n const { delayMs = 10, modifiers = {} } = options\n\n await pressDown(startX, startY, button, { modifiers })\n\n const steps = 5\n const dx = (endX - startX) / steps\n const dy = (endY - startY) / steps\n\n for (let i = 1; i <= steps; i++) {\n const currentX = Math.round(startX + dx * i)\n const currentY = Math.round(startY + dy * i)\n await emitMouseEvent(\"drag\", currentX, currentY, button, { modifiers, delayMs })\n }\n\n await release(endX, endY, button, { modifiers })\n }\n\n const scroll = async (\n x: number,\n y: number,\n direction: \"up\" | \"down\" | \"left\" | \"right\",\n options: MouseEventOptions = {},\n ): Promise<void> => {\n const { modifiers = {}, delayMs = 0 } = options\n\n let button: MouseButton\n switch (direction) {\n case \"up\":\n button = MouseButtons.WHEEL_UP\n break\n case \"down\":\n button = MouseButtons.WHEEL_DOWN\n break\n case \"left\":\n button = MouseButtons.WHEEL_LEFT\n break\n case \"right\":\n button = MouseButtons.WHEEL_RIGHT\n break\n }\n\n await emitMouseEvent(\"scroll\", x, y, button, { modifiers, delayMs })\n }\n\n const getCurrentPosition = (): MousePosition => {\n return { ...currentPosition }\n }\n\n const getPressedButtons = (): MouseButton[] => {\n return Array.from(buttonsPressed)\n }\n\n return {\n // Core interaction methods\n moveTo,\n click,\n doubleClick,\n pressDown,\n release,\n drag,\n scroll,\n\n // State getters\n getCurrentPosition,\n getPressedButtons,\n\n // Low-level event emission (for advanced use cases)\n emitMouseEvent,\n }\n}\n",
|
|
7
|
+
"import { CliRenderer, type CliRendererConfig } from \"../renderer\"\nimport { resolveRenderLib } from \"../zig\"\nimport { createMockKeys } from \"./mock-keys\"\nimport { createMockMouse } from \"./mock-mouse\"\n\nexport interface TestRendererOptions extends CliRendererConfig {\n width?: number\n height?: number\n}\nexport interface TestRenderer extends CliRenderer {}\nexport type MockInput = ReturnType<typeof createMockKeys>\nexport type MockMouse = ReturnType<typeof createMockMouse>\n\nconst decoder = new TextDecoder()\n\nexport async function createTestRenderer(options: TestRendererOptions): Promise<{\n renderer: TestRenderer\n mockInput: MockInput\n mockMouse: MockMouse\n renderOnce: () => Promise<void>\n captureCharFrame: () => string\n resize: (width: number, height: number) => void\n}> {\n process.env.OTUI_USE_CONSOLE = \"false\"\n const renderer = await setupTestRenderer({\n ...options,\n useAlternateScreen: false,\n useConsole: false,\n })\n\n renderer.disableStdoutInterception()\n\n const mockInput = createMockKeys(renderer)\n const mockMouse = createMockMouse(renderer)\n\n const renderOnce = async () => {\n //@ts-expect-error - this is a test renderer\n await renderer.loop()\n }\n\n return {\n renderer,\n mockInput,\n mockMouse,\n renderOnce,\n captureCharFrame: () => {\n const currentBuffer = renderer.currentRenderBuffer\n const frameBytes = currentBuffer.getRealCharBytes(true)\n return decoder.decode(frameBytes)\n },\n resize: (width: number, height: number) => {\n //@ts-expect-error - this is a test renderer\n renderer.processResize(width, height)\n },\n }\n}\n\nasync function setupTestRenderer(config: TestRendererOptions) {\n const stdin = config.stdin || process.stdin\n const stdout = config.stdout || process.stdout\n\n const width = config.width || stdout.columns || 80\n const height = config.height || stdout.rows || 24\n const renderHeight =\n config.experimental_splitHeight && config.experimental_splitHeight > 0 ? config.experimental_splitHeight : height\n\n const ziglib = resolveRenderLib()\n const rendererPtr = ziglib.createRenderer(width, renderHeight, { testing: true })\n if (!rendererPtr) {\n throw new Error(\"Failed to create test renderer\")\n }\n if (config.useThread === undefined) {\n config.useThread = true\n }\n\n if (process.platform === \"linux\") {\n config.useThread = false\n }\n ziglib.setUseThread(rendererPtr, config.useThread)\n\n const renderer = new CliRenderer(ziglib, rendererPtr, stdin, stdout, width, height, config)\n\n // Do not setup the terminal for testing as we will not actualy output anything to the terminal\n // await renderer.setupTerminal()\n\n return renderer\n}\n"
|
|
8
|
+
],
|
|
9
|
+
"mappings": "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",
|
|
10
|
+
"debugId": "DDD6AB1840C0574664756E2164756E21",
|
|
11
|
+
"names": []
|
|
12
|
+
}
|
package/text-buffer.d.ts
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import type { StyledText } from "./lib/styled-text";
|
|
2
2
|
import { RGBA } from "./lib/RGBA";
|
|
3
|
-
import { type RenderLib } from "./zig";
|
|
3
|
+
import { type LineInfo, type RenderLib } from "./zig";
|
|
4
4
|
import { type Pointer } from "bun:ffi";
|
|
5
5
|
import { type WidthMethod } from "./types";
|
|
6
6
|
export interface TextChunk {
|
|
@@ -14,24 +14,21 @@ export declare class TextBuffer {
|
|
|
14
14
|
private lib;
|
|
15
15
|
private bufferPtr;
|
|
16
16
|
private _length;
|
|
17
|
-
private _capacity;
|
|
18
17
|
private _lineInfo?;
|
|
19
|
-
|
|
20
|
-
|
|
18
|
+
private _destroyed;
|
|
19
|
+
constructor(lib: RenderLib, ptr: Pointer);
|
|
20
|
+
static create(widthMethod: WidthMethod): TextBuffer;
|
|
21
|
+
private guard;
|
|
21
22
|
setStyledText(text: StyledText): void;
|
|
22
23
|
setDefaultFg(fg: RGBA | null): void;
|
|
23
24
|
setDefaultBg(bg: RGBA | null): void;
|
|
24
25
|
setDefaultAttributes(attributes: number | null): void;
|
|
25
26
|
resetDefaults(): void;
|
|
26
27
|
get length(): number;
|
|
27
|
-
get capacity(): number;
|
|
28
28
|
get ptr(): Pointer;
|
|
29
29
|
getSelectedText(): string;
|
|
30
30
|
getPlainText(): string;
|
|
31
|
-
get lineInfo():
|
|
32
|
-
lineStarts: number[];
|
|
33
|
-
lineWidths: number[];
|
|
34
|
-
};
|
|
31
|
+
get lineInfo(): LineInfo;
|
|
35
32
|
setSelection(start: number, end: number, bgColor?: RGBA, fgColor?: RGBA): void;
|
|
36
33
|
resetSelection(): void;
|
|
37
34
|
setLocalSelection(anchorX: number, anchorY: number, focusX: number, focusY: number, bgColor?: RGBA, fgColor?: RGBA): boolean;
|
|
@@ -47,5 +44,7 @@ export declare class TextBuffer {
|
|
|
47
44
|
replaceChunkGroup(index: number, text: string, fg?: RGBA, bg?: RGBA, attributes?: number): void;
|
|
48
45
|
replaceEncodedChunkGroup(index: number, textBytes: Uint8Array, fg?: RGBA, bg?: RGBA, attributes?: number): void;
|
|
49
46
|
get chunkGroupCount(): number;
|
|
47
|
+
setWrapWidth(width: number | null): void;
|
|
48
|
+
setWrapMode(mode: "char" | "word"): void;
|
|
50
49
|
destroy(): void;
|
|
51
50
|
}
|
package/types.d.ts
CHANGED
|
@@ -2,6 +2,7 @@ import type { RGBA } from "./lib/RGBA";
|
|
|
2
2
|
import type { EventEmitter } from "events";
|
|
3
3
|
import type { Selection } from "./lib/selection";
|
|
4
4
|
import type { Renderable } from "./Renderable";
|
|
5
|
+
import type { KeyHandler } from "./lib";
|
|
5
6
|
export declare const TextAttributes: {
|
|
6
7
|
NONE: number;
|
|
7
8
|
BOLD: number;
|
|
@@ -49,6 +50,10 @@ export interface RenderContext extends EventEmitter {
|
|
|
49
50
|
requestSelectionUpdate: () => void;
|
|
50
51
|
currentFocusedRenderable: Renderable | null;
|
|
51
52
|
focusRenderable: (renderable: Renderable) => void;
|
|
53
|
+
registerLifecyclePass: (renderable: Renderable) => void;
|
|
54
|
+
unregisterLifecyclePass: (renderable: Renderable) => void;
|
|
55
|
+
getLifecyclePasses: () => Set<Renderable>;
|
|
56
|
+
keyInput: KeyHandler;
|
|
52
57
|
}
|
|
53
58
|
export type Timeout = ReturnType<typeof setTimeout> | undefined;
|
|
54
59
|
export interface ViewportBounds {
|
package/utils.d.ts
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import { Renderable } from "./Renderable";
|
|
1
2
|
export declare function createTextAttributes({ bold, italic, underline, dim, blink, inverse, hidden, strikethrough, }?: {
|
|
2
3
|
bold?: boolean;
|
|
3
4
|
italic?: boolean;
|
|
@@ -8,3 +9,4 @@ export declare function createTextAttributes({ bold, italic, underline, dim, bli
|
|
|
8
9
|
hidden?: boolean;
|
|
9
10
|
strikethrough?: boolean;
|
|
10
11
|
}): number;
|
|
12
|
+
export declare function visualizeRenderableTree(renderable: Renderable, maxDepth?: number): void;
|
package/zig.d.ts
CHANGED
|
@@ -9,9 +9,16 @@ export declare enum LogLevel {
|
|
|
9
9
|
Info = 2,
|
|
10
10
|
Debug = 3
|
|
11
11
|
}
|
|
12
|
+
export interface LineInfo {
|
|
13
|
+
lineStarts: number[];
|
|
14
|
+
lineWidths: number[];
|
|
15
|
+
maxLineWidth: number;
|
|
16
|
+
}
|
|
12
17
|
export interface RenderLib {
|
|
13
|
-
createRenderer: (width: number, height: number
|
|
14
|
-
|
|
18
|
+
createRenderer: (width: number, height: number, options?: {
|
|
19
|
+
testing: boolean;
|
|
20
|
+
}) => Pointer | null;
|
|
21
|
+
destroyRenderer: (renderer: Pointer) => void;
|
|
15
22
|
setUseThread: (renderer: Pointer, useThread: boolean) => void;
|
|
16
23
|
setBackgroundColor: (renderer: Pointer, color: RGBA) => void;
|
|
17
24
|
setRenderOffset: (renderer: Pointer, offset: number) => void;
|
|
@@ -33,6 +40,8 @@ export interface RenderLib {
|
|
|
33
40
|
bufferGetRespectAlpha: (buffer: Pointer) => boolean;
|
|
34
41
|
bufferSetRespectAlpha: (buffer: Pointer, respectAlpha: boolean) => void;
|
|
35
42
|
bufferGetId: (buffer: Pointer) => string;
|
|
43
|
+
bufferGetRealCharSize: (buffer: Pointer) => number;
|
|
44
|
+
bufferWriteResolvedChars: (buffer: Pointer, outputBuffer: Uint8Array, addLineBreaks: boolean) => number;
|
|
36
45
|
bufferDrawText: (buffer: Pointer, text: string, x: number, y: number, color: RGBA, bgColor?: RGBA, attributes?: number) => void;
|
|
37
46
|
bufferSetCellWithAlphaBlending: (buffer: Pointer, x: number, y: number, char: string, color: RGBA, bgColor: RGBA, attributes?: number) => void;
|
|
38
47
|
bufferSetCell: (buffer: Pointer, x: number, y: number, char: string, color: RGBA, bgColor: RGBA, attributes?: number) => void;
|
|
@@ -58,11 +67,10 @@ export interface RenderLib {
|
|
|
58
67
|
enableKittyKeyboard: (renderer: Pointer, flags: number) => void;
|
|
59
68
|
disableKittyKeyboard: (renderer: Pointer) => void;
|
|
60
69
|
setupTerminal: (renderer: Pointer, useAlternateScreen: boolean) => void;
|
|
61
|
-
|
|
70
|
+
queryPixelResolution: (renderer: Pointer) => void;
|
|
71
|
+
createTextBuffer: (widthMethod: WidthMethod) => TextBuffer;
|
|
62
72
|
destroyTextBuffer: (buffer: Pointer) => void;
|
|
63
|
-
textBufferGetCharPtr: (buffer: Pointer) => Pointer;
|
|
64
73
|
textBufferGetLength: (buffer: Pointer) => number;
|
|
65
|
-
textBufferResize: (buffer: Pointer, newLength: number) => void;
|
|
66
74
|
textBufferReset: (buffer: Pointer) => void;
|
|
67
75
|
textBufferSetSelection: (buffer: Pointer, start: number, end: number, bgColor: RGBA | null, fgColor: RGBA | null) => void;
|
|
68
76
|
textBufferResetSelection: (buffer: Pointer) => void;
|
|
@@ -73,14 +81,10 @@ export interface RenderLib {
|
|
|
73
81
|
textBufferSetDefaultAttributes: (buffer: Pointer, attributes: number | null) => void;
|
|
74
82
|
textBufferResetDefaults: (buffer: Pointer) => void;
|
|
75
83
|
textBufferWriteChunk: (buffer: Pointer, textBytes: Uint8Array, fg: RGBA | null, bg: RGBA | null, attributes: number | null) => number;
|
|
76
|
-
textBufferGetCapacity: (buffer: Pointer) => number;
|
|
77
84
|
textBufferFinalizeLineInfo: (buffer: Pointer) => void;
|
|
78
85
|
textBufferGetLineCount: (buffer: Pointer) => number;
|
|
79
86
|
textBufferGetLineInfoDirect: (buffer: Pointer, lineStartsPtr: Pointer, lineWidthsPtr: Pointer) => void;
|
|
80
|
-
textBufferGetLineInfo: (buffer: Pointer) =>
|
|
81
|
-
lineStarts: number[];
|
|
82
|
-
lineWidths: number[];
|
|
83
|
-
};
|
|
87
|
+
textBufferGetLineInfo: (buffer: Pointer) => LineInfo;
|
|
84
88
|
textBufferGetSelection: (buffer: Pointer) => {
|
|
85
89
|
start: number;
|
|
86
90
|
end: number;
|
|
@@ -102,6 +106,8 @@ export interface RenderLib {
|
|
|
102
106
|
textBufferRemoveChunkGroup: (buffer: Pointer, index: number) => number;
|
|
103
107
|
textBufferReplaceChunkGroup: (buffer: Pointer, index: number, textBytes: Uint8Array, fg: RGBA | null, bg: RGBA | null, attributes: number | null) => number;
|
|
104
108
|
textBufferGetChunkGroupCount: (buffer: Pointer) => number;
|
|
109
|
+
textBufferSetWrapWidth: (buffer: Pointer, width: number) => void;
|
|
110
|
+
textBufferSetWrapMode: (buffer: Pointer, mode: "char" | "word") => void;
|
|
105
111
|
getArenaAllocatedBytes: () => number;
|
|
106
112
|
getTerminalCapabilities: (renderer: Pointer) => any;
|
|
107
113
|
processCapabilityResponse: (renderer: Pointer, response: string) => void;
|