ReactNative封装iOS/Android原生组件

简介

为了提高代码的复用,减少重复工作,我们经常会把比较独立的功能模块封装成组件或者library

在iOS我们通常封装为静态库(.framework和.a),

在Android中我们通常封装为Android Library(gradle模块),

在JavaScript中我们通常封装成node package,

而在ReactNative中我们要封装与原始有关的库我们需要,结合iOS静态库(.a)、Android Library和node package一起封装成ReactNative用的node package。

目录结构

testmodule-wyh
.
├── android   // 存放Android library
│   ├── build
│   ├── build.gradle
│   ├── libs
│   ├── proguard-rules.pro
│   ├── src
│   └── testmodule.iml
├── component // 存放js桥接过来的组件以及api
│   └── TestView.js
├── index.js
├── ios // 存放iOS library
│   ├── TestModule
│   └── TestModule.xcodeproj
└── package.json // node包的配置文件

源码地址

制作

装备工作

创建ReactNative工程

我们需要先创建一个ReactNative工程,使用如下命令创建。

$ react native init A7011CreatNativeModule

创建存放封装库的目录

$ cd A7011CreatNativeModule/node_modules
$ mkdir testmodule-wyh
$ cd testmodule-wyh
$ mkdir ios
$ mkdir android
$ mkdir component

封装iOS库

步骤

1. 静态库创建

由于ReactNative的组件都是一个个静态库,我们发布到npm给别人使用的话,也需要建立静态库。我们使用xcode建立静态库,取名为Testmodule。建立之后,我们将创建的静态库中的文件全部copy到node_modules/Testmodule/ios目录下。 ios文件目录如下:

testmodule-wyh
.
├── android   // 存放Android library
├── component // 存放js桥接过来的组件以及api
└── ios // 存放iOS library
    ├── Testmodule
    │   ├── Testmodule.h
    │   └── Testmodule.m
    └── TestModule.xcodeproj
2. 引入到目标工程

使用xcode打开A7011CreatNativeModule/ios/下的iOS工程,将BGNativeModuleExample静态库工程拖动到工程中的Library中。

image-20180713184731573
  1. 为什么要拖到RN工程中呢?

    因为,不拖到RN工程中去的话,用的RN库中的API,就会报错,所以需要RN工程的基础环境。

  2. 为什么这里要先把静态库放到node_modules/Testmodule/ios,再拖到RN工程中去呢?

    那是因为,如果直接把创建的静态库的TestModule.xcodeproj文件拖到工程中去,无法编辑这个静态库,只有在node_modules/Testmodule/ios才能编辑。目前还不知道原因

3. 编写原生代码

咱们这里主要是演示做封装RN的node package,所以就封装一个最简单的组件,一个背景色为红色的View。

  1. 创建TestView Class

       // TestView.h
       #import <UIKit/UIKit.h>
       
       @interface TestView : UIView
           
       @end
       
           
       // TestView.m
       #import "TestView.h"
       
       @implementation TestView
       - (instancetype)init
       {
           self = [super init];
           if (self) {
               self.backgroundColor = [UIColor redColor];
           }
           return self;
       }
       
       @end    
    
  2. 创建桥接类

    // TestViewManager.h
    #import <React/RCTBridgeModule.h>
    
    @interface TestViewManager : RCTViewManager <RCTBridgeModule>
    
    @end
        
        
    // TestViewManager.m
    #import "TestViewManager.h"
    #import <React/RCTViewManager.h>
    #import "TestView.h"
    
    @interface TestViewManager : RCTViewManager
    @end
    
    @implementation TestViewManager
    
    RCT_EXPORT_MODULE()
    
    - (UIView *)view
    {
        return [[TestView alloc] init];
    }
    
    @end
    
    
  3. 需要更复杂的交互,可以参考React Native官网的iOS原生模块原生UI组件

4. React组件创建
  1. 接下来你需要一些Javascript代码来让这个视图变成一个可用的React组件:

    // A7011CreatNativeModule/node_modules/testmodule-wyh/component/TestView.js
    // TestView.js
    import {requireNativeComponent} from 'react-native';
    
    const TestView = requireNativeComponent('TestView', null);
    export default TestView;
    
  2. 在路径A7011CreatNativeModule/node_modules/testmodule-wyh创建index.js文件,导出TestView组件

    // index.js
    import TestView from './component/TestView'
    
    export default TestView;
    
5. 组件使用

在RN工程里面的App组件中,使用这个组件

// App.js
import React, {Component} from 'react';
// ......
export default class App extends Component<Props> {
  render() {
    return (
      <View style={styles.container}>
      // ......
        <TestView style={{width: 100, height: 100}} /> // TestView组件
      </View>
    );
  }
}
// ......
});

6. 运行效果
image

注意

引入第三方库

