美文网首页Android开发互联网科技Android开发
React Native 与 嵌入Android原生与Activ

React Native 与 嵌入Android原生与Activ

作者: 881ef7b85f62 | 来源:发表于2019-08-14 11:03 被阅读7次

    前言

    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)即可获取免费领取方式!

    重要的事说三遍,关注!关注!关注!

    相关文章

      网友评论

        本文标题:React Native 与 嵌入Android原生与Activ

        本文链接:https://www.haomeiwen.com/subject/kazwjctx.html