钩子
简介
Cordova 钩子代表特殊的脚本,可以由应用程序和插件开发人员,甚至您自己的构建系统添加,以自定义 cordova 命令。
Cordova 钩子允许您在 cordova 命令周围执行特殊活动。例如,您可能有一个自定义工具来检查您的 javascript 文件中的代码格式。并且,您希望在每次构建之前运行此工具。在这种情况下,您可以使用“before_build”钩子,并指示 cordova 运行时在每次构建之前调用自定义工具。
钩子可能与您的应用程序活动相关,例如 before_build
、after_build
等。或者,它们可能与应用程序的插件相关。例如,before_plugin_add
、after_plugin_add
等钩子适用于插件相关活动。这些钩子可以与应用程序中的所有插件相关联,也可以仅特定于一个插件。
Cordova 支持以下钩子类型
钩子类型 | 关联的 Cordova 命令 | 描述 |
---|---|---|
before_platform_add | cordova platform add |
在添加平台之前和之后执行。 |
after_platform_add | ||
before_platform_rm | cordova platform rm |
在删除平台之前和之后执行。 |
after_platform_rm | ||
before_platform_ls | cordova platform ls |
在列出已安装和可用的平台之前和之后执行。 |
after_platform_ls | ||
before_prepare | cordova prepare cordova platform add cordova build cordova run |
在准备您的应用程序之前和之后执行。 |
after_prepare | ||
before_compile | cordova compile cordova build |
在编译您的应用程序之前和之后执行。 |
after_compile | ||
before_deploy | cordova emulate cordova run |
在部署您的应用程序之前执行。 |
before_build | cordova build |
在构建您的应用程序之前和之后执行。 |
after_build | ||
before_emulate | cordova emulate |
在模拟您的应用程序之前和之后执行。 |
after_emulate | ||
before_run | cordova run |
在运行您的应用程序之前和之后执行。 |
after_run | ||
before_serve | cordova serve |
在服务您的应用程序之前和之后执行。 |
after_serve | ||
before_clean | cordova clean |
在清理您的应用程序之前和之后执行。 |
after_clean | ||
before_plugin_add | cordova plugin add |
在添加插件之前和之后执行。 |
after_plugin_add | ||
before_plugin_rm | cordova plugin rm |
在删除插件之前和之后执行。 |
after_plugin_rm | ||
before_plugin_ls | cordova plugin ls |
在列出应用程序中的插件之前和之后执行。 |
after_plugin_ls | ||
before_plugin_install | cordova plugin add |
在安装插件(到平台)之前和之后执行。plugin.xml 中的插件钩子仅针对正在安装的插件执行 |
after_plugin_install | ||
before_plugin_uninstall | cordova plugin rm |
在卸载插件(从平台)之前执行。plugin.xml 中的插件钩子仅针对正在安装的插件执行 |
定义钩子的方式
Config.xml
可以使用 <hook>
元素在项目的 config.xml
中定义钩子,例如
<hook type="before_build" src="scripts/appBeforeBuild.bat" />
<hook type="before_build" src="scripts/appBeforeBuild.js" />
<hook type="before_plugin_install" src="scripts/appBeforePluginInstall.js" />
<platform name="android">
<hook type="before_build" src="scripts/android/appAndroidBeforeBuild.bat" />
<hook type="before_build" src="scripts/android/appAndroidBeforeBuild.js" />
<hook type="before_plugin_install" src="scripts/android/appAndroidBeforePluginInstall.js" />
...
</platform>
插件钩子 (plugin.xml)
作为插件开发人员,您可以使用 plugin.xml
中的 <hook>
元素定义钩子脚本,如下所示
<hook type="before_plugin_install" src="scripts/beforeInstall.js" />
<hook type="after_build" src="scripts/afterBuild.js" />
<platform name="android">
<hook type="before_plugin_install" src="scripts/androidBeforeInstall.js" />
<hook type="before_build" src="scripts/androidBeforeBuild.js" />
...
</platform>
before_plugin_install
、after_plugin_install
、before_plugin_uninstall
插件钩子将专门为正在安装/卸载的插件触发。
钩子执行的顺序
基于钩子定义
一个给定钩子的钩子脚本按照它们在文件中的出现顺序依次运行,config.xml
中的应用程序钩子在 plugins/.../plugin.xml
中的插件钩子之前运行。
基于内部执行顺序
钩子的内部执行顺序是固定的。
示例 1 (cordova platform add)
如果存在与 before_platform_add
、after_platform_add
、before_prepare
、after_prepare
、before_plugin_install
和 after_plugin_install
相关的钩子(并假设您的项目上安装了一个插件),则添加新平台将按以下顺序执行钩子
before_platform_add
before_prepare
after_prepare
before_plugin_install
after_plugin_install
after_platform_add
示例 2 (cordova build)
如果存在与 before_prepare
、after_prepare
、before_compile
、after_compile
、before_build
和 after_build
相关的钩子 - 运行构建命令将按以下顺序执行钩子
before_build
before_prepare
after_prepare
before_compile
after_compile
after_build
脚本接口
Javascript
如果您正在使用 Node.js 编写钩子,则应使用以下模块定义
module.exports = function(context) {
...
}
这是一个展示 context
对象内容的示例
{
// The type of hook being run
hook: 'before_plugin_install',
// Absolute path to the hook script that is currently executing
scriptLocation: '/foo/scripts/appBeforePluginInstall.js',
// The CLI command that lead to this hook being executed
cmdLine: 'cordova plugin add plugin-withhooks',
// The options associated with the current operation.
// WARNING: The contents of this object vary among the different
// operations and are currently not documented anywhere.
opts: {
projectRoot: '/foo',
cordova: {
platforms: ['android'],
plugins: ['plugin-withhooks'],
version: '0.21.7-dev'
},
// Information about the plugin currently operated on.
// This object will only be passed to plugin hooks scripts.
plugin: {
id: 'plugin-withhooks',
pluginInfo: { /* ... */ },
platform: 'android',
dir: '/foo/plugins/plugin-withhooks'
}
},
// A reference to Cordova's API
cordova: { /* ... */ }
}
您还可以使用 context.requireCordovaModule
在脚本中需要其他 Cordova 模块,如下所示
const cordovaCommon = context.requireCordovaModule('cordova-common');
您可以使用 Promise 使脚本异步。这是一个仅等待一秒钟然后打印等待花费的毫秒数的示例
module.exports = context => {
return new Promise(resolve => {
const start = Date.now();
setTimeout(() => resolve(Date.now() - start), 1000);
}).then(msWaited => {
console.log(`${context.scriptLocation} waited ${msWaited} ms`);
});
};
非 javascript
非 javascript 脚本通过 Node child_process spawn 从项目的根目录运行,并将根目录作为第一个参数传递。所有其他选项都使用环境变量传递给脚本
环境变量名称 | 描述 |
---|---|
CORDOVA_VERSION | Cordova-CLI 的版本。 |
CORDOVA_PLATFORMS | 命令适用的平台的逗号分隔列表(例如:android、ios)。 |
CORDOVA_PLUGINS | 命令适用的插件 ID 的逗号分隔列表(例如:cordova-plugin-file-transfer、cordova-plugin-file)。 |
CORDOVA_HOOK | 正在执行的钩子的路径。 |
CORDOVA_CMDLINE | 传递给 cordova 的确切命令行参数(例如:cordova run ios --emulate)。 |
如果脚本返回非零退出代码,则父 cordova 命令将被中止。
注意:我们强烈建议使用 Node.js 编写您的钩子,以便它们是跨平台的,请参阅上面的 Javascript 部分。
Windows 怪癖
如果您在 Windows 上工作,并且您的钩子脚本不是 *.bat
文件,Cordova CLI 将期望 shebang 行作为脚本的第一行。这样它就知道需要使用哪个解释器来启动脚本。Python 脚本的 shebang 行可能如下所示
#!/usr/bin/env python
示例用法
此示例演示了 Cordova 钩子的用法,以跟踪到控制台输出 Android 平台生成的 .apk 文件的大小。
创建空白 Cordova 应用程序,并将以下定义添加到 config.xml
,以告诉 Cordova 在每个平台构建后运行 afterBuild.js
脚本。
<hook type="after_build" src="scripts/afterBuild.js" />
创建 scripts/afterBuild.js
文件,并添加以下实现。我们使用 fs.stat
方法的异步版本来演示如何在钩子中使用异步函数。
const fs = require('fs');
const util = require('util');
const stat = util.promisify(fs.stat);
module.exports = function(ctx) {
// Make sure android platform is part of build
if (!ctx.opts.platforms.includes('android')) return;
const platformRoot = path.join(ctx.opts.projectRoot, 'platforms/android');
const apkFileLocation = path.join(platformRoot, 'build/outputs/apk/android-debug.apk');
return stat(apkFileLocation).then(stats => {
console.log(`Size of ${apkFileLocation} is ${stats.size} bytes`);
});
};
示例中的参数 ctx
由 Cordova 传递,代表执行上下文,例如脚本的完整路径、目标平台、命令行参数等,并且还公开了其他辅助功能。有关更多详细信息,请参阅上面的 脚本接口
部分。
您现在可以添加 android 平台并执行构建。
cordova platform add android
..
cordova build
..
Size of path\to\app\platforms\android\build\outputs\apk\android-debug.apk is 1821193 bytes