| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189 |
- import { useFilteredList } from "@opencode-ai/ui/hooks"
- import { createEffect, on, Component, Show, For, onCleanup, Switch, Match, createMemo, createSignal } from "solid-js"
- import { createStore } from "solid-js/store"
- import { createFocusSignal } from "@solid-primitives/active-element"
- import { useLocal } from "@/context/local"
- import { useFile } from "@/context/file"
- import {
- ContentPart,
- DEFAULT_PROMPT,
- isPromptEqual,
- Prompt,
- usePrompt,
- ImageAttachmentPart,
- AgentPart,
- FileAttachmentPart,
- } from "@/context/prompt"
- import { useLayout } from "@/context/layout"
- import { useSDK } from "@/context/sdk"
- import { useParams } from "@solidjs/router"
- import { useSync } from "@/context/sync"
- import { useComments } from "@/context/comments"
- import { Button } from "@opencode-ai/ui/button"
- import { Icon } from "@opencode-ai/ui/icon"
- import { ProviderIcon } from "@opencode-ai/ui/provider-icon"
- import type { IconName } from "@opencode-ai/ui/icons/provider"
- import { Tooltip, TooltipKeybind } from "@opencode-ai/ui/tooltip"
- import { IconButton } from "@opencode-ai/ui/icon-button"
- import { Select } from "@opencode-ai/ui/select"
- import { useDialog } from "@opencode-ai/ui/context/dialog"
- import { ModelSelectorPopover } from "@/components/dialog-select-model"
- import { DialogSelectModelUnpaid } from "@/components/dialog-select-model-unpaid"
- import { useProviders } from "@/hooks/use-providers"
- import { useCommand } from "@/context/command"
- import { Persist, persisted } from "@/utils/persist"
- import { SessionContextUsage } from "@/components/session-context-usage"
- import { usePermission } from "@/context/permission"
- import { useLanguage } from "@/context/language"
- import { createTextFragment, getCursorPosition, setCursorPosition, setRangeEdge } from "./prompt-input/editor-dom"
- import { createPromptAttachments, ACCEPTED_FILE_TYPES } from "./prompt-input/attachments"
- import { navigatePromptHistory, prependHistoryEntry, promptLength } from "./prompt-input/history"
- import { createPromptSubmit } from "./prompt-input/submit"
- import { PromptPopover, type AtOption, type SlashCommand } from "./prompt-input/slash-popover"
- import { PromptContextItems } from "./prompt-input/context-items"
- import { PromptImageAttachments } from "./prompt-input/image-attachments"
- import { PromptDragOverlay } from "./prompt-input/drag-overlay"
- import { promptPlaceholder } from "./prompt-input/placeholder"
- import { ImagePreview } from "@opencode-ai/ui/image-preview"
- interface PromptInputProps {
- class?: string
- ref?: (el: HTMLDivElement) => void
- newSessionWorktree?: string
- onNewSessionWorktreeReset?: () => void
- onSubmit?: () => void
- }
- const EXAMPLES = [
- "prompt.example.1",
- "prompt.example.2",
- "prompt.example.3",
- "prompt.example.4",
- "prompt.example.5",
- "prompt.example.6",
- "prompt.example.7",
- "prompt.example.8",
- "prompt.example.9",
- "prompt.example.10",
- "prompt.example.11",
- "prompt.example.12",
- "prompt.example.13",
- "prompt.example.14",
- "prompt.example.15",
- "prompt.example.16",
- "prompt.example.17",
- "prompt.example.18",
- "prompt.example.19",
- "prompt.example.20",
- "prompt.example.21",
- "prompt.example.22",
- "prompt.example.23",
- "prompt.example.24",
- "prompt.example.25",
- ] as const
- export const PromptInput: Component<PromptInputProps> = (props) => {
- const sdk = useSDK()
- const sync = useSync()
- const local = useLocal()
- const files = useFile()
- const prompt = usePrompt()
- const commentCount = createMemo(() => prompt.context.items().filter((item) => !!item.comment?.trim()).length)
- const layout = useLayout()
- const comments = useComments()
- const params = useParams()
- const dialog = useDialog()
- const providers = useProviders()
- const command = useCommand()
- const permission = usePermission()
- const language = useLanguage()
- let editorRef!: HTMLDivElement
- let fileInputRef!: HTMLInputElement
- let scrollRef!: HTMLDivElement
- let slashPopoverRef!: HTMLDivElement
- const mirror = { input: false }
- const scrollCursorIntoView = () => {
- const container = scrollRef
- const selection = window.getSelection()
- if (!container || !selection || selection.rangeCount === 0) return
- const range = selection.getRangeAt(0)
- if (!editorRef.contains(range.startContainer)) return
- const rect = range.getBoundingClientRect()
- if (!rect.height) return
- const containerRect = container.getBoundingClientRect()
- const top = rect.top - containerRect.top + container.scrollTop
- const bottom = rect.bottom - containerRect.top + container.scrollTop
- const padding = 12
- if (top < container.scrollTop + padding) {
- container.scrollTop = Math.max(0, top - padding)
- return
- }
- if (bottom > container.scrollTop + container.clientHeight - padding) {
- container.scrollTop = bottom - container.clientHeight + padding
- }
- }
- const queueScroll = () => {
- requestAnimationFrame(scrollCursorIntoView)
- }
- const sessionKey = createMemo(() => `${params.dir}${params.id ? "/" + params.id : ""}`)
- const tabs = createMemo(() => layout.tabs(sessionKey))
- const view = createMemo(() => layout.view(sessionKey))
- const commentInReview = (path: string) => {
- const sessionID = params.id
- if (!sessionID) return false
- const diffs = sync.data.session_diff[sessionID]
- if (!diffs) return false
- return diffs.some((diff) => diff.file === path)
- }
- const openComment = (item: { path: string; commentID?: string; commentOrigin?: "review" | "file" }) => {
- if (!item.commentID) return
- const focus = { file: item.path, id: item.commentID }
- comments.setActive(focus)
- const wantsReview = item.commentOrigin === "review" || (item.commentOrigin !== "file" && commentInReview(item.path))
- if (wantsReview) {
- if (!view().reviewPanel.opened()) view().reviewPanel.open()
- layout.fileTree.open()
- layout.fileTree.setTab("changes")
- requestAnimationFrame(() => comments.setFocus(focus))
- return
- }
- if (!view().reviewPanel.opened()) view().reviewPanel.open()
- layout.fileTree.open()
- layout.fileTree.setTab("all")
- const tab = files.tab(item.path)
- tabs().open(tab)
- files.load(item.path)
- requestAnimationFrame(() => comments.setFocus(focus))
- }
- const recent = createMemo(() => {
- const all = tabs().all()
- const active = tabs().active()
- const order = active ? [active, ...all.filter((x) => x !== active)] : all
- const seen = new Set<string>()
- const paths: string[] = []
- for (const tab of order) {
- const path = files.pathFromTab(tab)
- if (!path) continue
- if (seen.has(path)) continue
- seen.add(path)
- paths.push(path)
- }
- return paths
- })
- const info = createMemo(() => (params.id ? sync.session.get(params.id) : undefined))
- const status = createMemo(
- () =>
- sync.data.session_status[params.id ?? ""] ?? {
- type: "idle",
- },
- )
- const working = createMemo(() => status()?.type !== "idle")
- const imageAttachments = createMemo(() =>
- prompt.current().filter((part): part is ImageAttachmentPart => part.type === "image"),
- )
- const [store, setStore] = createStore<{
- popover: "at" | "slash" | null
- historyIndex: number
- savedPrompt: Prompt | null
- placeholder: number
- dragging: boolean
- mode: "normal" | "shell"
- applyingHistory: boolean
- }>({
- popover: null,
- historyIndex: -1,
- savedPrompt: null,
- placeholder: Math.floor(Math.random() * EXAMPLES.length),
- dragging: false,
- mode: "normal",
- applyingHistory: false,
- })
- const placeholder = createMemo(() =>
- promptPlaceholder({
- mode: store.mode,
- commentCount: commentCount(),
- example: language.t(EXAMPLES[store.placeholder]),
- t: (key, params) => language.t(key as Parameters<typeof language.t>[0], params as never),
- }),
- )
- const MAX_HISTORY = 100
- const [history, setHistory] = persisted(
- Persist.global("prompt-history", ["prompt-history.v1"]),
- createStore<{
- entries: Prompt[]
- }>({
- entries: [],
- }),
- )
- const [shellHistory, setShellHistory] = persisted(
- Persist.global("prompt-history-shell", ["prompt-history-shell.v1"]),
- createStore<{
- entries: Prompt[]
- }>({
- entries: [],
- }),
- )
- const applyHistoryPrompt = (p: Prompt, position: "start" | "end") => {
- const length = position === "start" ? 0 : promptLength(p)
- setStore("applyingHistory", true)
- prompt.set(p, length)
- requestAnimationFrame(() => {
- editorRef.focus()
- setCursorPosition(editorRef, length)
- setStore("applyingHistory", false)
- queueScroll()
- })
- }
- const getCaretState = () => {
- const selection = window.getSelection()
- const textLength = promptLength(prompt.current())
- if (!selection || selection.rangeCount === 0) {
- return { collapsed: false, cursorPosition: 0, textLength }
- }
- const anchorNode = selection.anchorNode
- if (!anchorNode || !editorRef.contains(anchorNode)) {
- return { collapsed: false, cursorPosition: 0, textLength }
- }
- return {
- collapsed: selection.isCollapsed,
- cursorPosition: getCursorPosition(editorRef),
- textLength,
- }
- }
- const isFocused = createFocusSignal(() => editorRef)
- createEffect(() => {
- params.id
- if (params.id) return
- const interval = setInterval(() => {
- setStore("placeholder", (prev) => (prev + 1) % EXAMPLES.length)
- }, 6500)
- onCleanup(() => clearInterval(interval))
- })
- const [composing, setComposing] = createSignal(false)
- const isImeComposing = (event: KeyboardEvent) => event.isComposing || composing() || event.keyCode === 229
- createEffect(() => {
- if (!isFocused()) setStore("popover", null)
- })
- // Safety: reset composing state on focus change to prevent stuck state
- // This handles edge cases where compositionend event may not fire
- createEffect(() => {
- if (!isFocused()) setComposing(false)
- })
- const agentList = createMemo(() =>
- sync.data.agent
- .filter((agent) => !agent.hidden && agent.mode !== "primary")
- .map((agent): AtOption => ({ type: "agent", name: agent.name, display: agent.name })),
- )
- const handleAtSelect = (option: AtOption | undefined) => {
- if (!option) return
- if (option.type === "agent") {
- addPart({ type: "agent", name: option.name, content: "@" + option.name, start: 0, end: 0 })
- } else {
- addPart({ type: "file", path: option.path, content: "@" + option.path, start: 0, end: 0 })
- }
- }
- const atKey = (x: AtOption | undefined) => {
- if (!x) return ""
- return x.type === "agent" ? `agent:${x.name}` : `file:${x.path}`
- }
- const {
- flat: atFlat,
- active: atActive,
- setActive: setAtActive,
- onInput: atOnInput,
- onKeyDown: atOnKeyDown,
- } = useFilteredList<AtOption>({
- items: async (query) => {
- const agents = agentList()
- const open = recent()
- const seen = new Set(open)
- const pinned: AtOption[] = open.map((path) => ({ type: "file", path, display: path, recent: true }))
- const paths = await files.searchFilesAndDirectories(query)
- const fileOptions: AtOption[] = paths
- .filter((path) => !seen.has(path))
- .map((path) => ({ type: "file", path, display: path }))
- return [...agents, ...pinned, ...fileOptions]
- },
- key: atKey,
- filterKeys: ["display"],
- groupBy: (item) => {
- if (item.type === "agent") return "agent"
- if (item.recent) return "recent"
- return "file"
- },
- sortGroupsBy: (a, b) => {
- const rank = (category: string) => {
- if (category === "agent") return 0
- if (category === "recent") return 1
- return 2
- }
- return rank(a.category) - rank(b.category)
- },
- onSelect: handleAtSelect,
- })
- const slashCommands = createMemo<SlashCommand[]>(() => {
- const builtin = command.options
- .filter((opt) => !opt.disabled && !opt.id.startsWith("suggested.") && opt.slash)
- .map((opt) => ({
- id: opt.id,
- trigger: opt.slash!,
- title: opt.title,
- description: opt.description,
- keybind: opt.keybind,
- type: "builtin" as const,
- }))
- const custom = sync.data.command.map((cmd) => ({
- id: `custom.${cmd.name}`,
- trigger: cmd.name,
- title: cmd.name,
- description: cmd.description,
- type: "custom" as const,
- source: cmd.source,
- }))
- return [...custom, ...builtin]
- })
- const handleSlashSelect = (cmd: SlashCommand | undefined) => {
- if (!cmd) return
- setStore("popover", null)
- if (cmd.type === "custom") {
- const text = `/${cmd.trigger} `
- editorRef.innerHTML = ""
- editorRef.textContent = text
- prompt.set([{ type: "text", content: text, start: 0, end: text.length }], text.length)
- requestAnimationFrame(() => {
- editorRef.focus()
- const range = document.createRange()
- const sel = window.getSelection()
- range.selectNodeContents(editorRef)
- range.collapse(false)
- sel?.removeAllRanges()
- sel?.addRange(range)
- })
- return
- }
- editorRef.innerHTML = ""
- prompt.set([{ type: "text", content: "", start: 0, end: 0 }], 0)
- command.trigger(cmd.id, "slash")
- }
- const {
- flat: slashFlat,
- active: slashActive,
- setActive: setSlashActive,
- onInput: slashOnInput,
- onKeyDown: slashOnKeyDown,
- refetch: slashRefetch,
- } = useFilteredList<SlashCommand>({
- items: slashCommands,
- key: (x) => x?.id,
- filterKeys: ["trigger", "title", "description"],
- onSelect: handleSlashSelect,
- })
- const createPill = (part: FileAttachmentPart | AgentPart) => {
- const pill = document.createElement("span")
- pill.textContent = part.content
- pill.setAttribute("data-type", part.type)
- if (part.type === "file") pill.setAttribute("data-path", part.path)
- if (part.type === "agent") pill.setAttribute("data-name", part.name)
- pill.setAttribute("contenteditable", "false")
- pill.style.userSelect = "text"
- pill.style.cursor = "default"
- return pill
- }
- const isNormalizedEditor = () =>
- Array.from(editorRef.childNodes).every((node) => {
- if (node.nodeType === Node.TEXT_NODE) {
- const text = node.textContent ?? ""
- if (!text.includes("\u200B")) return true
- if (text !== "\u200B") return false
- const prev = node.previousSibling
- const next = node.nextSibling
- const prevIsBr = prev?.nodeType === Node.ELEMENT_NODE && (prev as HTMLElement).tagName === "BR"
- const nextIsBr = next?.nodeType === Node.ELEMENT_NODE && (next as HTMLElement).tagName === "BR"
- if (!prevIsBr && !nextIsBr) return false
- if (nextIsBr && !prevIsBr && prev) return false
- return true
- }
- if (node.nodeType !== Node.ELEMENT_NODE) return false
- const el = node as HTMLElement
- if (el.dataset.type === "file") return true
- if (el.dataset.type === "agent") return true
- return el.tagName === "BR"
- })
- const renderEditor = (parts: Prompt) => {
- editorRef.innerHTML = ""
- for (const part of parts) {
- if (part.type === "text") {
- editorRef.appendChild(createTextFragment(part.content))
- continue
- }
- if (part.type === "file" || part.type === "agent") {
- editorRef.appendChild(createPill(part))
- }
- }
- }
- createEffect(
- on(
- () => sync.data.command,
- () => slashRefetch(),
- { defer: true },
- ),
- )
- // Auto-scroll active command into view when navigating with keyboard
- createEffect(() => {
- const activeId = slashActive()
- if (!activeId || !slashPopoverRef) return
- requestAnimationFrame(() => {
- const element = slashPopoverRef.querySelector(`[data-slash-id="${activeId}"]`)
- element?.scrollIntoView({ block: "nearest", behavior: "smooth" })
- })
- })
- const selectPopoverActive = () => {
- if (store.popover === "at") {
- const items = atFlat()
- if (items.length === 0) return
- const active = atActive()
- const item = items.find((entry) => atKey(entry) === active) ?? items[0]
- handleAtSelect(item)
- return
- }
- if (store.popover === "slash") {
- const items = slashFlat()
- if (items.length === 0) return
- const active = slashActive()
- const item = items.find((entry) => entry.id === active) ?? items[0]
- handleSlashSelect(item)
- }
- }
- createEffect(
- on(
- () => prompt.current(),
- (currentParts) => {
- const inputParts = currentParts.filter((part) => part.type !== "image")
- if (mirror.input) {
- mirror.input = false
- if (isNormalizedEditor()) return
- const selection = window.getSelection()
- let cursorPosition: number | null = null
- if (selection && selection.rangeCount > 0 && editorRef.contains(selection.anchorNode)) {
- cursorPosition = getCursorPosition(editorRef)
- }
- renderEditor(inputParts)
- if (cursorPosition !== null) {
- setCursorPosition(editorRef, cursorPosition)
- }
- return
- }
- const domParts = parseFromDOM()
- if (isNormalizedEditor() && isPromptEqual(inputParts, domParts)) return
- const selection = window.getSelection()
- let cursorPosition: number | null = null
- if (selection && selection.rangeCount > 0 && editorRef.contains(selection.anchorNode)) {
- cursorPosition = getCursorPosition(editorRef)
- }
- renderEditor(inputParts)
- if (cursorPosition !== null) {
- setCursorPosition(editorRef, cursorPosition)
- }
- },
- ),
- )
- const parseFromDOM = (): Prompt => {
- const parts: Prompt = []
- let position = 0
- let buffer = ""
- const flushText = () => {
- const content = buffer.replace(/\r\n?/g, "\n").replace(/\u200B/g, "")
- buffer = ""
- if (!content) return
- parts.push({ type: "text", content, start: position, end: position + content.length })
- position += content.length
- }
- const pushFile = (file: HTMLElement) => {
- const content = file.textContent ?? ""
- parts.push({
- type: "file",
- path: file.dataset.path!,
- content,
- start: position,
- end: position + content.length,
- })
- position += content.length
- }
- const pushAgent = (agent: HTMLElement) => {
- const content = agent.textContent ?? ""
- parts.push({
- type: "agent",
- name: agent.dataset.name!,
- content,
- start: position,
- end: position + content.length,
- })
- position += content.length
- }
- const visit = (node: Node) => {
- if (node.nodeType === Node.TEXT_NODE) {
- buffer += node.textContent ?? ""
- return
- }
- if (node.nodeType !== Node.ELEMENT_NODE) return
- const el = node as HTMLElement
- if (el.dataset.type === "file") {
- flushText()
- pushFile(el)
- return
- }
- if (el.dataset.type === "agent") {
- flushText()
- pushAgent(el)
- return
- }
- if (el.tagName === "BR") {
- buffer += "\n"
- return
- }
- for (const child of Array.from(el.childNodes)) {
- visit(child)
- }
- }
- const children = Array.from(editorRef.childNodes)
- children.forEach((child, index) => {
- const isBlock = child.nodeType === Node.ELEMENT_NODE && ["DIV", "P"].includes((child as HTMLElement).tagName)
- visit(child)
- if (isBlock && index < children.length - 1) {
- buffer += "\n"
- }
- })
- flushText()
- if (parts.length === 0) parts.push(...DEFAULT_PROMPT)
- return parts
- }
- const handleInput = () => {
- const rawParts = parseFromDOM()
- const images = imageAttachments()
- const cursorPosition = getCursorPosition(editorRef)
- const rawText = rawParts.map((p) => ("content" in p ? p.content : "")).join("")
- const trimmed = rawText.replace(/\u200B/g, "").trim()
- const hasNonText = rawParts.some((part) => part.type !== "text")
- const shouldReset = trimmed.length === 0 && !hasNonText && images.length === 0
- if (shouldReset) {
- setStore("popover", null)
- if (store.historyIndex >= 0 && !store.applyingHistory) {
- setStore("historyIndex", -1)
- setStore("savedPrompt", null)
- }
- if (prompt.dirty()) {
- mirror.input = true
- prompt.set(DEFAULT_PROMPT, 0)
- }
- queueScroll()
- return
- }
- const shellMode = store.mode === "shell"
- if (!shellMode) {
- const atMatch = rawText.substring(0, cursorPosition).match(/@(\S*)$/)
- const slashMatch = rawText.match(/^\/(\S*)$/)
- if (atMatch) {
- atOnInput(atMatch[1])
- setStore("popover", "at")
- } else if (slashMatch) {
- slashOnInput(slashMatch[1])
- setStore("popover", "slash")
- } else {
- setStore("popover", null)
- }
- } else {
- setStore("popover", null)
- }
- if (store.historyIndex >= 0 && !store.applyingHistory) {
- setStore("historyIndex", -1)
- setStore("savedPrompt", null)
- }
- mirror.input = true
- prompt.set([...rawParts, ...images], cursorPosition)
- queueScroll()
- }
- const addPart = (part: ContentPart) => {
- const selection = window.getSelection()
- if (!selection || selection.rangeCount === 0) return
- const cursorPosition = getCursorPosition(editorRef)
- const currentPrompt = prompt.current()
- const rawText = currentPrompt.map((p) => ("content" in p ? p.content : "")).join("")
- const textBeforeCursor = rawText.substring(0, cursorPosition)
- const atMatch = textBeforeCursor.match(/@(\S*)$/)
- if (part.type === "file" || part.type === "agent") {
- const pill = createPill(part)
- const gap = document.createTextNode(" ")
- const range = selection.getRangeAt(0)
- if (atMatch) {
- const start = atMatch.index ?? cursorPosition - atMatch[0].length
- setRangeEdge(editorRef, range, "start", start)
- setRangeEdge(editorRef, range, "end", cursorPosition)
- }
- range.deleteContents()
- range.insertNode(gap)
- range.insertNode(pill)
- range.setStartAfter(gap)
- range.collapse(true)
- selection.removeAllRanges()
- selection.addRange(range)
- } else if (part.type === "text") {
- const range = selection.getRangeAt(0)
- const fragment = createTextFragment(part.content)
- const last = fragment.lastChild
- range.deleteContents()
- range.insertNode(fragment)
- if (last) {
- if (last.nodeType === Node.TEXT_NODE) {
- const text = last.textContent ?? ""
- if (text === "\u200B") {
- range.setStart(last, 0)
- }
- if (text !== "\u200B") {
- range.setStart(last, text.length)
- }
- }
- if (last.nodeType !== Node.TEXT_NODE) {
- range.setStartAfter(last)
- }
- }
- range.collapse(true)
- selection.removeAllRanges()
- selection.addRange(range)
- }
- handleInput()
- setStore("popover", null)
- }
- const addToHistory = (prompt: Prompt, mode: "normal" | "shell") => {
- const currentHistory = mode === "shell" ? shellHistory : history
- const setCurrentHistory = mode === "shell" ? setShellHistory : setHistory
- const next = prependHistoryEntry(currentHistory.entries, prompt)
- if (next === currentHistory.entries) return
- setCurrentHistory("entries", next)
- }
- const navigateHistory = (direction: "up" | "down") => {
- const result = navigatePromptHistory({
- direction,
- entries: store.mode === "shell" ? shellHistory.entries : history.entries,
- historyIndex: store.historyIndex,
- currentPrompt: prompt.current(),
- savedPrompt: store.savedPrompt,
- })
- if (!result.handled) return false
- setStore("historyIndex", result.historyIndex)
- setStore("savedPrompt", result.savedPrompt)
- applyHistoryPrompt(result.prompt, result.cursor)
- return true
- }
- const { addImageAttachment, removeImageAttachment, handlePaste } = createPromptAttachments({
- editor: () => editorRef,
- isFocused,
- isDialogActive: () => !!dialog.active,
- setDragging: (value) => setStore("dragging", value),
- addPart,
- })
- const { abort, handleSubmit } = createPromptSubmit({
- info,
- imageAttachments,
- commentCount,
- mode: () => store.mode,
- working,
- editor: () => editorRef,
- queueScroll,
- promptLength,
- addToHistory,
- resetHistoryNavigation: () => {
- setStore("historyIndex", -1)
- setStore("savedPrompt", null)
- },
- setMode: (mode) => setStore("mode", mode),
- setPopover: (popover) => setStore("popover", popover),
- newSessionWorktree: props.newSessionWorktree,
- onNewSessionWorktreeReset: props.onNewSessionWorktreeReset,
- onSubmit: props.onSubmit,
- })
- const handleKeyDown = (event: KeyboardEvent) => {
- if (event.key === "Backspace") {
- const selection = window.getSelection()
- if (selection && selection.isCollapsed) {
- const node = selection.anchorNode
- const offset = selection.anchorOffset
- if (node && node.nodeType === Node.TEXT_NODE) {
- const text = node.textContent ?? ""
- if (/^\u200B+$/.test(text) && offset > 0) {
- const range = document.createRange()
- range.setStart(node, 0)
- range.collapse(true)
- selection.removeAllRanges()
- selection.addRange(range)
- }
- }
- }
- }
- if (event.key === "!" && store.mode === "normal") {
- const cursorPosition = getCursorPosition(editorRef)
- if (cursorPosition === 0) {
- setStore("mode", "shell")
- setStore("popover", null)
- event.preventDefault()
- return
- }
- }
- if (store.mode === "shell") {
- const { collapsed, cursorPosition, textLength } = getCaretState()
- if (event.key === "Escape") {
- setStore("mode", "normal")
- event.preventDefault()
- return
- }
- if (event.key === "Backspace" && collapsed && cursorPosition === 0 && textLength === 0) {
- setStore("mode", "normal")
- event.preventDefault()
- return
- }
- }
- // Handle Shift+Enter BEFORE IME check - Shift+Enter is never used for IME input
- // and should always insert a newline regardless of composition state
- if (event.key === "Enter" && event.shiftKey) {
- addPart({ type: "text", content: "\n", start: 0, end: 0 })
- event.preventDefault()
- return
- }
- if (event.key === "Enter" && isImeComposing(event)) {
- return
- }
- const ctrl = event.ctrlKey && !event.metaKey && !event.altKey && !event.shiftKey
- if (store.popover) {
- if (event.key === "Tab") {
- selectPopoverActive()
- event.preventDefault()
- return
- }
- const nav = event.key === "ArrowUp" || event.key === "ArrowDown" || event.key === "Enter"
- const ctrlNav = ctrl && (event.key === "n" || event.key === "p")
- if (nav || ctrlNav) {
- if (store.popover === "at") {
- atOnKeyDown(event)
- event.preventDefault()
- return
- }
- if (store.popover === "slash") {
- slashOnKeyDown(event)
- }
- event.preventDefault()
- return
- }
- }
- if (ctrl && event.code === "KeyG") {
- if (store.popover) {
- setStore("popover", null)
- event.preventDefault()
- return
- }
- if (working()) {
- abort()
- event.preventDefault()
- }
- return
- }
- if (event.key === "ArrowUp" || event.key === "ArrowDown") {
- if (event.altKey || event.ctrlKey || event.metaKey) return
- const { collapsed } = getCaretState()
- if (!collapsed) return
- const cursorPosition = getCursorPosition(editorRef)
- const textLength = promptLength(prompt.current())
- const textContent = prompt
- .current()
- .map((part) => ("content" in part ? part.content : ""))
- .join("")
- const isEmpty = textContent.trim() === "" || textLength <= 1
- const hasNewlines = textContent.includes("\n")
- const inHistory = store.historyIndex >= 0
- const atStart = cursorPosition <= (isEmpty ? 1 : 0)
- const atEnd = cursorPosition >= (isEmpty ? textLength - 1 : textLength)
- const allowUp = isEmpty || atStart || (!hasNewlines && !inHistory) || (inHistory && atEnd)
- const allowDown = isEmpty || atEnd || (!hasNewlines && !inHistory) || (inHistory && atStart)
- if (event.key === "ArrowUp") {
- if (!allowUp) return
- if (navigateHistory("up")) {
- event.preventDefault()
- }
- return
- }
- if (!allowDown) return
- if (navigateHistory("down")) {
- event.preventDefault()
- }
- return
- }
- // Note: Shift+Enter is handled earlier, before IME check
- if (event.key === "Enter" && !event.shiftKey) {
- handleSubmit(event)
- }
- if (event.key === "Escape") {
- if (store.popover) {
- setStore("popover", null)
- } else if (working()) {
- abort()
- }
- }
- }
- return (
- <div class="relative size-full _max-h-[320px] flex flex-col gap-3">
- <PromptPopover
- popover={store.popover}
- setSlashPopoverRef={(el) => (slashPopoverRef = el)}
- atFlat={atFlat()}
- atActive={atActive() ?? undefined}
- atKey={atKey}
- setAtActive={setAtActive}
- onAtSelect={handleAtSelect}
- slashFlat={slashFlat()}
- slashActive={slashActive() ?? undefined}
- setSlashActive={setSlashActive}
- onSlashSelect={handleSlashSelect}
- commandKeybind={command.keybind}
- t={(key) => language.t(key as Parameters<typeof language.t>[0])}
- />
- <form
- onSubmit={handleSubmit}
- classList={{
- "group/prompt-input": true,
- "bg-surface-raised-stronger-non-alpha shadow-xs-border relative": true,
- "rounded-[14px] overflow-clip focus-within:shadow-xs-border": true,
- "border-icon-info-active border-dashed": store.dragging,
- [props.class ?? ""]: !!props.class,
- }}
- >
- <PromptDragOverlay dragging={store.dragging} label={language.t("prompt.dropzone.label")} />
- <PromptContextItems
- items={prompt.context.items()}
- active={(item) => {
- const active = comments.active()
- return !!item.commentID && item.commentID === active?.id && item.path === active?.file
- }}
- openComment={openComment}
- remove={(item) => {
- if (item.commentID) comments.remove(item.path, item.commentID)
- prompt.context.remove(item.key)
- }}
- t={(key) => language.t(key as Parameters<typeof language.t>[0])}
- />
- <PromptImageAttachments
- attachments={imageAttachments()}
- onOpen={(attachment) =>
- dialog.show(() => <ImagePreview src={attachment.dataUrl} alt={attachment.filename} />)
- }
- onRemove={removeImageAttachment}
- removeLabel={language.t("prompt.attachment.remove")}
- />
- <div class="relative max-h-[240px] overflow-y-auto" ref={(el) => (scrollRef = el)}>
- <div
- data-component="prompt-input"
- ref={(el) => {
- editorRef = el
- props.ref?.(el)
- }}
- role="textbox"
- aria-multiline="true"
- aria-label={placeholder()}
- contenteditable="true"
- onInput={handleInput}
- onPaste={handlePaste}
- onCompositionStart={() => setComposing(true)}
- onCompositionEnd={() => setComposing(false)}
- onKeyDown={handleKeyDown}
- classList={{
- "select-text": true,
- "w-full p-3 pr-12 text-14-regular text-text-strong focus:outline-none whitespace-pre-wrap": true,
- "[&_[data-type=file]]:text-syntax-property": true,
- "[&_[data-type=agent]]:text-syntax-type": true,
- "font-mono!": store.mode === "shell",
- }}
- />
- <Show when={!prompt.dirty()}>
- <div class="absolute top-0 inset-x-0 p-3 pr-12 text-14-regular text-text-weak pointer-events-none whitespace-nowrap truncate">
- {placeholder()}
- </div>
- </Show>
- </div>
- <div class="relative p-3 flex items-center justify-between gap-2">
- <div class="flex items-center gap-2 min-w-0 flex-1">
- <Switch>
- <Match when={store.mode === "shell"}>
- <div class="flex items-center gap-2 px-2 h-6">
- <Icon name="console" size="small" class="text-icon-primary" />
- <span class="text-12-regular text-text-primary">{language.t("prompt.mode.shell")}</span>
- <span class="text-12-regular text-text-weak">{language.t("prompt.mode.shell.exit")}</span>
- </div>
- </Match>
- <Match when={store.mode === "normal"}>
- <TooltipKeybind
- placement="top"
- gutter={8}
- title={language.t("command.agent.cycle")}
- keybind={command.keybind("agent.cycle")}
- >
- <Select
- options={local.agent.list().map((agent) => agent.name)}
- current={local.agent.current()?.name ?? ""}
- onSelect={local.agent.set}
- class={`capitalize ${local.model.variant.list().length > 0 ? "max-w-full" : "max-w-[120px]"}`}
- valueClass="truncate"
- variant="ghost"
- />
- </TooltipKeybind>
- <Show
- when={providers.paid().length > 0}
- fallback={
- <TooltipKeybind
- placement="top"
- gutter={8}
- title={language.t("command.model.choose")}
- keybind={command.keybind("model.choose")}
- >
- <Button
- as="div"
- variant="ghost"
- class="px-2 min-w-0 max-w-[240px]"
- onClick={() => dialog.show(() => <DialogSelectModelUnpaid />)}
- >
- <Show when={local.model.current()?.provider?.id}>
- <ProviderIcon id={local.model.current()!.provider.id as IconName} class="size-4 shrink-0" />
- </Show>
- <span class="truncate">
- {local.model.current()?.name ?? language.t("dialog.model.select.title")}
- </span>
- <Icon name="chevron-down" size="small" class="shrink-0" />
- </Button>
- </TooltipKeybind>
- }
- >
- <TooltipKeybind
- placement="top"
- gutter={8}
- title={language.t("command.model.choose")}
- keybind={command.keybind("model.choose")}
- >
- <ModelSelectorPopover
- triggerAs={Button}
- triggerProps={{ variant: "ghost", class: "min-w-0 max-w-[240px]" }}
- >
- <Show when={local.model.current()?.provider?.id}>
- <ProviderIcon id={local.model.current()!.provider.id as IconName} class="size-4 shrink-0" />
- </Show>
- <span class="truncate">
- {local.model.current()?.name ?? language.t("dialog.model.select.title")}
- </span>
- <Icon name="chevron-down" size="small" class="shrink-0" />
- </ModelSelectorPopover>
- </TooltipKeybind>
- </Show>
- <Show when={local.model.variant.list().length > 0}>
- <TooltipKeybind
- placement="top"
- gutter={8}
- title={language.t("command.model.variant.cycle")}
- keybind={command.keybind("model.variant.cycle")}
- >
- <Button
- data-action="model-variant-cycle"
- variant="ghost"
- class="text-text-base _hidden group-hover/prompt-input:inline-block capitalize text-12-regular"
- onClick={() => local.model.variant.cycle()}
- >
- {local.model.variant.current() ?? language.t("common.default")}
- </Button>
- </TooltipKeybind>
- </Show>
- <Show when={permission.permissionsEnabled() && params.id}>
- <TooltipKeybind
- placement="top"
- gutter={8}
- title={language.t("command.permissions.autoaccept.enable")}
- keybind={command.keybind("permissions.autoaccept")}
- >
- <Button
- variant="ghost"
- onClick={() => permission.toggleAutoAccept(params.id!, sdk.directory)}
- classList={{
- "_hidden group-hover/prompt-input:flex size-6 items-center justify-center": true,
- "text-text-base": !permission.isAutoAccepting(params.id!, sdk.directory),
- "hover:bg-surface-success-base": permission.isAutoAccepting(params.id!, sdk.directory),
- }}
- aria-label={
- permission.isAutoAccepting(params.id!, sdk.directory)
- ? language.t("command.permissions.autoaccept.disable")
- : language.t("command.permissions.autoaccept.enable")
- }
- aria-pressed={permission.isAutoAccepting(params.id!, sdk.directory)}
- >
- <Icon
- name="chevron-double-right"
- size="small"
- classList={{ "text-icon-success-base": permission.isAutoAccepting(params.id!, sdk.directory) }}
- />
- </Button>
- </TooltipKeybind>
- </Show>
- </Match>
- </Switch>
- </div>
- <div class="flex items-center gap-1 shrink-0">
- <input
- ref={fileInputRef}
- type="file"
- accept={ACCEPTED_FILE_TYPES.join(",")}
- class="hidden"
- onChange={(e) => {
- const file = e.currentTarget.files?.[0]
- if (file) addImageAttachment(file)
- e.currentTarget.value = ""
- }}
- />
- <div class="flex items-center gap-1 mr-1">
- <SessionContextUsage />
- <Show when={store.mode === "normal"}>
- <Tooltip placement="top" value={language.t("prompt.action.attachFile")}>
- <Button
- type="button"
- variant="ghost"
- class="size-6 px-1"
- onClick={() => fileInputRef.click()}
- aria-label={language.t("prompt.action.attachFile")}
- >
- <Icon name="photo" class="size-4.5" />
- </Button>
- </Tooltip>
- </Show>
- </div>
- <Tooltip
- placement="top"
- inactive={!prompt.dirty() && !working()}
- value={
- <Switch>
- <Match when={working()}>
- <div class="flex items-center gap-2">
- <span>{language.t("prompt.action.stop")}</span>
- <span class="text-icon-base text-12-medium text-[10px]!">{language.t("common.key.esc")}</span>
- </div>
- </Match>
- <Match when={true}>
- <div class="flex items-center gap-2">
- <span>{language.t("prompt.action.send")}</span>
- <Icon name="enter" size="small" class="text-icon-base" />
- </div>
- </Match>
- </Switch>
- }
- >
- <IconButton
- type="submit"
- disabled={!prompt.dirty() && !working() && commentCount() === 0}
- icon={working() ? "stop" : "arrow-up"}
- variant="primary"
- class="h-6 w-4.5"
- aria-label={working() ? language.t("prompt.action.stop") : language.t("prompt.action.send")}
- />
- </Tooltip>
- </div>
- </div>
- </form>
- </div>
- )
- }
|