mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-06-06 07:51:53 +00:00
Wraps + heights are column-dependent, so a width change must remeasure every row and the renderer must repaint the full viewport. - Key virtualRows on cols so React remounts wrapped rows on resize. - Snap back to bottom after sticky-mode resize once React rerenders. - Reserve a scrollbar + gap column in transcriptBodyWidth (non-termux). - Full repaint on any viewport height change (was: shrink-only). - ScrollBox scrollHeight uses deepest child bottom so sticky-bottom math can reach the real final rendered row after reflow. - DECSTBM fast-path now requires full container rect match.
282 lines
8.9 KiB
TypeScript
282 lines
8.9 KiB
TypeScript
import { PassThrough } from 'stream'
|
|
|
|
import { Box, renderSync, ScrollBox, type ScrollBoxHandle, Text } from '@hermes/ink'
|
|
import React, { useLayoutEffect, useRef } from 'react'
|
|
import { describe, expect, it } from 'vitest'
|
|
|
|
import { useVirtualHistory, virtualHistorySnapshotKey } from '../hooks/useVirtualHistory.js'
|
|
|
|
interface Item {
|
|
height: number
|
|
heightAfterResize?: number
|
|
key: string
|
|
}
|
|
|
|
interface Exposed {
|
|
scroll: ScrollBoxHandle | null
|
|
virtualHistory: ReturnType<typeof useVirtualHistory>
|
|
}
|
|
|
|
const delay = (ms: number) => new Promise(resolve => setTimeout(resolve, ms))
|
|
|
|
const makeStreams = () => {
|
|
const stdout = new PassThrough()
|
|
const stdin = new PassThrough()
|
|
const stderr = new PassThrough()
|
|
|
|
Object.assign(stdout, { columns: 80, isTTY: false, rows: 20 })
|
|
Object.assign(stdin, { isTTY: false })
|
|
Object.assign(stderr, { isTTY: false })
|
|
stdout.on('data', () => {})
|
|
|
|
return { stderr, stdin, stdout }
|
|
}
|
|
|
|
const mountedSpan = (items: readonly Item[], virtualHistory: ReturnType<typeof useVirtualHistory>) => {
|
|
let height = 0
|
|
|
|
for (let index = virtualHistory.start; index < virtualHistory.end; index++) {
|
|
height += items[index]?.height ?? 0
|
|
}
|
|
|
|
return { bottom: virtualHistory.topSpacer + height, top: virtualHistory.topSpacer }
|
|
}
|
|
|
|
const viewportIsMounted = (items: readonly Item[], virtualHistory: ReturnType<typeof useVirtualHistory>, scroll: ScrollBoxHandle) => {
|
|
const span = mountedSpan(items, virtualHistory)
|
|
const top = scroll.getScrollTop()
|
|
const bottom = top + scroll.getViewportHeight()
|
|
|
|
return top >= span.top && bottom <= span.bottom
|
|
}
|
|
|
|
const itemHeightForColumns = (item: Item | undefined, columns: number) =>
|
|
columns >= 80 ? (item?.heightAfterResize ?? item?.height ?? 1) : (item?.height ?? 1)
|
|
|
|
function Harness({
|
|
columns = 80,
|
|
expose,
|
|
height = 10,
|
|
items,
|
|
maxMounted = 16
|
|
}: {
|
|
columns?: number
|
|
expose: React.MutableRefObject<Exposed | null>
|
|
height?: number
|
|
items: readonly Item[]
|
|
maxMounted?: number
|
|
}) {
|
|
const scrollRef = useRef<ScrollBoxHandle | null>(null)
|
|
|
|
const virtualHistory = useVirtualHistory(scrollRef, items, columns, {
|
|
coldStartCount: 16,
|
|
estimateHeight: index => itemHeightForColumns(items[index], columns),
|
|
maxMounted,
|
|
overscan: 2
|
|
})
|
|
|
|
useLayoutEffect(() => {
|
|
expose.current = { scroll: scrollRef.current, virtualHistory }
|
|
})
|
|
|
|
return React.createElement(
|
|
ScrollBox,
|
|
{ flexDirection: 'column', height, ref: scrollRef, stickyScroll: true },
|
|
React.createElement(
|
|
Box,
|
|
{ flexDirection: 'column', width: '100%' },
|
|
virtualHistory.topSpacer > 0 ? React.createElement(Box, { height: virtualHistory.topSpacer }) : null,
|
|
...items
|
|
.slice(virtualHistory.start, virtualHistory.end)
|
|
.map(item =>
|
|
React.createElement(
|
|
Box,
|
|
{
|
|
height: itemHeightForColumns(item, columns),
|
|
key: item.key,
|
|
ref: virtualHistory.measureRef(item.key)
|
|
},
|
|
React.createElement(Text, null, item.key)
|
|
)
|
|
),
|
|
virtualHistory.bottomSpacer > 0 ? React.createElement(Box, { height: virtualHistory.bottomSpacer }) : null
|
|
)
|
|
)
|
|
}
|
|
|
|
describe('useVirtualHistory offset cache reuse', () => {
|
|
it('includes viewport height in the external-store snapshot key', () => {
|
|
const base = {
|
|
getPendingDelta: () => 0,
|
|
getScrollTop: () => 20,
|
|
isSticky: () => false
|
|
}
|
|
|
|
const short = virtualHistorySnapshotKey({
|
|
...base,
|
|
getViewportHeight: () => 5
|
|
} as ScrollBoxHandle)
|
|
|
|
const tall = virtualHistorySnapshotKey({
|
|
...base,
|
|
getViewportHeight: () => 25
|
|
} as ScrollBoxHandle)
|
|
|
|
expect(short).not.toBe(tall)
|
|
})
|
|
|
|
it('remounts enough tail rows after the scroll viewport grows', async () => {
|
|
const items = Array.from({ length: 100 }, (_, index) => ({ height: 1, key: `item-${index}` }))
|
|
const expose = { current: null as Exposed | null }
|
|
const streams = makeStreams()
|
|
|
|
const instance = renderSync(React.createElement(Harness, { expose, height: 4, items, maxMounted: 80 }), {
|
|
patchConsole: false,
|
|
stderr: streams.stderr as NodeJS.WriteStream,
|
|
stdin: streams.stdin as NodeJS.ReadStream,
|
|
stdout: streams.stdout as NodeJS.WriteStream
|
|
})
|
|
|
|
try {
|
|
await delay(20)
|
|
instance.rerender(React.createElement(Harness, { expose, height: 9, items, maxMounted: 80 }))
|
|
await delay(80)
|
|
|
|
expect(viewportIsMounted(items, expose.current!.virtualHistory, expose.current!.scroll!)).toBe(true)
|
|
} finally {
|
|
instance.unmount()
|
|
instance.cleanup()
|
|
}
|
|
})
|
|
|
|
it('recomputes tail coverage when wrapped rows shrink after a width resize', async () => {
|
|
const items = Array.from({ length: 100 }, (_, index) => ({
|
|
height: 4,
|
|
heightAfterResize: 1,
|
|
key: `item-${index}`
|
|
}))
|
|
|
|
const expose = { current: null as Exposed | null }
|
|
const streams = makeStreams()
|
|
|
|
const instance = renderSync(
|
|
React.createElement(Harness, { columns: 40, expose, height: 10, items, maxMounted: 80 }),
|
|
{
|
|
patchConsole: false,
|
|
stderr: streams.stderr as NodeJS.WriteStream,
|
|
stdin: streams.stdin as NodeJS.ReadStream,
|
|
stdout: streams.stdout as NodeJS.WriteStream
|
|
}
|
|
)
|
|
|
|
try {
|
|
await delay(20)
|
|
instance.rerender(React.createElement(Harness, { columns: 80, expose, height: 10, items, maxMounted: 80 }))
|
|
await delay(80)
|
|
|
|
const resizedItems = items.map(item => ({ height: item.heightAfterResize!, key: item.key }))
|
|
|
|
expect(viewportIsMounted(resizedItems, expose.current!.virtualHistory, expose.current!.scroll!)).toBe(true)
|
|
} finally {
|
|
instance.unmount()
|
|
instance.cleanup()
|
|
}
|
|
})
|
|
|
|
it('keeps sticky scroll at the bottom when one tall tail row resizes', async () => {
|
|
const items = [{ height: 90, heightAfterResize: 50, key: 'tail' }]
|
|
const expose = { current: null as Exposed | null }
|
|
const streams = makeStreams()
|
|
|
|
const instance = renderSync(
|
|
React.createElement(Harness, { columns: 70, expose, height: 18, items, maxMounted: 80 }),
|
|
{
|
|
patchConsole: false,
|
|
stderr: streams.stderr as NodeJS.WriteStream,
|
|
stdin: streams.stdin as NodeJS.ReadStream,
|
|
stdout: streams.stdout as NodeJS.WriteStream
|
|
}
|
|
)
|
|
|
|
try {
|
|
await delay(20)
|
|
instance.rerender(React.createElement(Harness, { columns: 120, expose, height: 36, items, maxMounted: 80 }))
|
|
await delay(80)
|
|
|
|
const scroll = expose.current!.scroll!
|
|
|
|
expect(scroll.getScrollTop()).toBe(scroll.getScrollHeight() - scroll.getViewportHeight())
|
|
} finally {
|
|
instance.unmount()
|
|
instance.cleanup()
|
|
}
|
|
})
|
|
|
|
it('recomputes offsets after a mounted row height changes', async () => {
|
|
const tall = [
|
|
{ height: 6, key: 'a' },
|
|
{ height: 6, key: 'b' },
|
|
{ height: 6, key: 'c' }
|
|
]
|
|
|
|
const short = tall.map(item => ({ ...item, height: 2 }))
|
|
const expose = { current: null as Exposed | null }
|
|
const streams = makeStreams()
|
|
|
|
const instance = renderSync(React.createElement(Harness, { expose, items: tall }), {
|
|
patchConsole: false,
|
|
stderr: streams.stderr as NodeJS.WriteStream,
|
|
stdin: streams.stdin as NodeJS.ReadStream,
|
|
stdout: streams.stdout as NodeJS.WriteStream
|
|
})
|
|
|
|
try {
|
|
await delay(20)
|
|
expect(expose.current!.virtualHistory.offsets[tall.length]).toBe(18)
|
|
|
|
instance.rerender(React.createElement(Harness, { expose, items: short }))
|
|
await delay(40)
|
|
|
|
expect(expose.current!.virtualHistory.offsets[short.length]).toBe(6)
|
|
expect(expose.current!.virtualHistory.bottomSpacer).toBe(0)
|
|
} finally {
|
|
instance.unmount()
|
|
instance.cleanup()
|
|
}
|
|
})
|
|
|
|
it('ignores stale reused offset-array entries after the item count shrinks', async () => {
|
|
const beforeShrink = Array.from({ length: 1400 }, (_, index) => ({ height: 1, key: `old${index}` }))
|
|
const afterShrink = Array.from({ length: 800 }, (_, index) => ({ height: 7, key: `new${index}` }))
|
|
const expose = { current: null as Exposed | null }
|
|
const streams = makeStreams()
|
|
|
|
const instance = renderSync(React.createElement(Harness, { expose, items: beforeShrink }), {
|
|
patchConsole: false,
|
|
stderr: streams.stderr as NodeJS.WriteStream,
|
|
stdin: streams.stdin as NodeJS.ReadStream,
|
|
stdout: streams.stdout as NodeJS.WriteStream
|
|
})
|
|
|
|
try {
|
|
await delay(20)
|
|
instance.rerender(React.createElement(Harness, { expose, items: afterShrink }))
|
|
await delay(20)
|
|
|
|
const scroll = expose.current!.scroll!
|
|
const transcriptHeight = expose.current!.virtualHistory.offsets[afterShrink.length] ?? 0
|
|
|
|
expect(transcriptHeight).toBe(5600)
|
|
expect(scroll.getScrollTop()).toBe(transcriptHeight - scroll.getViewportHeight())
|
|
|
|
scroll.scrollBy(-1)
|
|
await delay(80)
|
|
|
|
expect(scroll.getPendingDelta()).toBe(0)
|
|
expect(viewportIsMounted(afterShrink, expose.current!.virtualHistory, scroll)).toBe(true)
|
|
} finally {
|
|
instance.unmount()
|
|
instance.cleanup()
|
|
}
|
|
})
|
|
})
|