-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathMenu.js
More file actions
147 lines (132 loc) · 3.21 KB
/
Copy pathMenu.js
File metadata and controls
147 lines (132 loc) · 3.21 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
import React, { useRef, useState, useEffect } from 'react';
import {
useWindowDimensions,
Animated,
Text,
StyleSheet
} from 'react-native';
import CloseButton from './CloseButton';
const Menu = (props) => {
const {
shouldShow,
title,
children,
onClose,
showDefaultCloseButton,
animationDuration,
style,
backgroundColor
} = props;
const { height } = useWindowDimensions();
// translate entire menu on/off-screen
const beforeShow = shouldShow ? height : 0;
const afterShow = shouldShow ? 0 : height;
// fade entire menu in/out
const beforeOpacity = shouldShow ? 0 : 1;
const afterOpacity = shouldShow ? 1 : 0;
// translate and opacity ref values
const translateAnim = useRef(new Animated.Value(beforeShow)).current;
const opacityAnim = useRef(new Animated.Value(beforeOpacity)).current;
// call onClose prop onPress of close button
const [shouldClose, setShouldClose] = useState(false);
// translate menu on/off-screen
useEffect(() => {
if (shouldShow) {
// translate entire menu on-screen instantly
Animated.timing(
translateAnim, {
toValue: afterShow,
duration: 0,
useNativeDriver: true
}
).start();
} else {
// translate entire menu off-screen upon animation completion
const timer = setTimeout(() => {
Animated.timing(
translateAnim, {
toValue: afterShow,
duration: 0,
useNativeDriver: true
}
).start();
return (() => {
clearTimeout(timer);
});
}, animationDuration);
}
}, [shouldShow, shouldClose]);
// fade menu in/out
useEffect(() => {
Animated.timing(
opacityAnim, {
toValue: afterOpacity,
duration: animationDuration,
useNativeDriver: true
}
).start();
}, [shouldShow]);
// close this menu from this close button
useEffect(() => {
onClose();
setShouldClose(false);
}, [shouldClose]);
return (
<Animated.View
onTouchEnd={(e) => {
if (e.target == e.currentTarget)
setShouldClose(true);
}}
style={[
style ?? styles.container,
{
backgroundColor: backgroundColor,
transform: [{ translateY: translateAnim }],
opacity: opacityAnim
}
]}
>
{title && (
<Text style={styles.title}>{title}</Text>
)}
{children}
{showDefaultCloseButton && (
<CloseButton
onClose={() => setShouldClose(true)}
/>
)}
</Animated.View>
);
}
const styles = StyleSheet.create({
container: {
position: 'absolute',
height: '100%',
width: '100%',
display: 'flex',
justifyContent: 'flex-end',
alignItems: 'center'
},
title: {
position: 'absolute',
top: 60,
paddingLeft: 2,
fontSize: 14,
fontWeight: '500',
textAlign: 'center',
textTransform: 'uppercase',
letterSpacing: 2,
color: 'white'
},
});
Menu.defaultProps = {
shouldShow: true,
title: undefined,
children: <></>,
onClose: (() => {}),
showDefaultCloseButton: true,
animationDuration: 300,
style: undefined,
backgroundColor: 'transparent'
};
export default Menu;