1 | import * as React from 'react';
|
2 | import { HostComponent, Keyboard, TextInput } from 'react-native';
|
3 |
|
4 | type InputRef = React.ElementRef<HostComponent<unknown>> | undefined;
|
5 |
|
6 | export default function useKeyboardManager(isEnabled: () => boolean) {
|
7 |
|
8 |
|
9 | const previouslyFocusedTextInputRef = React.useRef<InputRef>(undefined);
|
10 | const startTimestampRef = React.useRef<number>(0);
|
11 | const keyboardTimeoutRef = React.useRef<any>();
|
12 |
|
13 | const clearKeyboardTimeout = React.useCallback(() => {
|
14 | if (keyboardTimeoutRef.current !== undefined) {
|
15 | clearTimeout(keyboardTimeoutRef.current);
|
16 | keyboardTimeoutRef.current = undefined;
|
17 | }
|
18 | }, []);
|
19 |
|
20 | const onPageChangeStart = React.useCallback(() => {
|
21 | if (!isEnabled()) {
|
22 | return;
|
23 | }
|
24 |
|
25 | clearKeyboardTimeout();
|
26 |
|
27 | const input: InputRef = TextInput.State.currentlyFocusedInput();
|
28 |
|
29 |
|
30 | input?.blur();
|
31 |
|
32 |
|
33 | previouslyFocusedTextInputRef.current = input;
|
34 |
|
35 |
|
36 | startTimestampRef.current = Date.now();
|
37 | }, [clearKeyboardTimeout, isEnabled]);
|
38 |
|
39 | const onPageChangeConfirm = React.useCallback(
|
40 | (force: boolean) => {
|
41 | if (!isEnabled()) {
|
42 | return;
|
43 | }
|
44 |
|
45 | clearKeyboardTimeout();
|
46 |
|
47 | if (force) {
|
48 |
|
49 |
|
50 |
|
51 | Keyboard.dismiss();
|
52 | } else {
|
53 | const input = previouslyFocusedTextInputRef.current;
|
54 |
|
55 |
|
56 |
|
57 | input?.blur();
|
58 | }
|
59 |
|
60 |
|
61 | previouslyFocusedTextInputRef.current = undefined;
|
62 | },
|
63 | [clearKeyboardTimeout, isEnabled]
|
64 | );
|
65 |
|
66 | const onPageChangeCancel = React.useCallback(() => {
|
67 | if (!isEnabled()) {
|
68 | return;
|
69 | }
|
70 |
|
71 | clearKeyboardTimeout();
|
72 |
|
73 |
|
74 | const input = previouslyFocusedTextInputRef.current;
|
75 |
|
76 | if (input) {
|
77 |
|
78 |
|
79 |
|
80 |
|
81 |
|
82 |
|
83 |
|
84 | if (Date.now() - startTimestampRef.current < 100) {
|
85 | keyboardTimeoutRef.current = setTimeout(() => {
|
86 | input?.focus();
|
87 | previouslyFocusedTextInputRef.current = undefined;
|
88 | }, 100);
|
89 | } else {
|
90 | input?.focus();
|
91 | previouslyFocusedTextInputRef.current = undefined;
|
92 | }
|
93 | }
|
94 | }, [clearKeyboardTimeout, isEnabled]);
|
95 |
|
96 | React.useEffect(() => {
|
97 | return () => clearKeyboardTimeout();
|
98 | }, [clearKeyboardTimeout]);
|
99 |
|
100 | return {
|
101 | onPageChangeStart,
|
102 | onPageChangeConfirm,
|
103 | onPageChangeCancel,
|
104 | };
|
105 | }
|