chore: storybook tweaks
parent
ad40b65b0b
commit
b746aec493
|
|
@ -425,13 +425,60 @@ const TOOL_SAMPLES = {
|
|||
// Fake data generators
|
||||
// ---------------------------------------------------------------------------
|
||||
const SESSION_ID = "playground-session"
|
||||
const DEFAULT_SESSION = { id: SESSION_ID, title: "Timeline Playground" }
|
||||
|
||||
function mkUser(text: string, extra: Part[] = []): { message: UserMessage; parts: Part[] } {
|
||||
function record(value: unknown): value is Record<string, unknown> {
|
||||
return !!value && typeof value === "object" && !Array.isArray(value)
|
||||
}
|
||||
|
||||
function normalize(raw: unknown) {
|
||||
if (Array.isArray(raw)) {
|
||||
const info = raw.find((row) => record(row) && row.type === "session" && record(row.data))?.data
|
||||
if (!record(info) || typeof info.id !== "string") {
|
||||
throw new Error("No session found in JSON")
|
||||
}
|
||||
|
||||
const part = new Map<string, Part[]>()
|
||||
const messages = raw.flatMap((row) => {
|
||||
if (!record(row) || !record(row.data)) return []
|
||||
if (row.type === "part" && typeof row.data.messageID === "string") {
|
||||
const list = part.get(row.data.messageID) ?? []
|
||||
list.push(row.data as Part)
|
||||
part.set(row.data.messageID, list)
|
||||
return []
|
||||
}
|
||||
if (row.type !== "message" || typeof row.data.id !== "string") return []
|
||||
return [{ info: row.data as Message, parts: [] as Part[] }]
|
||||
})
|
||||
|
||||
return {
|
||||
info,
|
||||
messages: messages.map((msg) => ({
|
||||
info: msg.info,
|
||||
parts: part.get(msg.info.id) ?? [],
|
||||
})),
|
||||
}
|
||||
}
|
||||
|
||||
if (!record(raw) || !record(raw.info) || typeof raw.info.id !== "string" || !Array.isArray(raw.messages)) {
|
||||
throw new Error("Expected an `opencode export` JSON file")
|
||||
}
|
||||
|
||||
return {
|
||||
info: raw.info,
|
||||
messages: raw.messages.flatMap((row) => {
|
||||
if (!record(row) || !record(row.info) || typeof row.info.id !== "string") return []
|
||||
return [{ info: row.info as Message, parts: Array.isArray(row.parts) ? (row.parts as Part[]) : [] }]
|
||||
}),
|
||||
}
|
||||
}
|
||||
|
||||
function mkUser(text: string, extra: Part[] = [], sessionID = SESSION_ID): { message: UserMessage; parts: Part[] } {
|
||||
const id = uid()
|
||||
return {
|
||||
message: {
|
||||
id,
|
||||
sessionID: SESSION_ID,
|
||||
sessionID,
|
||||
role: "user",
|
||||
time: { created: Date.now() },
|
||||
agent: "code",
|
||||
|
|
@ -445,10 +492,10 @@ function mkUser(text: string, extra: Part[] = []): { message: UserMessage; parts
|
|||
}
|
||||
}
|
||||
|
||||
function mkAssistant(parentID: string): AssistantMessage {
|
||||
function mkAssistant(parentID: string, sessionID = SESSION_ID): AssistantMessage {
|
||||
return {
|
||||
id: uid(),
|
||||
sessionID: SESSION_ID,
|
||||
sessionID,
|
||||
role: "assistant",
|
||||
time: { created: Date.now(), completed: Date.now() + 3000 },
|
||||
parentID,
|
||||
|
|
@ -1010,12 +1057,16 @@ function Playground() {
|
|||
messages: [],
|
||||
parts: {},
|
||||
})
|
||||
const [session, setSession] = createSignal({ ...DEFAULT_SESSION })
|
||||
const [loaded, setLoaded] = createSignal("")
|
||||
const [issue, setIssue] = createSignal("")
|
||||
|
||||
// ---- CSS overrides ----
|
||||
const [css, setCss] = createStore<Record<string, string>>({})
|
||||
const [defaults, setDefaults] = createStore<Record<string, string>>({})
|
||||
let styleEl: HTMLStyleElement | undefined
|
||||
let previewRef: HTMLDivElement | undefined
|
||||
let pick: HTMLInputElement | undefined
|
||||
|
||||
/** Read computed styles from the DOM to seed slider defaults */
|
||||
const readDefaults = () => {
|
||||
|
|
@ -1074,10 +1125,10 @@ function Playground() {
|
|||
const userMessages = createMemo(() => state.messages.filter((m): m is UserMessage => m.role === "user"))
|
||||
|
||||
const data = createMemo(() => ({
|
||||
session: [{ id: SESSION_ID }],
|
||||
session: [session()],
|
||||
session_status: {},
|
||||
session_diff: {},
|
||||
message: { [SESSION_ID]: state.messages },
|
||||
message: { [session().id]: state.messages },
|
||||
part: state.parts,
|
||||
provider: {
|
||||
all: [{ id: "anthropic", models: { "claude-sonnet-4-20250514": { name: "Claude Sonnet" } } }],
|
||||
|
|
@ -1109,8 +1160,8 @@ function Playground() {
|
|||
const id = lastAssistantID()
|
||||
if (id) return id
|
||||
// Create a minimal placeholder turn
|
||||
const user = mkUser("...")
|
||||
const asst = mkAssistant(user.message.id)
|
||||
const user = mkUser("...", [], session().id)
|
||||
const asst = mkAssistant(user.message.id, session().id)
|
||||
setState(
|
||||
produce((draft) => {
|
||||
draft.messages.push(user.message)
|
||||
|
|
@ -1136,8 +1187,8 @@ function Playground() {
|
|||
// ---- User message helpers ----
|
||||
const addUser = (variant: keyof typeof USER_VARIANTS) => {
|
||||
const v = USER_VARIANTS[variant]
|
||||
const user = mkUser(v.text, v.parts)
|
||||
const asst = mkAssistant(user.message.id)
|
||||
const user = mkUser(v.text, v.parts, session().id)
|
||||
const asst = mkAssistant(user.message.id, session().id)
|
||||
setState(
|
||||
produce((draft) => {
|
||||
draft.messages.push(user.message)
|
||||
|
|
@ -1164,8 +1215,8 @@ function Playground() {
|
|||
|
||||
// ---- Composite helpers (create full turns with user + assistant) ----
|
||||
const addFullTurn = (userText: string, parts: Part[]) => {
|
||||
const user = mkUser(userText)
|
||||
const asst = mkAssistant(user.message.id)
|
||||
const user = mkUser(userText, [], session().id)
|
||||
const asst = mkAssistant(user.message.id, session().id)
|
||||
setState(
|
||||
produce((draft) => {
|
||||
draft.messages.push(user.message)
|
||||
|
|
@ -1222,9 +1273,91 @@ function Playground() {
|
|||
addReasoningFullTurn()
|
||||
}
|
||||
|
||||
const interrupt = () => {
|
||||
const user = userMessages().at(-1)
|
||||
if (!user) return
|
||||
const now = Date.now()
|
||||
|
||||
setState(
|
||||
produce((draft) => {
|
||||
const msg = draft.messages.findLast(
|
||||
(item): item is AssistantMessage => item.role === "assistant" && item.parentID === user.id,
|
||||
)
|
||||
|
||||
if (msg) {
|
||||
const time = msg.time ?? { created: now }
|
||||
msg.time = { ...time, completed: time.completed ?? now }
|
||||
msg.error = { name: "MessageAbortedError", message: "Interrupted" }
|
||||
return
|
||||
}
|
||||
|
||||
const asst = mkAssistant(user.id, session().id)
|
||||
asst.time = { created: now, completed: now }
|
||||
asst.error = { name: "MessageAbortedError", message: "Interrupted" }
|
||||
draft.messages.push(asst)
|
||||
draft.parts[asst.id] = []
|
||||
}),
|
||||
)
|
||||
}
|
||||
|
||||
const load = (raw: unknown, name: string) => {
|
||||
const next = normalize(raw)
|
||||
const id = typeof next.info.id === "string" && next.info.id ? next.info.id : SESSION_ID
|
||||
const messages = next.messages.map((msg) => ({
|
||||
...msg.info,
|
||||
sessionID: typeof msg.info.sessionID === "string" ? msg.info.sessionID : id,
|
||||
}))
|
||||
const parts = Object.fromEntries(
|
||||
next.messages.map((msg, idx) => {
|
||||
const info = messages[idx]
|
||||
return [
|
||||
info.id,
|
||||
msg.parts.map((part) => ({
|
||||
...part,
|
||||
messageID: typeof part.messageID === "string" ? part.messageID : info.id,
|
||||
sessionID: typeof part.sessionID === "string" ? part.sessionID : info.sessionID,
|
||||
})),
|
||||
]
|
||||
}),
|
||||
)
|
||||
|
||||
batch(() => {
|
||||
setSession({
|
||||
...DEFAULT_SESSION,
|
||||
...next.info,
|
||||
id,
|
||||
title: typeof next.info.title === "string" && next.info.title ? next.info.title : name,
|
||||
})
|
||||
setState({ messages, parts })
|
||||
setLoaded(name)
|
||||
setIssue("")
|
||||
})
|
||||
}
|
||||
|
||||
const importFile = async (event: Event) => {
|
||||
const input = event.currentTarget as HTMLInputElement
|
||||
const file = input.files?.[0]
|
||||
if (!file) return
|
||||
|
||||
setIssue("")
|
||||
|
||||
try {
|
||||
load(JSON.parse(await file.text()), file.name)
|
||||
} catch (err) {
|
||||
setIssue(err instanceof Error ? err.message : String(err))
|
||||
} finally {
|
||||
input.value = ""
|
||||
}
|
||||
}
|
||||
|
||||
const clearAll = () => {
|
||||
setState({ messages: [], parts: {} })
|
||||
seq = 0
|
||||
batch(() => {
|
||||
setState({ messages: [], parts: {} })
|
||||
setSession({ ...DEFAULT_SESSION })
|
||||
setLoaded("")
|
||||
setIssue("")
|
||||
seq = 0
|
||||
})
|
||||
}
|
||||
|
||||
// ---- CSS export ----
|
||||
|
|
@ -1393,6 +1526,35 @@ function Playground() {
|
|||
</button>
|
||||
<Show when={panels.generators}>
|
||||
<div style={{ padding: "0 12px 12px", display: "flex", "flex-direction": "column", gap: "6px" }}>
|
||||
{/* ---- Session import ---- */}
|
||||
<div style={sectionLabel}>Import session</div>
|
||||
<div style={{ "font-size": "10px", color: "var(--text-weaker)", "margin-bottom": "2px" }}>
|
||||
Replaces the current timeline with an `opencode export` JSON file
|
||||
</div>
|
||||
<div style={{ display: "flex", "flex-wrap": "wrap", gap: "4px" }}>
|
||||
<button style={btnAccent} onClick={() => pick?.click()}>
|
||||
Import session
|
||||
</button>
|
||||
<input
|
||||
ref={pick!}
|
||||
type="file"
|
||||
accept=".json,application/json"
|
||||
onChange={importFile}
|
||||
style={{ display: "none" }}
|
||||
/>
|
||||
</div>
|
||||
<Show when={loaded()}>
|
||||
<div style={{ "font-size": "10px", color: "var(--text-weaker)", "line-height": "1.4" }}>
|
||||
{loaded()} • {session().title || session().id} • {state.messages.length} message
|
||||
{state.messages.length === 1 ? "" : "s"}
|
||||
</div>
|
||||
</Show>
|
||||
<Show when={issue()}>
|
||||
<div style={{ "font-size": "10px", color: "var(--text-on-critical-base)", "line-height": "1.4" }}>
|
||||
{issue()}
|
||||
</div>
|
||||
</Show>
|
||||
|
||||
{/* ---- User messages ---- */}
|
||||
<div style={sectionLabel}>User messages</div>
|
||||
<div style={{ "font-size": "10px", color: "var(--text-weaker)", "margin-bottom": "2px" }}>
|
||||
|
|
@ -1407,6 +1569,19 @@ function Playground() {
|
|||
)}
|
||||
</For>
|
||||
</div>
|
||||
<div style={{ display: "flex", "flex-wrap": "wrap", gap: "4px" }}>
|
||||
<button
|
||||
style={{
|
||||
...btnDanger,
|
||||
opacity: userMessages().length === 0 ? "0.5" : "1",
|
||||
cursor: userMessages().length === 0 ? "not-allowed" : "pointer",
|
||||
}}
|
||||
disabled={userMessages().length === 0}
|
||||
onClick={interrupt}
|
||||
>
|
||||
Interrupt last
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{/* ---- Text and reasoning blocks ---- */}
|
||||
<div style={{ ...sectionLabel, "margin-top": "8px" }}>Text and reasoning blocks</div>
|
||||
|
|
@ -1716,7 +1891,7 @@ function Playground() {
|
|||
"font-size": "14px",
|
||||
}}
|
||||
>
|
||||
Click a generator button to add messages
|
||||
Click a generator button or import a session
|
||||
</div>
|
||||
}
|
||||
>
|
||||
|
|
@ -1729,7 +1904,7 @@ function Playground() {
|
|||
{(msg) => (
|
||||
<div style={{ width: "100%" }}>
|
||||
<SessionTurn
|
||||
sessionID={SESSION_ID}
|
||||
sessionID={session().id}
|
||||
messageID={msg.id}
|
||||
messages={state.messages}
|
||||
active={false}
|
||||
|
|
|
|||
Loading…
Reference in New Issue