Skip to content

Commit 263dcf7

Browse files
authored
fix: restore prompt focus after footer selection (anomalyco#20841)
1 parent 7994dce commit 263dcf7

3 files changed

Lines changed: 126 additions & 17 deletions

File tree

Lines changed: 88 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,88 @@
1+
import type { Locator, Page } from "@playwright/test"
2+
import { test, expect } from "../fixtures"
3+
import { promptAgentSelector, promptModelSelector, promptSelector } from "../selectors"
4+
5+
type Probe = {
6+
agent?: string
7+
model?: { providerID: string; modelID: string; name?: string }
8+
models?: Array<{ providerID: string; modelID: string; name: string }>
9+
agents?: Array<{ name: string }>
10+
}
11+
12+
async function probe(page: Page): Promise<Probe | null> {
13+
return page.evaluate(() => {
14+
const win = window as Window & {
15+
__opencode_e2e?: {
16+
model?: {
17+
current?: Probe
18+
}
19+
}
20+
}
21+
return win.__opencode_e2e?.model?.current ?? null
22+
})
23+
}
24+
25+
async function state(page: Page) {
26+
const value = await probe(page)
27+
if (!value) throw new Error("Failed to resolve model selection probe")
28+
return value
29+
}
30+
31+
async function ready(page: Page) {
32+
const prompt = page.locator(promptSelector)
33+
await prompt.click()
34+
await expect(prompt).toBeFocused()
35+
await prompt.pressSequentially("focus")
36+
return prompt
37+
}
38+
39+
async function body(prompt: Locator) {
40+
return prompt.evaluate((el) => (el as HTMLElement).innerText)
41+
}
42+
43+
test("agent select returns focus to the prompt", async ({ page, gotoSession }) => {
44+
await gotoSession()
45+
46+
const prompt = await ready(page)
47+
48+
const info = await state(page)
49+
const next = info.agents?.map((item) => item.name).find((name) => name !== info.agent)
50+
test.skip(!next, "only one agent available")
51+
if (!next) return
52+
53+
await page.locator(`${promptAgentSelector} [data-slot="select-select-trigger"]`).first().click()
54+
55+
const item = page.locator('[data-slot="select-select-item"]').filter({ hasText: next }).first()
56+
await expect(item).toBeVisible()
57+
await item.click({ force: true })
58+
59+
await expect(page.locator(`${promptAgentSelector} [data-slot="select-select-trigger-value"]`).first()).toHaveText(
60+
next,
61+
)
62+
await expect(prompt).toBeFocused()
63+
await prompt.pressSequentially(" agent")
64+
await expect.poll(() => body(prompt)).toContain("focus agent")
65+
})
66+
67+
test("model select returns focus to the prompt", async ({ page, gotoSession }) => {
68+
await gotoSession()
69+
70+
const prompt = await ready(page)
71+
72+
const info = await state(page)
73+
const key = info.model ? `${info.model.providerID}:${info.model.modelID}` : null
74+
const next = info.models?.find((item) => `${item.providerID}:${item.modelID}` !== key)
75+
test.skip(!next, "only one model available")
76+
if (!next) return
77+
78+
await page.locator(`${promptModelSelector} [data-action="prompt-model"]`).first().click()
79+
80+
const item = page.locator(`[data-slot="list-item"][data-key="${next.providerID}:${next.modelID}"]`).first()
81+
await expect(item).toBeVisible()
82+
await item.click({ force: true })
83+
84+
await expect(page.locator(`${promptModelSelector} [data-action="prompt-model"] span`).first()).toHaveText(next.name)
85+
await expect(prompt).toBeFocused()
86+
await prompt.pressSequentially(" model")
87+
await expect.poll(() => body(prompt)).toContain("focus model")
88+
})

packages/app/src/components/dialog-select-model.tsx

Lines changed: 20 additions & 15 deletions
Original file line numberDiff line numberDiff line change
@@ -86,32 +86,39 @@ const ModelList: Component<{
8686
}
8787

8888
type ModelSelectorTriggerProps = Omit<ComponentProps<typeof Kobalte.Trigger>, "as" | "ref">
89+
type Dismiss = "escape" | "outside" | "select" | "manage" | "provider"
8990

9091
export function ModelSelectorPopover(props: {
9192
provider?: string
9293
model?: ModelState
9394
children?: JSX.Element
9495
triggerAs?: ValidComponent
9596
triggerProps?: ModelSelectorTriggerProps
97+
onClose?: (cause: "escape" | "select") => void
9698
}) {
9799
const [store, setStore] = createStore<{
98100
open: boolean
99-
dismiss: "escape" | "outside" | null
101+
dismiss: Dismiss | null
100102
}>({
101103
open: false,
102104
dismiss: null,
103105
})
104106
const dialog = useDialog()
105107

106-
const handleManage = () => {
108+
const close = (dismiss: Dismiss) => {
109+
setStore("dismiss", dismiss)
107110
setStore("open", false)
111+
}
112+
113+
const handleManage = () => {
114+
close("manage")
108115
void import("./dialog-manage-models").then((x) => {
109116
dialog.show(() => <x.DialogManageModels />)
110117
})
111118
}
112119

113120
const handleConnectProvider = () => {
114-
setStore("open", false)
121+
close("provider")
115122
void import("./dialog-select-provider").then((x) => {
116123
dialog.show(() => <x.DialogSelectProvider />)
117124
})
@@ -136,29 +143,27 @@ export function ModelSelectorPopover(props: {
136143
<Kobalte.Content
137144
class="w-72 h-80 flex flex-col p-2 rounded-md border border-border-base bg-surface-raised-stronger-non-alpha shadow-md z-50 outline-none overflow-hidden"
138145
onEscapeKeyDown={(event) => {
139-
setStore("dismiss", "escape")
140-
setStore("open", false)
146+
close("escape")
141147
event.preventDefault()
142148
event.stopPropagation()
143149
}}
144-
onPointerDownOutside={() => {
145-
setStore("dismiss", "outside")
146-
setStore("open", false)
147-
}}
148-
onFocusOutside={() => {
149-
setStore("dismiss", "outside")
150-
setStore("open", false)
151-
}}
150+
onPointerDownOutside={() => close("outside")}
151+
onFocusOutside={() => close("outside")}
152152
onCloseAutoFocus={(event) => {
153-
if (store.dismiss === "outside") event.preventDefault()
153+
const dismiss = store.dismiss
154+
if (dismiss === "outside") event.preventDefault()
155+
if (dismiss === "escape" || dismiss === "select") {
156+
event.preventDefault()
157+
props.onClose?.(dismiss)
158+
}
154159
setStore("dismiss", null)
155160
}}
156161
>
157162
<Kobalte.Title class="sr-only">{language.t("dialog.model.select.title")}</Kobalte.Title>
158163
<ModelList
159164
provider={props.provider}
160165
model={props.model}
161-
onSelect={() => setStore("open", false)}
166+
onSelect={() => close("select")}
162167
class="p-1"
163168
action={
164169
<div class="flex items-center gap-1">

packages/app/src/components/prompt-input.tsx

Lines changed: 18 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -502,6 +502,15 @@ export const PromptInput: Component<PromptInputProps> = (props) => {
502502
return getCursorPosition(editorRef)
503503
}
504504

505+
const restoreFocus = () => {
506+
requestAnimationFrame(() => {
507+
const cursor = prompt.cursor() ?? promptLength(prompt.current())
508+
editorRef.focus()
509+
setCursorPosition(editorRef, cursor)
510+
queueScroll()
511+
})
512+
}
513+
505514
const renderEditorWithCursor = (parts: Prompt) => {
506515
const cursor = currentCursor()
507516
renderEditor(parts)
@@ -1471,7 +1480,10 @@ export const PromptInput: Component<PromptInputProps> = (props) => {
14711480
size="normal"
14721481
options={agentNames()}
14731482
current={local.agent.current()?.name ?? ""}
1474-
onSelect={local.agent.set}
1483+
onSelect={(value) => {
1484+
local.agent.set(value)
1485+
restoreFocus()
1486+
}}
14751487
class="capitalize max-w-[160px] text-text-base"
14761488
valueClass="truncate text-13-regular text-text-base"
14771489
triggerStyle={control()}
@@ -1535,6 +1547,7 @@ export const PromptInput: Component<PromptInputProps> = (props) => {
15351547
class: "min-w-0 max-w-[320px] text-13-regular text-text-base group",
15361548
"data-action": "prompt-model",
15371549
}}
1550+
onClose={restoreFocus}
15381551
>
15391552
<Show when={local.model.current()?.provider?.id}>
15401553
<ProviderIcon
@@ -1563,7 +1576,10 @@ export const PromptInput: Component<PromptInputProps> = (props) => {
15631576
options={variants()}
15641577
current={local.model.variant.current() ?? "default"}
15651578
label={(x) => (x === "default" ? language.t("common.default") : x)}
1566-
onSelect={(x) => local.model.variant.set(x === "default" ? undefined : x)}
1579+
onSelect={(value) => {
1580+
local.model.variant.set(value === "default" ? undefined : value)
1581+
restoreFocus()
1582+
}}
15671583
class="capitalize max-w-[160px] text-text-base"
15681584
valueClass="truncate text-13-regular text-text-base"
15691585
triggerStyle={control()}

0 commit comments

Comments
 (0)