React Native 與 Android原生Activity互相跳轉頁面,RN作為混合開發,肯定需要與原生直接的頁面跳轉,這裡也屬於和原生端通信的知識模塊。我們知道Android的頁面跳轉是通過Intent、Rn是通過路由,而兩者直接頁面互相跳轉就需要原生借助JS暴露接口給Rn來實現瞭。閑話不說直接上圖:
一、AS創建一個普通的Activity類,顯示HelloWorld
@Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(createView()); } private View createView() { LinearLayout ll= new LinearLayout(this); ll.setGravity(Gravity.CENTER); ll.setLayoutParams(new RelativeLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT)); // 設置文字 TextView mTextView = new TextView(this); mTextView.setText("hello world"); LinearLayout.LayoutParams mLayoutParams = new LinearLayout.LayoutParams( ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT); // 在父類佈局中添加它,及佈局樣式 ll.addView(mTextView, mLayoutParams); return ll; }
二、添加MyIntentModule類,並繼承ReactContextBaseJavaModule實現其方法和構造函數。在該類中添加方法,註意:方法頭要加@ReactMethod
/** * 原生Activity與React交互——模塊 */ public class MyIntentModule extends ReactContextBaseJavaModule { public MyIntentModule(ReactApplicationContext reactContext) { super(reactContext); } @Override public String getName() { return "IntentMoudle"; } //註意:記住getName方法中的命名名稱,JS中調用需要 @ReactMethod public void startActivityFromJS(String name, String params){ try{ Activity currentActivity = getCurrentActivity(); if(null!=currentActivity){ Class toActivity = Class.forName(name); Intent intent = new Intent(currentActivity,toActivity); intent.putExtra("params", params); currentActivity.startActivity(intent); } }catch(Exception e){ throw new JSApplicationIllegalArgumentException( "不能打開Activity : "+e.getMessage()); } } @ReactMethod public void dataToJS(Callback successBack, Callback errorBack){ try{ Activity currentActivity = getCurrentActivity(); String result = currentActivity.getIntent().getStringExtra("data"); if (TextUtils.isEmpty(result)){ result = "沒有數據"; } successBack.invoke(result); }catch (Exception e){ errorBack.invoke(e.getMessage()); } } //註意:startActivityFromJS、dataToJS方法添加RN註解(@ReactMethod),否則該方法將不被添加到RN中 }
三、添加MyReactPackage類,實現ReactPackage接口裡的方法暴露給RN調用,在重寫方法createNativeModules裡註冊上一步添加的模塊:
/** * 註冊模塊 */ public class MyReactPackage implements ReactPackage { @Override public List createNativeModules(ReactApplicationContext reactContext) { return Arrays.asList(new MyIntentModule(reactContext)); } @Override public List createViewManagers(ReactApplicationContext reactContext) { return Collections.emptyList(); } }
四、接著在MainApplication中的getPackages方法中註冊到ReactPackage中:
@Override protected List getPackages() { return Arrays.asList( new MainReactPackage(), new MyReactPackage() ); }
五、接下來的工作便是RN的Index.js代碼:
*從RN跳轉到Android
import React, { Component } from 'react'; import { View, NativeModules, TouchableNativeFeedback, ToastAndroid } from 'react-native';
export default class App extends Component<{}> { _onPressButton() { console.log("You tapped the button!"); NativeModules .IntentMoudle .startActivityFromJS("com.myreactdemo.MyActivity", null); } render() { return ( <View> <TouchableNativeFeedback onPress={this._onPressButton}> <Text>跳轉到原生頁面<!--Text> <!--TouchableNativeFeedback> <!--View> ); } }
*從Android跳轉到RN
可以在rn中拿到activity跳轉傳遞的值,值的傳遞跟普通activity之間的跳轉沒有差別:
[javascript]view plaincopy
- getData(){
- NativeModules.IntentModule.dataToJS((msg)=>{
- console.log(msg);
- letbase=require('./constant');
- base.columnID=msg;
- //ToastAndroid.show('JS界面:從Activity中傳輸過來的數據為:'+base.columnID,ToastAndroid.SHORT);
- },
- (result)=>{
- ToastAndroid.show('JS界面:錯誤信息為:'+result,ToastAndroid.SHORT);
- })
- }
拿到這個值之後存在瞭常量類裡,就是通過這個常量來實現跳轉到不同的界面,之後的事情就迎刃而解瞭:
[java]view plaincopy
- componentDidMount(){
- letbase=require('./constant');
- //ToastAndroid.show(base.columnID,ToastAndroid.SHORT);
- letid=base.columnID;
- if(id=="3"){
- const{navigator}=this.props;
- if(navigator){
- navigator.push({
- name:'secondPage',
- component:secondPage,
- })
- }
- }if(id=="4"){
- const{navigator}=this.props;
- if(navigator){
- navigator.push({
- name:'otherPage',
- component:otherPage,
- })
- }
- }
- }