@vsuryav/agent-sim 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.
- package/README.md +25 -0
- package/bin/agent-sim.js +25 -0
- package/package.json +72 -0
- package/src/app-paths.ts +29 -0
- package/src/app-sync.test.ts +75 -0
- package/src/app-sync.ts +110 -0
- package/src/cli.ts +129 -0
- package/src/collector/claude-code.test.ts +102 -0
- package/src/collector/claude-code.ts +133 -0
- package/src/collector/codex-cli.test.ts +116 -0
- package/src/collector/codex-cli.ts +149 -0
- package/src/collector/db.test.ts +59 -0
- package/src/collector/db.ts +125 -0
- package/src/collector/names.test.ts +21 -0
- package/src/collector/names.ts +28 -0
- package/src/collector/personality.test.ts +40 -0
- package/src/collector/personality.ts +46 -0
- package/src/collector/remote-sync.test.ts +31 -0
- package/src/collector/remote-sync.ts +171 -0
- package/src/collector/sync.test.ts +67 -0
- package/src/collector/sync.ts +148 -0
- package/src/collector/types.ts +1 -0
- package/src/engine/bootstrap/state.ts +3 -0
- package/src/engine/buddy/CompanionSprite.tsx +371 -0
- package/src/engine/buddy/companion.ts +133 -0
- package/src/engine/buddy/prompt.ts +36 -0
- package/src/engine/buddy/sprites.ts +514 -0
- package/src/engine/buddy/types.ts +148 -0
- package/src/engine/buddy/useBuddyNotification.tsx +98 -0
- package/src/engine/ink/Ansi.tsx +292 -0
- package/src/engine/ink/bidi.ts +139 -0
- package/src/engine/ink/clearTerminal.ts +74 -0
- package/src/engine/ink/colorize.ts +231 -0
- package/src/engine/ink/components/AlternateScreen.tsx +80 -0
- package/src/engine/ink/components/App.tsx +658 -0
- package/src/engine/ink/components/AppContext.ts +21 -0
- package/src/engine/ink/components/Box.tsx +214 -0
- package/src/engine/ink/components/Button.tsx +192 -0
- package/src/engine/ink/components/ClockContext.tsx +112 -0
- package/src/engine/ink/components/CursorDeclarationContext.ts +32 -0
- package/src/engine/ink/components/ErrorOverview.tsx +109 -0
- package/src/engine/ink/components/Link.tsx +42 -0
- package/src/engine/ink/components/Newline.tsx +39 -0
- package/src/engine/ink/components/NoSelect.tsx +68 -0
- package/src/engine/ink/components/RawAnsi.tsx +57 -0
- package/src/engine/ink/components/ScrollBox.tsx +237 -0
- package/src/engine/ink/components/Spacer.tsx +20 -0
- package/src/engine/ink/components/StdinContext.ts +49 -0
- package/src/engine/ink/components/TerminalFocusContext.tsx +52 -0
- package/src/engine/ink/components/TerminalSizeContext.tsx +7 -0
- package/src/engine/ink/components/Text.tsx +254 -0
- package/src/engine/ink/constants.ts +2 -0
- package/src/engine/ink/dom.ts +484 -0
- package/src/engine/ink/events/click-event.ts +38 -0
- package/src/engine/ink/events/dispatcher.ts +233 -0
- package/src/engine/ink/events/emitter.ts +39 -0
- package/src/engine/ink/events/event-handlers.ts +73 -0
- package/src/engine/ink/events/event.ts +11 -0
- package/src/engine/ink/events/focus-event.ts +21 -0
- package/src/engine/ink/events/input-event.ts +205 -0
- package/src/engine/ink/events/keyboard-event.ts +51 -0
- package/src/engine/ink/events/terminal-event.ts +107 -0
- package/src/engine/ink/events/terminal-focus-event.ts +19 -0
- package/src/engine/ink/focus.ts +181 -0
- package/src/engine/ink/frame.ts +124 -0
- package/src/engine/ink/get-max-width.ts +27 -0
- package/src/engine/ink/global.d.ts +18 -0
- package/src/engine/ink/hit-test.ts +130 -0
- package/src/engine/ink/hooks/use-animation-frame.ts +57 -0
- package/src/engine/ink/hooks/use-app.ts +8 -0
- package/src/engine/ink/hooks/use-declared-cursor.ts +73 -0
- package/src/engine/ink/hooks/use-input.ts +92 -0
- package/src/engine/ink/hooks/use-interval.ts +67 -0
- package/src/engine/ink/hooks/use-search-highlight.ts +53 -0
- package/src/engine/ink/hooks/use-selection.ts +104 -0
- package/src/engine/ink/hooks/use-stdin.ts +8 -0
- package/src/engine/ink/hooks/use-tab-status.ts +72 -0
- package/src/engine/ink/hooks/use-terminal-focus.ts +16 -0
- package/src/engine/ink/hooks/use-terminal-title.ts +31 -0
- package/src/engine/ink/hooks/use-terminal-viewport.ts +96 -0
- package/src/engine/ink/ink.tsx +1723 -0
- package/src/engine/ink/instances.ts +10 -0
- package/src/engine/ink/layout/engine.ts +6 -0
- package/src/engine/ink/layout/geometry.ts +97 -0
- package/src/engine/ink/layout/node.ts +152 -0
- package/src/engine/ink/layout/yoga.ts +308 -0
- package/src/engine/ink/line-width-cache.ts +24 -0
- package/src/engine/ink/log-update.ts +773 -0
- package/src/engine/ink/measure-element.ts +23 -0
- package/src/engine/ink/measure-text.ts +47 -0
- package/src/engine/ink/node-cache.ts +54 -0
- package/src/engine/ink/optimizer.ts +93 -0
- package/src/engine/ink/output.ts +797 -0
- package/src/engine/ink/parse-keypress.ts +801 -0
- package/src/engine/ink/reconciler.ts +512 -0
- package/src/engine/ink/render-border.ts +231 -0
- package/src/engine/ink/render-node-to-output.ts +1462 -0
- package/src/engine/ink/render-to-screen.ts +231 -0
- package/src/engine/ink/renderer.ts +178 -0
- package/src/engine/ink/root.ts +184 -0
- package/src/engine/ink/screen.ts +1486 -0
- package/src/engine/ink/searchHighlight.ts +93 -0
- package/src/engine/ink/selection.ts +917 -0
- package/src/engine/ink/squash-text-nodes.ts +92 -0
- package/src/engine/ink/stringWidth.ts +222 -0
- package/src/engine/ink/styles.ts +771 -0
- package/src/engine/ink/supports-hyperlinks.ts +57 -0
- package/src/engine/ink/tabstops.ts +46 -0
- package/src/engine/ink/terminal-focus-state.ts +47 -0
- package/src/engine/ink/terminal-querier.ts +212 -0
- package/src/engine/ink/terminal.ts +248 -0
- package/src/engine/ink/termio/ansi.ts +75 -0
- package/src/engine/ink/termio/csi.ts +319 -0
- package/src/engine/ink/termio/dec.ts +60 -0
- package/src/engine/ink/termio/esc.ts +67 -0
- package/src/engine/ink/termio/osc.ts +493 -0
- package/src/engine/ink/termio/parser.ts +394 -0
- package/src/engine/ink/termio/sgr.ts +308 -0
- package/src/engine/ink/termio/tokenize.ts +319 -0
- package/src/engine/ink/termio/types.ts +236 -0
- package/src/engine/ink/useTerminalNotification.ts +126 -0
- package/src/engine/ink/warn.ts +9 -0
- package/src/engine/ink/widest-line.ts +19 -0
- package/src/engine/ink/wrap-text.ts +74 -0
- package/src/engine/ink/wrapAnsi.ts +20 -0
- package/src/engine/native-ts/yoga-layout/enums.ts +134 -0
- package/src/engine/native-ts/yoga-layout/index.ts +2578 -0
- package/src/engine/stubs/bootstrap-state.ts +4 -0
- package/src/engine/stubs/debug.ts +6 -0
- package/src/engine/stubs/log.ts +4 -0
- package/src/engine/utils/debug.ts +5 -0
- package/src/engine/utils/earlyInput.ts +4 -0
- package/src/engine/utils/env.ts +15 -0
- package/src/engine/utils/envUtils.ts +4 -0
- package/src/engine/utils/execFileNoThrow.ts +24 -0
- package/src/engine/utils/fullscreen.ts +4 -0
- package/src/engine/utils/intl.ts +9 -0
- package/src/engine/utils/log.ts +3 -0
- package/src/engine/utils/semver.ts +13 -0
- package/src/engine/utils/sliceAnsi.ts +10 -0
- package/src/engine/utils/theme.ts +17 -0
- package/src/game/App.tsx +141 -0
- package/src/game/agents/behavior.ts +249 -0
- package/src/game/agents/speech.ts +57 -0
- package/src/game/canvas.ts +98 -0
- package/src/game/launch.ts +36 -0
- package/src/game/ship/ShipView.tsx +145 -0
- package/src/game/ship/ship-map.ts +172 -0
- package/src/game/ui/AgentBio.tsx +72 -0
- package/src/game/ui/HUD.tsx +63 -0
- package/src/game/ui/StatusBar.tsx +49 -0
- package/src/game/useKeyboard.ts +62 -0
- package/src/main.tsx +22 -0
- package/src/run-interactive.ts +74 -0
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import { createContext } from 'react'
|
|
2
|
+
|
|
3
|
+
export type Props = {
|
|
4
|
+
/**
|
|
5
|
+
* Exit (unmount) the whole Ink app.
|
|
6
|
+
*/
|
|
7
|
+
readonly exit: (error?: Error) => void
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* `AppContext` is a React context, which exposes a method to manually exit the app (unmount).
|
|
12
|
+
*/
|
|
13
|
+
// eslint-disable-next-line @typescript-eslint/naming-convention
|
|
14
|
+
const AppContext = createContext<Props>({
|
|
15
|
+
exit() {},
|
|
16
|
+
})
|
|
17
|
+
|
|
18
|
+
// eslint-disable-next-line custom-rules/no-top-level-side-effects
|
|
19
|
+
AppContext.displayName = 'InternalAppContext'
|
|
20
|
+
|
|
21
|
+
export default AppContext
|
|
@@ -0,0 +1,214 @@
|
|
|
1
|
+
import { c as _c } from "react/compiler-runtime";
|
|
2
|
+
import '../global.d.ts';
|
|
3
|
+
import React, { type PropsWithChildren, type Ref } from 'react';
|
|
4
|
+
import type { Except } from 'type-fest';
|
|
5
|
+
import type { DOMElement } from '../dom.js';
|
|
6
|
+
import type { ClickEvent } from '../events/click-event.js';
|
|
7
|
+
import type { FocusEvent } from '../events/focus-event.js';
|
|
8
|
+
import type { KeyboardEvent } from '../events/keyboard-event.js';
|
|
9
|
+
import type { Styles } from '../styles.js';
|
|
10
|
+
import * as warn from '../warn.js';
|
|
11
|
+
export type Props = Except<Styles, 'textWrap'> & {
|
|
12
|
+
ref?: Ref<DOMElement>;
|
|
13
|
+
/**
|
|
14
|
+
* Tab order index. Nodes with `tabIndex >= 0` participate in
|
|
15
|
+
* Tab/Shift+Tab cycling; `-1` means programmatically focusable only.
|
|
16
|
+
*/
|
|
17
|
+
tabIndex?: number;
|
|
18
|
+
/**
|
|
19
|
+
* Focus this element when it mounts. Like the HTML `autofocus`
|
|
20
|
+
* attribute — the FocusManager calls `focus(node)` during the
|
|
21
|
+
* reconciler's `commitMount` phase.
|
|
22
|
+
*/
|
|
23
|
+
autoFocus?: boolean;
|
|
24
|
+
/**
|
|
25
|
+
* Fired on left-button click (press + release without drag). Only works
|
|
26
|
+
* inside `<AlternateScreen>` where mouse tracking is enabled — no-op
|
|
27
|
+
* otherwise. The event bubbles from the deepest hit Box up through
|
|
28
|
+
* ancestors; call `event.stopImmediatePropagation()` to stop bubbling.
|
|
29
|
+
*/
|
|
30
|
+
onClick?: (event: ClickEvent) => void;
|
|
31
|
+
onFocus?: (event: FocusEvent) => void;
|
|
32
|
+
onFocusCapture?: (event: FocusEvent) => void;
|
|
33
|
+
onBlur?: (event: FocusEvent) => void;
|
|
34
|
+
onBlurCapture?: (event: FocusEvent) => void;
|
|
35
|
+
onKeyDown?: (event: KeyboardEvent) => void;
|
|
36
|
+
onKeyDownCapture?: (event: KeyboardEvent) => void;
|
|
37
|
+
/**
|
|
38
|
+
* Fired when the mouse moves into this Box's rendered rect. Like DOM
|
|
39
|
+
* `mouseenter`, does NOT bubble — moving between children does not
|
|
40
|
+
* re-fire on the parent. Only works inside `<AlternateScreen>` where
|
|
41
|
+
* mode-1003 mouse tracking is enabled.
|
|
42
|
+
*/
|
|
43
|
+
onMouseEnter?: () => void;
|
|
44
|
+
/** Fired when the mouse moves out of this Box's rendered rect. */
|
|
45
|
+
onMouseLeave?: () => void;
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
/**
|
|
49
|
+
* `<Box>` is an essential Ink component to build your layout. It's like `<div style="display: flex">` in the browser.
|
|
50
|
+
*/
|
|
51
|
+
function Box(t0) {
|
|
52
|
+
const $ = _c(42);
|
|
53
|
+
let autoFocus;
|
|
54
|
+
let children;
|
|
55
|
+
let flexDirection;
|
|
56
|
+
let flexGrow;
|
|
57
|
+
let flexShrink;
|
|
58
|
+
let flexWrap;
|
|
59
|
+
let onBlur;
|
|
60
|
+
let onBlurCapture;
|
|
61
|
+
let onClick;
|
|
62
|
+
let onFocus;
|
|
63
|
+
let onFocusCapture;
|
|
64
|
+
let onKeyDown;
|
|
65
|
+
let onKeyDownCapture;
|
|
66
|
+
let onMouseEnter;
|
|
67
|
+
let onMouseLeave;
|
|
68
|
+
let ref;
|
|
69
|
+
let style;
|
|
70
|
+
let tabIndex;
|
|
71
|
+
if ($[0] !== t0) {
|
|
72
|
+
const {
|
|
73
|
+
children: t1,
|
|
74
|
+
flexWrap: t2,
|
|
75
|
+
flexDirection: t3,
|
|
76
|
+
flexGrow: t4,
|
|
77
|
+
flexShrink: t5,
|
|
78
|
+
ref: t6,
|
|
79
|
+
tabIndex: t7,
|
|
80
|
+
autoFocus: t8,
|
|
81
|
+
onClick: t9,
|
|
82
|
+
onFocus: t10,
|
|
83
|
+
onFocusCapture: t11,
|
|
84
|
+
onBlur: t12,
|
|
85
|
+
onBlurCapture: t13,
|
|
86
|
+
onMouseEnter: t14,
|
|
87
|
+
onMouseLeave: t15,
|
|
88
|
+
onKeyDown: t16,
|
|
89
|
+
onKeyDownCapture: t17,
|
|
90
|
+
...t18
|
|
91
|
+
} = t0;
|
|
92
|
+
children = t1;
|
|
93
|
+
ref = t6;
|
|
94
|
+
tabIndex = t7;
|
|
95
|
+
autoFocus = t8;
|
|
96
|
+
onClick = t9;
|
|
97
|
+
onFocus = t10;
|
|
98
|
+
onFocusCapture = t11;
|
|
99
|
+
onBlur = t12;
|
|
100
|
+
onBlurCapture = t13;
|
|
101
|
+
onMouseEnter = t14;
|
|
102
|
+
onMouseLeave = t15;
|
|
103
|
+
onKeyDown = t16;
|
|
104
|
+
onKeyDownCapture = t17;
|
|
105
|
+
style = t18;
|
|
106
|
+
flexWrap = t2 === undefined ? "nowrap" : t2;
|
|
107
|
+
flexDirection = t3 === undefined ? "row" : t3;
|
|
108
|
+
flexGrow = t4 === undefined ? 0 : t4;
|
|
109
|
+
flexShrink = t5 === undefined ? 1 : t5;
|
|
110
|
+
warn.ifNotInteger(style.margin, "margin");
|
|
111
|
+
warn.ifNotInteger(style.marginX, "marginX");
|
|
112
|
+
warn.ifNotInteger(style.marginY, "marginY");
|
|
113
|
+
warn.ifNotInteger(style.marginTop, "marginTop");
|
|
114
|
+
warn.ifNotInteger(style.marginBottom, "marginBottom");
|
|
115
|
+
warn.ifNotInteger(style.marginLeft, "marginLeft");
|
|
116
|
+
warn.ifNotInteger(style.marginRight, "marginRight");
|
|
117
|
+
warn.ifNotInteger(style.padding, "padding");
|
|
118
|
+
warn.ifNotInteger(style.paddingX, "paddingX");
|
|
119
|
+
warn.ifNotInteger(style.paddingY, "paddingY");
|
|
120
|
+
warn.ifNotInteger(style.paddingTop, "paddingTop");
|
|
121
|
+
warn.ifNotInteger(style.paddingBottom, "paddingBottom");
|
|
122
|
+
warn.ifNotInteger(style.paddingLeft, "paddingLeft");
|
|
123
|
+
warn.ifNotInteger(style.paddingRight, "paddingRight");
|
|
124
|
+
warn.ifNotInteger(style.gap, "gap");
|
|
125
|
+
warn.ifNotInteger(style.columnGap, "columnGap");
|
|
126
|
+
warn.ifNotInteger(style.rowGap, "rowGap");
|
|
127
|
+
$[0] = t0;
|
|
128
|
+
$[1] = autoFocus;
|
|
129
|
+
$[2] = children;
|
|
130
|
+
$[3] = flexDirection;
|
|
131
|
+
$[4] = flexGrow;
|
|
132
|
+
$[5] = flexShrink;
|
|
133
|
+
$[6] = flexWrap;
|
|
134
|
+
$[7] = onBlur;
|
|
135
|
+
$[8] = onBlurCapture;
|
|
136
|
+
$[9] = onClick;
|
|
137
|
+
$[10] = onFocus;
|
|
138
|
+
$[11] = onFocusCapture;
|
|
139
|
+
$[12] = onKeyDown;
|
|
140
|
+
$[13] = onKeyDownCapture;
|
|
141
|
+
$[14] = onMouseEnter;
|
|
142
|
+
$[15] = onMouseLeave;
|
|
143
|
+
$[16] = ref;
|
|
144
|
+
$[17] = style;
|
|
145
|
+
$[18] = tabIndex;
|
|
146
|
+
} else {
|
|
147
|
+
autoFocus = $[1];
|
|
148
|
+
children = $[2];
|
|
149
|
+
flexDirection = $[3];
|
|
150
|
+
flexGrow = $[4];
|
|
151
|
+
flexShrink = $[5];
|
|
152
|
+
flexWrap = $[6];
|
|
153
|
+
onBlur = $[7];
|
|
154
|
+
onBlurCapture = $[8];
|
|
155
|
+
onClick = $[9];
|
|
156
|
+
onFocus = $[10];
|
|
157
|
+
onFocusCapture = $[11];
|
|
158
|
+
onKeyDown = $[12];
|
|
159
|
+
onKeyDownCapture = $[13];
|
|
160
|
+
onMouseEnter = $[14];
|
|
161
|
+
onMouseLeave = $[15];
|
|
162
|
+
ref = $[16];
|
|
163
|
+
style = $[17];
|
|
164
|
+
tabIndex = $[18];
|
|
165
|
+
}
|
|
166
|
+
const t1 = style.overflowX ?? style.overflow ?? "visible";
|
|
167
|
+
const t2 = style.overflowY ?? style.overflow ?? "visible";
|
|
168
|
+
let t3;
|
|
169
|
+
if ($[19] !== flexDirection || $[20] !== flexGrow || $[21] !== flexShrink || $[22] !== flexWrap || $[23] !== style || $[24] !== t1 || $[25] !== t2) {
|
|
170
|
+
t3 = {
|
|
171
|
+
flexWrap,
|
|
172
|
+
flexDirection,
|
|
173
|
+
flexGrow,
|
|
174
|
+
flexShrink,
|
|
175
|
+
...style,
|
|
176
|
+
overflowX: t1,
|
|
177
|
+
overflowY: t2
|
|
178
|
+
};
|
|
179
|
+
$[19] = flexDirection;
|
|
180
|
+
$[20] = flexGrow;
|
|
181
|
+
$[21] = flexShrink;
|
|
182
|
+
$[22] = flexWrap;
|
|
183
|
+
$[23] = style;
|
|
184
|
+
$[24] = t1;
|
|
185
|
+
$[25] = t2;
|
|
186
|
+
$[26] = t3;
|
|
187
|
+
} else {
|
|
188
|
+
t3 = $[26];
|
|
189
|
+
}
|
|
190
|
+
let t4;
|
|
191
|
+
if ($[27] !== autoFocus || $[28] !== children || $[29] !== onBlur || $[30] !== onBlurCapture || $[31] !== onClick || $[32] !== onFocus || $[33] !== onFocusCapture || $[34] !== onKeyDown || $[35] !== onKeyDownCapture || $[36] !== onMouseEnter || $[37] !== onMouseLeave || $[38] !== ref || $[39] !== t3 || $[40] !== tabIndex) {
|
|
192
|
+
t4 = <ink-box ref={ref} tabIndex={tabIndex} autoFocus={autoFocus} onClick={onClick} onFocus={onFocus} onFocusCapture={onFocusCapture} onBlur={onBlur} onBlurCapture={onBlurCapture} onMouseEnter={onMouseEnter} onMouseLeave={onMouseLeave} onKeyDown={onKeyDown} onKeyDownCapture={onKeyDownCapture} style={t3}>{children}</ink-box>;
|
|
193
|
+
$[27] = autoFocus;
|
|
194
|
+
$[28] = children;
|
|
195
|
+
$[29] = onBlur;
|
|
196
|
+
$[30] = onBlurCapture;
|
|
197
|
+
$[31] = onClick;
|
|
198
|
+
$[32] = onFocus;
|
|
199
|
+
$[33] = onFocusCapture;
|
|
200
|
+
$[34] = onKeyDown;
|
|
201
|
+
$[35] = onKeyDownCapture;
|
|
202
|
+
$[36] = onMouseEnter;
|
|
203
|
+
$[37] = onMouseLeave;
|
|
204
|
+
$[38] = ref;
|
|
205
|
+
$[39] = t3;
|
|
206
|
+
$[40] = tabIndex;
|
|
207
|
+
$[41] = t4;
|
|
208
|
+
} else {
|
|
209
|
+
t4 = $[41];
|
|
210
|
+
}
|
|
211
|
+
return t4;
|
|
212
|
+
}
|
|
213
|
+
export default Box;
|
|
214
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","PropsWithChildren","Ref","Except","DOMElement","ClickEvent","FocusEvent","KeyboardEvent","Styles","warn","Props","ref","tabIndex","autoFocus","onClick","event","onFocus","onFocusCapture","onBlur","onBlurCapture","onKeyDown","onKeyDownCapture","onMouseEnter","onMouseLeave","Box","t0","$","_c","children","flexDirection","flexGrow","flexShrink","flexWrap","style","t1","t2","t3","t4","t5","t6","t7","t8","t9","t10","t11","t12","t13","t14","t15","t16","t17","t18","undefined","ifNotInteger","margin","marginX","marginY","marginTop","marginBottom","marginLeft","marginRight","padding","paddingX","paddingY","paddingTop","paddingBottom","paddingLeft","paddingRight","gap","columnGap","rowGap","overflowX","overflow","overflowY"],"sources":["Box.tsx"],"sourcesContent":["import '../global.d.ts'\nimport React, { type PropsWithChildren, type Ref } from 'react'\nimport type { Except } from 'type-fest'\nimport type { DOMElement } from '../dom.js'\nimport type { ClickEvent } from '../events/click-event.js'\nimport type { FocusEvent } from '../events/focus-event.js'\nimport type { KeyboardEvent } from '../events/keyboard-event.js'\nimport type { Styles } from '../styles.js'\nimport * as warn from '../warn.js'\n\nexport type Props = Except<Styles, 'textWrap'> & {\n  ref?: Ref<DOMElement>\n  /**\n   * Tab order index. Nodes with `tabIndex >= 0` participate in\n   * Tab/Shift+Tab cycling; `-1` means programmatically focusable only.\n   */\n  tabIndex?: number\n  /**\n   * Focus this element when it mounts. Like the HTML `autofocus`\n   * attribute — the FocusManager calls `focus(node)` during the\n   * reconciler's `commitMount` phase.\n   */\n  autoFocus?: boolean\n  /**\n   * Fired on left-button click (press + release without drag). Only works\n   * inside `<AlternateScreen>` where mouse tracking is enabled — no-op\n   * otherwise. The event bubbles from the deepest hit Box up through\n   * ancestors; call `event.stopImmediatePropagation()` to stop bubbling.\n   */\n  onClick?: (event: ClickEvent) => void\n  onFocus?: (event: FocusEvent) => void\n  onFocusCapture?: (event: FocusEvent) => void\n  onBlur?: (event: FocusEvent) => void\n  onBlurCapture?: (event: FocusEvent) => void\n  onKeyDown?: (event: KeyboardEvent) => void\n  onKeyDownCapture?: (event: KeyboardEvent) => void\n  /**\n   * Fired when the mouse moves into this Box's rendered rect. Like DOM\n   * `mouseenter`, does NOT bubble — moving between children does not\n   * re-fire on the parent. Only works inside `<AlternateScreen>` where\n   * mode-1003 mouse tracking is enabled.\n   */\n  onMouseEnter?: () => void\n  /** Fired when the mouse moves out of this Box's rendered rect. */\n  onMouseLeave?: () => void\n}\n\n/**\n * `<Box>` is an essential Ink component to build your layout. It's like `<div style=\"display: flex\">` in the browser.\n */\nfunction Box({\n  children,\n  flexWrap = 'nowrap',\n  flexDirection = 'row',\n  flexGrow = 0,\n  flexShrink = 1,\n  ref,\n  tabIndex,\n  autoFocus,\n  onClick,\n  onFocus,\n  onFocusCapture,\n  onBlur,\n  onBlurCapture,\n  onMouseEnter,\n  onMouseLeave,\n  onKeyDown,\n  onKeyDownCapture,\n  ...style\n}: PropsWithChildren<Props>): React.ReactNode {\n  // Warn if spacing values are not integers to prevent fractional layout dimensions\n  warn.ifNotInteger(style.margin, 'margin')\n  warn.ifNotInteger(style.marginX, 'marginX')\n  warn.ifNotInteger(style.marginY, 'marginY')\n  warn.ifNotInteger(style.marginTop, 'marginTop')\n  warn.ifNotInteger(style.marginBottom, 'marginBottom')\n  warn.ifNotInteger(style.marginLeft, 'marginLeft')\n  warn.ifNotInteger(style.marginRight, 'marginRight')\n  warn.ifNotInteger(style.padding, 'padding')\n  warn.ifNotInteger(style.paddingX, 'paddingX')\n  warn.ifNotInteger(style.paddingY, 'paddingY')\n  warn.ifNotInteger(style.paddingTop, 'paddingTop')\n  warn.ifNotInteger(style.paddingBottom, 'paddingBottom')\n  warn.ifNotInteger(style.paddingLeft, 'paddingLeft')\n  warn.ifNotInteger(style.paddingRight, 'paddingRight')\n  warn.ifNotInteger(style.gap, 'gap')\n  warn.ifNotInteger(style.columnGap, 'columnGap')\n  warn.ifNotInteger(style.rowGap, 'rowGap')\n\n  return (\n    <ink-box\n      ref={ref}\n      tabIndex={tabIndex}\n      autoFocus={autoFocus}\n      onClick={onClick}\n      onFocus={onFocus}\n      onFocusCapture={onFocusCapture}\n      onBlur={onBlur}\n      onBlurCapture={onBlurCapture}\n      onMouseEnter={onMouseEnter}\n      onMouseLeave={onMouseLeave}\n      onKeyDown={onKeyDown}\n      onKeyDownCapture={onKeyDownCapture}\n      style={{\n        flexWrap,\n        flexDirection,\n        flexGrow,\n        flexShrink,\n        ...style,\n        overflowX: style.overflowX ?? style.overflow ?? 'visible',\n        overflowY: style.overflowY ?? style.overflow ?? 'visible',\n      }}\n    >\n      {children}\n    </ink-box>\n  )\n}\n\nexport default Box\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -0,0 +1,192 @@
|
|
|
1
|
+
import { c as _c } from "react/compiler-runtime";
|
|
2
|
+
import React, { type Ref, useCallback, useEffect, useRef, useState } from 'react';
|
|
3
|
+
import type { Except } from 'type-fest';
|
|
4
|
+
import type { DOMElement } from '../dom.js';
|
|
5
|
+
import type { ClickEvent } from '../events/click-event.js';
|
|
6
|
+
import type { FocusEvent } from '../events/focus-event.js';
|
|
7
|
+
import type { KeyboardEvent } from '../events/keyboard-event.js';
|
|
8
|
+
import type { Styles } from '../styles.js';
|
|
9
|
+
import Box from './Box.js';
|
|
10
|
+
type ButtonState = {
|
|
11
|
+
focused: boolean;
|
|
12
|
+
hovered: boolean;
|
|
13
|
+
active: boolean;
|
|
14
|
+
};
|
|
15
|
+
export type Props = Except<Styles, 'textWrap'> & {
|
|
16
|
+
ref?: Ref<DOMElement>;
|
|
17
|
+
/**
|
|
18
|
+
* Called when the button is activated via Enter, Space, or click.
|
|
19
|
+
*/
|
|
20
|
+
onAction: () => void;
|
|
21
|
+
/**
|
|
22
|
+
* Tab order index. Defaults to 0 (in tab order).
|
|
23
|
+
* Set to -1 for programmatically focusable only.
|
|
24
|
+
*/
|
|
25
|
+
tabIndex?: number;
|
|
26
|
+
/**
|
|
27
|
+
* Focus this button when it mounts.
|
|
28
|
+
*/
|
|
29
|
+
autoFocus?: boolean;
|
|
30
|
+
/**
|
|
31
|
+
* Render prop receiving the interactive state. Use this to
|
|
32
|
+
* style children based on focus/hover/active — Button itself
|
|
33
|
+
* is intentionally unstyled.
|
|
34
|
+
*
|
|
35
|
+
* If not provided, children render as-is (no state-dependent styling).
|
|
36
|
+
*/
|
|
37
|
+
children: ((state: ButtonState) => React.ReactNode) | React.ReactNode;
|
|
38
|
+
};
|
|
39
|
+
function Button(t0) {
|
|
40
|
+
const $ = _c(30);
|
|
41
|
+
let autoFocus;
|
|
42
|
+
let children;
|
|
43
|
+
let onAction;
|
|
44
|
+
let ref;
|
|
45
|
+
let style;
|
|
46
|
+
let t1;
|
|
47
|
+
if ($[0] !== t0) {
|
|
48
|
+
({
|
|
49
|
+
onAction,
|
|
50
|
+
tabIndex: t1,
|
|
51
|
+
autoFocus,
|
|
52
|
+
children,
|
|
53
|
+
ref,
|
|
54
|
+
...style
|
|
55
|
+
} = t0);
|
|
56
|
+
$[0] = t0;
|
|
57
|
+
$[1] = autoFocus;
|
|
58
|
+
$[2] = children;
|
|
59
|
+
$[3] = onAction;
|
|
60
|
+
$[4] = ref;
|
|
61
|
+
$[5] = style;
|
|
62
|
+
$[6] = t1;
|
|
63
|
+
} else {
|
|
64
|
+
autoFocus = $[1];
|
|
65
|
+
children = $[2];
|
|
66
|
+
onAction = $[3];
|
|
67
|
+
ref = $[4];
|
|
68
|
+
style = $[5];
|
|
69
|
+
t1 = $[6];
|
|
70
|
+
}
|
|
71
|
+
const tabIndex = t1 === undefined ? 0 : t1;
|
|
72
|
+
const [isFocused, setIsFocused] = useState(false);
|
|
73
|
+
const [isHovered, setIsHovered] = useState(false);
|
|
74
|
+
const [isActive, setIsActive] = useState(false);
|
|
75
|
+
const activeTimer = useRef(null);
|
|
76
|
+
let t2;
|
|
77
|
+
let t3;
|
|
78
|
+
if ($[7] === Symbol.for("react.memo_cache_sentinel")) {
|
|
79
|
+
t2 = () => () => {
|
|
80
|
+
if (activeTimer.current) {
|
|
81
|
+
clearTimeout(activeTimer.current);
|
|
82
|
+
}
|
|
83
|
+
};
|
|
84
|
+
t3 = [];
|
|
85
|
+
$[7] = t2;
|
|
86
|
+
$[8] = t3;
|
|
87
|
+
} else {
|
|
88
|
+
t2 = $[7];
|
|
89
|
+
t3 = $[8];
|
|
90
|
+
}
|
|
91
|
+
useEffect(t2, t3);
|
|
92
|
+
let t4;
|
|
93
|
+
if ($[9] !== onAction) {
|
|
94
|
+
t4 = e => {
|
|
95
|
+
if (e.key === "return" || e.key === " ") {
|
|
96
|
+
e.preventDefault();
|
|
97
|
+
setIsActive(true);
|
|
98
|
+
onAction();
|
|
99
|
+
if (activeTimer.current) {
|
|
100
|
+
clearTimeout(activeTimer.current);
|
|
101
|
+
}
|
|
102
|
+
activeTimer.current = setTimeout(_temp, 100, setIsActive);
|
|
103
|
+
}
|
|
104
|
+
};
|
|
105
|
+
$[9] = onAction;
|
|
106
|
+
$[10] = t4;
|
|
107
|
+
} else {
|
|
108
|
+
t4 = $[10];
|
|
109
|
+
}
|
|
110
|
+
const handleKeyDown = t4;
|
|
111
|
+
let t5;
|
|
112
|
+
if ($[11] !== onAction) {
|
|
113
|
+
t5 = _e => {
|
|
114
|
+
onAction();
|
|
115
|
+
};
|
|
116
|
+
$[11] = onAction;
|
|
117
|
+
$[12] = t5;
|
|
118
|
+
} else {
|
|
119
|
+
t5 = $[12];
|
|
120
|
+
}
|
|
121
|
+
const handleClick = t5;
|
|
122
|
+
let t6;
|
|
123
|
+
if ($[13] === Symbol.for("react.memo_cache_sentinel")) {
|
|
124
|
+
t6 = _e_0 => setIsFocused(true);
|
|
125
|
+
$[13] = t6;
|
|
126
|
+
} else {
|
|
127
|
+
t6 = $[13];
|
|
128
|
+
}
|
|
129
|
+
const handleFocus = t6;
|
|
130
|
+
let t7;
|
|
131
|
+
if ($[14] === Symbol.for("react.memo_cache_sentinel")) {
|
|
132
|
+
t7 = _e_1 => setIsFocused(false);
|
|
133
|
+
$[14] = t7;
|
|
134
|
+
} else {
|
|
135
|
+
t7 = $[14];
|
|
136
|
+
}
|
|
137
|
+
const handleBlur = t7;
|
|
138
|
+
let t8;
|
|
139
|
+
if ($[15] === Symbol.for("react.memo_cache_sentinel")) {
|
|
140
|
+
t8 = () => setIsHovered(true);
|
|
141
|
+
$[15] = t8;
|
|
142
|
+
} else {
|
|
143
|
+
t8 = $[15];
|
|
144
|
+
}
|
|
145
|
+
const handleMouseEnter = t8;
|
|
146
|
+
let t9;
|
|
147
|
+
if ($[16] === Symbol.for("react.memo_cache_sentinel")) {
|
|
148
|
+
t9 = () => setIsHovered(false);
|
|
149
|
+
$[16] = t9;
|
|
150
|
+
} else {
|
|
151
|
+
t9 = $[16];
|
|
152
|
+
}
|
|
153
|
+
const handleMouseLeave = t9;
|
|
154
|
+
let t10;
|
|
155
|
+
if ($[17] !== children || $[18] !== isActive || $[19] !== isFocused || $[20] !== isHovered) {
|
|
156
|
+
const state = {
|
|
157
|
+
focused: isFocused,
|
|
158
|
+
hovered: isHovered,
|
|
159
|
+
active: isActive
|
|
160
|
+
};
|
|
161
|
+
t10 = typeof children === "function" ? children(state) : children;
|
|
162
|
+
$[17] = children;
|
|
163
|
+
$[18] = isActive;
|
|
164
|
+
$[19] = isFocused;
|
|
165
|
+
$[20] = isHovered;
|
|
166
|
+
$[21] = t10;
|
|
167
|
+
} else {
|
|
168
|
+
t10 = $[21];
|
|
169
|
+
}
|
|
170
|
+
const content = t10;
|
|
171
|
+
let t11;
|
|
172
|
+
if ($[22] !== autoFocus || $[23] !== content || $[24] !== handleClick || $[25] !== handleKeyDown || $[26] !== ref || $[27] !== style || $[28] !== tabIndex) {
|
|
173
|
+
t11 = <Box ref={ref} tabIndex={tabIndex} autoFocus={autoFocus} onKeyDown={handleKeyDown} onClick={handleClick} onFocus={handleFocus} onBlur={handleBlur} onMouseEnter={handleMouseEnter} onMouseLeave={handleMouseLeave} {...style}>{content}</Box>;
|
|
174
|
+
$[22] = autoFocus;
|
|
175
|
+
$[23] = content;
|
|
176
|
+
$[24] = handleClick;
|
|
177
|
+
$[25] = handleKeyDown;
|
|
178
|
+
$[26] = ref;
|
|
179
|
+
$[27] = style;
|
|
180
|
+
$[28] = tabIndex;
|
|
181
|
+
$[29] = t11;
|
|
182
|
+
} else {
|
|
183
|
+
t11 = $[29];
|
|
184
|
+
}
|
|
185
|
+
return t11;
|
|
186
|
+
}
|
|
187
|
+
function _temp(setter) {
|
|
188
|
+
return setter(false);
|
|
189
|
+
}
|
|
190
|
+
export default Button;
|
|
191
|
+
export type { ButtonState };
|
|
192
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","Ref","useCallback","useEffect","useRef","useState","Except","DOMElement","ClickEvent","FocusEvent","KeyboardEvent","Styles","Box","ButtonState","focused","hovered","active","Props","ref","onAction","tabIndex","autoFocus","children","state","ReactNode","Button","t0","$","_c","style","t1","undefined","isFocused","setIsFocused","isHovered","setIsHovered","isActive","setIsActive","activeTimer","t2","t3","Symbol","for","current","clearTimeout","t4","e","key","preventDefault","setTimeout","_temp","handleKeyDown","t5","_e","handleClick","t6","_e_0","handleFocus","t7","_e_1","handleBlur","t8","handleMouseEnter","t9","handleMouseLeave","t10","content","t11","setter"],"sources":["Button.tsx"],"sourcesContent":["import React, {\n  type Ref,\n  useCallback,\n  useEffect,\n  useRef,\n  useState,\n} from 'react'\nimport type { Except } from 'type-fest'\nimport type { DOMElement } from '../dom.js'\nimport type { ClickEvent } from '../events/click-event.js'\nimport type { FocusEvent } from '../events/focus-event.js'\nimport type { KeyboardEvent } from '../events/keyboard-event.js'\nimport type { Styles } from '../styles.js'\nimport Box from './Box.js'\n\ntype ButtonState = {\n  focused: boolean\n  hovered: boolean\n  active: boolean\n}\n\nexport type Props = Except<Styles, 'textWrap'> & {\n  ref?: Ref<DOMElement>\n  /**\n   * Called when the button is activated via Enter, Space, or click.\n   */\n  onAction: () => void\n  /**\n   * Tab order index. Defaults to 0 (in tab order).\n   * Set to -1 for programmatically focusable only.\n   */\n  tabIndex?: number\n  /**\n   * Focus this button when it mounts.\n   */\n  autoFocus?: boolean\n  /**\n   * Render prop receiving the interactive state. Use this to\n   * style children based on focus/hover/active — Button itself\n   * is intentionally unstyled.\n   *\n   * If not provided, children render as-is (no state-dependent styling).\n   */\n  children: ((state: ButtonState) => React.ReactNode) | React.ReactNode\n}\n\nfunction Button({\n  onAction,\n  tabIndex = 0,\n  autoFocus,\n  children,\n  ref,\n  ...style\n}: Props): React.ReactNode {\n  const [isFocused, setIsFocused] = useState(false)\n  const [isHovered, setIsHovered] = useState(false)\n  const [isActive, setIsActive] = useState(false)\n\n  const activeTimer = useRef<ReturnType<typeof setTimeout> | null>(null)\n\n  useEffect(() => {\n    return () => {\n      if (activeTimer.current) clearTimeout(activeTimer.current)\n    }\n  }, [])\n\n  const handleKeyDown = useCallback(\n    (e: KeyboardEvent) => {\n      if (e.key === 'return' || e.key === ' ') {\n        e.preventDefault()\n        setIsActive(true)\n        onAction()\n        if (activeTimer.current) clearTimeout(activeTimer.current)\n        activeTimer.current = setTimeout(\n          setter => setter(false),\n          100,\n          setIsActive,\n        )\n      }\n    },\n    [onAction],\n  )\n\n  const handleClick = useCallback(\n    (_e: ClickEvent) => {\n      onAction()\n    },\n    [onAction],\n  )\n\n  const handleFocus = useCallback((_e: FocusEvent) => setIsFocused(true), [])\n  const handleBlur = useCallback((_e: FocusEvent) => setIsFocused(false), [])\n  const handleMouseEnter = useCallback(() => setIsHovered(true), [])\n  const handleMouseLeave = useCallback(() => setIsHovered(false), [])\n\n  const state: ButtonState = {\n    focused: isFocused,\n    hovered: isHovered,\n    active: isActive,\n  }\n  const content = typeof children === 'function' ? children(state) : children\n\n  return (\n    <Box\n      ref={ref}\n      tabIndex={tabIndex}\n      autoFocus={autoFocus}\n      onKeyDown={handleKeyDown}\n      onClick={handleClick}\n      onFocus={handleFocus}\n      onBlur={handleBlur}\n      onMouseEnter={handleMouseEnter}\n      onMouseLeave={handleMouseLeave}\n      {...style}\n    >\n      {content}\n    </Box>\n  )\n}\n\nexport default Button\nexport type { ButtonState }\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import { c as _c } from "react/compiler-runtime";
|
|
2
|
+
import React, { createContext, useEffect, useState } from 'react';
|
|
3
|
+
import { FRAME_INTERVAL_MS } from '../constants.js';
|
|
4
|
+
import { useTerminalFocus } from '../hooks/use-terminal-focus.js';
|
|
5
|
+
export type Clock = {
|
|
6
|
+
subscribe: (onChange: () => void, keepAlive: boolean) => () => void;
|
|
7
|
+
now: () => number;
|
|
8
|
+
setTickInterval: (ms: number) => void;
|
|
9
|
+
};
|
|
10
|
+
export function createClock(tickIntervalMs: number): Clock {
|
|
11
|
+
const subscribers = new Map<() => void, boolean>();
|
|
12
|
+
let interval: ReturnType<typeof setInterval> | null = null;
|
|
13
|
+
let currentTickIntervalMs = tickIntervalMs;
|
|
14
|
+
let startTime = 0;
|
|
15
|
+
// Snapshot of the current tick's time, ensuring all subscribers in the same
|
|
16
|
+
// tick see the same value (keeps animations synchronized)
|
|
17
|
+
let tickTime = 0;
|
|
18
|
+
function tick(): void {
|
|
19
|
+
tickTime = Date.now() - startTime;
|
|
20
|
+
for (const onChange of subscribers.keys()) {
|
|
21
|
+
onChange();
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
function updateInterval(): void {
|
|
25
|
+
const anyKeepAlive = [...subscribers.values()].some(Boolean);
|
|
26
|
+
if (anyKeepAlive) {
|
|
27
|
+
if (interval) {
|
|
28
|
+
clearInterval(interval);
|
|
29
|
+
interval = null;
|
|
30
|
+
}
|
|
31
|
+
if (startTime === 0) {
|
|
32
|
+
startTime = Date.now();
|
|
33
|
+
}
|
|
34
|
+
interval = setInterval(tick, currentTickIntervalMs);
|
|
35
|
+
} else if (interval) {
|
|
36
|
+
clearInterval(interval);
|
|
37
|
+
interval = null;
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
return {
|
|
41
|
+
subscribe(onChange, keepAlive) {
|
|
42
|
+
subscribers.set(onChange, keepAlive);
|
|
43
|
+
updateInterval();
|
|
44
|
+
return () => {
|
|
45
|
+
subscribers.delete(onChange);
|
|
46
|
+
updateInterval();
|
|
47
|
+
};
|
|
48
|
+
},
|
|
49
|
+
now() {
|
|
50
|
+
if (startTime === 0) {
|
|
51
|
+
startTime = Date.now();
|
|
52
|
+
}
|
|
53
|
+
// When the clock interval is running, return the synchronized tickTime
|
|
54
|
+
// so all subscribers in the same tick see the same value.
|
|
55
|
+
// When paused (no keepAlive subscribers), return real-time to avoid
|
|
56
|
+
// returning a stale tickTime from the last tick before the pause.
|
|
57
|
+
if (interval && tickTime) {
|
|
58
|
+
return tickTime;
|
|
59
|
+
}
|
|
60
|
+
return Date.now() - startTime;
|
|
61
|
+
},
|
|
62
|
+
setTickInterval(ms) {
|
|
63
|
+
if (ms === currentTickIntervalMs) return;
|
|
64
|
+
currentTickIntervalMs = ms;
|
|
65
|
+
updateInterval();
|
|
66
|
+
}
|
|
67
|
+
};
|
|
68
|
+
}
|
|
69
|
+
export const ClockContext = createContext<Clock | null>(null);
|
|
70
|
+
const BLURRED_TICK_INTERVAL_MS = FRAME_INTERVAL_MS * 2;
|
|
71
|
+
|
|
72
|
+
// Own component so App.tsx doesn't re-render when the clock is created.
|
|
73
|
+
// The clock value is stable (created once via useState), so the provider
|
|
74
|
+
// never causes consumer re-renders on its own.
|
|
75
|
+
export function ClockProvider(t0) {
|
|
76
|
+
const $ = _c(7);
|
|
77
|
+
const {
|
|
78
|
+
children
|
|
79
|
+
} = t0;
|
|
80
|
+
const [clock] = useState(_temp);
|
|
81
|
+
const focused = useTerminalFocus();
|
|
82
|
+
let t1;
|
|
83
|
+
let t2;
|
|
84
|
+
if ($[0] !== clock || $[1] !== focused) {
|
|
85
|
+
t1 = () => {
|
|
86
|
+
clock.setTickInterval(focused ? FRAME_INTERVAL_MS : BLURRED_TICK_INTERVAL_MS);
|
|
87
|
+
};
|
|
88
|
+
t2 = [clock, focused];
|
|
89
|
+
$[0] = clock;
|
|
90
|
+
$[1] = focused;
|
|
91
|
+
$[2] = t1;
|
|
92
|
+
$[3] = t2;
|
|
93
|
+
} else {
|
|
94
|
+
t1 = $[2];
|
|
95
|
+
t2 = $[3];
|
|
96
|
+
}
|
|
97
|
+
useEffect(t1, t2);
|
|
98
|
+
let t3;
|
|
99
|
+
if ($[4] !== children || $[5] !== clock) {
|
|
100
|
+
t3 = <ClockContext.Provider value={clock}>{children}</ClockContext.Provider>;
|
|
101
|
+
$[4] = children;
|
|
102
|
+
$[5] = clock;
|
|
103
|
+
$[6] = t3;
|
|
104
|
+
} else {
|
|
105
|
+
t3 = $[6];
|
|
106
|
+
}
|
|
107
|
+
return t3;
|
|
108
|
+
}
|
|
109
|
+
function _temp() {
|
|
110
|
+
return createClock(FRAME_INTERVAL_MS);
|
|
111
|
+
}
|
|
112
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["React","createContext","useEffect","useState","FRAME_INTERVAL_MS","useTerminalFocus","Clock","subscribe","onChange","keepAlive","now","setTickInterval","ms","createClock","tickIntervalMs","subscribers","Map","interval","ReturnType","setInterval","currentTickIntervalMs","startTime","tickTime","tick","Date","keys","updateInterval","anyKeepAlive","values","some","Boolean","clearInterval","set","delete","ClockContext","BLURRED_TICK_INTERVAL_MS","ClockProvider","t0","$","_c","children","clock","_temp","focused","t1","t2","t3"],"sources":["ClockContext.tsx"],"sourcesContent":["import React, { createContext, useEffect, useState } from 'react'\nimport { FRAME_INTERVAL_MS } from '../constants.js'\nimport { useTerminalFocus } from '../hooks/use-terminal-focus.js'\n\nexport type Clock = {\n  subscribe: (onChange: () => void, keepAlive: boolean) => () => void\n  now: () => number\n  setTickInterval: (ms: number) => void\n}\n\nexport function createClock(tickIntervalMs: number): Clock {\n  const subscribers = new Map<() => void, boolean>()\n  let interval: ReturnType<typeof setInterval> | null = null\n  let currentTickIntervalMs = tickIntervalMs\n  let startTime = 0\n  // Snapshot of the current tick's time, ensuring all subscribers in the same\n  // tick see the same value (keeps animations synchronized)\n  let tickTime = 0\n\n  function tick(): void {\n    tickTime = Date.now() - startTime\n    for (const onChange of subscribers.keys()) {\n      onChange()\n    }\n  }\n\n  function updateInterval(): void {\n    const anyKeepAlive = [...subscribers.values()].some(Boolean)\n\n    if (anyKeepAlive) {\n      if (interval) {\n        clearInterval(interval)\n        interval = null\n      }\n      if (startTime === 0) {\n        startTime = Date.now()\n      }\n      interval = setInterval(tick, currentTickIntervalMs)\n    } else if (interval) {\n      clearInterval(interval)\n      interval = null\n    }\n  }\n\n  return {\n    subscribe(onChange, keepAlive) {\n      subscribers.set(onChange, keepAlive)\n      updateInterval()\n      return () => {\n        subscribers.delete(onChange)\n        updateInterval()\n      }\n    },\n\n    now() {\n      if (startTime === 0) {\n        startTime = Date.now()\n      }\n      // When the clock interval is running, return the synchronized tickTime\n      // so all subscribers in the same tick see the same value.\n      // When paused (no keepAlive subscribers), return real-time to avoid\n      // returning a stale tickTime from the last tick before the pause.\n      if (interval && tickTime) {\n        return tickTime\n      }\n      return Date.now() - startTime\n    },\n\n    setTickInterval(ms) {\n      if (ms === currentTickIntervalMs) return\n      currentTickIntervalMs = ms\n      updateInterval()\n    },\n  }\n}\n\nexport const ClockContext = createContext<Clock | null>(null)\n\nconst BLURRED_TICK_INTERVAL_MS = FRAME_INTERVAL_MS * 2\n\n// Own component so App.tsx doesn't re-render when the clock is created.\n// The clock value is stable (created once via useState), so the provider\n// never causes consumer re-renders on its own.\nexport function ClockProvider({\n  children,\n}: {\n  children: React.ReactNode\n}): React.ReactNode {\n  const [clock] = useState(() => createClock(FRAME_INTERVAL_MS))\n  const focused = useTerminalFocus()\n\n  useEffect(() => {\n    clock.setTickInterval(\n      focused ? FRAME_INTERVAL_MS : BLURRED_TICK_INTERVAL_MS,\n    )\n  }, [clock, focused])\n\n  return <ClockContext.Provider value={clock}>{children}</ClockContext.Provider>\n}\n"],"mappings":";AAAA,OAAOA,KAAK,IAAIC,aAAa,EAAEC,SAAS,EAAEC,QAAQ,QAAQ,OAAO;AACjE,SAASC,iBAAiB,QAAQ,iBAAiB;AACnD,SAASC,gBAAgB,QAAQ,gCAAgC;AAEjE,OAAO,KAAKC,KAAK,GAAG;EAClBC,SAAS,EAAE,CAACC,QAAQ,EAAE,GAAG,GAAG,IAAI,EAAEC,SAAS,EAAE,OAAO,EAAE,GAAG,GAAG,GAAG,IAAI;EACnEC,GAAG,EAAE,GAAG,GAAG,MAAM;EACjBC,eAAe,EAAE,CAACC,EAAE,EAAE,MAAM,EAAE,GAAG,IAAI;AACvC,CAAC;AAED,OAAO,SAASC,WAAWA,CAACC,cAAc,EAAE,MAAM,CAAC,EAAER,KAAK,CAAC;EACzD,MAAMS,WAAW,GAAG,IAAIC,GAAG,CAAC,GAAG,GAAG,IAAI,EAAE,OAAO,CAAC,CAAC,CAAC;EAClD,IAAIC,QAAQ,EAAEC,UAAU,CAAC,OAAOC,WAAW,CAAC,GAAG,IAAI,GAAG,IAAI;EAC1D,IAAIC,qBAAqB,GAAGN,cAAc;EAC1C,IAAIO,SAAS,GAAG,CAAC;EACjB;EACA;EACA,IAAIC,QAAQ,GAAG,CAAC;EAEhB,SAASC,IAAIA,CAAA,CAAE,EAAE,IAAI,CAAC;IACpBD,QAAQ,GAAGE,IAAI,CAACd,GAAG,CAAC,CAAC,GAAGW,SAAS;IACjC,KAAK,MAAMb,QAAQ,IAAIO,WAAW,CAACU,IAAI,CAAC,CAAC,EAAE;MACzCjB,QAAQ,CAAC,CAAC;IACZ;EACF;EAEA,SAASkB,cAAcA,CAAA,CAAE,EAAE,IAAI,CAAC;IAC9B,MAAMC,YAAY,GAAG,CAAC,GAAGZ,WAAW,CAACa,MAAM,CAAC,CAAC,CAAC,CAACC,IAAI,CAACC,OAAO,CAAC;IAE5D,IAAIH,YAAY,EAAE;MAChB,IAAIV,QAAQ,EAAE;QACZc,aAAa,CAACd,QAAQ,CAAC;QACvBA,QAAQ,GAAG,IAAI;MACjB;MACA,IAAII,SAAS,KAAK,CAAC,EAAE;QACnBA,SAAS,GAAGG,IAAI,CAACd,GAAG,CAAC,CAAC;MACxB;MACAO,QAAQ,GAAGE,WAAW,CAACI,IAAI,EAAEH,qBAAqB,CAAC;IACrD,CAAC,MAAM,IAAIH,QAAQ,EAAE;MACnBc,aAAa,CAACd,QAAQ,CAAC;MACvBA,QAAQ,GAAG,IAAI;IACjB;EACF;EAEA,OAAO;IACLV,SAASA,CAACC,QAAQ,EAAEC,SAAS,EAAE;MAC7BM,WAAW,CAACiB,GAAG,CAACxB,QAAQ,EAAEC,SAAS,CAAC;MACpCiB,cAAc,CAAC,CAAC;MAChB,OAAO,MAAM;QACXX,WAAW,CAACkB,MAAM,CAACzB,QAAQ,CAAC;QAC5BkB,cAAc,CAAC,CAAC;MAClB,CAAC;IACH,CAAC;IAEDhB,GAAGA,CAAA,EAAG;MACJ,IAAIW,SAAS,KAAK,CAAC,EAAE;QACnBA,SAAS,GAAGG,IAAI,CAACd,GAAG,CAAC,CAAC;MACxB;MACA;MACA;MACA;MACA;MACA,IAAIO,QAAQ,IAAIK,QAAQ,EAAE;QACxB,OAAOA,QAAQ;MACjB;MACA,OAAOE,IAAI,CAACd,GAAG,CAAC,CAAC,GAAGW,SAAS;IAC/B,CAAC;IAEDV,eAAeA,CAACC,EAAE,EAAE;MAClB,IAAIA,EAAE,KAAKQ,qBAAqB,EAAE;MAClCA,qBAAqB,GAAGR,EAAE;MAC1Bc,cAAc,CAAC,CAAC;IAClB;EACF,CAAC;AACH;AAEA,OAAO,MAAMQ,YAAY,GAAGjC,aAAa,CAACK,KAAK,GAAG,IAAI,CAAC,CAAC,IAAI,CAAC;AAE7D,MAAM6B,wBAAwB,GAAG/B,iBAAiB,GAAG,CAAC;;AAEtD;AACA;AACA;AACA,OAAO,SAAAgC,cAAAC,EAAA;EAAA,MAAAC,CAAA,GAAAC,EAAA;EAAuB;IAAAC;EAAA,IAAAH,EAI7B;EACC,OAAAI,KAAA,IAAgBtC,QAAQ,CAACuC,KAAoC,CAAC;EAC9D,MAAAC,OAAA,GAAgBtC,gBAAgB,CAAC,CAAC;EAAA,IAAAuC,EAAA;EAAA,IAAAC,EAAA;EAAA,IAAAP,CAAA,QAAAG,KAAA,IAAAH,CAAA,QAAAK,OAAA;IAExBC,EAAA,GAAAA,CAAA;MACRH,KAAK,CAAA9B,eAAgB,CACnBgC,OAAO,GAAPvC,iBAAsD,GAAtD+B,wBACF,CAAC;IAAA,CACF;IAAEU,EAAA,IAACJ,KAAK,EAAEE,OAAO,CAAC;IAAAL,CAAA,MAAAG,KAAA;IAAAH,CAAA,MAAAK,OAAA;IAAAL,CAAA,MAAAM,EAAA;IAAAN,CAAA,MAAAO,EAAA;EAAA;IAAAD,EAAA,GAAAN,CAAA;IAAAO,EAAA,GAAAP,CAAA;EAAA;EAJnBpC,SAAS,CAAC0C,EAIT,EAAEC,EAAgB,CAAC;EAAA,IAAAC,EAAA;EAAA,IAAAR,CAAA,QAAAE,QAAA,IAAAF,CAAA,QAAAG,KAAA;IAEbK,EAAA,0BAA8BL,KAAK,CAALA,MAAI,CAAC,CAAGD,SAAO,CAAE,wBAAwB;IAAAF,CAAA,MAAAE,QAAA;IAAAF,CAAA,MAAAG,KAAA;IAAAH,CAAA,MAAAQ,EAAA;EAAA;IAAAA,EAAA,GAAAR,CAAA;EAAA;EAAA,OAAvEQ,EAAuE;AAAA;AAdzE,SAAAJ,MAAA;EAAA,OAK0B7B,WAAW,CAACT,iBAAiB,CAAC;AAAA","ignoreList":[]}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { createContext } from 'react'
|
|
2
|
+
import type { DOMElement } from '../dom.js'
|
|
3
|
+
|
|
4
|
+
export type CursorDeclaration = {
|
|
5
|
+
/** Display column (terminal cell width) within the declared node */
|
|
6
|
+
readonly relativeX: number
|
|
7
|
+
/** Line number within the declared node */
|
|
8
|
+
readonly relativeY: number
|
|
9
|
+
/** The ink-box DOMElement whose yoga layout provides the absolute origin */
|
|
10
|
+
readonly node: DOMElement
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Setter for the declared cursor position.
|
|
15
|
+
*
|
|
16
|
+
* The optional second argument makes `null` a conditional clear: the
|
|
17
|
+
* declaration is only cleared if the currently-declared node matches
|
|
18
|
+
* `clearIfNode`. This makes the hook safe for sibling components
|
|
19
|
+
* (e.g. list items) that transfer focus among themselves — without the
|
|
20
|
+
* node check, a newly-unfocused item's clear could clobber a
|
|
21
|
+
* newly-focused sibling's set depending on layout-effect order.
|
|
22
|
+
*/
|
|
23
|
+
export type CursorDeclarationSetter = (
|
|
24
|
+
declaration: CursorDeclaration | null,
|
|
25
|
+
clearIfNode?: DOMElement | null,
|
|
26
|
+
) => void
|
|
27
|
+
|
|
28
|
+
const CursorDeclarationContext = createContext<CursorDeclarationSetter>(
|
|
29
|
+
() => {},
|
|
30
|
+
)
|
|
31
|
+
|
|
32
|
+
export default CursorDeclarationContext
|