-
-
Notifications
You must be signed in to change notification settings - Fork 6.5k
/
DOMElement.ts
131 lines (112 loc) · 3.06 KB
/
DOMElement.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
/**
* Copyright (c) Meta Platforms, Inc. and affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
import type {Config, NewPlugin, Printer, Refs} from '../types';
import {
printChildren,
printComment,
printElement,
printElementAsLeaf,
printProps,
printText,
} from './lib/markup';
const ELEMENT_NODE = 1;
const TEXT_NODE = 3;
const COMMENT_NODE = 8;
const FRAGMENT_NODE = 11;
const ELEMENT_REGEXP = /^((HTML|SVG)\w*)?Element$/;
const testHasAttribute = (val: any) => {
try {
return typeof val.hasAttribute === 'function' && val.hasAttribute('is');
} catch {
return false;
}
};
const isCustomElement = (val: any) => {
const tagName = val?.tagName;
return (
(typeof tagName === 'string' && tagName.includes('-')) ||
testHasAttribute(val)
);
};
const testNode = (val: any) => {
const constructorName = val.constructor.name;
const {nodeType} = val;
return (
(nodeType === ELEMENT_NODE &&
(ELEMENT_REGEXP.test(constructorName) || isCustomElement(val))) ||
(nodeType === TEXT_NODE && constructorName === 'Text') ||
(nodeType === COMMENT_NODE && constructorName === 'Comment') ||
(nodeType === FRAGMENT_NODE && constructorName === 'DocumentFragment')
);
};
export const test: NewPlugin['test'] = (val: any) =>
(val?.constructor?.name || isCustomElement(val)) && testNode(val);
type HandledType = Element | Text | Comment | DocumentFragment;
function nodeIsText(node: HandledType): node is Text {
return node.nodeType === TEXT_NODE;
}
function nodeIsComment(node: HandledType): node is Comment {
return node.nodeType === COMMENT_NODE;
}
function nodeIsFragment(node: HandledType): node is DocumentFragment {
return node.nodeType === FRAGMENT_NODE;
}
export const serialize: NewPlugin['serialize'] = (
node: HandledType,
config: Config,
indentation: string,
depth: number,
refs: Refs,
printer: Printer,
) => {
if (nodeIsText(node)) {
return printText(node.data, config);
}
if (nodeIsComment(node)) {
return printComment(node.data, config);
}
const type = nodeIsFragment(node)
? 'DocumentFragment'
: node.tagName.toLowerCase();
if (++depth > config.maxDepth) {
return printElementAsLeaf(type, config);
}
return printElement(
type,
printProps(
nodeIsFragment(node)
? []
: Array.from(node.attributes, attr => attr.name).sort(),
nodeIsFragment(node)
? {}
: [...node.attributes].reduce<Record<string, string>>(
(props, attribute) => {
props[attribute.name] = attribute.value;
return props;
},
{},
),
config,
indentation + config.indent,
depth,
refs,
printer,
),
printChildren(
Array.prototype.slice.call(node.childNodes || node.children),
config,
indentation + config.indent,
depth,
refs,
printer,
),
config,
indentation,
);
};
const plugin: NewPlugin = {serialize, test};
export default plugin;