From 54888932eb4f2cbbb2ce8667ac96abe77baed956 Mon Sep 17 00:00:00 2001 From: YuanboXue-Amber Date: Mon, 3 Jun 2024 18:37:27 +0200 Subject: [PATCH 1/4] add fix to globals that are problematic when using multiple windows --- packages/virtual-core/src/index.ts | 67 +++++++++++++++++++++--------- packages/virtual-core/src/utils.ts | 12 ++++-- 2 files changed, 56 insertions(+), 23 deletions(-) diff --git a/packages/virtual-core/src/index.ts b/packages/virtual-core/src/index.ts index a8012a19..5db9823e 100644 --- a/packages/virtual-core/src/index.ts +++ b/packages/virtual-core/src/index.ts @@ -67,6 +67,10 @@ export const observeElementRect = ( if (!element) { return } + const targetWindow = element.ownerDocument.defaultView + if (!targetWindow) { + return + } const handler = (rect: Rect) => { const { width, height } = rect @@ -79,7 +83,7 @@ export const observeElementRect = ( return () => {} } - const observer = new ResizeObserver((entries) => { + const observer = new targetWindow.ResizeObserver((entries) => { const entry = entries[0] if (entry?.borderBoxSize) { const box = entry.borderBoxSize[0] @@ -134,13 +138,21 @@ export const observeElementOffset = ( if (!element) { return } + const targetWindow = element.ownerDocument.defaultView + if (!targetWindow) { + return + } let offset = 0 const fallback = supportsScrollend ? () => undefined - : debounce(() => { - cb(offset, false) - }, instance.options.isScrollingResetDelay) + : debounce( + targetWindow, + () => { + cb(offset, false) + }, + instance.options.isScrollingResetDelay, + ) const createHandler = (isScrolling: boolean) => () => { offset = element[instance.options.horizontal ? 'scrollLeft' : 'scrollTop'] @@ -168,13 +180,21 @@ export const observeWindowOffset = ( if (!element) { return } + const targetWindow = element.window + if (!targetWindow) { + return + } let offset = 0 const fallback = supportsScrollend ? () => undefined - : debounce(() => { - cb(offset, false) - }, instance.options.isScrollingResetDelay) + : debounce( + targetWindow, + () => { + cb(offset, false) + }, + instance.options.isScrollingResetDelay, + ) const createHandler = (isScrolling: boolean) => () => { offset = element[instance.options.horizontal ? 'scrollX' : 'scrollY'] @@ -307,8 +327,9 @@ export class Virtualizer< private unsubs: (void | (() => void))[] = [] options!: Required> scrollElement: TScrollElement | null = null + private targetWindow: (Window & typeof globalThis) | null = null isScrolling: boolean = false - private scrollToIndexTimeoutId: ReturnType | null = null + private scrollToIndexTimeoutId: number | null = null measurementsCache: VirtualItem[] = [] private itemSizeCache = new Map() private pendingMeasuredCacheIndexes: number[] = [] @@ -330,15 +351,17 @@ export class Virtualizer< const get = () => { if (_ro) { return _ro - } else if (typeof ResizeObserver !== 'undefined') { - return (_ro = new ResizeObserver((entries) => { - entries.forEach((entry) => { - this._measureElement(entry.target as TItemElement, entry) - }) - })) - } else { + } + + if (typeof ResizeObserver === 'undefined' || !this.targetWindow) { return null } + + return (_ro = new this.targetWindow.ResizeObserver((entries) => { + entries.forEach((entry) => { + this._measureElement(entry.target as TItemElement, entry) + }) + })) } return { @@ -432,6 +455,12 @@ export class Virtualizer< this.scrollElement = scrollElement + if (this.scrollElement && 'ownerDocument' in this.scrollElement) { + this.targetWindow = this.scrollElement.ownerDocument.defaultView + } else { + this.targetWindow = this.scrollElement?.window ?? null + } + this._scrollToOffset(this.scrollOffset, { adjustments: undefined, behavior: undefined, @@ -807,8 +836,8 @@ export class Virtualizer< private isDynamicMode = () => this.measureElementCache.size > 0 private cancelScrollToIndex = () => { - if (this.scrollToIndexTimeoutId !== null) { - clearTimeout(this.scrollToIndexTimeoutId) + if (this.scrollToIndexTimeoutId !== null && this.targetWindow) { + this.targetWindow.clearTimeout(this.scrollToIndexTimeoutId) this.scrollToIndexTimeoutId = null } } @@ -849,8 +878,8 @@ export class Virtualizer< this._scrollToOffset(toOffset, { adjustments: undefined, behavior }) - if (behavior !== 'smooth' && this.isDynamicMode()) { - this.scrollToIndexTimeoutId = setTimeout(() => { + if (behavior !== 'smooth' && this.isDynamicMode() && this.targetWindow) { + this.scrollToIndexTimeoutId = this.targetWindow.setTimeout(() => { this.scrollToIndexTimeoutId = null const elementInDOM = this.measureElementCache.has( diff --git a/packages/virtual-core/src/utils.ts b/packages/virtual-core/src/utils.ts index f48116dd..41ee42a6 100644 --- a/packages/virtual-core/src/utils.ts +++ b/packages/virtual-core/src/utils.ts @@ -78,10 +78,14 @@ export function notUndefined(value: T | undefined, msg?: string): T { export const approxEqual = (a: number, b: number) => Math.abs(a - b) < 1 -export const debounce = (fn: Function, ms: number) => { - let timeoutId: ReturnType +export const debounce = ( + targetWindow: Window & typeof globalThis, + fn: Function, + ms: number, +) => { + let timeoutId: number return function (this: any, ...args: any[]) { - clearTimeout(timeoutId) - timeoutId = setTimeout(() => fn.apply(this, args), ms) + targetWindow.clearTimeout(timeoutId) + timeoutId = targetWindow.setTimeout(() => fn.apply(this, args), ms) } } From 31ac3e45d4104e11c7d24b00fbadd419909d5d98 Mon Sep 17 00:00:00 2001 From: YuanboXue-Amber Date: Tue, 4 Jun 2024 14:46:39 +0200 Subject: [PATCH 2/4] use instance.targetWindow --- packages/virtual-core/src/index.ts | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/packages/virtual-core/src/index.ts b/packages/virtual-core/src/index.ts index 5db9823e..169b3791 100644 --- a/packages/virtual-core/src/index.ts +++ b/packages/virtual-core/src/index.ts @@ -67,7 +67,7 @@ export const observeElementRect = ( if (!element) { return } - const targetWindow = element.ownerDocument.defaultView + const targetWindow = instance.targetWindow if (!targetWindow) { return } @@ -138,7 +138,7 @@ export const observeElementOffset = ( if (!element) { return } - const targetWindow = element.ownerDocument.defaultView + const targetWindow = instance.targetWindow if (!targetWindow) { return } @@ -180,7 +180,7 @@ export const observeWindowOffset = ( if (!element) { return } - const targetWindow = element.window + const targetWindow = instance.targetWindow if (!targetWindow) { return } @@ -327,7 +327,7 @@ export class Virtualizer< private unsubs: (void | (() => void))[] = [] options!: Required> scrollElement: TScrollElement | null = null - private targetWindow: (Window & typeof globalThis) | null = null + targetWindow: (Window & typeof globalThis) | null = null isScrolling: boolean = false private scrollToIndexTimeoutId: number | null = null measurementsCache: VirtualItem[] = [] From 4d2088f5f309954825d2760c8c731fd9a197f733 Mon Sep 17 00:00:00 2001 From: YuanboXue-Amber Date: Tue, 4 Jun 2024 16:06:52 +0200 Subject: [PATCH 3/4] apply code review --- packages/virtual-core/src/index.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/virtual-core/src/index.ts b/packages/virtual-core/src/index.ts index 169b3791..ae9cca58 100644 --- a/packages/virtual-core/src/index.ts +++ b/packages/virtual-core/src/index.ts @@ -353,7 +353,7 @@ export class Virtualizer< return _ro } - if (typeof ResizeObserver === 'undefined' || !this.targetWindow) { + if (!this.targetWindow || !this.targetWindow.ResizeObserver) { return null } From 8a60b2632b7f94937a5890589c26a892d28e1ee2 Mon Sep 17 00:00:00 2001 From: YuanboXue-Amber Date: Wed, 5 Jun 2024 10:27:09 +0200 Subject: [PATCH 4/4] apply suggestions --- packages/virtual-core/src/index.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/virtual-core/src/index.ts b/packages/virtual-core/src/index.ts index ae9cca58..ddddf9b0 100644 --- a/packages/virtual-core/src/index.ts +++ b/packages/virtual-core/src/index.ts @@ -79,7 +79,7 @@ export const observeElementRect = ( handler(element.getBoundingClientRect()) - if (typeof ResizeObserver === 'undefined') { + if (!targetWindow.ResizeObserver) { return () => {} }