| 'use strict'; |
| |
| /** |
| * @typedef {import('../lib/types').XastElement} XastElement |
| */ |
| |
| const { detachNodeFromParent } = require('../lib/xast.js'); |
| const { elemsGroups } = require('./_collections.js'); |
| |
| exports.name = 'removeUselessDefs'; |
| exports.description = 'removes elements in <defs> without id'; |
| |
| /** |
| * Removes content of defs and properties that aren't rendered directly without ids. |
| * |
| * @author Lev Solntsev |
| * |
| * @type {import('./plugins-types').Plugin<'removeUselessDefs'>} |
| */ |
| exports.fn = () => { |
| return { |
| element: { |
| enter: (node, parentNode) => { |
| if (node.name === 'defs') { |
| /** |
| * @type {XastElement[]} |
| */ |
| const usefulNodes = []; |
| collectUsefulNodes(node, usefulNodes); |
| if (usefulNodes.length === 0) { |
| detachNodeFromParent(node, parentNode); |
| } |
| // TODO remove legacy parentNode in v4 |
| for (const usefulNode of usefulNodes) { |
| Object.defineProperty(usefulNode, 'parentNode', { |
| writable: true, |
| value: node, |
| }); |
| } |
| node.children = usefulNodes; |
| } else if ( |
| elemsGroups.nonRendering.has(node.name) && |
| node.attributes.id == null |
| ) { |
| detachNodeFromParent(node, parentNode); |
| } |
| }, |
| }, |
| }; |
| }; |
| |
| /** |
| * @type {(node: XastElement, usefulNodes: XastElement[]) => void} |
| */ |
| const collectUsefulNodes = (node, usefulNodes) => { |
| for (const child of node.children) { |
| if (child.type === 'element') { |
| if (child.attributes.id != null || child.name === 'style') { |
| usefulNodes.push(child); |
| } else { |
| collectUsefulNodes(child, usefulNodes); |
| } |
| } |
| } |
| }; |