import type { HTMLNode, CommentOrTextAST, ElementAST, AST } from './types';
export const splitHead = (str: string, sep: string) => {
const idx = str.indexOf(sep);
if (idx === -1) return [str];
return [str.slice(0, idx), str.slice(idx + sep.length)];
};
const unquote = (str: string) => {
const car = str.charAt(0);
const end = str.length - 1;
const isQuoteStart = car === '"' || car === "'";
if (isQuoteStart && car === str.charAt(end)) {
return str.slice(1, end);
}
return str;
};
const formatAttributes = (attributes: string[]) => {
return attributes.map((attribute) => {
const parts = splitHead(attribute.trim(), '=');
const key = parts[0];
const value = typeof parts[1] === 'string' ? unquote(parts[1]) : null;
return { key, value };
});
};
export const format = (nodes: HTMLNode[]): AST[] => {
return nodes.map((node) => {
if (node.type === 'element') {
const children = format(node.children);
const item: ElementAST = {
type: 'element',
tagName: node.tagName.toLowerCase(),
attributes: formatAttributes(node.attributes),
children,
};
return item;
}
const item: CommentOrTextAST = {
type: node.type,
content: node.content,
};
return item;
});
};