mirror of
https://github.com/status-im/react-native.git
synced 2025-01-25 00:39:03 +00:00
72670bf8d2
Summary: This adds support for both automagical sticky section headers in `SectionList` as well as the more free-form `stickyHeaderIndices` on `FlatList` or `VirtualizedList`. The basic concept is to take the initial `stickySectionHeaders` and remap them to the indices corresponding to the mounted subset in the render window. The main trick here is that the currently stuck header might itself be outside of the render window, so we need to search the gap to see if that's the case and render it (with spacers above and below it instead of one big spacer). In the `SectionList` we simply pre-compute the sticky headers at the same time as when we scan the sections to determine the flattened length and pass those to `VirtualizedList`. This also requires some updates to `ScrollView` to work in the churny environment of `VirtualizedList`. We propogate the keys on the children to the animated wrappers so that as items are removed and the indices of the remaining items change, react can keep proper track of them. We also fix the scroll back case where new headers are rendered from the top down and aren't updated with the `setNextLayoutY` callback because the `onLayout` call for the next header happened before it was mounted. This is done by just tracking all the layout values in a map and providing them to the sticky components at render time. This might also improve perf a little by property configuring the animations syncronously instead of waiting for the `onLayout` callback. We also need to protect against stale onLayout callbacks and other fun stuff. == Test Plan == https://www.facebook.com/groups/react.native.community/permalink/940332509435661/ Scroll a lot with and without debug mode on. Make sure spinner still spins and there are no crashes (lots of crashes during development due to the animated configuration being non-monotonic if anything stale values get through). Also made sure that tapping a row to change it's height would properly update the animation configurations so the collision point would still be correct. Reviewed By: yungsters Differential Revision: D4695065 fbshipit-source-id: 855c4e31c8f8b450d32150dbdb2e07f1a9f9f98e
123 lines
3.6 KiB
JavaScript
123 lines
3.6 KiB
JavaScript
/**
|
|
* Copyright (c) 2015-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 ScrollViewStickyHeader
|
|
* @flow
|
|
*/
|
|
'use strict';
|
|
|
|
const Animated = require('Animated');
|
|
const React = require('React');
|
|
const StyleSheet = require('StyleSheet');
|
|
|
|
type Props = {
|
|
children?: React.Element<*>,
|
|
nextHeaderLayoutY: ?number,
|
|
onLayout: (event: Object) => void,
|
|
scrollAnimatedValue: Animated.Value,
|
|
};
|
|
|
|
class ScrollViewStickyHeader extends React.Component {
|
|
props: Props;
|
|
state: {
|
|
measured: boolean,
|
|
layoutY: number,
|
|
layoutHeight: number,
|
|
nextHeaderLayoutY: ?number,
|
|
};
|
|
|
|
constructor(props: Props, context: Object) {
|
|
super(props, context);
|
|
this.state = {
|
|
measured: false,
|
|
layoutY: 0,
|
|
layoutHeight: 0,
|
|
nextHeaderLayoutY: props.nextHeaderLayoutY,
|
|
};
|
|
}
|
|
|
|
setNextHeaderY(y: number) {
|
|
this.setState({ nextHeaderLayoutY: y });
|
|
}
|
|
|
|
_onLayout = (event) => {
|
|
this.setState({
|
|
measured: true,
|
|
layoutY: event.nativeEvent.layout.y,
|
|
layoutHeight: event.nativeEvent.layout.height,
|
|
});
|
|
|
|
this.props.onLayout(event);
|
|
const child = React.Children.only(this.props.children);
|
|
if (child.props.onLayout) {
|
|
child.props.onLayout(event);
|
|
}
|
|
};
|
|
|
|
render() {
|
|
const {measured, layoutHeight, layoutY, nextHeaderLayoutY} = this.state;
|
|
|
|
let translateY;
|
|
if (measured) {
|
|
// The interpolation looks like:
|
|
// - Negative scroll: no translation
|
|
// - From 0 to the y of the header: no translation. This will cause the header
|
|
// to scroll normally until it reaches the top of the scroll view.
|
|
// - From header y to when the next header y hits the bottom edge of the header: translate
|
|
// equally to scroll. This will cause the header to stay at the top of the scroll view.
|
|
// - Past the collision with the next header y: no more translation. This will cause the
|
|
// header to continue scrolling up and make room for the next sticky header.
|
|
// In the case that there is no next header just translate equally to
|
|
// scroll indefinetly.
|
|
const inputRange = [-1, 0, layoutY];
|
|
const outputRange: Array<number> = [0, 0, 0];
|
|
// Sometimes headers jump around so we make sure we don't violate the monotonic inputRange
|
|
// condition.
|
|
const collisionPoint = (nextHeaderLayoutY || 0) - layoutHeight;
|
|
if (collisionPoint >= layoutY) {
|
|
inputRange.push(collisionPoint, collisionPoint + 1);
|
|
outputRange.push(collisionPoint - layoutY, collisionPoint - layoutY);
|
|
} else {
|
|
inputRange.push(layoutY + 1);
|
|
outputRange.push(1);
|
|
}
|
|
translateY = this.props.scrollAnimatedValue.interpolate({
|
|
inputRange,
|
|
outputRange,
|
|
});
|
|
} else {
|
|
translateY = 0;
|
|
}
|
|
|
|
const child = React.Children.only(this.props.children);
|
|
|
|
return (
|
|
<Animated.View
|
|
collapsable={false}
|
|
onLayout={this._onLayout}
|
|
style={[child.props.style, styles.header, {transform: [{translateY}]}]}>
|
|
{React.cloneElement(child, {
|
|
style: styles.fill, // We transfer the child style to the wrapper.
|
|
onLayout: undefined, // we call this manually through our this._onLayout
|
|
})}
|
|
</Animated.View>
|
|
);
|
|
}
|
|
}
|
|
|
|
const styles = StyleSheet.create({
|
|
header: {
|
|
zIndex: 10,
|
|
},
|
|
fill: {
|
|
flex: 1,
|
|
},
|
|
});
|
|
|
|
module.exports = ScrollViewStickyHeader;
|