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, shouldFocusTerminalOnKeyDown, } 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 = `
` const focused = focusTerminalById("one") expect(focused).toBe(true) expect(document.activeElement?.tagName).toBe("TEXTAREA") }) test("falls back to terminal element focus", () => { document.body.innerHTML = `
` 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("shouldFocusTerminalOnKeyDown", () => { test("skips pure modifier keys", () => { expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "Meta", metaKey: true }))).toBe(false) expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "Control", ctrlKey: true }))).toBe(false) expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "Alt", altKey: true }))).toBe(false) expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "Shift", shiftKey: true }))).toBe(false) }) test("skips shortcut key combos", () => { expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "c", metaKey: true }))).toBe(false) expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "c", ctrlKey: true }))).toBe(false) expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "ArrowLeft", altKey: true }))).toBe(false) }) test("keeps plain typing focused on terminal", () => { expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "a" }))).toBe(true) expect(shouldFocusTerminalOnKeyDown(new KeyboardEvent("keydown", { key: "A", shiftKey: true }))).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() }) }) })