mirror of
https://github.com/harivansh-afk/sandbox-agent.git
synced 2026-04-17 12:04:15 +00:00
- Add delete button on ended sessions (visible on hover) - Darken ended sessions with opacity and "ended" pill badge - Sort ended sessions to bottom of list - Add token usage pill in chat header - Disable input when session ended - Add Official Skills dropdown with SDK and Rivet presets - Format session IDs shorter with full ID on hover - Add arrow icon to "Configure persistence" link - Add agent logo SVGs Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
407 lines
14 KiB
TypeScript
407 lines
14 KiB
TypeScript
import { ChevronDown, ChevronRight, FolderOpen, Loader2, Plus, Trash2 } from "lucide-react";
|
|
import { useCallback, useEffect, useRef, useState } from "react";
|
|
import type { SandboxAgent } from "sandbox-agent";
|
|
import { formatJson } from "../../utils/format";
|
|
|
|
type SkillEntry = {
|
|
name: string;
|
|
config: { sources: Array<{ source: string; type: string; ref?: string | null; subpath?: string | null; skills?: string[] | null }> };
|
|
};
|
|
|
|
const SkillsTab = ({
|
|
getClient,
|
|
}: {
|
|
getClient: () => SandboxAgent;
|
|
}) => {
|
|
const officialSkills = [
|
|
{
|
|
name: "Sandbox Agent SDK",
|
|
skillId: "sandbox-agent",
|
|
source: "rivet-dev/skills",
|
|
summary: "Skills bundle for fast Sandbox Agent SDK setup and consistent workflows.",
|
|
},
|
|
{
|
|
name: "Rivet",
|
|
skillId: "rivet",
|
|
source: "rivet-dev/skills",
|
|
summary: "Open-source platform for building, deploying, and scaling AI agents.",
|
|
features: [
|
|
"Session Persistence",
|
|
"Resumable Sessions",
|
|
"Multi-Agent Support",
|
|
"Realtime Events",
|
|
"Tool Call Visibility",
|
|
],
|
|
},
|
|
];
|
|
|
|
const [directory, setDirectory] = useState("/");
|
|
const [entries, setEntries] = useState<SkillEntry[]>([]);
|
|
const [loading, setLoading] = useState(false);
|
|
const [error, setError] = useState<string | null>(null);
|
|
const [copiedId, setCopiedId] = useState<string | null>(null);
|
|
const [showSdkSkills, setShowSdkSkills] = useState(false);
|
|
const dropdownRef = useRef<HTMLDivElement | null>(null);
|
|
|
|
useEffect(() => {
|
|
if (!showSdkSkills) return;
|
|
const handler = (event: MouseEvent) => {
|
|
if (!dropdownRef.current) return;
|
|
if (!dropdownRef.current.contains(event.target as Node)) {
|
|
setShowSdkSkills(false);
|
|
}
|
|
};
|
|
document.addEventListener("mousedown", handler);
|
|
return () => document.removeEventListener("mousedown", handler);
|
|
}, [showSdkSkills]);
|
|
|
|
// Add form state
|
|
const [editing, setEditing] = useState(false);
|
|
const [editName, setEditName] = useState("");
|
|
const [editSource, setEditSource] = useState("");
|
|
const [editType, setEditType] = useState("github");
|
|
const [editRef, setEditRef] = useState("");
|
|
const [editSubpath, setEditSubpath] = useState("");
|
|
const [editSkills, setEditSkills] = useState("");
|
|
const [editError, setEditError] = useState<string | null>(null);
|
|
const [saving, setSaving] = useState(false);
|
|
|
|
const loadAll = useCallback(async (dir: string) => {
|
|
setLoading(true);
|
|
setError(null);
|
|
try {
|
|
const configPath = `${dir === "/" ? "" : dir}/.sandbox-agent/config/skills.json`;
|
|
const bytes = await getClient().readFsFile({ path: configPath });
|
|
const text = new TextDecoder().decode(bytes);
|
|
if (!text.trim()) {
|
|
setEntries([]);
|
|
return;
|
|
}
|
|
const map = JSON.parse(text) as Record<string, SkillEntry["config"]>;
|
|
setEntries(
|
|
Object.entries(map).map(([name, config]) => ({ name, config })),
|
|
);
|
|
} catch {
|
|
// File doesn't exist yet or is empty — that's fine
|
|
setEntries([]);
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
}, [getClient]);
|
|
|
|
useEffect(() => {
|
|
loadAll(directory);
|
|
}, [directory, loadAll]);
|
|
|
|
const startAdd = () => {
|
|
setEditing(true);
|
|
setEditName("");
|
|
setEditSource("rivet-dev/skills");
|
|
setEditType("github");
|
|
setEditRef("");
|
|
setEditSubpath("");
|
|
setEditSkills("sandbox-agent");
|
|
setEditError(null);
|
|
};
|
|
|
|
const cancelEdit = () => {
|
|
setEditing(false);
|
|
setEditName("");
|
|
setEditSource("");
|
|
setEditType("github");
|
|
setEditRef("");
|
|
setEditSubpath("");
|
|
setEditSkills("");
|
|
setEditError(null);
|
|
};
|
|
|
|
const save = async () => {
|
|
const name = editName.trim();
|
|
if (!name) {
|
|
setEditError("Name is required");
|
|
return;
|
|
}
|
|
const source = editSource.trim();
|
|
if (!source) {
|
|
setEditError("Source is required");
|
|
return;
|
|
}
|
|
|
|
const skillEntry: SkillEntry["config"]["sources"][0] = {
|
|
source,
|
|
type: editType,
|
|
};
|
|
if (editRef.trim()) skillEntry.ref = editRef.trim();
|
|
if (editSubpath.trim()) skillEntry.subpath = editSubpath.trim();
|
|
const skillsList = editSkills.trim()
|
|
? editSkills.split(",").map((s) => s.trim()).filter(Boolean)
|
|
: null;
|
|
if (skillsList && skillsList.length > 0) skillEntry.skills = skillsList;
|
|
|
|
const config = { sources: [skillEntry] };
|
|
|
|
setSaving(true);
|
|
setEditError(null);
|
|
try {
|
|
await getClient().setSkillsConfig(
|
|
{ directory, skillName: name },
|
|
config,
|
|
);
|
|
cancelEdit();
|
|
await loadAll(directory);
|
|
} catch (err) {
|
|
setEditError(err instanceof Error ? err.message : "Failed to save");
|
|
} finally {
|
|
setSaving(false);
|
|
}
|
|
};
|
|
|
|
const remove = async (name: string) => {
|
|
try {
|
|
await getClient().deleteSkillsConfig({ directory, skillName: name });
|
|
await loadAll(directory);
|
|
} catch (err) {
|
|
setError(err instanceof Error ? err.message : "Failed to delete");
|
|
}
|
|
};
|
|
|
|
const fallbackCopy = (text: string) => {
|
|
const textarea = document.createElement("textarea");
|
|
textarea.value = text;
|
|
textarea.style.position = "fixed";
|
|
textarea.style.opacity = "0";
|
|
document.body.appendChild(textarea);
|
|
textarea.select();
|
|
document.execCommand("copy");
|
|
document.body.removeChild(textarea);
|
|
};
|
|
|
|
const copyText = async (id: string, text: string) => {
|
|
try {
|
|
if (navigator.clipboard && window.isSecureContext) {
|
|
await navigator.clipboard.writeText(text);
|
|
} else {
|
|
fallbackCopy(text);
|
|
}
|
|
setCopiedId(id);
|
|
window.setTimeout(() => {
|
|
setCopiedId((current) => (current === id ? null : current));
|
|
}, 1800);
|
|
} catch {
|
|
setError("Failed to copy snippet");
|
|
}
|
|
};
|
|
|
|
const applySkillPreset = (skill: typeof officialSkills[0]) => {
|
|
setEditing(true);
|
|
setEditName(skill.skillId);
|
|
setEditSource(skill.source);
|
|
setEditType("github");
|
|
setEditRef("");
|
|
setEditSubpath("");
|
|
setEditSkills(skill.skillId);
|
|
setEditError(null);
|
|
setShowSdkSkills(false);
|
|
};
|
|
|
|
const copySkillToInput = async (skillId: string) => {
|
|
const skill = officialSkills.find((s) => s.skillId === skillId);
|
|
if (skill) {
|
|
applySkillPreset(skill);
|
|
await copyText(`skill-input-${skillId}`, skillId);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<>
|
|
<div className="inline-row" style={{ marginBottom: 12, justifyContent: "space-between" }}>
|
|
<span className="card-meta">Skills Configuration</span>
|
|
<div className="inline-row" style={{ gap: 6 }}>
|
|
<div style={{ position: "relative" }} ref={dropdownRef}>
|
|
<button
|
|
className="button secondary small"
|
|
onClick={() => setShowSdkSkills((prev) => !prev)}
|
|
title="Toggle official skills list"
|
|
>
|
|
{showSdkSkills ? <ChevronDown className="button-icon" style={{ width: 12, height: 12 }} /> : <ChevronRight className="button-icon" style={{ width: 12, height: 12 }} />}
|
|
Official Skills
|
|
</button>
|
|
{showSdkSkills && (
|
|
<div
|
|
style={{
|
|
position: "absolute",
|
|
top: "100%",
|
|
right: 0,
|
|
marginTop: 4,
|
|
width: 320,
|
|
background: "var(--bg)",
|
|
border: "1px solid var(--border)",
|
|
borderRadius: 8,
|
|
padding: 12,
|
|
zIndex: 100,
|
|
boxShadow: "0 4px 12px rgba(0,0,0,0.5)",
|
|
}}
|
|
>
|
|
<div className="card-meta" style={{ marginBottom: 8 }}>
|
|
Pick a skill to auto-fill the form.
|
|
</div>
|
|
{officialSkills.map((skill) => (
|
|
<div
|
|
key={skill.name}
|
|
style={{
|
|
border: "1px solid var(--border)",
|
|
borderRadius: 6,
|
|
padding: "8px 10px",
|
|
background: "var(--surface-2)",
|
|
marginBottom: 6,
|
|
}}
|
|
>
|
|
<div className="inline-row" style={{ justifyContent: "space-between", gap: 8, marginBottom: 4 }}>
|
|
<div style={{ fontWeight: 500, fontSize: 12 }}>{skill.name}</div>
|
|
<button className="button ghost small" onClick={() => void copySkillToInput(skill.skillId)}>
|
|
{copiedId === `skill-input-${skill.skillId}` ? "Filled" : "Use"}
|
|
</button>
|
|
</div>
|
|
<div className="card-meta" style={{ fontSize: 10, marginBottom: skill.features ? 6 : 0 }}>{skill.summary}</div>
|
|
{skill.features && (
|
|
<div style={{ display: "flex", flexWrap: "wrap", gap: 4 }}>
|
|
{skill.features.map((feature) => (
|
|
<span key={feature} className="pill accent" style={{ fontSize: 9 }}>
|
|
{feature}
|
|
</span>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
))}
|
|
</div>
|
|
)}
|
|
</div>
|
|
{!editing && (
|
|
<button className="button secondary small" onClick={startAdd}>
|
|
<Plus className="button-icon" style={{ width: 12, height: 12 }} />
|
|
Add
|
|
</button>
|
|
)}
|
|
</div>
|
|
</div>
|
|
|
|
<div className="inline-row" style={{ marginBottom: 12, gap: 6 }}>
|
|
<FolderOpen size={14} className="muted" style={{ flexShrink: 0 }} />
|
|
<input
|
|
className="setup-input mono"
|
|
value={directory}
|
|
onChange={(e) => setDirectory(e.target.value)}
|
|
placeholder="/"
|
|
style={{ flex: 1, fontSize: 11 }}
|
|
/>
|
|
</div>
|
|
|
|
{error && <div className="banner error">{error}</div>}
|
|
{loading && <div className="card-meta">Loading...</div>}
|
|
|
|
{editing && (
|
|
<div className="card" style={{ marginBottom: 12 }}>
|
|
<div className="card-header">
|
|
<span className="card-title">Add Skill Source</span>
|
|
</div>
|
|
<div style={{ marginTop: 8 }}>
|
|
<input
|
|
className="setup-input"
|
|
value={editName}
|
|
onChange={(e) => { setEditName(e.target.value); setEditError(null); }}
|
|
placeholder="skill-name"
|
|
style={{ marginBottom: 6, width: "100%", boxSizing: "border-box" }}
|
|
/>
|
|
<div className="inline-row" style={{ marginBottom: 6, gap: 4 }}>
|
|
<select
|
|
className="setup-select"
|
|
value={editType}
|
|
onChange={(e) => setEditType(e.target.value)}
|
|
style={{ width: 90 }}
|
|
>
|
|
<option value="github">github</option>
|
|
<option value="local">local</option>
|
|
<option value="git">git</option>
|
|
</select>
|
|
<input
|
|
className="setup-input mono"
|
|
value={editSource}
|
|
onChange={(e) => { setEditSource(e.target.value); setEditError(null); }}
|
|
placeholder={editType === "github" ? "owner/repo" : editType === "local" ? "/path/to/skill" : "https://..."}
|
|
style={{ flex: 1 }}
|
|
/>
|
|
</div>
|
|
<input
|
|
className="setup-input"
|
|
value={editSkills}
|
|
onChange={(e) => setEditSkills(e.target.value)}
|
|
placeholder="Skills filter (comma-separated, optional)"
|
|
style={{ marginBottom: 6, width: "100%", boxSizing: "border-box" }}
|
|
/>
|
|
{editType !== "local" && (
|
|
<div className="inline-row" style={{ gap: 4 }}>
|
|
<input
|
|
className="setup-input mono"
|
|
value={editRef}
|
|
onChange={(e) => setEditRef(e.target.value)}
|
|
placeholder="Branch/tag (optional)"
|
|
style={{ flex: 1 }}
|
|
/>
|
|
<input
|
|
className="setup-input mono"
|
|
value={editSubpath}
|
|
onChange={(e) => setEditSubpath(e.target.value)}
|
|
placeholder="Subpath (optional)"
|
|
style={{ flex: 1 }}
|
|
/>
|
|
</div>
|
|
)}
|
|
{editError && <div className="banner error" style={{ marginTop: 4 }}>{editError}</div>}
|
|
</div>
|
|
<div className="card-actions">
|
|
<button className="button primary small" onClick={save} disabled={saving}>
|
|
{saving ? <Loader2 className="button-icon spinner-icon" /> : null}
|
|
Save
|
|
</button>
|
|
<button className="button ghost small" onClick={cancelEdit}>
|
|
Cancel
|
|
</button>
|
|
</div>
|
|
</div>
|
|
)}
|
|
|
|
{entries.length === 0 && !editing && !loading && (
|
|
<div className="card-meta">
|
|
No skills configured in this directory.
|
|
</div>
|
|
)}
|
|
|
|
{entries.map((entry) => (
|
|
<div key={entry.name} className="card" style={{ marginBottom: 8 }}>
|
|
<div className="card-header">
|
|
<span className="card-title">{entry.name}</span>
|
|
<div className="card-header-pills">
|
|
<span className="pill accent">
|
|
{entry.config.sources.length} source{entry.config.sources.length !== 1 ? "s" : ""}
|
|
</span>
|
|
<button
|
|
className="button ghost small"
|
|
onClick={() => remove(entry.name)}
|
|
title="Remove"
|
|
style={{ padding: "2px 4px" }}
|
|
>
|
|
<Trash2 size={12} />
|
|
</button>
|
|
</div>
|
|
</div>
|
|
<pre className="code-block" style={{ marginTop: 4, fontSize: 10 }}>
|
|
{formatJson(entry.config)}
|
|
</pre>
|
|
</div>
|
|
))}
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default SkillsTab;
|