带有 Create-react-app 的 Cordova

IT技术 javascript reactjs cordova create-react-app
2021-04-16 11:11:59

我使用create-react-app创建了一个 ReactJs 应用程序,然后使用npm run build. 在我用 Cordova 创建的www文件夹中,我只是从create-react-app 的构建文件夹中复制所有文件,这很好。

我想知道如何关联 Cordova 的事件,例如:

function startApp() {
  // your app logic
}
if (window.cordova) {
  document.addEventListener('deviceready', startApp, false);
} else {
  startApp();
}

例如,我想调用 .js 文件中缩小的 JS 文件startApp()或者是否有任何其他工作流程可用于使 Cordova 事件与 react 应用程序一起使用。

一个小例子会有所帮助。

是否可以完全使用构建文件并直接在 Cordova 中使用 React App?鉴于有将 ES6 代码转换为 ES5 和所有代码的 Webpack 设置,我不确定这将如何工作。

我是 Cordova 的新手,正在努力解决这个集成方面的问题。

4个回答

我发现可以让这两个工作,并将在这里发布给其他寻找相同的人。也许还有其他方法可以做到这一点,但这对我有用。

所以基本上我们将使用(说)创建一个 Cordova 应用程序:cordova create testapp com.test.testapp testapp 这会给我一个文件夹结构,如下所示:

testapp
        --hooks
        --platforms
        --plugins
        --www
        --config.xml

现在在 testapp 文件夹中我们运行: create-react-app testappReact 这将在 testapp 文件夹中添加我的react应用程序。您的 React 应用程序将在 /src 目录中有一个主要的 index.js。

我 index.js 确保将您的主要逻辑包装在一个函数中,然后像这样调用该函数和 Cordova 对象:

import React from 'react';
import ReactDOM from 'react-dom';
import App from './App';
import './index.css';


const startApp = () => {
ReactDOM.render(
  <App />,
  document.getElementById('root')
);
}

if(!window.cordova) {
  startApp()
} else {
  document.addEventListener('deviceready', startApp, false)
}

现在应该这样做了,您的应用程序将拥有 Cordova 实例以及您的应用程序内的 navigator.camera 等设备对象。

同样在您的react应用程序 index.html 中,它可以在公共文件夹中找到,从您将在 Codova www 文件夹中找到的 index.html 中复制 html。现在我们可以删除 www 文件夹中的所有文件。我们稍后将手动或通过脚本将所有文件从 react apps build 文件夹复制到 Cordova www 文件夹。

所以我的 index.html 看起来像下面这样,注意作为脚本包含的 cordova.js 文件。

<!DOCTYPE html>
<!--
    Licensed to the Apache Software Foundation (ASF) under one
    or more contributor license agreements.  See the NOTICE file
    distributed with this work for additional information
    regarding copyright ownership.  The ASF licenses this file
    to you under the Apache License, Version 2.0 (the
    "License"); you may not use this file except in compliance
    with the License.  You may obtain a copy of the License at

    http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing,
    software distributed under the License is distributed on an
    "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
     KIND, either express or implied.  See the License for the
    specific language governing permissions and limitations
    under the License.
-->
<html>

<head>
    <!--
        Customize this policy to fit your own app's needs. For more guidance, see:
            https://github.com/apache/cordova-plugin-whitelist/blob/master/README.md#content-security-policy
        Some notes:
            * gap: is required only on iOS (when using UIWebView) and is needed for JS->native communication
            * https://ssl.gstatic.com is required only on Android and is needed for TalkBack to function properly
            * Disables use of inline scripts in order to mitigate risk of XSS vulnerabilities. To change this:
                * Enable inline JS: add 'unsafe-inline' to default-src
        -->
    <meta http-equiv="Content-Security-Policy" content="default-src * 'unsafe-inline' 'unsafe-eval'; style-src 'self' 'unsafe-inline'; media-src *; img-src * data: content:;">
    <meta name="format-detection" content="telephone=no">
    <meta name="msapplication-tap-highlight" content="no">
    <meta name="viewport" content="user-scalable=no, initial-scale=1, maximum-scale=1, minimum-scale=1, width=device-width">
    <link rel="shortcut icon" href="%PUBLIC_URL%/favicon.ico">

    <!-- Latest compiled and minified CSS -->
    <title>React App</title>
</head>

<body>
    <div id="root"></div>
   <script type="text/javascript" src="cordova.js"></script>
</body>

</html>

