Compare commits
62 Commits
codenomad/
...
v0.12.2-de
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
6a4e548d2c | ||
|
|
ad943b2bd4 | ||
|
|
6dac8a6209 | ||
|
|
bec1af6523 | ||
|
|
1719802c0f | ||
|
|
3719dcecf8 | ||
|
|
3dae143830 | ||
|
|
f050273a8e | ||
|
|
8f955cf21c | ||
|
|
a893fca66e | ||
|
|
4f8aba5658 | ||
|
|
219e012c1b | ||
|
|
17716a730b | ||
|
|
c57170d122 | ||
|
|
24c1b7e8ad | ||
|
|
3c76f9776c | ||
|
|
80a02b68b9 | ||
|
|
c766b5ab62 | ||
|
|
133e937772 | ||
|
|
95df743339 | ||
|
|
cd6266757d | ||
|
|
ec0bffe0c2 | ||
|
|
ed322a16bf | ||
|
|
044e46cd6b | ||
|
|
38f75ab06d | ||
|
|
b6bf58ea8f | ||
|
|
2c27fc53ad | ||
|
|
4c5acefa07 | ||
|
|
224cab6a42 | ||
|
|
48b2d7c5ee | ||
|
|
594809538d | ||
|
|
13802537b4 | ||
|
|
ca2b3c232f | ||
|
|
c51e71c7a2 | ||
|
|
482313f662 | ||
|
|
9a4d378238 | ||
|
|
5d5fbfb5f2 | ||
|
|
d147ad49ff | ||
|
|
9b435e3621 | ||
|
|
ab9e188b02 | ||
|
|
2991de528a | ||
|
|
f1bd681618 | ||
|
|
b91dbb1a60 | ||
|
|
688b127c6d | ||
|
|
0f9c99e3bd | ||
|
|
1122070b9c | ||
|
|
57b81f00f8 | ||
|
|
362105fe78 | ||
|
|
5834d2df1b | ||
|
|
ef4c8ef425 | ||
|
|
5f755a7e1c | ||
|
|
8607fab5b5 | ||
|
|
0368fe8248 | ||
|
|
b970281fa7 | ||
|
|
8e5a7fc213 | ||
|
|
15f362e8b5 | ||
|
|
7bbd0a1787 | ||
|
|
f8aae56728 | ||
|
|
027d7fc97d | ||
|
|
e90aef4b3c | ||
|
|
e4e89008b2 | ||
|
|
96fe1b86dd |
56
.github/workflows/build-and-upload.yml
vendored
56
.github/workflows/build-and-upload.yml
vendored
@@ -61,7 +61,21 @@ jobs:
|
|||||||
|
|
||||||
- name: Set workspace versions
|
- name: Set workspace versions
|
||||||
if: ${{ inputs.set_versions && inputs.version != '' }}
|
if: ${{ inputs.set_versions && inputs.version != '' }}
|
||||||
run: npm version ${VERSION} --workspaces --include-workspace-root --no-git-tag-version --allow-same-version
|
shell: bash
|
||||||
|
env:
|
||||||
|
NPM_CONFIG_FETCH_RETRIES: 5
|
||||||
|
NPM_CONFIG_FETCH_RETRY_MINTIMEOUT: 20000
|
||||||
|
NPM_CONFIG_FETCH_RETRY_MAXTIMEOUT: 120000
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
for attempt in 1 2 3; do
|
||||||
|
if npm version "${VERSION}" --workspaces --include-workspace-root --no-git-tag-version --allow-same-version; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
echo "npm version failed (attempt $attempt/3); retrying..." >&2
|
||||||
|
sleep $((attempt * 10))
|
||||||
|
done
|
||||||
|
exit 1
|
||||||
|
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: npm ci --workspaces --include=optional
|
run: npm ci --workspaces --include=optional
|
||||||
@@ -72,6 +86,37 @@ jobs:
|
|||||||
- name: Build macOS binaries (Electron)
|
- name: Build macOS binaries (Electron)
|
||||||
run: npm run build:mac --workspace @neuralnomads/codenomad-electron-app
|
run: npm run build:mac --workspace @neuralnomads/codenomad-electron-app
|
||||||
|
|
||||||
|
- name: Ad-hoc sign Electron macOS app bundles (seal resources)
|
||||||
|
shell: bash
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
release_root="packages/electron-app/release"
|
||||||
|
apps=()
|
||||||
|
while IFS= read -r -d '' app; do
|
||||||
|
apps+=("$app")
|
||||||
|
done < <(find "$release_root" -type d -name 'CodeNomad.app' -print0)
|
||||||
|
|
||||||
|
if [ "${#apps[@]}" -eq 0 ]; then
|
||||||
|
echo "No CodeNomad.app found under $release_root" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# GitHub macOS runners typically have no signing identity. Without any signature,
|
||||||
|
# the shipped .app can fail Gatekeeper with:
|
||||||
|
# code has no resources but signature indicates they must be present
|
||||||
|
# Ad-hoc signing seals bundle resources and makes the signature internally consistent.
|
||||||
|
if security find-identity -p codesigning -v | grep -q "0 valid identities found"; then
|
||||||
|
echo "No valid macOS codesigning identity found; applying ad-hoc signature"
|
||||||
|
for app in "${apps[@]}"; do
|
||||||
|
echo "codesign (adhoc): $app"
|
||||||
|
codesign --force --deep --sign - "$app"
|
||||||
|
codesign --verify --deep --strict --verbose=2 "$app"
|
||||||
|
done
|
||||||
|
else
|
||||||
|
echo "macOS codesigning identity present; skipping ad-hoc signing"
|
||||||
|
fi
|
||||||
|
|
||||||
- name: Repackage Electron macOS zips (ditto)
|
- name: Repackage Electron macOS zips (ditto)
|
||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
@@ -84,9 +129,12 @@ jobs:
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
release_root="packages/electron-app/release"
|
release_root="packages/electron-app/release"
|
||||||
shopt -s nullglob globstar
|
# macOS GitHub runners ship /bin/bash 3.2 which doesn't support `shopt -s globstar`.
|
||||||
|
# Use find to locate built app bundles instead of ** globs.
|
||||||
apps=("$release_root"/**/CodeNomad.app)
|
apps=()
|
||||||
|
while IFS= read -r -d '' app; do
|
||||||
|
apps+=("$app")
|
||||||
|
done < <(find "$release_root" -type d -name 'CodeNomad.app' -print0)
|
||||||
if [ "${#apps[@]}" -eq 0 ]; then
|
if [ "${#apps[@]}" -eq 0 ]; then
|
||||||
echo "No CodeNomad.app found under $release_root" >&2
|
echo "No CodeNomad.app found under $release_root" >&2
|
||||||
exit 1
|
exit 1
|
||||||
|
|||||||
13
package-lock.json
generated
13
package-lock.json
generated
@@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "codenomad-workspace",
|
"name": "codenomad-workspace",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "codenomad-workspace",
|
"name": "codenomad-workspace",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"7zip-bin": "^5.2.0",
|
"7zip-bin": "^5.2.0",
|
||||||
@@ -11985,7 +11985,7 @@
|
|||||||
},
|
},
|
||||||
"packages/electron-app": {
|
"packages/electron-app": {
|
||||||
"name": "@neuralnomads/codenomad-electron-app",
|
"name": "@neuralnomads/codenomad-electron-app",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@codenomad/ui": "file:../ui",
|
"@codenomad/ui": "file:../ui",
|
||||||
@@ -11995,6 +11995,7 @@
|
|||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"7zip-bin": "^5.2.0",
|
"7zip-bin": "^5.2.0",
|
||||||
"app-builder-bin": "^4.2.0",
|
"app-builder-bin": "^4.2.0",
|
||||||
|
"cross-env": "^7.0.3",
|
||||||
"electron": "39.0.0",
|
"electron": "39.0.0",
|
||||||
"electron-builder": "^24.0.0",
|
"electron-builder": "^24.0.0",
|
||||||
"electron-vite": "4.0.1",
|
"electron-vite": "4.0.1",
|
||||||
@@ -12021,7 +12022,7 @@
|
|||||||
},
|
},
|
||||||
"packages/server": {
|
"packages/server": {
|
||||||
"name": "@neuralnomads/codenomad",
|
"name": "@neuralnomads/codenomad",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@fastify/cors": "^8.5.0",
|
"@fastify/cors": "^8.5.0",
|
||||||
@@ -12062,7 +12063,7 @@
|
|||||||
},
|
},
|
||||||
"packages/tauri-app": {
|
"packages/tauri-app": {
|
||||||
"name": "@codenomad/tauri-app",
|
"name": "@codenomad/tauri-app",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@tauri-apps/cli": "^2.9.4"
|
"@tauri-apps/cli": "^2.9.4"
|
||||||
@@ -12070,7 +12071,7 @@
|
|||||||
},
|
},
|
||||||
"packages/ui": {
|
"packages/ui": {
|
||||||
"name": "@codenomad/ui",
|
"name": "@codenomad/ui",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@git-diff-view/solid": "^0.0.8",
|
"@git-diff-view/solid": "^0.0.8",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "codenomad-workspace",
|
"name": "codenomad-workspace",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"private": true,
|
"private": true,
|
||||||
"description": "CodeNomad monorepo workspace",
|
"description": "CodeNomad monorepo workspace",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
|
|||||||
@@ -431,7 +431,9 @@ export class CliProcessManager extends EventEmitter {
|
|||||||
|
|
||||||
if (options.dev) {
|
if (options.dev) {
|
||||||
const devServer = process.env.VITE_DEV_SERVER_URL || process.env.ELECTRON_RENDERER_URL || "http://localhost:3000"
|
const devServer = process.env.VITE_DEV_SERVER_URL || process.env.ELECTRON_RENDERER_URL || "http://localhost:3000"
|
||||||
args.push("--ui-dev-server", devServer, "--log-level", "debug")
|
const rawLogLevel = (process.env.CLI_LOG_LEVEL ?? "info").trim()
|
||||||
|
const logLevel = rawLogLevel.length > 0 ? rawLogLevel.toLowerCase() : "info"
|
||||||
|
args.push("--ui-dev-server", devServer, "--log-level", logLevel)
|
||||||
}
|
}
|
||||||
|
|
||||||
return args
|
return args
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@neuralnomads/codenomad-electron-app",
|
"name": "@neuralnomads/codenomad-electron-app",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"description": "CodeNomad - AI coding assistant",
|
"description": "CodeNomad - AI coding assistant",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"author": {
|
"author": {
|
||||||
@@ -15,7 +15,10 @@
|
|||||||
},
|
},
|
||||||
"homepage": "https://github.com/NeuralNomadsAI/CodeNomad",
|
"homepage": "https://github.com/NeuralNomadsAI/CodeNomad",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "electron-vite dev",
|
"dev": "npm run dev:info",
|
||||||
|
"dev:info": "cross-env CLI_LOG_LEVEL=info electron-vite dev",
|
||||||
|
"dev:debug": "cross-env CLI_LOG_LEVEL=debug electron-vite dev",
|
||||||
|
"dev:trace": "cross-env CLI_LOG_LEVEL=trace electron-vite dev",
|
||||||
"dev:electron": "NODE_ENV=development ELECTRON_ENABLE_LOGGING=1 NODE_OPTIONS=\"--import tsx\" electron electron/main/main.ts",
|
"dev:electron": "NODE_ENV=development ELECTRON_ENABLE_LOGGING=1 NODE_OPTIONS=\"--import tsx\" electron electron/main/main.ts",
|
||||||
"build": "electron-vite build",
|
"build": "electron-vite build",
|
||||||
"typecheck": "tsc --noEmit -p tsconfig.json",
|
"typecheck": "tsc --noEmit -p tsconfig.json",
|
||||||
@@ -42,6 +45,7 @@
|
|||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"7zip-bin": "^5.2.0",
|
"7zip-bin": "^5.2.0",
|
||||||
"app-builder-bin": "^4.2.0",
|
"app-builder-bin": "^4.2.0",
|
||||||
|
"cross-env": "^7.0.3",
|
||||||
"electron": "39.0.0",
|
"electron": "39.0.0",
|
||||||
"electron-builder": "^24.0.0",
|
"electron-builder": "^24.0.0",
|
||||||
"electron-vite": "4.0.1",
|
"electron-vite": "4.0.1",
|
||||||
|
|||||||
@@ -4,6 +4,6 @@
|
|||||||
"private": true,
|
"private": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@opencode-ai/plugin": "1.2.10"
|
"@opencode-ai/plugin": "1.2.14"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
4
packages/server/package-lock.json
generated
4
packages/server/package-lock.json
generated
@@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "@neuralnomads/codenomad",
|
"name": "@neuralnomads/codenomad",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "@neuralnomads/codenomad",
|
"name": "@neuralnomads/codenomad",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@fastify/cors": "^8.5.0",
|
"@fastify/cors": "^8.5.0",
|
||||||
"@fastify/reply-from": "^9.8.0",
|
"@fastify/reply-from": "^9.8.0",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@neuralnomads/codenomad",
|
"name": "@neuralnomads/codenomad",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"description": "CodeNomad Server",
|
"description": "CodeNomad Server",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"author": {
|
"author": {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@codenomad/tauri-app",
|
"name": "@codenomad/tauri-app",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"private": true,
|
"private": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "@codenomad/ui",
|
"name": "@codenomad/ui",
|
||||||
"version": "0.11.4",
|
"version": "0.12.2",
|
||||||
"private": true,
|
"private": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
@@ -496,17 +496,24 @@ const App: Component = () => {
|
|||||||
const isActiveInstance = () => activeInstanceId() === instance.id
|
const isActiveInstance = () => activeInstanceId() === instance.id
|
||||||
const isVisible = () => isActiveInstance() && !showFolderSelection()
|
const isVisible = () => isActiveInstance() && !showFolderSelection()
|
||||||
return (
|
return (
|
||||||
<div class="flex-1 min-h-0 overflow-hidden" style={{ display: isVisible() ? "flex" : "none" }}>
|
<div
|
||||||
<InstanceMetadataProvider instance={instance}>
|
class="flex-1 min-h-0 overflow-hidden"
|
||||||
<InstanceShell
|
style={{ display: isVisible() ? "flex" : "none" }}
|
||||||
instance={instance}
|
data-instance-id={instance.id}
|
||||||
escapeInDebounce={escapeInDebounce()}
|
data-instance-active={isActiveInstance() ? "true" : "false"}
|
||||||
paletteCommands={paletteCommands}
|
data-instance-visible={isVisible() ? "true" : "false"}
|
||||||
onCloseSession={(sessionId) => handleCloseSession(instance.id, sessionId)}
|
>
|
||||||
onNewSession={() => handleNewSession(instance.id)}
|
<InstanceMetadataProvider instance={instance}>
|
||||||
handleSidebarAgentChange={(sessionId, agent) => handleSidebarAgentChange(instance.id, sessionId, agent)}
|
<InstanceShell
|
||||||
handleSidebarModelChange={(sessionId, model) => handleSidebarModelChange(instance.id, sessionId, model)}
|
instance={instance}
|
||||||
onExecuteCommand={executeCommand}
|
isActiveInstance={isActiveInstance()}
|
||||||
|
escapeInDebounce={escapeInDebounce()}
|
||||||
|
paletteCommands={paletteCommands}
|
||||||
|
onCloseSession={(sessionId) => handleCloseSession(instance.id, sessionId)}
|
||||||
|
onNewSession={() => handleNewSession(instance.id)}
|
||||||
|
handleSidebarAgentChange={(sessionId, agent) => handleSidebarAgentChange(instance.id, sessionId, agent)}
|
||||||
|
handleSidebarModelChange={(sessionId, model) => handleSidebarModelChange(instance.id, sessionId, model)}
|
||||||
|
onExecuteCommand={executeCommand}
|
||||||
tabBarOffset={isPhoneLayout() && mobileFullscreenMode() ? 0 : instanceTabBarHeight()}
|
tabBarOffset={isPhoneLayout() && mobileFullscreenMode() ? 0 : instanceTabBarHeight()}
|
||||||
mobileFullscreenMode={isPhoneLayout() && mobileFullscreenMode()}
|
mobileFullscreenMode={isPhoneLayout() && mobileFullscreenMode()}
|
||||||
onEnterMobileFullscreen={() => void enterMobileFullscreen()}
|
onEnterMobileFullscreen={() => void enterMobileFullscreen()}
|
||||||
|
|||||||
@@ -62,6 +62,9 @@ const log = getLogger("session")
|
|||||||
|
|
||||||
interface InstanceShellProps {
|
interface InstanceShellProps {
|
||||||
instance: Instance
|
instance: Instance
|
||||||
|
// Provided by App-level instance tabs; lets us pause heavy rendering
|
||||||
|
// work for inactive instances while keeping them mounted for fast switching.
|
||||||
|
isActiveInstance?: boolean
|
||||||
escapeInDebounce: boolean
|
escapeInDebounce: boolean
|
||||||
paletteCommands: Accessor<Command[]>
|
paletteCommands: Accessor<Command[]>
|
||||||
onCloseSession: (sessionId: string) => Promise<void> | void
|
onCloseSession: (sessionId: string) => Promise<void> | void
|
||||||
@@ -115,6 +118,7 @@ const InstanceShell2: Component<InstanceShellProps> = (props) => {
|
|||||||
const desktopQuery = useMediaQuery("(min-width: 1280px)")
|
const desktopQuery = useMediaQuery("(min-width: 1280px)")
|
||||||
|
|
||||||
const tabletQuery = useMediaQuery("(min-width: 768px)")
|
const tabletQuery = useMediaQuery("(min-width: 768px)")
|
||||||
|
const compactHeaderQuery = useMediaQuery("(max-width: 1024px)")
|
||||||
|
|
||||||
const layoutMode = createMemo<LayoutMode>(() => {
|
const layoutMode = createMemo<LayoutMode>(() => {
|
||||||
if (desktopQuery()) return "desktop"
|
if (desktopQuery()) return "desktop"
|
||||||
@@ -123,6 +127,7 @@ const InstanceShell2: Component<InstanceShellProps> = (props) => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
const isPhoneLayout = createMemo(() => layoutMode() === "phone")
|
const isPhoneLayout = createMemo(() => layoutMode() === "phone")
|
||||||
|
const compactHeaderLayout = createMemo(() => isPhoneLayout() || compactHeaderQuery())
|
||||||
const mobileFullscreen = createMemo(() => props.mobileFullscreenMode && isPhoneLayout())
|
const mobileFullscreen = createMemo(() => props.mobileFullscreenMode && isPhoneLayout())
|
||||||
const compactPromptLayout = createMemo(() => layoutMode() !== "desktop")
|
const compactPromptLayout = createMemo(() => layoutMode() !== "desktop")
|
||||||
const leftPinningSupported = createMemo(() => layoutMode() !== "phone")
|
const leftPinningSupported = createMemo(() => layoutMode() !== "phone")
|
||||||
@@ -596,7 +601,7 @@ const InstanceShell2: Component<InstanceShellProps> = (props) => {
|
|||||||
<AppBar position="sticky" color="default" elevation={0} class="border-b border-base">
|
<AppBar position="sticky" color="default" elevation={0} class="border-b border-base">
|
||||||
<Toolbar variant="dense" class="session-toolbar flex flex-wrap items-center gap-2 py-0 min-h-[40px]">
|
<Toolbar variant="dense" class="session-toolbar flex flex-wrap items-center gap-2 py-0 min-h-[40px]">
|
||||||
<Show
|
<Show
|
||||||
when={!isPhoneLayout()}
|
when={!compactHeaderLayout()}
|
||||||
fallback={
|
fallback={
|
||||||
<div class="flex flex-col w-full gap-1.5">
|
<div class="flex flex-col w-full gap-1.5">
|
||||||
<div class="flex flex-wrap items-center justify-between gap-2 w-full">
|
<div class="flex flex-wrap items-center justify-between gap-2 w-full">
|
||||||
@@ -634,8 +639,8 @@ const InstanceShell2: Component<InstanceShellProps> = (props) => {
|
|||||||
</button>
|
</button>
|
||||||
<span class="connection-status-shortcut-hint kbd-hint">
|
<span class="connection-status-shortcut-hint kbd-hint">
|
||||||
<Kbd shortcut="cmd+shift+p" />
|
<Kbd shortcut="cmd+shift+p" />
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex-1 flex items-center justify-center min-w-0">
|
<div class="flex-1 flex items-center justify-center min-w-0">
|
||||||
<span
|
<span
|
||||||
@@ -646,7 +651,7 @@ const InstanceShell2: Component<InstanceShellProps> = (props) => {
|
|||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Show when={!props.mobileFullscreenMode}>
|
<Show when={isPhoneLayout() && !props.mobileFullscreenMode}>
|
||||||
<IconButton
|
<IconButton
|
||||||
color="inherit"
|
color="inherit"
|
||||||
onClick={props.onEnterMobileFullscreen}
|
onClick={props.onEnterMobileFullscreen}
|
||||||
@@ -670,16 +675,18 @@ const InstanceShell2: Component<InstanceShellProps> = (props) => {
|
|||||||
{rightAppBarButtonIcon()}
|
{rightAppBarButtonIcon()}
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</Show>
|
</Show>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex flex-wrap items-center justify-center gap-2 pb-1">
|
<div class="flex flex-wrap items-center justify-center gap-2 pb-1">
|
||||||
<ContextMeter
|
<Show when={!showingInfoView()}>
|
||||||
usedTokens={tokenStats().used}
|
<ContextMeter
|
||||||
availableTokens={tokenStats().avail}
|
usedTokens={tokenStats().used}
|
||||||
formatTokens={formatTokenTotal}
|
availableTokens={tokenStats().avail}
|
||||||
usedLabel={t("instanceShell.metrics.usedLabel")}
|
formatTokens={formatTokenTotal}
|
||||||
availableLabel={t("instanceShell.metrics.availableLabel")}
|
usedLabel={t("instanceShell.metrics.usedLabel")}
|
||||||
/>
|
availableLabel={t("instanceShell.metrics.availableLabel")}
|
||||||
|
/>
|
||||||
|
</Show>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
@@ -796,12 +803,14 @@ const InstanceShell2: Component<InstanceShellProps> = (props) => {
|
|||||||
>
|
>
|
||||||
<For each={cachedSessionIds()}>
|
<For each={cachedSessionIds()}>
|
||||||
{(sessionId) => {
|
{(sessionId) => {
|
||||||
const isActive = () => activeSessionIdForInstance() === sessionId
|
const isActive = () => Boolean(props.isActiveInstance) && activeSessionIdForInstance() === sessionId
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
class="session-cache-pane flex flex-col flex-1 min-h-0"
|
class="session-cache-pane flex flex-col flex-1 min-h-0"
|
||||||
style={{ display: isActive() ? "flex" : "none" }}
|
style={{ display: isActive() ? "flex" : "none" }}
|
||||||
data-session-id={sessionId}
|
data-session-id={sessionId}
|
||||||
|
data-instance-id={props.instance.id}
|
||||||
|
data-session-active={isActive() ? "true" : "false"}
|
||||||
aria-hidden={!isActive()}
|
aria-hidden={!isActive()}
|
||||||
>
|
>
|
||||||
<SessionView
|
<SessionView
|
||||||
@@ -837,7 +846,10 @@ const InstanceShell2: Component<InstanceShellProps> = (props) => {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div class="instance-shell2 flex flex-col flex-1 min-h-0">
|
<div
|
||||||
|
class="instance-shell2 flex flex-col flex-1 min-h-0"
|
||||||
|
data-instance-id={props.instance.id}
|
||||||
|
>
|
||||||
<Show when={hasSessions()} fallback={<InstanceWelcomeView instance={props.instance} />}>
|
<Show when={hasSessions()} fallback={<InstanceWelcomeView instance={props.instance} />}>
|
||||||
{sessionLayout}
|
{sessionLayout}
|
||||||
</Show>
|
</Show>
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ import {
|
|||||||
type Accessor,
|
type Accessor,
|
||||||
type Component,
|
type Component,
|
||||||
} from "solid-js"
|
} from "solid-js"
|
||||||
import type { ToolState } from "@opencode-ai/sdk"
|
import type { ToolState } from "@opencode-ai/sdk/v2"
|
||||||
import type { FileContent, FileNode, File as GitFileStatus } from "@opencode-ai/sdk/v2/client"
|
import type { FileContent, FileNode, File as GitFileStatus } from "@opencode-ai/sdk/v2/client"
|
||||||
import IconButton from "@suid/material/IconButton"
|
import IconButton from "@suid/material/IconButton"
|
||||||
import MenuOpenIcon from "@suid/icons-material/MenuOpen"
|
import MenuOpenIcon from "@suid/icons-material/MenuOpen"
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { For, Show, type Accessor, type Component, type JSX } from "solid-js"
|
import { For, Show, createMemo, type Accessor, type Component, type JSX } from "solid-js"
|
||||||
|
|
||||||
import { MonacoDiffViewer } from "../../../../file-viewer/monaco-diff-viewer"
|
import { MonacoDiffViewer } from "../../../../file-viewer/monaco-diff-viewer"
|
||||||
|
|
||||||
@@ -32,14 +32,18 @@ interface ChangesTabProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const ChangesTab: Component<ChangesTabProps> = (props) => {
|
const ChangesTab: Component<ChangesTabProps> = (props) => {
|
||||||
const renderContent = (): JSX.Element => {
|
const sessionId = createMemo(() => props.activeSessionId())
|
||||||
const sessionId = props.activeSessionId()
|
const hasSession = createMemo(() => Boolean(sessionId() && sessionId() !== "info"))
|
||||||
|
const diffs = createMemo(() => (hasSession() ? props.activeSessionDiffs() : null))
|
||||||
|
|
||||||
const hasSession = Boolean(sessionId && sessionId !== "info")
|
const sorted = createMemo<any[]>(() => {
|
||||||
const diffs = hasSession ? props.activeSessionDiffs() : null
|
const list = diffs()
|
||||||
|
if (!Array.isArray(list)) return []
|
||||||
|
return [...list].sort((a, b) => String(a.file || "").localeCompare(String(b.file || "")))
|
||||||
|
})
|
||||||
|
|
||||||
const sorted = Array.isArray(diffs) ? [...diffs].sort((a, b) => String(a.file || "").localeCompare(String(b.file || ""))) : []
|
const totals = createMemo(() => {
|
||||||
const totals = sorted.reduce(
|
return sorted().reduce(
|
||||||
(acc, item) => {
|
(acc, item) => {
|
||||||
acc.additions += typeof item.additions === "number" ? item.additions : 0
|
acc.additions += typeof item.additions === "number" ? item.additions : 0
|
||||||
acc.deletions += typeof item.deletions === "number" ? item.deletions : 0
|
acc.deletions += typeof item.deletions === "number" ? item.deletions : 0
|
||||||
@@ -47,41 +51,61 @@ const ChangesTab: Component<ChangesTabProps> = (props) => {
|
|||||||
},
|
},
|
||||||
{ additions: 0, deletions: 0 },
|
{ additions: 0, deletions: 0 },
|
||||||
)
|
)
|
||||||
|
})
|
||||||
|
|
||||||
const mostChanged = sorted.length
|
const mostChanged = createMemo<any | null>(() => {
|
||||||
? sorted.reduce((best, item) => {
|
const items = sorted()
|
||||||
const bestAdd = typeof (best as any)?.additions === "number" ? (best as any).additions : 0
|
if (items.length === 0) return null
|
||||||
const bestDel = typeof (best as any)?.deletions === "number" ? (best as any).deletions : 0
|
return items.reduce((best, item) => {
|
||||||
const bestScore = bestAdd + bestDel
|
const bestAdd = typeof (best as any)?.additions === "number" ? (best as any).additions : 0
|
||||||
|
const bestDel = typeof (best as any)?.deletions === "number" ? (best as any).deletions : 0
|
||||||
|
const bestScore = bestAdd + bestDel
|
||||||
|
|
||||||
const add = typeof (item as any)?.additions === "number" ? (item as any).additions : 0
|
const add = typeof (item as any)?.additions === "number" ? (item as any).additions : 0
|
||||||
const del = typeof (item as any)?.deletions === "number" ? (item as any).deletions : 0
|
const del = typeof (item as any)?.deletions === "number" ? (item as any).deletions : 0
|
||||||
const score = add + del
|
const score = add + del
|
||||||
|
|
||||||
if (score > bestScore) return item
|
if (score > bestScore) return item
|
||||||
if (score < bestScore) return best
|
if (score < bestScore) return best
|
||||||
return String(item.file || "").localeCompare(String((best as any)?.file || "")) < 0 ? item : best
|
return String(item.file || "").localeCompare(String((best as any)?.file || "")) < 0 ? item : best
|
||||||
}, sorted[0])
|
}, items[0])
|
||||||
: null
|
})
|
||||||
|
|
||||||
// Auto-select the most-changed file if none selected.
|
const selectedFileData = createMemo<any | null>(() => {
|
||||||
const currentSelected = props.selectedFile()
|
const currentSelected = props.selectedFile()
|
||||||
const selectedFileData = sorted.find((f) => f.file === currentSelected) || mostChanged
|
const items = sorted()
|
||||||
|
if (currentSelected) {
|
||||||
const scopeKey = `${props.instanceId}:${hasSession ? sessionId : "no-session"}`
|
const match = items.find((f) => f.file === currentSelected)
|
||||||
|
if (match) return match
|
||||||
const emptyViewerMessage = () => {
|
|
||||||
if (!hasSession) return props.t("instanceShell.sessionChanges.noSessionSelected")
|
|
||||||
if (diffs === undefined) return props.t("instanceShell.sessionChanges.loading")
|
|
||||||
if (!Array.isArray(diffs) || diffs.length === 0) return props.t("instanceShell.sessionChanges.empty")
|
|
||||||
return props.t("instanceShell.filesShell.viewerEmpty")
|
|
||||||
}
|
}
|
||||||
|
return mostChanged()
|
||||||
|
})
|
||||||
|
|
||||||
|
const scopeKey = createMemo(() => `${props.instanceId}:${hasSession() ? sessionId() : "no-session"}`)
|
||||||
|
|
||||||
|
const emptyViewerMessage = createMemo(() => {
|
||||||
|
if (!hasSession()) return props.t("instanceShell.sessionChanges.noSessionSelected")
|
||||||
|
const currentDiffs = diffs()
|
||||||
|
if (currentDiffs === undefined) return props.t("instanceShell.sessionChanges.loading")
|
||||||
|
if (!Array.isArray(currentDiffs) || currentDiffs.length === 0) return props.t("instanceShell.sessionChanges.empty")
|
||||||
|
return props.t("instanceShell.filesShell.viewerEmpty")
|
||||||
|
})
|
||||||
|
|
||||||
|
const headerPath = createMemo(() => {
|
||||||
|
const file = selectedFileData()
|
||||||
|
return file?.file ? String(file.file) : props.t("instanceShell.rightPanel.tabs.changes")
|
||||||
|
})
|
||||||
|
|
||||||
|
const renderContent = (): JSX.Element => {
|
||||||
|
const sortedList = sorted()
|
||||||
|
const totalsValue = totals()
|
||||||
|
const selected = selectedFileData()
|
||||||
|
|
||||||
const renderViewer = () => (
|
const renderViewer = () => (
|
||||||
<div class="file-viewer-panel flex-1">
|
<div class="file-viewer-panel flex-1">
|
||||||
<div class="file-viewer-content file-viewer-content--monaco">
|
<div class="file-viewer-content file-viewer-content--monaco">
|
||||||
<Show
|
<Show
|
||||||
when={selectedFileData && hasSession && Array.isArray(diffs) && diffs.length > 0 ? selectedFileData : null}
|
when={selected && hasSession() && sortedList.length > 0 ? selected : null}
|
||||||
fallback={
|
fallback={
|
||||||
<div class="file-viewer-empty">
|
<div class="file-viewer-empty">
|
||||||
<span class="file-viewer-empty-text">{emptyViewerMessage()}</span>
|
<span class="file-viewer-empty-text">{emptyViewerMessage()}</span>
|
||||||
@@ -90,7 +114,7 @@ const ChangesTab: Component<ChangesTabProps> = (props) => {
|
|||||||
>
|
>
|
||||||
{(file) => (
|
{(file) => (
|
||||||
<MonacoDiffViewer
|
<MonacoDiffViewer
|
||||||
scopeKey={scopeKey}
|
scopeKey={scopeKey()}
|
||||||
path={String(file().file || "")}
|
path={String(file().file || "")}
|
||||||
before={String((file() as any).before || "")}
|
before={String((file() as any).before || "")}
|
||||||
after={String((file() as any).after || "")}
|
after={String((file() as any).after || "")}
|
||||||
@@ -109,11 +133,11 @@ const ChangesTab: Component<ChangesTabProps> = (props) => {
|
|||||||
)
|
)
|
||||||
|
|
||||||
const renderListPanel = () => (
|
const renderListPanel = () => (
|
||||||
<Show when={sorted.length > 0} fallback={renderEmptyList()}>
|
<Show when={sortedList.length > 0} fallback={renderEmptyList()}>
|
||||||
<For each={sorted}>
|
<For each={sortedList}>
|
||||||
{(item) => (
|
{(item) => (
|
||||||
<div
|
<div
|
||||||
class={`file-list-item ${selectedFileData?.file === item.file ? "file-list-item-active" : ""}`}
|
class={`file-list-item ${selected?.file === item.file ? "file-list-item-active" : ""}`}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
props.onSelectFile(item.file, props.isPhoneLayout())
|
props.onSelectFile(item.file, props.isPhoneLayout())
|
||||||
}}
|
}}
|
||||||
@@ -134,11 +158,11 @@ const ChangesTab: Component<ChangesTabProps> = (props) => {
|
|||||||
)
|
)
|
||||||
|
|
||||||
const renderListOverlay = () => (
|
const renderListOverlay = () => (
|
||||||
<Show when={sorted.length > 0} fallback={renderEmptyList()}>
|
<Show when={sortedList.length > 0} fallback={renderEmptyList()}>
|
||||||
<For each={sorted}>
|
<For each={sortedList}>
|
||||||
{(item) => (
|
{(item) => (
|
||||||
<div
|
<div
|
||||||
class={`file-list-item ${selectedFileData?.file === item.file ? "file-list-item-active" : ""}`}
|
class={`file-list-item ${selected?.file === item.file ? "file-list-item-active" : ""}`}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
props.onSelectFile(item.file, true)
|
props.onSelectFile(item.file, true)
|
||||||
}}
|
}}
|
||||||
@@ -159,8 +183,6 @@ const ChangesTab: Component<ChangesTabProps> = (props) => {
|
|||||||
</Show>
|
</Show>
|
||||||
)
|
)
|
||||||
|
|
||||||
const headerPath = () => (selectedFileData?.file ? selectedFileData.file : props.t("instanceShell.rightPanel.tabs.changes"))
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SplitFilePanel
|
<SplitFilePanel
|
||||||
header={
|
header={
|
||||||
@@ -171,10 +193,10 @@ const ChangesTab: Component<ChangesTabProps> = (props) => {
|
|||||||
|
|
||||||
<div class="files-tab-stats" style={{ flex: "0 0 auto" }}>
|
<div class="files-tab-stats" style={{ flex: "0 0 auto" }}>
|
||||||
<span class="files-tab-stat files-tab-stat-additions">
|
<span class="files-tab-stat files-tab-stat-additions">
|
||||||
<span class="files-tab-stat-value">+{totals.additions}</span>
|
<span class="files-tab-stat-value">+{totalsValue.additions}</span>
|
||||||
</span>
|
</span>
|
||||||
<span class="files-tab-stat files-tab-stat-deletions">
|
<span class="files-tab-stat files-tab-stat-deletions">
|
||||||
<span class="files-tab-stat-value">-{totals.deletions}</span>
|
<span class="files-tab-stat-value">-{totalsValue.deletions}</span>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { For, Show, type Accessor, type Component, type JSX } from "solid-js"
|
import { For, Show, createMemo, type Accessor, type Component, type JSX } from "solid-js"
|
||||||
import type { File as GitFileStatus } from "@opencode-ai/sdk/v2/client"
|
import type { File as GitFileStatus } from "@opencode-ai/sdk/v2/client"
|
||||||
|
|
||||||
import { RefreshCw } from "lucide-solid"
|
import { RefreshCw } from "lucide-solid"
|
||||||
@@ -46,17 +46,18 @@ interface GitChangesTabProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const GitChangesTab: Component<GitChangesTabProps> = (props) => {
|
const GitChangesTab: Component<GitChangesTabProps> = (props) => {
|
||||||
const renderContent = (): JSX.Element => {
|
const sessionId = createMemo(() => props.activeSessionId())
|
||||||
const sessionId = props.activeSessionId()
|
const hasSession = createMemo(() => Boolean(sessionId() && sessionId() !== "info"))
|
||||||
|
const entries = createMemo(() => (hasSession() ? props.entries() : null))
|
||||||
|
|
||||||
const hasSession = Boolean(sessionId && sessionId !== "info")
|
const sorted = createMemo<GitFileStatus[]>(() => {
|
||||||
const entries = hasSession ? props.entries() : null
|
const list = entries()
|
||||||
|
if (!Array.isArray(list)) return []
|
||||||
|
return [...list].sort((a, b) => String(a.path || "").localeCompare(String(b.path || "")))
|
||||||
|
})
|
||||||
|
|
||||||
const sorted = Array.isArray(entries)
|
const totals = createMemo(() => {
|
||||||
? [...entries].sort((a, b) => String(a.path || "").localeCompare(String(b.path || "")))
|
return sorted().reduce(
|
||||||
: []
|
|
||||||
|
|
||||||
const totals = sorted.reduce(
|
|
||||||
(acc, item) => {
|
(acc, item) => {
|
||||||
acc.additions += typeof item.added === "number" ? item.added : 0
|
acc.additions += typeof item.added === "number" ? item.added : 0
|
||||||
acc.deletions += typeof item.removed === "number" ? item.removed : 0
|
acc.deletions += typeof item.removed === "number" ? item.removed : 0
|
||||||
@@ -64,21 +65,33 @@ const GitChangesTab: Component<GitChangesTabProps> = (props) => {
|
|||||||
},
|
},
|
||||||
{ additions: 0, deletions: 0 },
|
{ additions: 0, deletions: 0 },
|
||||||
)
|
)
|
||||||
|
})
|
||||||
|
|
||||||
const nonDeleted = sorted.filter((item) => item && item.status !== "deleted")
|
const nonDeleted = createMemo(() => sorted().filter((item) => item && item.status !== "deleted"))
|
||||||
|
|
||||||
const emptyViewerMessage = () => {
|
|
||||||
if (!hasSession) return "Select a session to view changes."
|
|
||||||
if (entries === null) return "Loading git changes…"
|
|
||||||
if (nonDeleted.length === 0) return "No git changes yet."
|
|
||||||
return "No file selected."
|
|
||||||
}
|
|
||||||
|
|
||||||
|
const selectedEntry = createMemo<GitFileStatus | null>(() => {
|
||||||
|
const list = sorted()
|
||||||
const selectedPath = props.selectedPath()
|
const selectedPath = props.selectedPath()
|
||||||
const fallbackPath = props.mostChangedPath()
|
const fallbackPath = props.mostChangedPath()
|
||||||
const selectedEntry =
|
const found =
|
||||||
sorted.find((item) => item.path === selectedPath) ||
|
list.find((item) => item.path === selectedPath) ||
|
||||||
(fallbackPath ? sorted.find((item) => item.path === fallbackPath) : null)
|
(fallbackPath ? list.find((item) => item.path === fallbackPath) : undefined)
|
||||||
|
return found ?? null
|
||||||
|
})
|
||||||
|
|
||||||
|
const emptyViewerMessage = createMemo(() => {
|
||||||
|
if (!hasSession()) return "Select a session to view changes."
|
||||||
|
const currentEntries = entries()
|
||||||
|
if (currentEntries === null) return "Loading git changes…"
|
||||||
|
if (nonDeleted().length === 0) return "No git changes yet."
|
||||||
|
return "No file selected."
|
||||||
|
})
|
||||||
|
|
||||||
|
const renderContent = (): JSX.Element => {
|
||||||
|
const totalsValue = totals()
|
||||||
|
const selected = selectedEntry()
|
||||||
|
const sortedList = sorted()
|
||||||
|
const nonDeletedList = nonDeleted()
|
||||||
|
|
||||||
const renderViewer = () => (
|
const renderViewer = () => (
|
||||||
<div class="file-viewer-panel flex-1">
|
<div class="file-viewer-panel flex-1">
|
||||||
@@ -91,12 +104,12 @@ const GitChangesTab: Component<GitChangesTabProps> = (props) => {
|
|||||||
fallback={
|
fallback={
|
||||||
<Show
|
<Show
|
||||||
when={
|
when={
|
||||||
selectedEntry &&
|
selected &&
|
||||||
props.selectedBefore() !== null &&
|
props.selectedBefore() !== null &&
|
||||||
props.selectedAfter() !== null &&
|
props.selectedAfter() !== null &&
|
||||||
selectedEntry.status !== "deleted"
|
selected.status !== "deleted"
|
||||||
? {
|
? {
|
||||||
path: selectedEntry.path,
|
path: selected.path,
|
||||||
before: props.selectedBefore() as string,
|
before: props.selectedBefore() as string,
|
||||||
after: props.selectedAfter() as string,
|
after: props.selectedAfter() as string,
|
||||||
}
|
}
|
||||||
@@ -109,16 +122,16 @@ const GitChangesTab: Component<GitChangesTabProps> = (props) => {
|
|||||||
}
|
}
|
||||||
>
|
>
|
||||||
{(file) => (
|
{(file) => (
|
||||||
<MonacoDiffViewer
|
<MonacoDiffViewer
|
||||||
scopeKey={props.scopeKey()}
|
scopeKey={props.scopeKey()}
|
||||||
path={String(file().path || "")}
|
path={String(file().path || "")}
|
||||||
before={String((file() as any).before || "")}
|
before={String((file() as any).before || "")}
|
||||||
after={String((file() as any).after || "")}
|
after={String((file() as any).after || "")}
|
||||||
viewMode={props.diffViewMode()}
|
viewMode={props.diffViewMode()}
|
||||||
contextMode={props.diffContextMode()}
|
contextMode={props.diffContextMode()}
|
||||||
wordWrap={props.diffWordWrapMode()}
|
wordWrap={props.diffWordWrapMode()}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</Show>
|
</Show>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
@@ -141,8 +154,8 @@ const GitChangesTab: Component<GitChangesTabProps> = (props) => {
|
|||||||
const renderEmptyList = () => <div class="p-3 text-xs text-secondary">{emptyViewerMessage()}</div>
|
const renderEmptyList = () => <div class="p-3 text-xs text-secondary">{emptyViewerMessage()}</div>
|
||||||
|
|
||||||
const renderListPanel = () => (
|
const renderListPanel = () => (
|
||||||
<Show when={nonDeleted.length > 0} fallback={renderEmptyList()}>
|
<Show when={nonDeletedList.length > 0} fallback={renderEmptyList()}>
|
||||||
<For each={sorted}>
|
<For each={sortedList}>
|
||||||
{(item) => (
|
{(item) => (
|
||||||
<div
|
<div
|
||||||
class={`file-list-item ${props.selectedPath() === item.path ? "file-list-item-active" : ""}`}
|
class={`file-list-item ${props.selectedPath() === item.path ? "file-list-item-active" : ""}`}
|
||||||
@@ -173,8 +186,8 @@ const GitChangesTab: Component<GitChangesTabProps> = (props) => {
|
|||||||
)
|
)
|
||||||
|
|
||||||
const renderListOverlay = () => (
|
const renderListOverlay = () => (
|
||||||
<Show when={nonDeleted.length > 0} fallback={renderEmptyList()}>
|
<Show when={nonDeletedList.length > 0} fallback={renderEmptyList()}>
|
||||||
<For each={sorted}>
|
<For each={sortedList}>
|
||||||
{(item) => (
|
{(item) => (
|
||||||
<div
|
<div
|
||||||
class={`file-list-item ${props.selectedPath() === item.path ? "file-list-item-active" : ""}`}
|
class={`file-list-item ${props.selectedPath() === item.path ? "file-list-item-active" : ""}`}
|
||||||
@@ -204,19 +217,19 @@ const GitChangesTab: Component<GitChangesTabProps> = (props) => {
|
|||||||
)
|
)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SplitFilePanel
|
<SplitFilePanel
|
||||||
header={
|
header={
|
||||||
<>
|
<>
|
||||||
<span class="files-tab-selected-path" title={selectedEntry?.path || "Git Changes"}>
|
<span class="files-tab-selected-path" title={selected?.path || "Git Changes"}>
|
||||||
<span class="file-path-text">{selectedEntry?.path || "Git Changes"}</span>
|
<span class="file-path-text">{selected?.path || "Git Changes"}</span>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<div class="files-tab-stats" style={{ flex: "0 0 auto" }}>
|
<div class="files-tab-stats" style={{ flex: "0 0 auto" }}>
|
||||||
<span class="files-tab-stat files-tab-stat-additions">
|
<span class="files-tab-stat files-tab-stat-additions">
|
||||||
<span class="files-tab-stat-value">+{totals.additions}</span>
|
<span class="files-tab-stat-value">+{totalsValue.additions}</span>
|
||||||
</span>
|
</span>
|
||||||
<span class="files-tab-stat files-tab-stat-deletions">
|
<span class="files-tab-stat files-tab-stat-deletions">
|
||||||
<span class="files-tab-stat-value">-{totals.deletions}</span>
|
<span class="files-tab-stat-value">-{totalsValue.deletions}</span>
|
||||||
</span>
|
</span>
|
||||||
<Show when={props.statusError()}>{(err) => <span class="text-error">{err()}</span>}</Show>
|
<Show when={props.statusError()}>{(err) => <span class="text-error">{err()}</span>}</Show>
|
||||||
</div>
|
</div>
|
||||||
@@ -226,23 +239,23 @@ const GitChangesTab: Component<GitChangesTabProps> = (props) => {
|
|||||||
class="files-header-icon-button"
|
class="files-header-icon-button"
|
||||||
title={props.t("instanceShell.rightPanel.actions.refresh")}
|
title={props.t("instanceShell.rightPanel.actions.refresh")}
|
||||||
aria-label={props.t("instanceShell.rightPanel.actions.refresh")}
|
aria-label={props.t("instanceShell.rightPanel.actions.refresh")}
|
||||||
disabled={!hasSession || props.statusLoading() || entries === null}
|
disabled={!hasSession() || props.statusLoading() || entries() === null}
|
||||||
style={{ "margin-left": "auto" }}
|
style={{ "margin-left": "auto" }}
|
||||||
onClick={() => props.onRefresh()}
|
onClick={() => props.onRefresh()}
|
||||||
>
|
>
|
||||||
<RefreshCw class={`h-4 w-4${props.statusLoading() ? " animate-spin" : ""}`} />
|
<RefreshCw class={`h-4 w-4${props.statusLoading() ? " animate-spin" : ""}`} />
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<DiffToolbar
|
<DiffToolbar
|
||||||
viewMode={props.diffViewMode()}
|
viewMode={props.diffViewMode()}
|
||||||
contextMode={props.diffContextMode()}
|
contextMode={props.diffContextMode()}
|
||||||
wordWrapMode={props.diffWordWrapMode()}
|
wordWrapMode={props.diffWordWrapMode()}
|
||||||
onViewModeChange={props.onViewModeChange}
|
onViewModeChange={props.onViewModeChange}
|
||||||
onContextModeChange={props.onContextModeChange}
|
onContextModeChange={props.onContextModeChange}
|
||||||
onWordWrapModeChange={props.onWordWrapModeChange}
|
onWordWrapModeChange={props.onWordWrapModeChange}
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
list={{ panel: renderListPanel, overlay: renderListOverlay }}
|
list={{ panel: renderListPanel, overlay: renderListOverlay }}
|
||||||
viewer={renderViewer()}
|
viewer={renderViewer()}
|
||||||
listOpen={props.listOpen()}
|
listOpen={props.listOpen()}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { For, Show, type Accessor, type Component } from "solid-js"
|
import { For, Show, type Accessor, type Component } from "solid-js"
|
||||||
import type { ToolState } from "@opencode-ai/sdk"
|
import type { ToolState } from "@opencode-ai/sdk/v2"
|
||||||
import { Accordion } from "@kobalte/core"
|
import { Accordion } from "@kobalte/core"
|
||||||
import { Tooltip } from "@kobalte/core/tooltip"
|
import { Tooltip } from "@kobalte/core/tooltip"
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { batch, createMemo, type Accessor } from "solid-js"
|
import { batch, createMemo, type Accessor } from "solid-js"
|
||||||
import type { ToolState } from "@opencode-ai/sdk"
|
import type { ToolState } from "@opencode-ai/sdk/v2"
|
||||||
import type { Session } from "../../../types/session"
|
import type { Session } from "../../../types/session"
|
||||||
import {
|
import {
|
||||||
activeParentSessionId,
|
activeParentSessionId,
|
||||||
|
|||||||
@@ -92,7 +92,6 @@ export function Markdown(props: MarkdownProps) {
|
|||||||
const globalCache = cacheHandle.get<RenderCache>()
|
const globalCache = cacheHandle.get<RenderCache>()
|
||||||
if (globalCache && cacheMatches(globalCache)) {
|
if (globalCache && cacheMatches(globalCache)) {
|
||||||
setHtml(globalCache.html)
|
setHtml(globalCache.html)
|
||||||
part.renderCache = globalCache
|
|
||||||
notifyRendered()
|
notifyRendered()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -100,14 +99,11 @@ export function Markdown(props: MarkdownProps) {
|
|||||||
const commitCacheEntry = (renderedHtml: string) => {
|
const commitCacheEntry = (renderedHtml: string) => {
|
||||||
const cacheEntry: RenderCache = { text, html: renderedHtml, theme: themeKey, mode: version }
|
const cacheEntry: RenderCache = { text, html: renderedHtml, theme: themeKey, mode: version }
|
||||||
setHtml(renderedHtml)
|
setHtml(renderedHtml)
|
||||||
part.renderCache = cacheEntry
|
|
||||||
cacheHandle.set(cacheEntry)
|
cacheHandle.set(cacheEntry)
|
||||||
notifyRendered()
|
notifyRendered()
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!highlightEnabled) {
|
if (!highlightEnabled) {
|
||||||
part.renderCache = undefined
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const rendered = await renderMarkdown(text, { suppressHighlight: true })
|
const rendered = await renderMarkdown(text, { suppressHighlight: true })
|
||||||
|
|
||||||
@@ -185,7 +181,6 @@ export function Markdown(props: MarkdownProps) {
|
|||||||
if (latestRequestedText === text) {
|
if (latestRequestedText === text) {
|
||||||
const cacheEntry: RenderCache = { text, html: rendered, theme: themeKey, mode: version }
|
const cacheEntry: RenderCache = { text, html: rendered, theme: themeKey, mode: version }
|
||||||
setHtml(rendered)
|
setHtml(rendered)
|
||||||
part.renderCache = cacheEntry
|
|
||||||
cacheHandle.set(cacheEntry)
|
cacheHandle.set(cacheEntry)
|
||||||
notifyRendered()
|
notifyRendered()
|
||||||
}
|
}
|
||||||
@@ -202,5 +197,15 @@ export function Markdown(props: MarkdownProps) {
|
|||||||
|
|
||||||
const proseClass = () => "markdown-body"
|
const proseClass = () => "markdown-body"
|
||||||
|
|
||||||
return <div ref={containerRef} class={proseClass()} innerHTML={html()} />
|
return (
|
||||||
|
<div
|
||||||
|
ref={containerRef}
|
||||||
|
class={proseClass()}
|
||||||
|
data-view="markdown"
|
||||||
|
data-part-id={resolved().partId}
|
||||||
|
data-markdown-theme={resolved().themeKey}
|
||||||
|
data-markdown-highlight={resolved().highlightEnabled ? "true" : "false"}
|
||||||
|
innerHTML={html()}
|
||||||
|
/>
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
9
packages/ui/src/components/message-anchors.ts
Normal file
9
packages/ui/src/components/message-anchors.ts
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
export const MESSAGE_ANCHOR_PREFIX = "message-anchor-"
|
||||||
|
|
||||||
|
export function getMessageAnchorId(messageId: string) {
|
||||||
|
return `${MESSAGE_ANCHOR_PREFIX}${messageId}`
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getMessageIdFromAnchorId(anchorId: string) {
|
||||||
|
return anchorId.startsWith(MESSAGE_ANCHOR_PREFIX) ? anchorId.slice(MESSAGE_ANCHOR_PREFIX.length) : anchorId
|
||||||
|
}
|
||||||
@@ -1,64 +0,0 @@
|
|||||||
import { Index, type Accessor } from "solid-js"
|
|
||||||
import VirtualItem from "./virtual-item"
|
|
||||||
import MessageBlock from "./message-block"
|
|
||||||
import type { InstanceMessageStore } from "../stores/message-v2/instance-store"
|
|
||||||
|
|
||||||
export function getMessageAnchorId(messageId: string) {
|
|
||||||
return `message-anchor-${messageId}`
|
|
||||||
}
|
|
||||||
|
|
||||||
const VIRTUAL_ITEM_MARGIN_PX = 800
|
|
||||||
|
|
||||||
interface MessageBlockListProps {
|
|
||||||
instanceId: string
|
|
||||||
sessionId: string
|
|
||||||
store: () => InstanceMessageStore
|
|
||||||
messageIds: () => string[]
|
|
||||||
lastAssistantIndex: () => number
|
|
||||||
showThinking: () => boolean
|
|
||||||
thinkingDefaultExpanded: () => boolean
|
|
||||||
showUsageMetrics: () => boolean
|
|
||||||
scrollContainer: Accessor<HTMLDivElement | undefined>
|
|
||||||
loading?: boolean
|
|
||||||
onRevert?: (messageId: string) => void
|
|
||||||
onFork?: (messageId?: string) => void
|
|
||||||
onContentRendered?: () => void
|
|
||||||
setBottomSentinel: (element: HTMLDivElement | null) => void
|
|
||||||
suspendMeasurements?: () => boolean
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function MessageBlockList(props: MessageBlockListProps) {
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<Index each={props.messageIds()}>
|
|
||||||
{(messageId, index) => (
|
|
||||||
<VirtualItem
|
|
||||||
id={getMessageAnchorId(messageId())}
|
|
||||||
cacheKey={messageId()}
|
|
||||||
scrollContainer={props.scrollContainer}
|
|
||||||
threshold={VIRTUAL_ITEM_MARGIN_PX}
|
|
||||||
placeholderClass="message-stream-placeholder"
|
|
||||||
virtualizationEnabled={() => !props.loading}
|
|
||||||
suspendMeasurements={props.suspendMeasurements}
|
|
||||||
>
|
|
||||||
<MessageBlock
|
|
||||||
messageId={messageId()}
|
|
||||||
instanceId={props.instanceId}
|
|
||||||
sessionId={props.sessionId}
|
|
||||||
store={props.store}
|
|
||||||
messageIndex={index}
|
|
||||||
lastAssistantIndex={props.lastAssistantIndex}
|
|
||||||
showThinking={props.showThinking}
|
|
||||||
thinkingDefaultExpanded={props.thinkingDefaultExpanded}
|
|
||||||
showUsageMetrics={props.showUsageMetrics}
|
|
||||||
onRevert={props.onRevert}
|
|
||||||
onFork={props.onFork}
|
|
||||||
onContentRendered={props.onContentRendered}
|
|
||||||
/>
|
|
||||||
</VirtualItem>
|
|
||||||
)}
|
|
||||||
</Index>
|
|
||||||
<div ref={props.setBottomSentinel} aria-hidden="true" style={{ height: "1px" }} />
|
|
||||||
</>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { For, Match, Show, Switch, createEffect, createMemo, createSignal, untrack } from "solid-js"
|
import { For, Match, Show, Switch, createEffect, createMemo, createSignal, onCleanup, untrack } from "solid-js"
|
||||||
import { ChevronsDownUp, ChevronsUpDown, ExternalLink, FoldVertical, Trash2 } from "lucide-solid"
|
import { ChevronsDownUp, ChevronsUpDown, ExternalLink, FoldVertical, ListStart, Trash } from "lucide-solid"
|
||||||
import MessageItem from "./message-item"
|
import MessageItem from "./message-item"
|
||||||
import ToolCall from "./tool-call"
|
import ToolCall from "./tool-call"
|
||||||
import type { InstanceMessageStore } from "../stores/message-v2/instance-store"
|
import type { InstanceMessageStore } from "../stores/message-v2/instance-store"
|
||||||
@@ -12,8 +12,17 @@ import { formatTokenTotal } from "../lib/formatters"
|
|||||||
import { sessions, setActiveParentSession, setActiveSession } from "../stores/sessions"
|
import { sessions, setActiveParentSession, setActiveSession } from "../stores/sessions"
|
||||||
import { setActiveInstanceId } from "../stores/instances"
|
import { setActiveInstanceId } from "../stores/instances"
|
||||||
import { showAlertDialog } from "../stores/alerts"
|
import { showAlertDialog } from "../stores/alerts"
|
||||||
import { deleteMessagePart } from "../stores/session-actions"
|
import { deleteMessage } from "../stores/session-actions"
|
||||||
import { useI18n } from "../lib/i18n"
|
import { useI18n } from "../lib/i18n"
|
||||||
|
import type { DeleteHoverState } from "../types/delete-hover"
|
||||||
|
|
||||||
|
function DeleteUpToIcon() {
|
||||||
|
return (
|
||||||
|
<span class="relative inline-block w-3.5 h-3.5" aria-hidden="true">
|
||||||
|
<ListStart class="absolute inset-0 w-3.5 h-3.5" aria-hidden="true" />
|
||||||
|
</span>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
const TOOL_ICON = "🔧"
|
const TOOL_ICON = "🔧"
|
||||||
const USER_BORDER_COLOR = "var(--message-user-border)"
|
const USER_BORDER_COLOR = "var(--message-user-border)"
|
||||||
@@ -23,10 +32,10 @@ const TOOL_BORDER_COLOR = "var(--message-tool-border)"
|
|||||||
type ToolCallPart = Extract<ClientPart, { type: "tool" }>
|
type ToolCallPart = Extract<ClientPart, { type: "tool" }>
|
||||||
|
|
||||||
|
|
||||||
type ToolState = import("@opencode-ai/sdk").ToolState
|
type ToolState = import("@opencode-ai/sdk/v2").ToolState
|
||||||
type ToolStateRunning = import("@opencode-ai/sdk").ToolStateRunning
|
type ToolStateRunning = import("@opencode-ai/sdk/v2").ToolStateRunning
|
||||||
type ToolStateCompleted = import("@opencode-ai/sdk").ToolStateCompleted
|
type ToolStateCompleted = import("@opencode-ai/sdk/v2").ToolStateCompleted
|
||||||
type ToolStateError = import("@opencode-ai/sdk").ToolStateError
|
type ToolStateError = import("@opencode-ai/sdk/v2").ToolStateError
|
||||||
|
|
||||||
function isToolStateRunning(state: ToolState | undefined): state is ToolStateRunning {
|
function isToolStateRunning(state: ToolState | undefined): state is ToolStateRunning {
|
||||||
return Boolean(state && state.status === "running")
|
return Boolean(state && state.status === "running")
|
||||||
@@ -194,8 +203,13 @@ interface MessageContentItemProps {
|
|||||||
messageIndex: number
|
messageIndex: number
|
||||||
lastAssistantIndex: () => number
|
lastAssistantIndex: () => number
|
||||||
onRevert?: (messageId: string) => void
|
onRevert?: (messageId: string) => void
|
||||||
|
onDeleteMessagesUpTo?: (messageId: string) => void | Promise<void>
|
||||||
onFork?: (messageId?: string) => void
|
onFork?: (messageId?: string) => void
|
||||||
onContentRendered?: () => void
|
onContentRendered?: () => void
|
||||||
|
showDeleteMessage?: boolean
|
||||||
|
onDeleteHoverChange?: (state: DeleteHoverState) => void
|
||||||
|
selectedMessageIds?: () => Set<string>
|
||||||
|
onToggleSelectedMessage?: (messageId: string, selected: boolean) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
function isSupportedPartType(part: unknown): boolean {
|
function isSupportedPartType(part: unknown): boolean {
|
||||||
@@ -282,7 +296,12 @@ function MessageContentItem(props: MessageContentItemProps) {
|
|||||||
sessionId={props.sessionId}
|
sessionId={props.sessionId}
|
||||||
isQueued={isQueued()}
|
isQueued={isQueued()}
|
||||||
showAgentMeta={showAgentMeta()}
|
showAgentMeta={showAgentMeta()}
|
||||||
|
showDeleteMessage={props.showDeleteMessage}
|
||||||
|
onDeleteHoverChange={props.onDeleteHoverChange}
|
||||||
|
selectedMessageIds={props.selectedMessageIds}
|
||||||
|
onToggleSelectedMessage={props.onToggleSelectedMessage}
|
||||||
onRevert={props.onRevert}
|
onRevert={props.onRevert}
|
||||||
|
onDeleteMessagesUpTo={props.onDeleteMessagesUpTo}
|
||||||
onFork={props.onFork}
|
onFork={props.onFork}
|
||||||
onContentRendered={props.onContentRendered}
|
onContentRendered={props.onContentRendered}
|
||||||
/>
|
/>
|
||||||
@@ -298,11 +317,41 @@ interface ToolCallItemProps {
|
|||||||
messageId: string
|
messageId: string
|
||||||
partId: string
|
partId: string
|
||||||
onContentRendered?: () => void
|
onContentRendered?: () => void
|
||||||
|
showDeleteMessage?: boolean
|
||||||
|
deleteHover?: () => DeleteHoverState
|
||||||
|
onDeleteHoverChange?: (state: DeleteHoverState) => void
|
||||||
|
onDeleteMessagesUpTo?: (messageId: string) => void | Promise<void>
|
||||||
|
selectedMessageIds?: () => Set<string>
|
||||||
|
selectedToolPartKeys?: () => Set<string>
|
||||||
|
onToggleSelectedMessage?: (messageId: string, selected: boolean) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
function ToolCallItem(props: ToolCallItemProps) {
|
function ToolCallItem(props: ToolCallItemProps) {
|
||||||
const { t } = useI18n()
|
const { t } = useI18n()
|
||||||
const [deleting, setDeleting] = createSignal(false)
|
const [deletingMessage, setDeletingMessage] = createSignal(false)
|
||||||
|
const [deletingUpTo, setDeletingUpTo] = createSignal(false)
|
||||||
|
|
||||||
|
const isSelectedForDeletion = () => Boolean(props.selectedMessageIds?.().has(props.messageId))
|
||||||
|
|
||||||
|
const isSelectedToolPartForDeletion = () => Boolean(props.selectedToolPartKeys?.().has(`${props.messageId}:${props.partId}`))
|
||||||
|
|
||||||
|
const isDeleteOverlayActive = () => {
|
||||||
|
if (isSelectedForDeletion()) return true
|
||||||
|
if (isSelectedToolPartForDeletion()) return true
|
||||||
|
const hover = props.deleteHover?.() ?? ({ kind: "none" } as DeleteHoverState)
|
||||||
|
if (hover.kind === "message") {
|
||||||
|
return hover.messageId === props.messageId
|
||||||
|
}
|
||||||
|
if (hover.kind === "deleteUpTo") {
|
||||||
|
const ids = props.store().getSessionMessageIds(props.sessionId)
|
||||||
|
const targetIndex = ids.indexOf(hover.messageId)
|
||||||
|
if (targetIndex === -1) return false
|
||||||
|
const currentIndex = ids.indexOf(props.messageId)
|
||||||
|
if (currentIndex === -1) return false
|
||||||
|
return currentIndex >= targetIndex
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
const record = createMemo(() => props.store().getMessage(props.messageId))
|
const record = createMemo(() => props.store().getMessage(props.messageId))
|
||||||
const messageInfo = createMemo(() => props.store().getMessageInfo(props.messageId))
|
const messageInfo = createMemo(() => props.store().getMessageInfo(props.messageId))
|
||||||
@@ -319,14 +368,6 @@ function ToolCallItem(props: ToolCallItemProps) {
|
|||||||
const messageVersion = createMemo(() => record()?.revision ?? 0)
|
const messageVersion = createMemo(() => record()?.revision ?? 0)
|
||||||
const partVersion = createMemo(() => partEntry()?.revision ?? 0)
|
const partVersion = createMemo(() => partEntry()?.revision ?? 0)
|
||||||
|
|
||||||
const deleteDisabled = createMemo(() => {
|
|
||||||
if (deleting()) return true
|
|
||||||
// Avoid deleting while a tool is actively running to prevent confusing UI states.
|
|
||||||
if (isToolStateRunning(toolState())) return true
|
|
||||||
// Avoid deleting permission prompts from here; those are interactive.
|
|
||||||
return Boolean(toolPart()?.pendingPermission)
|
|
||||||
})
|
|
||||||
|
|
||||||
const taskSessionId = createMemo(() => {
|
const taskSessionId = createMemo(() => {
|
||||||
const state = toolState()
|
const state = toolState()
|
||||||
if (!state) return ""
|
if (!state) return ""
|
||||||
@@ -350,38 +391,72 @@ function ToolCallItem(props: ToolCallItemProps) {
|
|||||||
navigateToTaskSession(location)
|
navigateToTaskSession(location)
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleDeleteToolPart = async (event: MouseEvent) => {
|
const handleDeleteMessage = async (event: MouseEvent) => {
|
||||||
event.preventDefault()
|
event.preventDefault()
|
||||||
event.stopPropagation()
|
event.stopPropagation()
|
||||||
|
|
||||||
if (deleteDisabled()) return
|
if (!props.showDeleteMessage) return
|
||||||
|
if (deletingMessage()) return
|
||||||
|
|
||||||
setDeleting(true)
|
setDeletingMessage(true)
|
||||||
try {
|
try {
|
||||||
await deleteMessagePart(props.instanceId, props.sessionId, props.messageId, props.partId)
|
await deleteMessage(props.instanceId, props.sessionId, props.messageId)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
showAlertDialog(t("messageBlock.tool.deletePart.failed.message"), {
|
showAlertDialog(t("messageItem.actions.deleteMessageFailedMessage"), {
|
||||||
title: t("messageBlock.tool.deletePart.failed.title"),
|
title: t("messageItem.actions.deleteMessageFailedTitle"),
|
||||||
detail: error instanceof Error ? error.message : String(error),
|
detail: error instanceof Error ? error.message : String(error),
|
||||||
variant: "error",
|
variant: "error",
|
||||||
})
|
})
|
||||||
} finally {
|
} finally {
|
||||||
setDeleting(false)
|
setDeletingMessage(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleDeleteUpTo = async (event: MouseEvent) => {
|
||||||
|
event.preventDefault()
|
||||||
|
event.stopPropagation()
|
||||||
|
if (!props.showDeleteMessage) return
|
||||||
|
if (!props.onDeleteMessagesUpTo) return
|
||||||
|
if (deletingUpTo()) return
|
||||||
|
|
||||||
|
setDeletingUpTo(true)
|
||||||
|
try {
|
||||||
|
await props.onDeleteMessagesUpTo(props.messageId)
|
||||||
|
} finally {
|
||||||
|
setDeletingUpTo(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Show when={toolPart()}>
|
<Show when={toolPart()}>
|
||||||
{(resolvedToolPart) => (
|
{(resolvedToolPart) => (
|
||||||
<>
|
<div class="delete-hover-scope" data-delete-part-hover={isDeleteOverlayActive() ? "true" : undefined}>
|
||||||
<div class="tool-call-header-label">
|
<div class="tool-call-header-label">
|
||||||
<div class="tool-call-header-meta">
|
<div class="tool-call-header-meta">
|
||||||
|
<Show when={props.showDeleteMessage}>
|
||||||
|
<input
|
||||||
|
class="message-select-checkbox"
|
||||||
|
type="checkbox"
|
||||||
|
checked={isSelectedForDeletion()}
|
||||||
|
onClick={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
}}
|
||||||
|
onChange={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
const next = Boolean((event.currentTarget as HTMLInputElement).checked)
|
||||||
|
props.onToggleSelectedMessage?.(props.messageId, next)
|
||||||
|
}}
|
||||||
|
aria-label={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
title={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
/>
|
||||||
|
</Show>
|
||||||
|
|
||||||
<span class="tool-call-icon">{TOOL_ICON}</span>
|
<span class="tool-call-icon">{TOOL_ICON}</span>
|
||||||
<span>{t("messageBlock.tool.header")}</span>
|
<span>{t("messageBlock.tool.header")}</span>
|
||||||
<span class="tool-name">{toolName() || t("messageBlock.tool.unknown")}</span>
|
<span class="tool-name">{toolName() || t("messageBlock.tool.unknown")}</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex items-center gap-2">
|
<div class="flex items-center gap-0">
|
||||||
<Show when={taskSessionId()}>
|
<Show when={taskSessionId()}>
|
||||||
<button
|
<button
|
||||||
class="tool-call-header-button"
|
class="tool-call-header-button"
|
||||||
@@ -395,16 +470,33 @@ function ToolCallItem(props: ToolCallItemProps) {
|
|||||||
</button>
|
</button>
|
||||||
</Show>
|
</Show>
|
||||||
|
|
||||||
<button
|
<Show when={props.showDeleteMessage}>
|
||||||
class="tool-call-header-button"
|
<button
|
||||||
type="button"
|
class="tool-call-header-button"
|
||||||
disabled={deleteDisabled()}
|
type="button"
|
||||||
onClick={handleDeleteToolPart}
|
disabled={!props.onDeleteMessagesUpTo || deletingUpTo()}
|
||||||
title={deleting() ? t("messageBlock.tool.deletePart.deleting") : t("messageBlock.tool.deletePart.label")}
|
onClick={handleDeleteUpTo}
|
||||||
aria-label={deleting() ? t("messageBlock.tool.deletePart.deleting") : t("messageBlock.tool.deletePart.label")}
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "deleteUpTo", messageId: props.messageId })}
|
||||||
>
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
<Trash2 class="w-3.5 h-3.5" aria-hidden="true" />
|
title={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
</button>
|
aria-label={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
|
>
|
||||||
|
<DeleteUpToIcon />
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
class="tool-call-header-button"
|
||||||
|
type="button"
|
||||||
|
disabled={deletingMessage()}
|
||||||
|
onClick={handleDeleteMessage}
|
||||||
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "message", messageId: props.messageId })}
|
||||||
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
|
title={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
aria-label={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
>
|
||||||
|
<Trash class="w-3.5 h-3.5" aria-hidden="true" />
|
||||||
|
</button>
|
||||||
|
</Show>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -418,7 +510,7 @@ function ToolCallItem(props: ToolCallItemProps) {
|
|||||||
sessionId={props.sessionId}
|
sessionId={props.sessionId}
|
||||||
onContentRendered={props.onContentRendered}
|
onContentRendered={props.onContentRendered}
|
||||||
/>
|
/>
|
||||||
</>
|
</div>
|
||||||
)}
|
)}
|
||||||
</Show>
|
</Show>
|
||||||
)
|
)
|
||||||
@@ -470,7 +562,13 @@ interface MessageBlockProps {
|
|||||||
showThinking: () => boolean
|
showThinking: () => boolean
|
||||||
thinkingDefaultExpanded: () => boolean
|
thinkingDefaultExpanded: () => boolean
|
||||||
showUsageMetrics: () => boolean
|
showUsageMetrics: () => boolean
|
||||||
|
deleteHover?: () => DeleteHoverState
|
||||||
|
onDeleteHoverChange?: (state: DeleteHoverState) => void
|
||||||
|
selectedMessageIds?: () => Set<string>
|
||||||
|
selectedToolPartKeys?: () => Set<string>
|
||||||
|
onToggleSelectedMessage?: (messageId: string, selected: boolean) => void
|
||||||
onRevert?: (messageId: string) => void
|
onRevert?: (messageId: string) => void
|
||||||
|
onDeleteMessagesUpTo?: (messageId: string) => void | Promise<void>
|
||||||
onFork?: (messageId?: string) => void
|
onFork?: (messageId?: string) => void
|
||||||
onContentRendered?: () => void
|
onContentRendered?: () => void
|
||||||
}
|
}
|
||||||
@@ -481,6 +579,30 @@ export default function MessageBlock(props: MessageBlockProps) {
|
|||||||
const messageInfo = createMemo(() => props.store().getMessageInfo(props.messageId))
|
const messageInfo = createMemo(() => props.store().getMessageInfo(props.messageId))
|
||||||
const sessionCache = getSessionRenderCache(props.instanceId, props.sessionId)
|
const sessionCache = getSessionRenderCache(props.instanceId, props.sessionId)
|
||||||
|
|
||||||
|
const isDeleteMessageHovered = () => {
|
||||||
|
const hover = props.deleteHover?.() ?? ({ kind: "none" } as DeleteHoverState)
|
||||||
|
|
||||||
|
const selected = props.selectedMessageIds?.() ?? new Set<string>()
|
||||||
|
if (selected.has(props.messageId)) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hover.kind === "message") {
|
||||||
|
return hover.messageId === props.messageId
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hover.kind === "deleteUpTo") {
|
||||||
|
const ids = props.store().getSessionMessageIds(props.sessionId)
|
||||||
|
const targetIndex = ids.indexOf(hover.messageId)
|
||||||
|
if (targetIndex === -1) return false
|
||||||
|
const currentIndex = ids.indexOf(props.messageId)
|
||||||
|
if (currentIndex === -1) return false
|
||||||
|
return currentIndex >= targetIndex
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
const block = createMemo<MessageDisplayBlock | null>(() => {
|
const block = createMemo<MessageDisplayBlock | null>(() => {
|
||||||
const current = record()
|
const current = record()
|
||||||
if (!current) return null
|
if (!current) return null
|
||||||
@@ -668,9 +790,13 @@ export default function MessageBlock(props: MessageBlockProps) {
|
|||||||
return (
|
return (
|
||||||
<Show when={block()}>
|
<Show when={block()}>
|
||||||
{(resolvedBlock) => (
|
{(resolvedBlock) => (
|
||||||
<div class="message-stream-block" data-message-id={resolvedBlock().record.id}>
|
<div
|
||||||
|
class="message-stream-block"
|
||||||
|
data-message-id={resolvedBlock().record.id}
|
||||||
|
data-delete-message-hover={isDeleteMessageHovered() ? "true" : undefined}
|
||||||
|
>
|
||||||
<For each={resolvedBlock().items}>
|
<For each={resolvedBlock().items}>
|
||||||
{(item) => (
|
{(item, index) => (
|
||||||
<Switch>
|
<Switch>
|
||||||
<Match when={item.type === "content"}>
|
<Match when={item.type === "content"}>
|
||||||
<MessageContentItem
|
<MessageContentItem
|
||||||
@@ -681,7 +807,12 @@ export default function MessageBlock(props: MessageBlockProps) {
|
|||||||
startPartId={(item as ContentDisplayItem).startPartId}
|
startPartId={(item as ContentDisplayItem).startPartId}
|
||||||
messageIndex={props.messageIndex}
|
messageIndex={props.messageIndex}
|
||||||
lastAssistantIndex={props.lastAssistantIndex}
|
lastAssistantIndex={props.lastAssistantIndex}
|
||||||
|
showDeleteMessage={index() === 0}
|
||||||
|
onDeleteHoverChange={props.onDeleteHoverChange}
|
||||||
onRevert={props.onRevert}
|
onRevert={props.onRevert}
|
||||||
|
onDeleteMessagesUpTo={props.onDeleteMessagesUpTo}
|
||||||
|
selectedMessageIds={props.selectedMessageIds}
|
||||||
|
onToggleSelectedMessage={props.onToggleSelectedMessage}
|
||||||
onFork={props.onFork}
|
onFork={props.onFork}
|
||||||
onContentRendered={props.onContentRendered}
|
onContentRendered={props.onContentRendered}
|
||||||
/>
|
/>
|
||||||
@@ -697,6 +828,13 @@ export default function MessageBlock(props: MessageBlockProps) {
|
|||||||
store={props.store}
|
store={props.store}
|
||||||
messageId={toolItem.messageId}
|
messageId={toolItem.messageId}
|
||||||
partId={toolItem.partId}
|
partId={toolItem.partId}
|
||||||
|
showDeleteMessage={index() === 0}
|
||||||
|
deleteHover={props.deleteHover}
|
||||||
|
onDeleteHoverChange={props.onDeleteHoverChange}
|
||||||
|
onDeleteMessagesUpTo={props.onDeleteMessagesUpTo}
|
||||||
|
selectedMessageIds={props.selectedMessageIds}
|
||||||
|
selectedToolPartKeys={props.selectedToolPartKeys}
|
||||||
|
onToggleSelectedMessage={props.onToggleSelectedMessage}
|
||||||
onContentRendered={props.onContentRendered}
|
onContentRendered={props.onContentRendered}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@@ -709,6 +847,14 @@ export default function MessageBlock(props: MessageBlockProps) {
|
|||||||
part={(item as StepDisplayItem).part}
|
part={(item as StepDisplayItem).part}
|
||||||
messageInfo={(item as StepDisplayItem).messageInfo}
|
messageInfo={(item as StepDisplayItem).messageInfo}
|
||||||
showAgentMeta
|
showAgentMeta
|
||||||
|
showDeleteMessage={index() === 0}
|
||||||
|
instanceId={props.instanceId}
|
||||||
|
sessionId={props.sessionId}
|
||||||
|
messageId={props.messageId}
|
||||||
|
onDeleteHoverChange={props.onDeleteHoverChange}
|
||||||
|
onDeleteMessagesUpTo={props.onDeleteMessagesUpTo}
|
||||||
|
selectedMessageIds={props.selectedMessageIds}
|
||||||
|
onToggleSelectedMessage={props.onToggleSelectedMessage}
|
||||||
/>
|
/>
|
||||||
</Match>
|
</Match>
|
||||||
<Match when={item.type === "step-finish"}>
|
<Match when={item.type === "step-finish"}>
|
||||||
@@ -718,6 +864,14 @@ export default function MessageBlock(props: MessageBlockProps) {
|
|||||||
messageInfo={(item as StepDisplayItem).messageInfo}
|
messageInfo={(item as StepDisplayItem).messageInfo}
|
||||||
showUsage={props.showUsageMetrics()}
|
showUsage={props.showUsageMetrics()}
|
||||||
borderColor={(item as StepDisplayItem).accentColor}
|
borderColor={(item as StepDisplayItem).accentColor}
|
||||||
|
showDeleteMessage={index() === 0}
|
||||||
|
instanceId={props.instanceId}
|
||||||
|
sessionId={props.sessionId}
|
||||||
|
messageId={props.messageId}
|
||||||
|
onDeleteHoverChange={props.onDeleteHoverChange}
|
||||||
|
onDeleteMessagesUpTo={props.onDeleteMessagesUpTo}
|
||||||
|
selectedMessageIds={props.selectedMessageIds}
|
||||||
|
onToggleSelectedMessage={props.onToggleSelectedMessage}
|
||||||
/>
|
/>
|
||||||
</Match>
|
</Match>
|
||||||
<Match when={item.type === "compaction"}>
|
<Match when={item.type === "compaction"}>
|
||||||
@@ -728,7 +882,11 @@ export default function MessageBlock(props: MessageBlockProps) {
|
|||||||
instanceId={props.instanceId}
|
instanceId={props.instanceId}
|
||||||
sessionId={props.sessionId}
|
sessionId={props.sessionId}
|
||||||
messageId={(item as CompactionDisplayItem).messageId}
|
messageId={(item as CompactionDisplayItem).messageId}
|
||||||
partId={(item as CompactionDisplayItem).partId}
|
showDeleteMessage={index() === 0}
|
||||||
|
onDeleteHoverChange={props.onDeleteHoverChange}
|
||||||
|
onDeleteMessagesUpTo={props.onDeleteMessagesUpTo}
|
||||||
|
selectedMessageIds={props.selectedMessageIds}
|
||||||
|
onToggleSelectedMessage={props.onToggleSelectedMessage}
|
||||||
/>
|
/>
|
||||||
</Match>
|
</Match>
|
||||||
<Match when={item.type === "reasoning"}>
|
<Match when={item.type === "reasoning"}>
|
||||||
@@ -738,9 +896,13 @@ export default function MessageBlock(props: MessageBlockProps) {
|
|||||||
instanceId={props.instanceId}
|
instanceId={props.instanceId}
|
||||||
sessionId={props.sessionId}
|
sessionId={props.sessionId}
|
||||||
messageId={(item as ReasoningDisplayItem).messageId}
|
messageId={(item as ReasoningDisplayItem).messageId}
|
||||||
partId={(item as ReasoningDisplayItem).partId}
|
|
||||||
showAgentMeta={(item as ReasoningDisplayItem).showAgentMeta}
|
showAgentMeta={(item as ReasoningDisplayItem).showAgentMeta}
|
||||||
defaultExpanded={(item as ReasoningDisplayItem).defaultExpanded}
|
defaultExpanded={(item as ReasoningDisplayItem).defaultExpanded}
|
||||||
|
showDeleteMessage={index() === 0}
|
||||||
|
onDeleteHoverChange={props.onDeleteHoverChange}
|
||||||
|
onDeleteMessagesUpTo={props.onDeleteMessagesUpTo}
|
||||||
|
selectedMessageIds={props.selectedMessageIds}
|
||||||
|
onToggleSelectedMessage={props.onToggleSelectedMessage}
|
||||||
/>
|
/>
|
||||||
</Match>
|
</Match>
|
||||||
</Switch>
|
</Switch>
|
||||||
@@ -759,6 +921,14 @@ interface StepCardProps {
|
|||||||
showAgentMeta?: boolean
|
showAgentMeta?: boolean
|
||||||
showUsage?: boolean
|
showUsage?: boolean
|
||||||
borderColor?: string
|
borderColor?: string
|
||||||
|
showDeleteMessage?: boolean
|
||||||
|
instanceId?: string
|
||||||
|
sessionId?: string
|
||||||
|
messageId?: string
|
||||||
|
onDeleteHoverChange?: (state: DeleteHoverState) => void
|
||||||
|
onDeleteMessagesUpTo?: (messageId: string) => void | Promise<void>
|
||||||
|
selectedMessageIds?: () => Set<string>
|
||||||
|
onToggleSelectedMessage?: (messageId: string, selected: boolean) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
interface CompactionCardProps {
|
interface CompactionCardProps {
|
||||||
@@ -768,12 +938,18 @@ interface CompactionCardProps {
|
|||||||
instanceId: string
|
instanceId: string
|
||||||
sessionId: string
|
sessionId: string
|
||||||
messageId: string
|
messageId: string
|
||||||
partId: string
|
showDeleteMessage?: boolean
|
||||||
|
onDeleteHoverChange?: (state: DeleteHoverState) => void
|
||||||
|
onDeleteMessagesUpTo?: (messageId: string) => void | Promise<void>
|
||||||
|
selectedMessageIds?: () => Set<string>
|
||||||
|
onToggleSelectedMessage?: (messageId: string, selected: boolean) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
function CompactionCard(props: CompactionCardProps) {
|
function CompactionCard(props: CompactionCardProps) {
|
||||||
const { t } = useI18n()
|
const { t } = useI18n()
|
||||||
const [deleting, setDeleting] = createSignal(false)
|
const [deletingMessage, setDeletingMessage] = createSignal(false)
|
||||||
|
const [deletingUpTo, setDeletingUpTo] = createSignal(false)
|
||||||
|
const isSelectedForDeletion = () => Boolean(props.selectedMessageIds?.().has(props.messageId))
|
||||||
const isAuto = () => Boolean((props.part as any)?.auto)
|
const isAuto = () => Boolean((props.part as any)?.auto)
|
||||||
const label = () => (isAuto() ? t("messageBlock.compaction.autoLabel") : t("messageBlock.compaction.manualLabel"))
|
const label = () => (isAuto() ? t("messageBlock.compaction.autoLabel") : t("messageBlock.compaction.manualLabel"))
|
||||||
const borderColor = () => props.borderColor ?? (isAuto() ? "var(--session-status-compacting-fg)" : USER_BORDER_COLOR)
|
const borderColor = () => props.borderColor ?? (isAuto() ? "var(--session-status-compacting-fg)" : USER_BORDER_COLOR)
|
||||||
@@ -781,44 +957,98 @@ function CompactionCard(props: CompactionCardProps) {
|
|||||||
const containerClass = () =>
|
const containerClass = () =>
|
||||||
`message-compaction-card ${isAuto() ? "message-compaction-card--auto" : "message-compaction-card--manual"}`
|
`message-compaction-card ${isAuto() ? "message-compaction-card--auto" : "message-compaction-card--manual"}`
|
||||||
|
|
||||||
const canDelete = () => Boolean(props.partId) && !deleting()
|
const canDeleteMessage = () => Boolean(props.showDeleteMessage) && !deletingMessage()
|
||||||
|
|
||||||
const handleDelete = async (event: MouseEvent) => {
|
const handleDeleteMessage = async (event: MouseEvent) => {
|
||||||
event.preventDefault()
|
event.preventDefault()
|
||||||
event.stopPropagation()
|
event.stopPropagation()
|
||||||
if (!canDelete()) return
|
if (!props.showDeleteMessage) return
|
||||||
setDeleting(true)
|
if (!canDeleteMessage()) return
|
||||||
|
setDeletingMessage(true)
|
||||||
try {
|
try {
|
||||||
await deleteMessagePart(props.instanceId, props.sessionId, props.messageId, props.partId)
|
await deleteMessage(props.instanceId, props.sessionId, props.messageId)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
showAlertDialog(t("messagePart.actions.deleteFailedMessage"), {
|
showAlertDialog(t("messageItem.actions.deleteMessageFailedMessage"), {
|
||||||
title: t("messagePart.actions.deleteFailedTitle"),
|
title: t("messageItem.actions.deleteMessageFailedTitle"),
|
||||||
detail: error instanceof Error ? error.message : String(error),
|
detail: error instanceof Error ? error.message : String(error),
|
||||||
variant: "error",
|
variant: "error",
|
||||||
})
|
})
|
||||||
} finally {
|
} finally {
|
||||||
setDeleting(false)
|
setDeletingMessage(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleDeleteUpTo = async (event: MouseEvent) => {
|
||||||
|
event.preventDefault()
|
||||||
|
event.stopPropagation()
|
||||||
|
if (!props.showDeleteMessage) return
|
||||||
|
if (!props.onDeleteMessagesUpTo) return
|
||||||
|
if (deletingUpTo()) return
|
||||||
|
|
||||||
|
setDeletingUpTo(true)
|
||||||
|
try {
|
||||||
|
await props.onDeleteMessagesUpTo(props.messageId)
|
||||||
|
} finally {
|
||||||
|
setDeletingUpTo(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
class={`${containerClass()} relative`}
|
class={`delete-hover-scope ${containerClass()} relative`}
|
||||||
style={{ "border-left": `4px solid ${borderColor()}` }}
|
style={{ "border-left": `4px solid ${borderColor()}` }}
|
||||||
role="status"
|
role="status"
|
||||||
aria-label={t("messageBlock.compaction.ariaLabel")}
|
aria-label={t("messageBlock.compaction.ariaLabel")}
|
||||||
>
|
>
|
||||||
<button
|
<div class="absolute right-2 top-1/2 -translate-y-1/2 flex items-center gap-1">
|
||||||
type="button"
|
<Show when={props.showDeleteMessage}>
|
||||||
class="tool-call-header-button absolute right-2 top-1/2 -translate-y-1/2"
|
<button
|
||||||
disabled={!canDelete()}
|
type="button"
|
||||||
onClick={handleDelete}
|
class="tool-call-header-button"
|
||||||
title={t("messagePart.actions.deleteTitle")}
|
disabled={!props.onDeleteMessagesUpTo || deletingUpTo()}
|
||||||
>
|
onClick={handleDeleteUpTo}
|
||||||
{deleting() ? t("messagePart.actions.deleting") : t("messagePart.actions.delete")}
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "deleteUpTo", messageId: props.messageId })}
|
||||||
</button>
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
|
title={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
|
aria-label={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
|
>
|
||||||
|
<DeleteUpToIcon />
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="tool-call-header-button"
|
||||||
|
disabled={!canDeleteMessage()}
|
||||||
|
onClick={handleDeleteMessage}
|
||||||
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "message", messageId: props.messageId })}
|
||||||
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
|
title={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
aria-label={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
>
|
||||||
|
<Trash class="w-3.5 h-3.5" aria-hidden="true" />
|
||||||
|
</button>
|
||||||
|
</Show>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="message-compaction-row">
|
<div class="message-compaction-row">
|
||||||
|
<Show when={props.showDeleteMessage}>
|
||||||
|
<input
|
||||||
|
class="message-select-checkbox"
|
||||||
|
type="checkbox"
|
||||||
|
checked={isSelectedForDeletion()}
|
||||||
|
onClick={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
}}
|
||||||
|
onChange={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
const next = Boolean((event.currentTarget as HTMLInputElement).checked)
|
||||||
|
props.onToggleSelectedMessage?.(props.messageId, next)
|
||||||
|
}}
|
||||||
|
aria-label={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
title={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
/>
|
||||||
|
</Show>
|
||||||
|
|
||||||
<FoldVertical class="message-compaction-icon w-4 h-4" aria-hidden="true" />
|
<FoldVertical class="message-compaction-icon w-4 h-4" aria-hidden="true" />
|
||||||
<span class="message-compaction-label">{label()}</span>
|
<span class="message-compaction-label">{label()}</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -828,6 +1058,9 @@ function CompactionCard(props: CompactionCardProps) {
|
|||||||
|
|
||||||
function StepCard(props: StepCardProps) {
|
function StepCard(props: StepCardProps) {
|
||||||
const { t } = useI18n()
|
const { t } = useI18n()
|
||||||
|
const [deletingMessage, setDeletingMessage] = createSignal(false)
|
||||||
|
const [deletingUpTo, setDeletingUpTo] = createSignal(false)
|
||||||
|
const isSelectedForDeletion = () => Boolean(props.messageId && props.selectedMessageIds?.().has(props.messageId))
|
||||||
const timestamp = () => {
|
const timestamp = () => {
|
||||||
const value = props.messageInfo?.time?.created ?? (props.part as any)?.time?.start ?? Date.now()
|
const value = props.messageInfo?.time?.created ?? (props.part as any)?.time?.start ?? Date.now()
|
||||||
const date = new Date(value)
|
const date = new Date(value)
|
||||||
@@ -872,6 +1105,42 @@ function StepCard(props: StepCardProps) {
|
|||||||
|
|
||||||
const finishStyle = () => (props.borderColor ? { "border-left-color": props.borderColor } : undefined)
|
const finishStyle = () => (props.borderColor ? { "border-left-color": props.borderColor } : undefined)
|
||||||
|
|
||||||
|
const canDeleteMessage = () =>
|
||||||
|
Boolean(props.showDeleteMessage && props.instanceId && props.sessionId && props.messageId) && !deletingMessage()
|
||||||
|
|
||||||
|
const handleDeleteMessage = async (event: MouseEvent) => {
|
||||||
|
event.preventDefault()
|
||||||
|
event.stopPropagation()
|
||||||
|
if (!canDeleteMessage()) return
|
||||||
|
setDeletingMessage(true)
|
||||||
|
try {
|
||||||
|
await deleteMessage(props.instanceId!, props.sessionId!, props.messageId!)
|
||||||
|
} catch (error) {
|
||||||
|
showAlertDialog(t("messageItem.actions.deleteMessageFailedMessage"), {
|
||||||
|
title: t("messageItem.actions.deleteMessageFailedTitle"),
|
||||||
|
detail: error instanceof Error ? error.message : String(error),
|
||||||
|
variant: "error",
|
||||||
|
})
|
||||||
|
} finally {
|
||||||
|
setDeletingMessage(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleDeleteUpTo = async (event: MouseEvent) => {
|
||||||
|
event.preventDefault()
|
||||||
|
event.stopPropagation()
|
||||||
|
if (!props.messageId) return
|
||||||
|
if (!props.onDeleteMessagesUpTo) return
|
||||||
|
if (deletingUpTo()) return
|
||||||
|
|
||||||
|
setDeletingUpTo(true)
|
||||||
|
try {
|
||||||
|
await props.onDeleteMessagesUpTo(props.messageId)
|
||||||
|
} finally {
|
||||||
|
setDeletingUpTo(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
const renderUsageChips = (usage: NonNullable<ReturnType<typeof usageStats>>) => {
|
const renderUsageChips = (usage: NonNullable<ReturnType<typeof usageStats>>) => {
|
||||||
const entries = [
|
const entries = [
|
||||||
@@ -902,17 +1171,83 @@ function StepCard(props: StepCardProps) {
|
|||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<div class={`message-step-card message-step-finish message-step-finish-flush`} style={finishStyle()}>
|
<div class={`message-step-card message-step-finish message-step-finish-flush relative`} style={finishStyle()}>
|
||||||
|
<Show when={props.showDeleteMessage && props.messageId}>
|
||||||
|
<input
|
||||||
|
class="message-select-checkbox absolute left-2 top-1/2 -translate-y-1/2"
|
||||||
|
type="checkbox"
|
||||||
|
checked={isSelectedForDeletion()}
|
||||||
|
onClick={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
}}
|
||||||
|
onChange={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
const next = Boolean((event.currentTarget as HTMLInputElement).checked)
|
||||||
|
props.onToggleSelectedMessage?.(props.messageId!, next)
|
||||||
|
}}
|
||||||
|
aria-label={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
title={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
/>
|
||||||
|
</Show>
|
||||||
|
|
||||||
|
<Show when={props.showDeleteMessage}>
|
||||||
|
<div class="absolute right-2 top-1/2 -translate-y-1/2 flex items-center gap-1">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="message-action-button"
|
||||||
|
disabled={!props.onDeleteMessagesUpTo || deletingUpTo()}
|
||||||
|
onClick={handleDeleteUpTo}
|
||||||
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "deleteUpTo", messageId: props.messageId! })}
|
||||||
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
|
title={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
|
aria-label={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
|
>
|
||||||
|
<DeleteUpToIcon />
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="message-action-button"
|
||||||
|
disabled={!canDeleteMessage()}
|
||||||
|
onClick={handleDeleteMessage}
|
||||||
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "message", messageId: props.messageId! })}
|
||||||
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
|
title={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
aria-label={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
>
|
||||||
|
<Trash class="w-3.5 h-3.5" aria-hidden="true" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</Show>
|
||||||
|
|
||||||
{renderUsageChips(usage)}
|
{renderUsageChips(usage)}
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div class={`message-step-card message-step-start`}>
|
<div class={`message-step-card message-step-start relative`}>
|
||||||
<div class="message-step-heading">
|
<div class="message-step-heading">
|
||||||
<div class="message-step-title">
|
<div class="message-step-title">
|
||||||
<div class="message-step-title-left">
|
<div class="message-step-title-left">
|
||||||
|
<Show when={props.showDeleteMessage && props.messageId}>
|
||||||
|
<input
|
||||||
|
class="message-select-checkbox"
|
||||||
|
type="checkbox"
|
||||||
|
checked={isSelectedForDeletion()}
|
||||||
|
onClick={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
}}
|
||||||
|
onChange={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
const next = Boolean((event.currentTarget as HTMLInputElement).checked)
|
||||||
|
props.onToggleSelectedMessage?.(props.messageId!, next)
|
||||||
|
}}
|
||||||
|
aria-label={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
title={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
/>
|
||||||
|
</Show>
|
||||||
|
|
||||||
<Show when={props.showAgentMeta && (agentIdentifier() || modelIdentifier())}>
|
<Show when={props.showAgentMeta && (agentIdentifier() || modelIdentifier())}>
|
||||||
<span class="message-step-meta-inline">
|
<span class="message-step-meta-inline">
|
||||||
<Show when={agentIdentifier()}>{(value) => <span>{t("messageBlock.step.agentLabel", { agent: value() })}</span>}</Show>
|
<Show when={agentIdentifier()}>{(value) => <span>{t("messageBlock.step.agentLabel", { agent: value() })}</span>}</Show>
|
||||||
@@ -939,15 +1274,27 @@ interface ReasoningCardProps {
|
|||||||
instanceId: string
|
instanceId: string
|
||||||
sessionId: string
|
sessionId: string
|
||||||
messageId: string
|
messageId: string
|
||||||
partId: string
|
|
||||||
showAgentMeta?: boolean
|
showAgentMeta?: boolean
|
||||||
defaultExpanded?: boolean
|
defaultExpanded?: boolean
|
||||||
|
showDeleteMessage?: boolean
|
||||||
|
onDeleteHoverChange?: (state: DeleteHoverState) => void
|
||||||
|
onDeleteMessagesUpTo?: (messageId: string) => void | Promise<void>
|
||||||
|
selectedMessageIds?: () => Set<string>
|
||||||
|
onToggleSelectedMessage?: (messageId: string, selected: boolean) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
function ReasoningCard(props: ReasoningCardProps) {
|
function ReasoningCard(props: ReasoningCardProps) {
|
||||||
const { t } = useI18n()
|
const { t } = useI18n()
|
||||||
const [expanded, setExpanded] = createSignal(Boolean(props.defaultExpanded))
|
const [expanded, setExpanded] = createSignal(Boolean(props.defaultExpanded))
|
||||||
const [deleting, setDeleting] = createSignal(false)
|
const [deletingMessage, setDeletingMessage] = createSignal(false)
|
||||||
|
const [deletingUpTo, setDeletingUpTo] = createSignal(false)
|
||||||
|
const isSelectedForDeletion = () => Boolean(props.selectedMessageIds?.().has(props.messageId))
|
||||||
|
|
||||||
|
let headerEl: HTMLDivElement | undefined
|
||||||
|
let actionsEl: HTMLDivElement | undefined
|
||||||
|
let primaryEl: HTMLSpanElement | undefined
|
||||||
|
let metaMeasureEl: HTMLSpanElement | undefined
|
||||||
|
const [showMetaInline, setShowMetaInline] = createSignal(true)
|
||||||
|
|
||||||
createEffect(() => {
|
createEffect(() => {
|
||||||
setExpanded(Boolean(props.defaultExpanded))
|
setExpanded(Boolean(props.defaultExpanded))
|
||||||
@@ -974,6 +1321,35 @@ function ReasoningCard(props: ReasoningCardProps) {
|
|||||||
return modelID
|
return modelID
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const hasMeta = () => Boolean(props.showAgentMeta && (agentIdentifier() || modelIdentifier()))
|
||||||
|
|
||||||
|
const updateMetaLayout = () => {
|
||||||
|
if (!hasMeta()) return
|
||||||
|
if (!headerEl || !actionsEl || !primaryEl || !metaMeasureEl) return
|
||||||
|
|
||||||
|
const headerWidth = headerEl.getBoundingClientRect().width
|
||||||
|
const actionsWidth = actionsEl.getBoundingClientRect().width
|
||||||
|
const primaryWidth = primaryEl.getBoundingClientRect().width
|
||||||
|
const metaWidth = metaMeasureEl.getBoundingClientRect().width
|
||||||
|
|
||||||
|
const availableLeft = Math.max(0, headerWidth - actionsWidth - 12)
|
||||||
|
setShowMetaInline(primaryWidth + metaWidth + 8 <= availableLeft)
|
||||||
|
}
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
if (!hasMeta() || typeof ResizeObserver === "undefined") {
|
||||||
|
setShowMetaInline(true)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
updateMetaLayout()
|
||||||
|
const observer = new ResizeObserver(() => updateMetaLayout())
|
||||||
|
if (headerEl) observer.observe(headerEl)
|
||||||
|
if (actionsEl) observer.observe(actionsEl)
|
||||||
|
if (primaryEl) observer.observe(primaryEl)
|
||||||
|
onCleanup(() => observer.disconnect())
|
||||||
|
})
|
||||||
|
|
||||||
const reasoningText = () => {
|
const reasoningText = () => {
|
||||||
const part = props.part as any
|
const part = props.part as any
|
||||||
if (!part) return ""
|
if (!part) return ""
|
||||||
@@ -1014,30 +1390,45 @@ function ReasoningCard(props: ReasoningCardProps) {
|
|||||||
const viewHideLabel = () =>
|
const viewHideLabel = () =>
|
||||||
expanded() ? t("messageBlock.reasoning.indicator.hide") : t("messageBlock.reasoning.indicator.view")
|
expanded() ? t("messageBlock.reasoning.indicator.hide") : t("messageBlock.reasoning.indicator.view")
|
||||||
|
|
||||||
const hasDeleteTarget = () => Boolean(props.partId)
|
const canDeleteMessage = () => Boolean(props.showDeleteMessage) && !deletingMessage()
|
||||||
const canDelete = () => hasDeleteTarget() && !deleting()
|
|
||||||
|
|
||||||
const handleDelete = async (event: MouseEvent) => {
|
const handleDeleteMessage = async (event: MouseEvent) => {
|
||||||
event.preventDefault()
|
event.preventDefault()
|
||||||
event.stopPropagation()
|
event.stopPropagation()
|
||||||
if (!canDelete()) return
|
if (!props.showDeleteMessage) return
|
||||||
setDeleting(true)
|
if (!canDeleteMessage()) return
|
||||||
|
setDeletingMessage(true)
|
||||||
try {
|
try {
|
||||||
await deleteMessagePart(props.instanceId, props.sessionId, props.messageId, props.partId)
|
await deleteMessage(props.instanceId, props.sessionId, props.messageId)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
showAlertDialog(t("messagePart.actions.deleteFailedMessage"), {
|
showAlertDialog(t("messageItem.actions.deleteMessageFailedMessage"), {
|
||||||
title: t("messagePart.actions.deleteFailedTitle"),
|
title: t("messageItem.actions.deleteMessageFailedTitle"),
|
||||||
detail: error instanceof Error ? error.message : String(error),
|
detail: error instanceof Error ? error.message : String(error),
|
||||||
variant: "error",
|
variant: "error",
|
||||||
})
|
})
|
||||||
} finally {
|
} finally {
|
||||||
setDeleting(false)
|
setDeletingMessage(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleDeleteUpTo = async (event: MouseEvent) => {
|
||||||
|
event.preventDefault()
|
||||||
|
event.stopPropagation()
|
||||||
|
if (!props.showDeleteMessage) return
|
||||||
|
if (!props.onDeleteMessagesUpTo) return
|
||||||
|
if (deletingUpTo()) return
|
||||||
|
|
||||||
|
setDeletingUpTo(true)
|
||||||
|
try {
|
||||||
|
await props.onDeleteMessagesUpTo(props.messageId)
|
||||||
|
} finally {
|
||||||
|
setDeletingUpTo(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div class="message-reasoning-card">
|
<div class="delete-hover-scope message-reasoning-card">
|
||||||
<div class="message-reasoning-header">
|
<div class="message-reasoning-header" ref={(el) => (headerEl = el)}>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="message-reasoning-toggle"
|
class="message-reasoning-toggle"
|
||||||
@@ -1045,9 +1436,30 @@ function ReasoningCard(props: ReasoningCardProps) {
|
|||||||
aria-expanded={expanded()}
|
aria-expanded={expanded()}
|
||||||
aria-label={expanded() ? t("messageBlock.reasoning.collapseAriaLabel") : t("messageBlock.reasoning.expandAriaLabel")}
|
aria-label={expanded() ? t("messageBlock.reasoning.collapseAriaLabel") : t("messageBlock.reasoning.expandAriaLabel")}
|
||||||
>
|
>
|
||||||
<span class="message-reasoning-label flex flex-wrap items-center gap-2">
|
<span class="message-reasoning-label">
|
||||||
<span>{t("messageBlock.reasoning.thinkingLabel")}</span>
|
<span class="message-reasoning-label-primary" ref={(el) => (primaryEl = el)}>
|
||||||
<Show when={props.showAgentMeta && (agentIdentifier() || modelIdentifier())}>
|
<Show when={props.showDeleteMessage}>
|
||||||
|
<input
|
||||||
|
class="message-select-checkbox"
|
||||||
|
type="checkbox"
|
||||||
|
checked={isSelectedForDeletion()}
|
||||||
|
onClick={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
}}
|
||||||
|
onChange={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
const next = Boolean((event.currentTarget as HTMLInputElement).checked)
|
||||||
|
props.onToggleSelectedMessage?.(props.messageId, next)
|
||||||
|
}}
|
||||||
|
aria-label={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
title={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
/>
|
||||||
|
</Show>
|
||||||
|
|
||||||
|
<span>{t("messageBlock.reasoning.thinkingLabel")}</span>
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<Show when={hasMeta() && showMetaInline()}>
|
||||||
<span class="message-step-meta-inline">
|
<span class="message-step-meta-inline">
|
||||||
<Show when={agentIdentifier()}>
|
<Show when={agentIdentifier()}>
|
||||||
{(value) => (
|
{(value) => (
|
||||||
@@ -1061,10 +1473,28 @@ function ReasoningCard(props: ReasoningCardProps) {
|
|||||||
</Show>
|
</Show>
|
||||||
</span>
|
</span>
|
||||||
</Show>
|
</Show>
|
||||||
|
|
||||||
|
<Show when={hasMeta()}>
|
||||||
|
<span
|
||||||
|
ref={(el) => (metaMeasureEl = el)}
|
||||||
|
class="message-step-meta-inline message-step-meta-inline--measure"
|
||||||
|
>
|
||||||
|
<Show when={agentIdentifier()}>
|
||||||
|
{(value) => (
|
||||||
|
<span class="font-medium text-[var(--message-assistant-border)]">{t("messageBlock.step.agentLabel", { agent: value() })}</span>
|
||||||
|
)}
|
||||||
|
</Show>
|
||||||
|
<Show when={modelIdentifier()}>
|
||||||
|
{(value) => (
|
||||||
|
<span class="font-medium text-[var(--message-assistant-border)]">{t("messageBlock.step.modelLabel", { model: value() })}</span>
|
||||||
|
)}
|
||||||
|
</Show>
|
||||||
|
</span>
|
||||||
|
</Show>
|
||||||
</span>
|
</span>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<div class="message-reasoning-actions">
|
<div class="message-reasoning-actions" ref={(el) => (actionsEl = el)}>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="message-action-button"
|
class="message-action-button"
|
||||||
@@ -1081,16 +1511,31 @@ function ReasoningCard(props: ReasoningCardProps) {
|
|||||||
</Show>
|
</Show>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<Show when={hasDeleteTarget()}>
|
<Show when={props.showDeleteMessage}>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="message-action-button"
|
class="message-action-button"
|
||||||
onClick={handleDelete}
|
onClick={handleDeleteUpTo}
|
||||||
disabled={!canDelete()}
|
disabled={!props.onDeleteMessagesUpTo || deletingUpTo()}
|
||||||
aria-label={t("messagePart.actions.deleteTitle")}
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "deleteUpTo", messageId: props.messageId })}
|
||||||
title={t("messagePart.actions.deleteTitle")}
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
|
aria-label={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
|
title={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
>
|
>
|
||||||
<Trash2 class="w-3.5 h-3.5" aria-hidden="true" />
|
<DeleteUpToIcon />
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="message-action-button"
|
||||||
|
onClick={handleDeleteMessage}
|
||||||
|
disabled={!canDeleteMessage()}
|
||||||
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "message", messageId: props.messageId })}
|
||||||
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
|
aria-label={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
title={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
>
|
||||||
|
<Trash class="w-3.5 h-3.5" aria-hidden="true" />
|
||||||
</button>
|
</button>
|
||||||
</Show>
|
</Show>
|
||||||
|
|
||||||
@@ -1098,6 +1543,23 @@ function ReasoningCard(props: ReasoningCardProps) {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<Show when={hasMeta() && !showMetaInline()}>
|
||||||
|
<div class="message-reasoning-meta-row">
|
||||||
|
<span class="message-step-meta-inline">
|
||||||
|
<Show when={agentIdentifier()}>
|
||||||
|
{(value) => (
|
||||||
|
<span class="font-medium text-[var(--message-assistant-border)]">{t("messageBlock.step.agentLabel", { agent: value() })}</span>
|
||||||
|
)}
|
||||||
|
</Show>
|
||||||
|
<Show when={modelIdentifier()}>
|
||||||
|
{(value) => (
|
||||||
|
<span class="font-medium text-[var(--message-assistant-border)]">{t("messageBlock.step.modelLabel", { model: value() })}</span>
|
||||||
|
)}
|
||||||
|
</Show>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</Show>
|
||||||
|
|
||||||
<Show when={expanded()}>
|
<Show when={expanded()}>
|
||||||
<div class="message-reasoning-expanded">
|
<div class="message-reasoning-expanded">
|
||||||
<div class="message-reasoning-body">
|
<div class="message-reasoning-body">
|
||||||
|
|||||||
@@ -1,14 +1,24 @@
|
|||||||
import { For, Show, createSignal } from "solid-js"
|
import { For, Show, createEffect, createSignal, onCleanup } from "solid-js"
|
||||||
import { Copy, ExternalLink, Split, Trash2, Undo } from "lucide-solid"
|
import { Portal } from "solid-js/web"
|
||||||
import type { MessageInfo, ClientPart } from "../types/message"
|
import { Copy, ListStart, Split, Trash, Undo } from "lucide-solid"
|
||||||
|
import type { MessageInfo, ClientPart, SDKAssistantMessageV2 } from "../types/message"
|
||||||
import { partHasRenderableText } from "../types/message"
|
import { partHasRenderableText } from "../types/message"
|
||||||
import type { MessageRecord } from "../stores/message-v2/types"
|
import type { MessageRecord } from "../stores/message-v2/types"
|
||||||
import MessagePart from "./message-part"
|
import MessagePart from "./message-part"
|
||||||
import { copyToClipboard } from "../lib/clipboard"
|
import { copyToClipboard } from "../lib/clipboard"
|
||||||
import { useI18n } from "../lib/i18n"
|
import { useI18n } from "../lib/i18n"
|
||||||
import { showAlertDialog } from "../stores/alerts"
|
import { showAlertDialog } from "../stores/alerts"
|
||||||
import { deleteMessagePart } from "../stores/session-actions"
|
import { deleteMessage } from "../stores/session-actions"
|
||||||
import { isTauriHost } from "../lib/runtime-env"
|
import { isTauriHost } from "../lib/runtime-env"
|
||||||
|
import type { DeleteHoverState } from "../types/delete-hover"
|
||||||
|
|
||||||
|
function DeleteUpToIcon() {
|
||||||
|
return (
|
||||||
|
<span class="relative inline-block w-3.5 h-3.5" aria-hidden="true">
|
||||||
|
<ListStart class="absolute inset-0 w-3.5 h-3.5" aria-hidden="true" />
|
||||||
|
</span>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
interface MessageItemProps {
|
interface MessageItemProps {
|
||||||
record: MessageRecord
|
record: MessageRecord
|
||||||
@@ -18,15 +28,112 @@ interface MessageItemProps {
|
|||||||
isQueued?: boolean
|
isQueued?: boolean
|
||||||
parts: ClientPart[]
|
parts: ClientPart[]
|
||||||
onRevert?: (messageId: string) => void
|
onRevert?: (messageId: string) => void
|
||||||
|
selectedMessageIds?: () => Set<string>
|
||||||
|
onToggleSelectedMessage?: (messageId: string, selected: boolean) => void
|
||||||
|
onDeleteMessagesUpTo?: (messageId: string) => void | Promise<void>
|
||||||
onFork?: (messageId?: string) => void
|
onFork?: (messageId?: string) => void
|
||||||
showAgentMeta?: boolean
|
showAgentMeta?: boolean
|
||||||
onContentRendered?: () => void
|
onContentRendered?: () => void
|
||||||
|
showDeleteMessage?: boolean
|
||||||
|
onDeleteHoverChange?: (state: DeleteHoverState) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function MessageItem(props: MessageItemProps) {
|
export default function MessageItem(props: MessageItemProps) {
|
||||||
const { t } = useI18n()
|
const { t } = useI18n()
|
||||||
const [copied, setCopied] = createSignal(false)
|
const [copied, setCopied] = createSignal(false)
|
||||||
const [deletingParts, setDeletingParts] = createSignal<Set<string>>(new Set())
|
const [deletingMessage, setDeletingMessage] = createSignal(false)
|
||||||
|
const [deletingUpTo, setDeletingUpTo] = createSignal(false)
|
||||||
|
|
||||||
|
type ImagePreviewState = {
|
||||||
|
url: string
|
||||||
|
name: string
|
||||||
|
anchor: HTMLElement
|
||||||
|
}
|
||||||
|
|
||||||
|
const [imagePreview, setImagePreview] = createSignal<ImagePreviewState | null>(null)
|
||||||
|
|
||||||
|
const clamp = (value: number, min: number, max: number) => Math.min(max, Math.max(min, value))
|
||||||
|
|
||||||
|
const getImagePreviewPosition = () => {
|
||||||
|
const state = imagePreview()
|
||||||
|
if (!state) return null
|
||||||
|
|
||||||
|
const rect = state.anchor.getBoundingClientRect()
|
||||||
|
|
||||||
|
// Outer box: 320px image + 8px padding on each side.
|
||||||
|
const padding = 8
|
||||||
|
const maxImage = 320
|
||||||
|
const gap = 8
|
||||||
|
const chrome = padding * 2
|
||||||
|
const outerWidth = maxImage + chrome
|
||||||
|
const outerHeight = maxImage + chrome
|
||||||
|
|
||||||
|
const viewportW = window.innerWidth
|
||||||
|
const viewportH = window.innerHeight
|
||||||
|
|
||||||
|
const left = clamp(rect.left, 8, Math.max(8, viewportW - outerWidth - 8))
|
||||||
|
|
||||||
|
const fitsAbove = rect.top >= outerHeight + gap + 8
|
||||||
|
const preferredTop = fitsAbove ? rect.top - outerHeight - gap : rect.bottom + gap
|
||||||
|
const top = clamp(preferredTop, 8, Math.max(8, viewportH - outerHeight - 8))
|
||||||
|
|
||||||
|
return { left, top }
|
||||||
|
}
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
const active = imagePreview()
|
||||||
|
if (!active) return
|
||||||
|
|
||||||
|
// If the user scrolls (message stream scroll container) or resizes, the anchor moves.
|
||||||
|
// Hide the popover to avoid showing it in the wrong place.
|
||||||
|
const hide = () => setImagePreview(null)
|
||||||
|
window.addEventListener("scroll", hide, true)
|
||||||
|
window.addEventListener("resize", hide)
|
||||||
|
onCleanup(() => {
|
||||||
|
window.removeEventListener("scroll", hide, true)
|
||||||
|
window.removeEventListener("resize", hide)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
const isSelectedForDeletion = () => Boolean(props.selectedMessageIds?.().has(props.record.id))
|
||||||
|
|
||||||
|
let topRowEl: HTMLDivElement | undefined
|
||||||
|
let actionsEl: HTMLDivElement | undefined
|
||||||
|
let speakerPrimaryEl: HTMLDivElement | undefined
|
||||||
|
let metaMeasureEl: HTMLSpanElement | undefined
|
||||||
|
const [showMetaInline, setShowMetaInline] = createSignal(true)
|
||||||
|
|
||||||
|
const metaText = () => agentMeta()
|
||||||
|
|
||||||
|
const updateMetaLayout = () => {
|
||||||
|
const text = metaText()
|
||||||
|
if (!text) return
|
||||||
|
if (!topRowEl || !actionsEl || !speakerPrimaryEl || !metaMeasureEl) return
|
||||||
|
|
||||||
|
const rowWidth = topRowEl.getBoundingClientRect().width
|
||||||
|
const actionsWidth = actionsEl.getBoundingClientRect().width
|
||||||
|
const primaryWidth = speakerPrimaryEl.getBoundingClientRect().width
|
||||||
|
const metaWidth = metaMeasureEl.getBoundingClientRect().width
|
||||||
|
|
||||||
|
// Allow for the flex gap between left and actions.
|
||||||
|
const availableLeft = Math.max(0, rowWidth - actionsWidth - 12)
|
||||||
|
setShowMetaInline(primaryWidth + metaWidth + 8 <= availableLeft)
|
||||||
|
}
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
const text = metaText()
|
||||||
|
if (!text || typeof ResizeObserver === "undefined") {
|
||||||
|
setShowMetaInline(true)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
updateMetaLayout()
|
||||||
|
const observer = new ResizeObserver(() => updateMetaLayout())
|
||||||
|
if (topRowEl) observer.observe(topRowEl)
|
||||||
|
if (actionsEl) observer.observe(actionsEl)
|
||||||
|
if (speakerPrimaryEl) observer.observe(speakerPrimaryEl)
|
||||||
|
onCleanup(() => observer.disconnect())
|
||||||
|
})
|
||||||
|
|
||||||
const isUser = () => props.record.role === "user"
|
const isUser = () => props.record.role === "user"
|
||||||
const createdTimestamp = () => props.messageInfo?.time?.created ?? props.record.createdAt
|
const createdTimestamp = () => props.messageInfo?.time?.created ?? props.record.createdAt
|
||||||
@@ -123,6 +230,11 @@ export default function MessageItem(props: MessageItemProps) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const showImagePreview = (anchor: HTMLElement, url: string, name: string) => {
|
||||||
|
if (!url) return
|
||||||
|
setImagePreview({ anchor, url, name })
|
||||||
|
}
|
||||||
|
|
||||||
const errorMessage = () => {
|
const errorMessage = () => {
|
||||||
const info = props.messageInfo
|
const info = props.messageInfo
|
||||||
if (!info || info.role !== "assistant" || !info.error) return null
|
if (!info || info.role !== "assistant" || !info.error) return null
|
||||||
@@ -190,47 +302,30 @@ export default function MessageItem(props: MessageItemProps) {
|
|||||||
setTimeout(() => setCopied(false), 2000)
|
setTimeout(() => setCopied(false), 2000)
|
||||||
}
|
}
|
||||||
|
|
||||||
const deletableTextPartId = () => {
|
const handleDeleteMessage = async () => {
|
||||||
const part = props.parts.find((candidate) => {
|
if (deletingMessage()) return
|
||||||
if (!candidate || candidate.type !== "text") return false
|
setDeletingMessage(true)
|
||||||
const id = (candidate as any).id
|
|
||||||
if (typeof id !== "string" || id.length === 0) return false
|
|
||||||
return !Boolean((candidate as any).synthetic)
|
|
||||||
})
|
|
||||||
return (part as any)?.id as string | undefined
|
|
||||||
}
|
|
||||||
|
|
||||||
const isDeletingPart = (partId?: string) => {
|
|
||||||
if (!partId) return false
|
|
||||||
return deletingParts().has(partId)
|
|
||||||
}
|
|
||||||
|
|
||||||
const setPartDeleting = (partId: string, value: boolean) => {
|
|
||||||
setDeletingParts((prev) => {
|
|
||||||
const next = new Set(prev)
|
|
||||||
if (value) {
|
|
||||||
next.add(partId)
|
|
||||||
} else {
|
|
||||||
next.delete(partId)
|
|
||||||
}
|
|
||||||
return next
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleDeletePart = async (partId?: string) => {
|
|
||||||
if (!partId) return
|
|
||||||
if (isDeletingPart(partId)) return
|
|
||||||
setPartDeleting(partId, true)
|
|
||||||
try {
|
try {
|
||||||
await deleteMessagePart(props.instanceId, props.sessionId, props.record.id, partId)
|
await deleteMessage(props.instanceId, props.sessionId, props.record.id)
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
showAlertDialog(t("messagePart.actions.deleteFailedMessage"), {
|
showAlertDialog(t("messageItem.actions.deleteMessageFailedMessage"), {
|
||||||
title: t("messagePart.actions.deleteFailedTitle"),
|
title: t("messageItem.actions.deleteMessageFailedTitle"),
|
||||||
detail: error instanceof Error ? error.message : String(error),
|
detail: error instanceof Error ? error.message : String(error),
|
||||||
variant: "error",
|
variant: "error",
|
||||||
})
|
})
|
||||||
} finally {
|
} finally {
|
||||||
setPartDeleting(partId, false)
|
setDeletingMessage(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleDeleteUpTo = async () => {
|
||||||
|
if (!props.onDeleteMessagesUpTo) return
|
||||||
|
if (deletingUpTo()) return
|
||||||
|
setDeletingUpTo(true)
|
||||||
|
try {
|
||||||
|
await props.onDeleteMessagesUpTo(props.record.id)
|
||||||
|
} finally {
|
||||||
|
setDeletingUpTo(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -258,8 +353,16 @@ export default function MessageItem(props: MessageItemProps) {
|
|||||||
if (!info || info.role !== "assistant") return ""
|
if (!info || info.role !== "assistant") return ""
|
||||||
const modelID = info.modelID || ""
|
const modelID = info.modelID || ""
|
||||||
const providerID = info.providerID || ""
|
const providerID = info.providerID || ""
|
||||||
if (modelID && providerID) return `${providerID}/${modelID}`
|
|
||||||
return modelID
|
const base = modelID && providerID ? `${providerID}/${modelID}` : modelID
|
||||||
|
if (!base) return ""
|
||||||
|
|
||||||
|
const variant = (info as SDKAssistantMessageV2).variant
|
||||||
|
if (typeof variant === "string" && variant.trim().length > 0) {
|
||||||
|
return `${base} (${variant.trim()})`
|
||||||
|
}
|
||||||
|
|
||||||
|
return base
|
||||||
}
|
}
|
||||||
|
|
||||||
const agentMeta = () => {
|
const agentMeta = () => {
|
||||||
@@ -278,28 +381,68 @@ export default function MessageItem(props: MessageItemProps) {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div class={containerClass()}>
|
<div
|
||||||
|
class={containerClass()}
|
||||||
|
data-view="message-item"
|
||||||
|
data-instance-id={props.instanceId}
|
||||||
|
data-session-id={props.sessionId}
|
||||||
|
data-message-id={props.record.id}
|
||||||
|
data-message-role={isUser() ? "user" : "assistant"}
|
||||||
|
data-message-status={props.record.status}
|
||||||
|
>
|
||||||
<header class={`message-item-header ${isUser() ? "pb-0.5" : "pb-0"}`}>
|
<header class={`message-item-header ${isUser() ? "pb-0.5" : "pb-0"}`}>
|
||||||
<div class="message-item-header-row message-item-header-row--top">
|
<div class="message-item-header-row message-item-header-row--top" ref={(el) => (topRowEl = el)}>
|
||||||
<div class="message-speaker">
|
<div class="message-header-left">
|
||||||
<span class="message-speaker-label" data-role={isUser() ? "user" : "assistant"}>
|
<div class="message-speaker-primary" ref={(el) => (speakerPrimaryEl = el)}>
|
||||||
{speakerLabel()}
|
<Show when={props.showDeleteMessage}>
|
||||||
</span>
|
<input
|
||||||
|
class="message-select-checkbox"
|
||||||
|
type="checkbox"
|
||||||
|
checked={isSelectedForDeletion()}
|
||||||
|
onClick={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
}}
|
||||||
|
onChange={(event) => {
|
||||||
|
event.stopPropagation()
|
||||||
|
const next = Boolean((event.currentTarget as HTMLInputElement).checked)
|
||||||
|
props.onToggleSelectedMessage?.(props.record.id, next)
|
||||||
|
}}
|
||||||
|
aria-label={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
title={t("messageItem.selection.checkboxAriaLabel")}
|
||||||
|
/>
|
||||||
|
</Show>
|
||||||
|
|
||||||
|
<span class="message-speaker-label" data-role={isUser() ? "user" : "assistant"}>
|
||||||
|
{speakerLabel()}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Show when={metaText() && showMetaInline()}>
|
||||||
|
<span class="message-agent-meta-inline">{metaText()}</span>
|
||||||
|
</Show>
|
||||||
|
|
||||||
|
<Show when={metaText()}>
|
||||||
|
<span
|
||||||
|
ref={(el) => (metaMeasureEl = el)}
|
||||||
|
class="message-agent-meta-inline message-agent-meta-inline--measure"
|
||||||
|
>
|
||||||
|
{metaText()}
|
||||||
|
</span>
|
||||||
|
</Show>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="message-item-actions">
|
<div class="message-item-actions" ref={(el) => (actionsEl = el)}>
|
||||||
<Show when={isUser()}>
|
<Show when={isUser()}>
|
||||||
<div class="message-action-group">
|
<div class="message-action-group">
|
||||||
<Show when={props.onRevert}>
|
<button
|
||||||
<button
|
class="message-action-button"
|
||||||
class="message-action-button"
|
onClick={handleCopy}
|
||||||
onClick={handleRevert}
|
title={copyLabel()}
|
||||||
title={t("messageItem.actions.revert")}
|
aria-label={copyLabel()}
|
||||||
aria-label={t("messageItem.actions.revert")}
|
>
|
||||||
>
|
<Copy class="w-3.5 h-3.5" aria-hidden="true" />
|
||||||
<Undo class="w-3.5 h-3.5" aria-hidden="true" />
|
</button>
|
||||||
</button>
|
|
||||||
</Show>
|
|
||||||
<Show when={props.onFork}>
|
<Show when={props.onFork}>
|
||||||
<button
|
<button
|
||||||
class="message-action-button"
|
class="message-action-button"
|
||||||
@@ -310,14 +453,43 @@ export default function MessageItem(props: MessageItemProps) {
|
|||||||
<Split class="w-3.5 h-3.5" aria-hidden="true" />
|
<Split class="w-3.5 h-3.5" aria-hidden="true" />
|
||||||
</button>
|
</button>
|
||||||
</Show>
|
</Show>
|
||||||
<button
|
|
||||||
class="message-action-button"
|
<Show when={props.onRevert}>
|
||||||
onClick={handleCopy}
|
<button
|
||||||
title={copyLabel()}
|
class="message-action-button"
|
||||||
aria-label={copyLabel()}
|
onClick={handleRevert}
|
||||||
>
|
title={t("messageItem.actions.revertTitle")}
|
||||||
<Copy class="w-3.5 h-3.5" aria-hidden="true" />
|
aria-label={t("messageItem.actions.revertTitle")}
|
||||||
</button>
|
>
|
||||||
|
<Undo class="w-3.5 h-3.5" aria-hidden="true" />
|
||||||
|
</button>
|
||||||
|
</Show>
|
||||||
|
|
||||||
|
<Show when={props.showDeleteMessage}>
|
||||||
|
<button
|
||||||
|
class="message-action-button"
|
||||||
|
onClick={() => void handleDeleteUpTo()}
|
||||||
|
disabled={!props.onDeleteMessagesUpTo || deletingUpTo()}
|
||||||
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "deleteUpTo", messageId: props.record.id })}
|
||||||
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
|
title={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
|
aria-label={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
|
>
|
||||||
|
<DeleteUpToIcon />
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
class="message-action-button"
|
||||||
|
onClick={handleDeleteMessage}
|
||||||
|
disabled={deletingMessage()}
|
||||||
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "message", messageId: props.record.id })}
|
||||||
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
|
title={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
aria-label={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
>
|
||||||
|
<Trash class="w-3.5 h-3.5" aria-hidden="true" />
|
||||||
|
</button>
|
||||||
|
</Show>
|
||||||
</div>
|
</div>
|
||||||
</Show>
|
</Show>
|
||||||
<Show when={!isUser()}>
|
<Show when={!isUser()}>
|
||||||
@@ -331,18 +503,30 @@ export default function MessageItem(props: MessageItemProps) {
|
|||||||
<Copy class="w-3.5 h-3.5" aria-hidden="true" />
|
<Copy class="w-3.5 h-3.5" aria-hidden="true" />
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<Show when={deletableTextPartId()}>
|
<Show when={props.showDeleteMessage}>
|
||||||
{(partId) => (
|
<button
|
||||||
<button
|
class="message-action-button"
|
||||||
class="message-action-button"
|
onClick={() => void handleDeleteUpTo()}
|
||||||
onClick={() => void handleDeletePart(partId())}
|
disabled={!props.onDeleteMessagesUpTo || deletingUpTo()}
|
||||||
disabled={isDeletingPart(partId())}
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "deleteUpTo", messageId: props.record.id })}
|
||||||
title={isDeletingPart(partId()) ? t("messagePart.actions.deleting") : t("messagePart.actions.delete")}
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
aria-label={isDeletingPart(partId()) ? t("messagePart.actions.deleting") : t("messagePart.actions.delete")}
|
title={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
>
|
aria-label={t("messageItem.actions.deleteMessagesUpTo")}
|
||||||
<Trash2 class="w-3.5 h-3.5" aria-hidden="true" />
|
>
|
||||||
</button>
|
<DeleteUpToIcon />
|
||||||
)}
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
class="message-action-button"
|
||||||
|
onClick={handleDeleteMessage}
|
||||||
|
disabled={deletingMessage()}
|
||||||
|
onMouseEnter={() => props.onDeleteHoverChange?.({ kind: "message", messageId: props.record.id })}
|
||||||
|
onMouseLeave={() => props.onDeleteHoverChange?.({ kind: "none" })}
|
||||||
|
title={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
aria-label={deletingMessage() ? t("messageItem.actions.deletingMessage") : t("messageItem.actions.deleteMessage")}
|
||||||
|
>
|
||||||
|
<Trash class="w-3.5 h-3.5" aria-hidden="true" />
|
||||||
|
</button>
|
||||||
</Show>
|
</Show>
|
||||||
</div>
|
</div>
|
||||||
</Show>
|
</Show>
|
||||||
@@ -350,12 +534,10 @@ export default function MessageItem(props: MessageItemProps) {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Show when={agentMeta()}>
|
<Show when={metaText() && !showMetaInline()}>
|
||||||
{(meta) => (
|
<div class="message-item-header-row message-item-header-row--meta">
|
||||||
<div class="message-item-header-row message-item-header-row--bottom">
|
<span class="message-agent-meta-block">{metaText()}</span>
|
||||||
<span class="message-agent-meta">{meta()}</span>
|
</div>
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</Show>
|
</Show>
|
||||||
|
|
||||||
</header>
|
</header>
|
||||||
@@ -378,16 +560,20 @@ export default function MessageItem(props: MessageItemProps) {
|
|||||||
</Show>
|
</Show>
|
||||||
|
|
||||||
<For each={messageParts()}>
|
<For each={messageParts()}>
|
||||||
{(part) => (
|
{(part) => {
|
||||||
<MessagePart
|
return (
|
||||||
part={part}
|
<div class="message-part-shell">
|
||||||
messageType={props.record.role}
|
<MessagePart
|
||||||
instanceId={props.instanceId}
|
part={part}
|
||||||
sessionId={props.sessionId}
|
messageType={props.record.role}
|
||||||
primaryUserTextPartId={primaryUserTextPartId()}
|
instanceId={props.instanceId}
|
||||||
onRendered={props.onContentRendered}
|
sessionId={props.sessionId}
|
||||||
/>
|
primaryUserTextPartId={primaryUserTextPartId()}
|
||||||
)}
|
onRendered={props.onContentRendered}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}}
|
||||||
</For>
|
</For>
|
||||||
|
|
||||||
<Show when={fileAttachments().length > 0}>
|
<Show when={fileAttachments().length > 0}>
|
||||||
@@ -397,7 +583,16 @@ export default function MessageItem(props: MessageItemProps) {
|
|||||||
const name = getAttachmentName(attachment)
|
const name = getAttachmentName(attachment)
|
||||||
const isImage = isImageAttachment(attachment)
|
const isImage = isImageAttachment(attachment)
|
||||||
return (
|
return (
|
||||||
<div class={`attachment-chip ${isImage ? "attachment-chip-image" : ""}`} title={name}>
|
<div
|
||||||
|
class={`attachment-chip ${isImage ? "attachment-chip-image" : ""}`}
|
||||||
|
title={name}
|
||||||
|
onMouseEnter={(e) => {
|
||||||
|
if (!isImage) return
|
||||||
|
const el = e.currentTarget as HTMLElement
|
||||||
|
showImagePreview(el, attachment.url || "", name)
|
||||||
|
}}
|
||||||
|
onMouseLeave={() => setImagePreview(null)}
|
||||||
|
>
|
||||||
<Show when={isImage} fallback={
|
<Show when={isImage} fallback={
|
||||||
<svg class="h-3 w-3" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
<svg class="h-3 w-3" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||||
<path
|
<path
|
||||||
@@ -425,24 +620,6 @@ export default function MessageItem(props: MessageItemProps) {
|
|||||||
</svg>
|
</svg>
|
||||||
</button>
|
</button>
|
||||||
</Show>
|
</Show>
|
||||||
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
onClick={() => void handleDeletePart(attachment.id)}
|
|
||||||
class="attachment-remove"
|
|
||||||
disabled={isDeletingPart(attachment.id)}
|
|
||||||
aria-label={t("messagePart.actions.deleteTitle")}
|
|
||||||
title={t("messagePart.actions.deleteTitle")}
|
|
||||||
>
|
|
||||||
<svg class="h-3 w-3" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
|
||||||
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12" />
|
|
||||||
</svg>
|
|
||||||
</button>
|
|
||||||
<Show when={isImage}>
|
|
||||||
<div class="attachment-chip-preview">
|
|
||||||
<img src={attachment.url} alt={name} />
|
|
||||||
</div>
|
|
||||||
</Show>
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
@@ -450,6 +627,31 @@ export default function MessageItem(props: MessageItemProps) {
|
|||||||
</div>
|
</div>
|
||||||
</Show>
|
</Show>
|
||||||
|
|
||||||
|
<Show when={imagePreview()}>
|
||||||
|
{(stateAccessor) => {
|
||||||
|
const state = stateAccessor()
|
||||||
|
const pos = () => getImagePreviewPosition()
|
||||||
|
return (
|
||||||
|
<Portal>
|
||||||
|
<Show when={pos()}>
|
||||||
|
{(posAccessor) => {
|
||||||
|
const coords = posAccessor()
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
class="attachment-image-popover"
|
||||||
|
style={{ left: `${coords.left}px`, top: `${coords.top}px` }}
|
||||||
|
aria-hidden="true"
|
||||||
|
>
|
||||||
|
<img src={state.url} alt={state.name} />
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
</Show>
|
||||||
|
</Portal>
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
</Show>
|
||||||
|
|
||||||
<Show when={props.record.status === "sending"}>
|
<Show when={props.record.status === "sending"}>
|
||||||
<div class="message-sending">
|
<div class="message-sending">
|
||||||
<span class="generating-spinner">●</span> {t("messageItem.status.sending")}
|
<span class="generating-spinner">●</span> {t("messageItem.status.sending")}
|
||||||
|
|||||||
@@ -131,7 +131,12 @@ export default function MessagePart(props: MessagePartProps) {
|
|||||||
<Switch>
|
<Switch>
|
||||||
<Match when={partType() === "text"}>
|
<Match when={partType() === "text"}>
|
||||||
<Show when={!shouldHideTextPart() && partHasRenderableText(props.part)}>
|
<Show when={!shouldHideTextPart() && partHasRenderableText(props.part)}>
|
||||||
<div class={canRenderMarkdown() ? markdownContainerClass() : textContainerClass()} data-role={textContainerRole()}>
|
<div
|
||||||
|
class={canRenderMarkdown() ? markdownContainerClass() : textContainerClass()}
|
||||||
|
data-role={textContainerRole()}
|
||||||
|
data-part-type="text"
|
||||||
|
data-part-id={typeof (props.part as any)?.id === "string" ? (props.part as any).id : undefined}
|
||||||
|
>
|
||||||
<Show when={canRenderMarkdown()} fallback={<span class="text-primary">{plainTextContent()}</span>}>
|
<Show when={canRenderMarkdown()} fallback={<span class="text-primary">{plainTextContent()}</span>}>
|
||||||
<Markdown
|
<Markdown
|
||||||
part={createTextPartForMarkdown()}
|
part={createTextPartForMarkdown()}
|
||||||
|
|||||||
@@ -1,12 +1,18 @@
|
|||||||
import type { Component } from "solid-js"
|
import type { Component } from "solid-js"
|
||||||
import MessageBlock from "./message-block"
|
import MessageBlock from "./message-block"
|
||||||
import type { InstanceMessageStore } from "../stores/message-v2/instance-store"
|
import type { InstanceMessageStore } from "../stores/message-v2/instance-store"
|
||||||
|
import type { DeleteHoverState } from "../types/delete-hover"
|
||||||
|
|
||||||
interface MessagePreviewProps {
|
interface MessagePreviewProps {
|
||||||
instanceId: string
|
instanceId: string
|
||||||
sessionId: string
|
sessionId: string
|
||||||
messageId: string
|
messageId: string
|
||||||
store: () => InstanceMessageStore
|
store: () => InstanceMessageStore
|
||||||
|
deleteHover?: () => DeleteHoverState
|
||||||
|
onDeleteHoverChange?: (state: DeleteHoverState) => void
|
||||||
|
onDeleteMessagesUpTo?: (messageId: string) => void | Promise<void>
|
||||||
|
selectedMessageIds?: () => Set<string>
|
||||||
|
onToggleSelectedMessage?: (messageId: string, selected: boolean) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
const MessagePreview: Component<MessagePreviewProps> = (props) => {
|
const MessagePreview: Component<MessagePreviewProps> = (props) => {
|
||||||
@@ -24,6 +30,11 @@ const MessagePreview: Component<MessagePreviewProps> = (props) => {
|
|||||||
showThinking={() => false}
|
showThinking={() => false}
|
||||||
thinkingDefaultExpanded={() => false}
|
thinkingDefaultExpanded={() => false}
|
||||||
showUsageMetrics={() => false}
|
showUsageMetrics={() => false}
|
||||||
|
deleteHover={props.deleteHover}
|
||||||
|
onDeleteHoverChange={props.onDeleteHoverChange}
|
||||||
|
onDeleteMessagesUpTo={props.onDeleteMessagesUpTo}
|
||||||
|
selectedMessageIds={props.selectedMessageIds}
|
||||||
|
onToggleSelectedMessage={props.onToggleSelectedMessage}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,12 +1,14 @@
|
|||||||
import { For, Show, createEffect, createMemo, createSignal, onCleanup, type Component } from "solid-js"
|
import { For, Show, createEffect, createMemo, createSignal, onCleanup, on, untrack, type Component, type Accessor } from "solid-js"
|
||||||
import MessagePreview from "./message-preview"
|
import MessagePreview from "./message-preview"
|
||||||
import { messageStoreBus } from "../stores/message-v2/bus"
|
import { messageStoreBus } from "../stores/message-v2/bus"
|
||||||
import type { ClientPart } from "../types/message"
|
import type { ClientPart } from "../types/message"
|
||||||
import type { MessageRecord } from "../stores/message-v2/types"
|
import type { MessageRecord } from "../stores/message-v2/types"
|
||||||
import { buildRecordDisplayData } from "../stores/message-v2/record-display-cache"
|
import { buildRecordDisplayData } from "../stores/message-v2/record-display-cache"
|
||||||
|
import { getPartCharCount } from "../lib/token-utils"
|
||||||
import { getToolIcon } from "./tool-call/utils"
|
import { getToolIcon } from "./tool-call/utils"
|
||||||
import { User as UserIcon, Bot as BotIcon, FoldVertical, ShieldAlert } from "lucide-solid"
|
import { User as UserIcon, Bot as BotIcon, FoldVertical, ShieldAlert } from "lucide-solid"
|
||||||
import { useI18n } from "../lib/i18n"
|
import { useI18n } from "../lib/i18n"
|
||||||
|
import type { DeleteHoverState } from "../types/delete-hover"
|
||||||
|
|
||||||
export type TimelineSegmentType = "user" | "assistant" | "tool" | "compaction"
|
export type TimelineSegmentType = "user" | "assistant" | "tool" | "compaction"
|
||||||
|
|
||||||
@@ -19,18 +21,38 @@ export interface TimelineSegment {
|
|||||||
shortLabel?: string
|
shortLabel?: string
|
||||||
variant?: "auto" | "manual"
|
variant?: "auto" | "manual"
|
||||||
toolPartIds?: string[]
|
toolPartIds?: string[]
|
||||||
|
partIds?: string[]
|
||||||
|
partId?: string
|
||||||
|
totalChars: number
|
||||||
}
|
}
|
||||||
|
|
||||||
interface MessageTimelineProps {
|
interface MessageTimelineProps {
|
||||||
segments: TimelineSegment[]
|
segments: TimelineSegment[]
|
||||||
onSegmentClick?: (segment: TimelineSegment) => void
|
onSegmentClick?: (segment: TimelineSegment) => void
|
||||||
activeMessageId?: string | null
|
onToggleSelection?: (id: string) => void
|
||||||
|
onLongPressSelection?: (segment: TimelineSegment) => void
|
||||||
|
onSelectRange?: (id: string) => void
|
||||||
|
onClearSelection?: () => void
|
||||||
|
selectedIds?: Accessor<Set<string>>
|
||||||
|
expandedMessageIds?: Accessor<Set<string>>
|
||||||
|
// Optional: restrict histogram/xray overlay to only show for these message ids.
|
||||||
|
// Used to hide ribs for messages before the last compaction.
|
||||||
|
deletableMessageIds?: Accessor<Set<string>>
|
||||||
|
activeSegmentId?: string | null
|
||||||
instanceId: string
|
instanceId: string
|
||||||
sessionId: string
|
sessionId: string
|
||||||
showToolSegments?: boolean
|
showToolSegments?: boolean
|
||||||
|
deleteHover?: () => DeleteHoverState
|
||||||
|
onDeleteHoverChange?: (state: DeleteHoverState) => void
|
||||||
|
onDeleteMessagesUpTo?: (messageId: string) => void | Promise<void>
|
||||||
|
selectedMessageIds?: () => Set<string>
|
||||||
|
onToggleSelectedMessage?: (messageId: string, selected: boolean) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
const MAX_TOOLTIP_LENGTH = 220
|
const MAX_TOOLTIP_LENGTH = 220
|
||||||
|
const LONG_PRESS_MS = 500
|
||||||
|
const JITTER_THRESHOLD = 10
|
||||||
|
const ABSOLUTE_TOKEN_CAP = 10000
|
||||||
|
|
||||||
type ToolCallPart = Extract<ClientPart, { type: "tool" }>
|
type ToolCallPart = Extract<ClientPart, { type: "tool" }>
|
||||||
|
|
||||||
@@ -38,10 +60,8 @@ interface PendingSegment {
|
|||||||
type: TimelineSegmentType
|
type: TimelineSegmentType
|
||||||
texts: string[]
|
texts: string[]
|
||||||
reasoningTexts: string[]
|
reasoningTexts: string[]
|
||||||
toolTitles: string[]
|
partIds: string[]
|
||||||
toolTypeLabels: string[]
|
totalChars: number
|
||||||
toolIcons: string[]
|
|
||||||
toolPartIds: string[]
|
|
||||||
hasPrimaryText: boolean
|
hasPrimaryText: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -171,18 +191,13 @@ export function buildTimelineSegments(
|
|||||||
pending = null
|
pending = null
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
const isToolSegment = pending.type === "tool"
|
const label = segmentLabel(pending.type)
|
||||||
const label = isToolSegment
|
const shortLabel = undefined
|
||||||
? pending.toolTypeLabels[0] || segmentLabel("tool")
|
const tooltip = formatTextsTooltip(
|
||||||
: segmentLabel(pending.type)
|
[...pending.texts, ...pending.reasoningTexts],
|
||||||
const shortLabel = isToolSegment ? pending.toolIcons[0] || getToolIcon("tool") : undefined
|
pending.type === "user" ? t("messageTimeline.tooltip.userFallback") : t("messageTimeline.tooltip.assistantFallback"),
|
||||||
const tooltip = isToolSegment
|
)
|
||||||
? formatToolTooltip(pending.toolTitles, t)
|
|
||||||
: formatTextsTooltip(
|
|
||||||
[...pending.texts, ...pending.reasoningTexts],
|
|
||||||
pending.type === "user" ? t("messageTimeline.tooltip.userFallback") : t("messageTimeline.tooltip.assistantFallback"),
|
|
||||||
)
|
|
||||||
|
|
||||||
result.push({
|
result.push({
|
||||||
id: `${record.id}:${segmentIndex}`,
|
id: `${record.id}:${segmentIndex}`,
|
||||||
messageId: record.id,
|
messageId: record.id,
|
||||||
@@ -190,16 +205,24 @@ export function buildTimelineSegments(
|
|||||||
label,
|
label,
|
||||||
tooltip,
|
tooltip,
|
||||||
shortLabel,
|
shortLabel,
|
||||||
toolPartIds: isToolSegment ? pending.toolPartIds : undefined,
|
partIds: pending.partIds,
|
||||||
|
totalChars: pending.totalChars,
|
||||||
})
|
})
|
||||||
segmentIndex += 1
|
segmentIndex += 1
|
||||||
pending = null
|
pending = null
|
||||||
}
|
}
|
||||||
|
|
||||||
const ensureSegment = (type: TimelineSegmentType): PendingSegment => {
|
const ensureSegment = (type: TimelineSegmentType): PendingSegment => {
|
||||||
if (!pending || pending.type !== type) {
|
if (!pending || pending.type !== type) {
|
||||||
flushPending()
|
flushPending()
|
||||||
pending = { type, texts: [], reasoningTexts: [], toolTitles: [], toolTypeLabels: [], toolIcons: [], toolPartIds: [], hasPrimaryText: type !== "assistant" }
|
pending = {
|
||||||
|
type,
|
||||||
|
texts: [],
|
||||||
|
reasoningTexts: [],
|
||||||
|
partIds: [],
|
||||||
|
totalChars: 0,
|
||||||
|
hasPrimaryText: type !== "assistant",
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return pending!
|
return pending!
|
||||||
}
|
}
|
||||||
@@ -211,14 +234,21 @@ export function buildTimelineSegments(
|
|||||||
if (!part || typeof part !== "object") continue
|
if (!part || typeof part !== "object") continue
|
||||||
|
|
||||||
if (part.type === "tool") {
|
if (part.type === "tool") {
|
||||||
const target = ensureSegment("tool")
|
flushPending()
|
||||||
const toolPart = part as ToolCallPart
|
const toolPart = part as ToolCallPart
|
||||||
target.toolTitles.push(getToolTitle(toolPart, t))
|
const partId = typeof toolPart.id === "string" ? toolPart.id : ""
|
||||||
target.toolTypeLabels.push(getToolTypeLabel(toolPart, t))
|
const title = getToolTitle(toolPart, t)
|
||||||
target.toolIcons.push(getToolIcon(typeof toolPart.tool === "string" ? toolPart.tool : "tool"))
|
result.push({
|
||||||
if (typeof toolPart.id === "string" && toolPart.id.length > 0) {
|
id: `${record.id}:${segmentIndex}`,
|
||||||
target.toolPartIds.push(toolPart.id)
|
messageId: record.id,
|
||||||
}
|
type: "tool",
|
||||||
|
label: getToolTypeLabel(toolPart, t) || segmentLabel("tool"),
|
||||||
|
tooltip: formatToolTooltip([title], t),
|
||||||
|
shortLabel: getToolIcon(typeof toolPart.tool === "string" ? toolPart.tool : "tool"),
|
||||||
|
toolPartIds: partId ? [partId] : undefined,
|
||||||
|
totalChars: getPartCharCount(part),
|
||||||
|
})
|
||||||
|
segmentIndex += 1
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -228,13 +258,18 @@ export function buildTimelineSegments(
|
|||||||
const target = ensureSegment(defaultContentType)
|
const target = ensureSegment(defaultContentType)
|
||||||
if (target) {
|
if (target) {
|
||||||
target.reasoningTexts.push(text)
|
target.reasoningTexts.push(text)
|
||||||
|
if (typeof (part as any).id === "string" && (part as any).id.length > 0) {
|
||||||
|
target.partIds.push((part as any).id)
|
||||||
|
}
|
||||||
|
target.totalChars += getPartCharCount(part)
|
||||||
}
|
}
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
if (part.type === "compaction") {
|
if (part.type === "compaction") {
|
||||||
flushPending()
|
flushPending()
|
||||||
const isAuto = Boolean((part as any)?.auto)
|
const isAuto = Boolean((part as any)?.auto)
|
||||||
|
const partId = typeof (part as any)?.id === "string" ? ((part as any).id as string) : ""
|
||||||
result.push({
|
result.push({
|
||||||
id: `${record.id}:${segmentIndex}`,
|
id: `${record.id}:${segmentIndex}`,
|
||||||
messageId: record.id,
|
messageId: record.id,
|
||||||
@@ -242,6 +277,8 @@ export function buildTimelineSegments(
|
|||||||
label: segmentLabel("compaction"),
|
label: segmentLabel("compaction"),
|
||||||
tooltip: isAuto ? t("messageTimeline.tooltip.compaction.auto") : t("messageTimeline.tooltip.compaction.manual"),
|
tooltip: isAuto ? t("messageTimeline.tooltip.compaction.auto") : t("messageTimeline.tooltip.compaction.manual"),
|
||||||
variant: isAuto ? "auto" : "manual",
|
variant: isAuto ? "auto" : "manual",
|
||||||
|
partId,
|
||||||
|
totalChars: 0,
|
||||||
})
|
})
|
||||||
segmentIndex += 1
|
segmentIndex += 1
|
||||||
continue
|
continue
|
||||||
@@ -250,19 +287,23 @@ export function buildTimelineSegments(
|
|||||||
if (part.type === "step-start" || part.type === "step-finish") {
|
if (part.type === "step-start" || part.type === "step-finish") {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
const text = collectTextFromPart(part, t)
|
const text = collectTextFromPart(part, t)
|
||||||
if (text.trim().length === 0) continue
|
if (text.trim().length === 0) continue
|
||||||
const target = ensureSegment(defaultContentType)
|
const target = ensureSegment(defaultContentType)
|
||||||
if (target) {
|
if (target) {
|
||||||
target.texts.push(text)
|
target.texts.push(text)
|
||||||
target.hasPrimaryText = true
|
target.hasPrimaryText = true
|
||||||
|
if (typeof (part as any).id === "string" && (part as any).id.length > 0) {
|
||||||
|
target.partIds.push((part as any).id)
|
||||||
|
}
|
||||||
|
target.totalChars += getPartCharCount(part)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
flushPending()
|
flushPending()
|
||||||
|
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -278,7 +319,14 @@ const MessageTimeline: Component<MessageTimelineProps> = (props) => {
|
|||||||
let hoverTimer: number | null = null
|
let hoverTimer: number | null = null
|
||||||
let closeTimer: number | null = null
|
let closeTimer: number | null = null
|
||||||
const showTools = () => props.showToolSegments ?? true
|
const showTools = () => props.showToolSegments ?? true
|
||||||
|
const deleteHover = () => props.deleteHover?.() ?? { kind: "none" as const }
|
||||||
|
|
||||||
|
const isHistogramEligible = (segment: TimelineSegment): boolean => {
|
||||||
|
const allowed = props.deletableMessageIds?.()
|
||||||
|
if (!allowed) return true
|
||||||
|
return allowed.has(segment.messageId)
|
||||||
|
}
|
||||||
|
|
||||||
const registerButtonRef = (segmentId: string, element: HTMLButtonElement | null) => {
|
const registerButtonRef = (segmentId: string, element: HTMLButtonElement | null) => {
|
||||||
if (element) {
|
if (element) {
|
||||||
buttonRefs.set(segmentId, element)
|
buttonRefs.set(segmentId, element)
|
||||||
@@ -286,7 +334,7 @@ const MessageTimeline: Component<MessageTimelineProps> = (props) => {
|
|||||||
buttonRefs.delete(segmentId)
|
buttonRefs.delete(segmentId)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const clearHoverTimer = () => {
|
const clearHoverTimer = () => {
|
||||||
if (hoverTimer !== null && typeof window !== "undefined") {
|
if (hoverTimer !== null && typeof window !== "undefined") {
|
||||||
window.clearTimeout(hoverTimer)
|
window.clearTimeout(hoverTimer)
|
||||||
@@ -312,8 +360,11 @@ const MessageTimeline: Component<MessageTimelineProps> = (props) => {
|
|||||||
setHoverAnchorRect(null)
|
setHoverAnchorRect(null)
|
||||||
}, 160)
|
}, 160)
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleMouseEnter = (segment: TimelineSegment, event: MouseEvent) => {
|
const handleMouseEnter = (segment: TimelineSegment, event: MouseEvent) => {
|
||||||
|
// Suppress previews during long-press selection gestures.
|
||||||
|
if (longPressTimer !== null) return
|
||||||
|
|
||||||
if (typeof window === "undefined") return
|
if (typeof window === "undefined") return
|
||||||
clearHoverTimer()
|
clearHoverTimer()
|
||||||
clearCloseTimer()
|
clearCloseTimer()
|
||||||
@@ -328,7 +379,7 @@ const MessageTimeline: Component<MessageTimelineProps> = (props) => {
|
|||||||
const handleMouseLeave = () => {
|
const handleMouseLeave = () => {
|
||||||
scheduleClose()
|
scheduleClose()
|
||||||
}
|
}
|
||||||
|
|
||||||
createEffect(() => {
|
createEffect(() => {
|
||||||
if (typeof window === "undefined") return
|
if (typeof window === "undefined") return
|
||||||
const anchor = hoverAnchorRect()
|
const anchor = hoverAnchorRect()
|
||||||
@@ -350,13 +401,235 @@ const MessageTimeline: Component<MessageTimelineProps> = (props) => {
|
|||||||
clearCloseTimer()
|
clearCloseTimer()
|
||||||
})
|
})
|
||||||
|
|
||||||
createEffect(() => {
|
// --- Selection & histogram rib state ---
|
||||||
const activeId = props.activeMessageId
|
const isSelectionActive = createMemo(() => (props.selectedIds?.().size ?? 0) > 0)
|
||||||
|
|
||||||
|
// Segments eligible for xray ribs. We intentionally exclude messages before
|
||||||
|
// the last compaction (when provided by the parent) to avoid misleading token
|
||||||
|
// weights for content that's no longer in context.
|
||||||
|
const xraySegments = createMemo(() => {
|
||||||
|
if (!isSelectionActive()) return [] as TimelineSegment[]
|
||||||
|
return props.segments.filter((segment) => isHistogramEligible(segment))
|
||||||
|
})
|
||||||
|
|
||||||
|
// Stable layout offsets per badge (relative to scroll content), recomputed only
|
||||||
|
// on activation, resize, or expansion — NOT on every scroll frame.
|
||||||
|
const [badgeOffsets, setBadgeOffsets] = createSignal<Record<string, { layoutTop: number; height: number }>>({})
|
||||||
|
const [windowWidth, setWindowWidth] = createSignal(typeof window !== "undefined" ? window.innerWidth : 1200)
|
||||||
|
let scrollContainerRef: HTMLDivElement | undefined
|
||||||
|
let xrayOverlayRef: HTMLDivElement | undefined
|
||||||
|
|
||||||
|
// Full layout recomputation: reads every badge's getBoundingClientRect once,
|
||||||
|
// then stores offsets relative to the scroll content so they survive scrolling.
|
||||||
|
const computeBadgeLayout = () => {
|
||||||
|
if (!isSelectionActive() || !scrollContainerRef) return
|
||||||
|
const containerRect = scrollContainerRef.getBoundingClientRect()
|
||||||
|
const scrollTop = scrollContainerRef.scrollTop
|
||||||
|
const offsets: Record<string, { layoutTop: number; height: number }> = {}
|
||||||
|
|
||||||
|
for (const [id, element] of buttonRefs.entries()) {
|
||||||
|
if (!element) continue
|
||||||
|
const rect = element.getBoundingClientRect()
|
||||||
|
// Store position relative to scroll content (survives scrolling).
|
||||||
|
offsets[id] = {
|
||||||
|
layoutTop: rect.top - containerRect.top + scrollTop,
|
||||||
|
height: rect.height,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
setBadgeOffsets(offsets)
|
||||||
|
if (xrayOverlayRef) {
|
||||||
|
xrayOverlayRef.style.setProperty("--xray-scroll-y", `${-scrollTop}px`)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof window !== "undefined") {
|
||||||
|
setWindowWidth(window.innerWidth)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleScroll = () => {
|
||||||
|
if (!isSelectionActive()) return
|
||||||
|
if (!scrollContainerRef || !xrayOverlayRef) return
|
||||||
|
xrayOverlayRef.style.setProperty("--xray-scroll-y", `${-scrollContainerRef.scrollTop}px`)
|
||||||
|
}
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
if (isSelectionActive()) {
|
||||||
|
computeBadgeLayout()
|
||||||
|
if (typeof window !== "undefined") {
|
||||||
|
// Deferred pass: tool segments become visible when selection activates,
|
||||||
|
// but they may need a layout pass before getBoundingClientRect is accurate.
|
||||||
|
requestAnimationFrame(computeBadgeLayout)
|
||||||
|
window.addEventListener("resize", computeBadgeLayout)
|
||||||
|
onCleanup(() => {
|
||||||
|
window.removeEventListener("resize", computeBadgeLayout)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
// Re-compute badge layout after expansion changes (tools become visible in DOM)
|
||||||
|
createEffect(() => {
|
||||||
|
props.expandedMessageIds?.()
|
||||||
|
if (isSelectionActive()) {
|
||||||
|
requestAnimationFrame(computeBadgeLayout)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const maxRibWidth = createMemo(() => Math.round(windowWidth() * 0.5))
|
||||||
|
|
||||||
|
// Compute fresh char counts from the store. segment.totalChars can be stale for
|
||||||
|
// tool parts whose output arrived after the timeline segment was first built.
|
||||||
|
const liveSegmentChars = createMemo(() => {
|
||||||
|
if (!isSelectionActive()) return {} as Record<string, number>
|
||||||
|
const result: Record<string, number> = {}
|
||||||
|
const resolvedStore = store()
|
||||||
|
|
||||||
|
// Compute live char counts by reading only the parts that the segment
|
||||||
|
// references (partIds/toolPartIds). This stays accurate for streamed tool
|
||||||
|
// outputs without scanning every part in the message.
|
||||||
|
for (const segment of xraySegments()) {
|
||||||
|
const record = resolvedStore.getMessage(segment.messageId)
|
||||||
|
if (!record) {
|
||||||
|
result[segment.id] = segment.totalChars
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
const ids = [...(segment.partIds ?? []), ...(segment.toolPartIds ?? [])]
|
||||||
|
let chars = 0
|
||||||
|
for (const partId of ids) {
|
||||||
|
const part = record.parts?.[partId]?.data
|
||||||
|
if (!part) continue
|
||||||
|
chars += getPartCharCount(part)
|
||||||
|
}
|
||||||
|
|
||||||
|
result[segment.id] = chars > 0 ? chars : segment.totalChars
|
||||||
|
}
|
||||||
|
|
||||||
|
return result
|
||||||
|
})
|
||||||
|
|
||||||
|
// Pre-compute aggregate tokens per message: O(n) once, O(1) per lookup.
|
||||||
|
// Avoids the previous O(n²) pattern of iterating all segments inside each <For> item.
|
||||||
|
const aggregateTokensByMessageId = createMemo(() => {
|
||||||
|
const chars = liveSegmentChars()
|
||||||
|
const result: Record<string, number> = {}
|
||||||
|
for (const s of xraySegments()) {
|
||||||
|
result[s.messageId] = (result[s.messageId] ?? 0) + (chars[s.id] ?? s.totalChars)
|
||||||
|
}
|
||||||
|
for (const id of Object.keys(result)) {
|
||||||
|
result[id] = Math.max(Math.round(result[id] / 4), 1)
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
})
|
||||||
|
|
||||||
|
const getSegmentTokens = (segment: TimelineSegment): number => {
|
||||||
|
const isExpanded = props.expandedMessageIds?.().has(segment.messageId) ?? false
|
||||||
|
// When tools are hidden (not expanded, not in selection mode), assistant/user
|
||||||
|
// bars show aggregate tokens for the whole message. When tools are visible
|
||||||
|
// (expanded or selection mode active), each segment shows its own tokens to
|
||||||
|
// avoid double-counting.
|
||||||
|
if (!isExpanded && !isSelectionActive() && (segment.type === "assistant" || segment.type === "user")) {
|
||||||
|
return aggregateTokensByMessageId()[segment.messageId] ?? 1
|
||||||
|
}
|
||||||
|
const chars = liveSegmentChars()[segment.id] ?? segment.totalChars
|
||||||
|
return Math.max(Math.round(chars / 4), 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
const getMessageAggregateTokens = (messageId: string): number => {
|
||||||
|
return aggregateTokensByMessageId()[messageId] ?? 1
|
||||||
|
}
|
||||||
|
|
||||||
|
const formatTokenLabel = (tokens: number): string => {
|
||||||
|
if (tokens >= 1000000) return `${(tokens / 1000000).toFixed(1)}M`
|
||||||
|
if (tokens >= 1000) return `${(tokens / 1000).toFixed(1)}K`
|
||||||
|
return String(tokens)
|
||||||
|
}
|
||||||
|
|
||||||
|
const maxTokens = createMemo(() => {
|
||||||
|
let max = 0
|
||||||
|
for (const s of xraySegments()) {
|
||||||
|
const tokens = getSegmentTokens(s)
|
||||||
|
if (tokens > max) max = tokens
|
||||||
|
}
|
||||||
|
return Math.max(max, 1)
|
||||||
|
})
|
||||||
|
|
||||||
|
// --- Long-press for mobile selection ---
|
||||||
|
let longPressTimer: number | null = null
|
||||||
|
let wasLongPress = false
|
||||||
|
let pressStartPos = { x: 0, y: 0 }
|
||||||
|
|
||||||
|
const handlePointerDown = (segment: TimelineSegment, event: PointerEvent) => {
|
||||||
|
if (event.button !== 0) return
|
||||||
|
wasLongPress = false
|
||||||
|
pressStartPos = { x: event.clientX, y: event.clientY }
|
||||||
|
|
||||||
|
clearHoverTimer()
|
||||||
|
clearCloseTimer()
|
||||||
|
|
||||||
|
if (longPressTimer !== null && typeof window !== "undefined") {
|
||||||
|
window.clearTimeout(longPressTimer)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof window !== "undefined") {
|
||||||
|
longPressTimer = window.setTimeout(() => {
|
||||||
|
longPressTimer = null
|
||||||
|
wasLongPress = true
|
||||||
|
|
||||||
|
// Scroll anchoring: preserve visual position of the pressed badge.
|
||||||
|
const btn = buttonRefs.get(segment.id)
|
||||||
|
let anchorOffset: number | null = null
|
||||||
|
if (btn && scrollContainerRef) {
|
||||||
|
anchorOffset = btn.offsetTop - scrollContainerRef.scrollTop
|
||||||
|
}
|
||||||
|
|
||||||
|
if (props.onLongPressSelection) {
|
||||||
|
props.onLongPressSelection(segment)
|
||||||
|
} else {
|
||||||
|
props.onToggleSelection?.(segment.id)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (anchorOffset !== null && btn && scrollContainerRef) {
|
||||||
|
const desired = btn.offsetTop - anchorOffset
|
||||||
|
if (Math.abs(scrollContainerRef.scrollTop - desired) > 1) {
|
||||||
|
scrollContainerRef.scrollTop = desired
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, LONG_PRESS_MS)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handlePointerUp = () => {
|
||||||
|
if (longPressTimer !== null && typeof window !== "undefined") {
|
||||||
|
window.clearTimeout(longPressTimer)
|
||||||
|
longPressTimer = null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handlePointerMove = (event: PointerEvent) => {
|
||||||
|
if (longPressTimer !== null) {
|
||||||
|
const dist = Math.sqrt(
|
||||||
|
Math.pow(event.clientX - pressStartPos.x, 2) +
|
||||||
|
Math.pow(event.clientY - pressStartPos.y, 2),
|
||||||
|
)
|
||||||
|
if (dist > JITTER_THRESHOLD) {
|
||||||
|
if (typeof window !== "undefined") {
|
||||||
|
window.clearTimeout(longPressTimer)
|
||||||
|
}
|
||||||
|
longPressTimer = null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleContextMenu = (event: MouseEvent) => {
|
||||||
|
if (wasLongPress) {
|
||||||
|
event.preventDefault()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
createEffect(on(() => props.activeSegmentId, (activeId) => {
|
||||||
if (!activeId) return
|
if (!activeId) return
|
||||||
const targetSegment = props.segments.find((segment) => segment.messageId === activeId)
|
const element = buttonRefs.get(activeId)
|
||||||
if (!targetSegment) return
|
|
||||||
const element = buttonRefs.get(targetSegment.id)
|
|
||||||
if (!element) return
|
if (!element) return
|
||||||
const timer = typeof window !== "undefined" ? window.setTimeout(() => {
|
const timer = typeof window !== "undefined" ? window.setTimeout(() => {
|
||||||
element.scrollIntoView({ block: "nearest", behavior: "smooth" })
|
element.scrollIntoView({ block: "nearest", behavior: "smooth" })
|
||||||
@@ -366,7 +639,7 @@ const MessageTimeline: Component<MessageTimelineProps> = (props) => {
|
|||||||
window.clearTimeout(timer)
|
window.clearTimeout(timer)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
})
|
}))
|
||||||
|
|
||||||
createEffect(() => {
|
createEffect(() => {
|
||||||
const element = tooltipElement()
|
const element = tooltipElement()
|
||||||
@@ -383,92 +656,265 @@ const MessageTimeline: Component<MessageTimelineProps> = (props) => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
const previewData = createMemo(() => {
|
const previewData = createMemo(() => {
|
||||||
|
|
||||||
const segment = hoveredSegment()
|
const segment = hoveredSegment()
|
||||||
if (!segment) return null
|
if (!segment) return null
|
||||||
const record = store().getMessage(segment.messageId)
|
const record = store().getMessage(segment.messageId)
|
||||||
if (!record) return null
|
if (!record) return null
|
||||||
return { messageId: segment.messageId }
|
return { messageId: segment.messageId }
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// Pre-computed set of messageIds that have at least one tool segment.
|
||||||
|
// Used by groupRole() inside <For> to avoid O(n) .some() per segment → O(1) .has().
|
||||||
|
const messagesWithTools = createMemo(() => {
|
||||||
|
const set = new Set<string>()
|
||||||
|
for (const s of props.segments) {
|
||||||
|
if (s.type === "tool") set.add(s.messageId)
|
||||||
|
}
|
||||||
|
return set
|
||||||
|
})
|
||||||
|
|
||||||
|
// Pre-computed index map for session message ordering.
|
||||||
|
// Used by isDeleteHovered() to replace O(n) indexOf with O(1) Map.get().
|
||||||
|
const messageIdToSessionIndex = createMemo(() => {
|
||||||
|
const ids = store().getSessionMessageIds(props.sessionId)
|
||||||
|
const map = new Map<string, number>()
|
||||||
|
for (let i = 0; i < ids.length; i++) map.set(ids[i], i)
|
||||||
|
return map
|
||||||
|
})
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div class="message-timeline" role="navigation" aria-label={t("messageTimeline.ariaLabel")}>
|
<div class="message-timeline-container">
|
||||||
<For each={props.segments}>
|
<div
|
||||||
{(segment) => {
|
ref={scrollContainerRef}
|
||||||
onCleanup(() => buttonRefs.delete(segment.id))
|
class={`message-timeline${isSelectionActive() ? " message-timeline--selection-active" : ""}`}
|
||||||
const isActive = () => props.activeMessageId === segment.messageId
|
role="navigation"
|
||||||
|
aria-label={t("messageTimeline.ariaLabel")}
|
||||||
|
onScroll={handleScroll}
|
||||||
|
>
|
||||||
|
<For each={props.segments}>
|
||||||
|
{(segment, segIndex) => {
|
||||||
|
onCleanup(() => buttonRefs.delete(segment.id))
|
||||||
|
const isActive = () => props.activeSegmentId === segment.id
|
||||||
|
const isSelected = () => props.selectedIds?.().has(segment.id)
|
||||||
|
|
||||||
const hasActivePermission = () => {
|
const isDeleteHovered = () => {
|
||||||
if (segment.type !== "tool") return false
|
const hover = deleteHover() as DeleteHoverState
|
||||||
const partIds = segment.toolPartIds ?? []
|
if (hover.kind === "message") {
|
||||||
if (partIds.length === 0) return false
|
return hover.messageId === segment.messageId
|
||||||
for (const partId of partIds) {
|
}
|
||||||
const permissionState = store().getPermissionState(segment.messageId, partId)
|
|
||||||
if (permissionState?.active) return true
|
if (hover.kind === "deleteUpTo") {
|
||||||
|
const indexMap = messageIdToSessionIndex()
|
||||||
|
const targetIndex = indexMap.get(hover.messageId)
|
||||||
|
if (targetIndex === undefined) return false
|
||||||
|
const segmentIndex = indexMap.get(segment.messageId)
|
||||||
|
if (segmentIndex === undefined) return false
|
||||||
|
return segmentIndex >= targetIndex
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
}
|
}
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
const isHidden = () => segment.type === "tool" && !(showTools() || isActive() || hasActivePermission())
|
const isDeleteSelected = () => {
|
||||||
|
const selected = props.selectedMessageIds?.()
|
||||||
|
if (!selected) return false
|
||||||
|
return selected.has(segment.messageId)
|
||||||
|
}
|
||||||
|
|
||||||
const shortLabelContent = () => {
|
const hasActivePermission = () => {
|
||||||
if (segment.type === "tool") {
|
if (segment.type !== "tool") return false
|
||||||
if (hasActivePermission()) {
|
const partIds = segment.toolPartIds ?? []
|
||||||
return <ShieldAlert class="message-timeline-icon" aria-hidden="true" />
|
if (partIds.length === 0) return false
|
||||||
|
for (const partId of partIds) {
|
||||||
|
const permissionState = store().getPermissionState(segment.messageId, partId)
|
||||||
|
if (permissionState?.active) return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
const isExpanded = () => props.expandedMessageIds?.().has(segment.messageId) ?? false
|
||||||
|
const isHidden = () =>
|
||||||
|
segment.type === "tool" &&
|
||||||
|
!(showTools() || isExpanded() || isSelectionActive() || isActive() || hasActivePermission() || isDeleteHovered() || isDeleteSelected())
|
||||||
|
|
||||||
|
// Group visual indicators: tools belong to the same message as their
|
||||||
|
// assistant. Uses messageId for correctness (not positional adjacency).
|
||||||
|
const groupRole = (): "child" | "parent" | "none" => {
|
||||||
|
if (segment.type === "tool") return "child"
|
||||||
|
if (segment.type === "assistant" && messagesWithTools().has(segment.messageId)) return "parent"
|
||||||
|
return "none"
|
||||||
|
}
|
||||||
|
const isGroupStart = () => {
|
||||||
|
if (segment.type !== "tool") return false
|
||||||
|
const idx = segIndex()
|
||||||
|
const prev = idx > 0 ? props.segments[idx - 1] : null
|
||||||
|
// First tool in the message's run: either nothing before, or previous
|
||||||
|
// segment is from a different message or is not a tool.
|
||||||
|
return !prev || prev.type !== "tool" || prev.messageId !== segment.messageId
|
||||||
|
}
|
||||||
|
|
||||||
|
const shortLabelContent = () => {
|
||||||
|
if (segment.type === "tool") {
|
||||||
|
if (hasActivePermission()) {
|
||||||
|
return <ShieldAlert class="message-timeline-icon" aria-hidden="true" />
|
||||||
|
}
|
||||||
|
return segment.shortLabel ?? getToolIcon("tool")
|
||||||
}
|
}
|
||||||
return segment.shortLabel ?? getToolIcon("tool")
|
if (segment.type === "compaction") {
|
||||||
|
return <FoldVertical class="message-timeline-icon" aria-hidden="true" />
|
||||||
|
}
|
||||||
|
if (segment.type === "user") {
|
||||||
|
return <UserIcon class="message-timeline-icon" aria-hidden="true" />
|
||||||
|
}
|
||||||
|
return <BotIcon class="message-timeline-icon" aria-hidden="true" />
|
||||||
}
|
}
|
||||||
if (segment.type === "compaction") {
|
|
||||||
return <FoldVertical class="message-timeline-icon" aria-hidden="true" />
|
|
||||||
}
|
|
||||||
if (segment.type === "user") {
|
|
||||||
return <UserIcon class="message-timeline-icon" aria-hidden="true" />
|
|
||||||
}
|
|
||||||
return <BotIcon class="message-timeline-icon" aria-hidden="true" />
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<button
|
<button
|
||||||
ref={(el) => registerButtonRef(segment.id, el)}
|
ref={(el) => registerButtonRef(segment.id, el)}
|
||||||
type="button"
|
type="button"
|
||||||
data-variant={segment.variant}
|
data-variant={segment.variant}
|
||||||
class={`message-timeline-segment message-timeline-${segment.type} ${hasActivePermission() ? "message-timeline-segment-permission" : ""} ${segment.type === "compaction" ? `message-timeline-compaction-${segment.variant ?? "manual"}` : ""} ${isActive() ? "message-timeline-segment-active" : ""} ${isHidden() ? "message-timeline-segment-hidden" : ""}`}
|
class={`message-timeline-segment message-timeline-${segment.type} ${hasActivePermission() ? "message-timeline-segment-permission" : ""} ${segment.type === "compaction" ? `message-timeline-compaction-${segment.variant ?? "manual"}` : ""} ${isActive() ? "message-timeline-segment-active" : ""} ${isHidden() ? "message-timeline-segment-hidden" : ""} ${isSelected() ? "message-timeline-segment-selected" : ""} ${isDeleteSelected() ? "message-timeline-segment-delete-selected" : ""} ${groupRole() !== "none" ? `message-timeline-group-${groupRole()}` : ""} ${isGroupStart() ? "message-timeline-group-start" : ""}`}
|
||||||
|
|
||||||
aria-current={isActive() ? "true" : undefined}
|
data-delete-hover={isDeleteHovered() || isDeleteSelected() || isSelected() ? "true" : undefined}
|
||||||
aria-hidden={isHidden() ? "true" : undefined}
|
|
||||||
onClick={() => props.onSegmentClick?.(segment)}
|
aria-current={isActive() ? "true" : undefined}
|
||||||
onMouseEnter={(event) => handleMouseEnter(segment, event)}
|
aria-hidden={isHidden() ? "true" : undefined}
|
||||||
onMouseLeave={handleMouseLeave}
|
onClick={(event) => {
|
||||||
>
|
if (wasLongPress) {
|
||||||
<span class="message-timeline-label message-timeline-label-full">{segment.label}</span>
|
wasLongPress = false
|
||||||
<span class="message-timeline-label message-timeline-label-short">{shortLabelContent()}</span>
|
return
|
||||||
</button>
|
}
|
||||||
)
|
|
||||||
}}
|
// Capture scroll anchor before selection changes may toggle
|
||||||
</For>
|
// tool segment visibility, which shifts timeline layout.
|
||||||
<Show when={previewData()}>
|
const btn = buttonRefs.get(segment.id)
|
||||||
{(data) => {
|
let anchorOffset: number | null = null
|
||||||
onCleanup(() => setTooltipElement(null))
|
if (btn && scrollContainerRef) {
|
||||||
return (
|
anchorOffset = btn.offsetTop - scrollContainerRef.scrollTop
|
||||||
<div
|
}
|
||||||
ref={(element) => setTooltipElement(element)}
|
|
||||||
class="message-timeline-tooltip"
|
const isMultiSelectActive = (props.selectedIds?.().size ?? 0) > 0
|
||||||
style={{ top: `${tooltipCoords().top}px`, left: `${tooltipCoords().left}px` }}
|
|
||||||
onMouseEnter={() => clearCloseTimer()}
|
if (event.shiftKey) {
|
||||||
onMouseLeave={() => scheduleClose()}
|
props.onSelectRange?.(segment.id)
|
||||||
>
|
} else if (event.ctrlKey || event.metaKey) {
|
||||||
<MessagePreview
|
props.onToggleSelection?.(segment.id)
|
||||||
messageId={data().messageId}
|
} else if (isMultiSelectActive) {
|
||||||
instanceId={props.instanceId}
|
// In selection mode, plain click scrolls to the message
|
||||||
sessionId={props.sessionId}
|
// instead of clearing. Selection is cleared by clicking
|
||||||
store={store}
|
// anywhere inside the chat container or pressing Esc.
|
||||||
/>
|
props.onSegmentClick?.(segment)
|
||||||
</div>
|
} else {
|
||||||
)
|
props.onSegmentClick?.(segment)
|
||||||
}}
|
}
|
||||||
|
|
||||||
|
// Restore scroll anchor: keep the clicked badge at the same
|
||||||
|
// visual position after hidden tools appear or disappear.
|
||||||
|
if (anchorOffset !== null && btn && scrollContainerRef) {
|
||||||
|
const desired = btn.offsetTop - anchorOffset
|
||||||
|
if (Math.abs(scrollContainerRef.scrollTop - desired) > 1) {
|
||||||
|
scrollContainerRef.scrollTop = desired
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
onPointerDown={(e) => handlePointerDown(segment, e)}
|
||||||
|
onPointerUp={handlePointerUp}
|
||||||
|
onPointerCancel={handlePointerUp}
|
||||||
|
onPointerMove={handlePointerMove}
|
||||||
|
onContextMenu={handleContextMenu}
|
||||||
|
onMouseEnter={(event) => handleMouseEnter(segment, event)}
|
||||||
|
onMouseLeave={handleMouseLeave}
|
||||||
|
>
|
||||||
|
<span class="message-timeline-label message-timeline-label-full">{segment.label}</span>
|
||||||
|
<span class="message-timeline-label message-timeline-label-short">{shortLabelContent()}</span>
|
||||||
|
</button>
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
</For>
|
||||||
|
<Show when={previewData()}>
|
||||||
|
{(data) => {
|
||||||
|
onCleanup(() => setTooltipElement(null))
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
ref={(element) => setTooltipElement(element)}
|
||||||
|
class="message-timeline-tooltip"
|
||||||
|
style={{ top: `${tooltipCoords().top}px`, left: `${tooltipCoords().left}px` }}
|
||||||
|
onMouseEnter={() => clearCloseTimer()}
|
||||||
|
onMouseLeave={() => scheduleClose()}
|
||||||
|
>
|
||||||
|
<MessagePreview
|
||||||
|
messageId={data().messageId}
|
||||||
|
instanceId={props.instanceId}
|
||||||
|
sessionId={props.sessionId}
|
||||||
|
store={store}
|
||||||
|
deleteHover={props.deleteHover}
|
||||||
|
onDeleteHoverChange={props.onDeleteHoverChange}
|
||||||
|
onDeleteMessagesUpTo={props.onDeleteMessagesUpTo}
|
||||||
|
selectedMessageIds={props.selectedMessageIds}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
</Show>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Show when={isSelectionActive()}>
|
||||||
|
<div
|
||||||
|
ref={(el) => {
|
||||||
|
xrayOverlayRef = el
|
||||||
|
if (xrayOverlayRef && scrollContainerRef) {
|
||||||
|
xrayOverlayRef.style.setProperty("--xray-scroll-y", `${-scrollContainerRef.scrollTop}px`)
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
class="message-timeline-xray-overlay"
|
||||||
|
style={{ "--max-rib-width": `${maxRibWidth()}px` }}
|
||||||
|
>
|
||||||
|
<div class="message-timeline-xray-overlay-inner">
|
||||||
|
<For each={xraySegments()}>
|
||||||
|
{(segment) => {
|
||||||
|
const pos = () => {
|
||||||
|
const offset = badgeOffsets()[segment.id]
|
||||||
|
if (!offset) return null
|
||||||
|
return { top: offset.layoutTop + offset.height / 2 }
|
||||||
|
}
|
||||||
|
const tokens = () => getSegmentTokens(segment)
|
||||||
|
const relativeWeight = () => tokens() / maxTokens()
|
||||||
|
const absoluteWeight = () => Math.min(tokens() / ABSOLUTE_TOKEN_CAP, 1.0)
|
||||||
|
const isOverflow = () => tokens() > ABSOLUTE_TOKEN_CAP
|
||||||
|
const isParent = segment.type === "assistant" || segment.type === "user"
|
||||||
|
const displayTokens = () =>
|
||||||
|
isParent ? getMessageAggregateTokens(segment.messageId) : tokens()
|
||||||
|
return (
|
||||||
|
<Show when={pos()}>
|
||||||
|
<div
|
||||||
|
class="message-timeline-xray-rib"
|
||||||
|
style={{
|
||||||
|
top: `${pos()!.top}px`,
|
||||||
|
left: "var(--xray-overhang)",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<span class="message-timeline-xray-token-label">
|
||||||
|
{formatTokenLabel(displayTokens())}
|
||||||
|
</span>
|
||||||
|
<div
|
||||||
|
class="message-timeline-relative-bar"
|
||||||
|
style={{ "--segment-weight": relativeWeight() }}
|
||||||
|
/>
|
||||||
|
<div
|
||||||
|
class={`message-timeline-absolute-bar${isOverflow() ? " message-timeline-absolute-bar-overflow" : ""}`}
|
||||||
|
style={{ "--segment-weight": absoluteWeight() }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</Show>
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
</For>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</Show>
|
</Show>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
export default MessageTimeline
|
export default MessageTimeline
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import { getAttachments, removeAttachment } from "../../stores/attachments"
|
|||||||
import { instances } from "../../stores/instances"
|
import { instances } from "../../stores/instances"
|
||||||
import { loadMessages, sendMessage, forkSession, renameSession, isSessionMessagesLoading, setActiveParentSession, setActiveSession, runShellCommand, abortSession } from "../../stores/sessions"
|
import { loadMessages, sendMessage, forkSession, renameSession, isSessionMessagesLoading, setActiveParentSession, setActiveSession, runShellCommand, abortSession } from "../../stores/sessions"
|
||||||
import { isSessionBusy as getSessionBusyStatus } from "../../stores/session-status"
|
import { isSessionBusy as getSessionBusyStatus } from "../../stores/session-status"
|
||||||
|
import { deleteMessage } from "../../stores/session-actions"
|
||||||
import { showAlertDialog } from "../../stores/alerts"
|
import { showAlertDialog } from "../../stores/alerts"
|
||||||
import { getLogger } from "../../lib/logger"
|
import { getLogger } from "../../lib/logger"
|
||||||
import { requestData } from "../../lib/opencode-api"
|
import { requestData } from "../../lib/opencode-api"
|
||||||
@@ -55,12 +56,22 @@ export const SessionView: Component<SessionViewProps> = (props) => {
|
|||||||
|
|
||||||
const attachments = createMemo(() => getAttachments(props.instanceId, props.sessionId))
|
const attachments = createMemo(() => getAttachments(props.instanceId, props.sessionId))
|
||||||
|
|
||||||
|
const MESSAGE_SCROLL_CACHE_SCOPE = "message-stream"
|
||||||
|
|
||||||
let promptInputApi: PromptInputApi | null = null
|
let promptInputApi: PromptInputApi | null = null
|
||||||
let pendingPromptText: string | null = null
|
let pendingPromptText: string | null = null
|
||||||
let pendingSelectionInsert: { text: string; mode: PromptInsertMode } | null = null
|
let pendingSelectionInsert: { text: string; mode: PromptInsertMode } | null = null
|
||||||
|
|
||||||
let scrollToBottomHandle: (() => void) | undefined
|
let scrollToBottomHandle: (() => void) | undefined
|
||||||
let rootRef: HTMLDivElement | undefined
|
let rootRef: HTMLDivElement | undefined
|
||||||
|
|
||||||
|
function shouldScrollToBottomOnActivate() {
|
||||||
|
const current = session()
|
||||||
|
if (!current) return true
|
||||||
|
const snapshot = messageStore().getScrollSnapshot(current.id, MESSAGE_SCROLL_CACHE_SCOPE)
|
||||||
|
return !snapshot || snapshot.atBottom
|
||||||
|
}
|
||||||
|
|
||||||
function scheduleScrollToBottom() {
|
function scheduleScrollToBottom() {
|
||||||
if (!scrollToBottomHandle) return
|
if (!scrollToBottomHandle) return
|
||||||
requestAnimationFrame(() => {
|
requestAnimationFrame(() => {
|
||||||
@@ -69,6 +80,7 @@ export const SessionView: Component<SessionViewProps> = (props) => {
|
|||||||
}
|
}
|
||||||
createEffect(() => {
|
createEffect(() => {
|
||||||
if (!props.isActive) return
|
if (!props.isActive) return
|
||||||
|
if (!shouldScrollToBottomOnActivate()) return
|
||||||
scheduleScrollToBottom()
|
scheduleScrollToBottom()
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -225,6 +237,35 @@ export const SessionView: Component<SessionViewProps> = (props) => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function handleDeleteMessagesUpTo(messageId: string) {
|
||||||
|
const ids = messageStore().getSessionMessageIds(props.sessionId)
|
||||||
|
const index = ids.indexOf(messageId)
|
||||||
|
if (index === -1) return
|
||||||
|
|
||||||
|
const restoredText = getUserMessageText(messageId)
|
||||||
|
const toDelete = ids.slice(index)
|
||||||
|
|
||||||
|
try {
|
||||||
|
for (let idx = toDelete.length - 1; idx >= 0; idx -= 1) {
|
||||||
|
await deleteMessage(props.instanceId, props.sessionId, toDelete[idx])
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
log.error("Failed to delete messages up to", error)
|
||||||
|
showAlertDialog(t("sessionView.alerts.deleteUpToFailed.message"), {
|
||||||
|
title: t("sessionView.alerts.deleteUpToFailed.title"),
|
||||||
|
variant: "error",
|
||||||
|
})
|
||||||
|
} finally {
|
||||||
|
if (restoredText) {
|
||||||
|
if (promptInputApi) {
|
||||||
|
promptInputApi.setPromptText(restoredText, { focus: true })
|
||||||
|
} else {
|
||||||
|
pendingPromptText = restoredText
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async function handleFork(messageId?: string) {
|
async function handleFork(messageId?: string) {
|
||||||
if (!messageId) {
|
if (!messageId) {
|
||||||
log.warn("Fork requires a user message id")
|
log.warn("Fork requires a user message id")
|
||||||
@@ -283,14 +324,17 @@ export const SessionView: Component<SessionViewProps> = (props) => {
|
|||||||
<MessageSection
|
<MessageSection
|
||||||
instanceId={props.instanceId}
|
instanceId={props.instanceId}
|
||||||
sessionId={activeSession.id}
|
sessionId={activeSession.id}
|
||||||
loading={messagesLoading()}
|
loading={messagesLoading()}
|
||||||
onRevert={handleRevert}
|
onRevert={handleRevert}
|
||||||
onFork={handleFork}
|
onDeleteMessagesUpTo={handleDeleteMessagesUpTo}
|
||||||
isActive={props.isActive}
|
onFork={handleFork}
|
||||||
|
isActive={props.isActive}
|
||||||
registerScrollToBottom={(fn) => {
|
registerScrollToBottom={(fn) => {
|
||||||
scrollToBottomHandle = fn
|
scrollToBottomHandle = fn
|
||||||
if (props.isActive) {
|
if (props.isActive) {
|
||||||
scheduleScrollToBottom()
|
if (shouldScrollToBottomOnActivate()) {
|
||||||
|
scheduleScrollToBottom()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,4 +1,4 @@
|
|||||||
import type { ToolState } from "@opencode-ai/sdk"
|
import type { ToolState } from "@opencode-ai/sdk/v2"
|
||||||
import { getRelativePath, isToolStateCompleted, isToolStateError, isToolStateRunning } from "./utils"
|
import { getRelativePath, isToolStateCompleted, isToolStateError, isToolStateRunning } from "./utils"
|
||||||
import { tGlobal } from "../../lib/i18n"
|
import { tGlobal } from "../../lib/i18n"
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import type { Accessor, JSXElement } from "solid-js"
|
import type { Accessor, JSXElement } from "solid-js"
|
||||||
import type { ToolState } from "@opencode-ai/sdk"
|
import type { ToolState } from "@opencode-ai/sdk/v2"
|
||||||
import type { TextPart } from "../../types/message"
|
import type { TextPart } from "../../types/message"
|
||||||
import { Markdown } from "../markdown"
|
import { Markdown } from "../markdown"
|
||||||
import type { MarkdownRenderOptions, ToolScrollHelpers } from "./types"
|
import type { MarkdownRenderOptions, ToolScrollHelpers } from "./types"
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { createMemo, Show, For, createEffect, type Accessor } from "solid-js"
|
import { createMemo, Show, For, createEffect, type Accessor } from "solid-js"
|
||||||
import type { ToolState } from "@opencode-ai/sdk"
|
import type { ToolState } from "@opencode-ai/sdk/v2"
|
||||||
import type { QuestionRequest } from "@opencode-ai/sdk/v2"
|
import type { QuestionRequest } from "@opencode-ai/sdk/v2"
|
||||||
import { useI18n } from "../../lib/i18n"
|
import { useI18n } from "../../lib/i18n"
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { For, Show, createEffect, createMemo, createSignal, untrack } from "solid-js"
|
import { For, Show, createEffect, createMemo, createSignal, untrack } from "solid-js"
|
||||||
import type { ToolState } from "@opencode-ai/sdk"
|
import type { ToolState } from "@opencode-ai/sdk/v2"
|
||||||
import type { ToolRenderer } from "../types"
|
import type { ToolRenderer } from "../types"
|
||||||
import { ensureMarkdownContent, getDefaultToolAction, getToolIcon, getToolName, readToolStatePayload } from "../utils"
|
import { ensureMarkdownContent, getDefaultToolAction, getToolIcon, getToolName, readToolStatePayload } from "../utils"
|
||||||
import { resolveTitleForTool } from "../tool-title"
|
import { resolveTitleForTool } from "../tool-title"
|
||||||
@@ -178,28 +178,116 @@ export const taskRenderer: ToolRenderer = {
|
|||||||
void loadMessages(instanceId, id)
|
void loadMessages(instanceId, id)
|
||||||
})
|
})
|
||||||
|
|
||||||
const childToolKeys = createMemo(() => {
|
const [childToolKeys, setChildToolKeys] = createSignal<string[]>([])
|
||||||
const id = childSessionId()
|
|
||||||
if (!id) return [] as string[]
|
|
||||||
if (!childSessionLoaded()) return [] as string[]
|
|
||||||
|
|
||||||
// React to session changes, but do the scan untracked to avoid
|
let indexedSessionId = ""
|
||||||
// subscribing to every message/part node in the store.
|
let indexedMessageCount = 0
|
||||||
|
let indexedMessageTail = ""
|
||||||
|
const indexedPartCounts = new Map<string, number>()
|
||||||
|
|
||||||
|
function resetChildToolIndex(nextSessionId: string) {
|
||||||
|
indexedSessionId = nextSessionId
|
||||||
|
indexedMessageCount = 0
|
||||||
|
indexedMessageTail = ""
|
||||||
|
indexedPartCounts.clear()
|
||||||
|
setChildToolKeys([])
|
||||||
|
}
|
||||||
|
|
||||||
|
function scanMessageToolParts(messageId: string, startIndex: number) {
|
||||||
|
const record = store.getMessage(messageId)
|
||||||
|
if (!record) return [] as string[]
|
||||||
|
|
||||||
|
const partIds = record.partIds
|
||||||
|
const keys: string[] = []
|
||||||
|
for (let idx = startIndex; idx < partIds.length; idx += 1) {
|
||||||
|
const partId = partIds[idx]
|
||||||
|
const entry = record.parts?.[partId]
|
||||||
|
const data = entry?.data
|
||||||
|
if (!data || (data as any).type !== "tool") continue
|
||||||
|
keys.push(`${messageId}::${partId}`)
|
||||||
|
}
|
||||||
|
indexedPartCounts.set(messageId, partIds.length)
|
||||||
|
return keys
|
||||||
|
}
|
||||||
|
|
||||||
|
function fullRescanChildTools(sessionId: string, messageIds: string[]) {
|
||||||
|
indexedSessionId = sessionId
|
||||||
|
indexedMessageCount = messageIds.length
|
||||||
|
indexedMessageTail = messageIds[messageIds.length - 1] ?? ""
|
||||||
|
indexedPartCounts.clear()
|
||||||
|
|
||||||
|
const nextKeys: string[] = []
|
||||||
|
for (const messageId of messageIds) {
|
||||||
|
nextKeys.push(...scanMessageToolParts(messageId, 0))
|
||||||
|
}
|
||||||
|
setChildToolKeys(nextKeys)
|
||||||
|
}
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
const id = childSessionId()
|
||||||
|
const loaded = childSessionLoaded()
|
||||||
|
|
||||||
|
if (!id || !loaded) {
|
||||||
|
if (indexedSessionId) {
|
||||||
|
resetChildToolIndex("")
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// We use the session revision as the reactive change point, but avoid
|
||||||
|
// rescanning the entire session on every update.
|
||||||
store.getSessionRevision(id)
|
store.getSessionRevision(id)
|
||||||
return untrack(() => {
|
|
||||||
|
untrack(() => {
|
||||||
const messageIds = store.getSessionMessageIds(id)
|
const messageIds = store.getSessionMessageIds(id)
|
||||||
const keys: string[] = []
|
|
||||||
for (const messageId of messageIds) {
|
if (!indexedSessionId || indexedSessionId !== id) {
|
||||||
const record = store.getMessage(messageId)
|
fullRescanChildTools(id, messageIds)
|
||||||
if (!record) continue
|
return
|
||||||
for (const partId of record.partIds) {
|
}
|
||||||
const entry = record.parts?.[partId]
|
|
||||||
const data = entry?.data
|
// Detect structural changes (reorder/shrink) and fall back to a full rescan.
|
||||||
if (!data || (data as any).type !== "tool") continue
|
if (messageIds.length < indexedMessageCount) {
|
||||||
keys.push(`${messageId}::${partId}`)
|
fullRescanChildTools(id, messageIds)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (indexedMessageCount > 0) {
|
||||||
|
const expectedTailIndex = indexedMessageCount - 1
|
||||||
|
if (expectedTailIndex >= 0 && messageIds[expectedTailIndex] !== indexedMessageTail) {
|
||||||
|
fullRescanChildTools(id, messageIds)
|
||||||
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return keys
|
|
||||||
|
const appendedKeys: string[] = []
|
||||||
|
|
||||||
|
// Scan any new messages appended since last index.
|
||||||
|
for (let idx = indexedMessageCount; idx < messageIds.length; idx += 1) {
|
||||||
|
const messageId = messageIds[idx]
|
||||||
|
appendedKeys.push(...scanMessageToolParts(messageId, 0))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Scan a small window of recent messages for newly appended parts.
|
||||||
|
// Deltas typically affect the most recent tool call, so this avoids
|
||||||
|
// iterating every message on every revision.
|
||||||
|
const existingCount = Math.min(indexedMessageCount, messageIds.length)
|
||||||
|
const windowStart = Math.max(0, existingCount - 3)
|
||||||
|
for (let idx = windowStart; idx < existingCount; idx += 1) {
|
||||||
|
const messageId = messageIds[idx]
|
||||||
|
const previousPartCount = indexedPartCounts.get(messageId) ?? 0
|
||||||
|
const record = store.getMessage(messageId)
|
||||||
|
const nextPartCount = record?.partIds.length ?? 0
|
||||||
|
if (nextPartCount > previousPartCount) {
|
||||||
|
appendedKeys.push(...scanMessageToolParts(messageId, previousPartCount))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
indexedMessageCount = messageIds.length
|
||||||
|
indexedMessageTail = messageIds[messageIds.length - 1] ?? ""
|
||||||
|
|
||||||
|
if (appendedKeys.length > 0) {
|
||||||
|
setChildToolKeys((prev) => [...prev, ...appendedKeys])
|
||||||
|
}
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
const promptContent = createMemo(() => {
|
const promptContent = createMemo(() => {
|
||||||
@@ -354,7 +442,7 @@ export const taskRenderer: ToolRenderer = {
|
|||||||
scrollHelpers ? (event) => scrollHelpers.handleScroll(event as Event & { currentTarget: HTMLDivElement }) : undefined
|
scrollHelpers ? (event) => scrollHelpers.handleScroll(event as Event & { currentTarget: HTMLDivElement }) : undefined
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<div class="tool-call-task-summary">
|
<div class="tool-call-task-summary">
|
||||||
<For each={childToolKeys()}>
|
<For each={childToolKeys()}>
|
||||||
{(key) => (
|
{(key) => (
|
||||||
<Show when={renderToolCall}>
|
<Show when={renderToolCall}>
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { For, Show } from "solid-js"
|
import { For, Show } from "solid-js"
|
||||||
import type { ToolState } from "@opencode-ai/sdk"
|
import type { ToolState } from "@opencode-ai/sdk/v2"
|
||||||
import type { ToolRenderer } from "../types"
|
import type { ToolRenderer } from "../types"
|
||||||
import { readToolStatePayload } from "../utils"
|
import { readToolStatePayload } from "../utils"
|
||||||
import { useI18n, tGlobal } from "../../../lib/i18n"
|
import { useI18n, tGlobal } from "../../../lib/i18n"
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import type { ToolState } from "@opencode-ai/sdk"
|
import type { ToolState } from "@opencode-ai/sdk/v2"
|
||||||
import type { ToolRendererContext, ToolRenderer, ToolCallPart } from "./types"
|
import type { ToolRendererContext, ToolRenderer, ToolCallPart } from "./types"
|
||||||
import { getDefaultToolAction, getToolName, isToolStateCompleted, isToolStateRunning } from "./utils"
|
import { getDefaultToolAction, getToolName, isToolStateCompleted, isToolStateRunning } from "./utils"
|
||||||
import { enMessages } from "../../lib/i18n/messages/en"
|
import { enMessages } from "../../lib/i18n/messages/en"
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import type { Accessor, JSXElement } from "solid-js"
|
import type { Accessor, JSXElement } from "solid-js"
|
||||||
import type { ToolState } from "@opencode-ai/sdk"
|
import type { ToolState } from "@opencode-ai/sdk/v2"
|
||||||
import type { ClientPart } from "../../types/message"
|
import type { ClientPart } from "../../types/message"
|
||||||
|
|
||||||
export type ToolCallPart = Extract<ClientPart, { type: "tool" }>
|
export type ToolCallPart = Extract<ClientPart, { type: "tool" }>
|
||||||
|
|||||||
@@ -1,15 +1,15 @@
|
|||||||
import { isRenderableDiffText } from "../../lib/diff-utils"
|
import { isRenderableDiffText } from "../../lib/diff-utils"
|
||||||
import { getLanguageFromPath } from "../../lib/markdown"
|
import { getLanguageFromPath } from "../../lib/markdown"
|
||||||
import type { ToolState } from "@opencode-ai/sdk"
|
import type { ToolState } from "@opencode-ai/sdk/v2"
|
||||||
import type { DiffPayload } from "./types"
|
import type { DiffPayload } from "./types"
|
||||||
import { getLogger } from "../../lib/logger"
|
import { getLogger } from "../../lib/logger"
|
||||||
import { tGlobal } from "../../lib/i18n"
|
import { tGlobal } from "../../lib/i18n"
|
||||||
const log = getLogger("session")
|
const log = getLogger("session")
|
||||||
|
|
||||||
|
|
||||||
export type ToolStateRunning = import("@opencode-ai/sdk").ToolStateRunning
|
export type ToolStateRunning = import("@opencode-ai/sdk/v2").ToolStateRunning
|
||||||
export type ToolStateCompleted = import("@opencode-ai/sdk").ToolStateCompleted
|
export type ToolStateCompleted = import("@opencode-ai/sdk/v2").ToolStateCompleted
|
||||||
export type ToolStateError = import("@opencode-ai/sdk").ToolStateError
|
export type ToolStateError = import("@opencode-ai/sdk/v2").ToolStateError
|
||||||
|
|
||||||
export const diffCapableTools = new Set(["edit", "patch"])
|
export const diffCapableTools = new Set(["edit", "patch"])
|
||||||
|
|
||||||
|
|||||||
933
packages/ui/src/components/virtual-follow-list.tsx
Normal file
933
packages/ui/src/components/virtual-follow-list.tsx
Normal file
@@ -0,0 +1,933 @@
|
|||||||
|
import { Index, Show, createEffect, createMemo, createSignal, onCleanup, type Accessor, type JSX } from "solid-js"
|
||||||
|
import VirtualItem from "./virtual-item"
|
||||||
|
|
||||||
|
const DEFAULT_SCROLL_SENTINEL_MARGIN_PX = 48
|
||||||
|
const USER_SCROLL_INTENT_WINDOW_MS = 600
|
||||||
|
const SCROLL_INTENT_KEYS = new Set(["ArrowUp", "ArrowDown", "PageUp", "PageDown", "Home", "End", " ", "Spacebar"])
|
||||||
|
|
||||||
|
export interface VirtualFollowListApi {
|
||||||
|
scrollToTop: (opts?: { immediate?: boolean }) => void
|
||||||
|
scrollToBottom: (opts?: { immediate?: boolean; suppressAutoAnchor?: boolean }) => void
|
||||||
|
scrollToKey: (
|
||||||
|
key: string,
|
||||||
|
opts?: { behavior?: ScrollBehavior; block?: ScrollLogicalPosition; setAutoScroll?: boolean },
|
||||||
|
) => void
|
||||||
|
notifyContentRendered: () => void
|
||||||
|
setAutoScroll: (enabled: boolean) => void
|
||||||
|
getAutoScroll: () => boolean
|
||||||
|
getScrollElement: () => HTMLDivElement | undefined
|
||||||
|
getShellElement: () => HTMLDivElement | undefined
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface VirtualFollowListState {
|
||||||
|
autoScroll: Accessor<boolean>
|
||||||
|
showScrollTopButton: Accessor<boolean>
|
||||||
|
showScrollBottomButton: Accessor<boolean>
|
||||||
|
scrollButtonsCount: Accessor<number>
|
||||||
|
activeKey: Accessor<string | null>
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface VirtualFollowListProps<T> {
|
||||||
|
items: Accessor<T[]>
|
||||||
|
getKey: (item: T, index: number) => string
|
||||||
|
renderItem: (item: T, index: number) => JSX.Element
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Optional stable DOM id for the item wrapper.
|
||||||
|
* Defaults to the key itself.
|
||||||
|
*/
|
||||||
|
getAnchorId?: (key: string) => string
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Decode an item key from an observed wrapper element id.
|
||||||
|
* Defaults to identity.
|
||||||
|
*/
|
||||||
|
getKeyFromAnchorId?: (anchorId: string) => string
|
||||||
|
|
||||||
|
overscanPx?: number
|
||||||
|
scrollSentinelMarginPx?: number
|
||||||
|
virtualizationEnabled?: Accessor<boolean>
|
||||||
|
suspendMeasurements?: Accessor<boolean>
|
||||||
|
loading?: Accessor<boolean>
|
||||||
|
isActive?: Accessor<boolean>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When switching back to an inactive (cached) pane, the list historically
|
||||||
|
* re-pinned to the bottom if autoScroll was enabled.
|
||||||
|
*
|
||||||
|
* Disable this to preserve the existing scroll position across pane switches.
|
||||||
|
*/
|
||||||
|
scrollToBottomOnActivate?: Accessor<boolean>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Controls whether the list should scroll to bottom the first time items
|
||||||
|
* appear (default behavior for chat streams).
|
||||||
|
*
|
||||||
|
* Set to false when an outer component restores scroll from a cache.
|
||||||
|
*/
|
||||||
|
initialScrollToBottom?: Accessor<boolean>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initial value for the internal autoScroll signal.
|
||||||
|
* Useful when restoring scroll state (e.g. start in non-follow mode).
|
||||||
|
*/
|
||||||
|
initialAutoScroll?: Accessor<boolean>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When this value changes, the list resets internal follow/anchor state.
|
||||||
|
* Useful when reusing the same list instance across different datasets.
|
||||||
|
*/
|
||||||
|
resetKey?: Accessor<string | number>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If this value changes and autoScroll is enabled, the list will
|
||||||
|
* anchor-scroll to the bottom (unless suppressed).
|
||||||
|
*/
|
||||||
|
followToken?: Accessor<string | number>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Optional hooks to render content inside the scroll container.
|
||||||
|
* Useful for empty/loading states that should scroll with the list.
|
||||||
|
*/
|
||||||
|
renderBeforeItems?: Accessor<JSX.Element>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render content inside the shell, above timeline/sidebar layers.
|
||||||
|
* (Quote popovers, etc.)
|
||||||
|
*/
|
||||||
|
renderOverlay?: Accessor<JSX.Element>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provide localized labels for built-in controls.
|
||||||
|
*/
|
||||||
|
scrollToTopAriaLabel?: Accessor<string>
|
||||||
|
scrollToBottomAriaLabel?: Accessor<string>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Receive element refs for external logic (selection, geometry, etc.)
|
||||||
|
*/
|
||||||
|
onScrollElementChange?: (element: HTMLDivElement | undefined) => void
|
||||||
|
onShellElementChange?: (element: HTMLDivElement | undefined) => void
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Callbacks for consumers.
|
||||||
|
*/
|
||||||
|
onScroll?: () => void
|
||||||
|
onMouseUp?: (event: MouseEvent) => void
|
||||||
|
onClick?: (event: MouseEvent) => void
|
||||||
|
onActiveKeyChange?: (key: string | null) => void
|
||||||
|
registerApi?: (api: VirtualFollowListApi) => void
|
||||||
|
registerState?: (state: VirtualFollowListState) => void
|
||||||
|
renderControls?: (state: VirtualFollowListState, api: VirtualFollowListApi) => JSX.Element
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function VirtualFollowList<T>(props: VirtualFollowListProps<T>) {
|
||||||
|
const getAnchorId = (key: string) => (props.getAnchorId ? props.getAnchorId(key) : key)
|
||||||
|
const getKeyFromAnchorId = (anchorId: string) => (props.getKeyFromAnchorId ? props.getKeyFromAnchorId(anchorId) : anchorId)
|
||||||
|
|
||||||
|
const [scrollElement, setScrollElement] = createSignal<HTMLDivElement | undefined>()
|
||||||
|
const [shellElement, setShellElement] = createSignal<HTMLDivElement | undefined>()
|
||||||
|
const [topSentinel, setTopSentinel] = createSignal<HTMLDivElement | null>(null)
|
||||||
|
const [bottomSentinelSignal, setBottomSentinelSignal] = createSignal<HTMLDivElement | null>(null)
|
||||||
|
const bottomSentinel = () => bottomSentinelSignal()
|
||||||
|
|
||||||
|
const isActive = () => (props.isActive ? props.isActive() : true)
|
||||||
|
const scrollToBottomOnActivate = () => (props.scrollToBottomOnActivate ? props.scrollToBottomOnActivate() : true)
|
||||||
|
const initialScrollToBottom = () => (props.initialScrollToBottom ? props.initialScrollToBottom() : true)
|
||||||
|
const initialAutoScroll = () => (props.initialAutoScroll ? props.initialAutoScroll() : true)
|
||||||
|
const isLoading = () => Boolean(props.loading?.())
|
||||||
|
const virtualizationEnabled = () => (props.virtualizationEnabled ? props.virtualizationEnabled() : true)
|
||||||
|
const measurementsSuspended = () => Boolean(props.suspendMeasurements?.())
|
||||||
|
|
||||||
|
const [autoScroll, setAutoScroll] = createSignal(Boolean(initialAutoScroll()))
|
||||||
|
const [showScrollTopButton, setShowScrollTopButton] = createSignal(false)
|
||||||
|
const [showScrollBottomButton, setShowScrollBottomButton] = createSignal(false)
|
||||||
|
const [topSentinelVisible, setTopSentinelVisible] = createSignal(true)
|
||||||
|
const [bottomSentinelVisible, setBottomSentinelVisible] = createSignal(true)
|
||||||
|
const [activeKey, setActiveKey] = createSignal<string | null>(null)
|
||||||
|
|
||||||
|
const [anchorLock, setAnchorLock] = createSignal<{ key: string; block: ScrollLogicalPosition } | null>(null)
|
||||||
|
|
||||||
|
const scrollButtonsCount = createMemo(() => (showScrollTopButton() ? 1 : 0) + (showScrollBottomButton() ? 1 : 0))
|
||||||
|
|
||||||
|
let containerRef: HTMLDivElement | undefined
|
||||||
|
let shellRef: HTMLDivElement | undefined
|
||||||
|
let pendingScrollFrame: number | null = null
|
||||||
|
let pendingAnchorScroll: number | null = null
|
||||||
|
let pendingAnchorCorrectionFrame: number | null = null
|
||||||
|
let pendingScrollCompensationScheduled = false
|
||||||
|
let pendingScrollCompensations = new Map<string, number>()
|
||||||
|
let scrollCompensationGen = 0
|
||||||
|
let pendingActiveScroll = false
|
||||||
|
let suppressAutoScrollOnce = false
|
||||||
|
let pendingInitialScroll = true
|
||||||
|
let scrollToBottomFrame: number | null = null
|
||||||
|
let scrollToBottomDelayedFrame: number | null = null
|
||||||
|
|
||||||
|
let lastKnownScrollTop = 0
|
||||||
|
let lastUserScrollIntentDirection: "up" | "down" | null = null
|
||||||
|
|
||||||
|
let userScrollIntentUntil = 0
|
||||||
|
let detachScrollIntentListeners: (() => void) | undefined
|
||||||
|
|
||||||
|
let lastResetKey: string | number | undefined
|
||||||
|
|
||||||
|
const state: VirtualFollowListState = {
|
||||||
|
autoScroll,
|
||||||
|
showScrollTopButton,
|
||||||
|
showScrollBottomButton,
|
||||||
|
scrollButtonsCount,
|
||||||
|
activeKey,
|
||||||
|
}
|
||||||
|
|
||||||
|
function markUserScrollIntent(direction?: "up" | "down" | null) {
|
||||||
|
const now = typeof performance !== "undefined" ? performance.now() : Date.now()
|
||||||
|
userScrollIntentUntil = now + USER_SCROLL_INTENT_WINDOW_MS
|
||||||
|
if (direction) {
|
||||||
|
lastUserScrollIntentDirection = direction
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function hasUserScrollIntent() {
|
||||||
|
const now = typeof performance !== "undefined" ? performance.now() : Date.now()
|
||||||
|
return now <= userScrollIntentUntil
|
||||||
|
}
|
||||||
|
|
||||||
|
function attachScrollIntentListeners(element: HTMLDivElement | undefined) {
|
||||||
|
if (detachScrollIntentListeners) {
|
||||||
|
detachScrollIntentListeners()
|
||||||
|
detachScrollIntentListeners = undefined
|
||||||
|
}
|
||||||
|
if (!element) return
|
||||||
|
const handleWheelIntent = (event: WheelEvent) => {
|
||||||
|
const dir: "up" | "down" | null = event.deltaY < 0 ? "up" : event.deltaY > 0 ? "down" : null
|
||||||
|
markUserScrollIntent(dir)
|
||||||
|
}
|
||||||
|
const handlePointerIntent = () => markUserScrollIntent(null)
|
||||||
|
const handleKeyIntent = (event: KeyboardEvent) => {
|
||||||
|
if (!SCROLL_INTENT_KEYS.has(event.key)) return
|
||||||
|
const key = event.key
|
||||||
|
const dir: "up" | "down" | null =
|
||||||
|
key === "ArrowUp" || key === "PageUp" || key === "Home"
|
||||||
|
? "up"
|
||||||
|
: key === "ArrowDown" || key === "PageDown" || key === "End"
|
||||||
|
? "down"
|
||||||
|
: key === " " || key === "Spacebar"
|
||||||
|
? event.shiftKey
|
||||||
|
? "up"
|
||||||
|
: "down"
|
||||||
|
: null
|
||||||
|
markUserScrollIntent(dir)
|
||||||
|
}
|
||||||
|
element.addEventListener("wheel", handleWheelIntent, { passive: true })
|
||||||
|
element.addEventListener("pointerdown", handlePointerIntent)
|
||||||
|
element.addEventListener("touchstart", handlePointerIntent, { passive: true })
|
||||||
|
element.addEventListener("keydown", handleKeyIntent)
|
||||||
|
detachScrollIntentListeners = () => {
|
||||||
|
element.removeEventListener("wheel", handleWheelIntent)
|
||||||
|
element.removeEventListener("pointerdown", handlePointerIntent)
|
||||||
|
element.removeEventListener("touchstart", handlePointerIntent)
|
||||||
|
element.removeEventListener("keydown", handleKeyIntent)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateScrollIndicatorsFromVisibility() {
|
||||||
|
const hasItems = props.items().length > 0
|
||||||
|
const bottomVisible = bottomSentinelVisible()
|
||||||
|
const topVisible = topSentinelVisible()
|
||||||
|
setShowScrollBottomButton(hasItems && !bottomVisible)
|
||||||
|
setShowScrollTopButton(hasItems && !topVisible)
|
||||||
|
}
|
||||||
|
|
||||||
|
function clearScrollToBottomFrames() {
|
||||||
|
if (scrollToBottomFrame !== null) {
|
||||||
|
cancelAnimationFrame(scrollToBottomFrame)
|
||||||
|
scrollToBottomFrame = null
|
||||||
|
}
|
||||||
|
if (scrollToBottomDelayedFrame !== null) {
|
||||||
|
cancelAnimationFrame(scrollToBottomDelayedFrame)
|
||||||
|
scrollToBottomDelayedFrame = null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function scrollToBottom(immediate = false, options?: { suppressAutoAnchor?: boolean }) {
|
||||||
|
if (!containerRef) return
|
||||||
|
if (anchorLock()) {
|
||||||
|
clearAnchorLock()
|
||||||
|
}
|
||||||
|
const sentinel = bottomSentinel()
|
||||||
|
const behavior: ScrollBehavior = immediate ? "auto" : "smooth"
|
||||||
|
const suppressAutoAnchor = options?.suppressAutoAnchor ?? !immediate
|
||||||
|
if (suppressAutoAnchor) {
|
||||||
|
suppressAutoScrollOnce = true
|
||||||
|
}
|
||||||
|
sentinel?.scrollIntoView({ block: "end", inline: "nearest", behavior })
|
||||||
|
setAutoScroll(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
function requestScrollToBottom(immediate = true) {
|
||||||
|
if (!isActive()) {
|
||||||
|
pendingActiveScroll = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (!containerRef || !bottomSentinel()) {
|
||||||
|
pendingActiveScroll = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
pendingActiveScroll = false
|
||||||
|
clearScrollToBottomFrames()
|
||||||
|
scrollToBottomFrame = requestAnimationFrame(() => {
|
||||||
|
scrollToBottomFrame = null
|
||||||
|
scrollToBottomDelayedFrame = requestAnimationFrame(() => {
|
||||||
|
scrollToBottomDelayedFrame = null
|
||||||
|
scrollToBottom(immediate)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolvePendingActiveScroll() {
|
||||||
|
if (!pendingActiveScroll) return
|
||||||
|
if (!isActive()) return
|
||||||
|
requestScrollToBottom(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
function scrollToTop(immediate = false) {
|
||||||
|
if (!containerRef) return
|
||||||
|
const behavior: ScrollBehavior = immediate ? "auto" : "smooth"
|
||||||
|
if (anchorLock()) {
|
||||||
|
clearAnchorLock()
|
||||||
|
}
|
||||||
|
setAutoScroll(false)
|
||||||
|
topSentinel()?.scrollIntoView({ block: "start", inline: "nearest", behavior })
|
||||||
|
}
|
||||||
|
|
||||||
|
function scheduleAnchorScroll(immediate = false) {
|
||||||
|
if (!autoScroll()) return
|
||||||
|
if (!isActive()) {
|
||||||
|
pendingActiveScroll = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const sentinel = bottomSentinel()
|
||||||
|
if (!sentinel) {
|
||||||
|
pendingActiveScroll = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (pendingAnchorScroll !== null) {
|
||||||
|
cancelAnimationFrame(pendingAnchorScroll)
|
||||||
|
pendingAnchorScroll = null
|
||||||
|
}
|
||||||
|
pendingAnchorScroll = requestAnimationFrame(() => {
|
||||||
|
pendingAnchorScroll = null
|
||||||
|
sentinel.scrollIntoView({ block: "end", inline: "nearest", behavior: immediate ? "auto" : "smooth" })
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function clearAnchorLock() {
|
||||||
|
setAnchorLock(null)
|
||||||
|
if (pendingAnchorCorrectionFrame !== null) {
|
||||||
|
cancelAnimationFrame(pendingAnchorCorrectionFrame)
|
||||||
|
pendingAnchorCorrectionFrame = null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function computeDesiredOffset(block: ScrollLogicalPosition, container: HTMLElement, anchorRect: DOMRect) {
|
||||||
|
if (block === "end") {
|
||||||
|
return Math.max(0, container.clientHeight - anchorRect.height)
|
||||||
|
}
|
||||||
|
if (block === "center") {
|
||||||
|
return Math.max(0, container.clientHeight / 2 - anchorRect.height / 2)
|
||||||
|
}
|
||||||
|
// Default to start.
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
function applyAnchorCorrection() {
|
||||||
|
const lock = anchorLock()
|
||||||
|
if (!lock) return
|
||||||
|
if (autoScroll()) return
|
||||||
|
if (!containerRef) return
|
||||||
|
if (typeof document === "undefined") return
|
||||||
|
|
||||||
|
const anchorId = getAnchorId(lock.key)
|
||||||
|
const anchor = document.getElementById(anchorId)
|
||||||
|
if (!anchor) return
|
||||||
|
|
||||||
|
const containerRect = containerRef.getBoundingClientRect()
|
||||||
|
const anchorRect = anchor.getBoundingClientRect()
|
||||||
|
const currentOffset = anchorRect.top - containerRect.top
|
||||||
|
const desiredOffset = computeDesiredOffset(lock.block, containerRef, anchorRect)
|
||||||
|
const delta = currentOffset - desiredOffset
|
||||||
|
if (!Number.isFinite(delta) || Math.abs(delta) < 0.5) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const nextTop = containerRef.scrollTop + delta
|
||||||
|
const maxScrollTop = Math.max(containerRef.scrollHeight - containerRef.clientHeight, 0)
|
||||||
|
containerRef.scrollTop = Math.min(maxScrollTop, Math.max(0, nextTop))
|
||||||
|
}
|
||||||
|
|
||||||
|
function scheduleAnchorCorrection() {
|
||||||
|
if (pendingAnchorCorrectionFrame !== null) return
|
||||||
|
pendingAnchorCorrectionFrame = requestAnimationFrame(() => {
|
||||||
|
pendingAnchorCorrectionFrame = null
|
||||||
|
applyAnchorCorrection()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleContentRendered() {
|
||||||
|
scheduleAnchorScroll()
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleScroll() {
|
||||||
|
if (!containerRef) return
|
||||||
|
if (pendingScrollFrame !== null) {
|
||||||
|
cancelAnimationFrame(pendingScrollFrame)
|
||||||
|
}
|
||||||
|
const isUserScroll = hasUserScrollIntent()
|
||||||
|
pendingScrollFrame = requestAnimationFrame(() => {
|
||||||
|
pendingScrollFrame = null
|
||||||
|
if (!containerRef) return
|
||||||
|
const previousScrollTop = lastKnownScrollTop
|
||||||
|
const currentScrollTop = containerRef.scrollTop
|
||||||
|
const deltaScrollTop = currentScrollTop - previousScrollTop
|
||||||
|
if (currentScrollTop !== lastKnownScrollTop) {
|
||||||
|
lastKnownScrollTop = currentScrollTop
|
||||||
|
}
|
||||||
|
const atBottom = bottomSentinelVisible()
|
||||||
|
|
||||||
|
const beforeAutoScroll = autoScroll()
|
||||||
|
|
||||||
|
const inferredDirection: "up" | "down" | null =
|
||||||
|
lastUserScrollIntentDirection ?? (deltaScrollTop < 0 ? "up" : deltaScrollTop > 0 ? "down" : null)
|
||||||
|
|
||||||
|
// If the user scrolls manually, exit key-anchored mode.
|
||||||
|
if (isUserScroll && anchorLock()) {
|
||||||
|
clearAnchorLock()
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isUserScroll) {
|
||||||
|
// If the user is actively scrolling upward, exit follow-to-bottom mode
|
||||||
|
// immediately. The bottom sentinel can remain "visible" for a short
|
||||||
|
// distance due to its observer margin, which otherwise keeps autoScroll
|
||||||
|
// enabled and makes the list feel stuck.
|
||||||
|
if (inferredDirection === "up" && deltaScrollTop < -0.5 && autoScroll()) {
|
||||||
|
if (pendingAnchorScroll !== null) {
|
||||||
|
cancelAnimationFrame(pendingAnchorScroll)
|
||||||
|
pendingAnchorScroll = null
|
||||||
|
}
|
||||||
|
setAutoScroll(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do not re-enable follow mode while the user's current scroll intent
|
||||||
|
// is upward. This prevents transient anchor/pin scrolls from pulling
|
||||||
|
// the list back into autoScroll(true).
|
||||||
|
if (inferredDirection !== "up") {
|
||||||
|
if (atBottom) {
|
||||||
|
if (!autoScroll()) setAutoScroll(true)
|
||||||
|
} else if (autoScroll()) {
|
||||||
|
setAutoScroll(false)
|
||||||
|
}
|
||||||
|
} else if (!atBottom && autoScroll()) {
|
||||||
|
// If the user is scrolling up and we are no longer at the bottom,
|
||||||
|
// ensure follow mode is disabled.
|
||||||
|
setAutoScroll(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
props.onScroll?.()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function setContainerRef(element: HTMLDivElement | null) {
|
||||||
|
containerRef = element || undefined
|
||||||
|
setScrollElement(containerRef)
|
||||||
|
props.onScrollElementChange?.(containerRef)
|
||||||
|
attachScrollIntentListeners(containerRef)
|
||||||
|
lastKnownScrollTop = containerRef?.scrollTop ?? 0
|
||||||
|
lastUserScrollIntentDirection = null
|
||||||
|
if (!containerRef) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
resolvePendingActiveScroll()
|
||||||
|
}
|
||||||
|
|
||||||
|
function scheduleScrollCompensation(key: string, delta: number) {
|
||||||
|
if (!containerRef) return
|
||||||
|
if (!delta || !Number.isFinite(delta)) return
|
||||||
|
if (typeof document === "undefined") return
|
||||||
|
|
||||||
|
// Only compensate while the user scrolls upward (testing default).
|
||||||
|
if (!hasUserScrollIntent() || lastUserScrollIntentDirection !== "up") return
|
||||||
|
if (autoScroll() || anchorLock()) return
|
||||||
|
|
||||||
|
const anchorId = getAnchorId(key)
|
||||||
|
const anchor = document.getElementById(anchorId)
|
||||||
|
if (!anchor) return
|
||||||
|
const containerRect = containerRef.getBoundingClientRect()
|
||||||
|
const rect = anchor.getBoundingClientRect()
|
||||||
|
// Determine whether the item was fully above the viewport *before* the
|
||||||
|
// height delta applied. Items can expand downward into the viewport; in that
|
||||||
|
// case we still need to compensate to keep existing visible content stable.
|
||||||
|
const bottomAfter = rect.bottom
|
||||||
|
const bottomBefore = bottomAfter - delta
|
||||||
|
const wasAboveViewport = bottomBefore < containerRect.top
|
||||||
|
if (!wasAboveViewport) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const next = (pendingScrollCompensations.get(key) ?? 0) + delta
|
||||||
|
pendingScrollCompensations.set(key, next)
|
||||||
|
|
||||||
|
if (pendingScrollCompensationScheduled) return
|
||||||
|
pendingScrollCompensationScheduled = true
|
||||||
|
const gen = scrollCompensationGen
|
||||||
|
|
||||||
|
// Flush in a microtask so compensation lands before the next paint.
|
||||||
|
queueMicrotask(() => {
|
||||||
|
if (gen !== scrollCompensationGen) return
|
||||||
|
pendingScrollCompensationScheduled = false
|
||||||
|
if (!containerRef) return
|
||||||
|
if (!hasUserScrollIntent() || lastUserScrollIntentDirection !== "up") {
|
||||||
|
pendingScrollCompensations = new Map()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (autoScroll() || anchorLock()) {
|
||||||
|
pendingScrollCompensations = new Map()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
let applied = 0
|
||||||
|
let count = 0
|
||||||
|
for (const pendingDelta of pendingScrollCompensations.values()) {
|
||||||
|
if (!pendingDelta) continue
|
||||||
|
applied += pendingDelta
|
||||||
|
count += 1
|
||||||
|
}
|
||||||
|
pendingScrollCompensations = new Map()
|
||||||
|
if (!applied) return
|
||||||
|
|
||||||
|
const before = containerRef.scrollTop
|
||||||
|
const maxScrollTop = Math.max(containerRef.scrollHeight - containerRef.clientHeight, 0)
|
||||||
|
const nextTop = Math.min(maxScrollTop, Math.max(0, before + applied))
|
||||||
|
if (nextTop !== before) {
|
||||||
|
containerRef.scrollTop = nextTop
|
||||||
|
lastKnownScrollTop = nextTop
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
let pendingAutoPin = false
|
||||||
|
function scheduleAutoPinToBottom() {
|
||||||
|
if (!containerRef) return
|
||||||
|
if (pendingAutoPin) return
|
||||||
|
pendingAutoPin = true
|
||||||
|
const gen = scrollCompensationGen
|
||||||
|
|
||||||
|
// Flush in a microtask so adjustments land before the next paint.
|
||||||
|
queueMicrotask(() => {
|
||||||
|
if (gen !== scrollCompensationGen) return
|
||||||
|
pendingAutoPin = false
|
||||||
|
if (!containerRef) return
|
||||||
|
if (!autoScroll()) return
|
||||||
|
if (anchorLock()) return
|
||||||
|
|
||||||
|
const maxScrollTop = Math.max(containerRef.scrollHeight - containerRef.clientHeight, 0)
|
||||||
|
if (containerRef.scrollTop !== maxScrollTop) {
|
||||||
|
containerRef.scrollTop = maxScrollTop
|
||||||
|
lastKnownScrollTop = maxScrollTop
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function setShellRef(element: HTMLDivElement | null) {
|
||||||
|
shellRef = element || undefined
|
||||||
|
setShellElement(shellRef)
|
||||||
|
props.onShellElementChange?.(shellRef)
|
||||||
|
}
|
||||||
|
|
||||||
|
function setBottomSentinel(element: HTMLDivElement | null) {
|
||||||
|
setBottomSentinelSignal(element)
|
||||||
|
resolvePendingActiveScroll()
|
||||||
|
}
|
||||||
|
|
||||||
|
const api: VirtualFollowListApi = {
|
||||||
|
scrollToTop: (opts) => scrollToTop(Boolean(opts?.immediate)),
|
||||||
|
scrollToBottom: (opts) => scrollToBottom(Boolean(opts?.immediate), { suppressAutoAnchor: opts?.suppressAutoAnchor }),
|
||||||
|
scrollToKey: (key, opts) => {
|
||||||
|
if (typeof document === "undefined") return
|
||||||
|
const anchorId = getAnchorId(key)
|
||||||
|
const behavior = opts?.behavior ?? "smooth"
|
||||||
|
const block = opts?.block ?? "start"
|
||||||
|
const nextAutoScroll = opts?.setAutoScroll ?? false
|
||||||
|
setAutoScroll(nextAutoScroll)
|
||||||
|
if (!nextAutoScroll) {
|
||||||
|
if (anchorLock()) {
|
||||||
|
clearAnchorLock()
|
||||||
|
}
|
||||||
|
setAnchorLock({ key, block })
|
||||||
|
} else {
|
||||||
|
if (anchorLock()) {
|
||||||
|
clearAnchorLock()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const first = document.getElementById(anchorId)
|
||||||
|
first?.scrollIntoView({ block, behavior })
|
||||||
|
// When using virtualization, the placeholder height can be stale until the
|
||||||
|
// item mounts/measures. Re-run scrollIntoView() on the next frame to
|
||||||
|
// stabilize the final position.
|
||||||
|
requestAnimationFrame(() => {
|
||||||
|
const second = document.getElementById(anchorId)
|
||||||
|
second?.scrollIntoView({ block, behavior })
|
||||||
|
})
|
||||||
|
},
|
||||||
|
notifyContentRendered: () => handleContentRendered(),
|
||||||
|
setAutoScroll: (enabled) => setAutoScroll(Boolean(enabled)),
|
||||||
|
getAutoScroll: () => autoScroll(),
|
||||||
|
getScrollElement: () => scrollElement(),
|
||||||
|
getShellElement: () => shellElement(),
|
||||||
|
}
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
props.registerApi?.(api)
|
||||||
|
})
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
props.registerState?.(state)
|
||||||
|
})
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
const nextKey = props.resetKey?.()
|
||||||
|
if (nextKey === undefined) return
|
||||||
|
if (lastResetKey === undefined) {
|
||||||
|
lastResetKey = nextKey
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (nextKey === lastResetKey) return
|
||||||
|
lastResetKey = nextKey
|
||||||
|
|
||||||
|
// Reset internal state when consumers swap datasets (e.g. session switch).
|
||||||
|
if (pendingScrollFrame !== null) {
|
||||||
|
cancelAnimationFrame(pendingScrollFrame)
|
||||||
|
pendingScrollFrame = null
|
||||||
|
}
|
||||||
|
if (pendingAnchorScroll !== null) {
|
||||||
|
cancelAnimationFrame(pendingAnchorScroll)
|
||||||
|
pendingAnchorScroll = null
|
||||||
|
}
|
||||||
|
if (pendingAnchorCorrectionFrame !== null) {
|
||||||
|
cancelAnimationFrame(pendingAnchorCorrectionFrame)
|
||||||
|
pendingAnchorCorrectionFrame = null
|
||||||
|
}
|
||||||
|
clearScrollToBottomFrames()
|
||||||
|
|
||||||
|
scrollCompensationGen += 1
|
||||||
|
pendingScrollCompensationScheduled = false
|
||||||
|
pendingScrollCompensations = new Map()
|
||||||
|
pendingAutoPin = false
|
||||||
|
|
||||||
|
suppressAutoScrollOnce = false
|
||||||
|
pendingActiveScroll = false
|
||||||
|
pendingInitialScroll = true
|
||||||
|
|
||||||
|
setAnchorLock(null)
|
||||||
|
setActiveKey(null)
|
||||||
|
setShowScrollTopButton(false)
|
||||||
|
setShowScrollBottomButton(false)
|
||||||
|
setTopSentinelVisible(true)
|
||||||
|
setBottomSentinelVisible(true)
|
||||||
|
setAutoScroll(Boolean(initialAutoScroll()))
|
||||||
|
|
||||||
|
lastKnownScrollTop = containerRef?.scrollTop ?? 0
|
||||||
|
lastUserScrollIntentDirection = null
|
||||||
|
})
|
||||||
|
|
||||||
|
let lastActiveState = false
|
||||||
|
createEffect(() => {
|
||||||
|
const active = isActive()
|
||||||
|
if (active) {
|
||||||
|
resolvePendingActiveScroll()
|
||||||
|
if (!lastActiveState && autoScroll() && scrollToBottomOnActivate()) {
|
||||||
|
requestScrollToBottom(true)
|
||||||
|
|
||||||
|
// When switching back to a cached session pane, items can mount/measure
|
||||||
|
// after the initial scroll jump. Re-pin once layout settles so the
|
||||||
|
// viewport stays at the bottom.
|
||||||
|
requestAnimationFrame(() => {
|
||||||
|
requestAnimationFrame(() => {
|
||||||
|
scheduleAutoPinToBottom()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
} else if (autoScroll() && scrollToBottomOnActivate()) {
|
||||||
|
pendingActiveScroll = true
|
||||||
|
}
|
||||||
|
lastActiveState = active
|
||||||
|
})
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
const loading = isLoading()
|
||||||
|
if (loading) {
|
||||||
|
// Keep the initial scroll pending while loading so we can
|
||||||
|
// anchor to the bottom as soon as items appear.
|
||||||
|
pendingInitialScroll = true
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!pendingInitialScroll) return
|
||||||
|
|
||||||
|
const container = scrollElement()
|
||||||
|
const sentinel = bottomSentinel()
|
||||||
|
if (!container || !sentinel || props.items().length === 0) return
|
||||||
|
|
||||||
|
if (!initialScrollToBottom()) {
|
||||||
|
// An outer component is responsible for restoring scroll.
|
||||||
|
pendingInitialScroll = false
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure we're in follow-to-bottom mode for the initial position.
|
||||||
|
if (anchorLock()) {
|
||||||
|
clearAnchorLock()
|
||||||
|
}
|
||||||
|
setAutoScroll(true)
|
||||||
|
|
||||||
|
pendingInitialScroll = false
|
||||||
|
// Scroll synchronously so the first paint prefers bottom content.
|
||||||
|
scrollToBottom(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
let previousFollowToken: string | number | undefined
|
||||||
|
createEffect(() => {
|
||||||
|
const token = props.followToken?.()
|
||||||
|
if (token === undefined) {
|
||||||
|
previousFollowToken = token
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (previousFollowToken === undefined) {
|
||||||
|
previousFollowToken = token
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (token === previousFollowToken) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
previousFollowToken = token
|
||||||
|
if (suppressAutoScrollOnce) {
|
||||||
|
suppressAutoScrollOnce = false
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (autoScroll()) scheduleAnchorScroll(true)
|
||||||
|
})
|
||||||
|
|
||||||
|
// Drop anchor lock if the anchored key is removed.
|
||||||
|
createEffect(() => {
|
||||||
|
const lock = anchorLock()
|
||||||
|
if (!lock) return
|
||||||
|
const keys = props.items().map((item, idx) => props.getKey(item, idx))
|
||||||
|
if (!keys.includes(lock.key)) {
|
||||||
|
clearAnchorLock()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
if (props.items().length === 0) {
|
||||||
|
setShowScrollTopButton(false)
|
||||||
|
setShowScrollBottomButton(false)
|
||||||
|
setAutoScroll(true)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
updateScrollIndicatorsFromVisibility()
|
||||||
|
})
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
const container = scrollElement()
|
||||||
|
const topTarget = topSentinel()
|
||||||
|
const bottomTarget = bottomSentinel()
|
||||||
|
if (!container || !topTarget || !bottomTarget) return
|
||||||
|
if (typeof IntersectionObserver === "undefined") return
|
||||||
|
|
||||||
|
const margin = props.scrollSentinelMarginPx ?? DEFAULT_SCROLL_SENTINEL_MARGIN_PX
|
||||||
|
|
||||||
|
const observer = new IntersectionObserver(
|
||||||
|
(entries) => {
|
||||||
|
let visibilityChanged = false
|
||||||
|
for (const entry of entries) {
|
||||||
|
if (entry.target === topTarget) {
|
||||||
|
setTopSentinelVisible(entry.isIntersecting)
|
||||||
|
visibilityChanged = true
|
||||||
|
} else if (entry.target === bottomTarget) {
|
||||||
|
setBottomSentinelVisible(entry.isIntersecting)
|
||||||
|
visibilityChanged = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (visibilityChanged) {
|
||||||
|
updateScrollIndicatorsFromVisibility()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ root: container, threshold: 0, rootMargin: `${margin}px 0px ${margin}px 0px` },
|
||||||
|
)
|
||||||
|
observer.observe(topTarget)
|
||||||
|
observer.observe(bottomTarget)
|
||||||
|
onCleanup(() => observer.disconnect())
|
||||||
|
})
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
const container = scrollElement()
|
||||||
|
const items = props.items()
|
||||||
|
if (!container || items.length === 0) return
|
||||||
|
if (typeof document === "undefined") return
|
||||||
|
if (typeof IntersectionObserver === "undefined") return
|
||||||
|
|
||||||
|
const observer = new IntersectionObserver(
|
||||||
|
(entries) => {
|
||||||
|
let best: IntersectionObserverEntry | null = null
|
||||||
|
for (const entry of entries) {
|
||||||
|
if (!entry.isIntersecting) continue
|
||||||
|
if (!best || entry.boundingClientRect.top < best.boundingClientRect.top) {
|
||||||
|
best = entry
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (best) {
|
||||||
|
const anchorId = (best.target as HTMLElement).id
|
||||||
|
const key = getKeyFromAnchorId(anchorId)
|
||||||
|
setActiveKey((current) => (current === key ? current : key))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ root: container, rootMargin: "-10% 0px -80% 0px", threshold: 0 },
|
||||||
|
)
|
||||||
|
|
||||||
|
const anchorIds = items.map((item, idx) => getAnchorId(props.getKey(item, idx)))
|
||||||
|
anchorIds.forEach((anchorId) => {
|
||||||
|
const anchor = document.getElementById(anchorId)
|
||||||
|
if (anchor) observer.observe(anchor)
|
||||||
|
})
|
||||||
|
|
||||||
|
onCleanup(() => observer.disconnect())
|
||||||
|
})
|
||||||
|
|
||||||
|
createEffect(() => {
|
||||||
|
const key = activeKey()
|
||||||
|
props.onActiveKeyChange?.(key)
|
||||||
|
})
|
||||||
|
|
||||||
|
onCleanup(() => {
|
||||||
|
if (pendingScrollFrame !== null) {
|
||||||
|
cancelAnimationFrame(pendingScrollFrame)
|
||||||
|
}
|
||||||
|
if (pendingAnchorScroll !== null) {
|
||||||
|
cancelAnimationFrame(pendingAnchorScroll)
|
||||||
|
}
|
||||||
|
if (pendingAnchorCorrectionFrame !== null) {
|
||||||
|
cancelAnimationFrame(pendingAnchorCorrectionFrame)
|
||||||
|
}
|
||||||
|
scrollCompensationGen += 1
|
||||||
|
pendingScrollCompensationScheduled = false
|
||||||
|
pendingScrollCompensations = new Map()
|
||||||
|
clearScrollToBottomFrames()
|
||||||
|
if (detachScrollIntentListeners) {
|
||||||
|
detachScrollIntentListeners()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const controls = () => {
|
||||||
|
if (props.renderControls) {
|
||||||
|
return props.renderControls(state, api)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Avoid hardcoded user-visible strings; require consumers to supply
|
||||||
|
// localized aria labels when using the default controls.
|
||||||
|
if (!props.scrollToTopAriaLabel || !props.scrollToBottomAriaLabel) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
const labelTop = props.scrollToTopAriaLabel()
|
||||||
|
const labelBottom = props.scrollToBottomAriaLabel()
|
||||||
|
return (
|
||||||
|
<Show when={showScrollTopButton() || showScrollBottomButton()}>
|
||||||
|
<div class="message-scroll-button-wrapper">
|
||||||
|
<Show when={showScrollTopButton()}>
|
||||||
|
<button type="button" class="message-scroll-button" onClick={() => scrollToTop()} aria-label={labelTop}>
|
||||||
|
<span class="message-scroll-icon" aria-hidden="true">
|
||||||
|
↑
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</Show>
|
||||||
|
<Show when={showScrollBottomButton()}>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="message-scroll-button"
|
||||||
|
onClick={() => scrollToBottom(false, { suppressAutoAnchor: false })}
|
||||||
|
aria-label={labelBottom}
|
||||||
|
>
|
||||||
|
<span class="message-scroll-icon" aria-hidden="true">
|
||||||
|
↓
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</Show>
|
||||||
|
</div>
|
||||||
|
</Show>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div class="message-stream-shell" ref={setShellRef}>
|
||||||
|
<div
|
||||||
|
class="message-stream"
|
||||||
|
ref={setContainerRef}
|
||||||
|
onScroll={handleScroll}
|
||||||
|
onMouseUp={(event) => props.onMouseUp?.(event)}
|
||||||
|
onClick={(event) => props.onClick?.(event)}
|
||||||
|
>
|
||||||
|
<div ref={setTopSentinel} aria-hidden="true" style={{ height: "1px" }} />
|
||||||
|
{props.renderBeforeItems?.()}
|
||||||
|
<Index each={props.items()}>
|
||||||
|
{(item, index) => {
|
||||||
|
const key = () => props.getKey(item(), index)
|
||||||
|
const anchorId = () => getAnchorId(key())
|
||||||
|
const overscanPx = props.overscanPx ?? 800
|
||||||
|
const suspendMeasurements = () => measurementsSuspended() || !isActive()
|
||||||
|
return (
|
||||||
|
<VirtualItem
|
||||||
|
id={anchorId()}
|
||||||
|
cacheKey={key()}
|
||||||
|
scrollContainer={scrollElement}
|
||||||
|
threshold={overscanPx}
|
||||||
|
placeholderClass="message-stream-placeholder"
|
||||||
|
virtualizationEnabled={virtualizationEnabled}
|
||||||
|
suspendMeasurements={suspendMeasurements}
|
||||||
|
onHeightChange={(nextHeight, previousHeight) => {
|
||||||
|
const delta = nextHeight - previousHeight
|
||||||
|
|
||||||
|
// Follow mode: keep the viewport pinned to the bottom as
|
||||||
|
// items mount/measure and change height.
|
||||||
|
if (delta && autoScroll() && !anchorLock()) {
|
||||||
|
scheduleAutoPinToBottom()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Key-anchored mode: keep the target key in view when
|
||||||
|
// items above it mount/measure and shift layout.
|
||||||
|
if (anchorLock() && !autoScroll()) {
|
||||||
|
scheduleAnchorCorrection()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Free-scroll mode: if items above the viewport change height
|
||||||
|
// while scrolling upward, compensate scrollTop so visible
|
||||||
|
// content stays stable.
|
||||||
|
if (delta) {
|
||||||
|
scheduleScrollCompensation(key(), delta)
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{() => props.renderItem(item(), index)}
|
||||||
|
</VirtualItem>
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
</Index>
|
||||||
|
<div ref={setBottomSentinel} aria-hidden="true" style={{ height: "1px" }} />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{controls()}
|
||||||
|
|
||||||
|
{props.renderOverlay?.()}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
import { JSX, Accessor, children as resolveChildren, createEffect, createMemo, createSignal, onCleanup } from "solid-js"
|
import { JSX, Accessor, createEffect, createMemo, createSignal, onCleanup } from "solid-js"
|
||||||
|
|
||||||
const sizeCache = new Map<string, number>()
|
const sizeCache = new Map<string, number>()
|
||||||
const DEFAULT_MARGIN_PX = 600
|
const DEFAULT_MARGIN_PX = 600
|
||||||
const MIN_PLACEHOLDER_HEIGHT = 32
|
const MIN_PLACEHOLDER_HEIGHT = 400
|
||||||
const VISIBILITY_BUFFER_PX = 48
|
const VISIBILITY_BUFFER_PX = 0
|
||||||
|
|
||||||
type ObserverRoot = Element | Document | null
|
type ObserverRoot = Element | Document | null
|
||||||
|
|
||||||
@@ -54,11 +54,64 @@ function shouldRenderEntry(entry: IntersectionObserverEntry) {
|
|||||||
if (!rootBounds) {
|
if (!rootBounds) {
|
||||||
return entry.isIntersecting
|
return entry.isIntersecting
|
||||||
}
|
}
|
||||||
const distanceAbove = rootBounds.top - entry.boundingClientRect.bottom
|
|
||||||
const distanceBelow = entry.boundingClientRect.top - rootBounds.bottom
|
// Above the root: compare bottom edge to root top.
|
||||||
if (distanceAbove > VISIBILITY_BUFFER_PX || distanceBelow > VISIBILITY_BUFFER_PX) {
|
if (entry.boundingClientRect.bottom < rootBounds.top) {
|
||||||
|
const distance = rootBounds.top - entry.boundingClientRect.bottom
|
||||||
|
return distance <= VISIBILITY_BUFFER_PX
|
||||||
|
}
|
||||||
|
|
||||||
|
// Below the root: compare top edge to root bottom.
|
||||||
|
if (entry.boundingClientRect.top > rootBounds.bottom) {
|
||||||
|
const distance = entry.boundingClientRect.top - rootBounds.bottom
|
||||||
|
return distance <= VISIBILITY_BUFFER_PX
|
||||||
|
}
|
||||||
|
|
||||||
|
// Overlapping the root bounds.
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
function getViewportRect(): { top: number; bottom: number } {
|
||||||
|
if (typeof window === "undefined") {
|
||||||
|
return { top: 0, bottom: 0 }
|
||||||
|
}
|
||||||
|
return { top: 0, bottom: window.innerHeight }
|
||||||
|
}
|
||||||
|
|
||||||
|
function isRenderableRoot(root: ObserverRoot): boolean {
|
||||||
|
if (!root) return true
|
||||||
|
if (root instanceof Document) return true
|
||||||
|
if (typeof window === "undefined") return false
|
||||||
|
|
||||||
|
const element = root as Element
|
||||||
|
const style = window.getComputedStyle(element as Element)
|
||||||
|
if (style.display === "none" || style.visibility === "hidden") {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
const rect = (element as Element).getBoundingClientRect()
|
||||||
|
return rect.width > 0 && rect.height > 0
|
||||||
|
}
|
||||||
|
|
||||||
|
function shouldRenderByRects(params: {
|
||||||
|
wrapperRect: DOMRect
|
||||||
|
rootRect: { top: number; bottom: number }
|
||||||
|
margin: number
|
||||||
|
}): boolean {
|
||||||
|
const { wrapperRect, rootRect, margin } = params
|
||||||
|
const threshold = margin + VISIBILITY_BUFFER_PX
|
||||||
|
|
||||||
|
// Above the root: compare bottom edge to root top.
|
||||||
|
if (wrapperRect.bottom < rootRect.top) {
|
||||||
|
const distance = rootRect.top - wrapperRect.bottom
|
||||||
|
return distance <= threshold
|
||||||
|
}
|
||||||
|
|
||||||
|
// Below the root: compare top edge to root bottom.
|
||||||
|
if (wrapperRect.top > rootRect.bottom) {
|
||||||
|
const distance = wrapperRect.top - rootRect.bottom
|
||||||
|
return distance <= threshold
|
||||||
|
}
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -103,7 +156,7 @@ function subscribeToSharedObserver(
|
|||||||
|
|
||||||
interface VirtualItemProps {
|
interface VirtualItemProps {
|
||||||
cacheKey: string
|
cacheKey: string
|
||||||
children: JSX.Element
|
children: JSX.Element | (() => JSX.Element)
|
||||||
scrollContainer?: Accessor<HTMLElement | undefined | null>
|
scrollContainer?: Accessor<HTMLElement | undefined | null>
|
||||||
threshold?: number
|
threshold?: number
|
||||||
minPlaceholderHeight?: number
|
minPlaceholderHeight?: number
|
||||||
@@ -114,14 +167,22 @@ interface VirtualItemProps {
|
|||||||
forceVisible?: Accessor<boolean>
|
forceVisible?: Accessor<boolean>
|
||||||
suspendMeasurements?: Accessor<boolean>
|
suspendMeasurements?: Accessor<boolean>
|
||||||
onMeasured?: () => void
|
onMeasured?: () => void
|
||||||
|
onHeightChange?: (nextHeight: number, previousHeight: number) => void
|
||||||
id?: string
|
id?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function VirtualItem(props: VirtualItemProps) {
|
export default function VirtualItem(props: VirtualItemProps) {
|
||||||
const resolved = resolveChildren(() => props.children)
|
const resolveContent = () => (typeof props.children === "function" ? (props.children as () => JSX.Element)() : props.children)
|
||||||
const cachedHeight = sizeCache.get(props.cacheKey)
|
const cachedHeight = sizeCache.get(props.cacheKey)
|
||||||
const [isIntersecting, setIsIntersecting] = createSignal(true)
|
const fallbackPlaceholderHeight = () => props.minPlaceholderHeight ?? MIN_PLACEHOLDER_HEIGHT
|
||||||
const [measuredHeight, setMeasuredHeight] = createSignal(cachedHeight ?? 0)
|
// Default to hidden until we can determine visibility.
|
||||||
|
// This avoids keeping heavy DOM alive when IntersectionObserver
|
||||||
|
// doesn't fire (common for hidden/zero-sized scroll roots).
|
||||||
|
const [isIntersecting, setIsIntersecting] = createSignal(false)
|
||||||
|
// Keep measuredHeight aligned with the *effective layout height* while hidden.
|
||||||
|
// When content first mounts, onHeightChange deltas should reflect the DOM's
|
||||||
|
// placeholder height (not 0), otherwise scroll compensation can overshoot.
|
||||||
|
const [measuredHeight, setMeasuredHeight] = createSignal(cachedHeight ?? fallbackPlaceholderHeight())
|
||||||
const [hasMeasured, setHasMeasured] = createSignal(cachedHeight !== undefined)
|
const [hasMeasured, setHasMeasured] = createSignal(cachedHeight !== undefined)
|
||||||
let hasReportedMeasurement = Boolean(cachedHeight && cachedHeight > 0)
|
let hasReportedMeasurement = Boolean(cachedHeight && cachedHeight > 0)
|
||||||
let pendingVisibility: boolean | null = null
|
let pendingVisibility: boolean | null = null
|
||||||
@@ -148,12 +209,12 @@ export default function VirtualItem(props: VirtualItemProps) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
const virtualizationEnabled = () => (props.virtualizationEnabled ? props.virtualizationEnabled() : true)
|
const virtualizationEnabled = () => (props.virtualizationEnabled ? props.virtualizationEnabled() : true)
|
||||||
|
const measurementsSuspended = () => Boolean(props.suspendMeasurements?.())
|
||||||
const shouldHideContent = createMemo(() => {
|
const shouldHideContent = createMemo(() => {
|
||||||
if (props.forceVisible?.()) return false
|
if (props.forceVisible?.()) return false
|
||||||
if (!virtualizationEnabled()) return false
|
if (!virtualizationEnabled()) return false
|
||||||
return !isIntersecting()
|
return !isIntersecting()
|
||||||
})
|
})
|
||||||
const measurementsSuspended = () => Boolean(props.suspendMeasurements?.())
|
|
||||||
|
|
||||||
let wrapperRef: HTMLDivElement | undefined
|
let wrapperRef: HTMLDivElement | undefined
|
||||||
|
|
||||||
@@ -180,9 +241,14 @@ export default function VirtualItem(props: VirtualItemProps) {
|
|||||||
if (!Number.isFinite(nextHeight) || nextHeight < 0) {
|
if (!Number.isFinite(nextHeight) || nextHeight < 0) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
const before = measuredHeight()
|
||||||
const normalized = nextHeight
|
const normalized = nextHeight
|
||||||
const previous = sizeCache.get(props.cacheKey) ?? measuredHeight()
|
const previous = sizeCache.get(props.cacheKey) ?? measuredHeight()
|
||||||
const shouldKeepPrevious = previous > 0 && (normalized === 0 || (normalized > 0 && normalized < previous))
|
// Only keep the previous measurement when the element reports 0 height.
|
||||||
|
// Allow shrinkage so placeholder height matches real content height;
|
||||||
|
// keeping the max height can cause mount/unmount jitter near the
|
||||||
|
// virtualization boundary.
|
||||||
|
const shouldKeepPrevious = previous > 0 && normalized === 0
|
||||||
if (shouldKeepPrevious) {
|
if (shouldKeepPrevious) {
|
||||||
if (!hasReportedMeasurement) {
|
if (!hasReportedMeasurement) {
|
||||||
hasReportedMeasurement = true
|
hasReportedMeasurement = true
|
||||||
@@ -191,6 +257,7 @@ export default function VirtualItem(props: VirtualItemProps) {
|
|||||||
setHasMeasured(true)
|
setHasMeasured(true)
|
||||||
sizeCache.set(props.cacheKey, previous)
|
sizeCache.set(props.cacheKey, previous)
|
||||||
setMeasuredHeight(previous)
|
setMeasuredHeight(previous)
|
||||||
|
if (previous !== before) props.onHeightChange?.(previous, before)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if (normalized > 0) {
|
if (normalized > 0) {
|
||||||
@@ -202,11 +269,15 @@ export default function VirtualItem(props: VirtualItemProps) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
setMeasuredHeight(normalized)
|
setMeasuredHeight(normalized)
|
||||||
|
if (normalized !== before) props.onHeightChange?.(normalized, before)
|
||||||
}
|
}
|
||||||
|
|
||||||
function updateMeasuredHeight() {
|
function updateMeasuredHeight() {
|
||||||
if (!contentRef || measurementsSuspended()) return
|
if (!contentRef || measurementsSuspended()) return
|
||||||
const next = contentRef.offsetHeight
|
// Prefer subpixel-accurate height for scroll compensation.
|
||||||
|
// offsetHeight rounds to integers which can accumulate error.
|
||||||
|
const rect = contentRef.getBoundingClientRect()
|
||||||
|
const next = Math.max(0, Math.round(rect.height * 2) / 2)
|
||||||
if (next === measuredHeight()) return
|
if (next === measuredHeight()) return
|
||||||
persistMeasurement(next)
|
persistMeasurement(next)
|
||||||
}
|
}
|
||||||
@@ -229,15 +300,60 @@ export default function VirtualItem(props: VirtualItemProps) {
|
|||||||
function refreshIntersectionObserver(targetRoot: Element | Document | null) {
|
function refreshIntersectionObserver(targetRoot: Element | Document | null) {
|
||||||
cleanupIntersectionObserver()
|
cleanupIntersectionObserver()
|
||||||
if (!wrapperRef) {
|
if (!wrapperRef) {
|
||||||
setIsIntersecting(true)
|
setIsIntersecting(false)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if (typeof IntersectionObserver === "undefined") {
|
if (typeof IntersectionObserver === "undefined") {
|
||||||
setIsIntersecting(true)
|
setIsIntersecting(true)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
const margin = props.threshold ?? DEFAULT_MARGIN_PX
|
const margin = props.threshold ?? DEFAULT_MARGIN_PX
|
||||||
intersectionCleanup = subscribeToSharedObserver(wrapperRef, targetRoot, margin, (entry) => {
|
|
||||||
|
// If the scroll root is hidden / 0x0, IntersectionObserver can report
|
||||||
|
// `isIntersecting` in unexpected ways (often "true" with null rootBounds),
|
||||||
|
// which keeps heavy DOM alive in background tabs.
|
||||||
|
//
|
||||||
|
// In that state, force-hide and skip attaching the observer. When the
|
||||||
|
// pane becomes visible again, VirtualItem will re-run this setup and
|
||||||
|
// re-attach the observer.
|
||||||
|
const renderable = isRenderableRoot(targetRoot)
|
||||||
|
if (!renderable) {
|
||||||
|
setIsIntersecting(false)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Avoid doing an eager geometry read here.
|
||||||
|
// During large list hydration / initial layout, wrapper rects can be
|
||||||
|
// transiently 0/incorrect and cause many offscreen items to mount.
|
||||||
|
// Rely on the observer callback (which we harden below) to determine
|
||||||
|
// visibility.
|
||||||
|
|
||||||
|
const wrapperEl = wrapperRef
|
||||||
|
intersectionCleanup = subscribeToSharedObserver(wrapperEl, targetRoot, margin, (entry) => {
|
||||||
|
// IntersectionObserver can produce transient false-positives during pane
|
||||||
|
// activation/layout transitions (e.g. `isIntersecting: true` for items far
|
||||||
|
// outside the scroll root). For element roots, prefer explicit rect math.
|
||||||
|
if (targetRoot && !(targetRoot instanceof Document)) {
|
||||||
|
// When rootBounds is null we cannot trust the entry; treat as hidden.
|
||||||
|
if (entry.rootBounds === null) {
|
||||||
|
queueVisibility(false)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
const rootRect = (targetRoot as Element).getBoundingClientRect()
|
||||||
|
const visible = shouldRenderByRects({
|
||||||
|
wrapperRect: wrapperEl.getBoundingClientRect(),
|
||||||
|
rootRect: { top: rootRect.top, bottom: rootRect.bottom },
|
||||||
|
margin,
|
||||||
|
})
|
||||||
|
queueVisibility(visible)
|
||||||
|
return
|
||||||
|
} catch {
|
||||||
|
// Fall through to the entry-based heuristic.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const nextVisible = shouldRenderEntry(entry)
|
const nextVisible = shouldRenderEntry(entry)
|
||||||
queueVisibility(nextVisible)
|
queueVisibility(nextVisible)
|
||||||
})
|
})
|
||||||
@@ -283,7 +399,7 @@ export default function VirtualItem(props: VirtualItemProps) {
|
|||||||
setMeasuredHeight(cached)
|
setMeasuredHeight(cached)
|
||||||
setHasMeasured(true)
|
setHasMeasured(true)
|
||||||
} else {
|
} else {
|
||||||
setMeasuredHeight(0)
|
setMeasuredHeight(fallbackPlaceholderHeight())
|
||||||
setHasMeasured(false)
|
setHasMeasured(false)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@@ -320,10 +436,9 @@ export default function VirtualItem(props: VirtualItemProps) {
|
|||||||
const placeholderClass = () => ["virtual-item-placeholder", props.placeholderClass].filter(Boolean).join(" ")
|
const placeholderClass = () => ["virtual-item-placeholder", props.placeholderClass].filter(Boolean).join(" ")
|
||||||
const lazyContent = createMemo<JSX.Element | null>(() => {
|
const lazyContent = createMemo<JSX.Element | null>(() => {
|
||||||
if (shouldHideContent()) return null
|
if (shouldHideContent()) return null
|
||||||
return resolved()
|
return resolveContent()
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div ref={setWrapperRef} id={props.id} class={wrapperClass()} style={{ width: "100%" }}>
|
<div ref={setWrapperRef} id={props.id} class={wrapperClass()} style={{ width: "100%" }}>
|
||||||
<div
|
<div
|
||||||
@@ -340,4 +455,3 @@ export default function VirtualItem(props: VirtualItemProps) {
|
|||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -23,7 +23,6 @@ export const messagingMessages = {
|
|||||||
"messageSection.quote.copy": "Copy",
|
"messageSection.quote.copy": "Copy",
|
||||||
"messageSection.quote.copied": "Copied!",
|
"messageSection.quote.copied": "Copied!",
|
||||||
"messageSection.quote.copyFailed": "Copy failed",
|
"messageSection.quote.copyFailed": "Copy failed",
|
||||||
|
|
||||||
"messageTimeline.ariaLabel": "Message timeline",
|
"messageTimeline.ariaLabel": "Message timeline",
|
||||||
"messageTimeline.segment.user.label": "You",
|
"messageTimeline.segment.user.label": "You",
|
||||||
"messageTimeline.segment.assistant.label": "Asst",
|
"messageTimeline.segment.assistant.label": "Asst",
|
||||||
@@ -35,13 +34,12 @@ export const messagingMessages = {
|
|||||||
"messageTimeline.tooltip.compaction.manual": "User Compaction",
|
"messageTimeline.tooltip.compaction.manual": "User Compaction",
|
||||||
"messageTimeline.text.filePrefix": "[File] {filename}",
|
"messageTimeline.text.filePrefix": "[File] {filename}",
|
||||||
"messageTimeline.text.attachment": "Attachment",
|
"messageTimeline.text.attachment": "Attachment",
|
||||||
|
|
||||||
"messageBlock.tool.header": "Tool Call",
|
"messageBlock.tool.header": "Tool Call",
|
||||||
"messageBlock.tool.unknown": "unknown",
|
"messageBlock.tool.unknown": "unknown",
|
||||||
"messageBlock.tool.goToSession.label": "Go to Session",
|
"messageBlock.tool.goToSession.label": "Go to Session",
|
||||||
"messageBlock.tool.goToSession.title": "Go to session",
|
"messageBlock.tool.goToSession.title": "Go to session",
|
||||||
"messageBlock.tool.goToSession.unavailableTitle": "Session not available yet",
|
"messageBlock.tool.goToSession.unavailableTitle": "Session not available yet",
|
||||||
"messageBlock.tool.deletePart.label": "Delete",
|
"messageBlock.tool.deletePart.label": "Delete Part",
|
||||||
"messageBlock.tool.deletePart.deleting": "Deleting...",
|
"messageBlock.tool.deletePart.deleting": "Deleting...",
|
||||||
"messageBlock.tool.deletePart.title": "Delete this tool call output",
|
"messageBlock.tool.deletePart.title": "Delete this tool call output",
|
||||||
"messageBlock.tool.deletePart.failed.title": "Delete failed",
|
"messageBlock.tool.deletePart.failed.title": "Delete failed",
|
||||||
@@ -71,17 +69,38 @@ export const messagingMessages = {
|
|||||||
"messageItem.speaker.you": "You",
|
"messageItem.speaker.you": "You",
|
||||||
"messageItem.speaker.assistant": "Assistant",
|
"messageItem.speaker.assistant": "Assistant",
|
||||||
"messageItem.actions.revert": "Revert",
|
"messageItem.actions.revert": "Revert",
|
||||||
"messageItem.actions.revertTitle": "Revert to this message",
|
"messageItem.actions.revertTitle": "Undo changes up to here (deletes messages)",
|
||||||
"messageItem.actions.fork": "Fork",
|
"messageItem.actions.fork": "Fork",
|
||||||
"messageItem.actions.forkTitle": "Fork from this message",
|
"messageItem.actions.forkTitle": "Fork from this message",
|
||||||
"messageItem.actions.copy": "Copy",
|
"messageItem.actions.copy": "Copy",
|
||||||
"messageItem.actions.copyTitle": "Copy message",
|
"messageItem.actions.copyTitle": "Copy message",
|
||||||
"messageItem.actions.copied": "Copied!",
|
"messageItem.actions.copied": "Copied!",
|
||||||
|
"messageItem.actions.deleteMessage": "Delete message (doesn't undo changes)",
|
||||||
|
"messageItem.actions.deleteMessagesUpTo": "Delete messages up to here (doesn't undo changes)",
|
||||||
|
"messageItem.actions.deletingMessage": "Deleting...",
|
||||||
|
"messageItem.actions.deleteMessageFailedTitle": "Delete failed",
|
||||||
|
"messageItem.actions.deleteMessageFailedMessage": "Failed to delete message",
|
||||||
|
|
||||||
|
"messageItem.selection.checkboxAriaLabel": "Select message for deletion",
|
||||||
|
|
||||||
|
"messageSection.bulkDelete.toolbarAriaLabel": "Selected items ({count})",
|
||||||
|
"messageSection.bulkDelete.deleteSelectedTitle": "Delete selected items",
|
||||||
|
"messageSection.bulkDelete.selectAllTitle": "Select all messages",
|
||||||
|
"messageSection.bulkDelete.moreOptionsTitle": "More options",
|
||||||
|
"messageSection.bulkDelete.selectionModeLabel": "Selection",
|
||||||
|
"messageSection.bulkDelete.selectionModeAll": "All",
|
||||||
|
"messageSection.bulkDelete.selectionModeTools": "Tools only",
|
||||||
|
"messageSection.bulkDelete.selectionHint.toggle": "Select item",
|
||||||
|
"messageSection.bulkDelete.selectionHint.range": "Select range",
|
||||||
|
"messageSection.bulkDelete.selectionHint.clear": "Clear Selection",
|
||||||
|
"messageSection.bulkDelete.cancelTitle": "Cancel selection",
|
||||||
|
"messageSection.bulkDelete.failedTitle": "Delete failed",
|
||||||
|
"messageSection.bulkDelete.failedMessage": "Failed to delete selected items",
|
||||||
"messageItem.status.queued": "QUEUED",
|
"messageItem.status.queued": "QUEUED",
|
||||||
"messageItem.status.generating": "Generating...",
|
"messageItem.status.generating": "Generating...",
|
||||||
"messageItem.status.sending": "Sending...",
|
"messageItem.status.sending": "Sending...",
|
||||||
"messageItem.status.failedToSend": "Message failed to send",
|
"messageItem.status.failedToSend": "Message failed to send",
|
||||||
"messagePart.actions.delete": "Delete",
|
"messagePart.actions.delete": "Delete Part",
|
||||||
"messagePart.actions.deleting": "Deleting...",
|
"messagePart.actions.deleting": "Deleting...",
|
||||||
"messagePart.actions.deleteTitle": "Delete this item",
|
"messagePart.actions.deleteTitle": "Delete this item",
|
||||||
"messagePart.actions.deleteFailedTitle": "Delete failed",
|
"messagePart.actions.deleteFailedTitle": "Delete failed",
|
||||||
|
|||||||
@@ -67,6 +67,8 @@ export const sessionMessages = {
|
|||||||
"sessionView.alerts.abortFailed.title": "Stop failed",
|
"sessionView.alerts.abortFailed.title": "Stop failed",
|
||||||
"sessionView.alerts.revertFailed.message": "Failed to revert to message",
|
"sessionView.alerts.revertFailed.message": "Failed to revert to message",
|
||||||
"sessionView.alerts.revertFailed.title": "Revert failed",
|
"sessionView.alerts.revertFailed.title": "Revert failed",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.message": "Failed to delete messages",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.title": "Delete failed",
|
||||||
"sessionView.alerts.forkFailed.message": "Failed to fork session",
|
"sessionView.alerts.forkFailed.message": "Failed to fork session",
|
||||||
"sessionView.alerts.forkFailed.title": "Fork failed",
|
"sessionView.alerts.forkFailed.title": "Fork failed",
|
||||||
"sessionView.attachments.expandPastedTextAriaLabel": "Expand pasted text",
|
"sessionView.attachments.expandPastedTextAriaLabel": "Expand pasted text",
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ export const messagingMessages = {
|
|||||||
"messageBlock.tool.goToSession.label": "Ir a sesión",
|
"messageBlock.tool.goToSession.label": "Ir a sesión",
|
||||||
"messageBlock.tool.goToSession.title": "Ir a la sesión",
|
"messageBlock.tool.goToSession.title": "Ir a la sesión",
|
||||||
"messageBlock.tool.goToSession.unavailableTitle": "La sesión aún no está disponible",
|
"messageBlock.tool.goToSession.unavailableTitle": "La sesión aún no está disponible",
|
||||||
"messageBlock.tool.deletePart.label": "Eliminar",
|
"messageBlock.tool.deletePart.label": "Eliminar parte",
|
||||||
"messageBlock.tool.deletePart.deleting": "Eliminando...",
|
"messageBlock.tool.deletePart.deleting": "Eliminando...",
|
||||||
"messageBlock.tool.deletePart.title": "Eliminar esta salida de herramienta",
|
"messageBlock.tool.deletePart.title": "Eliminar esta salida de herramienta",
|
||||||
"messageBlock.tool.deletePart.failed.title": "Error al eliminar",
|
"messageBlock.tool.deletePart.failed.title": "Error al eliminar",
|
||||||
@@ -71,17 +71,38 @@ export const messagingMessages = {
|
|||||||
"messageItem.speaker.you": "Tú",
|
"messageItem.speaker.you": "Tú",
|
||||||
"messageItem.speaker.assistant": "Asistente",
|
"messageItem.speaker.assistant": "Asistente",
|
||||||
"messageItem.actions.revert": "Revertir",
|
"messageItem.actions.revert": "Revertir",
|
||||||
"messageItem.actions.revertTitle": "Revertir a este mensaje",
|
"messageItem.actions.revertTitle": "Deshacer cambios hasta aqui (elimina mensajes)",
|
||||||
"messageItem.actions.fork": "Fork",
|
"messageItem.actions.fork": "Fork",
|
||||||
"messageItem.actions.forkTitle": "Fork desde este mensaje",
|
"messageItem.actions.forkTitle": "Fork desde este mensaje",
|
||||||
"messageItem.actions.copy": "Copiar",
|
"messageItem.actions.copy": "Copiar",
|
||||||
"messageItem.actions.copyTitle": "Copiar mensaje",
|
"messageItem.actions.copyTitle": "Copiar mensaje",
|
||||||
"messageItem.actions.copied": "¡Copiado!",
|
"messageItem.actions.copied": "¡Copiado!",
|
||||||
|
"messageItem.actions.deleteMessage": "Eliminar mensaje (no deshace cambios)",
|
||||||
|
"messageItem.actions.deleteMessagesUpTo": "Eliminar mensajes hasta aqui (no deshace cambios)",
|
||||||
|
"messageItem.actions.deletingMessage": "Eliminando...",
|
||||||
|
"messageItem.actions.deleteMessageFailedTitle": "Error al eliminar",
|
||||||
|
"messageItem.actions.deleteMessageFailedMessage": "No se pudo eliminar el mensaje",
|
||||||
|
|
||||||
|
"messageItem.selection.checkboxAriaLabel": "Seleccionar mensaje para eliminar",
|
||||||
|
|
||||||
|
"messageSection.bulkDelete.toolbarAriaLabel": "Elementos seleccionados ({count})",
|
||||||
|
"messageSection.bulkDelete.deleteSelectedTitle": "Eliminar elementos seleccionados",
|
||||||
|
"messageSection.bulkDelete.selectAllTitle": "Seleccionar todos los mensajes",
|
||||||
|
"messageSection.bulkDelete.moreOptionsTitle": "Más opciones",
|
||||||
|
"messageSection.bulkDelete.selectionModeLabel": "Selección",
|
||||||
|
"messageSection.bulkDelete.selectionModeAll": "Todo",
|
||||||
|
"messageSection.bulkDelete.selectionModeTools": "Solo herramientas",
|
||||||
|
"messageSection.bulkDelete.selectionHint.toggle": "Seleccionar elemento",
|
||||||
|
"messageSection.bulkDelete.selectionHint.range": "Seleccionar rango",
|
||||||
|
"messageSection.bulkDelete.selectionHint.clear": "Borrar selección",
|
||||||
|
"messageSection.bulkDelete.cancelTitle": "Cancelar selección",
|
||||||
|
"messageSection.bulkDelete.failedTitle": "Error al eliminar",
|
||||||
|
"messageSection.bulkDelete.failedMessage": "No se pudieron eliminar los elementos seleccionados",
|
||||||
"messageItem.status.queued": "EN COLA",
|
"messageItem.status.queued": "EN COLA",
|
||||||
"messageItem.status.generating": "Generando...",
|
"messageItem.status.generating": "Generando...",
|
||||||
"messageItem.status.sending": "Enviando...",
|
"messageItem.status.sending": "Enviando...",
|
||||||
"messageItem.status.failedToSend": "No se pudo enviar el mensaje",
|
"messageItem.status.failedToSend": "No se pudo enviar el mensaje",
|
||||||
"messagePart.actions.delete": "Eliminar",
|
"messagePart.actions.delete": "Eliminar parte",
|
||||||
"messagePart.actions.deleting": "Eliminando...",
|
"messagePart.actions.deleting": "Eliminando...",
|
||||||
"messagePart.actions.deleteTitle": "Eliminar este elemento",
|
"messagePart.actions.deleteTitle": "Eliminar este elemento",
|
||||||
"messagePart.actions.deleteFailedTitle": "Error al eliminar",
|
"messagePart.actions.deleteFailedTitle": "Error al eliminar",
|
||||||
|
|||||||
@@ -67,6 +67,8 @@ export const sessionMessages = {
|
|||||||
"sessionView.alerts.abortFailed.title": "No se pudo detener",
|
"sessionView.alerts.abortFailed.title": "No se pudo detener",
|
||||||
"sessionView.alerts.revertFailed.message": "No se pudo revertir al mensaje",
|
"sessionView.alerts.revertFailed.message": "No se pudo revertir al mensaje",
|
||||||
"sessionView.alerts.revertFailed.title": "No se pudo revertir",
|
"sessionView.alerts.revertFailed.title": "No se pudo revertir",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.message": "No se pudieron eliminar los mensajes",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.title": "Error al eliminar",
|
||||||
"sessionView.alerts.forkFailed.message": "No se pudo hacer fork de la sesión",
|
"sessionView.alerts.forkFailed.message": "No se pudo hacer fork de la sesión",
|
||||||
"sessionView.alerts.forkFailed.title": "No se pudo hacer fork",
|
"sessionView.alerts.forkFailed.title": "No se pudo hacer fork",
|
||||||
"sessionView.attachments.expandPastedTextAriaLabel": "Expandir texto pegado",
|
"sessionView.attachments.expandPastedTextAriaLabel": "Expandir texto pegado",
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ export const messagingMessages = {
|
|||||||
"messageBlock.tool.goToSession.label": "Aller à la session",
|
"messageBlock.tool.goToSession.label": "Aller à la session",
|
||||||
"messageBlock.tool.goToSession.title": "Aller à la session",
|
"messageBlock.tool.goToSession.title": "Aller à la session",
|
||||||
"messageBlock.tool.goToSession.unavailableTitle": "Session pas encore disponible",
|
"messageBlock.tool.goToSession.unavailableTitle": "Session pas encore disponible",
|
||||||
"messageBlock.tool.deletePart.label": "Supprimer",
|
"messageBlock.tool.deletePart.label": "Supprimer la partie",
|
||||||
"messageBlock.tool.deletePart.deleting": "Suppression...",
|
"messageBlock.tool.deletePart.deleting": "Suppression...",
|
||||||
"messageBlock.tool.deletePart.title": "Supprimer cette sortie d'outil",
|
"messageBlock.tool.deletePart.title": "Supprimer cette sortie d'outil",
|
||||||
"messageBlock.tool.deletePart.failed.title": "Échec de suppression",
|
"messageBlock.tool.deletePart.failed.title": "Échec de suppression",
|
||||||
@@ -71,17 +71,38 @@ export const messagingMessages = {
|
|||||||
"messageItem.speaker.you": "Vous",
|
"messageItem.speaker.you": "Vous",
|
||||||
"messageItem.speaker.assistant": "Assistant",
|
"messageItem.speaker.assistant": "Assistant",
|
||||||
"messageItem.actions.revert": "Revenir",
|
"messageItem.actions.revert": "Revenir",
|
||||||
"messageItem.actions.revertTitle": "Revenir à ce message",
|
"messageItem.actions.revertTitle": "Annuler les changements jusqu'ici (supprime les messages)",
|
||||||
"messageItem.actions.fork": "Fork",
|
"messageItem.actions.fork": "Fork",
|
||||||
"messageItem.actions.forkTitle": "Fork depuis ce message",
|
"messageItem.actions.forkTitle": "Fork depuis ce message",
|
||||||
"messageItem.actions.copy": "Copier",
|
"messageItem.actions.copy": "Copier",
|
||||||
"messageItem.actions.copyTitle": "Copier le message",
|
"messageItem.actions.copyTitle": "Copier le message",
|
||||||
"messageItem.actions.copied": "Copié !",
|
"messageItem.actions.copied": "Copié !",
|
||||||
|
"messageItem.actions.deleteMessage": "Supprimer le message (sans annuler les changements)",
|
||||||
|
"messageItem.actions.deleteMessagesUpTo": "Supprimer les messages jusqu'ici (sans annuler les changements)",
|
||||||
|
"messageItem.actions.deletingMessage": "Suppression...",
|
||||||
|
"messageItem.actions.deleteMessageFailedTitle": "Échec de suppression",
|
||||||
|
"messageItem.actions.deleteMessageFailedMessage": "Impossible de supprimer le message",
|
||||||
|
|
||||||
|
"messageItem.selection.checkboxAriaLabel": "Sélectionner le message pour suppression",
|
||||||
|
|
||||||
|
"messageSection.bulkDelete.toolbarAriaLabel": "Éléments sélectionnés ({count})",
|
||||||
|
"messageSection.bulkDelete.deleteSelectedTitle": "Supprimer les éléments sélectionnés",
|
||||||
|
"messageSection.bulkDelete.selectAllTitle": "Tout sélectionner",
|
||||||
|
"messageSection.bulkDelete.moreOptionsTitle": "Plus d'options",
|
||||||
|
"messageSection.bulkDelete.selectionModeLabel": "Sélection",
|
||||||
|
"messageSection.bulkDelete.selectionModeAll": "Tous",
|
||||||
|
"messageSection.bulkDelete.selectionModeTools": "Outils uniquement",
|
||||||
|
"messageSection.bulkDelete.selectionHint.toggle": "Selectionner un element",
|
||||||
|
"messageSection.bulkDelete.selectionHint.range": "Selectionner une plage",
|
||||||
|
"messageSection.bulkDelete.selectionHint.clear": "Effacer la selection",
|
||||||
|
"messageSection.bulkDelete.cancelTitle": "Annuler la sélection",
|
||||||
|
"messageSection.bulkDelete.failedTitle": "Échec de suppression",
|
||||||
|
"messageSection.bulkDelete.failedMessage": "Impossible de supprimer les éléments sélectionnés",
|
||||||
"messageItem.status.queued": "EN FILE",
|
"messageItem.status.queued": "EN FILE",
|
||||||
"messageItem.status.generating": "Génération...",
|
"messageItem.status.generating": "Génération...",
|
||||||
"messageItem.status.sending": "Envoi...",
|
"messageItem.status.sending": "Envoi...",
|
||||||
"messageItem.status.failedToSend": "Échec de l'envoi du message",
|
"messageItem.status.failedToSend": "Échec de l'envoi du message",
|
||||||
"messagePart.actions.delete": "Supprimer",
|
"messagePart.actions.delete": "Supprimer la partie",
|
||||||
"messagePart.actions.deleting": "Suppression...",
|
"messagePart.actions.deleting": "Suppression...",
|
||||||
"messagePart.actions.deleteTitle": "Supprimer cet élément",
|
"messagePart.actions.deleteTitle": "Supprimer cet élément",
|
||||||
"messagePart.actions.deleteFailedTitle": "Échec de suppression",
|
"messagePart.actions.deleteFailedTitle": "Échec de suppression",
|
||||||
|
|||||||
@@ -67,6 +67,8 @@ export const sessionMessages = {
|
|||||||
"sessionView.alerts.abortFailed.title": "Échec de l'arrêt",
|
"sessionView.alerts.abortFailed.title": "Échec de l'arrêt",
|
||||||
"sessionView.alerts.revertFailed.message": "Impossible de revenir au message",
|
"sessionView.alerts.revertFailed.message": "Impossible de revenir au message",
|
||||||
"sessionView.alerts.revertFailed.title": "Échec du retour",
|
"sessionView.alerts.revertFailed.title": "Échec du retour",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.message": "Impossible de supprimer les messages",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.title": "Échec de suppression",
|
||||||
"sessionView.alerts.forkFailed.message": "Impossible de forker la session",
|
"sessionView.alerts.forkFailed.message": "Impossible de forker la session",
|
||||||
"sessionView.alerts.forkFailed.title": "Échec du fork",
|
"sessionView.alerts.forkFailed.title": "Échec du fork",
|
||||||
"sessionView.attachments.expandPastedTextAriaLabel": "Développer le texte collé",
|
"sessionView.attachments.expandPastedTextAriaLabel": "Développer le texte collé",
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ export const messagingMessages = {
|
|||||||
"messageBlock.tool.goToSession.label": "セッションへ移動",
|
"messageBlock.tool.goToSession.label": "セッションへ移動",
|
||||||
"messageBlock.tool.goToSession.title": "セッションへ移動",
|
"messageBlock.tool.goToSession.title": "セッションへ移動",
|
||||||
"messageBlock.tool.goToSession.unavailableTitle": "セッションはまだ利用できません",
|
"messageBlock.tool.goToSession.unavailableTitle": "セッションはまだ利用できません",
|
||||||
"messageBlock.tool.deletePart.label": "削除",
|
"messageBlock.tool.deletePart.label": "パートを削除",
|
||||||
"messageBlock.tool.deletePart.deleting": "削除中...",
|
"messageBlock.tool.deletePart.deleting": "削除中...",
|
||||||
"messageBlock.tool.deletePart.title": "このツール出力を削除",
|
"messageBlock.tool.deletePart.title": "このツール出力を削除",
|
||||||
"messageBlock.tool.deletePart.failed.title": "削除に失敗しました",
|
"messageBlock.tool.deletePart.failed.title": "削除に失敗しました",
|
||||||
@@ -71,17 +71,38 @@ export const messagingMessages = {
|
|||||||
"messageItem.speaker.you": "あなた",
|
"messageItem.speaker.you": "あなた",
|
||||||
"messageItem.speaker.assistant": "アシスタント",
|
"messageItem.speaker.assistant": "アシスタント",
|
||||||
"messageItem.actions.revert": "戻す",
|
"messageItem.actions.revert": "戻す",
|
||||||
"messageItem.actions.revertTitle": "このメッセージまで戻す",
|
"messageItem.actions.revertTitle": "ここまでの変更を元に戻す(メッセージを削除)",
|
||||||
"messageItem.actions.fork": "フォーク",
|
"messageItem.actions.fork": "フォーク",
|
||||||
"messageItem.actions.forkTitle": "このメッセージからフォーク",
|
"messageItem.actions.forkTitle": "このメッセージからフォーク",
|
||||||
"messageItem.actions.copy": "コピー",
|
"messageItem.actions.copy": "コピー",
|
||||||
"messageItem.actions.copyTitle": "メッセージをコピー",
|
"messageItem.actions.copyTitle": "メッセージをコピー",
|
||||||
"messageItem.actions.copied": "コピーしました!",
|
"messageItem.actions.copied": "コピーしました!",
|
||||||
|
"messageItem.actions.deleteMessage": "メッセージを削除(変更は元に戻さない)",
|
||||||
|
"messageItem.actions.deleteMessagesUpTo": "ここまでのメッセージを削除(変更は元に戻さない)",
|
||||||
|
"messageItem.actions.deletingMessage": "削除中...",
|
||||||
|
"messageItem.actions.deleteMessageFailedTitle": "削除に失敗しました",
|
||||||
|
"messageItem.actions.deleteMessageFailedMessage": "メッセージの削除に失敗しました",
|
||||||
|
|
||||||
|
"messageItem.selection.checkboxAriaLabel": "削除するメッセージを選択",
|
||||||
|
|
||||||
|
"messageSection.bulkDelete.toolbarAriaLabel": "選択した項目({count})",
|
||||||
|
"messageSection.bulkDelete.deleteSelectedTitle": "選択した項目を削除",
|
||||||
|
"messageSection.bulkDelete.selectAllTitle": "すべて選択",
|
||||||
|
"messageSection.bulkDelete.moreOptionsTitle": "その他のオプション",
|
||||||
|
"messageSection.bulkDelete.selectionModeLabel": "選択",
|
||||||
|
"messageSection.bulkDelete.selectionModeAll": "すべて",
|
||||||
|
"messageSection.bulkDelete.selectionModeTools": "ツールのみ",
|
||||||
|
"messageSection.bulkDelete.selectionHint.toggle": "項目を選択",
|
||||||
|
"messageSection.bulkDelete.selectionHint.range": "範囲を選択",
|
||||||
|
"messageSection.bulkDelete.selectionHint.clear": "選択を解除",
|
||||||
|
"messageSection.bulkDelete.cancelTitle": "選択をキャンセル",
|
||||||
|
"messageSection.bulkDelete.failedTitle": "削除に失敗しました",
|
||||||
|
"messageSection.bulkDelete.failedMessage": "選択した項目の削除に失敗しました",
|
||||||
"messageItem.status.queued": "待機中",
|
"messageItem.status.queued": "待機中",
|
||||||
"messageItem.status.generating": "生成中...",
|
"messageItem.status.generating": "生成中...",
|
||||||
"messageItem.status.sending": "送信中...",
|
"messageItem.status.sending": "送信中...",
|
||||||
"messageItem.status.failedToSend": "メッセージの送信に失敗しました",
|
"messageItem.status.failedToSend": "メッセージの送信に失敗しました",
|
||||||
"messagePart.actions.delete": "削除",
|
"messagePart.actions.delete": "パートを削除",
|
||||||
"messagePart.actions.deleting": "削除中...",
|
"messagePart.actions.deleting": "削除中...",
|
||||||
"messagePart.actions.deleteTitle": "この項目を削除",
|
"messagePart.actions.deleteTitle": "この項目を削除",
|
||||||
"messagePart.actions.deleteFailedTitle": "削除に失敗しました",
|
"messagePart.actions.deleteFailedTitle": "削除に失敗しました",
|
||||||
|
|||||||
@@ -67,6 +67,8 @@ export const sessionMessages = {
|
|||||||
"sessionView.alerts.abortFailed.title": "停止に失敗",
|
"sessionView.alerts.abortFailed.title": "停止に失敗",
|
||||||
"sessionView.alerts.revertFailed.message": "メッセージへ戻せませんでした",
|
"sessionView.alerts.revertFailed.message": "メッセージへ戻せませんでした",
|
||||||
"sessionView.alerts.revertFailed.title": "復元に失敗",
|
"sessionView.alerts.revertFailed.title": "復元に失敗",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.message": "メッセージの削除に失敗しました",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.title": "削除に失敗しました",
|
||||||
"sessionView.alerts.forkFailed.message": "セッションのフォークに失敗しました",
|
"sessionView.alerts.forkFailed.message": "セッションのフォークに失敗しました",
|
||||||
"sessionView.alerts.forkFailed.title": "フォークに失敗",
|
"sessionView.alerts.forkFailed.title": "フォークに失敗",
|
||||||
"sessionView.attachments.expandPastedTextAriaLabel": "貼り付けたテキストを展開",
|
"sessionView.attachments.expandPastedTextAriaLabel": "貼り付けたテキストを展開",
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ export const messagingMessages = {
|
|||||||
"messageBlock.tool.goToSession.label": "Перейти к сессии",
|
"messageBlock.tool.goToSession.label": "Перейти к сессии",
|
||||||
"messageBlock.tool.goToSession.title": "Перейти к сессии",
|
"messageBlock.tool.goToSession.title": "Перейти к сессии",
|
||||||
"messageBlock.tool.goToSession.unavailableTitle": "Сессия пока недоступна",
|
"messageBlock.tool.goToSession.unavailableTitle": "Сессия пока недоступна",
|
||||||
"messageBlock.tool.deletePart.label": "Удалить",
|
"messageBlock.tool.deletePart.label": "Удалить часть",
|
||||||
"messageBlock.tool.deletePart.deleting": "Удаление...",
|
"messageBlock.tool.deletePart.deleting": "Удаление...",
|
||||||
"messageBlock.tool.deletePart.title": "Удалить этот вывод инструмента",
|
"messageBlock.tool.deletePart.title": "Удалить этот вывод инструмента",
|
||||||
"messageBlock.tool.deletePart.failed.title": "Ошибка удаления",
|
"messageBlock.tool.deletePart.failed.title": "Ошибка удаления",
|
||||||
@@ -71,17 +71,38 @@ export const messagingMessages = {
|
|||||||
"messageItem.speaker.you": "Вы",
|
"messageItem.speaker.you": "Вы",
|
||||||
"messageItem.speaker.assistant": "Ассистент",
|
"messageItem.speaker.assistant": "Ассистент",
|
||||||
"messageItem.actions.revert": "Откатить",
|
"messageItem.actions.revert": "Откатить",
|
||||||
"messageItem.actions.revertTitle": "Откатиться к этому сообщению",
|
"messageItem.actions.revertTitle": "Отменить изменения до этого места (удалит сообщения)",
|
||||||
"messageItem.actions.fork": "Форк",
|
"messageItem.actions.fork": "Форк",
|
||||||
"messageItem.actions.forkTitle": "Форкнуть от этого сообщения",
|
"messageItem.actions.forkTitle": "Форкнуть от этого сообщения",
|
||||||
"messageItem.actions.copy": "Копировать",
|
"messageItem.actions.copy": "Копировать",
|
||||||
"messageItem.actions.copyTitle": "Копировать сообщение",
|
"messageItem.actions.copyTitle": "Копировать сообщение",
|
||||||
"messageItem.actions.copied": "Скопировано!",
|
"messageItem.actions.copied": "Скопировано!",
|
||||||
|
"messageItem.actions.deleteMessage": "Удалить сообщение (без отката изменений)",
|
||||||
|
"messageItem.actions.deleteMessagesUpTo": "Удалить сообщения до этого места (без отката изменений)",
|
||||||
|
"messageItem.actions.deletingMessage": "Удаление...",
|
||||||
|
"messageItem.actions.deleteMessageFailedTitle": "Ошибка удаления",
|
||||||
|
"messageItem.actions.deleteMessageFailedMessage": "Не удалось удалить сообщение",
|
||||||
|
|
||||||
|
"messageItem.selection.checkboxAriaLabel": "Выбрать сообщение для удаления",
|
||||||
|
|
||||||
|
"messageSection.bulkDelete.toolbarAriaLabel": "Выбранные элементы ({count})",
|
||||||
|
"messageSection.bulkDelete.deleteSelectedTitle": "Удалить выбранные элементы",
|
||||||
|
"messageSection.bulkDelete.selectAllTitle": "Выбрать все сообщения",
|
||||||
|
"messageSection.bulkDelete.moreOptionsTitle": "Больше настроек",
|
||||||
|
"messageSection.bulkDelete.selectionModeLabel": "Выбор",
|
||||||
|
"messageSection.bulkDelete.selectionModeAll": "Все",
|
||||||
|
"messageSection.bulkDelete.selectionModeTools": "Только инструменты",
|
||||||
|
"messageSection.bulkDelete.selectionHint.toggle": "Выбрать элемент",
|
||||||
|
"messageSection.bulkDelete.selectionHint.range": "Выбрать диапазон",
|
||||||
|
"messageSection.bulkDelete.selectionHint.clear": "Очистить выбор",
|
||||||
|
"messageSection.bulkDelete.cancelTitle": "Отменить выбор",
|
||||||
|
"messageSection.bulkDelete.failedTitle": "Ошибка удаления",
|
||||||
|
"messageSection.bulkDelete.failedMessage": "Не удалось удалить выбранные элементы",
|
||||||
"messageItem.status.queued": "В ОЧЕРЕДИ",
|
"messageItem.status.queued": "В ОЧЕРЕДИ",
|
||||||
"messageItem.status.generating": "Генерация…",
|
"messageItem.status.generating": "Генерация…",
|
||||||
"messageItem.status.sending": "Отправка…",
|
"messageItem.status.sending": "Отправка…",
|
||||||
"messageItem.status.failedToSend": "Не удалось отправить сообщение",
|
"messageItem.status.failedToSend": "Не удалось отправить сообщение",
|
||||||
"messagePart.actions.delete": "Удалить",
|
"messagePart.actions.delete": "Удалить часть",
|
||||||
"messagePart.actions.deleting": "Удаление...",
|
"messagePart.actions.deleting": "Удаление...",
|
||||||
"messagePart.actions.deleteTitle": "Удалить этот элемент",
|
"messagePart.actions.deleteTitle": "Удалить этот элемент",
|
||||||
"messagePart.actions.deleteFailedTitle": "Ошибка удаления",
|
"messagePart.actions.deleteFailedTitle": "Ошибка удаления",
|
||||||
|
|||||||
@@ -67,6 +67,8 @@ export const sessionMessages = {
|
|||||||
"sessionView.alerts.abortFailed.title": "Не удалось остановить",
|
"sessionView.alerts.abortFailed.title": "Не удалось остановить",
|
||||||
"sessionView.alerts.revertFailed.message": "Не удалось откатиться к сообщению",
|
"sessionView.alerts.revertFailed.message": "Не удалось откатиться к сообщению",
|
||||||
"sessionView.alerts.revertFailed.title": "Не удалось откатиться",
|
"sessionView.alerts.revertFailed.title": "Не удалось откатиться",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.message": "Не удалось удалить сообщения",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.title": "Ошибка удаления",
|
||||||
"sessionView.alerts.forkFailed.message": "Не удалось форкнуть сессию",
|
"sessionView.alerts.forkFailed.message": "Не удалось форкнуть сессию",
|
||||||
"sessionView.alerts.forkFailed.title": "Не удалось форкнуть",
|
"sessionView.alerts.forkFailed.title": "Не удалось форкнуть",
|
||||||
"sessionView.attachments.expandPastedTextAriaLabel": "Развернуть вставленный текст",
|
"sessionView.attachments.expandPastedTextAriaLabel": "Развернуть вставленный текст",
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ export const messagingMessages = {
|
|||||||
"messageBlock.tool.goToSession.label": "前往会话",
|
"messageBlock.tool.goToSession.label": "前往会话",
|
||||||
"messageBlock.tool.goToSession.title": "前往会话",
|
"messageBlock.tool.goToSession.title": "前往会话",
|
||||||
"messageBlock.tool.goToSession.unavailableTitle": "会话尚不可用",
|
"messageBlock.tool.goToSession.unavailableTitle": "会话尚不可用",
|
||||||
"messageBlock.tool.deletePart.label": "删除",
|
"messageBlock.tool.deletePart.label": "删除部分",
|
||||||
"messageBlock.tool.deletePart.deleting": "正在删除...",
|
"messageBlock.tool.deletePart.deleting": "正在删除...",
|
||||||
"messageBlock.tool.deletePart.title": "删除此工具输出",
|
"messageBlock.tool.deletePart.title": "删除此工具输出",
|
||||||
"messageBlock.tool.deletePart.failed.title": "删除失败",
|
"messageBlock.tool.deletePart.failed.title": "删除失败",
|
||||||
@@ -71,17 +71,38 @@ export const messagingMessages = {
|
|||||||
"messageItem.speaker.you": "你",
|
"messageItem.speaker.you": "你",
|
||||||
"messageItem.speaker.assistant": "助手",
|
"messageItem.speaker.assistant": "助手",
|
||||||
"messageItem.actions.revert": "回退",
|
"messageItem.actions.revert": "回退",
|
||||||
"messageItem.actions.revertTitle": "回退到这条消息",
|
"messageItem.actions.revertTitle": "撤销到此处的更改(会删除消息)",
|
||||||
"messageItem.actions.fork": "分叉",
|
"messageItem.actions.fork": "分叉",
|
||||||
"messageItem.actions.forkTitle": "从这条消息分叉",
|
"messageItem.actions.forkTitle": "从这条消息分叉",
|
||||||
"messageItem.actions.copy": "复制",
|
"messageItem.actions.copy": "复制",
|
||||||
"messageItem.actions.copyTitle": "复制消息",
|
"messageItem.actions.copyTitle": "复制消息",
|
||||||
"messageItem.actions.copied": "已复制!",
|
"messageItem.actions.copied": "已复制!",
|
||||||
|
"messageItem.actions.deleteMessage": "删除消息(不会撤销更改)",
|
||||||
|
"messageItem.actions.deleteMessagesUpTo": "删除到此处的消息(不会撤销更改)",
|
||||||
|
"messageItem.actions.deletingMessage": "正在删除...",
|
||||||
|
"messageItem.actions.deleteMessageFailedTitle": "删除失败",
|
||||||
|
"messageItem.actions.deleteMessageFailedMessage": "无法删除消息",
|
||||||
|
|
||||||
|
"messageItem.selection.checkboxAriaLabel": "选择要删除的消息",
|
||||||
|
|
||||||
|
"messageSection.bulkDelete.toolbarAriaLabel": "已选择的项目({count})",
|
||||||
|
"messageSection.bulkDelete.deleteSelectedTitle": "删除已选择的项目",
|
||||||
|
"messageSection.bulkDelete.selectAllTitle": "全选消息",
|
||||||
|
"messageSection.bulkDelete.moreOptionsTitle": "更多选项",
|
||||||
|
"messageSection.bulkDelete.selectionModeLabel": "选择",
|
||||||
|
"messageSection.bulkDelete.selectionModeAll": "全部",
|
||||||
|
"messageSection.bulkDelete.selectionModeTools": "仅工具",
|
||||||
|
"messageSection.bulkDelete.selectionHint.toggle": "选择项目",
|
||||||
|
"messageSection.bulkDelete.selectionHint.range": "选择范围",
|
||||||
|
"messageSection.bulkDelete.selectionHint.clear": "清除选择",
|
||||||
|
"messageSection.bulkDelete.cancelTitle": "取消选择",
|
||||||
|
"messageSection.bulkDelete.failedTitle": "删除失败",
|
||||||
|
"messageSection.bulkDelete.failedMessage": "无法删除已选择的项目",
|
||||||
"messageItem.status.queued": "排队中",
|
"messageItem.status.queued": "排队中",
|
||||||
"messageItem.status.generating": "正在生成...",
|
"messageItem.status.generating": "正在生成...",
|
||||||
"messageItem.status.sending": "正在发送...",
|
"messageItem.status.sending": "正在发送...",
|
||||||
"messageItem.status.failedToSend": "消息发送失败",
|
"messageItem.status.failedToSend": "消息发送失败",
|
||||||
"messagePart.actions.delete": "删除",
|
"messagePart.actions.delete": "删除部分",
|
||||||
"messagePart.actions.deleting": "正在删除...",
|
"messagePart.actions.deleting": "正在删除...",
|
||||||
"messagePart.actions.deleteTitle": "删除此项",
|
"messagePart.actions.deleteTitle": "删除此项",
|
||||||
"messagePart.actions.deleteFailedTitle": "删除失败",
|
"messagePart.actions.deleteFailedTitle": "删除失败",
|
||||||
|
|||||||
@@ -67,6 +67,8 @@ export const sessionMessages = {
|
|||||||
"sessionView.alerts.abortFailed.title": "停止失败",
|
"sessionView.alerts.abortFailed.title": "停止失败",
|
||||||
"sessionView.alerts.revertFailed.message": "回退到消息失败",
|
"sessionView.alerts.revertFailed.message": "回退到消息失败",
|
||||||
"sessionView.alerts.revertFailed.title": "回退失败",
|
"sessionView.alerts.revertFailed.title": "回退失败",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.message": "无法删除消息",
|
||||||
|
"sessionView.alerts.deleteUpToFailed.title": "删除失败",
|
||||||
"sessionView.alerts.forkFailed.message": "分叉会话失败",
|
"sessionView.alerts.forkFailed.message": "分叉会话失败",
|
||||||
"sessionView.alerts.forkFailed.title": "分叉失败",
|
"sessionView.alerts.forkFailed.title": "分叉失败",
|
||||||
"sessionView.attachments.expandPastedTextAriaLabel": "展开粘贴的文本",
|
"sessionView.attachments.expandPastedTextAriaLabel": "展开粘贴的文本",
|
||||||
|
|||||||
@@ -127,17 +127,23 @@ async function ensureLanguages(content: string) {
|
|||||||
if (highlightSuppressed) {
|
if (highlightSuppressed) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
// Parse code fences to extract language tokens
|
|
||||||
// Updated regex to capture optional language tokens and handle trailing annotations
|
|
||||||
const codeBlockRegex = /```[ \t]*([A-Za-z0-9_.+#-]+)?[^`]*?```/g
|
|
||||||
const foundLanguages = new Set<string>()
|
|
||||||
let match
|
|
||||||
|
|
||||||
while ((match = codeBlockRegex.exec(content)) !== null) {
|
// Extract code-fence language tokens via `marked` so we correctly handle code blocks
|
||||||
const langToken = match[1]
|
// that contain backticks (e.g. JS template literals). Regex-based fence scans tend
|
||||||
if (langToken && langToken.trim()) {
|
// to miss these and prevent languages from loading.
|
||||||
foundLanguages.add(langToken.trim())
|
const foundLanguages = new Set<string>()
|
||||||
}
|
try {
|
||||||
|
const tokens = marked.lexer(content) as any
|
||||||
|
marked.walkTokens(tokens, (token: any) => {
|
||||||
|
if (token?.type !== "code") return
|
||||||
|
const langToken = typeof token.lang === "string" ? token.lang : ""
|
||||||
|
if (langToken.trim()) {
|
||||||
|
foundLanguages.add(langToken.trim())
|
||||||
|
}
|
||||||
|
})
|
||||||
|
} catch {
|
||||||
|
// If tokenization fails for any reason, skip language preloading.
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Queue language loading tasks
|
// Queue language loading tasks
|
||||||
|
|||||||
66
packages/ui/src/lib/token-utils.ts
Normal file
66
packages/ui/src/lib/token-utils.ts
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
import type { ClientPart } from "../types/message"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Count the total character content of a message part.
|
||||||
|
*
|
||||||
|
* Used by both the xray histogram overlay (message-timeline) and the
|
||||||
|
* bulk-delete toolbar token pills (message-section) so both surfaces
|
||||||
|
* derive token estimates from the same logic.
|
||||||
|
*
|
||||||
|
* Note: For tool parts we intentionally only count `state.input` and
|
||||||
|
* `state.output`. We exclude `state.metadata` from token estimation since
|
||||||
|
* metadata can contain large or verbose diagnostic payloads that are not
|
||||||
|
* representative of model context.
|
||||||
|
*/
|
||||||
|
export function getPartCharCount(part: ClientPart): number {
|
||||||
|
if (!part) return 0
|
||||||
|
let count = 0
|
||||||
|
|
||||||
|
if (typeof (part as any).text === "string") {
|
||||||
|
count += (part as any).text.length
|
||||||
|
}
|
||||||
|
|
||||||
|
if (part.type === "tool") {
|
||||||
|
const state = (part as any).state
|
||||||
|
// Tool calls may be compacted server-side. When that happens we treat the
|
||||||
|
// tool payload as effectively absent from context for token estimation.
|
||||||
|
const compacted = (state as any)?.time?.compacted
|
||||||
|
if (compacted !== undefined && compacted !== null) {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
if (state) {
|
||||||
|
if (state.input) {
|
||||||
|
try {
|
||||||
|
count += JSON.stringify(state.input).length
|
||||||
|
} catch {}
|
||||||
|
}
|
||||||
|
if (state.output) {
|
||||||
|
if (typeof state.output === "string") {
|
||||||
|
count += state.output.length
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
count += JSON.stringify(state.output).length
|
||||||
|
} catch {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Array.isArray((part as any).content)) {
|
||||||
|
count += (part as any).content.reduce((acc: number, entry: unknown) => {
|
||||||
|
if (typeof entry === "string") return acc + entry.length
|
||||||
|
if (entry && typeof entry === "object") {
|
||||||
|
let entryCount = (String((entry as any).text || "")).length + (String((entry as any).value || "")).length
|
||||||
|
if (Array.isArray((entry as any).content)) {
|
||||||
|
entryCount += (entry as any).content.reduce((innerAcc: number, sub: unknown) => {
|
||||||
|
if (typeof sub === "string") return innerAcc + sub.length
|
||||||
|
return innerAcc + (String((sub as any)?.text || "")).length
|
||||||
|
}, 0)
|
||||||
|
}
|
||||||
|
return acc + entryCount
|
||||||
|
}
|
||||||
|
return acc
|
||||||
|
}, 0)
|
||||||
|
}
|
||||||
|
return count
|
||||||
|
}
|
||||||
@@ -117,6 +117,7 @@ export function applyPartDeltaV2(
|
|||||||
partId: input.partId,
|
partId: input.partId,
|
||||||
field: input.field,
|
field: input.field,
|
||||||
delta: input.delta,
|
delta: input.delta,
|
||||||
|
bumpSessionRevision: false,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -189,7 +189,14 @@ export interface InstanceMessageStore {
|
|||||||
hydrateMessages: (sessionId: string, inputs: MessageUpsertInput[], infos?: Iterable<MessageInfo>) => void
|
hydrateMessages: (sessionId: string, inputs: MessageUpsertInput[], infos?: Iterable<MessageInfo>) => void
|
||||||
upsertMessage: (input: MessageUpsertInput) => void
|
upsertMessage: (input: MessageUpsertInput) => void
|
||||||
applyPartUpdate: (input: PartUpdateInput) => void
|
applyPartUpdate: (input: PartUpdateInput) => void
|
||||||
applyPartDelta: (input: { messageId: string; partId: string; field: string; delta: string; bumpRevision?: boolean }) => void
|
applyPartDelta: (input: {
|
||||||
|
messageId: string
|
||||||
|
partId: string
|
||||||
|
field: string
|
||||||
|
delta: string
|
||||||
|
bumpRevision?: boolean
|
||||||
|
bumpSessionRevision: boolean
|
||||||
|
}) => void
|
||||||
removeMessage: (messageId: string) => void
|
removeMessage: (messageId: string) => void
|
||||||
removeMessagePart: (messageId: string, partId: string) => void
|
removeMessagePart: (messageId: string, partId: string) => void
|
||||||
bufferPendingPart: (entry: PendingPartEntry) => void
|
bufferPendingPart: (entry: PendingPartEntry) => void
|
||||||
@@ -211,6 +218,9 @@ export interface InstanceMessageStore {
|
|||||||
getScrollSnapshot: (sessionId: string, scope: string) => ScrollSnapshot | undefined
|
getScrollSnapshot: (sessionId: string, scope: string) => ScrollSnapshot | undefined
|
||||||
getSessionRevision: (sessionId: string) => number
|
getSessionRevision: (sessionId: string) => number
|
||||||
getSessionMessageIds: (sessionId: string) => string[]
|
getSessionMessageIds: (sessionId: string) => string[]
|
||||||
|
// Index of the most recent message in the session that contains a compaction part.
|
||||||
|
// Returns -1 if there has been no compaction.
|
||||||
|
getLastCompactionMessageIndex: (sessionId: string) => number
|
||||||
getMessage: (messageId: string) => MessageRecord | undefined
|
getMessage: (messageId: string) => MessageRecord | undefined
|
||||||
getLatestTodoSnapshot: (sessionId: string) => LatestTodoSnapshot | undefined
|
getLatestTodoSnapshot: (sessionId: string) => LatestTodoSnapshot | undefined
|
||||||
clearSession: (sessionId: string) => void
|
clearSession: (sessionId: string) => void
|
||||||
@@ -224,6 +234,24 @@ export function createInstanceMessageStore(instanceId: string, hooks?: MessageSt
|
|||||||
|
|
||||||
const messageInfoCache = new Map<string, MessageInfo>()
|
const messageInfoCache = new Map<string, MessageInfo>()
|
||||||
|
|
||||||
|
function getLastCompactionMessageIndex(sessionId: string): number {
|
||||||
|
if (!sessionId) return -1
|
||||||
|
const ids = state.sessions[sessionId]?.messageIds ?? []
|
||||||
|
// Scan from the end: we only care about the most recent compaction.
|
||||||
|
for (let i = ids.length - 1; i >= 0; i--) {
|
||||||
|
const messageId = ids[i]
|
||||||
|
const record = state.messages[messageId]
|
||||||
|
if (!record || !Array.isArray(record.partIds) || record.partIds.length === 0) continue
|
||||||
|
for (const partId of record.partIds) {
|
||||||
|
const part = record.parts[partId]?.data
|
||||||
|
if ((part as any)?.type === "compaction") {
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
|
||||||
function isCompletedTodoPart(part: ClientPart | undefined): boolean {
|
function isCompletedTodoPart(part: ClientPart | undefined): boolean {
|
||||||
if (!part || (part as any).type !== "tool") {
|
if (!part || (part as any).type !== "tool") {
|
||||||
return false
|
return false
|
||||||
@@ -598,7 +626,14 @@ export function createInstanceMessageStore(instanceId: string, hooks?: MessageSt
|
|||||||
bumpSessionRevision(message.sessionId)
|
bumpSessionRevision(message.sessionId)
|
||||||
}
|
}
|
||||||
|
|
||||||
function applyPartDelta(input: { messageId: string; partId: string; field: string; delta: string; bumpRevision?: boolean }) {
|
function applyPartDelta(input: {
|
||||||
|
messageId: string
|
||||||
|
partId: string
|
||||||
|
field: string
|
||||||
|
delta: string
|
||||||
|
bumpRevision?: boolean
|
||||||
|
bumpSessionRevision?: boolean
|
||||||
|
}) {
|
||||||
if (!input?.messageId || !input.partId || !input.field || typeof input.delta !== "string") {
|
if (!input?.messageId || !input.partId || !input.field || typeof input.delta !== "string") {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -632,7 +667,7 @@ export function createInstanceMessageStore(instanceId: string, hooks?: MessageSt
|
|||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
|
|
||||||
if (applied) {
|
if (applied && (input.bumpSessionRevision ?? true)) {
|
||||||
bumpSessionRevision(message.sessionId)
|
bumpSessionRevision(message.sessionId)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1124,8 +1159,8 @@ export function createInstanceMessageStore(instanceId: string, hooks?: MessageSt
|
|||||||
|
|
||||||
function clearInstance() {
|
function clearInstance() {
|
||||||
messageInfoCache.clear()
|
messageInfoCache.clear()
|
||||||
setState(reconcile(createInitialState(instanceId)))
|
setState(reconcile(createInitialState(instanceId)))
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|
||||||
@@ -1158,11 +1193,11 @@ export function createInstanceMessageStore(instanceId: string, hooks?: MessageSt
|
|||||||
setScrollSnapshot,
|
setScrollSnapshot,
|
||||||
getScrollSnapshot,
|
getScrollSnapshot,
|
||||||
getSessionRevision: getSessionRevisionValue,
|
getSessionRevision: getSessionRevisionValue,
|
||||||
getSessionMessageIds: (sessionId: string) => state.sessions[sessionId]?.messageIds ?? [],
|
getSessionMessageIds: (sessionId: string) => state.sessions[sessionId]?.messageIds ?? [],
|
||||||
getMessage: (messageId: string) => state.messages[messageId],
|
getLastCompactionMessageIndex,
|
||||||
getLatestTodoSnapshot: (sessionId: string) => state.latestTodos[sessionId],
|
getMessage: (messageId: string) => state.messages[messageId],
|
||||||
clearSession,
|
getLatestTodoSnapshot: (sessionId: string) => state.latestTodos[sessionId],
|
||||||
clearInstance,
|
clearSession,
|
||||||
|
clearInstance,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ import { providers, sessions, withSession } from "./session-state"
|
|||||||
import { getDefaultModel, isModelValid } from "./session-models"
|
import { getDefaultModel, isModelValid } from "./session-models"
|
||||||
import { updateSessionInfo } from "./message-v2/session-info"
|
import { updateSessionInfo } from "./message-v2/session-info"
|
||||||
import { messageStoreBus } from "./message-v2/bus"
|
import { messageStoreBus } from "./message-v2/bus"
|
||||||
import { removeMessagePartV2 } from "./message-v2/bridge"
|
import { removeMessagePartV2, removeMessageV2 } from "./message-v2/bridge"
|
||||||
import { getLogger } from "../lib/logger"
|
import { getLogger } from "../lib/logger"
|
||||||
import { requestData } from "../lib/opencode-api"
|
import { requestData } from "../lib/opencode-api"
|
||||||
|
|
||||||
@@ -439,8 +439,33 @@ async function deleteMessagePart(instanceId: string, sessionId: string, messageI
|
|||||||
updateSessionInfo(instanceId, sessionId)
|
updateSessionInfo(instanceId, sessionId)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function deleteMessage(instanceId: string, sessionId: string, messageId: string): Promise<void> {
|
||||||
|
if (!instanceId || !sessionId || !messageId) return
|
||||||
|
const instance = instances().get(instanceId)
|
||||||
|
if (!instance || !instance.client) {
|
||||||
|
throw new Error("Instance not ready")
|
||||||
|
}
|
||||||
|
|
||||||
|
const worktreeSlug = getWorktreeSlugForSession(instanceId, sessionId)
|
||||||
|
const client = getOrCreateWorktreeClient(instanceId, worktreeSlug)
|
||||||
|
|
||||||
|
// The SDK generator does not currently expose a typed method for deleting a message,
|
||||||
|
// but the API is available at DELETE /session/:sessionID/message/:messageID.
|
||||||
|
await requestData(
|
||||||
|
(client as any).client.delete({
|
||||||
|
url: `/session/${encodeURIComponent(sessionId)}/message/${encodeURIComponent(messageId)}`,
|
||||||
|
}),
|
||||||
|
"session.message.delete",
|
||||||
|
)
|
||||||
|
|
||||||
|
// Optimistic removal; SSE will also broadcast a message-removed event.
|
||||||
|
removeMessageV2(instanceId, messageId)
|
||||||
|
updateSessionInfo(instanceId, sessionId)
|
||||||
|
}
|
||||||
|
|
||||||
export {
|
export {
|
||||||
abortSession,
|
abortSession,
|
||||||
|
deleteMessage,
|
||||||
deleteMessagePart,
|
deleteMessagePart,
|
||||||
executeCustomCommand,
|
executeCustomCommand,
|
||||||
renameSession,
|
renameSession,
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
@import "./messaging/message-base.css";
|
@import "./messaging/message-base.css";
|
||||||
@import "./messaging/prompt-input.css";
|
@import "./messaging/prompt-input.css";
|
||||||
@import "./messaging/message-section.css";
|
@import "./messaging/message-section.css";
|
||||||
@import "./messaging/message-block-list.css";
|
@import "./messaging/virtual-follow-list.css";
|
||||||
|
@import "./messaging/message-selection.css";
|
||||||
|
@import "./messaging/delete-overlays.css";
|
||||||
@import "./messaging/message-timeline.css";
|
@import "./messaging/message-timeline.css";
|
||||||
@import "./messaging/tool-call.css";
|
@import "./messaging/tool-call.css";
|
||||||
@import "./messaging/log-view.css";
|
@import "./messaging/log-view.css";
|
||||||
@@ -110,4 +112,3 @@
|
|||||||
.reasoning-label {
|
.reasoning-label {
|
||||||
font-weight: var(--font-weight-medium);
|
font-weight: var(--font-weight-medium);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
35
packages/ui/src/styles/messaging/delete-overlays.css
Normal file
35
packages/ui/src/styles/messaging/delete-overlays.css
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
/* Hover overlays for destructive actions (delete part / delete message). */
|
||||||
|
|
||||||
|
.message-stream-block[data-delete-message-hover="true"] {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-stream-block[data-delete-message-hover="true"]::before {
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
inset: -2px;
|
||||||
|
background: var(--status-error-bg);
|
||||||
|
border-radius: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
/* Overlay must sit above the message cards (they have opaque backgrounds). */
|
||||||
|
z-index: 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-part-shell {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.delete-hover-scope[data-delete-part-hover="true"] {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.delete-hover-scope[data-delete-part-hover="true"]::before {
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
inset: -2px;
|
||||||
|
background: var(--status-error-bg);
|
||||||
|
border-radius: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
/* Overlay must sit above the part card background. */
|
||||||
|
z-index: 10;
|
||||||
|
}
|
||||||
@@ -8,7 +8,8 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.message-item-header {
|
.message-item-header {
|
||||||
@apply flex flex-col gap-0.5;
|
@apply flex flex-col;
|
||||||
|
gap: 0.25rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.message-item-header-row {
|
.message-item-header-row {
|
||||||
@@ -19,12 +20,58 @@
|
|||||||
@apply flex justify-between items-start gap-2.5;
|
@apply flex justify-between items-start gap-2.5;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.message-item-header-row--meta {
|
||||||
|
@apply w-full;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-header-left {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
flex: 1 1 auto;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
.message-item-header-row--bottom {
|
.message-item-header-row--bottom {
|
||||||
@apply flex items-start;
|
@apply flex items-start;
|
||||||
}
|
}
|
||||||
|
|
||||||
.message-speaker {
|
.message-speaker {
|
||||||
@apply flex flex-col gap-0.5 text-xs;
|
/* Allow agent meta to wrap to a second row with comfortable spacing. */
|
||||||
|
@apply flex flex-wrap items-center gap-x-2 gap-y-0.5 text-xs;
|
||||||
|
flex: 1 1 auto;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-speaker-primary {
|
||||||
|
@apply inline-flex items-center;
|
||||||
|
white-space: nowrap;
|
||||||
|
flex: 0 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-agent-meta-inline {
|
||||||
|
@apply text-[11px] font-medium;
|
||||||
|
color: var(--message-assistant-border);
|
||||||
|
white-space: nowrap;
|
||||||
|
flex: 0 0 auto;
|
||||||
|
line-height: 1.1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-agent-meta-inline--measure {
|
||||||
|
position: fixed;
|
||||||
|
left: -9999px;
|
||||||
|
top: -9999px;
|
||||||
|
visibility: hidden;
|
||||||
|
pointer-events: none;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-agent-meta-block {
|
||||||
|
@apply text-[11px] font-medium;
|
||||||
|
color: var(--message-assistant-border);
|
||||||
|
overflow-wrap: anywhere;
|
||||||
|
word-break: break-word;
|
||||||
|
line-height: 1.15;
|
||||||
}
|
}
|
||||||
|
|
||||||
.message-speaker-label {
|
.message-speaker-label {
|
||||||
@@ -46,19 +93,19 @@
|
|||||||
|
|
||||||
.message-item-actions {
|
.message-item-actions {
|
||||||
@apply flex items-center gap-2;
|
@apply flex items-center gap-2;
|
||||||
|
flex: 0 0 auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.message-action-group {
|
.message-action-group {
|
||||||
@apply flex items-center gap-2;
|
@apply flex items-center gap-0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.message-action-button {
|
.message-action-button {
|
||||||
@apply bg-transparent border border-[var(--border-base)] text-[var(--text-muted)] cursor-pointer px-3 py-0.5 rounded text-xs font-semibold leading-none transition-all duration-200 flex items-center justify-center h-6;
|
@apply bg-transparent border-0 text-[var(--text-muted)] cursor-pointer px-2 py-0.5 rounded text-xs font-semibold leading-none transition-all duration-200 flex items-center justify-center h-6;
|
||||||
}
|
}
|
||||||
|
|
||||||
.message-action-button:hover {
|
.message-action-button:hover {
|
||||||
background-color: var(--surface-hover);
|
background-color: var(--surface-hover);
|
||||||
border-color: var(--accent-primary);
|
|
||||||
color: var(--accent-primary);
|
color: var(--accent-primary);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -156,6 +203,27 @@
|
|||||||
border-top: 1px solid var(--border-base);
|
border-top: 1px solid var(--border-base);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Image attachment preview popover.
|
||||||
|
Rendered via a Portal to avoid being clipped by the message stream scroller. */
|
||||||
|
.attachment-image-popover {
|
||||||
|
position: fixed;
|
||||||
|
padding: 8px;
|
||||||
|
background-color: var(--surface-base);
|
||||||
|
border: 1px solid var(--border-base);
|
||||||
|
border-radius: 10px;
|
||||||
|
box-shadow: var(--popover-shadow);
|
||||||
|
z-index: 1000;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.attachment-image-popover img {
|
||||||
|
display: block;
|
||||||
|
max-width: 320px;
|
||||||
|
max-height: 320px;
|
||||||
|
border-radius: 8px;
|
||||||
|
object-fit: contain;
|
||||||
|
}
|
||||||
|
|
||||||
.message-error {
|
.message-error {
|
||||||
@apply text-xs mt-1;
|
@apply text-xs mt-1;
|
||||||
color: var(--status-error);
|
color: var(--status-error);
|
||||||
@@ -296,6 +364,12 @@
|
|||||||
color: var(--message-assistant-border);
|
color: var(--message-assistant-border);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Keep reasoning meta as a single unit so it drops to the next line when needed. */
|
||||||
|
.message-reasoning-label .message-step-meta-inline {
|
||||||
|
flex-wrap: nowrap;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
.message-step-reason {
|
.message-step-reason {
|
||||||
@apply text-[11px] font-medium;
|
@apply text-[11px] font-medium;
|
||||||
@@ -320,7 +394,7 @@
|
|||||||
|
|
||||||
.message-reasoning-header {
|
.message-reasoning-header {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: stretch;
|
align-items: flex-start;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
gap: 0.5rem;
|
gap: 0.5rem;
|
||||||
transition: background-color 0.2s ease, box-shadow 0.2s ease;
|
transition: background-color 0.2s ease, box-shadow 0.2s ease;
|
||||||
@@ -365,11 +439,36 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.message-reasoning-label {
|
.message-reasoning-label {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: nowrap;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
font-size: 0.75rem;
|
font-size: 0.75rem;
|
||||||
font-weight: var(--font-weight-medium);
|
font-weight: var(--font-weight-medium);
|
||||||
color: var(--message-assistant-border);
|
color: var(--message-assistant-border);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.message-reasoning-label-primary {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5rem;
|
||||||
|
white-space: nowrap;
|
||||||
|
flex: 0 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-step-meta-inline--measure {
|
||||||
|
position: fixed;
|
||||||
|
left: -9999px;
|
||||||
|
top: -9999px;
|
||||||
|
visibility: hidden;
|
||||||
|
pointer-events: none;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-reasoning-meta-row {
|
||||||
|
padding: 0 0.6rem 0.15rem 0.6rem;
|
||||||
|
}
|
||||||
|
|
||||||
.message-reasoning-meta {
|
.message-reasoning-meta {
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
|||||||
232
packages/ui/src/styles/messaging/message-selection.css
Normal file
232
packages/ui/src/styles/messaging/message-selection.css
Normal file
@@ -0,0 +1,232 @@
|
|||||||
|
/* Message multi-select delete mode UI. */
|
||||||
|
|
||||||
|
.message-select-checkbox {
|
||||||
|
width: 14px;
|
||||||
|
height: 14px;
|
||||||
|
margin-right: 0.5rem;
|
||||||
|
cursor: pointer;
|
||||||
|
accent-color: var(--status-error);
|
||||||
|
flex: 0 0 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-toolbar {
|
||||||
|
position: absolute;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
bottom: 1rem;
|
||||||
|
display: inline-flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
gap: 6px;
|
||||||
|
padding: 6px 10px;
|
||||||
|
/* Match other popups (dropdown-surface / panels) */
|
||||||
|
background-color: var(--surface-base);
|
||||||
|
border: 1px solid var(--border-base);
|
||||||
|
border-radius: 12px;
|
||||||
|
z-index: 50;
|
||||||
|
box-shadow: var(--panel-shadow-strong);
|
||||||
|
width: max-content;
|
||||||
|
max-width: min(80vw, 560px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-toolbar-row {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-token-group {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-count {
|
||||||
|
min-width: 28px;
|
||||||
|
height: 28px;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 0 8px;
|
||||||
|
border-radius: 999px;
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: 700;
|
||||||
|
font-variant-numeric: tabular-nums;
|
||||||
|
color: var(--accent-primary);
|
||||||
|
background: color-mix(in oklab, var(--surface-base) 85%, var(--accent-primary));
|
||||||
|
border: 1px solid color-mix(in oklab, var(--accent-primary) 50%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-count--before {
|
||||||
|
color: var(--text-muted);
|
||||||
|
background: color-mix(in oklab, var(--surface-base) 90%, var(--text-muted));
|
||||||
|
border-color: color-mix(in oklab, var(--text-muted) 30%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-count--selection {
|
||||||
|
color: var(--status-error);
|
||||||
|
background: color-mix(in oklab, var(--surface-base) 85%, var(--status-error));
|
||||||
|
border-color: color-mix(in oklab, var(--status-error) 40%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-count--after {
|
||||||
|
color: var(--status-success);
|
||||||
|
background: color-mix(in oklab, var(--surface-base) 85%, var(--status-success));
|
||||||
|
border-color: color-mix(in oklab, var(--status-success) 40%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-arrow {
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 700;
|
||||||
|
color: var(--text-muted);
|
||||||
|
line-height: 1;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-button {
|
||||||
|
width: 32px;
|
||||||
|
height: 32px;
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
background: transparent;
|
||||||
|
border: 1px solid color-mix(in oklab, var(--accent-primary) 30%, transparent);
|
||||||
|
border-radius: 10px;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background-color 0.15s ease, color 0.15s ease, border-color 0.15s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-button:hover {
|
||||||
|
background-color: color-mix(in oklab, var(--accent-primary) 15%, transparent);
|
||||||
|
border-color: var(--accent-primary);
|
||||||
|
color: var(--accent-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-button--delete {
|
||||||
|
color: var(--status-error);
|
||||||
|
border-color: color-mix(in oklab, var(--status-error) 30%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-button--delete:hover {
|
||||||
|
background-color: var(--status-error-bg);
|
||||||
|
border-color: var(--status-error);
|
||||||
|
color: var(--status-error);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-button--cancel:hover {
|
||||||
|
background-color: color-mix(in oklab, var(--text-muted) 12%, transparent);
|
||||||
|
border-color: var(--text-muted);
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-button--menu {
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-button--menu:hover {
|
||||||
|
color: var(--text-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-menu-container {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-menu {
|
||||||
|
right: 0;
|
||||||
|
bottom: calc(100% + 6px);
|
||||||
|
min-width: 150px;
|
||||||
|
width: max-content;
|
||||||
|
max-width: min(70vw, 220px);
|
||||||
|
padding: 2px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-menu-divider {
|
||||||
|
height: 1px;
|
||||||
|
margin: 3px 0;
|
||||||
|
background-color: var(--border-base);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-menu-row {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 3px;
|
||||||
|
padding: 2px 8px 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-menu-label {
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-muted);
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-menu-toggle {
|
||||||
|
display: inline-flex;
|
||||||
|
border-radius: 999px;
|
||||||
|
border: 1px solid var(--border-base);
|
||||||
|
background-color: var(--surface-secondary);
|
||||||
|
overflow: hidden;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-menu-toggle-button {
|
||||||
|
padding: 2px 8px;
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-muted);
|
||||||
|
background: transparent;
|
||||||
|
border: none;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: color 0.15s ease, background-color 0.15s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-menu-toggle-button[data-mode="all"] {
|
||||||
|
flex: 0 0 auto;
|
||||||
|
min-width: 56px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-menu-toggle-button[data-mode="tools"] {
|
||||||
|
flex: 1 1 auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-menu-toggle-button[data-active="true"] {
|
||||||
|
color: var(--text-primary);
|
||||||
|
background-color: color-mix(in oklab, var(--accent-primary) 18%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-menu .dropdown-item {
|
||||||
|
width: calc(100% - 8px);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
margin: 0 4px;
|
||||||
|
padding: 3px 8px;
|
||||||
|
font-size: 12px;
|
||||||
|
border-radius: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-button:focus-visible {
|
||||||
|
outline: none;
|
||||||
|
box-shadow: 0 0 0 2px color-mix(in oklab, var(--accent-primary) 45%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-hint-row {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 6px;
|
||||||
|
padding-top: 2px;
|
||||||
|
font-size: 10px;
|
||||||
|
color: var(--text-muted);
|
||||||
|
user-select: none;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-hint-text {
|
||||||
|
white-space: normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-delete-mode-hint-sep {
|
||||||
|
color: var(--text-muted);
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
@@ -6,6 +6,9 @@
|
|||||||
min-height: 0;
|
min-height: 0;
|
||||||
flex: 1 1 auto;
|
flex: 1 1 auto;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
/* Isolate stacking context so sidebar z-indices don't compete with
|
||||||
|
Portals (Command Palette, modals) that live at the body level. */
|
||||||
|
isolation: isolate;
|
||||||
}
|
}
|
||||||
|
|
||||||
.message-layout--with-timeline {
|
.message-layout--with-timeline {
|
||||||
@@ -51,6 +54,8 @@
|
|||||||
min-height: 0;
|
min-height: 0;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
position: relative;
|
||||||
|
z-index: 100;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -67,11 +72,16 @@
|
|||||||
gap: 0.35rem;
|
gap: 0.35rem;
|
||||||
padding: 0.25rem;
|
padding: 0.25rem;
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
|
overflow-x: visible;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
background-color: var(--surface-base);
|
background-color: var(--surface-base);
|
||||||
box-shadow: var(--panel-shadow);
|
box-shadow: var(--panel-shadow);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.message-timeline--selection-active {
|
||||||
|
padding-bottom: 4rem;
|
||||||
|
}
|
||||||
|
|
||||||
.message-timeline::-webkit-scrollbar {
|
.message-timeline::-webkit-scrollbar {
|
||||||
width: 5px;
|
width: 5px;
|
||||||
}
|
}
|
||||||
@@ -87,6 +97,7 @@
|
|||||||
border-radius: 0;
|
border-radius: 0;
|
||||||
border: 1px solid var(--border-base);
|
border: 1px solid var(--border-base);
|
||||||
background-color: var(--surface-secondary);
|
background-color: var(--surface-secondary);
|
||||||
|
position: relative;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
@@ -96,6 +107,40 @@
|
|||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
color: var(--text-primary);
|
color: var(--text-primary);
|
||||||
transition: transform 0.15s ease, background-color 0.2s ease, color 0.2s ease, border-color 0.2s ease, box-shadow 0.2s ease;
|
transition: transform 0.15s ease, background-color 0.2s ease, color 0.2s ease, border-color 0.2s ease, box-shadow 0.2s ease;
|
||||||
|
overflow: hidden;
|
||||||
|
user-select: none;
|
||||||
|
-webkit-user-select: none;
|
||||||
|
touch-action: manipulation;
|
||||||
|
-webkit-touch-callout: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-segment[data-delete-hover="true"]::before {
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
inset: 0;
|
||||||
|
background: var(--status-error-bg);
|
||||||
|
border-radius: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Ensure delete hover is visible even on the active segment styling. */
|
||||||
|
.message-timeline-segment[data-delete-hover="true"],
|
||||||
|
.message-timeline-segment[data-delete-hover="true"]:hover,
|
||||||
|
.message-timeline-segment[data-delete-hover="true"]:focus-visible,
|
||||||
|
.message-timeline-segment-active[data-delete-hover="true"],
|
||||||
|
.message-timeline-segment-active[data-delete-hover="true"]:hover,
|
||||||
|
.message-timeline-segment-active[data-delete-hover="true"]:focus-visible {
|
||||||
|
/* Let the ::before overlay provide the highlight (matches stream behavior). */
|
||||||
|
background-color: transparent !important;
|
||||||
|
border-color: transparent !important;
|
||||||
|
box-shadow: none !important;
|
||||||
|
transform: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-label {
|
||||||
|
position: relative;
|
||||||
|
z-index: 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
.message-timeline-segment.message-timeline-segment-hidden {
|
.message-timeline-segment.message-timeline-segment-hidden {
|
||||||
@@ -229,3 +274,162 @@
|
|||||||
.message-preview .message-item-base {
|
.message-preview .message-item-base {
|
||||||
font-size: 0.85rem;
|
font-size: 0.85rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* --- Selection & Histogram Ribs --- */
|
||||||
|
|
||||||
|
.message-timeline-segment-selected {
|
||||||
|
border-color: var(--accent-primary) !important;
|
||||||
|
background-color: color-mix(in oklab, var(--accent-primary) 25%, var(--surface-base)) !important;
|
||||||
|
box-shadow: 0 0 0 2px color-mix(in oklab, var(--accent-primary) 50%, transparent) inset !important;
|
||||||
|
color: var(--accent-primary) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* When a whole message is selected for deletion (via stream checkbox),
|
||||||
|
reflect that on all timeline segments for that message. */
|
||||||
|
.message-timeline-segment-delete-selected {
|
||||||
|
border-color: color-mix(in oklab, var(--status-error) 55%, transparent) !important;
|
||||||
|
background-color: color-mix(in oklab, var(--status-error) 18%, var(--surface-base)) !important;
|
||||||
|
box-shadow: 0 0 0 2px color-mix(in oklab, var(--status-error) 35%, transparent) inset !important;
|
||||||
|
color: var(--status-error) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-segment-delete-selected:hover,
|
||||||
|
.message-timeline-segment-delete-selected:focus-visible {
|
||||||
|
background-color: color-mix(in oklab, var(--status-error) 24%, var(--surface-base)) !important;
|
||||||
|
color: var(--status-error) !important;
|
||||||
|
transform: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-segment-selected:hover,
|
||||||
|
.message-timeline-segment-selected:focus-visible {
|
||||||
|
background-color: color-mix(in oklab, var(--accent-primary) 35%, var(--surface-base)) !important;
|
||||||
|
color: var(--accent-primary) !important;
|
||||||
|
transform: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* --- Group indicators: tools belong to the same message as their assistant --- */
|
||||||
|
|
||||||
|
/* Tool segments that are part of a group get a left accent border. */
|
||||||
|
.message-timeline-group-child {
|
||||||
|
border-left: 3px solid color-mix(in oklab, var(--accent-primary) 35%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The assistant "parent" at the bottom of a tool group gets the same border. */
|
||||||
|
.message-timeline-group-parent {
|
||||||
|
border-left: 3px solid color-mix(in oklab, var(--accent-primary) 35%, transparent);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Extra spacing before the first tool in a group to separate from the
|
||||||
|
preceding user/assistant badge. */
|
||||||
|
.message-timeline-group-start {
|
||||||
|
margin-top: 0.35rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Subtle extra spacing after the group parent (assistant) to separate
|
||||||
|
from the next user badge below. Uses adjacent sibling targeting. */
|
||||||
|
.message-timeline-group-parent + .message-timeline-user,
|
||||||
|
.message-timeline-group-parent + .message-timeline-compaction {
|
||||||
|
margin-top: 0.35rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-container {
|
||||||
|
position: relative;
|
||||||
|
flex: 1 1 auto;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
min-height: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-xray-overlay {
|
||||||
|
position: absolute;
|
||||||
|
inset: 0;
|
||||||
|
/* Extend the overlay box into the stream so ribs are not relying on
|
||||||
|
overflow-visible behavior (which is brittle around scroll containers). */
|
||||||
|
--xray-overhang: calc(var(--max-rib-width, 50vw) + 84px);
|
||||||
|
left: calc(-1 * var(--xray-overhang));
|
||||||
|
width: calc(100% + var(--xray-overhang));
|
||||||
|
overflow: hidden;
|
||||||
|
padding: 0.25rem;
|
||||||
|
pointer-events: none;
|
||||||
|
/* Above the scroll container background; still non-interactive. */
|
||||||
|
z-index: 2;
|
||||||
|
--xray-scroll-y: 0px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-xray-overlay-inner {
|
||||||
|
position: absolute;
|
||||||
|
inset: 0;
|
||||||
|
transform: translateY(var(--xray-scroll-y));
|
||||||
|
will-change: transform;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-xray-rib {
|
||||||
|
position: absolute;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: flex-end;
|
||||||
|
gap: 1px;
|
||||||
|
transform: translate(-100%, -50%);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-xray-token-label {
|
||||||
|
position: absolute;
|
||||||
|
right: 100%;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
margin-right: 4px;
|
||||||
|
height: 1.5rem;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 600;
|
||||||
|
font-variant-numeric: tabular-nums;
|
||||||
|
line-height: 1;
|
||||||
|
color: var(--text-primary);
|
||||||
|
background: var(--surface-base);
|
||||||
|
padding: 1px 5px;
|
||||||
|
border: 1px solid var(--border-base);
|
||||||
|
border-radius: 999px;
|
||||||
|
white-space: nowrap;
|
||||||
|
pointer-events: none;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-relative-bar {
|
||||||
|
height: 5px;
|
||||||
|
width: calc(var(--segment-weight) * var(--max-rib-width, 50vw));
|
||||||
|
background-color: color-mix(
|
||||||
|
in srgb,
|
||||||
|
var(--status-success) calc(100% - var(--segment-weight) * 100%),
|
||||||
|
var(--status-error) calc(var(--segment-weight) * 100%)
|
||||||
|
);
|
||||||
|
border-radius: 3px 0 0 3px;
|
||||||
|
transition: width 0.3s ease, background-color 0.3s ease;
|
||||||
|
box-shadow: -2px 0 4px rgba(0, 0, 0, 0.25);
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-absolute-bar {
|
||||||
|
height: 3px;
|
||||||
|
width: calc(var(--segment-weight) * var(--max-rib-width, 50vw));
|
||||||
|
background-color: var(--text-muted);
|
||||||
|
border-radius: 2px 0 0 2px;
|
||||||
|
transition: width 0.3s ease;
|
||||||
|
opacity: 0.5;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-absolute-bar-overflow {
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
|
||||||
|
.message-timeline-absolute-bar-overflow::before {
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
left: -1px;
|
||||||
|
top: -3px;
|
||||||
|
bottom: -3px;
|
||||||
|
width: 3px;
|
||||||
|
border-radius: 2px;
|
||||||
|
background: var(--status-error);
|
||||||
|
box-shadow: 0 0 6px 2px var(--status-error);
|
||||||
|
}
|
||||||
|
|||||||
@@ -21,9 +21,9 @@
|
|||||||
|
|
||||||
.tool-call-header-button {
|
.tool-call-header-button {
|
||||||
background-color: transparent;
|
background-color: transparent;
|
||||||
border: 1px solid var(--tool-call-border-color, var(--border-base));
|
|
||||||
color: var(--text-secondary);
|
color: var(--text-secondary);
|
||||||
padding: 0.15rem 0.75rem;
|
border: 0;
|
||||||
|
padding: 0.1rem 0.35rem;
|
||||||
border-radius: 0.375rem;
|
border-radius: 0.375rem;
|
||||||
font-size: 0.75rem;
|
font-size: 0.75rem;
|
||||||
font-weight: var(--font-weight-semibold);
|
font-weight: var(--font-weight-semibold);
|
||||||
@@ -37,7 +37,6 @@
|
|||||||
|
|
||||||
.tool-call-header-button:hover:not(:disabled) {
|
.tool-call-header-button:hover:not(:disabled) {
|
||||||
background-color: var(--surface-hover);
|
background-color: var(--surface-hover);
|
||||||
border-color: var(--accent-primary);
|
|
||||||
color: var(--accent-primary);
|
color: var(--accent-primary);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,12 +2,16 @@
|
|||||||
@apply flex-1 min-h-0 overflow-y-auto flex flex-col gap-0.5;
|
@apply flex-1 min-h-0 overflow-y-auto flex flex-col gap-0.5;
|
||||||
background-color: var(--surface-base);
|
background-color: var(--surface-base);
|
||||||
color: inherit;
|
color: inherit;
|
||||||
|
/* Prevent browser scroll anchoring fighting our virtualization compensation. */
|
||||||
|
overflow-anchor: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.message-stream-block {
|
.message-stream-block {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
gap: 0.0625rem;
|
gap: 0.0625rem;
|
||||||
|
|
||||||
|
contain: layout paint style;
|
||||||
}
|
}
|
||||||
|
|
||||||
.virtual-item-wrapper {
|
.virtual-item-wrapper {
|
||||||
4
packages/ui/src/types/delete-hover.ts
Normal file
4
packages/ui/src/types/delete-hover.ts
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
export type DeleteHoverState =
|
||||||
|
| { kind: "none" }
|
||||||
|
| { kind: "message"; messageId: string }
|
||||||
|
| { kind: "deleteUpTo"; messageId: string }
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
// SDK types
|
// SDK v2 types
|
||||||
import type {
|
import type {
|
||||||
EventMessageUpdated as MessageUpdateEvent,
|
EventMessageUpdated as MessageUpdateEvent,
|
||||||
EventMessageRemoved as MessageRemovedEvent,
|
EventMessageRemoved as MessageRemovedEvent,
|
||||||
@@ -6,7 +6,8 @@ import type {
|
|||||||
EventMessagePartRemoved as MessagePartRemovedEvent,
|
EventMessagePartRemoved as MessagePartRemovedEvent,
|
||||||
Part as SDKPart,
|
Part as SDKPart,
|
||||||
Message as SDKMessage,
|
Message as SDKMessage,
|
||||||
} from "@opencode-ai/sdk"
|
AssistantMessage as SDKAssistantMessageV2,
|
||||||
|
} from "@opencode-ai/sdk/v2"
|
||||||
|
|
||||||
import type { PermissionRequestLike } from "./permission"
|
import type { PermissionRequestLike } from "./permission"
|
||||||
|
|
||||||
@@ -17,7 +18,8 @@ export type {
|
|||||||
MessagePartUpdatedEvent,
|
MessagePartUpdatedEvent,
|
||||||
MessagePartRemovedEvent,
|
MessagePartRemovedEvent,
|
||||||
SDKPart,
|
SDKPart,
|
||||||
SDKMessage
|
SDKMessage,
|
||||||
|
SDKAssistantMessageV2,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Server streaming event: append-only delta updates.
|
// Server streaming event: append-only delta updates.
|
||||||
|
|||||||
Reference in New Issue
Block a user