对于刚接触rn开发的同学可能不是太了解,简单介绍一下。三端指的是安卓、IOS、H5。rn本身就是跨平台框架,为了让RN能在不改动代码的情况下同时兼容H5,只需要引入一些库和配置一些细节即可。
本模板采用目前react-native@0.6x
,react-navigation@5.x
新版本,搭建ts开发环境。
新版本相关库的API与老rn版本有些是不兼容的,请不要随意更改主版本!
不管是对于刚入门RN的小白,还是熟练运用RN老版本的熟手,都能轻松上手
照着官网配置开发环境,不熟悉客户端开发的前端可以选沙盒环境。但是建议一步步装客户端环境,毕竟现在跨平台时代。
首先新版本的RN已经不再使用react-native-cli
,也就是说我们不需要全局安装任何包。只要你的npm版本在5.2+,可以使用npx命令(npx介绍)即可。
npx react-native init MyApp --template react-native-template-typescript
npx生成rn for ts模板,安装完成后这就是一个官方可运行的RN项目了,接下来我们就来修改这个模板
为了使用新版本的@react-navigation v5
搭建完整可用的RN开发项目、以及我们要做的三端同构 ,需要新增一些npm依赖的包。
"dependencies": {
// ...模板自带的包
// ...
"@react-native-community/async-storage": "1.9.0", // 相当于localStorage(可选)
"@react-native-community/masked-view": "0.1.9", // 路由所需要的包 (必须)
"@react-navigation/stack": "5.2.10", // 路由所需要的包 (必须)
"react": "16.11.0",
"react-dom": "16.11.0",
"react-native": "0.62.2",
"react-native-gesture-handler": "1.6.0", // 路由所需要的包,原生手势系统 (必须)
"react-native-reanimated": "^1.8.0", // 路由所需要的包 (必须)
"react-native-safe-area-context": "0.7.3", // 路由所需要的包 (必须)
"react-native-screens": "^2.3.0", // 路由所需要的包 (必须)
"react-native-webview": "8.0.0", // RN打开webView容器,原生调用 (可选)
"react-redux": "5.0.7",
"redux": "4.0.0",
"redux-thunk": "2.3.0"
},
"devDependencies": {
// ...模板自带的包
// ...
"@babel/plugin-transform-runtime": "^7.9.0", // babel的插件包(必须)
"@babel/preset-typescript": "^7.9.0", // babel的编译ts(必须)
"@react-navigation/core": "3.4.2", // rn路由在H5里运行的核心包(必须)
"@react-navigation/web": "1.0.0-alpha.8", // rn路由在H5里运行的核心包(必须)
"babel-loader": "^8.1.0", // webpack loader(必须)
"file-loader": "3.0.1", // webpack loader(必须)
"html-webpack-plugin": "^4.2.0", // webpack plugin(必须)
"react-native-web": "0.12.2", // rn组件映射为WEB的dom(必须)
"webpack": "4.42.1", // 打包rn项目到h5(必须)
"webpack-cli": "3.3.2", // 打包rn项目到h5(必须)
"webpack-dev-server": "3.5.1" // 调试rn项目到h5(必须)
},
该装的都装上别漏了,建议梭哈
首先新建一个index.html
文件作为H5打包模板
根目录下index.ts
为RN为原生打包的入口,新建一个index.web.ts
当做H5的打包入口,注意在webpack中添加
resolve: {
extensions: [
'.web.ts',
'.web.tsx',
'.ts',
'.tsx',
'.js',
'.jsx'
],
alias: {
'react-native$': 'react-native-web'
}
},
extensions配置的目的是为了让我们的项目能够在import XX from './xx'
引入文件时,能够编写两个不同后缀的文件xx.web.ts
,xx.ts
。一个给RN打包给原生,一个给webpack打包给H5,用来对不同平台的代码做定制化开发,这一点在路由上是极其有用的。.web.ts一定要配置在.ts前面,让webpack优先找到属于H5的文件打包。
另外alias就是将react-native
整个库的组件映射为react-native-web
,实际上这个库的原理就是为RN每个组件写对应的dom然后传入props。因为你也可以自行拓展,有的原生组件react-native-web
是没有的,比如一开始提到的webView。
{
test: /\.(js|jsx|ts|tsx)$/,
use: [
{
loader: 'babel-loader',
options: {
cacheDirectory: false,
presets: [
'module:metro-react-native-babel-preset',
'@babel/preset-typescript'
],
plugins: [
'@babel/plugin-transform-runtime'
]
}
}
],
exclude: /node_modules/
}
其次配置babel-loader
来编译ts,这里尤其要注意:
不能使用ts-loader
来编译!
因为跟文件tsconfig.json
中我们配的jsx
识别为react-native
,是用来编译给客户端的,但是实际上面我们将其代理成了dom元素,打包会产生冲突导致loader编译错误。解决方案为直接使用babel-loader
进行配置编译TS,生成sourcemap
Redux状态管理和普通react一样,也可以用别的库,毕竟状态管理不涉及到渲染。主要说明React-navigation,这个RN的路由库,每个版本的API差异很大,V5的版本将API拆分为多个组@react-navigation/
,有很多种路由配置选择,我这里提供一种:
// router.ts
import 'react-native-gesture-handler';
import React from 'react';
import { connect } from 'react-redux';
import AsyncStorage from '@react-native-community/async-storage';
import {
InitialState,
useLinking,
NavigationContainerRef,
NavigationContainer,
DefaultTheme,
DarkTheme
} from '@react-navigation/native';
import {
createStackNavigator,
HeaderStyleInterpolators
} from '@react-navigation/stack';
import routes from './config';
type RootDrawerParamList = {
[key: string]: any;
};
const Stack = createStackNavigator<RootDrawerParamList>();
const HeaderNull = function(): React.ReactNode {
return null;
};
const MyApp = function() {
const NAVIGATION_PERSISTENCE_KEY = 'NAVIGATION_STATE';
const containerRef = React.useRef<NavigationContainerRef>(null);
const [initialState, setInitialState] = React.useState<
InitialState | undefined
>();
const [theme, setTheme] = React.useState(DefaultTheme);
return <NavigationContainer
ref={containerRef}
initialState={initialState}
onStateChange={async (state) => {
try {
await AsyncStorage.setItem(
NAVIGATION_PERSISTENCE_KEY,
JSON.stringify(state)
);
} catch (e) {
console.log(e);
}
}}
theme={theme}
>
<Stack.Navigator>
{(Object.keys(routes) as (keyof typeof routes)[]).map(
(name) => (
<Stack.Screen
key={name}
name={name}
component={routes[name].screen}
options={{
header: props => HeaderNull()
}}
/>
)
)}
</Stack.Navigator>
</NavigationContainer>;
};
const mapStateToProps = (state: any) => state;
export default connect(mapStateToProps)(MyApp);
简单来说就是把路由配置导入就行了,但是这个config
要抽离出来,因为我们在这个目录下还有一个route.web.ts
,同样要引用这个配置,但是引用配置的库却是不同的,在H5里我们使用@react-navigation/web
:
// router.web.ts
import { connect } from 'react-redux';
import { createSwitchNavigator } from '@react-navigation/core';
import { createBrowserApp } from '@react-navigation/web';
import routes from './config';
const MyNavigator = createSwitchNavigator(routes);
const MyApp = createBrowserApp(MyNavigator);
const mapStateToProps = state => state;
export default connect(mapStateToProps)(MyApp);
这样一来,我们就能保证路由的每个页面都是引用的相同的组件了
该项目最开始配置package.json
有两个原生模块模块包,需要使用link进行关联操作,否则构建客户端报错。在npm i
后执行:
npx react-native link react-native-gesture-handler
这个是react-navigation使用的原生的手势系统。npx react-native link @react-native-community/async-storage
这个类似于浏览器的localStorage原生缓存。
配置好npm启动命令
// package.json
"scripts": {
"android": "react-native run-android",
"ios": "react-native run-ios",
"start": "react-native start",
"web:dev": "webpack-dev-server --config webpack.config.web.ts --inline --hot --colors",
"web": "webpack -p --config webpack.config.web.ts"
},
- 原生客户端使用
npm run android
或npx react-native run-android
(npm v5.2+)启动 npm run web:dev
启动webpack-dev-server调试,和普通移动端开发一样npm run web
打包为生产环境代码到dist-H5
从上面的截图可以看到几点:
- UI样式上基本能一比一还原
- 路由动画不如原生(这个貌似图片看不到...)
- H5上首页有个红圈,这是我故意用了一个
react-native-web
不支持的组件,在H5就会标红这个区域。 - Native-3中有一个客户端半屏webView容器打开了第三方H5,是使用的原生RN组件,这个在H5里显然是没有的,所以遇到这种情况,需要单独对环境进行兼容处理。(我这里没有兼容,所以打包H5的时候你能看到webpack抛出的警告)