@pierre/diffs 1.2.9 → 1.2.11

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (52) hide show
  1. package/README.md +6 -6
  2. package/dist/components/CodeView.js +6 -6
  3. package/dist/components/CodeView.js.map +1 -1
  4. package/dist/components/File.d.ts.map +1 -1
  5. package/dist/components/File.js +6 -1
  6. package/dist/components/File.js.map +1 -1
  7. package/dist/components/FileDiff.js +1 -1
  8. package/dist/components/FileDiff.js.map +1 -1
  9. package/dist/components/FileStream.js +4 -2
  10. package/dist/components/FileStream.js.map +1 -1
  11. package/dist/components/UnresolvedFile.d.ts.map +1 -1
  12. package/dist/components/VirtualizedFile.d.ts +6 -2
  13. package/dist/components/VirtualizedFile.d.ts.map +1 -1
  14. package/dist/components/VirtualizedFile.js +82 -21
  15. package/dist/components/VirtualizedFile.js.map +1 -1
  16. package/dist/components/VirtualizedFileDiff.d.ts +7 -2
  17. package/dist/components/VirtualizedFileDiff.d.ts.map +1 -1
  18. package/dist/components/VirtualizedFileDiff.js +77 -15
  19. package/dist/components/VirtualizedFileDiff.js.map +1 -1
  20. package/dist/components/VirtulizerDevelopment.d.ts.map +1 -1
  21. package/dist/constants.d.ts.map +1 -1
  22. package/dist/highlighter/themes/themeResolver.d.ts +4 -1
  23. package/dist/highlighter/themes/themeResolver.d.ts.map +1 -1
  24. package/dist/managers/InteractionManager.js +1 -1
  25. package/dist/managers/InteractionManager.js.map +1 -1
  26. package/dist/managers/ResizeManager.js +1 -1
  27. package/dist/managers/ResizeManager.js.map +1 -1
  28. package/dist/react/jsx.d.ts.map +1 -1
  29. package/dist/renderers/DiffHunksRenderer.d.ts +3 -2
  30. package/dist/renderers/DiffHunksRenderer.d.ts.map +1 -1
  31. package/dist/renderers/DiffHunksRenderer.js +49 -2
  32. package/dist/renderers/DiffHunksRenderer.js.map +1 -1
  33. package/dist/renderers/FileRenderer.js +12 -0
  34. package/dist/renderers/FileRenderer.js.map +1 -1
  35. package/dist/types.d.ts +8 -0
  36. package/dist/types.d.ts.map +1 -1
  37. package/dist/utils/includesFileAnnotations.d.ts +17 -0
  38. package/dist/utils/includesFileAnnotations.d.ts.map +1 -0
  39. package/dist/utils/includesFileAnnotations.js +19 -0
  40. package/dist/utils/includesFileAnnotations.js.map +1 -0
  41. package/dist/utils/parseMergeConflictDiffFromFile.js.map +1 -1
  42. package/dist/utils/renderDiffWithHighlighter.js +4 -2
  43. package/dist/utils/renderDiffWithHighlighter.js.map +1 -1
  44. package/dist/utils/renderFileWithHighlighter.js +4 -2
  45. package/dist/utils/renderFileWithHighlighter.js.map +1 -1
  46. package/dist/worker/{wasm-BaDzIkIn.js → wasm-qE0LgnY3.js} +2 -2
  47. package/dist/worker/{wasm-BaDzIkIn.js.map → wasm-qE0LgnY3.js.map} +1 -1
  48. package/dist/worker/worker-portable.js +991 -266
  49. package/dist/worker/worker-portable.js.map +1 -1
  50. package/dist/worker/worker.js +8 -4
  51. package/dist/worker/worker.js.map +1 -1
  52. package/package.json +4 -10
