| t.js | |
|---|---|
| version 0.4.0 (source) t-js is freely distributable under the MIT license |  | 
| overviewt.js is a tree-traversal library. its only assumption is that the trees it traverses are made up of objects with 'children' arrays: it's entirely non-recursive, including the post-order traversal and  testingunit tests are provided courtesy of
 or viewed in most any system with a modern browser by opening the
 documentation is generated with the  | (function() { | 
| usagethe   | var _dfsPostOrder,
    t = {},
    root = this,
    isArray = function(o) {
        return Object.prototype.toString.call(o) == '[object Array]';
    };
/*global exports:true, module:true, define*/
if (typeof exports !== 'undefined') {
    if (typeof module !== 'undefined' && module.exports) {
        exports = module.exports = t;
    }
    exports.t = t;
} else {
    root['t'] = t;
} | 
| available functions |  | 
| t.bfs()perform a breadth-first search, executing the given callback at each node. 
 | t.bfs = function(node) {
    var cur, config, callback, children, i, length, par,
        queue = isArray(node)? node.slice(0) : [node],
        parents = [undefined];
    if (node == null) return node;
    if (arguments.length >= 3) {
        config = arguments[1];
        callback = arguments[2];
    } else {
        config = {};
        callback = arguments[1];
    }
    while (queue.length) {
        cur = queue.shift();
        par = parents.shift();
        callback.call(cur, cur, par);
        children = cur.children || [];
        for (i = 0, length = children.length; i < length; i++) {
            queue.push(children[i]);
            parents.push(cur);
        }
    }
    return node;
}; | 
| t.dfs()perform a depth-first search, executing the given callback at each node. in the pre-order case,  
 | t.dfs = function(node) {
    var cur, par, children, ctrl, i, ret, numArgs = arguments.length,
        nodes = isArray(node)? node.slice(0).reverse() : [node],
        config = numArgs === 3? arguments[1] : {},
        callback = arguments[numArgs === 3? 2 : 1],
        parents = [];
    if (typeof nodes[0] === 'undefined' && nodes.length === 1) return;
    if (config.order === 'post') {
        ret = _dfsPostOrder(nodes, config, callback);
        return isArray(node)? ret : ret[0];
    }
    for (i = nodes.length-1; i >= 0; i--)
        parents.push(undefined);
    while (nodes.length > 0) {
        cur = nodes.pop();
        par = parents.pop();
        ctrl = {};
        callback.call(cur, cur, par, ctrl);
        if (ctrl.stop) break;
        children = (cur && cur.children)? cur.children : [];
        for (i = ctrl.cutoff? -1 : children.length-1; i >= 0; i--) {
            nodes.push(children[i]);
            parents.push(cur);
        }
    }
    return node;
}; | 
| t.map()given a tree, return a tree of the same structure made up of the objects
returned by the callback which is executed at each node.  think of the
 
 | t.map = function() {
    var node = arguments[0],
        numArgs = arguments.length,
        config = numArgs === 3? arguments[1] : {},
        filter = config.filter,
        nodeFactory = arguments[numArgs === 3? 2 : 1],
        ret = isArray(node)? [] : undefined,
        last = function(l) { return l[l.length-1]; },
        parentStack = [];
    t.dfs(node, function(n, par, ctrl) {
        var curParent = last(parentStack),
            newNode = nodeFactory(n, curParent? curParent.ret : undefined);
        if (filter && ! newNode) {
            ctrl.cutoff = true;
            if (curParent && n === last(curParent.n.children)) {
                parentStack.pop();
                if (curParent.ret.children && ! curParent.ret.children.length)
                    delete curParent.ret.children;
            }
            return;
        }
        if (! par) {
            if (isArray(node))
                ret.push(newNode);
            else
                ret = newNode;
        } else {
            curParent.ret.children.push(newNode);
            if (n === last(curParent.n.children)) {
                parentStack.pop();
                if (curParent.ret.children && ! curParent.ret.children.length)
                    delete curParent.ret.children;
            }
        }
        if (n.children && n.children.length) {
            newNode.children = [];
            parentStack.push({n: n, ret: newNode});
        }
    });
    return ret;
}; | 
| t.filter()given a tree, return a tree of the same structure made up of the objects returned by the callback which is executed at each node. if, however, at a given node the callback returns a falsy value, then the current node and all of its descendents will be pruned from the output tree. 
 returns: a new tree, filtered by the callback function | t.filter = function(node, nodeFactory) {
    return t.map(node, {filter: true}, nodeFactory);
}; | 
| t.stroll()a walk through the trees... given two trees of similar structure, traverse both trees at the same time, executing the given callback with the pair of corresponding nodes as arguments. 
 | t.stroll = function(tree1, tree2, callback) {
    var i, children, node2,
        nodes2 = isArray(tree2)? tree2.slice(0).reverse() : [tree2],
        len = function(a) { return typeof a === 'undefined'? 0 : a.length; };
    t.dfs(tree1, function(node1, par, ctrl) {
        node2 = nodes2.pop();
        callback(node1, node2);
        if (node1 && node2 && len(node1.children) === len(node2.children))
            for (i = (node2.children || []).length-1; i >= 0; i--)
                nodes2.push(node2.children[i]);
        else
            ctrl.cutoff = true;
    });
}; | 
| t.find()given a tree and a truth test, return the first node that responds with a truthy value 
 returns: the found node | t.find = function(tree, callback) {
    var found;
    t.dfs(tree, function(node, par, ctrl) {
        if (callback.call(node, node, par)) {
            ctrl.stop = true;
            found = this;
        }
    });
    return found;
}; | 
| _dfsPostOrder()this is a module-private function used by  | _dfsPostOrder = function(nodes, config, callback) {
    var cur, par, ctrl, node, i,
        last = function(l) { return l[l.length-1]; },
        ret = [],
        stack = [{
            node: nodes.pop(),
            index: 0,
            ret: []
        }];
    while (stack.length) {
        cur = last(stack);
        node = cur.node;
        if (node.children && node.children.length) {
            if (cur.index < node.children.length) {
                stack.push({
                    node: node.children[cur.index++],
                    index: 0,
                    ret: []
                });
                continue;
            }
        }
        ctrl = {};
        par = stack[stack.length-2];
        if (par) {
            par.ret.push(callback.call(node, node, par.node, ctrl, cur.ret));
            stack.pop();
        } else {
            ret.push(callback.call(node, node, undefined, ctrl, cur.ret));
            stack.pop();
            if (nodes.length)
                stack.push({
                    node: nodes.pop(),
                    index: 0,
                    ret: []
                });
        }
    }
    return ret;
}; | 
| creditsthis library is of course heavily inspired by the work of
@jashkenas and others on  | }());
 |