描述
开 本: 16开纸 张: 胶版纸包 装: 平装-胶订是否套装: 否国际标准书号ISBN: 9787121334740
第1部分 欢迎来到.Node.js.桌面应用开发的世界
第1章 Electron和NW.js入门 .3
1.1 为什么要用.Node.js.构建桌面应用.4
1.1.1 桌面应用到.Web.应用,再回到桌面应用 .4
1.1.2 Node.js.桌面应用相比.Web.应用有什么优势 .6
1.2 NW.js.和.Electron.的起源 .8
1.3 NW.js.介绍 .9
1.3.1 使用.NW.js.构建.Hello.World.应用 .10
1.3.2 NW.js.有哪些特性.15
1.4 Electron.介绍 .18
1.4.1 Electron.是如何工作的以及它和.NW.js.的区别是什么 .19
1.4.2 使用.Electron.开发.Hello.World.应用 .19
1.4.3 Electron.有哪些特性.25
1.5 NW.js.和.Electron.支持创建哪类应用 .25
1.5.1 Slack .26
1.5.2 Light.Table .26
1.5.3 Game.Dev.Tycoon .27
1.5.4 Gitter .28
1.5.5 Macaw .29
1.5.6 Hyper .30
1.6 小结 .31
第2章 为你的首款桌面应用搭建基础架构.32
2.1 我们将构建什么应用 .33
2.2 创建应用 .34
2.2.1 安装.NW.js.和.Electron.34
2.2.2 为.NW.js.版本的应用创建文件和文件夹 .35
2.2.3 为.Electron.版本的应用创建文件和文件夹 .37
2.3 实现启动界面 .39
2.3.1 在工具条中展示用户个人文件夹信息 .40
2.3.2 显示用户个人文件夹中的文件和文件夹 .44
2.4 小结 .54
第3章 构建你的首款桌面应用 .56
3.1 浏览文件夹 .57
3.1.1 重构代码 .57
3.1.2 处理对文件夹的双击操作.61
3.2 实现快速搜索 .64
3.2.1 在工具条中增加搜索框.65
3.2.2 引入一个内存搜索库.65
3.2.3 在界面上触发搜索功能.67
3.3 改进应用内的导航功能 .71
3.3.1 实现当前文件夹路径可单击 .71
3.3.2 让应用随着文件夹路径的改变显示对应的文件夹内容 .74
3.3.3 实现使用默认应用打开对应的文件 .75
3.4 小结 .77
第4章 分发你的首款桌面应用 .79
4.1 对应用进行与分发相关的设置.80
4.2 对要分发的应用进行打包.83
4.2.1 使用一种.NW.js.的构建工具 .83
4.2.2 使用一种.Electron的构建工具 .84
4.2.3 设置应用的图标 .85
4.3 在多个操作系统中测试应用.91
4.3.1 Windows.操作系统.91
4.3.2 Linux.操作系统 .92
4.3.3 Mac.OS.系统 .92
4.4 小结 .92
第2部分 深度剖析
第5章 在NW.js和Electron中使用Node.js.97
5.1 什么是.Node.js .98
5.1.1 同步与异步 .98
5.1.2 流是一等公民 .101
5.1.3 事件 .105
5.1.4 模块 .106
5.2 Node.包管理器 .109
5.2.1 寻找应用需要的模块.109
5.2.2 使用.package.json记录安装的模块 .109
5.2.3 使用.npm.打包模块和应用.111
5.3 小结 .114
第6章 探索NW.js和Electron的内部机制.115
6.1 NW.js.内部是如何工作的 .116
6.1.1 使用同一个.V8.实例.117
6.1.2 集成主事件循环 .118
6.1.3 桥接.Node.js.和.Chromium.的.JavaScript.上下文 .119
6.2 Electron.内部是如何工作的.119
6.2.1 libchromiumcontent.介绍.120
6.2.2 Electron.中的组件.120
6.2.3 Electron.是如何将应用运行起来的 .121
6.3 Node.js是如何与NW.js以及Electron一起工作的 .122
6.3.1 Node.js.集成在.NW.js.的哪个位置 .122
6.3.2 在.NW.js中使用.Node.js.的缺点 .123
6.3.3 Electron.是怎么使用.Node.js.的 .123
6.4 小结 .124
第3部分 精通Node.js桌面应用开发
第7章 自定义桌面应用的外观.127
7.1 视窗的尺寸和模式 .127
7.1.1 配置.NW.js.应用的视窗尺寸 .128
7.1.2 配置.Electron.应用的视窗尺寸 .129
7.1.3 在.NW.js.中限制视窗的尺寸 .131
7.1.4 在.Electron.中限制视窗的尺寸 .133
7.2 无边框应用以及全屏应用.134
7.2.1 NW.js.中的全屏应用.135
7.2.2 Electron.中的全屏应用.138
7.2.3 无边框应用 .140
7.2.4 kiosk.应用 .145
7.3 小结 .149
第8章 创建托盘应用 .150
8.1 使用.NW.js.创建简单的托盘应用.151
8.2 使用.Electron.创建托盘应用.156
8.3 小结 .159
第9章 创建应用菜单以及上下文菜单.161
9.1 为应用添加菜单 .162
9.1.1 应用视窗菜单 .162
9.1.2 使用.NW.js.为.Mac.OS的应用创建菜单 .162
9.1.3 使用.Electron.为.Mac.OS的应用创建菜单 .163
9.1.4 为.Windows.和.Linux的应用创建菜单 .166
9.1.5 基于操作系统来选择渲染具体的菜单 .173
9.2 上下文菜单 .174
9.2.1 使用.NW.js.创建上下文菜单 .174
9.2.2 NW.js.中的上下文菜单是如何工作的 .179
9.2.3 设置菜单项图标 .180
9.2.4 使用.Electron.创建上下文菜单 .181
9.2.5 使用.Electron.添加上下文菜单 .184
9.3 小结 .185
第10章 拖曳文件以及定制界面.186
10.1 在应用中拖曳文件 .186
10.1.1 使用.NW.js.实现在应用中拖曳文件 .187
10.1.2 使用.Electron.实现拖曳功能 .190
10.2 模拟操作系统原生样式.191
10.2.1 检测用户的操作系统.191
10.2.2 使用.NW.js检测操作系统 .191
10.2.3 使用.Electron检测操作系统 .192
10.2.4 使用.CSS匹配用户操作系统的样式 .194
10.3 小结 .197
第11章 在应用中使用网络摄像头.198
11.1 使用.HTML5.媒体捕捉.API.来实现相片快照 .198
11.1.1 解读.NW.js.版的应用.199
11.1.2 使用.Electron.构建.Facebomb.应用 .205
11.2 小结 .210
第12章 存储应用数据 .211
12.1 应该使用哪种数据存储方案.211
12.2 使用.localStorage.API.存储便笺数据 .212
12.2.1 使用.Electron开发.Let.Me.Remember应用 .213
12.2.2 使用.NW.js开发.Let.Me.Remember应用 .216
12.3 将待办事项应用移植为桌面应用 .219
12.3.1 使用.NW.js.移植.TodoMVC.Web.应用 .219
12.3.2 使用.Electron.移植.TodoMVC.应用 .220
12.4 小结 .222
第13章 从剪贴板复制和粘贴数据.223
13.1 访问剪贴板数据 .223
13.1.1 使用.NW.js.创建.Pearls.应用 .224
13.1.2 使用.Electron.创建.Pearls.应用 .228
13.1.3 使用.Electron.将不同类型的数据写入剪贴板 .231
13.2 小结 .232
第14章 绑定键盘快捷键 .233
14.1 使用.NW.js.创建贪吃蛇游戏.234
14.1.1 使用.NW.js.在视窗获取焦点的时候实现键盘快捷键 .242
14.1.2 使用.NW.js.来创建全局键盘快捷键 .243
14.2 使用.Electron.为贪吃蛇游戏创建全局快捷键 .245
14.3 小结 .247
第15章 制作桌面通知 .248
15.1 关于你要构建的应用 .249
15.2 使用.Electron.构建.Watchy.应用.249
15.3 使用.NW.js.构建.Watchy.应用.254
15.4 小结 .257
第4部分 准备发布
第16章 测试桌面应用 .261
16.1 测试应用的不同方法 .262
16.1.1 测试驱动开发 .262
16.1.2 行为驱动开发 .264
16.1.3 不同层面的测试.265
16.2 单元测试 .265
16.2.1 使用.Mocha.编写测试.266
16.2.2 让待完成的测试变成执行通过的测试 .268
16.3 功能测试 .271
16.3.1 功能测试实践 .272
16.3.2 使用.NW.js.和.ChromeDriver.进行测试 .272
16.4 使用Spectron测试Electron应用.273
16.5 集成测试 .275
16.5.1 Cucumber.介绍.276
16.5.2 使用.Cucumber和.Spectron对.Electron应用进行自动化测试.277
16.6 小结 .280
第17章 调试并提升应用性能 .281
17.1 了解你要调试的是什么.282
17.1.1 确定问题根本原因的位置 .283
17.1.2 使用浏览器开发者工具进行调试 .284
17.2 修复.bug .287
17.2.1 使用.Node.js.的调试器来调试应用 .288
17.2.2 使用.NW.js.的开发者工具来调试应用 .291
17.3 解决性能问题 .296
17.3.1 Network选项卡.296
17.3.2 Timeline选项卡.297
17.3.3 Pro.les选项卡.299
17.4 调试.Electron.应用 .301
17.5 小结 .307
第18章 为多平台打包应用 .308
18.1 为应用创建可执行文件.309
18.1.1 为.Windows.系统创建.NW.js.应用的可执行文件 .309
18.1.2 安装虚拟机 .309
18.1.3 为一个.NW.js应用创建针对.Windows系统的.exe文件.310
18.1.4 为一个.Electron.应用创建.Windows.系统的可执行文件 .311
18.2 为.Windows.的应用创建启动安装器 .314
18.2.1 使用.NW.js.创建.Windows.系统启动安装器 .314
18.2.2 使用.Electron创建.Windows系统启动安装器 .321
18.3 为.Mac.OS.创建.NW.js.应用的可执行文件 .324
18.3.1 创建.Mac.可执行应用.324
18.3.2 为.Mac.OS.创建.Electron.应用的可执行文件 .327
18.4 为.Linux.创建可执行应用.329
18.4.1 为.Linux.创建独立的.NW.js.应用文件 .330
18.4.2 为.Linux.创建独立的.Electron.应用文件 .331
18.5 小结 .333
附录A 安装.Node.js .335
译者序
Stack Over.ow 的联合创始人 Jeff Atwood 说过一句非常经典的话: Any application that can be written in JavaScript, will eventually be written in JavaScript,翻译过来就是:任何能使用 JavaScript 来编写的应用,最终都会用 JavaScript 来实现。这句话被誉为 Atwood 定律。事实上,这句话正在不同领域被一次一次地验证。以前 JavaScript 只是运行在浏览器沙箱环境中的脚本语言,而自从 2009 年 Node.js 问世后,JavaScript 在服务器端、物联网领域、移动原生应用开发领域,乃至桌面应用开发领域都大放异彩。
以往要开发桌面应用,针对 Windows、Linux 以及 Mac OS 三大平台要专门去学习各自平台的编程语言和框架,成本高昂而且要做一款支持兼容三种平台的桌面应用非常费时,基本都需要针对不同平台的不同团队才能实现。就我个人而言,几年前我一直想学习 Objective-C 以及 Cocoa 来开发 Mac OS X 桌面应用,但是始终没有成功。现如今, JavaScript 让这一切都变得无比简单。一名 Web 开发者就能开发出兼容三大操作系统的桌面应用。不仅大大降低了学习曲线,而且开发效率可以说呈指数级提升。这要归功于 NW.js 和 Electron 这两款目前最流行的使用 Web 技术开发桌面应用的开发框架。这两款框架将 Chromium 和 Node.js 非常好地结合起来, Chromium使得 Web 开发技术能够在桌面应用中得以施展, Node.js则提供了访问操作系统 API 的能力,两者的结合使得使用 JavaScript 开发桌面应用成为可能。
目前, NW.js 和 Electron 这两款框架在全世界各大公司被广泛使用。近几年红遍全球的 Slack就是使用 Electron 来开发他们的桌面应用的,国内阿里巴巴的企业应用—钉钉桌面应用,就是用 NW.js 来开发的,除此之外,全球范围内越来越多的桌面应用都在采用这两种框架进行开发。
本书是一本专门介绍如何使用 NW.js 和 Electron 框架来开发桌面应用的书。在国内,目前本书应该是第一本同时介绍 NW.js 和 Electron 开发桌面应用的图书。而且本书内容非常系统,从框架的背景介绍、教你开发第一款桌面应用、深入剖析框架内部原理、通过丰富的示例应用介绍框架提供的多个 API,再到应用的测试、调试、跨平台打包、构建和最终的发布,涵盖整个开发到发布流程中的所有环节。而且本书的每一章中都有大量的实用示例,通过实际的编码让你感受使用 NW.js 的 Electron 开发桌面应用的体验。书中每个示例应用都会分别介绍 NW.js 和 Electron 两个版本如何实现、过程中需要注意的地方有哪些,非常有实践价值。总的来说,本书不论是对于初学者还是有一定经验的开发者,都是一本相当好的学习使用 NW.js 和 Electron 开发桌面应用的图书。
最后,非常感谢电子工业出版社计算机出版分社的张春雨编辑对我的信任,将这本书交给我来翻译;感谢本书的责任编辑刘舫对本书的辛苦付出;还要感谢本书的原作者 Paul B. Jensen,翻译过程中遇到模棱两可的地方,通过 Twitter 联系他,他都能及时回复我,并给予详细的解释。
翻译和写书一样,都是需要花费大量精力和时间的事情,自从翻译完上一本《了不起的 Node.js》后,我就对自己说我再也不会干翻译图书的事情了,实在是太累了。但是,当出版社编辑找到我,给我看了原版样书后,我还是没忍住,因为虽然过程很累很苦,但是在书出版的那一刻,除了自己小小的虚荣心能够得到一点满足,更多的是一想到可以帮助到很多学习使用 NW.js 和 Electron 开发桌面应用的开发者,就觉得非常自豪,再累再苦都是值得的!当然,翻译过程中难免会有错误的地方,也希望大家能够多多指正!
谨以此书献给在背后默默支持我的家人,特别是我的两个孩子 —木木和一一,希望你们能够健康快乐地成长,爸爸爱你们!
Goddy Zhao 2017 年 12 月 12 日于上海
推荐序
Electron 框架诞生于 2013 年,那个时候 Node.js 才刚刚流行起来。整个社区因 JavaScript 能够在客户端和服务器端运行而兴奋不已,并且也在尝试使用 JavaScript 来开发桌面应用。
我个人也对 JavaScript 技术很热衷,而且 GUI 编程是我比较喜欢的领域。我自己写过一些 Node.js 的模块,这些模块对主流的 GUI 工具提供了 JavaScript 的绑定,不过都做得一般,也没有引起太多关注。
之后我发现了一个非常有趣的 Node.js模块,叫作 node-webkit:这个简单的模块可以实现在 WebKit 浏览器中插入执行 Node.js 代码。于是我有了一个点子,可以用它来开发一个具备完整功能的客户端开发框架:我可以用 Chromium 来显示 Web 页面,就像桌面窗口一样,然后其他的都用 Node.js 来控制!
当时 node-webkit 的开发并不活跃,于是我接手了这个项目并进行重写,将它打造成一个完善的用于桌面应用开发的框架。当我完成第一版的时候,它可以用于开发小型的跨平台应用,效果奇好!
与此同时, GitHub正在秘密开发一款基于 Web技术的 Atom编辑器,而且他们非常希望可以有一个更好的工具来替代目前 Atom不尽如人意的 Web运行时。 GitHub曾尝试将 Atom迁移到 node-webkit,但是遇到了很多问题。我和他们的开发者碰了面并且最终我们达成一致:由我来开发一款新的框架,让开发者使用 Node.js
技术和浏览器相关技术就可以开发桌面程序,然后再帮他们把 Atom迁移过来。
这款新的框架起初命名为 atom-shell;一年后,在正式开源的时候将其改名为 Electron。Electron是从零开始开发的,并且使用了和 node-webkit完全不同的底层架构,它可以让开发者开发大型且复杂的桌面端应用。(如今, node-webkit交由其他开发者在维护开发,项目状态也比较活跃。它现在叫 NW.js,使用也很广泛。)
因为使用 Electron可以既简单又快速地构建出复杂的跨平台应用,所以它得到了许多开发者的关注,发展也很迅速。现如今,许多大公司都基于 Electron 开发了他们的桌面端产品,除此之外,小型创业公司也围绕这个平台在构建他们的业务。
使用 Electron 和 NW.js 开发桌面应用要求开发者掌握一些新的概念。桌面应用开发和前端程序开发截然不同,对于初学者来说也更难。不过本书可以帮助到大家。
本书将带你一览 Electron 和 NW.js 丰富的 API、教你如何开发桌面应用。你会学到许多使用 JavaScript 开发桌面应用的技术细节,包括如何构建和分发应用,以及如何将现有应用集成到桌面应用中的一些深度小技巧。本书还涵盖了一些高级话题,如调试、分析以及在不同平台发布应用,哪怕是有经验的开发者也可以从中学到不少东西。
我建议所有想要开发桌面应用的读者都来阅读本书。读完后你会惊讶于使用 JavaScript 和 Web 技术来进行跨平台的桌面应用开发是一件多么简单的事情。
Cheng Zhao Electron框架开发者
序言
几年前我在一家叫 Axisto Media 的公司工作时,我们需要为一个健康行业大会开发一款桌面应用,用来展示大会的视频、议题信息以及海报。当时这款应用是用 Adobe AIR 开发的。但是开发过程并不容易,而且客户需要进行一些操作才能让应用在他们的计算机中运行起来。好在我们后来找到了更好的解决方案。
我大概从 2013 年年底开始学习 NW.js(那个时候它叫 node-webkit)。我发现使用 NW.js 开发的桌面应用客户用起来更方便,因为他们不再需要安装 Adobe Flash 播放器,也不用把应用文件放到 U 盘里来加载。他们只需双击应用就可以运行了。不仅如此,我们还能提供 Linux版本,而且其技术栈和我们的业务本身的技术栈很契合,因为我们在其他地方也都使用 Node.js 技术。
我抓住了机会,使用 NW.js 去重新构建这款桌面应用并且摒弃一切,勇往直前。 NW.js 让一切都变得更加简单,这得益于它可以从大会网站的 Web 应用重用 HTML、 CSS和 JS代码,我们可以让桌面应用看上去样式更加统一。这是一个巨大的好处。
我当时对这个框架非常满意,于是决定在 2014 年 6 月的伦敦 Node.js 用户组聚会上进行分享。后来我就把演示稿放到了网上。几个月后,我发现这个演示稿在 SlideShare 网站上很快被查看了 20 000次。这太棒了,我以为这事就这样了。
然而并没有。
2014 年 12 月,我收到了一封来自 Manning 出版社 Erin Twohey 的电子邮件,他问我是否有兴趣写一本关于 node-webkit 的书。这简直太棒了,我立刻就投入到这本书的写作中。
那段时间发生了很多事情。Node.js 社区 fork 了 Node.js 项目并命名为 IO.js,他们加快了平台新特性的开发,后来 IO.js 项目又合并回了 Node.js项目。 node-webkit 框架切换到了 IO.js,并且由于它使用了 Blink 而非 WebKit,因此改名为 NW.js。一年过去了,本书的写作也临近尾声,就在这个时候,我们发现了另外一个可以用 Node.js 开发桌面应用的框架,叫 Electron。仔细一看,我发现 Electron 和 NW.js 很像,而且它的作者以前就是开发 NW.js 的。于是我们决定将 Electron 也写到本书中。
写一本书同时涵盖两种 Node.js 桌面应用开发框架是一个挑战,不过最终还是完成了。本书涵盖了使用 NW.js 和 Electron 开发桌面应用的基础知识。尽管本书没有面面俱到地介绍这两个框架,但是足够让你了解它们的大部分特性以及如何使用的知识,这样你就可以根据你的需求,选择其中一个框架来构建桌面应用。
对于开发者来说这是一个很好的时代,有了像 NW.js 和 Electron 这样的工具,构建桌面应用变得再简单不过。我希望你喜欢这本书,如果对于这两个框架有问题想问我的话,可以通过我的电子邮箱 [email protected] 或者通过我的 Twitter账号 @paulbjensen联系我。
Paul B. Jensen
致谢
写书是非常艰难的项目之一,它需要投入大量的时间和精力。同时,还需要不少人的协助。我要感谢的人很多,他们都或多或少帮助过我。首先我要感谢 Manning 出版社负责本书的团队: Erin Twohey、Ana Romac、 Candace Gillhoolhey、Rebecca Rinehart、Aleksandar Dragosavljevic、Toni Bowers、
ˊMehmed Pasic、Karen Gulliver、Katie Tennant、Janet Vail 以及 Lynn Beighley。促成本书的工作量之大是你难以想象的,在此过程中他们都极力帮助我完善本书。我还要感谢技术审校 Clive Harber 和以下这些审校人员: Angelo Costa、Daniel Baktiar、 Darko Bozhinovski、Deepak Karanth、Fernando Monteiro Kobayashi、Jeff Smith、 Matt Borack、Nicolas Boulet-Lavoie、Olivier Ducatteeuw、Patrick Regan、Patrick Rein、Robert Walsh、Rocio Chongtay、Stephen Byrne、Toni La.. hdekorpi、William Wheeler、Yogesh Poojari 以及 Marcelo Pires;同时感谢 Natko Stipanicˇ ev 在图片方面提供的帮助。
感谢 Marjan Bace 给我写作本书的机会。能为 Manning写书是一种荣誉;我的书架上有不少 Manning出版社的书,现在他们的书架上多了一本我的书。还要感谢 Michael Stephens 在写书之初帮助我制订了本书的大纲、感谢他在面对我各种拖稿的时候能够妥善处理、感谢他当我遇到个人困难的时候给予理解。
感谢我的开发编辑—Cynthia Kane。她完成了最困难的工作—激励我完成每一个章节的内容。这是我的第一本书,你可以想象这个过程有多么痛苦。我有一份电子邮件归档,里面包含超过 150封邮件,这些都是她在我写书阶段发给我的,那个时候我在伦敦、阿姆斯特丹、爱尔兰、意大利、纽约,然后又到阿姆斯特丹,最后又回到伦敦。在非常困难的 2016 年,Cynthia 始终耐心地激励我将本书完成,而且尽管有时区问题,她都能及时地提供支持和帮助。万分感激,谢谢 Cynthia。
感谢 Roger Wang和 Cheng Zhao开发了 NW.js 和 Electron—没有他们的努力,这本书压根就不可能存在。
感谢在伦敦 Starcount 的 Edwina Dunn 和 Clive Humby,很荣幸可以和他们共事,非常感激他们给予我的支持。
感谢 Purple Seven 的 Stuart Nicolle。Stuart 当年带我入职并带我领略了如何从艺术和戏剧分析世界中收集有用的信息。
感谢我的家人:我的母亲 Jette、妹妹 Maria 和她的伙伴 Mark,已故的 Gran Lis 以及 Brenda 和 Jim。他们抚育我成人、在我人生道路上为我披荆斩棘。
我还想特别感谢 Fiona。她容忍了我写书过程中的一切,甚至更多。本书能够成功出版和她对我的支持和爱是分不开的。
最后我还想提一下我的父亲 Wily,他是一名硬件和软件工程师,非常聪明却又不好相处。虽然我们从未正眼看过对方,但还是要感谢有这样一位父亲。
关于代码
本书包含诸多示例代码,有标明序号的多行代码,也有直接在正文中的单行代码。不论是哪种形式,代码都是以等宽字体的形式来表示的,以此来和正文进行区分。大多数情况下,源代码都是格式化过的;为了适应书页的空间添加了必要的换行和缩进。除此之外,当有专门解释源代码的文字时,代码注释通常就被去掉了。一般在多行代码以及高亮显示的重要概念时会有代码注解。
本书中的示例代码可以通过出版社网站 www.manning.com/books/cross-platform-desktop-applications或 GitHub的 http://github.com/paulbjensen/cross-platform-desktop-applications 下载得到。
作者在线
购买本书之后,读者可以免费访问 Manning 出版社的私有论坛,在那里,可以对本书进行评论、提问技术问题,作者和论坛中的其他用户都会给予解答。要访问和订阅该论坛,请访问 www.manning.com/books/cross-platform-desktop-applications。
读者服务
轻松注册成为博文视点社区用户(www.broadview.com.cn),扫码直达本书页面。
提交勘误:对书中内容的修改意见可在提交勘误处提交,若被采纳,将获赠博文视点社区积分(在你购买电子书时,积分可用来抵扣相应金额)。
与我们交流:在页面下方读者评论处留下疑问或观点,与我们和其他读者一同学习交流。
页面入口: http://www.broadview.com.cn/33474
你会被这么容易就能开发桌面应用而感到震惊的!
评论
还没有评论。