You've already forked joplin
mirror of
https://github.com/laurent22/joplin.git
synced 2025-07-03 23:50:33 +02:00
69 lines
1.8 KiB
TypeScript
69 lines
1.8 KiB
TypeScript
![]() |
import * as React from 'react';
|
||
|
import { useCallback, useMemo, useRef } from 'react';
|
||
|
import { Animated, StyleSheet, Pressable, ViewProps, PressableProps } from 'react-native';
|
||
|
|
||
|
interface Props {
|
||
|
// Nodes that need to change opacity but shouldn't be included in the main touchable
|
||
|
beforePressable: React.ReactNode;
|
||
|
// Children of the main pressable
|
||
|
children: React.ReactNode;
|
||
|
onPress: ()=> void;
|
||
|
|
||
|
pressableProps?: PressableProps;
|
||
|
containerProps?: ViewProps;
|
||
|
}
|
||
|
|
||
|
// A TouchableOpacity that can contain multiple pressable items still within the region that
|
||
|
// changes opacity
|
||
|
const MultiTouchableOpacity: React.FC<Props> = props => {
|
||
|
// See https://blog.logrocket.com/react-native-touchable-vs-pressable-components/
|
||
|
// for more about animating Pressable buttons.
|
||
|
const fadeAnim = useRef(new Animated.Value(1)).current;
|
||
|
|
||
|
const animationDuration = 100; // ms
|
||
|
const onPressIn = useCallback(() => {
|
||
|
// Fade out.
|
||
|
Animated.timing(fadeAnim, {
|
||
|
toValue: 0.5,
|
||
|
duration: animationDuration,
|
||
|
useNativeDriver: true,
|
||
|
}).start();
|
||
|
}, [fadeAnim]);
|
||
|
const onPressOut = useCallback(() => {
|
||
|
// Fade in.
|
||
|
Animated.timing(fadeAnim, {
|
||
|
toValue: 1,
|
||
|
duration: animationDuration,
|
||
|
useNativeDriver: true,
|
||
|
}).start();
|
||
|
}, [fadeAnim]);
|
||
|
|
||
|
const button = (
|
||
|
<Pressable
|
||
|
accessibilityRole='button'
|
||
|
{...props.pressableProps}
|
||
|
onPress={props.onPress}
|
||
|
onPressIn={onPressIn}
|
||
|
onPressOut={onPressOut}
|
||
|
>
|
||
|
{props.children}
|
||
|
</Pressable>
|
||
|
);
|
||
|
|
||
|
const styles = useMemo(() => {
|
||
|
return StyleSheet.create({
|
||
|
container: { opacity: fadeAnim },
|
||
|
});
|
||
|
}, [fadeAnim]);
|
||
|
|
||
|
const containerProps = props.containerProps ?? {};
|
||
|
return (
|
||
|
<Animated.View {...containerProps} style={[styles.container, props.containerProps.style]}>
|
||
|
{props.beforePressable}
|
||
|
{button}
|
||
|
</Animated.View>
|
||
|
);
|
||
|
};
|
||
|
|
||
|
export default MultiTouchableOpacity;
|