mirror of
https://github.com/anomalyco/opencode.git
synced 2026-04-24 14:55:19 +00:00
161 lines
5.0 KiB
TypeScript
161 lines
5.0 KiB
TypeScript
import { describe, expect, test } from "bun:test"
|
|
import { createMemo, createRoot } from "solid-js"
|
|
import { createStore } from "solid-js/store"
|
|
import {
|
|
createOpenReviewFile,
|
|
createOpenSessionFileTab,
|
|
createSessionTabs,
|
|
focusTerminalById,
|
|
getTabReorderIndex,
|
|
} from "./helpers"
|
|
|
|
describe("createOpenReviewFile", () => {
|
|
test("opens and loads selected review file", () => {
|
|
const calls: string[] = []
|
|
const openReviewFile = createOpenReviewFile({
|
|
showAllFiles: () => calls.push("show"),
|
|
tabForPath: (path) => {
|
|
calls.push(`tab:${path}`)
|
|
return `file://${path}`
|
|
},
|
|
openTab: (tab) => calls.push(`open:${tab}`),
|
|
setActive: (tab) => calls.push(`active:${tab}`),
|
|
loadFile: (path) => calls.push(`load:${path}`),
|
|
})
|
|
|
|
openReviewFile("src/a.ts")
|
|
|
|
expect(calls).toEqual(["show", "load:src/a.ts", "tab:src/a.ts", "open:file://src/a.ts", "active:file://src/a.ts"])
|
|
})
|
|
})
|
|
|
|
describe("createOpenSessionFileTab", () => {
|
|
test("activates the opened file tab", () => {
|
|
const calls: string[] = []
|
|
const openTab = createOpenSessionFileTab({
|
|
normalizeTab: (value) => {
|
|
calls.push(`normalize:${value}`)
|
|
return `file://${value}`
|
|
},
|
|
openTab: (tab) => calls.push(`open:${tab}`),
|
|
pathFromTab: (tab) => {
|
|
calls.push(`path:${tab}`)
|
|
return tab.slice("file://".length)
|
|
},
|
|
loadFile: (path) => calls.push(`load:${path}`),
|
|
openReviewPanel: () => calls.push("review"),
|
|
setActive: (tab) => calls.push(`active:${tab}`),
|
|
})
|
|
|
|
openTab("src/a.ts")
|
|
|
|
expect(calls).toEqual([
|
|
"normalize:src/a.ts",
|
|
"open:file://src/a.ts",
|
|
"path:file://src/a.ts",
|
|
"load:src/a.ts",
|
|
"review",
|
|
"active:file://src/a.ts",
|
|
])
|
|
})
|
|
})
|
|
|
|
describe("focusTerminalById", () => {
|
|
test("focuses textarea when present", () => {
|
|
document.body.innerHTML = `<div id="terminal-wrapper-one"><div data-component="terminal"><textarea></textarea></div></div>`
|
|
|
|
const focused = focusTerminalById("one")
|
|
|
|
expect(focused).toBe(true)
|
|
expect(document.activeElement?.tagName).toBe("TEXTAREA")
|
|
})
|
|
|
|
test("falls back to terminal element focus", () => {
|
|
document.body.innerHTML = `<div id="terminal-wrapper-two"><div data-component="terminal" tabindex="0"></div></div>`
|
|
const terminal = document.querySelector('[data-component="terminal"]') as HTMLElement
|
|
let pointerDown = false
|
|
terminal.addEventListener("pointerdown", () => {
|
|
pointerDown = true
|
|
})
|
|
|
|
const focused = focusTerminalById("two")
|
|
|
|
expect(focused).toBe(true)
|
|
expect(document.activeElement).toBe(terminal)
|
|
expect(pointerDown).toBe(true)
|
|
})
|
|
})
|
|
|
|
describe("getTabReorderIndex", () => {
|
|
test("returns target index for valid drag reorder", () => {
|
|
expect(getTabReorderIndex(["a", "b", "c"], "a", "c")).toBe(2)
|
|
})
|
|
|
|
test("returns undefined for unknown droppable id", () => {
|
|
expect(getTabReorderIndex(["a", "b", "c"], "a", "missing")).toBeUndefined()
|
|
})
|
|
})
|
|
|
|
describe("createSessionTabs", () => {
|
|
test("normalizes the effective file tab", () => {
|
|
createRoot((dispose) => {
|
|
const [state] = createStore({
|
|
active: undefined as string | undefined,
|
|
all: ["file://src/a.ts", "context"],
|
|
})
|
|
const tabs = createMemo(() => ({ active: () => state.active, all: () => state.all }))
|
|
const result = createSessionTabs({
|
|
tabs,
|
|
pathFromTab: (tab) => (tab.startsWith("file://") ? tab.slice("file://".length) : undefined),
|
|
normalizeTab: (tab) => (tab.startsWith("file://") ? `norm:${tab.slice("file://".length)}` : tab),
|
|
})
|
|
|
|
expect(result.activeTab()).toBe("norm:src/a.ts")
|
|
expect(result.activeFileTab()).toBe("norm:src/a.ts")
|
|
expect(result.closableTab()).toBe("norm:src/a.ts")
|
|
dispose()
|
|
})
|
|
})
|
|
|
|
test("prefers context and review fallbacks when no file tab is active", () => {
|
|
createRoot((dispose) => {
|
|
const [state] = createStore({
|
|
active: undefined as string | undefined,
|
|
all: ["context"],
|
|
})
|
|
const tabs = createMemo(() => ({ active: () => state.active, all: () => state.all }))
|
|
const result = createSessionTabs({
|
|
tabs,
|
|
pathFromTab: () => undefined,
|
|
normalizeTab: (tab) => tab,
|
|
review: () => true,
|
|
hasReview: () => true,
|
|
})
|
|
|
|
expect(result.activeTab()).toBe("context")
|
|
expect(result.closableTab()).toBe("context")
|
|
dispose()
|
|
})
|
|
|
|
createRoot((dispose) => {
|
|
const [state] = createStore({
|
|
active: undefined as string | undefined,
|
|
all: [],
|
|
})
|
|
const tabs = createMemo(() => ({ active: () => state.active, all: () => state.all }))
|
|
const result = createSessionTabs({
|
|
tabs,
|
|
pathFromTab: () => undefined,
|
|
normalizeTab: (tab) => tab,
|
|
review: () => true,
|
|
hasReview: () => true,
|
|
})
|
|
|
|
expect(result.activeTab()).toBe("review")
|
|
expect(result.activeFileTab()).toBeUndefined()
|
|
expect(result.closableTab()).toBeUndefined()
|
|
dispose()
|
|
})
|
|
})
|
|
})
|