对 DOM 的操作
探索 HTMLElement 类型
自从标准化以来的 20 多年里,JavaScript 已经取得了长足的进步。尽管在 2020 年,JavaScript 可以在服务器、数据科学甚至物联网设备上使用,但重要的是要记住它最广泛的应用场景:web 浏览器。
网站由 HTML 和/或 XML 文档组成。这些文档是静态的,不会改变。*文档对象模型(DOM)*是浏览器实现的一个编程接口,用于使静态网站变得可操作。DOM API 可用于更改文档结构、样式和内容。该 API 十分强大,强大到无数前端框架(jQuery、React 以及 Angular 等)都是围绕它开发的,其使得动态网站的开发变得更加容易。
TypeScript 是 JavaScript 的一个类型超集,并且它提供了 DOM API 的类型定义。这些定义在任何默认的 TypeScript 项目中都可以使用。在 lib.dom.d.ts 文件的两万多行定义中,有一个脱颖而出:HTMLElement。这个类型是使用 TypeScript 进行 DOM 操作的关键。
你可以探索 DOM 类型定义的源代码
基本示例
给定一个简化的 index.html 文件:
html<!DOCTYPE html><html lang="zh-CN"><head><title>TypeScript DOM 操作</title></head><body><div id="app"></div><!-- 假设 index.js 是 index.ts 的编译输出 --><script src="index.js"></script></body></html>
让我们探讨一个使用 TypeScript 编写的脚本,该脚本将 <p>Hello, World!</p> 元素添加到了 #app 元素中。
ts// 1. 使用 id 属性选择 div 元素const app = document.getElementById("app");// 2.以编程方式创建一个新的 <p></p> 元素const p = document.createElement("p");// 3. 添加文本内容p.textContent = "Hello, World!";// 4. 将 p 元素附加到 div 元素中app?.appendChild(p);
编译并运行 index.html 页面后,生成的 HTML 将为:
html<div id="app"><p>Hello, World!</p></div>
Document 接口
TypeScript 代码的第一行使用了全局变量 document。检查这个变量显示它是由 lib.dom.d.ts 文件中的 Document 接口定义的。代码片段包含对两个方法的调用:getElementById 和 createElement。
Document.getElementById
该方法的定义如下:
tsgetElementById(elementId: string): HTMLElement | null;
向它传递一个元素 id 字符串,它将返回 HTMLElement 或 null。这个方法引入了最重要的类型之一:HTMLElement。它是每个其他元素接口的基础接口。例如,代码示例中的 p 变量的类型为 HTMLParagraphElement。另外,请注意这个方法可能返回 null。这是因为该方法在运行时无法确定是否能找到指定的元素。在代码片段的最后一行,我们使用了新的*可选链接(optional chaining)*运算符来调用 appendChild。
Document.createElement
这个方法的定义如下(我省略了已弃用的定义):
tscreateElement<K extends keyof HTMLElementTagNameMap>(tagName: K, options?: ElementCreationOptions): HTMLElementTagNameMap[K];createElement(tagName: string, options?: ElementCreationOptions): HTMLElement;
这是一个重载的函数定义。第二个重载是最简单的,与 getElementById 方法的工作方式很相似。向它传递任何 string,它将返回标准的 HTMLElement。这个定义使开发者能够创建独特的 HTML 元素标签。
例如,document.createElement('xyz') 返回一个 <xyz></xyz> 元素,显然这不是 HTML 规范中指定的元素。
对于感兴趣的人来说,你可以使用
document.getElementsByTagName与自定义标签元素进行交互。
对于 createElement 的第一个定义,它使用了一些高级的泛型模式。将其拆解成几小部分来分析会更容易理解,从泛型表达式开始: <K extends keyof HTMLElementTagNameMap>。这个表达式定义了一个泛型参数 K,它被限制为符合 HTMLElementTagNameMap 接口的键。这个映射接口包含了每个指定的 HTML 标签名及其对应的类型接口。例如,这里是前 5 个映射值:
tsinterface HTMLElementTagNameMap {"a": HTMLAnchorElement;"abbr": HTMLElement;"address": HTMLElement;"applet": HTMLAppletElement;"area": HTMLAreaElement;...}
一些元素并没有独特的属性,因此它们只返回 HTMLElement,但其他类型确实有独特的属性和方法,所以它们返回特定的接口(这些接口将继承或实现 HTMLElement)。
现在,我们探索一下 createElement 定义的其余部分:(tagName: K, options?: ElementCreationOptions): HTMLElementTagNameMap[K]。第一个参数 tagName 被定义为泛型参数 K。TypeScript 解释器足够智能,能够从这个参数推断出泛型参数。这意味着开发者在使用这个方法时不必指定泛型参数;传递给 tagName 参数的任何值都将被推断为 K,因此可以在定义的其余部分中使用。实际情况也正是如此;返回值 HTMLElementTagNameMap[K] 获取 tagName 参数,并使用它返回相应的类型。这个定义就是代码片段中 p 变量获得 HTMLParagraphElement 类型的方式。如果代码是 document.createElement('a'),那么它将是 HTMLAnchorElement 类型的元素。
Node 接口
document.getElementById 函数会返回 HTMLElement。HTMLElement 接口继承自 Element 接口,而 Element 又继承自 Node 接口。这种原型扩展允许所有 HTMLElements 使用一组标准方法的子集。在代码片段中,我们使用定义在 Node 接口上的属性来将新的 p 元素附加到网站上。
Node.appendChild
代码片段的最后一行是 app?.appendChild(p)。前面的 document.getElementById 部分详细介绍了使用可选链接运算符的原因,因为 app 在运行时可能为 null。appendChild 方法的定义如下:
tsappendChild<T extends Node>(newChild: T): T;
这个方法的工作方式与 createElement 方法类似,因为泛型参数 T 是从 newChild 参数推断出来的。T 被限制为符合另一个基础接口 Node。
children 和 childNodes 的区别
之前,本文详细介绍了 HTMLElement 接口继承自 Element,而 Element 又继承自 Node。在 DOM API 中,有一个称为*子级(children)*元素的概念。例如,在以下 HTML 中,p 标签是 div 元素的子级
tsx<div><p>Hello, World</p><p>TypeScript!</p></div>;const div = document.getElementsByTagName("div")[0];div.children;// HTMLCollection(2) [p, p]div.childNodes;// NodeList(2) [p, p]
在获取 div 元素后,children 属性将返回一个包含 HTMLParagraphElements 的 HTMLCollection 列表。childNodes 属性将返回一个类似的包含节点的 NodeList 列表。每个 p 标签仍会是 HTMLParagraphElements 类型,但 NodeList 可以包含 HTMLCollection 列表无法包含的其他 HTML节点。
修改 HTML 代码,删除其中一个 p 标签,但保留文本内容。
tsx<div><p>Hello, World</p>TypeScript!</div>;const div = document.getElementsByTagName("div")[0];div.children;// HTMLCollection(1) [p]div.childNodes;// NodeList(2) [p, text]
观察两个列表的变化。children 现在只包含 <p>Hello, World</p> 元素,而 childNodes 包含一个 text 节点,而不是两个 p 节点。NodeList 中的 text 部分是包含 TypeScript! 文本的 Node。children 列表中不包含这个 Node,因为它不被视为 HTMLElement。
querySelector 和 querySelectorAll 方法
这两个方法都是很好的工具,可以获取符合更多独特约束条件的 DOM 元素列表。它们在 lib.dom.d.ts 中定义如下:
ts/*** 返回 node 的后代中匹配选择器的第一个元素。*/querySelector<K extends keyof HTMLElementTagNameMap>(selectors: K): HTMLElementTagNameMap[K] | null;querySelector<K extends keyof SVGElementTagNameMap>(selectors: K): SVGElementTagNameMap[K] | null;querySelector<E extends Element = Element>(selectors: string): E | null;/*** 返回 node 的所有后代元素中匹配选择器的元素。*/querySelectorAll<K extends keyof HTMLElementTagNameMap>(selectors: K): NodeListOf<HTMLElementTagNameMap[K]>;querySelectorAll<K extends keyof SVGElementTagNameMap>(selectors: K): NodeListOf<SVGElementTagNameMap[K]>;querySelectorAll<E extends Element = Element>(selectors: string): NodeListOf<E>;
querySelectorAll 的定义类似于 getElementsByTagName,不同之处在于它返回一个新的类型:NodeListOf。这个返回类型本质上是标准 JavaScript 列表元素的自定义实现。可以说,将 NodeListOf<E> 替换为 E[] 将会得到非常相似的用户体验。NodeListOf 只实现了以下属性和方法:length、item(index)、forEach((value, key, parent) => void) 和数字索引。此外,这个方法返回元素列表,而不是从 .childNodes 方法返回的节点列表。虽然这可能看起来有些矛盾,但请注意 Element 接口继承自 Node 接口。
要查看这些方法的实际应用,我们可以修改现有的代码如下:
tsx<ul><li>第一个 :)</li><li>第二个!</li><li>第三个也不错。</li></ul>;const first = document.querySelector("li"); // 返回第一个 li 元素const all = document.querySelectorAll("li"); // 返回所有 li 元素的列表
想了解更多?
lib.dom.d.ts 类型定义的最大好处就是它们反映了 Mozilla Developer Network (MDN) 文档网站中的类型注释。例如,HTMLElement 接口在 MDN 上有专门的 HTMLElement 页面进行了记录。这些页面列出了所有可用的属性、方法,有时还会提供示例。这些页面的另一个优点是它们提供了指向相应标准文档的链接。这是 W3C 对 HTMLElement 的建议。
参考资料: