create-expo-cljs-app
Version:
Create a react native application with Expo and Shadow-CLJS!
426 lines (323 loc) • 16.2 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.GestureDetector = void 0;
var _react = _interopRequireWildcard(require("react"));
var _gesture = require("./gesture");
var _reanimatedWrapper = require("./reanimatedWrapper");
var _handlersRegistry = require("../handlersRegistry");
var _RNGestureHandlerModule = _interopRequireDefault(require("../../RNGestureHandlerModule"));
var _gestureHandlerCommon = require("../gestureHandlerCommon");
var _gestureStateManager = require("./gestureStateManager");
var _FlingGestureHandler = require("../FlingGestureHandler");
var _ForceTouchGestureHandler = require("../ForceTouchGestureHandler");
var _LongPressGestureHandler = require("../LongPressGestureHandler");
var _PanGestureHandler = require("../PanGestureHandler");
var _TapGestureHandler = require("../TapGestureHandler");
var _State = require("../../State");
var _EventType = require("../../EventType");
var _Reanimated$default$c, _Reanimated$default;
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function () { return cache; }; return cache; }
function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
const ALLOWED_PROPS = [..._gestureHandlerCommon.baseGestureHandlerWithMonitorProps, ..._TapGestureHandler.tapGestureHandlerProps, ..._PanGestureHandler.panGestureHandlerProps, ..._PanGestureHandler.panGestureHandlerCustomNativeProps, ..._LongPressGestureHandler.longPressGestureHandlerProps, ..._ForceTouchGestureHandler.forceTouchGestureHandlerProps, ..._FlingGestureHandler.flingGestureHandlerProps];
function convertToHandlerTag(ref) {
if (typeof ref === 'number') {
return ref;
} else if (ref instanceof _gesture.BaseGesture) {
return ref.handlerTag;
} else {
var _ref$current$handlerT, _ref$current;
// @ts-ignore in this case it should be a ref either to gesture object or
// a gesture handler component, in both cases handlerTag property exists
return (_ref$current$handlerT = (_ref$current = ref.current) === null || _ref$current === void 0 ? void 0 : _ref$current.handlerTag) !== null && _ref$current$handlerT !== void 0 ? _ref$current$handlerT : -1;
}
}
function extractValidHandlerTags(interactionGroup) {
var _interactionGroup$map, _interactionGroup$map2;
return (_interactionGroup$map = interactionGroup === null || interactionGroup === void 0 ? void 0 : (_interactionGroup$map2 = interactionGroup.map(convertToHandlerTag)) === null || _interactionGroup$map2 === void 0 ? void 0 : _interactionGroup$map2.filter(tag => tag > 0)) !== null && _interactionGroup$map !== void 0 ? _interactionGroup$map : [];
}
function dropHandlers(preparedGesture) {
for (const handler of preparedGesture.config) {
_RNGestureHandlerModule.default.dropGestureHandler(handler.handlerTag);
(0, _handlersRegistry.unregisterHandler)(handler.handlerTag);
}
}
function attachHandlers({
preparedGesture,
gestureConfig,
gesture,
viewTag,
useAnimated
}) {
if (!preparedGesture.firstExecution) {
gestureConfig === null || gestureConfig === void 0 ? void 0 : gestureConfig.initialize();
} else {
preparedGesture.firstExecution = false;
} // use setImmediate to extract handlerTags, because all refs should be initialized
// when it's ran
setImmediate(() => {
gestureConfig === null || gestureConfig === void 0 ? void 0 : gestureConfig.prepare();
});
for (const handler of gesture) {
_RNGestureHandlerModule.default.createGestureHandler(handler.handlerName, handler.handlerTag, (0, _gestureHandlerCommon.filterConfig)(handler.config, ALLOWED_PROPS));
(0, _handlersRegistry.registerHandler)(handler.handlerTag, handler); // use setImmediate to extract handlerTags, because all refs should be initialized
// when it's ran
setImmediate(() => {
let requireToFail = [];
if (handler.config.requireToFail) {
requireToFail = extractValidHandlerTags(handler.config.requireToFail);
}
let simultaneousWith = [];
if (handler.config.simultaneousWith) {
simultaneousWith = extractValidHandlerTags(handler.config.simultaneousWith);
}
_RNGestureHandlerModule.default.updateGestureHandler(handler.handlerTag, (0, _gestureHandlerCommon.filterConfig)(handler.config, ALLOWED_PROPS, {
simultaneousHandlers: simultaneousWith,
waitFor: requireToFail
}));
});
}
preparedGesture.config = gesture;
for (const gesture of preparedGesture.config) {
_RNGestureHandlerModule.default.attachGestureHandler(gesture.handlerTag, viewTag, !useAnimated // send direct events when using animatedGesture, device events otherwise
);
}
if (preparedGesture.animatedHandlers) {
preparedGesture.animatedHandlers.value = gesture.map(g => g.handlers);
}
}
function updateHandlers(preparedGesture, gestureConfig, gesture) {
gestureConfig === null || gestureConfig === void 0 ? void 0 : gestureConfig.prepare();
for (let i = 0; i < gesture.length; i++) {
const handler = preparedGesture.config[i];
gesture[i].handlerTag = handler.handlerTag;
gesture[i].handlers.handlerTag = handler.handlerTag;
} // use setImmediate to extract handlerTags, because when it's ran, all refs should be updated
// and handlerTags in BaseGesture references should be updated in the loop above (we need to wait
// in case of external relations)
setImmediate(() => {
for (let i = 0; i < gesture.length; i++) {
const handler = preparedGesture.config[i];
handler.config = gesture[i].config;
handler.handlers = gesture[i].handlers;
handler.handlers.handlerTag = handler.handlerTag;
const requireToFail = extractValidHandlerTags(handler.config.requireToFail);
const simultaneousWith = extractValidHandlerTags(handler.config.simultaneousWith);
_RNGestureHandlerModule.default.updateGestureHandler(handler.handlerTag, (0, _gestureHandlerCommon.filterConfig)(handler.config, ALLOWED_PROPS, {
simultaneousHandlers: simultaneousWith,
waitFor: requireToFail
}));
(0, _handlersRegistry.registerHandler)(handler.handlerTag, handler);
}
if (preparedGesture.animatedHandlers) {
preparedGesture.animatedHandlers.value = preparedGesture.config.map(g => g.handlers);
}
});
}
function needsToReattach(preparedGesture, gesture) {
if (gesture.length !== preparedGesture.config.length) {
return true;
}
for (let i = 0; i < gesture.length; i++) {
if (gesture[i].handlerName !== preparedGesture.config[i].handlerName) {
return true;
}
}
return false;
}
function useAnimatedGesture(preparedGesture) {
if (!_reanimatedWrapper.Reanimated) {
return;
}
function isStateChangeEvent(event) {
'worklet'; // @ts-ignore Yes, the oldState prop is missing on GestureTouchEvent, that's the point
return event.oldState != null;
}
function isTouchEvent(event) {
'worklet';
return event.eventType != null;
}
function getHandler(type, gesture) {
'worklet';
switch (type) {
case _gesture.CALLBACK_TYPE.BEGAN:
return gesture.onBegin;
case _gesture.CALLBACK_TYPE.START:
return gesture.onStart;
case _gesture.CALLBACK_TYPE.UPDATE:
return gesture.onUpdate;
case _gesture.CALLBACK_TYPE.END:
return gesture.onEnd;
case _gesture.CALLBACK_TYPE.FINALIZE:
return gesture.onFinalize;
case _gesture.CALLBACK_TYPE.TOUCHES_DOWN:
return gesture.onTouchesDown;
case _gesture.CALLBACK_TYPE.TOUCHES_MOVE:
return gesture.onTouchesMove;
case _gesture.CALLBACK_TYPE.TOUCHES_UP:
return gesture.onTouchesUp;
case _gesture.CALLBACK_TYPE.TOUCHES_CANCELLED:
return gesture.onTouchesCancelled;
}
}
function touchEventTypeToCallbackType(eventType) {
'worklet';
switch (eventType) {
case _EventType.EventType.TOUCHES_DOWN:
return _gesture.CALLBACK_TYPE.TOUCHES_DOWN;
case _EventType.EventType.TOUCHES_MOVE:
return _gesture.CALLBACK_TYPE.TOUCHES_MOVE;
case _EventType.EventType.TOUCHES_UP:
return _gesture.CALLBACK_TYPE.TOUCHES_UP;
case _EventType.EventType.TOUCHES_CANCELLED:
return _gesture.CALLBACK_TYPE.TOUCHES_CANCELLED;
}
return _gesture.CALLBACK_TYPE.UNDEFINED;
}
function runWorklet(type, gesture, event, ...args) {
'worklet';
const handler = getHandler(type, gesture);
if (gesture.isWorklet[type]) {
// @ts-ignore Logic below makes sure the correct event is send to the
// correct handler.
handler === null || handler === void 0 ? void 0 : handler(event, ...args);
} else if (handler) {
console.warn('Animated gesture callback must be a worklet');
}
} // Hooks are called conditionally, but the condition is whether the
// react-native-reanimated is installed, which shouldn't change while running
// eslint-disable-next-line react-hooks/rules-of-hooks
const sharedHandlersCallbacks = _reanimatedWrapper.Reanimated.useSharedValue(null); // not every gesture needs a state controller, init them lazily
const stateControllers = [];
const callback = event => {
'worklet';
const currentCallback = sharedHandlersCallbacks.value;
if (!currentCallback) {
return;
}
for (let i = 0; i < currentCallback.length; i++) {
const gesture = currentCallback[i];
if (event.handlerTag === gesture.handlerTag) {
if (isStateChangeEvent(event)) {
if (event.oldState === _State.State.UNDETERMINED && event.state === _State.State.BEGAN) {
runWorklet(_gesture.CALLBACK_TYPE.BEGAN, gesture, event);
} else if ((event.oldState === _State.State.BEGAN || event.oldState === _State.State.UNDETERMINED) && event.state === _State.State.ACTIVE) {
runWorklet(_gesture.CALLBACK_TYPE.START, gesture, event);
} else if (event.oldState !== event.state && event.state === _State.State.END) {
if (event.oldState === _State.State.ACTIVE) {
runWorklet(_gesture.CALLBACK_TYPE.END, gesture, event, true);
}
runWorklet(_gesture.CALLBACK_TYPE.FINALIZE, gesture, event, true);
} else if ((event.state === _State.State.FAILED || event.state === _State.State.CANCELLED) && event.state !== event.oldState) {
if (event.oldState === _State.State.ACTIVE) {
runWorklet(_gesture.CALLBACK_TYPE.END, gesture, event, false);
}
runWorklet(_gesture.CALLBACK_TYPE.FINALIZE, gesture, event, false);
}
} else if (isTouchEvent(event)) {
if (!stateControllers[i]) {
stateControllers[i] = _gestureStateManager.GestureStateManager.create(event.handlerTag);
}
if (event.eventType !== _EventType.EventType.UNDETERMINED) {
runWorklet(touchEventTypeToCallbackType(event.eventType), gesture, event, stateControllers[i]);
}
} else {
runWorklet(_gesture.CALLBACK_TYPE.UPDATE, gesture, event);
}
}
}
}; // eslint-disable-next-line react-hooks/rules-of-hooks
const event = _reanimatedWrapper.Reanimated.useEvent(callback, ['onGestureHandlerStateChange', 'onGestureHandlerEvent'], true);
preparedGesture.animatedEventHandler = event;
preparedGesture.animatedHandlers = sharedHandlersCallbacks;
}
const GestureDetector = props => {
var _gestureConfig$toGest, _gestureConfig$toGest2;
const gestureConfig = props.gesture;
const gesture = (_gestureConfig$toGest = gestureConfig === null || gestureConfig === void 0 ? void 0 : (_gestureConfig$toGest2 = gestureConfig.toGestureArray) === null || _gestureConfig$toGest2 === void 0 ? void 0 : _gestureConfig$toGest2.call(gestureConfig)) !== null && _gestureConfig$toGest !== void 0 ? _gestureConfig$toGest : [];
const useAnimated = gesture.find(gesture => gesture.handlers.isWorklet.reduce((prev, current) => prev || current)) != null;
const viewRef = (0, _react.useRef)(null);
const firstRenderRef = (0, _react.useRef)(true);
const preparedGesture = _react.default.useRef({
config: gesture,
animatedEventHandler: null,
animatedHandlers: null,
firstExecution: true,
useAnimated: useAnimated
}).current;
if (useAnimated !== preparedGesture.useAnimated) {
throw new Error('You cannot change whether you are using gesture or animatedGesture while the app is running');
}
if (preparedGesture.firstExecution) {
var _gestureConfig$initia;
gestureConfig === null || gestureConfig === void 0 ? void 0 : (_gestureConfig$initia = gestureConfig.initialize) === null || _gestureConfig$initia === void 0 ? void 0 : _gestureConfig$initia.call(gestureConfig);
}
if (useAnimated) {
// Whether animatedGesture or gesture is used shouldn't change
// during while an app is running
// eslint-disable-next-line react-hooks/rules-of-hooks
useAnimatedGesture(preparedGesture);
}
(0, _react.useEffect)(() => {
firstRenderRef.current = true;
const viewTag = (0, _gestureHandlerCommon.findNodeHandle)(viewRef.current);
attachHandlers({
preparedGesture,
gestureConfig,
gesture,
viewTag,
useAnimated
});
return () => {
dropHandlers(preparedGesture);
};
}, []);
(0, _react.useEffect)(() => {
if (!firstRenderRef.current) {
const viewTag = (0, _gestureHandlerCommon.findNodeHandle)(viewRef.current);
if (needsToReattach(preparedGesture, gesture)) {
dropHandlers(preparedGesture);
attachHandlers({
preparedGesture,
gestureConfig,
gesture,
viewTag,
useAnimated
});
} else {
updateHandlers(preparedGesture, gestureConfig, gesture);
}
} else {
firstRenderRef.current = false;
}
}, [props]);
if (useAnimated) {
return /*#__PURE__*/_react.default.createElement(AnimatedWrap, {
ref: viewRef,
onGestureHandlerEvent: preparedGesture.animatedEventHandler
}, props.children);
} else {
return /*#__PURE__*/_react.default.createElement(Wrap, {
ref: viewRef
}, props.children);
}
};
exports.GestureDetector = GestureDetector;
class Wrap extends _react.default.Component {
render() {
// I don't think that fighting with types over such a simple function is worth it
// The only thing it does is add 'collapsable: false' to the child component
// to make sure it is in the native view hierarchy so the detector can find
// correct viewTag to attach to.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const child = _react.default.Children.only(this.props.children);
return /*#__PURE__*/_react.default.cloneElement(child, {
collapsable: false
}, // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
child.props.children);
}
}
const AnimatedWrap = (_Reanimated$default$c = _reanimatedWrapper.Reanimated === null || _reanimatedWrapper.Reanimated === void 0 ? void 0 : (_Reanimated$default = _reanimatedWrapper.Reanimated.default) === null || _Reanimated$default === void 0 ? void 0 : _Reanimated$default.createAnimatedComponent(Wrap)) !== null && _Reanimated$default$c !== void 0 ? _Reanimated$default$c : Wrap;
//# sourceMappingURL=GestureDetector.js.map