mirror of
https://github.com/status-im/react-native.git
synced 2025-01-25 00:39:03 +00:00
a3457486e3
Summary: We really need a better list view - so here it is! Main changes from existing `ListView`: * Items are "virtualized" to limit memory - that is, items outside of the render window are unmounted and their memory is reclaimed. This means that instance state is not preserved when items scroll out of the render window. * No `DataSource` - just a simple `data` prop of shape `Array<any>`. By default, they are expected to be of the shape `{key: string}` but a custom `rowExtractor` function can be provided for different shapes, e.g. graphql data where you want to map `id` to `key`. Note the underlying `VirtualizedList` is much more flexible. * Fancy `scrollTo` functionality: `scrollToEnd`, `scrollToIndex`, and `scrollToItem` in addition to the normal `scrollToOffset`. * Built-in pull to refresh support - set set the `onRefresh` and `refreshing` props. * Rendering additional rows is usually done with low priority, after any interactions/animations complete, unless we're about to run out of rendered content. This should help apps feel more responsive. * Component props replace render functions, e.g. `ItemComponent: ReactClass<{item: Item, index: number}>` replaces `renderRow: (...) => React.Element<*>` * Supports dynamic items automatically by using `onLayout`, or `getItemLayout` can be provided for a perf boost and smoother `scrollToIndex` and scroll bar behavior. * Visibility callback replaced with more powerful viewability callback and works in vertical and horizontal mode on at least Android and iOS, but probably other platforms as well. Extra power comes from the `viewablePercentThreshold` that lets the client decide when an item should be considered viewable. Demo: https://www.facebook.com/groups/576288835853049/permalink/753923058089625/ Reviewed By: yungsters Differential Revision: D4412469 fbshipit-source-id: e2d891490bf76fe14df49294ecddf78a58adcf23
99 lines
2.7 KiB
JavaScript
99 lines
2.7 KiB
JavaScript
/**
|
|
* Copyright (c) 2013-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*
|
|
* @providesModule ViewabilityHelper
|
|
* @flow
|
|
*/
|
|
'use strict';
|
|
|
|
const invariant = require('invariant');
|
|
|
|
export type Viewable = {item: any, key: string, index: ?number, isViewable: boolean, section?: any};
|
|
|
|
/**
|
|
* A row is said to be in a "viewable" state when either of the following
|
|
* is true:
|
|
* - Occupying >= viewablePercentThreshold of the viewport
|
|
* - Entirely visible on screen
|
|
*/
|
|
const ViewabilityHelper = {
|
|
computeViewableItems(
|
|
viewablePercentThreshold: number,
|
|
itemCount: number,
|
|
scrollOffset: number,
|
|
viewportHeight: number,
|
|
getFrameMetrics: (index: number) => ?{length: number, offset: number},
|
|
renderRange?: {first: number, last: number}, // Optional optimization to reduce the scan size
|
|
): Array<number> {
|
|
const viewableIndices = [];
|
|
if (itemCount === 0) {
|
|
return viewableIndices;
|
|
}
|
|
let firstVisible = -1;
|
|
const {first, last} = renderRange || {first: 0, last: itemCount - 1};
|
|
invariant(
|
|
last < itemCount,
|
|
'Invalid render range ' + JSON.stringify({renderRange, itemCount})
|
|
);
|
|
for (let idx = first; idx <= last; idx++) {
|
|
const metrics = getFrameMetrics(idx);
|
|
if (!metrics) {
|
|
continue;
|
|
}
|
|
const top = metrics.offset - scrollOffset;
|
|
const bottom = top + metrics.length;
|
|
if ((top < viewportHeight) && (bottom > 0)) {
|
|
firstVisible = idx;
|
|
if (_isViewable(
|
|
viewablePercentThreshold,
|
|
top,
|
|
bottom,
|
|
viewportHeight
|
|
)) {
|
|
viewableIndices.push(idx);
|
|
}
|
|
} else if (firstVisible >= 0) {
|
|
break;
|
|
}
|
|
}
|
|
return viewableIndices;
|
|
},
|
|
};
|
|
|
|
|
|
function _isViewable(
|
|
viewablePercentThreshold: number,
|
|
top: number,
|
|
bottom: number,
|
|
viewportHeight: number
|
|
): bool {
|
|
return _isEntirelyVisible(top, bottom, viewportHeight) ||
|
|
_getPercentOccupied(top, bottom, viewportHeight) >=
|
|
viewablePercentThreshold;
|
|
}
|
|
|
|
function _getPercentOccupied(
|
|
top: number,
|
|
bottom: number,
|
|
viewportHeight: number
|
|
): number {
|
|
let visibleHeight = Math.min(bottom, viewportHeight) - Math.max(top, 0);
|
|
visibleHeight = Math.max(0, visibleHeight);
|
|
return Math.max(0, visibleHeight * 100 / viewportHeight);
|
|
}
|
|
|
|
function _isEntirelyVisible(
|
|
top: number,
|
|
bottom: number,
|
|
viewportHeight: number
|
|
): bool {
|
|
return top >= 0 && bottom <= viewportHeight && bottom > top;
|
|
}
|
|
|
|
module.exports = ViewabilityHelper;
|