react-native-tab-view 3.2.1 → 3.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/commonjs/PagerViewAdapter.js +2 -1
- package/lib/commonjs/PagerViewAdapter.js.map +1 -1
- package/lib/commonjs/PanResponderAdapter.js +2 -1
- package/lib/commonjs/PanResponderAdapter.js.map +1 -1
- package/lib/commonjs/SceneMap.js +9 -12
- package/lib/commonjs/SceneMap.js.map +1 -1
- package/lib/commonjs/SceneView.js +54 -101
- package/lib/commonjs/SceneView.js.map +1 -1
- package/lib/commonjs/TabBar.js +327 -327
- package/lib/commonjs/TabBar.js.map +1 -1
- package/lib/commonjs/TabBarIndicator.js +81 -99
- package/lib/commonjs/TabBarIndicator.js.map +1 -1
- package/lib/commonjs/TabBarItem.js +184 -161
- package/lib/commonjs/TabBarItem.js.map +1 -1
- package/lib/module/PagerViewAdapter.js +2 -1
- package/lib/module/PagerViewAdapter.js.map +1 -1
- package/lib/module/PanResponderAdapter.js +2 -1
- package/lib/module/PanResponderAdapter.js.map +1 -1
- package/lib/module/SceneMap.js +9 -14
- package/lib/module/SceneMap.js.map +1 -1
- package/lib/module/SceneView.js +53 -98
- package/lib/module/SceneView.js.map +1 -1
- package/lib/module/TabBar.js +323 -323
- package/lib/module/TabBar.js.map +1 -1
- package/lib/module/TabBarIndicator.js +74 -92
- package/lib/module/TabBarIndicator.js.map +1 -1
- package/lib/module/TabBarItem.js +178 -154
- package/lib/module/TabBarItem.js.map +1 -1
- package/lib/typescript/SceneMap.d.ts +5 -3
- package/lib/typescript/SceneView.d.ts +1 -18
- package/lib/typescript/TabBar.d.ts +7 -38
- package/lib/typescript/TabBarIndicator.d.ts +2 -10
- package/lib/typescript/TabBarItem.d.ts +3 -5
- package/package.json +4 -1
- package/src/PagerViewAdapter.tsx +6 -1
- package/src/PanResponderAdapter.tsx +6 -3
- package/src/SceneMap.tsx +11 -7
- package/src/SceneView.tsx +70 -106
- package/src/TabBar.tsx +451 -391
- package/src/TabBarIndicator.tsx +108 -114
- package/src/TabBarItem.tsx +214 -185
package/src/SceneMap.tsx
CHANGED
|
@@ -1,19 +1,23 @@
|
|
|
1
1
|
import * as React from 'react';
|
|
2
2
|
import type { SceneRendererProps } from './types';
|
|
3
3
|
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
4
|
+
type SceneProps = {
|
|
5
|
+
route: any;
|
|
6
|
+
} & Omit<SceneRendererProps, 'layout'>;
|
|
7
|
+
|
|
8
|
+
const SceneComponent = React.memo(
|
|
9
|
+
<T extends { component: React.ComponentType<any> } & SceneProps>({
|
|
10
|
+
component,
|
|
11
|
+
...rest
|
|
12
|
+
}: T) => {
|
|
9
13
|
return React.createElement(component, rest);
|
|
10
14
|
}
|
|
11
|
-
|
|
15
|
+
);
|
|
12
16
|
|
|
13
17
|
export default function SceneMap<T extends any>(scenes: {
|
|
14
18
|
[key: string]: React.ComponentType<T>;
|
|
15
19
|
}) {
|
|
16
|
-
return ({ route, jumpTo, position }:
|
|
20
|
+
return ({ route, jumpTo, position }: SceneProps) => (
|
|
17
21
|
<SceneComponent
|
|
18
22
|
key={route.key}
|
|
19
23
|
component={scenes[route.key]}
|
package/src/SceneView.tsx
CHANGED
|
@@ -17,121 +17,85 @@ type Props<T extends Route> = SceneRendererProps &
|
|
|
17
17
|
style?: StyleProp<ViewStyle>;
|
|
18
18
|
};
|
|
19
19
|
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
20
|
+
export default function SceneView<T extends Route>({
|
|
21
|
+
children,
|
|
22
|
+
navigationState,
|
|
23
|
+
lazy,
|
|
24
|
+
layout,
|
|
25
|
+
index,
|
|
26
|
+
lazyPreloadDistance,
|
|
27
|
+
addEnterListener,
|
|
28
|
+
style,
|
|
29
|
+
}: Props<T>) {
|
|
30
|
+
const [isLoading, setIsLoading] = React.useState(
|
|
31
|
+
Math.abs(navigationState.index - index) > lazyPreloadDistance
|
|
32
|
+
);
|
|
33
|
+
|
|
34
|
+
if (
|
|
35
|
+
isLoading &&
|
|
36
|
+
Math.abs(navigationState.index - index) <= lazyPreloadDistance
|
|
37
|
+
) {
|
|
38
|
+
// Always render the route when it becomes focused
|
|
39
|
+
setIsLoading(false);
|
|
39
40
|
}
|
|
40
41
|
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
42
|
+
React.useEffect(() => {
|
|
43
|
+
const handleEnter = (value: number) => {
|
|
44
|
+
// If we're entering the current route, we need to load it
|
|
45
|
+
if (value === index) {
|
|
46
|
+
setIsLoading((prevState) => {
|
|
47
|
+
if (prevState) {
|
|
48
|
+
return false;
|
|
49
|
+
}
|
|
50
|
+
return prevState;
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
let unsubscribe: (() => void) | undefined;
|
|
56
|
+
let timer: NodeJS.Timeout;
|
|
46
57
|
|
|
47
|
-
|
|
48
|
-
if (this.props.lazy) {
|
|
58
|
+
if (lazy && isLoading) {
|
|
49
59
|
// If lazy mode is enabled, listen to when we enter screens
|
|
50
|
-
|
|
51
|
-
} else if (
|
|
60
|
+
unsubscribe = addEnterListener(handleEnter);
|
|
61
|
+
} else if (isLoading) {
|
|
52
62
|
// If lazy mode is not enabled, render the scene with a delay if not loaded already
|
|
53
63
|
// This improves the initial startup time as the scene is no longer blocking
|
|
54
|
-
|
|
55
|
-
() => this.setState({ loading: false }),
|
|
56
|
-
0
|
|
57
|
-
);
|
|
64
|
+
timer = setTimeout(() => setIsLoading(false), 0);
|
|
58
65
|
}
|
|
59
|
-
}
|
|
60
66
|
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
67
|
+
return () => {
|
|
68
|
+
unsubscribe?.();
|
|
69
|
+
clearTimeout(timer);
|
|
70
|
+
};
|
|
71
|
+
}, [addEnterListener, index, isLoading, lazy]);
|
|
72
|
+
|
|
73
|
+
const focused = navigationState.index === index;
|
|
74
|
+
|
|
75
|
+
return (
|
|
76
|
+
<View
|
|
77
|
+
accessibilityElementsHidden={!focused}
|
|
78
|
+
importantForAccessibility={focused ? 'auto' : 'no-hide-descendants'}
|
|
79
|
+
style={[
|
|
80
|
+
styles.route,
|
|
81
|
+
// If we don't have the layout yet, make the focused screen fill the container
|
|
82
|
+
// This avoids delay before we are able to render pages side by side
|
|
83
|
+
layout.width
|
|
84
|
+
? { width: layout.width }
|
|
85
|
+
: focused
|
|
86
|
+
? StyleSheet.absoluteFill
|
|
87
|
+
: null,
|
|
88
|
+
style,
|
|
89
|
+
]}
|
|
90
|
+
>
|
|
91
|
+
{
|
|
92
|
+
// Only render the route only if it's either focused or layout is available
|
|
93
|
+
// When layout is not available, we must not render unfocused routes
|
|
94
|
+
// so that the focused route can fill the screen
|
|
95
|
+
focused || layout.width ? children({ loading: isLoading }) : null
|
|
72
96
|
}
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
componentWillUnmount() {
|
|
77
|
-
this.unsubscribe?.();
|
|
78
|
-
|
|
79
|
-
if (this.timerHandler) {
|
|
80
|
-
clearTimeout(this.timerHandler);
|
|
81
|
-
this.timerHandler = undefined;
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
private timerHandler: NodeJS.Timeout | undefined;
|
|
86
|
-
|
|
87
|
-
private unsubscribe: (() => void) | null = null;
|
|
88
|
-
|
|
89
|
-
private handleEnter = (value: number) => {
|
|
90
|
-
const { index } = this.props;
|
|
91
|
-
|
|
92
|
-
// If we're entering the current route, we need to load it
|
|
93
|
-
if (value === index) {
|
|
94
|
-
this.setState((prevState) => {
|
|
95
|
-
if (prevState.loading) {
|
|
96
|
-
return { loading: false };
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
return null;
|
|
100
|
-
});
|
|
101
|
-
}
|
|
102
|
-
};
|
|
103
|
-
|
|
104
|
-
render() {
|
|
105
|
-
const { navigationState, index, layout, style } = this.props;
|
|
106
|
-
const { loading } = this.state;
|
|
107
|
-
|
|
108
|
-
const focused = navigationState.index === index;
|
|
109
|
-
|
|
110
|
-
return (
|
|
111
|
-
<View
|
|
112
|
-
accessibilityElementsHidden={!focused}
|
|
113
|
-
importantForAccessibility={focused ? 'auto' : 'no-hide-descendants'}
|
|
114
|
-
style={[
|
|
115
|
-
styles.route,
|
|
116
|
-
// If we don't have the layout yet, make the focused screen fill the container
|
|
117
|
-
// This avoids delay before we are able to render pages side by side
|
|
118
|
-
layout.width
|
|
119
|
-
? { width: layout.width }
|
|
120
|
-
: focused
|
|
121
|
-
? StyleSheet.absoluteFill
|
|
122
|
-
: null,
|
|
123
|
-
style,
|
|
124
|
-
]}
|
|
125
|
-
>
|
|
126
|
-
{
|
|
127
|
-
// Only render the route only if it's either focused or layout is available
|
|
128
|
-
// When layout is not available, we must not render unfocused routes
|
|
129
|
-
// so that the focused route can fill the screen
|
|
130
|
-
focused || layout.width ? this.props.children({ loading }) : null
|
|
131
|
-
}
|
|
132
|
-
</View>
|
|
133
|
-
);
|
|
134
|
-
}
|
|
97
|
+
</View>
|
|
98
|
+
);
|
|
135
99
|
}
|
|
136
100
|
|
|
137
101
|
const styles = StyleSheet.create({
|