我是React Native的新手,但是想知道是否可以从Java调用React Native JavaScript方法?我有一个小众设备,有一个物理按钮,我想用它来按下时触发一个方法。
我看过以下插件https://github.com/artemyarulin/react-native-eval,但Android实施似乎已过时。
我已经google了一下但是所有的例子我都可以找到从React Native调用Java方法的引用而不是相反。
非常感谢提前。
修改
我尝试使用@ide建议的方法实现解决方案。这是我到目前为止的尝试:
在我的MainActivity中,我有:
private String latestBarcode = "";
...
public String getLastestBarcode() {
return this.latestBarcode;
}
public void setLatestBarcode(String barcode) {
this.latestBarcode = barcode;
}
@Override
public void onBarcodeEvent(final BarcodeReadEvent event) {
runOnUiThread(new Runnable(){
@Override
public void run() {
String barcodeData = event.getBarcodeData();
String tstp = event.getTimestamp();
setLatestBarcode(barcodeData);
Log.d("MainActivity", " TEST - barcodeData "+ barcodeData + " tstp : "+ tstp);
}
});
}
...
当用户按下物理硬件按钮时,此代码从设备获取条形码值。
在我的自定义React Native模块
中@ReactMethod
public void latestBarcode(Callback callback) {
final Activity activity = getCurrentActivity();
if(activity != null && activity instanceof MainActivity){
callback.invoke(((MainActivity) activity).getLastestBarcode());
}
else {
callback.invoke("");
}
}
此方法获取值并将其公开给React Native。这工作我可以读取值,但我需要在用户按下按钮时触发一个方法,所以这里是我的React Native模块,因为它现在完全是:
import android.app.Activity;
import android.util.Log;
import com.facebook.react.bridge.NativeModule;
import com.facebook.react.bridge.Callback;
import com.facebook.react.bridge.ReactContextBaseJavaModule;
import com.facebook.react.bridge.ReactApplicationContext;
import com.facebook.react.bridge.ReactContext;
import com.facebook.react.bridge.ReactMethod;
import com.facebook.react.modules.core.DeviceEventManagerModule;
import com.facebook.react.bridge.WritableMap;
import com.facebook.react.bridge.Arguments;
import javax.annotation.Nullable;
// Import MainActivity
import com.myapp.MainActivity;
public class HoneywellCT50Module extends ReactContextBaseJavaModule {
private ReactContext reactContext;
public HoneywellCT50Module(ReactApplicationContext reactContext) {
super(reactContext);
this.reactContext = reactContext;
}
@ReactMethod
public void latestBarcode(Callback callback) {
final Activity activity = getCurrentActivity();
if(activity != null && activity instanceof MainActivity){
WritableMap params = Arguments.createMap(); // <--- Added WritableMap
sendEvent("BarcodeRecieved", params); // <--- Added sendEvent
callback.invoke(((MainActivity) activity).getLastestBarcode());
}
else {
callback.invoke("");
}
}
// Added sendEvent
private void sendEvent(
String eventName,
@Nullable WritableMap params
) {
this.reactContext
.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class)
.emit(eventName, params);
}
@Override
public String getName() {
return "HoneywellCT50";
}
}
在React Native中:
// I import DeviceEventEmitter
import { DeviceEventEmitter } from 'react-native'
componentDidMount () {
// Listen for Honeywell Scans
DeviceEventEmitter.addListener('BarcodeRecieved', this.getBarcodeValue)
}
// I want to trigger this method
getBarcodeValue () {
console.log('Event Received')
// This gets the barcode value, works ok
HoneywellCT50.latestBarcode((value) => {
this.setState({ barcode: value })
})
}
使用上面的代码我可以构建没有Java / build错误的项目,我看不到JS错误,但是我无法触发事件。任何人都可以从这里帮我指出正确的方向吗?
答案 0 :(得分:3)
您要做的是从Java发出一个发送到JavaScript的事件。您将编写一个自定义本机模块,从JavaScript订阅其事件,然后在您的本机代码中发出事件。请参阅本指南:http://facebook.github.io/react-native/docs/native-modules-android.html#sending-events-to-javascript
答案 1 :(得分:2)
好的,所以我得到了一些代码。我不确定这是否是最好的解决方案,但现在我是理解的极限,欢迎任何有关以下内容的反馈:
MainActivity.java
@Override
public void onBarcodeEvent(final BarcodeReadEvent event) {
runOnUiThread(new Runnable(){
@Override
public void run() {
String barcodeData = event.getBarcodeData();
String tstp = event.getTimestamp();
setLatestBarcode(barcodeData);
Log.d("MainActivity", " TEST - barcodeData "+ barcodeData + " tstp : "+ tstp);
}
});
WritableMap params = Arguments.createMap(); // <--- added
getReactInstanceManager().getCurrentReactContext() // <--- added
.getJSModule(DeviceEventManagerModule.RCTDeviceEventEmitter.class) // <--- added
.emit("BarcodeReceived", params); // <--- added
}
这可行我可以通过在我的JS中使用以下内容来触发React方法:
componentWillUnmount () {
this._barcodeListener.remove()
}
componentWillMount () {
// Listen for Honeywell Scans
this.getBarcodeValue = DeviceEventEmitter.addListener('BarcodeReceived', this.getBarcodeValue)
}
...
getBarcodeValue () {
console.log('Event Received')
HoneywellCT50.latestBarcode((value) => {
this.setState({ barcode: value }, () => {
console.log(this.state.barcode)
})
})
}
如果有经验的人读到这个,我很想知道为什么我上面的编辑中发布的代码不起作用。是因为我尝试在@ReactMethod
?