前言
RN作为混合开发,肯定需要与原生直接的页面跳转,这里也属于和原生端通信的知识模块。我们知道Android的页面跳转是通过Intent、Rn是通过路由,而两者直接页面互相跳转就需要原生借助JS暴露接口给Rn来实现了。先上效果图:
Demo源码地址:https://github.com/aiyangtianci/MyReactDemo
运行项目:React Native 配置环境,运行第一个RN项目
集成项目:React Native 教程——集成到现有原生应用
第一步,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<NativeModule> createNativeModules(ReactApplicationContext reactContext) {
return Arrays.<NativeModule>asList(new MyIntentModule(reactContext));
}
@Override
public List<ViewManager> createViewManagers(ReactApplicationContext reactContext) {
return Collections.emptyList();
}
}
第四步,在MainApplication中的getPackages方法中注册到ReactPackage中:
@Override
protected List<ReactPackage> getPackages() {
return Arrays.<ReactPackage>asList(
new MainReactPackage(),
new MyReactPackage()
);
}
第五步,接下来的工作便是RN的Index.js代码:
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 plain copy
getData() {
NativeModules.IntentModule.dataToJS((msg) => {
console.log(msg);
let base = require('./constant');
base.columnID = msg;
//ToastAndroid.show('JS界面:从Activity中传输过来的数据为:' + base.columnID, ToastAndroid.SHORT);
},
(result) => {
ToastAndroid.show('JS界面:错误信息为:' + result, ToastAndroid.SHORT);
})
}
拿到这个值之后存在了常量类里,就是通过这个常量来实现跳转到不同的界面,之后的事情就迎刃而解了:
[java] view plain copy
componentDidMount() {
let base = require('./constant');
//ToastAndroid.show(base.columnID, ToastAndroid.SHORT);
let id = 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,
})
}
}
}
第七步,运行安装:
输入命令启动应用:
1.进入项目根目录:
cd MyReactDemo
2.运行:
react-native run-android
*如果设备没安装上AwesomeProject,可直接用Android studio或eclipse开发工具打开AwesomeProject文件里的Android项目,点击run运行安装即可。运行应用首先需要启动开发服务器(Packager)
npm start
学习分享,共勉
题外话,我从事Android开发已经五年了,此前我指导过不少同行。但很少跟大家一起探讨,正好最近我花了一个多月的时间整理出来一份包括不限于高级UI、性能优化、移动架构师、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter等全方面的Android进阶实践技术,今天暂且开放给有需要的人,若有关于此方面可以转发+关注+点赞后领取,或者评论与我一起交流探讨。
资料免费领取方式:转发+关注+点赞后,加入点击链接加入群聊:Android高级开发交流群(818520403)即可获取免费领取方式!
重要的事说三遍,关注!关注!关注!
网友评论