NW.js 簡(jiǎn)介與使用方法
簡(jiǎn)介
NW.js (原名 node-webkit)是一個(gè)基于 Chromium 和 node.js 的應(yīng)用運(yùn)行時(shí),通過(guò)它可以用 HTML 和 JavaScript 編寫(xiě)原生應(yīng)用程序。它還允許您從 DOM 調(diào)用 Node.js 的模塊 ,實(shí)現(xiàn)了一個(gè)用所有 Web 技術(shù)來(lái)寫(xiě)原生應(yīng)用程序的新的開(kāi)發(fā)模式。
(1)以網(wǎng)絡(luò)最流行的技術(shù)編寫(xiě)原生應(yīng)用程序的新方法
(2)基于HTML5, CSS3, JS and WebGL而編寫(xiě)
(3)完全支持nodejs所有api及第三方模塊
(4)可以使用DOM直接調(diào)用nodejs模塊
(5)容易打包和分發(fā)
(6)支持運(yùn)行環(huán)境包括32位和64位的Window、Linux和Mac OS
使用方法如下:
一、下載nw
1.下載 NW.js(官網(wǎng):http://nwjs.io/)
這里面normal這個(gè)算是運(yùn)行時(shí)吧,sdk那個(gè)是一些工具箱,建議都下下來(lái)~
2.下載 Enigma Virtual Box(官網(wǎng):http://enigmaprotector.com/)
二、配置 package.json 文件
{
"name": "nw-demo",
"version": "0.0.1",
"main": "index.html"
}
更多的可用如下:
{
"main": "app/index.html",
"name": "WeixinMenuEditor",
"description": "使用nw.js封裝的一個(gè)微信公眾號(hào)菜單編輯器App",
"version": "0.0.1",
"keywords": [ "微信", "菜單編輯器" ],
"window": {
"title": "微信菜單編輯器",
"icon": "app/static/img/weixin_logo.jpg",
"toolbar": true,
"frame": true,
"width": 1008,
"height": 750,
"position": "center",
"min_width": 400,
"min_height": 200
},
"webkit": {
"plugin": true,
"java": false,
"page-cache": false
},
"chromium-args" :"-allow-file-access-from-files"
}
- title : 字符串,設(shè)置默認(rèn) title。
- width/height : 主窗口的大小。
- toolbar : bool 值。是否顯示導(dǎo)航欄。
- icon : 窗口的 icon。
- position :字符串。窗口打開(kāi)時(shí)的位置,可以設(shè)置為“null”、“center”或者“mouse”。
- min_width/min_height : 窗口的最小值。
- max_width/max_height : 窗口顯示的最大值。
- resizable : bool 值。是否允許調(diào)整窗口大小。
- always-on-top : bool 值。窗口置頂。
- fullscreen : bool 值。是否全屏顯示。
- show_in_taskbar : 是否在任務(wù)欄顯示圖標(biāo)。
- frame : bool 值。如果設(shè)置為 false,程序?qū)o(wú)邊框顯示。
- "chromium-args" :"-allow-file-access-from-files" 相當(dāng)于給谷歌瀏覽器添加啟動(dòng)參數(shù)一樣,這行代碼允許angularjs直接訪問(wèn)本地json文件。
三、生成exe
項(xiàng)目目錄如下:

將html項(xiàng)目壓縮成zip,并改名為nw,輸入以下命令
copy /b nw.exe+app.nw firstApp.exe
四、打發(fā)包發(fā)布
打開(kāi) Enigma Virtual Box 程序(enigmavb.exe),界面應(yīng)該是這樣的:

然后在 Enter Input File Name 處選擇上一步生成的 test.exe 文件,Enter Output Name 可以默認(rèn);
之后再點(diǎn)擊下面的 Add 按鈕,將 nwjs 文件夾(名稱不一定是 nwjs ,就是最開(kāi)始第一步 NW.js 環(huán)境的那個(gè)文件夾)下除 nw.exe 和 test.nw 以及 test.exe 之外的所有文件加載上,然后點(diǎn)擊 Process ,等待執(zhí)行成功即可,這時(shí)候會(huì)在相應(yīng)的路徑下生成一個(gè)新的 .exe 文件(我們暫且叫做 newtest.exe),此時(shí)的 newtest.exe 文件即可在任意的 Windows 環(huán)境下運(yùn)行了,你可以拷貝給你的小伙伴去 Show 一下。
下面是nw使用過(guò)程中的一些坑
1.如果只希望當(dāng)前應(yīng)用獲取焦點(diǎn)才執(zhí)行快捷鍵,看看這個(gè)庫(kù)用js設(shè)置快捷鍵
// 加載本地ui庫(kù)
var gui = require('nw.gui');
var option = {
key: "Ctrl+R",
active: function () {
alert("全局快捷鍵" + this.key + "按下");
},
failed: function (msg) {
//創(chuàng)建快捷鍵失敗
alert(msg);
}
};
// 創(chuàng)建快捷鍵
var shortcut = new gui.Shortcut(option);
// 注冊(cè)全局快捷鍵
gui.App.registerGlobalHotKey(shortcut);
// 解除注冊(cè),在應(yīng)用結(jié)束的時(shí)候執(zhí)行
gui.App.unregisterGlobalHotKey(shortcut);
2.nw.js不能對(duì)頁(yè)面多次刷新,各種不正常,這是由于刷新頁(yè)面后重新加載js文件對(duì)變量重新賦值引起的bug。 解決方案
nw.js 讀取和保存文件
<html>
<head>
<meta charset="utf-8"/>
<title>nw.js實(shí)現(xiàn)文件讀寫(xiě)</title>
</head>
<body>
<input id="readFile" type="file" >讀取文件</input>
<!-- 默認(rèn)文件名為filename.html -->
<input id="writeFile" nwsaveas="filename.html" type="file">保存文件</input>
<p></p>
<script>
//nw.js提供的讀寫(xiě)文件模塊
var fs = require("fs");
//讀文件
var chooser = document.querySelector('#readFile');
chooser.addEventListener("change", function (evt) {
//用戶選擇的文件
var filePath = this.value.toString();
document.querySelector("p").innerHTML = "讀取文件從" + filePath;
fs.readFile(filePath, function (err, data) {
if (err) {
layer.msg("讀取文件失敗! :" + err.message);
return;
} else {
console.log(data);
alert(data);
}
})
});
//寫(xiě)文件
chooser = document.querySelector('#writeFile');
chooser.addEventListener("change", function (evt) {
//用戶選擇的文件
var filePath = this.value.toString();
document.querySelector("p").innerHTML = "寫(xiě)入文件到:" + filePath;
//把hello寫(xiě)入文件
fs.writeFile(filePath, "Hello!\n", function (err) {
if (err) {
alert("保存失敗!");
}
});
});
</script>
</body>
</html>
3.使用nwjs的'fs'直接保存cancas為本地圖片,在網(wǎng)上找到的方法都是彈出選擇框保存,但我需要直接保存圖片到指定路徑,不能彈出對(duì)話框讓用戶選擇。kailniris給了一個(gè)解決方案,可行,代碼如下:
var fs = require('fs');
var c = document.getElementById("myCanvas");
var ctx = c.getContext("2d");
ctx.moveTo(0, 0);
ctx.lineTo(200, 100);
ctx.stroke();
<canvas id="myCanvas" width="200" height="100" style="border:1px solid #000000;">
</canvas>
base64Data = c.toDataURL("image/png").replace(/^data:image\/png;base64,/, "")
fs.writeFile("c:/Dev/test.png", base64Data, 'base64', function (err) {
if (err) {
console.log("err", err);
} else {
return res.json({ 'status': 'success' });
}
});
用html2canvas把html頁(yè)面轉(zhuǎn)換為圖片,再把圖片保存到本地。貼一下代碼(需要導(dǎo)入html2canvas.js和jquery):
//要保存圖片的文件路徑
var filePath = templateDir + filename + '.html';
//要保存的html頁(yè)面
var editerDocument = window.editor.edit.iframe.get().contentWindow.document;
html2canvas(editerDocument.body, {
onrendered: function (canvas) {
var base64Data = canvas.toDataURL("image/png").replace(/^data:image\/png;base64,/, "")
var fs = require("fs");
fs.writeFile(templateDir + filename + '.png', base64Data, 'base64', function (err) {
if (err) {
alert("保存模板失敗!");
}
$('#model_template_name').modal("hide");
layer.msg("模板已保存為" + filename);
});
}
});
4.在app.js里引用Node內(nèi)置模塊
//調(diào)用NodeJs內(nèi)置模塊
$scope.fs = require('fs');
//讀取配置文件
$scope.readConfig = function () {
try {
var configStr = $scope.fs.readFileSync(config.weixin.path, 'utf8');
console.log(configStr);
var obj = eval('(' + configStr + ')');
$scope.weixin.appid = obj.appid;
$scope.weixin.appsecret = obj.appsecret;
$scope.weixin.qrcodeurl = obj.qrcodeurl;
}
catch (e) {
console.log(e);
alert("讀取微信配置文件失敗");
}
}
//寫(xiě)入配置文件
$scope.writeConfig = function () {
try {
var configStr = JSON.stringify($scope.weixin);
$scope.fs.writeFileSync(config.weixin.path, configStr, {encoding: 'utf8'});
return true;
}
catch (e) {
console.log(e);
alert("寫(xiě)入微信配置文件失敗");
return false;
}
}
5.引用第三方模塊wechat-api
//調(diào)用NodeJs第三方模塊
$scope.wechatApi = require('wechat-api');
$scope.query = function () {
var api = new $scope.wechatApi($scope.weixin.appid, $scope.weixin.appsecret);
api.getMenu(function (err, result) {
if (err) {
console.log(err);
alert("查詢菜單異常");
} else {
load(result);
$scope.$apply();//需要手動(dòng)刷新
}
});
};
更多詳細(xì)的可以參考 http://liuxp.me/nwjs/References/Window/ 中文文檔
總結(jié)
以上所述是小編給大家介紹的NW.js 簡(jiǎn)介與使用方法,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
w3c聲明下可運(yùn)行 兼容性比較好的js對(duì)聯(lián)廣告集合
最近有美工朋友找我們要兼容性比較好的對(duì)聯(lián)廣告代碼,我們給他一個(gè)他竟然不能運(yùn)行,經(jīng)過(guò)排查,這是因?yàn)楝F(xiàn)在的廣告也用了一樣代碼,函數(shù)命名重復(fù)。2011-07-07
JS使用canvas實(shí)現(xiàn)基本的截圖功能
這篇文章主要給大家介紹了使用JS中的canvas實(shí)現(xiàn)基本的截圖功能,文中有詳細(xì)的實(shí)現(xiàn)思路和實(shí)現(xiàn)過(guò)程,文章通過(guò)代碼示例講解的非常詳細(xì),很感興趣的同學(xué)可以參考一下2023-08-08
給頁(yè)面渲染時(shí)間加速 干掉Dom Level 0 Event
我們?nèi)サ羰录壎ǖ倪壿?發(fā)現(xiàn)只渲染dom元素,不綁定事件的時(shí)間,僅僅125ms,可見(jiàn)事件綁定的時(shí)間消耗還是很大的 ,尤其是第一種方式,也就是Dom Level 0 Event,最為耗時(shí)2012-12-12
js實(shí)現(xiàn)鼠標(biāo)點(diǎn)擊文本框自動(dòng)選中內(nèi)容的方法
這篇文章主要介紹了js實(shí)現(xiàn)鼠標(biāo)點(diǎn)擊文本框自動(dòng)選中內(nèi)容的方法,涉及javascript鼠標(biāo)點(diǎn)擊事件onClick及選擇事件select的使用技巧,非常簡(jiǎn)單實(shí)用,需要的朋友可以參考下2015-08-08
JavaScript自定義等待wait函數(shù)實(shí)例分析
這篇文章主要介紹了JavaScript自定義等待wait函數(shù),實(shí)例分析了自定義等待函數(shù)的實(shí)現(xiàn)與使用技巧,需要的朋友可以參考下2015-03-03
JS實(shí)現(xiàn)的圖片選擇順序切換和循環(huán)切換功能示例【測(cè)試可用】
這篇文章主要介紹了JS實(shí)現(xiàn)的圖片選擇順序切換和循環(huán)切換功能,結(jié)合完整實(shí)例形式分析了JavaScript基于事件響應(yīng)與樣式動(dòng)態(tài)操作實(shí)現(xiàn)圖片切換相關(guān)操作技巧,需要的朋友可以參考下2018-12-12
一個(gè)html5播放視頻的video控件只支持android的默認(rèn)格式mp4和3gp
寫(xiě)了個(gè)html5播放視頻的video控件,只支持mp4和3gp(android和ios默認(rèn)支持的格式就寫(xiě)了這個(gè)) ,需要的朋友可以參考下2014-05-05
js substring從右邊獲取指定長(zhǎng)度字符串(示例代碼)
本篇文章主要是對(duì)js substring從右邊獲取指定長(zhǎng)度字符串的示例代碼進(jìn)行了介紹,需要的朋友可以過(guò)來(lái)參考下,希望對(duì)大家有所幫助2013-12-12
layui添加動(dòng)態(tài)菜單與選項(xiàng)卡
這篇文章主要為大家詳細(xì)介紹了layui添加動(dòng)態(tài)菜單與選項(xiàng)卡,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-07-07

