List View component to handle cell recycling, positioning, measuring, and more.
import { ListView, ListViewItem } from 'react-scrollable-list-view';
// In your render...
<ListView
runwayItems={7}
runwayItemsOpposite={5}
aveCellHeight={250}
>
{
Array.apply(null, Array(3))
.forEach((_, i) =>
<ListViewItem height={250} key={i}><div>Item at Index: {i}</div></ListViewItem>
)
}
</ListView>
More options and configurations below.
- Npm:
npm install -S react-scrollable-list-view
react
and react-dom
are external dependencies for this component. This means you need to have
is installed and available in project.
If you build with webpack or similar build system then having them installed as a dependency is
enough. npm install -S react react-dom
CSS similar to what is below is included with the Component automatically with the help of Styled-Components.
.ListView
position: absolute
top: 0
right: 0
bottom: 0
left: 0
overflow-y: scroll
overflow-x: hidden
// Smooth momentum scrolling
-webkit-overflow-scrolling: touch
.ListView-content
position: absolute
top: 0
right: 0
bottom: 0
left: 0
.ListView-runway
position: absolute
height: 1px
width: 1px
This Component was written for Periscope so we could asynchronously fetch more broadcasts, only load images into the DOM when they are ready to be shown, recycle cells, allow sticky cells, and more.
I used infinite scroller: Google Developers article as a basis for methodology and naming conventions. It is a great read and I suggest you check it out.
Most of the cell recycling is handled by React as long as we set everything up correctly.
Here is a nonexhaustive list of what I feel this component does well:
- Cell Recycling
- Fixed height cells
- Sticky headers
- Desktop and Mobile Perf
- Async data calls
- Reflowing
Exports:
ListView
ListViewItem
ListViewStickyItem
<ListView aveCellHeight={250}>
{ getListItems( )}
</ListView>
Required. The aveCellHeight gives us a number to set the initial render heights of a cell (if the cell doesn't provide one), and allows us to estimate the runway needed for the current list. READ: less messuring makes everything faster.
Required. This is your list. Ideally this is an Array of ListViewItem
s or ListViewStickyItem
s with height prop on them.
Each of these elements should also have a unique key on the root for cell recycling to work correctly.
Optional. This flag tells the ListView if it needs to call loadMore
if it reaches the loadMoreItemOffset
in the scrolled
list.
Optional. This allows you to put a heading in your scrollable list. The benefit of knowing it is a header is it allows us to render it directly in the flow of the list, so we avoid expensive measuring.
Optional. This element will be rendered at the end of the list when ever loadMore
returns a Promise and the promise is
still pending resolution.
Optional function.
- Return: a promise that resolves in a boolean.
Optional. Number of items from the end of the list to call loadMore at.
Optional. Number of items to instantiate beyond current view in the scroll direction.
Optional. Number of items to instantiate beyond current view in the opposite direction.
Optional. Index to start as anchor item.
<ListView aveCellHeight={250}>
{
Array.apply(null, Array(3))
.forEach((_, i) =>
<ListViewItem height={250} key={i}><div>Item at Index: {i}</div></ListViewItem>
)
}
</ListView>
This is a general item and we only count the height
prop in our flow if we recognize it as one of our Item types.
Optional. This is heighly suggested but not required. If this is set we do not measure this element after render at all.
If this is not set we render using the aveCellHeight
of the ListView
, and once it has entered the DOM we measure the
rendered height. We use this height to decide if we need to reflow the items after this one. If there is no height
set, on rezise we remeasure as well.
<ListView aveCellHeight={250}>
{
Array.apply(null, Array(3))
.forEach((_, i) =>
<ListViewStickyItem height={250} key={i}><div>Item at Index: {i}</div></ListViewStickyItem>
)
}
</ListView>
This is an extension of ListViewItem
, but when this item hits the top of the list it will get the className is-stuck
,
and we apply some inline CSS to position it fixed
. In addition we make sure a stuck item is always included in the DOM
and not recycled. Subsequent StickyItems will bump the previous stuck item out of position.
Optional. This is heighly suggested but not required. If this is set we do not measure this element after render at all.
If this is not set we render using the aveCellHeight
of the ListView
, and once it has entered the DOM we measure the
rendered height. We use this height to decide if we need to reflow the items after this one. If there is no height
set, on rezise we remeasure as well.
This Component was built and developed with some specific use cases in mind. If you have suggestions PRs and issues are welcome.