Android已有項(xiàng)目集成ReactNative

我們知道從0開始搭建ReactNative項(xiàng)目的話,參照官方的Getting Started來的話基本是沒有問題的袜硫。但是如果我們想把現(xiàn)有的項(xiàng)目集成RN的話,雖然有官方的Integration with Existing Apps集成步驟,但是有個(gè)別的地方說的還是不清不楚孽惰,可能對(duì)于初學(xué)者埋下了坑點(diǎn)。本人也是第一次集成嗤无,現(xiàn)在講集成的點(diǎn)點(diǎn)滴滴寫出來猴贰,本文基本思路還是引用官方的集成步驟來的对雪。

開發(fā)環(huán)境搭建

  • 新建Android Studio工程為AndroidRnDemo,項(xiàng)目SDK版本等信息
    //最低版本
    minSdkVersion 19
    //目標(biāo)版本
    targetSdkVersion 26
    
    // support v7
    implementation 'com.android.support:appcompat-v7:26.1.0'
    
    //gradle
    classpath 'com.android.tools.build:gradle:3.0.1'
  • 配置ReactNative的js環(huán)境

我們進(jìn)入到AndroidRnDemo的項(xiàng)目根目錄,接下來我們開始在終端輸入:

npm init

我們可以看到init讓你生成一個(gè)package.json文件米绕,當(dāng)然這里里面的配置信息就是讓你剛剛init填寫的配置參數(shù):

//init命令提示輸入的信息
name: (AndroidRnDemo) androidrndemo
version: (1.0.0) 
description: 
entry point: (index.js) 
test command: 
git repository: 
keywords: 
author: loen
license: (ISC) 

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

添加ReactNative到項(xiàng)目

接下來我們將React瑟捣、ReactNative添加到項(xiàng)目中:

npm install --save react react-native

安裝完成之后,我們?cè)偃ヅ渲?flowconfig栅干,這個(gè)是對(duì)js代碼做約束的迈套,建議配置(也可以不配置)。

curl -o .flowconfig  https://raw.githubusercontent.com/facebook/react-native/master/.flowconfig

當(dāng)然你也可以在項(xiàng)目根目錄新建.flowconfig非驮,然后把這個(gè)鏈接的文件內(nèi)容復(fù)制到.flowconfig中交汤。

添加start到package.json文件

這樣我們就可以時(shí)時(shí)的調(diào)用本地調(diào)試服務(wù)進(jìn)行熱加載了

"scripts": {
    "test": "echo \"Error: no test specified\" && exit 1",
    "start": "node node_modules/react-native/local-cli/cli.js start"
},

好了雏赦,現(xiàn)在看下我們的項(xiàng)目目錄吧

基本目錄

添加ReactNative到Android項(xiàng)目

添加ReactNative依賴到Android項(xiàng)目

配置你的maven

添加ReactNative的依賴到你的app目錄下的build.gradle中:

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation 'com.android.support:appcompat-v7:26.1.0'
    ...
     // From node_modules.
    api "com.facebook.react:react-native:+"
}

配置項(xiàng)目根目錄下的build.gradle:

這里有個(gè)小插曲:

//官方配置
allprojects {
    repositories {
        ...
        maven {
            // All of React Native (JS, Android binaries) is installed from npm
            url "$rootDir/../node_modules/react-native/android"
        }
    }
    //...
}
// 由于官方的android項(xiàng)目是放在android/目錄下劫笙,所以他的路徑是這樣的
"$rootDir/../node_modules/react-native/android"
//而我們?yōu)榱朔奖鉇S編譯,直接放在項(xiàng)目根目錄配置RN的星岗,所以這里的路徑要改一下
 url "$rootDir/../node_modules/react-native/android"
 
 //我們的目錄正確配置
allprojects {
    repositories {
        maven {
            url "$rootDir/node_modules/react-native/android"
        }
        jcenter()
    }
}

為了防止個(gè)別機(jī)型.so庫(kù)和findbugsbug問題填大,建議在app/build.gradle中增加下面的代碼:

