123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455 |
- /*!
- * Modified jQuery treeTable plugin
- * ================================
- * Based on jQuery treeTable Plugin 2.3.0
- *
- * http://ludo.cubicphuse.nl/jquery-plugins/treeTable/doc/
- *
- * Copyright 2011, Ludo van den Boom
- * Dual licensed under the MIT or GPL Version 2 licenses.
- *
- * Modifications:
- * - remove option clickableNodeNames
- * + add option clickableElement:
- * Specify element in row node on which expander event will be listened for.
- * (default) If not set event will be listened on the row itself.
- * + add option doubleclickMode:
- * If true, listen for double click event.
- * (default) If false, listen for click event.
- * + add option initialRootState:
- * Overwrites initialState setting for root nodes. Can be "expanded" or "collapsed".
- * (default) "collapsed"
- * + add option initialIndent:
- * Global indent modifier.
- * (default) 0
- * + add function $.fn.moveBranch:
- * Move +this+ node's entire branch before/after target element.
- * (arg) +where+ >> where to move the branch ("before" or "after")
- * (arg) +element+ >> target element
- * + add function $.fn.selectBranch:
- * Select +this+ node's entire branch.
- * + add function $.fn.branchLast:
- * Select +this+ node's entire branch last node.
- * + add function $.fn.nodeParent:
- * Return +this+ node's parent.
- * + add callback onNodeInit
- * (arg) +node+ >> initialized node
- * (arg) +expandable+ >> is node expandable?
- * (arg) +isRootNode+ >> is node a root node?
- * + add callback onNodeReinit
- * (arg) +node+ >> reinitialized node
- * (arg) +expandable+ >> is node expandable?
- * (arg) +isRootNode+ >> is node a root node?
- * Modified internal functions:
- * + added selectBranch function
- * + added branchLast function
- * - removed move function
- * + added insert function
- * + modified initialize function
- * + added reinitialize function
- * + modified indent function
- * + modified getPaddingLeft function
- *
- * Modified 2013, loostro:
- * Released under the MIT license.
- */
- (function($) {
- // Helps to make options available to all functions
- // TODO: This gives problems when there are both expandable and non-expandable
- // trees on a page. The options shouldn't be global to all these instances!
- var options;
- var defaultPaddingLeft;
- var persistStore;
-
- $.fn.treeTable = function(opts) {
- options = $.extend({}, $.fn.treeTable.defaults, opts);
-
- if(options.persist) {
- persistStore = new Persist.Store(options.persistStoreName);
- }
-
- return this.each(function() {
- $(this).addClass("treeTable").find("tbody tr").each(function() {
- // Skip initialized nodes.
- if (!$(this).hasClass('initialized')) {
- var isRootNode = ($(this)[0].className.search(options.childPrefix) == -1);
-
- // To optimize performance of indentation, I retrieve the padding-left
- // value of the first root node. This way I only have to call +css+
- // once.
- if (isRootNode && isNaN(defaultPaddingLeft)) {
- defaultPaddingLeft = options.initialIndent + parseInt($($(this).children("td")[options.treeColumn]).css('padding-left'), 10);
- }
-
- // Set child nodes to initial state if we're in expandable mode.
- if(!isRootNode && options.expandable && options.initialState == "collapsed") {
- $(this).addClass('ui-helper-hidden');
- }
-
- // If we're not in expandable mode, initialize all nodes.
- // If we're in expandable mode, only initialize root nodes.
- if(!options.expandable || isRootNode) {
- initialize($(this));
- }
- }
- });
- });
- };
-
- $.fn.treeTable.defaults = {
- childPrefix: "child-of-",
- clickableElement: false,
- doubleclickMode: false,
- expandable: true,
- indent: 19,
- initialIndent: 0,
- initialState: "collapsed",
- initialRootState: "collapsed",
- onNodeShow: null,
- onNodeHide: null,
- onExpandableInit: null,
- onNonExpandableInit: null,
- treeColumn: 0,
- persist: false,
- persistStoreName: 'treeTable',
- stringExpand: "Expand",
- stringCollapse: "Collapse"
- };
-
- //Expand all nodes
- $.fn.expandAll = function() {
- $(this).find("tr").each(function() {
- $(this).expand();
- });
- };
-
- //Collapse all nodes
- $.fn.collapseAll = function() {
- $(this).find("tr").each(function() {
- $(this).collapse();
- });
- };
-
- // Recursively hide all node's children in a tree
- $.fn.collapse = function() {
- return this.each(function() {
- $(this).removeClass("expanded").addClass("collapsed");
-
- if (options.persist) {
- persistNodeState($(this));
- }
-
- childrenOf($(this)).each(function() {
- if(!$(this).hasClass("collapsed")) {
- $(this).collapse();
- }
-
- $(this).addClass('ui-helper-hidden');
-
- if($.isFunction(options.onNodeHide)) {
- options.onNodeHide.call(this);
- }
-
- });
- });
- };
-
- // Recursively show all node's children in a tree
- $.fn.expand = function() {
- return this.each(function() {
- $(this).removeClass("collapsed").addClass("expanded");
-
- if (options.persist) {
- persistNodeState($(this));
- }
-
- childrenOf($(this)).each(function() {
- initialize($(this));
-
- if($(this).is(".expanded.parent")) {
- $(this).expand();
- }
-
- $(this).removeClass('ui-helper-hidden');
-
- if($.isFunction(options.onNodeShow)) {
- options.onNodeShow.call(this);
- }
- });
- });
- };
-
- // Reveal a node by expanding all ancestors
- $.fn.reveal = function() {
- $(ancestorsOf($(this)).reverse()).each(function() {
- initialize($(this));
- $(this).expand().show();
- });
-
- return this;
- };
-
- // Add an entire branch to +destination+
- $.fn.appendBranchTo = function(destination) {
- var node = $(this);
- var parent = parentOf(node);
- var target = $(destination);
-
- var ancestorNames = $.map(ancestorsOf(target), function(a) { return a.id; });
-
- // Conditions:
- // 1: +node+ should not be inserted in a location in a branch if this would
- // result in +node+ being an ancestor of itself.
- // 2: +node+ should not have a parent OR the destination should not be the
- // same as +node+'s current parent (this last condition prevents +node+
- // from being moved to the same location where it already is).
- // 3: +node+ should not be inserted as a child of +node+ itself.
- if($.inArray(node[0].id, ancestorNames) == -1 && (!parent || (target.id != parent[0].id)) && target.id != node[0].id) {
- insert(node, 'after', target); // Move nodes to new location
- if(parent) { node.removeClass(options.childPrefix + parent[0].id); } // Remove parent
- node.addClass(options.childPrefix + target[0].id); // Set new parent
- indent(node, ancestorsOf(node).length * options.indent); // Set new indentation
- }
-
- return this;
- };
-
- // Move +this+ node's entire branch before/after target +element+.
- $.fn.moveBranch = function(where, element) {
- // use appendBranchTo to handle 'in' action
- if(where == 'in') { $(this).appendBranchTo(element); return; }
- // sanity check
- if($.inArray(where, ['before','after']) == -1) { return; }
-
- var node = $(this);
- var parent = parentOf(node);
-
- var target = $(element);
- var targetParent = parentOf(target);
-
- var ancestorNames = $.map(ancestorsOf(target), function(a) { return a.id; });
-
- // Conditions:
- // 1: +node+ should not be inserted in a location in a branch if this would
- // result in +node+ being an ancestor of itself.
- // 2: +node+ should not be inserted before/after itself.
- if($.inArray(node[0].id, ancestorNames) == -1 && target[0].id != node[0].id) {
- insert(node, where, target); // Move nodes to new location
- if(parent) { node.removeClass(options.childPrefix + parent[0].id); } // Remove parent
- if(targetParent) { node.addClass(options.childPrefix + targetParent[0].id); } // Set new parent
- indent(node, ancestorsOf(node).length * options.indent); // Set new indentation
- }
-
- return this;
- };
-
- // Add reverse() function from JS Arrays
- $.fn.reverse = function() {
- return this.pushStack(this.get().reverse(), arguments);
- };
-
- // Toggle an entire branch
- $.fn.toggleBranch = function() {
- if($(this).hasClass("collapsed")) {
- $(this).expand();
- } else {
- $(this).collapse();
- }
-
- return this;
- };
-
- // Get node's parent
- $.fn.nodeParent = function () {
- var $node = $(this);
- var match = $node[0].className.match(new RegExp(options.childPrefix+'[^\\s]+', 'i'));
- return (match) ? $('#node-'+match[0].substring(14)) : null;
- }
-
- // Reinitialize node
- $.fn.nodeReinitialize = function() {
- reinitialize($(this));
-
- return this;
- };
-
- // Select an entire branch
- $.fn.selectBranch = function() {
- return selectBranch(this);
- };
-
- // Select branch last node
- $.fn.branchLast = function() {
- return branchLast(this);
- };
-
- // === Private functions
-
- function ancestorsOf(node) {
- var ancestors = [];
- while(node = parentOf(node)) {
- ancestors[ancestors.length] = node[0];
- }
- return ancestors;
- };
-
- function childrenOf(node) {
- return $(node).siblings("tr." + options.childPrefix + node[0].id);
- };
-
- // note: this function assumes that
- // last node of branch is the one with the highest index
- function branchLast(node) {
- return (childrenOf(node).length) ? branchLast(childrenOf(node).last()) : $(node);
- };
-
- // note: this function assumes that
- // all nodes between node and branchLast(node) belong to this branch
- // that is true after initializing treeTable and as long as you use only
- // provided API to move treeTable rows
- //
- // If node has no children, return that node
- function selectBranch(node) {
- var nodelast = branchLast(node);
- return (node[0].id === nodelast[0].id) ? node : node.nextUntil(nodelast).addBack().add(nodelast);
- };
-
- function getPaddingLeft(node) {
- return ancestorsOf(node).length * options.indent;
- }
-
- function indent(node, value) {
- var cell = $(node.children("td")[options.treeColumn]);
- cell[0].style.paddingLeft = options.initialIndent + value + 'px';
-
- childrenOf(node).each(function() {
- indent($(this), value + options.indent);
- });
- };
-
- function initialize(node) {
- if(!node.hasClass("initialized")) {
- node.addClass("initialized");
-
- var isRootNode = (node[0].className.search(options.childPrefix) == -1);
- var childNodes = childrenOf(node);
- var expandable = childNodes.length > 0;
-
- if(!node.hasClass("parent") && expandable) {
- node.addClass("parent");
- }
-
- if($.isFunction(options.onNodeInit)) {
- options.onNodeInit.call(this, node, expandable, isRootNode);
- }
-
- if(expandable) {
- indent(node, getPaddingLeft(node));
-
- if(options.expandable) {
- var handle = (options.clickableElement) ? node.find(options.clickableElement) : node;
- handle.attr('title', options.stringExpand).addClass('expander');
-
- handle.on((options.doubleclickMode) ? 'dblclick' : 'click', function(e){
- e.preventDefault;
- node.toggleBranch();
- });
-
- if (options.persist && getPersistedNodeState(node)) {
- node.addClass('expanded');
- }
-
- // Check for a class set explicitly by the user, otherwise set the default class
- if(!(node.hasClass("expanded") || node.hasClass("collapsed"))) {
- (isRootNode) ? node.addClass(options.initialRootState) : node.addClass(options.initialState);
- }
-
- if(node.hasClass("expanded")) {
- node.expand();
- }
- }
- }
- }
- };
-
- function reinitialize(node) {
- if(node.hasClass("initialized")) {
- node.removeClass('initialized').removeClass('parent')
- .removeClass('expanded').removeClass('collapsed');
-
- var isRootNode = (node[0].className.search(options.childPrefix) == -1);
- var childNodes = childrenOf(node);
- var expandable = childNodes.length > 0;
-
- if(options.expandable) {
- var handle = (options.clickableElement) ? node.find(options.clickableElement) : node;
- handle.removeAttr('title').removeClass('expander');
-
- handle.off((options.doubleclickMode) ? 'dblclick' : 'click');
- }
-
- if($.isFunction(options.onNodeReinit)) {
- options.onNodeReinit.call(this, node, expandable, isRootNode);
- }
-
- if(expandable) { node.addClass('expanded'); }
- initialize(node);
- }
- };
-
- // note: this function assumes that
- // all nodes between node and branchLast(node) belong to this branch
- // that is true after initializing treeTable and as long as you use only
- // provided API to move treeTable rows
- function insert(node, where, target) {
- if(where == 'before') {
- // no problems here
- // simply insert before +target+
- selectBranch(node).insertBefore(target);
- }
- if(where == 'after') {
- // target may have children
- // insert after +target+ last child
- var targetlast = branchLast(target);
- selectBranch(node).insertAfter(targetlast);
- }
- };
-
- function parentOf(node) {
- var classNames = node[0].className.split(' ');
-
- for(var key=0; key<classNames.length; key++) {
- if(classNames[key].match(options.childPrefix)) {
- return $(node).siblings("#" + classNames[key].substring(options.childPrefix.length));
- }
- }
-
- return null;
- };
-
- //saving state functions, not critical, so will not generate alerts on error
- function persistNodeState(node) {
- if(node.hasClass('expanded')) {
- try {
- persistStore.set(node.attr('id'), '1');
- } catch (err) {
-
- }
- } else {
- try {
- persistStore.remove(node.attr('id'));
- } catch (err) {
-
- }
- }
- }
-
- function getPersistedNodeState(node) {
- try {
- return persistStore.get(node.attr('id')) == '1';
- } catch (err) {
- return false;
- }
- }
- })(jQuery);
-
|