From a4a7af758d3531937f537fc53ad1bd00f5215355 Mon Sep 17 00:00:00 2001 From: Ruslan Lesiutin Date: Wed, 21 Feb 2024 14:14:34 +0000 Subject: [PATCH] fix: use ResizeObserver global from parentNode realm to support case with multiple realms --- src/AutoSizer.ts | 55 ++++++++++++++++++++++++------------------------ 1 file changed, 27 insertions(+), 28 deletions(-) diff --git a/src/AutoSizer.ts b/src/AutoSizer.ts index fbb539a..78cfcf2 100644 --- a/src/AutoSizer.ts +++ b/src/AutoSizer.ts @@ -29,41 +29,40 @@ export class AutoSizer extends Component { componentDidMount() { const { nonce } = this.props; + const parentNode = this._autoSizer ? this._autoSizer.parentNode : null; if ( - this._autoSizer && - this._autoSizer.parentNode && - this._autoSizer.parentNode.ownerDocument && - this._autoSizer.parentNode.ownerDocument.defaultView && - this._autoSizer.parentNode instanceof - this._autoSizer.parentNode.ownerDocument.defaultView.HTMLElement + parentNode != null && + parentNode.ownerDocument && + parentNode.ownerDocument.defaultView && + parentNode instanceof parentNode.ownerDocument.defaultView.HTMLElement ) { // Delay access of parentNode until mount. // This handles edge-cases where the component has already been unmounted before its ref has been set, // As well as libraries like react-lite which have a slightly different lifecycle. - this._parentNode = this._autoSizer.parentNode; - - // Defer requiring resize handler in order to support server-side rendering. - // See issue #41 - if (this._parentNode != null) { - if (typeof ResizeObserver !== "undefined") { - this._resizeObserver = new ResizeObserver(() => { - // Guard against "ResizeObserver loop limit exceeded" error; - // could be triggered if the state update causes the ResizeObserver handler to run long. - // See https://github.com/bvaughn/react-virtualized-auto-sizer/issues/55 - this._timeoutId = setTimeout(this._onResize, 0); - }); - this._resizeObserver.observe(this._parentNode); - } else { - this._detectElementResize = createDetectElementResize(nonce); - this._detectElementResize.addResizeListener( - this._parentNode, - this._onResize - ); - } - - this._onResize(); + this._parentNode = parentNode; + + // Use ResizeObserver from the same context where parentNode (which we will observe) was defined + // Using just global can result into onResize events not being emitted in cases with multiple realms + const ResizeObserverInstance = + parentNode.ownerDocument.defaultView.ResizeObserver; + + if (ResizeObserverInstance != null) { + this._resizeObserver = new ResizeObserverInstance(() => { + // Guard against "ResizeObserver loop limit exceeded" error; + // could be triggered if the state update causes the ResizeObserver handler to run long. + // See https://github.com/bvaughn/react-virtualized-auto-sizer/issues/55 + this._timeoutId = setTimeout(this._onResize, 0); + }); + this._resizeObserver.observe(parentNode); + } else { + // Defer requiring resize handler in order to support server-side rendering. + // See issue #41 + this._detectElementResize = createDetectElementResize(nonce); + this._detectElementResize.addResizeListener(parentNode, this._onResize); } + + this._onResize(); } }