android {

    defaultConfig {
       ndk {
            //選擇要添加的對(duì)應(yīng)cpu類型的.so庫(kù)。
            abiFilters 'armeabi', "armeabi-v7a","armeabi-v7a","x86"
        }
    }
    
    //...
    configurations.all {
        resolutionStrategy.force 'com.google.code.findbugs:jsr305:3.0.0'
    }
}

AndroidManifest.xml

添加權(quán)限:

<uses-permission android:name="android.permission.INTERNET"/>
/**設(shè)置調(diào)試 的權(quán)限**/
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>
<uses-permission android:name="android.permission.SYSTEM_OVERLAY_WINDOW" />

添加debug模式Activity(正式打包注釋掉就好了)

<activity android:name="com.facebook.react.devsupport.DevSettingsActivity"/>

debug模式下需要懸浮窗的權(quán)限俏橘,這個(gè)需要手動(dòng)設(shè)置允华,每部手機(jī)姿勢(shì)不一樣,具體請(qǐng)百度調(diào)整姿勢(shì)寥掐。如果沒有設(shè)置的話靴寂,個(gè)別手機(jī)在debug模式下reload會(huì)出現(xiàn)異常

android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl$W@fc0db15 -- permission denied for this window type

添加ReactNative界面

添加index.js

首先我們?cè)陧?xiàng)目根目錄添加index.js,這個(gè)文件作為RN的入口文件。

import React from 'react';
import {AppRegistry, StyleSheet, Text, View} from 'react-native';

class HelloWorld extends React.Component {
  render() {
    return (
      <View style={styles.container}>
        <Text style={styles.hello}>Hello, World</Text>
      </View>
    );
  }
}
var styles = StyleSheet.create({
  container: {
    flex: 1,
    justifyContent: 'center',
  },
  hello: {
    fontSize: 20,
    textAlign: 'center',
    margin: 10,
  },
});

AppRegistry.registerComponent('AndroidRnDemoApp', () => HelloWorld);

注意 AppRegistry.registerComponent('AndroidRnDemoApp', () => HelloWorld);里面的AndroidRnDemoApp,這個(gè)作為接下來我們要綁定的activity的入口通道名召耘。

配置Android動(dòng)態(tài)權(quán)限--->(非必須百炬,針對(duì)targetSdk>=23版本(android 6.0))

我們的項(xiàng)目目標(biāo)sdk是26,所以我們需要去代碼請(qǐng)求窗口浮層的權(quán)限污它,為了方便起見呢剖踊,我們將在MainActivity里面配置這個(gè)權(quán)限:

public class MainActivity extends AppCompatActivity {

    static final int OVERLAY_PERMISSION_REQ_CODE = 1000;
    private AppCompatButton mBtnRn;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initEvent();
    }

    private void initEvent() {
        initView();
        checkAppPermission();
    }

    private void initView() {
        mBtnRn = (AppCompatButton) findViewById(R.id.btn_rn);
        mBtnRn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //跳轉(zhuǎn)RN頁(yè)面
            }
        });
    }

    private void checkAppPermission() {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            if (!Settings.canDrawOverlays(this)) {
                Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,
                        Uri.parse("package:" + getPackageName()));
                startActivityForResult(intent, OVERLAY_PERMISSION_REQ_CODE);
            }
        }
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        if (requestCode == OVERLAY_PERMISSION_REQ_CODE) {
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
                if (!Settings.canDrawOverlays(this)) {
                    //SYSTEM_ALERT_WINDOW被拒絕
                }
            }
        }
    }
}

添加RN Activity界面

我們新建一個(gè)Activity作為RN界面展示的容器庶弃。這里姑且叫BaseRnActivity吧。

public class BaseRnActivity extends Activity implements DefaultHardwareBackBtnHandler {

    private ReactRootView mReactRootView;
    private ReactInstanceManager mReactInstanceManager;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        mReactRootView = new ReactRootView(this);
        mReactInstanceManager = ReactInstanceManager.builder()
                .setApplication(getApplication())
                .setBundleAssetName("index.android.bundle")
                .setJSMainModulePath("index")
                .addPackage(new MainReactPackage())
                .setUseDeveloperSupport(BuildConfig.DEBUG)
                .setInitialLifecycleState(LifecycleState.RESUMED)
                .build();

