Library 의 / 키가 활성화. App.search slot 이 lazy 채워지고 (run loop 가 SwitchPane(Search) 받을 때),
debounce 200 ms 후 kebab-app::search 호출, 선택된 hit 의 chunk 를 preview pane 에 표시.
g 키로 $EDITOR (vim/nvim/code/cursor 자동 감지) 에서 citation 위치 열림.
핵심:
- SearchState 본체 (`app.rs` 의 forward decl 채움) — input / mode / hits /
selected_hit / input_dirty_at / last_query / searching / preview.
- `src/search.rs` (신규):
- `render_search(f, area, state)` — 3-pane layout (input bar / 결과 리스트 / preview).
각 hit 는 §1.5 dense 4-line format (rank.score URI / heading / snippet).
- `handle_key_search`: typing → input + dirty mark. Tab → mode 순환. Enter →
immediate refresh. j/k → 선택 이동 + preview invalidate. g → editor jump
(RAII raw-mode suspend). Esc → Library 복귀.
- `build_jump_command(citation, editor_env, workspace_root)` 가 vim 류
`+<line> path` / VS Code `code -g path:line` / cursor `cursor -g`
자동 분기. unit test 로 잠금.
- `jump_to_citation` 가 raw-mode + AltScreen 을 RAII 로 suspend/restore
(panic 안전).
- run-loop hook 4 함수: `debounce_due` / `fire_search` /
`refresh_preview` (private to crate).
- run.rs:
- Pane::Search arm 이 `handle_key_search` 로 dispatch + `render_search`.
- SwitchPane(Search) 시 `app.search = Some(SearchState::default())` lazy init.
- Idle tick 마다 debounce_due → fire_search, preview None → refresh_preview.
- 테스트 13개 (`tests/search.rs`) — Esc/typing/backspace/Tab cycle/Enter
refresh/j-k 이동/jump cmd vim+code+args/render w/hits/empty render/no slot.
Spec deviation (HOTFIXES `2026-05-02 P9-2`):
- `render_search<B: Backend>` generic 제거 (P9-1 와 동일 사유 — ratatui 0.28
Frame backend-agnostic).
- `jump_to_citation` 가 `workspace_root: &Path` 인자 추가. Citation.path 가
workspace 상대 라 editor 호출 시 절대 경로 필요. spec literal 의 시그니처
는 unimplementable.
Docs (sync rule):
- README: TUI 행 \"Library + Search 패널, ask/inspect 진행 중\" + Quick start
의 `kebab tui` 코멘트 갱신.
- HANDOFF: 한 줄 요약 + Phase status (P9 1/5 → 2/5) + deviation 한 줄 추가.
- HOTFIXES: P9-2 entry 추가.
- tasks/p9/p9-2 status: completed.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
191 lines
7.0 KiB
Rust
191 lines
7.0 KiB
Rust
//! Run loop — owns the event poll + render cycle. Pane-specific
|
|
//! key handlers are dispatched on focus.
|
|
|
|
use anyhow::Result;
|
|
use crossterm::event::{self, Event, KeyEventKind};
|
|
use ratatui::Frame;
|
|
use ratatui::layout::{Constraint, Direction, Layout, Rect};
|
|
use ratatui::style::{Modifier, Style};
|
|
use ratatui::text::{Line, Span};
|
|
use ratatui::widgets::{Block, Borders, Paragraph};
|
|
use std::time::Duration;
|
|
|
|
use crate::app::{App, KeyOutcome, Pane, SearchState};
|
|
use crate::error_popup::{ErrorOverlay, render_error_overlay};
|
|
use crate::library::{handle_key_library, refresh_docs, render_library};
|
|
use crate::search::{
|
|
debounce_due, fire_search, handle_key_search, refresh_preview, render_search,
|
|
};
|
|
use crate::terminal::TuiTerminal;
|
|
|
|
/// Poll interval for crossterm's `event::poll`. Short enough that a
|
|
/// pending data refresh shows up promptly, long enough that an idle
|
|
/// app doesn't spin the CPU.
|
|
const POLL_INTERVAL: Duration = Duration::from_millis(150);
|
|
|
|
pub(crate) fn run_loop(app: &mut App) -> Result<()> {
|
|
let mut terminal = TuiTerminal::enter()?;
|
|
|
|
while !app.should_quit {
|
|
// Per-pane idle work BEFORE rendering so the frame reflects
|
|
// freshly-loaded state.
|
|
if app.error_overlay.is_none() {
|
|
match app.focus {
|
|
Pane::Library => {
|
|
if app.library.inner.needs_refresh {
|
|
if let Err(e) = refresh_docs(app) {
|
|
app.error_overlay = Some(ErrorOverlay::from_anyhow(&e));
|
|
}
|
|
}
|
|
}
|
|
Pane::Search => {
|
|
let due = app
|
|
.search
|
|
.as_ref()
|
|
.map(debounce_due)
|
|
.unwrap_or(false);
|
|
if due {
|
|
if let Err(e) = fire_search(app) {
|
|
app.error_overlay = Some(ErrorOverlay::from_anyhow(&e));
|
|
}
|
|
}
|
|
// Lazy preview fetch when selection lacks one.
|
|
let needs_preview = app
|
|
.search
|
|
.as_ref()
|
|
.map(|s| s.preview.is_none() && !s.hits.is_empty())
|
|
.unwrap_or(false);
|
|
if needs_preview {
|
|
if let Err(e) = refresh_preview(app) {
|
|
app.error_overlay = Some(ErrorOverlay::from_anyhow(&e));
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
terminal.inner.draw(|f| render_root(f, app))?;
|
|
|
|
if event::poll(POLL_INTERVAL)? {
|
|
match event::read()? {
|
|
Event::Key(key) if key.kind == KeyEventKind::Press => {
|
|
let outcome = match app.focus {
|
|
Pane::Library => handle_key_library(app, key),
|
|
Pane::Search => handle_key_search(app, key),
|
|
// p9-3/4/5 plug their handlers here as their
|
|
// crates land. Until then, those panes accept
|
|
// only `q` / `Esc` to return.
|
|
Pane::Ask | Pane::Inspect | Pane::Jobs => {
|
|
handle_key_unimplemented_pane(app, key)
|
|
}
|
|
};
|
|
match outcome {
|
|
KeyOutcome::Quit => app.should_quit = true,
|
|
KeyOutcome::SwitchPane(p) => {
|
|
app.focus = p;
|
|
// Lazy-init pane state on first switch.
|
|
if p == Pane::Search && app.search.is_none() {
|
|
app.search = Some(SearchState::default());
|
|
}
|
|
}
|
|
KeyOutcome::Refresh => {
|
|
// Library uses needs_refresh; Search uses
|
|
// input_dirty_at — pane-specific. The next
|
|
// loop iteration's idle pass services it.
|
|
}
|
|
KeyOutcome::Continue => {}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Stub key handler for panes whose authoring task has not landed
|
|
/// yet. `q` / `Esc` returns to Library; everything else is a no-op.
|
|
fn handle_key_unimplemented_pane(
|
|
app: &mut App,
|
|
key: crossterm::event::KeyEvent,
|
|
) -> KeyOutcome {
|
|
use crossterm::event::KeyCode;
|
|
if app.error_overlay.is_some() {
|
|
app.error_overlay = None;
|
|
return KeyOutcome::Continue;
|
|
}
|
|
match key.code {
|
|
KeyCode::Char('q') | KeyCode::Esc => KeyOutcome::SwitchPane(Pane::Library),
|
|
_ => KeyOutcome::Continue,
|
|
}
|
|
}
|
|
|
|
fn render_root(f: &mut Frame, app: &App) {
|
|
let outer = Layout::default()
|
|
.direction(Direction::Vertical)
|
|
.constraints([
|
|
Constraint::Length(1),
|
|
Constraint::Min(1),
|
|
Constraint::Length(1),
|
|
])
|
|
.split(f.area());
|
|
render_header(f, outer[0], app);
|
|
match app.focus {
|
|
Pane::Library => render_library(f, outer[1], app),
|
|
Pane::Search => render_search(f, outer[1], app),
|
|
// p9-3/4/5 panes are not yet rendered; placeholder is the
|
|
// Library frame — focus state already reads "Search" /
|
|
// "Ask" / etc. in the header so the user is not misled.
|
|
_ => render_library(f, outer[1], app),
|
|
}
|
|
render_footer(f, outer[2], app);
|
|
if let Some(err) = &app.error_overlay {
|
|
render_error_overlay(f, f.area(), err);
|
|
}
|
|
}
|
|
|
|
fn render_header(f: &mut Frame, area: Rect, app: &App) {
|
|
let pane_label = match app.focus {
|
|
Pane::Library => "Library",
|
|
Pane::Search => "Search",
|
|
Pane::Ask => "Ask",
|
|
Pane::Inspect => "Inspect",
|
|
Pane::Jobs => "Jobs",
|
|
};
|
|
let line = Line::from(vec![
|
|
Span::styled(
|
|
"kebab",
|
|
Style::default().add_modifier(Modifier::BOLD),
|
|
),
|
|
Span::raw(" / "),
|
|
Span::raw(pane_label),
|
|
]);
|
|
f.render_widget(Paragraph::new(line), area);
|
|
}
|
|
|
|
fn render_footer(f: &mut Frame, area: Rect, app: &App) {
|
|
let hints = match app.focus {
|
|
Pane::Library => {
|
|
if app.library.inner.filter_edit.is_some() {
|
|
"Tab=field Enter=apply Esc=cancel"
|
|
} else {
|
|
"j/k=move gg=top G=bottom f=filter /=search ?=ask Enter=inspect q=quit"
|
|
}
|
|
}
|
|
Pane::Search => "type=query Tab=mode Enter=search j/k=move g=open in $EDITOR Esc=back",
|
|
Pane::Ask => "Ask pane not yet implemented (lands with p9-3) — q to return",
|
|
Pane::Inspect => "Inspect pane not yet implemented (lands with p9-4) — q to return",
|
|
Pane::Jobs => "Jobs pane not yet implemented — q to return",
|
|
};
|
|
let line = Line::from(Span::styled(
|
|
hints,
|
|
Style::default().add_modifier(Modifier::DIM),
|
|
));
|
|
f.render_widget(
|
|
Paragraph::new(line).block(Block::default().borders(Borders::TOP)),
|
|
area,
|
|
);
|
|
}
|