@@ -1 +1 @@
1
- {"version":3,"file":"VirtualizedFileDiff.js","names":["position: { top: number; height: number } | undefined","anchor: NumericScrollLineAnchor | undefined","result: DiffLayoutCheckpoint | undefined","hunkOffsets: number[]","firstVisibleHunk: number | undefined","centerHunk: number | undefined","overflowCounter: number | undefined","lineHeight","lineIndexes: number[]","offsets: number[]"],"sources":["../../src/components/VirtualizedFileDiff.ts"],"sourcesContent":["import { DEFAULT_COLLAPSED_CONTEXT_THRESHOLD } from '../constants';\nimport type {\n ExpansionDirections,\n FileDiffMetadata,\n Hunk,\n HunkSeparators,\n NumericScrollLineAnchor,\n PendingCodeViewLayoutReset,\n RenderRange,\n RenderWindow,\n SelectionSide,\n StickySpecs,\n ThemeTypes,\n VirtualFileMetrics,\n} from '../types';\nimport { areDiffTargetsEqual } from '../utils/areDiffTargetsEqual';\nimport { areObjectsEqual } from '../utils/areObjectsEqual';\nimport { areOptionsEqual } from '../utils/areOptionsEqual';\nimport { computeEstimatedDiffHeights } from '../utils/computeEstimatedDiffHeights';\nimport {\n computeVirtualFileMetrics,\n getVirtualFileHeaderRegion,\n getVirtualFilePaddingBottom,\n} from '../utils/computeVirtualFileMetrics';\nimport { iterateOverDiff } from '../utils/iterateOverDiff';\nimport { parseDiffFromFile } from '../utils/parseDiffFromFile';\nimport {\n getExpandedRegion,\n getLeadingHunkSeparatorLayout,\n getTrailingExpandedRegion,\n getTrailingHunkSeparatorLayout,\n} from '../utils/virtualDiffLayout';\nimport type { WorkerPoolManager } from '../worker';\nimport type { CodeView } from './CodeView';\nimport {\n FileDiff,\n type FileDiffOptions,\n type FileDiffRenderProps,\n} from './FileDiff';\nimport type { Virtualizer } from './Virtualizer';\n\ninterface DiffLayoutCheckpoint {\n renderedLineIndex: number;\n lineIndex: number;\n top: number;\n}\n\ninterface DiffLayoutCache {\n // Sparse map: view-specific line index -> measured height delta from the\n // baseline line height. Only stores lines that differ from the estimate.\n heightDeltas: Map<number, number>;\n measuredHeightDeltaTotal: number;\n // Baseline estimated heights for the active diff content. These are preserved\n // across style/collapse toggles and cleared only when estimate inputs change.\n estimatedSplitHeight: number | undefined;\n estimatedUnifiedHeight: number | undefined;\n // Sparse measured positions used to resume deep geometry scans near a target\n // diff line, rendered row, or scroll offset instead of replaying layout from\n // the first hunk.\n checkpoints: DiffLayoutCheckpoint[];\n // Total renderable diff rows for the current diff style and expansion state.\n totalLines: number;\n}\n\ninterface ResetLayoutCacheOptions {\n forceSimpleRecompute?: boolean;\n includeEstimatedHeights?: boolean;\n}\n\nconst LAYOUT_CHECKPOINT_INTERVAL = 5_000;\n\nlet instanceId = -1;\n\nexport class VirtualizedFileDiff<\n LAnnotation = undefined,\n> extends FileDiff<LAnnotation> {\n override readonly __id: string = `little-virtualized-file-diff:${++instanceId}`;\n\n public top: number | undefined;\n public height: number = 0;\n private metrics: VirtualFileMetrics;\n private cache: DiffLayoutCache = {\n heightDeltas: new Map(),\n measuredHeightDeltaTotal: 0,\n estimatedSplitHeight: undefined,\n estimatedUnifiedHeight: undefined,\n checkpoints: [],\n totalLines: 0,\n };\n private isVisible: boolean = false;\n private isSetup: boolean = false;\n private virtualizer: Virtualizer | CodeView<LAnnotation>;\n private layoutDirty = true;\n private forceRenderOverride: true | undefined;\n private currentCollapsed: boolean | undefined;\n\n constructor(\n options: FileDiffOptions<LAnnotation> | undefined,\n virtualizer: Virtualizer | CodeView<LAnnotation>,\n metrics?: Partial<VirtualFileMetrics>,\n workerManager?: WorkerPoolManager,\n isContainerManaged = false\n ) {\n super(options, workerManager, isContainerManaged);\n this.virtualizer = virtualizer;\n this.metrics = computeVirtualFileMetrics(metrics);\n }\n\n public setMetrics(\n metrics?: Partial<VirtualFileMetrics>,\n force = false\n ): void {\n const nextMetrics = computeVirtualFileMetrics(metrics);\n if (!force && areObjectsEqual(this.metrics, nextMetrics)) {\n return;\n }\n\n this.metrics = nextMetrics;\n this.resetLayoutCache({ includeEstimatedHeights: true });\n }\n\n // Get the height for a line, using cached value if available.\n // If not cached and hasMetadataLine is true, adds lineHeight for the metadata.\n private getLineHeight(lineIndex: number, hasMetadataLine = false): number {\n return (\n this.getEstimatedLineHeight(hasMetadataLine) +\n (this.cache.heightDeltas.get(lineIndex) ?? 0)\n );\n }\n\n private getEstimatedLineHeight(hasMetadataLine = false): number {\n const multiplier = hasMetadataLine ? 2 : 1;\n return this.metrics.lineHeight * multiplier;\n }\n\n override setOptions(options: FileDiffOptions<LAnnotation> | undefined): void {\n if (this.isAdvancedMode()) {\n throw new Error(\n 'VirtualizedFileDiff.setOptions cannot be used inside CodeView. Update CodeView options instead.'\n );\n }\n\n if (options == null) return;\n const { options: previousOptions } = this;\n const optionsChanged = !areOptionsEqual(previousOptions, options);\n const layoutChanged =\n optionsChanged && hasDiffLayoutOptionChanged(previousOptions, options);\n\n super.setOptions(options);\n\n if (layoutChanged) {\n this.resetLayoutCache({\n forceSimpleRecompute: true,\n includeEstimatedHeights: hasDiffEstimateOptionChanged(\n previousOptions,\n options\n ),\n });\n }\n // Any option can affect rendered DOM; only layout-affecting options clear\n // the measured height cache above.\n if (optionsChanged) {\n this.forceRenderOverride = true;\n }\n if (optionsChanged && this.isSimpleMode()) {\n this.virtualizer.instanceChanged(this, layoutChanged);\n }\n }\n\n override setThemeType(themeType: ThemeTypes): void {\n if (this.isAdvancedMode()) {\n throw new Error(\n 'VirtualizedFileDiff.setThemeType cannot be used inside CodeView. Update CodeView options instead.'\n );\n }\n\n super.setThemeType(themeType);\n }\n\n private resetLayoutCache({\n forceSimpleRecompute = false,\n includeEstimatedHeights = false,\n }: ResetLayoutCacheOptions = {}): void {\n this.layoutDirty = true;\n if (this.cache.heightDeltas.size > 0) {\n this.cache.heightDeltas.clear();\n }\n if (this.cache.measuredHeightDeltaTotal !== 0) {\n this.cache.measuredHeightDeltaTotal = 0;\n }\n if (this.cache.checkpoints.length > 0) {\n this.cache.checkpoints.length = 0;\n }\n if (this.cache.totalLines !== 0) {\n this.cache.totalLines = 0;\n }\n if (includeEstimatedHeights) {\n this.cache.estimatedSplitHeight = undefined;\n this.cache.estimatedUnifiedHeight = undefined;\n }\n if (this.renderRange != null) {\n this.renderRange = undefined;\n }\n // NOTE(amadeus): In CodeView we intentionally batch computes to all happen\n // at the same time, so we shouldn't trigger this there.\n if (forceSimpleRecompute && this.isSimpleMode()) {\n this.computeApproximateSize();\n }\n }\n\n // Measure rendered lines and update height cache.\n // Called after render to reconcile estimated vs actual heights.\n // Definitely need to optimize this in cases where there aren't any custom\n // line heights or in cases of extremely large files...\n public reconcileHeights(): boolean {\n let hasHeightChange = false;\n const { overflow = 'scroll' } = this.options;\n if (this.fileContainer == null || this.fileDiff == null) {\n if (this.height !== 0) {\n hasHeightChange = true;\n }\n this.height = 0;\n return hasHeightChange;\n }\n this.top = this.getVirtualizedTop();\n // NOTE(amadeus): We can probably be a lot smarter about this, and we\n // should be thinking about ways to improve this\n // If the file has no annotations and we are using the scroll variant, then\n // we can probably skip everything\n if (\n overflow === 'scroll' &&\n this.lineAnnotations.length === 0 &&\n !this.isResizeDebuggingEnabled()\n ) {\n return hasHeightChange;\n }\n const diffStyle = this.getDiffStyle();\n const codeGroups =\n diffStyle === 'split'\n ? [this.codeDeletions, this.codeAdditions]\n : [this.codeUnified];\n\n for (const codeGroup of codeGroups) {\n if (codeGroup == null) continue;\n const content = codeGroup.children[1];\n if (!(content instanceof HTMLElement)) continue;\n for (const line of content.children) {\n if (!(line instanceof HTMLElement)) continue;\n\n const lineIndexAttr = line.dataset.lineIndex;\n if (lineIndexAttr == null) continue;\n\n const lineIndex = parseLineIndex(lineIndexAttr, diffStyle);\n let measuredHeight = line.getBoundingClientRect().height;\n let hasMetadata = false;\n // Annotations or noNewline metadata increase the size of the their\n // attached line\n if (\n line.nextElementSibling instanceof HTMLElement &&\n ('lineAnnotation' in line.nextElementSibling.dataset ||\n 'noNewline' in line.nextElementSibling.dataset)\n ) {\n if ('noNewline' in line.nextElementSibling.dataset) {\n hasMetadata = true;\n }\n measuredHeight +=\n line.nextElementSibling.getBoundingClientRect().height;\n }\n const estimatedHeight = this.getEstimatedLineHeight(hasMetadata);\n const previousDelta = this.cache.heightDeltas.get(lineIndex) ?? 0;\n const nextDelta = measuredHeight - estimatedHeight;\n\n if (nextDelta === previousDelta) {\n continue;\n }\n\n hasHeightChange = true;\n this.cache.measuredHeightDeltaTotal += nextDelta - previousDelta;\n if (nextDelta === 0) {\n this.cache.heightDeltas.delete(lineIndex);\n } else {\n this.cache.heightDeltas.set(lineIndex, nextDelta);\n }\n }\n }\n\n if (hasHeightChange || this.isResizeDebuggingEnabled()) {\n this.computeApproximateSize(true);\n }\n return hasHeightChange;\n }\n\n public onRender = (dirty: boolean): boolean => {\n if (this.fileContainer == null) {\n return false;\n }\n if (dirty) {\n this.top = this.getVirtualizedTop();\n }\n return this.render();\n };\n\n // Prepares this item for CodeView layout by binding the latest diff, syncing\n // its virtualized top, and returning an approximate height. This method is\n // called while downstream items are being re-positioned, so later changes\n // should keep clean instances on a cached-height fast path.\n public prepareCodeViewItem(\n fileDiff: FileDiffMetadata,\n top: number,\n reset?: PendingCodeViewLayoutReset\n ): number {\n const targetChanged = !areDiffTargetsEqual(this.fileDiff, fileDiff);\n let shouldResetLayoutCache =\n reset?.resetDiffLayoutCache === true || targetChanged;\n let includeEstimatedHeights =\n targetChanged ||\n (reset?.resetDiffLayoutCache === true &&\n reset.includeEstimatedDiffHeights);\n\n if (reset?.metrics != null) {\n this.metrics = computeVirtualFileMetrics(reset.metrics);\n shouldResetLayoutCache = true;\n includeEstimatedHeights = true;\n }\n\n const { collapsed = false } = this.options;\n if (this.currentCollapsed !== collapsed) {\n this.currentCollapsed = collapsed;\n shouldResetLayoutCache = true;\n }\n\n if (shouldResetLayoutCache) {\n this.resetLayoutCache({ includeEstimatedHeights });\n }\n this.fileDiff = fileDiff;\n this.top = top;\n this.computeApproximateSize();\n return this.height;\n }\n\n public getLinePosition(\n lineNumber: number,\n side: SelectionSide = 'additions'\n ): { top: number; height: number } | undefined {\n if (this.fileDiff == null) {\n return undefined;\n }\n\n const targetLineIndexes = this.getLineIndex(lineNumber, side);\n if (targetLineIndexes == null) {\n return undefined;\n }\n\n const {\n disableFileHeader = false,\n expandUnchanged = false,\n collapsed = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n const diffStyle = this.getDiffStyle();\n const hunkSeparators = this.getHunkSeparatorType();\n const targetLineIndex =\n diffStyle === 'split' ? targetLineIndexes[1] : targetLineIndexes[0];\n this.approximateLayoutCheckpoints();\n const checkpoint = this.getLayoutCheckpointBeforeLineIndex(targetLineIndex);\n let top =\n checkpoint?.top ??\n getVirtualFileHeaderRegion(this.metrics, disableFileHeader);\n\n if (collapsed) {\n return { top, height: 0 };\n }\n\n let position: { top: number; height: number } | undefined;\n iterateOverDiff({\n diff: this.fileDiff,\n diffStyle,\n startingLine: checkpoint?.renderedLineIndex ?? 0,\n expandedHunks: expandUnchanged\n ? true\n : this.hunksRenderer.getExpandedHunksMap(),\n collapsedContextThreshold,\n callback: ({\n hunkIndex,\n hunk,\n collapsedBefore,\n collapsedAfter,\n deletionLine,\n additionLine,\n }) => {\n const lineIndex =\n diffStyle === 'split'\n ? (additionLine?.splitLineIndex ?? deletionLine?.splitLineIndex)\n : (additionLine?.unifiedLineIndex ??\n deletionLine?.unifiedLineIndex);\n if (lineIndex == null) {\n throw new Error(\n 'VirtualizedFileDiff.getLinePosition: missing line index data'\n );\n }\n\n if (collapsedBefore > 0) {\n const separator = getLeadingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n hunkIndex,\n hunkSpecs: hunk?.hunkSpecs,\n });\n if (separator != null) {\n top += separator.gapBefore;\n if (\n targetLineIndex >= lineIndex - collapsedBefore &&\n targetLineIndex < lineIndex\n ) {\n position = {\n top,\n height: separator.height,\n };\n return true;\n }\n top += separator.height + separator.gapAfter;\n }\n }\n\n const lineHeight = this.getLineHeight(\n lineIndex,\n (additionLine?.noEOFCR ?? false) || (deletionLine?.noEOFCR ?? false)\n );\n if (lineIndex === targetLineIndex) {\n position = {\n top,\n height: lineHeight,\n };\n return true;\n }\n top += lineHeight;\n\n if (collapsedAfter > 0) {\n const separator = getTrailingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n });\n if (separator != null) {\n if (\n targetLineIndex > lineIndex &&\n targetLineIndex <= lineIndex + collapsedAfter\n ) {\n position = {\n top: top + separator.gapBefore,\n height: separator.height,\n };\n return true;\n }\n top += separator.totalHeight;\n }\n }\n\n return false;\n },\n });\n\n return position;\n }\n\n public getNumericScrollAnchor(\n localViewportTop: number\n ): NumericScrollLineAnchor | undefined {\n if (this.fileDiff == null) {\n return undefined;\n }\n\n const {\n disableFileHeader = false,\n expandUnchanged = false,\n collapsed = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n if (collapsed) {\n return undefined;\n }\n\n const diffStyle = this.getDiffStyle();\n const hunkSeparators = this.getHunkSeparatorType();\n\n this.approximateLayoutCheckpoints();\n const checkpoint = this.getLayoutCheckpointBeforeTop(localViewportTop);\n let top =\n checkpoint?.top ??\n getVirtualFileHeaderRegion(this.metrics, disableFileHeader);\n let anchor: NumericScrollLineAnchor | undefined;\n\n // This may end up being quite expensive on extremely large files, we may\n // need to figure out how to anchor on different regions, or utilize\n // renderRange to shortcut this for us somehow\n iterateOverDiff({\n diff: this.fileDiff,\n diffStyle,\n startingLine: checkpoint?.renderedLineIndex ?? 0,\n expandedHunks: expandUnchanged\n ? true\n : this.hunksRenderer.getExpandedHunksMap(),\n collapsedContextThreshold,\n callback: ({\n hunkIndex,\n hunk,\n collapsedBefore,\n collapsedAfter,\n deletionLine,\n additionLine,\n }) => {\n const lineIndex =\n diffStyle === 'split'\n ? (additionLine?.splitLineIndex ?? deletionLine?.splitLineIndex)\n : (additionLine?.unifiedLineIndex ??\n deletionLine?.unifiedLineIndex);\n if (lineIndex == null) {\n throw new Error(\n 'VirtualizedFileDiff.getNumericScrollAnchor: missing line index data'\n );\n }\n\n if (collapsedBefore > 0) {\n const separator = getLeadingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n hunkIndex,\n hunkSpecs: hunk?.hunkSpecs,\n });\n if (separator != null) {\n top += separator.totalHeight;\n }\n }\n\n if (top >= localViewportTop) {\n if (deletionLine != null) {\n anchor = {\n lineNumber: deletionLine.lineNumber,\n side: 'deletions',\n top,\n };\n } else if (additionLine != null) {\n anchor = {\n lineNumber: additionLine.lineNumber,\n side: 'additions',\n top,\n };\n }\n if (anchor != null) {\n return true;\n }\n }\n\n const lineHeight = this.getLineHeight(\n lineIndex,\n (additionLine?.noEOFCR ?? false) || (deletionLine?.noEOFCR ?? false)\n );\n top += lineHeight;\n\n if (collapsedAfter > 0) {\n const separator = getTrailingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n });\n if (separator != null) {\n top += separator.totalHeight;\n }\n }\n\n return false;\n },\n });\n\n return anchor;\n }\n\n public getVirtualizedHeight(): number {\n return this.height;\n }\n\n public getAdvancedStickySpecs(\n windowSpecs?: RenderWindow\n ): StickySpecs | undefined {\n if (this.top == null || this.fileDiff == null) {\n return undefined;\n }\n if (this.options.collapsed === true) {\n return { topOffset: this.top, height: this.height };\n }\n const renderRange =\n windowSpecs != null\n ? this.computeRenderRangeFromWindow(\n this.fileDiff,\n this.top,\n windowSpecs\n )\n : this.renderRange;\n if (renderRange == null) {\n return undefined;\n }\n const { bufferBefore, bufferAfter, totalLines } = renderRange;\n // Rendered items flow contiguously in the sticky container with no buffer\n // spacers, so a header-only item (totalLines === 0, none of its rows fall\n // inside the window) must report where its header actually sits in that\n // flow, which depends on which side of the window its content is on:\n // - content ABOVE the window (item starts above window.top): the header\n // sits at the item's bottom so the following item connects, so offset by\n // bufferAfter.\n // - content BELOW the window (item starts at/after window.top, e.g. a\n // trailing header peeking in at the bottom): the header renders at the\n // item's top with nothing after it, so no offset. Always adding\n // bufferAfter here made getStickyBounds over-measure the sticky\n // container for that trailing case.\n let headerOnlyOffset = 0;\n if (totalLines === 0) {\n const activeWindow = windowSpecs ?? this.virtualizer.getWindowSpecs();\n if (this.top < activeWindow.top) {\n headerOnlyOffset = bufferAfter;\n }\n }\n return {\n topOffset: this.top + bufferBefore + headerOnlyOffset,\n height: this.height - (bufferBefore + bufferAfter),\n };\n }\n\n override cleanUp(recycle = false): void {\n if (this.fileContainer != null && this.isSimpleMode()) {\n this.getSimpleVirtualizer()?.disconnect(this.fileContainer);\n }\n if (!recycle) {\n this.resetLayoutCache({ includeEstimatedHeights: true });\n }\n this.isSetup = false;\n super.cleanUp(recycle);\n }\n\n override expandHunk = (\n hunkIndex: number,\n direction: ExpansionDirections,\n expansionLineCountOverride?: number\n ): void => {\n this.hunksRenderer.expandHunk(\n hunkIndex,\n direction,\n expansionLineCountOverride\n );\n this.forceRenderOverride = true;\n this.resetLayoutCache({ includeEstimatedHeights: true });\n if (this.isSimpleMode()) {\n this.computeApproximateSize();\n }\n this.virtualizer.instanceChanged(this, true);\n };\n\n public setVisibility(visible: boolean): void {\n if (this.isAdvancedMode() || this.fileContainer == null) {\n return;\n }\n this.renderRange = undefined;\n if (visible && !this.isVisible) {\n this.top = this.getVirtualizedTop();\n this.isVisible = true;\n } else if (!visible && this.isVisible) {\n this.isVisible = false;\n this.rerender();\n }\n }\n\n override rerender(): void {\n if (\n !this.enabled ||\n (this.fileDiff == null &&\n this.additionFile == null &&\n this.deletionFile == null)\n ) {\n return;\n }\n this.forceRenderOverride = true;\n this.virtualizer.instanceChanged(this, false);\n }\n\n // Compute the approximate size from the cached baseline estimate plus any\n // measured height deltas observed in rendered rows.\n // The reason we refer to this as `approximate size` is because heights my\n // dynamically change for a number of reasons so we can never be fully sure\n // if the height is 100% accurate\n private computeApproximateSize(force = false): void {\n const shouldValidateSize = this.isResizeDebuggingEnabled();\n if (!force && !this.layoutDirty && !shouldValidateSize) {\n return;\n }\n\n const isFirstCompute = this.height === 0;\n this.height = 0;\n this.cache.checkpoints = [];\n this.cache.totalLines = 0;\n if (this.fileDiff == null) {\n this.layoutDirty = false;\n return;\n }\n\n const { disableFileHeader = false, collapsed = false } = this.options;\n const headerRegion = getVirtualFileHeaderRegion(\n this.metrics,\n disableFileHeader\n );\n\n this.height += headerRegion;\n if (collapsed) {\n this.layoutDirty = false;\n return;\n }\n\n this.height =\n this.getActiveEstimatedHeight() + this.cache.measuredHeightDeltaTotal;\n\n if (shouldValidateSize && !isFirstCompute) {\n this.validateComputedHeight();\n }\n this.layoutDirty = false;\n }\n\n private getActiveEstimatedHeight(): number {\n this.ensureEstimatedDiffHeights();\n const estimatedHeight =\n this.getDiffStyle() === 'split'\n ? this.cache.estimatedSplitHeight\n : this.cache.estimatedUnifiedHeight;\n if (estimatedHeight == null) {\n throw new Error(\n 'VirtualizedFileDiff.getActiveEstimatedHeight: missing estimated height'\n );\n }\n return estimatedHeight;\n }\n\n private ensureEstimatedDiffHeights(): void {\n if (this.fileDiff == null) {\n this.cache.estimatedSplitHeight = undefined;\n this.cache.estimatedUnifiedHeight = undefined;\n return;\n }\n if (\n this.cache.estimatedSplitHeight != null &&\n this.cache.estimatedUnifiedHeight != null\n ) {\n return;\n }\n\n const {\n disableFileHeader = false,\n expandUnchanged = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n const { splitHeight, unifiedHeight } = computeEstimatedDiffHeights({\n fileDiff: this.fileDiff,\n metrics: this.metrics,\n disableFileHeader,\n hunkSeparators: this.getHunkSeparatorType(),\n expandUnchanged,\n expandedHunks: this.hunksRenderer.getExpandedHunksMap(),\n collapsedContextThreshold,\n });\n this.cache.estimatedSplitHeight = splitHeight;\n this.cache.estimatedUnifiedHeight = unifiedHeight;\n }\n\n private validateComputedHeight(): void {\n if (this.fileContainer == null || this.fileDiff == null) {\n return;\n }\n\n const rect = this.fileContainer.getBoundingClientRect();\n if (rect.height !== this.height) {\n console.log(\n 'VirtualizedFileDiff.computeApproximateSize: computed height doesnt match',\n {\n name: this.fileDiff.name,\n elementHeight: rect.height,\n computedHeight: this.height,\n }\n );\n } else {\n console.log(\n 'VirtualizedFileDiff.computeApproximateSize: computed height IS CORRECT'\n );\n }\n }\n\n override render({\n fileContainer,\n oldFile,\n newFile,\n fileDiff,\n forceRender = false,\n ...props\n }: FileDiffRenderProps<LAnnotation> = {}): boolean {\n const { forceRenderOverride, isSetup } = this;\n this.forceRenderOverride = undefined;\n\n this.fileDiff ??=\n fileDiff ??\n (oldFile != null && newFile != null\n ? // NOTE(amadeus): We might be forcing ourselves to double up the\n // computation of fileDiff (in the super.render() call), so we might want\n // to figure out a way to avoid that. That also could be just as simple as\n // passing through fileDiff though... so maybe we good?\n parseDiffFromFile(oldFile, newFile, this.options.parseDiffOptions)\n : undefined);\n\n fileContainer = this.getOrCreateFileContainer(fileContainer);\n\n if (this.fileDiff == null) {\n console.error(\n 'VirtualizedFileDiff.render: attempting to virtually render when we dont have the correct data'\n );\n return false;\n }\n\n if (!isSetup) {\n this.computeApproximateSize();\n const virtualizer = this.getSimpleVirtualizer();\n this.top ??= this.getVirtualizedTop();\n if (this.isAdvancedMode()) {\n this.isVisible = true;\n } else {\n if (virtualizer == null) {\n throw new Error(\n 'VirtualizedFileDiff.render: simple virtualizer is not available'\n );\n }\n virtualizer.connect(fileContainer, this);\n this.isVisible = virtualizer.isInstanceVisible(\n this.top ?? 0,\n this.height\n );\n }\n this.isSetup = true;\n } else {\n this.top ??= this.getVirtualizedTop();\n }\n\n if (!this.isVisible && this.isSimpleMode()) {\n return this.renderPlaceholder(this.height);\n }\n\n const windowSpecs = this.virtualizer.getWindowSpecs();\n const fileTop = this.top ?? 0;\n const renderRange = this.computeRenderRangeFromWindow(\n this.fileDiff,\n fileTop,\n windowSpecs\n );\n return super.render({\n fileDiff: this.fileDiff,\n fileContainer,\n renderRange,\n oldFile,\n newFile,\n forceRender: forceRenderOverride ?? forceRender,\n ...props,\n });\n }\n\n public syncVirtualizedTop(): void {\n this.top = this.getVirtualizedTop();\n }\n\n protected override shouldDisableVirtualizationBuffers(): boolean {\n return this.isAdvancedMode() || super.shouldDisableVirtualizationBuffers();\n }\n\n private isSimpleMode(): boolean {\n return this.virtualizer.type === 'simple';\n }\n\n private isAdvancedMode(): boolean {\n return this.virtualizer.type === 'advanced';\n }\n\n private getVirtualizedTop(): number | undefined {\n if (this.virtualizer.type === 'advanced') {\n return this.virtualizer.getLocalTopForInstance(this);\n }\n return this.fileContainer != null\n ? this.virtualizer.getOffsetInScrollContainer(this.fileContainer)\n : 0;\n }\n\n private getSimpleVirtualizer(): Virtualizer | undefined {\n return this.virtualizer.type === 'simple' ? this.virtualizer : undefined;\n }\n\n private isResizeDebuggingEnabled(): boolean {\n return this.getSimpleVirtualizer()?.config.resizeDebugging ?? false;\n }\n\n private getDiffStyle(): 'split' | 'unified' {\n return this.options.diffStyle ?? 'split';\n }\n\n private getHunkSeparatorType(): HunkSeparators {\n return getOptionHunkSeparatorType(this.options.hunkSeparators);\n }\n\n private approximateLayoutCheckpoints(): void {\n if (\n this.cache.checkpoints.length > 0 ||\n this.fileDiff == null ||\n this.fileDiff.hunks.length === 0 ||\n this.options.collapsed === true\n ) {\n return;\n }\n\n const {\n disableFileHeader = false,\n expandUnchanged = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n const finalHunkIndex = this.fileDiff.hunks.length - 1;\n const diffStyle = this.getDiffStyle();\n const hunkSeparators = this.getHunkSeparatorType();\n const expandedHunks = expandUnchanged\n ? true\n : this.hunksRenderer.getExpandedHunksMap();\n const heightDeltaPrefix = createHeightDeltaPrefix(this.cache.heightDeltas);\n let top = getVirtualFileHeaderRegion(this.metrics, disableFileHeader);\n let renderedLineIndex = 0;\n\n const processRows = ({\n rowCount,\n startLineIndex,\n preSeparatorHeight = 0,\n postSeparatorHeight = 0,\n metadataOffsets = [],\n }: {\n rowCount: number;\n startLineIndex: number;\n preSeparatorHeight?: number;\n postSeparatorHeight?: number;\n metadataOffsets?: number[];\n }) => {\n if (rowCount <= 0) {\n return;\n }\n\n const blockStart = renderedLineIndex;\n const blockEnd = renderedLineIndex + rowCount;\n let nextCheckpoint = getNextCheckpointIndex(blockStart);\n while (nextCheckpoint < blockEnd) {\n const offset = nextCheckpoint - blockStart;\n const checkpointTop =\n top +\n (offset > 0 ? preSeparatorHeight : 0) +\n offset * this.metrics.lineHeight +\n countMetadataOffsetsBefore(metadataOffsets, offset) *\n this.metrics.lineHeight +\n sumHeightDeltas(\n heightDeltaPrefix,\n startLineIndex,\n startLineIndex + offset\n );\n this.cache.checkpoints.push({\n renderedLineIndex: nextCheckpoint,\n lineIndex: startLineIndex + offset,\n top: checkpointTop,\n });\n nextCheckpoint += LAYOUT_CHECKPOINT_INTERVAL;\n }\n\n top +=\n preSeparatorHeight +\n rowCount * this.metrics.lineHeight +\n metadataOffsets.length * this.metrics.lineHeight +\n sumHeightDeltas(\n heightDeltaPrefix,\n startLineIndex,\n startLineIndex + rowCount\n ) +\n postSeparatorHeight;\n renderedLineIndex = blockEnd;\n };\n\n for (\n let hunkIndex = 0;\n hunkIndex < this.fileDiff.hunks.length;\n hunkIndex++\n ) {\n const hunk = this.fileDiff.hunks[hunkIndex];\n if (hunk == null) {\n throw new Error(\n 'VirtualizedFileDiff.approximateLayoutCheckpoints: invalid hunk index'\n );\n }\n\n const leadingRegion = getExpandedRegion({\n isPartial: this.fileDiff.isPartial,\n rangeSize: hunk.collapsedBefore,\n expandedHunks,\n hunkIndex,\n collapsedContextThreshold,\n });\n const leadingSeparatorHeight =\n leadingRegion.collapsedLines > 0\n ? (getLeadingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n hunkIndex,\n hunkSpecs: hunk.hunkSpecs,\n })?.totalHeight ?? 0)\n : 0;\n\n processRows({\n rowCount: leadingRegion.fromStart,\n startLineIndex:\n (diffStyle === 'split'\n ? hunk.splitLineStart\n : hunk.unifiedLineStart) - leadingRegion.rangeSize,\n });\n\n let pendingLeadingSeparatorHeight = leadingSeparatorHeight;\n processRows({\n rowCount: leadingRegion.fromEnd,\n startLineIndex:\n (diffStyle === 'split'\n ? hunk.splitLineStart\n : hunk.unifiedLineStart) - leadingRegion.fromEnd,\n preSeparatorHeight: pendingLeadingSeparatorHeight,\n });\n if (leadingRegion.fromEnd > 0) {\n pendingLeadingSeparatorHeight = 0;\n }\n\n const trailingRegion =\n hunkIndex === finalHunkIndex\n ? getTrailingExpandedRegion({\n fileDiff: this.fileDiff,\n hunkIndex,\n expandedHunks,\n collapsedContextThreshold,\n errorPrefix: 'VirtualizedFileDiff',\n })\n : undefined;\n const trailingSeparatorHeight =\n trailingRegion != null && trailingRegion.collapsedLines > 0\n ? (getTrailingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n })?.totalHeight ?? 0)\n : 0;\n const trailingExpandedCount =\n trailingRegion != null\n ? trailingRegion.fromStart + trailingRegion.fromEnd\n : 0;\n\n const hunkBodyRowCount =\n diffStyle === 'split' ? hunk.splitLineCount : hunk.unifiedLineCount;\n const hunkBodyStartLineIndex =\n diffStyle === 'split' ? hunk.splitLineStart : hunk.unifiedLineStart;\n processRows({\n rowCount: hunkBodyRowCount,\n startLineIndex: hunkBodyStartLineIndex,\n preSeparatorHeight: pendingLeadingSeparatorHeight,\n postSeparatorHeight:\n trailingExpandedCount === 0 ? trailingSeparatorHeight : 0,\n metadataOffsets: getHunkMetadataOffsets({\n diffStyle,\n hunk,\n rowCount: hunkBodyRowCount,\n }),\n });\n\n if (trailingRegion != null && trailingExpandedCount > 0) {\n processRows({\n rowCount: trailingExpandedCount,\n startLineIndex: hunkBodyStartLineIndex + hunkBodyRowCount,\n postSeparatorHeight: trailingSeparatorHeight,\n });\n }\n }\n\n this.cache.totalLines = renderedLineIndex;\n }\n\n // Find the nearest sparse layout checkpoint at or before an active\n // diff-style line index. Diff checkpoints also store the dense rendered-row\n // index, so deep line-position lookups can resume iteration from that\n // rendered row and replay only the nearby layout work instead of walking\n // from the first hunk.\n private getLayoutCheckpointBeforeLineIndex(\n lineIndex: number\n ): DiffLayoutCheckpoint | undefined {\n if (lineIndex <= 0 || this.cache.checkpoints.length === 0) {\n return undefined;\n }\n\n let low = 0;\n let high = this.cache.checkpoints.length - 1;\n let result: DiffLayoutCheckpoint | undefined;\n\n while (low <= high) {\n const mid = (low + high) >> 1;\n const checkpoint = this.cache.checkpoints[mid];\n if (checkpoint == null) {\n throw new Error('VirtualizedFileDiff: invalid checkpoint index');\n }\n if (checkpoint.lineIndex <= lineIndex) {\n result = checkpoint;\n low = mid + 1;\n } else {\n high = mid - 1;\n }\n }\n\n return result;\n }\n\n // Find the nearest sparse layout checkpoint at or before a scroll offset.\n // Render-range scans start from this checkpoint so variable-height diffs\n // only replay nearby rows. When `hunkLineCount` is provided, step backward\n // to a rendered hunk boundary so buffer calculations can reuse absolute hunk\n // offsets safely.\n private getLayoutCheckpointBeforeTop(\n top: number,\n hunkLineCount?: number\n ): DiffLayoutCheckpoint | undefined {\n let low = 0;\n let high = this.cache.checkpoints.length - 1;\n let resultIndex = -1;\n\n while (low <= high) {\n const mid = (low + high) >> 1;\n const checkpoint = this.cache.checkpoints[mid];\n if (checkpoint == null) {\n throw new Error('VirtualizedFileDiff: invalid checkpoint index');\n }\n if (checkpoint.top <= top) {\n resultIndex = mid;\n low = mid + 1;\n } else {\n high = mid - 1;\n }\n }\n\n if (hunkLineCount == null) {\n return resultIndex >= 0 ? this.cache.checkpoints[resultIndex] : undefined;\n }\n\n for (let index = resultIndex; index >= 0; index--) {\n const checkpoint = this.cache.checkpoints[index];\n if (checkpoint == null) {\n throw new Error('VirtualizedFileDiff: invalid checkpoint index');\n }\n if (checkpoint.renderedLineIndex % hunkLineCount === 0) {\n return checkpoint;\n }\n }\n\n return undefined;\n }\n\n private getExpandedLineCount(\n fileDiff: FileDiffMetadata,\n diffStyle: 'split' | 'unified'\n ): number {\n let count = 0;\n if (fileDiff.isPartial) {\n for (const hunk of fileDiff.hunks) {\n count +=\n diffStyle === 'split' ? hunk.splitLineCount : hunk.unifiedLineCount;\n }\n return count;\n }\n\n const {\n expandUnchanged = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n const expandedHunks = expandUnchanged\n ? true\n : this.hunksRenderer.getExpandedHunksMap();\n\n for (const [hunkIndex, hunk] of fileDiff.hunks.entries()) {\n const hunkCount =\n diffStyle === 'split' ? hunk.splitLineCount : hunk.unifiedLineCount;\n count += hunkCount;\n const collapsedBefore = Math.max(hunk.collapsedBefore, 0);\n const { fromStart, fromEnd, renderAll } = getExpandedRegion({\n isPartial: fileDiff.isPartial,\n rangeSize: collapsedBefore,\n expandedHunks,\n hunkIndex,\n collapsedContextThreshold,\n });\n if (collapsedBefore > 0) {\n count += renderAll ? collapsedBefore : fromStart + fromEnd;\n }\n }\n\n const trailingRegion = getTrailingExpandedRegion({\n fileDiff,\n hunkIndex: fileDiff.hunks.length - 1,\n expandedHunks,\n collapsedContextThreshold,\n errorPrefix: 'VirtualizedFileDiff',\n });\n if (trailingRegion != null) {\n count += trailingRegion.fromStart + trailingRegion.fromEnd;\n }\n\n return count;\n }\n\n private computeRenderRangeFromWindow(\n fileDiff: FileDiffMetadata,\n fileTop: number,\n { top, bottom }: RenderWindow\n ): RenderRange {\n const {\n disableFileHeader = false,\n expandUnchanged = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n const { hunkLineCount, lineHeight } = this.metrics;\n const diffStyle = this.getDiffStyle();\n const hunkSeparators = this.getHunkSeparatorType();\n const fileHeight = this.height;\n let lineCount =\n this.cache.totalLines > 0\n ? this.cache.totalLines\n : this.getExpandedLineCount(fileDiff, diffStyle);\n\n const headerRegion = getVirtualFileHeaderRegion(\n this.metrics,\n disableFileHeader\n );\n const paddingBottom =\n fileDiff.hunks.length > 0 ? getVirtualFilePaddingBottom(this.metrics) : 0;\n\n // File is outside render window\n if (fileTop < top - fileHeight || fileTop > bottom) {\n return {\n startingLine: 0,\n totalLines: 0,\n bufferBefore: 0,\n bufferAfter: fileHeight - headerRegion - paddingBottom,\n };\n }\n\n // Whole file is under hunkLineCount, just render it all\n if (lineCount <= hunkLineCount || fileDiff.hunks.length === 0) {\n return {\n startingLine: 0,\n totalLines: hunkLineCount,\n bufferBefore: 0,\n bufferAfter: 0,\n };\n }\n\n this.approximateLayoutCheckpoints();\n lineCount = this.cache.totalLines > 0 ? this.cache.totalLines : lineCount;\n\n const estimatedTargetLines = Math.ceil(\n Math.max(bottom - top, 0) / lineHeight\n );\n const totalLines =\n Math.ceil(estimatedTargetLines / hunkLineCount) * hunkLineCount +\n hunkLineCount;\n const totalHunks = totalLines / hunkLineCount;\n const overflowHunks = totalHunks;\n const hunkOffsets: number[] = [];\n // Halfway between top & bottom, represented as an absolute position\n const viewportCenter = (top + bottom) / 2;\n // Start the scan before the viewport so we collect hunk offsets that may be\n // needed for bufferBefore. This only chooses the scan origin; the returned\n // render range is still computed from the visible window below.\n const checkpoint = this.getLayoutCheckpointBeforeTop(\n Math.max(0, top - fileTop - totalLines * lineHeight * 2),\n hunkLineCount\n );\n\n let absoluteLineTop = fileTop + (checkpoint?.top ?? headerRegion);\n let currentLine = checkpoint?.renderedLineIndex ?? 0;\n let firstVisibleHunk: number | undefined;\n let centerHunk: number | undefined;\n let overflowCounter: number | undefined;\n\n iterateOverDiff({\n diff: fileDiff,\n diffStyle,\n startingLine: checkpoint?.renderedLineIndex ?? 0,\n expandedHunks: expandUnchanged\n ? true\n : this.hunksRenderer.getExpandedHunksMap(),\n collapsedContextThreshold,\n callback: ({\n hunkIndex,\n hunk,\n collapsedBefore,\n collapsedAfter,\n deletionLine,\n additionLine,\n }) => {\n const splitLineIndex =\n additionLine != null\n ? additionLine.splitLineIndex\n : deletionLine.splitLineIndex;\n const unifiedLineIndex =\n additionLine != null\n ? additionLine.unifiedLineIndex\n : deletionLine.unifiedLineIndex;\n const hasMetadata =\n (additionLine?.noEOFCR ?? false) || (deletionLine?.noEOFCR ?? false);\n const leadingSeparator =\n collapsedBefore > 0\n ? getLeadingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n hunkIndex,\n hunkSpecs: hunk?.hunkSpecs,\n })\n : undefined;\n const gapAdjustment = leadingSeparator?.totalHeight ?? 0;\n\n absoluteLineTop += gapAdjustment;\n\n const isAtHunkBoundary = currentLine % hunkLineCount === 0;\n const currentHunk = Math.floor(currentLine / hunkLineCount);\n\n // Track the boundary positional offset at a hunk\n if (isAtHunkBoundary) {\n hunkOffsets[currentHunk] =\n absoluteLineTop - (fileTop + headerRegion + gapAdjustment);\n\n // Check if we should bail (overflow complete)\n if (overflowCounter != null) {\n if (overflowCounter <= 0) {\n return true;\n }\n overflowCounter--;\n }\n }\n\n const lineHeight = this.getLineHeight(\n diffStyle === 'split' ? splitLineIndex : unifiedLineIndex,\n hasMetadata\n );\n\n // Track visible region\n if (absoluteLineTop > top - lineHeight && absoluteLineTop < bottom) {\n firstVisibleHunk ??= currentHunk;\n }\n\n // Track which hunk contains the viewport center\n // If viewport center is above this line and we haven't set centerHunk yet,\n // this is the first line at or past the center\n if (\n centerHunk == null &&\n absoluteLineTop + lineHeight > viewportCenter\n ) {\n centerHunk = currentHunk;\n }\n\n // Start overflow when we are out of the viewport at a hunk boundary\n if (\n overflowCounter == null &&\n absoluteLineTop >= bottom &&\n isAtHunkBoundary\n ) {\n overflowCounter = overflowHunks;\n }\n\n currentLine++;\n absoluteLineTop += lineHeight;\n\n if (collapsedAfter > 0) {\n absoluteLineTop +=\n getTrailingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n })?.totalHeight ?? 0;\n }\n\n return false;\n },\n });\n\n // No visible lines found\n if (firstVisibleHunk == null) {\n return {\n startingLine: 0,\n totalLines: 0,\n bufferBefore: 0,\n bufferAfter: fileHeight - headerRegion - paddingBottom,\n };\n }\n\n // Calculate balanced startingLine centered around the viewport center\n // Fall back to firstVisibleHunk if center wasn't found (e.g., center in a gap)\n centerHunk ??= firstVisibleHunk;\n const idealStartHunk = Math.round(centerHunk - totalHunks / 2);\n\n // Clamp startHunk: at the beginning, reduce totalLines; at the end, shift startHunk back\n const maxStartHunk = Math.max(\n 0,\n Math.ceil(lineCount / hunkLineCount) - totalHunks\n );\n const startHunk = Math.max(0, Math.min(idealStartHunk, maxStartHunk));\n const startingLine = startHunk * hunkLineCount;\n\n // If we wanted to start before 0, reduce totalLines by the clamped amount\n const clampedTotalLines =\n idealStartHunk < 0\n ? totalLines + idealStartHunk * hunkLineCount\n : totalLines;\n\n // Use hunkOffsets array for efficient buffer calculations\n const bufferBefore = hunkOffsets[startHunk] ?? 0;\n\n // Calculate bufferAfter using hunkOffset if available, otherwise use cumulative height\n const finalHunkIndex = startHunk + clampedTotalLines / hunkLineCount;\n const bufferAfter =\n finalHunkIndex < hunkOffsets.length\n ? fileHeight -\n headerRegion -\n hunkOffsets[finalHunkIndex] -\n // We gotta subtract the bottom padding off of the buffer\n paddingBottom\n : // We stopped early, calculate from current position\n fileHeight -\n (absoluteLineTop - fileTop) -\n // We gotta subtract the bottom padding off of the buffer\n paddingBottom;\n\n return {\n startingLine,\n totalLines: clampedTotalLines,\n bufferBefore,\n bufferAfter,\n };\n }\n}\n\ninterface HeightDeltaPrefix {\n lineIndexes: number[];\n prefixTotals: number[];\n}\n\nfunction createHeightDeltaPrefix(\n heightDeltas: Map<number, number>\n): HeightDeltaPrefix {\n const entries = Array.from(heightDeltas).sort((a, b) => a[0] - b[0]);\n const lineIndexes: number[] = [];\n const prefixTotals = [0];\n let total = 0;\n for (const [lineIndex, delta] of entries) {\n lineIndexes.push(lineIndex);\n total += delta;\n prefixTotals.push(total);\n }\n return { lineIndexes, prefixTotals };\n}\n\nfunction sumHeightDeltas(\n { lineIndexes, prefixTotals }: HeightDeltaPrefix,\n startLineIndex: number,\n endLineIndex: number\n): number {\n if (startLineIndex >= endLineIndex || lineIndexes.length === 0) {\n return 0;\n }\n const start = lowerBound(lineIndexes, startLineIndex);\n const end = lowerBound(lineIndexes, endLineIndex);\n return (prefixTotals[end] ?? 0) - (prefixTotals[start] ?? 0);\n}\n\nfunction lowerBound(values: number[], target: number): number {\n let low = 0;\n let high = values.length;\n while (low < high) {\n const mid = (low + high) >> 1;\n const value = values[mid];\n if (value == null) {\n throw new Error('VirtualizedFileDiff: invalid prefix index');\n }\n if (value < target) {\n low = mid + 1;\n } else {\n high = mid;\n }\n }\n return low;\n}\n\nfunction getNextCheckpointIndex(renderedLineIndex: number): number {\n return (\n Math.ceil(renderedLineIndex / LAYOUT_CHECKPOINT_INTERVAL) *\n LAYOUT_CHECKPOINT_INTERVAL\n );\n}\n\nfunction countMetadataOffsetsBefore(\n metadataOffsets: number[],\n offset: number\n): number {\n let count = 0;\n for (const metadataOffset of metadataOffsets) {\n if (metadataOffset < offset) {\n count++;\n }\n }\n return count;\n}\n\nfunction getHunkMetadataOffsets({\n diffStyle,\n hunk,\n rowCount,\n}: {\n diffStyle: 'split' | 'unified';\n hunk: Hunk;\n rowCount: number;\n}): number[] {\n if (rowCount <= 0 || (!hunk.noEOFCRAdditions && !hunk.noEOFCRDeletions)) {\n return [];\n }\n\n const lastContent = hunk.hunkContent.at(-1);\n if (lastContent == null) {\n return [];\n }\n\n if (lastContent.type === 'context') {\n return [rowCount - 1];\n }\n\n const splitCount = Math.max(lastContent.deletions, lastContent.additions);\n const unifiedCount = lastContent.deletions + lastContent.additions;\n if (diffStyle === 'split') {\n return splitCount > 0 && (hunk.noEOFCRAdditions || hunk.noEOFCRDeletions)\n ? [rowCount - 1]\n : [];\n }\n\n const offsets: number[] = [];\n const contentStartOffset = rowCount - unifiedCount;\n if (lastContent.deletions > 0 && hunk.noEOFCRDeletions) {\n offsets.push(contentStartOffset + lastContent.deletions - 1);\n }\n if (lastContent.additions > 0 && hunk.noEOFCRAdditions) {\n offsets.push(rowCount - 1);\n }\n return offsets;\n}\n\nfunction hasDiffLayoutOptionChanged<LAnnotation>(\n previousOptions: FileDiffOptions<LAnnotation>,\n nextOptions: FileDiffOptions<LAnnotation>\n): boolean {\n return (\n (previousOptions.diffStyle ?? 'split') !==\n (nextOptions.diffStyle ?? 'split') ||\n (previousOptions.overflow ?? 'scroll') !==\n (nextOptions.overflow ?? 'scroll') ||\n (previousOptions.collapsed ?? false) !== (nextOptions.collapsed ?? false) ||\n (previousOptions.disableLineNumbers ?? false) !==\n (nextOptions.disableLineNumbers ?? false) ||\n (previousOptions.disableFileHeader ?? false) !==\n (nextOptions.disableFileHeader ?? false) ||\n (previousOptions.diffIndicators ?? 'bars') !==\n (nextOptions.diffIndicators ?? 'bars') ||\n (previousOptions.hunkSeparators ?? 'line-info') !==\n (nextOptions.hunkSeparators ?? 'line-info') ||\n (previousOptions.expandUnchanged ?? false) !==\n (nextOptions.expandUnchanged ?? false) ||\n (previousOptions.collapsedContextThreshold ??\n DEFAULT_COLLAPSED_CONTEXT_THRESHOLD) !==\n (nextOptions.collapsedContextThreshold ??\n DEFAULT_COLLAPSED_CONTEXT_THRESHOLD) ||\n previousOptions.unsafeCSS !== nextOptions.unsafeCSS\n );\n}\n\nfunction hasDiffEstimateOptionChanged<LAnnotation>(\n previousOptions: FileDiffOptions<LAnnotation>,\n nextOptions: FileDiffOptions<LAnnotation>\n): boolean {\n return (\n (previousOptions.disableFileHeader ?? false) !==\n (nextOptions.disableFileHeader ?? false) ||\n (previousOptions.hunkSeparators ?? 'line-info') !==\n (nextOptions.hunkSeparators ?? 'line-info') ||\n (previousOptions.expandUnchanged ?? false) !==\n (nextOptions.expandUnchanged ?? false) ||\n (previousOptions.collapsedContextThreshold ??\n DEFAULT_COLLAPSED_CONTEXT_THRESHOLD) !==\n (nextOptions.collapsedContextThreshold ??\n DEFAULT_COLLAPSED_CONTEXT_THRESHOLD)\n );\n}\n\nfunction getOptionHunkSeparatorType<LAnnotation>(\n hunkSeparators: FileDiffOptions<LAnnotation>['hunkSeparators'] | undefined\n): HunkSeparators {\n return typeof hunkSeparators === 'function'\n ? 'custom'\n : (hunkSeparators ?? 'line-info');\n}\n\n// Extracts the view-specific line index from the data-line-index attribute.\n// Format is \"unifiedIndex,splitIndex\"\nfunction parseLineIndex(\n lineIndexAttr: string,\n diffStyle: 'split' | 'unified'\n): number {\n const [unifiedIndex, splitIndex] = lineIndexAttr.split(',').map(Number);\n return diffStyle === 'split' ? splitIndex : unifiedIndex;\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"VirtualizedFileDiff.js","names":["position: { top: number; height: number } | undefined","anchor: NumericScrollLineAnchor | undefined","result: DiffLayoutCheckpoint | undefined","hunkOffsets: number[]","firstVisibleHunk: number | undefined","centerHunk: number | undefined","overflowCounter: number | undefined","lineHeight","height: number | undefined","lineIndexes: number[]","offsets: number[]"],"sources":["../../src/components/VirtualizedFileDiff.ts"],"sourcesContent":["import { DEFAULT_COLLAPSED_CONTEXT_THRESHOLD } from '../constants';\nimport type {\n DiffLineAnnotation,\n ExpansionDirections,\n FileDiffMetadata,\n Hunk,\n HunkSeparators,\n NumericScrollLineAnchor,\n PendingCodeViewLayoutReset,\n RenderRange,\n RenderWindow,\n SelectionSide,\n StickySpecs,\n ThemeTypes,\n VirtualFileMetrics,\n} from '../types';\nimport { areDiffTargetsEqual } from '../utils/areDiffTargetsEqual';\nimport { areObjectsEqual } from '../utils/areObjectsEqual';\nimport { areOptionsEqual } from '../utils/areOptionsEqual';\nimport { computeEstimatedDiffHeights } from '../utils/computeEstimatedDiffHeights';\nimport {\n computeVirtualFileMetrics,\n getVirtualFileHeaderRegion,\n getVirtualFilePaddingBottom,\n} from '../utils/computeVirtualFileMetrics';\nimport {\n FILE_ANNOTATION_DOM_KEY,\n FILE_ANNOTATION_LINE_NUMBER,\n includesFileAnnotations,\n shouldRenderFileAnnotations,\n} from '../utils/includesFileAnnotations';\nimport { iterateOverDiff } from '../utils/iterateOverDiff';\nimport { parseDiffFromFile } from '../utils/parseDiffFromFile';\nimport {\n getExpandedRegion,\n getLeadingHunkSeparatorLayout,\n getTrailingExpandedRegion,\n getTrailingHunkSeparatorLayout,\n} from '../utils/virtualDiffLayout';\nimport type { WorkerPoolManager } from '../worker';\nimport type { CodeView } from './CodeView';\nimport {\n FileDiff,\n type FileDiffOptions,\n type FileDiffRenderProps,\n} from './FileDiff';\nimport type { Virtualizer } from './Virtualizer';\n\ninterface DiffLayoutCheckpoint {\n renderedLineIndex: number;\n lineIndex: number;\n top: number;\n}\n\ninterface DiffLayoutCache {\n // Sparse map: view-specific line index -> measured height delta from the\n // baseline line height. Only stores lines that differ from the estimate.\n heightDeltas: Map<number, number>;\n // Aggregate delta from estimated height, including source row deltas and the\n // zero-baseline file annotation row height.\n measuredHeightDeltaTotal: number;\n // Baseline estimated heights for the active diff content. These are preserved\n // across style/collapse toggles and cleared only when estimate inputs change.\n estimatedSplitHeight: number | undefined;\n estimatedUnifiedHeight: number | undefined;\n // Sparse measured positions used to resume deep geometry scans near a target\n // diff line, rendered row, or scroll offset instead of replaying layout from\n // the first hunk.\n checkpoints: DiffLayoutCheckpoint[];\n // Total renderable diff rows for the current diff style and expansion state.\n totalLines: number;\n // Measured height for the file annotation row. Starts at 0 so\n // unmeasured annotations behave like all other unmeasured annotations.\n fileAnnotationHeight: number;\n}\n\ninterface ResetLayoutCacheOptions {\n forceSimpleRecompute?: boolean;\n includeEstimatedHeights?: boolean;\n}\n\nconst LAYOUT_CHECKPOINT_INTERVAL = 5_000;\n\nlet instanceId = -1;\n\nexport class VirtualizedFileDiff<\n LAnnotation = undefined,\n> extends FileDiff<LAnnotation> {\n override readonly __id: string = `little-virtualized-file-diff:${++instanceId}`;\n\n public top: number | undefined;\n public height: number = 0;\n private metrics: VirtualFileMetrics;\n private cache: DiffLayoutCache = {\n heightDeltas: new Map(),\n measuredHeightDeltaTotal: 0,\n estimatedSplitHeight: undefined,\n estimatedUnifiedHeight: undefined,\n checkpoints: [],\n totalLines: 0,\n fileAnnotationHeight: 0,\n };\n private isVisible: boolean = false;\n private isSetup: boolean = false;\n private virtualizer: Virtualizer | CodeView<LAnnotation>;\n private layoutDirty = true;\n private forceRenderOverride: true | undefined;\n private currentCollapsed: boolean | undefined;\n\n constructor(\n options: FileDiffOptions<LAnnotation> | undefined,\n virtualizer: Virtualizer | CodeView<LAnnotation>,\n metrics?: Partial<VirtualFileMetrics>,\n workerManager?: WorkerPoolManager,\n isContainerManaged = false\n ) {\n super(options, workerManager, isContainerManaged);\n this.virtualizer = virtualizer;\n this.metrics = computeVirtualFileMetrics(metrics);\n }\n\n public setMetrics(\n metrics?: Partial<VirtualFileMetrics>,\n force = false\n ): void {\n const nextMetrics = computeVirtualFileMetrics(metrics);\n if (!force && areObjectsEqual(this.metrics, nextMetrics)) {\n return;\n }\n\n this.metrics = nextMetrics;\n this.resetLayoutCache({ includeEstimatedHeights: true });\n }\n\n override setLineAnnotations(\n lineAnnotations: DiffLineAnnotation<LAnnotation>[]\n ): void {\n if (this.syncLineAnnotations(lineAnnotations)) {\n this.resetLayoutCache({ includeEstimatedHeights: false });\n }\n }\n\n private syncLineAnnotations(\n lineAnnotations: DiffLineAnnotation<LAnnotation>[] | undefined\n ): boolean {\n if (\n lineAnnotations == null ||\n lineAnnotations === this.lineAnnotations ||\n (lineAnnotations.length === 0 && this.lineAnnotations.length === 0)\n ) {\n return false;\n }\n super.setLineAnnotations(lineAnnotations);\n return true;\n }\n\n private setFileAnnotationHeight(nextHeight: number): boolean {\n const previousHeight = this.cache.fileAnnotationHeight;\n if (nextHeight === previousHeight) {\n return false;\n }\n\n this.cache.fileAnnotationHeight = nextHeight;\n this.cache.measuredHeightDeltaTotal += nextHeight - previousHeight;\n return true;\n }\n\n private hasFileAnnotations(\n fileDiff: FileDiffMetadata | undefined = this.fileDiff\n ): boolean {\n if (fileDiff == null || !includesFileAnnotations(this.lineAnnotations)) {\n return false;\n }\n return this.lineAnnotations.some((annotation) => {\n if (annotation.lineNumber !== FILE_ANNOTATION_LINE_NUMBER) {\n return false;\n }\n // Lets ensure for singled sided diffs that the sides match\n if (fileDiff.type === 'new') {\n return annotation.side === 'additions';\n }\n if (fileDiff.type === 'deleted') {\n return annotation.side === 'deletions';\n }\n return true;\n });\n }\n\n // Get the height for a line, using cached value if available.\n // If not cached and hasMetadataLine is true, adds lineHeight for the metadata.\n private getLineHeight(lineIndex: number, hasMetadataLine = false): number {\n return (\n this.getEstimatedLineHeight(hasMetadataLine) +\n (this.cache.heightDeltas.get(lineIndex) ?? 0)\n );\n }\n\n private getEstimatedLineHeight(hasMetadataLine = false): number {\n const multiplier = hasMetadataLine ? 2 : 1;\n return this.metrics.lineHeight * multiplier;\n }\n\n override setOptions(options: FileDiffOptions<LAnnotation> | undefined): void {\n if (this.isAdvancedMode()) {\n throw new Error(\n 'VirtualizedFileDiff.setOptions cannot be used inside CodeView. Update CodeView options instead.'\n );\n }\n\n if (options == null) return;\n const { options: previousOptions } = this;\n const optionsChanged = !areOptionsEqual(previousOptions, options);\n const layoutChanged =\n optionsChanged && hasDiffLayoutOptionChanged(previousOptions, options);\n\n super.setOptions(options);\n\n if (layoutChanged) {\n this.resetLayoutCache({\n forceSimpleRecompute: true,\n includeEstimatedHeights: hasDiffEstimateOptionChanged(\n previousOptions,\n options\n ),\n });\n }\n // Any option can affect rendered DOM; only layout-affecting options clear\n // the measured height cache above.\n if (optionsChanged) {\n this.forceRenderOverride = true;\n }\n if (optionsChanged && this.isSimpleMode()) {\n this.virtualizer.instanceChanged(this, layoutChanged);\n }\n }\n\n override setThemeType(themeType: ThemeTypes): void {\n if (this.isAdvancedMode()) {\n throw new Error(\n 'VirtualizedFileDiff.setThemeType cannot be used inside CodeView. Update CodeView options instead.'\n );\n }\n\n super.setThemeType(themeType);\n }\n\n private resetLayoutCache({\n forceSimpleRecompute = false,\n includeEstimatedHeights = false,\n }: ResetLayoutCacheOptions = {}): void {\n this.layoutDirty = true;\n this.cache.fileAnnotationHeight = 0;\n if (this.cache.heightDeltas.size > 0) {\n this.cache.heightDeltas.clear();\n }\n if (this.cache.measuredHeightDeltaTotal !== 0) {\n this.cache.measuredHeightDeltaTotal = 0;\n }\n if (this.cache.checkpoints.length > 0) {\n this.cache.checkpoints.length = 0;\n }\n if (this.cache.totalLines !== 0) {\n this.cache.totalLines = 0;\n }\n if (includeEstimatedHeights) {\n this.cache.estimatedSplitHeight = undefined;\n this.cache.estimatedUnifiedHeight = undefined;\n }\n if (this.renderRange != null) {\n this.renderRange = undefined;\n }\n // NOTE(amadeus): In CodeView we intentionally batch computes to all happen\n // at the same time, so we shouldn't trigger this there.\n if (forceSimpleRecompute && this.isSimpleMode()) {\n this.computeApproximateSize();\n }\n }\n\n // Measure rendered lines and update height cache.\n // Called after render to reconcile estimated vs actual heights.\n // Definitely need to optimize this in cases where there aren't any custom\n // line heights or in cases of extremely large files...\n public reconcileHeights(): boolean {\n let hasHeightChange = false;\n const { overflow = 'scroll' } = this.options;\n if (this.fileContainer == null || this.fileDiff == null) {\n if (this.height !== 0) {\n hasHeightChange = true;\n }\n this.height = 0;\n return hasHeightChange;\n }\n this.top = this.getVirtualizedTop();\n // NOTE(amadeus): We can probably be a lot smarter about this, and we\n // should be thinking about ways to improve this\n // If the file has no annotations and we are using the scroll variant, then\n // we can probably skip everything\n if (\n overflow === 'scroll' &&\n this.lineAnnotations.length === 0 &&\n !this.isResizeDebuggingEnabled()\n ) {\n return hasHeightChange;\n }\n const diffStyle = this.getDiffStyle();\n const codeGroups =\n diffStyle === 'split'\n ? [this.codeDeletions, this.codeAdditions]\n : [this.codeUnified];\n\n const hasFileAnnotations = this.hasFileAnnotations(this.fileDiff);\n if (\n this.renderRange != null &&\n hasFileAnnotations &&\n shouldRenderFileAnnotations(this.renderRange)\n ) {\n const fileAnnotationHeight = measureFileAnnotationHeight(codeGroups);\n const nextFileAnnotationHeight = fileAnnotationHeight ?? 0;\n if (this.setFileAnnotationHeight(nextFileAnnotationHeight)) {\n hasHeightChange = true;\n }\n } else if (!hasFileAnnotations && this.setFileAnnotationHeight(0)) {\n hasHeightChange = true;\n }\n\n for (const codeGroup of codeGroups) {\n if (codeGroup == null) continue;\n const content = codeGroup.children[1];\n if (!(content instanceof HTMLElement)) continue;\n for (const line of content.children) {\n if (!(line instanceof HTMLElement)) continue;\n\n const lineIndexAttr = line.dataset.lineIndex;\n if (lineIndexAttr == null) continue;\n\n const lineIndex = parseLineIndex(lineIndexAttr, diffStyle);\n let measuredHeight = line.getBoundingClientRect().height;\n let hasMetadata = false;\n // Annotations or noNewline metadata increase the size of the their\n // attached line\n if (\n line.nextElementSibling instanceof HTMLElement &&\n ('lineAnnotation' in line.nextElementSibling.dataset ||\n 'noNewline' in line.nextElementSibling.dataset)\n ) {\n if ('noNewline' in line.nextElementSibling.dataset) {\n hasMetadata = true;\n }\n measuredHeight +=\n line.nextElementSibling.getBoundingClientRect().height;\n }\n const estimatedHeight = this.getEstimatedLineHeight(hasMetadata);\n const previousDelta = this.cache.heightDeltas.get(lineIndex) ?? 0;\n const nextDelta = measuredHeight - estimatedHeight;\n\n if (nextDelta === previousDelta) {\n continue;\n }\n\n hasHeightChange = true;\n this.cache.measuredHeightDeltaTotal += nextDelta - previousDelta;\n if (nextDelta === 0) {\n this.cache.heightDeltas.delete(lineIndex);\n } else {\n this.cache.heightDeltas.set(lineIndex, nextDelta);\n }\n }\n }\n\n if (hasHeightChange || this.isResizeDebuggingEnabled()) {\n this.computeApproximateSize(true);\n }\n return hasHeightChange;\n }\n\n public onRender = (dirty: boolean): boolean => {\n if (this.fileContainer == null) {\n return false;\n }\n if (dirty) {\n this.top = this.getVirtualizedTop();\n }\n return this.render();\n };\n\n // Prepares this item for CodeView layout by binding the latest diff, syncing\n // its virtualized top, and returning an approximate height. This method is\n // called while downstream items are being re-positioned, so later changes\n // should keep clean instances on a cached-height fast path.\n public prepareCodeViewItem(\n fileDiff: FileDiffMetadata,\n top: number,\n reset?: PendingCodeViewLayoutReset,\n lineAnnotations?: DiffLineAnnotation<LAnnotation>[]\n ): number {\n const targetChanged = !areDiffTargetsEqual(this.fileDiff, fileDiff);\n const annotationsChanged = this.syncLineAnnotations(lineAnnotations);\n let shouldResetLayoutCache =\n reset?.resetDiffLayoutCache === true ||\n targetChanged ||\n annotationsChanged;\n let includeEstimatedHeights =\n targetChanged ||\n (reset?.resetDiffLayoutCache === true &&\n reset.includeEstimatedDiffHeights);\n\n if (reset?.metrics != null) {\n this.metrics = computeVirtualFileMetrics(reset.metrics);\n shouldResetLayoutCache = true;\n includeEstimatedHeights = true;\n }\n\n const { collapsed = false } = this.options;\n if (this.currentCollapsed !== collapsed) {\n this.currentCollapsed = collapsed;\n shouldResetLayoutCache = true;\n }\n\n if (shouldResetLayoutCache) {\n this.resetLayoutCache({ includeEstimatedHeights });\n }\n this.fileDiff = fileDiff;\n this.top = top;\n this.computeApproximateSize();\n return this.height;\n }\n\n public getLinePosition(\n lineNumber: number,\n side: SelectionSide = 'additions'\n ): { top: number; height: number } | undefined {\n if (this.fileDiff == null || lineNumber < 1) {\n return undefined;\n }\n\n const targetLineIndexes = this.getLineIndex(lineNumber, side);\n if (targetLineIndexes == null) {\n return undefined;\n }\n\n const {\n disableFileHeader = false,\n expandUnchanged = false,\n collapsed = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n const diffStyle = this.getDiffStyle();\n const hunkSeparators = this.getHunkSeparatorType();\n const targetLineIndex =\n diffStyle === 'split' ? targetLineIndexes[1] : targetLineIndexes[0];\n this.approximateLayoutCheckpoints();\n const headerRegion = getVirtualFileHeaderRegion(\n this.metrics,\n disableFileHeader\n );\n const checkpoint = this.getLayoutCheckpointBeforeLineIndex(targetLineIndex);\n let top = checkpoint?.top ?? headerRegion + this.cache.fileAnnotationHeight;\n\n if (collapsed) {\n return { top: headerRegion, height: 0 };\n }\n\n let position: { top: number; height: number } | undefined;\n iterateOverDiff({\n diff: this.fileDiff,\n diffStyle,\n startingLine: checkpoint?.renderedLineIndex ?? 0,\n expandedHunks: expandUnchanged\n ? true\n : this.hunksRenderer.getExpandedHunksMap(),\n collapsedContextThreshold,\n callback: ({\n hunkIndex,\n hunk,\n collapsedBefore,\n collapsedAfter,\n deletionLine,\n additionLine,\n }) => {\n const lineIndex =\n diffStyle === 'split'\n ? (additionLine?.splitLineIndex ?? deletionLine?.splitLineIndex)\n : (additionLine?.unifiedLineIndex ??\n deletionLine?.unifiedLineIndex);\n if (lineIndex == null) {\n throw new Error(\n 'VirtualizedFileDiff.getLinePosition: missing line index data'\n );\n }\n\n if (collapsedBefore > 0) {\n const separator = getLeadingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n hunkIndex,\n hunkSpecs: hunk?.hunkSpecs,\n });\n if (separator != null) {\n top += separator.gapBefore;\n if (\n targetLineIndex >= lineIndex - collapsedBefore &&\n targetLineIndex < lineIndex\n ) {\n position = {\n top,\n height: separator.height,\n };\n return true;\n }\n top += separator.height + separator.gapAfter;\n }\n }\n\n const lineHeight = this.getLineHeight(\n lineIndex,\n (additionLine?.noEOFCR ?? false) || (deletionLine?.noEOFCR ?? false)\n );\n if (lineIndex === targetLineIndex) {\n position = {\n top,\n height: lineHeight,\n };\n return true;\n }\n top += lineHeight;\n\n if (collapsedAfter > 0) {\n const separator = getTrailingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n });\n if (separator != null) {\n if (\n targetLineIndex > lineIndex &&\n targetLineIndex <= lineIndex + collapsedAfter\n ) {\n position = {\n top: top + separator.gapBefore,\n height: separator.height,\n };\n return true;\n }\n top += separator.totalHeight;\n }\n }\n\n return false;\n },\n });\n\n return position;\n }\n\n public getNumericScrollAnchor(\n localViewportTop: number\n ): NumericScrollLineAnchor | undefined {\n if (this.fileDiff == null) {\n return undefined;\n }\n\n const {\n disableFileHeader = false,\n expandUnchanged = false,\n collapsed = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n if (collapsed) {\n return undefined;\n }\n\n const diffStyle = this.getDiffStyle();\n const hunkSeparators = this.getHunkSeparatorType();\n\n this.approximateLayoutCheckpoints();\n const checkpoint = this.getLayoutCheckpointBeforeTop(localViewportTop);\n let top =\n checkpoint?.top ??\n getVirtualFileHeaderRegion(this.metrics, disableFileHeader) +\n this.cache.fileAnnotationHeight;\n let anchor: NumericScrollLineAnchor | undefined;\n\n // This may end up being quite expensive on extremely large files, we may\n // need to figure out how to anchor on different regions, or utilize\n // renderRange to shortcut this for us somehow\n iterateOverDiff({\n diff: this.fileDiff,\n diffStyle,\n startingLine: checkpoint?.renderedLineIndex ?? 0,\n expandedHunks: expandUnchanged\n ? true\n : this.hunksRenderer.getExpandedHunksMap(),\n collapsedContextThreshold,\n callback: ({\n hunkIndex,\n hunk,\n collapsedBefore,\n collapsedAfter,\n deletionLine,\n additionLine,\n }) => {\n const lineIndex =\n diffStyle === 'split'\n ? (additionLine?.splitLineIndex ?? deletionLine?.splitLineIndex)\n : (additionLine?.unifiedLineIndex ??\n deletionLine?.unifiedLineIndex);\n if (lineIndex == null) {\n throw new Error(\n 'VirtualizedFileDiff.getNumericScrollAnchor: missing line index data'\n );\n }\n\n if (collapsedBefore > 0) {\n const separator = getLeadingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n hunkIndex,\n hunkSpecs: hunk?.hunkSpecs,\n });\n if (separator != null) {\n top += separator.totalHeight;\n }\n }\n\n if (top >= localViewportTop) {\n if (deletionLine != null) {\n anchor = {\n lineNumber: deletionLine.lineNumber,\n side: 'deletions',\n top,\n };\n } else if (additionLine != null) {\n anchor = {\n lineNumber: additionLine.lineNumber,\n side: 'additions',\n top,\n };\n }\n if (anchor != null) {\n return true;\n }\n }\n\n const lineHeight = this.getLineHeight(\n lineIndex,\n (additionLine?.noEOFCR ?? false) || (deletionLine?.noEOFCR ?? false)\n );\n top += lineHeight;\n\n if (collapsedAfter > 0) {\n const separator = getTrailingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n });\n if (separator != null) {\n top += separator.totalHeight;\n }\n }\n\n return false;\n },\n });\n\n return anchor;\n }\n\n public getVirtualizedHeight(): number {\n return this.height;\n }\n\n public getAdvancedStickySpecs(\n windowSpecs?: RenderWindow\n ): StickySpecs | undefined {\n if (this.top == null || this.fileDiff == null) {\n return undefined;\n }\n if (this.options.collapsed === true) {\n return { topOffset: this.top, height: this.height };\n }\n const renderRange =\n windowSpecs != null\n ? this.computeRenderRangeFromWindow(\n this.fileDiff,\n this.top,\n windowSpecs\n )\n : this.renderRange;\n if (renderRange == null) {\n return undefined;\n }\n const { bufferBefore, bufferAfter, totalLines } = renderRange;\n // Rendered items flow contiguously in the sticky container with no buffer\n // spacers, so a header-only item (totalLines === 0, none of its rows fall\n // inside the window) must report where its header actually sits in that\n // flow, which depends on which side of the window its content is on:\n // - content ABOVE the window (item starts above window.top): the header\n // sits at the item's bottom so the following item connects, so offset by\n // bufferAfter.\n // - content BELOW the window (item starts at/after window.top, e.g. a\n // trailing header peeking in at the bottom): the header renders at the\n // item's top with nothing after it, so no offset. Always adding\n // bufferAfter here made getStickyBounds over-measure the sticky\n // container for that trailing case.\n let headerOnlyOffset = 0;\n if (totalLines === 0) {\n const activeWindow = windowSpecs ?? this.virtualizer.getWindowSpecs();\n if (this.top < activeWindow.top) {\n headerOnlyOffset = bufferAfter;\n }\n }\n return {\n topOffset: this.top + bufferBefore + headerOnlyOffset,\n height: this.height - (bufferBefore + bufferAfter),\n };\n }\n\n override cleanUp(recycle = false): void {\n if (this.fileContainer != null && this.isSimpleMode()) {\n this.getSimpleVirtualizer()?.disconnect(this.fileContainer);\n }\n if (!recycle) {\n this.resetLayoutCache({ includeEstimatedHeights: true });\n }\n this.isSetup = false;\n super.cleanUp(recycle);\n }\n\n override expandHunk = (\n hunkIndex: number,\n direction: ExpansionDirections,\n expansionLineCountOverride?: number\n ): void => {\n this.hunksRenderer.expandHunk(\n hunkIndex,\n direction,\n expansionLineCountOverride\n );\n this.forceRenderOverride = true;\n this.resetLayoutCache({ includeEstimatedHeights: true });\n if (this.isSimpleMode()) {\n this.computeApproximateSize();\n }\n this.virtualizer.instanceChanged(this, true);\n };\n\n public setVisibility(visible: boolean): void {\n if (this.isAdvancedMode() || this.fileContainer == null) {\n return;\n }\n this.renderRange = undefined;\n if (visible && !this.isVisible) {\n this.top = this.getVirtualizedTop();\n this.isVisible = true;\n } else if (!visible && this.isVisible) {\n this.isVisible = false;\n this.rerender();\n }\n }\n\n override rerender(): void {\n if (\n !this.enabled ||\n (this.fileDiff == null &&\n this.additionFile == null &&\n this.deletionFile == null)\n ) {\n return;\n }\n this.forceRenderOverride = true;\n this.virtualizer.instanceChanged(this, false);\n }\n\n // Compute the approximate size from the cached baseline estimate plus any\n // measured height deltas observed in rendered rows.\n // The reason we refer to this as `approximate size` is because heights my\n // dynamically change for a number of reasons so we can never be fully sure\n // if the height is 100% accurate\n private computeApproximateSize(force = false): void {\n const shouldValidateSize = this.isResizeDebuggingEnabled();\n if (!force && !this.layoutDirty && !shouldValidateSize) {\n return;\n }\n\n const isFirstCompute = this.height === 0;\n this.height = 0;\n this.cache.checkpoints = [];\n this.cache.totalLines = 0;\n if (this.fileDiff == null) {\n this.layoutDirty = false;\n return;\n }\n\n const { disableFileHeader = false, collapsed = false } = this.options;\n const headerRegion = getVirtualFileHeaderRegion(\n this.metrics,\n disableFileHeader\n );\n\n this.height += headerRegion;\n if (collapsed) {\n this.layoutDirty = false;\n return;\n }\n\n this.height =\n this.getActiveEstimatedHeight() + this.cache.measuredHeightDeltaTotal;\n\n if (shouldValidateSize && !isFirstCompute) {\n this.validateComputedHeight();\n }\n this.layoutDirty = false;\n }\n\n private getActiveEstimatedHeight(): number {\n this.ensureEstimatedDiffHeights();\n const estimatedHeight =\n this.getDiffStyle() === 'split'\n ? this.cache.estimatedSplitHeight\n : this.cache.estimatedUnifiedHeight;\n if (estimatedHeight == null) {\n throw new Error(\n 'VirtualizedFileDiff.getActiveEstimatedHeight: missing estimated height'\n );\n }\n return estimatedHeight;\n }\n\n private ensureEstimatedDiffHeights(): void {\n if (this.fileDiff == null) {\n this.cache.estimatedSplitHeight = undefined;\n this.cache.estimatedUnifiedHeight = undefined;\n return;\n }\n if (\n this.cache.estimatedSplitHeight != null &&\n this.cache.estimatedUnifiedHeight != null\n ) {\n return;\n }\n\n const {\n disableFileHeader = false,\n expandUnchanged = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n const { splitHeight, unifiedHeight } = computeEstimatedDiffHeights({\n fileDiff: this.fileDiff,\n metrics: this.metrics,\n disableFileHeader,\n hunkSeparators: this.getHunkSeparatorType(),\n expandUnchanged,\n expandedHunks: this.hunksRenderer.getExpandedHunksMap(),\n collapsedContextThreshold,\n });\n this.cache.estimatedSplitHeight = splitHeight;\n this.cache.estimatedUnifiedHeight = unifiedHeight;\n }\n\n private validateComputedHeight(): void {\n if (this.fileContainer == null || this.fileDiff == null) {\n return;\n }\n\n const rect = this.fileContainer.getBoundingClientRect();\n if (rect.height !== this.height) {\n console.log(\n 'VirtualizedFileDiff.computeApproximateSize: computed height doesnt match',\n {\n name: this.fileDiff.name,\n elementHeight: rect.height,\n computedHeight: this.height,\n }\n );\n } else {\n console.log(\n 'VirtualizedFileDiff.computeApproximateSize: computed height IS CORRECT'\n );\n }\n }\n\n override render({\n fileContainer,\n oldFile,\n newFile,\n fileDiff,\n forceRender = false,\n lineAnnotations,\n ...props\n }: FileDiffRenderProps<LAnnotation> = {}): boolean {\n const { forceRenderOverride, isSetup } = this;\n this.forceRenderOverride = undefined;\n const annotationsChanged = this.syncLineAnnotations(lineAnnotations);\n if (annotationsChanged) {\n this.resetLayoutCache({ includeEstimatedHeights: false });\n }\n\n this.fileDiff ??=\n fileDiff ??\n (oldFile != null && newFile != null\n ? // NOTE(amadeus): We might be forcing ourselves to double up the\n // computation of fileDiff (in the super.render() call), so we might want\n // to figure out a way to avoid that. That also could be just as simple as\n // passing through fileDiff though... so maybe we good?\n parseDiffFromFile(oldFile, newFile, this.options.parseDiffOptions)\n : undefined);\n\n fileContainer = this.getOrCreateFileContainer(fileContainer);\n\n if (this.fileDiff == null) {\n console.error(\n 'VirtualizedFileDiff.render: attempting to virtually render when we dont have the correct data'\n );\n return false;\n }\n\n if (!isSetup) {\n this.computeApproximateSize();\n const virtualizer = this.getSimpleVirtualizer();\n this.top ??= this.getVirtualizedTop();\n if (this.isAdvancedMode()) {\n this.isVisible = true;\n } else {\n if (virtualizer == null) {\n throw new Error(\n 'VirtualizedFileDiff.render: simple virtualizer is not available'\n );\n }\n virtualizer.connect(fileContainer, this);\n this.isVisible = virtualizer.isInstanceVisible(\n this.top ?? 0,\n this.height\n );\n }\n this.isSetup = true;\n } else {\n this.top ??= this.getVirtualizedTop();\n }\n\n if (!this.isVisible && this.isSimpleMode()) {\n return this.renderPlaceholder(this.height);\n }\n\n const windowSpecs = this.virtualizer.getWindowSpecs();\n const fileTop = this.top ?? 0;\n const renderRange = this.computeRenderRangeFromWindow(\n this.fileDiff,\n fileTop,\n windowSpecs\n );\n return super.render({\n fileDiff: this.fileDiff,\n fileContainer,\n renderRange,\n oldFile,\n newFile,\n lineAnnotations,\n forceRender: (forceRenderOverride ?? forceRender) || annotationsChanged,\n ...props,\n });\n }\n\n public syncVirtualizedTop(): void {\n this.top = this.getVirtualizedTop();\n }\n\n protected override shouldDisableVirtualizationBuffers(): boolean {\n return this.isAdvancedMode() || super.shouldDisableVirtualizationBuffers();\n }\n\n private isSimpleMode(): boolean {\n return this.virtualizer.type === 'simple';\n }\n\n private isAdvancedMode(): boolean {\n return this.virtualizer.type === 'advanced';\n }\n\n private getVirtualizedTop(): number | undefined {\n if (this.virtualizer.type === 'advanced') {\n return this.virtualizer.getLocalTopForInstance(this);\n }\n return this.fileContainer != null\n ? this.virtualizer.getOffsetInScrollContainer(this.fileContainer)\n : 0;\n }\n\n private getSimpleVirtualizer(): Virtualizer | undefined {\n return this.virtualizer.type === 'simple' ? this.virtualizer : undefined;\n }\n\n private isResizeDebuggingEnabled(): boolean {\n return this.getSimpleVirtualizer()?.config.resizeDebugging ?? false;\n }\n\n private getDiffStyle(): 'split' | 'unified' {\n return this.options.diffStyle ?? 'split';\n }\n\n private getHunkSeparatorType(): HunkSeparators {\n return getOptionHunkSeparatorType(this.options.hunkSeparators);\n }\n\n private approximateLayoutCheckpoints(): void {\n if (\n this.cache.checkpoints.length > 0 ||\n this.fileDiff == null ||\n this.fileDiff.hunks.length === 0 ||\n this.options.collapsed === true\n ) {\n return;\n }\n\n const {\n disableFileHeader = false,\n expandUnchanged = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n const finalHunkIndex = this.fileDiff.hunks.length - 1;\n const diffStyle = this.getDiffStyle();\n const hunkSeparators = this.getHunkSeparatorType();\n const expandedHunks = expandUnchanged\n ? true\n : this.hunksRenderer.getExpandedHunksMap();\n const heightDeltaPrefix = createHeightDeltaPrefix(this.cache.heightDeltas);\n let top =\n getVirtualFileHeaderRegion(this.metrics, disableFileHeader) +\n this.cache.fileAnnotationHeight;\n let renderedLineIndex = 0;\n\n const processRows = ({\n rowCount,\n startLineIndex,\n preSeparatorHeight = 0,\n postSeparatorHeight = 0,\n metadataOffsets = [],\n }: {\n rowCount: number;\n startLineIndex: number;\n preSeparatorHeight?: number;\n postSeparatorHeight?: number;\n metadataOffsets?: number[];\n }) => {\n if (rowCount <= 0) {\n return;\n }\n\n const blockStart = renderedLineIndex;\n const blockEnd = renderedLineIndex + rowCount;\n let nextCheckpoint = getNextCheckpointIndex(blockStart);\n while (nextCheckpoint < blockEnd) {\n const offset = nextCheckpoint - blockStart;\n const checkpointTop =\n top +\n (offset > 0 ? preSeparatorHeight : 0) +\n offset * this.metrics.lineHeight +\n countMetadataOffsetsBefore(metadataOffsets, offset) *\n this.metrics.lineHeight +\n sumHeightDeltas(\n heightDeltaPrefix,\n startLineIndex,\n startLineIndex + offset\n );\n this.cache.checkpoints.push({\n renderedLineIndex: nextCheckpoint,\n lineIndex: startLineIndex + offset,\n top: checkpointTop,\n });\n nextCheckpoint += LAYOUT_CHECKPOINT_INTERVAL;\n }\n\n top +=\n preSeparatorHeight +\n rowCount * this.metrics.lineHeight +\n metadataOffsets.length * this.metrics.lineHeight +\n sumHeightDeltas(\n heightDeltaPrefix,\n startLineIndex,\n startLineIndex + rowCount\n ) +\n postSeparatorHeight;\n renderedLineIndex = blockEnd;\n };\n\n for (\n let hunkIndex = 0;\n hunkIndex < this.fileDiff.hunks.length;\n hunkIndex++\n ) {\n const hunk = this.fileDiff.hunks[hunkIndex];\n if (hunk == null) {\n throw new Error(\n 'VirtualizedFileDiff.approximateLayoutCheckpoints: invalid hunk index'\n );\n }\n\n const leadingRegion = getExpandedRegion({\n isPartial: this.fileDiff.isPartial,\n rangeSize: hunk.collapsedBefore,\n expandedHunks,\n hunkIndex,\n collapsedContextThreshold,\n });\n const leadingSeparatorHeight =\n leadingRegion.collapsedLines > 0\n ? (getLeadingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n hunkIndex,\n hunkSpecs: hunk.hunkSpecs,\n })?.totalHeight ?? 0)\n : 0;\n\n processRows({\n rowCount: leadingRegion.fromStart,\n startLineIndex:\n (diffStyle === 'split'\n ? hunk.splitLineStart\n : hunk.unifiedLineStart) - leadingRegion.rangeSize,\n });\n\n let pendingLeadingSeparatorHeight = leadingSeparatorHeight;\n processRows({\n rowCount: leadingRegion.fromEnd,\n startLineIndex:\n (diffStyle === 'split'\n ? hunk.splitLineStart\n : hunk.unifiedLineStart) - leadingRegion.fromEnd,\n preSeparatorHeight: pendingLeadingSeparatorHeight,\n });\n if (leadingRegion.fromEnd > 0) {\n pendingLeadingSeparatorHeight = 0;\n }\n\n const trailingRegion =\n hunkIndex === finalHunkIndex\n ? getTrailingExpandedRegion({\n fileDiff: this.fileDiff,\n hunkIndex,\n expandedHunks,\n collapsedContextThreshold,\n errorPrefix: 'VirtualizedFileDiff',\n })\n : undefined;\n const trailingSeparatorHeight =\n trailingRegion != null && trailingRegion.collapsedLines > 0\n ? (getTrailingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n })?.totalHeight ?? 0)\n : 0;\n const trailingExpandedCount =\n trailingRegion != null\n ? trailingRegion.fromStart + trailingRegion.fromEnd\n : 0;\n\n const hunkBodyRowCount =\n diffStyle === 'split' ? hunk.splitLineCount : hunk.unifiedLineCount;\n const hunkBodyStartLineIndex =\n diffStyle === 'split' ? hunk.splitLineStart : hunk.unifiedLineStart;\n processRows({\n rowCount: hunkBodyRowCount,\n startLineIndex: hunkBodyStartLineIndex,\n preSeparatorHeight: pendingLeadingSeparatorHeight,\n postSeparatorHeight:\n trailingExpandedCount === 0 ? trailingSeparatorHeight : 0,\n metadataOffsets: getHunkMetadataOffsets({\n diffStyle,\n hunk,\n rowCount: hunkBodyRowCount,\n }),\n });\n\n if (trailingRegion != null && trailingExpandedCount > 0) {\n processRows({\n rowCount: trailingExpandedCount,\n startLineIndex: hunkBodyStartLineIndex + hunkBodyRowCount,\n postSeparatorHeight: trailingSeparatorHeight,\n });\n }\n }\n\n this.cache.totalLines = renderedLineIndex;\n }\n\n // Find the nearest sparse layout checkpoint at or before an active\n // diff-style line index. Diff checkpoints also store the dense rendered-row\n // index, so deep line-position lookups can resume iteration from that\n // rendered row and replay only the nearby layout work instead of walking\n // from the first hunk.\n private getLayoutCheckpointBeforeLineIndex(\n lineIndex: number\n ): DiffLayoutCheckpoint | undefined {\n if (lineIndex <= 0 || this.cache.checkpoints.length === 0) {\n return undefined;\n }\n\n let low = 0;\n let high = this.cache.checkpoints.length - 1;\n let result: DiffLayoutCheckpoint | undefined;\n\n while (low <= high) {\n const mid = (low + high) >> 1;\n const checkpoint = this.cache.checkpoints[mid];\n if (checkpoint == null) {\n throw new Error('VirtualizedFileDiff: invalid checkpoint index');\n }\n if (checkpoint.lineIndex <= lineIndex) {\n result = checkpoint;\n low = mid + 1;\n } else {\n high = mid - 1;\n }\n }\n\n return result;\n }\n\n // Find the nearest sparse layout checkpoint at or before a scroll offset.\n // Render-range scans start from this checkpoint so variable-height diffs\n // only replay nearby rows. When `hunkLineCount` is provided, step backward\n // to a rendered hunk boundary so buffer calculations can reuse absolute hunk\n // offsets safely.\n private getLayoutCheckpointBeforeTop(\n top: number,\n hunkLineCount?: number\n ): DiffLayoutCheckpoint | undefined {\n let low = 0;\n let high = this.cache.checkpoints.length - 1;\n let resultIndex = -1;\n\n while (low <= high) {\n const mid = (low + high) >> 1;\n const checkpoint = this.cache.checkpoints[mid];\n if (checkpoint == null) {\n throw new Error('VirtualizedFileDiff: invalid checkpoint index');\n }\n if (checkpoint.top <= top) {\n resultIndex = mid;\n low = mid + 1;\n } else {\n high = mid - 1;\n }\n }\n\n if (hunkLineCount == null) {\n return resultIndex >= 0 ? this.cache.checkpoints[resultIndex] : undefined;\n }\n\n for (let index = resultIndex; index >= 0; index--) {\n const checkpoint = this.cache.checkpoints[index];\n if (checkpoint == null) {\n throw new Error('VirtualizedFileDiff: invalid checkpoint index');\n }\n if (checkpoint.renderedLineIndex % hunkLineCount === 0) {\n return checkpoint;\n }\n }\n\n return undefined;\n }\n\n private getExpandedLineCount(\n fileDiff: FileDiffMetadata,\n diffStyle: 'split' | 'unified'\n ): number {\n let count = 0;\n if (fileDiff.isPartial) {\n for (const hunk of fileDiff.hunks) {\n count +=\n diffStyle === 'split' ? hunk.splitLineCount : hunk.unifiedLineCount;\n }\n return count;\n }\n\n const {\n expandUnchanged = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n const expandedHunks = expandUnchanged\n ? true\n : this.hunksRenderer.getExpandedHunksMap();\n\n for (const [hunkIndex, hunk] of fileDiff.hunks.entries()) {\n const hunkCount =\n diffStyle === 'split' ? hunk.splitLineCount : hunk.unifiedLineCount;\n count += hunkCount;\n const collapsedBefore = Math.max(hunk.collapsedBefore, 0);\n const { fromStart, fromEnd, renderAll } = getExpandedRegion({\n isPartial: fileDiff.isPartial,\n rangeSize: collapsedBefore,\n expandedHunks,\n hunkIndex,\n collapsedContextThreshold,\n });\n if (collapsedBefore > 0) {\n count += renderAll ? collapsedBefore : fromStart + fromEnd;\n }\n }\n\n const trailingRegion = getTrailingExpandedRegion({\n fileDiff,\n hunkIndex: fileDiff.hunks.length - 1,\n expandedHunks,\n collapsedContextThreshold,\n errorPrefix: 'VirtualizedFileDiff',\n });\n if (trailingRegion != null) {\n count += trailingRegion.fromStart + trailingRegion.fromEnd;\n }\n\n return count;\n }\n\n private computeRenderRangeFromWindow(\n fileDiff: FileDiffMetadata,\n fileTop: number,\n { top, bottom }: RenderWindow\n ): RenderRange {\n const {\n disableFileHeader = false,\n expandUnchanged = false,\n collapsedContextThreshold = DEFAULT_COLLAPSED_CONTEXT_THRESHOLD,\n } = this.options;\n const { hunkLineCount, lineHeight } = this.metrics;\n const diffStyle = this.getDiffStyle();\n const hunkSeparators = this.getHunkSeparatorType();\n const fileHeight = this.height;\n let lineCount =\n this.cache.totalLines > 0\n ? this.cache.totalLines\n : this.getExpandedLineCount(fileDiff, diffStyle);\n\n const headerRegion = getVirtualFileHeaderRegion(\n this.metrics,\n disableFileHeader\n );\n const paddingBottom =\n fileDiff.hunks.length > 0 ? getVirtualFilePaddingBottom(this.metrics) : 0;\n const { fileAnnotationHeight } = this.cache;\n const codeRegionTop = headerRegion + fileAnnotationHeight;\n const codeHeight = Math.max(\n 0,\n fileHeight - headerRegion - fileAnnotationHeight - paddingBottom\n );\n const hasFileAnnotations = this.hasFileAnnotations(fileDiff);\n const fileAnnotationTop = fileTop + headerRegion;\n const measuredFileAnnotationVisible =\n fileAnnotationHeight > 0 &&\n hasFileAnnotations &&\n fileAnnotationTop < bottom &&\n fileAnnotationTop + fileAnnotationHeight > top;\n\n // File is outside render window\n if (fileTop < top - fileHeight || fileTop > bottom) {\n return {\n startingLine: 0,\n totalLines: 0,\n bufferBefore: 0,\n bufferAfter: fileHeight - headerRegion - paddingBottom,\n };\n }\n\n // Whole file is under hunkLineCount, just render it all\n if (lineCount <= hunkLineCount || fileDiff.hunks.length === 0) {\n return {\n startingLine: 0,\n totalLines: hunkLineCount,\n bufferBefore: 0,\n bufferAfter: 0,\n };\n }\n\n this.approximateLayoutCheckpoints();\n lineCount = this.cache.totalLines > 0 ? this.cache.totalLines : lineCount;\n\n const estimatedTargetLines = Math.ceil(\n Math.max(bottom - top, 0) / lineHeight\n );\n const totalLines =\n Math.ceil(estimatedTargetLines / hunkLineCount) * hunkLineCount +\n hunkLineCount;\n const totalHunks = totalLines / hunkLineCount;\n const overflowHunks = totalHunks;\n const hunkOffsets: number[] = [];\n // Halfway between top & bottom, represented as an absolute position\n const viewportCenter = (top + bottom) / 2;\n // Start the scan before the viewport so we collect hunk offsets that may be\n // needed for bufferBefore. This only chooses the scan origin; the returned\n // render range is still computed from the visible window below.\n const checkpoint = this.getLayoutCheckpointBeforeTop(\n Math.max(0, top - fileTop - totalLines * lineHeight * 2),\n hunkLineCount\n );\n\n let absoluteLineTop = fileTop + (checkpoint?.top ?? codeRegionTop);\n let currentLine = checkpoint?.renderedLineIndex ?? 0;\n let firstVisibleHunk: number | undefined;\n let centerHunk: number | undefined;\n let overflowCounter: number | undefined;\n\n iterateOverDiff({\n diff: fileDiff,\n diffStyle,\n startingLine: checkpoint?.renderedLineIndex ?? 0,\n expandedHunks: expandUnchanged\n ? true\n : this.hunksRenderer.getExpandedHunksMap(),\n collapsedContextThreshold,\n callback: ({\n hunkIndex,\n hunk,\n collapsedBefore,\n collapsedAfter,\n deletionLine,\n additionLine,\n }) => {\n const splitLineIndex =\n additionLine != null\n ? additionLine.splitLineIndex\n : deletionLine.splitLineIndex;\n const unifiedLineIndex =\n additionLine != null\n ? additionLine.unifiedLineIndex\n : deletionLine.unifiedLineIndex;\n const hasMetadata =\n (additionLine?.noEOFCR ?? false) || (deletionLine?.noEOFCR ?? false);\n const leadingSeparator =\n collapsedBefore > 0\n ? getLeadingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n hunkIndex,\n hunkSpecs: hunk?.hunkSpecs,\n })\n : undefined;\n const gapAdjustment = leadingSeparator?.totalHeight ?? 0;\n\n absoluteLineTop += gapAdjustment;\n\n const isAtHunkBoundary = currentLine % hunkLineCount === 0;\n const currentHunk = Math.floor(currentLine / hunkLineCount);\n\n // Track the boundary positional offset at a hunk\n if (isAtHunkBoundary) {\n hunkOffsets[currentHunk] =\n absoluteLineTop - (fileTop + codeRegionTop + gapAdjustment);\n\n // Check if we should bail (overflow complete)\n if (overflowCounter != null) {\n if (overflowCounter <= 0) {\n return true;\n }\n overflowCounter--;\n }\n }\n\n const lineHeight = this.getLineHeight(\n diffStyle === 'split' ? splitLineIndex : unifiedLineIndex,\n hasMetadata\n );\n\n // Track visible region\n if (absoluteLineTop > top - lineHeight && absoluteLineTop < bottom) {\n firstVisibleHunk ??= currentHunk;\n }\n\n // Track which hunk contains the viewport center\n // If viewport center is above this line and we haven't set centerHunk yet,\n // this is the first line at or past the center\n if (\n centerHunk == null &&\n absoluteLineTop + lineHeight > viewportCenter\n ) {\n centerHunk = currentHunk;\n }\n\n // Start overflow when we are out of the viewport at a hunk boundary\n if (\n overflowCounter == null &&\n absoluteLineTop >= bottom &&\n isAtHunkBoundary\n ) {\n overflowCounter = overflowHunks;\n }\n\n currentLine++;\n absoluteLineTop += lineHeight;\n\n if (collapsedAfter > 0) {\n absoluteLineTop +=\n getTrailingHunkSeparatorLayout({\n type: hunkSeparators,\n metrics: this.metrics,\n })?.totalHeight ?? 0;\n }\n\n return false;\n },\n });\n\n // No visible lines found\n if (firstVisibleHunk == null) {\n if (measuredFileAnnotationVisible) {\n firstVisibleHunk = 0;\n centerHunk = 0;\n } else {\n return {\n startingLine: 0,\n totalLines: 0,\n bufferBefore: 0,\n bufferAfter: fileHeight - headerRegion - paddingBottom,\n };\n }\n }\n\n // Calculate balanced startingLine centered around the viewport center\n // Fall back to firstVisibleHunk if center wasn't found (e.g., center in a gap)\n centerHunk ??= firstVisibleHunk;\n const idealStartHunk = Math.round(centerHunk - totalHunks / 2);\n\n // Clamp startHunk: at the beginning, reduce totalLines; at the end, shift startHunk back\n const maxStartHunk = Math.max(\n 0,\n Math.ceil(lineCount / hunkLineCount) - totalHunks\n );\n const startHunk = Math.max(0, Math.min(idealStartHunk, maxStartHunk));\n const startingLine = startHunk * hunkLineCount;\n\n // If we wanted to start before 0, reduce totalLines by the clamped amount\n const clampedTotalLines =\n idealStartHunk < 0\n ? totalLines + idealStartHunk * hunkLineCount\n : totalLines;\n\n // Use hunkOffsets array for efficient buffer calculations\n const codeBufferBefore = hunkOffsets[startHunk] ?? 0;\n const bufferBefore =\n startingLine === 0 ? 0 : fileAnnotationHeight + codeBufferBefore;\n\n // Calculate bufferAfter using hunkOffset if available, otherwise use cumulative height\n const finalHunkIndex = startHunk + clampedTotalLines / hunkLineCount;\n const bufferAfter =\n finalHunkIndex < hunkOffsets.length\n ? codeHeight - hunkOffsets[finalHunkIndex]\n : // We stopped early, calculate from current position\n codeHeight - (absoluteLineTop - fileTop - codeRegionTop);\n\n return {\n startingLine,\n totalLines: clampedTotalLines,\n bufferBefore,\n bufferAfter: Math.max(0, bufferAfter),\n };\n }\n}\n\nfunction measureFileAnnotationHeight(\n codeGroups: (HTMLElement | undefined)[]\n): number | undefined {\n let height: number | undefined;\n for (const codeGroup of codeGroups) {\n if (codeGroup == null) continue;\n const content = codeGroup.children[1];\n if (!(content instanceof HTMLElement)) continue;\n for (const child of content.children) {\n if (!(child instanceof HTMLElement)) continue;\n if (child.dataset.lineAnnotation !== FILE_ANNOTATION_DOM_KEY) {\n continue;\n }\n height = Math.max(height ?? 0, child.getBoundingClientRect().height);\n }\n }\n return height;\n}\n\ninterface HeightDeltaPrefix {\n lineIndexes: number[];\n prefixTotals: number[];\n}\n\nfunction createHeightDeltaPrefix(\n heightDeltas: Map<number, number>\n): HeightDeltaPrefix {\n const entries = Array.from(heightDeltas).sort((a, b) => a[0] - b[0]);\n const lineIndexes: number[] = [];\n const prefixTotals = [0];\n let total = 0;\n for (const [lineIndex, delta] of entries) {\n lineIndexes.push(lineIndex);\n total += delta;\n prefixTotals.push(total);\n }\n return { lineIndexes, prefixTotals };\n}\n\nfunction sumHeightDeltas(\n { lineIndexes, prefixTotals }: HeightDeltaPrefix,\n startLineIndex: number,\n endLineIndex: number\n): number {\n if (startLineIndex >= endLineIndex || lineIndexes.length === 0) {\n return 0;\n }\n const start = lowerBound(lineIndexes, startLineIndex);\n const end = lowerBound(lineIndexes, endLineIndex);\n return (prefixTotals[end] ?? 0) - (prefixTotals[start] ?? 0);\n}\n\nfunction lowerBound(values: number[], target: number): number {\n let low = 0;\n let high = values.length;\n while (low < high) {\n const mid = (low + high) >> 1;\n const value = values[mid];\n if (value == null) {\n throw new Error('VirtualizedFileDiff: invalid prefix index');\n }\n if (value < target) {\n low = mid + 1;\n } else {\n high = mid;\n }\n }\n return low;\n}\n\nfunction getNextCheckpointIndex(renderedLineIndex: number): number {\n return (\n Math.ceil(renderedLineIndex / LAYOUT_CHECKPOINT_INTERVAL) *\n LAYOUT_CHECKPOINT_INTERVAL\n );\n}\n\nfunction countMetadataOffsetsBefore(\n metadataOffsets: number[],\n offset: number\n): number {\n let count = 0;\n for (const metadataOffset of metadataOffsets) {\n if (metadataOffset < offset) {\n count++;\n }\n }\n return count;\n}\n\nfunction getHunkMetadataOffsets({\n diffStyle,\n hunk,\n rowCount,\n}: {\n diffStyle: 'split' | 'unified';\n hunk: Hunk;\n rowCount: number;\n}): number[] {\n if (rowCount <= 0 || (!hunk.noEOFCRAdditions && !hunk.noEOFCRDeletions)) {\n return [];\n }\n\n const lastContent = hunk.hunkContent.at(-1);\n if (lastContent == null) {\n return [];\n }\n\n if (lastContent.type === 'context') {\n return [rowCount - 1];\n }\n\n const splitCount = Math.max(lastContent.deletions, lastContent.additions);\n const unifiedCount = lastContent.deletions + lastContent.additions;\n if (diffStyle === 'split') {\n return splitCount > 0 && (hunk.noEOFCRAdditions || hunk.noEOFCRDeletions)\n ? [rowCount - 1]\n : [];\n }\n\n const offsets: number[] = [];\n const contentStartOffset = rowCount - unifiedCount;\n if (lastContent.deletions > 0 && hunk.noEOFCRDeletions) {\n offsets.push(contentStartOffset + lastContent.deletions - 1);\n }\n if (lastContent.additions > 0 && hunk.noEOFCRAdditions) {\n offsets.push(rowCount - 1);\n }\n return offsets;\n}\n\nfunction hasDiffLayoutOptionChanged<LAnnotation>(\n previousOptions: FileDiffOptions<LAnnotation>,\n nextOptions: FileDiffOptions<LAnnotation>\n): boolean {\n return (\n (previousOptions.diffStyle ?? 'split') !==\n (nextOptions.diffStyle ?? 'split') ||\n (previousOptions.overflow ?? 'scroll') !==\n (nextOptions.overflow ?? 'scroll') ||\n (previousOptions.collapsed ?? false) !== (nextOptions.collapsed ?? false) ||\n (previousOptions.disableLineNumbers ?? false) !==\n (nextOptions.disableLineNumbers ?? false) ||\n (previousOptions.disableFileHeader ?? false) !==\n (nextOptions.disableFileHeader ?? false) ||\n (previousOptions.diffIndicators ?? 'bars') !==\n (nextOptions.diffIndicators ?? 'bars') ||\n (previousOptions.hunkSeparators ?? 'line-info') !==\n (nextOptions.hunkSeparators ?? 'line-info') ||\n (previousOptions.expandUnchanged ?? false) !==\n (nextOptions.expandUnchanged ?? false) ||\n (previousOptions.collapsedContextThreshold ??\n DEFAULT_COLLAPSED_CONTEXT_THRESHOLD) !==\n (nextOptions.collapsedContextThreshold ??\n DEFAULT_COLLAPSED_CONTEXT_THRESHOLD) ||\n previousOptions.unsafeCSS !== nextOptions.unsafeCSS\n );\n}\n\nfunction hasDiffEstimateOptionChanged<LAnnotation>(\n previousOptions: FileDiffOptions<LAnnotation>,\n nextOptions: FileDiffOptions<LAnnotation>\n): boolean {\n return (\n (previousOptions.disableFileHeader ?? false) !==\n (nextOptions.disableFileHeader ?? false) ||\n (previousOptions.hunkSeparators ?? 'line-info') !==\n (nextOptions.hunkSeparators ?? 'line-info') ||\n (previousOptions.expandUnchanged ?? false) !==\n (nextOptions.expandUnchanged ?? false) ||\n (previousOptions.collapsedContextThreshold ??\n DEFAULT_COLLAPSED_CONTEXT_THRESHOLD) !==\n (nextOptions.collapsedContextThreshold ??\n DEFAULT_COLLAPSED_CONTEXT_THRESHOLD)\n );\n}\n\nfunction getOptionHunkSeparatorType<LAnnotation>(\n hunkSeparators: FileDiffOptions<LAnnotation>['hunkSeparators'] | undefined\n): HunkSeparators {\n return typeof hunkSeparators === 'function'\n ? 'custom'\n : (hunkSeparators ?? 'line-info');\n}\n\n// Extracts the view-specific line index from the data-line-index attribute.\n// Format is \"unifiedIndex,splitIndex\"\nfunction parseLineIndex(\n lineIndexAttr: string,\n diffStyle: 'split' | 'unified'\n): number {\n const [unifiedIndex, splitIndex] = lineIndexAttr.split(',').map(Number);\n return diffStyle === 'split' ? splitIndex : unifiedIndex;\n}\n"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"VirtulizerDevelopment.d.ts","names":["CodeView","Virtualizer","_1","sideEffect"],"sources":["../../src/components/VirtulizerDevelopment.d.ts"],"sourcesContent":["import type { CodeView } from './CodeView';\nimport type { Virtualizer } from './Virtualizer';\n\n// FIXME(amadeus): REMOVE ME AFTER RELEASING VIRTUALIZATION\ndeclare global {\n interface Window {\n // oxlint-disable-next-line typescript/no-explicit-any\n __INSTANCE?: CodeView<any> | Virtualizer;\n __TOGGLE?: () => void;\n __LOG?: boolean;\n }\n}\n"],"mappings":";;;;;AACiD,QAAAE,MAAA,CAAA;EAAA,UAAA,MAAA,CAAA;;IAMLC,UAAAD,CAAA,EAA3BF,QAA2B,CAAA,GAAA,CAAA,GAAXC,WAAW;IAAA,QAAA,CAAA,EAAA,GAAA,GAAA,IAAA"}
1
+ {"version":3,"file":"VirtulizerDevelopment.d.ts","names":["CodeView","Virtualizer","_0","sideEffect"],"sources":["../../src/components/VirtulizerDevelopment.d.ts"],"sourcesContent":["import type { CodeView } from './CodeView';\nimport type { Virtualizer } from './Virtualizer';\n\n// FIXME(amadeus): REMOVE ME AFTER RELEASING VIRTUALIZATION\ndeclare global {\n interface Window {\n // oxlint-disable-next-line typescript/no-explicit-any\n __INSTANCE?: CodeView<any> | Virtualizer;\n __TOGGLE?: () => void;\n __LOG?: boolean;\n }\n}\n"],"mappings":";;;;;AACiD,QAAAE,MAAA,CAAA;EAAA,UAAA,MAAA,CAAA;;IAMLC,UAAAD,CAAA,EAA3BF,QAA2B,CAAA,GAAA,CAAA,GAAXC,WAAW;IAAA,QAAA,CAAA,EAAA,GAAA,GAAA,IAAA"}
@@ -1 +1 @@
1
- {"version":3,"file":"constants.d.ts","names":["CodeViewLayout","HunkExpansionRegion","RenderRange","SmoothScrollSettings","ThemesType","VirtualFileMetrics","DIFFS_TAG_NAME","DIFFS_DEVELOPMENT_BUILD","COMMIT_METADATA_SPLIT","RegExp","GIT_DIFF_FILE_BREAK_REGEX","UNIFIED_DIFF_FILE_BREAK_REGEX","FILE_CONTEXT_BLOB","HUNK_HEADER","SPLIT_WITH_NEWLINES","FILENAME_HEADER_REGEX","FILENAME_HEADER_REGEX_GIT","ALTERNATE_FILE_NAMES_GIT","INDEX_LINE_METADATA","MERGE_CONFLICT_START_MARKER_REGEX","MERGE_CONFLICT_BASE_MARKER_REGEX","MERGE_CONFLICT_SEPARATOR_MARKER_REGEX","MERGE_CONFLICT_END_MARKER_REGEX","HEADER_PREFIX_SLOT_ID","HEADER_METADATA_SLOT_ID","CUSTOM_HEADER_SLOT_ID","DEFAULT_THEMES","THEME_CSS_ATTRIBUTE","UNSAFE_CSS_ATTRIBUTE","CORE_CSS_ATTRIBUTE","DIFFS_SCROLLBAR_MEASURE_ATTRIBUTE","DIFFS_SCROLLBAR_GUTTER_MEASURED_PROPERTY","DEFAULT_COLLAPSED_CONTEXT_THRESHOLD","DEFAULT_TOKENIZE_MAX_LENGTH","DEFAULT_VIRTUAL_FILE_METRICS","DEFAULT_CODE_VIEW_FILE_METRICS","DEFAULT_CODE_VIEW_LAYOUT","DEFAULT_SMOOTH_SCROLL_SETTINGS","DEFAULT_EXPANDED_REGION","DEFAULT_RENDER_RANGE","EMPTY_RENDER_RANGE"],"sources":["../src/constants.d.ts"],"sourcesContent":["import type { CodeViewLayout, HunkExpansionRegion, RenderRange, SmoothScrollSettings, ThemesType, VirtualFileMetrics } from './types';\nexport declare const DIFFS_TAG_NAME: \"diffs-container\";\nexport declare const DIFFS_DEVELOPMENT_BUILD: boolean;\nexport declare const COMMIT_METADATA_SPLIT: RegExp;\nexport declare const GIT_DIFF_FILE_BREAK_REGEX: RegExp;\nexport declare const UNIFIED_DIFF_FILE_BREAK_REGEX: RegExp;\nexport declare const FILE_CONTEXT_BLOB: RegExp;\nexport declare const HUNK_HEADER: RegExp;\nexport declare const SPLIT_WITH_NEWLINES: RegExp;\nexport declare const FILENAME_HEADER_REGEX: RegExp;\nexport declare const FILENAME_HEADER_REGEX_GIT: RegExp;\nexport declare const ALTERNATE_FILE_NAMES_GIT: RegExp;\nexport declare const INDEX_LINE_METADATA: RegExp;\nexport declare const MERGE_CONFLICT_START_MARKER_REGEX: RegExp;\nexport declare const MERGE_CONFLICT_BASE_MARKER_REGEX: RegExp;\nexport declare const MERGE_CONFLICT_SEPARATOR_MARKER_REGEX: RegExp;\nexport declare const MERGE_CONFLICT_END_MARKER_REGEX: RegExp;\nexport declare const HEADER_PREFIX_SLOT_ID = \"header-prefix\";\nexport declare const HEADER_METADATA_SLOT_ID = \"header-metadata\";\nexport declare const CUSTOM_HEADER_SLOT_ID = \"header-custom\";\nexport declare const DEFAULT_THEMES: ThemesType;\nexport declare const THEME_CSS_ATTRIBUTE = \"data-theme-css\";\nexport declare const UNSAFE_CSS_ATTRIBUTE = \"data-unsafe-css\";\nexport declare const CORE_CSS_ATTRIBUTE = \"data-core-css\";\nexport declare const DIFFS_SCROLLBAR_MEASURE_ATTRIBUTE = \"data-diffs-scrollbar-measure\";\nexport declare const DIFFS_SCROLLBAR_GUTTER_MEASURED_PROPERTY = \"--diffs-scrollbar-gutter-measured\";\nexport declare const DEFAULT_COLLAPSED_CONTEXT_THRESHOLD = 1;\nexport declare const DEFAULT_TOKENIZE_MAX_LENGTH = 100000;\nexport declare const DEFAULT_VIRTUAL_FILE_METRICS: VirtualFileMetrics;\nexport declare const DEFAULT_CODE_VIEW_FILE_METRICS: VirtualFileMetrics;\nexport declare const DEFAULT_CODE_VIEW_LAYOUT: CodeViewLayout;\nexport declare const DEFAULT_SMOOTH_SCROLL_SETTINGS: SmoothScrollSettings;\nexport declare const DEFAULT_EXPANDED_REGION: HunkExpansionRegion;\nexport declare const DEFAULT_RENDER_RANGE: RenderRange;\nexport declare const EMPTY_RENDER_RANGE: RenderRange;\n//# sourceMappingURL=constants.d.ts.map"],"mappings":";;;cACqBM;cACAC;AADAD,cAEAE,qBAFiC,EAEVC,MAFU;AACjCF,cAEAG,yBAFgC,EAELD,MAFK;AAChCD,cAEAG,6BAF6B,EAEEF,MAFF;AAC7BC,cAEAE,iBAFiC,EAEdH,MAFQA;AAC3BE,cAEAE,WAFAF,EAEaF,MAFwB;AACrCG,cAEAE,mBAFmBL,EAEEA,MAFI;AACzBI,cAEAE,qBAFmB,EAEIN,MAFJ;AACnBK,cAEAE,yBAF2B,EAEAP,MAFA;AAC3BM,cAEAE,wBAFuBR,EAEGA,MAFG;AAC7BO,cAEAE,mBAFiC,EAEZT,MAFMA;AAC3BQ,cAEAE,iCAFgC,EAEGV,MAFH;AAChCS,cAEAE,gCAF2B,EAEOX,MAFP;AAC3BU,cAEAE,qCAFmCZ,EAEIA,MAFE;AACzCW,cAEAE,+BAFkCb,EAEDA,MAFO;AACxCY,cAEAE,qBAAAA,GAF6C,eAANd;AACvCa,cAEAE,uBAAAA,GAFuC,iBAAA;AACvCD,cAEAE,qBAAAA,GAFqB,eAAA;AACrBD,cAEAE,cAFuB,EAEPtB,UAFO;AACvBqB,cAEAE,mBAAAA,GAFqB,gBAAA;AACrBD,cAEAE,oBAAAA,GAFgBxB,iBAAU;AAC1BuB,cAEAE,kBAAAA,GAFmB,eAAA;AACnBD,cAEAE,iCAAAA,GAFoB,8BAAA;AACpBD,cAEAE,wCAAAA,GAFkB,mCAAA;AAClBD,cAEAE,mCAAAA,GAFiC,CAAA;AACjCD,cAEAE,2BAAAA,GAFwC,MAAA;AACxCD,cAEAE,4BAFmC,EAEL7B,kBAFK;AACnC4B,cAEAE,8BAF2B,EAEK9B,kBAFL;AAC3B6B,cAEAE,wBAFgD,EAEtBpC,cAFIK;AAC9B8B,cAEAE,8BAFgChC,EAEAF,oBAFkB;AAClDiC,cAEAE,uBAF0BtC,EAEDC,mBAFe;AACxCoC,cAEAE,oBAFoD,EAE9BrC,WAFUC;AAChCmC,cAEAE,kBAF4C,EAExBtC,WAFKD"}
1
+ {"version":3,"file":"constants.d.ts","names":["CodeViewLayout","HunkExpansionRegion","RenderRange","SmoothScrollSettings","ThemesType","VirtualFileMetrics","DIFFS_TAG_NAME","DIFFS_DEVELOPMENT_BUILD","COMMIT_METADATA_SPLIT","RegExp","GIT_DIFF_FILE_BREAK_REGEX","UNIFIED_DIFF_FILE_BREAK_REGEX","FILE_CONTEXT_BLOB","HUNK_HEADER","SPLIT_WITH_NEWLINES","FILENAME_HEADER_REGEX","FILENAME_HEADER_REGEX_GIT","ALTERNATE_FILE_NAMES_GIT","INDEX_LINE_METADATA","MERGE_CONFLICT_START_MARKER_REGEX","MERGE_CONFLICT_BASE_MARKER_REGEX","MERGE_CONFLICT_SEPARATOR_MARKER_REGEX","MERGE_CONFLICT_END_MARKER_REGEX","HEADER_PREFIX_SLOT_ID","HEADER_METADATA_SLOT_ID","CUSTOM_HEADER_SLOT_ID","DEFAULT_THEMES","THEME_CSS_ATTRIBUTE","UNSAFE_CSS_ATTRIBUTE","CORE_CSS_ATTRIBUTE","DIFFS_SCROLLBAR_MEASURE_ATTRIBUTE","DIFFS_SCROLLBAR_GUTTER_MEASURED_PROPERTY","DEFAULT_COLLAPSED_CONTEXT_THRESHOLD","DEFAULT_TOKENIZE_MAX_LENGTH","DEFAULT_VIRTUAL_FILE_METRICS","DEFAULT_CODE_VIEW_FILE_METRICS","DEFAULT_CODE_VIEW_LAYOUT","DEFAULT_SMOOTH_SCROLL_SETTINGS","DEFAULT_EXPANDED_REGION","DEFAULT_RENDER_RANGE","EMPTY_RENDER_RANGE"],"sources":["../src/constants.d.ts"],"sourcesContent":["import type { CodeViewLayout, HunkExpansionRegion, RenderRange, SmoothScrollSettings, ThemesType, VirtualFileMetrics } from './types';\nexport declare const DIFFS_TAG_NAME: \"diffs-container\";\nexport declare const DIFFS_DEVELOPMENT_BUILD: boolean;\nexport declare const COMMIT_METADATA_SPLIT: RegExp;\nexport declare const GIT_DIFF_FILE_BREAK_REGEX: RegExp;\nexport declare const UNIFIED_DIFF_FILE_BREAK_REGEX: RegExp;\nexport declare const FILE_CONTEXT_BLOB: RegExp;\nexport declare const HUNK_HEADER: RegExp;\nexport declare const SPLIT_WITH_NEWLINES: RegExp;\nexport declare const FILENAME_HEADER_REGEX: RegExp;\nexport declare const FILENAME_HEADER_REGEX_GIT: RegExp;\nexport declare const ALTERNATE_FILE_NAMES_GIT: RegExp;\nexport declare const INDEX_LINE_METADATA: RegExp;\nexport declare const MERGE_CONFLICT_START_MARKER_REGEX: RegExp;\nexport declare const MERGE_CONFLICT_BASE_MARKER_REGEX: RegExp;\nexport declare const MERGE_CONFLICT_SEPARATOR_MARKER_REGEX: RegExp;\nexport declare const MERGE_CONFLICT_END_MARKER_REGEX: RegExp;\nexport declare const HEADER_PREFIX_SLOT_ID = \"header-prefix\";\nexport declare const HEADER_METADATA_SLOT_ID = \"header-metadata\";\nexport declare const CUSTOM_HEADER_SLOT_ID = \"header-custom\";\nexport declare const DEFAULT_THEMES: ThemesType;\nexport declare const THEME_CSS_ATTRIBUTE = \"data-theme-css\";\nexport declare const UNSAFE_CSS_ATTRIBUTE = \"data-unsafe-css\";\nexport declare const CORE_CSS_ATTRIBUTE = \"data-core-css\";\nexport declare const DIFFS_SCROLLBAR_MEASURE_ATTRIBUTE = \"data-diffs-scrollbar-measure\";\nexport declare const DIFFS_SCROLLBAR_GUTTER_MEASURED_PROPERTY = \"--diffs-scrollbar-gutter-measured\";\nexport declare const DEFAULT_COLLAPSED_CONTEXT_THRESHOLD = 1;\nexport declare const DEFAULT_TOKENIZE_MAX_LENGTH = 100000;\nexport declare const DEFAULT_VIRTUAL_FILE_METRICS: VirtualFileMetrics;\nexport declare const DEFAULT_CODE_VIEW_FILE_METRICS: VirtualFileMetrics;\nexport declare const DEFAULT_CODE_VIEW_LAYOUT: CodeViewLayout;\nexport declare const DEFAULT_SMOOTH_SCROLL_SETTINGS: SmoothScrollSettings;\nexport declare const DEFAULT_EXPANDED_REGION: HunkExpansionRegion;\nexport declare const DEFAULT_RENDER_RANGE: RenderRange;\nexport declare const EMPTY_RENDER_RANGE: RenderRange;\n//# sourceMappingURL=constants.d.ts.map"],"mappings":";;;cACqBM;cACAC;AADAD,cAEAE,qBAFiC,EAEVC,MAFU;AACjCF,cAEAG,yBAFgC,EAELD,MAFK;AAChCD,cAEAG,6BAF6B,EAEEF,MAFF;AAC7BC,cAEAE,iBAFiC,EAEdH,MAFQA;AAC3BE,cAEAE,WAFAF,EAEaF,MAFwB;AACrCG,cAEAE,mBAFmBL,EAEEA,MAFI;AACzBI,cAEAE,qBAFmB,EAEIN,MAFJ;AACnBK,cAEAE,yBAF2B,EAEAP,MAFA;AAC3BM,cAEAE,wBAFuBR,EAEGA,MAFG;AAC7BO,cAEAE,mBAFiC,EAEZT,MAFMA;AAC3BQ,cAEAE,iCAFgC,EAEGV,MAFH;AAChCS,cAEAE,gCAF2B,EAEOX,MAFP;AAC3BU,cAEAE,qCAFmCZ,EAEIA,MAFE;AACzCW,cAEAE,+BAFkCb,EAEDA,MAFO;AACxCY,cAEAE,qBAAAA,GAF6C,eAANd;AACvCa,cAEAE,uBAAAA,GAFuC,iBAAA;AACvCD,cAEAE,qBAAAA,GAFqB,eAAA;AACrBD,cAEAE,cAFuB,EAEPtB,UAFO;AACvBqB,cAEAE,mBAAAA,GAFqB,gBAAA;AACrBD,cAEAE,oBAAAA,GAF0B,iBAAA;AAC1BD,cAEAE,kBAAAA,GAFmB,eAAA;AACnBD,cAEAE,iCAAAA,GAFoB,8BAAA;AACpBD,cAEAE,wCAAAA,GAFkB,mCAAA;AAClBD,cAEAE,mCAAAA,GAFiC,CAAA;AACjCD,cAEAE,2BAAAA,GAFwC,MAAA;AACxCD,cAEAE,4BAFmC,EAEL7B,kBAFK;AACnC4B,cAEAE,8BAF2B,EAEK9B,kBAFL;AAC3B6B,cAEAE,wBAFgD,EAEtBpC,cAFIK;AAC9B8B,cAEAE,8BAFgChC,EAEAF,oBAFkB;AAClDiC,cAEAE,uBAF0BtC,EAEDC,mBAFe;AACxCoC,cAEAE,oBAFoD,EAE9BrC,WAFUC;AAChCmC,cAEAE,kBAF4C,EAExBtC,WAFKD"}
@@ -1,5 +1,8 @@
1
+ import { ThemeRegistrationResolved } from "../../types.js";
2
+ import * as _pierre_theming0 from "@pierre/theming";
3
+
1
4
  //#region src/highlighter/themes/themeResolver.d.ts
2
- declare const themeResolver: any;
5
+ declare const themeResolver: _pierre_theming0.ThemeResolver<ThemeRegistrationResolved>;
3
6
  //#endregion
4
7
  export { themeResolver };
5
8
  //# sourceMappingURL=themeResolver.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"themeResolver.d.ts","names":["themeResolver"],"sources":["../../../src/highlighter/themes/themeResolver.d.ts"],"sourcesContent":["export declare const themeResolver: any;\n//# sourceMappingURL=themeResolver.d.ts.map"],"mappings":";cAAqBA"}
1
+ {"version":3,"file":"themeResolver.d.ts","names":["ThemeRegistrationResolved","themeResolver","_pierre_theming0","ThemeResolver"],"sources":["../../../src/highlighter/themes/themeResolver.d.ts"],"sourcesContent":["import type { ThemeRegistrationResolved } from '../../types';\nexport declare const themeResolver: import(\"@pierre/theming\").ThemeResolver<ThemeRegistrationResolved>;\n//# sourceMappingURL=themeResolver.d.ts.map"],"mappings":";;;;cACqBC,eAAgFC,gBAAAA,CAAvCC,cAAcH"}
@@ -1073,7 +1073,7 @@ function selectionPointFromAnnotationSlotName(slotName) {
1073
1073
  const match = /^annotation-(?:(additions|deletions)-)?(\d+)$/.exec(slotName);
1074
1074
  if (match == null) return;
1075
1075
  const lineNumber = Number.parseInt(match[2], 10);
1076
- if (!Number.isFinite(lineNumber)) return;
1076
+ if (!Number.isFinite(lineNumber) || lineNumber <= 0) return;
1077
1077
  return {
1078
1078
  lineNumber,
1079
1079
  side: match[1]