iOS封装静态库的时候,经常会用到需要引入的系统库和第三方库;系统库引入很简单,只要在TestModule.xcodeproj->Build Phases->Link Bnary with Libraries添加系统库就行;而第三方库有可能是.a也有可能是.framework,.a拷贝到静态库根目录里面,接下就和系统库添加一样了,而.framework就复杂了,他不能直接引用到封装的静态库中,需要引用到目标iOS工程中去才可以。

桥接ViewController

封装Android库

步骤

1. 创建Android library

用Android studio打开RN工程中的Android工程,新建file->new->new module..->Android library,并命名为testmodule

image-20180716164450271
2. RN的Android工程中配置模块
  • app工程中的build.gradle文件中的dependencies添加一行compile project(':testmodule-wyh'),让主工程app依赖我们新创建的Library。
  • 我们还需要让新创建的Library依赖react native,和上面差不多,只需要在我们新创建的testmodule-wyh下的build.gradle中的dependencies添加一行compile "com.facebook.react:react-native:+"就行了。
3. 编写原生代码

Android原生代码的编写,前两步和iOS步骤类似,多了一个创建ReactPackage实现,并在MainApplication的getPackages方法里面注册。

  1. 创建TestView Class

    package com.example.testmodule;
    
    import android.content.Context;
    import android.graphics.Color;
    import android.view.View;
    
    public class TestView extends View {
    
        public TestView(Context context) {
            super(context);
            this.setBackgroundColor(Color.rgb(255,0,0));
        }
    }
    
  2. 创建桥接类

    package com.example.testmodule;
    
    import com.facebook.react.uimanager.SimpleViewManager;
    import com.facebook.react.uimanager.ThemedReactContext;
    
    public class TestViewManager extends SimpleViewManager<TestView> {
        public static final String REACT_CLASS = "TestView";
    
        @Override
        public String getName() {
            return REACT_CLASS;
        }
    
        @Override
        protected TestView createViewInstance(ThemedReactContext reactContext) {
            return new TestView(reactContext);
        }
    }
    
    
  3. 需要创建一个Package注册类

    package com.example.testmodule;
    
    import com.facebook.react.ReactPackage;
    import com.facebook.react.bridge.NativeModule;
    import com.facebook.react.bridge.ReactApplicationContext;
    import com.facebook.react.uimanager.ViewManager;
    
    import java.util.ArrayList;
    import java.util.Collections;
    import java.util.List;
    
    public class TestPackage implements ReactPackage {
        @Override
        public List<NativeModule> createNativeModules(ReactApplicationContext reactContext) {
            return Collections.emptyList();
        }
    
        @Override
        public List<ViewManager> createViewManagers(ReactApplicationContext reactContext) {
            List<ViewManager> modules = new ArrayList<>();
            modules.add(new TestViewManager());
            return modules;
        }
    }
    
  4. 在MainApplication.java中注册这个package

    package com.a7011creatnativemodule;
    // ......
    public class MainApplication extends Application implements ReactApplication {
      private final ReactNativeHost mReactNativeHost = new ReactNativeHost(this) {
       //.......
        @Override
        protected List<ReactPackage> getPackages() {
          return Arrays.<ReactPackage>asList(
              new MainReactPackage(),
              new TestPackage(),
              // .......
          );
        }
     //.......
      };
    
      //.......
    }
    
    
  5. 需要更复杂的交互,可以参考React Native官网的android原生模块原生UI组件

4. 组件创建使用

之后的步骤在封装iOS库的4、5、6步已经做了

5. 运行效果

直接在点击run,看结果


image

注意

Gradle问题
  1. com.android.tools.build:gradle工具也有2升级为3以后,dependencie依赖由compile变为implementation,而用React Native init ProjectName创建的rn项目还是用的compile,所以创建的时候要调整Gradle.build的dependencie和minSdkVersion以适用目标项目。
  2. 一定要记得创建ReactPackage的接口实现类,这样当执行react-native link testmodule-wyh才会自动添加到依赖。

整理成node package

  1. 将封装的Android library里面的所以文件移到node_modules/Testmodule/android里面,形成的目录结构如下:

    .
    ├── android
    │   ├── build.gradle
    │   ├── libs
    │   ├── proguard-rules.pro
    │   ├── src
    │   ├── androidTest
    │   ├── main
    │   │   ├── AndroidManifest.xml
    │   │   ├── java
    │   │   │   └── com
    │   │   │       └── example
    │   │   │           └── testmodule
    │   │   │               ├── TestPackage.java
    │   │   │               ├── TestView.java
    │   │   │               └── TestViewManager.java
    │   │   └── res
    │   │       ├── drawable
    │   │       └── values
    │   │           └── strings.xml
    │   └── test
    │   └── testmodule.iml
    ├── component
    │   └── TestView.js
    ├── index.js
    ├── ios
    │   ├── TestModule
    │   │   ├── TestView.h
    │   │   ├── TestView.m
    │   │   ├── TestViewManager.h
    │   │   └── TestViewManager.m
    │   └── TestModule.xcodeproj
    
  2. 在路径node_modules/Testmodule-wyh/使用npm init创建一个package.json文件,全部使用默认项就行,这样Testmodule-wyh node package就封装完成了。

    // package.json
    {
      "name": "testmodule-wyh",
      "version": "1.0.0",
      "description": "test",
      "main": "index.js",
      "directories": {
        "lib": "lib"
      },
      "scripts": {
        "test": "echo \"Error: no test specified\" && exit 1"
      },
      "author": "",
      "license": "ISC"
    }
    
    

