Infinite virtual scroller for Zero. Built on top of Tanstack Virtual.
Features:
- Bidirectional infinite scrolling (load more items at top or bottom)
- Permalink support (jump to and highlight a specific item by ID)
- State persistence (restore scroll position across navigation)
- Dynamic page sizing based on viewport
- Only fixed row heights are currently supported.
This guide explains how to add @rocicorp/zero-virtual to your own Zero app, using the demo as a reference.
A working Zero setup. See Hello Zero for a minimal starting point.
1. Install
npm install @rocicorp/zero-virtual2. Define your page and single-row queries
useZeroVirtualizer fetches rows in pages and can also look up a single row by ID for permalink support. Define these using Zero's defineQuery / defineQueries helpers. See demo/queries.ts for an example:
import {defineQueries, defineQuery} from '@rocicorp/zero';
import {zql} from './schema.ts';
export type ItemStart = Pick<Item, 'id' | 'created'>;
export const queries = defineQueries({
item: {
// Fetches a single item by ID (used for permalink resolution)
getSingleQuery: defineQuery(({args: {id}}: {args: {id: string}}) =>
zql.item.where('id', id).one(),
),
// Fetches a page of items given pagination parameters
getPageQuery: defineQuery(
({
args: {limit, start, dir},
}: {
args: {
limit: number;
start: ItemStart | null;
dir: 'forward' | 'backward';
};
}) => {
let q = zql.item
.limit(limit)
.orderBy('created', dir === 'forward' ? 'desc' : 'asc');
if (start) {
q = q.start(start, {inclusive: false});
}
return q;
},
),
},
});3. Use useZeroVirtualizer in your component
import {
useZeroVirtualizer,
useHistoryPermalinkState,
} from '@rocicorp/zero-virtual/react';
import {useCallback, useRef} from 'react';
function getRowKey(item: Item) {
return item.id;
}
function toStartRow(item: Item): ItemStart {
return {id: item.id, created: item.created};
}
export function ItemList() {
const parentRef = useRef<HTMLDivElement>(null);
const [permalinkState, setPermalinkState] =
useHistoryPermalinkState<ItemStart>();
const {virtualizer, rowAt} = useZeroVirtualizer({
listContextParams: {},
getScrollElement: useCallback(() => parentRef.current, []),
estimateSize: useCallback(() => 48, []),
getRowKey,
toStartRow,
getPageQuery: useCallback(
({limit, start, dir}) => ({
query: queries.item.getPageQuery({limit, start, dir}),
}),
[],
),
getSingleQuery: useCallback(
({id}) => ({
query: queries.item.getSingleQuery({id}),
}),
[],
),
permalinkState,
onPermalinkStateChange: setPermalinkState,
});
const virtualItems = virtualizer.getVirtualItems();
return (
<div ref={parentRef} style={{overflow: 'auto', height: '100vh'}}>
<div style={{height: virtualizer.getTotalSize(), position: 'relative'}}>
{virtualItems.map(virtualRow => {
const row = rowAt(virtualRow.index);
return (
<div
key={virtualRow.key}
data-index={virtualRow.index}
style={{
position: 'absolute',
transform: `translateY(${virtualRow.start}px)`,
}}
>
{row ? row.title : 'Loading...'}
</div>
);
})}
</div>
</div>
);
}Query functions receive an options object and return a QueryResult:
type GetPageQueryOptions<TStartRow> = {
limit: number;
start: TStartRow | null;
dir: 'forward' | 'backward';
settled: boolean;
};
type GetSingleQueryOptions = {
id: string;
settled: boolean;
};
type QueryResult<TReturn> = {query: ...; options?: UseQueryOptions};The settled flag indicates whether the list has been idle for settleTime ms (default 2000). Use this to vary query options based on scroll state — for example, using a shorter TTL while scrolling and a longer one when settled:
getPageQuery: ({limit, start, dir, settled}) => ({
query: queries.item.getPageQuery({limit, start, dir}),
options: {ttl: settled ? '5m' : '10s'},
}),useZeroVirtualizer tracks whether the user has stopped scrolling:
settled(returned) —truewhen the list has been idle forsettleTimemssettleTime(option) — how long to wait before considering the list settled (default 2000ms)onSettled(option) — callback fired whensettledtransitions totrue, useful for deferred side effects like syncing search params to the URL
A ready-made hook that persists virtualizer scroll/pagination state in window.history.state, so back/forward navigation restores position automatically:
const [permalinkState, setPermalinkState] =
useHistoryPermalinkState<MyStartRow>();Pass a custom key if you have multiple virtualizers on the same page:
const [state, setState] = useHistoryPermalinkState<MyStartRow>('myList');For a complete working example including sorting, permalinks, and scroll-position persistence, see demo/App.tsx.
First, install dependencies from the repo root:
pnpm iThen cd into the demo directory for the remaining steps:
cd demoRun Docker:
pnpm dev:db-upIn a second terminal, run the zero-cache server:
cd demo
pnpm dev:zero-cacheIn a third terminal, run the Vite dev server:
cd demo
pnpm dev:uiReleases are published to npm automatically via CI when a version tag is pushed. To cut a new patch release:
git checkout main
git reset --hard origin/main
pnpm version patch
git push
git push --tagsUse pnpm version minor or pnpm version major for larger releases.