Enhanced and multifunctional tree walker
@moyuyc/walk-tree"">
@moyuyc/walk-tree"">
Enhanced and multifunctional tree walker
npm install @moyuyc/walk-tree
# or use yarn
yarn add @moyuyc/walk-tree
const walkTree = require('@moyuyc/walk-tree')
walkTree(
{
name: 'root',
children: [{ name: 'c1' }, { name: 'c2' }, { name: 'c3', children: { name: 'c31' } }]
},
node => {
console.log(node.name)
}
)
// prints
// root / c1 / c2 / c3 / c31
// by order
tree
{T} - Type T
should extends Objectwalker
{(node, ctx: Context) => {}} - Iterator for each node by orderopts
Object {object}opts.path
{string} - The child’s path on recursive struction (optional, default 'children'
)opts.order
{‘pre’ | ‘post’ | ‘bfs’}pre
means walking the node before walking it’s children node by dfs post
means walking the node after walking it’s children node by dfs bfs
means walking the node by bfs 'pre'
)opts.skipVisited
{boolean}true
)opts.uniquePath
{Function | string | null}node=>node
)opts.state
{any}context.state
on customized wayReturns any walkedTree {T}
A traversal context.
Four operations are available. Note that depending on the traversal order, some operations have no effects.
remove
walk(rootNode, (node, ctx) => {
if (node.name === 'remove-me') {
return ctx.remove()
}
})
replace
walk(rootNode, (node, ctx) => {
if (node.name === 'replace-me') {
return ctx.replace({ name: 'new-me' })
}
})
break
Stop traversal now.
walk(rootNode, (node, ctx) => {
if (node.name === 'stop') {
return ctx.break()
}
})
skip
Skip current node, children won’t be visited.
walk(rootNode, (node, ctx) => {
if (node.name === 'skip') {
return ctx.skip()
}
})
parent
Get the parent of the current node.
depth
Get the depth of the current node. The depth is the number of ancestors the current node has.
level
Get the level of current node. The level is the number of ancestors+1 the current node has.
index
Get the index of the current node.
If you are using DFS you can use the following utility function:
const getPath = context =>
(context.cursor.stack.xs || []).reduce((path, item) => {
if (item.node) {
path.push(item.node)
}
return path
}, [])
If you are really concerned about performance, you could read items from the stack directly. Each item has a node and index property that you can use. The first item in the stack can be discarded and will have a node set to null. Be aware that you should not mutate the stack, or it will break the traversal.
If you are using BFS, things gets more complex. A simple hacky way to do so is to traverse the tree using DFS first. You can ad a path property to your nodes using the method above. And then do your regular BFS traversal using that path property.
The core algorithm of traverse credits to tree-crawl
walk-tree has the different with tree-crawl
Because tree-crawl has no idea about the intrinsic structure of your tree, you have to remove the node yourself. Context#remove
only notifies the traversal code that the structure of the tree has changed.
Otherwise walk-tree would infers your tree by option path
so don’t requires additional remove action.
This library is written and maintained by imcuttle, moyuyc95@gmail.com.
MIT