2022-11-01 22:01:28 +08:00
|
|
|
export const Fragment = "FRAGMENT";
|
|
|
|
|
|
|
|
export type Tag = {
|
|
|
|
name: string;
|
|
|
|
attrs?: Record<string, string | undefined>;
|
|
|
|
body: Tag[] | string;
|
|
|
|
} | string;
|
|
|
|
|
|
|
|
function htmlEscape(s: string): string {
|
2024-03-27 04:04:34 +08:00
|
|
|
if (typeof s !== "string") {
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2023-10-29 19:10:30 +08:00
|
|
|
s = s.replace(/&/g, "&")
|
2022-11-01 22:01:28 +08:00
|
|
|
.replace(/</g, "<")
|
|
|
|
.replace(/>/g, ">")
|
2023-10-29 19:10:30 +08:00
|
|
|
.replace(/"/g, """)
|
|
|
|
.replace(/\n/g, "<br>");
|
|
|
|
|
|
|
|
let oldS = s;
|
|
|
|
do {
|
|
|
|
oldS = s;
|
2024-07-30 21:17:34 +08:00
|
|
|
s = s.replace(/ {2}/g, " ");
|
2023-10-29 19:10:30 +08:00
|
|
|
} while (s !== oldS);
|
|
|
|
return s;
|
2022-11-01 22:01:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export function renderHtml(t: Tag | null): string {
|
|
|
|
if (!t) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
if (typeof t === "string") {
|
|
|
|
return htmlEscape(t);
|
|
|
|
}
|
|
|
|
const attrs = t.attrs
|
|
|
|
? " " + Object.entries(t.attrs)
|
|
|
|
.filter(([, value]) => value !== undefined)
|
|
|
|
.map(([k, v]) => `${k}="${htmlEscape(v!)}"`).join(
|
|
|
|
" ",
|
|
|
|
)
|
|
|
|
: "";
|
|
|
|
const body = typeof t.body === "string"
|
|
|
|
? htmlEscape(t.body)
|
|
|
|
: t.body.map(renderHtml).join("");
|
|
|
|
if (t.name === Fragment) {
|
|
|
|
return body;
|
|
|
|
}
|
2023-10-29 19:10:30 +08:00
|
|
|
return `<${t.name}${attrs}>${body}</${t.name}>`;
|
2022-11-01 22:01:28 +08:00
|
|
|
}
|