mirror of
https://github.com/status-im/react-native.git
synced 2025-01-15 03:56:03 +00:00
Fixed missing rows on UIExplorer <ListView> - Grid Layout example
Summary: public I was looking into the missing panels at the bottom of the <ListView> - Grid Layout example, and found that it was caused by several problems, some in the example and some in ListView itself. The first problem seemed to be a bug in the `_getDistanceFromEnd()` method, which calculates whether the ListView needs to load more content based on the distance of the visible content from the bottom of the scrollview. This was previously using the function Math.max(scrollProperties.contentLength, scrollProperties.visibleLength) - scrollProperties.visibleLength - scrollProperties.offset to calculate the amount the user could scroll before they run out of content. This sort-of works in most cases because `scrollProperties.contentLength` is usually longer than `scrollProperties.visibleLength`, so this would generally evaluate to scrollProperties.contentLength - scrollProperties.visibleLength - scrollProperties.offset which meant that it would be positive as long as there was content still to be displayed offscreen, and negative when you reached the end of the content. This logic breaks down if `contentLength` is less than `visibleLength`, however. For example, if you have 300pts of content loaded, and your scrollView is 500pts tall, and your scroll position is zero, this evaluates to Math.max(300, 500) - 500 - 0 = 0 In other words, the algorithm is saying that you have zero pts of scroll content remaining before you need to reload. But actually, the bottom 200pts of the screen are empty, so you're really 200pts in debt, and need to load extra rows to fill that space. The correct algorithm is simply to get rid of the `Math.max` and just use scrollProperties.contentLength - scrollProperties.visibleLength - scrollProperties.offset I originally thought that this was the cause of the gap, but it isn't, because ListView has `DEFAULT_SCROLL_RENDER_AHEAD = 1000`, which means that it tries to load at least 1000pts more content than is currently visible, to avoid gaps. This masked the bug, so in practice it wasn't causing an issue. The next problem I found was that there is an implict assumption in ListView that the first page of content you load is sufficient to cover the screen, or rather, that the first _ second page is sufficient. The constants `DEFAULT_INITIAL_ROWS = 10` and `DEFAULT_PAGE_SIZE = 1`, mean that when the ListView first loads, the following happens: 1. It loads 10 rows of content. 2. It checks if `_getDistanceFromEnd() < DEFAULT_SCROLL_RENDER_AHEAD` (1000). 3. If it is, it loads another `DEFAULT_PAGE_SIZE` rows of content, then stops. In the case of the ListView Grid Layout example, this meant that it first loaded 10 cells, then loaded another 1, for a total of 11. The problem was that going from 10 to 11 cells isn't sufficient to fill the visible scroll area, and it doesn't change the `contentSize` (since the cells wrap onto the same line), and since ListView doesn't try to load any more until the `contentSize` or `scrollOffset ` changes, it stops loading new rows at that point. I tried fixing this by calling `_renderMoreRowsIfNeeded()` after `_pageInNewRows()` so that it will continue to fetch new rows until the `_getDistanceFromEnd()` is less than the threshold, rather than stopping after the first page and waiting until the `contentSize` or `scrollOffset` change, but although this solves the problem for the Grid Layout example, it leads to over-fetching in the more common case of a standard row-based ListView. In the end, I just increased the `pageSize` to 3 for the Grid Layout example, which makes more sense anyway since loading a page that is not a multiple of the number of cells per row confuses the `_renderMoreRowsIfNeeded` algorithm, and leads to gaps at the bottom of the view. This solved the problem, however there was still a "pop-in" effect, where the additional rows were paged in after the ListView appeared. This was simply a misconfiguration in the example itself: The default of 10 rows was insufficient to fill the screen, so I changed the `initialListSize` prop to `20`. Reviewed By: javache Differential Revision: D2911690 fb-gh-sync-id: 8d6bd78843335fb091e7e24f7c2e6a416b0321d3 shipit-source-id: 8d6bd78843335fb091e7e24f7c2e6a416b0321d3
This commit is contained in:
parent
bab48182d9
commit
e7005f7f54
@ -67,6 +67,9 @@ var ListViewGridLayoutExample = React.createClass({
|
|||||||
<ListView
|
<ListView
|
||||||
contentContainerStyle={styles.list}
|
contentContainerStyle={styles.list}
|
||||||
dataSource={this.state.dataSource}
|
dataSource={this.state.dataSource}
|
||||||
|
initialListSize={21}
|
||||||
|
pageSize={3} // should be a multiple of the no. of visible cells per row
|
||||||
|
scrollRenderAheadDistance={500}
|
||||||
renderRow={this._renderRow}
|
renderRow={this._renderRow}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
@ -32,7 +32,6 @@ var {
|
|||||||
UIManager,
|
UIManager,
|
||||||
} = NativeModules;
|
} = NativeModules;
|
||||||
|
|
||||||
var PAGE_SIZE = 4;
|
|
||||||
var THUMB_URLS = [
|
var THUMB_URLS = [
|
||||||
require('./Thumbnails/like.png'),
|
require('./Thumbnails/like.png'),
|
||||||
require('./Thumbnails/dislike.png'),
|
require('./Thumbnails/dislike.png'),
|
||||||
@ -182,8 +181,8 @@ var ListViewPagingExample = React.createClass({
|
|||||||
renderSectionHeader={this.renderSectionHeader}
|
renderSectionHeader={this.renderSectionHeader}
|
||||||
renderRow={this.renderRow}
|
renderRow={this.renderRow}
|
||||||
initialListSize={10}
|
initialListSize={10}
|
||||||
pageSize={PAGE_SIZE}
|
pageSize={4}
|
||||||
scrollRenderAheadDistance={2000}
|
scrollRenderAheadDistance={500}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
@ -147,11 +147,15 @@ var ListView = React.createClass({
|
|||||||
*/
|
*/
|
||||||
onEndReached: PropTypes.func,
|
onEndReached: PropTypes.func,
|
||||||
/**
|
/**
|
||||||
* Threshold in pixels for onEndReached.
|
* Threshold in pixels (virtual, not physical) for calling onEndReached.
|
||||||
*/
|
*/
|
||||||
onEndReachedThreshold: PropTypes.number,
|
onEndReachedThreshold: PropTypes.number,
|
||||||
/**
|
/**
|
||||||
* Number of rows to render per event loop.
|
* Number of rows to render per event loop. Note: if your 'rows' are actually
|
||||||
|
* cells, i.e. they don't span the full width of your view (as in the
|
||||||
|
* ListViewGridLayoutExample), you should set the pageSize to be a multiple
|
||||||
|
* of the number of cells per row, otherwise you're likely to see gaps at
|
||||||
|
* the edge of the ListView as new pages are loaded.
|
||||||
*/
|
*/
|
||||||
pageSize: PropTypes.number,
|
pageSize: PropTypes.number,
|
||||||
/**
|
/**
|
||||||
@ -512,11 +516,7 @@ var ListView = React.createClass({
|
|||||||
},
|
},
|
||||||
|
|
||||||
_getDistanceFromEnd: function(scrollProperties) {
|
_getDistanceFromEnd: function(scrollProperties) {
|
||||||
var maxLength = Math.max(
|
return scrollProperties.contentLength - scrollProperties.visibleLength - scrollProperties.offset;
|
||||||
scrollProperties.contentLength,
|
|
||||||
scrollProperties.visibleLength
|
|
||||||
);
|
|
||||||
return maxLength - scrollProperties.visibleLength - scrollProperties.offset;
|
|
||||||
},
|
},
|
||||||
|
|
||||||
_updateVisibleRows: function(updatedFrames) {
|
_updateVisibleRows: function(updatedFrames) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user