2天时间,聊今年最热的 Agent、上下文工程、AI 产品创新等话题。2025 年最后一场~ 了解详情
写点什么

使用 node-webkit 构建桌面应用程序(二)

  • 2015-05-19
  • 本文字数:4445 字

    阅读完需:约 15 分钟

在这片文章中,我们将使用 node-webkit 来编写一个实际的例子,来说明如何使用 HTML5+CSS3 技术编写桌面应用。我们将通过给 Moco 工具编写一个简单的设计器来说明一些具体的技术,比如和本地文件的交互,实现简单的上下文菜单等。

Moco 简介

Moco 是一个用于简化服务器搭建的轻量级的测试工具,使用 Moco 可以很容易地创建一个“假的”服务器,这个服务器的行为就好像真实的服务器一样。Moco 不但可以将集成测试变得更加简单,还可以搭建一个稳定的后端,从而使得纯前端的 Web 开发(或者前后端分离开发)变得平坦而容易。

Moco 需要一个配置文件来指明 request/response 的对应关系:

复制代码
[
{
"request": {
"method": "post",
"uri": "/action.do",
"json_paths": {
"$.type": "events"
}
},
"response": {
"status": 200,
"file": "spec/fixtures/events.json"
}
}
]

上例这个配置文件指定了:当发往 /action.do 的 POST 请求中,包含了 JSONPath 如 $.type=events 时,Moco 会响应一个 200 的状态码,并且会返回文件 spec/fixtures/events.json 的内容作为响应。

每个配置文件中可以指定多条规则,当请求到达时,Moco 会依次匹配,直到找到对应的规则,如果没有发现匹配,则返回 400 错误。

模板项目

我在 github 上创建了一个 node-webkit 的模板工程。你可以很从这个模板工程开始你的 node-webkit 之旅:

复制代码
$ git clone git@github.com:abruzzi/node-webkit-boilerplate.git
$ cd node-webkit-boilerplate
$ npm install
$ grunt

这个 grunt 命令会下载 node-webkit,并构建应用程序:

构建之后,可以执行

复制代码
$ ./launch.sh

启动应用程序,如果你看到了下面这个界面,说明基本的环境已经就绪,然后就可以进行下一步了。

引入 bootstrap

bootstrap 可以帮助我们快速地从零开始搭建一个专业的 Web 页面,bootstrap 提供了丰富的特性,开箱即用的组件,我们这里只使用其中的布局器和表单组件。

使用 bootstrap 的一个模板 HTML 如下:

复制代码
<html lang="en">
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>node-webkit application</title>
<!-- Bootstrap -->
<link href="css/bootstrap.min.css" rel="stylesheet">
</head>
<body>
<h1>Hello, node-webkit!</h1>
<script src="js/jquery.min.js"></script>
<script src="js/bootstrap.min.js"></script>
<script src="js/app.js"></script>
</body>
</html>

有了 bootstrap,接下来可以很容易地创建出专业的页面,比如在下面这个页面中,我们创建了几个列表:

复制代码
<div class="row">
<div class="media">
<a class="pull-left" href="#">
<img class="media-object" src="http://placehold.it/64x64">
</a>
<div class="media-body">
<h4 class="media-heading">Listing object</h4>
Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod
tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam,
quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo
consequat.
</div>
</div>
...
...
</div>

上面片段中的 http://placehold.it/ 是一个提供图片占位符的服务,你只需要指定占位符的大小如 64x64,这个服务就会返回给你一张对应尺寸的占位符。

有了这些元素之后就更像是一个桌面应用程序了。

Moco Services Designer

如前面所述,我们这篇文章中将要开发一个实际例子:moco 服务器需要一个配置文件,但是 moco 本身提供了丰富的特性,因此要记住所有的选项非常困难,所以需要一个图形界面支持用户通过简单的操作就可以导出 moco 的配置文件:

点击界面中的加号会生成一个新的规则,每个规则分为请求和响应两部分,这里我们只提供最简单的选项:

  1. 用户可以选择使用那种 HTTP 动词(GET/POST/PUT/DELETE);
  2. 用户可以指定一个 URI;
  3. 用户可以指定一个字符串作为返回;
  4. 用户可以定义多条规则;
  5. 访问本地文件。

首先在界面上有一个蓝色的加号:

复制代码
<div class="row">
<div class="col-lg-12 col-md-12">
<a id="addRule" href="#" class="pull-right">
<span class="glyphicon glyphicon-plus"></span>
</a>
</div>
</div>
<div id="rules">
</div>

点击这个加号可以在页面上 ID 为 rules 元素中添加一个新的 HTML 片段:

