Javascript 之所以为 Javascript,很重要的原因是浏览器为其实现了 DOM 和 BOM 接口(参考)。但在 Node.js 环境下,并没有 DOM 和 BOM,所以 html 文档与其他类型的文档并无不同,很难进行处理。而前端大量的 js 类库在 Node.js 环境下也毫无用武之地。这篇文档就来探究一下,如何在 Node.js 端进行 DOM 操作。
使用jsdom
虽然 Node.js 没有 DOM 支持,但我们可以自己构建一个 DOM,而 jsdom
这个库就是用来帮助我们完成这个任务的,它使用 javascript 构建了符合 W3C 标准的 DOM。
安装
jsnode 本身是 javascript 的,但它有一个依赖项 Contextify
是用 C++ 编写的。这个依赖项的功能是处理页面内的 script
标签中的 js 代码,将 javascript 在一个独立的模拟浏览器环境下运行。所以这个依赖项提供了一个非常重要的特性。从 Node.js 中加载一个基于 C++ 的模块,可能在 windows 下出现一些问题,如果在 windows 下安装不成功,请尝试安装32位版的 Node.js 再试。另外 Node.js 加载本地模块还需要 Python 的支持。
-
Windows
- 请确认安装了 visual studio;
- 安装 Python 2.7(点击下载),并把Python安装路径添加到系统变量;
- 安装最新版的 Node.js,最好是x86版本的(我用x64版本测试成功);
- 使用
npm install npm
更新npm到最新版本; - 使用
npm install jsdom
安装jsdom。
-
Mac OS
- 请确认安装了 Xcode;
- 需要安装 “Command line tools for XCode”;
- 启动过 Xcode,并同意授权条款;
- 使用
npm install jsdom
安装jsdom。
-
Linux(文档不详,未验证)
- 请确认安装了 make 等编译工具;
- 使用
npm install jsdom
安装jsdom; - 使用 make 编译 Contextify。
简单模式
jsdom 提供了 env()
方法可以很轻松地使用 jsdom 的强大功能。其定义如下:
jsdom.env(string, [scripts], [config], callback);
- 参数
string
:可以是一个 url、文件名或 HTML 片断。 - 参数
script
:类型为 string 或 string 数组,表示 javascript 文件的路径或 url,这些 javascript 文件最后会通过script
标签引用,并附加到 body 末尾。 - 参数
config
:类似于 jquery 的一些函数参数列表的定义思路,env()
允许不使用 string、script、callback 等参数,而直接传入一个 config 对象包含各种配置信息,当然,config 中也可以定义一些其他的配置选项。 - 参数
callback
:回调方法,callback 接受两个参数,error
和window
。- 参数
error
:可能是一个 object,为初始化 window 时抛出的错误;也可能是一个数组,为DOM收集到的脚本错误。 - 参数
window
:生成的一个全新的 window 对象。
- 参数
喜闻乐见的示例1:使用 url
const jsdom = require('jsdom');
jsdom.env(
'http://nodejs.org/dist/',
['http://code.jquery.com/jquery.js'],
(errors, window) => {
console.log('there have been', window.$('a').length, 'nodejs releases!');
}
);
此示例直接把要加载的 html 的 url 传入进行处理。注意传入的 jquery 完全是前端使用的js类型,不需要额外对引入的类库做任何处理即可在 Node.js 端使用它们进行文档处理。不过与前端代码稍有不同的是我们要留意此时 window 域内的方法对象并不能像在前端代码中那样直接获取,例如,我们并不能直接使用 $
,而是需要使用 window.$
,如果为了简便使用,可以定义好 var $=windwo.$
。
喜闻乐见的示例2:使用 html 片断
const jsdom = require('jsdom');
jsdom.env(
'<p><a class="the-link" href="https://github.com/tmpvar/jsdom">jsdom\'s Homepage</a></p>',
['http://code.jquery.com/jquery.js'],
(errors, window) => {
console.log('contents of a.the-link:', window.$('a.the-link').text());
}
);
类似的,我们可以通过 Node.js 的 fs
模块读取本地的 view,再传入 env()
作处理。
喜闻乐见的示例3:使用 config
对象
const fs = require('node:fs');
const jsdom = require('jsdom');
const jquery = fs.readFileSync('./jquery.js', 'utf-8');
jsdom.env({
done(errors, window) {
const $ = window.$;
console.log('HN Links');
$('td.title:not(:last) a').each(function () {
console.log(' -', $(this).text());
});
},
src: [jquery],
url: 'http://news.ycombinator.com/'
});
这个示例展示了通过 config
对象使用 env()
方法,这段代码同时也说明了 javascript 代码也是可以直接读取本地文件并插入到 html 内的。
高级模式
上面的用法应该已经能满足一定的需求,但它的不足在于不支持 html 代码中定义的外部 script、img 等资源。还有,在简单模式下,一次只能有一个 window 对象存在,不能同时处理多个文档。如果有这类的需要,我们可以使用高级模式。jsdom 使用的高级模式的方法是 jsdom()
。请参见如下代码:
const jsdom = require('jsdom').jsdom;
const doc = jsdom(markup, level, options);
const window = doc.parentWindow;
与简单模式不同的是,它不是采用异步回调的方式,而是采用同步的方式,直接将 window 对象暴露出来,这意味着在同一段代码域中,可以有多个完全隔离的 window 对象,可以同时处理多个文档。第二行向 jsdom()
方法传入了三个参数:
- 参数
markup
:markup 表示是的传入的 HTML/XML 文档(文档片断或空字符串也支持),我们也可以传入 null 或 undefined 来生成空白的 html 文档。 - 参数
level
:指定 DOM 级别,默认是 DOM 3级。 - 参数
option
:option 对象与简单模式下的 option 参数基本相同,不过它拥有一个features
选项控制着对外部资源的加载情况,具体如下:FetchExternalResources
- 默认值:
["script"]
- 允许值:
["script", "img", "css", "frame", "iframe", "link"]
或false
——是否允许从文件系统或网络获取文件,以及什么类型的文件。
- 默认值:
ProcessExternalResources
- 默认值:
["script"]
- 允许值:
["script"]
或false
——是否允许执行外部脚本。
- 默认值:
SkipExternalResources
- 默认值:
false
- 允许值:
/url/
或false
——指定不下载、不执行的资源 URL 的正则。
- 默认值:
Canvas支持
jsdom 本身不对 canvas 支持,会把它当成 div 处理,如果需要支持 canvas,则需要添加 canvas
包,安装方法(点击查看详情):
npm install canvas
注意事项
-
区分两种 window 使用 jsdom 确实能模拟出一个 window,让我们能在 Node.js 端书写前端代码,但一定要区分这个 window 与浏览器运行时的 window 对象不是一回事。在浏览器渲染页面时,浏览器找不到我们定义在 Node.js 端 window 上的函数、变量。在 Node.js 端进行的一些事件绑定也只会绑到模拟出来的 dom 上,浏览器呈现的时候并不会具有这些事件的绑定。
-
script 参数指定的 javascript 永远在 body 最末 虽然没有看到具体文档说明,但在我测试下来呈现这样的情况,因此,如果有多个 javascript、多种引入方式,特别是在 Node.js 端向 html 插入 script 节点时,特别要注意先后依赖关系,建议不要通过 script 参数传入 js 文件,而是直接以 script 标签写在 html 中。
-
jsdom 似乎不支持 ajax 没有在 jsdom 中看到有关 xmlHttpRequest 的实现,测试结果也是无法使用 ajax。可能需要针对 Node.js 重写 zepto 的 ajax 方法。