最后在你的 react 应用程序的 package.json 中添加以下行: .... "homepage": "../www" .... 这将确保你的最终构建文件指向正确的路径。我们还可以在 package.json 构建脚本中添加以下几行。

  "scripts": {
    "start": "react-scripts start",
    ***"build": "react-scripts build && robocopy .\\build ..\\www /MIR",***
    "test": "react-scripts test --env=jsdom",
    "eject": "react-scripts eject",
    "deploy": "npm run build&&gh-pages -d build"
  }

它可以是基于操作系统(Windows/Linux 等)的 robocopy 或 cp-r。

我们应该准备好使用cordova build android/ios构建我们的Cordova应用程序。

$ cordova plugin add cordova-plugin-whitelist. 默认情况下,只允许导航到 file:// URL。要允许其他 URL,您必须将 <allow-navigation> 标记添加到您的 config.xml SO
2021-05-27 11:11:59
对于 Mac 使用cp -a ./build/. ../www/而不是 robocopy,-a是一个改进版本,-r构建后的点确保所有隐藏文件也被复制(如果有的话)
2021-06-02 11:11:59
robocopy是一个 Windows 工具。在 Linux 中使用 "build": "react-scripts build && cp -r ./build/* ../www/" 代替。
2021-06-02 11:11:59
如果您想拥有特定于平台的代码,您会怎么做?仅使用 Cordova,平台特定代码位于 merges/ 中。
2021-06-07 11:11:59

我解决了这个问题。以下是我为寻找解决方案的人提供的分步格式:

  1. 复制/创建一个新React项目(使用创建create-react-app)就在Cordovaapp 目录中。
  2. 清除应用程序www文件夹中的所有内容Cordova
  3. cd到 React 项目文件夹(您刚刚复制/创建)并打开package.json.
  4. dependencies添加"homepage": "./",和内部脚本之前更改build"build": "react-scripts build && robocopy .\\build ..\\www /MIR",
  5. 不要npm run build在同一个(React的)目录及回父(Cordova),然后选择文件夹build,并emulate在需要的平台项目。
  6. 额外提示:如果您<Router>在项目中使用,请将其更改为<HashRouter>否则您将看到空白显示,因为屏幕上不会呈现任何内容。
谢谢你,你的额外提示,救了我!
2021-05-23 11:11:59
@Halt 没试过。请让我们知道它是否有效。虽然以上是适用于Windows的命令
2021-05-26 11:11:59
奖金提示拯救了我的一天!
2021-05-29 11:11:59
我尝试了各种方法,这是唯一对我有用的方法。
2021-06-04 11:11:59
为什么不使用cp -rf ./build/ ../www/
2021-06-15 11:11:59

我认为很难找到如何解决这个问题的完整指南。我是这样解决的,从头到尾,为了能够在 Windows 上的模拟 Android 设备上运行 Create React App:

首先创建一个 React 应用程序或使用您现有的应用程序。

npx create-react-app my-app

https://github.com/facebook/create-react-app#creating-an-app

然后安装cordova:

npm install -g cordova

https://cordova.apache.org/docs/en/latest/guide/cli/

my-app在我的例子中,文件夹内创建一个新的cordova应用程序

cordova create hello com.example.hello HelloWorld

将目录更改为hello或您称为 Cordova 应用程序的名称。

cordova platform add ios
cordova platform add android

运行cordova requirements以查看构建项目所需的内容。

在此处输入图片说明

因为我使用的是 Windows,所以在这个例子中我只会为 Android 构建它。

cordova platform remove ios

并确认我只有 Android cordova platform ls

在此处输入图片说明

根据cordova requirements命令安装你需要的东西因为我是全新安装的,所以我需要一切:Java 开发工具包 (JDK) 8、Gradle 和 Android SDK。链接可以在这里找到:

https://cordova.apache.org/docs/en/latest/guide/platforms/android/index.html#requirements-and-support

或者:

https://www.oracle.com/java/technologies/javase/javase-jdk8-downloads.html

https://gradle.org/install/

https://developer.android.com/studio/index.html

安装后打开Android Studio。我选择了标准安装,但失败并显示以下警告:

无法安装英特尔 HAXM。详情请查看安装日志:“C:\Users\Oscar\AppData\Local\Temp\haxm_log.txt” Intel® HAXM 安装失败。要安装英特尔® HAXM,请按照以下说明进行操作:https ://software.intel.com/android/articles/installation-instructions-for-intel-hardware-accelerated-execution-manager-windows 安装程序日志位于

