diff --git a/src/user-event/event-builder/scroll-view.ts b/src/user-event/event-builder/scroll-view.ts
index b322ec099..6a9d8e5de 100644
--- a/src/user-event/event-builder/scroll-view.ts
+++ b/src/user-event/event-builder/scroll-view.ts
@@ -1,9 +1,3 @@
-/**
- * Experimental values:
- * - iOS: `{"contentInset": {"bottom": 0, "left": 0, "right": 0, "top": 0}, "contentOffset": {"x": 0, "y": 5.333333333333333}, "contentSize": {"height": 1676.6666259765625, "width": 390}, "layoutMeasurement": {"height": 753, "width": 390}, "zoomScale": 1}`
- * - Android: `{"contentInset": {"bottom": 0, "left": 0, "right": 0, "top": 0}, "contentOffset": {"x": 0, "y": 31.619047164916992}, "contentSize": {"height": 1624.761962890625, "width": 411.4285583496094}, "layoutMeasurement": {"height": 785.5238037109375, "width": 411.4285583496094}, "responderIgnoreScroll": true, "target": 139, "velocity": {"x": -1.3633992671966553, "y": -1.3633992671966553}}`
- */
-
/**
* Scroll position of a scrollable element.
*/
@@ -12,16 +6,41 @@ export interface ContentOffset {
x: number;
}
+/**
+ * Other options for constructing a scroll event.
+ */
+export type ScrollEventOptions = {
+ contentSize?: {
+ height: number;
+ width: number;
+ };
+ layoutMeasurement?: {
+ height: number;
+ width: number;
+ };
+};
+
+/**
+ * Experimental values:
+ * - iOS: `{"contentInset": {"bottom": 0, "left": 0, "right": 0, "top": 0}, "contentOffset": {"x": 0, "y": 5.333333333333333}, "contentSize": {"height": 1676.6666259765625, "width": 390}, "layoutMeasurement": {"height": 753, "width": 390}, "zoomScale": 1}`
+ * - Android: `{"contentInset": {"bottom": 0, "left": 0, "right": 0, "top": 0}, "contentOffset": {"x": 0, "y": 31.619047164916992}, "contentSize": {"height": 1624.761962890625, "width": 411.4285583496094}, "layoutMeasurement": {"height": 785.5238037109375, "width": 411.4285583496094}, "responderIgnoreScroll": true, "target": 139, "velocity": {"x": -1.3633992671966553, "y": -1.3633992671966553}}`
+ */
export const ScrollViewEventBuilder = {
- scroll: (offset: ContentOffset = { y: 0, x: 0 }) => {
+ scroll: (
+ offset: ContentOffset = { y: 0, x: 0 },
+ options?: ScrollEventOptions
+ ) => {
return {
nativeEvent: {
contentInset: { bottom: 0, left: 0, right: 0, top: 0 },
contentOffset: { y: offset.y, x: offset.x },
- contentSize: { height: 0, width: 0 },
+ contentSize: {
+ height: options?.contentSize?.height ?? 0,
+ width: options?.contentSize?.width ?? 0,
+ },
layoutMeasurement: {
- height: 0,
- width: 0,
+ height: options?.layoutMeasurement?.height ?? 0,
+ width: options?.layoutMeasurement?.width ?? 0,
},
responderIgnoreScroll: true,
target: 0,
diff --git a/src/user-event/scroll/__tests__/__snapshots__/scrollTo-flatList.tsx.snap b/src/user-event/scroll/__tests__/__snapshots__/scroll-to-flat-list.test.tsx.snap
similarity index 100%
rename from src/user-event/scroll/__tests__/__snapshots__/scrollTo-flatList.tsx.snap
rename to src/user-event/scroll/__tests__/__snapshots__/scroll-to-flat-list.test.tsx.snap
diff --git a/src/user-event/scroll/__tests__/__snapshots__/scrollTo.test.tsx.snap b/src/user-event/scroll/__tests__/__snapshots__/scroll-to.test.tsx.snap
similarity index 100%
rename from src/user-event/scroll/__tests__/__snapshots__/scrollTo.test.tsx.snap
rename to src/user-event/scroll/__tests__/__snapshots__/scroll-to.test.tsx.snap
diff --git a/src/user-event/scroll/__tests__/scrollTo-flatList.tsx b/src/user-event/scroll/__tests__/scroll-to-flat-list.test.tsx
similarity index 63%
rename from src/user-event/scroll/__tests__/scrollTo-flatList.tsx
rename to src/user-event/scroll/__tests__/scroll-to-flat-list.test.tsx
index 67dcc1a28..30cff931e 100644
--- a/src/user-event/scroll/__tests__/scrollTo-flatList.tsx
+++ b/src/user-event/scroll/__tests__/scroll-to-flat-list.test.tsx
@@ -1,7 +1,8 @@
import * as React from 'react';
-import { FlatList, ScrollViewProps, Text } from 'react-native';
-import { EventEntry, createEventLogger } from '../../../test-utils';
+import { FlatList, ScrollViewProps, Text, View } from 'react-native';
import { render, screen } from '../../..';
+import '../../../matchers/extend-expect';
+import { EventEntry, createEventLogger } from '../../../test-utils';
import { userEvent } from '../..';
const data = ['A', 'B', 'C', 'D', 'E', 'F', 'G'];
@@ -68,3 +69,46 @@ describe('scrollTo() with FlatList', () => {
]);
});
});
+
+const DATA = new Array(100).fill(0).map((_, i) => `Item ${i}`);
+
+function Scrollable() {
+ return (
+
+ }
+ initialNumToRender={10}
+ updateCellsBatchingPeriod={0}
+ />
+
+ );
+}
+
+function Item({ title }: { title: string }) {
+ return (
+
+ {title}
+
+ );
+}
+
+test('scrollTo with contentSize and layoutMeasurement update FlatList content', async () => {
+ render();
+ const user = userEvent.setup();
+
+ expect(screen.getByText('Item 0')).toBeOnTheScreen();
+ expect(screen.getByText('Item 7')).toBeOnTheScreen();
+ expect(screen.queryByText('Item 15')).not.toBeOnTheScreen();
+
+ await user.scrollTo(screen.getByTestId('flat-list'), {
+ y: 300,
+ contentSize: { width: 240, height: 480 },
+ layoutMeasurement: { width: 240, height: 480 },
+ });
+
+ expect(screen.getByText('Item 0')).toBeOnTheScreen();
+ expect(screen.getByText('Item 7')).toBeOnTheScreen();
+ expect(screen.getByText('Item 15')).toBeOnTheScreen();
+});
diff --git a/src/user-event/scroll/__tests__/scrollTo.test.tsx b/src/user-event/scroll/__tests__/scroll-to.test.tsx
similarity index 100%
rename from src/user-event/scroll/__tests__/scrollTo.test.tsx
rename to src/user-event/scroll/__tests__/scroll-to.test.tsx
diff --git a/src/user-event/scroll/scroll-to.ts b/src/user-event/scroll/scroll-to.ts
index 6ddd066be..082152e17 100644
--- a/src/user-event/scroll/scroll-to.ts
+++ b/src/user-event/scroll/scroll-to.ts
@@ -5,8 +5,8 @@ import { EventBuilder } from '../event-builder';
import { ErrorWithStack } from '../../helpers/errors';
import { isHostScrollView } from '../../helpers/host-component-names';
import { pick } from '../../helpers/object';
-import { dispatchEvent, wait } from '../utils';
import { ContentOffset } from '../event-builder/scroll-view';
+import { dispatchEvent, wait } from '../utils';
import {
createScrollSteps,
inertialInterpolator,
@@ -14,7 +14,18 @@ import {
} from './utils';
import { getElementScrollOffset, setElementScrollOffset } from './state';
-export interface VerticalScrollToOptions {
+interface CommonScrollToOptions {
+ contentSize?: {
+ height: number;
+ width: number;
+ };
+ layoutMeasurement?: {
+ height: number;
+ width: number;
+ };
+}
+
+export interface VerticalScrollToOptions extends CommonScrollToOptions {
y: number;
momentumY?: number;
@@ -23,7 +34,7 @@ export interface VerticalScrollToOptions {
momentumX?: never;
}
-export interface HorizontalScrollToOptions {
+export interface HorizontalScrollToOptions extends CommonScrollToOptions {
x: number;
momentumX?: number;
@@ -50,13 +61,20 @@ export async function scrollTo(
ensureScrollViewDirection(element, options);
+ dispatchEvent(
+ element,
+ 'contentSizeChange',
+ options.contentSize?.width ?? 0,
+ options.contentSize?.height ?? 0
+ );
+
const initialPosition = getElementScrollOffset(element);
const dragSteps = createScrollSteps(
{ y: options.y, x: options.x },
initialPosition,
linearInterpolator
);
- await emitDragScrollEvents(this.config, element, dragSteps);
+ await emitDragScrollEvents(this.config, element, dragSteps, options);
const momentumStart = dragSteps.at(-1) ?? initialPosition;
const momentumSteps = createScrollSteps(
@@ -64,7 +82,7 @@ export async function scrollTo(
momentumStart,
inertialInterpolator
);
- await emitMomentumScrollEvents(this.config, element, momentumSteps);
+ await emitMomentumScrollEvents(this.config, element, momentumSteps, options);
const finalPosition =
momentumSteps.at(-1) ?? dragSteps.at(-1) ?? initialPosition;
@@ -74,7 +92,8 @@ export async function scrollTo(
async function emitDragScrollEvents(
config: UserEventConfig,
element: ReactTestInstance,
- scrollSteps: ContentOffset[]
+ scrollSteps: ContentOffset[],
+ scrollOptions: ScrollToOptions
) {
if (scrollSteps.length === 0) {
return;
@@ -84,7 +103,7 @@ async function emitDragScrollEvents(
dispatchEvent(
element,
'scrollBeginDrag',
- EventBuilder.ScrollView.scroll(scrollSteps[0])
+ EventBuilder.ScrollView.scroll(scrollSteps[0], scrollOptions)
);
// Note: experimentally, in case of drag scroll the last scroll step
@@ -95,7 +114,7 @@ async function emitDragScrollEvents(
dispatchEvent(
element,
'scroll',
- EventBuilder.ScrollView.scroll(scrollSteps[i])
+ EventBuilder.ScrollView.scroll(scrollSteps[i], scrollOptions)
);
}
@@ -104,14 +123,15 @@ async function emitDragScrollEvents(
dispatchEvent(
element,
'scrollEndDrag',
- EventBuilder.ScrollView.scroll(lastStep)
+ EventBuilder.ScrollView.scroll(lastStep, scrollOptions)
);
}
async function emitMomentumScrollEvents(
config: UserEventConfig,
element: ReactTestInstance,
- scrollSteps: ContentOffset[]
+ scrollSteps: ContentOffset[],
+ scrollOptions: ScrollToOptions
) {
if (scrollSteps.length === 0) {
return;
@@ -121,7 +141,7 @@ async function emitMomentumScrollEvents(
dispatchEvent(
element,
'momentumScrollBegin',
- EventBuilder.ScrollView.scroll(scrollSteps[0])
+ EventBuilder.ScrollView.scroll(scrollSteps[0], scrollOptions)
);
// Note: experimentally, in case of momentum scroll the last scroll step
@@ -132,7 +152,7 @@ async function emitMomentumScrollEvents(
dispatchEvent(
element,
'scroll',
- EventBuilder.ScrollView.scroll(scrollSteps[i])
+ EventBuilder.ScrollView.scroll(scrollSteps[i], scrollOptions)
);
}
@@ -141,7 +161,7 @@ async function emitMomentumScrollEvents(
dispatchEvent(
element,
'momentumScrollEnd',
- EventBuilder.ScrollView.scroll(lastStep)
+ EventBuilder.ScrollView.scroll(lastStep, scrollOptions)
);
}
diff --git a/src/user-event/utils/dispatch-event.ts b/src/user-event/utils/dispatch-event.ts
index a8c6ad91c..76a9d3384 100644
--- a/src/user-event/utils/dispatch-event.ts
+++ b/src/user-event/utils/dispatch-event.ts
@@ -6,12 +6,12 @@ import act from '../../act';
*
* @param element element trigger event on
* @param eventName name of the event
- * @param event event payload
+ * @param event event payload(s)
*/
export function dispatchEvent(
element: ReactTestInstance,
eventName: string,
- event: unknown
+ ...event: unknown[]
) {
const handler = getEventHandler(element, eventName);
if (!handler) {
@@ -20,7 +20,7 @@ export function dispatchEvent(
// This will be called synchronously.
void act(() => {
- handler(event);
+ handler(...event);
});
}
diff --git a/website/docs/UserEvent.md b/website/docs/UserEvent.md
index 363e2a094..68effc391 100644
--- a/website/docs/UserEvent.md
+++ b/website/docs/UserEvent.md
@@ -204,9 +204,13 @@ scrollTo(
options: {
y: number,
momentumY?: number,
+ contentSize?: { width: number, height: number },
+ layoutMeasurement?: { width: number, height: number },
} | {
x: number,
momentumX?: number,
+ contentSize?: { width: number, height: number },
+ layoutMeasurement?: { width: number, height: number },
}
```
@@ -219,22 +223,26 @@ await user.scrollTo(scrollView, { y: 100, momentumY: 200 });
This helper simulates user scrolling a host `ScrollView` element.
-This function supports only host `ScrollView` elements, passing other element types will result in error. Note that `FlatList` is accepted as it renders to a host `ScrolLView` element, however in the current iteration we focus only on base `ScrollView` only features.
+This function supports only host `ScrollView` elements, passing other element types will result in error. Note that `FlatList` is accepted as it renders to a host `ScrolLView` element.
Scroll interaction should match `ScrollView` element direction. For vertical scroll view (default or explicit `horizontal={false}`) you should pass only `y` (and optionally also `momentumY`) option, for horizontal scroll view (`horizontal={true}`) you should pass only `x` (and optionally `momentumX`) option.
Each scroll interaction consists of a mandatory drag scroll part which simulates user dragging the scroll view with his finger (`y` or `x` option). This may optionally be followed by a momentum scroll movement which simulates the inertial movement of scroll view content after the user lifts his finger up (`momentumY` or `momentumX` options).
-### Options {#type-options}
+### Options {#scroll-to-options}
- `y` - target vertical drag scroll position
- `x` - target horizontal drag scroll position
- `momentumY` - target vertical momentum scroll position
- `momentumX` - target horizontal momentum scroll position
+- `contentSize` - passed to `ScrollView` events and enabling `FlatList` updates
+- `layoutMeasurement` - passed to `ScrollView` events and enabling `FlatList` updates
User Event will generate a number of intermediate scroll steps to simulate user scroll interaction. You should not rely on exact number or values of these scrolls steps as they might be change in the future version.
-This function will remember where the last scroll ended, so subsequent scroll interaction will starts from that positition. The initial scroll position will be assumed to be `{ y: 0, x: 0 }`.
+This function will remember where the last scroll ended, so subsequent scroll interaction will starts from that position. The initial scroll position will be assumed to be `{ y: 0, x: 0 }`.
+
+In order to simulate a `FlatList` (and other controls based on `VirtualizedList`) scrolling, you should pass the `contentSize` and `layoutMeasurement` options, which enable the underlying logic to update the currently visible window.
### Sequence of events