0fcf66fc22
Interactive Prompts: - Unified InteractivePrompt type across all 3 adapters (Claude/Codex/Gemini) - InteractivePromptOverlay component with options, text input, countdown - Gemini + Codex pane monitors detect tool confirmation, ask user, plan approval - respondInteractivePrompt routing: permission → respondPermission, options → _selectOption - Claude AskUserQuestion nested questions[0] structure parsing Cross-AI Review: - Client-generated reviewId, removed pendingReview state - FloatingReviewPanel uses CSS display:none instead of unmount (keeps hooks alive) - Child review sessions default to YOLO/bypass permission mode - Send back to parent, send to existing/new review, tab switching, end review - Collapsed review cards with read-only panel for ended reviews - Full reconnect support: active + ended reviews restore correctly AskUserQuestion Tool Card UI: - Dedicated renderer replaces raw JSON display - Options shown with selected (green) / unselected (gray) indicators - Free text answers shown in quoted format with green border - Collapsed summary: question → answer - Shared parseAskQuestionInput utility (client + server) - Historical tool results attached via _result on tool_use blocks Adapter Fixes: - Session→adapter mapping persisted in SQLite (survives server restart) - SESSION_CREATED deferred for pendingRekey adapters (Codex/Gemini) - session-rekeyed handler sends complete SESSION_CREATED with adapter + cwd - Gemini: auto-accept folder trust, privacy notice, IDE nudge, YOLO * prompt - Claude: auto-accept bypass permissions confirmation (v2.1.85+) - Port fallback (EADDRINUSE → try +1), statusLine shell script wrapper Other: - Desktop Enter sends / Shift+Enter newline; Mobile Enter newline - Strip CLAWTAP_REF marker from session list - Active sessions tab shows adapter badge - Rename CLAUDE_UI_PASSWORD → CLAWTAP_PASSWORD Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
19 lines
765 B
TypeScript
19 lines
765 B
TypeScript
/** Parse Claude's AskUserQuestion nested input structure into a flat format */
|
|
export function parseAskQuestionInput(input: any): {
|
|
question: string;
|
|
header?: string;
|
|
options?: { label: string; value: string }[];
|
|
} {
|
|
const q = input?.questions?.[0] || input || {};
|
|
const question = q.question || q.text || input?.question || input?.text || '';
|
|
const header = q.header;
|
|
const rawOpts = q.options || input?.options;
|
|
const options = Array.isArray(rawOpts) && rawOpts.length > 0
|
|
? rawOpts.map((o: any, i: number) => ({
|
|
value: typeof o === 'string' ? String(i) : (o.value ?? String(i)),
|
|
label: typeof o === 'string' ? o : (o.label || o.text || `Option ${i + 1}`),
|
|
}))
|
|
: undefined;
|
|
return { question, header, options };
|
|
}
|