复制代码
$("#addRule").on("click", function() {
$("#rules").append(fragement);
});

这个片段来自另外一个文件:

复制代码
<div class="row">
<div class="panel panel-primary">
<div class="panel-heading">Rule
<a href="#" class="pull-right remove">
<span class="glyphicon glyphicon-minus"></span>
</a>
</div>
<ul class="list-group">
<li class="list-group-item">
<div class="input-group">
<select name="method">
<option value="get" selected="true">GET</option>
<option value="post">POST</option>
<option value="put">PUT</option>
<option value="delete">DELETE</option>
</select>
</div>
</li>
<li class="list-group-item">
Request
<div class="form-group">
<input type="text" class="form-control" name="uri" placeholder="/checkout">
</div>
</li>
<li class="list-group-item">
Response
<div class="form-group">
<textarea class="form-control" name="text" placeholder="content"></textarea>
</div>
</li>
</ul>
</div>
</div>

由于涉及本地文件的存取,我们需要引入 node.js 中的 fs 模块,首先在 app 目录中创建一个 node_modules 目录,并在该目录中创建一个 file.js 文件,内容如下:

复制代码
var fs = require('fs');
function File() {
function save(path, text) {
fs.writeFile(path, text);
}
function read(path, callback) {
fs.readFile(path, 'utf-8', callback);
}
this.save = save;
this.read = read;
}
module.exports = new File;

我们导出了 save 和 read 两个方法可供使用,然后在外部的 app.js 中可以通过 require 的方式引入该文件:

复制代码
var file = require('file.js');
var fragement = null;
file.read("rule.html", function(error, content) {
fragement = content;
});

要将页面上的内容保存到本地文件系统中,需要使用 node-webkit 提供的 nwsaveas。

复制代码
<input id="save" type="file" nwsaveas style="display:none" accept="text/*"/>

有了这个 nwsaveas,我们可以为它注册一个 change 事件的回调,当用户选择了文件名(选择文件是一个典型的异步操作,保存文件的代码其实不知道用户在何时完成选择)之后,这个回调会执行。

一个简单的做法是定义一个隐藏的按钮,ID 为 save,一个可见的按钮 ID 为 export,当 export 按钮被点击时,它触发 save 的点击,并且为 change 绑定回调函数。

复制代码
$("#export").on("click", function() {
chooseFile("#save");
});
function chooseFile(id) {
var chooser = $(id);
chooser.change(function(evt) {
var config = generateMocoConf();
var filename = $(this).val();
file.save(filename, config);
});
chooser.trigger('click');
}

这样,当用户完成选择之后,我们就有了文件名,这时候就可以将内容存入外部文件了。

当用户最终导出文件时,通过遍历所有的规则,然后生成一个 JSON:

复制代码
function generateMocoConf() {
var rules = $("#rules .row");
var config = [];
$(rules).each(function(index, item) {
var method = $('select[name="method"]', item).val();
config.push({
request: {
method: method,
uri: $('input[name="uri"]', item).val()
},
response: {
text: $('textarea[name="text"]', item).val()
}
});
});
return JSON.stringify(config, null, 4);
}

右键菜单

到目前为止一切都还很顺利,你可以添加规则,保存规则到文件。但是你可能已经发现了,拷贝 / 粘贴等不工作,而且右键菜单也不见了。

我们需要自己构建一个右键菜单,比如最简单的拷贝 / 剪切 / 粘贴:

复制代码
function Menu(cutLabel, copyLabel, pasteLabel) {
var gui = require('nw.gui'),
menu = new gui.Menu(),
cut = new gui.MenuItem({
label: cutLabel || "Cut",
click: function() {
document.execCommand("cut");
}
}),
copy = new gui.MenuItem({
label: copyLabel || "Copy",
click: function() {
document.execCommand("copy");
}
}),
paste = new gui.MenuItem({
label: pasteLabel || "Paste",
click: function() {
document.execCommand("paste");
}
});
menu.append(cut);
menu.append(copy);
menu.append(paste);
return menu;
}

nw.gui 是 node-webkit 提供的一个包,它可以访问整个应用程序级别的对象,比如 window,application 等,详细的文档在这里 [bd1]

我们可以通过这个包来创建一些 MenuItem,并且在这些 Item 上绑定事件。最后,当鼠标右键时,显示这些菜单即可:

复制代码
var menu = new Menu();
$(document).on("contextmenu", function(e) {
e.preventDefault();
menu.popup(e.originalEvent.x, e.originalEvent.y);
});

完整的代码请参看 github 上的 repo

