1
0
mirror of https://github.com/svg/svgo.git synced 2025-04-19 10:22:15 +03:00
svgo/lib/xast.test.js

121 lines
2.6 KiB
JavaScript

/**
* @typedef {import('./types.js').XastElement} XastElement
* @typedef {import('./types.js').XastRoot} XastRoot
*/
import { visit, visitSkip, detachNodeFromParent } from './xast.js';
/**
* @type {(children: XastElement[]) => XastRoot}
*/
const root = (children) => {
return { type: 'root', children };
};
/**
* @type {(
* name: string,
* attrs?: ?Record<string, string>,
* children?: XastElement[]
* ) => XastElement}
*/
const x = (name, attrs = null, children = []) => {
return { type: 'element', name, attributes: attrs || {}, children };
};
test('visit enters into nodes', () => {
const ast = root([x('g', null, [x('rect'), x('circle')]), x('ellipse')]);
/**
* @type {string[]}
*/
const entered = [];
visit(ast, {
root: {
enter: (node) => {
entered.push(node.type);
},
},
element: {
enter: (node) => {
entered.push(`${node.type}:${node.name}`);
},
},
});
expect(entered).toStrictEqual([
'root',
'element:g',
'element:rect',
'element:circle',
'element:ellipse',
]);
});
test('visit exits from nodes', () => {
const ast = root([x('g', null, [x('rect'), x('circle')]), x('ellipse')]);
/**
* @type {string[]}
*/
const exited = [];
visit(ast, {
root: {
exit: (node) => {
exited.push(node.type);
},
},
element: {
exit: (node) => {
exited.push(`${node.type}:${node.name}`);
},
},
});
expect(exited).toStrictEqual([
'element:rect',
'element:circle',
'element:g',
'element:ellipse',
'root',
]);
});
test('visit skips entering children if node is detached', () => {
const ast = root([x('g', null, [x('rect'), x('circle')]), x('ellipse')]);
/**
* @type {string[]}
*/
const entered = [];
visit(ast, {
element: {
enter: (node, parentNode) => {
entered.push(node.name);
if (node.name === 'g') {
detachNodeFromParent(node, parentNode);
}
},
},
});
expect(entered).toStrictEqual(['g', 'ellipse']);
expect(ast).toStrictEqual(root([x('ellipse')]));
});
test('visit skips entering children when symbol is passed', () => {
const ast = root([x('g', null, [x('rect'), x('circle')]), x('ellipse')]);
/**
* @type {string[]}
*/
const entered = [];
visit(ast, {
element: {
enter: (node) => {
entered.push(node.name);
if (node.name === 'g') {
return visitSkip;
}
},
},
});
expect(entered).toStrictEqual(['g', 'ellipse']);
expect(ast).toStrictEqual(
root([x('g', null, [x('rect'), x('circle')]), x('ellipse')]),
);
});