pcapr-local 0.1.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (203) hide show
  1. data/.document +5 -0
  2. data/LICENSE.txt +20 -0
  3. data/README.md +64 -0
  4. data/Rakefile +57 -0
  5. data/VERSION +1 -0
  6. data/bin/pcap2par +49 -0
  7. data/bin/startpcapr +40 -0
  8. data/bin/stoppcapr +33 -0
  9. data/bin/xtractr +5 -0
  10. data/lib/environment.rb +106 -0
  11. data/lib/exe/xtractr +0 -0
  12. data/lib/mu/pcap.rb +110 -0
  13. data/lib/mu/pcap/ethernet.rb +148 -0
  14. data/lib/mu/pcap/header.rb +75 -0
  15. data/lib/mu/pcap/io_pair.rb +67 -0
  16. data/lib/mu/pcap/io_wrapper.rb +76 -0
  17. data/lib/mu/pcap/ip.rb +61 -0
  18. data/lib/mu/pcap/ipv4.rb +257 -0
  19. data/lib/mu/pcap/ipv6.rb +148 -0
  20. data/lib/mu/pcap/packet.rb +104 -0
  21. data/lib/mu/pcap/pkthdr.rb +155 -0
  22. data/lib/mu/pcap/reader.rb +61 -0
  23. data/lib/mu/pcap/reader/http_family.rb +170 -0
  24. data/lib/mu/pcap/sctp.rb +367 -0
  25. data/lib/mu/pcap/sctp/chunk.rb +123 -0
  26. data/lib/mu/pcap/sctp/chunk/data.rb +134 -0
  27. data/lib/mu/pcap/sctp/chunk/init.rb +100 -0
  28. data/lib/mu/pcap/sctp/chunk/init_ack.rb +68 -0
  29. data/lib/mu/pcap/sctp/parameter.rb +110 -0
  30. data/lib/mu/pcap/sctp/parameter/ip_address.rb +48 -0
  31. data/lib/mu/pcap/stream_packetizer.rb +72 -0
  32. data/lib/mu/pcap/tcp.rb +505 -0
  33. data/lib/mu/pcap/udp.rb +69 -0
  34. data/lib/mu/scenario/pcap.rb +164 -0
  35. data/lib/mu/scenario/pcap/fields.rb +50 -0
  36. data/lib/mu/scenario/pcap/rtp.rb +71 -0
  37. data/lib/pcapr_local.rb +159 -0
  38. data/lib/pcapr_local/config.rb +336 -0
  39. data/lib/pcapr_local/db.rb +197 -0
  40. data/lib/pcapr_local/scanner.rb +250 -0
  41. data/lib/pcapr_local/server.rb +178 -0
  42. data/lib/pcapr_local/www/favicon.ico +0 -0
  43. data/lib/pcapr_local/www/favicon.png +0 -0
  44. data/lib/pcapr_local/www/home/index.html +138 -0
  45. data/lib/pcapr_local/www/static/image/16x16/Cancel.png +0 -0
  46. data/lib/pcapr_local/www/static/image/16x16/Cancel.png.1 +0 -0
  47. data/lib/pcapr_local/www/static/image/16x16/Download.png +0 -0
  48. data/lib/pcapr_local/www/static/image/16x16/Folder3.png +0 -0
  49. data/lib/pcapr_local/www/static/image/16x16/Full Size.png +0 -0
  50. data/lib/pcapr_local/www/static/image/16x16/Minus.png +0 -0
  51. data/lib/pcapr_local/www/static/image/16x16/Plus.png +0 -0
  52. data/lib/pcapr_local/www/static/image/16x16/Search.png +0 -0
  53. data/lib/pcapr_local/www/static/image/16x16/User.png +0 -0
  54. data/lib/pcapr_local/www/static/image/48x48/Phone.png +0 -0
  55. data/lib/pcapr_local/www/static/image/48x48/Video.png +0 -0
  56. data/lib/pcapr_local/www/static/image/bar-orange.gif +0 -0
  57. data/lib/pcapr_local/www/static/image/beta.png +0 -0
  58. data/lib/pcapr_local/www/static/image/bg.png +0 -0
  59. data/lib/pcapr_local/www/static/image/blockquote.png +0 -0
  60. data/lib/pcapr_local/www/static/image/body-bg.png +0 -0
  61. data/lib/pcapr_local/www/static/image/body-h3.png +0 -0
  62. data/lib/pcapr_local/www/static/image/body-hl1-bg.png +0 -0
  63. data/lib/pcapr_local/www/static/image/body-hl1-h3.png +0 -0
  64. data/lib/pcapr_local/www/static/image/body-hl1-readmore.png +0 -0
  65. data/lib/pcapr_local/www/static/image/body-hl2-bg.png +0 -0
  66. data/lib/pcapr_local/www/static/image/body-hl2-h3.png +0 -0
  67. data/lib/pcapr_local/www/static/image/body-hl2-readmore.png +0 -0
  68. data/lib/pcapr_local/www/static/image/body-hl3-bg.png +0 -0
  69. data/lib/pcapr_local/www/static/image/body-hl3-h3.png +0 -0
  70. data/lib/pcapr_local/www/static/image/body-hl3-readmore.png +0 -0
  71. data/lib/pcapr_local/www/static/image/body-hl4-bg.png +0 -0
  72. data/lib/pcapr_local/www/static/image/body-hl4-h3.png +0 -0
  73. data/lib/pcapr_local/www/static/image/body-hl4-readmore.png +0 -0
  74. data/lib/pcapr_local/www/static/image/body-hl5-h3.png +0 -0
  75. data/lib/pcapr_local/www/static/image/body-hl6-h3.png +0 -0
  76. data/lib/pcapr_local/www/static/image/body-hl7-h3.png +0 -0
  77. data/lib/pcapr_local/www/static/image/body-hl8-h3.png +0 -0
  78. data/lib/pcapr_local/www/static/image/body-readmore.png +0 -0
  79. data/lib/pcapr_local/www/static/image/bottom-bg.png +0 -0
  80. data/lib/pcapr_local/www/static/image/bottom-l.png +0 -0
  81. data/lib/pcapr_local/www/static/image/bottom-r.png +0 -0
  82. data/lib/pcapr_local/www/static/image/btn-search.png +0 -0
  83. data/lib/pcapr_local/www/static/image/bullet-1.png +0 -0
  84. data/lib/pcapr_local/www/static/image/bullet-2.png +0 -0
  85. data/lib/pcapr_local/www/static/image/bullet-3.png +0 -0
  86. data/lib/pcapr_local/www/static/image/bullet-4.png +0 -0
  87. data/lib/pcapr_local/www/static/image/bullet-5.png +0 -0
  88. data/lib/pcapr_local/www/static/image/bullet-6.png +0 -0
  89. data/lib/pcapr_local/www/static/image/bullet-7.png +0 -0
  90. data/lib/pcapr_local/www/static/image/bullet-hl1.png +0 -0
  91. data/lib/pcapr_local/www/static/image/bullet-hl2.png +0 -0
  92. data/lib/pcapr_local/www/static/image/bullet-hl3.png +0 -0
  93. data/lib/pcapr_local/www/static/image/bullet-hl4.png +0 -0
  94. data/lib/pcapr_local/www/static/image/bullet-pathway.png +0 -0
  95. data/lib/pcapr_local/www/static/image/bullet-section1.png +0 -0
  96. data/lib/pcapr_local/www/static/image/bullet-section2.png +0 -0
  97. data/lib/pcapr_local/www/static/image/collapsed.gif +0 -0
  98. data/lib/pcapr_local/www/static/image/crosslink.png +0 -0
  99. data/lib/pcapr_local/www/static/image/expanded.gif +0 -0
  100. data/lib/pcapr_local/www/static/image/favicon.ico +0 -0
  101. data/lib/pcapr_local/www/static/image/favicon.png +0 -0
  102. data/lib/pcapr_local/www/static/image/icon-author.png +0 -0
  103. data/lib/pcapr_local/www/static/image/icon-created.png +0 -0
  104. data/lib/pcapr_local/www/static/image/p-expand.gif +0 -0
  105. data/lib/pcapr_local/www/static/image/pcapr-logo.png +0 -0
  106. data/lib/pcapr_local/www/static/image/powered-by.png +0 -0
  107. data/lib/pcapr_local/www/static/image/section1-bg.png +0 -0
  108. data/lib/pcapr_local/www/static/image/section1-h3.png +0 -0
  109. data/lib/pcapr_local/www/static/image/section1-readmore.png +0 -0
  110. data/lib/pcapr_local/www/static/image/section2-bg.png +0 -0
  111. data/lib/pcapr_local/www/static/image/section2-h3.png +0 -0
  112. data/lib/pcapr_local/www/static/image/section2-readmore.png +0 -0
  113. data/lib/pcapr_local/www/static/image/status-alert.png +0 -0
  114. data/lib/pcapr_local/www/static/image/status-download.png +0 -0
  115. data/lib/pcapr_local/www/static/image/status-info.png +0 -0
  116. data/lib/pcapr_local/www/static/image/status-note.png +0 -0
  117. data/lib/pcapr_local/www/static/image/tab-round.png +0 -0
  118. data/lib/pcapr_local/www/static/image/throbber.gif +0 -0
  119. data/lib/pcapr_local/www/static/image/user.jpg +0 -0
  120. data/lib/pcapr_local/www/static/script/closet/async.js +421 -0
  121. data/lib/pcapr_local/www/static/script/closet/closet.api.js +241 -0
  122. data/lib/pcapr_local/www/static/script/closet/closet.folders.js +94 -0
  123. data/lib/pcapr_local/www/static/script/closet/closet.js +187 -0
  124. data/lib/pcapr_local/www/static/script/closet/closet.mr.js +219 -0
  125. data/lib/pcapr_local/www/static/script/closet/closet.options.js +359 -0
  126. data/lib/pcapr_local/www/static/script/closet/closet.quantity.js +73 -0
  127. data/lib/pcapr_local/www/static/script/closet/closet.render.js +205 -0
  128. data/lib/pcapr_local/www/static/script/closet/closet.report.js +86 -0
  129. data/lib/pcapr_local/www/static/script/closet/closet.reports.http.js +135 -0
  130. data/lib/pcapr_local/www/static/script/closet/closet.reports.overview.js +163 -0
  131. data/lib/pcapr_local/www/static/script/closet/closet.reports.sip.js +159 -0
  132. data/lib/pcapr_local/www/static/script/closet/closet.reports.tcp.js +72 -0
  133. data/lib/pcapr_local/www/static/script/closet/closet.reports.visualize.js +263 -0
  134. data/lib/pcapr_local/www/static/script/closet/closet.util.js +40 -0
  135. data/lib/pcapr_local/www/static/script/jquery/jquery-1.4.2.min.js +154 -0
  136. data/lib/pcapr_local/www/static/script/jquery/jquery-ui.js +10921 -0
  137. data/lib/pcapr_local/www/static/script/jquery/jquery.flot.js +2123 -0
  138. data/lib/pcapr_local/www/static/script/jquery/jquery.flot.selection.js +184 -0
  139. data/lib/pcapr_local/www/static/script/jquery/jquery.flot.stack.js +184 -0
  140. data/lib/pcapr_local/www/static/script/jquery/jquery.form.js +643 -0
  141. data/lib/pcapr_local/www/static/script/jquery/jquery.jsonp.min.js +3 -0
  142. data/lib/pcapr_local/www/static/script/jquery/jquery.menu.js +142 -0
  143. data/lib/pcapr_local/www/static/script/jquery/jquery.suggest.js +308 -0
  144. data/lib/pcapr_local/www/static/script/jquery/jquery.ui.core.js +203 -0
  145. data/lib/pcapr_local/www/static/script/jquery/jquery.ui.slider.js +629 -0
  146. data/lib/pcapr_local/www/static/script/jquery/jquery.ui.sortable.js +1055 -0
  147. data/lib/pcapr_local/www/static/script/jquery/jquery.ui.widget.js +236 -0
  148. data/lib/pcapr_local/www/static/script/json2.js +481 -0
  149. data/lib/pcapr_local/www/static/script/sammy/plugins/sammy.cache.js +115 -0
  150. data/lib/pcapr_local/www/static/script/sammy/plugins/sammy.template.js +117 -0
  151. data/lib/pcapr_local/www/static/script/sammy/sammy.js +1696 -0
  152. data/lib/pcapr_local/www/static/script/tipsy/jquery.tipsy.js +104 -0
  153. data/lib/pcapr_local/www/static/style/c3p0.css +116 -0
  154. data/lib/pcapr_local/www/static/style/jquery.suggest.css +27 -0
  155. data/lib/pcapr_local/www/static/style/page.css +1113 -0
  156. data/lib/pcapr_local/www/static/style/tipsy.css +7 -0
  157. data/lib/pcapr_local/www/templates/browse.services.template +10 -0
  158. data/lib/pcapr_local/www/templates/browse.template +77 -0
  159. data/lib/pcapr_local/www/templates/flows.template +38 -0
  160. data/lib/pcapr_local/www/templates/pcap.template +63 -0
  161. data/lib/pcapr_local/www/templates/sip.calls.template +35 -0
  162. data/lib/pcapr_local/www/templates/statistics.template +6 -0
  163. data/lib/pcapr_local/xtractr.rb +179 -0
  164. data/lib/pcapr_local/xtractr/instance.rb +172 -0
  165. data/pcapr-local.gemspec +297 -0
  166. data/test/mu/pcap/reader/tc_http_family.rb +251 -0
  167. data/test/mu/pcap/tc_ethernet.rb +71 -0
  168. data/test/mu/pcap/tc_header.rb +56 -0
  169. data/test/mu/pcap/tc_ipv4.rb +103 -0
  170. data/test/mu/pcap/tc_ipv6.rb +83 -0
  171. data/test/mu/pcap/tc_packet.rb +44 -0
  172. data/test/mu/pcap/tc_pair.rb +58 -0
  173. data/test/mu/pcap/tc_pkthdr.rb +33 -0
  174. data/test/mu/pcap/tc_reader.rb +76 -0
  175. data/test/mu/pcap/tc_tcp.rb +426 -0
  176. data/test/mu/pcap/tc_udp.rb +33 -0
  177. data/test/mu/pcap/tc_wrapper.rb +80 -0
  178. data/test/mu/scenario/pcap/tc_fields.rb +67 -0
  179. data/test/mu/scenario/pcap/tc_rtp.rb +135 -0
  180. data/test/mu/scenario/sip_signalled_call_1.pcap +0 -0
  181. data/test/mu/scenario/tc_pcap.rb +190 -0
  182. data/test/mu/scenario/test_data/arp.pcap +0 -0
  183. data/test/mu/scenario/test_data/dns.pcap +0 -0
  184. data/test/mu/scenario/test_data/http-v6.pcap +0 -0
  185. data/test/mu/scenario/test_data/http.pcap +0 -0
  186. data/test/mu/scenario/test_data/http_chunked.pcap +0 -0
  187. data/test/mu/scenario/test_data/http_deflate.pcap +0 -0
  188. data/test/mu/scenario/test_data/httpauth3.pcap +0 -0
  189. data/test/mu/scenario/test_data/icmp.pcap +0 -0
  190. data/test/mu/scenario/test_data/sip_signalled_call_1.pcap +0 -0
  191. data/test/mu/tc_pcap.rb +39 -0
  192. data/test/mu/testcase.rb +86 -0
  193. data/test/pcapr_local/arp.pcap +0 -0
  194. data/test/pcapr_local/data.js +3 -0
  195. data/test/pcapr_local/http_chunked.pcap +0 -0
  196. data/test/pcapr_local/tc_api.rb +181 -0
  197. data/test/pcapr_local/test.tgz +0 -0
  198. data/test/pcapr_local/test_scanner.rb +241 -0
  199. data/test/pcapr_local/test_xtractr.rb +219 -0
  200. data/test/pcapr_local/testcase.rb +107 -0
  201. data/test/test_export_to_scenario.sh +25 -0
  202. data/test/test_pcapr_local.rb +29 -0
  203. metadata +450 -0
