mirror of
https://github.com/lWolvesl/claw-code.git
synced 2026-04-02 22:51:51 +08:00
222 lines
14 KiB
Markdown
222 lines
14 KiB
Markdown
# TUI Enhancement Plan — Claw Code (`rusty-claude-cli`)
|
||
|
||
## Executive Summary
|
||
|
||
This plan covers a comprehensive analysis of the current terminal user interface and proposes phased enhancements that will transform the existing REPL/prompt CLI into a polished, modern TUI experience — while preserving the existing clean architecture and test coverage.
|
||
|
||
---
|
||
|
||
## 1. Current Architecture Analysis
|
||
|
||
### Crate Map
|
||
|
||
| Crate | Purpose | Lines | TUI Relevance |
|
||
|---|---|---|---|
|
||
| `rusty-claude-cli` | Main binary: REPL loop, arg parsing, rendering, API bridge | ~3,600 | **Primary TUI surface** |
|
||
| `runtime` | Session, conversation loop, config, permissions, compaction | ~5,300 | Provides data/state |
|
||
| `api` | Anthropic HTTP client + SSE streaming | ~1,500 | Provides stream events |
|
||
| `commands` | Slash command metadata/parsing/help | ~470 | Drives command dispatch |
|
||
| `tools` | 18 built-in tool implementations | ~3,500 | Tool execution display |
|
||
|
||
### Current TUI Components
|
||
|
||
| Component | File | What It Does Today | Quality |
|
||
|---|---|---|---|
|
||
| **Input** | `input.rs` (269 lines) | `rustyline`-based line editor with slash-command tab completion, Shift+Enter newline, history | ✅ Solid |
|
||
| **Rendering** | `render.rs` (641 lines) | Markdown→terminal rendering (headings, lists, tables, code blocks with syntect highlighting, blockquotes), spinner widget | ✅ Good |
|
||
| **App/REPL loop** | `main.rs` (3,159 lines) | The monolithic `LiveCli` struct: REPL loop, all slash command handlers, streaming output, tool call display, permission prompting, session management | ⚠️ Monolithic |
|
||
| **Alt App** | `app.rs` (398 lines) | An earlier `CliApp` prototype with `ConversationClient`, stream event handling, `TerminalRenderer`, output format support | ⚠️ Appears unused/legacy |
|
||
|
||
### Key Dependencies
|
||
|
||
- **crossterm 0.28** — terminal control (cursor, colors, clear)
|
||
- **pulldown-cmark 0.13** — Markdown parsing
|
||
- **syntect 5** — syntax highlighting
|
||
- **rustyline 15** — line editing with completion
|
||
- **serde_json** — tool I/O formatting
|
||
|
||
### Strengths
|
||
|
||
1. **Clean rendering pipeline**: Markdown rendering is well-structured with state tracking, table rendering, code highlighting
|
||
2. **Rich tool display**: Tool calls get box-drawing borders (`╭─ name ─╮`), results show ✓/✗ icons
|
||
3. **Comprehensive slash commands**: 15 commands covering model switching, permissions, sessions, config, diff, export
|
||
4. **Session management**: Full persistence, resume, list, switch, compaction
|
||
5. **Permission prompting**: Interactive Y/N approval for restricted tool calls
|
||
6. **Thorough tests**: Every formatting function, every parse path has unit tests
|
||
|
||
### Weaknesses & Gaps
|
||
|
||
1. **`main.rs` is a 3,159-line monolith** — all REPL logic, formatting, API bridging, session management, and tests in one file
|
||
2. **No alternate-screen / full-screen layout** — everything is inline scrolling output
|
||
3. **No progress bars** — only a single braille spinner; no indication of streaming progress or token counts during generation
|
||
4. **No visual diff rendering** — `/diff` just dumps raw git diff text
|
||
5. **No syntax highlighting in streamed output** — markdown rendering only applies to tool results, not to the main assistant response stream
|
||
6. **No status bar / HUD** — model, tokens, session info not visible during interaction
|
||
7. **No image/attachment preview** — `SendUserMessage` resolves attachments but never displays them
|
||
8. **Streaming is char-by-char with artificial delay** — `stream_markdown` sleeps 8ms per whitespace-delimited chunk
|
||
9. **No color theme customization** — hardcoded `ColorTheme::default()`
|
||
10. **No resize handling** — no terminal size awareness for wrapping, truncation, or layout
|
||
11. **Dual app structs** — `app.rs` has a separate `CliApp` that duplicates `LiveCli` from `main.rs`
|
||
12. **No pager for long outputs** — `/status`, `/config`, `/memory` can overflow the viewport
|
||
13. **Tool results not collapsible** — large bash outputs flood the screen
|
||
14. **No thinking/reasoning indicator** — when the model is in "thinking" mode, no visual distinction
|
||
15. **No auto-complete for tool arguments** — only slash command names complete
|
||
|
||
---
|
||
|
||
## 2. Enhancement Plan
|
||
|
||
### Phase 0: Structural Cleanup (Foundation)
|
||
|
||
**Goal**: Break the monolith, remove dead code, establish the module structure for TUI work.
|
||
|
||
| Task | Description | Effort |
|
||
|---|---|---|
|
||
| 0.1 | **Extract `LiveCli` into `app.rs`** — Move the entire `LiveCli` struct, its impl, and helpers (`format_*`, `render_*`, session management) out of `main.rs` into focused modules: `app.rs` (core), `format.rs` (report formatting), `session_manager.rs` (session CRUD) | M |
|
||
| 0.2 | **Remove or merge the legacy `CliApp`** — The existing `app.rs` has an unused `CliApp` with its own `ConversationClient`-based rendering. Either delete it or merge its unique features (stream event handler pattern) into the active `LiveCli` | S |
|
||
| 0.3 | **Extract `main.rs` arg parsing** — The current `parse_args()` is a hand-rolled parser that duplicates the clap-based `args.rs`. Consolidate on the hand-rolled parser (it's more feature-complete) and move it to `args.rs`, or adopt clap fully | S |
|
||
| 0.4 | **Create a `tui/` module** — Introduce `crates/rusty-claude-cli/src/tui/mod.rs` as the namespace for all new TUI components: `status_bar.rs`, `layout.rs`, `tool_panel.rs`, etc. | S |
|
||
|
||
### Phase 1: Status Bar & Live HUD
|
||
|
||
**Goal**: Persistent information display during interaction.
|
||
|
||
| Task | Description | Effort |
|
||
|---|---|---|
|
||
| 1.1 | **Terminal-size-aware status line** — Use `crossterm::terminal::size()` to render a bottom-pinned status bar showing: model name, permission mode, session ID, cumulative token count, estimated cost | M |
|
||
| 1.2 | **Live token counter** — Update the status bar in real-time as `AssistantEvent::Usage` and `AssistantEvent::TextDelta` events arrive during streaming | M |
|
||
| 1.3 | **Turn duration timer** — Show elapsed time for the current turn (the `showTurnDuration` config already exists in Config tool but isn't wired up) | S |
|
||
| 1.4 | **Git branch indicator** — Display the current git branch in the status bar (already parsed via `parse_git_status_metadata`) | S |
|
||
|
||
### Phase 2: Enhanced Streaming Output
|
||
|
||
**Goal**: Make the main response stream visually rich and responsive.
|
||
|
||
| Task | Description | Effort |
|
||
|---|---|---|
|
||
| 2.1 | **Live markdown rendering** — Instead of raw text streaming, buffer text deltas and incrementally render Markdown as it arrives (heading detection, bold/italic, inline code). The existing `TerminalRenderer::render_markdown` can be adapted for incremental use | L |
|
||
| 2.2 | **Thinking indicator** — When extended thinking/reasoning is active, show a distinct animated indicator (e.g., `🧠 Reasoning...` with pulsing dots or a different spinner) instead of the generic `🦀 Thinking...` | S |
|
||
| 2.3 | **Streaming progress bar** — Add an optional horizontal progress indicator below the spinner showing approximate completion (based on max_tokens vs. output_tokens so far) | M |
|
||
| 2.4 | **Remove artificial stream delay** — The current `stream_markdown` sleeps 8ms per chunk. For tool results this is fine, but for the main response stream it should be immediate or configurable | S |
|
||
|
||
### Phase 3: Tool Call Visualization
|
||
|
||
**Goal**: Make tool execution legible and navigable.
|
||
|
||
| Task | Description | Effort |
|
||
|---|---|---|
|
||
| 3.1 | **Collapsible tool output** — For tool results longer than N lines (configurable, default 15), show a summary with `[+] Expand` hint; pressing a key reveals the full output. Initially implement as truncation with a "full output saved to file" fallback | M |
|
||
| 3.2 | **Syntax-highlighted tool results** — When tool results contain code (detected by tool name — `bash` stdout, `read_file` content, `REPL` output), apply syntect highlighting rather than rendering as plain text | M |
|
||
| 3.3 | **Tool call timeline** — For multi-tool turns, show a compact summary: `🔧 bash → ✓ | read_file → ✓ | edit_file → ✓ (3 tools, 1.2s)` after all tool calls complete | S |
|
||
| 3.4 | **Diff-aware edit_file display** — When `edit_file` succeeds, show a colored unified diff of the change instead of just `✓ edit_file: path` | M |
|
||
| 3.5 | **Permission prompt enhancement** — Style the approval prompt with box drawing, color the tool name, show a one-line summary of what the tool will do | S |
|
||
|
||
### Phase 4: Enhanced Slash Commands & Navigation
|
||
|
||
**Goal**: Improve information display and add missing features.
|
||
|
||
| Task | Description | Effort |
|
||
|---|---|---|
|
||
| 4.1 | **Colored `/diff` output** — Parse the git diff and render it with red/green coloring for removals/additions, similar to `delta` or `diff-so-fancy` | M |
|
||
| 4.2 | **Pager for long outputs** — When `/status`, `/config`, `/memory`, or `/diff` produce output longer than the terminal height, pipe through an internal pager (scroll with j/k/q) or external `$PAGER` | M |
|
||
| 4.3 | **`/search` command** — Add a new command to search conversation history by keyword | M |
|
||
| 4.4 | **`/undo` command** — Undo the last file edit by restoring from the `originalFile` data in `write_file`/`edit_file` tool results | M |
|
||
| 4.5 | **Interactive session picker** — Replace the text-based `/session list` with an interactive fuzzy-filterable list (up/down arrows to select, enter to switch) | L |
|
||
| 4.6 | **Tab completion for tool arguments** — Extend `SlashCommandHelper` to complete file paths after `/export`, model names after `/model`, session IDs after `/session switch` | M |
|
||
|
||
### Phase 5: Color Themes & Configuration
|
||
|
||
**Goal**: User-customizable visual appearance.
|
||
|
||
| Task | Description | Effort |
|
||
|---|---|---|
|
||
| 5.1 | **Named color themes** — Add `dark` (current default), `light`, `solarized`, `catppuccin` themes. Wire to the existing `Config` tool's `theme` setting | M |
|
||
| 5.2 | **ANSI-256 / truecolor detection** — Detect terminal capabilities and fall back gracefully (no colors → 16 colors → 256 → truecolor) | M |
|
||
| 5.3 | **Configurable spinner style** — Allow choosing between braille dots, bar, moon phases, etc. | S |
|
||
| 5.4 | **Banner customization** — Make the ASCII art banner optional or configurable via settings | S |
|
||
|
||
### Phase 6: Full-Screen TUI Mode (Stretch)
|
||
|
||
**Goal**: Optional alternate-screen layout for power users.
|
||
|
||
| Task | Description | Effort |
|
||
|---|---|---|
|
||
| 6.1 | **Add `ratatui` dependency** — Introduce `ratatui` (terminal UI framework) as an optional dependency for the full-screen mode | S |
|
||
| 6.2 | **Split-pane layout** — Top pane: conversation with scrollback; Bottom pane: input area; Right sidebar (optional): tool status/todo list | XL |
|
||
| 6.3 | **Scrollable conversation view** — Navigate past messages with PgUp/PgDn, search within conversation | L |
|
||
| 6.4 | **Keyboard shortcuts panel** — Show `?` help overlay with all keybindings | M |
|
||
| 6.5 | **Mouse support** — Click to expand tool results, scroll conversation, select text for copy | L |
|
||
|
||
---
|
||
|
||
## 3. Priority Recommendation
|
||
|
||
### Immediate (High Impact, Moderate Effort)
|
||
|
||
1. **Phase 0** — Essential cleanup. The 3,159-line `main.rs` is the #1 maintenance risk and blocks clean TUI additions.
|
||
2. **Phase 1.1–1.2** — Status bar with live tokens. Highest-impact UX win: users constantly want to know token usage.
|
||
3. **Phase 2.4** — Remove artificial delay. Low effort, immediately noticeable improvement.
|
||
4. **Phase 3.1** — Collapsible tool output. Large bash outputs currently wreck readability.
|
||
|
||
### Near-Term (Next Sprint)
|
||
|
||
5. **Phase 2.1** — Live markdown rendering. Makes the core interaction feel polished.
|
||
6. **Phase 3.2** — Syntax-highlighted tool results.
|
||
7. **Phase 3.4** — Diff-aware edit display.
|
||
8. **Phase 4.1** — Colored diff for `/diff`.
|
||
|
||
### Longer-Term
|
||
|
||
9. **Phase 5** — Color themes (user demand-driven).
|
||
10. **Phase 4.2–4.6** — Enhanced navigation and commands.
|
||
11. **Phase 6** — Full-screen mode (major undertaking, evaluate after earlier phases ship).
|
||
|
||
---
|
||
|
||
## 4. Architecture Recommendations
|
||
|
||
### Module Structure After Phase 0
|
||
|
||
```
|
||
crates/rusty-claude-cli/src/
|
||
├── main.rs # Entrypoint, arg dispatch only (~100 lines)
|
||
├── args.rs # CLI argument parsing (consolidate existing two parsers)
|
||
├── app.rs # LiveCli struct, REPL loop, turn execution
|
||
├── format.rs # All report formatting (status, cost, model, permissions, etc.)
|
||
├── session_mgr.rs # Session CRUD: create, resume, list, switch, persist
|
||
├── init.rs # Repo initialization (unchanged)
|
||
├── input.rs # Line editor (unchanged, minor extensions)
|
||
├── render.rs # TerminalRenderer, Spinner (extended)
|
||
└── tui/
|
||
├── mod.rs # TUI module root
|
||
├── status_bar.rs # Persistent bottom status line
|
||
├── tool_panel.rs # Tool call visualization (boxes, timelines, collapsible)
|
||
├── diff_view.rs # Colored diff rendering
|
||
├── pager.rs # Internal pager for long outputs
|
||
└── theme.rs # Color theme definitions and selection
|
||
```
|
||
|
||
### Key Design Principles
|
||
|
||
1. **Keep the inline REPL as the default** — Full-screen TUI should be opt-in (`--tui` flag)
|
||
2. **Everything testable without a terminal** — All formatting functions take `&mut impl Write`, never assume stdout directly
|
||
3. **Streaming-first** — Rendering should work incrementally, not buffering the entire response
|
||
4. **Respect `crossterm` for all terminal control** — Don't mix raw ANSI escape codes with crossterm (the current codebase does this in the startup banner)
|
||
5. **Feature-gate heavy dependencies** — `ratatui` should be behind a `full-tui` feature flag
|
||
|
||
---
|
||
|
||
## 5. Risk Assessment
|
||
|
||
| Risk | Mitigation |
|
||
|---|---|
|
||
| Breaking the working REPL during refactor | Phase 0 is pure restructuring with existing test coverage as safety net |
|
||
| Terminal compatibility issues (tmux, SSH, Windows) | Rely on crossterm's abstraction; test in degraded environments |
|
||
| Performance regression with rich rendering | Profile before/after; keep the fast path (raw streaming) always available |
|
||
| Scope creep into Phase 6 | Ship Phases 0–3 as a coherent release before starting Phase 6 |
|
||
| `app.rs` vs `main.rs` confusion | Phase 0.2 explicitly resolves this by removing the legacy `CliApp` |
|
||
|
||
---
|
||
|
||
*Generated: 2026-03-31 | Workspace: `rust/` | Branch: `dev/rust`*
|