From 4af98bf0a422292717cd1dfe3e84b31ee845ba05 Mon Sep 17 00:00:00 2001 From: Nick Gerleman Date: Thu, 3 Aug 2023 16:59:07 -0700 Subject: [PATCH] Right align scrollToIndex in RTL (#38737) Summary: Pull Request resolved: https://github.com/facebook/react-native/pull/38737 This fixes up behavior on Android so that `scrollToIndex` aligns the right edge of the cell of the given index to the right edge of the scrollview viewport. We do not incorporate RTL on iOS which inverts x/y coordinates from scroller (but not layout). Changelog: [General][Fixed] - Right align scrollToIndex in RTL Reviewed By: lenaic Differential Revision: D47978637 fbshipit-source-id: 638de1ab47bfa034190789d8564fafec7a17630c --- .../Lists/VirtualizedList.js | 70 +++++++++---------- 1 file changed, 35 insertions(+), 35 deletions(-) diff --git a/packages/virtualized-lists/Lists/VirtualizedList.js b/packages/virtualized-lists/Lists/VirtualizedList.js index 350e500e1e0d01..f9d3a5aef7792c 100644 --- a/packages/virtualized-lists/Lists/VirtualizedList.js +++ b/packages/virtualized-lists/Lists/VirtualizedList.js @@ -276,10 +276,23 @@ class VirtualizedList extends StateSafePureComponent { scrollRef.scrollTo({ animated, - ...this._cartesianScrollOffset(offset), + ...this._scrollToParamsFromOffset(offset), }); } + _scrollToParamsFromOffset(offset: number): {x?: number, y?: number} { + const {horizontal, rtl} = this._orientation(); + if (horizontal && rtl) { + // Add the visible length of the scrollview so that the offset is right-aligned + const cartOffset = this._listMetrics.cartesianOffset( + offset + this._scrollMetrics.visibleLength, + ); + return horizontal ? {x: cartOffset} : {y: cartOffset}; + } else { + return horizontal ? {x: offset} : {y: offset}; + } + } + recordInteraction() { this._nestedChildLists.forEach(childList => { childList.recordInteraction(); @@ -1480,39 +1493,6 @@ class VirtualizedList extends StateSafePureComponent { : metrics.width; } - _flowRelativeScrollOffset( - metrics: $ReadOnly<{ - x: number, - y: number, - ... - }>, - contentSize: $ReadOnly<{ - width: number, - height: number, - ... - }>, - ): number { - let offset = this._selectOffset(metrics); - - const {horizontal, rtl} = this._orientation(); - if (horizontal && rtl && Platform.OS !== 'ios') { - offset = this._selectLength(contentSize) - offset; - } - - return offset; - } - - _cartesianScrollOffset(offset: number): {x?: number, y?: number} { - const {horizontal, rtl} = this._orientation(); - const normalizedOffset = - horizontal && rtl && Platform.OS !== 'ios' - ? this._listMetrics.getContentLength() - offset - : offset; - - const cartOffset = this._listMetrics.cartesianOffset(normalizedOffset); - return horizontal ? {x: cartOffset} : {y: cartOffset}; - } - _selectOffset({x, y}: $ReadOnly<{x: number, y: number, ...}>): number { return this._orientation().horizontal ? x : y; } @@ -1688,7 +1668,7 @@ class VirtualizedList extends StateSafePureComponent { const timestamp = e.timeStamp; let visibleLength = this._selectLength(e.nativeEvent.layoutMeasurement); let contentLength = this._selectLength(e.nativeEvent.contentSize); - let offset = this._flowRelativeScrollOffset( + let offset = this._offsetFromScrollEvent( e.nativeEvent.contentOffset, e.nativeEvent.contentSize, ); @@ -1755,6 +1735,26 @@ class VirtualizedList extends StateSafePureComponent { this._scheduleCellsToRenderUpdate(); }; + _offsetFromScrollEvent( + contentOffset: $ReadOnly<{ + x: number, + y: number, + ... + }>, + contentSize: $ReadOnly<{ + width: number, + height: number, + ... + }>, + ): number { + const {horizontal, rtl} = this._orientation(); + if (Platform.OS === 'ios' || !(horizontal && rtl)) { + return this._selectOffset(contentOffset); + } + + return this._selectLength(contentSize) - this._selectOffset(contentOffset); + } + _scheduleCellsToRenderUpdate(opts?: {allowImmediateExecution?: boolean}) { const allowImmediateExecution = opts?.allowImmediateExecution ?? true;