作者简介

邱俊涛,ThoughtWorks 咨询师,喜欢技术,崇尚轻量级的开发 / 工作方式,痛恨冗长的会议和各种繁琐的流程。他还是《JavaScript 核心概念及实践》一书的作者,个人博客是 http://icodeit.org ,博客上经常会有各种技术的分享。


感谢张凯峰对本文的策划,丁晓昀对本文的审校。

给InfoQ 中文站投稿或者参与内容翻译工作,请邮件至 editors@cn.infoq.com 。也欢迎大家通过新浪微博( @InfoQ @丁晓昀),微信(微信号: InfoQChina )关注我们,并与我们的编辑和其他读者朋友交流。

2015-05-19 10:054570

评论

发布
暂无评论
发现更多内容

等保服务费包括哪些?谁来付?企业吗?

行云管家

等保 等保测评 等保服务

响应“人工智能+”,枫清科技行业智能体赋能政企AI规模化应用

Fabarta

人工智能 智能体

监控系统如何选型:Zabbix vs Prometheus

巴辉特

Prometheus zabbix

INTERSPEECH 2025 | 数据堂诚邀您参加MLC-SLM挑战赛暨研讨会

数据堂

人工智能 Interspeech2025 语音研究

昆仑万维「Skywork UniPic 2.0」开源,统一多模态模型再迎新突破

新消费日报

说说内存泄漏的常见场景和排查方案?

王磊

鸿蒙NEXT开发中使用星闪服务

最新动态

计算机视觉前沿:自监督学习与高效学习挑战

qife122

计算机视觉 自监督学习

SeaTunnel MCP Server 入选《中国信通院开源商业产品及企业典型案例集(2025)》

白鲸开源

AI 开源商业化 Apache SeaTunnel 白鲸开源 MCP Server

分享一个人开发一套MES系统的历程——智能制造实践者

万界星空科技

数字化 制造业 mes 生产管理软件

SSL证书安装不成功是怎么回事?有哪些原因?

防火墙后吃泡面

硬件成本降52%,快钱支付引入OceanBase后的降本增效

老纪的技术唠嗑局

数据库设计 OceanBase 开源

用好CAD图层特性管理器,绘图更高效!

在路上

cad cad看图 CAD看图王

Apache DolphinScheduler 7 月社区月报 | 关键修复与性能优化全面推进

白鲸开源

Java 大数据 开源 大数据任务调度 Apache DolphinScheduler

混合云是什么意思?有哪些利弊?

行云管家

云计算 数字化 云服务 混合云

CAD中如何修改图层的颜色、线型和线宽?

在路上

cad CAD看图王

鸿蒙 NEXT 开发之后台任务开发服务框架学习笔记

最新动态

全球社交媒体监控如何帮助政府把握国际舆论脉搏?

沃观Wovision

海外舆情监控 沃观Wovision 舆情监测系统

基于视觉推理的Img2LaTeX转换技术突破

qife122

计算机视觉 视觉语言模型

Qwen3-Coder-480B-A35B-Instruct发布,可能是目前最好的编程模型

qife122

人工智能 开源

从东南亚出发:小程序容器技术如何助力 App 快速打入全球市场?

Byte_Me

小程序 App 出海

GreatSQL备份报错"PROCESS权限不足"分析与解决

GreatSQL

Apache RocketMQ EventBridge:为什么 GenAI 需要 EDA?

阿里巴巴云原生

阿里云 云原生 Apache RocketMQ EventBridge

FFmpeg 8.0 将集成 Whisper,支持实时字幕和转录;DeepMind 生物声学模型从鸟类拓展到哺乳昆虫和两栖丨日报

声网

从“字”到“画”:基于Elasticsearch Serverless 的多模态商品搜索实践

阿里云大数据AI技术

大数据 elasticsearch Serverless 搜索 向量检索

腾讯云EdgeOne:个人用户的加速与安全利器,免费开启一站式边缘优化

穿过生命散发芬芳

EdgeOne

海外品牌声誉管理新思路:社交媒体监控的精准化转型

沃观Wovision

社交媒体 海外舆情监控 沃观Wovision 舆情监测系统

搭建人文社科学生与AI时代的实质性连接,南京财经大学的这门“新文科”课程这样做

ModelWhale

人工智能 南京财经大学 文科+AI 文科

邀您参与 “直通乌镇” Spring AI Alibaba 开源竞技挑战赛!

阿里巴巴云原生

阿里云 云原生 Spring AI Alibabab

使用node-webkit构建桌面应用程序(二)_架构/框架_邱俊涛_InfoQ精选文章