-
Notifications
You must be signed in to change notification settings - Fork 2.6k
Expand file tree
/
Copy pathReactView.java
More file actions
246 lines (209 loc) · 8.07 KB
/
ReactView.java
File metadata and controls
246 lines (209 loc) · 8.07 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
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
package com.reactnativenavigation.react;
import static com.reactnativenavigation.utils.CoordinatorLayoutUtils.matchParentLP;
import android.annotation.SuppressLint;
import android.content.Context;
import android.os.Bundle;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.FrameLayout;
import androidx.annotation.RestrictTo;
import com.facebook.react.ReactApplication;
import com.facebook.react.ReactHost;
import com.facebook.react.bridge.ReactContext;
import com.facebook.react.interfaces.fabric.ReactSurface;
import com.facebook.react.uimanager.UIManagerHelper;
import com.facebook.react.uimanager.common.UIManagerType;
import com.facebook.react.uimanager.events.EventDispatcher;
import com.reactnativenavigation.react.events.ComponentType;
import com.reactnativenavigation.react.events.EventEmitter;
import com.reactnativenavigation.viewcontrollers.viewcontroller.IReactView;
import com.reactnativenavigation.viewcontrollers.viewcontroller.ScrollEventListener;
import com.reactnativenavigation.views.component.Renderable;
@SuppressLint("ViewConstructor")
public class ReactView extends FrameLayout implements IReactView, Renderable {
private static final String TAG = "RNN.ReactView";
/**
* Upper bound on deferred lifecycle retries (~20s at ~60fps-style scheduling).
* After this we emit with React context even if {@link #isRendered()} is still false,
* matching legacy behavior for edge cases.
*/
private static final int MAX_DEFERRED_LIFECYCLE_RETRIES = 1200;
private final String componentId;
private final String componentName;
private boolean isAttachedToReactInstance = false;
private final ReactSurface reactSurface;
private boolean pendingWillAppear;
private boolean pendingDidAppear;
private ComponentType pendingWillType;
private ComponentType pendingDidType;
private int deferredLifecycleRetryCount;
private boolean deferredLifecycleFlushPosted;
public ReactView(final Context context, String componentId, String componentName) {
super(context);
this.componentId = componentId;
this.componentName = componentName;
final Bundle opts = new Bundle();
opts.putString("componentId", componentId);
reactSurface = getReactHost().createSurface(context, componentName, opts);
addView(reactSurface.getView(), matchParentLP());
}
@Override
protected void onAttachedToWindow() {
super.onAttachedToWindow();
start();
}
public void start() {
if (isAttachedToReactInstance) return;
isAttachedToReactInstance = true;
reactSurface.start();
}
@Override
public boolean isReady() {
return isAttachedToReactInstance;
}
@Override
public ReactView asView() {
return this;
}
@Override
public void destroy() {
clearDeferredLifecycleEvents();
reactSurface.stop();
}
public void sendComponentWillStart(ComponentType type) {
this.post(() -> handleSendComponentWillStart(type));
}
public void sendComponentStart(ComponentType type) {
this.post(() -> handleSendComponentStart(type));
}
public void sendComponentStop(ComponentType type) {
clearDeferredLifecycleEvents();
ReactContext currentReactContext = getReactContext();
if (currentReactContext != null) {
new EventEmitter(currentReactContext).emitComponentDidDisappear(componentId, componentName, type);
}
}
private void handleSendComponentWillStart(ComponentType type) {
if (!shouldDeferLifecycleEmit()) {
emitComponentWillAppear(type);
return;
}
pendingWillAppear = true;
pendingWillType = type;
scheduleFlushDeferredLifecycleEvents();
}
private void handleSendComponentStart(ComponentType type) {
if (!shouldDeferLifecycleEmit()) {
emitComponentDidAppear(type);
return;
}
pendingDidAppear = true;
pendingDidType = type;
scheduleFlushDeferredLifecycleEvents();
}
/**
* Wait until the Fabric surface has mounted content so JS has typically committed the screen and
* {@code bindComponent} can receive appear events (parity with iOS {@code RNNReactView} pending appear).
*/
private boolean shouldDeferLifecycleEmit() {
return getReactContext() == null || !isRendered();
}
private void scheduleFlushDeferredLifecycleEvents() {
if (deferredLifecycleFlushPosted) return;
deferredLifecycleFlushPosted = true;
post(() -> {
deferredLifecycleFlushPosted = false;
flushDeferredLifecycleEvents();
});
}
private void flushDeferredLifecycleEvents() {
if (!pendingWillAppear && !pendingDidAppear) {
deferredLifecycleRetryCount = 0;
return;
}
boolean strictReady = !shouldDeferLifecycleEmit();
if (!strictReady) {
if (++deferredLifecycleRetryCount <= MAX_DEFERRED_LIFECYCLE_RETRIES) {
scheduleFlushDeferredLifecycleEvents();
return;
}
deferredLifecycleRetryCount = 0;
if (getReactContext() == null) {
Log.w(TAG, "Deferred component lifecycle events dropped (React context not ready)");
clearDeferredLifecycleEvents();
return;
}
// Degraded: context exists but surface not reporting children yet — emit once (legacy timing).
} else {
deferredLifecycleRetryCount = 0;
}
if (pendingWillAppear) {
pendingWillAppear = false;
emitComponentWillAppear(pendingWillType);
}
if (pendingDidAppear) {
pendingDidAppear = false;
emitComponentDidAppear(pendingDidType);
}
}
private void clearDeferredLifecycleEvents() {
pendingWillAppear = false;
pendingDidAppear = false;
deferredLifecycleRetryCount = 0;
deferredLifecycleFlushPosted = false;
}
private void emitComponentWillAppear(ComponentType type) {
ReactContext currentReactContext = getReactContext();
if (currentReactContext != null) {
new EventEmitter(currentReactContext).emitComponentWillAppear(componentId, componentName, type);
}
}
private void emitComponentDidAppear(ComponentType type) {
ReactContext currentReactContext = getReactContext();
if (currentReactContext != null) {
new EventEmitter(currentReactContext).emitComponentDidAppear(componentId, componentName, type);
}
}
@Override
public void sendOnNavigationButtonPressed(String buttonId) {
ReactContext currentReactContext = getReactContext();
if (currentReactContext != null) {
new EventEmitter(currentReactContext).emitOnNavigationButtonPressed(componentId, buttonId);
}
}
@Override
public ScrollEventListener getScrollEventListener() {
return new ScrollEventListener(getEventDispatcher());
}
@Override
public void dispatchTouchEventToJs(MotionEvent event) {
View view = reactSurface.getView();
if (view != null) {
view.onTouchEvent(event);
}
}
@Override
public boolean isRendered() {
ViewGroup view = reactSurface.getView();
if (view != null) {
return view.getChildCount() >= 1;
}
return false;
}
public EventDispatcher getEventDispatcher() {
ReactContext reactContext = getReactContext();
return reactContext == null ? null : UIManagerHelper.getEventDispatcher(reactContext, UIManagerType.FABRIC);
}
@RestrictTo(RestrictTo.Scope.TESTS)
public String getComponentName() {
return componentName;
}
private ReactHost getReactHost() {
return ((ReactApplication) getContext().getApplicationContext()).getReactHost();
}
private ReactContext getReactContext() {
return getReactHost().getCurrentReactContext();
}
}