2016-05-04 15:49:02 -07:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* The examples provided by Facebook are for non-commercial testing and
|
|
|
|
* evaluation purposes only.
|
|
|
|
*
|
|
|
|
* Facebook reserves all rights not expressly granted.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
|
|
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NON INFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* FACEBOOK BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
|
|
|
|
* AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. *
|
|
|
|
*
|
|
|
|
* @providesModule SwipeableRow
|
|
|
|
* @flow
|
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const Animated = require('Animated');
|
|
|
|
const PanResponder = require('PanResponder');
|
|
|
|
const React = require('React');
|
|
|
|
const StyleSheet = require('StyleSheet');
|
|
|
|
const View = require('View');
|
|
|
|
|
|
|
|
const {PropTypes} = React;
|
|
|
|
|
2016-06-06 13:47:21 -07:00
|
|
|
const emptyFunction = require('fbjs/lib/emptyFunction');
|
2016-05-16 20:21:47 -07:00
|
|
|
|
2016-05-04 15:49:02 -07:00
|
|
|
// Position of the left of the swipable item when closed
|
|
|
|
const CLOSED_LEFT_POSITION = 0;
|
2016-05-04 19:50:09 -07:00
|
|
|
// Minimum swipe distance before we recognize it as such
|
|
|
|
const HORIZONTAL_SWIPE_DISTANCE_THRESHOLD = 15;
|
2016-06-06 14:49:23 -07:00
|
|
|
// Distance left of closed position to bounce back when right-swiping from closed
|
|
|
|
const RIGHT_SWIPE_BOUNCE_BACK_DISTANCE = 30;
|
|
|
|
// Factor to divide by to get slow speed; i.e. 4 means 1/4 of full speed
|
|
|
|
const SLOW_SPEED_SWIPE_FACTOR = 4;
|
|
|
|
/**
|
|
|
|
* Max distance of right swipe to allow (right swipes do functionally nothing).
|
|
|
|
* Must be multiplied by SLOW_SPEED_SWIPE_FACTOR because gestureState.dx tracks
|
|
|
|
* how far the finger swipes, and not the actual animation distance.
|
|
|
|
*/
|
|
|
|
const RIGHT_SWIPE_THRESHOLD = 30 * SLOW_SPEED_SWIPE_FACTOR;
|
2016-05-23 10:21:07 -07:00
|
|
|
// Time, in milliseconds, of how long the animated swipe should be
|
|
|
|
const SWIPE_DURATION = 200;
|
2016-05-04 15:49:02 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a swipable row that allows taps on the main item and a custom View
|
|
|
|
* on the item hidden behind the row
|
|
|
|
*/
|
|
|
|
const SwipeableRow = React.createClass({
|
|
|
|
_panResponder: {},
|
|
|
|
_previousLeft: CLOSED_LEFT_POSITION,
|
|
|
|
|
|
|
|
propTypes: {
|
2016-05-05 11:48:25 -07:00
|
|
|
isOpen: PropTypes.bool,
|
2016-05-19 21:39:58 -07:00
|
|
|
maxSwipeDistance: PropTypes.number.isRequired,
|
2016-06-06 14:49:23 -07:00
|
|
|
onOpen: PropTypes.func.isRequired,
|
2016-05-16 20:21:47 -07:00
|
|
|
onSwipeEnd: PropTypes.func.isRequired,
|
|
|
|
onSwipeStart: PropTypes.func.isRequired,
|
2016-05-04 15:49:02 -07:00
|
|
|
/**
|
|
|
|
* A ReactElement that is unveiled when the user swipes
|
|
|
|
*/
|
|
|
|
slideoutView: PropTypes.node.isRequired,
|
|
|
|
/**
|
|
|
|
* The minimum swipe distance required before fully animating the swipe. If
|
|
|
|
* the user swipes less than this distance, the item will return to its
|
|
|
|
* previous (open/close) position.
|
|
|
|
*/
|
|
|
|
swipeThreshold: PropTypes.number.isRequired,
|
|
|
|
},
|
|
|
|
|
|
|
|
getInitialState(): Object {
|
|
|
|
return {
|
|
|
|
currentLeft: new Animated.Value(this._previousLeft),
|
2016-05-16 20:21:47 -07:00
|
|
|
/**
|
|
|
|
* In order to render component A beneath component B, A must be rendered
|
|
|
|
* before B. However, this will cause "flickering", aka we see A briefly
|
|
|
|
* then B. To counter this, _isSwipeableViewRendered flag is used to set
|
|
|
|
* component A to be transparent until component B is loaded.
|
|
|
|
*/
|
|
|
|
isSwipeableViewRendered: false,
|
2016-05-19 21:40:03 -07:00
|
|
|
rowHeight: (null: ?number),
|
2016-05-04 15:49:02 -07:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
getDefaultProps(): Object {
|
|
|
|
return {
|
2016-05-05 11:48:25 -07:00
|
|
|
isOpen: false,
|
2016-05-19 21:39:58 -07:00
|
|
|
maxSwipeDistance: 0,
|
2016-06-06 14:49:23 -07:00
|
|
|
onOpen: emptyFunction,
|
2016-05-16 20:21:47 -07:00
|
|
|
onSwipeEnd: emptyFunction,
|
|
|
|
onSwipeStart: emptyFunction,
|
|
|
|
swipeThreshold: 30,
|
2016-05-04 15:49:02 -07:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
componentWillMount(): void {
|
|
|
|
this._panResponder = PanResponder.create({
|
2016-05-16 20:21:47 -07:00
|
|
|
onStartShouldSetPanResponder: (event, gestureState) => true,
|
|
|
|
// Don't capture child's start events
|
|
|
|
onStartShouldSetPanResponderCapture: (event, gestureState) => false,
|
|
|
|
onMoveShouldSetPanResponder: (event, gestureState) => false,
|
2016-05-04 19:50:09 -07:00
|
|
|
onMoveShouldSetPanResponderCapture: this._handleMoveShouldSetPanResponderCapture,
|
2016-05-16 20:21:47 -07:00
|
|
|
onPanResponderGrant: this._handlePanResponderGrant,
|
2016-05-04 15:49:02 -07:00
|
|
|
onPanResponderMove: this._handlePanResponderMove,
|
|
|
|
onPanResponderRelease: this._handlePanResponderEnd,
|
2016-05-16 20:21:47 -07:00
|
|
|
onPanResponderTerminationRequest: this._onPanResponderTerminationRequest,
|
2016-05-04 15:49:02 -07:00
|
|
|
onPanResponderTerminate: this._handlePanResponderEnd,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2016-05-05 11:48:25 -07:00
|
|
|
componentWillReceiveProps(nextProps: Object): void {
|
|
|
|
/**
|
|
|
|
* We do not need an "animateOpen(noCallback)" because this animation is
|
|
|
|
* handled internally by this component.
|
|
|
|
*/
|
|
|
|
if (this.props.isOpen && !nextProps.isOpen) {
|
2016-05-16 20:21:47 -07:00
|
|
|
this._animateToClosedPosition();
|
2016-05-05 11:48:25 -07:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2016-05-24 18:20:12 -07:00
|
|
|
render(): ReactElement<any> {
|
2016-05-04 15:49:02 -07:00
|
|
|
// The view hidden behind the main view
|
2016-05-19 21:39:58 -07:00
|
|
|
let slideOutView;
|
|
|
|
if (this.state.isSwipeableViewRendered) {
|
|
|
|
slideOutView = (
|
2016-05-19 21:40:03 -07:00
|
|
|
<View style={[
|
|
|
|
styles.slideOutContainer,
|
|
|
|
{height: this.state.rowHeight},
|
|
|
|
]}>
|
2016-05-19 21:39:58 -07:00
|
|
|
{this.props.slideoutView}
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
}
|
2016-05-04 15:49:02 -07:00
|
|
|
|
2016-05-19 21:39:58 -07:00
|
|
|
// The swipeable item
|
2016-05-05 20:44:48 -07:00
|
|
|
const swipeableView = (
|
2016-05-04 15:49:02 -07:00
|
|
|
<Animated.View
|
2016-05-05 20:44:48 -07:00
|
|
|
onLayout={this._onSwipeableViewLayout}
|
2016-05-19 21:39:58 -07:00
|
|
|
style={[
|
|
|
|
styles.swipeableContainer,
|
|
|
|
{
|
|
|
|
transform: [{translateX: this.state.currentLeft}],
|
|
|
|
},
|
|
|
|
]}>
|
2016-05-04 15:49:02 -07:00
|
|
|
{this.props.children}
|
|
|
|
</Animated.View>
|
|
|
|
);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<View
|
2016-05-19 21:40:03 -07:00
|
|
|
{...this._panResponder.panHandlers}>
|
2016-05-04 15:49:02 -07:00
|
|
|
{slideOutView}
|
2016-05-05 20:44:48 -07:00
|
|
|
{swipeableView}
|
2016-05-04 15:49:02 -07:00
|
|
|
</View>
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
2016-05-05 20:44:48 -07:00
|
|
|
_onSwipeableViewLayout(event: Object): void {
|
2016-05-19 21:39:58 -07:00
|
|
|
if (!this.state.isSwipeableViewRendered) {
|
2016-05-16 20:21:47 -07:00
|
|
|
this.setState({
|
|
|
|
isSwipeableViewRendered: true,
|
2016-05-19 21:40:03 -07:00
|
|
|
rowHeight: event.nativeEvent.layout.height,
|
2016-05-16 20:21:47 -07:00
|
|
|
});
|
2016-05-05 20:44:48 -07:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2016-05-04 19:50:09 -07:00
|
|
|
_handleMoveShouldSetPanResponderCapture(
|
|
|
|
event: Object,
|
|
|
|
gestureState: Object,
|
|
|
|
): boolean {
|
2016-05-16 20:21:47 -07:00
|
|
|
// Decides whether a swipe is responded to by this component or its child
|
2016-05-31 16:23:51 -07:00
|
|
|
return gestureState.dy < 10 && this._isValidSwipe(gestureState);
|
2016-05-04 15:49:02 -07:00
|
|
|
},
|
|
|
|
|
2016-05-16 20:21:47 -07:00
|
|
|
_handlePanResponderGrant(event: Object, gestureState: Object): void {
|
2016-05-04 15:49:02 -07:00
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
_handlePanResponderMove(event: Object, gestureState: Object): void {
|
2016-06-06 14:49:23 -07:00
|
|
|
if (this._isSwipingExcessivelyRightFromClosedPosition(gestureState)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-05-16 20:21:47 -07:00
|
|
|
this.props.onSwipeStart();
|
2016-05-31 16:23:51 -07:00
|
|
|
|
2016-06-06 14:49:23 -07:00
|
|
|
if (this._isSwipingRightFromClosed(gestureState)) {
|
|
|
|
this._swipeSlowSpeed(gestureState);
|
|
|
|
} else {
|
|
|
|
this._swipeFullSpeed(gestureState);
|
2016-05-31 16:23:51 -07:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2016-06-06 14:49:23 -07:00
|
|
|
_isSwipingRightFromClosed(gestureState: Object): boolean {
|
2016-05-31 16:23:51 -07:00
|
|
|
return this._previousLeft === CLOSED_LEFT_POSITION && gestureState.dx > 0;
|
2016-05-04 15:49:02 -07:00
|
|
|
},
|
|
|
|
|
2016-06-06 14:49:23 -07:00
|
|
|
_swipeFullSpeed(gestureState: Object): void {
|
|
|
|
this.state.currentLeft.setValue(this._previousLeft + gestureState.dx);
|
|
|
|
},
|
|
|
|
|
|
|
|
_swipeSlowSpeed(gestureState: Object): void {
|
|
|
|
this.state.currentLeft.setValue(
|
|
|
|
this._previousLeft + gestureState.dx / SLOW_SPEED_SWIPE_FACTOR,
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
|
|
|
_isSwipingExcessivelyRightFromClosedPosition(gestureState: Object): boolean {
|
|
|
|
/**
|
|
|
|
* We want to allow a BIT of right swipe, to allow users to know that
|
|
|
|
* swiping is available, but swiping right does not do anything
|
|
|
|
* functionally.
|
|
|
|
*/
|
|
|
|
return (
|
|
|
|
this._isSwipingRightFromClosed(gestureState) &&
|
|
|
|
gestureState.dx > RIGHT_SWIPE_THRESHOLD
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
|
|
|
_onPanResponderTerminationRequest(
|
|
|
|
event: Object,
|
|
|
|
gestureState: Object,
|
|
|
|
): boolean {
|
2016-05-16 20:21:47 -07:00
|
|
|
return false;
|
2016-05-04 15:49:02 -07:00
|
|
|
},
|
|
|
|
|
2016-06-06 14:49:23 -07:00
|
|
|
_animateTo(toValue: number, callback: Function = emptyFunction): void {
|
2016-05-16 20:21:47 -07:00
|
|
|
Animated.timing(
|
|
|
|
this.state.currentLeft,
|
|
|
|
{
|
2016-05-23 10:21:07 -07:00
|
|
|
duration: SWIPE_DURATION,
|
2016-05-16 20:21:47 -07:00
|
|
|
toValue: toValue,
|
|
|
|
},
|
|
|
|
).start(() => {
|
2016-05-04 15:49:02 -07:00
|
|
|
this._previousLeft = toValue;
|
2016-06-06 14:49:23 -07:00
|
|
|
callback();
|
2016-05-04 15:49:02 -07:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2016-05-16 20:21:47 -07:00
|
|
|
_animateToOpenPosition(): void {
|
2016-05-19 21:39:58 -07:00
|
|
|
this._animateTo(-this.props.maxSwipeDistance);
|
2016-05-04 15:49:02 -07:00
|
|
|
},
|
|
|
|
|
2016-05-16 20:21:47 -07:00
|
|
|
_animateToClosedPosition(): void {
|
2016-05-04 15:49:02 -07:00
|
|
|
this._animateTo(CLOSED_LEFT_POSITION);
|
|
|
|
},
|
|
|
|
|
2016-06-06 14:49:23 -07:00
|
|
|
_animateRightSwipeBounceBack(): void {
|
|
|
|
/**
|
|
|
|
* When swiping right, we want to bounce back past closed position on release
|
|
|
|
* so users know they should swipe right to get content.
|
|
|
|
*/
|
|
|
|
this._animateTo(
|
|
|
|
-RIGHT_SWIPE_BOUNCE_BACK_DISTANCE,
|
|
|
|
this._animateToClosedPosition,
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
2016-05-16 20:21:47 -07:00
|
|
|
// Ignore swipes due to user's finger moving slightly when tapping
|
|
|
|
_isValidSwipe(gestureState: Object): boolean {
|
|
|
|
return Math.abs(gestureState.dx) > HORIZONTAL_SWIPE_DISTANCE_THRESHOLD;
|
|
|
|
},
|
|
|
|
|
2016-05-04 15:49:02 -07:00
|
|
|
_handlePanResponderEnd(event: Object, gestureState: Object): void {
|
|
|
|
const horizontalDistance = gestureState.dx;
|
|
|
|
|
2016-06-06 14:49:23 -07:00
|
|
|
if (this._isSwipingRightFromClosed(gestureState)) {
|
|
|
|
this.props.onOpen();
|
|
|
|
this._animateRightSwipeBounceBack();
|
|
|
|
} else if (Math.abs(horizontalDistance) > this.props.swipeThreshold) {
|
|
|
|
// Overswiped
|
|
|
|
|
2016-05-16 20:21:47 -07:00
|
|
|
if (horizontalDistance < 0) {
|
|
|
|
// Swiped left
|
2016-06-06 14:49:23 -07:00
|
|
|
this.props.onOpen();
|
2016-05-16 20:21:47 -07:00
|
|
|
this._animateToOpenPosition();
|
|
|
|
} else {
|
|
|
|
// Swiped right
|
|
|
|
this._animateToClosedPosition();
|
|
|
|
}
|
|
|
|
} else {
|
2016-06-06 14:49:23 -07:00
|
|
|
// Swiping from closed but let go before fully
|
|
|
|
|
2016-05-16 20:21:47 -07:00
|
|
|
if (this._previousLeft === CLOSED_LEFT_POSITION) {
|
|
|
|
this._animateToClosedPosition();
|
|
|
|
} else {
|
|
|
|
this._animateToOpenPosition();
|
|
|
|
}
|
2016-05-04 15:49:02 -07:00
|
|
|
}
|
|
|
|
|
2016-05-16 20:21:47 -07:00
|
|
|
this.props.onSwipeEnd();
|
2016-05-04 15:49:02 -07:00
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
const styles = StyleSheet.create({
|
|
|
|
slideOutContainer: {
|
2016-05-17 01:35:47 -07:00
|
|
|
bottom: 0,
|
|
|
|
left: 0,
|
|
|
|
position: 'absolute',
|
|
|
|
right: 0,
|
|
|
|
top: 0,
|
2016-05-04 15:49:02 -07:00
|
|
|
},
|
2016-05-19 21:39:58 -07:00
|
|
|
swipeableContainer: {
|
|
|
|
flex: 1,
|
|
|
|
},
|
2016-05-04 15:49:02 -07:00
|
|
|
});
|
|
|
|
|
|
|
|
module.exports = SwipeableRow;
|