GT2/GT2-Android/node_modules/react-native-gesture-handler/GestureHandler.js

614 lines
17 KiB
JavaScript

import React from 'react';
import {
findNodeHandle,
requireNativeComponent,
Animated,
NativeModules,
ScrollView,
Slider,
Switch,
TextInput,
ToolbarAndroid,
ViewPagerAndroid,
DrawerLayoutAndroid,
WebView,
StyleSheet,
FlatList,
Platform,
} from 'react-native';
import ReactNativeBridgeEventPlugin from 'react-native/Libraries/Renderer/shims/ReactNativeBridgeEventPlugin';
import deepEqual from 'fbjs/lib/areEqual';
import PropTypes from 'prop-types';
import gestureHandlerRootHOC from './gestureHandlerRootHOC';
import Swipeable from './Swipeable';
import DrawerLayout from './DrawerLayout';
const RNGestureHandlerModule = NativeModules.RNGestureHandlerModule;
/* Wrap JS responder calls and notify gesture handler manager */
const { UIManager } = NativeModules;
const {
setJSResponder: oldSetJSResponder,
clearJSResponder: oldClearJSResponder,
} = UIManager;
UIManager.setJSResponder = (tag, blockNativeResponder) => {
RNGestureHandlerModule.handleSetJSResponder(tag, blockNativeResponder);
oldSetJSResponder(tag, blockNativeResponder);
};
UIManager.clearJSResponder = () => {
RNGestureHandlerModule.handleClearJSResponder();
oldClearJSResponder();
};
ReactNativeBridgeEventPlugin.processEventTypes({
directEventTypes: {
topGestureHandlerEvent: { registrationName: 'onGestureHandlerEvent' },
topGestureHandlerStateChange: {
registrationName: 'onGestureHandlerStateChange',
},
},
});
const State = RNGestureHandlerModule.State;
let handlerTag = 1;
const handlerIDToTag = {};
const GestureHandlerPropTypes = {
id: PropTypes.string,
enabled: PropTypes.bool,
waitFor: PropTypes.oneOfType([
PropTypes.string,
PropTypes.arrayOf(PropTypes.string),
]),
simultaneousHandlers: PropTypes.oneOfType([
PropTypes.string,
PropTypes.arrayOf(PropTypes.string),
]),
shouldCancelWhenOutside: PropTypes.bool,
hitSlop: PropTypes.oneOfType([
PropTypes.number,
PropTypes.shape({
left: PropTypes.number,
top: PropTypes.number,
right: PropTypes.number,
bottom: PropTypes.number,
vertical: PropTypes.number,
horizontal: PropTypes.number,
}),
]),
onGestureEvent: PropTypes.oneOfType([PropTypes.func, PropTypes.object]),
onHandlerStateChange: PropTypes.func,
onBegan: PropTypes.func,
onFailed: PropTypes.func,
onCancelled: PropTypes.func,
onActivated: PropTypes.func,
onEnded: PropTypes.func,
};
const stateToPropMappings = {
[State.BEGAN]: 'onBegan',
[State.FAILED]: 'onFailed',
[State.CANCELLED]: 'onCancelled',
[State.ACTIVE]: 'onActivated',
[State.END]: 'onEnded',
};
function canUseNativeParam(param) {
return (
param !== undefined &&
typeof param !== 'function' &&
(typeof param !== 'object' || !('__isNative' in param))
);
}
function transformIntoHandlerTags(handlerIDs) {
if (!Array.isArray(handlerIDs)) {
handlerIDs = [handlerIDs];
}
// converts handler string IDs into their numeric tags
return handlerIDs
.map(handlerID => handlerIDToTag[handlerID] || -1)
.filter(handlerTag => handlerTag > 0);
}
function filterConfig(props, validProps, defaults = {}) {
const res = { ...defaults };
Object.keys(validProps).forEach(key => {
const value = props[key];
if (canUseNativeParam(value)) {
let value = props[key];
if (key === 'simultaneousHandlers' || key === 'waitFor') {
value = transformIntoHandlerTags(props[key]);
} else if (key === 'hitSlop') {
if (typeof value !== 'object') {
value = { top: value, left: value, bottom: value, right: value };
}
}
res[key] = value;
}
});
return res;
}
function createHandler(handlerName, propTypes = null, config = {}) {
class Handler extends React.Component {
static propTypes = {
...GestureHandlerPropTypes,
...propTypes,
};
constructor(props) {
super(props);
this._handlerTag = handlerTag++;
this._config = {};
if (props.id) {
if (handlerIDToTag[props.id] !== undefined) {
throw new Error(`Handler with ID "${props.id}" already registered`);
}
handlerIDToTag[props.id] = this._handlerTag;
}
}
_onGestureHandlerEvent = event => {
if (event.nativeEvent.handlerTag === this._handlerTag) {
this.props.onGestureEvent && this.props.onGestureEvent(event);
} else {
this.props.onGestureHandlerEvent &&
this.props.onGestureHandlerEvent(event);
}
};
_onGestureHandlerStateChange = event => {
if (event.nativeEvent.handlerTag === this._handlerTag) {
this.props.onHandlerStateChange &&
this.props.onHandlerStateChange(event);
const stateEventName = stateToPropMappings[event.nativeEvent.state];
if (typeof this.props[stateEventName] === 'function') {
this.props[stateEventName](event);
}
} else {
this.props.onGestureHandlerStateChange &&
this.props.onGestureHandlerStateChange(event);
}
};
_refHandler = node => {
this._viewNode = node;
const child = React.Children.only(this.props.children);
const { ref } = child;
if (typeof ref === 'function') {
ref(node);
}
};
componentWillUnmount() {
RNGestureHandlerModule.dropGestureHandler(this._handlerTag);
if (this.props.id) {
delete handlerIDToTag[this.props.id];
}
}
componentDidMount() {
this._viewTag = findNodeHandle(this._viewNode);
this._config = filterConfig(
this.props,
this.constructor.propTypes,
config
);
RNGestureHandlerModule.createGestureHandler(
handlerName,
this._handlerTag,
this._config
);
RNGestureHandlerModule.attachGestureHandler(
this._handlerTag,
this._viewTag
);
}
componentDidUpdate(prevProps, prevState) {
const viewTag = findNodeHandle(this._viewNode);
if (this._viewTag !== viewTag) {
this._viewTag = viewTag;
RNGestureHandlerModule.attachGestureHandler(this._handlerTag, viewTag);
}
const newConfig = filterConfig(
this.props,
this.constructor.propTypes,
config
);
if (!deepEqual(this._config, newConfig)) {
this._config = newConfig;
RNGestureHandlerModule.updateGestureHandler(
this._handlerTag,
this._config
);
}
}
render() {
let gestureEventHandler = this._onGestureHandlerEvent;
const { onGestureEvent, onGestureHandlerEvent } = this.props;
if (onGestureEvent && typeof onGestureEvent !== 'function') {
// If it's not a mathod it should be an native Animated.event
// object. We set it directly as the handler for the view
// In this case nested handlers are not going to be supported
if (onGestureHandlerEvent) {
throw new Error(
'Nesting touch handlers with native animated driver is not supported yet'
);
}
gestureEventHandler = this.props.onGestureEvent;
} else {
if (
onGestureHandlerEvent &&
typeof onGestureHandlerEvent !== 'function'
) {
throw new Error(
'Nesting touch handlers with native animated driver is not supported yet'
);
}
}
let gestureStateEventHandler = this._onGestureHandlerStateChange;
const { onHandlerStateChange, onGestureHandlerStateChange } = this.props;
if (onHandlerStateChange && typeof onHandlerStateChange !== 'function') {
// If it's not a method it should be an native Animated.event
// object. We set it directly as the handler for the view
// In this case nested handlers are not going to be supported
if (onGestureHandlerStateChange) {
throw new Error(
'Nesting touch handlers with native animated driver is not supported yet'
);
}
gestureStateEventHandler = this.props.onHandlerStateChange;
} else {
if (
onGestureHandlerStateChange &&
typeof onGestureHandlerStateChange !== 'function'
) {
throw new Error(
'Nesting touch handlers with native animated driver is not supported yet'
);
}
}
const child = React.Children.only(this.props.children);
return React.cloneElement(child, {
ref: this._refHandler,
collapsable: false,
onGestureHandlerEvent: gestureEventHandler,
onGestureHandlerStateChange: gestureStateEventHandler,
});
}
}
return Handler;
}
const NativeViewGestureHandler = createHandler('NativeViewGestureHandler', {
shouldActivateOnStart: PropTypes.bool,
disallowInterruption: PropTypes.bool,
});
const TapGestureHandler = createHandler(
'TapGestureHandler',
{
maxDurationMs: PropTypes.number,
maxDelayMs: PropTypes.number,
numberOfTaps: PropTypes.number,
},
{}
);
const LongPressGestureHandler = createHandler(
'LongPressGestureHandler',
{
minDurationMs: PropTypes.number,
maxDist: PropTypes.number,
},
{}
);
const PanGestureHandler = createHandler(
'PanGestureHandler',
{
minDeltaX: PropTypes.number,
minDeltaY: PropTypes.number,
maxDeltaX: PropTypes.number,
maxDeltaY: PropTypes.number,
minOffsetX: PropTypes.number,
minOffsetY: PropTypes.number,
minDist: PropTypes.number,
minVelocity: PropTypes.number,
minVelocityX: PropTypes.number,
minVelocityY: PropTypes.number,
minPointers: PropTypes.number,
maxPointers: PropTypes.number,
avgTouches: PropTypes.bool,
},
{}
);
const PinchGestureHandler = createHandler('PinchGestureHandler', {}, {});
const RotationGestureHandler = createHandler('RotationGestureHandler', {}, {});
const NATIVE_WRAPPER_BIND_BLACKLIST = new Set(['replaceState', 'isMounted']);
const NATIVE_WRAPPER_PROPS_FILTER = {
// accept all gesture handler prop types plus native wrapper specific ones
...NativeViewGestureHandler.propTypes,
// we want to pass gesture event handlers if registered
onGestureHandlerEvent: PropTypes.func,
onGestureHandlerStateChange: PropTypes.func,
};
function createNativeWrapper(Component, config = {}) {
class ComponentWrapper extends React.Component {
static propTypes = {
...Component.propTypes,
};
_refHandler = node => {
// bind native component's methods
for (let methodName in node) {
const method = node[methodName];
if (
!methodName.startsWith('_') && // private methods
!methodName.startsWith('component') && // lifecycle methods
!NATIVE_WRAPPER_BIND_BLACKLIST.has(methodName) && // other
typeof method === 'function' &&
this[methodName] === undefined
) {
this[methodName] = method;
}
}
};
render() {
// filter out props that should be passed to gesture handler wrapper
const gestureHandlerProps = Object.keys(this.props).reduce(
(props, key) => {
if (key in NATIVE_WRAPPER_PROPS_FILTER) {
props[key] = this.props[key];
}
return props;
},
{ ...config } // watch out not to modify config
);
return (
<NativeViewGestureHandler {...gestureHandlerProps}>
<Component {...this.props} ref={this._refHandler} />
</NativeViewGestureHandler>
);
}
}
return ComponentWrapper;
}
const WrappedScrollView = createNativeWrapper(ScrollView, {
disallowInterruption: true,
});
const WrappedSlider = createNativeWrapper(Slider, {
shouldCancelWhenOutside: false,
shouldActivateOnStart: true,
disallowInterruption: true,
});
const WrappedSwitch = createNativeWrapper(Switch, {
shouldCancelWhenOutside: false,
shouldActivateOnStart: true,
disallowInterruption: true,
});
const WrappedTextInput = createNativeWrapper(TextInput);
const WrappedWebView = createNativeWrapper(WebView);
const WrappedToolbarAndroid = createNativeWrapper(ToolbarAndroid);
const WrappedViewPagerAndroid = createNativeWrapper(ViewPagerAndroid, {
disallowInterruption: true,
});
const WrappedDrawerLayoutAndroid = createNativeWrapper(DrawerLayoutAndroid, {
disallowInterruption: true,
});
WrappedDrawerLayoutAndroid.positions = DrawerLayoutAndroid.positions;
State.print = state => {
const keys = Object.keys(State);
for (let i = 0; i < keys.length; i++) {
if (state === State[keys[i]]) {
return keys[i];
}
}
};
const RawButton = createNativeWrapper(
requireNativeComponent('RNGestureHandlerButton', null),
{
shouldCancelWhenOutside: false,
shouldActivateOnStart: false,
}
);
/* Buttons */
class BaseButton extends React.Component {
static propTypes = {
...RawButton.propTypes,
onPress: PropTypes.func,
onActiveStateChange: PropTypes.func,
};
constructor(props) {
super(props);
this._lastActive = false;
}
_handleEvent = ({ nativeEvent }) => {
const { state, oldState, pointerInside } = nativeEvent;
const active = pointerInside && state === State.ACTIVE;
if (active !== this._lastActive && this.props.onActiveStateChange) {
this.props.onActiveStateChange(active);
}
if (
oldState === State.ACTIVE &&
state !== State.CANCELLED &&
this._lastActive &&
this.props.onPress
) {
this.props.onPress(active);
}
this._lastActive = active;
};
// Normally, the parent would execute it's handler first,
// then forward the event to listeners. However, here our handler
// is virtually only forwarding events to listeners, so we reverse the order
// to keep the proper order of the callbacks (from "raw" ones to "processed").
_onHandlerStateChange = e => {
this.props.onHandlerStateChange && this.props.onHandlerStateChange(e);
this._handleEvent(e);
};
_onGestureEvent = e => {
this.props.onGestureEvent && this.props.onGestureEvent(e);
this._handleEvent(e);
};
render() {
return (
<RawButton
{...this.props}
onGestureEvent={this._onGestureEvent}
onHandlerStateChange={this._onHandlerStateChange}
/>
);
}
}
const AnimatedBaseButton = Animated.createAnimatedComponent(BaseButton);
const btnStyles = StyleSheet.create({
underlay: {
position: 'absolute',
left: 0,
right: 0,
bottom: 0,
top: 0,
},
});
class RectButton extends React.Component {
static propTypes = BaseButton.propTypes;
static defaultProps = {
activeOpacity: 0.105,
underlayColor: 'black',
};
constructor(props) {
super(props);
this._opacity = new Animated.Value(0);
}
_onActiveStateChange = active => {
if (Platform.OS !== 'android') {
this._opacity.setValue(active ? this.props.activeOpacity : 0);
}
this.props.onActiveStateChange && this.props.onActiveStateChange(active);
};
render() {
const { children, ...rest } = this.props;
return (
<BaseButton {...rest} onActiveStateChange={this._onActiveStateChange}>
<Animated.View
style={[
btnStyles.underlay,
{ opacity: this._opacity },
{ backgroundColor: this.props.underlayColor },
]}
/>
{children}
</BaseButton>
);
}
}
class BorderlessButton extends React.Component {
static propTypes = {
...BaseButton.propTypes,
borderless: PropTypes.bool,
};
static defaultProps = {
activeOpacity: 0.3,
borderless: true,
};
constructor(props) {
super(props);
this._opacity = new Animated.Value(1);
}
_onActiveStateChange = active => {
if (Platform.OS !== 'android') {
this._opacity.setValue(active ? this.props.activeOpacity : 1);
}
this.props.onActiveStateChange && this.props.onActiveStateChange(active);
};
render() {
const { children, style, ...rest } = this.props;
return (
<AnimatedBaseButton
{...rest}
onActiveStateChange={this._onActiveStateChange}
style={[style, Platform.OS === 'ios' && { opacity: this._opacity }]}>
{children}
</AnimatedBaseButton>
);
}
}
/* Other */
const FlatListWithGHScroll = props => (
<FlatList
{...props}
renderScrollComponent={props => <WrappedScrollView {...props} />}
/>
);
export {
WrappedScrollView as ScrollView,
WrappedSlider as Slider,
WrappedSwitch as Switch,
WrappedTextInput as TextInput,
WrappedToolbarAndroid as ToolbarAndroid,
WrappedViewPagerAndroid as ViewPagerAndroid,
WrappedDrawerLayoutAndroid as DrawerLayoutAndroid,
WrappedWebView as WebView,
NativeViewGestureHandler,
TapGestureHandler,
LongPressGestureHandler,
PanGestureHandler,
PinchGestureHandler,
RotationGestureHandler,
State,
/* Buttons */
RawButton,
BaseButton,
RectButton,
BorderlessButton,
/* Other */
FlatListWithGHScroll as FlatList,
gestureHandlerRootHOC,
Swipeable,
DrawerLayout,
};