        //這里的AndroidRnDemoApp必須對(duì)應(yīng)“index.js”中的“AppRegistry.registerComponent()”的第一個(gè)參數(shù)
        mReactRootView.startReactApplication(mReactInstanceManager, "AndroidRnDemoApp", null);
        //加載ReactRootView到布局中
        setContentView(mReactRootView);
    }

    @Override
    public void invokeDefaultOnBackPressed() {
        super.onBackPressed();
    }


    /**
     * ReactInstanceManager生命周期同activity
     */

    @Override
    protected void onPause() {
        super.onPause();

        if (mReactInstanceManager != null) {
            mReactInstanceManager.onHostPause(this);
        }
    }

    @Override
    protected void onResume() {
        super.onResume();

        if (mReactInstanceManager != null) {
            mReactInstanceManager.onHostResume(this, this);
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();

        if (mReactInstanceManager != null) {
            mReactInstanceManager.onHostDestroy(this);
        }
    }

    @Override
    public void onBackPressed() {
        if (mReactInstanceManager != null) {
            mReactInstanceManager.onBackPressed();
        } else {
            super.onBackPressed();
        }
    }

    @Override
    public boolean onKeyUp(int keyCode, KeyEvent event) {
        if (keyCode == KeyEvent.KEYCODE_MENU && mReactInstanceManager != null) {
            mReactInstanceManager.showDevOptionsDialog();
            return true;
        }
        return super.onKeyUp(keyCode, event);
    }
}

最后不要忘了在AndroidManifest中添加:

<activity
  android:name=".BaseRnActivity"
  android:theme="@style/Theme.AppCompat.Light.NoActionBar">
</activity>

RN打離線包到Android

由于我們?cè)贐aseRnActivity的ReactInstanceManager中setBundleAssetName("index.android.bundle")了android離線包德澈,所以在運(yùn)行之前我們app之前先打個(gè)離線包JSBundle到android項(xiàng)目中歇攻。

首先在項(xiàng)目app/src/main下面必須要?jiǎng)?chuàng)建一個(gè)assets目錄,然后我們就開始打離線包啦:

//注意路徑
react-native bundle --platform android --dev false --entry-file index.js --bundle-output app/src/main/assets/index.android.bundle --assets-dest app/src/main/res/

# 控制臺(tái)這樣輸出的話 梆造,表示打包成功
Loading dependency graph, done.
bundle: start
bundle: finish
bundle: Writing bundle output to: app/src/main/assets/index.android.bundle
bundle: Done writing bundle output

運(yùn)行ReactNative

由于我們是已有項(xiàng)目集成RN,所以我們就不可以使用命令react-native run-android了缴守。沒關(guān)系我們手動(dòng)編譯唄。

//先開啟本地react native服務(wù)
adb reverse tcp:8081 tcp:8081
npm start

這里需要注意的是連接Debug server的話镇辉,需要我們手機(jī)連接pc的代理斧散,具體請(qǐng)參考

ok 運(yùn)行成功

運(yùn)行成功

遇到的坑

  1. 未添加NDK的so庫(kù)
java.lang.UnsatisfiedLinkError: could find DSO to load: libreactnativejni.so
java.lang.UnsatisfiedLinkError: dlopen failed: "xxx/libgnustl_shared.so" is 32-bit instead of 64-bit

這個(gè)錯(cuò)誤前面在我集成的項(xiàng)目中已經(jīng)添加了 ,如果你沒有添加摊聋,那就報(bào)錯(cuò)了鸡捐。

android {

    defaultConfig {
       ndk {
            //選擇要添加的對(duì)應(yīng)cpu類型的.so庫(kù)。
            abiFilters 'armeabi', "armeabi-v7a","armeabi-v7a","x86"
        }
    }
    
    //...
    configurations.all {
        resolutionStrategy.force 'com.google.code.findbugs:jsr305:3.0.0'
    }
}

2.依賴錯(cuò)誤:

Error:Execution failed for task ':app:prepareDebugAndroidTestDependencies'.

在app/build.gradle中添加

configurations.all {
    resolutionStrategy.force 'com.google.code.findbugs:jsr305:3.0.0'
}

