cordova3.X 运用grunt快速生成plugin自定义插件骨架

Cordova提供了一组设备相关的API,通过这组API,移动应用能够以JavaScript访问原生的设备功能,如摄像头、麦克风等。Cordova还提供了一组统一的JavaScript类库,以及为这些类库所用的设备相关的原生后台代码。

一、安装cordova
npm install -g cordova
二、创建项目
cordova create hello com.blue.sky.hybrid.app.hello HelloWorld
三、添加平台支持
cd hello
cordova platform add android`(前提是系统上面要安装了对应移动系统的SDK)
四、自定义插件

学了这么多, 准备自己写个自定义插件,变在命令行输入:cordova plugin create com.blue.sky.test 发现不管用. 查阅资料后, 发现没有这个命令, 网上大家都是手动创建目录, 觉得太麻烦, 于是用grunt 写了一个命令通过模板生成cordova plugin骨架。

》》网上的方法大概是这样:

cordova3.X之后,插件不能自己手动添加了,手动添加后,只要cordova build,数据立即被抹去.
因此,3.X后要添加插件,需要用 cordova plungin add "你本地插件的路径" 的方式来添加插件,.
1.新建一个文件夹,命名为你插件的名字,如TestPlugin
2.在文件夹里再新建2个文件夹和1个文件.两个文件夹分别是src和www.其中src中放你插件的java代码,www中放对应的js文件;与src和www文件夹同级,建立plugin.xml

》》运用grunt模板生成cordova plugin骨架

思路:cordova plugin 主要是三个文件:
1、继承cordovaActivity的Native实现类
2、编写javascript代码
3、编写plugin.xml配置文件

既然是这样, 便可以运用grunt通过模板生成cordova plugin骨架。
首先看一下项目代码结构:


第一步:制定cordova plugin模板

模板内容如下:
**src/android/template.txt **继承cordovaActivity的Native实现类

package <%=pkgName%>;
 
import java.util.TimeZone;
 
import org.apache.cordova.CordovaWebView;
import org.apache.cordova.CallbackContext;
import org.apache.cordova.CordovaPlugin;
import org.apache.cordova.CordovaInterface;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
 
import android.provider.Settings;
 
public class <%=className%> extends CordovaPlugin {
 
    public <%=className%>() {
 
    }
 
    /**
     * Executes the request and returns PluginResult.
     *
     * @param action            The action to execute.
     * @param args              JSONArry of arguments for the plugin.
     * @param callbackContext   The callback id used when calling back into JavaScript.
     * @return                  True if the action was valid, false if not.
     */
    public boolean execute(String action, JSONArray args, CallbackContext callbackContext) throws JSONException {
 
        ///TODO 自定义实现
 
        return true;
    }
 
}

**www/template.txt **javascript 模板

var channel = require('cordova/channel'),
  utils = require('cordova/utils'),
  exec = require('cordova/exec'),
  cordova = require('cordova');
 
function <%=className%>() {
 
}
 
module.exports = new <%=className%>();

plugin.xml 插件编译生成android 项目代码配置文件

<?xml version="1.0" encoding="UTF-8"?>
<!--插件id号,与package.json保持一致 版本号,与package.json保持一致-->
<plugin xmlns="http://apache.org/cordova/ns/plugins/1.0"
        id="<%=pkgName%>" version="0.1">
    <!--插件在cordova下的名称,Test.js中exec的接口名称,保持一致-->
    <name><%=className%></name>
    <description>Cordova Plugin</description>
    <license>Apache 2.0</license>
    <!--与package.json保持一致-->
    <keywords></keywords>
    <repo></repo>
    <issue></issue>
    <!--插件js接口文件配置信息,插件在android-->
    <!--src="www/Test.js"为已经写好的js文件路径,与js中调用的类名保持一致-->
    <js-module src="www/<%=className%>.js" name="<%=className%>">
        <!--插件在js中调用的类名-->
        <clobbers target="<%=className%>" />
    </js-module>
 
    <!-- android -->
    <platform name="android">
        <config-file target="res/xml/config.xml" parent="/*">
            <!--插件在java端的接口名称,与之前文件中的接口名称保持一致-->
            <feature name="<%=className%>">
                <!--该插件接口对应的java代码路径-->
                <param name="android-package" value="<%=pkgName%><%=className%>"/>
            </feature>
        </config-file>
 
        <!--该插件需要的权限申明,根据需要自行定义-->
        <config-file target="AndroidManifest.xml" parent="/*">
            <uses-permission android:name="android.permission.INTERNET" />
            <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
            <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
        </config-file>
 
        <!--源文件的路径和目标文件路径,src为已经编写好的java代码路径,target-dir为需要生成的android工程中该java源码路径,与上面的java代码路径保持一致-->
        <source-file src="<%=sourceSrc%>" target-dir="<%=targetDir%>"/>
 
    </platform>
 
</plugin>

运用Node.js grunt根据模板生成骨架代码:
grunt plugin:create:com.blue.sky.test:Test

module.exports = function(grunt) {
    grunt.registerTask('plugin:create', '自定义插件 参数一 包名  参数二 插件类名', function (arg1, arg2, arg3) {
        grunt.log.writeln(">>>>length:" + arguments.length);
        if (arguments.length === 2) {
     
          var pkgName = arg1;
          var fileName = arg2;
          var platform = arg3 || "android";
          var pluginDir = "temp/" + arg1;
          var tplNativeCode = "template/src/" + platform + "/template.js";
          var tplJSCode = "template/www/template.js";
          var tplPlugin = "template/plugin.xml";
     
          var srcFileName = pluginDir + "/src/" + platform + "/" + arg2 + ".java";
          var jsFileName = pluginDir + "/www/" + arg2 + ".js";
          var configFileName = pluginDir + "/plugin.xml";
     
          grunt.log.writeln("start create plugin:" + arg1);
     
          grunt.file.mkdir(pluginDir);
     
          // 创建插件java类
          grunt.file.mkdir(pluginDir + "/src/" + platform);
          grunt.file.write(srcFileName, grunt.file.read(tplNativeCode));
          var content = grunt.file.read(srcFileName);
          var text = grunt.template.process(content, {data: {"pkgName": pkgName + "." + fileName, "className": fileName}});
          grunt.file.write(srcFileName, text);
     
          // 创建插件javascript
          grunt.file.mkdir(pluginDir + "/www");
          grunt.file.write(jsFileName, grunt.file.read(tplJSCode));
          var jsContent = grunt.file.read(jsFileName);
          var jsText = grunt.template.process(jsContent, {data: {"className": fileName}});
          grunt.file.write(jsFileName, jsText);
     
          // 创建插件配置文件plugin.xml
          var configContent = grunt.file.read(tplPlugin);
          var configText = grunt.template.process(configContent,
            {
              data: {
                "pkgName": pkgName,
                "className": fileName,
                "sourceSrc":"src/"+ platform + "/" + fileName +  ".java",
                "targetDir":"src/" + pkgName.replace(/\./g,"/")
              }
            }
          );
          grunt.file.write(configFileName, configText);
     
     
          grunt.log.writeln("create plugin success");
     
        } else {
          grunt.log.writeln("命令格式错误。 grunt plugin:create 包名 插件类名");
        }
    });
}

使用cordova plugin add "本地自定义插件代码"

cordova plugin add "D:\Project\workspace\phonegap\hello\temp\com.blue.sky.test"

运行之后, 在plugins 目录下面会看到有com.blue.sky.test插件(请看项目结果图)。

运行cordova run android 命令打包程序到手机
运行后,查看platforms目录下面生成了自定义的相关代码,如下图所示:

总结

通过运用grunt生成cordova plugin 可以很方面的创建plugin骨架, 省去繁琐的步骤。当然, 这个demo只是实现了android平台的plugin。如果要支持ios、wp也比较简单,只需要加相应的模板以及映射关系即可。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,793评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,567评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,342评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,825评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,814评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,680评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,033评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,687评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,175评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,668评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,775评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,419评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,020评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,206评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,092评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,510评论 2 343

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,596评论 18 139
  • 主要的目的就在这了,我要学习一下怎么写cordova插件,如何调试,如何写config plugin.xml文件...
    wyude阅读 318评论 0 0
  • 创建Cordova项目 一、环境配置(mac版): 1.安装node.js(https://nodejs.org/...
    XDUZ阅读 1,341评论 1 3
  • 所有项目的构建都是有生命周期的,这个生命周期包括:项目清理、初始化、编译、测试、打包、集成测试、验证、部署、站点生...
    zlcook阅读 2,753评论 0 21
  • Cordova自定义插件实战# 使用前提 这篇文章是之前发表在CSDN上的,拿过来充数用的,其实那个时候也写了不少...
    one_cup阅读 1,546评论 5 3