| 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 | |
| }) | |
| } |