使用

  1. testmodule-wyh node package拷贝到桌面,然后将iOS和Android项目中与Testmodule-wyh有关的代码全部删除

  2. 在RN项目中package.json中使用本地路径的方式引用制作的testmodule-wyh node package

    // package.json
    {
      "name": "A7011CreatNativeModule",
      "version": "0.0.1",
      "private": true,
      "scripts": {
        "start": "node node_modules/react-native/local-cli/cli.js start",
        "test": "jest"
      },
      "dependencies": {
        "react": "16.4.1",
        "react-native": "0.56.0",
        "testmodule": "file:/Users/wangyinghui/Desktop/testmodule-wyh",
      },
      "devDependencies": {
        "babel-jest": "23.4.0",
        "babel-preset-react-native": "^5",
        "jest": "23.4.0",
        "react-test-renderer": "16.4.1"
      },
      "jest": {
        "preset": "react-native"
      }
    }
    
  3. 在RN项目根目录执行如下指令

    $ yarn install
    $ react-native link testmodule-wyh
    
  4. 看到如下提示表示成功

    image
  5. 运行iOS和Android查看效果

    $ react-native run-android
    $ react-native run-ios
    

注意事项

  1. iOS和Android的命名不一定要和封装的node package名一直。

  2. 默认iOS和Android的library都要在根目录,如果不在根目录需要在packag.json指明如下

    {
      "name": "react-native-maps",
      // ...
      "rnpm": {
        "android": {
          "sourceDir": "./lib/android"
        }
      }
    }
    

上传代码库

GitHub,创建一个代码仓库,将node package 代码上传,然后在package.json添加repository如下:

"repository" :  
  { 
      "type" : "git",
      "url" : "https://github.com/<yourusername>/testmodule-wyh.git"  
  }  

repository 属性不写也可以,但是最好建一个 github 项目然后把地址写进来,方便以后维护。

发布

注册npm账号

发布node package到npm服务器上,需要npm的账号,注册哪怕npm账号有两种形式

  • npm官网注册

  • 使用命令行注册

    $ npm adduser
    Username: XXX # 账户名
    Password: *** # 密码
    Email: (this IS public) XXX@XXX.com # 邮箱
    

如果用命令行注册的话,最好是在npm官网验证一下

登录npm账户

首次需要登录,npm login(需要输入用户名,密码,还有邮箱) 存储证书到本地,后面就不需要每次都登录的

$ npm login
Username: wangyinghui
Password: 
Email: (this IS public) iyinghui@163.com
Logged in as wangyinghui on http://registry.npmjs.org/.

开始发布

$ npm publish
+ testmodule-wyh@1.0.2

发布时报错

1. cnpm造成的报错

$ npm publish
npm ERR! registry error parsing json
npm ERR! publish Failed PUT 413
npm ERR! Unexpected token < in JSON at position 0
npm ERR! <html>
npm ERR! <head><title>413 Request Entity Too Large</title></head>
npm ERR! <body bgcolor="white">
npm ERR! <center><h1>413 Request Entity Too Large</h1></center>
npm ERR! <hr><center>nginx/1.4.6 (Ubuntu)</center>
npm ERR! </body>
npm ERR! </html>
npm ERR! 

npm ERR! A complete log of this run can be found in:
npm ERR!     /Users/wangyinghui/.npm/_logs/2018-07-13T02_27_29_082Z-debug.log

出现原因:使用的是淘宝源cnpm,登陆到的是cnpm

解决方法:切换到npmjs的网址,代码如下:

$ npm config set registry http://registry.npmjs.org/

发布完成之后,如果还想回到之前的cnpm,使用下面的命令

$ npm config set registry https://registry.npm.taobao.org

npm 安装git项目的几种方式

1. 直接通过用户名安装

#   直接利用用户名与仓库名进行安装
$ npm install yiifaa/yii-es6-amd
#   或者为了提醒自己,加上github前缀进行区分
$ npm install github:yiifaa/yii-es6-amd1234

2. 通过地址安装

#   这样适合安装公司内部的git服务器上的项目
$ npm install git+https://git@github.com/yiifaa/yii-es6-amd.git#v1.0.0
#   或者以ssh的方式
$ npm install git+ssh://git@github.com/yiifaa/yii-es6-amd.git#v1.0.0

参考文章

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,918评论 25 707
  • 道路蜿蜒 前程似锦 一个星期以来,简素莫名的给自己增加了许多压力,焦虑感十足。主要是来源于向左而生你007-49...
    简素七年阅读 340评论 0 1