{"version":3,"file":"index.mjs","sources":["../../src/src/core/utils.ts","../../src/src/core/cache.ts","../../src/src/core/environment.ts","../../src/src/core/store.ts","../../src/src/core/scroller.ts","../../src/src/core/resizer.ts","../../src/src/solid/ListItem.tsx","../../src/src/solid/RangedFor.tsx","../../src/src/solid/utils.ts","../../src/src/solid/Virtualizer.tsx","../../src/src/solid/VList.tsx","../../src/src/solid/WindowVirtualizer.tsx"],"sourcesContent":["/** @internal */\nexport const NULL = null;\n\n/** @internal */\nexport const { min, max, abs, floor } = Math;\n\n/**\n * @internal\n */\nexport const clamp = (\n  value: number,\n  minValue: number,\n  maxValue: number\n): number => min(maxValue, max(minValue, value));\n\n/**\n * @internal\n */\nexport const sort = <T extends number>(arr: readonly T[]): T[] => {\n  return [...arr].sort((a, b) => a - b);\n};\n\n/**\n * @internal\n */\nexport const microtask: (fn: () => void) => void =\n  typeof queueMicrotask === \"function\"\n    ? queueMicrotask\n    : (fn) => {\n        Promise.resolve().then(fn);\n      };\n\n/**\n * @internal\n */\nexport const once = <V>(fn: () => V): (() => V) => {\n  let called: undefined | boolean;\n  let cache: V;\n\n  return () => {\n    if (!called) {\n      called = true;\n      cache = fn();\n    }\n    return cache;\n  };\n};\n","import { type InternalCacheSnapshot, type ItemsRange } from \"./types\";\nimport { clamp, floor, max, min, sort } from \"./utils\";\n\ntype Writeable<T> = {\n  -readonly [key in keyof T]: Writeable<T[key]>;\n};\n\n/** @internal */\nexport const UNCACHED = -1;\n\n/**\n * @internal\n */\nexport type Cache = {\n  readonly _length: number;\n  // sizes\n  readonly _sizes: number[];\n  readonly _defaultItemSize: number;\n  // offsets\n  readonly _computedOffsetIndex: number;\n  readonly _offsets: number[];\n};\n\nconst fill = (array: number[], length: number, prepend?: boolean): number[] => {\n  const key = prepend ? \"unshift\" : \"push\";\n  for (let i = 0; i < length; i++) {\n    array[key](UNCACHED);\n  }\n  return array;\n};\n\n/**\n * @internal\n */\nexport const getItemSize = (cache: Cache, index: number): number => {\n  const size = cache._sizes[index]!;\n  return size === UNCACHED ? cache._defaultItemSize : size;\n};\n\n/**\n * @internal\n */\nexport const setItemSize = (\n  cache: Writeable<Cache>,\n  index: number,\n  size: number\n): boolean => {\n  const isInitialMeasurement = cache._sizes[index] === UNCACHED;\n  cache._sizes[index] = size;\n  // mark as dirty\n  cache._computedOffsetIndex = min(index, cache._computedOffsetIndex);\n  return isInitialMeasurement;\n};\n\n/**\n * @internal\n */\nexport const computeOffset = (\n  cache: Writeable<Cache>,\n  index: number\n): number => {\n  if (!cache._length) return 0;\n  if (cache._computedOffsetIndex >= index) {\n    return cache._offsets[index]!;\n  }\n\n  if (cache._computedOffsetIndex < 0) {\n    // first offset must be 0 to avoid returning NaN, which can cause infinite rerender.\n    // https://github.com/inokawa/virtua/pull/160\n    cache._offsets[0] = 0;\n    cache._computedOffsetIndex = 0;\n  }\n  let i = cache._computedOffsetIndex;\n  let top = cache._offsets[i]!;\n  while (i < index) {\n    top += getItemSize(cache, i);\n    cache._offsets[++i] = top;\n  }\n  // mark as measured\n  cache._computedOffsetIndex = index;\n  return top;\n};\n\n/**\n * @internal\n */\nexport const computeTotalSize = (cache: Cache): number => {\n  if (!cache._length) return 0;\n  return (\n    computeOffset(cache, cache._length - 1) +\n    getItemSize(cache, cache._length - 1)\n  );\n};\n\n/**\n * Finds the index of an item in the cache whose computed offset is closest to the specified offset.\n *\n * @internal\n */\nexport const findIndex = (\n  cache: Cache,\n  offset: number,\n  low: number = 0,\n  high: number = cache._length - 1\n): number => {\n  // Find with binary search\n  while (low <= high) {\n    const mid = floor((low + high) / 2);\n    const itemOffset = computeOffset(cache, mid);\n    if (itemOffset <= offset) {\n      if (itemOffset + getItemSize(cache, mid) > offset) {\n        return mid;\n      }\n      low = mid + 1;\n    } else {\n      high = mid - 1;\n    }\n  }\n  return clamp(low, 0, cache._length - 1);\n};\n\n/**\n * @internal\n */\nexport const computeRange = (\n  cache: Cache,\n  scrollOffset: number,\n  viewportSize: number,\n  prevStartIndex: number\n): ItemsRange => {\n  // Clamp because prevStartIndex may exceed the limit when children decreased a lot after scrolling\n  prevStartIndex = min(prevStartIndex, cache._length - 1);\n\n  if (computeOffset(cache, prevStartIndex) <= scrollOffset) {\n    // search forward\n    // start <= end, prevStartIndex <= start\n    const end = findIndex(cache, scrollOffset + viewportSize, prevStartIndex);\n    return [findIndex(cache, scrollOffset, prevStartIndex, end), end];\n  } else {\n    // search backward\n    // start <= end, start <= prevStartIndex\n    const start = findIndex(cache, scrollOffset, undefined, prevStartIndex);\n    return [start, findIndex(cache, scrollOffset + viewportSize, start)];\n  }\n};\n\n/**\n * @internal\n */\nexport const estimateDefaultItemSize = (\n  cache: Writeable<Cache>,\n  startIndex: number\n): number => {\n  let measuredCountBeforeStart = 0;\n  // This function will be called after measurement so measured size array must be longer than 0\n  const measuredSizes: number[] = [];\n  cache._sizes.forEach((s, i) => {\n    if (s !== UNCACHED) {\n      measuredSizes.push(s);\n      if (i < startIndex) {\n        measuredCountBeforeStart++;\n      }\n    }\n  });\n\n  // Discard cache for now\n  cache._computedOffsetIndex = -1;\n\n  // Calculate median\n  const sorted = sort(measuredSizes);\n  const len = sorted.length;\n  const mid = (len / 2) | 0;\n  const median =\n    len % 2 === 0 ? (sorted[mid - 1]! + sorted[mid]!) / 2 : sorted[mid]!;\n\n  const prevDefaultItemSize = cache._defaultItemSize;\n\n  // Calculate diff of unmeasured items before start\n  return (\n    ((cache._defaultItemSize = median) - prevDefaultItemSize) *\n    max(startIndex - measuredCountBeforeStart, 0)\n  );\n};\n\n/**\n * @internal\n */\nexport const initCache = (\n  length: number,\n  itemSize: number,\n  snapshot?: InternalCacheSnapshot\n): Cache => {\n  return {\n    _defaultItemSize: snapshot ? snapshot[1] : itemSize,\n    _sizes:\n      snapshot && snapshot[0]\n        ? // https://github.com/inokawa/virtua/issues/441\n          fill(\n            snapshot[0].slice(0, min(length, snapshot[0].length)),\n            max(0, length - snapshot[0].length)\n          )\n        : fill([], length),\n    _length: length,\n    _computedOffsetIndex: -1,\n    _offsets: fill([], length),\n  };\n};\n\n/**\n * @internal\n */\nexport const takeCacheSnapshot = (cache: Cache): InternalCacheSnapshot => {\n  return [cache._sizes.slice(), cache._defaultItemSize];\n};\n\n/**\n * @internal\n */\nexport const updateCacheLength = (\n  cache: Writeable<Cache>,\n  length: number,\n  isShift?: boolean\n): number => {\n  const diff = length - cache._length;\n\n  cache._computedOffsetIndex = isShift\n    ? // Discard cache for now\n      -1\n    : min(length - 1, cache._computedOffsetIndex);\n  cache._length = length;\n\n  if (diff > 0) {\n    // Added\n    fill(cache._offsets, diff);\n    fill(cache._sizes, diff, isShift);\n    return cache._defaultItemSize * diff;\n  } else {\n    // Removed\n    cache._offsets.splice(diff);\n    return (\n      isShift ? cache._sizes.splice(0, -diff) : cache._sizes.splice(diff)\n    ).reduce(\n      (acc, removed) =>\n        acc - (removed === UNCACHED ? cache._defaultItemSize : removed),\n      0\n    );\n  }\n};\n","import { once } from \"./utils\";\n\n/**\n * @internal\n */\nexport const isBrowser = typeof window !== \"undefined\";\n\nconst getDocumentElement = () => document.documentElement;\n\n/**\n * @internal\n */\nexport const getCurrentDocument = (node: HTMLElement): Document =>\n  node.ownerDocument;\n\n/**\n * @internal\n */\nexport const getCurrentWindow = (doc: Document) => doc.defaultView!;\n\n/**\n * @internal\n */\nexport const isRTLDocument = /*#__PURE__*/ once((): boolean => {\n  // TODO support SSR in rtl\n  return isBrowser\n    ? getComputedStyle(getDocumentElement()).direction === \"rtl\"\n    : false;\n});\n\n/**\n * Currently, all browsers on iOS/iPadOS are WebKit, including WebView.\n * @internal\n */\nexport const isIOSWebKit = /*#__PURE__*/ once((): boolean => {\n  return /iP(hone|od|ad)/.test(navigator.userAgent);\n});\n\n/**\n * @internal\n */\nexport const isSmoothScrollSupported = /*#__PURE__*/ once((): boolean => {\n  return \"scrollBehavior\" in getDocumentElement().style;\n});\n","import {\n  initCache,\n  getItemSize as _getItemSize,\n  computeTotalSize,\n  computeOffset as computeStartOffset,\n  UNCACHED,\n  setItemSize,\n  estimateDefaultItemSize,\n  updateCacheLength,\n  computeRange,\n  takeCacheSnapshot,\n  findIndex,\n} from \"./cache\";\nimport { isIOSWebKit } from \"./environment\";\nimport type {\n  CacheSnapshot,\n  InternalCacheSnapshot,\n  ItemResize,\n  ItemsRange,\n} from \"./types\";\nimport { abs, max, min, NULL } from \"./utils\";\n\nconst MAX_INT_32 = 0x7fffffff;\n\nconst SCROLL_IDLE = 0;\nconst SCROLL_DOWN = 1;\nconst SCROLL_UP = 2;\ntype ScrollDirection =\n  | typeof SCROLL_IDLE\n  | typeof SCROLL_DOWN\n  | typeof SCROLL_UP;\n\nconst SCROLL_BY_NATIVE = 0;\nconst SCROLL_BY_MANUAL_SCROLL = 1;\nconst SCROLL_BY_SHIFT = 2;\ntype ScrollMode =\n  | typeof SCROLL_BY_NATIVE\n  | typeof SCROLL_BY_MANUAL_SCROLL\n  | typeof SCROLL_BY_SHIFT;\n\n/** @internal */\nexport const ACTION_SCROLL = 1;\n/** @internal */\nexport const ACTION_SCROLL_END = 2;\n/** @internal */\nexport const ACTION_ITEM_RESIZE = 3;\n/** @internal */\nexport const ACTION_VIEWPORT_RESIZE = 4;\n/** @internal */\nexport const ACTION_ITEMS_LENGTH_CHANGE = 5;\n/** @internal */\nexport const ACTION_START_OFFSET_CHANGE = 6;\n/** @internal */\nexport const ACTION_MANUAL_SCROLL = 7;\n/** @internal */\nexport const ACTION_BEFORE_MANUAL_SMOOTH_SCROLL = 8;\n\ntype Actions =\n  | [type: typeof ACTION_SCROLL, offset: number]\n  | [type: typeof ACTION_SCROLL_END, dummy?: void]\n  | [type: typeof ACTION_ITEM_RESIZE, entries: ItemResize[]]\n  | [type: typeof ACTION_VIEWPORT_RESIZE, size: number]\n  | [\n      type: typeof ACTION_ITEMS_LENGTH_CHANGE,\n      arg: [length: number, isShift?: boolean | undefined],\n    ]\n  | [type: typeof ACTION_START_OFFSET_CHANGE, offset: number]\n  | [type: typeof ACTION_MANUAL_SCROLL, dummy?: void]\n  | [type: typeof ACTION_BEFORE_MANUAL_SMOOTH_SCROLL, offset: number];\n\n/** @internal */\nexport const UPDATE_VIRTUAL_STATE = 0b0001;\n/** @internal */\nexport const UPDATE_SIZE_EVENT = 0b0010;\n/** @internal */\nexport const UPDATE_SCROLL_EVENT = 0b0100;\n/** @internal */\nexport const UPDATE_SCROLL_END_EVENT = 0b1000;\n\n/**\n * @internal\n */\nexport const getScrollSize = (store: VirtualStore): number => {\n  return max(store.$getTotalSize(), store.$getViewportSize());\n};\n\n/**\n * @internal\n */\nexport const isInitialMeasurementDone = (store: VirtualStore): boolean => {\n  return !!store.$getViewportSize();\n};\n\ntype Subscriber = (sync?: boolean) => void;\n\n/** @internal */\nexport type StateVersion =\n  number & {} /* hack for typescript to pretend as not falsy */;\n\n/**\n * @internal\n */\nexport type VirtualStore = {\n  $getStateVersion(): StateVersion;\n  $getCacheSnapshot(): CacheSnapshot;\n  $getRange(): ItemsRange;\n  $findStartIndex(): number;\n  $findEndIndex(): number;\n  $isUnmeasuredItem(index: number): boolean;\n  $getItemOffset(index: number): number;\n  $getItemSize(index: number): number;\n  $getItemsLength(): number;\n  $getScrollOffset(): number;\n  $isScrolling(): boolean;\n  $getViewportSize(): number;\n  $getStartSpacerSize(): number;\n  $getTotalSize(): number;\n  _flushJump(): [number, boolean];\n  $subscribe(target: number, cb: Subscriber): () => void;\n  $update(...action: Actions): void;\n  _hasUnmeasuredItemsInFrozenRange(): boolean;\n};\n\n/**\n * @internal\n */\nexport const createVirtualStore = (\n  elementsCount: number,\n  itemSize: number = 40,\n  overscan: number = 4,\n  ssrCount: number = 0,\n  cacheSnapshot?: CacheSnapshot | undefined,\n  shouldAutoEstimateItemSize: boolean = false\n): VirtualStore => {\n  let isSSR = !!ssrCount;\n  let stateVersion: StateVersion = 1;\n  let viewportSize = 0;\n  let startSpacerSize = 0;\n  let scrollOffset = 0;\n  let jump = 0;\n  let pendingJump = 0;\n  let _flushedJump = 0;\n  let _scrollDirection: ScrollDirection = SCROLL_IDLE;\n  let _scrollMode: ScrollMode = SCROLL_BY_NATIVE;\n  let _frozenRange: ItemsRange | null = isSSR\n    ? [0, max(ssrCount - 1, 0)]\n    : NULL;\n  let _prevRange: ItemsRange = [0, 0];\n  let _totalMeasuredSize = 0;\n\n  const cache = initCache(\n    elementsCount,\n    itemSize,\n    cacheSnapshot as unknown as InternalCacheSnapshot | undefined\n  );\n  const subscribers = new Set<[number, Subscriber]>();\n  const getRelativeScrollOffset = () => scrollOffset - startSpacerSize;\n  const getVisibleOffset = () => getRelativeScrollOffset() + pendingJump + jump;\n  const getRange = (offset: number) => {\n    return computeRange(cache, offset, viewportSize, _prevRange[0]);\n  };\n  const getTotalSize = (): number => computeTotalSize(cache);\n  const getItemOffset = (index: number): number => {\n    return computeStartOffset(cache, index) - pendingJump;\n  };\n  const getItemSize = (index: number): number => {\n    return _getItemSize(cache, index);\n  };\n\n  const applyJump = (j: number) => {\n    if (j) {\n      // In iOS WebKit browsers, updating scroll position will stop scrolling so it have to be deferred during scrolling.\n      if (isIOSWebKit() && _scrollDirection !== SCROLL_IDLE) {\n        pendingJump += j;\n      } else {\n        jump += j;\n      }\n    }\n  };\n\n  return {\n    $getStateVersion: () => stateVersion,\n    $getCacheSnapshot: () => {\n      return takeCacheSnapshot(cache) as unknown as CacheSnapshot;\n    },\n    $getRange: () => {\n      let startIndex: number;\n      let endIndex: number;\n      if (_flushedJump) {\n        // Return previous range for consistent render until next scroll event comes in.\n        // And it must be clamped. https://github.com/inokawa/virtua/issues/597\n        [startIndex, endIndex] = _prevRange;\n      } else {\n        [startIndex, endIndex] = _prevRange = getRange(\n          max(0, getVisibleOffset())\n        );\n        if (_frozenRange) {\n          startIndex = min(startIndex, _frozenRange[0]);\n          endIndex = max(endIndex, _frozenRange[1]);\n        }\n      }\n\n      if (_scrollDirection !== SCROLL_DOWN) {\n        startIndex -= max(0, overscan);\n      }\n      if (_scrollDirection !== SCROLL_UP) {\n        endIndex += max(0, overscan);\n      }\n      return [max(startIndex, 0), min(endIndex, cache._length - 1)];\n    },\n    $findStartIndex: () => findIndex(cache, getVisibleOffset()),\n    $findEndIndex: () => findIndex(cache, getVisibleOffset() + viewportSize),\n    $isUnmeasuredItem: (index) => cache._sizes[index] === UNCACHED,\n    _hasUnmeasuredItemsInFrozenRange: () => {\n      if (!_frozenRange) return false;\n      return cache._sizes\n        .slice(\n          max(0, _frozenRange[0] - 1),\n          min(cache._length - 1, _frozenRange[1] + 1) + 1\n        )\n        .includes(UNCACHED);\n    },\n    $getItemOffset: getItemOffset,\n    $getItemSize: getItemSize,\n    $getItemsLength: () => cache._length,\n    $getScrollOffset: () => scrollOffset,\n    $isScrolling: () => _scrollDirection !== SCROLL_IDLE,\n    $getViewportSize: () => viewportSize,\n    $getStartSpacerSize: () => startSpacerSize,\n    $getTotalSize: getTotalSize,\n    _flushJump: () => {\n      _flushedJump = jump;\n      jump = 0;\n      return [\n        _flushedJump,\n        // Use absolute position not to exceed scrollable bounds\n        _scrollMode === SCROLL_BY_SHIFT ||\n          // https://github.com/inokawa/virtua/discussions/475\n          getRelativeScrollOffset() + viewportSize >= getTotalSize(),\n      ];\n    },\n    $subscribe: (target, cb) => {\n      const sub: [number, Subscriber] = [target, cb];\n      subscribers.add(sub);\n      return () => {\n        subscribers.delete(sub);\n      };\n    },\n    $update: (type, payload): void => {\n      let shouldFlushPendingJump: boolean | undefined;\n      let shouldSync: boolean | undefined;\n      let mutated = 0;\n\n      switch (type) {\n        case ACTION_SCROLL: {\n          const flushedJump = _flushedJump;\n          _flushedJump = 0;\n\n          const delta = payload - scrollOffset;\n          const distance = abs(delta);\n\n          // Scroll event after jump compensation is not reliable because it may result in the opposite direction.\n          // The delta of artificial scroll may not be equal with the jump because it may be batched with other scrolls.\n          // And at least in latest Chrome/Firefox/Safari in 2023, setting value to scrollTop/scrollLeft can lose subpixel because its integer (sometimes float probably depending on dpr).\n          const isJustJumped = flushedJump && distance < abs(flushedJump) + 1;\n\n          // Scroll events are dispatched enough so it's ok to skip some of them.\n          if (\n            !isJustJumped &&\n            // Ignore until manual scrolling\n            _scrollMode === SCROLL_BY_NATIVE\n          ) {\n            _scrollDirection = delta < 0 ? SCROLL_UP : SCROLL_DOWN;\n          }\n\n          // TODO This will cause glitch in reverse infinite scrolling. Disable this until better solution is found.\n          // if (\n          //   pendingJump &&\n          //   ((_scrollDirection === SCROLL_UP &&\n          //     payload - max(pendingJump, 0) <= 0) ||\n          //     (_scrollDirection === SCROLL_DOWN &&\n          //       payload - min(pendingJump, 0) >= getScrollOffsetMax()))\n          // ) {\n          //   // Flush if almost reached to start or end\n          //   shouldFlushPendingJump = true;\n          // }\n\n          if (isSSR) {\n            _frozenRange = NULL;\n            isSSR = false;\n          }\n\n          scrollOffset = payload;\n          mutated = UPDATE_SCROLL_EVENT;\n\n          // Skip if offset is not changed\n          // Scroll offset may exceed min or max especially in Safari's elastic scrolling.\n          const relativeOffset = getRelativeScrollOffset();\n          if (\n            relativeOffset >= -viewportSize &&\n            relativeOffset <= getTotalSize()\n          ) {\n            mutated += UPDATE_VIRTUAL_STATE;\n\n            // Update synchronously if scrolled a lot\n            shouldSync = distance > viewportSize;\n          }\n          break;\n        }\n        case ACTION_SCROLL_END: {\n          mutated = UPDATE_SCROLL_END_EVENT;\n          if (_scrollDirection !== SCROLL_IDLE) {\n            shouldFlushPendingJump = true;\n            mutated += UPDATE_VIRTUAL_STATE;\n          }\n          _scrollDirection = SCROLL_IDLE;\n          _scrollMode = SCROLL_BY_NATIVE;\n          _frozenRange = NULL;\n          break;\n        }\n        case ACTION_ITEM_RESIZE: {\n          const updated = payload.filter(\n            ([index, size]) => cache._sizes[index] !== size\n          );\n\n          // Skip if all items are cached and not updated\n          if (!updated.length) {\n            break;\n          }\n\n          // Calculate jump by resize to minimize junks in appearance\n          applyJump(\n            updated.reduce((acc, [index, size]) => {\n              if (\n                // Keep distance from end during shifting\n                _scrollMode === SCROLL_BY_SHIFT ||\n                (_frozenRange\n                  ? // https://github.com/inokawa/virtua/issues/380\n                    // https://github.com/inokawa/virtua/issues/590\n                    !isSSR && index < _frozenRange[0]\n                  : // Otherwise we should maintain visible position\n                    getItemOffset(index) +\n                      // https://github.com/inokawa/virtua/issues/385\n                      (_scrollDirection === SCROLL_IDLE &&\n                      _scrollMode === SCROLL_BY_NATIVE\n                        ? getItemSize(index)\n                        : 0) <\n                    getRelativeScrollOffset())\n              ) {\n                acc += size - getItemSize(index);\n              }\n              return acc;\n            }, 0)\n          );\n\n          // Update item sizes\n          for (const [index, size] of updated) {\n            const prevSize = getItemSize(index);\n            const isInitialMeasurement = setItemSize(cache, index, size);\n\n            if (shouldAutoEstimateItemSize) {\n              _totalMeasuredSize += isInitialMeasurement\n                ? size\n                : size - prevSize;\n            }\n          }\n\n          // Estimate initial item size from measured sizes\n          if (\n            shouldAutoEstimateItemSize &&\n            viewportSize &&\n            // If the total size is lower than the viewport, the item may be a empty state\n            _totalMeasuredSize > viewportSize\n          ) {\n            applyJump(\n              estimateDefaultItemSize(\n                cache,\n                findIndex(cache, getVisibleOffset())\n              )\n            );\n            shouldAutoEstimateItemSize = false;\n          }\n\n          mutated = UPDATE_VIRTUAL_STATE + UPDATE_SIZE_EVENT;\n\n          // Synchronous update is necessary in current design to minimize visible glitch in concurrent rendering.\n          // However this seems to be the main cause of the errors from ResizeObserver.\n          // https://github.com/inokawa/virtua/issues/470\n          //\n          // And in React, synchronous update with flushSync after asynchronous update will overtake the asynchronous one.\n          // If items resize happens just after scroll, race condition can occur depending on implementation.\n          shouldSync = true;\n          break;\n        }\n        case ACTION_VIEWPORT_RESIZE: {\n          if (viewportSize !== payload) {\n            viewportSize = payload;\n            mutated = UPDATE_VIRTUAL_STATE + UPDATE_SIZE_EVENT;\n          }\n          break;\n        }\n        case ACTION_ITEMS_LENGTH_CHANGE: {\n          if (payload[1]) {\n            applyJump(updateCacheLength(cache, payload[0], true));\n            _scrollMode = SCROLL_BY_SHIFT;\n            mutated = UPDATE_VIRTUAL_STATE;\n          } else {\n            updateCacheLength(cache, payload[0]);\n            // https://github.com/inokawa/virtua/issues/552\n            // https://github.com/inokawa/virtua/issues/557\n            mutated = UPDATE_VIRTUAL_STATE;\n          }\n          break;\n        }\n        case ACTION_START_OFFSET_CHANGE: {\n          startSpacerSize = payload;\n          break;\n        }\n        case ACTION_MANUAL_SCROLL: {\n          _scrollMode = SCROLL_BY_MANUAL_SCROLL;\n          break;\n        }\n        case ACTION_BEFORE_MANUAL_SMOOTH_SCROLL: {\n          _frozenRange = getRange(payload);\n          mutated = UPDATE_VIRTUAL_STATE;\n          break;\n        }\n      }\n\n      if (mutated) {\n        stateVersion = (stateVersion & MAX_INT_32) + 1;\n\n        if (shouldFlushPendingJump && pendingJump) {\n          jump += pendingJump;\n          pendingJump = 0;\n        }\n\n        subscribers.forEach(([target, cb]) => {\n          // Early return to skip React's computation\n          if (!(mutated & target)) {\n            return;\n          }\n          // https://github.com/facebook/react/issues/25191\n          // https://github.com/facebook/react/blob/a5fc797db14c6e05d4d5c4dbb22a0dd70d41f5d5/packages/react-reconciler/src/ReactFiberWorkLoop.js#L1443-L1447\n          cb(shouldSync);\n        });\n      }\n    },\n  };\n};\n","import {\n  getCurrentDocument,\n  getCurrentWindow,\n  isIOSWebKit,\n  isRTLDocument,\n  isSmoothScrollSupported,\n} from \"./environment\";\nimport {\n  ACTION_SCROLL,\n  type VirtualStore,\n  ACTION_SCROLL_END,\n  UPDATE_SIZE_EVENT,\n  ACTION_MANUAL_SCROLL,\n  ACTION_BEFORE_MANUAL_SMOOTH_SCROLL,\n  ACTION_START_OFFSET_CHANGE,\n  isInitialMeasurementDone,\n} from \"./store\";\nimport { type ScrollToIndexOpts } from \"./types\";\nimport { clamp, microtask, NULL } from \"./utils\";\n\nconst timeout = setTimeout;\n\nconst debounce = <T extends () => void>(fn: T, ms: number) => {\n  let id: ReturnType<typeof setTimeout> | undefined | null;\n\n  const cancel = () => {\n    if (id != NULL) {\n      clearTimeout(id);\n    }\n  };\n  const debouncedFn = () => {\n    cancel();\n    id = timeout(() => {\n      id = NULL;\n      fn();\n    }, ms);\n  };\n  debouncedFn._cancel = cancel;\n  return debouncedFn;\n};\n\n/**\n * scrollLeft is negative value in rtl direction.\n *\n * left  right\n * 0     100    spec compliant (ltr)\n * -100  0      spec compliant (rtl)\n * https://github.com/othree/jquery.rtl-scroll-type\n */\nconst normalizeOffset = (offset: number, isHorizontal: boolean): number => {\n  if (isHorizontal && isRTLDocument()) {\n    return -offset;\n  } else {\n    return offset;\n  }\n};\n\nconst createScrollObserver = (\n  store: VirtualStore,\n  viewport: HTMLElement | Window,\n  isHorizontal: boolean,\n  getScrollOffset: () => number,\n  updateScrollOffset: (\n    value: number,\n    shift: boolean,\n    isMomentumScrolling: boolean\n  ) => void,\n  getStartOffset?: () => number\n) => {\n  const now = Date.now;\n\n  let lastScrollTime = 0;\n  let wheeling = false;\n  let touching = false;\n  let justTouchEnded = false;\n  let stillMomentumScrolling = false;\n\n  const onScrollEnd = debounce(() => {\n    if (wheeling || touching) {\n      wheeling = false;\n\n      // Wait while wheeling or touching\n      onScrollEnd();\n      return;\n    }\n\n    justTouchEnded = false;\n\n    store.$update(ACTION_SCROLL_END);\n  }, 150);\n\n  const onScroll = () => {\n    lastScrollTime = now();\n\n    if (justTouchEnded) {\n      stillMomentumScrolling = true;\n    }\n\n    if (getStartOffset) {\n      store.$update(ACTION_START_OFFSET_CHANGE, getStartOffset());\n    }\n    store.$update(ACTION_SCROLL, getScrollOffset());\n\n    onScrollEnd();\n  };\n\n  // Infer scroll state also from wheel events\n  // Sometimes scroll events do not fire when frame dropped even if the visual have been already scrolled\n  const onWheel = ((e: WheelEvent) => {\n    if (\n      wheeling ||\n      // Scroll start should be detected with scroll event\n      !store.$isScrolling() ||\n      // Probably a pinch-to-zoom gesture\n      e.ctrlKey\n    ) {\n      return;\n    }\n\n    const timeDelta = now() - lastScrollTime;\n    if (\n      // Check if wheel event occurs some time after scrolling\n      150 > timeDelta &&\n      50 < timeDelta &&\n      // Get delta before checking deltaMode for firefox behavior\n      // https://github.com/w3c/uievents/issues/181#issuecomment-392648065\n      // https://bugzilla.mozilla.org/show_bug.cgi?id=1392460#c34\n      (isHorizontal ? e.deltaX : e.deltaY)\n    ) {\n      wheeling = true;\n    }\n  }) as (e: Event) => void; // FIXME type error. why only here?\n\n  const onTouchStart = () => {\n    touching = true;\n    justTouchEnded = stillMomentumScrolling = false;\n  };\n  const onTouchEnd = () => {\n    touching = false;\n    if (isIOSWebKit()) {\n      justTouchEnded = true;\n    }\n  };\n\n  viewport.addEventListener(\"scroll\", onScroll);\n  viewport.addEventListener(\"wheel\", onWheel, { passive: true });\n  viewport.addEventListener(\"touchstart\", onTouchStart, { passive: true });\n  viewport.addEventListener(\"touchend\", onTouchEnd, { passive: true });\n\n  return {\n    _dispose: () => {\n      viewport.removeEventListener(\"scroll\", onScroll);\n      viewport.removeEventListener(\"wheel\", onWheel);\n      viewport.removeEventListener(\"touchstart\", onTouchStart);\n      viewport.removeEventListener(\"touchend\", onTouchEnd);\n      onScrollEnd._cancel();\n    },\n    _fixScrollJump: () => {\n      const [jump, shift] = store._flushJump();\n      if (!jump) return;\n      updateScrollOffset(\n        normalizeOffset(jump, isHorizontal),\n        shift,\n        stillMomentumScrolling\n      );\n      stillMomentumScrolling = false;\n\n      if (shift && store.$getViewportSize() > store.$getTotalSize()) {\n        // In this case applying jump may not cause scroll.\n        // Current logic expects scroll event occurs after applying jump so we dispatch it manually.\n        store.$update(ACTION_SCROLL, getScrollOffset());\n      }\n    },\n  };\n};\n\ntype ScrollObserver = ReturnType<typeof createScrollObserver>;\n\n/**\n * @internal\n */\nexport type Scroller = {\n  $observe: (viewportElement: HTMLElement) => void;\n  $dispose(): void;\n  $scrollTo: (offset: number) => void;\n  $scrollBy: (offset: number) => void;\n  $scrollToIndex: (index: number, opts?: ScrollToIndexOpts) => void;\n  $fixScrollJump: () => void;\n};\n\n/**\n * @internal\n */\nexport const createScroller = (\n  store: VirtualStore,\n  isHorizontal: boolean\n): Scroller => {\n  let viewportElement: HTMLElement | undefined;\n  let scrollObserver: ScrollObserver | undefined;\n  let cancelScroll: (() => void) | undefined;\n  const scrollOffsetKey = isHorizontal ? \"scrollLeft\" : \"scrollTop\";\n  const overflowKey = isHorizontal ? \"overflowX\" : \"overflowY\";\n\n  // The given offset will be clamped by browser\n  // https://drafts.csswg.org/cssom-view/#dom-element-scrolltop\n  const scheduleImperativeScroll = async (\n    getTargetOffset: () => number,\n    smooth?: boolean\n  ) => {\n    if (!viewportElement) {\n      // Wait for element assign. The element may be undefined if scrollRef prop is used and scroll is scheduled on mount.\n      microtask(() => scheduleImperativeScroll(getTargetOffset, smooth));\n      return;\n    }\n\n    if (cancelScroll) {\n      // Cancel waiting scrollTo\n      cancelScroll();\n    }\n\n    const waitForMeasurement = (): [Promise<void>, () => void] => {\n      // Wait for the scroll destination items to be measured.\n      // The measurement will be done asynchronously and the timing is not predictable so we use promise.\n      let queue: (() => void) | undefined;\n      return [\n        new Promise<void>((resolve, reject) => {\n          queue = resolve;\n          cancelScroll = reject;\n\n          // Resize event may not happen when the window/tab is not visible, or during browser back in Safari.\n          // We have to wait for the initial measurement to avoid failing imperative scroll on mount.\n          // https://github.com/inokawa/virtua/issues/450\n          if (isInitialMeasurementDone(store)) {\n            // Reject when items around scroll destination completely measured\n            timeout(reject, 150);\n          }\n        }),\n        store.$subscribe(UPDATE_SIZE_EVENT, () => {\n          queue && queue();\n        }),\n      ];\n    };\n\n    if (smooth && isSmoothScrollSupported()) {\n      while (true) {\n        store.$update(ACTION_BEFORE_MANUAL_SMOOTH_SCROLL, getTargetOffset());\n\n        if (!store._hasUnmeasuredItemsInFrozenRange()) {\n          break;\n        }\n\n        const [promise, unsubscribe] = waitForMeasurement();\n\n        try {\n          await promise;\n        } catch (e) {\n          // canceled\n          return;\n        } finally {\n          unsubscribe();\n        }\n      }\n\n      viewportElement.scrollTo({\n        [isHorizontal ? \"left\" : \"top\"]: normalizeOffset(\n          getTargetOffset(),\n          isHorizontal\n        ),\n        behavior: \"smooth\",\n      });\n    } else {\n      while (true) {\n        const [promise, unsubscribe] = waitForMeasurement();\n\n        try {\n          viewportElement[scrollOffsetKey] = normalizeOffset(\n            getTargetOffset(),\n            isHorizontal\n          );\n          store.$update(ACTION_MANUAL_SCROLL);\n\n          await promise;\n        } catch (e) {\n          // canceled or finished\n          return;\n        } finally {\n          unsubscribe();\n        }\n      }\n    }\n  };\n\n  return {\n    $observe(viewport) {\n      viewportElement = viewport;\n\n      scrollObserver = createScrollObserver(\n        store,\n        viewport,\n        isHorizontal,\n        () => normalizeOffset(viewport[scrollOffsetKey], isHorizontal),\n        (jump, shift, isMomentumScrolling) => {\n          // If we update scroll position while touching on iOS, the position will be reverted.\n          // However iOS WebKit fires touch events only once at the beginning of momentum scrolling.\n          // That means we have no reliable way to confirm still touched or not if user touches more than once during momentum scrolling...\n          // This is a hack for the suspectable situations, inspired by https://github.com/prud/ios-overflow-scroll-to-top\n          if (isMomentumScrolling) {\n            const style = viewport.style;\n            const prev = style[overflowKey];\n            style[overflowKey] = \"hidden\";\n            timeout(() => {\n              style[overflowKey] = prev;\n            });\n          }\n\n          if (shift) {\n            viewport[scrollOffsetKey] = store.$getScrollOffset() + jump;\n            // https://github.com/inokawa/virtua/issues/357\n            cancelScroll && cancelScroll();\n          } else {\n            viewport[scrollOffsetKey] += jump;\n          }\n        }\n      );\n    },\n    $dispose() {\n      scrollObserver && scrollObserver._dispose();\n    },\n    $scrollTo(offset) {\n      scheduleImperativeScroll(() => offset);\n    },\n    $scrollBy(offset) {\n      offset += store.$getScrollOffset();\n      scheduleImperativeScroll(() => offset);\n    },\n    $scrollToIndex(index, { align, smooth, offset = 0 } = {}) {\n      index = clamp(index, 0, store.$getItemsLength() - 1);\n\n      if (align === \"nearest\") {\n        const itemOffset = store.$getItemOffset(index);\n        const scrollOffset = store.$getScrollOffset();\n\n        if (itemOffset < scrollOffset) {\n          align = \"start\";\n        } else if (\n          itemOffset + store.$getItemSize(index) >\n          scrollOffset + store.$getViewportSize()\n        ) {\n          align = \"end\";\n        } else {\n          // already completely visible\n          return;\n        }\n      }\n\n      scheduleImperativeScroll(() => {\n        return (\n          offset +\n          store.$getStartSpacerSize() +\n          store.$getItemOffset(index) +\n          (align === \"end\"\n            ? store.$getItemSize(index) - store.$getViewportSize()\n            : align === \"center\"\n              ? (store.$getItemSize(index) - store.$getViewportSize()) / 2\n              : 0)\n        );\n      }, smooth);\n    },\n    $fixScrollJump: () => {\n      scrollObserver && scrollObserver._fixScrollJump();\n    },\n  };\n};\n\n/**\n * @internal\n */\nexport type WindowScroller = {\n  $observe(containerElement: HTMLElement): void;\n  $dispose(): void;\n  $scrollToIndex: (index: number, opts?: ScrollToIndexOpts) => void;\n  $fixScrollJump: () => void;\n};\n\n/**\n * @internal\n */\nexport const createWindowScroller = (\n  store: VirtualStore,\n  isHorizontal: boolean\n): WindowScroller => {\n  let containerElement: HTMLElement | undefined;\n  let scrollObserver: ScrollObserver | undefined;\n  let cancelScroll: (() => void) | undefined;\n\n  const calcOffsetToViewport = (\n    node: HTMLElement,\n    viewport: HTMLElement,\n    window: Window,\n    isHorizontal: boolean,\n    offset: number = 0\n  ): number => {\n    // TODO calc offset only when it changes (maybe impossible)\n    const offsetKey = isHorizontal ? \"offsetLeft\" : \"offsetTop\";\n    const offsetSum =\n      offset +\n      (isHorizontal && isRTLDocument()\n        ? window.innerWidth - node[offsetKey] - node.offsetWidth\n        : node[offsetKey]);\n\n    const parent = node.offsetParent;\n    if (node === viewport || !parent) {\n      return offsetSum;\n    }\n\n    return calcOffsetToViewport(\n      parent as HTMLElement,\n      viewport,\n      window,\n      isHorizontal,\n      offsetSum\n    );\n  };\n\n  const scheduleImperativeScroll = async (\n    getTargetOffset: () => number,\n    smooth?: boolean\n  ) => {\n    if (!containerElement) {\n      // Wait for element assign\n      microtask(() => scheduleImperativeScroll(getTargetOffset, smooth));\n      return;\n    }\n\n    if (cancelScroll) {\n      cancelScroll();\n    }\n\n    const waitForMeasurement = (): [Promise<void>, () => void] => {\n      let queue: (() => void) | undefined;\n      return [\n        new Promise<void>((resolve, reject) => {\n          queue = resolve;\n          cancelScroll = reject;\n\n          if (isInitialMeasurementDone(store)) {\n            timeout(reject, 150);\n          }\n        }),\n        store.$subscribe(UPDATE_SIZE_EVENT, () => {\n          queue && queue();\n        }),\n      ];\n    };\n\n    const window = getCurrentWindow(getCurrentDocument(containerElement));\n\n    if (smooth && isSmoothScrollSupported()) {\n      while (true) {\n        store.$update(ACTION_BEFORE_MANUAL_SMOOTH_SCROLL, getTargetOffset());\n\n        if (!store._hasUnmeasuredItemsInFrozenRange()) {\n          break;\n        }\n\n        const [promise, unsubscribe] = waitForMeasurement();\n\n        try {\n          await promise;\n        } catch (e) {\n          return;\n        } finally {\n          unsubscribe();\n        }\n      }\n\n      window.scroll({\n        [isHorizontal ? \"left\" : \"top\"]: normalizeOffset(\n          getTargetOffset(),\n          isHorizontal\n        ),\n        behavior: \"smooth\",\n      });\n    } else {\n      while (true) {\n        const [promise, unsubscribe] = waitForMeasurement();\n\n        try {\n          window.scroll({\n            [isHorizontal ? \"left\" : \"top\"]: normalizeOffset(\n              getTargetOffset(),\n              isHorizontal\n            ),\n          });\n          store.$update(ACTION_MANUAL_SCROLL);\n\n          await promise;\n        } catch (e) {\n          return;\n        } finally {\n          unsubscribe();\n        }\n      }\n    }\n  };\n\n  return {\n    $observe(container) {\n      containerElement = container;\n      const scrollOffsetKey = isHorizontal ? \"scrollX\" : \"scrollY\";\n\n      const document = getCurrentDocument(container);\n      const window = getCurrentWindow(document);\n      const documentBody = document.body;\n\n      scrollObserver = createScrollObserver(\n        store,\n        window,\n        isHorizontal,\n        () => normalizeOffset(window[scrollOffsetKey], isHorizontal),\n        (jump, shift) => {\n          // TODO support case two window scrollers exist in the same view\n          if (shift) {\n            window.scroll({\n              [isHorizontal ? \"left\" : \"top\"]: store.$getScrollOffset() + jump,\n            });\n          } else {\n            window.scrollBy(isHorizontal ? jump : 0, isHorizontal ? 0 : jump);\n          }\n        },\n        () =>\n          calcOffsetToViewport(container, documentBody, window, isHorizontal)\n      );\n    },\n    $dispose() {\n      scrollObserver && scrollObserver._dispose();\n      containerElement = undefined;\n    },\n    $fixScrollJump: () => {\n      scrollObserver && scrollObserver._fixScrollJump();\n    },\n    $scrollToIndex(index, { align, smooth, offset = 0 } = {}) {\n      if (!containerElement) return;\n\n      index = clamp(index, 0, store.$getItemsLength() - 1);\n\n      if (align === \"nearest\") {\n        const itemOffset = store.$getItemOffset(index);\n        const scrollOffset = store.$getScrollOffset();\n\n        if (itemOffset < scrollOffset) {\n          align = \"start\";\n        } else if (\n          itemOffset + store.$getItemSize(index) >\n          scrollOffset + store.$getViewportSize()\n        ) {\n          align = \"end\";\n        } else {\n          return;\n        }\n      }\n\n      const document = getCurrentDocument(containerElement);\n      const window = getCurrentWindow(document);\n      const html = document.documentElement;\n      const getScrollbarSize = () =>\n        store.$getViewportSize() -\n        (isHorizontal ? html.clientWidth : html.clientHeight);\n\n      scheduleImperativeScroll(() => {\n        return (\n          offset +\n          // Calculate target scroll position including container's offset from document\n          calcOffsetToViewport(\n            containerElement!,\n            document.body,\n            window,\n            isHorizontal\n          ) +\n          // store._getStartSpacerSize() +\n          store.$getItemOffset(index) +\n          (align === \"end\"\n            ? store.$getItemSize(index) -\n              (store.$getViewportSize() - getScrollbarSize())\n            : align === \"center\"\n              ? (store.$getItemSize(index) -\n                  (store.$getViewportSize() - getScrollbarSize())) /\n                2\n              : 0)\n        );\n      }, smooth);\n    },\n  };\n};\n\n/**\n * @internal\n */\nexport type GridScroller = {\n  $observe: (viewportElement: HTMLElement) => void;\n  $dispose(): void;\n  $scrollTo: (offsetX: number, offsetY: number) => void;\n  $scrollBy: (offsetX: number, offsetY: number) => void;\n  $scrollToIndex: (indexX: number, indexY: number) => void;\n  $fixScrollJump: () => void;\n};\n\n/**\n * @internal\n */\nexport const createGridScroller = (\n  vStore: VirtualStore,\n  hStore: VirtualStore\n): GridScroller => {\n  const vScroller = createScroller(vStore, false);\n  const hScroller = createScroller(hStore, true);\n  return {\n    $observe(viewportElement) {\n      vScroller.$observe(viewportElement);\n      hScroller.$observe(viewportElement);\n    },\n    $dispose() {\n      vScroller.$dispose();\n      hScroller.$dispose();\n    },\n    $scrollTo(offsetX, offsetY) {\n      vScroller.$scrollTo(offsetY);\n      hScroller.$scrollTo(offsetX);\n    },\n    $scrollBy(offsetX, offsetY) {\n      vScroller.$scrollBy(offsetY);\n      hScroller.$scrollBy(offsetX);\n    },\n    $scrollToIndex(indexX, indexY) {\n      vScroller.$scrollToIndex(indexY);\n      hScroller.$scrollToIndex(indexX);\n    },\n    $fixScrollJump() {\n      vScroller.$fixScrollJump();\n      hScroller.$fixScrollJump();\n    },\n  };\n};\n","import { getCurrentDocument, getCurrentWindow } from \"./environment\";\nimport {\n  ACTION_ITEM_RESIZE,\n  ACTION_VIEWPORT_RESIZE,\n  type VirtualStore,\n} from \"./store\";\nimport { type ItemResize } from \"./types\";\nimport { max, NULL } from \"./utils\";\n\nconst createResizeObserver = (cb: ResizeObserverCallback) => {\n  let ro: ResizeObserver | undefined;\n\n  return {\n    _observe(e: HTMLElement) {\n      // Initialize ResizeObserver lazily for SSR\n      // https://www.w3.org/TR/resize-observer/#intro\n      (\n        ro ||\n        // https://bugs.chromium.org/p/chromium/issues/detail?id=1491739\n        (ro = new (getCurrentWindow(getCurrentDocument(e)).ResizeObserver)(cb))\n      ).observe(e);\n    },\n    _unobserve(e: HTMLElement) {\n      ro!.unobserve(e);\n    },\n    _dispose() {\n      ro && ro.disconnect();\n    },\n  };\n};\n\n/**\n * @internal\n */\nexport type ItemResizeObserver = (el: HTMLElement, i: number) => () => void;\n\ninterface ListResizer {\n  $observeRoot(viewportElement: HTMLElement): void;\n  $observeItem: ItemResizeObserver;\n  $dispose(): void;\n}\n\n/**\n * @internal\n */\nexport const createResizer = (\n  store: VirtualStore,\n  isHorizontal: boolean\n): ListResizer => {\n  let viewportElement: HTMLElement | undefined;\n  const sizeKey = isHorizontal ? \"width\" : \"height\";\n  const mountedIndexes = new WeakMap<Element, number>();\n\n  const resizeObserver = createResizeObserver((entries) => {\n    const resizes: ItemResize[] = [];\n    for (const { target, contentRect } of entries) {\n      // Skip zero-sized rects that may be observed under `display: none` style\n      if (!(target as HTMLElement).offsetParent) continue;\n\n      if (target === viewportElement) {\n        store.$update(ACTION_VIEWPORT_RESIZE, contentRect[sizeKey]);\n      } else {\n        const index = mountedIndexes.get(target);\n        if (index != NULL) {\n          resizes.push([index, contentRect[sizeKey]]);\n        }\n      }\n    }\n\n    if (resizes.length) {\n      store.$update(ACTION_ITEM_RESIZE, resizes);\n    }\n  });\n\n  return {\n    $observeRoot(viewport: HTMLElement) {\n      resizeObserver._observe((viewportElement = viewport));\n    },\n    $observeItem: (el: HTMLElement, i: number) => {\n      mountedIndexes.set(el, i);\n      resizeObserver._observe(el);\n      return () => {\n        mountedIndexes.delete(el);\n        resizeObserver._unobserve(el);\n      };\n    },\n    $dispose: resizeObserver._dispose,\n  };\n};\n\ninterface WindowListResizer {\n  $observeRoot(container: HTMLElement): void;\n  $observeItem: ItemResizeObserver;\n  $dispose(): void;\n}\n\n/**\n * @internal\n */\nexport const createWindowResizer = (\n  store: VirtualStore,\n  isHorizontal: boolean\n): WindowListResizer => {\n  const sizeKey = isHorizontal ? \"width\" : \"height\";\n  const windowSizeKey = isHorizontal ? \"innerWidth\" : \"innerHeight\";\n  const mountedIndexes = new WeakMap<Element, number>();\n\n  const resizeObserver = createResizeObserver((entries) => {\n    const resizes: ItemResize[] = [];\n    for (const { target, contentRect } of entries) {\n      // Skip zero-sized rects that may be observed under `display: none` style\n      if (!(target as HTMLElement).offsetParent) continue;\n\n      const index = mountedIndexes.get(target);\n      if (index != NULL) {\n        resizes.push([index, contentRect[sizeKey]]);\n      }\n    }\n\n    if (resizes.length) {\n      store.$update(ACTION_ITEM_RESIZE, resizes);\n    }\n  });\n\n  let cleanupOnWindowResize: (() => void) | undefined;\n\n  return {\n    $observeRoot(container) {\n      const window = getCurrentWindow(getCurrentDocument(container));\n      const onWindowResize = () => {\n        store.$update(ACTION_VIEWPORT_RESIZE, window[windowSizeKey]);\n      };\n      window.addEventListener(\"resize\", onWindowResize);\n      onWindowResize();\n\n      cleanupOnWindowResize = () => {\n        window.removeEventListener(\"resize\", onWindowResize);\n      };\n    },\n    $observeItem: (el: HTMLElement, i: number) => {\n      mountedIndexes.set(el, i);\n      resizeObserver._observe(el);\n      return () => {\n        mountedIndexes.delete(el);\n        resizeObserver._unobserve(el);\n      };\n    },\n    $dispose() {\n      cleanupOnWindowResize && cleanupOnWindowResize();\n      resizeObserver._dispose();\n    },\n  };\n};\n\n/**\n * @internal\n */\nexport const createGridResizer = (\n  vStore: VirtualStore,\n  hStore: VirtualStore\n) => {\n  let viewportElement: HTMLElement | undefined;\n\n  const heightKey = \"height\";\n  const widthKey = \"width\";\n  const mountedIndexes = new WeakMap<\n    Element,\n    [rowIndex: number, colIndex: number]\n  >();\n\n  type CellSize = [height: number, width: number];\n  const maybeCachedRowIndexes = new Set<number>();\n  const maybeCachedColIndexes = new Set<number>();\n  const sizeCache = new Map<string, CellSize>();\n  const getKey = (rowIndex: number, colIndex: number): string =>\n    `${rowIndex}-${colIndex}`;\n\n  const resizeObserver = createResizeObserver((entries) => {\n    const resizedRows = new Set<number>();\n    const resizedCols = new Set<number>();\n    for (const { target, contentRect } of entries) {\n      // Skip zero-sized rects that may be observed under `display: none` style\n      if (!(target as HTMLElement).offsetParent) continue;\n\n      if (target === viewportElement) {\n        vStore.$update(ACTION_VIEWPORT_RESIZE, contentRect[heightKey]);\n        hStore.$update(ACTION_VIEWPORT_RESIZE, contentRect[widthKey]);\n      } else {\n        const cell = mountedIndexes.get(target);\n        if (cell) {\n          const [rowIndex, colIndex] = cell;\n          const key = getKey(rowIndex, colIndex);\n          const prevSize = sizeCache.get(key);\n          const size: CellSize = [\n            contentRect[heightKey],\n            contentRect[widthKey],\n          ];\n          let rowResized: boolean | undefined;\n          let colResized: boolean | undefined;\n          if (!prevSize) {\n            rowResized = colResized = true;\n          } else {\n            if (prevSize[0] !== size[0]) {\n              rowResized = true;\n            }\n            if (prevSize[1] !== size[1]) {\n              colResized = true;\n            }\n          }\n          if (rowResized) {\n            resizedRows.add(rowIndex);\n          }\n          if (colResized) {\n            resizedCols.add(colIndex);\n          }\n          if (rowResized || colResized) {\n            sizeCache.set(key, size);\n          }\n        }\n      }\n    }\n\n    if (resizedRows.size) {\n      const heightResizes: ItemResize[] = [];\n      resizedRows.forEach((rowIndex) => {\n        let maxHeight = 0;\n        maybeCachedColIndexes.forEach((colIndex) => {\n          const size = sizeCache.get(getKey(rowIndex, colIndex));\n          if (size) {\n            maxHeight = max(maxHeight, size[0]);\n          }\n        });\n        if (maxHeight) {\n          heightResizes.push([rowIndex, maxHeight]);\n        }\n      });\n      vStore.$update(ACTION_ITEM_RESIZE, heightResizes);\n    }\n    if (resizedCols.size) {\n      const widthResizes: ItemResize[] = [];\n      resizedCols.forEach((colIndex) => {\n        let maxWidth = 0;\n        maybeCachedRowIndexes.forEach((rowIndex) => {\n          const size = sizeCache.get(getKey(rowIndex, colIndex));\n          if (size) {\n            maxWidth = max(maxWidth, size[1]);\n          }\n        });\n        if (maxWidth) {\n          widthResizes.push([colIndex, maxWidth]);\n        }\n      });\n      hStore.$update(ACTION_ITEM_RESIZE, widthResizes);\n    }\n  });\n\n  return {\n    $observeRoot(viewport: HTMLElement) {\n      resizeObserver._observe((viewportElement = viewport));\n    },\n    $observeItem(el: HTMLElement, rowIndex: number, colIndex: number) {\n      mountedIndexes.set(el, [rowIndex, colIndex]);\n      maybeCachedRowIndexes.add(rowIndex);\n      maybeCachedColIndexes.add(colIndex);\n      resizeObserver._observe(el);\n      return () => {\n        mountedIndexes.delete(el);\n        resizeObserver._unobserve(el);\n      };\n    },\n    $dispose: resizeObserver._dispose,\n  };\n};\n\n/**\n * @internal\n */\nexport type GridResizer = ReturnType<typeof createGridResizer>;\n","/**\n * @jsxImportSource solid-js\n */\nimport { ItemResizeObserver, isRTLDocument } from \"../core\";\nimport {\n  Component,\n  JSX,\n  createEffect,\n  createMemo,\n  mergeProps,\n  onCleanup,\n  type ValidComponent,\n} from \"solid-js\";\nimport { Dynamic } from \"solid-js/web\";\n\ninterface ListItemProps {\n  _children: JSX.Element;\n  _resizer: ItemResizeObserver;\n  _index: number;\n  _offset: number;\n  _hide: boolean;\n  _isHorizontal: boolean;\n  _as?: ValidComponent;\n}\n\n/**\n * @internal\n */\nexport const ListItem: Component<ListItemProps> = (props) => {\n  let elementRef: HTMLDivElement | undefined;\n  props = mergeProps<[Partial<ListItemProps>, ListItemProps]>(\n    { _as: \"div\" },\n    props\n  );\n\n  // The index may be changed if elements are inserted to or removed from the start of props.children\n  createEffect(() => {\n    if (!elementRef) return;\n    onCleanup(props._resizer(elementRef, props._index));\n  });\n\n  const style = createMemo(() => {\n    const isHorizontal = props._isHorizontal;\n    const style: JSX.CSSProperties = {\n      position: \"absolute\",\n      [isHorizontal ? \"height\" : \"width\"]: \"100%\",\n      [isHorizontal ? \"top\" : \"left\"]: \"0px\",\n      [isHorizontal ? (isRTLDocument() ? \"right\" : \"left\") : \"top\"]:\n        props._offset + \"px\",\n      visibility: props._hide ? \"hidden\" : \"visible\",\n    };\n    if (isHorizontal) {\n      style.display = \"flex\";\n    }\n    return style;\n  });\n\n  return (\n    <Dynamic component={props._as} ref={elementRef} style={style()}>\n      {props._children}\n    </Dynamic>\n  );\n};\n","/**\n * @jsxImportSource solid-js\n */\nimport {\n  createMemo,\n  createRoot,\n  createSignal,\n  onCleanup,\n  JSX,\n  Signal,\n  Accessor,\n} from \"solid-js\";\nimport { ItemsRange } from \"../core\";\n\ninterface RenderedNode<T> {\n  _data: Signal<T>;\n  _element: JSX.Element;\n  _dispose: () => void;\n}\n\n/**\n * https://github.com/solidjs/solid/blob/main/packages/solid/src/reactive/array.ts\n * https://github.com/solidjs/solid/blob/main/packages/solid/src/render/flow.ts\n * https://github.com/solidjs/solid/discussions/366\n * @internal\n */\nexport const RangedFor = <T,>(props: {\n  _each: T[];\n  _range: ItemsRange;\n  _render: (data: Accessor<T>, index: number) => JSX.Element;\n}): JSX.Element => {\n  let prev = new Map<number, RenderedNode<T>>();\n\n  onCleanup(() => {\n    for (const node of prev.values()) {\n      node._dispose();\n    }\n  });\n\n  return createMemo(() => {\n    const list = props._each;\n    const [start, end] = props._range;\n    const current = new Map<number, RenderedNode<T>>();\n    const items: JSX.Element[] = [];\n\n    for (let i = start; i <= end; i++) {\n      const newData: T = list[i]!;\n      const lookup = prev.get(i);\n      items.push(\n        lookup\n          ? lookup._element\n          : createRoot((dispose) => {\n              const data = createSignal(newData);\n              const result = props._render(data[0], i);\n              current.set(i, {\n                _data: data,\n                _element: result,\n                _dispose: dispose,\n              });\n              return result;\n            })\n      );\n      if (lookup) {\n        if (newData !== lookup._data) {\n          lookup._data[1](\n            newData as Exclude<T, Function> // TODO improve type\n          );\n        }\n        current.set(i, lookup);\n      }\n    }\n    for (const [key, node] of prev.entries()) {\n      if (!current.has(key)) {\n        node._dispose();\n      }\n    }\n    prev = current;\n    return items;\n  }) as unknown as JSX.Element; // TODO improve type\n};\n","import { ItemsRange } from \"../core\";\n\n/**\n * @internal\n */\nexport const isSameRange = (prev: ItemsRange, next: ItemsRange): boolean => {\n  return prev[0] === next[0] && prev[1] === next[1];\n};\n","/**\n * @jsxImportSource solid-js\n */\nimport {\n  onMount,\n  onCleanup,\n  createEffect,\n  createSignal,\n  createMemo,\n  JSX,\n  on,\n  createComputed,\n  type ValidComponent,\n  mergeProps,\n} from \"solid-js\";\nimport { Dynamic } from \"solid-js/web\";\nimport {\n  UPDATE_SCROLL_EVENT,\n  UPDATE_SCROLL_END_EVENT,\n  UPDATE_VIRTUAL_STATE,\n  createVirtualStore,\n  ACTION_ITEMS_LENGTH_CHANGE,\n  getScrollSize,\n  ACTION_START_OFFSET_CHANGE,\n  createResizer,\n  createScroller,\n  ItemsRange,\n  ScrollToIndexOpts,\n} from \"../core\";\nimport { ListItem } from \"./ListItem\";\nimport { RangedFor } from \"./RangedFor\";\nimport { isSameRange } from \"./utils\";\n\n/**\n * Methods of {@link Virtualizer}.\n */\nexport interface VirtualizerHandle {\n  /**\n   * Get current scrollTop, or scrollLeft if horizontal: true.\n   */\n  readonly scrollOffset: number;\n  /**\n   * Get current scrollHeight, or scrollWidth if horizontal: true.\n   */\n  readonly scrollSize: number;\n  /**\n   * Get current offsetHeight, or offsetWidth if horizontal: true.\n   */\n  readonly viewportSize: number;\n  /**\n   * Find the start index of visible range of items.\n   */\n  findStartIndex: () => number;\n  /**\n   * Find the end index of visible range of items.\n   */\n  findEndIndex: () => number;\n  /**\n   * Get item offset from start.\n   * @param index index of item\n   */\n  getItemOffset(index: number): number;\n  /**\n   * Get item size.\n   * @param index index of item\n   */\n  getItemSize(index: number): number;\n  /**\n   * Scroll to the item specified by index.\n   * @param index index of item\n   * @param opts options\n   */\n  scrollToIndex(index: number, opts?: ScrollToIndexOpts): void;\n  /**\n   * Scroll to the given offset.\n   * @param offset offset from start\n   */\n  scrollTo(offset: number): void;\n  /**\n   * Scroll by the given offset.\n   * @param offset offset from current position\n   */\n  scrollBy(offset: number): void;\n}\n\n/**\n * Props of {@link Virtualizer}.\n */\nexport interface VirtualizerProps<T> {\n  /**\n   * Get reference to {@link VirtualizerHandle}.\n   */\n  ref?: (handle?: VirtualizerHandle) => void;\n  /**\n   * The data items rendered by this component.\n   */\n  data: T[];\n  /**\n   * The elements renderer function.\n   */\n  children: (data: T, index: number) => JSX.Element;\n  /**\n   * Number of items to render above/below the visible bounds of the list. Lower value will give better performance but you can increase to avoid showing blank items in fast scrolling.\n   * @defaultValue 4\n   */\n  overscan?: number;\n  /**\n   * Component or element type for container element.\n   * @defaultValue \"div\"\n   */\n  as?: ValidComponent;\n  /**\n   * Component or element type for item element.\n   * @defaultValue \"div\"\n   */\n  item?: ValidComponent;\n  /**\n   * Reference to the scrollable element. The default will get the direct parent element of virtualizer.\n   */\n  scrollRef?: HTMLElement;\n  /**\n   * Item size hint for unmeasured items. It will help to reduce scroll jump when items are measured if used properly.\n   *\n   * - If not set, initial item sizes will be automatically estimated from measured sizes. This is recommended for most cases.\n   * - If set, you can opt out estimation and use the value as initial item size.\n   */\n  itemSize?: number;\n  /**\n   * While true is set, scroll position will be maintained from the end not usual start when items are added to/removed from start. It's recommended to set false if you add to/remove from mid/end of the list because it can cause unexpected behavior. This prop is useful for reverse infinite scrolling.\n   */\n  shift?: boolean;\n  /**\n   * If true, rendered as a horizontally scrollable list. Otherwise rendered as a vertically scrollable list.\n   */\n  horizontal?: boolean;\n  /**\n   * If you put an element before virtualizer, you have to define its height with this prop.\n   */\n  startMargin?: number;\n  /**\n   * Callback invoked whenever scroll offset changes.\n   * @param offset Current scrollTop, or scrollLeft if horizontal: true.\n   */\n  onScroll?: (offset: number) => void;\n  /**\n   * Callback invoked when scrolling stops.\n   */\n  onScrollEnd?: () => void;\n}\n\n/**\n * Customizable list virtualizer for advanced usage. See {@link VirtualizerProps} and {@link VirtualizerHandle}.\n */\nexport const Virtualizer = <T,>(props: VirtualizerProps<T>): JSX.Element => {\n  let containerRef: HTMLDivElement | undefined;\n  const { itemSize, horizontal = false, overscan } = props;\n  props = mergeProps<[Partial<VirtualizerProps<T>>, VirtualizerProps<T>]>(\n    { as: \"div\" },\n    props\n  );\n\n  const store = createVirtualStore(\n    props.data.length,\n    itemSize,\n    overscan,\n    undefined,\n    undefined,\n    !itemSize\n  );\n  const resizer = createResizer(store, horizontal);\n  const scroller = createScroller(store, horizontal);\n\n  const [stateVersion, setRerender] = createSignal(store.$getStateVersion());\n\n  const unsubscribeStore = store.$subscribe(UPDATE_VIRTUAL_STATE, () => {\n    setRerender(store.$getStateVersion());\n  });\n\n  const unsubscribeOnScroll = store.$subscribe(UPDATE_SCROLL_EVENT, () => {\n    props.onScroll?.(store.$getScrollOffset());\n  });\n  const unsubscribeOnScrollEnd = store.$subscribe(\n    UPDATE_SCROLL_END_EVENT,\n    () => {\n      props.onScrollEnd?.();\n    }\n  );\n\n  const range = createMemo<ItemsRange>((prev) => {\n    stateVersion();\n    const next = store.$getRange();\n    if (prev && isSameRange(prev, next)) {\n      return prev;\n    }\n    return next;\n  });\n  const isScrolling = createMemo(() => stateVersion() && store.$isScrolling());\n  const totalSize = createMemo(() => stateVersion() && store.$getTotalSize());\n\n  onMount(() => {\n    if (props.ref) {\n      props.ref({\n        get scrollOffset() {\n          return store.$getScrollOffset();\n        },\n        get scrollSize() {\n          return getScrollSize(store);\n        },\n        get viewportSize() {\n          return store.$getViewportSize();\n        },\n        findStartIndex: store.$findStartIndex,\n        findEndIndex: store.$findEndIndex,\n        getItemOffset: store.$getItemOffset,\n        getItemSize: store.$getItemSize,\n        scrollToIndex: scroller.$scrollToIndex,\n        scrollTo: scroller.$scrollTo,\n        scrollBy: scroller.$scrollBy,\n      });\n    }\n\n    const scrollable = props.scrollRef || containerRef!.parentElement!;\n    resizer.$observeRoot(scrollable);\n    scroller.$observe(scrollable);\n\n    onCleanup(() => {\n      if (props.ref) {\n        props.ref();\n      }\n\n      unsubscribeStore();\n      unsubscribeOnScroll();\n      unsubscribeOnScrollEnd();\n      resizer.$dispose();\n      scroller.$dispose();\n    });\n  });\n\n  createComputed(\n    on(\n      () => props.data.length,\n      (count) => {\n        if (count !== store.$getItemsLength()) {\n          store.$update(ACTION_ITEMS_LENGTH_CHANGE, [count, props.shift]);\n        }\n      }\n    )\n  );\n\n  createComputed(\n    on(\n      () => props.startMargin || 0,\n      (value) => {\n        if (value !== store.$getStartSpacerSize()) {\n          store.$update(ACTION_START_OFFSET_CHANGE, value);\n        }\n      }\n    )\n  );\n\n  createEffect(\n    on(stateVersion, () => {\n      scroller.$fixScrollJump();\n    })\n  );\n\n  return (\n    <Dynamic\n      component={props.as}\n      ref={containerRef}\n      style={{\n        // contain: \"content\",\n        \"overflow-anchor\": \"none\", // opt out browser's scroll anchoring because it will conflict to scroll anchoring of virtualizer\n        flex: \"none\", // flex style can break layout\n        position: \"relative\",\n        visibility: \"hidden\", // TODO replace with other optimization methods\n        width: horizontal ? totalSize() + \"px\" : \"100%\",\n        height: horizontal ? \"100%\" : totalSize() + \"px\",\n        \"pointer-events\": isScrolling() ? \"none\" : undefined,\n      }}\n    >\n      <RangedFor\n        _each={props.data}\n        _range={range()}\n        _render={(data, index) => {\n          const offset = createMemo(() => {\n            stateVersion();\n            return store.$getItemOffset(index);\n          });\n          const hide = createMemo(() => {\n            stateVersion();\n            return store.$isUnmeasuredItem(index);\n          });\n\n          return (\n            <ListItem\n              _as={props.item}\n              _index={index}\n              _resizer={resizer.$observeItem}\n              _offset={offset()}\n              _hide={hide()}\n              _children={props.children(data(), index)}\n              _isHorizontal={horizontal}\n            />\n          );\n        }}\n      />\n    </Dynamic>\n  );\n};\n","/**\n * @jsxImportSource solid-js\n */\nimport { JSX } from \"solid-js\";\nimport { ViewportComponentAttributes } from \"./types\";\nimport {\n  Virtualizer,\n  VirtualizerHandle,\n  VirtualizerProps,\n} from \"./Virtualizer\";\n\n/**\n * Methods of {@link VList}.\n */\nexport interface VListHandle extends VirtualizerHandle {}\n\n/**\n * Props of {@link VList}.\n */\nexport interface VListProps<T>\n  extends Pick<\n      VirtualizerProps<T>,\n      | \"ref\"\n      | \"data\"\n      | \"children\"\n      | \"overscan\"\n      | \"itemSize\"\n      | \"shift\"\n      | \"horizontal\"\n      | \"onScroll\"\n      | \"onScrollEnd\"\n    >,\n    ViewportComponentAttributes {}\n\n/**\n * Virtualized list component. See {@link VListProps} and {@link VListHandle}.\n */\nexport const VList = <T,>(props: VListProps<T>): JSX.Element => {\n  const {\n    ref,\n    data,\n    children,\n    overscan,\n    itemSize,\n    shift,\n    horizontal,\n    onScroll,\n    onScrollEnd,\n    style,\n    ...attrs\n  } = props;\n\n  return (\n    <div\n      {...attrs}\n      style={{\n        display: horizontal ? \"inline-block\" : \"block\",\n        [horizontal ? \"overflow-x\" : \"overflow-y\"]: \"auto\",\n        contain: \"strict\",\n        width: \"100%\",\n        height: \"100%\",\n        ...props.style,\n      }}\n    >\n      <Virtualizer\n        ref={props.ref}\n        data={props.data}\n        overscan={props.overscan}\n        itemSize={props.itemSize}\n        shift={props.shift}\n        horizontal={horizontal}\n        onScroll={props.onScroll}\n        onScrollEnd={props.onScrollEnd}\n      >\n        {props.children}\n      </Virtualizer>\n    </div>\n  );\n};\n","/**\n * @jsxImportSource solid-js\n */\nimport {\n  onMount,\n  onCleanup,\n  createEffect,\n  createSignal,\n  createMemo,\n  JSX,\n  on,\n  createComputed,\n} from \"solid-js\";\nimport {\n  UPDATE_SCROLL_END_EVENT,\n  UPDATE_VIRTUAL_STATE,\n  createVirtualStore,\n  ACTION_ITEMS_LENGTH_CHANGE,\n  UPDATE_SCROLL_EVENT,\n  createWindowResizer,\n  createWindowScroller,\n  ItemsRange,\n  ScrollToIndexOpts,\n} from \"../core\";\nimport { ListItem } from \"./ListItem\";\nimport { RangedFor } from \"./RangedFor\";\nimport { isSameRange } from \"./utils\";\n\n/**\n * Methods of {@link WindowVirtualizer}.\n */\nexport interface WindowVirtualizerHandle {\n  /**\n   * Find the start index of visible range of items.\n   */\n  findStartIndex: () => number;\n  /**\n   * Find the end index of visible range of items.\n   */\n  findEndIndex: () => number;\n  /**\n   * Scroll to the item specified by index.\n   * @param index index of item\n   * @param opts options\n   */\n  scrollToIndex(index: number, opts?: ScrollToIndexOpts): void;\n}\n\n/**\n * Props of {@link WindowVirtualizer}.\n */\nexport interface WindowVirtualizerProps<T> {\n  /**\n   * Get reference to {@link WindowVirtualizerHandle}.\n   */\n  ref?: (handle?: WindowVirtualizerHandle) => void;\n  /**\n   * The data items rendered by this component.\n   */\n  data: T[];\n  /**\n   * The elements renderer function.\n   */\n  children: (data: T, index: number) => JSX.Element;\n  /**\n   * Number of items to render above/below the visible bounds of the list. Lower value will give better performance but you can increase to avoid showing blank items in fast scrolling.\n   * @defaultValue 4\n   */\n  overscan?: number;\n  /**\n   * Item size hint for unmeasured items. It will help to reduce scroll jump when items are measured if used properly.\n   *\n   * - If not set, initial item sizes will be automatically estimated from measured sizes. This is recommended for most cases.\n   * - If set, you can opt out estimation and use the value as initial item size.\n   */\n  itemSize?: number;\n  /**\n   * While true is set, scroll position will be maintained from the end not usual start when items are added to/removed from start. It's recommended to set false if you add to/remove from mid/end of the list because it can cause unexpected behavior. This prop is useful for reverse infinite scrolling.\n   */\n  shift?: boolean;\n  /**\n   * If true, rendered as a horizontally scrollable list. Otherwise rendered as a vertically scrollable list.\n   */\n  horizontal?: boolean;\n  /**\n   * Callback invoked whenever scroll offset changes.\n   */\n  onScroll?: () => void;\n  /**\n   * Callback invoked when scrolling stops.\n   */\n  onScrollEnd?: () => void;\n}\n\n/**\n * {@link Virtualizer} controlled by the window scrolling. See {@link WindowVirtualizerProps} and {@link WindowVirtualizerHandle}.\n */\nexport const WindowVirtualizer = <T,>(\n  props: WindowVirtualizerProps<T>\n): JSX.Element => {\n  let containerRef: HTMLDivElement | undefined;\n\n  const {\n    ref: _ref,\n    data: _data,\n    children: _children,\n    overscan,\n    itemSize,\n    shift: _shift,\n    horizontal = false,\n    onScrollEnd: _onScrollEnd,\n  } = props;\n\n  const store = createVirtualStore(\n    props.data.length,\n    itemSize,\n    overscan,\n    undefined,\n    undefined,\n    !itemSize\n  );\n  const resizer = createWindowResizer(store, horizontal);\n  const scroller = createWindowScroller(store, horizontal);\n\n  const [stateVersion, setRerender] = createSignal(store.$getStateVersion());\n\n  const unsubscribeStore = store.$subscribe(UPDATE_VIRTUAL_STATE, () => {\n    setRerender(store.$getStateVersion());\n  });\n\n  const unsubscribeOnScroll = store.$subscribe(UPDATE_SCROLL_EVENT, () => {\n    // https://github.com/inokawa/virtua/discussions/580\n    props.onScroll?.();\n  });\n  const unsubscribeOnScrollEnd = store.$subscribe(\n    UPDATE_SCROLL_END_EVENT,\n    () => {\n      props.onScrollEnd?.();\n    }\n  );\n\n  const range = createMemo<ItemsRange>((prev) => {\n    stateVersion();\n    const next = store.$getRange();\n    if (prev && isSameRange(prev, next)) {\n      return prev;\n    }\n    return next;\n  });\n  const isScrolling = createMemo(() => stateVersion() && store.$isScrolling());\n  const totalSize = createMemo(() => stateVersion() && store.$getTotalSize());\n\n  onMount(() => {\n    if (props.ref) {\n      props.ref({\n        findStartIndex: store.$findStartIndex,\n        findEndIndex: store.$findEndIndex,\n        scrollToIndex: scroller.$scrollToIndex,\n      });\n    }\n\n    resizer.$observeRoot(containerRef!);\n    scroller.$observe(containerRef!);\n\n    onCleanup(() => {\n      if (props.ref) {\n        props.ref();\n      }\n\n      unsubscribeStore();\n      unsubscribeOnScroll();\n      unsubscribeOnScrollEnd();\n      resizer.$dispose();\n      scroller.$dispose();\n    });\n  });\n\n  createComputed(\n    on(\n      () => props.data.length,\n      (len) => {\n        if (len !== store.$getItemsLength()) {\n          store.$update(ACTION_ITEMS_LENGTH_CHANGE, [len, props.shift]);\n        }\n      }\n    )\n  );\n\n  createEffect(\n    on(stateVersion, () => {\n      scroller.$fixScrollJump();\n    })\n  );\n\n  return (\n    <div\n      ref={containerRef}\n      style={{\n        // contain: \"content\",\n        \"overflow-anchor\": \"none\", // opt out browser's scroll anchoring because it will conflict to scroll anchoring of virtualizer\n        flex: \"none\", // flex style can break layout\n        position: \"relative\",\n        visibility: \"hidden\", // TODO replace with other optimization methods\n        width: horizontal ? totalSize() + \"px\" : \"100%\",\n        height: horizontal ? \"100%\" : totalSize() + \"px\",\n        \"pointer-events\": isScrolling() ? \"none\" : undefined,\n      }}\n    >\n      <RangedFor\n        _each={props.data}\n        _range={range()}\n        _render={(data, index) => {\n          const offset = createMemo(() => {\n            stateVersion();\n            return store.$getItemOffset(index);\n          });\n          const hide = createMemo(() => {\n            stateVersion();\n            return store.$isUnmeasuredItem(index);\n          });\n\n          return (\n            <ListItem\n              _index={index}\n              _resizer={resizer.$observeItem}\n              _offset={offset()}\n              _hide={hide()}\n              _children={props.children(data(), index)}\n              _isHorizontal={horizontal}\n            />\n          );\n        }}\n      />\n    </div>\n  );\n};\n"],"names":["NULL","min","max","abs","floor","Math","clamp","value","minValue","maxValue","microtask","queueMicrotask","fn","Promise","resolve","then","once","called","cache","UNCACHED","fill","array","length","prepend","key","i","getItemSize","index","size","_sizes","_defaultItemSize","setItemSize","isInitialMeasurement","_computedOffsetIndex","computeOffset","_length","_offsets","top","findIndex","offset","low","high","mid","itemOffset","updateCacheLength","isShift","diff","splice","reduce","acc","removed","isBrowser","window","getDocumentElement","document","documentElement","getCurrentDocument","node","ownerDocument","getCurrentWindow","doc","defaultView","isRTLDocument","getComputedStyle","direction","isIOSWebKit","test","navigator","userAgent","isSmoothScrollSupported","style","isInitialMeasurementDone","store","$getViewportSize","createVirtualStore","elementsCount","itemSize","overscan","ssrCount","cacheSnapshot","shouldAutoEstimateItemSize","isSSR","stateVersion","viewportSize","startSpacerSize","scrollOffset","jump","pendingJump","_flushedJump","_scrollDirection","_scrollMode","_frozenRange","_prevRange","_totalMeasuredSize","initCache","subscribers","Set","getRelativeScrollOffset","getVisibleOffset","getRange","prevStartIndex","end","start","undefined","computeRange","getTotalSize","computeTotalSize","getItemOffset","computeStartOffset","_getItemSize","applyJump","j","$getStateVersion","$getCacheSnapshot","slice","takeCacheSnapshot","$getRange","startIndex","endIndex","$findStartIndex","$findEndIndex","$isUnmeasuredItem","_hasUnmeasuredItemsInFrozenRange","includes","$getItemOffset","$getItemSize","$getItemsLength","$getScrollOffset","$isScrolling","$getStartSpacerSize","$getTotalSize","_flushJump","$subscribe","target","cb","sub","add","delete","$update","type","payload","shouldFlushPendingJump","shouldSync","mutated","flushedJump","delta","distance","relativeOffset","updated","filter","prevSize","measuredCountBeforeStart","measuredSizes","forEach","s","push","sorted","arr","sort","a","b","len","median","prevDefaultItemSize","estimateDefaultItemSize","UPDATE_VIRTUAL_STATE","timeout","setTimeout","normalizeOffset","isHorizontal","createScrollObserver","viewport","getScrollOffset","updateScrollOffset","getStartOffset","now","Date","lastScrollTime","wheeling","touching","justTouchEnded","stillMomentumScrolling","onScrollEnd","id","cancel","clearTimeout","debouncedFn","_cancel","debounce","onScroll","onWheel","e","ctrlKey","timeDelta","deltaX","deltaY","onTouchStart","onTouchEnd","addEventListener","passive","_dispose","removeEventListener","_fixScrollJump","shift","createResizeObserver","ro","_observe","observe","_unobserve","unobserve","disconnect","ListItem","props","elementRef","mergeProps","_as","createEffect","onCleanup","_resizer","_index","createMemo","_isHorizontal","position","_offset","visibility","_hide","display","_$createComponent","Dynamic","component","ref","r$","children","_children","RangedFor","prev","Map","values","list","_each","_range","current","items","newData","lookup","get","_element","createRoot","dispose","data","createSignal","result","_render","set","_data","entries","has","isSameRange","next","Virtualizer","containerRef","horizontal","as","resizer","viewportElement","sizeKey","mountedIndexes","WeakMap","resizeObserver","resizes","contentRect","offsetParent","$observeRoot","$observeItem","el","$dispose","createResizer","scroller","scrollObserver","cancelScroll","scrollOffsetKey","overflowKey","scheduleImperativeScroll","async","getTargetOffset","smooth","waitForMeasurement","queue","reject","promise","unsubscribe","scrollTo","behavior","$observe","isMomentumScrolling","$scrollTo","$scrollBy","$scrollToIndex","align","$fixScrollJump","createScroller","setRerender","unsubscribeStore","unsubscribeOnScroll","_a","call","unsubscribeOnScrollEnd","range","isScrolling","totalSize","onMount","scrollSize","getScrollSize","findStartIndex","findEndIndex","scrollToIndex","scrollBy","scrollable","scrollRef","parentElement","createComputed","on","count","startMargin","flex","width","height","hide","item","VList","attrs","_el$","_tmpl$","_$spread","_$mergeProps","contain","_$insert","_ref$","WindowVirtualizer","_ref","_shift","_onScrollEnd","windowSizeKey","cleanupOnWindowResize","container","onWindowResize","createWindowResizer","containerElement","calcOffsetToViewport","offsetKey","offsetSum","innerWidth","offsetWidth","parent","scroll","documentBody","body","html","getScrollbarSize","clientWidth","clientHeight","createWindowScroller","_$use","setProperty","_$effect","_p$","_v$","_v$2","_v$3","removeProperty","t"],"mappings":"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"}