项目作者: imcuttle

项目描述 :
Enhanced and multifunctional tree walker
高级语言: JavaScript
项目地址: git://github.com/imcuttle/walk-tree.git
创建时间: 2018-09-17T17:11:26Z
项目社区:https://github.com/imcuttle/walk-tree

开源协议:MIT License

下载


@moyuyc/walk-tree

Build status
Test coverage
@moyuyc/walk-tree"">NPM version
@moyuyc/walk-tree"">NPM Downloads
Prettier
Conventional Commits

Enhanced and multifunctional tree walker

Installation

  1. npm install @moyuyc/walk-tree
  2. # or use yarn
  3. yarn add @moyuyc/walk-tree

Usage

  1. const walkTree = require('@moyuyc/walk-tree')
  2. walkTree(
  3. {
  4. name: 'root',
  5. children: [{ name: 'c1' }, { name: 'c2' }, { name: 'c3', children: { name: 'c31' } }]
  6. },
  7. node => {
  8. console.log(node.name)
  9. }
  10. )
  11. // prints
  12. // root / c1 / c2 / c3 / c31
  13. // by order

API

walk

index.js:54-138

Parameters

  • tree {T} - Type T should extends Object
  • walker {(node, ctx: Context) => {}} - Iterator for each node by order
  • opts 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
      (optional, default 'pre')
    • opts.skipVisited {boolean}
      Should skip the node which has been visited. (optional, default true)
    • opts.uniquePath {Function | string | null}
      The unique’s path for determining the node has been visited (same node) (optional, default node=>node)
    • opts.state {any}
      Inject in context.state on customized way

Returns any walkedTree {T}

Context

A traversal context.

Four operations are available. Note that depending on the traversal order, some operations have no effects.

remove

  1. walk(rootNode, (node, ctx) => {
  2. if (node.name === 'remove-me') {
  3. return ctx.remove()
  4. }
  5. })

replace

  1. walk(rootNode, (node, ctx) => {
  2. if (node.name === 'replace-me') {
  3. return ctx.replace({ name: 'new-me' })
  4. }
  5. })

break

Stop traversal now.

  1. walk(rootNode, (node, ctx) => {
  2. if (node.name === 'stop') {
  3. return ctx.break()
  4. }
  5. })

skip

Skip current node, children won’t be visited.

  1. walk(rootNode, (node, ctx) => {
  2. if (node.name === 'skip') {
  3. return ctx.skip()
  4. }
  5. })

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.

FAQ

How can I get the path of the current node tree-crawl#37?

tl;dr It’s easy for DFS, less easy for BFS

If you are using DFS you can use the following utility function:

  1. const getPath = context =>
  2. (context.cursor.stack.xs || []).reduce((path, item) => {
  3. if (item.node) {
  4. path.push(item.node)
  5. }
  6. return path
  7. }, [])

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.

Credit

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.

Authors

This library is written and maintained by imcuttle, moyuyc95@gmail.com.

License

MIT