Skip to content

Commit ffbd288

Browse files
authored
Support react-native-youtube (#5903)
react-native-youtubte is a popular library which wraps the native youtube library. The native lib attempts to detect if the player is hidden behind other views in order to prevent developers from playing videos in the background. Since the overlay container was always attached to hierarchy, the library stopped playback as it mistakingly detected the player was used in the background. This commit simply attaches the overlay container only when needed so as long as no overlays are displayed, the lib can be used.
1 parent 030d756 commit ffbd288

File tree

4 files changed

+65
-18
lines changed

4 files changed

+65
-18
lines changed
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,28 @@
11
package com.reactnativenavigation.presentation;
22

3+
import android.view.View;
34
import android.view.ViewGroup;
45

56
import com.reactnativenavigation.utils.CommandListener;
7+
import com.reactnativenavigation.utils.ViewUtils;
68
import com.reactnativenavigation.viewcontrollers.ViewController;
79
import com.reactnativenavigation.views.BehaviourDelegate;
810

911
import java.util.HashMap;
1012

13+
import static com.reactnativenavigation.utils.CollectionUtils.*;
1114
import static com.reactnativenavigation.utils.CoordinatorLayoutUtils.matchParentWithBehaviour;
1215

1316
public class OverlayManager {
1417
private final HashMap<String, ViewController> overlayRegistry = new HashMap<>();
18+
private ViewGroup contentLayout;
19+
20+
public void setContentLayout(ViewGroup contentLayout) {
21+
this.contentLayout = contentLayout;
22+
}
1523

1624
public void show(ViewGroup overlaysContainer, ViewController overlay, CommandListener listener) {
25+
if (overlaysContainer.getParent() == null) contentLayout.addView(overlaysContainer);
1726
overlayRegistry.put(overlay.getId(), overlay);
1827
overlay.addOnAppearedListener(() -> listener.onSuccess(overlay.getId()));
1928
overlaysContainer.addView(overlay.getView(), matchParentWithBehaviour(new BehaviourDelegate(overlay)));
@@ -24,17 +33,14 @@ public void dismiss(String componentId, CommandListener listener) {
2433
if (overlay == null) {
2534
listener.onError("Could not dismiss Overlay. Overlay with id " + componentId + " was not found.");
2635
} else {
27-
overlay.destroy();
28-
overlayRegistry.remove(componentId);
36+
destroyOverlay(overlay);
2937
listener.onSuccess(componentId);
3038
}
3139
}
3240

3341
public void destroy() {
34-
for (ViewController view : overlayRegistry.values()) {
35-
view.destroy();
36-
}
37-
overlayRegistry.clear();
42+
forEach(overlayRegistry.values(), this::destroyOverlay);
43+
contentLayout = null;
3844
}
3945

4046
public int size() {
@@ -44,4 +50,15 @@ public int size() {
4450
public ViewController findControllerById(String id) {
4551
return overlayRegistry.get(id);
4652
}
53+
54+
private void destroyOverlay(ViewController overlay) {
55+
View parent = (View) overlay.getView().getParent();
56+
overlay.destroy();
57+
overlayRegistry.remove(overlay.getId());
58+
if (isEmpty()) ViewUtils.removeFromParent(parent);
59+
}
60+
61+
private boolean isEmpty() {
62+
return size() == 0;
63+
}
4764
}

lib/android/app/src/main/java/com/reactnativenavigation/viewcontrollers/navigator/Navigator.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -64,9 +64,9 @@ public void setEventEmitter(EventEmitter eventEmitter) {
6464

6565
public void setContentLayout(ViewGroup contentLayout) {
6666
this.contentLayout = contentLayout;
67+
overlayManager.setContentLayout(contentLayout);
6768
contentLayout.addView(rootLayout);
6869
contentLayout.addView(modalsLayout);
69-
contentLayout.addView(overlaysLayout);
7070
}
7171

7272
public Navigator(final Activity activity, ChildControllersRegistry childRegistry, ModalStack modalStack, OverlayManager overlayManager, RootPresenter rootPresenter) {

lib/android/app/src/test/java/com/reactnativenavigation/viewcontrollers/navigator/NavigatorTest.java

+11-2
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
import android.os.Bundle;
44
import android.view.View;
5+
import android.view.ViewGroup;
56
import android.widget.FrameLayout;
67

78
import com.facebook.react.ReactInstanceManager;
@@ -118,6 +119,14 @@ public boolean onViewDisappear(View view) {
118119
activityController.postCreate(Bundle.EMPTY);
119120
}
120121

122+
@Test
123+
public void setContentLayout() {
124+
ViewGroup contentLayout = Mockito.mock(ViewGroup.class);
125+
uut.setContentLayout(contentLayout);
126+
127+
verify(overlayManager).setContentLayout(contentLayout);
128+
}
129+
121130
@Test
122131
public void bindViews() {
123132
verify(rootPresenter).setRootContainer(uut.getRootLayout());
@@ -149,11 +158,11 @@ public void setRoot_delegatesToRootPresenter() {
149158
@Test
150159
public void setRoot_clearsSplashLayout() {
151160
FrameLayout content = activity.findViewById(android.R.id.content);
152-
assertThat(content.getChildCount()).isEqualTo(4); // 3 frame layouts and the default splash layout
161+
assertThat(content.getChildCount()).isEqualTo(3); // 2 frame layouts (root and modal containers) and the default splash layout
153162

154163
uut.setRoot(child2, new CommandListenerAdapter(), reactInstanceManager);
155164

156-
assertThat(content.getChildCount()).isEqualTo(3);
165+
assertThat(content.getChildCount()).isEqualTo(2);
157166
}
158167

159168
@Test

lib/android/app/src/test/java/com/reactnativenavigation/viewcontrollers/overlay/OverlayManagerTest.java

+30-9
Original file line numberDiff line numberDiff line change
@@ -26,33 +26,43 @@ public class OverlayManagerTest extends BaseTest {
2626
private OverlayManager uut;
2727
private SimpleViewController overlay1;
2828
private SimpleViewController overlay2;
29-
private FrameLayout root;
29+
private FrameLayout contentLayout;
30+
private FrameLayout overlayContainer;
3031

3132
@Override
3233
public void beforeEach() {
3334
Activity activity = newActivity();
34-
root = new FrameLayout(activity);
35-
root.layout(0, 0, 1000, 1000);
36-
activity.setContentView(root);
35+
contentLayout = new FrameLayout(activity);
36+
contentLayout.layout(0, 0, 1000, 1000);
37+
activity.setContentView(contentLayout);
38+
overlayContainer = new FrameLayout(activity);
3739

3840
ChildControllersRegistry childRegistry = new ChildControllersRegistry();
3941
overlay1 = spy(new SimpleViewController(activity, childRegistry, OVERLAY_ID_1, new Options()));
4042
overlay2 = spy(new SimpleViewController(activity, childRegistry, OVERLAY_ID_2, new Options()));
4143
uut = new OverlayManager();
44+
uut.setContentLayout(contentLayout);
45+
}
46+
47+
@Test
48+
public void show_attachesOverlayContainerToContentLayout() {
49+
uut.show(overlayContainer, overlay1, new CommandListenerAdapter());
50+
assertThat(overlayContainer.getParent()).isEqualTo(contentLayout);
51+
uut.show(overlayContainer, overlay2, new CommandListenerAdapter());
4252
}
4353

4454
@Test
4555
public void show() {
4656
CommandListenerAdapter listener = spy(new CommandListenerAdapter());
47-
uut.show(root, overlay1, listener);
57+
uut.show(overlayContainer, overlay1, listener);
4858
verify(listener, times(1)).onSuccess(OVERLAY_ID_1);
49-
assertThat(overlay1.getView().getParent()).isEqualTo(root);
59+
assertThat(overlay1.getView().getParent()).isEqualTo(overlayContainer);
5060
assertMatchParent(overlay1.getView());
5161
}
5262

5363
@Test
5464
public void dismiss() {
55-
uut.show(root, overlay1, new CommandListenerAdapter());
65+
uut.show(overlayContainer, overlay1, new CommandListenerAdapter());
5666
assertThat(uut.size()).isOne();
5767
CommandListener listener = spy(new CommandListenerAdapter());
5868
uut.dismiss(overlay1.getId(), listener);
@@ -70,11 +80,22 @@ public void dismiss_rejectIfOverlayNotFound() {
7080

7181
@Test
7282
public void dismiss_onViewReturnedToFront() {
73-
uut.show(root, overlay1, new CommandListenerAdapter());
74-
uut.show(root, overlay2, new CommandListenerAdapter());
83+
uut.show(overlayContainer, overlay1, new CommandListenerAdapter());
84+
uut.show(overlayContainer, overlay2, new CommandListenerAdapter());
7585
verify(overlay1, times(0)).onViewBroughtToFront();
7686

7787
uut.dismiss(OVERLAY_ID_2, new CommandListenerAdapter());
7888
verify(overlay1, times(1)).onViewBroughtToFront();
7989
}
90+
91+
@Test
92+
public void dismiss_overlayContainerIsRemovedIfAllOverlaysAreDismissed() {
93+
uut.show(overlayContainer, overlay1, new CommandListenerAdapter());
94+
uut.show(overlayContainer, overlay2, new CommandListenerAdapter());
95+
96+
uut.dismiss(OVERLAY_ID_2, new CommandListenerAdapter());
97+
assertThat(overlayContainer.getParent()).isEqualTo(contentLayout);
98+
uut.dismiss(OVERLAY_ID_1, new CommandListenerAdapter());
99+
assertThat(overlayContainer.getParent()).isNull();
100+
}
80101
}

0 commit comments

Comments
 (0)