C:\Users\Oscar\AppData\Local\Temp\haxm_log.txt 安装程序日志内容: === Logging started: 2020-07-10 16:39:27 === 这台计算机不支持 Intel Virtualization Technology (VT- x) 或者它专门由 Hyper-V 使用。无法安装 HAXM。请确保在 Windows 功能中禁用 Hyper-V,或参阅英特尔 HAXM 文档了解更多信息。

然而,我无论如何都可以启动应用程序并添加在配置下找到的 Android 虚拟设备 (AVD)。

在此处输入图片说明

我选择添加一个Pixel XL带有R系统映像。

但是cordova requirements再次运行我可以看到我需要一个 API 级别为 28 的 Android 目标。R 是级别 30。

在此处输入图片说明

因此Pie我安装了 API 级别 28 x86_64 并创建了一个新的虚拟设备。

在此处输入图片说明

AVD Manager我没有打开SDK manager而是打开并下载了 Android 9.0 Pie SDK。

在此处输入图片说明

现在一切看起来都不错:

在此处输入图片说明

然后运行cordova emulate android以测试默认的 Cordova 应用程序。

如果它有效,它应该是这样的:

在此处输入图片说明

将目录更改为my-app.

在依赖package.json"homepage": "./",之前编辑和添加

在此处输入图片说明

感谢@BlackBeard。来源:https : //stackoverflow.com/a/46785362/3850405

npm run build

清除所有内容,my-app\hello\www然后将所有内容复制my-app\buildmy-app\hello\www

瞧:

在此处输入图片说明

如果您不进行编辑my-app package.json和添加"homepage": "./",,它将如下所示:

在此处输入图片说明

得到教训:

1.

如果您<Router>在项目中使用,请将其更改为<HashRouter>否则您将看到空白显示,因为屏幕上不会呈现任何内容。适用于 iOS 和 Android。

来源:https : //stackoverflow.com/a/46785362/3850405

2.

您需要一个白名单来允许 URL。从文档:

默认情况下,只允许导航到 file:// URL。要允许其他 URL,您必须在 config.xml 中添加标签:

https://cordova.apache.org/docs/en/latest/reference/cordova-plugin-whitelist/

像这样安装:

cordova plugin add cordova-plugin-whitelist

然后编辑config.xml位于应用程序根目录中的 which 并添加以下任何一项:

<!-- Allow links to example.com -->
<allow-navigation href="http://example.com/*" />

<!-- Wildcards are allowed for the protocol, as a prefix
     to the host, or as a suffix to the path -->
<allow-navigation href="*://*.example.com/*" />

<!-- A wildcard can be used to whitelist the entire network,
     over HTTP and HTTPS.
     *NOT RECOMMENDED* -->
<allow-navigation href="*" />

来源:https : //stackoverflow.com/a/30327204/3850405

3.

即使您使用的是白名单,您可能仍然需要访问不支持 https 的 http API。默认情况下这是不允许的,可能会导致一些真正的头痛。通过编辑config.xml并在下面添加以下内容来解决此问题<platform name="android">

<edit-config xmlns:android="http://schemas.android.com/apk/res/android"  file="app/src/main/AndroidManifest.xml" mode="merge" target="/manifest/application">     <application android:usesCleartextTraffic="true" /></edit-config>

鉴于您没有浏览到 URL,任何 API 调用都必须指定实际服务器。我通常使用 Axios,所以我们只需要将我们的服务器添加到默认 URL。例子:

import axios, { AxiosPromise, AxiosRequestConfig, Method } from 'axios';

const getConfig = (url: string, method: Method, params?: any, data?: any) => {
     const config: AxiosRequestConfig = {
         url: 'http://192.168.1.249' + url,
         method: method,
         responseType: 'json',
         params: params,
         data: data,
         headers: { 'X-Requested-With': 'XMLHttpRequest' },
    }
    return config;
}

export const sendRequest = (url: string, method: Method, params?: any, data?: any): AxiosPromise<any> => {
    return axios(getConfig(url, method))
}

然后像这样调用:

const path = '/api/test/'

export const initialLoad = (number: number): AxiosPromise<InitialLoadDto> => {
    return sendRequest(path + 'InitialLoad/' + number, 'get');
}

npm i -g react.cordova

https://www.npmjs.com/package/react.cordova

cli 为您完成所有工作。它已经被修复,现在工作得很好。

*我写了这个

在推荐您自己编写的工具时,添加免责声明(“我写了这个”)
2021-05-26 11:11:59
哇....谢谢你写的。我陷入了让 livereload 工作在我添加 Cordova 的 create-react-app 上的问题。这很棒!
2021-06-20 11:11:59