|
"use strict";
|
|
module.exports = {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
serializeOne: serializeOne,
|
|
|
|
|
|
|
|
|
|
ɵescapeMatchingClosingTag: escapeMatchingClosingTag,
|
|
ɵescapeClosingCommentTag: escapeClosingCommentTag,
|
|
ɵescapeProcessingInstructionContent: escapeProcessingInstructionContent
|
|
};
|
|
|
|
var utils = require('./utils');
|
|
var NAMESPACE = utils.NAMESPACE;
|
|
|
|
var hasRawContent = {
|
|
STYLE: true,
|
|
SCRIPT: true,
|
|
XMP: true,
|
|
IFRAME: true,
|
|
NOEMBED: true,
|
|
NOFRAMES: true,
|
|
PLAINTEXT: true
|
|
};
|
|
|
|
var emptyElements = {
|
|
area: true,
|
|
base: true,
|
|
basefont: true,
|
|
bgsound: true,
|
|
br: true,
|
|
col: true,
|
|
embed: true,
|
|
frame: true,
|
|
hr: true,
|
|
img: true,
|
|
input: true,
|
|
keygen: true,
|
|
link: true,
|
|
meta: true,
|
|
param: true,
|
|
source: true,
|
|
track: true,
|
|
wbr: true
|
|
};
|
|
|
|
var extraNewLine = {
|
|
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
const ESCAPE_REGEXP = /[&<>\u00A0]/g;
|
|
const ESCAPE_ATTR_REGEXP = /[&"<>\u00A0]/g;
|
|
|
|
function escape(s) {
|
|
if (!ESCAPE_REGEXP.test(s)) {
|
|
|
|
return s;
|
|
}
|
|
|
|
return s.replace(ESCAPE_REGEXP, (c) => {
|
|
switch (c) {
|
|
case "&":
|
|
return "&";
|
|
case "<":
|
|
return "<";
|
|
case ">":
|
|
return ">";
|
|
case "\u00A0":
|
|
return " ";
|
|
}
|
|
});
|
|
}
|
|
|
|
function escapeAttr(s) {
|
|
if (!ESCAPE_ATTR_REGEXP.test(s)) {
|
|
|
|
return s;
|
|
}
|
|
|
|
return s.replace(ESCAPE_ATTR_REGEXP, (c) => {
|
|
switch (c) {
|
|
case "<":
|
|
return "<";
|
|
case ">":
|
|
return ">";
|
|
case "&":
|
|
return "&";
|
|
case '"':
|
|
return """;
|
|
case "\u00A0":
|
|
return " ";
|
|
}
|
|
});
|
|
}
|
|
|
|
function attrname(a) {
|
|
var ns = a.namespaceURI;
|
|
if (!ns)
|
|
return a.localName;
|
|
if (ns === NAMESPACE.XML)
|
|
return 'xml:' + a.localName;
|
|
if (ns === NAMESPACE.XLINK)
|
|
return 'xlink:' + a.localName;
|
|
|
|
if (ns === NAMESPACE.XMLNS) {
|
|
if (a.localName === 'xmlns') return 'xmlns';
|
|
else return 'xmlns:' + a.localName;
|
|
}
|
|
return a.name;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function escapeMatchingClosingTag(rawText, parentTag) {
|
|
const parentClosingTag = '</' + parentTag;
|
|
if (!rawText.toLowerCase().includes(parentClosingTag)) {
|
|
return rawText;
|
|
}
|
|
const result = [...rawText];
|
|
const matches = rawText.matchAll(new RegExp(parentClosingTag, 'ig'));
|
|
for (const match of matches) {
|
|
result[match.index] = '<';
|
|
}
|
|
return result.join('');
|
|
}
|
|
|
|
const CLOSING_COMMENT_REGEXP = /--!?>/;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function escapeClosingCommentTag(rawContent) {
|
|
if (!CLOSING_COMMENT_REGEXP.test(rawContent)) {
|
|
return rawContent;
|
|
}
|
|
return rawContent.replace(/(--\!?)>/g, '$1>');
|
|
}
|
|
|
|
|
|
|
|
|
|
function escapeProcessingInstructionContent(rawContent) {
|
|
return rawContent.includes('>')
|
|
? rawContent.replaceAll('>', '>')
|
|
: rawContent;
|
|
}
|
|
|
|
function serializeOne(kid, parent) {
|
|
var s = '';
|
|
switch(kid.nodeType) {
|
|
case 1:
|
|
var ns = kid.namespaceURI;
|
|
var html = ns === NAMESPACE.HTML;
|
|
var tagname = (html || ns === NAMESPACE.SVG || ns === NAMESPACE.MATHML) ? kid.localName : kid.tagName;
|
|
|
|
s += '<' + tagname;
|
|
|
|
for(var j = 0, k = kid._numattrs; j < k; j++) {
|
|
var a = kid._attr(j);
|
|
s += ' ' + attrname(a);
|
|
if (a.value !== undefined) s += '="' + escapeAttr(a.value) + '"';
|
|
}
|
|
s += '>';
|
|
|
|
if (!(html && emptyElements[tagname])) {
|
|
var ss = kid.serialize();
|
|
|
|
|
|
if (hasRawContent[tagname.toUpperCase()]) {
|
|
ss = escapeMatchingClosingTag(ss, tagname);
|
|
}
|
|
if (html && extraNewLine[tagname] && ss.charAt(0)==='\n') s += '\n';
|
|
|
|
s += ss;
|
|
s += '</' + tagname + '>';
|
|
}
|
|
break;
|
|
case 3:
|
|
case 4:
|
|
var parenttag;
|
|
if (parent.nodeType === 1 &&
|
|
parent.namespaceURI === NAMESPACE.HTML)
|
|
parenttag = parent.tagName;
|
|
else
|
|
parenttag = '';
|
|
|
|
if (hasRawContent[parenttag] ||
|
|
(parenttag==='NOSCRIPT' && parent.ownerDocument._scripting_enabled)) {
|
|
s += kid.data;
|
|
} else {
|
|
s += escape(kid.data);
|
|
}
|
|
break;
|
|
case 8:
|
|
s += '<!--' + escapeClosingCommentTag(kid.data) + '-->';
|
|
break;
|
|
case 7:
|
|
const content = escapeProcessingInstructionContent(kid.data);
|
|
s += '<?' + kid.target + ' ' + content + '?>';
|
|
break;
|
|
case 10:
|
|
s += '<!DOCTYPE ' + kid.name;
|
|
|
|
if (false) {
|
|
|
|
if (kid.publicID) {
|
|
s += ' PUBLIC "' + kid.publicId + '"';
|
|
}
|
|
|
|
if (kid.systemId) {
|
|
s += ' "' + kid.systemId + '"';
|
|
}
|
|
}
|
|
|
|
s += '>';
|
|
break;
|
|
default:
|
|
utils.InvalidStateError();
|
|
}
|
|
return s;
|
|
}
|
|
|