@@ -0,0 +1,1055 @@
1
+ /*
2
+ * jQuery UI Sortable 1.8
3
+ *
4
+ * Copyright (c) 2010 AUTHORS.txt (http://jqueryui.com/about)
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
+ * jquery.ui.core.js
12
+ * jquery.ui.mouse.js
13
+ * jquery.ui.widget.js
14
+ */
15
+ (function($) {
16
+
17
+ $.widget("ui.sortable", $.ui.mouse, {
18
+ widgetEventPrefix: "sort",
19
+ options: {
20
+ appendTo: "parent",
21
+ axis: false,
22
+ connectWith: false,
23
+ containment: false,
24
+ cursor: 'auto',
25
+ cursorAt: false,
26
+ dropOnEmpty: true,
27
+ forcePlaceholderSize: false,
28
+ forceHelperSize: false,
29
+ grid: false,
30
+ handle: false,
31
+ helper: "original",
32
+ items: '> *',
33
+ opacity: false,
34
+ placeholder: false,
35
+ revert: false,
36
+ scroll: true,
37
+ scrollSensitivity: 20,
38
+ scrollSpeed: 20,
39
+ scope: "default",
40
+ tolerance: "intersect",
41
+ zIndex: 1000
42
+ },
43
+ _create: function() {
44
+
45
+ var o = this.options;
46
+ this.containerCache = {};
47
+ this.element.addClass("ui-sortable");
48
+
49
+ //Get the items
50
+ this.refresh();
51
+
52
+ //Let's determine if the items are floating
53
+ this.floating = this.items.length ? (/left|right/).test(this.items[0].item.css('float')) : false;
54
+
55
+ //Let's determine the parent's offset
56
+ this.offset = this.element.offset();
57
+
58
+ //Initialize mouse events for interaction
59
+ this._mouseInit();
60
+
61
+ },
62
+
63
+ destroy: function() {
64
+ this.element
65
+ .removeClass("ui-sortable ui-sortable-disabled")
66
+ .removeData("sortable")
67
+ .unbind(".sortable");
68
+ this._mouseDestroy();
69
+
70
+ for ( var i = this.items.length - 1; i >= 0; i-- )
71
+ this.items[i].item.removeData("sortable-item");
72
+
73
+ return this;
74
+ },
75
+
76
+ _mouseCapture: function(event, overrideHandle) {
77
+
78
+ if (this.reverting) {
79
+ return false;
80
+ }
81
+
82
+ if(this.options.disabled || this.options.type == 'static') return false;
83
+
84
+ //We have to refresh the items data once first
85
+ this._refreshItems(event);
86
+
87
+ //Find out if the clicked node (or one of its parents) is a actual item in this.items
88
+ var currentItem = null, self = this, nodes = $(event.target).parents().each(function() {
89
+ if($.data(this, 'sortable-item') == self) {
90
+ currentItem = $(this);
91
+ return false;
92
+ }
93
+ });
94
+ if($.data(event.target, 'sortable-item') == self) currentItem = $(event.target);
95
+
96
+ if(!currentItem) return false;
97
+ if(this.options.handle && !overrideHandle) {
98
+ var validHandle = false;
99
+
100
+ $(this.options.handle, currentItem).find("*").andSelf().each(function() { if(this == event.target) validHandle = true; });
101
+ if(!validHandle) return false;
102
+ }
103
+
104
+ this.currentItem = currentItem;
105
+ this._removeCurrentsFromItems();
106
+ return true;
107
+
108
+ },
109
+
110
+ _mouseStart: function(event, overrideHandle, noActivation) {
111
+
112
+ var o = this.options, self = this;
113
+ this.currentContainer = this;
114
+
115
+ //We only need to call refreshPositions, because the refreshItems call has been moved to mouseCapture
116
+ this.refreshPositions();
117
+
118
+ //Create and append the visible helper
119
+ this.helper = this._createHelper(event);
120
+
121
+ //Cache the helper size
122
+ this._cacheHelperProportions();
123
+
124
+ /*
125
+ * - Position generation -
126
+ * This block generates everything position related - it's the core of draggables.
127
+ */
128
+
129
+ //Cache the margins of the original element
130
+ this._cacheMargins();
131
+
132
+ //Get the next scrolling parent
133
+ this.scrollParent = this.helper.scrollParent();
134
+
135
+ //The element's absolute position on the page minus margins
136
+ this.offset = this.currentItem.offset();
137
+ this.offset = {
138
+ top: this.offset.top - this.margins.top,
139
+ left: this.offset.left - this.margins.left
140
+ };
141
+
142
+ // Only after we got the offset, we can change the helper's position to absolute
143
+ // TODO: Still need to figure out a way to make relative sorting possible
144
+ this.helper.css("position", "absolute");
145
+ this.cssPosition = this.helper.css("position");
146
+
147
+ $.extend(this.offset, {
148
+ click: { //Where the click happened, relative to the element
149
+ left: event.pageX - this.offset.left,
150
+ top: event.pageY - this.offset.top
151
+ },
152
+ parent: this._getParentOffset(),
153
+ relative: this._getRelativeOffset() //This is a relative to absolute position minus the actual position calculation - only used for relative positioned helper
154
+ });
155
+
156
+ //Generate the original position
157
+ this.originalPosition = this._generatePosition(event);
158
+ this.originalPageX = event.pageX;
159
+ this.originalPageY = event.pageY;
160
+
161
+ //Adjust the mouse offset relative to the helper if 'cursorAt' is supplied
162
+ (o.cursorAt && this._adjustOffsetFromHelper(o.cursorAt));
163
+
164
+ //Cache the former DOM position
165
+ this.domPosition = { prev: this.currentItem.prev()[0], parent: this.currentItem.parent()[0] };
166
+
167
+ //If the helper is not the original, hide the original so it's not playing any role during the drag, won't cause anything bad this way
168
+ if(this.helper[0] != this.currentItem[0]) {
169
+ this.currentItem.hide();
170
+ }
171
+
172
+ //Create the placeholder
173
+ this._createPlaceholder();
174
+
175
+ //Set a containment if given in the options
176
+ if(o.containment)
177
+ this._setContainment();
178
+
179
+ if(o.cursor) { // cursor option
180
+ if ($('body').css("cursor")) this._storedCursor = $('body').css("cursor");
181
+ $('body').css("cursor", o.cursor);
182
+ }
183
+
184
+ if(o.opacity) { // opacity option
185
+ if (this.helper.css("opacity")) this._storedOpacity = this.helper.css("opacity");
186
+ this.helper.css("opacity", o.opacity);
187
+ }
188
+
189
+ if(o.zIndex) { // zIndex option
190
+ if (this.helper.css("zIndex")) this._storedZIndex = this.helper.css("zIndex");
191
+ this.helper.css("zIndex", o.zIndex);
192
+ }
193
+
194
+ //Prepare scrolling
195
+ if(this.scrollParent[0] != document && this.scrollParent[0].tagName != 'HTML')
196
+ this.overflowOffset = this.scrollParent.offset();
197
+
198
+ //Call callbacks
199
+ this._trigger("start", event, this._uiHash());
200
+
201
+ //Recache the helper size
202
+ if(!this._preserveHelperProportions)
203
+ this._cacheHelperProportions();
204
+
205
+
206
+ //Post 'activate' events to possible containers
207
+ if(!noActivation) {
208
+ for (var i = this.containers.length - 1; i >= 0; i--) { this.containers[i]._trigger("activate", event, self._uiHash(this)); }
209
+ }
210
+
211
+ //Prepare possible droppables
212
+ if($.ui.ddmanager)
213
+ $.ui.ddmanager.current = this;
214
+
215
+ if ($.ui.ddmanager && !o.dropBehaviour)
216
+ $.ui.ddmanager.prepareOffsets(this, event);
217
+
218
+ this.dragging = true;
219
+
220
+ this.helper.addClass("ui-sortable-helper");
221
+ this._mouseDrag(event); //Execute the drag once - this causes the helper not to be visible before getting its correct position
222
+ return true;
223
+
224
+ },
225
+
226
+ _mouseDrag: function(event) {
227
+
228
+ //Compute the helpers position
229
+ this.position = this._generatePosition(event);
230
+ this.positionAbs = this._convertPositionTo("absolute");
231
+
232
+ if (!this.lastPositionAbs) {
233
+ this.lastPositionAbs = this.positionAbs;
234
+ }
235
+
236
+ //Do scrolling
237
+ if(this.options.scroll) {
238
+ var o = this.options, scrolled = false;
239
+ if(this.scrollParent[0] != document && this.scrollParent[0].tagName != 'HTML') {
240
+
241
+ if((this.overflowOffset.top + this.scrollParent[0].offsetHeight) - event.pageY < o.scrollSensitivity)
242
+ this.scrollParent[0].scrollTop = scrolled = this.scrollParent[0].scrollTop + o.scrollSpeed;
243
+ else if(event.pageY - this.overflowOffset.top < o.scrollSensitivity)
244
+ this.scrollParent[0].scrollTop = scrolled = this.scrollParent[0].scrollTop - o.scrollSpeed;
245
+
246
+ if((this.overflowOffset.left + this.scrollParent[0].offsetWidth) - event.pageX < o.scrollSensitivity)
247
+ this.scrollParent[0].scrollLeft = scrolled = this.scrollParent[0].scrollLeft + o.scrollSpeed;
248
+ else if(event.pageX - this.overflowOffset.left < o.scrollSensitivity)
249
+ this.scrollParent[0].scrollLeft = scrolled = this.scrollParent[0].scrollLeft - o.scrollSpeed;
250
+
251
+ } else {
252
+
253
+ if(event.pageY - $(document).scrollTop() < o.scrollSensitivity)
254
+ scrolled = $(document).scrollTop($(document).scrollTop() - o.scrollSpeed);
255
+ else if($(window).height() - (event.pageY - $(document).scrollTop()) < o.scrollSensitivity)
256
+ scrolled = $(document).scrollTop($(document).scrollTop() + o.scrollSpeed);
257
+
258
+ if(event.pageX - $(document).scrollLeft() < o.scrollSensitivity)
259
+ scrolled = $(document).scrollLeft($(document).scrollLeft() - o.scrollSpeed);
260
+ else if($(window).width() - (event.pageX - $(document).scrollLeft()) < o.scrollSensitivity)
261
+ scrolled = $(document).scrollLeft($(document).scrollLeft() + o.scrollSpeed);
262
+
263
+ }
264
+
265
+ if(scrolled !== false && $.ui.ddmanager && !o.dropBehaviour)
266
+ $.ui.ddmanager.prepareOffsets(this, event);
267
+ }
268
+
269
+ //Regenerate the absolute position used for position checks
270
+ this.positionAbs = this._convertPositionTo("absolute");
271
+
272
+ //Set the helper position
273
+ if(!this.options.axis || this.options.axis != "y") this.helper[0].style.left = this.position.left+'px';
274
+ if(!this.options.axis || this.options.axis != "x") this.helper[0].style.top = this.position.top+'px';
275
+
276
+ //Rearrange
277
+ for (var i = this.items.length - 1; i >= 0; i--) {
278
+
279
+ //Cache variables and intersection, continue if no intersection
280
+ var item = this.items[i], itemElement = item.item[0], intersection = this._intersectsWithPointer(item);
281
+ if (!intersection) continue;
282
+
283
+ if(itemElement != this.currentItem[0] //cannot intersect with itself
284
+ && this.placeholder[intersection == 1 ? "next" : "prev"]()[0] != itemElement //no useless actions that have been done before
285
+ && !$.ui.contains(this.placeholder[0], itemElement) //no action if the item moved is the parent of the item checked
286
+ && (this.options.type == 'semi-dynamic' ? !$.ui.contains(this.element[0], itemElement) : true)
287
+ //&& itemElement.parentNode == this.placeholder[0].parentNode // only rearrange items within the same container
288
+ ) {
289
+
290
+ this.direction = intersection == 1 ? "down" : "up";
291
+
292
+ if (this.options.tolerance == "pointer" || this._intersectsWithSides(item)) {
293
+ this._rearrange(event, item);
294
+ } else {
295
+ break;
296
+ }
297
+
298
+ this._trigger("change", event, this._uiHash());
299
+ break;
300
+ }
301
+ }
302
+
303
+ //Post events to containers
304
+ this._contactContainers(event);
305
+
306
+ //Interconnect with droppables
307
+ if($.ui.ddmanager) $.ui.ddmanager.drag(this, event);
308
+
309
+ //Call callbacks
310
+ this._trigger('sort', event, this._uiHash());
311
+
312
+ this.lastPositionAbs = this.positionAbs;
313
+ return false;
314
+
315
+ },
316
+
317
+ _mouseStop: function(event, noPropagation) {
318
+
319
+ if(!event) return;
320
+
321
+ //If we are using droppables, inform the manager about the drop
322
+ if ($.ui.ddmanager && !this.options.dropBehaviour)
323
+ $.ui.ddmanager.drop(this, event);
324
+
325
+ if(this.options.revert) {
326
+ var self = this;
327
+ var cur = self.placeholder.offset();
328
+
329
+ self.reverting = true;
330
+
331
+ $(this.helper).animate({
332
+ left: cur.left - this.offset.parent.left - self.margins.left + (this.offsetParent[0] == document.body ? 0 : this.offsetParent[0].scrollLeft),
333
+ top: cur.top - this.offset.parent.top - self.margins.top + (this.offsetParent[0] == document.body ? 0 : this.offsetParent[0].scrollTop)
334
+ }, parseInt(this.options.revert, 10) || 500, function() {
335
+ self._clear(event);
336
+ });
337
+ } else {
338
+ this._clear(event, noPropagation);
339
+ }
340
+
341
+ return false;
342
+
343
+ },
344
+
345
+ cancel: function() {
346
+
347
+ var self = this;
348
+
349
+ if(this.dragging) {
350
+
351
+ this._mouseUp();
352
+
353
+ if(this.options.helper == "original")
354
+ this.currentItem.css(this._storedCSS).removeClass("ui-sortable-helper");
355
+ else
356
+ this.currentItem.show();
357
+
358
+ //Post deactivating events to containers
359
+ for (var i = this.containers.length - 1; i >= 0; i--){
360
+ this.containers[i]._trigger("deactivate", null, self._uiHash(this));
361
+ if(this.containers[i].containerCache.over) {
362
+ this.containers[i]._trigger("out", null, self._uiHash(this));
363
+ this.containers[i].containerCache.over = 0;
364
+ }
365
+ }
366
+
367
+ }
368
+
369
+ //$(this.placeholder[0]).remove(); would have been the jQuery way - unfortunately, it unbinds ALL events from the original node!
370
+ if(this.placeholder[0].parentNode) this.placeholder[0].parentNode.removeChild(this.placeholder[0]);
371
+ if(this.options.helper != "original" && this.helper && this.helper[0].parentNode) this.helper.remove();
372
+
373
+ $.extend(this, {
374
+ helper: null,
375
+ dragging: false,
376
+ reverting: false,
377
+ _noFinalSort: null
378
+ });
379
+
380
+ if(this.domPosition.prev) {
381
+ $(this.domPosition.prev).after(this.currentItem);
382
+ } else {
383
+ $(this.domPosition.parent).prepend(this.currentItem);
384
+ }
385
+
386
+ return this;
387
+
388
+ },
389
+
390
+ serialize: function(o) {
391
+
392
+ var items = this._getItemsAsjQuery(o && o.connected);
393
+ var str = []; o = o || {};
394
+
395
+ $(items).each(function() {
396
+ var res = ($(o.item || this).attr(o.attribute || 'id') || '').match(o.expression || (/(.+)[-=_](.+)/));
397
+ if(res) str.push((o.key || res[1]+'[]')+'='+(o.key && o.expression ? res[1] : res[2]));
398
+ });
399
+
400
+ return str.join('&');
401
+
402
+ },
403
+
404
+ toArray: function(o) {
405
+
406
+ var items = this._getItemsAsjQuery(o && o.connected);
407
+ var ret = []; o = o || {};
408
+
409
+ items.each(function() { ret.push($(o.item || this).attr(o.attribute || 'id') || ''); });
410
+ return ret;
411
+
412
+ },
413
+
414
+ /* Be careful with the following core functions */
415
+ _intersectsWith: function(item) {
416
+
417
+ var x1 = this.positionAbs.left,
418
+ x2 = x1 + this.helperProportions.width,
419
+ y1 = this.positionAbs.top,
420
+ y2 = y1 + this.helperProportions.height;
421
+
422
+ var l = item.left,
423
+ r = l + item.width,
424
+ t = item.top,
425
+ b = t + item.height;
426
+
427
+ var dyClick = this.offset.click.top,
428
+ dxClick = this.offset.click.left;
429
+
430
+ var isOverElement = (y1 + dyClick) > t && (y1 + dyClick) < b && (x1 + dxClick) > l && (x1 + dxClick) < r;
431
+
432
+ if( this.options.tolerance == "pointer"
433
+ || this.options.forcePointerForContainers
434
+ || (this.options.tolerance != "pointer" && this.helperProportions[this.floating ? 'width' : 'height'] > item[this.floating ? 'width' : 'height'])
435
+ ) {
436
+ return isOverElement;
437
+ } else {
438
+
439
+ return (l < x1 + (this.helperProportions.width / 2) // Right Half
440
+ && x2 - (this.helperProportions.width / 2) < r // Left Half
441
+ && t < y1 + (this.helperProportions.height / 2) // Bottom Half
442
+ && y2 - (this.helperProportions.height / 2) < b ); // Top Half
443
+
444
+ }
445
+ },
446
+
447
+ _intersectsWithPointer: function(item) {
448
+
449
+ var isOverElementHeight = $.ui.isOverAxis(this.positionAbs.top + this.offset.click.top, item.top, item.height),
450
+ isOverElementWidth = $.ui.isOverAxis(this.positionAbs.left + this.offset.click.left, item.left, item.width),
451
+ isOverElement = isOverElementHeight && isOverElementWidth,
452
+ verticalDirection = this._getDragVerticalDirection(),
453
+ horizontalDirection = this._getDragHorizontalDirection();
454
+
455
+ if (!isOverElement)
456
+ return false;
457
+
458
+ return this.floating ?
459
+ ( ((horizontalDirection && horizontalDirection == "right") || verticalDirection == "down") ? 2 : 1 )
460
+ : ( verticalDirection && (verticalDirection == "down" ? 2 : 1) );
461
+
462
+ },
463
+
464
+ _intersectsWithSides: function(item) {
465
+
466
+ var isOverBottomHalf = $.ui.isOverAxis(this.positionAbs.top + this.offset.click.top, item.top + (item.height/2), item.height),
467
+ isOverRightHalf = $.ui.isOverAxis(this.positionAbs.left + this.offset.click.left, item.left + (item.width/2), item.width),
468
+ verticalDirection = this._getDragVerticalDirection(),
469
+ horizontalDirection = this._getDragHorizontalDirection();
470
+
471
+ if (this.floating && horizontalDirection) {
472
+ return ((horizontalDirection == "right" && isOverRightHalf) || (horizontalDirection == "left" && !isOverRightHalf));
473
+ } else {
474
+ return verticalDirection && ((verticalDirection == "down" && isOverBottomHalf) || (verticalDirection == "up" && !isOverBottomHalf));
475
+ }
476
+
477
+ },
478
+
479
+ _getDragVerticalDirection: function() {
480
+ var delta = this.positionAbs.top - this.lastPositionAbs.top;
481
+ return delta != 0 && (delta > 0 ? "down" : "up");
482
+ },
483
+
484
+ _getDragHorizontalDirection: function() {
485
+ var delta = this.positionAbs.left - this.lastPositionAbs.left;
486
+ return delta != 0 && (delta > 0 ? "right" : "left");
487
+ },
488
+
489
+ refresh: function(event) {
490
+ this._refreshItems(event);
491
+ this.refreshPositions();
492
+ return this;
493
+ },
494
+
495
+ _connectWith: function() {
496
+ var options = this.options;
497
+ return options.connectWith.constructor == String
498
+ ? [options.connectWith]
499
+ : options.connectWith;
500
+ },
501
+
502
+ _getItemsAsjQuery: function(connected) {
503
+
504
+ var self = this;
505
+ var items = [];
506
+ var queries = [];
507
+ var connectWith = this._connectWith();
508
+
509
+ if(connectWith && connected) {
510
+ for (var i = connectWith.length - 1; i >= 0; i--){
511
+ var cur = $(connectWith[i]);
512
+ for (var j = cur.length - 1; j >= 0; j--){
513
+ var inst = $.data(cur[j], 'sortable');
514
+ if(inst && inst != this && !inst.options.disabled) {
515
+ queries.push([$.isFunction(inst.options.items) ? inst.options.items.call(inst.element) : $(inst.options.items, inst.element).not(".ui-sortable-helper").not('.ui-sortable-placeholder'), inst]);
516
+ }
517
+ };
518
+ };
519
+ }
520
+
521
+ queries.push([$.isFunction(this.options.items) ? this.options.items.call(this.element, null, { options: this.options, item: this.currentItem }) : $(this.options.items, this.element).not(".ui-sortable-helper").not('.ui-sortable-placeholder'), this]);
522
+
523
+ for (var i = queries.length - 1; i >= 0; i--){
524
+ queries[i][0].each(function() {
525
+ items.push(this);
526
+ });
527
+ };
528
+
529
+ return $(items);
530
+
531
+ },
532
+
533
+ _removeCurrentsFromItems: function() {
534
+
535
+ var list = this.currentItem.find(":data(sortable-item)");
536
+
537
+ for (var i=0; i < this.items.length; i++) {
538
+
539
+ for (var j=0; j < list.length; j++) {
540
+ if(list[j] == this.items[i].item[0])
541
+ this.items.splice(i,1);
542
+ };
543
+
544
+ };
545
+
546
+ },
547
+
548
+ _refreshItems: function(event) {
549
+
550
+ this.items = [];
551
+ this.containers = [this];
552
+ var items = this.items;
553
+ var self = this;
554
+ var queries = [[$.isFunction(this.options.items) ? this.options.items.call(this.element[0], event, { item: this.currentItem }) : $(this.options.items, this.element), this]];
555
+ var connectWith = this._connectWith();
556
+
557
+ if(connectWith) {
558
+ for (var i = connectWith.length - 1; i >= 0; i--){
559
+ var cur = $(connectWith[i]);
560
+ for (var j = cur.length - 1; j >= 0; j--){
561
+ var inst = $.data(cur[j], 'sortable');
562
+ if(inst && inst != this && !inst.options.disabled) {
563
+ queries.push([$.isFunction(inst.options.items) ? inst.options.items.call(inst.element[0], event, { item: this.currentItem }) : $(inst.options.items, inst.element), inst]);
564
+ this.containers.push(inst);
565
+ }
566
+ };
567
+ };
568
+ }
569
+
570
+ for (var i = queries.length - 1; i >= 0; i--) {
571
+ var targetData = queries[i][1];
572
+ var _queries = queries[i][0];
573
+
574
+ for (var j=0, queriesLength = _queries.length; j < queriesLength; j++) {
575
+ var item = $(_queries[j]);
576
+
577
+ item.data('sortable-item', targetData); // Data for target checking (mouse manager)
578
+
579
+ items.push({
580
+ item: item,
581
+ instance: targetData,
582
+ width: 0, height: 0,
583
+ left: 0, top: 0
584
+ });
585
+ };
586
+ };
587
+
588
+ },
589
+
590
+ refreshPositions: function(fast) {
591
+
592
+ //This has to be redone because due to the item being moved out/into the offsetParent, the offsetParent's position will change
593
+ if(this.offsetParent && this.helper) {
594
+ this.offset.parent = this._getParentOffset();
595
+ }
596
+
597
+ for (var i = this.items.length - 1; i >= 0; i--){
598
+ var item = this.items[i];
599
+
600
+ var t = this.options.toleranceElement ? $(this.options.toleranceElement, item.item) : item.item;
601
+
602
+ if (!fast) {
603
+ item.width = t.outerWidth();
604
+ item.height = t.outerHeight();
605
+ }
606
+
607
+ var p = t.offset();
608
+ item.left = p.left;
609
+ item.top = p.top;
610
+ };
611
+
612
+ if(this.options.custom && this.options.custom.refreshContainers) {
613
+ this.options.custom.refreshContainers.call(this);
614
+ } else {
615
+ for (var i = this.containers.length - 1; i >= 0; i--){
616
+ var p = this.containers[i].element.offset();
617
+ this.containers[i].containerCache.left = p.left;
618
+ this.containers[i].containerCache.top = p.top;
619
+ this.containers[i].containerCache.width = this.containers[i].element.outerWidth();
620
+ this.containers[i].containerCache.height = this.containers[i].element.outerHeight();
621
+ };
622
+ }
623
+
624
+ return this;
625
+ },
626
+
627
+ _createPlaceholder: function(that) {
628
+
629
+ var self = that || this, o = self.options;
630
+
631
+ if(!o.placeholder || o.placeholder.constructor == String) {
632
+ var className = o.placeholder;
633
+ o.placeholder = {
634
+ element: function() {
635
+
636
+ var el = $(document.createElement(self.currentItem[0].nodeName))
637
+ .addClass(className || self.currentItem[0].className+" ui-sortable-placeholder")
638
+ .removeClass("ui-sortable-helper")[0];
639
+
640
+ if(!className)
641
+ el.style.visibility = "hidden";
642
+
643
+ return el;
644
+ },
645
+ update: function(container, p) {
646
+
647
+ // 1. If a className is set as 'placeholder option, we don't force sizes - the class is responsible for that
648
+ // 2. The option 'forcePlaceholderSize can be enabled to force it even if a class name is specified
649
+ if(className && !o.forcePlaceholderSize) return;
650
+
651
+ //If the element doesn't have a actual height by itself (without styles coming from a stylesheet), it receives the inline height from the dragged item
652
+ if(!p.height()) { p.height(self.currentItem.innerHeight() - parseInt(self.currentItem.css('paddingTop')||0, 10) - parseInt(self.currentItem.css('paddingBottom')||0, 10)); };
653
+ if(!p.width()) { p.width(self.currentItem.innerWidth() - parseInt(self.currentItem.css('paddingLeft')||0, 10) - parseInt(self.currentItem.css('paddingRight')||0, 10)); };
654
+ }
655
+ };
656
+ }
657
+
658
+ //Create the placeholder
659
+ self.placeholder = $(o.placeholder.element.call(self.element, self.currentItem));
660
+
661
+ //Append it after the actual current item
662
+ self.currentItem.after(self.placeholder);
663
+
664
+ //Update the size of the placeholder (TODO: Logic to fuzzy, see line 316/317)
665
+ o.placeholder.update(self, self.placeholder);
666
+
667
+ },
668
+
669
+ _contactContainers: function(event) {
670
+
671
+ // get innermost container that intersects with item
672
+ var innermostContainer = null, innermostIndex = null;
673
+
674
+
675
+ for (var i = this.containers.length - 1; i >= 0; i--){
676
+
677
+ // never consider a container that's located within the item itself
678
+ if($.ui.contains(this.currentItem[0], this.containers[i].element[0]))
679
+ continue;
680
+
681
+ if(this._intersectsWith(this.containers[i].containerCache)) {
682
+
683
+ // if we've already found a container and it's more "inner" than this, then continue
684
+ if(innermostContainer && $.ui.contains(this.containers[i].element[0], innermostContainer.element[0]))
685
+ continue;
686
+
687
+ innermostContainer = this.containers[i];
688
+ innermostIndex = i;
689
+
690
+ } else {
691
+ // container doesn't intersect. trigger "out" event if necessary
692
+ if(this.containers[i].containerCache.over) {
693
+ this.containers[i]._trigger("out", event, this._uiHash(this));
694
+ this.containers[i].containerCache.over = 0;
695
+ }
696
+ }
697
+
698
+ }
699
+
700
+ // if no intersecting containers found, return
701
+ if(!innermostContainer) return;
702
+
703
+ // move the item into the container if it's not there already
704
+ if(this.containers.length === 1) {
705
+ this.containers[innermostIndex]._trigger("over", event, this._uiHash(this));
706
+ this.containers[innermostIndex].containerCache.over = 1;
707
+ } else if(this.currentContainer != this.containers[innermostIndex]) {
708
+
709
+ //When entering a new container, we will find the item with the least distance and append our item near it
710
+ var dist = 10000; var itemWithLeastDistance = null; var base = this.positionAbs[this.containers[innermostIndex].floating ? 'left' : 'top'];
711
+ for (var j = this.items.length - 1; j >= 0; j--) {
712
+ if(!$.ui.contains(this.containers[innermostIndex].element[0], this.items[j].item[0])) continue;
713
+ var cur = this.items[j][this.containers[innermostIndex].floating ? 'left' : 'top'];
714
+ if(Math.abs(cur - base) < dist) {
715
+ dist = Math.abs(cur - base); itemWithLeastDistance = this.items[j];
716
+ }
717
+ }
718
+
719
+ if(!itemWithLeastDistance && !this.options.dropOnEmpty) //Check if dropOnEmpty is enabled
720
+ return;
721
+
722
+ this.currentContainer = this.containers[innermostIndex];
723
+ itemWithLeastDistance ? this._rearrange(event, itemWithLeastDistance, null, true) : this._rearrange(event, null, this.containers[innermostIndex].element, true);
724
+ this._trigger("change", event, this._uiHash());
725
+ this.containers[innermostIndex]._trigger("change", event, this._uiHash(this));
726
+
727
+ //Update the placeholder
728
+ this.options.placeholder.update(this.currentContainer, this.placeholder);
729
+
730
+ this.containers[innermostIndex]._trigger("over", event, this._uiHash(this));
731
+ this.containers[innermostIndex].containerCache.over = 1;
732
+ }
733
+
734
+
735
+ },
736
+
737
+ _createHelper: function(event) {
738
+
739
+ var o = this.options;
740
+ var helper = $.isFunction(o.helper) ? $(o.helper.apply(this.element[0], [event, this.currentItem])) : (o.helper == 'clone' ? this.currentItem.clone() : this.currentItem);
741
+
742
+ if(!helper.parents('body').length) //Add the helper to the DOM if that didn't happen already
743
+ $(o.appendTo != 'parent' ? o.appendTo : this.currentItem[0].parentNode)[0].appendChild(helper[0]);
744
+
745
+ if(helper[0] == this.currentItem[0])
746
+ this._storedCSS = { width: this.currentItem[0].style.width, height: this.currentItem[0].style.height, position: this.currentItem.css("position"), top: this.currentItem.css("top"), left: this.currentItem.css("left") };
747
+
748
+ if(helper[0].style.width == '' || o.forceHelperSize) helper.width(this.currentItem.width());
749
+ if(helper[0].style.height == '' || o.forceHelperSize) helper.height(this.currentItem.height());
750
+
751
+ return helper;
752
+
753
+ },
754
+
755
+ _adjustOffsetFromHelper: function(obj) {
756
+ if (typeof obj == 'string') {
757
+ obj = obj.split(' ');
758
+ }
759
+ if ($.isArray(obj)) {
760
+ obj = {left: +obj[0], top: +obj[1] || 0};
761
+ }
762
+ if ('left' in obj) {
763
+ this.offset.click.left = obj.left + this.margins.left;
764
+ }
765
+ if ('right' in obj) {
766
+ this.offset.click.left = this.helperProportions.width - obj.right + this.margins.left;
767
+ }
768
+ if ('top' in obj) {
769
+ this.offset.click.top = obj.top + this.margins.top;
770
+ }
771
+ if ('bottom' in obj) {
772
+ this.offset.click.top = this.helperProportions.height - obj.bottom + this.margins.top;
773
+ }
774
+ },
775
+
776
+ _getParentOffset: function() {
777
+
778
+
779
+ //Get the offsetParent and cache its position
780
+ this.offsetParent = this.helper.offsetParent();
781
+ var po = this.offsetParent.offset();
782
+
783
+ // This is a special case where we need to modify a offset calculated on start, since the following happened:
784
+ // 1. The position of the helper is absolute, so it's position is calculated based on the next positioned parent
785
+ // 2. The actual offset parent is a child of the scroll parent, and the scroll parent isn't the document, which means that
786
+ // the scroll is included in the initial calculation of the offset of the parent, and never recalculated upon drag
787
+ if(this.cssPosition == 'absolute' && this.scrollParent[0] != document && $.ui.contains(this.scrollParent[0], this.offsetParent[0])) {
788
+ po.left += this.scrollParent.scrollLeft();
789
+ po.top += this.scrollParent.scrollTop();
790
+ }
791
+
792
+ if((this.offsetParent[0] == document.body) //This needs to be actually done for all browsers, since pageX/pageY includes this information
793
+ || (this.offsetParent[0].tagName && this.offsetParent[0].tagName.toLowerCase() == 'html' && $.browser.msie)) //Ugly IE fix
794
+ po = { top: 0, left: 0 };
795
+
796
+ return {
797
+ top: po.top + (parseInt(this.offsetParent.css("borderTopWidth"),10) || 0),
798
+ left: po.left + (parseInt(this.offsetParent.css("borderLeftWidth"),10) || 0)
799
+ };
800
+
801
+ },
802
+
803
+ _getRelativeOffset: function() {
804
+
805
+ if(this.cssPosition == "relative") {
806
+ var p = this.currentItem.position();
807
+ return {
808
+ top: p.top - (parseInt(this.helper.css("top"),10) || 0) + this.scrollParent.scrollTop(),
809
+ left: p.left - (parseInt(this.helper.css("left"),10) || 0) + this.scrollParent.scrollLeft()
810
+ };
811
+ } else {
812
+ return { top: 0, left: 0 };
813
+ }
814
+
815
+ },
816
+
817
+ _cacheMargins: function() {
818
+ this.margins = {
819
+ left: (parseInt(this.currentItem.css("marginLeft"),10) || 0),
820
+ top: (parseInt(this.currentItem.css("marginTop"),10) || 0)
821
+ };
822
+ },
823
+
824
+ _cacheHelperProportions: function() {
825
+ this.helperProportions = {
826
+ width: this.helper.outerWidth(),
827
+ height: this.helper.outerHeight()
828
+ };
829
+ },
830
+
831
+ _setContainment: function() {
832
+
833
+ var o = this.options;
834
+ if(o.containment == 'parent') o.containment = this.helper[0].parentNode;
835
+ if(o.containment == 'document' || o.containment == 'window') this.containment = [
836
+ 0 - this.offset.relative.left - this.offset.parent.left,
837
+ 0 - this.offset.relative.top - this.offset.parent.top,
838
+ $(o.containment == 'document' ? document : window).width() - this.helperProportions.width - this.margins.left,
839
+ ($(o.containment == 'document' ? document : window).height() || document.body.parentNode.scrollHeight) - this.helperProportions.height - this.margins.top
840
+ ];
841
+
842
+ if(!(/^(document|window|parent)$/).test(o.containment)) {
843
+ var ce = $(o.containment)[0];
844
+ var co = $(o.containment).offset();
845
+ var over = ($(ce).css("overflow") != 'hidden');
846
+
847
+ this.containment = [
848
+ co.left + (parseInt($(ce).css("borderLeftWidth"),10) || 0) + (parseInt($(ce).css("paddingLeft"),10) || 0) - this.margins.left,
849
+ co.top + (parseInt($(ce).css("borderTopWidth"),10) || 0) + (parseInt($(ce).css("paddingTop"),10) || 0) - this.margins.top,
850
+ co.left+(over ? Math.max(ce.scrollWidth,ce.offsetWidth) : ce.offsetWidth) - (parseInt($(ce).css("borderLeftWidth"),10) || 0) - (parseInt($(ce).css("paddingRight"),10) || 0) - this.helperProportions.width - this.margins.left,
851
+ co.top+(over ? Math.max(ce.scrollHeight,ce.offsetHeight) : ce.offsetHeight) - (parseInt($(ce).css("borderTopWidth"),10) || 0) - (parseInt($(ce).css("paddingBottom"),10) || 0) - this.helperProportions.height - this.margins.top
852
+ ];
853
+ }
854
+
855
+ },
856
+
857
+ _convertPositionTo: function(d, pos) {
858
+
859
+ if(!pos) pos = this.position;
860
+ var mod = d == "absolute" ? 1 : -1;
861
+ var o = this.options, scroll = this.cssPosition == 'absolute' && !(this.scrollParent[0] != document && $.ui.contains(this.scrollParent[0], this.offsetParent[0])) ? this.offsetParent : this.scrollParent, scrollIsRootNode = (/(html|body)/i).test(scroll[0].tagName);
862
+
863
+ return {
864
+ top: (
865
+ pos.top // The absolute mouse position
866
+ + this.offset.relative.top * mod // Only for relative positioned nodes: Relative offset from element to offset parent
867
+ + this.offset.parent.top * mod // The offsetParent's offset without borders (offset + border)
868
+ - ($.browser.safari && this.cssPosition == 'fixed' ? 0 : ( this.cssPosition == 'fixed' ? -this.scrollParent.scrollTop() : ( scrollIsRootNode ? 0 : scroll.scrollTop() ) ) * mod)
869
+ ),
870
+ left: (
871
+ pos.left // The absolute mouse position
872
+ + this.offset.relative.left * mod // Only for relative positioned nodes: Relative offset from element to offset parent
873
+ + this.offset.parent.left * mod // The offsetParent's offset without borders (offset + border)
874
+ - ($.browser.safari && this.cssPosition == 'fixed' ? 0 : ( this.cssPosition == 'fixed' ? -this.scrollParent.scrollLeft() : scrollIsRootNode ? 0 : scroll.scrollLeft() ) * mod)
875
+ )
876
+ };
877
+
878
+ },
879
+
880
+ _generatePosition: function(event) {
881
+
882
+ var o = this.options, scroll = this.cssPosition == 'absolute' && !(this.scrollParent[0] != document && $.ui.contains(this.scrollParent[0], this.offsetParent[0])) ? this.offsetParent : this.scrollParent, scrollIsRootNode = (/(html|body)/i).test(scroll[0].tagName);
883
+
884
+ // This is another very weird special case that only happens for relative elements:
885
+ // 1. If the css position is relative
886
+ // 2. and the scroll parent is the document or similar to the offset parent
887
+ // we have to refresh the relative offset during the scroll so there are no jumps
888
+ if(this.cssPosition == 'relative' && !(this.scrollParent[0] != document && this.scrollParent[0] != this.offsetParent[0])) {
889
+ this.offset.relative = this._getRelativeOffset();
890
+ }
891
+
892
+ var pageX = event.pageX;
893
+ var pageY = event.pageY;
894
+
895
+ /*
896
+ * - Position constraining -
897
+ * Constrain the position to a mix of grid, containment.
898
+ */
899
+
900
+ if(this.originalPosition) { //If we are not dragging yet, we won't check for options
901
+
902
+ if(this.containment) {
903
+ if(event.pageX - this.offset.click.left < this.containment[0]) pageX = this.containment[0] + this.offset.click.left;
904
+ if(event.pageY - this.offset.click.top < this.containment[1]) pageY = this.containment[1] + this.offset.click.top;
905
+ if(event.pageX - this.offset.click.left > this.containment[2]) pageX = this.containment[2] + this.offset.click.left;
906
+ if(event.pageY - this.offset.click.top > this.containment[3]) pageY = this.containment[3] + this.offset.click.top;
907
+ }
908
+
909
+ if(o.grid) {
910
+ var top = this.originalPageY + Math.round((pageY - this.originalPageY) / o.grid[1]) * o.grid[1];
911
+ pageY = this.containment ? (!(top - this.offset.click.top < this.containment[1] || top - this.offset.click.top > this.containment[3]) ? top : (!(top - this.offset.click.top < this.containment[1]) ? top - o.grid[1] : top + o.grid[1])) : top;
912
+
913
+ var left = this.originalPageX + Math.round((pageX - this.originalPageX) / o.grid[0]) * o.grid[0];
914
+ pageX = this.containment ? (!(left - this.offset.click.left < this.containment[0] || left - this.offset.click.left > this.containment[2]) ? left : (!(left - this.offset.click.left < this.containment[0]) ? left - o.grid[0] : left + o.grid[0])) : left;
915
+ }
916
+
917
+ }
918
+
919
+ return {
920
+ top: (
921
+ pageY // The absolute mouse position
922
+ - this.offset.click.top // Click offset (relative to the element)
923
+ - this.offset.relative.top // Only for relative positioned nodes: Relative offset from element to offset parent
924
+ - this.offset.parent.top // The offsetParent's offset without borders (offset + border)
925
+ + ($.browser.safari && this.cssPosition == 'fixed' ? 0 : ( this.cssPosition == 'fixed' ? -this.scrollParent.scrollTop() : ( scrollIsRootNode ? 0 : scroll.scrollTop() ) ))
926
+ ),
927
+ left: (
928
+ pageX // The absolute mouse position
929
+ - this.offset.click.left // Click offset (relative to the element)
930
+ - this.offset.relative.left // Only for relative positioned nodes: Relative offset from element to offset parent
931
+ - this.offset.parent.left // The offsetParent's offset without borders (offset + border)
932
+ + ($.browser.safari && this.cssPosition == 'fixed' ? 0 : ( this.cssPosition == 'fixed' ? -this.scrollParent.scrollLeft() : scrollIsRootNode ? 0 : scroll.scrollLeft() ))
933
+ )
934
+ };
935
+
936
+ },
937
+
938
+ _rearrange: function(event, i, a, hardRefresh) {
939
+
940
+ a ? a[0].appendChild(this.placeholder[0]) : i.item[0].parentNode.insertBefore(this.placeholder[0], (this.direction == 'down' ? i.item[0] : i.item[0].nextSibling));
941
+
942
+ //Various things done here to improve the performance:
943
+ // 1. we create a setTimeout, that calls refreshPositions
944
+ // 2. on the instance, we have a counter variable, that get's higher after every append
945
+ // 3. on the local scope, we copy the counter variable, and check in the timeout, if it's still the same
946
+ // 4. this lets only the last addition to the timeout stack through
947
+ this.counter = this.counter ? ++this.counter : 1;
948
+ var self = this, counter = this.counter;
949
+
950
+ window.setTimeout(function() {
951
+ if(counter == self.counter) self.refreshPositions(!hardRefresh); //Precompute after each DOM insertion, NOT on mousemove
952
+ },0);
953
+
954
+ },
955
+
956
+ _clear: function(event, noPropagation) {
957
+
958
+ this.reverting = false;
959
+ // We delay all events that have to be triggered to after the point where the placeholder has been removed and
960
+ // everything else normalized again
961
+ var delayedTriggers = [], self = this;
962
+
963
+ // We first have to update the dom position of the actual currentItem
964
+ // Note: don't do it if the current item is already removed (by a user), or it gets reappended (see #4088)
965
+ if(!this._noFinalSort && this.currentItem[0].parentNode) this.placeholder.before(this.currentItem);
966
+ this._noFinalSort = null;
967
+
968
+ if(this.helper[0] == this.currentItem[0]) {
969
+ for(var i in this._storedCSS) {
970
+ if(this._storedCSS[i] == 'auto' || this._storedCSS[i] == 'static') this._storedCSS[i] = '';
971
+ }
972
+ this.currentItem.css(this._storedCSS).removeClass("ui-sortable-helper");
973
+ } else {
974
+ this.currentItem.show();
975
+ }
976
+
977
+ if(this.fromOutside && !noPropagation) delayedTriggers.push(function(event) { this._trigger("receive", event, this._uiHash(this.fromOutside)); });
978
+ if((this.fromOutside || this.domPosition.prev != this.currentItem.prev().not(".ui-sortable-helper")[0] || this.domPosition.parent != this.currentItem.parent()[0]) && !noPropagation) delayedTriggers.push(function(event) { this._trigger("update", event, this._uiHash()); }); //Trigger update callback if the DOM position has changed
979
+ if(!$.ui.contains(this.element[0], this.currentItem[0])) { //Node was moved out of the current element
980
+ if(!noPropagation) delayedTriggers.push(function(event) { this._trigger("remove", event, this._uiHash()); });
981
+ for (var i = this.containers.length - 1; i >= 0; i--){
982
+ if($.ui.contains(this.containers[i].element[0], this.currentItem[0]) && !noPropagation) {
983
+ delayedTriggers.push((function(c) { return function(event) { c._trigger("receive", event, this._uiHash(this)); }; }).call(this, this.containers[i]));
984
+ delayedTriggers.push((function(c) { return function(event) { c._trigger("update", event, this._uiHash(this)); }; }).call(this, this.containers[i]));
985
+ }
986
+ };
987
+ };
988
+
989
+ //Post events to containers
990
+ for (var i = this.containers.length - 1; i >= 0; i--){
991
+ if(!noPropagation) delayedTriggers.push((function(c) { return function(event) { c._trigger("deactivate", event, this._uiHash(this)); }; }).call(this, this.containers[i]));
992
+ if(this.containers[i].containerCache.over) {
993
+ delayedTriggers.push((function(c) { return function(event) { c._trigger("out", event, this._uiHash(this)); }; }).call(this, this.containers[i]));
994
+ this.containers[i].containerCache.over = 0;
995
+ }
996
+ }
997
+
998
+ //Do what was originally in plugins
999
+ if(this._storedCursor) $('body').css("cursor", this._storedCursor); //Reset cursor
1000
+ if(this._storedOpacity) this.helper.css("opacity", this._storedOpacity); //Reset opacity
1001
+ if(this._storedZIndex) this.helper.css("zIndex", this._storedZIndex == 'auto' ? '' : this._storedZIndex); //Reset z-index
1002
+
1003
+ this.dragging = false;
1004
+ if(this.cancelHelperRemoval) {
1005
+ if(!noPropagation) {
1006
+ this._trigger("beforeStop", event, this._uiHash());
1007
+ for (var i=0; i < delayedTriggers.length; i++) { delayedTriggers[i].call(this, event); }; //Trigger all delayed events
1008
+ this._trigger("stop", event, this._uiHash());
1009
+ }
1010
+ return false;
1011
+ }
1012
+
1013
+ if(!noPropagation) this._trigger("beforeStop", event, this._uiHash());
1014
+
1015
+ //$(this.placeholder[0]).remove(); would have been the jQuery way - unfortunately, it unbinds ALL events from the original node!
1016
+ this.placeholder[0].parentNode.removeChild(this.placeholder[0]);
1017
+
1018
+ if(this.helper[0] != this.currentItem[0]) this.helper.remove(); this.helper = null;
1019
+
1020
+ if(!noPropagation) {
1021
+ for (var i=0; i < delayedTriggers.length; i++) { delayedTriggers[i].call(this, event); }; //Trigger all delayed events
1022
+ this._trigger("stop", event, this._uiHash());
1023
+ }
1024
+
1025
+ this.fromOutside = false;
1026
+ return true;
1027
+
1028
+ },
1029
+
1030
+ _trigger: function() {
1031
+ if ($.Widget.prototype._trigger.apply(this, arguments) === false) {
1032
+ this.cancel();
1033
+ }
1034
+ },
1035
+
1036
+ _uiHash: function(inst) {
1037
+ var self = inst || this;
1038
+ return {
1039
+ helper: self.helper,
1040
+ placeholder: self.placeholder || $([]),
1041
+ position: self.position,
1042
+ originalPosition: self.originalPosition,
1043
+ offset: self.positionAbs,
1044
+ item: self.currentItem,
1045
+ sender: inst ? inst.element : null
1046
+ };
1047
+ }
1048
+
1049
+ });
1050
+
1051
+ $.extend($.ui.sortable, {
1052
+ version: "1.8"
1053
+ });
1054
+
1055
+ })(jQuery);