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