2015-10-15 14:18:41 +00:00
|
|
|
/**
|
|
|
|
* Copyright 2004-present Facebook. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* @providesModule RecyclerViewBackedScrollView
|
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
var React = require('React');
|
|
|
|
var ScrollResponder = require('ScrollResponder');
|
|
|
|
var ScrollView = require('ScrollView');
|
2015-11-11 09:19:03 +00:00
|
|
|
var View = require('View');
|
|
|
|
var StyleSheet = require('StyleSheet');
|
2015-10-15 14:18:41 +00:00
|
|
|
|
|
|
|
var requireNativeComponent = require('requireNativeComponent');
|
|
|
|
|
2015-10-16 10:24:38 +00:00
|
|
|
var INNERVIEW = 'InnerView';
|
|
|
|
|
2015-10-15 14:18:41 +00:00
|
|
|
/**
|
|
|
|
* Wrapper around android native recycler view.
|
|
|
|
*
|
|
|
|
* It simply renders rows passed as children in a separate recycler view cells
|
2015-12-15 17:08:39 +00:00
|
|
|
* similarly to how `ScrollView` is doing it. Thanks to the fact that it uses
|
2015-10-15 14:18:41 +00:00
|
|
|
* native `RecyclerView` though, rows that are out of sight are going to be
|
2015-12-15 17:08:39 +00:00
|
|
|
* automatically detached (similarly on how this would work with
|
2015-10-15 14:18:41 +00:00
|
|
|
* `removeClippedSubviews = true` on a `ScrollView.js`).
|
|
|
|
*
|
|
|
|
* CAUTION: This is an experimental component and should only be used together
|
|
|
|
* with javascript implementation of list view (see ListView.js). In order to
|
|
|
|
* use it pass this component as `renderScrollComponent` to the list view. For
|
|
|
|
* now only horizontal scrolling is supported.
|
|
|
|
*
|
|
|
|
* Example:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* getInitialState: function() {
|
|
|
|
* var ds = new ListView.DataSource({rowHasChanged: (r1, r2) => r1 !== r2});
|
|
|
|
* return {
|
|
|
|
* dataSource: ds.cloneWithRows(['row 1', 'row 2']),
|
|
|
|
* };
|
|
|
|
* },
|
|
|
|
*
|
|
|
|
* render: function() {
|
|
|
|
* return (
|
|
|
|
* <ListView
|
|
|
|
* dataSource={this.state.dataSource}
|
|
|
|
* renderRow={rowData => <Text>{rowData}</Text>}
|
|
|
|
* renderScrollComponent={props => <RecyclerViewBackedScrollView {...props} />}
|
|
|
|
* />
|
|
|
|
* );
|
|
|
|
* },
|
|
|
|
* ```
|
|
|
|
*/
|
|
|
|
var RecyclerViewBackedScrollView = React.createClass({
|
|
|
|
|
|
|
|
propTypes: {
|
|
|
|
...ScrollView.propTypes,
|
|
|
|
},
|
|
|
|
|
2015-10-16 14:43:09 +00:00
|
|
|
mixins: [ScrollResponder.Mixin],
|
2015-10-15 14:18:41 +00:00
|
|
|
|
|
|
|
getInitialState: function() {
|
|
|
|
return this.scrollResponderMixinGetInitialState();
|
|
|
|
},
|
|
|
|
|
|
|
|
getScrollResponder: function() {
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2015-10-16 14:43:09 +00:00
|
|
|
setNativeProps: function(props: Object) {
|
|
|
|
this.refs[INNERVIEW].setNativeProps(props);
|
|
|
|
},
|
|
|
|
|
2015-11-20 15:36:23 +00:00
|
|
|
_handleContentSizeChange: function(event) {
|
|
|
|
var {width, height} = event.nativeEvent;
|
|
|
|
this.props.onContentSizeChange(width, height);
|
|
|
|
},
|
|
|
|
|
2015-10-15 14:18:41 +00:00
|
|
|
render: function() {
|
|
|
|
var props = {
|
|
|
|
...this.props,
|
|
|
|
onTouchStart: this.scrollResponderHandleTouchStart,
|
|
|
|
onTouchMove: this.scrollResponderHandleTouchMove,
|
|
|
|
onTouchEnd: this.scrollResponderHandleTouchEnd,
|
|
|
|
onScrollBeginDrag: this.scrollResponderHandleScrollBeginDrag,
|
|
|
|
onScrollEndDrag: this.scrollResponderHandleScrollEndDrag,
|
|
|
|
onMomentumScrollBegin: this.scrollResponderHandleMomentumScrollBegin,
|
|
|
|
onMomentumScrollEnd: this.scrollResponderHandleMomentumScrollEnd,
|
|
|
|
onStartShouldSetResponder: this.scrollResponderHandleStartShouldSetResponder,
|
|
|
|
onStartShouldSetResponderCapture: this.scrollResponderHandleStartShouldSetResponderCapture,
|
|
|
|
onScrollShouldSetResponder: this.scrollResponderHandleScrollShouldSetResponder,
|
|
|
|
onResponderGrant: this.scrollResponderHandleResponderGrant,
|
|
|
|
onResponderRelease: this.scrollResponderHandleResponderRelease,
|
|
|
|
onResponderReject: this.scrollResponderHandleResponderReject,
|
|
|
|
onScroll: this.scrollResponderHandleScroll,
|
|
|
|
style: ([{flex: 1}, this.props.style]: ?Array<any>),
|
2015-10-16 10:24:38 +00:00
|
|
|
ref: INNERVIEW,
|
2015-10-15 14:18:41 +00:00
|
|
|
};
|
2015-11-11 09:19:03 +00:00
|
|
|
|
2015-11-20 15:36:23 +00:00
|
|
|
if (this.props.onContentSizeChange) {
|
|
|
|
props.onContentSizeChange = this._handleContentSizeChange;
|
|
|
|
}
|
|
|
|
|
2015-11-11 09:19:03 +00:00
|
|
|
var wrappedChildren = React.Children.map(this.props.children, (child) => {
|
|
|
|
if (!child) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<View
|
|
|
|
collapsable={false}
|
|
|
|
style={styles.absolute}>
|
|
|
|
{child}
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2015-10-15 14:18:41 +00:00
|
|
|
return (
|
2015-11-11 09:19:03 +00:00
|
|
|
<NativeAndroidRecyclerView {...props}>
|
|
|
|
{wrappedChildren}
|
|
|
|
</NativeAndroidRecyclerView>
|
2015-10-15 14:18:41 +00:00
|
|
|
);
|
|
|
|
},
|
2015-11-11 09:19:03 +00:00
|
|
|
});
|
2015-10-15 14:18:41 +00:00
|
|
|
|
2015-11-11 09:19:03 +00:00
|
|
|
var styles = StyleSheet.create({
|
|
|
|
absolute: {
|
|
|
|
position: 'absolute',
|
|
|
|
top: 0,
|
|
|
|
left: 0,
|
|
|
|
right: 0,
|
|
|
|
},
|
2015-10-15 14:18:41 +00:00
|
|
|
});
|
|
|
|
|
2015-11-18 13:32:46 +00:00
|
|
|
var NativeAndroidRecyclerView = requireNativeComponent(
|
|
|
|
'AndroidRecyclerViewBackedScrollView',
|
|
|
|
RecyclerViewBackedScrollView
|
|
|
|
);
|
2015-10-15 14:18:41 +00:00
|
|
|
|
|
|
|
module.exports = RecyclerViewBackedScrollView;
|