3.待補(bǔ)充...

Demo地址

github

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末麻裁,一起剝皮案震驚了整個(gè)濱河市箍镜,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌煎源,老刑警劉巖色迂,帶你破解...
    沈念sama閱讀 212,383評(píng)論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異手销,居然都是意外死亡歇僧,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,522評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門锋拖,熙熙樓的掌柜王于貴愁眉苦臉地迎上來诈悍,“玉大人,你說我怎么就攤上這事兽埃〗那” “怎么了?”我有些...
    開封第一講書人閱讀 157,852評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵柄错,是天一觀的道長(zhǎng)舷夺。 經(jīng)常有香客問我,道長(zhǎng)售貌,這世上最難降的妖魔是什么给猾? 我笑而不...
    開封第一講書人閱讀 56,621評(píng)論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮颂跨,結(jié)果婚禮上敢伸,老公的妹妹穿的比我還像新娘。我一直安慰自己毫捣,他們只是感情好详拙,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,741評(píng)論 6 386
  • 文/花漫 我一把揭開白布帝际。 她就那樣靜靜地躺著,像睡著了一般饶辙。 火紅的嫁衣襯著肌膚如雪蹲诀。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,929評(píng)論 1 290
  • 那天弃揽,我揣著相機(jī)與錄音脯爪,去河邊找鬼。 笑死矿微,一個(gè)胖子當(dāng)著我的面吹牛痕慢,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播涌矢,決...
    沈念sama閱讀 39,076評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼掖举,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了娜庇?” 一聲冷哼從身側(cè)響起塔次,我...
    開封第一講書人閱讀 37,803評(píng)論 0 268
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎名秀,沒想到半個(gè)月后励负,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,265評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡匕得,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,582評(píng)論 2 327
  • 正文 我和宋清朗相戀三年继榆,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片汁掠。...
    茶點(diǎn)故事閱讀 38,716評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡略吨,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出调塌,到底是詐尸還是另有隱情晋南,我是刑警寧澤,帶...
    沈念sama閱讀 34,395評(píng)論 4 333
  • 正文 年R本政府宣布羔砾,位于F島的核電站,受9級(jí)特大地震影響偶妖,放射性物質(zhì)發(fā)生泄漏姜凄。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,039評(píng)論 3 316
  • 文/蒙蒙 一趾访、第九天 我趴在偏房一處隱蔽的房頂上張望态秧。 院中可真熱鬧,春花似錦扼鞋、人聲如沸申鱼。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,798評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)捐友。三九已至淫半,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間匣砖,已是汗流浹背科吭。 一陣腳步聲響...
    開封第一講書人閱讀 32,027評(píng)論 1 266
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留猴鲫,地道東北人对人。 一個(gè)月前我還...
    沈念sama閱讀 46,488評(píng)論 2 361
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像拂共,于是被迫代替她去往敵國(guó)和親牺弄。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,612評(píng)論 2 350

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,839評(píng)論 25 707
  • 背景近期由于項(xiàng)目需要宜狐,在已經(jīng)開發(fā)了幾個(gè)版本的原生App中集成了ReactNative猖闪,新版的模塊都使用RN開發(fā)。此...
    kevinlandering閱讀 2,419評(píng)論 0 6
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理肌厨,服務(wù)發(fā)現(xiàn)培慌,斷路器,智...
    卡卡羅2017閱讀 134,637評(píng)論 18 139
  • “把金錢當(dāng)做某種行為的外部獎(jiǎng)勵(lì)時(shí)柑爸,行為主體就失去了對(duì)這項(xiàng)活動(dòng)的內(nèi)在興趣"獎(jiǎng)勵(lì)只能帶來短期的爆發(fā)吵护,就像是少量咖啡因只...
    榛果Latte閱讀 214評(píng)論 0 1
  • 我叫肖楓,我是風(fēng)云酒吧的駐唱表鳍,今天是我第五天來到這里馅而。在這里,我還交了一個(gè)新朋友譬圣,他叫周林瓮恭,周林的性格很豪...
    7b67f3280cce閱讀 319評(píng)論 0 1