yzcode-cli 1.0.1 → 1.0.3
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/assistant/sessionHistory.ts +87 -0
- package/bootstrap/state.ts +1769 -0
- package/bridge/bridgeApi.ts +539 -0
- package/bridge/bridgeConfig.ts +48 -0
- package/bridge/bridgeDebug.ts +135 -0
- package/bridge/bridgeEnabled.ts +202 -0
- package/bridge/bridgeMain.ts +2999 -0
- package/bridge/bridgeMessaging.ts +461 -0
- package/bridge/bridgePermissionCallbacks.ts +43 -0
- package/bridge/bridgePointer.ts +210 -0
- package/bridge/bridgeStatusUtil.ts +163 -0
- package/bridge/bridgeUI.ts +530 -0
- package/bridge/capacityWake.ts +56 -0
- package/bridge/codeSessionApi.ts +168 -0
- package/bridge/createSession.ts +384 -0
- package/bridge/debugUtils.ts +141 -0
- package/bridge/envLessBridgeConfig.ts +165 -0
- package/bridge/flushGate.ts +71 -0
- package/bridge/inboundAttachments.ts +175 -0
- package/bridge/inboundMessages.ts +80 -0
- package/bridge/initReplBridge.ts +569 -0
- package/bridge/jwtUtils.ts +256 -0
- package/bridge/pollConfig.ts +110 -0
- package/bridge/pollConfigDefaults.ts +82 -0
- package/bridge/remoteBridgeCore.ts +1008 -0
- package/bridge/replBridge.ts +2406 -0
- package/bridge/replBridgeHandle.ts +36 -0
- package/bridge/replBridgeTransport.ts +370 -0
- package/bridge/sessionIdCompat.ts +57 -0
- package/bridge/sessionRunner.ts +550 -0
- package/bridge/trustedDevice.ts +210 -0
- package/bridge/types.ts +262 -0
- package/bridge/workSecret.ts +127 -0
- package/buddy/CompanionSprite.tsx +371 -0
- package/buddy/companion.ts +133 -0
- package/buddy/prompt.ts +36 -0
- package/buddy/sprites.ts +514 -0
- package/buddy/types.ts +148 -0
- package/buddy/useBuddyNotification.tsx +98 -0
- package/coordinator/coordinatorMode.ts +369 -0
- package/memdir/findRelevantMemories.ts +141 -0
- package/memdir/memdir.ts +507 -0
- package/memdir/memoryAge.ts +53 -0
- package/memdir/memoryScan.ts +94 -0
- package/memdir/memoryTypes.ts +271 -0
- package/memdir/paths.ts +278 -0
- package/memdir/teamMemPaths.ts +292 -0
- package/memdir/teamMemPrompts.ts +100 -0
- package/migrations/migrateAutoUpdatesToSettings.ts +61 -0
- package/migrations/migrateBypassPermissionsAcceptedToSettings.ts +40 -0
- package/migrations/migrateEnableAllProjectMcpServersToSettings.ts +118 -0
- package/migrations/migrateFennecToOpus.ts +45 -0
- package/migrations/migrateLegacyOpusToCurrent.ts +57 -0
- package/migrations/migrateOpusToOpus1m.ts +43 -0
- package/migrations/migrateReplBridgeEnabledToRemoteControlAtStartup.ts +22 -0
- package/migrations/migrateSonnet1mToSonnet45.ts +48 -0
- package/migrations/migrateSonnet45ToSonnet46.ts +67 -0
- package/migrations/resetAutoModeOptInForDefaultOffer.ts +51 -0
- package/migrations/resetProToOpusDefault.ts +51 -0
- package/native-ts/color-diff/index.ts +999 -0
- package/native-ts/file-index/index.ts +370 -0
- package/native-ts/yoga-layout/enums.ts +134 -0
- package/native-ts/yoga-layout/index.ts +2578 -0
- package/outputStyles/loadOutputStylesDir.ts +98 -0
- package/package.json +22 -5
- package/plugins/builtinPlugins.ts +159 -0
- package/plugins/bundled/index.ts +23 -0
- package/schemas/hooks.ts +222 -0
- package/screens/Doctor.tsx +575 -0
- package/screens/REPL.tsx +5006 -0
- package/screens/ResumeConversation.tsx +399 -0
- package/server/createDirectConnectSession.ts +88 -0
- package/server/directConnectManager.ts +213 -0
- package/server/types.ts +57 -0
- package/skills/bundled/batch.ts +124 -0
- package/skills/bundled/claudeApi.ts +196 -0
- package/skills/bundled/claudeApiContent.ts +75 -0
- package/skills/bundled/claudeInChrome.ts +34 -0
- package/skills/bundled/debug.ts +103 -0
- package/skills/bundled/index.ts +79 -0
- package/skills/bundled/keybindings.ts +339 -0
- package/skills/bundled/loop.ts +92 -0
- package/skills/bundled/loremIpsum.ts +282 -0
- package/skills/bundled/remember.ts +82 -0
- package/skills/bundled/scheduleRemoteAgents.ts +447 -0
- package/skills/bundled/simplify.ts +69 -0
- package/skills/bundled/skillify.ts +197 -0
- package/skills/bundled/stuck.ts +79 -0
- package/skills/bundled/updateConfig.ts +475 -0
- package/skills/bundled/verify/SKILL.md +3 -0
- package/skills/bundled/verify/examples/cli.md +3 -0
- package/skills/bundled/verify/examples/server.md +3 -0
- package/skills/bundled/verify.ts +30 -0
- package/skills/bundled/verifyContent.ts +13 -0
- package/skills/bundledSkills.ts +220 -0
- package/skills/loadSkillsDir.ts +1086 -0
- package/skills/mcpSkillBuilders.ts +44 -0
- package/tasks/DreamTask/DreamTask.ts +157 -0
- package/tasks/InProcessTeammateTask/InProcessTeammateTask.tsx +126 -0
- package/tasks/InProcessTeammateTask/types.ts +121 -0
- package/tasks/LocalAgentTask/LocalAgentTask.tsx +683 -0
- package/tasks/LocalMainSessionTask.ts +479 -0
- package/tasks/LocalShellTask/LocalShellTask.tsx +523 -0
- package/tasks/LocalShellTask/guards.ts +41 -0
- package/tasks/LocalShellTask/killShellTasks.ts +76 -0
- package/tasks/RemoteAgentTask/RemoteAgentTask.tsx +856 -0
- package/tasks/pillLabel.ts +82 -0
- package/tasks/stopTask.ts +100 -0
- package/tasks/types.ts +46 -0
- package/upstreamproxy/relay.ts +455 -0
- package/upstreamproxy/upstreamproxy.ts +285 -0
- package/vim/motions.ts +82 -0
- package/vim/operators.ts +556 -0
- package/vim/textObjects.ts +186 -0
- package/vim/transitions.ts +490 -0
- package/vim/types.ts +199 -0
- package/voice/voiceModeEnabled.ts +54 -0
|
@@ -0,0 +1,371 @@
|
|
|
1
|
+
import { c as _c } from "react/compiler-runtime";
|
|
2
|
+
import { feature } from 'bun:bundle';
|
|
3
|
+
import figures from 'figures';
|
|
4
|
+
import React, { useEffect, useRef, useState } from 'react';
|
|
5
|
+
import { useTerminalSize } from '../hooks/useTerminalSize.js';
|
|
6
|
+
import { stringWidth } from '../ink/stringWidth.js';
|
|
7
|
+
import { Box, Text } from '../ink.js';
|
|
8
|
+
import { useAppState, useSetAppState } from '../state/AppState.js';
|
|
9
|
+
import type { AppState } from '../state/AppStateStore.js';
|
|
10
|
+
import { getGlobalConfig } from '../utils/config.js';
|
|
11
|
+
import { isFullscreenActive } from '../utils/fullscreen.js';
|
|
12
|
+
import type { Theme } from '../utils/theme.js';
|
|
13
|
+
import { getCompanion } from './companion.js';
|
|
14
|
+
import { renderFace, renderSprite, spriteFrameCount } from './sprites.js';
|
|
15
|
+
import { RARITY_COLORS } from './types.js';
|
|
16
|
+
const TICK_MS = 500;
|
|
17
|
+
const BUBBLE_SHOW = 20; // ticks → ~10s at 500ms
|
|
18
|
+
const FADE_WINDOW = 6; // last ~3s the bubble dims so you know it's about to go
|
|
19
|
+
const PET_BURST_MS = 2500; // how long hearts float after /buddy pet
|
|
20
|
+
|
|
21
|
+
// Idle sequence: mostly rest (frame 0), occasional fidget (frames 1-2), rare blink.
|
|
22
|
+
// Sequence indices map to sprite frames; -1 means "blink on frame 0".
|
|
23
|
+
const IDLE_SEQUENCE = [0, 0, 0, 0, 1, 0, 0, 0, -1, 0, 0, 2, 0, 0, 0];
|
|
24
|
+
|
|
25
|
+
// Hearts float up-and-out over 5 ticks (~2.5s). Prepended above the sprite.
|
|
26
|
+
const H = figures.heart;
|
|
27
|
+
const PET_HEARTS = [` ${H} ${H} `, ` ${H} ${H} ${H} `, ` ${H} ${H} ${H} `, `${H} ${H} ${H} `, '· · · '];
|
|
28
|
+
function wrap(text: string, width: number): string[] {
|
|
29
|
+
const words = text.split(' ');
|
|
30
|
+
const lines: string[] = [];
|
|
31
|
+
let cur = '';
|
|
32
|
+
for (const w of words) {
|
|
33
|
+
if (cur.length + w.length + 1 > width && cur) {
|
|
34
|
+
lines.push(cur);
|
|
35
|
+
cur = w;
|
|
36
|
+
} else {
|
|
37
|
+
cur = cur ? `${cur} ${w}` : w;
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
if (cur) lines.push(cur);
|
|
41
|
+
return lines;
|
|
42
|
+
}
|
|
43
|
+
function SpeechBubble(t0) {
|
|
44
|
+
const $ = _c(31);
|
|
45
|
+
const {
|
|
46
|
+
text,
|
|
47
|
+
color,
|
|
48
|
+
fading,
|
|
49
|
+
tail
|
|
50
|
+
} = t0;
|
|
51
|
+
let T0;
|
|
52
|
+
let borderColor;
|
|
53
|
+
let t1;
|
|
54
|
+
let t2;
|
|
55
|
+
let t3;
|
|
56
|
+
let t4;
|
|
57
|
+
let t5;
|
|
58
|
+
let t6;
|
|
59
|
+
if ($[0] !== color || $[1] !== fading || $[2] !== text) {
|
|
60
|
+
const lines = wrap(text, 30);
|
|
61
|
+
borderColor = fading ? "inactive" : color;
|
|
62
|
+
T0 = Box;
|
|
63
|
+
t1 = "column";
|
|
64
|
+
t2 = "round";
|
|
65
|
+
t3 = borderColor;
|
|
66
|
+
t4 = 1;
|
|
67
|
+
t5 = 34;
|
|
68
|
+
let t7;
|
|
69
|
+
if ($[11] !== fading) {
|
|
70
|
+
t7 = (l, i) => <Text key={i} italic={true} dimColor={!fading} color={fading ? "inactive" : undefined}>{l}</Text>;
|
|
71
|
+
$[11] = fading;
|
|
72
|
+
$[12] = t7;
|
|
73
|
+
} else {
|
|
74
|
+
t7 = $[12];
|
|
75
|
+
}
|
|
76
|
+
t6 = lines.map(t7);
|
|
77
|
+
$[0] = color;
|
|
78
|
+
$[1] = fading;
|
|
79
|
+
$[2] = text;
|
|
80
|
+
$[3] = T0;
|
|
81
|
+
$[4] = borderColor;
|
|
82
|
+
$[5] = t1;
|
|
83
|
+
$[6] = t2;
|
|
84
|
+
$[7] = t3;
|
|
85
|
+
$[8] = t4;
|
|
86
|
+
$[9] = t5;
|
|
87
|
+
$[10] = t6;
|
|
88
|
+
} else {
|
|
89
|
+
T0 = $[3];
|
|
90
|
+
borderColor = $[4];
|
|
91
|
+
t1 = $[5];
|
|
92
|
+
t2 = $[6];
|
|
93
|
+
t3 = $[7];
|
|
94
|
+
t4 = $[8];
|
|
95
|
+
t5 = $[9];
|
|
96
|
+
t6 = $[10];
|
|
97
|
+
}
|
|
98
|
+
let t7;
|
|
99
|
+
if ($[13] !== T0 || $[14] !== t1 || $[15] !== t2 || $[16] !== t3 || $[17] !== t4 || $[18] !== t5 || $[19] !== t6) {
|
|
100
|
+
t7 = <T0 flexDirection={t1} borderStyle={t2} borderColor={t3} paddingX={t4} width={t5}>{t6}</T0>;
|
|
101
|
+
$[13] = T0;
|
|
102
|
+
$[14] = t1;
|
|
103
|
+
$[15] = t2;
|
|
104
|
+
$[16] = t3;
|
|
105
|
+
$[17] = t4;
|
|
106
|
+
$[18] = t5;
|
|
107
|
+
$[19] = t6;
|
|
108
|
+
$[20] = t7;
|
|
109
|
+
} else {
|
|
110
|
+
t7 = $[20];
|
|
111
|
+
}
|
|
112
|
+
const bubble = t7;
|
|
113
|
+
if (tail === "right") {
|
|
114
|
+
let t8;
|
|
115
|
+
if ($[21] !== borderColor) {
|
|
116
|
+
t8 = <Text color={borderColor}>─</Text>;
|
|
117
|
+
$[21] = borderColor;
|
|
118
|
+
$[22] = t8;
|
|
119
|
+
} else {
|
|
120
|
+
t8 = $[22];
|
|
121
|
+
}
|
|
122
|
+
let t9;
|
|
123
|
+
if ($[23] !== bubble || $[24] !== t8) {
|
|
124
|
+
t9 = <Box flexDirection="row" alignItems="center">{bubble}{t8}</Box>;
|
|
125
|
+
$[23] = bubble;
|
|
126
|
+
$[24] = t8;
|
|
127
|
+
$[25] = t9;
|
|
128
|
+
} else {
|
|
129
|
+
t9 = $[25];
|
|
130
|
+
}
|
|
131
|
+
return t9;
|
|
132
|
+
}
|
|
133
|
+
let t8;
|
|
134
|
+
if ($[26] !== borderColor) {
|
|
135
|
+
t8 = <Box flexDirection="column" alignItems="flex-end" paddingRight={6}><Text color={borderColor}>╲ </Text><Text color={borderColor}>╲</Text></Box>;
|
|
136
|
+
$[26] = borderColor;
|
|
137
|
+
$[27] = t8;
|
|
138
|
+
} else {
|
|
139
|
+
t8 = $[27];
|
|
140
|
+
}
|
|
141
|
+
let t9;
|
|
142
|
+
if ($[28] !== bubble || $[29] !== t8) {
|
|
143
|
+
t9 = <Box flexDirection="column" alignItems="flex-end" marginRight={1}>{bubble}{t8}</Box>;
|
|
144
|
+
$[28] = bubble;
|
|
145
|
+
$[29] = t8;
|
|
146
|
+
$[30] = t9;
|
|
147
|
+
} else {
|
|
148
|
+
t9 = $[30];
|
|
149
|
+
}
|
|
150
|
+
return t9;
|
|
151
|
+
}
|
|
152
|
+
export const MIN_COLS_FOR_FULL_SPRITE = 100;
|
|
153
|
+
const SPRITE_BODY_WIDTH = 12;
|
|
154
|
+
const NAME_ROW_PAD = 2; // focused state wraps name in spaces: ` name `
|
|
155
|
+
const SPRITE_PADDING_X = 2;
|
|
156
|
+
const BUBBLE_WIDTH = 36; // SpeechBubble box (34) + tail column
|
|
157
|
+
const NARROW_QUIP_CAP = 24;
|
|
158
|
+
function spriteColWidth(nameWidth: number): number {
|
|
159
|
+
return Math.max(SPRITE_BODY_WIDTH, nameWidth + NAME_ROW_PAD);
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
// Width the sprite area consumes. PromptInput subtracts this so text wraps
|
|
163
|
+
// correctly. In fullscreen the bubble floats over scrollback (no extra
|
|
164
|
+
// width); in non-fullscreen it sits inline and needs BUBBLE_WIDTH more.
|
|
165
|
+
// Narrow terminals: 0 — REPL.tsx stacks the one-liner on its own row
|
|
166
|
+
// (above input in fullscreen, below in scrollback), so no reservation.
|
|
167
|
+
export function companionReservedColumns(terminalColumns: number, speaking: boolean): number {
|
|
168
|
+
if (!feature('BUDDY')) return 0;
|
|
169
|
+
const companion = getCompanion();
|
|
170
|
+
if (!companion || getGlobalConfig().companionMuted) return 0;
|
|
171
|
+
if (terminalColumns < MIN_COLS_FOR_FULL_SPRITE) return 0;
|
|
172
|
+
const nameWidth = stringWidth(companion.name);
|
|
173
|
+
const bubble = speaking && !isFullscreenActive() ? BUBBLE_WIDTH : 0;
|
|
174
|
+
return spriteColWidth(nameWidth) + SPRITE_PADDING_X + bubble;
|
|
175
|
+
}
|
|
176
|
+
export function CompanionSprite(): React.ReactNode {
|
|
177
|
+
const reaction = useAppState(s => s.companionReaction);
|
|
178
|
+
const petAt = useAppState(s => s.companionPetAt);
|
|
179
|
+
const focused = useAppState(s => s.footerSelection === 'companion');
|
|
180
|
+
const setAppState = useSetAppState();
|
|
181
|
+
const {
|
|
182
|
+
columns
|
|
183
|
+
} = useTerminalSize();
|
|
184
|
+
const [tick, setTick] = useState(0);
|
|
185
|
+
const lastSpokeTick = useRef(0);
|
|
186
|
+
// Sync-during-render (not useEffect) so the first post-pet render already
|
|
187
|
+
// has petStartTick=tick and petAge=0 — otherwise frame 0 is skipped.
|
|
188
|
+
const [{
|
|
189
|
+
petStartTick,
|
|
190
|
+
forPetAt
|
|
191
|
+
}, setPetStart] = useState({
|
|
192
|
+
petStartTick: 0,
|
|
193
|
+
forPetAt: petAt
|
|
194
|
+
});
|
|
195
|
+
if (petAt !== forPetAt) {
|
|
196
|
+
setPetStart({
|
|
197
|
+
petStartTick: tick,
|
|
198
|
+
forPetAt: petAt
|
|
199
|
+
});
|
|
200
|
+
}
|
|
201
|
+
useEffect(() => {
|
|
202
|
+
const timer = setInterval(setT => setT((t: number) => t + 1), TICK_MS, setTick);
|
|
203
|
+
return () => clearInterval(timer);
|
|
204
|
+
}, []);
|
|
205
|
+
useEffect(() => {
|
|
206
|
+
if (!reaction) return;
|
|
207
|
+
lastSpokeTick.current = tick;
|
|
208
|
+
const timer = setTimeout(setA => setA((prev: AppState) => prev.companionReaction === undefined ? prev : {
|
|
209
|
+
...prev,
|
|
210
|
+
companionReaction: undefined
|
|
211
|
+
}), BUBBLE_SHOW * TICK_MS, setAppState);
|
|
212
|
+
return () => clearTimeout(timer);
|
|
213
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps -- tick intentionally captured at reaction-change, not tracked
|
|
214
|
+
}, [reaction, setAppState]);
|
|
215
|
+
if (!feature('BUDDY')) return null;
|
|
216
|
+
const companion = getCompanion();
|
|
217
|
+
if (!companion || getGlobalConfig().companionMuted) return null;
|
|
218
|
+
const color = RARITY_COLORS[companion.rarity];
|
|
219
|
+
const colWidth = spriteColWidth(stringWidth(companion.name));
|
|
220
|
+
const bubbleAge = reaction ? tick - lastSpokeTick.current : 0;
|
|
221
|
+
const fading = reaction !== undefined && bubbleAge >= BUBBLE_SHOW - FADE_WINDOW;
|
|
222
|
+
const petAge = petAt ? tick - petStartTick : Infinity;
|
|
223
|
+
const petting = petAge * TICK_MS < PET_BURST_MS;
|
|
224
|
+
|
|
225
|
+
// Narrow terminals: collapse to one-line face. When speaking, the quip
|
|
226
|
+
// replaces the name beside the face (no room for a bubble).
|
|
227
|
+
if (columns < MIN_COLS_FOR_FULL_SPRITE) {
|
|
228
|
+
const quip = reaction && reaction.length > NARROW_QUIP_CAP ? reaction.slice(0, NARROW_QUIP_CAP - 1) + '…' : reaction;
|
|
229
|
+
const label = quip ? `"${quip}"` : focused ? ` ${companion.name} ` : companion.name;
|
|
230
|
+
return <Box paddingX={1} alignSelf="flex-end">
|
|
231
|
+
<Text>
|
|
232
|
+
{petting && <Text color="autoAccept">{figures.heart} </Text>}
|
|
233
|
+
<Text bold color={color}>
|
|
234
|
+
{renderFace(companion)}
|
|
235
|
+
</Text>{' '}
|
|
236
|
+
<Text italic dimColor={!focused && !reaction} bold={focused} inverse={focused && !reaction} color={reaction ? fading ? 'inactive' : color : focused ? color : undefined}>
|
|
237
|
+
{label}
|
|
238
|
+
</Text>
|
|
239
|
+
</Text>
|
|
240
|
+
</Box>;
|
|
241
|
+
}
|
|
242
|
+
const frameCount = spriteFrameCount(companion.species);
|
|
243
|
+
const heartFrame = petting ? PET_HEARTS[petAge % PET_HEARTS.length] : null;
|
|
244
|
+
let spriteFrame: number;
|
|
245
|
+
let blink = false;
|
|
246
|
+
if (reaction || petting) {
|
|
247
|
+
// Excited: cycle all fidget frames fast
|
|
248
|
+
spriteFrame = tick % frameCount;
|
|
249
|
+
} else {
|
|
250
|
+
const step = IDLE_SEQUENCE[tick % IDLE_SEQUENCE.length]!;
|
|
251
|
+
if (step === -1) {
|
|
252
|
+
spriteFrame = 0;
|
|
253
|
+
blink = true;
|
|
254
|
+
} else {
|
|
255
|
+
spriteFrame = step % frameCount;
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
const body = renderSprite(companion, spriteFrame).map(line => blink ? line.replaceAll(companion.eye, '-') : line);
|
|
259
|
+
const sprite = heartFrame ? [heartFrame, ...body] : body;
|
|
260
|
+
|
|
261
|
+
// Name row doubles as hint row — unfocused shows dim name + ↓ discovery,
|
|
262
|
+
// focused shows inverse name. The enter-to-open hint lives in
|
|
263
|
+
// PromptInputFooter's right column so this row stays one line and the
|
|
264
|
+
// sprite doesn't jump up when selected. flexShrink=0 stops the
|
|
265
|
+
// inline-bubble row wrapper from squeezing the sprite to fit.
|
|
266
|
+
const spriteColumn = <Box flexDirection="column" flexShrink={0} alignItems="center" width={colWidth}>
|
|
267
|
+
{sprite.map((line, i) => <Text key={i} color={i === 0 && heartFrame ? 'autoAccept' : color}>
|
|
268
|
+
{line}
|
|
269
|
+
</Text>)}
|
|
270
|
+
<Text italic bold={focused} dimColor={!focused} color={focused ? color : undefined} inverse={focused}>
|
|
271
|
+
{focused ? ` ${companion.name} ` : companion.name}
|
|
272
|
+
</Text>
|
|
273
|
+
</Box>;
|
|
274
|
+
if (!reaction) {
|
|
275
|
+
return <Box paddingX={1}>{spriteColumn}</Box>;
|
|
276
|
+
}
|
|
277
|
+
|
|
278
|
+
// Fullscreen: bubble renders separately via CompanionFloatingBubble in
|
|
279
|
+
// FullscreenLayout's bottomFloat slot (the bottom slot's overflowY:hidden
|
|
280
|
+
// would clip a position:absolute overlay here). Sprite body only.
|
|
281
|
+
// Non-fullscreen: bubble sits inline beside the sprite (input shrinks)
|
|
282
|
+
// because floating into Static scrollback can't be cleared.
|
|
283
|
+
if (isFullscreenActive()) {
|
|
284
|
+
return <Box paddingX={1}>{spriteColumn}</Box>;
|
|
285
|
+
}
|
|
286
|
+
return <Box flexDirection="row" alignItems="flex-end" paddingX={1} flexShrink={0}>
|
|
287
|
+
<SpeechBubble text={reaction} color={color} fading={fading} tail="right" />
|
|
288
|
+
{spriteColumn}
|
|
289
|
+
</Box>;
|
|
290
|
+
}
|
|
291
|
+
|
|
292
|
+
// Floating bubble overlay for fullscreen mode. Mounted in FullscreenLayout's
|
|
293
|
+
// bottomFloat slot (outside the overflowY:hidden clip) so it can extend into
|
|
294
|
+
// the ScrollBox region. CompanionSprite owns the clear-after-10s timer; this
|
|
295
|
+
// just reads companionReaction and renders the fade.
|
|
296
|
+
export function CompanionFloatingBubble() {
|
|
297
|
+
const $ = _c(8);
|
|
298
|
+
const reaction = useAppState(_temp);
|
|
299
|
+
let t0;
|
|
300
|
+
if ($[0] !== reaction) {
|
|
301
|
+
t0 = {
|
|
302
|
+
tick: 0,
|
|
303
|
+
forReaction: reaction
|
|
304
|
+
};
|
|
305
|
+
$[0] = reaction;
|
|
306
|
+
$[1] = t0;
|
|
307
|
+
} else {
|
|
308
|
+
t0 = $[1];
|
|
309
|
+
}
|
|
310
|
+
const [t1, setTick] = useState(t0);
|
|
311
|
+
const {
|
|
312
|
+
tick,
|
|
313
|
+
forReaction
|
|
314
|
+
} = t1;
|
|
315
|
+
if (reaction !== forReaction) {
|
|
316
|
+
setTick({
|
|
317
|
+
tick: 0,
|
|
318
|
+
forReaction: reaction
|
|
319
|
+
});
|
|
320
|
+
}
|
|
321
|
+
let t2;
|
|
322
|
+
let t3;
|
|
323
|
+
if ($[2] !== reaction) {
|
|
324
|
+
t2 = () => {
|
|
325
|
+
if (!reaction) {
|
|
326
|
+
return;
|
|
327
|
+
}
|
|
328
|
+
const timer = setInterval(_temp3, TICK_MS, setTick);
|
|
329
|
+
return () => clearInterval(timer);
|
|
330
|
+
};
|
|
331
|
+
t3 = [reaction];
|
|
332
|
+
$[2] = reaction;
|
|
333
|
+
$[3] = t2;
|
|
334
|
+
$[4] = t3;
|
|
335
|
+
} else {
|
|
336
|
+
t2 = $[3];
|
|
337
|
+
t3 = $[4];
|
|
338
|
+
}
|
|
339
|
+
useEffect(t2, t3);
|
|
340
|
+
if (!feature("BUDDY") || !reaction) {
|
|
341
|
+
return null;
|
|
342
|
+
}
|
|
343
|
+
const companion = getCompanion();
|
|
344
|
+
if (!companion || getGlobalConfig().companionMuted) {
|
|
345
|
+
return null;
|
|
346
|
+
}
|
|
347
|
+
const t4 = tick >= BUBBLE_SHOW - FADE_WINDOW;
|
|
348
|
+
let t5;
|
|
349
|
+
if ($[5] !== reaction || $[6] !== t4) {
|
|
350
|
+
t5 = <SpeechBubble text={reaction} color={RARITY_COLORS[companion.rarity]} fading={t4} tail="down" />;
|
|
351
|
+
$[5] = reaction;
|
|
352
|
+
$[6] = t4;
|
|
353
|
+
$[7] = t5;
|
|
354
|
+
} else {
|
|
355
|
+
t5 = $[7];
|
|
356
|
+
}
|
|
357
|
+
return t5;
|
|
358
|
+
}
|
|
359
|
+
function _temp3(set) {
|
|
360
|
+
return set(_temp2);
|
|
361
|
+
}
|
|
362
|
+
function _temp2(s_0) {
|
|
363
|
+
return {
|
|
364
|
+
...s_0,
|
|
365
|
+
tick: s_0.tick + 1
|
|
366
|
+
};
|
|
367
|
+
}
|
|
368
|
+
function _temp(s) {
|
|
369
|
+
return s.companionReaction;
|
|
370
|
+
}
|
|
371
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["feature","figures","React","useEffect","useRef","useState","useTerminalSize","stringWidth","Box","Text","useAppState","useSetAppState","AppState","getGlobalConfig","isFullscreenActive","Theme","getCompanion","renderFace","renderSprite","spriteFrameCount","RARITY_COLORS","TICK_MS","BUBBLE_SHOW","FADE_WINDOW","PET_BURST_MS","IDLE_SEQUENCE","H","heart","PET_HEARTS","wrap","text","width","words","split","lines","cur","w","length","push","SpeechBubble","t0","$","_c","color","fading","tail","T0","borderColor","t1","t2","t3","t4","t5","t6","t7","l","i","undefined","map","bubble","t8","t9","MIN_COLS_FOR_FULL_SPRITE","SPRITE_BODY_WIDTH","NAME_ROW_PAD","SPRITE_PADDING_X","BUBBLE_WIDTH","NARROW_QUIP_CAP","spriteColWidth","nameWidth","Math","max","companionReservedColumns","terminalColumns","speaking","companion","companionMuted","name","CompanionSprite","ReactNode","reaction","s","companionReaction","petAt","companionPetAt","focused","footerSelection","setAppState","columns","tick","setTick","lastSpokeTick","petStartTick","forPetAt","setPetStart","timer","setInterval","setT","t","clearInterval","current","setTimeout","setA","prev","clearTimeout","rarity","colWidth","bubbleAge","petAge","Infinity","petting","quip","slice","label","frameCount","species","heartFrame","spriteFrame","blink","step","body","line","replaceAll","eye","sprite","spriteColumn","CompanionFloatingBubble","_temp","forReaction","_temp3","set","_temp2","s_0"],"sources":["CompanionSprite.tsx"],"sourcesContent":["import { feature } from 'bun:bundle'\nimport figures from 'figures'\nimport React, { useEffect, useRef, useState } from 'react'\nimport { useTerminalSize } from '../hooks/useTerminalSize.js'\nimport { stringWidth } from '../ink/stringWidth.js'\nimport { Box, Text } from '../ink.js'\nimport { useAppState, useSetAppState } from '../state/AppState.js'\nimport type { AppState } from '../state/AppStateStore.js'\nimport { getGlobalConfig } from '../utils/config.js'\nimport { isFullscreenActive } from '../utils/fullscreen.js'\nimport type { Theme } from '../utils/theme.js'\nimport { getCompanion } from './companion.js'\nimport { renderFace, renderSprite, spriteFrameCount } from './sprites.js'\nimport { RARITY_COLORS } from './types.js'\n\nconst TICK_MS = 500\nconst BUBBLE_SHOW = 20 // ticks → ~10s at 500ms\nconst FADE_WINDOW = 6 // last ~3s the bubble dims so you know it's about to go\nconst PET_BURST_MS = 2500 // how long hearts float after /buddy pet\n\n// Idle sequence: mostly rest (frame 0), occasional fidget (frames 1-2), rare blink.\n// Sequence indices map to sprite frames; -1 means \"blink on frame 0\".\nconst IDLE_SEQUENCE = [0, 0, 0, 0, 1, 0, 0, 0, -1, 0, 0, 2, 0, 0, 0]\n\n// Hearts float up-and-out over 5 ticks (~2.5s). Prepended above the sprite.\nconst H = figures.heart\nconst PET_HEARTS = [\n  `   ${H}    ${H}   `,\n  `  ${H}  ${H}   ${H}  `,\n  ` ${H}   ${H}  ${H}   `,\n  `${H}  ${H}      ${H} `,\n  '·    ·   ·  ',\n]\n\nfunction wrap(text: string, width: number): string[] {\n  const words = text.split(' ')\n  const lines: string[] = []\n  let cur = ''\n  for (const w of words) {\n    if (cur.length + w.length + 1 > width && cur) {\n      lines.push(cur)\n      cur = w\n    } else {\n      cur = cur ? `${cur} ${w}` : w\n    }\n  }\n  if (cur) lines.push(cur)\n  return lines\n}\n\nfunction SpeechBubble({\n  text,\n  color,\n  fading,\n  tail,\n}: {\n  text: string\n  color: keyof Theme\n  fading: boolean\n  tail: 'down' | 'right'\n}): React.ReactNode {\n  const lines = wrap(text, 30)\n  const borderColor = fading ? 'inactive' : color\n  const bubble = (\n    <Box\n      flexDirection=\"column\"\n      borderStyle=\"round\"\n      borderColor={borderColor}\n      paddingX={1}\n      width={34}\n    >\n      {lines.map((l, i) => (\n        <Text\n          key={i}\n          italic\n          dimColor={!fading}\n          color={fading ? 'inactive' : undefined}\n        >\n          {l}\n        </Text>\n      ))}\n    </Box>\n  )\n  if (tail === 'right') {\n    return (\n      <Box flexDirection=\"row\" alignItems=\"center\">\n        {bubble}\n        <Text color={borderColor}>─</Text>\n      </Box>\n    )\n  }\n  return (\n    <Box flexDirection=\"column\" alignItems=\"flex-end\" marginRight={1}>\n      {bubble}\n      <Box flexDirection=\"column\" alignItems=\"flex-end\" paddingRight={6}>\n        <Text color={borderColor}>╲ </Text>\n        <Text color={borderColor}>╲</Text>\n      </Box>\n    </Box>\n  )\n}\n\nexport const MIN_COLS_FOR_FULL_SPRITE = 100\nconst SPRITE_BODY_WIDTH = 12\nconst NAME_ROW_PAD = 2 // focused state wraps name in spaces: ` name `\nconst SPRITE_PADDING_X = 2\nconst BUBBLE_WIDTH = 36 // SpeechBubble box (34) + tail column\nconst NARROW_QUIP_CAP = 24\n\nfunction spriteColWidth(nameWidth: number): number {\n  return Math.max(SPRITE_BODY_WIDTH, nameWidth + NAME_ROW_PAD)\n}\n\n// Width the sprite area consumes. PromptInput subtracts this so text wraps\n// correctly. In fullscreen the bubble floats over scrollback (no extra\n// width); in non-fullscreen it sits inline and needs BUBBLE_WIDTH more.\n// Narrow terminals: 0 — REPL.tsx stacks the one-liner on its own row\n// (above input in fullscreen, below in scrollback), so no reservation.\nexport function companionReservedColumns(\n  terminalColumns: number,\n  speaking: boolean,\n): number {\n  if (!feature('BUDDY')) return 0\n  const companion = getCompanion()\n  if (!companion || getGlobalConfig().companionMuted) return 0\n  if (terminalColumns < MIN_COLS_FOR_FULL_SPRITE) return 0\n  const nameWidth = stringWidth(companion.name)\n  const bubble = speaking && !isFullscreenActive() ? BUBBLE_WIDTH : 0\n  return spriteColWidth(nameWidth) + SPRITE_PADDING_X + bubble\n}\n\nexport function CompanionSprite(): React.ReactNode {\n  const reaction = useAppState(s => s.companionReaction)\n  const petAt = useAppState(s => s.companionPetAt)\n  const focused = useAppState(s => s.footerSelection === 'companion')\n  const setAppState = useSetAppState()\n  const { columns } = useTerminalSize()\n  const [tick, setTick] = useState(0)\n  const lastSpokeTick = useRef(0)\n  // Sync-during-render (not useEffect) so the first post-pet render already\n  // has petStartTick=tick and petAge=0 — otherwise frame 0 is skipped.\n  const [{ petStartTick, forPetAt }, setPetStart] = useState({\n    petStartTick: 0,\n    forPetAt: petAt,\n  })\n  if (petAt !== forPetAt) {\n    setPetStart({ petStartTick: tick, forPetAt: petAt })\n  }\n\n  useEffect(() => {\n    const timer = setInterval(\n      setT => setT((t: number) => t + 1),\n      TICK_MS,\n      setTick,\n    )\n    return () => clearInterval(timer)\n  }, [])\n\n  useEffect(() => {\n    if (!reaction) return\n    lastSpokeTick.current = tick\n    const timer = setTimeout(\n      setA =>\n        setA((prev: AppState) =>\n          prev.companionReaction === undefined\n            ? prev\n            : { ...prev, companionReaction: undefined },\n        ),\n      BUBBLE_SHOW * TICK_MS,\n      setAppState,\n    )\n    return () => clearTimeout(timer)\n    // eslint-disable-next-line react-hooks/exhaustive-deps -- tick intentionally captured at reaction-change, not tracked\n  }, [reaction, setAppState])\n\n  if (!feature('BUDDY')) return null\n  const companion = getCompanion()\n  if (!companion || getGlobalConfig().companionMuted) return null\n\n  const color = RARITY_COLORS[companion.rarity]\n  const colWidth = spriteColWidth(stringWidth(companion.name))\n\n  const bubbleAge = reaction ? tick - lastSpokeTick.current : 0\n  const fading =\n    reaction !== undefined && bubbleAge >= BUBBLE_SHOW - FADE_WINDOW\n\n  const petAge = petAt ? tick - petStartTick : Infinity\n  const petting = petAge * TICK_MS < PET_BURST_MS\n\n  // Narrow terminals: collapse to one-line face. When speaking, the quip\n  // replaces the name beside the face (no room for a bubble).\n  if (columns < MIN_COLS_FOR_FULL_SPRITE) {\n    const quip =\n      reaction && reaction.length > NARROW_QUIP_CAP\n        ? reaction.slice(0, NARROW_QUIP_CAP - 1) + '…'\n        : reaction\n    const label = quip\n      ? `\"${quip}\"`\n      : focused\n        ? ` ${companion.name} `\n        : companion.name\n    return (\n      <Box paddingX={1} alignSelf=\"flex-end\">\n        <Text>\n          {petting && <Text color=\"autoAccept\">{figures.heart} </Text>}\n          <Text bold color={color}>\n            {renderFace(companion)}\n          </Text>{' '}\n          <Text\n            italic\n            dimColor={!focused && !reaction}\n            bold={focused}\n            inverse={focused && !reaction}\n            color={\n              reaction\n                ? fading\n                  ? 'inactive'\n                  : color\n                : focused\n                  ? color\n                  : undefined\n            }\n          >\n            {label}\n          </Text>\n        </Text>\n      </Box>\n    )\n  }\n  const frameCount = spriteFrameCount(companion.species)\n  const heartFrame = petting ? PET_HEARTS[petAge % PET_HEARTS.length] : null\n\n  let spriteFrame: number\n  let blink = false\n  if (reaction || petting) {\n    // Excited: cycle all fidget frames fast\n    spriteFrame = tick % frameCount\n  } else {\n    const step = IDLE_SEQUENCE[tick % IDLE_SEQUENCE.length]!\n    if (step === -1) {\n      spriteFrame = 0\n      blink = true\n    } else {\n      spriteFrame = step % frameCount\n    }\n  }\n\n  const body = renderSprite(companion, spriteFrame).map(line =>\n    blink ? line.replaceAll(companion.eye, '-') : line,\n  )\n  const sprite = heartFrame ? [heartFrame, ...body] : body\n\n  // Name row doubles as hint row — unfocused shows dim name + ↓ discovery,\n  // focused shows inverse name. The enter-to-open hint lives in\n  // PromptInputFooter's right column so this row stays one line and the\n  // sprite doesn't jump up when selected. flexShrink=0 stops the\n  // inline-bubble row wrapper from squeezing the sprite to fit.\n  const spriteColumn = (\n    <Box\n      flexDirection=\"column\"\n      flexShrink={0}\n      alignItems=\"center\"\n      width={colWidth}\n    >\n      {sprite.map((line, i) => (\n        <Text key={i} color={i === 0 && heartFrame ? 'autoAccept' : color}>\n          {line}\n        </Text>\n      ))}\n      <Text\n        italic\n        bold={focused}\n        dimColor={!focused}\n        color={focused ? color : undefined}\n        inverse={focused}\n      >\n        {focused ? ` ${companion.name} ` : companion.name}\n      </Text>\n    </Box>\n  )\n\n  if (!reaction) {\n    return <Box paddingX={1}>{spriteColumn}</Box>\n  }\n\n  // Fullscreen: bubble renders separately via CompanionFloatingBubble in\n  // FullscreenLayout's bottomFloat slot (the bottom slot's overflowY:hidden\n  // would clip a position:absolute overlay here). Sprite body only.\n  // Non-fullscreen: bubble sits inline beside the sprite (input shrinks)\n  // because floating into Static scrollback can't be cleared.\n  if (isFullscreenActive()) {\n    return <Box paddingX={1}>{spriteColumn}</Box>\n  }\n  return (\n    <Box flexDirection=\"row\" alignItems=\"flex-end\" paddingX={1} flexShrink={0}>\n      <SpeechBubble\n        text={reaction}\n        color={color}\n        fading={fading}\n        tail=\"right\"\n      />\n      {spriteColumn}\n    </Box>\n  )\n}\n\n// Floating bubble overlay for fullscreen mode. Mounted in FullscreenLayout's\n// bottomFloat slot (outside the overflowY:hidden clip) so it can extend into\n// the ScrollBox region. CompanionSprite owns the clear-after-10s timer; this\n// just reads companionReaction and renders the fade.\nexport function CompanionFloatingBubble(): React.ReactNode {\n  const reaction = useAppState(s => s.companionReaction)\n  const [{ tick, forReaction }, setTick] = useState({\n    tick: 0,\n    forReaction: reaction,\n  })\n\n  // Reset tick synchronously when reaction changes (not in useEffect, which\n  // runs post-render and would show one stale-faded frame). Storing the\n  // reaction the tick is counting FOR alongside the tick itself means the\n  // fade computation never sees a tick from a previous reaction.\n  if (reaction !== forReaction) {\n    setTick({ tick: 0, forReaction: reaction })\n  }\n\n  useEffect(() => {\n    if (!reaction) return\n    const timer = setInterval(\n      set => set(s => ({ ...s, tick: s.tick + 1 })),\n      TICK_MS,\n      setTick,\n    )\n    return () => clearInterval(timer)\n  }, [reaction])\n\n  if (!feature('BUDDY') || !reaction) return null\n  const companion = getCompanion()\n  if (!companion || getGlobalConfig().companionMuted) return null\n\n  return (\n    <SpeechBubble\n      text={reaction}\n      color={RARITY_COLORS[companion.rarity]}\n      fading={tick >= BUBBLE_SHOW - FADE_WINDOW}\n      tail=\"down\"\n    />\n  )\n}\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
import { getGlobalConfig } from '../utils/config.js'
|
|
2
|
+
import {
|
|
3
|
+
type Companion,
|
|
4
|
+
type CompanionBones,
|
|
5
|
+
EYES,
|
|
6
|
+
HATS,
|
|
7
|
+
RARITIES,
|
|
8
|
+
RARITY_WEIGHTS,
|
|
9
|
+
type Rarity,
|
|
10
|
+
SPECIES,
|
|
11
|
+
STAT_NAMES,
|
|
12
|
+
type StatName,
|
|
13
|
+
} from './types.js'
|
|
14
|
+
|
|
15
|
+
// Mulberry32 — tiny seeded PRNG, good enough for picking ducks
|
|
16
|
+
function mulberry32(seed: number): () => number {
|
|
17
|
+
let a = seed >>> 0
|
|
18
|
+
return function () {
|
|
19
|
+
a |= 0
|
|
20
|
+
a = (a + 0x6d2b79f5) | 0
|
|
21
|
+
let t = Math.imul(a ^ (a >>> 15), 1 | a)
|
|
22
|
+
t = (t + Math.imul(t ^ (t >>> 7), 61 | t)) ^ t
|
|
23
|
+
return ((t ^ (t >>> 14)) >>> 0) / 4294967296
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
function hashString(s: string): number {
|
|
28
|
+
if (typeof Bun !== 'undefined') {
|
|
29
|
+
return Number(BigInt(Bun.hash(s)) & 0xffffffffn)
|
|
30
|
+
}
|
|
31
|
+
let h = 2166136261
|
|
32
|
+
for (let i = 0; i < s.length; i++) {
|
|
33
|
+
h ^= s.charCodeAt(i)
|
|
34
|
+
h = Math.imul(h, 16777619)
|
|
35
|
+
}
|
|
36
|
+
return h >>> 0
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
function pick<T>(rng: () => number, arr: readonly T[]): T {
|
|
40
|
+
return arr[Math.floor(rng() * arr.length)]!
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
function rollRarity(rng: () => number): Rarity {
|
|
44
|
+
const total = Object.values(RARITY_WEIGHTS).reduce((a, b) => a + b, 0)
|
|
45
|
+
let roll = rng() * total
|
|
46
|
+
for (const rarity of RARITIES) {
|
|
47
|
+
roll -= RARITY_WEIGHTS[rarity]
|
|
48
|
+
if (roll < 0) return rarity
|
|
49
|
+
}
|
|
50
|
+
return 'common'
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
const RARITY_FLOOR: Record<Rarity, number> = {
|
|
54
|
+
common: 5,
|
|
55
|
+
uncommon: 15,
|
|
56
|
+
rare: 25,
|
|
57
|
+
epic: 35,
|
|
58
|
+
legendary: 50,
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
// One peak stat, one dump stat, rest scattered. Rarity bumps the floor.
|
|
62
|
+
function rollStats(
|
|
63
|
+
rng: () => number,
|
|
64
|
+
rarity: Rarity,
|
|
65
|
+
): Record<StatName, number> {
|
|
66
|
+
const floor = RARITY_FLOOR[rarity]
|
|
67
|
+
const peak = pick(rng, STAT_NAMES)
|
|
68
|
+
let dump = pick(rng, STAT_NAMES)
|
|
69
|
+
while (dump === peak) dump = pick(rng, STAT_NAMES)
|
|
70
|
+
|
|
71
|
+
const stats = {} as Record<StatName, number>
|
|
72
|
+
for (const name of STAT_NAMES) {
|
|
73
|
+
if (name === peak) {
|
|
74
|
+
stats[name] = Math.min(100, floor + 50 + Math.floor(rng() * 30))
|
|
75
|
+
} else if (name === dump) {
|
|
76
|
+
stats[name] = Math.max(1, floor - 10 + Math.floor(rng() * 15))
|
|
77
|
+
} else {
|
|
78
|
+
stats[name] = floor + Math.floor(rng() * 40)
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
return stats
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
const SALT = 'friend-2026-401'
|
|
85
|
+
|
|
86
|
+
export type Roll = {
|
|
87
|
+
bones: CompanionBones
|
|
88
|
+
inspirationSeed: number
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
function rollFrom(rng: () => number): Roll {
|
|
92
|
+
const rarity = rollRarity(rng)
|
|
93
|
+
const bones: CompanionBones = {
|
|
94
|
+
rarity,
|
|
95
|
+
species: pick(rng, SPECIES),
|
|
96
|
+
eye: pick(rng, EYES),
|
|
97
|
+
hat: rarity === 'common' ? 'none' : pick(rng, HATS),
|
|
98
|
+
shiny: rng() < 0.01,
|
|
99
|
+
stats: rollStats(rng, rarity),
|
|
100
|
+
}
|
|
101
|
+
return { bones, inspirationSeed: Math.floor(rng() * 1e9) }
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
// Called from three hot paths (500ms sprite tick, per-keystroke PromptInput,
|
|
105
|
+
// per-turn observer) with the same userId → cache the deterministic result.
|
|
106
|
+
let rollCache: { key: string; value: Roll } | undefined
|
|
107
|
+
export function roll(userId: string): Roll {
|
|
108
|
+
const key = userId + SALT
|
|
109
|
+
if (rollCache?.key === key) return rollCache.value
|
|
110
|
+
const value = rollFrom(mulberry32(hashString(key)))
|
|
111
|
+
rollCache = { key, value }
|
|
112
|
+
return value
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
export function rollWithSeed(seed: string): Roll {
|
|
116
|
+
return rollFrom(mulberry32(hashString(seed)))
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
export function companionUserId(): string {
|
|
120
|
+
const config = getGlobalConfig()
|
|
121
|
+
return config.oauthAccount?.accountUuid ?? config.userID ?? 'anon'
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
// Regenerate bones from userId, merge with stored soul. Bones never persist
|
|
125
|
+
// so species renames and SPECIES-array edits can't break stored companions,
|
|
126
|
+
// and editing config.companion can't fake a rarity.
|
|
127
|
+
export function getCompanion(): Companion | undefined {
|
|
128
|
+
const stored = getGlobalConfig().companion
|
|
129
|
+
if (!stored) return undefined
|
|
130
|
+
const { bones } = roll(companionUserId())
|
|
131
|
+
// bones last so stale bones fields in old-format configs get overridden
|
|
132
|
+
return { ...stored, ...bones }
|
|
133
|
+
}
|
package/buddy/prompt.ts
ADDED
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { feature } from 'bun:bundle'
|
|
2
|
+
import type { Message } from '../types/message.js'
|
|
3
|
+
import type { Attachment } from '../utils/attachments.js'
|
|
4
|
+
import { getGlobalConfig } from '../utils/config.js'
|
|
5
|
+
import { getCompanion } from './companion.js'
|
|
6
|
+
|
|
7
|
+
export function companionIntroText(name: string, species: string): string {
|
|
8
|
+
return `# Companion
|
|
9
|
+
|
|
10
|
+
A small ${species} named ${name} sits beside the user's input box and occasionally comments in a speech bubble. You're not ${name} — it's a separate watcher.
|
|
11
|
+
|
|
12
|
+
When the user addresses ${name} directly (by name), its bubble will answer. Your job in that moment is to stay out of the way: respond in ONE line or less, or just answer any part of the message meant for you. Don't explain that you're not ${name} — they know. Don't narrate what ${name} might say — the bubble handles that.`
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export function getCompanionIntroAttachment(
|
|
16
|
+
messages: Message[] | undefined,
|
|
17
|
+
): Attachment[] {
|
|
18
|
+
if (!feature('BUDDY')) return []
|
|
19
|
+
const companion = getCompanion()
|
|
20
|
+
if (!companion || getGlobalConfig().companionMuted) return []
|
|
21
|
+
|
|
22
|
+
// Skip if already announced for this companion.
|
|
23
|
+
for (const msg of messages ?? []) {
|
|
24
|
+
if (msg.type !== 'attachment') continue
|
|
25
|
+
if (msg.attachment.type !== 'companion_intro') continue
|
|
26
|
+
if (msg.attachment.name === companion.name) return []
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
return [
|
|
30
|
+
{
|
|
31
|
+
type: 'companion_intro',
|
|
32
|
+
name: companion.name,
|
|
33
|
+
species: companion.species,
|
|
34
|
+
},
|
|
35
|
+
]
|
|
36
|
+
}
|