DEV Community

Cover image for React Native: How To Build  Bidirectional Infinite Scroll
Vishal Narkhede
Vishal Narkhede

Posted on • Originally published at getstream.io

React Native: How To Build Bidirectional Infinite Scroll

๐Ÿ‘‹ Introduction

โ€‹
There have been many discussions on Stack Overflow and GitHub around implementing infinite scroll using React Native, on top of FlatList or SectionList. I've found that there aren't any easy solutions out there for bidirectional infinite scroll, which works on both Android and iOS. Recently, while working on v3.0.0 of React Native Chat SDK at Stream, we added bidirectional infinite scroll to our chat components. We had to jump through plenty of hurdles to make it happen while maintaining a good user experience around scrolling (especially for Android). Thus we decided to publish an excellent and small open-source package, on top of FlatList, to make this task easier for other React Native developers.โ€‹

GitHub โญ : github.com/getstream/react-native-bidirectional-infinite-scroll

โ€‹

๐Ÿ”Œ Usage

As mentioned earlier, this package is a wrapper around FlatList, and so it can be used exactly like FlatList from React Native, with few additional props:

Usage

โ€‹

๐Ÿ”ฎ Quick Preview

Scroll down further, for the tutorial

iOS Android
Alt Text Alt Text

โ€‹

Let's Dive Deep

โ€‹โ€‹โ€‹

๐Ÿ“š What and How

This section will walk you through the hurdles of implementing bidirectional infinite scroll and how we solved them.
โ€‹

Support for onStartReached

FlatList from React Native has built-in support for infinite scroll in a single direction (from the end of the list). You can add a prop onEndReachedon FlatList. This function gets called when your scroll is near the end of the list, and thus you can append more items to the list from this function. You can Google for React Native infinite scrolling, and you will find plenty of examples for this. Unfortunately, the FlatList doesn't provide any similar prop for onStartReached for infinite scrolling in other directions.
โ€‹
We have added support for this prop as part of this package by simply adding the onScroll handler on FlatList, and executing the callback function (onStartReached) when the scroll is near the start of the list. If you take a look at the implementation of VirtualizedList, you will notice that onEndReachedfunction gets called only once per content length. That's there for a good purpose - to avoid redundant function calls on every scroll position change. Similar optimizations have been done for onStartReached within this package.
โ€‹

Race condition between onStartReached and onEndReached

To maintain a smooth scrolling experience, we need to manage the execution order of onStartReached and onEndReached. Because if both the callbacks happen at (almost) the same time, which means items will be added to the list from both directions. This may result in scroll jump, and that's not a good user experience. Thus it's essential to make sure one callback waits for the other callback to finish.
โ€‹

onStartReachedThreshold and onEndReachedThreshold

FlatList from React Native has a support for the prop onEndReachedThreshold, which is documented here
โ€‹

How far from the end (in units of visible length of the list) the bottom edge of the list must be from the end of the content to trigger the onEndReached callback.

Instead, it's easier to have a fixed value offset (distance from the end of the list) to trigger one of these callbacks. Thus we can maintain these two values within our implementation. So onStartReachedThreshold and onEndReachedThreshold props accept the number - distance from the end of the list to trigger one of these callbacks.
โ€‹

Smooth scrolling experience

FlatList from React Native accepts a prop - maintainVisibleContentPosition, which makes sure your scroll doesn't jump to the end of the list when more items are added to the list. But this prop is only supported on iOS for now. So taking some inspiration from this PR, we published our separate package to add support for this prop on Android - flat-list-mvcp. And thus @stream-io/flat-list-mvcp is a dependency of the react-native-bidirectional-scroll package.
โ€‹

๐Ÿ–ฅ Tutorial: Chat UI With Bidirectional Infinite Scroll

Now let's see how we can implement a chat message list, scrolling infinitely in both directions.

Setup

Let's start by creating a React Native app:

$ npx react-native init AwesomeChatList
$ cd AwesomeChatList
Enter fullscreen mode Exit fullscreen mode

โ€‹
Add the required dependencies:

$ yarn add react-native-bidirectional-infinite-scroll @stream-io/flat-list-mvcp
Enter fullscreen mode Exit fullscreen mode

Next, run the app:

$ npx react-native run-ios
Enter fullscreen mode Exit fullscreen mode

Note: The server will auto-refresh as you make changes to the code.

โ€‹

Create message list

Open this project in some editor (I love VS Code) and open App.js in the project's root directory.
โ€‹
We are going to populate the list with dummy messages. In a real application, these messages are queried against a server. For example, we can write a simple utility function to mock this API call for querying n number of messages.
โ€‹
Create a file - utils.js

Next, let's write a small UI component for the message bubble. Create a file named MessageBubble.js.โ€‹

โ€‹
Now let's first implement a simple list, which renders these messages on the first load. Copy the following content to App.js and hit save.

You should be able to see the list of messages as shown in the screenshot on the right.
โ€‹
Message List

Add Infinite Scroll

Next, let's implement infinite scroll to the list by adding onStartReached and onEndReached prop functions.

  • onStartReached - add 10 messages at the beginning of the list
  • onEndReached - add 10 messages at the end of the list โ€‹

Replace the contents of App.js with the following:
โ€‹

And that's it. If you scroll up or down, you will see new messages being loaded in the list and scroll position being maintained as well (both for Android and iOS)
โ€‹
Bidirectional Infinite Scroll
โ€‹

Send Message

In real chat applications, you don't actually "infinitely" scroll. There is always an end to the loadMoreRecentMessages function. And after this point, you will want the scroll to automatically move to the bottom of the list, as other users send a new message or if you send a new message.
โ€‹
Basically, after this point, we want to enable the "autoscroll to top" (bottom in our case, since the list is inverted) functionality of FlatList. You can do this by setting a prop enableAutoscrollToTop as true. Additionally, you can also set autoscrollToTopThreshold to true.
โ€‹

  • Let's try to simulate this scenario by keeping a counter on the number of times recent messages were queried. Once the counter crosses 2, let's stop querying the message. โ€‹
  • Let's add a "Send Message" button at the bottom to send a single message. โ€‹

Replace the contents of App.js with following

Now scroll to the bottom couple of times, until you can't load any more recent messages. Now try to send a single message by pressing the "Send Message" button. You will see scroll automatically scrolling to the bottom of the list. But if you scroll up a bit and then send a message - then the scroll position will be maintained. The autoscrollToTopThreshold prop controls this threshold.

Send Message
โ€‹

๐ŸŽ‰ Congratulations!! Whats more?!

You've implemented a bidirectional infinite scroll with React Native! Additionally following props are available to add more customisations:

  • activityIndicatorColor
  • enableAutoscrollToTop
  • autoscrollToTopThreshold
  • onStartReachedThreshold
  • onEndReachedThreshold
  • showDefaultLoadingIndicators
  • HeaderLoadingIndicator
  • FooterLoadingIndicator

I hope you found it useful. Feel free to add some questions, comments, and feedback here. And don't forget to put a โญ on github repo. Happy Coding!!

Top comments (2)

Collapse
 
alimobasheri profile image
MirAli Mobasheri

Thanks a lot for the article! Really helpful. ๐Ÿ™

Collapse
 
vishalnarkhede profile image
Vishal Narkhede • Edited

Glad that it helped. It was quite fun to work on this :)