react-native-gesture-handler
Version:
Declarative API exposing native platform touch and gesture system to React Native
199 lines (197 loc) • 6.44 kB
JavaScript
"use strict";
import * as React from 'react';
import { Animated, Platform, processColor, StyleSheet } from 'react-native';
import createNativeWrapper from '../handlers/createNativeWrapper';
import GestureHandlerButton from './GestureHandlerButton';
import { State } from '../State';
import { isFabric } from '../utils';
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
export const RawButton = createNativeWrapper(GestureHandlerButton, {
shouldCancelWhenOutside: false,
shouldActivateOnStart: false
});
let IS_FABRIC = null;
class InnerBaseButton extends React.Component {
static defaultProps = {
delayLongPress: 600
};
constructor(props) {
super(props);
this.lastActive = false;
this.longPressDetected = 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 (!this.longPressDetected && oldState === State.ACTIVE && state !== State.CANCELLED && this.lastActive && this.props.onPress) {
this.props.onPress(pointerInside);
}
if (!this.lastActive &&
// NativeViewGestureHandler sends different events based on platform
state === (Platform.OS !== 'android' ? State.ACTIVE : State.BEGAN) && pointerInside) {
this.longPressDetected = false;
if (this.props.onLongPress) {
this.longPressTimeout = setTimeout(this.onLongPress, this.props.delayLongPress);
}
} else if (
// Cancel longpress timeout if it's set and the finger moved out of the view
state === State.ACTIVE && !pointerInside && this.longPressTimeout !== undefined) {
clearTimeout(this.longPressTimeout);
this.longPressTimeout = undefined;
} else if (
// Cancel longpress timeout if it's set and the gesture has finished
this.longPressTimeout !== undefined && (state === State.END || state === State.CANCELLED || state === State.FAILED)) {
clearTimeout(this.longPressTimeout);
this.longPressTimeout = undefined;
}
this.lastActive = active;
};
onLongPress = () => {
this.longPressDetected = true;
this.props.onLongPress?.();
};
// 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?.(e);
this.handleEvent(e);
};
onGestureEvent = e => {
this.props.onGestureEvent?.(e);
this.handleEvent(e); // TODO: maybe it is not correct
};
render() {
const {
rippleColor: unprocessedRippleColor,
style,
...rest
} = this.props;
if (IS_FABRIC === null) {
IS_FABRIC = isFabric();
}
const rippleColor = IS_FABRIC ? unprocessedRippleColor : processColor(unprocessedRippleColor ?? undefined);
return /*#__PURE__*/_jsx(RawButton, {
ref: this.props.innerRef,
rippleColor: rippleColor,
style: [style, Platform.OS === 'ios' && {
cursor: undefined
}],
...rest,
onGestureEvent: this.onGestureEvent,
onHandlerStateChange: this.onHandlerStateChange
});
}
}
const AnimatedInnerBaseButton = Animated.createAnimatedComponent(InnerBaseButton);
export const BaseButton = /*#__PURE__*/React.forwardRef((props, ref) => /*#__PURE__*/_jsx(InnerBaseButton, {
innerRef: ref,
...props
}));
const AnimatedBaseButton = /*#__PURE__*/React.forwardRef((props, ref) => /*#__PURE__*/_jsx(AnimatedInnerBaseButton, {
innerRef: ref,
...props
}));
const btnStyles = StyleSheet.create({
underlay: {
position: 'absolute',
left: 0,
right: 0,
bottom: 0,
top: 0
}
});
class InnerRectButton extends React.Component {
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?.(active);
};
render() {
const {
children,
style,
...rest
} = this.props;
const resolvedStyle = StyleSheet.flatten(style) ?? {};
return /*#__PURE__*/_jsxs(BaseButton, {
...rest,
ref: this.props.innerRef,
style: resolvedStyle,
onActiveStateChange: this.onActiveStateChange,
children: [/*#__PURE__*/_jsx(Animated.View, {
style: [btnStyles.underlay, {
opacity: this.opacity,
backgroundColor: this.props.underlayColor,
borderRadius: resolvedStyle.borderRadius,
borderTopLeftRadius: resolvedStyle.borderTopLeftRadius,
borderTopRightRadius: resolvedStyle.borderTopRightRadius,
borderBottomLeftRadius: resolvedStyle.borderBottomLeftRadius,
borderBottomRightRadius: resolvedStyle.borderBottomRightRadius
}]
}), children]
});
}
}
export const RectButton = /*#__PURE__*/React.forwardRef((props, ref) => /*#__PURE__*/_jsx(InnerRectButton, {
innerRef: ref,
...props
}));
class InnerBorderlessButton extends React.Component {
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?.(active);
};
render() {
const {
children,
style,
innerRef,
...rest
} = this.props;
return /*#__PURE__*/_jsx(AnimatedBaseButton, {
...rest,
innerRef: innerRef,
onActiveStateChange: this.onActiveStateChange,
style: [style, Platform.OS === 'ios' && {
opacity: this.opacity
}],
children: children
});
}
}
export const BorderlessButton = /*#__PURE__*/React.forwardRef((props, ref) => /*#__PURE__*/_jsx(InnerBorderlessButton, {
innerRef: ref,
...props
}));
export { default as PureNativeButton } from './GestureHandlerButton';
//# sourceMappingURL=GestureButtons.js.map