AutorÃa | Ultima modificación | Ver Log |
YUI.add('tree-sortable', function (Y, NAME) {/*jshint expr:true, onevar:false *//**Extension for `Tree` that makes nodes sortable.@module tree@submodule tree-sortable@main tree-sortable**//**Extension for `Tree` that makes nodes sortable.@class Tree.Sortable@constructor@param {Object} [config] Configuration options.@param {Function} [config.sortComparator] Default comparator function to usewhen sorting a node's children if the node itself doesn't have a customcomparator function. If not specified, insertion order will be used bydefault.@param {Boolean} [config.sortReverse=false] If `true`, node children will besorted in reverse (descending) order by default. Otherwise they'll be sortedin ascending order.@extensionfor Tree**//**Fired after a node's children are re-sorted.@event sort@param {Tree.Node} node Node whose children were sorted.@param {Boolean} reverse `true` if the children were sorted in reverse(descending) order, `false` otherwise.@param {String} src Source of the event.**/var EVT_SORT = 'sort';function Sortable() {}Sortable.prototype = {// -- Public Properties ----------------------------------------------------/**If `true`, node children will be sorted in reverse (descending) order bydefault. Otherwise they'll be sorted in ascending order.@property {Boolean} sortReverse@default false**/sortReverse: false,// -- Lifecycle ------------------------------------------------------------initializer: function (config) {this.nodeExtensions = this.nodeExtensions.concat(Y.Tree.Node.Sortable);if (config) {if (config.sortComparator) {this.sortComparator = config.sortComparator;}if ('sortReverse' in config) {this.sortReverse = config.sortReverse;}}},// -- Public Methods -------------------------------------------------------/**Sorts the children of every node in this tree.A `sort` event will be fired for each node whose children are sorted, whichcan get very noisy. If this is a large tree, you may want to set the`silent` option to `true` to suppress these events.@method sort@param {Object} [options] Options.@param {Boolean} [options.silent] If `true`, no `sort` events will befired.@param {Function} [options.sortComparator] Custom comparator function touse. If specified, this will become the new comparator function foreach node, overwriting any previous comparator function that was setfor the node.@param {Boolean} [options.sortReverse] If `true`, children will besorted in reverse (descending) order. Otherwise they'll be sorted inascending order. This will become each node's new sort order,overwriting any previous sort order that was set for the node.@param {String} [options.src] Source of the sort operation. Will bepassed along to the `sort` event facade.@chainable**/sort: function (options) {return this.sortNode(this.rootNode, Y.merge(options, {deep: true}));},/**Default comparator function to use when sorting a node's children if thenode itself doesn't have a custom comparator function.If not specified, insertion order will be used by default.@method sortComparator@param {Tree.Node} node Node being sorted.@return {Number|String} Value by which the node should be sorted relative toits siblings.**/sortComparator: function (node) {return node.index();},/**Sorts the children of the specified node.By default, only the node's direct children are sorted. To sort all nodes inthe hierarchy (children, children's children, etc.), set the `deep` optionto `true`. If this is a very deep hierarchy, you may also want to set`silent` to true to avoid generating a flood of `sort` events.@method sortNode@param {Tree.Node} node Node whose children should be sorted.@param {Object} [options] Options.@param {Boolean} [options.deep=false] If `true`, all of this node'schildren (and their children, and so on) will be traversed andre-sorted as well.@param {Boolean} [options.silent] If `true`, no `sort` event will befired.@param {Function} [options.sortComparator] Custom comparator function touse. If specified, this will become the node's new comparatorfunction, overwriting any previous comparator function that was setfor the node.@param {Boolean} [options.sortReverse] If `true`, children will besorted in reverse (descending) order. Otherwise they'll be sorted inascending order. This will become the node's new sort order,overwriting any previous sort order that was set for the node.@param {String} [options.src] Source of the sort operation. Will bepassed along to the `sort` event facade.@chainable**/sortNode: function (node, options) {// Nothing to do if the node has no children.if (!node.children.length) {return this;}options || (options = {});if (options.deep) {// Unset the `deep` option so we don't cause an infinite loop.options = Y.merge(options, {deep: false});var self = this;// Traverse and sort all nodes (including this one).this.traverseNode(node, function (nodeToSort) {self.sortNode(nodeToSort, options);});return this;}var comparator = this._getSortComparator(node, options),reverse;if ('sortReverse' in options) {reverse = node.sortReverse = options.sortReverse;} else if ('sortReverse' in node) {reverse = node.sortReverse;} else {reverse = this.sortReverse;}node.children.sort(Y.rbind(this._sort, this, comparator, reverse));node._isIndexStale = true;if (!options.silent) {this.fire(EVT_SORT, {node : node,reverse: !!reverse,src : options.src});}return this;},// -- Protected Methods ----------------------------------------------------/**Compares value _a_ to value _b_ for sorting purposes.Values are assumed to be the result of calling a sortComparator function.@method _compare@param {Mixed} a First value to compare.@param {Mixed} b Second value to compare.@return {Number} `-1` if _a_ should come before _b_, `0` if they'reequivalent, `1` if _a_ should come after _b_.@protected**/_compare: function (a, b) {return a < b ? -1 : (a > b ? 1 : 0);},/**Compares value _a_ to value _b_ for sorting purposes, but sorts them inreverse (descending) order.Values are assumed to be the result of calling a sortComparator function.@method _compareReverse@param {Mixed} a First value to compare.@param {Mixed} b Second value to compare.@return {Number} `-1` if _a_ should come before _b_, `0` if they'reequivalent, `1` if _a_ should come after _b_.@protected**/_compareReverse: function (a, b) {return b < a ? -1 : (b > a ? 1 : 0);},/**Overrides `Tree#_getDefaultNodeIndex()` to provide insertion-time sortingfor nodes inserted without an explicit index.@method _getDefaultNodeIndex@param {Tree.Node} parent Parent node.@param {Tree.Node} node Node being inserted.@param {Object} [options] Options passed to `insertNode()`.@return {Number} Index at which _node_ should be inserted into _parent_'s`children` array.@protected**/_getDefaultNodeIndex: function (parent, node) {/*jshint bitwise:false */var children = parent.children,comparator = this._getSortComparator(parent),max = children.length,min = 0,reverse = 'sortReverse' in parent ? parent.sortReverse : this.sortReverse;if (!max) {return max;}// Special case: if the sortComparator is the default sortComparator,// cheat and just return the first or last index of the children array.//// This is necessary because the default sortComparator relies on// the node's index, which is always -1 for uninserted nodes.if (comparator._unboundComparator === Sortable.prototype.sortComparator) {return reverse ? 0 : max;}var compare = reverse ? this._compareReverse : this._compare,needle = comparator(node);// Perform an iterative binary search to determine the correct position// for the node based on the return value of the comparator function.var middle;while (min < max) {middle = (min + max) >> 1; // Divide by two and discard remainder.if (compare(comparator(children[middle]), needle) < 0) {min = middle + 1;} else {max = middle;}}return min;},/**Returns a sort comparator function derived from the given _node_ and_options_, and bound to the correct `thisObj` based on where it was found.@method _getSortComparator@param {Tree.Node} node Node on which to look for a `sortComparator`function.@param {Object} [options] Options object on which to look for a`sortComparator` function.@return {Function} Properly bound sort comparator function.@protected**/_getSortComparator: function (node, options) {var boundComparator,comparator,thisObj;if (options && options.sortComparator) {comparator = node.sortComparator = options.sortComparator;} else if (node.sortComparator) {comparator = node.sortComparator;thisObj = node;} else {comparator = this.sortComparator;thisObj = this;}boundComparator = function () {return comparator.apply(thisObj, arguments);};boundComparator._unboundComparator = comparator;return boundComparator;},/**Array sort function used by `sortNode()` to re-sort a node's children.@method _sort@param {Tree.Node} a First node to compare.@param {Tree.Node} b Second node to compare.@param {Function} comparator Comparator function.@param {Boolean} [reverse=false] If `true`, this will be a reverse(descending) comparison.@return {Number} `-1` if _a_ is less than _b_, `0` if equal, `1` if greater.@protected**/_sort: function (a, b, comparator, reverse) {return this[reverse ? '_compareReverse' : '_compare'](comparator(a), comparator(b));}};Y.Tree.Sortable = Sortable;/**@module tree@submodule tree-sortable**//**`Tree.Node` extension that adds methods useful for nodes in trees that use the`Tree.Sortable` extension.@class Tree.Node.Sortable@constructor@extensionfor Tree.Node**/function NodeSortable() {}NodeSortable.prototype = {/**Sorts this node's children.@method sort@param {Object} [options] Options.@param {Boolean} [options.silent] If `true`, no `sort` event will befired.@param {Function} [options.sortComparator] Custom comparator function touse. If specified, this will become the node's new comparatorfunction, overwriting any previous comparator function that was setfor the node.@param {Boolean} [options.sortReverse] If `true`, children will besorted in reverse (descending) order. Otherwise they'll be sorted inascending order. This will become the node's new sort order,overwriting any previous sort order that was set for the node.@param {String} [options.src] Source of the sort operation. Will bepassed along to the `sort` event facade.@chainable**/sort: function (options) {this.tree.sortNode(this, options);return this;}};Y.Tree.Node.Sortable = NodeSortable;}, '3.18.1', {"requires": ["tree"]});