mirror of
https://github.com/status-im/react-native.git
synced 2025-02-05 06:04:15 +00:00
Add SwipeableFlatList
Reviewed By: sahrens Differential Revision: D5912488 fbshipit-source-id: 3d2872a7712c00badcbd8341a7d058df14a9091a
This commit is contained in:
parent
b64e6c722c
commit
d8cc6e3c2b
189
Libraries/Experimental/SwipeableRow/SwipeableFlatList.js
Normal file
189
Libraries/Experimental/SwipeableRow/SwipeableFlatList.js
Normal file
@ -0,0 +1,189 @@
|
|||||||
|
/**
|
||||||
|
* 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 SwipeableFlatList
|
||||||
|
* @flow
|
||||||
|
* @format
|
||||||
|
*/
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
import type {Props as FlatListProps} from 'FlatList';
|
||||||
|
import type {renderItemType} from 'VirtualizedList';
|
||||||
|
|
||||||
|
const PropTypes = require('prop-types');
|
||||||
|
const React = require('React');
|
||||||
|
const SwipeableRow = require('SwipeableRow');
|
||||||
|
const FlatList = require('FlatList');
|
||||||
|
|
||||||
|
type SwipableListProps = {
|
||||||
|
/**
|
||||||
|
* To alert the user that swiping is possible, the first row can bounce
|
||||||
|
* on component mount.
|
||||||
|
*/
|
||||||
|
bounceFirstRowOnMount: boolean,
|
||||||
|
// Maximum distance to open to after a swipe
|
||||||
|
maxSwipeDistance: number | (Object => number),
|
||||||
|
// Callback method to render the view that will be unveiled on swipe
|
||||||
|
renderQuickActions: renderItemType,
|
||||||
|
};
|
||||||
|
|
||||||
|
type Props<ItemT> = SwipableListProps & FlatListProps<ItemT>;
|
||||||
|
|
||||||
|
type State = {
|
||||||
|
openRowKey: ?string,
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A container component that renders multiple SwipeableRow's in a FlatList
|
||||||
|
* implementation. This is designed to be a drop-in replacement for the
|
||||||
|
* standard React Native `FlatList`, so use it as if it were a FlatList, but
|
||||||
|
* with extra props, i.e.
|
||||||
|
*
|
||||||
|
* <SwipeableListView renderRow={..} renderQuickActions={..} {..FlatList props} />
|
||||||
|
*
|
||||||
|
* SwipeableRow can be used independently of this component, but the main
|
||||||
|
* benefit of using this component is
|
||||||
|
*
|
||||||
|
* - It ensures that at most 1 row is swiped open (auto closes others)
|
||||||
|
* - It can bounce the 1st row of the list so users know it's swipeable
|
||||||
|
* - Increase performance on iOS by locking list swiping when row swiping is occuring
|
||||||
|
* - More to come
|
||||||
|
*/
|
||||||
|
|
||||||
|
class SwipeableFlatList<ItemT> extends React.Component<Props<ItemT>, State> {
|
||||||
|
props: Props<ItemT>;
|
||||||
|
state: State;
|
||||||
|
|
||||||
|
_flatListRef: ?FlatList<ItemT> = null;
|
||||||
|
_shouldBounceFirstRowOnMount: boolean = false;
|
||||||
|
|
||||||
|
static propTypes = {
|
||||||
|
...FlatList.propTypes,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* To alert the user that swiping is possible, the first row can bounce
|
||||||
|
* on component mount.
|
||||||
|
*/
|
||||||
|
bounceFirstRowOnMount: PropTypes.bool.isRequired,
|
||||||
|
|
||||||
|
// Maximum distance to open to after a swipe
|
||||||
|
maxSwipeDistance: PropTypes.oneOfType([PropTypes.number, PropTypes.func])
|
||||||
|
.isRequired,
|
||||||
|
|
||||||
|
// Callback method to render the view that will be unveiled on swipe
|
||||||
|
renderQuickActions: PropTypes.func.isRequired,
|
||||||
|
};
|
||||||
|
|
||||||
|
static defaultProps = {
|
||||||
|
...FlatList.defaultProps,
|
||||||
|
bounceFirstRowOnMount: true,
|
||||||
|
renderQuickActions: () => null,
|
||||||
|
};
|
||||||
|
|
||||||
|
constructor(props: Props<ItemT>, context: any): void {
|
||||||
|
super(props, context);
|
||||||
|
this.state = {
|
||||||
|
openRowKey: null,
|
||||||
|
};
|
||||||
|
|
||||||
|
this._shouldBounceFirstRowOnMount = this.props.bounceFirstRowOnMount;
|
||||||
|
}
|
||||||
|
|
||||||
|
render(): React.Node {
|
||||||
|
return (
|
||||||
|
<FlatList
|
||||||
|
{...this.props}
|
||||||
|
ref={ref => {
|
||||||
|
this._flatListRef = ref;
|
||||||
|
}}
|
||||||
|
onScroll={this._onScroll}
|
||||||
|
renderItem={this._renderItem}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
_onScroll = (e): void => {
|
||||||
|
// Close any opens rows on ListView scroll
|
||||||
|
if (this.state.openRowKey) {
|
||||||
|
this.setState({
|
||||||
|
openRowKey: null,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
this.props.onScroll && this.props.onScroll(e);
|
||||||
|
};
|
||||||
|
|
||||||
|
_renderItem = (info: Object): ?React.Element<any> => {
|
||||||
|
const slideoutView = this.props.renderQuickActions(info);
|
||||||
|
const key = this.props.keyExtractor(info.item, info.index);
|
||||||
|
|
||||||
|
// If renderQuickActions is unspecified or returns falsey, don't allow swipe
|
||||||
|
if (!slideoutView) {
|
||||||
|
return this.props.renderItem(info);
|
||||||
|
}
|
||||||
|
|
||||||
|
let shouldBounceOnMount = false;
|
||||||
|
if (this._shouldBounceFirstRowOnMount) {
|
||||||
|
this._shouldBounceFirstRowOnMount = false;
|
||||||
|
shouldBounceOnMount = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SwipeableRow
|
||||||
|
slideoutView={slideoutView}
|
||||||
|
isOpen={key === this.state.openRowKey}
|
||||||
|
maxSwipeDistance={this._getMaxSwipeDistance(info)}
|
||||||
|
onOpen={() => this._onOpen(key)}
|
||||||
|
onClose={() => this._onClose(key)}
|
||||||
|
shouldBounceOnMount={shouldBounceOnMount}
|
||||||
|
onSwipeEnd={this._setListViewScrollable}
|
||||||
|
onSwipeStart={this._setListViewNotScrollable}>
|
||||||
|
{this.props.renderItem(info)}
|
||||||
|
</SwipeableRow>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// This enables rows having variable width slideoutView.
|
||||||
|
_getMaxSwipeDistance(info: Object): number {
|
||||||
|
if (typeof this.props.maxSwipeDistance === 'function') {
|
||||||
|
return this.props.maxSwipeDistance(info);
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.props.maxSwipeDistance;
|
||||||
|
}
|
||||||
|
|
||||||
|
_setListViewScrollableTo(value: boolean) {
|
||||||
|
if (this._flatListRef) {
|
||||||
|
this._flatListRef.setNativeProps({
|
||||||
|
scrollEnabled: value,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_setListViewScrollable = () => {
|
||||||
|
this._setListViewScrollableTo(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
_setListViewNotScrollable = () => {
|
||||||
|
this._setListViewScrollableTo(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
_onOpen(key: any): void {
|
||||||
|
this.setState({
|
||||||
|
openRowKey: key,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_onClose(key: any): void {
|
||||||
|
this.setState({
|
||||||
|
openRowKey: null,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = SwipeableFlatList;
|
@ -201,7 +201,7 @@ type OptionalProps<ItemT> = {
|
|||||||
*/
|
*/
|
||||||
viewabilityConfigCallbackPairs?: Array<ViewabilityConfigCallbackPair>,
|
viewabilityConfigCallbackPairs?: Array<ViewabilityConfigCallbackPair>,
|
||||||
};
|
};
|
||||||
type Props<ItemT> = RequiredProps<ItemT> &
|
export type Props<ItemT> = RequiredProps<ItemT> &
|
||||||
OptionalProps<ItemT> &
|
OptionalProps<ItemT> &
|
||||||
VirtualizedListProps;
|
VirtualizedListProps;
|
||||||
|
|
||||||
@ -209,7 +209,7 @@ const defaultProps = {
|
|||||||
...VirtualizedList.defaultProps,
|
...VirtualizedList.defaultProps,
|
||||||
numColumns: 1,
|
numColumns: 1,
|
||||||
};
|
};
|
||||||
type DefaultProps = typeof defaultProps;
|
export type DefaultProps = typeof defaultProps;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A performant interface for rendering simple, flat lists, supporting the most handy features:
|
* A performant interface for rendering simple, flat lists, supporting the most handy features:
|
||||||
|
@ -42,7 +42,7 @@ import type {
|
|||||||
|
|
||||||
type Item = any;
|
type Item = any;
|
||||||
|
|
||||||
type renderItemType = (info: any) => ?React.Element<any>;
|
export type renderItemType = (info: any) => ?React.Element<any>;
|
||||||
|
|
||||||
type ViewabilityHelperCallbackTuple = {
|
type ViewabilityHelperCallbackTuple = {
|
||||||
viewabilityHelper: ViewabilityHelper,
|
viewabilityHelper: ViewabilityHelper,
|
||||||
|
@ -46,6 +46,7 @@ const ReactNative = {
|
|||||||
get Switch() { return require('Switch'); },
|
get Switch() { return require('Switch'); },
|
||||||
get RefreshControl() { return require('RefreshControl'); },
|
get RefreshControl() { return require('RefreshControl'); },
|
||||||
get StatusBar() { return require('StatusBar'); },
|
get StatusBar() { return require('StatusBar'); },
|
||||||
|
get SwipeableFlatList() { return require('SwipeableFlatList'); },
|
||||||
get SwipeableListView() { return require('SwipeableListView'); },
|
get SwipeableListView() { return require('SwipeableListView'); },
|
||||||
get TabBarIOS() { return require('TabBarIOS'); },
|
get TabBarIOS() { return require('TabBarIOS'); },
|
||||||
get Text() { return require('Text'); },
|
get Text() { return require('Text'); },
|
||||||
|
@ -85,6 +85,10 @@ const ComponentExamples: Array<RNTesterExample> = [
|
|||||||
key: 'StatusBarExample',
|
key: 'StatusBarExample',
|
||||||
module: require('./StatusBarExample'),
|
module: require('./StatusBarExample'),
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
key: 'SwipeableFlatListExample',
|
||||||
|
module: require('./SwipeableFlatListExample')
|
||||||
|
},
|
||||||
{
|
{
|
||||||
key: 'SwipeableListViewExample',
|
key: 'SwipeableListViewExample',
|
||||||
module: require('./SwipeableListViewExample')
|
module: require('./SwipeableListViewExample')
|
||||||
|
@ -153,6 +153,11 @@ const ComponentExamples: Array<RNTesterExample> = [
|
|||||||
module: require('./StatusBarExample'),
|
module: require('./StatusBarExample'),
|
||||||
supportsTVOS: false,
|
supportsTVOS: false,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
key: 'SwipeableFlatListExample',
|
||||||
|
module: require('./SwipeableFlatListExample'),
|
||||||
|
supportsTVOS: false,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
key: 'SwipeableListViewExample',
|
key: 'SwipeableListViewExample',
|
||||||
module: require('./SwipeableListViewExample'),
|
module: require('./SwipeableListViewExample'),
|
||||||
|
151
RNTester/js/SwipeableFlatListExample.js
Normal file
151
RNTester/js/SwipeableFlatListExample.js
Normal file
@ -0,0 +1,151 @@
|
|||||||
|
/**
|
||||||
|
* 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 SwipeableFlatListExample
|
||||||
|
* @flow
|
||||||
|
* @format
|
||||||
|
*/
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const React = require('react');
|
||||||
|
const createReactClass = require('create-react-class');
|
||||||
|
const ReactNative = require('react-native');
|
||||||
|
const {
|
||||||
|
Image,
|
||||||
|
SwipeableFlatList,
|
||||||
|
TouchableHighlight,
|
||||||
|
StyleSheet,
|
||||||
|
Text,
|
||||||
|
View,
|
||||||
|
Alert,
|
||||||
|
} = ReactNative;
|
||||||
|
|
||||||
|
const RNTesterPage = require('./RNTesterPage');
|
||||||
|
|
||||||
|
const data = [
|
||||||
|
{
|
||||||
|
key: 'like',
|
||||||
|
icon: require('./Thumbnails/like.png'),
|
||||||
|
data: 'Like!',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'heart',
|
||||||
|
icon: require('./Thumbnails/heart.png'),
|
||||||
|
data: 'Heart!',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: 'party',
|
||||||
|
icon: require('./Thumbnails/party.png'),
|
||||||
|
data: 'Party!',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const SwipeableFlatListExample = createReactClass({
|
||||||
|
displayName: 'SwipeableFlatListExample',
|
||||||
|
statics: {
|
||||||
|
title: '<SwipeableFlatList>',
|
||||||
|
description: 'Performant, scrollable, swipeable list of data.',
|
||||||
|
},
|
||||||
|
|
||||||
|
render: function() {
|
||||||
|
return (
|
||||||
|
<RNTesterPage
|
||||||
|
title={this.props.navigator ? null : '<SwipeableListView>'}
|
||||||
|
noSpacer={true}
|
||||||
|
noScroll={true}>
|
||||||
|
<SwipeableFlatList
|
||||||
|
data={data}
|
||||||
|
bounceFirstRowOnMount={true}
|
||||||
|
maxSwipeDistance={160}
|
||||||
|
renderItem={this._renderItem.bind(this)}
|
||||||
|
renderQuickActions={this._renderQuickActions.bind(this)}
|
||||||
|
/>
|
||||||
|
</RNTesterPage>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
_renderItem: function({item}): ?React.Element<any> {
|
||||||
|
return (
|
||||||
|
<View style={styles.row}>
|
||||||
|
<Image style={styles.rowIcon} source={item.icon} />
|
||||||
|
<View style={styles.rowData}>
|
||||||
|
<Text style={styles.rowDataText}>
|
||||||
|
{item.data}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
</View>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
_renderQuickActions: function({item}: Object): ?React.Element<any> {
|
||||||
|
return (
|
||||||
|
<View style={styles.actionsContainer}>
|
||||||
|
<TouchableHighlight
|
||||||
|
style={styles.actionButton}
|
||||||
|
onPress={() => {
|
||||||
|
Alert.alert(
|
||||||
|
'Tips',
|
||||||
|
'You could do something with this edit action!',
|
||||||
|
);
|
||||||
|
}}>
|
||||||
|
<Text style={styles.actionButtonText}>Edit</Text>
|
||||||
|
</TouchableHighlight>
|
||||||
|
<TouchableHighlight
|
||||||
|
style={[styles.actionButton, styles.actionButtonDestructive]}
|
||||||
|
onPress={() => {
|
||||||
|
Alert.alert(
|
||||||
|
'Tips',
|
||||||
|
'You could do something with this remove action!',
|
||||||
|
);
|
||||||
|
}}>
|
||||||
|
<Text style={styles.actionButtonText}>Remove</Text>
|
||||||
|
</TouchableHighlight>
|
||||||
|
</View>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
var styles = StyleSheet.create({
|
||||||
|
row: {
|
||||||
|
flexDirection: 'row',
|
||||||
|
justifyContent: 'center',
|
||||||
|
alignItems: 'center',
|
||||||
|
padding: 10,
|
||||||
|
backgroundColor: '#F6F6F6',
|
||||||
|
},
|
||||||
|
rowIcon: {
|
||||||
|
width: 64,
|
||||||
|
height: 64,
|
||||||
|
marginRight: 20,
|
||||||
|
},
|
||||||
|
rowData: {
|
||||||
|
flex: 1,
|
||||||
|
},
|
||||||
|
rowDataText: {
|
||||||
|
fontSize: 24,
|
||||||
|
},
|
||||||
|
actionsContainer: {
|
||||||
|
flex: 1,
|
||||||
|
flexDirection: 'row',
|
||||||
|
justifyContent: 'flex-end',
|
||||||
|
alignItems: 'center',
|
||||||
|
},
|
||||||
|
actionButton: {
|
||||||
|
padding: 10,
|
||||||
|
width: 80,
|
||||||
|
backgroundColor: '#999999',
|
||||||
|
},
|
||||||
|
actionButtonDestructive: {
|
||||||
|
backgroundColor: '#FF0000',
|
||||||
|
},
|
||||||
|
actionButtonText: {
|
||||||
|
textAlign: 'center',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
module.exports = SwipeableFlatListExample;
|
Loading…
x
Reference in New Issue
Block a user