feat: ClawTap v0.2.0

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>
This commit is contained in:
kuannnn
2026-03-27 14:46:00 +08:00
parent 16f75379af
commit 0fcf66fc22
50 changed files with 2179 additions and 400 deletions
+96 -1
View File
@@ -13,6 +13,16 @@
// that will be refined through empirical testing with the actual Codex TUI.
import { EventEmitter } from 'events';
import { InteractivePrompt } from '../../types/messages.js';
function simpleHash(str: string): string {
let hash = 0;
for (let i = 0; i < str.length; i++) {
hash = ((hash << 5) - hash) + str.charCodeAt(i);
hash |= 0;
}
return Math.abs(hash).toString(36);
}
/** Minimal interface for the tmux manager dependency */
interface TmuxCapture {
@@ -42,6 +52,7 @@ export class CodexPaneMonitor {
private interval: ReturnType<typeof setInterval> | null = null;
private _lastContent: string = '';
private _lastResponseText: string = '';
private lastPromptId: string | null = null;
constructor(
sessionId: string,
@@ -84,7 +95,19 @@ export class CodexPaneMonitor {
if (content === this._lastContent) return;
this._lastContent = content;
// 1. Check for approval prompt (highest priority — blocks everything)
// 0. Check for interactive prompt (highest priority)
const interactivePrompt = this._detectPrompt(content);
if (interactivePrompt) {
if (interactivePrompt.requestId !== this.lastPromptId) {
this.lastPromptId = interactivePrompt.requestId;
this.emitter.emit('interactive-prompt', this.sessionId, interactivePrompt);
}
return; // Don't process streaming while prompt is showing
} else if (this.lastPromptId) {
this.lastPromptId = null;
}
// 1. Check for approval prompt (legacy — kept for backwards compat)
const approval = detectApprovalPrompt(content);
if (approval) {
this.emitter.emit('approval-prompt', this.sessionId, approval);
@@ -108,6 +131,78 @@ export class CodexPaneMonitor {
// Silently ignore — tmux window may have been killed
}
}
/**
* Detect an interactive prompt in the Codex CLI pane content.
* Returns an InteractivePrompt if one is detected, null otherwise.
*/
private _detectPrompt(content: string): InteractivePrompt | null {
// Command/File/Network Approval: "(y)" with proceed/run/make patterns
if (
content.includes('(y)') &&
(/proceed/i.test(content) || /Would you like to run/i.test(content) || /Would you like to make/i.test(content))
) {
const options = this._parseCodexOptions(content);
const lines = content.split('\n');
const tail = lines.slice(-20);
const promptLine = tail.find(l => /proceed|\brun\b|\bmake\b/i.test(l)) || 'Approve action';
const description = tail.join('\n').trim();
return {
requestId: `codex-perm-${simpleHash(description)}`,
promptType: 'permission',
title: typeof promptLine === 'string' ? promptLine.trim() : 'Approve action',
description,
options: options.length > 0 ? options : [
{ value: 'y', label: 'Yes' },
{ value: 'n', label: 'No' },
],
};
}
// User Input: "enter to submit" AND "esc to cancel" (but NOT approval patterns)
if (
/enter to submit/i.test(content) &&
/esc to cancel/i.test(content) &&
!content.includes('(y)')
) {
const lines = content.split('\n');
const tail = lines.slice(-20);
const options = this._parseCodexOptions(content);
const description = tail.join('\n').trim();
if (options.length > 0) {
return {
requestId: `codex-ask-${simpleHash(description)}`,
promptType: 'question',
title: 'User Input',
description,
options,
};
}
return {
requestId: `codex-ask-${simpleHash(description)}`,
promptType: 'question',
title: 'User Input',
description,
textInput: { placeholder: 'Type your response...' },
};
}
return null;
}
/**
* Parse Codex-style options from content.
* Matches patterns like "(y) Yes" or "(a) Always approve".
*/
private _parseCodexOptions(content: string): { value: string; label: string }[] {
const results: { value: string; label: string }[] = [];
const regex = /\(([a-z])\)\s+(.+?)(?:\n|$)/g;
let match: RegExpExecArray | null;
while ((match = regex.exec(content)) !== null) {
results.push({ value: match[1]!, label: match[2]!.trim() });
}
return results;
}
}
// =============================================================================