@applicaster/zapp-react-native-ui-components 15.0.0-alpha.8621453569 → 15.0.0-alpha.9331298916
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/Components/Cell/Cell.tsx +3 -8
- package/Components/MasterCell/DefaultComponents/BorderContainerView/__tests__/index.test.tsx +1 -16
- package/Components/MasterCell/DefaultComponents/BorderContainerView/index.tsx +2 -30
- package/Components/MasterCell/DefaultComponents/Text/index.tsx +8 -8
- package/Components/MasterCell/index.tsx +0 -2
- package/Components/ScreenRevealManager/ScreenRevealManager.ts +40 -8
- package/Components/ScreenRevealManager/__tests__/ScreenRevealManager.test.ts +86 -69
- package/Components/Tabs/TV/Tabs.tsx +3 -20
- package/Components/Transitioner/Scene.tsx +15 -2
- package/index.d.ts +0 -7
- package/package.json +5 -5
package/Components/Cell/Cell.tsx
CHANGED
|
@@ -208,14 +208,14 @@ export class CellComponent extends React.Component<Props, State> {
|
|
|
208
208
|
this.accessibilityManager.readText({
|
|
209
209
|
text: " ",
|
|
210
210
|
});
|
|
211
|
-
} else
|
|
211
|
+
} else {
|
|
212
212
|
this.accessibilityManager.readText({
|
|
213
213
|
text: `${positionLabel}`,
|
|
214
214
|
});
|
|
215
215
|
}
|
|
216
216
|
}
|
|
217
217
|
|
|
218
|
-
componentDidUpdate(prevProps: Readonly<Props
|
|
218
|
+
componentDidUpdate(prevProps: Readonly<Props>) {
|
|
219
219
|
if (prevProps.item !== this.props.item) {
|
|
220
220
|
this.setState({
|
|
221
221
|
hasFocusableInside: this.props.CellRenderer.hasFocusableInside?.(
|
|
@@ -224,12 +224,7 @@ export class CellComponent extends React.Component<Props, State> {
|
|
|
224
224
|
});
|
|
225
225
|
}
|
|
226
226
|
|
|
227
|
-
|
|
228
|
-
prevState.cellFocused !== this.state.cellFocused ||
|
|
229
|
-
this.state.hasFocusableInside
|
|
230
|
-
) {
|
|
231
|
-
this.handleAccessibilityFocus(this.props.index, this.props.dataLength);
|
|
232
|
-
}
|
|
227
|
+
this.handleAccessibilityFocus(this.props.index, this.props.dataLength);
|
|
233
228
|
}
|
|
234
229
|
|
|
235
230
|
render() {
|
package/Components/MasterCell/DefaultComponents/BorderContainerView/__tests__/index.test.tsx
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
import * as React from "react";
|
|
2
1
|
import {
|
|
3
2
|
BorderContainerView,
|
|
4
3
|
getBorderPadding, // Export for testing (using a double underscore prefix is a common convention)
|
|
5
4
|
} from "../index";
|
|
5
|
+
import * as React from "react";
|
|
6
6
|
import { render } from "@testing-library/react-native";
|
|
7
7
|
import { toNumberWithDefaultZero } from "@applicaster/zapp-react-native-utils/numberUtils";
|
|
8
8
|
import { View } from "react-native";
|
|
@@ -11,15 +11,6 @@ jest.mock("@applicaster/zapp-react-native-utils/numberUtils", () => ({
|
|
|
11
11
|
toNumberWithDefaultZero: jest.fn((value) => Number(value) || 0),
|
|
12
12
|
}));
|
|
13
13
|
|
|
14
|
-
jest.mock(
|
|
15
|
-
"@applicaster/zapp-react-native-utils/appUtils/accessibilityManager/hooks",
|
|
16
|
-
() => ({
|
|
17
|
-
useAccessibilityManager: jest.fn(() => ({
|
|
18
|
-
addHeading: jest.fn(),
|
|
19
|
-
})),
|
|
20
|
-
})
|
|
21
|
-
);
|
|
22
|
-
|
|
23
14
|
describe("BorderContainerView", () => {
|
|
24
15
|
describe("getBorderPadding", () => {
|
|
25
16
|
it("returns 0 for inside", () => {
|
|
@@ -51,8 +42,6 @@ describe("BorderContainerView", () => {
|
|
|
51
42
|
};
|
|
52
43
|
|
|
53
44
|
const borderPosition = null;
|
|
54
|
-
const mockEntry = { id: "test-entry" } as ZappEntry;
|
|
55
|
-
const mockHasFocusableInside = jest.fn(() => false);
|
|
56
45
|
|
|
57
46
|
const { queryByTestId } = render(
|
|
58
47
|
<BorderContainerView
|
|
@@ -63,10 +52,6 @@ describe("BorderContainerView", () => {
|
|
|
63
52
|
borderPaddingRight={toNumberWithDefaultZero(padding.paddingRight)}
|
|
64
53
|
borderPaddingBottom={toNumberWithDefaultZero(padding.paddingBottom)}
|
|
65
54
|
borderPaddingLeft={toNumberWithDefaultZero(padding.paddingLeft)}
|
|
66
|
-
hasFocusableInside={mockHasFocusableInside}
|
|
67
|
-
entry={mockEntry}
|
|
68
|
-
state="focused"
|
|
69
|
-
hasTextLabels={false}
|
|
70
55
|
>
|
|
71
56
|
<View testID="child" />
|
|
72
57
|
</BorderContainerView>
|
|
@@ -1,16 +1,10 @@
|
|
|
1
|
-
import React, { useMemo, useContext, useEffect } from "react";
|
|
2
|
-
import { ImageStyle, StyleSheet, View, ViewStyle } from "react-native";
|
|
3
|
-
import { useAccessibilityManager } from "@applicaster/zapp-react-native-utils/appUtils/accessibilityManager/hooks";
|
|
4
1
|
import { toNumberWithDefaultZero } from "@applicaster/zapp-react-native-utils/numberUtils";
|
|
5
|
-
import
|
|
2
|
+
import * as React from "react";
|
|
3
|
+
import { ImageStyle, StyleSheet, View, ViewStyle } from "react-native";
|
|
6
4
|
|
|
7
5
|
type BorderPosition = "inside" | "outside" | "center";
|
|
8
6
|
|
|
9
7
|
interface Props {
|
|
10
|
-
hasFocusableInside: (entry: ZappEntry) => boolean;
|
|
11
|
-
entry: ZappEntry;
|
|
12
|
-
state: CellState;
|
|
13
|
-
hasTextLabels: boolean;
|
|
14
8
|
style: ImageStyle | ViewStyle;
|
|
15
9
|
borderPosition: BorderPosition;
|
|
16
10
|
borderPaddingTop: number;
|
|
@@ -124,30 +118,8 @@ export const BorderContainerView = (props: Props) => {
|
|
|
124
118
|
borderPaddingLeft,
|
|
125
119
|
style,
|
|
126
120
|
children,
|
|
127
|
-
hasFocusableInside,
|
|
128
|
-
entry,
|
|
129
|
-
state,
|
|
130
|
-
hasTextLabels,
|
|
131
121
|
} = props;
|
|
132
122
|
|
|
133
|
-
const accessibilityManager = useAccessibilityManager();
|
|
134
|
-
const isMeasurement = useContext(MeasurementPortalContext);
|
|
135
|
-
|
|
136
|
-
const isImageOnlyCell = useMemo(
|
|
137
|
-
() =>
|
|
138
|
-
!hasFocusableInside(entry) &&
|
|
139
|
-
!hasTextLabels &&
|
|
140
|
-
state === "focused" &&
|
|
141
|
-
!isMeasurement,
|
|
142
|
-
[hasFocusableInside, entry, hasTextLabels, state, isMeasurement]
|
|
143
|
-
);
|
|
144
|
-
|
|
145
|
-
useEffect(() => {
|
|
146
|
-
if (isImageOnlyCell && entry?.title) {
|
|
147
|
-
accessibilityManager.addHeading(String(entry.title));
|
|
148
|
-
}
|
|
149
|
-
}, [isImageOnlyCell, entry?.title]);
|
|
150
|
-
|
|
151
123
|
const padding =
|
|
152
124
|
borderPosition === "outside"
|
|
153
125
|
? {
|
|
@@ -52,14 +52,14 @@ const _Text = ({
|
|
|
52
52
|
: textTransform(transformText, _label);
|
|
53
53
|
|
|
54
54
|
React.useLayoutEffect(() => {
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
55
|
+
// For FocusableCells with action buttons
|
|
56
|
+
if (otherProps.state) {
|
|
57
|
+
if (otherProps.state === "focused" && cellFocused === true) {
|
|
58
|
+
accessibilityManager.addHeading(textLabel);
|
|
59
|
+
}
|
|
60
|
+
} else {
|
|
61
|
+
if (cellFocused === true) {
|
|
62
|
+
accessibilityManager.addHeading(textLabel);
|
|
63
63
|
}
|
|
64
64
|
}
|
|
65
65
|
}, [cellFocused, otherProps.state, textLabel]);
|
|
@@ -1,6 +1,11 @@
|
|
|
1
1
|
import { makeListOf } from "@applicaster/zapp-react-native-utils/arrayUtils";
|
|
2
2
|
import { isFirstComponentGallery } from "@applicaster/zapp-react-native-utils/componentsUtils";
|
|
3
|
-
import {
|
|
3
|
+
import { race, Subject, Subscription } from "rxjs";
|
|
4
|
+
import { first } from "rxjs/operators";
|
|
5
|
+
|
|
6
|
+
import { withTimeout$ } from "@applicaster/zapp-react-native-utils/idleUtils";
|
|
7
|
+
|
|
8
|
+
const MAX_TIMEOUT_TO_WAIT_COMPONENTS_TO_LOAD = 2000; // 2 seconds
|
|
4
9
|
|
|
5
10
|
const INITIAL_NUMBER_TO_LOAD = 3;
|
|
6
11
|
|
|
@@ -34,7 +39,8 @@ const getNumberOfComponentsWaitToLoadBeforePresent = (
|
|
|
34
39
|
export class ScreenRevealManager {
|
|
35
40
|
public numberOfComponentsWaitToLoadBeforePresent: number;
|
|
36
41
|
private renderingState: Array<ComponentLoadingState>;
|
|
37
|
-
private
|
|
42
|
+
private subject$ = new Subject<void>();
|
|
43
|
+
private subscription: Subscription;
|
|
38
44
|
|
|
39
45
|
constructor(componentsToRender: ZappUIComponent[], callback: Callback) {
|
|
40
46
|
this.numberOfComponentsWaitToLoadBeforePresent =
|
|
@@ -45,32 +51,58 @@ export class ScreenRevealManager {
|
|
|
45
51
|
this.numberOfComponentsWaitToLoadBeforePresent
|
|
46
52
|
);
|
|
47
53
|
|
|
48
|
-
this.
|
|
54
|
+
this.subscription = race(
|
|
55
|
+
withTimeout$(MAX_TIMEOUT_TO_WAIT_COMPONENTS_TO_LOAD),
|
|
56
|
+
this.subject$
|
|
57
|
+
)
|
|
58
|
+
.pipe(first())
|
|
59
|
+
.subscribe(callback);
|
|
49
60
|
}
|
|
50
61
|
|
|
51
62
|
onLoadFinished = (index: number): void => {
|
|
63
|
+
const currentState = this.renderingState[index];
|
|
64
|
+
|
|
65
|
+
if (
|
|
66
|
+
COMPONENT_LOADING_STATE.LOADED_WITH_SUCCESS === currentState ||
|
|
67
|
+
COMPONENT_LOADING_STATE.LOADED_WITH_FAILURE === currentState
|
|
68
|
+
) {
|
|
69
|
+
return;
|
|
70
|
+
}
|
|
71
|
+
|
|
52
72
|
this.renderingState[index] = COMPONENT_LOADING_STATE.LOADED_WITH_SUCCESS;
|
|
53
73
|
|
|
54
74
|
if (
|
|
55
75
|
getNumberOfLoaded(this.renderingState) >=
|
|
56
76
|
this.numberOfComponentsWaitToLoadBeforePresent
|
|
57
77
|
) {
|
|
58
|
-
this.
|
|
78
|
+
this.subject$.next();
|
|
79
|
+
this.subject$.complete();
|
|
59
80
|
}
|
|
60
81
|
};
|
|
61
82
|
|
|
62
83
|
onLoadFailed = (index: number): void => {
|
|
84
|
+
const currentState = this.renderingState[index];
|
|
85
|
+
|
|
86
|
+
if (
|
|
87
|
+
COMPONENT_LOADING_STATE.LOADED_WITH_SUCCESS === currentState ||
|
|
88
|
+
COMPONENT_LOADING_STATE.LOADED_WITH_FAILURE === currentState
|
|
89
|
+
) {
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
|
|
63
93
|
this.renderingState[index] = COMPONENT_LOADING_STATE.LOADED_WITH_FAILURE;
|
|
64
94
|
|
|
65
95
|
if (
|
|
66
96
|
getNumberOfLoaded(this.renderingState) >=
|
|
67
97
|
this.numberOfComponentsWaitToLoadBeforePresent
|
|
68
98
|
) {
|
|
69
|
-
this.
|
|
99
|
+
this.subject$.next();
|
|
100
|
+
this.subject$.complete();
|
|
70
101
|
}
|
|
71
102
|
};
|
|
72
103
|
|
|
73
|
-
|
|
74
|
-
this.
|
|
75
|
-
|
|
104
|
+
dispose(): void {
|
|
105
|
+
this.subscription?.unsubscribe();
|
|
106
|
+
this.subject$.complete();
|
|
107
|
+
}
|
|
76
108
|
}
|
|
@@ -2,106 +2,123 @@ import {
|
|
|
2
2
|
ScreenRevealManager,
|
|
3
3
|
COMPONENT_LOADING_STATE,
|
|
4
4
|
} from "../ScreenRevealManager";
|
|
5
|
+
import { Subject } from "rxjs";
|
|
6
|
+
|
|
7
|
+
jest.mock("@applicaster/zapp-react-native-utils/arrayUtils", () => ({
|
|
8
|
+
makeListOf: jest.fn((value: any, length: number) =>
|
|
9
|
+
Array(length).fill(value)
|
|
10
|
+
),
|
|
11
|
+
}));
|
|
12
|
+
|
|
13
|
+
jest.mock("@applicaster/zapp-react-native-utils/componentsUtils", () => ({
|
|
14
|
+
isFirstComponentGallery: jest.fn(),
|
|
15
|
+
}));
|
|
16
|
+
|
|
17
|
+
jest.mock("@applicaster/zapp-react-native-utils/idleUtils", () => ({
|
|
18
|
+
withTimeout$: jest.fn(),
|
|
19
|
+
}));
|
|
20
|
+
|
|
21
|
+
import { makeListOf } from "@applicaster/zapp-react-native-utils/arrayUtils";
|
|
22
|
+
import { isFirstComponentGallery } from "@applicaster/zapp-react-native-utils/componentsUtils";
|
|
23
|
+
import { withTimeout$ } from "@applicaster/zapp-react-native-utils/idleUtils";
|
|
5
24
|
|
|
6
25
|
describe("ScreenRevealManager", () => {
|
|
7
|
-
|
|
26
|
+
let mockCallback: jest.Mock;
|
|
27
|
+
let timeout$: Subject<void>;
|
|
8
28
|
|
|
9
29
|
beforeEach(() => {
|
|
10
|
-
jest.
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
it("should initialize with the correct number of components to wait for", () => {
|
|
14
|
-
const componentsToRender: ZappUIComponent[] = [
|
|
15
|
-
{ component_type: "component1" },
|
|
16
|
-
{ component_type: "component2" },
|
|
17
|
-
{ component_type: "component3" },
|
|
18
|
-
];
|
|
30
|
+
jest.useFakeTimers();
|
|
31
|
+
mockCallback = jest.fn();
|
|
32
|
+
timeout$ = new Subject();
|
|
19
33
|
|
|
20
|
-
|
|
34
|
+
(withTimeout$ as jest.Mock).mockReturnValue(timeout$);
|
|
35
|
+
(isFirstComponentGallery as jest.Mock).mockReturnValue(false);
|
|
21
36
|
|
|
22
|
-
|
|
37
|
+
(makeListOf as jest.Mock).mockImplementation((value, length) =>
|
|
38
|
+
Array(length).fill(value)
|
|
39
|
+
);
|
|
40
|
+
});
|
|
23
41
|
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
]);
|
|
42
|
+
afterEach(() => {
|
|
43
|
+
jest.clearAllTimers();
|
|
44
|
+
jest.useRealTimers();
|
|
45
|
+
jest.resetAllMocks();
|
|
29
46
|
});
|
|
30
47
|
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
{ component_type: "component3" },
|
|
36
|
-
];
|
|
48
|
+
// ────────────────────────────────────────────────
|
|
49
|
+
it("should initialize with correct number of components and UNKNOWN state", () => {
|
|
50
|
+
const components = new Array(5).fill({});
|
|
51
|
+
const mgr = new ScreenRevealManager(components, mockCallback);
|
|
37
52
|
|
|
38
|
-
|
|
53
|
+
expect(mgr.numberOfComponentsWaitToLoadBeforePresent).toBe(3);
|
|
54
|
+
expect(makeListOf).toHaveBeenCalledWith(COMPONENT_LOADING_STATE.UNKNOWN, 3);
|
|
55
|
+
});
|
|
39
56
|
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
57
|
+
// ────────────────────────────────────────────────
|
|
58
|
+
it("should set numberOfComponentsWaitToLoadBeforePresent to 1 if first component is gallery", () => {
|
|
59
|
+
(isFirstComponentGallery as jest.Mock).mockReturnValue(true);
|
|
43
60
|
|
|
44
|
-
|
|
61
|
+
const components = new Array(5).fill({});
|
|
62
|
+
const mgr = new ScreenRevealManager(components, mockCallback);
|
|
63
|
+
|
|
64
|
+
expect(mgr.numberOfComponentsWaitToLoadBeforePresent).toBe(1);
|
|
45
65
|
});
|
|
46
66
|
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
{ component_type: "component3" },
|
|
52
|
-
];
|
|
67
|
+
// ────────────────────────────────────────────────
|
|
68
|
+
it("should trigger callback after all components load successfully", () => {
|
|
69
|
+
const components = new Array(3).fill({});
|
|
70
|
+
const mgr = new ScreenRevealManager(components, mockCallback);
|
|
53
71
|
|
|
54
|
-
|
|
72
|
+
mgr.onLoadFinished(0);
|
|
73
|
+
expect(mockCallback).not.toHaveBeenCalled();
|
|
55
74
|
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
manager.onLoadFailed(2);
|
|
75
|
+
mgr.onLoadFinished(1);
|
|
76
|
+
expect(mockCallback).not.toHaveBeenCalled();
|
|
59
77
|
|
|
78
|
+
mgr.onLoadFinished(2);
|
|
60
79
|
expect(mockCallback).toHaveBeenCalledTimes(1);
|
|
61
80
|
});
|
|
62
81
|
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
{ component_type: "component3" },
|
|
68
|
-
];
|
|
69
|
-
|
|
70
|
-
const manager = new ScreenRevealManager(componentsToRender, mockCallback);
|
|
82
|
+
// ────────────────────────────────────────────────
|
|
83
|
+
it("should trigger callback after some components fail to load but all finished", () => {
|
|
84
|
+
const components = new Array(3).fill({});
|
|
85
|
+
const mgr = new ScreenRevealManager(components, mockCallback);
|
|
71
86
|
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
87
|
+
mgr.onLoadFinished(0);
|
|
88
|
+
mgr.onLoadFailed(1);
|
|
89
|
+
mgr.onLoadFailed(2);
|
|
75
90
|
|
|
76
91
|
expect(mockCallback).toHaveBeenCalledTimes(1);
|
|
77
92
|
});
|
|
78
93
|
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
{ component_type: "component3" },
|
|
84
|
-
];
|
|
85
|
-
|
|
86
|
-
const manager = new ScreenRevealManager(componentsToRender, mockCallback);
|
|
94
|
+
// ────────────────────────────────────────────────
|
|
95
|
+
it("should not trigger callback twice when same component finishes twice", () => {
|
|
96
|
+
const components = new Array(3).fill({});
|
|
97
|
+
const mgr = new ScreenRevealManager(components, mockCallback);
|
|
87
98
|
|
|
88
|
-
|
|
89
|
-
|
|
99
|
+
mgr.onLoadFinished(0);
|
|
100
|
+
mgr.onLoadFinished(0); // duplicate
|
|
101
|
+
mgr.onLoadFinished(1);
|
|
102
|
+
mgr.onLoadFinished(2);
|
|
90
103
|
|
|
91
|
-
expect(mockCallback).
|
|
104
|
+
expect(mockCallback).toHaveBeenCalledTimes(1);
|
|
92
105
|
});
|
|
93
106
|
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
{ component_type: "component3" },
|
|
99
|
-
];
|
|
107
|
+
// ────────────────────────────────────────────────
|
|
108
|
+
it("should trigger callback when timeout$ emits before all loaded", () => {
|
|
109
|
+
const components = new Array(3).fill({});
|
|
110
|
+
new ScreenRevealManager(components, mockCallback);
|
|
100
111
|
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
manager.onLoadFinished(0);
|
|
112
|
+
timeout$.next(); // simulate timeout event from withTimeout$
|
|
104
113
|
|
|
105
114
|
expect(mockCallback).toHaveBeenCalledTimes(1);
|
|
106
115
|
});
|
|
116
|
+
|
|
117
|
+
// ────────────────────────────────────────────────
|
|
118
|
+
it("should not call callback if nothing loads and no timeout emitted", () => {
|
|
119
|
+
const components = new Array(3).fill({});
|
|
120
|
+
new ScreenRevealManager(components, mockCallback);
|
|
121
|
+
|
|
122
|
+
expect(mockCallback).not.toHaveBeenCalled();
|
|
123
|
+
});
|
|
107
124
|
});
|
|
@@ -8,7 +8,6 @@ import { isEmptyOrNil } from "@applicaster/zapp-react-native-utils/cellUtils";
|
|
|
8
8
|
import { focusManager } from "@applicaster/zapp-react-native-utils/appUtils/focusManager";
|
|
9
9
|
import { FocusableGroup } from "@applicaster/zapp-react-native-ui-components/Components/FocusableGroup";
|
|
10
10
|
import { Focusable } from "@applicaster/zapp-react-native-ui-components/Components/Focusable";
|
|
11
|
-
import { useAccessibilityManager } from "@applicaster/zapp-react-native-utils/appUtils/accessibilityManager/hooks";
|
|
12
11
|
import { Gutter } from "../Gutter";
|
|
13
12
|
import Tab from "./Tab";
|
|
14
13
|
import { getStyles } from "./styles";
|
|
@@ -29,14 +28,11 @@ const TabsComponent = ({
|
|
|
29
28
|
style,
|
|
30
29
|
selectedEntryIndex,
|
|
31
30
|
setSelectedEntry,
|
|
32
|
-
accessibility,
|
|
33
31
|
}: TabsProps & Partial<TabsSelectionContextType>) => {
|
|
34
32
|
const configuration = useConfiguration();
|
|
35
33
|
const config = applyFontConfig(configuration);
|
|
36
34
|
const styles = useMemo(() => getStyles(config), [config]);
|
|
37
35
|
|
|
38
|
-
const accessibilityManager = useAccessibilityManager({});
|
|
39
|
-
|
|
40
36
|
const {
|
|
41
37
|
tab_bar_gutter: horizontalGutter,
|
|
42
38
|
tab_bar_background_image: bgImage,
|
|
@@ -64,20 +60,10 @@ const TabsComponent = ({
|
|
|
64
60
|
);
|
|
65
61
|
|
|
66
62
|
const onListElementFocus = useCallback(
|
|
67
|
-
(index
|
|
68
|
-
if (isSelected) {
|
|
69
|
-
accessibilityManager.readText({
|
|
70
|
-
text: `${accessibility?.selectedHint} ${item.title}`,
|
|
71
|
-
});
|
|
72
|
-
} else {
|
|
73
|
-
accessibilityManager.readText({
|
|
74
|
-
text: `${accessibility?.hint} ${item.title}`,
|
|
75
|
-
});
|
|
76
|
-
}
|
|
77
|
-
|
|
63
|
+
(index) => {
|
|
78
64
|
scrollToSelectedIndex(index, VIEW_POSITION);
|
|
79
65
|
},
|
|
80
|
-
[scrollToSelectedIndex
|
|
66
|
+
[scrollToSelectedIndex]
|
|
81
67
|
);
|
|
82
68
|
|
|
83
69
|
const renderItem = useCallback(
|
|
@@ -108,7 +94,7 @@ const TabsComponent = ({
|
|
|
108
94
|
id={itemId}
|
|
109
95
|
testID={itemId}
|
|
110
96
|
preferredFocus={isSelected}
|
|
111
|
-
onFocus={() => onListElementFocus(index
|
|
97
|
+
onFocus={() => onListElementFocus(index)}
|
|
112
98
|
onPress={() => setSelectedEntry && setSelectedEntry(item)}
|
|
113
99
|
style={style}
|
|
114
100
|
>
|
|
@@ -163,9 +149,6 @@ const TabsComponent = ({
|
|
|
163
149
|
shouldUsePreferredFocus
|
|
164
150
|
isWithMemory={false}
|
|
165
151
|
nextFocusDown={parentFocus?.nextFocusDown}
|
|
166
|
-
onFocus={() => {
|
|
167
|
-
accessibilityManager.addHeading(accessibility?.announcement);
|
|
168
|
-
}}
|
|
169
152
|
>
|
|
170
153
|
<View style={tabs}>
|
|
171
154
|
<ImageBackground
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import React from "react";
|
|
1
|
+
import React, { useEffect } from "react";
|
|
2
2
|
import { equals } from "ramda";
|
|
3
3
|
import { Animated, ViewProps, ViewStyle } from "react-native";
|
|
4
4
|
import { useSafeAreaFrame } from "react-native-safe-area-context";
|
|
@@ -95,9 +95,22 @@ function SceneComponent({
|
|
|
95
95
|
});
|
|
96
96
|
|
|
97
97
|
const frame = useSafeAreaFrame();
|
|
98
|
+
|
|
99
|
+
const [memoFrame, setMemoFrame] = React.useState(frame);
|
|
100
|
+
|
|
101
|
+
useEffect(() => {
|
|
102
|
+
if (isActive) {
|
|
103
|
+
setMemoFrame((oldFrame) =>
|
|
104
|
+
oldFrame.width === frame.width && oldFrame.height === frame.height
|
|
105
|
+
? oldFrame
|
|
106
|
+
: frame
|
|
107
|
+
);
|
|
108
|
+
}
|
|
109
|
+
}, [isActive, frame.width, frame.height]);
|
|
110
|
+
|
|
98
111
|
const isAnimating = animating && overlayStyle;
|
|
99
112
|
|
|
100
|
-
const dimensions = isAnimating ? fullWidthDimensions :
|
|
113
|
+
const dimensions = isAnimating ? fullWidthDimensions : memoFrame;
|
|
101
114
|
|
|
102
115
|
return (
|
|
103
116
|
<Animated.View
|
package/index.d.ts
CHANGED
|
@@ -228,12 +228,6 @@ declare type TabsSelectionContextType = {
|
|
|
228
228
|
selectedEntryIndex: number;
|
|
229
229
|
};
|
|
230
230
|
|
|
231
|
-
declare type TabsAccessibility = {
|
|
232
|
-
announcement?: string;
|
|
233
|
-
selectedHint?: string;
|
|
234
|
-
hint?: string;
|
|
235
|
-
};
|
|
236
|
-
|
|
237
231
|
declare type TabsProps = {
|
|
238
232
|
entry: ZappEntry[];
|
|
239
233
|
groupId: string;
|
|
@@ -241,7 +235,6 @@ declare type TabsProps = {
|
|
|
241
235
|
focused?: boolean;
|
|
242
236
|
parentFocus?: ParentFocus;
|
|
243
237
|
style?: ReactViewStyle;
|
|
244
|
-
accessibility?: TabsAccessibility;
|
|
245
238
|
};
|
|
246
239
|
|
|
247
240
|
declare type TabContentProps = {
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@applicaster/zapp-react-native-ui-components",
|
|
3
|
-
"version": "15.0.0-alpha.
|
|
3
|
+
"version": "15.0.0-alpha.9331298916",
|
|
4
4
|
"description": "Applicaster Zapp React Native ui components for the Quick Brick App",
|
|
5
5
|
"main": "index.js",
|
|
6
6
|
"types": "index.d.ts",
|
|
@@ -28,10 +28,10 @@
|
|
|
28
28
|
},
|
|
29
29
|
"homepage": "https://github.com/applicaster/quickbrick#readme",
|
|
30
30
|
"dependencies": {
|
|
31
|
-
"@applicaster/applicaster-types": "15.0.0-alpha.
|
|
32
|
-
"@applicaster/zapp-react-native-bridge": "15.0.0-alpha.
|
|
33
|
-
"@applicaster/zapp-react-native-redux": "15.0.0-alpha.
|
|
34
|
-
"@applicaster/zapp-react-native-utils": "15.0.0-alpha.
|
|
31
|
+
"@applicaster/applicaster-types": "15.0.0-alpha.9331298916",
|
|
32
|
+
"@applicaster/zapp-react-native-bridge": "15.0.0-alpha.9331298916",
|
|
33
|
+
"@applicaster/zapp-react-native-redux": "15.0.0-alpha.9331298916",
|
|
34
|
+
"@applicaster/zapp-react-native-utils": "15.0.0-alpha.9331298916",
|
|
35
35
|
"promise": "^8.3.0",
|
|
36
36
|
"url": "^0.11.0",
|
|
37
37
|
"uuid": "^3.3.2"
|