1

轻松几步教你将React Native 项目运行在Web端

 3 years ago
source link: https://segmentfault.com/a/1190000039020824
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.

摘要:尽管 React Native 已经进入开源的第 6 个年头,距离发布 1.0 版本依旧是遥遥无期。“Learn once, write anywhere”,完全不影响 React Native 沦为“不会 JavaScript 也能用”的框架,那如何将在 React Native 项目中引入 react-native-web 呢?

react-native-web 简介

仓库地址: https://github.com/necolas/react-native-web

react-native-web 是由 前 Twitter 现 Facebook 工程师 Nicolas Gallagher 实现并维护的开源项目,是一个使 React Native 组件和 API 能运行在 Web 上的库,其和 React Native Windows, React Native macOS 等库将 React Native 拓展到一个又一个新的平台。目前推特、expo、大联盟足球、Flipkart、优步、《泰晤士报》、DataCamp 以及我们小作坊都在生产中使用了 react-native-web。Chrome、Firefox、Edge,Safari 7 +、IE 10+都支持通过 react-native-web 构建的 web 应用。当然值得注意的是,官方文档明确表示不支持 React Native 中不推荐使用的组件和 API,因此如果您项目中的某些功能依赖第三方库,可能那部分的功能在 web 端同构时需要额外处理。

浅显地认为react-native-web就是把React Native的组件和API都用适用于Web的标签和API再适配实现一遍,使其在Web上的行为和在原生应用上尽量保持一致,从文档中提到的 Alert 和 Setting 模块以及其对应的源码中大概能感受到一二,比如TextInput:

fA7Rn2R.png!mobile

VveINzq.png!mobile

因此也借鉴了 React Native 的一些代码,作为适配的依据。

aqeiIbB.png!mobile

如果您想基于 React Native 实现多端统一化方案,可参考去哪儿前端团队的实现方案:跨端开发, 仓库地址: https://github.com/qunarcorp/qrn-remax-unir

添加到React Native项目

一般来说新建 React Native 项目时可以选用 expo-cli 或者 react-native-cli 来创建。expo-cli 中已经预置了对web的支持,如下图所示.

v2eiUvB.png!mobile

而我们实际开发中可能用 react-native-cli脚手架来构建项目比较多些,那么如何引入 react-native-web呢?

我们先初始化项目:

QrEb6rV.png!mobile

此时我们的项目并不支持在web中使用:

JVreiim.png!mobile

为了项目能在web环境中运行,我们需要借助今天的主角--react-native-web,有请主角出台:

eyeEbyy.png!mobile

接着我们施展Copy大法,将我们初始化能用到的App.web.jsx、index.html、index.web.js、webpack.config.js这几个文件一把 down下来:

fIvu2m.png!mobile

Ur2qequ.png!mobile

react-native-web

然后在package.json中添加build和web的脚本:

BB3mam.png!mobile

j2uAJv7.png!mobile

就和 expo-cli 初始化的项目一样可以执行yarn web,这时会在本地8080端口运行一个服务,这时我们分别执行yarn ios 和 yarn android就能看到在ios模拟器和Android模拟器中显示和web端一模一样的页面,一次 react-native-web 的多端同构 Hello World 就成功实现了,当然这也意味着我们还可能编译成小程序,后续有机会一起探讨探讨!

ANJnuaM.png!mobile

此处的注意点:

  1. 代码能得以成功拷贝全靠梯子,当然也可以选择去网页下载;
  2. Android能得以成功运行,全靠给权限 sudo 755 android/gradlew ;
  3. React Native 入口文件需修改为 App.web,不然只有Web端才能读取 App.web.js ;
  4. 适当执行 ./gradlew clean 重新 yarn android 等多年经验积累骚操作排除故障.

探究代码

关键的操作在于那行Copy代码的命令,那究竟上文中提到到下载了4的文件到底做了啥呢?Copy攻城狮心中也有一个大大的问号,Talk is cheap, show me the code,打开文件看看那些代码吧!

常见的单页面应用入口,像下面代码中的 div 我们称其为“根” DOM节点,因为其中的所有内容都将由React DOM进行管理。在 当前案例 中,我们只是设置一些基本样式以使主体div具有完整的高度和宽度:

bIzqyaM.png!mobile

使用index.web.js可以在Web和移动端之间区分开来,通过.web.js扩展名可以使该文件仅在Web上使用,其他一些可用的扩展如.native.js、.ios.js和.android.js适用于移动端。当然,如果您希望将本不同端的代码都保存在一个index.js文件中,则可以使用import { Platform } from 'react-native'来按照条件区分不同平台的代码。可以参考React Native官方文档中有关平台特定代码的更多信息。

VZfeIzY.png!mobile

这与我们移动端的index.js非常相似,不过它还将您的应用程序挂载到根目录中index.html的div上。

webpack虽然是重点内容,但此处不过过多介绍,请前往官方文档阅读更加详细的内容,此案例中我们用到了三个插件:

  1. HtmlWebpackPlugin创建HTML;
  2. HotModuleReplacementPlugin用于热模块重装;
  3. DefinePlugin定义变量,例如__DEV__或NODE_ENV中react-native-web。

该文件是临时添加的文件,用于在使用React Native Web 同构之前验证我们的设置是否正常运行。最终,您可以删除此文件,因为App的入口js文件可以在移动端运行,也能在Web端运行。不过为了处理某些在Web上能运行而在移动端不能运行的业务,需要将代码抽离出来存放在 .web.js 为后缀的文件中。

后记

结合上述的简单案例,在后续实际业务中,我们可以逐步尝试同构业务到Web并逐步进行验证。

本文分享自华为云社区《React Native 项目 Web 端同构初探》,原文作者:胡琦。

点击关注,第一时间了解华为云新鲜技术~


About Joyk


Aggregate valuable and interesting links.
Joyk means Joy of geeK