-
-
Notifications
You must be signed in to change notification settings - Fork 158
Expand file tree
/
Copy pathindex.tsx
More file actions
124 lines (110 loc) · 3.12 KB
/
index.tsx
File metadata and controls
124 lines (110 loc) · 3.12 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
import { StackScreenProps } from '@react-navigation/stack';
import React, { useEffect, useState } from 'react';
import { Text, TextInput, View } from 'react-native';
import {
KeyboardGestureArea,
useKeyboardHandler,
} from 'react-native-keyboard-controller';
import Reanimated, {
scrollTo,
useAnimatedRef,
useAnimatedScrollHandler,
useAnimatedStyle,
useSharedValue,
} from 'react-native-reanimated';
import Message from '../../../components/Message';
import { history } from '../../../components/Message/data';
import { ExamplesStackParamList } from '../../../navigation/ExamplesStack';
import styles from './styles';
const AnimatedTextInput = Reanimated.createAnimatedComponent(TextInput);
const useKeyboardAnimation = ({ref, scroll}) => {
const progress = useSharedValue(0);
const height = useSharedValue(0);
useKeyboardHandler({
onMove: (e) => {
'worklet';
progress.value = e.progress;
height.value = e.height;
},
onInteractive: (e) => {
'worklet';
scrollTo(ref, 0, scroll.value, false);
progress.value = e.progress;
height.value = e.height;
},
});
return { height, progress };
};
type Props = StackScreenProps<ExamplesStackParamList>;
function InteractiveKeyboard({ navigation }: Props) {
const aRef = useAnimatedRef();
const scroll = useSharedValue(0);
const [interpolator, setInterpolator] = useState<'ios' | 'linear'>('linear');
const { height } = useKeyboardAnimation({ref: aRef, scroll: scroll});
const onScroll = useAnimatedScrollHandler({
onScroll: (e) => {
scroll.value = e.contentOffset.y;
},
})
useEffect(() => {
navigation.setOptions({
headerRight: () => (
<Text
style={styles.header}
onPress={() =>
setInterpolator(interpolator === 'ios' ? 'linear' : 'ios')
}
>
{interpolator}
</Text>
),
});
}, [interpolator]);
const scrollViewStyle = useAnimatedStyle(
() => ({
transform: [{ translateY: -height.value }, ...styles.inverted.transform],
}),
[]
);
const textInputStyle = useAnimatedStyle(
() => ({
height: 50,
width: '100%',
backgroundColor: '#BCBCBC',
transform: [{ translateY: -height.value }],
}),
[]
);
const fakeView = useAnimatedStyle(
() => ({
// TODO: don't update when onInteractive is fired
// height: height.value,
}),
[]
);
return (
<View style={styles.container}>
<KeyboardGestureArea
style={styles.content}
interpolator={interpolator}
showOnSwipeUp
>
<Reanimated.ScrollView
ref={aRef}
onScroll={onScroll}
showsVerticalScrollIndicator={false}
style={scrollViewStyle}
>
<View style={styles.inverted}>
<Reanimated.View style={fakeView} />
{history.map((message, index) => (
<Message key={index} {...message} />
))}
</View>
</Reanimated.ScrollView>
</KeyboardGestureArea>
<AnimatedTextInput style={textInputStyle} />
</View>
);
}
export default InteractiveKeyboard;