-
Notifications
You must be signed in to change notification settings - Fork 24.4k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Push ReactContext logic in derived classes (#44026)
Summary: Pull Request resolved: #44026 Changelog: [Android][Removed] Delete ReactContext.initializeWithInstance(). ReactContext now no longer contains legacy react instance methods. Please use BridgeReactInstance instead. Yet another attempt to land this (last one was D55505416). Copy-pasting below the amazing summary from RSNara. ## Context Prior, ReactContext used to implement bridge logic. For bridgeless mode, we created BridgelessReactContext < ReactContext ## Problem This could lead to failures: we could call bridge methods in bridgeless mode. ## Changes Primary change: - Make all the react instance methods inside ReactContext abstract. Secondary changes: Implement react instance methods in concrete subclasses: - **New:** BridgeReactContext: By delegating to CatalystInstance - **New:** ThemedReactContext: By delegating to inner ReactContext - **Unchanged:** BridgelessReactContext: By delegating to ReactHost ## Auxiliary changes This fixes ThemedReactContext in bridgeless mode. **Problem:** Prior, ThemedReactContext's react instance methods did not work in bridgeless mode: ThemedReactContext wasn't initialized in bridgeless mode, so all those methods had undefined behaviour. **Solution:** ThemedReactContext now implements all react instance methods, by just forwarding to the initialized ReactContext it decorates (which has an instance). NOTE: Intentionally not converting `BridgeReactContext` to Kotlin to minimize the risk of these changes. Reviewed By: RSNara Differential Revision: D55964787
- Loading branch information
1 parent
44d59ea
commit afba36b
Showing
6 changed files
with
350 additions
and
200 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
190 changes: 190 additions & 0 deletions
190
...react-native/ReactAndroid/src/main/java/com/facebook/react/bridge/BridgeReactContext.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,190 @@ | ||
/* | ||
* Copyright (c) Meta Platforms, Inc. and affiliates. | ||
* | ||
* This source code is licensed under the MIT license found in the | ||
* LICENSE file in the root directory of this source tree. | ||
*/ | ||
|
||
package com.facebook.react.bridge; | ||
|
||
import android.content.Context; | ||
import androidx.annotation.Nullable; | ||
import com.facebook.common.logging.FLog; | ||
import com.facebook.infer.annotation.Assertions; | ||
import com.facebook.infer.annotation.Nullsafe; | ||
import com.facebook.react.bridge.queue.ReactQueueConfiguration; | ||
import com.facebook.react.common.ReactConstants; | ||
import com.facebook.react.common.annotations.DeprecatedInNewArchitecture; | ||
import java.util.Collection; | ||
|
||
/** | ||
* This is the bridge-specific concrete subclass of ReactContext. ReactContext has many methods that | ||
* delegate to the react instance. This subclass implements those methods, by delegating to the | ||
* CatalystInstance. If you need to create a ReactContext within an "bridge context", please create | ||
* BridgeReactContext. | ||
*/ | ||
@DeprecatedInNewArchitecture | ||
@Nullsafe(Nullsafe.Mode.LOCAL) | ||
public class BridgeReactContext extends ReactApplicationContext { | ||
|
||
private static final String TAG = "BridgeReactContext"; | ||
|
||
private static final String EARLY_CALL_EXCEPTION_MESSAGE = | ||
"Called %s before setting the Catalyst instance!"; | ||
|
||
private static final String LATE_CALL_EXCEPTION_MESSAGE = | ||
"Called %s after destroying the Catalyst instance!"; | ||
|
||
private volatile boolean mDestroyed = false; | ||
private @Nullable CatalystInstance mCatalystInstance; | ||
|
||
public BridgeReactContext(Context context) { | ||
super(context); | ||
} | ||
|
||
public void initializeWithInstance(CatalystInstance catalystInstance) { | ||
if (catalystInstance == null) { | ||
throw new IllegalArgumentException("CatalystInstance cannot be null."); | ||
} | ||
if (mCatalystInstance != null) { | ||
throw new IllegalStateException("ReactContext has been already initialized"); | ||
} | ||
if (mDestroyed) { | ||
ReactSoftExceptionLogger.logSoftException( | ||
TAG, | ||
new IllegalStateException("Cannot initialize ReactContext after it has been destroyed.")); | ||
} | ||
|
||
mCatalystInstance = catalystInstance; | ||
|
||
ReactQueueConfiguration queueConfig = catalystInstance.getReactQueueConfiguration(); | ||
initializeMessageQueueThreads(queueConfig); | ||
initializeInteropModules(); | ||
} | ||
|
||
@Override | ||
public void destroy() { | ||
UiThreadUtil.assertOnUiThread(); | ||
|
||
mDestroyed = true; | ||
if (mCatalystInstance != null) { | ||
mCatalystInstance.destroy(); | ||
} | ||
} | ||
|
||
@Override | ||
public CatalystInstance getCatalystInstance() { | ||
return Assertions.assertNotNull(mCatalystInstance); | ||
} | ||
|
||
@Override | ||
public @Nullable UIManager getFabricUIManager() { | ||
CatalystInstance catalystInstance = getNonNullableCatalystInstanceOrThrow("getFabricUIManager"); | ||
UIManager uiManager = catalystInstance.getFabricUIManager(); | ||
return uiManager != null | ||
? uiManager | ||
: (UIManager) catalystInstance.getJSIModule(JSIModuleType.UIManager); | ||
} | ||
|
||
@Override | ||
public @Nullable JavaScriptContextHolder getJavaScriptContextHolder() { | ||
return mCatalystInstance != null ? mCatalystInstance.getJavaScriptContextHolder() : null; | ||
} | ||
|
||
@Override | ||
public <T extends JavaScriptModule> T getJSModule(Class<T> jsInterface) { | ||
CatalystInstance catalystInstance = getNonNullableCatalystInstanceOrThrow("getJSModule"); | ||
|
||
if (mInteropModuleRegistry != null | ||
&& mInteropModuleRegistry.shouldReturnInteropModule(jsInterface)) { | ||
return mInteropModuleRegistry.getInteropModule(jsInterface); | ||
} | ||
|
||
return catalystInstance.getJSModule(jsInterface); | ||
} | ||
|
||
@Override | ||
public @Nullable <T extends NativeModule> T getNativeModule(Class<T> nativeModuleInterface) { | ||
CatalystInstance catalystInstance = getNonNullableCatalystInstanceOrThrow("getNativeModule"); | ||
return catalystInstance.getNativeModule(nativeModuleInterface); | ||
} | ||
|
||
@Override | ||
public Collection<NativeModule> getNativeModules() { | ||
CatalystInstance catalystInstance = getNonNullableCatalystInstanceOrThrow("getNativeModules"); | ||
return catalystInstance.getNativeModules(); | ||
} | ||
|
||
@Override | ||
public @Nullable RuntimeExecutor getRuntimeExecutor() { | ||
CatalystInstance catalystInstance = getNonNullableCatalystInstanceOrThrow("getRuntimeExecutor"); | ||
return catalystInstance.getRuntimeExecutor(); | ||
} | ||
|
||
@Override | ||
public @Nullable String getSourceURL() { | ||
return mCatalystInstance != null ? mCatalystInstance.getSourceURL() : null; | ||
} | ||
|
||
@Override | ||
public void handleException(Exception e) { | ||
JSExceptionHandler jsExceptionHandler = getJSExceptionHandler(); | ||
if (hasActiveReactInstance() && jsExceptionHandler != null) { | ||
jsExceptionHandler.handleException(e); | ||
} else { | ||
FLog.e( | ||
ReactConstants.TAG, | ||
"Unable to handle Exception - catalystInstanceVariableExists: " | ||
+ (mCatalystInstance != null) | ||
+ " - isCatalystInstanceAlive: " | ||
+ hasActiveReactInstance() | ||
+ " - hasExceptionHandler: " | ||
+ (jsExceptionHandler != null), | ||
e); | ||
throw new IllegalStateException(e); | ||
} | ||
} | ||
|
||
@Deprecated | ||
@Override | ||
public boolean hasActiveCatalystInstance() { | ||
return hasActiveReactInstance(); | ||
} | ||
|
||
@Deprecated | ||
@Override | ||
public boolean hasActiveReactInstance() { | ||
return mCatalystInstance != null && !mCatalystInstance.isDestroyed(); | ||
} | ||
|
||
@Override | ||
public boolean hasCatalystInstance() { | ||
return mCatalystInstance != null; | ||
} | ||
|
||
@Override | ||
public <T extends NativeModule> boolean hasNativeModule(Class<T> nativeModuleInterface) { | ||
CatalystInstance catalystInstance = getNonNullableCatalystInstanceOrThrow("hasNativeModule"); | ||
return catalystInstance.hasNativeModule(nativeModuleInterface); | ||
} | ||
|
||
@Override | ||
public boolean isBridgeless() { | ||
return false; | ||
} | ||
|
||
@Override | ||
public void registerSegment(int segmentId, String path, Callback callback) { | ||
Assertions.assertNotNull(mCatalystInstance).registerSegment(segmentId, path); | ||
Assertions.assertNotNull(callback).invoke(); | ||
} | ||
|
||
private CatalystInstance getNonNullableCatalystInstanceOrThrow(String accessTarget) { | ||
if (mCatalystInstance != null) { | ||
return mCatalystInstance; | ||
} | ||
throw new IllegalStateException( | ||
String.format( | ||
mDestroyed ? LATE_CALL_EXCEPTION_MESSAGE : EARLY_CALL_EXCEPTION_MESSAGE, accessTarget)); | ||
} | ||
} |
20 changes: 0 additions & 20 deletions
20
...s/react-native/ReactAndroid/src/main/java/com/facebook/react/bridge/BridgeReactContext.kt
This file was deleted.
Oops, something went wrong.
Oops, something went wrong.