(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o'); }, teardown: function() { var $tree = $('#tree1'); $tree.tree('destroy'); $tree.remove(); } }); test("create jqtree from data", function() { $('#tree1').tree({ data: example_data }); equal( $('#tree1').children().length, 1, 'number of children on level 0' ); ok( $('#tree1').children().is('ul.jqtree-tree'), 'first element is ul.jqtree-tree' ); equal( $('#tree1 ul.jqtree-tree > li').length, 2, 'number of children on level 1' ); ok( $('#tree1 ul.jqtree-tree li:eq(0)').is('li.jqtree-folder.jqtree-closed'), 'first child is li.jqtree-folder.jqtree-closed' ); ok( $('#tree1 ul.jqtree-tree li:eq(0) > .jqtree-element > a.jqtree-toggler').is('a.jqtree-toggler.jqtree-closed'), 'button in first folder' ); equal( $('#tree1 ul.jqtree-tree li:eq(0) > .jqtree-element span.jqtree-title').text(), 'node1' ); }); test('toggle', function() { // create tree var $tree = $('#tree1'); var $node1; var node1; $tree.tree({ data: example_data }); $tree.bind( 'tree.open', function(e) { start(); ok(! isNodeClosed($node1), 'node1 is open'); // 2. close node1 $tree.tree('toggle', node1); stop(); } ); $tree.bind( 'tree.close', function(e) { start(); ok(isNodeClosed($node1), 'node1 is closed'); } ); var tree = $tree.tree('getTree'); node1 = tree.children[0]; $node1 = $tree.find('ul.jqtree-tree li:eq(0)'); // node1 is initially closed ok(isNodeClosed($node1), 'node1 is closed'); // 1. open node1 $tree.tree('toggle', node1); stop(); }); test("click event", function() { stop(); // create tree var $tree = $('#tree1'); $tree.tree({ data: example_data, selectable: true }); $tree.bind('tree.click', function(e) { equal(e.node.name, 'node1'); }); $tree.bind('tree.select', function(e) { start(); equal(e.node.name, 'node1'); }); // click on node1 var $node1 = $tree.find('ul.jqtree-tree li:first'); var $text_span = $node1.find('span:first'); $text_span.click(); }); test('saveState', function() { var $tree = $('#tree1'); var saved_state; function setState(state) { saved_state = state; } function getState() { return saved_state; } function createTree() { $tree.tree({ data: example_data, saveState: true, onSetStateFromStorage: setState, onGetStateFromStorage: getState, selectable: true }); } // create tree createTree(); // nodes are initially closed var tree = $tree.tree('getTree'); tree.iterate(function(node) { ok(! node.is_open, 'jqtree-closed'); return true; }); // open node1 $tree.tree('toggle', tree.children[0]); // node1 is open ok(tree.children[0].is_open, 'node1 is_open'); // select node2 $tree.tree('selectNode', tree.children[1]); // node2 is selected equal( $tree.tree('getSelectedNode').name, 'node2', 'getSelectedNode (1)' ); // create tree again $tree.tree('destroy'); createTree(); tree = $tree.tree('getTree'); ok(tree.children[0].is_open, 'node1 is_open'); ok(! tree.children[1].is_open, 'node2 is closed'); // node2 is selected equal( $tree.tree('getSelectedNode').name, 'node2', 'getSelectedNode (2)' ); }); test('getSelectedNode', function() { var $tree = $('#tree1'); // create tree $tree.tree({ data: example_data, selectable: true }); // there is no node selected equal( $tree.tree('getSelectedNode'), false, 'getSelectedNode' ); // select node1 var tree = $tree.tree('getTree'); var node1 = tree.children[0]; $tree.tree('selectNode', node1); // node1 is selected equal( $tree.tree('getSelectedNode').name, 'node1', 'getSelectedNode' ); }); test("toJson", function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); // 1. call toJson equal( $tree.tree('toJson'), '[{"name":"node1","id":123,"int_property":1,"str_property":"1",'+ '"children":[{"name":"child1","id":125,"int_property":2},{"name":'+ '"child2","id":126}]},{"name":"node2","id":124,"int_property":3,'+ '"str_property":"3","children":[{"name":"child3","id":127}]}]' ); // Check that properties 'children', 'parent' and 'element' still exist. var tree = $tree.tree('getTree'); equal(tree.children.length, 2); ok(tree.children[0].parent != undefined, 'parent'); ok($(tree.children[0].element).is('li'), 'element'); }); test('loadData', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data, autoOpen: true }); // first node is 'node1' equal( $tree.find('> ul > li:first .jqtree-element:first > span').text(), 'node1' ); // - load new data $tree.tree('loadData', example_data2); // first node is 'main' equal( $tree.find('> ul > li:first .jqtree-element:first > span').text(), 'main' ); // - load new data under node 'child3' $tree.tree('loadData', example_data); var child3 = $tree.tree('getNodeByName', 'child3'); var data = [ { label: 'c4', id: 200 }, { label: 'c5', id: 201, children: [ { label: 'c6', id: 202 } ] } ]; $tree.tree('loadData', data, child3); // first node in html is still 'node1' equal( $tree.find('li:eq(0)').find('.jqtree-element:eq(0) span.jqtree-title').text(), 'node1' ); // Node 'child3' now has a children 'c4' and 'c5' var $child3 = $tree.find('span:contains(child3)'); var $li = $child3.closest('li'); equal( $li.children('ul').children('li:eq(0)').find('.jqtree-element span.jqtree-title').text(), 'c4' ); // Node 'child3' must have toggler button ok( $child3.prev().is('a.jqtree-toggler'), "node 'child3' must have toggler button" ); // - select node 'c5' and load new data under 'child3' var c5 = $tree.tree('getNodeByName', 'c5'); $tree.tree('selectNode', c5); equal($tree.tree('getSelectedNode').name, 'c5'); var data2 = [ { label: 'c7' }, { label: 'c8' } ]; $tree.tree('loadData', data2, child3); // c5 must be deselected equal($tree.tree('getSelectedNode'), false); // - select c7; load new data under child3; note that c7 has no id $tree.tree('selectNode', $tree.tree('getNodeByName', 'c7')); equal($tree.tree('getSelectedNode').name, 'c7'); $tree.tree('loadData', [ 'c9' ], child3); equal($tree.tree('getSelectedNode'), false); // - select c9 (which has no id); load new nodes under child2 $tree.tree('selectNode', $tree.tree('getNodeByName', 'c9')); var child2 = $tree.tree('getNodeByName', 'child2'); $tree.tree('loadData', [ 'c10' ], child2); equal($tree.tree('getSelectedNode').name, 'c9'); }); test('openNode and closeNode', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); var node2 = $tree.tree('getNodeByName', 'node2'); equal(node2.name, 'node2'); equal(node2.is_open, undefined); // 1. open node2 $tree.tree('openNode', node2, false); equal(node2.is_open, true); equal(isNodeOpen($(node2.element)), true); // 2. close node2 $tree.tree('closeNode', node2, false); equal(node2.is_open, false); equal(isNodeClosed($(node2.element)), true); // 3. open child1 var node1 = $tree.tree('getNodeByName', 'node1'); var child1 = $tree.tree('getNodeByName', 'child1'); // add a child to child1 so it is a folder $tree.tree('appendNode', 'child1a', child1); // node1 is initialy closed equal(node1.is_open, undefined); // open child1 $tree.tree('openNode', child1, false); // node1 and child1 are now open1 equal(node1.is_open, true); equal(child1.is_open, true); }); test('selectNode', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data, selectable: true }); var node1 = $tree.tree('getTree').children[0]; var node2 = $tree.tree('getTree').children[1]; var child3 = node2.children[0]; equal(child3.name, 'child3'); equal(node1.is_open, undefined); equal(node2.is_open, undefined); equal(child3.is_open, undefined); // -- select node 'child3', which is a child of 'node2'; must_open_parents = true $tree.tree('selectNode', child3, true); equal($tree.tree('getSelectedNode').name, 'child3'); equal(node1.is_open, undefined); equal(node2.is_open, true); equal(child3.is_open, undefined); // -- select node 'node1' $tree.tree('selectNode', node1); equal($tree.tree('getSelectedNode').name, 'node1'); // -- is 'node1' selected? ok($tree.tree('isNodeSelected', node1)); }); test('selectNode when another node is selected', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data, selectable: true }); var node1 = $tree.tree('getTree').children[0]; var node2 = $tree.tree('getTree').children[1]; // -- select node 'node2' $tree.tree('selectNode', node2); equal($tree.tree('getSelectedNode').name, 'node2'); // -- setting event // -- is node 'node2' named 'deselected_node' in object's attributes? stop(); $tree.bind('tree.select', function(e) { start(); equal(e.deselected_node, node2); }); // -- select node 'node1'; node 'node2' is selected before it $tree.tree('selectNode', node1); equal($tree.tree('getSelectedNode').name, 'node1'); ok($tree.tree('isNodeSelected', node1)); }); test('click toggler', function() { // setup stop(); var $tree = $('#tree1'); $tree.tree({ data: example_data, selectable: true }); var $title = $tree.find('li:eq(0)').find('> .jqtree-element > span.jqtree-title'); equal($title.text(), 'node1'); var $toggler = $title.prev(); ok($toggler.is('a.jqtree-toggler.jqtree-closed')); $tree.bind('tree.open', function(e) { // 2. handle 'open' event start(); equal(e.node.name, 'node1'); stop(); // 3. click toggler again $toggler.click(); }); $tree.bind('tree.close', function(e) { start(); equal(e.node.name, 'node1'); }); // 1. click toggler of 'node1' $toggler.click(); }); test('getNodeById', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); var node2 = $tree.tree('getNodeByName', 'node2'); // 1. get 'node2' by id equal( $tree.tree('getNodeById', 124).name, 'node2' ); // 2. get id that does not exist equal($tree.tree('getNodeById', 333), null); // 3. get id by string equal( $tree.tree('getNodeById', '124').name, 'node2' ); // 4. add node with string id; search by int $tree.tree( 'appendNode', { label: 'abc', id: '234' } ); equal( $tree.tree('getNodeById', 234).name, 'abc' ); equal( $tree.tree('getNodeById', '234').name, 'abc' ); // 5. load subtree in node2 var subtree_data = [ { label: 'sub1', id: 200, children: [ {label: 'sub2', id: 201} ] } ]; $tree.tree('loadData', subtree_data, node2); var t = $tree.tree('getTree'); equal( $tree.tree('getNodeById', 200).name, 'sub1' ); equal( $tree.tree('getNodeById', 201).name, 'sub2' ); }); test('autoOpen', function() { var $tree = $('#tree1'); function formatOpenFolders() { var open_nodes = []; $tree.find('li').each(function() { var $li = $(this); if ($li.is('.jqtree-folder') && ! $li.is('.jqtree-closed')) { var label = $li.children('.jqtree-element').find('span').text(); open_nodes.push(label); }; }); return open_nodes.join(';'); } /* -l1n1 (level 0) ----l2n1 (1) ----l2n2 (1) -------l3n1 (2) ----------l4n1 (3) -l1n2 */ var data = [ { label: 'l1n1', children: [ 'l2n1', { label: 'l2n2', children: [ { label: 'l3n1', children: [ 'l4n1' ] } ] } ] }, 'l1n2' ]; // 1. autoOpen is false $tree.tree({ data: data, autoOpen: false }); equal(formatOpenFolders(), ''); $tree.tree('destroy'); // 2. autoOpen is true $tree.tree({ data: data, autoOpen: true }); equal(formatOpenFolders(), 'l1n1;l2n2;l3n1'); $tree.tree('destroy'); // 3. autoOpen level 1 $tree.tree({ data: data, autoOpen: 1 }); equal(formatOpenFolders(), 'l1n1;l2n2'); }); test('onCreateLi', function() { // 1. init tree with onCreateLi var $tree = $('#tree1'); $tree.tree({ data: example_data, onCreateLi: function(node, $li) { var $span = $li.children('.jqtree-element').find('span'); $span.html('_' + node.name + '_'); } }); equal( $tree.find('span:eq(0)').text(), '_node1_' ); }); test('save state', function() { // setup var state = null; // Fake $.cookie plugin for browsers that do not support localstorage $.cookie = function(key, param2, param3) { if (typeof param3 == 'object') { // set state = param2; } else { // get return state; } } // Remove state from localstorage if (typeof localStorage != 'undefined') { localStorage.setItem('my_tree', null); } // 1. init tree var $tree = $('#tree1'); $tree.tree({ data: example_data, selectable: true, saveState: 'my_tree' }); var tree = $tree.tree('getTree'); equal($tree.tree('getSelectedNode'), false); // 2. select node -> state is saved $tree.tree('selectNode', tree.children[0]); equal($tree.tree('getSelectedNode').name, 'node1'); // 3. init tree again $tree.tree('destroy'); $tree.tree({ data: example_data, selectable: true, saveState: 'my_tree' }); equal($tree.tree('getSelectedNode').name, 'node1'); $.cookie = null; }); test('generate hit areas', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); // 1. get hit areas var node = $tree.tree('getNodeById', 123); var hit_areas = $tree.tree('testGenerateHitAreas', node); var strings = $.map(hit_areas, function(hit_area) { return hit_area.node.name + ' ' + Position.getName(hit_area.position); }); equal(strings.join(';'), 'node1 none;node2 inside;node2 after'); }); test('removeNode', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data, selectable: true }); // 1. Remove selected node; node is 'child1' var child1 = $tree.tree('getNodeByName', 'child1'); $tree.tree('selectNode', child1); equal($tree.tree('getSelectedNode').name, 'child1'); $tree.tree('removeNode', child1); equal( formatTitles($tree), 'node1 child2 node2 child3' ); // getSelectedNode must now return false equal($tree.tree('getSelectedNode'), false); // 2. No node is selected; remove child3 $tree.tree('loadData', example_data); var child3 = $tree.tree('getNodeByName', 'child3'); $tree.tree('removeNode', child3); equal( formatTitles($tree), 'node1 child1 child2 node2' ); equal($tree.tree('getSelectedNode'), false); // 3. Remove parent of selected node $tree.tree('loadData', example_data); child1 = $tree.tree('getNodeByName', 'child1'); var node1 = $tree.tree('getNodeByName', 'node1'); $tree.tree('selectNode', child1); $tree.tree('removeNode', node1); // node is unselected equal($tree.tree('getSelectedNode'), false); // 4. Remove unselected node without an id $tree.tree('loadData', example_data2); var c1 = $tree.tree('getNodeByName', 'c1'); $tree.tree('removeNode', c1); equal( formatTitles($tree), 'main c2' ) }); test('appendNode', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); var node1 = $tree.tree('getNodeByName', 'node1'); // 1. Add child3 to node1 $tree.tree('appendNode', 'child3', node1); equal( formatTitles($(node1.element)), 'node1 child1 child2 child3' ); // 2. Add child4 to child1 var child1 = $tree.tree('getNodeByName', 'child1'); // Node 'child1' does not have a toggler button equal( $(child1.element).find('> .jqtree-element > .jqtree-toggler').length, 0 ); $tree.tree('appendNode', 'child4', child1); equal(formatTitles($(child1.element)), 'child1 child4'); // Node 'child1' must get a toggler button equal( $(child1.element).find('> .jqtree-element > .jqtree-toggler').length, 1 ); }); test('prependNode', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); var node1 = $tree.tree('getNodeByName', 'node1'); // 1. Prepend child0 to node1 $tree.tree('prependNode', 'child0', node1); equal( formatTitles($(node1.element)), 'node1 child0 child1 child2' ); }); test('init event', function() { // setup var $tree = $('#tree1'); $tree.bind('tree.init', function() { start(); // Check that we can call functions in 'tree.init' event equal($tree.tree('getNodeByName', 'node2').name, 'node2'); }); stop(); $tree.tree({ data: example_data }); }); test('updateNode', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); equal(formatTitles($tree), 'node1 child1 child2 node2 child3'); // -- update label var node2 = $tree.tree('getNodeByName', 'node2'); $tree.tree('updateNode', node2, 'CHANGED'); equal(formatTitles($tree), 'node1 child1 child2 CHANGED child3'); equal(node2.name, 'CHANGED'); // -- update data $tree.tree( 'updateNode', node2, { name: 'xyz', tag1: 'abc' } ); equal(formatTitles($tree), 'node1 child1 child2 xyz child3'); equal(node2.name, 'xyz'); equal(node2.tag1, 'abc'); // - update id equal(node2.id, 124); $tree.tree('updateNode', node2, {id: 555}); equal(node2.id, 555); equal(node2.name, 'xyz'); // get node by id var node_555 = $tree.tree('getNodeById', 555); equal(node_555.name, 'xyz'); var node_124 = $tree.tree('getNodeById', 124); equal(node_124, undefined); // update child1 var child1 = $tree.tree('getNodeByName', 'child1'); $tree.tree('updateNode', child1, 'child1a'); equal(formatTitles($tree), 'node1 child1a child2 xyz child3'); // select child1 $tree.tree('selectNode', child1); $tree.tree('updateNode', child1, 'child1b'); ok($(child1.element).hasClass('jqtree-selected')); }); test('moveNode', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); var child1 = $tree.tree('getNodeByName', 'child1'); var child2 = $tree.tree('getNodeByName', 'child2'); var node1 = $tree.tree('getNodeByName', 'node1'); var node2 = $tree.tree('getNodeByName', 'node2'); // -- Move child1 after node2 $tree.tree('moveNode', child1, node2, 'after'); equal(formatTitles($tree), 'node1 child2 node2 child3 child1'); // -- Check that illegal moves are skipped $tree.tree('moveNode', node1, child2, 'inside'); }); test('load on demand', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: [ { id: 1, label: 'node1', load_on_demand: true } ], dataUrl: '/tree/' }); $.mockjax({ url: '*', response: function(options) { equal(options.url, '/tree/', '2'); deepEqual(options.data, { 'node' : 1 }, '3') this.responseText = [ { id: 2, label: 'child1' } ]; }, logging: false }); // -- open node $tree.bind('tree.refresh', function(e) { start(); equal(formatTitles($tree), 'node1 child1', '4'); }); var node1 = $tree.tree('getNodeByName', 'node1'); equal(formatTitles($tree), 'node1', '1'); $tree.tree('openNode', node1, true); stop(); }); test('addNodeAfter', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); var node1 = $tree.tree('getNodeByName', 'node1'); // -- add node after node1 $tree.tree('addNodeAfter', 'node3', node1); equal(formatTitles($tree), 'node1 child1 child2 node3 node2 child3'); }); test('addNodeBefore', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); var node1 = $tree.tree('getNodeByName', 'node1'); // -- add node before node1 var new_node = $tree.tree('addNodeBefore', 'node3', node1); equal(formatTitles($tree), 'node3 node1 child1 child2 node2 child3'); }); test('addParentNode', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); var child3 = $tree.tree('getNodeByName', 'child3'); // -- add parent to child3 $tree.tree('addParentNode', 'node3', child3); equal(formatTitles($tree), 'node1 child1 child2 node2 node3 child3'); }); test('mouse events', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data, dragAndDrop: true, autoOpen: true }); $tree.tree('setMouseDelay', 0); function getTitleElement(node_name) { var node = $tree.tree('getNodeByName', node_name); var $el = $(node.element); return $($el.find('.jqtree-title')); } var $node1 = getTitleElement('node1'); var $child3 = getTitleElement('child3'); // Move node1 inside child3 // trigger mousedown event on node1 $node1.trigger( $.Event('mousedown', { which: 1 }) ); // trigger mouse move to child3 var child3_offset = $child3.offset(); $tree.trigger( $.Event('mousemove', { pageX: child3_offset.left, pageY: child3_offset.top }) ); $tree.trigger('mouseup'); equal( formatTitles($tree), 'node2 child3 node1 child1 child2' ); }); test('multiple select', function() { // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); var child1 = $tree.tree('getNodeByName', 'child1'); var child2 = $tree.tree('getNodeByName', 'child2'); // -- add nodes to selection // todo: more nodes as parameters? // todo: rename to 'selection.add' or 'selection' 'add'? $tree.tree('addToSelection', child1); $tree.tree('addToSelection', child2); // -- get selected nodes var selected_nodes = $tree.tree('getSelectedNodes'); equal( formatNodes(selected_nodes), 'child1 child2' ); }); test('keyboard', function() { // setup var $tree = $('#tree1'); function keyDown(key) { $tree.trigger( $.Event('keydown', { which: key }) ); } $tree.tree({ data: example_data }); var node1 = $tree.tree('getNodeByName', 'node1'); // select node1 $tree.tree('selectNode', node1); equal(node1.is_open, undefined); // - move down; -> node2 keyDown(40); equal($tree.tree('getSelectedNode').name, 'node2'); // - move up; -> back to node1 keyDown(38); equal($tree.tree('getSelectedNode').name, 'node1'); // - move right; open node1 keyDown(39); equal(node1.is_open, true); equal($tree.tree('getSelectedNode').name, 'node1'); // - select child3 and move up -> node2 $tree.tree('selectNode', $tree.tree('getNodeByName', 'child3')); keyDown(38); equal($tree.tree('getSelectedNode').name, 'node2'); // - move up -> child2 keyDown(38); equal($tree.tree('getSelectedNode').name, 'child2'); // - select node1 and move left -> close $tree.tree('selectNode', node1); keyDown(37); equal(node1.is_open, false); equal($tree.tree('getSelectedNode').name, 'node1'); }); test('getNodesByProperty', function(){ // setup var $tree = $('#tree1'); $tree.tree({ data: example_data }); var node2 = $tree.tree('getNodeByName', 'node2'); // 1. get 'node1' by property equal( $tree.tree('getNodesByProperty', 'int_property', 1)[0].name, 'node1' ); // 2. get property that does not exist in any node equal($tree.tree('getNodesByProperty', 'int_property', 123).length, 0); // 3. get string property equal( $tree.tree('getNodesByProperty', 'str_property', '1')[0].name, 'node1' ); // 4. add node with string id; search by int $tree.tree( 'appendNode', { label: 'abc', id: '234', str_property: '111', int_property: 111 } ); equal( $tree.tree('getNodesByProperty', 'int_property', 111)[0].name, 'abc' ); equal( $tree.tree('getNodesByProperty', 'str_property', '111')[0].name, 'abc' ); // 5. load subtree in node2 var subtree_data = [ { label: 'sub1', id: 200, int_property: 222, children: [ {label: 'sub2', id: 201, int_property: 444} ] } ]; $tree.tree('loadData', subtree_data, node2); var t = $tree.tree('getTree'); equal( $tree.tree('getNodesByProperty', 'int_property', 222)[0].name, 'sub1' ); equal( $tree.tree('getNodesByProperty', 'int_property', 444)[0].name, 'sub2' ); }); QUnit.module("Tree"); test('constructor', function() { // 1. Create node from string var node = new Node('n1'); equal(node.name, 'n1'); equal(node.children.length, 0); equal(node.parent, null); // 2. Create node from object node = new Node({ label: 'n2', id: 123, parent: 'abc', // parent must be ignored children: ['c'], // children must be ignored url: '/' }); equal(node.name, 'n2'); equal(node.id, 123); equal(node.url, '/'); equal(node.label, undefined); equal(node.children.length, 0); equal(node.parent, null); }); test("create tree from data", function() { function checkData(tree) { equal( formatNodes(tree.children), 'node1 node2', 'nodes on level 1' ); equal( formatNodes(tree.children[0].children), 'child1 child2', 'children of node1' ); equal( formatNodes(tree.children[1].children), 'child3', 'children of node2' ); equal( tree.children[0].id, 123, 'id' ); } // - create tree from example data var tree = new Node(null, true); tree.loadFromData(example_data); checkData(tree); // - create tree from new data format var data = [ { label: 'node1', id: 123, children: ['child1', 'child2'] // nodes are only defined by a string }, { label: 'node2', id: 124, children: ['child3'] } ]; var tree = new Node(null, true); tree.loadFromData(data); checkData(tree); }); test("addChild", function() { var tree = new Node('tree1', true); tree.addChild( new Node('abc') ); tree.addChild( new Node('def') ); equal( formatNodes(tree.children), 'abc def', 'children' ); var node = tree.children[0]; equal( node.parent.name, 'tree1', 'parent of node' ); }); test('addChildAtPosition', function() { var tree = new Node(null, true); tree.addChildAtPosition(new Node('abc'), 0); // first tree.addChildAtPosition(new Node('ghi'), 2); // index 2 does not exist tree.addChildAtPosition(new Node('def'), 1); tree.addChildAtPosition(new Node('123'), 0); equal( formatNodes(tree.children), '123 abc def ghi', 'children' ); }); test('removeChild', function() { var tree = new Node(null, true); var abc = new Node({'label': 'abc', 'id': 1}); var def = new Node({'label': 'def', 'id': 2}); var ghi = new Node({'label': 'ghi', 'id': 3}); tree.addChild(abc); tree.addChild(def); tree.addChild(ghi); var jkl = new Node({'label': 'jkl', 'id': 4}); def.addChild(jkl); equal( formatNodes(tree.children), 'abc def ghi', 'children' ); equal(tree.id_mapping[2].name, 'def'); equal(tree.id_mapping[4].name, 'jkl'); // remove 'def' tree.removeChild(def); equal( formatNodes(tree.children), 'abc ghi', 'children' ); equal(tree.id_mapping[2], null); equal(tree.id_mapping[4], null); // remove 'ghi' tree.removeChild(ghi); equal( formatNodes(tree.children), 'abc', 'children' ); // remove 'abc' tree.removeChild(abc); equal( formatNodes(tree.children), '', 'children' ); }); test('getChildIndex', function() { // setup var tree = new Node(null, true); var abc = new Node('abc'); var def = new Node('def'); var ghi = new Node('ghi'); tree.addChild(abc); tree.addChild(def); tree.addChild(ghi); // 1. Get child index of 'def' equal(tree.getChildIndex(def), 1); // 2. Get child index of non-existing node equal(tree.getChildIndex(new Node('xyz')), -1); }); test('hasChildren', function() { var tree = new Node(null, true); equal( tree.hasChildren(), false, 'tree without children' ); tree.addChild(new Node('abc')); equal( tree.hasChildren(), true, 'tree has children' ); }); test('iterate', function() { var tree = new Node(null, true); tree.loadFromData(example_data); // iterate over all the nodes var nodes = []; tree.iterate( function(node, level) { nodes.push(node); return true; } ); equal( formatNodes(nodes), 'node1 child1 child2 node2 child3', 'all nodes' ); // iterate over nodes on first level nodes = []; tree.iterate( function(node, level) { nodes.push(node); return false; } ); equal( formatNodes(nodes), 'node1 node2', 'nodes on first level' ); // add child 4 var node3 = tree.getNodeById(124).children[0]; node3.addChild( new Node('child4') ); // test level parameter nodes = []; tree.iterate( function(node, level) { nodes.push(node.name + ' ' + level); return true; } ); equal( nodes.join(','), 'node1 0,child1 1,child2 1,node2 0,child3 1,child4 2' ); }); test('moveNode', function() { var tree = new Node(null, true); tree.loadFromData(example_data); /* node1 ---child1 ---child2 node2 ---child3 */ var node1 = tree.children[0]; var node2 = tree.children[1]; var child1 = node1.children[0]; var child2 = node1.children[1]; equal(node2.name, 'node2', 'node2 name'); equal(child2.name, 'child2', 'child2 name'); // move child2 after node2 tree.moveNode(child2, node2, Position.AFTER); /* node1 ---child1 node2 ---child3 child2 */ equal( formatNodes(tree.children), 'node1 node2 child2', 'tree nodes at first level' ); equal( formatNodes(node1.children), 'child1', 'node1 children' ); // move child1 inside node2 // this means it's the first child tree.moveNode(child1, node2, Position.INSIDE); /* node1 node2 ---child1 ---child3 child2 */ equal( formatNodes(node2.children), 'child1 child3', 'node2 children' ); equal( formatNodes(node1.children), '', 'node1 has no children' ); // move child2 before child1 tree.moveNode(child2, child1, Position.BEFORE); /* node1 node2 ---child2 ---child1 ---child3 */ equal( formatNodes(node2.children), 'child2 child1 child3', 'node2 children' ); equal( formatNodes(tree.children), 'node1 node2', 'tree nodes at first level' ); }); test('initFromData', function() { var data = { label: 'main', children: [ 'c1', { label: 'c2', id: 201 } ] }; var node = new Node(null, true); node.initFromData(data); equal(node.name, 'main') equal( formatNodes(node.children), 'c1 c2', 'children' ); equal(node.children[1].id, 201); }); test('getData', function() { // 1. empty node var node = new Node(null, true); deepEqual(node.getData(), []); // 2.node with data node.loadFromData( [ { label: 'n1', children: [ { label: 'c1' } ] } ] ); deepEqual( node.getData(), [ { name: 'n1', children: [ { name: 'c1' } ] } ] ); }); test('addAfter', function() { // setup var tree = new Node(null, true); tree.loadFromData(example_data); /* -node1 ---c1 ---c2 -node2 ---c3 */ equal(formatNodes(tree.children), 'node1 node2'); // - Add 'node_b' after node2 var node2 = tree.getNodeByName('node2'); node2.addAfter('node_b'); equal(formatNodes(tree.children), 'node1 node2 node_b'); var node_b = tree.getNodeByName('node_b'); equal(node_b.name, 'node_b'); // - Add 'node_a' after node1 var node1 = tree.getNodeByName('node1'); node1.addAfter('node_a'); equal(formatNodes(tree.children), 'node1 node_a node2 node_b'); // - Add 'node_c' after node_b; new node is an object node_b.addAfter({ label: 'node_c', id: 789 }); var node_c = tree.getNodeByName('node_c'); equal(node_c.id, 789); equal(formatNodes(tree.children), 'node1 node_a node2 node_b node_c'); // - Add after root node; this is not possible equal(tree.addAfter('node_x'), null); }); test('addBefore', function() { // setup var tree = new Node(null, true); tree.loadFromData(example_data); // - Add 'node_0' before node1 var node1 = tree.getNodeByName('node1'); node1.addBefore('node0'); equal(formatNodes(tree.children), 'node0 node1 node2'); // - Add before root node; this is not possile equal(tree.addBefore('x'), null); }); test('addParent', function() { // setup var tree = new Node(null, true); tree.loadFromData(example_data); // - Add node 'root' as parent of node1 // Note that node also becomes a child of 'root' var node1 = tree.getNodeByName('node1'); node1.addParent('root'); var root = tree.getNodeByName('root'); equal(formatNodes(root.children), 'node1 node2'); // - Add parent to root node; not possible equal(tree.addParent('x'), null); }); test('remove', function() { // setup var tree = new Node(null, true); tree.loadFromData(example_data); var child1 = tree.getNodeByName('child1'); var node1 = tree.getNodeByName('node1'); equal(formatNodes(node1.children), 'child1 child2'); equal(child1.parent, node1); // 1. Remove child1 child1.remove(); equal(formatNodes(node1.children), 'child2'); equal(child1.parent, null); }); test('append', function() { // setup var tree = new Node(null, true); tree.loadFromData(example_data); var node1 = tree.getNodeByName('node1'); // 1. Add child3 to node1 node1.append('child3'); equal(formatNodes(node1.children), 'child1 child2 child3'); }); test('prepend', function() { // setup var tree = new Node(null, true); tree.loadFromData(example_data); var node1 = tree.getNodeByName('node1'); // 1. Prepend child0 to node1 node1.prepend('child0'); equal(formatNodes(node1.children), 'child0 child1 child2'); }); test('getNodeById', function() { // setup var tree = new Node(null, true); tree.loadFromData(example_data); // 1. Get node with id 124 var node = tree.getNodeById(124); equal(node.name, 'node2'); // 2. Delete node with id 124 and search again node.remove(); equal(tree.getNodeById(124), null); // 3. Add node with id 456 and search for it var child3 = tree.getNodeByName('child2'); child3.append({ id: 456, label: 'new node' }); node = tree.getNodeById(456); equal(node.name, 'new node'); }); test('getLevel', function() { // setup var tree = new Node(null, true); tree.loadFromData(example_data); // 1. get level for node1 and child1 equal(tree.getNodeByName('node1').getLevel(), 1); equal(tree.getNodeByName('child1').getLevel(), 2); }); test('loadFromData and id mapping', function() { // - get node from empty tree var tree = new Node(null, true); equal(tree.getNodeById(999), null); // - load example data in tree tree.loadFromData(example_data); equal(tree.getNodeById(124).name, 'node2'); var child2 = tree.getNodeById(126); child2.addChild( new Node({label: 'child4', id: 128}) ); child2.addChild( new Node({label: 'child5', id: 129}) ); // - load data in node child2 child2.loadFromData(['abc', 'def']); equal(tree.getNodeById(128), null); equal(child2.children.length, 2); equal(child2.children[0].name, 'abc'); }); test('removeChildren', function() { // - load example data var tree = new Node(null, true); tree.loadFromData(example_data); // add child4 and child5 var child2 = tree.getNodeById(126); equal(child2.name, 'child2'); child2.addChild( new Node({label: 'child4', id: 128}) ); child2.addChild( new Node({label: 'child5', id: 129}) ); equal(tree.getNodeById(128).name, 'child4'); // - remove children from child2 child2.removeChildren(); equal(tree.getNodeById(128), null); equal(child2.children.length, 0); }); test('node with id 0', function() { // - load node with id 0 var tree = new Node(null, true); tree.loadFromData([ { id: 0, label: 'mynode' } ]); // - get node by id var node = tree.getNodeById(0); equal(node.name, 'mynode'); // -- remove the node node.remove(); equal(tree.getNodeById(0), undefined); }); test('getPreviousSibling', function() { // setup var tree = new Node(null, true); tree.loadFromData(example_data); // - getPreviousSibling equal( tree.getNodeByName('child2').getPreviousSibling().name, 'child1' ); equal(tree.getPreviousSibling(), null); equal( tree.getNodeByName('child1').getPreviousSibling(), null ); }); test('getNextSibling', function() { // setup var tree = new Node(null, true); tree.loadFromData(example_data); // - getNextSibling equal( tree.getNodeByName('node1').getNextSibling().name, 'node2' ); equal( tree.getNodeByName('node2').getNextSibling(), null ); equal(tree.getNextSibling(), null); }); test('getNodesByProperty', function() { var tree = new Node(null, true); tree.loadFromData(example_data); nodes = tree.getNodesByProperty('name', 'child1'); equal(nodes.length, 1); equal(nodes[0].name, 'child1'); }); QUnit.module('util'); test('JSON.stringify', function() { function test_stringify(stringify) { equal(stringify('abc'), '"abc"'); equal(stringify(123), '123'); equal(stringify(true), 'true'); equal(stringify({abc: 'def'}), '{"abc":"def"}'); equal(stringify({}), '{}'); equal(stringify([1, 2, 3]), '[1,2,3]'); equal(stringify(null), 'null'); equal(stringify(Number.NEGATIVE_INFINITY), 'null'); // test escapable JSON.stringify("\u200c"); } test_stringify(JSON.stringify); test_stringify(get_json_stringify_function()); }); test('indexOf', function() { equal(indexOf([3, 2, 1], 1), 2); equal(_indexOf([3, 2, 1], 1), 2); equal(indexOf([4, 5, 6], 1), -1); equal(_indexOf([4, 5, 6], 1), -1); }); test('Position.getName', function() { equal(Position.getName(Position.BEFORE), 'before'); equal(Position.getName(Position.AFTER), 'after'); equal(Position.getName(Position.INSIDE), 'inside'); equal(Position.getName(Position.NONE), 'none'); }); test('Position.nameToIndex', function() { equal(Position.nameToIndex('before'), Position.BEFORE); equal(Position.nameToIndex('after'), Position.AFTER); equal(Position.nameToIndex(''), 0); }); },{"./node":2,"./util":3}],2:[function(require,module,exports){ (function() { var Node, Position; Position = { getName: function(position) { return Position.strings[position - 1]; }, nameToIndex: function(name) { var i, _i, _ref; for (i = _i = 1, _ref = Position.strings.length; 1 <= _ref ? _i <= _ref : _i >= _ref; i = 1 <= _ref ? ++_i : --_i) { if (Position.strings[i - 1] === name) { return i; } } return 0; } }; Position.BEFORE = 1; Position.AFTER = 2; Position.INSIDE = 3; Position.NONE = 4; Position.strings = ['before', 'after', 'inside', 'none']; Node = (function() { function Node(o, is_root, node_class) { if (is_root == null) { is_root = false; } if (node_class == null) { node_class = Node; } this.setData(o); this.children = []; this.parent = null; if (is_root) { this.id_mapping = {}; this.tree = this; this.node_class = node_class; } } Node.prototype.setData = function(o) { var key, value, _results; if (typeof o !== 'object') { return this.name = o; } else { _results = []; for (key in o) { value = o[key]; if (key === 'label') { _results.push(this.name = value); } else { _results.push(this[key] = value); } } return _results; } }; Node.prototype.initFromData = function(data) { var addChildren, addNode; addNode = (function(_this) { return function(node_data) { _this.setData(node_data); if (node_data.children) { return addChildren(node_data.children); } }; })(this); addChildren = (function(_this) { return function(children_data) { var child, node, _i, _len; for (_i = 0, _len = children_data.length; _i < _len; _i++) { child = children_data[_i]; node = new _this.tree.node_class(''); node.initFromData(child); _this.addChild(node); } return null; }; })(this); addNode(data); return null; }; /* Create tree from data. Structure of data is: [ { label: 'node1', children: [ { label: 'child1' }, { label: 'child2' } ] }, { label: 'node2' } ] */ Node.prototype.loadFromData = function(data) { var node, o, _i, _len; this.removeChildren(); for (_i = 0, _len = data.length; _i < _len; _i++) { o = data[_i]; node = new this.tree.node_class(o); this.addChild(node); if (typeof o === 'object' && o.children) { node.loadFromData(o.children); } } return null; }; /* Add child. tree.addChild( new Node('child1') ); */ Node.prototype.addChild = function(node) { this.children.push(node); return node._setParent(this); }; /* Add child at position. Index starts at 0. tree.addChildAtPosition( new Node('abc'), 1 ); */ Node.prototype.addChildAtPosition = function(node, index) { this.children.splice(index, 0, node); return node._setParent(this); }; Node.prototype._setParent = function(parent) { this.parent = parent; this.tree = parent.tree; return this.tree.addNodeToIndex(this); }; /* Remove child. This also removes the children of the node. tree.removeChild(tree.children[0]); */ Node.prototype.removeChild = function(node) { node.removeChildren(); return this._removeChild(node); }; Node.prototype._removeChild = function(node) { this.children.splice(this.getChildIndex(node), 1); return this.tree.removeNodeFromIndex(node); }; /* Get child index. var index = getChildIndex(node); */ Node.prototype.getChildIndex = function(node) { return $.inArray(node, this.children); }; /* Does the tree have children? if (tree.hasChildren()) { // } */ Node.prototype.hasChildren = function() { return this.children.length !== 0; }; Node.prototype.isFolder = function() { return this.hasChildren() || this.load_on_demand; }; /* Iterate over all the nodes in the tree. Calls callback with (node, level). The callback must return true to continue the iteration on current node. tree.iterate( function(node, level) { console.log(node.name); // stop iteration after level 2 return (level <= 2); } ); */ Node.prototype.iterate = function(callback) { var _iterate; _iterate = function(node, level) { var child, result, _i, _len, _ref; if (node.children) { _ref = node.children; for (_i = 0, _len = _ref.length; _i < _len; _i++) { child = _ref[_i]; result = callback(child, level); if (result && child.hasChildren()) { _iterate(child, level + 1); } } return null; } }; _iterate(this, 0); return null; }; /* Move node relative to another node. Argument position: Position.BEFORE, Position.AFTER or Position.Inside // move node1 after node2 tree.moveNode(node1, node2, Position.AFTER); */ Node.prototype.moveNode = function(moved_node, target_node, position) { if (moved_node.isParentOf(target_node)) { return; } moved_node.parent._removeChild(moved_node); if (position === Position.AFTER) { return target_node.parent.addChildAtPosition(moved_node, target_node.parent.getChildIndex(target_node) + 1); } else if (position === Position.BEFORE) { return target_node.parent.addChildAtPosition(moved_node, target_node.parent.getChildIndex(target_node)); } else if (position === Position.INSIDE) { return target_node.addChildAtPosition(moved_node, 0); } }; /* Get the tree as data. */ Node.prototype.getData = function() { var getDataFromNodes; getDataFromNodes = (function(_this) { return function(nodes) { var data, k, node, tmp_node, v, _i, _len; data = []; for (_i = 0, _len = nodes.length; _i < _len; _i++) { node = nodes[_i]; tmp_node = {}; for (k in node) { v = node[k]; if ((k !== 'parent' && k !== 'children' && k !== 'element' && k !== 'tree') && Object.prototype.hasOwnProperty.call(node, k)) { tmp_node[k] = v; } } if (node.hasChildren()) { tmp_node.children = getDataFromNodes(node.children); } data.push(tmp_node); } return data; }; })(this); return getDataFromNodes(this.children); }; Node.prototype.getNodeByName = function(name) { var result; result = null; this.iterate(function(node) { if (node.name === name) { result = node; return false; } else { return true; } }); return result; }; Node.prototype.addAfter = function(node_info) { var child_index, node; if (!this.parent) { return null; } else { node = new this.tree.node_class(node_info); child_index = this.parent.getChildIndex(this); this.parent.addChildAtPosition(node, child_index + 1); return node; } }; Node.prototype.addBefore = function(node_info) { var child_index, node; if (!this.parent) { return null; } else { node = new this.tree.node_class(node_info); child_index = this.parent.getChildIndex(this); this.parent.addChildAtPosition(node, child_index); return node; } }; Node.prototype.addParent = function(node_info) { var child, new_parent, original_parent, _i, _len, _ref; if (!this.parent) { return null; } else { new_parent = new this.tree.node_class(node_info); new_parent._setParent(this.tree); original_parent = this.parent; _ref = original_parent.children; for (_i = 0, _len = _ref.length; _i < _len; _i++) { child = _ref[_i]; new_parent.addChild(child); } original_parent.children = []; original_parent.addChild(new_parent); return new_parent; } }; Node.prototype.remove = function() { if (this.parent) { this.parent.removeChild(this); return this.parent = null; } }; Node.prototype.append = function(node_info) { var node; node = new this.tree.node_class(node_info); this.addChild(node); return node; }; Node.prototype.prepend = function(node_info) { var node; node = new this.tree.node_class(node_info); this.addChildAtPosition(node, 0); return node; }; Node.prototype.isParentOf = function(node) { var parent; parent = node.parent; while (parent) { if (parent === this) { return true; } parent = parent.parent; } return false; }; Node.prototype.getLevel = function() { var level, node; level = 0; node = this; while (node.parent) { level += 1; node = node.parent; } return level; }; Node.prototype.getNodeById = function(node_id) { return this.id_mapping[node_id]; }; Node.prototype.addNodeToIndex = function(node) { if (node.id != null) { return this.id_mapping[node.id] = node; } }; Node.prototype.removeNodeFromIndex = function(node) { if (node.id != null) { return delete this.id_mapping[node.id]; } }; Node.prototype.removeChildren = function() { this.iterate((function(_this) { return function(child) { _this.tree.removeNodeFromIndex(child); return true; }; })(this)); return this.children = []; }; Node.prototype.getPreviousSibling = function() { var previous_index; if (!this.parent) { return null; } else { previous_index = this.parent.getChildIndex(this) - 1; if (previous_index >= 0) { return this.parent.children[previous_index]; } else { return null; } } }; Node.prototype.getNextSibling = function() { var next_index; if (!this.parent) { return null; } else { next_index = this.parent.getChildIndex(this) + 1; if (next_index < this.parent.children.length) { return this.parent.children[next_index]; } else { return null; } } }; Node.prototype.getNodesByProperty = function(key, value) { return this.filter(function(node) { return node[key] === value; }); }; Node.prototype.filter = function(f) { var result; result = []; this.iterate(function(node) { if (f(node)) { result.push(node); } return true; }); return result; }; Node.prototype.getNextNode = function(include_children) { var next_sibling; if (include_children == null) { include_children = true; } if (include_children && this.hasChildren() && this.is_open) { return this.children[0]; } else { if (!this.parent) { return null; } else { next_sibling = this.getNextSibling(); if (next_sibling) { return next_sibling; } else { return this.parent.getNextNode(false); } } } }; Node.prototype.getPreviousNode = function() { var previous_sibling; if (!this.parent) { return null; } else { previous_sibling = this.getPreviousSibling(); if (previous_sibling) { if (!previous_sibling.hasChildren() || !previous_sibling.is_open) { return previous_sibling; } else { return previous_sibling.getLastChild(); } } else { if (this.parent.parent) { return this.parent; } else { return null; } } } }; Node.prototype.getLastChild = function() { var last_child; if (!this.hasChildren()) { return null; } else { last_child = this.children[this.children.length - 1]; if (!last_child.hasChildren() || !last_child.is_open) { return last_child; } else { return last_child.getLastChild(); } } }; return Node; })(); module.exports = { Node: Node, Position: Position }; }).call(this); },{}],3:[function(require,module,exports){ (function() { var get_json_stringify_function, html_escape, indexOf, isInt, _indexOf; _indexOf = function(array, item) { var i, value, _i, _len; for (i = _i = 0, _len = array.length; _i < _len; i = ++_i) { value = array[i]; if (value === item) { return i; } } return -1; }; indexOf = function(array, item) { if (array.indexOf) { return array.indexOf(item); } else { return _indexOf(array, item); } }; isInt = function(n) { return typeof n === 'number' && n % 1 === 0; }; get_json_stringify_function = function() { var json_escapable, json_meta, json_quote, json_str, stringify; json_escapable = /[\\\"\x00-\x1f\x7f-\x9f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g; json_meta = { '\b': '\\b', '\t': '\\t', '\n': '\\n', '\f': '\\f', '\r': '\\r', '"': '\\"', '\\': '\\\\' }; json_quote = function(string) { json_escapable.lastIndex = 0; if (json_escapable.test(string)) { return '"' + string.replace(json_escapable, function(a) { var c; c = json_meta[a]; return (typeof c === 'string' ? c : '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4)); }) + '"'; } else { return '"' + string + '"'; } }; json_str = function(key, holder) { var i, k, partial, v, value, _i, _len; value = holder[key]; switch (typeof value) { case 'string': return json_quote(value); case 'number': if (isFinite(value)) { return String(value); } else { return 'null'; } case 'boolean': case 'null': return String(value); case 'object': if (!value) { return 'null'; } partial = []; if (Object.prototype.toString.apply(value) === '[object Array]') { for (i = _i = 0, _len = value.length; _i < _len; i = ++_i) { v = value[i]; partial[i] = json_str(i, value) || 'null'; } return (partial.length === 0 ? '[]' : '[' + partial.join(',') + ']'); } for (k in value) { if (Object.prototype.hasOwnProperty.call(value, k)) { v = json_str(k, value); if (v) { partial.push(json_quote(k) + ':' + v); } } } return (partial.length === 0 ? '{}' : '{' + partial.join(',') + '}'); } }; stringify = function(value) { return json_str('', { '': value }); }; return stringify; }; if (!((this.JSON != null) && (this.JSON.stringify != null) && typeof this.JSON.stringify === 'function')) { if (this.JSON == null) { this.JSON = {}; } this.JSON.stringify = get_json_stringify_function(); } html_escape = function(string) { return ('' + string).replace(/&/g, '&').replace(//g, '>').replace(/"/g, '"').replace(/'/g, ''').replace(/\//g, '/'); }; module.exports = { _indexOf: _indexOf, get_json_stringify_function: get_json_stringify_function, html_escape: html_escape, indexOf: indexOf, isInt: isInt }; }).call(this); },{}]},{},[1])