refactor: rename wait_for_approval to ask_user
More general-purpose user intervention tool — not just approve/reject, but any question or input request. Renames across Rust backend, Vue frontend, prompts, and status strings. Tool: wait_for_approval → ask_user (param: reason → question) Status: WaitingApproval → WaitingUser, waiting_approval → waiting_user Enum: NeedsApproval → NeedsInput
This commit is contained in:
111
src/api/mod.rs
111
src/api/mod.rs
@@ -115,6 +115,15 @@ async fn serve_project_file(
|
||||
|
||||
match tokio::fs::read(&full_path).await {
|
||||
Ok(bytes) => {
|
||||
// Render markdown files as HTML
|
||||
if full_path.extension().is_some_and(|e| e == "md") {
|
||||
let md = String::from_utf8_lossy(&bytes);
|
||||
let html = render_markdown_page(&md, &file_path);
|
||||
return (
|
||||
[(axum::http::header::CONTENT_TYPE, "text/html; charset=utf-8".to_string())],
|
||||
html,
|
||||
).into_response();
|
||||
}
|
||||
let mime = mime_guess::from_path(&full_path)
|
||||
.first_or_octet_stream()
|
||||
.to_string();
|
||||
@@ -127,3 +136,105 @@ async fn serve_project_file(
|
||||
}
|
||||
}
|
||||
|
||||
fn render_markdown_page(markdown: &str, title: &str) -> String {
|
||||
use pulldown_cmark::{Parser, Options, html};
|
||||
let mut opts = Options::empty();
|
||||
opts.insert(Options::ENABLE_TABLES);
|
||||
opts.insert(Options::ENABLE_STRIKETHROUGH);
|
||||
opts.insert(Options::ENABLE_TASKLISTS);
|
||||
let parser = Parser::new_ext(markdown, opts);
|
||||
let mut body = String::new();
|
||||
html::push_html(&mut body, parser);
|
||||
|
||||
format!(r#"<!doctype html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>{title}</title>
|
||||
<style>
|
||||
:root {{
|
||||
--text-primary: #1a1a2e;
|
||||
--text-secondary: #6b7280;
|
||||
--bg-primary: #ffffff;
|
||||
--bg-secondary: #f7f8fa;
|
||||
--border: #e2e5ea;
|
||||
--accent: #2563eb;
|
||||
--error: #dc2626;
|
||||
}}
|
||||
body {{
|
||||
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
|
||||
background: var(--bg-primary);
|
||||
color: var(--text-primary);
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
}}
|
||||
.page {{
|
||||
max-width: 860px;
|
||||
margin: 0 auto;
|
||||
padding: 24px 32px;
|
||||
min-height: 100vh;
|
||||
}}
|
||||
.toolbar {{
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 16px;
|
||||
margin-bottom: 24px;
|
||||
padding-bottom: 12px;
|
||||
border-bottom: 1px solid var(--border);
|
||||
}}
|
||||
.toolbar a {{
|
||||
color: var(--accent);
|
||||
text-decoration: none;
|
||||
font-size: 14px;
|
||||
font-weight: 500;
|
||||
}}
|
||||
.toolbar a:hover {{ text-decoration: underline; }}
|
||||
.toolbar .title {{
|
||||
font-size: 14px;
|
||||
font-weight: 600;
|
||||
color: var(--text-secondary);
|
||||
letter-spacing: 0.5px;
|
||||
}}
|
||||
.body {{ line-height: 1.7; font-size: 15px; }}
|
||||
.body h1 {{ font-size: 24px; font-weight: 700; margin: 0 0 16px; padding-bottom: 8px; border-bottom: 2px solid var(--border); }}
|
||||
.body h2 {{ font-size: 20px; font-weight: 600; margin: 24px 0 12px; }}
|
||||
.body h3 {{ font-size: 16px; font-weight: 600; margin: 20px 0 8px; }}
|
||||
.body p {{ margin: 0 0 12px; }}
|
||||
.body ul, .body ol {{ margin: 0 0 12px; padding-left: 24px; }}
|
||||
.body li {{ margin-bottom: 4px; }}
|
||||
.body pre {{
|
||||
background: var(--bg-secondary);
|
||||
border: 1px solid var(--border);
|
||||
border-radius: 6px;
|
||||
padding: 12px 16px;
|
||||
overflow-x: auto;
|
||||
margin: 0 0 12px;
|
||||
font-size: 13px;
|
||||
line-height: 1.5;
|
||||
}}
|
||||
.body code {{ font-family: 'JetBrains Mono', 'Fira Code', monospace; font-size: 13px; }}
|
||||
.body :not(pre) > code {{ background: var(--bg-secondary); padding: 2px 6px; border-radius: 4px; }}
|
||||
.body table {{ width: 100%; border-collapse: collapse; margin: 0 0 12px; font-size: 14px; }}
|
||||
.body th, .body td {{ border: 1px solid var(--border); padding: 8px 12px; text-align: left; }}
|
||||
.body th {{ background: var(--bg-secondary); font-weight: 600; }}
|
||||
.body blockquote {{ border-left: 3px solid var(--accent); padding-left: 16px; margin: 0 0 12px; color: var(--text-secondary); }}
|
||||
.body a {{ color: var(--accent); text-decoration: none; }}
|
||||
.body a:hover {{ text-decoration: underline; }}
|
||||
.body img {{ max-width: 100%; border-radius: 6px; }}
|
||||
.body hr {{ border: none; border-top: 1px solid var(--border); margin: 20px 0; }}
|
||||
.body input[type="checkbox"] {{ margin-right: 6px; }}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="page">
|
||||
<div class="toolbar">
|
||||
<a href="javascript:history.back()">← 返回</a>
|
||||
<span class="title">{title}</span>
|
||||
</div>
|
||||
<div class="body">{body}</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>"#, title = title, body = body)
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user