updating jquery treeview version to 1.4.1.
[openemr.git] / library / js / jquery.treeview-1.4.1 / jquery.treeview.sortable.js
blob5d09dd0f970e8e2801b99f256c8e00c1bafa479e
1 /*
2  * jQuery UI Sortable
3  *
4  * Copyright (c) 2008 Paul Bakaus
5  * Dual licensed under the MIT (MIT-LICENSE.txt)
6  * and GPL (GPL-LICENSE.txt) licenses.
7  * 
8  * http://docs.jquery.com/UI/Sortables
9  *
10  * Depends:
11  *   ui.base.js
12  *
13  * Revision: $Id: ui.sortable.js 5262 2008-04-17 13:13:51Z paul.bakaus $
14  */
15 ;(function($) {
17         if (window.Node && Node.prototype && !Node.prototype.contains) {
18                 Node.prototype.contains = function (arg) {
19                         return !!(this.compareDocumentPosition(arg) & 16);
20                 };
21         }
24         $.widget("ui.sortableTree", $.extend($.ui.mouse, {
25                 init: function() {
27                         //Initialize needed constants
28                         var self = this, o = this.options;
29                         this.containerCache = {};
30                         this.element.addClass("ui-sortableTree");
31                         
32                         //Get the items
33                         this.refresh();
34                         
35                         //Let's determine the parent's offset
36                         if(!(/(relative|absolute|fixed)/).test(this.element.css('position'))) this.element.css('position', 'relative');
37                         this.offset = this.element.offset();
38         
39                         //Initialize mouse events for interaction
40                         this.mouseInit();
41                         
42                         //Prepare cursorAt
43                         if(o.cursorAt && o.cursorAt.constructor == Array)
44                                 o.cursorAt = { left: o.cursorAt[0], top: o.cursorAt[1] };
46                 },
47                 plugins: {},
48                 ui: function(inst) {
49                         return {
50                                 helper: (inst || this)["helper"],
51                                 position: (inst || this)["position"].current,
52                                 absolutePosition: (inst || this)["position"].absolute,
53                                 instance: this,
54                                 options: this.options,
55                                 element: this.element,
56                                 item: (inst || this)["currentItem"],
57                                 sender: inst ? inst.element : null
58                         };              
59                 },
60                 propagate: function(n,e,inst) {
61                         $.ui.plugin.call(this, n, [e, this.ui(inst)]);
62                         this.element.triggerHandler(n == "sort" ? n : "sort"+n, [e, this.ui(inst)], this.options[n]);
63                 },
64                 serialize: function(o) {
65                         
66                         var items = $(this.options.items, this.element).not('.ui-sortableTree-helper'); //Only the items of the sortable itself
67                         var str = []; o = o || {};
68                         
69                         items.each(function() {
70                                 var res = ($(this).attr(o.attribute || 'id') || '').match(o.expression || (/(.+)[-=_](.+)/));
71                                 if(res) str.push((o.key || res[1])+'[]='+(o.key ? res[1] : res[2]));
72                         });
73                         
74                         return str.join('&');
75                         
76                 },
77                 toArray: function(attr) {
78                         var items = $(this.options.items, this.element).not('.ui-sortableTree-helper'); //Only the items of the sortable itself
79                         var ret = [];
81                         items.each(function() { ret.push($(this).attr(attr || 'id')); });
82                         return ret;
83                 },
84                 enable: function() {
85                         this.element.removeClass("ui-sortableTree-disabled");
86                         this.options.disabled = false;
87                 },
88                 disable: function() {
89                         this.element.addClass("ui-sortableTree-disabled");
90                         this.options.disabled = true;
91                 },
92                 /* Be careful with the following core functions */
93                 intersectsWith: function(item) {
94                                         
95                         var x1 = this.position.absolute.left - 10, x2 = x1 + 10,
96                             y1 = this.position.absolute.top - 10, y2 = y1 + 10;
97                         var l = item.left, r = l + item.width, 
98                             t = item.top,  b = t + item.height;
99                         
100                         return (   l < x1 + (this.helperProportions.width  / 2)    // Right Half
101                                 &&     x2 - (this.helperProportions.width  / 2) < r    // Left Half
102                                 && t < y1 + (this.helperProportions.height / 2)        // Bottom Half
103                                 &&     y2 - (this.helperProportions.height / 2) < b ); // Top Half
104                         
105                 },
106                 intersectsWithEdge: function(item) {    
107                         var y1 = this.position.absolute.top - 10, y2 = y1 + 10;
108                         var t = item.top,  b = t + item.height;
110                         if(!this.intersectsWith(item.item.parents(".ui-sortableTree").data("sortableTree").containerCache)) return false;
112                         if (!( t < y1 + (this.helperProportions.height / 2)        // Bottom Half
113                                 &&     y2 - (this.helperProportions.height / 2) < b )) return false; // Top Half
115                         if(y2 > t && y1 < t) return 1; //Crosses top edge
116                         if(y1 < b && y2 > b) return 2; //Crosses bottom edge
117                         
118                         return false;
119                         
120                 },
121                 refresh: function() {
122                         this.refreshItems();
123                         this.refreshPositions();
124                 },
125                 refreshItems: function() {
126                         
127                         this.items = [];
128                         this.containers = [this];
129                         var items = this.items;
130                         var queries = [$(this.options.items, this.element)];
131                         
132                         if(this.options.connectWith) {
133                                 for (var i = this.options.connectWith.length - 1; i >= 0; i--){
134                                         var cur = $(this.options.connectWith[i]);
135                                         for (var j = cur.length - 1; j >= 0; j--){
136                                                 var inst = $.data(cur[j], 'sortableTree');
137                                                 if(inst && !inst.options.disabled) {
138                                                         queries.push($(inst.options.items, inst.element));
139                                                         this.containers.push(inst);
140                                                 }
141                                         };
142                                 };
143                         }
145                         for (var i = queries.length - 1; i >= 0; i--){
146                                 queries[i].each(function() {
147                                         $.data(this, 'sortableTree-item', true); // Data for target checking (mouse manager)
148                                         items.push({
149                                                 item: $(this),
150                                                 width: 0, height: 0,
151                                                 left: 0, top: 0
152                                         });
153                                 });
154                         };
156                 },
157                 refreshPositions: function(fast) {
158                         for (var i = this.items.length - 1; i >= 0; i--){
159                                 if(!fast) this.items[i].height                  = this.items[i].item.outerHeight();
160                                 this.items[i].top                                               = this.items[i].item.offset().top;
161                         };
162                         for (var i = this.containers.length - 1; i >= 0; i--){
163                                 var p =this.containers[i].element.offset();
164                                 this.containers[i].containerCache.left  = p.left;
165                                 this.containers[i].containerCache.top   = p.top;
166                                 this.containers[i].containerCache.width = this.containers[i].element.outerWidth();
167                                 this.containers[i].containerCache.height= this.containers[i].element.outerHeight();
168                         };
169                 },
170                 destroy: function() {
172                         this.element
173                                 .removeClass("ui-sortableTree ui-sortableTree-disabled")
174                                 .removeData("sortableTree")
175                                 .unbind(".sortableTree");
176                         this.mouseDestroy();
177                         
178                         for ( var i = this.items.length - 1; i >= 0; i-- )
179                                 this.items[i].item.removeData("sortableTree-item");
180                                 
181                 },
182                 contactContainers: function(e) {
183                         for (var i = this.containers.length - 1; i >= 0; i--){
185                                 if(this.intersectsWith(this.containers[i].containerCache)) {
186                                         if(!this.containers[i].containerCache.over) {
187                                                 
188                                                 if(this.currentContainer != this.containers[i]) {
189                                                         
190                                                         //When entering a new container, we will find the item with the least distance and append our item near it
191                                                         var dist = 10000; var itemWithLeastDistance = null; var base = this.position.absolute.top;
192                                                         for (var j = this.items.length - 1; j >= 0; j--) {
193                                                                 if(!this.containers[i].element[0].contains(this.items[j].item[0])) continue;
194                                                                 var cur = this.items[j].top;
195                                                                 if(Math.abs(cur - base) < dist) {
196                                                                         dist = Math.abs(cur - base); itemWithLeastDistance = this.items[j];
197                                                                 }
198                                                         }
199                                                         
200                                                         itemWithLeastDistance ? this.rearrange(e, itemWithLeastDistance) : this.rearrange(e, null, this.containers[i].element);
201                                                         this.propagate("change", e); //Call plugins and callbacks
202                                                         this.containers[i].propagate("change", e, this); //Call plugins and callbacks
203                                                         this.currentContainer = this.containers[i];
205                                                 }
206                                                 
207                                                 this.containers[i].propagate("over", e, this);
208                                                 this.containers[i].containerCache.over = 1;
209                                         }
210                                 } else {
211                                         if(this.containers[i].containerCache.over) {
212                                                 this.containers[i].propagate("out", e, this);
213                                                 this.containers[i].containerCache.over = 0;
214                                         }
215                                 }
216                                 
217                         };                      
218                 },
219                 mouseStart: function(e,el) {
221                         if(this.options.disabled || this.options.type == 'static') return false;
223                         //Find out if the clicked node (or one of its parents) is a actual item in this.items
224                         var currentItem = null, nodes = $(e.target).parents().each(function() { 
225                                 if($.data(this, 'sortableTree-item')) {
226                                         currentItem = $(this);
227                                         return false;
228                                 }
229                         });
230                         if($.data(e.target, 'sortableTree-item')) currentItem = $(e.target);
231                         
232                         if(!currentItem) return false;  
233                         if(this.options.handle) {
234                                 var validHandle = false;
235                                 $(this.options.handle, currentItem).each(function() { if(this == e.target) validHandle = true; });
236                                 if(!validHandle) return false;
237                         }
238                                 
239                         this.currentItem = currentItem;
240                         
241                         var o = this.options;
242                         this.currentContainer = this;
243                         this.refresh();
245                         //Create and append the visible helper
246                         this.helper = typeof o.helper == 'function' ? $(o.helper.apply(this.element[0], [e, this.currentItem])) : this.currentItem.clone();
247                         if(!this.helper.parents('body').length) this.helper.appendTo("body"); //Add the helper to the DOM if that didn't happen already
248                         this.helper.css({ position: 'absolute', clear: 'both' }).addClass('ui-sortableTree-helper'); //Position it absolutely and add a helper class
249                         
250                         //Prepare variables for position generation
251                         $.extend(this, {
252                                 offsetParent: this.helper.offsetParent(),
253                                 offsets: { absolute: this.currentItem.offset() }
254                         });
256                         //Save the first time position
257                         $.extend(this, {
258                                 position: {
259                                         current: { left: e.pageX, top: e.pageY },
260                                         absolute: { left: e.pageX, top: e.pageY },
261                                         dom: this.currentItem.prev()[0]
262                                 },
263                                 clickOffset: { left: -5, top: -5 }
264                         });
266                         this.propagate("start", e); //Call plugins and callbacks
267                         this.helperProportions = { width: this.helper.outerWidth(), height: this.helper.outerHeight() }; //Save and store the helper proportions
269                         for (var i = this.containers.length - 1; i >= 0; i--) {
270                                 this.containers[i].propagate("activate", e, this);
271                         } //Post 'activate' events to possible containers
272                         
273                         //Prepare possible droppables
274                         if($.ui.ddmanager) $.ui.ddmanager.current = this;
275                         if ($.ui.ddmanager && !o.dropBehaviour) $.ui.ddmanager.prepareOffsets(this, e);
277                         this.dragging = true;
278                         return true;
279                         
280                 },
281                 mouseStop: function(e) {
283                         if(this.newPositionAt) this.options.sortIndication.remove.call(this.currentItem, this.newPositionAt); //remove sort indicator
284                         this.propagate("stop", e); //Call plugins and trigger callbacks
286                         //If we are using droppables, inform the manager about the drop
287                         var dropped = ($.ui.ddmanager && !this.options.dropBehaviour) ? $.ui.ddmanager.drop(this, e) : false;
288                         if(!dropped && this.newPositionAt) this.newPositionAt[this.direction == 'down' ? 'before' : 'after'](this.currentItem); //Append to element to its new position
289                         
290                         if(this.position.dom != this.currentItem.prev()[0]) this.propagate("update", e); //Trigger update callback if the DOM position has changed
291                         if(!this.element[0].contains(this.currentItem[0])) { //Node was moved out of the current element
292                                 this.propagate("remove", e);
293                                 for (var i = this.containers.length - 1; i >= 0; i--){
294                                         if(this.containers[i].element[0].contains(this.currentItem[0])) {
295                                                 this.containers[i].propagate("update", e, this);
296                                                 this.containers[i].propagate("receive", e, this);
297                                         }
298                                 };
299                         };
300                         
301                         //Post events to containers
302                         for (var i = this.containers.length - 1; i >= 0; i--){
303                                 this.containers[i].propagate("deactivate", e, this);
304                                 if(this.containers[i].containerCache.over) {
305                                         this.containers[i].propagate("out", e, this);
306                                         this.containers[i].containerCache.over = 0;
307                                 }
308                         }
309                         
310                         this.dragging = false;
311                         if(this.cancelHelperRemoval) return false;
312                         this.helper.remove();
314                         return false;
315                         
316                 },
317                 mouseDrag: function(e) {
319                         //Compute the helpers position
320                         this.position.current = { top: e.pageY + 5, left: e.pageX + 5 };
321                         this.position.absolute = { left: e.pageX + 5, top: e.pageY + 5 };
323                         //Interconnect with droppables
324                         if($.ui.ddmanager) $.ui.ddmanager.drag(this, e);
325                         var intersectsWithDroppable = false;
326                         $.each($.ui.ddmanager.droppables, function() {
327                                 if(this.isover) intersectsWithDroppable = true;
328                         });
330                         //Rearrange
331                         if(intersectsWithDroppable) {
332                                 if(this.newPositionAt) this.options.sortIndication.remove.call(this.currentItem, this.newPositionAt);
333                         } else {
334                                 for (var i = this.items.length - 1; i >= 0; i--) {
335                                         
336                                         if(this.currentItem[0].contains(this.items[i].item[0])) continue;
337                                         
338                                         var intersection = this.intersectsWithEdge(this.items[i]);
339                                         if(!intersection) continue;
340         
341                                         this.direction = intersection == 1 ? "down" : "up";
342                                         this.rearrange(e, this.items[i]);
343                                         this.propagate("change", e); //Call plugins and callbacks
344                                         break;
345                                 }
346                         }
347                         
348                         //Post events to containers
349                         this.contactContainers(e);
351                         this.propagate("sort", e); //Call plugins and callbacks
352                         this.helper.css({ left: this.position.current.left+'px', top: this.position.current.top+'px' }); // Stick the helper to the cursor
353                         return false;
354                         
355                 },
356                 rearrange: function(e, i, a) {
357                         if(i) {
358                                 if(this.newPositionAt) this.options.sortIndication.remove.call(this.currentItem, this.newPositionAt);
359                                 this.newPositionAt = i.item;
360                                 this.options.sortIndication[this.direction].call(this.currentItem, this.newPositionAt);
361                         } else {
362                                 //Append
363                         }
364                 }
365         }));
366         
367         $.extend($.ui.sortableTree, {
368                 defaults: {
369                         items: '> *',
370                         zIndex: 1000,
371                         distance: 1
372                 },
373                 getter: "serialize toArray"
374         });
378 })(jQuery);