Sortable.js 26.8 KB
Newer Older
R
RubaXa 已提交
1 2 3 4 5 6 7
/**!
 * Sortable
 * @author	RubaXa   <trash@rubaxa.org>
 * @license MIT
 */


R
RubaXa 已提交
8
(function (factory) {
R
RubaXa 已提交
9 10
	"use strict";

R
RubaXa 已提交
11
	if (typeof define === "function" && define.amd) {
R
RubaXa 已提交
12
		define(factory);
R
RubaXa 已提交
13
	}
R
RubaXa 已提交
14
	else if (typeof module != "undefined" && typeof module.exports != "undefined") {
S
Scott Nelson 已提交
15 16
		module.exports = factory();
	}
R
RubaXa 已提交
17
	else if (typeof Package !== "undefined") {
18 19
		Sortable = factory();  // export for Meteor.js
	}
R
RubaXa 已提交
20
	else {
R
RubaXa 已提交
21
		/* jshint sub:true */
R
RubaXa 已提交
22 23
		window["Sortable"] = factory();
	}
R
RubaXa 已提交
24
})(function () {
R
RubaXa 已提交
25 26
	"use strict";

R
RubaXa 已提交
27
	var dragEl,
28
		parentEl,
R
RubaXa 已提交
29 30 31 32
		ghostEl,
		cloneEl,
		rootEl,
		nextEl,
R
RubaXa 已提交
33

34 35 36
		scrollEl,
		scrollParentEl,

R
RubaXa 已提交
37 38
		lastEl,
		lastCSS,
R
RubaXa 已提交
39

R
RubaXa 已提交
40 41 42
		oldIndex,
		newIndex,

R
RubaXa 已提交
43
		activeGroup,
R
RubaXa 已提交
44
		autoScroll = {},
R
RubaXa 已提交
45

R
RubaXa 已提交
46 47
		tapEvt,
		touchEvt,
R
RubaXa 已提交
48

C
ChiefORZ 已提交
49 50
		moved,

R
RubaXa 已提交
51 52 53
		/** @const */
		RSPACE = /\s+/g,

R
RubaXa 已提交
54
		expando = 'Sortable' + (new Date).getTime(),
R
RubaXa 已提交
55

R
RubaXa 已提交
56 57 58
		win = window,
		document = win.document,
		parseInt = win.parseInt,
R
RubaXa 已提交
59

R
RubaXa 已提交
60 61
		supportDraggable = !!('draggable' in document.createElement('div')),

R
RubaXa 已提交
62
		_silent = false,
R
RubaXa 已提交
63

R
RubaXa 已提交
64
		abs = Math.abs,
R
RubaXa 已提交
65
		slice = [].slice,
R
RubaXa 已提交
66

67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132
		touchDragOverListeners = [],

		_autoScroll = _throttle(function (/**Event*/evt, /**Object*/options, /**HTMLElement*/rootEl) {
			// Bug: https://bugzilla.mozilla.org/show_bug.cgi?id=505521
			if (rootEl && options.scroll) {
				var el,
					rect,
					sens = options.scrollSensitivity,
					speed = options.scrollSpeed,

					x = evt.clientX,
					y = evt.clientY,

					winWidth = window.innerWidth,
					winHeight = window.innerHeight,

					vx,
					vy
				;

				// Delect scrollEl
				if (scrollParentEl !== rootEl) {
					scrollEl = options.scroll;
					scrollParentEl = rootEl;

					if (scrollEl === true) {
						scrollEl = rootEl;

						do {
							if ((scrollEl.offsetWidth < scrollEl.scrollWidth) ||
								(scrollEl.offsetHeight < scrollEl.scrollHeight)
							) {
								break;
							}
							/* jshint boss:true */
						} while (scrollEl = scrollEl.parentNode);
					}
				}

				if (scrollEl) {
					el = scrollEl;
					rect = scrollEl.getBoundingClientRect();
					vx = (abs(rect.right - x) <= sens) - (abs(rect.left - x) <= sens);
					vy = (abs(rect.bottom - y) <= sens) - (abs(rect.top - y) <= sens);
				}


				if (!(vx || vy)) {
					vx = (winWidth - x <= sens) - (x <= sens);
					vy = (winHeight - y <= sens) - (y <= sens);

					/* jshint expr:true */
					(vx || vy) && (el = win);
				}


				if (autoScroll.vx !== vx || autoScroll.vy !== vy || autoScroll.el !== el) {
					autoScroll.el = el;
					autoScroll.vx = vx;
					autoScroll.vy = vy;

					clearInterval(autoScroll.pid);

					if (el) {
						autoScroll.pid = setInterval(function () {
							if (el === win) {
133
								win.scrollTo(win.pageXOffset + vx * speed, win.pageYOffset + vy * speed);
134 135 136 137 138 139 140 141 142
							} else {
								vy && (el.scrollTop += vy * speed);
								vx && (el.scrollLeft += vx * speed);
							}
						}, 24);
					}
				}
			}
		}, 30)
R
RubaXa 已提交
143 144 145
	;


146

R
RubaXa 已提交
147 148 149
	/**
	 * @class  Sortable
	 * @param  {HTMLElement}  el
150
	 * @param  {Object}       [options]
R
RubaXa 已提交
151
	 */
R
RubaXa 已提交
152
	function Sortable(el, options) {
R
RubaXa 已提交
153
		this.el = el; // root element
154
		this.options = options = _extend({}, options);
R
RubaXa 已提交
155 156


157 158 159 160
		// Export instance
		el[expando] = this;


R
RubaXa 已提交
161
		// Default options
162 163
		var defaults = {
			group: Math.random(),
R
RubaXa 已提交
164
			sort: true,
R
RubaXa 已提交
165
			disabled: false,
166 167
			store: null,
			handle: null,
R
RubaXa 已提交
168 169 170
			scroll: true,
			scrollSensitivity: 30,
			scrollSpeed: 10,
R
RubaXa 已提交
171
			draggable: /[uo]l/i.test(el.nodeName) ? 'li' : '>*',
172 173
			ghostClass: 'sortable-ghost',
			ignore: 'a, img',
174
			filter: null,
R
RubaXa 已提交
175 176 177
			animation: 0,
			setData: function (dataTransfer, dragEl) {
				dataTransfer.setData('Text', dragEl.textContent);
178 179
			},
			dropBubble: false,
R
RubaXa 已提交
180
			dragoverBubble: false,
181
			dataIdAttr: 'data-id',
182
			delay: 0,
C
ChiefORZ 已提交
183 184 185
			forceFallback: false,
			fallbackClass: 'sortable-fallback',
			fallbackOnBody: false
R
RubaXa 已提交
186
		};
187

R
RubaXa 已提交
188

189 190
		// Set default options
		for (var name in defaults) {
R
RubaXa 已提交
191
			!(name in options) && (options[name] = defaults[name]);
192
		}
R
RubaXa 已提交
193

R
RubaXa 已提交
194 195
		var group = options.group;

R
RubaXa 已提交
196 197
		if (!group || typeof group != 'object') {
			group = options.group = { name: group };
R
RubaXa 已提交
198 199
		}

R
RubaXa 已提交
200

R
RubaXa 已提交
201
		['pull', 'put'].forEach(function (key) {
R
RubaXa 已提交
202 203
			if (!(key in group)) {
				group[key] = true;
R
RubaXa 已提交
204 205 206 207
			}
		});


208
		options.groups = ' ' + group.name + (group.put.join ? ' ' + group.put.join(' ') : '') + ' ';
R
RubaXa 已提交
209 210


R
* JSDoc  
RubaXa 已提交
211
		// Bind all private methods
R
RubaXa 已提交
212 213
		for (var fn in this) {
			if (fn.charAt(0) === '_') {
214
				this[fn] = this[fn].bind(this);
R
RubaXa 已提交
215 216 217
			}
		}

R
RubaXa 已提交
218 219
		// Setup drag mode
		this.nativeDraggable = options.forceFallback ? false : supportDraggable;
R
RubaXa 已提交
220 221 222 223 224

		// Bind events
		_on(el, 'mousedown', this._onTapStart);
		_on(el, 'touchstart', this._onTapStart);

R
RubaXa 已提交
225
		if (this.nativeDraggable) {
226 227 228
			_on(el, 'dragover', this);
			_on(el, 'dragenter', this);
		}
R
RubaXa 已提交
229 230

		touchDragOverListeners.push(this._onDragOver);
231 232 233

		// Restore sorting
		options.store && this.sort(options.store.get(this));
R
RubaXa 已提交
234 235 236
	}


237
	Sortable.prototype = /** @lends Sortable.prototype */ {
R
RubaXa 已提交
238 239
		constructor: Sortable,

240
		_onTapStart: function (/** Event|TouchEvent */evt) {
241 242
			var _this = this,
				el = this.el,
243 244 245 246 247 248
				options = this.options,
				type = evt.type,
				touch = evt.touches && evt.touches[0],
				target = (touch || evt).target,
				originalTarget = target,
				filter = options.filter;
R
RubaXa 已提交
249 250


251 252
			if (type === 'mousedown' && evt.button !== 0 || options.disabled) {
				return; // only left button or enabled
R
RubaXa 已提交
253
			}
R
RubaXa 已提交
254

255
			target = _closest(target, options.draggable, el);
256

257 258
			if (!target) {
				return;
259
			}
260 261 262 263 264 265 266

			// get the index of the dragged element within its parent
			oldIndex = _index(target);

			// Check filter
			if (typeof filter === 'function') {
				if (filter.call(this, evt, target, this)) {
267
					_dispatchEvent(_this, originalTarget, 'filter', target, el, oldIndex);
268 269 270
					evt.preventDefault();
					return; // cancel dnd
				}
271
			}
272 273 274
			else if (filter) {
				filter = filter.split(',').some(function (criteria) {
					criteria = _closest(originalTarget, criteria.trim(), el);
275

276
					if (criteria) {
277
						_dispatchEvent(_this, criteria, 'filter', target, el, oldIndex);
278 279 280 281 282 283 284
						return true;
					}
				});

				if (filter) {
					evt.preventDefault();
					return; // cancel dnd
285 286 287 288
				}
			}


289 290 291
			if (options.handle && !_closest(originalTarget, options.handle, el)) {
				return;
			}
292

293 294 295

			// Prepare `dragstart`
			this._prepareDragStart(evt, touch, target);
296 297
		},

298 299 300 301 302 303
		_prepareDragStart: function (/** Event */evt, /** Touch */touch, /** HTMLElement */target) {
			var _this = this,
				el = _this.el,
				options = _this.options,
				ownerDocument = el.ownerDocument,
				dragStartFn;
304 305 306 307

			if (target && !dragEl && (target.parentNode === el)) {
				tapEvt = evt;

308
				rootEl = el;
309
				dragEl = target;
310
				parentEl = target.parentNode;
311
				nextEl = dragEl.nextSibling;
312 313 314 315 316 317
				activeGroup = options.group;

				dragStartFn = function () {
					// Delayed drag has been triggered
					// we can re-enable the events: touchmove/mousemove
					_this._disableDelayedDrag();
318

319 320
					// Make the element draggable
					dragEl.draggable = true;
321

322 323 324 325 326 327 328 329 330 331 332 333 334 335
					// Disable "draggable"
					options.ignore.split(',').forEach(function (criteria) {
						_find(dragEl, criteria.trim(), _disableDraggable);
					});

					// Bind the events: dragstart/dragend
					_this._triggerDragStart(touch);
				};

				_on(ownerDocument, 'mouseup', _this._onDrop);
				_on(ownerDocument, 'touchend', _this._onDrop);
				_on(ownerDocument, 'touchcancel', _this._onDrop);

				if (options.delay) {
336 337
					// If the user moves the pointer or let go the click or touch
					// before the delay has been reached:
338
					// disable the delayed drag
339 340 341
					_on(ownerDocument, 'mouseup', _this._disableDelayedDrag);
					_on(ownerDocument, 'touchend', _this._disableDelayedDrag);
					_on(ownerDocument, 'touchcancel', _this._disableDelayedDrag);
342 343 344 345 346 347
					_on(ownerDocument, 'mousemove', _this._disableDelayedDrag);
					_on(ownerDocument, 'touchmove', _this._disableDelayedDrag);

					_this._dragStartTimer = setTimeout(dragStartFn, options.delay);
				} else {
					dragStartFn();
348 349 350
				}
			}
		},
R
RubaXa 已提交
351

352 353
		_disableDelayedDrag: function () {
			var ownerDocument = this.el.ownerDocument;
354

355
			clearTimeout(this._dragStartTimer);
356 357 358
			_off(ownerDocument, 'mouseup', this._disableDelayedDrag);
			_off(ownerDocument, 'touchend', this._disableDelayedDrag);
			_off(ownerDocument, 'touchcancel', this._disableDelayedDrag);
359 360 361
			_off(ownerDocument, 'mousemove', this._disableDelayedDrag);
			_off(ownerDocument, 'touchmove', this._disableDelayedDrag);
		},
R
RubaXa 已提交
362

363 364 365 366 367 368 369 370
		_triggerDragStart: function (/** Touch */touch) {
			if (touch) {
				// Touch device support
				tapEvt = {
					target: dragEl,
					clientX: touch.clientX,
					clientY: touch.clientY
				};
371

372
				this._onDragStart(tapEvt, 'touch');
R
RubaXa 已提交
373
			}
R
RubaXa 已提交
374
			else if (!this.nativeDraggable) {
375 376 377 378 379
				this._onDragStart(tapEvt, true);
			}
			else {
				_on(dragEl, 'dragend', this);
				_on(rootEl, 'dragstart', this._onDragStart);
380 381
			}

382 383 384 385 386
			try {
				if (document.selection) {
					document.selection.empty();
				} else {
					window.getSelection().removeAllRanges();
387
				}
388
			} catch (err) {
389
			}
390
		},
391

392 393 394 395
		_dragStarted: function () {
			if (rootEl && dragEl) {
				// Apply effect
				_toggleClass(dragEl, this.options.ghostClass, true);
R
RubaXa 已提交
396

397
				Sortable.active = this;
398

399
				// Drag start event
400
				_dispatchEvent(this, rootEl, 'start', dragEl, rootEl, oldIndex);
401
			}
R
RubaXa 已提交
402 403
		},

R
RubaXa 已提交
404 405
		_emulateDragOver: function () {
			if (touchEvt) {
R
RubaXa 已提交
406
				if (this._lastX === touchEvt.clientX && this._lastY === touchEvt.clientY) {
407 408
					return;
				}
R
RubaXa 已提交
409 410 411

				this._lastX = touchEvt.clientX;
				this._lastY = touchEvt.clientY;
412

R
RubaXa 已提交
413 414
				_css(ghostEl, 'display', 'none');

R
RubaXa 已提交
415
				var target = document.elementFromPoint(touchEvt.clientX, touchEvt.clientY),
R
RubaXa 已提交
416
					parent = target,
417
					groupName = ' ' + this.options.group.name + '',
R
RubaXa 已提交
418
					i = touchDragOverListeners.length;
R
RubaXa 已提交
419

R
RubaXa 已提交
420 421
				if (parent) {
					do {
422
						if (parent[expando] && parent[expando].options.groups.indexOf(groupName) > -1) {
R
RubaXa 已提交
423 424 425 426 427 428 429 430 431 432 433 434 435
							while (i--) {
								touchDragOverListeners[i]({
									clientX: touchEvt.clientX,
									clientY: touchEvt.clientY,
									target: target,
									rootEl: parent
								});
							}

							break;
						}

						target = parent; // store last element
L
Larry Davis 已提交
436
					}
R
RubaXa 已提交
437 438
					/* jshint boss:true */
					while (parent = parent.parentNode);
R
RubaXa 已提交
439 440 441 442 443 444 445
				}

				_css(ghostEl, 'display', '');
			}
		},


R
RubaXa 已提交
446 447
		_onTouchMove: function (/**TouchEvent*/evt) {
			if (tapEvt) {
448
				// only set the status to dragging, when we are actually dragging
R
RubaXa 已提交
449
				if (!Sortable.active) {
450 451
					this._dragStarted();
				}
R
RubaXa 已提交
452

453 454
				// as well as creating the ghost element on the document body
				this._appendGhost();
R
RubaXa 已提交
455

R
RubaXa 已提交
456
				var touch = evt.touches ? evt.touches[0] : evt,
R
RubaXa 已提交
457 458
					dx = touch.clientX - tapEvt.clientX,
					dy = touch.clientY - tapEvt.clientY,
R
RubaXa 已提交
459
					translate3d = evt.touches ? 'translate3d(' + dx + 'px,' + dy + 'px,0)' : 'translate(' + dx + 'px,' + dy + 'px)';
R
RubaXa 已提交
460

A
Alex Wild 已提交
461
				moved = true;
R
RubaXa 已提交
462
				touchEvt = touch;
R
RubaXa 已提交
463 464 465 466 467 468

				_css(ghostEl, 'webkitTransform', translate3d);
				_css(ghostEl, 'mozTransform', translate3d);
				_css(ghostEl, 'msTransform', translate3d);
				_css(ghostEl, 'transform', translate3d);

M
Marius Petcu 已提交
469
				evt.preventDefault();
R
RubaXa 已提交
470 471 472
			}
		},

R
RubaXa 已提交
473 474
		_appendGhost: function () {
			if (!ghostEl) {
R
RubaXa 已提交
475 476 477
				var rect = dragEl.getBoundingClientRect(),
					css = _css(dragEl),
					ghostRect;
R
RubaXa 已提交
478

479
				ghostEl = dragEl.cloneNode(true);
R
RubaXa 已提交
480

C
ChiefORZ 已提交
481 482 483
				_toggleClass(ghostEl, this.options.ghostClass, false);
				_toggleClass(ghostEl, this.options.fallbackClass, true);

R
RubaXa 已提交
484 485
				_css(ghostEl, 'top', rect.top - parseInt(css.marginTop, 10));
				_css(ghostEl, 'left', rect.left - parseInt(css.marginLeft, 10));
R
RubaXa 已提交
486 487
				_css(ghostEl, 'width', rect.width);
				_css(ghostEl, 'height', rect.height);
R
RubaXa 已提交
488 489 490 491
				_css(ghostEl, 'opacity', '0.8');
				_css(ghostEl, 'position', 'fixed');
				_css(ghostEl, 'zIndex', '100000');

C
ChiefORZ 已提交
492
				this.options.fallbackOnBody && document.body.appendChild(ghostEl) || rootEl.appendChild(ghostEl);
R
RubaXa 已提交
493 494 495

				// Fixing dimensions.
				ghostRect = ghostEl.getBoundingClientRect();
R
RubaXa 已提交
496 497
				_css(ghostEl, 'width', rect.width * 2 - ghostRect.width);
				_css(ghostEl, 'height', rect.height * 2 - ghostRect.height);
498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513
			}
		},

		_onDragStart: function (/**Event*/evt, /**boolean*/useFallback) {
			var dataTransfer = evt.dataTransfer,
				options = this.options;

			this._offUpEvents();

			if (activeGroup.pull == 'clone') {
				cloneEl = dragEl.cloneNode(true);
				_css(cloneEl, 'display', 'none');
				rootEl.insertBefore(cloneEl, dragEl);
			}

			if (useFallback) {
R
RubaXa 已提交
514

R
RubaXa 已提交
515 516 517 518 519 520 521 522 523 524
				if (useFallback === 'touch') {
					// Bind touch events
					_on(document, 'touchmove', this._onTouchMove);
					_on(document, 'touchend', this._onDrop);
					_on(document, 'touchcancel', this._onDrop);
				} else {
					// Old brwoser
					_on(document, 'mousemove', this._onTouchMove);
					_on(document, 'mouseup', this._onDrop);
				}
R
RubaXa 已提交
525

526
				this._loopId = setInterval(this._emulateDragOver, 50);
R
RubaXa 已提交
527 528
			}
			else {
R
RubaXa 已提交
529 530 531 532
				if (dataTransfer) {
					dataTransfer.effectAllowed = 'move';
					options.setData && options.setData.call(this, dataTransfer, dragEl);
				}
R
RubaXa 已提交
533

R
RubaXa 已提交
534
				_on(document, 'drop', this);
535
				setTimeout(this._dragStarted, 0);
R
RubaXa 已提交
536 537 538
			}
		},

R
RubaXa 已提交
539
		_onDragOver: function (/**Event*/evt) {
R
RubaXa 已提交
540 541 542 543 544 545
			var el = this.el,
				target,
				dragRect,
				revert,
				options = this.options,
				group = options.group,
R
RubaXa 已提交
546
				groupPut = group.put,
547 548
				isOwner = (activeGroup === group),
				canSort = options.sort;
R
RubaXa 已提交
549

R
RubaXa 已提交
550 551
			if (evt.preventDefault !== void 0) {
				evt.preventDefault();
552
				!options.dragoverBubble && evt.stopPropagation();
R
RubaXa 已提交
553
			}
R
RubaXa 已提交
554

A
Alex Wild 已提交
555 556
			moved = true;

R
RubaXa 已提交
557
			if (activeGroup && !options.disabled &&
558
				(isOwner
R
RubaXa 已提交
559
					? canSort || (revert = !rootEl.contains(dragEl)) // Reverting item into the original list
R
RubaXa 已提交
560 561 562 563
					: activeGroup.pull && groupPut && (
						(activeGroup.name === group.name) || // by Name
						(groupPut.indexOf && ~groupPut.indexOf(activeGroup.name)) // by Array
					)
564
				) &&
R
RubaXa 已提交
565
				(evt.rootEl === void 0 || evt.rootEl === this.el) // touch fallback
R
RubaXa 已提交
566
			) {
R
RubaXa 已提交
567 568 569 570 571 572 573
				// Smart auto-scrolling
				_autoScroll(evt, options, this.el);

				if (_silent) {
					return;
				}

R
RubaXa 已提交
574
				target = _closest(evt.target, options.draggable, el);
R
RubaXa 已提交
575 576 577
				dragRect = dragEl.getBoundingClientRect();


578
				if (revert) {
R
RubaXa 已提交
579 580
					_cloneHide(true);

581 582 583 584 585 586 587
					if (cloneEl || nextEl) {
						rootEl.insertBefore(dragEl, cloneEl || nextEl);
					}
					else if (!canSort) {
						rootEl.appendChild(dragEl);
					}

R
RubaXa 已提交
588 589
					return;
				}
R
RubaXa 已提交
590

R
RubaXa 已提交
591

R
RubaXa 已提交
592
				if ((el.children.length === 0) || (el.children[0] === ghostEl) ||
R
RubaXa 已提交
593
					(el === evt.target) && (target = _ghostInBottom(el, evt))
R
RubaXa 已提交
594
				) {
R
RubaXa 已提交
595 596 597 598 599 600
					if (target) {
						if (target.animated) {
							return;
						}
						targetRect = target.getBoundingClientRect();
					}
R
RubaXa 已提交
601

R
RubaXa 已提交
602 603
					_cloneHide(isOwner);

R
RubaXa 已提交
604
					if (_onMove(rootEl, el, dragEl, dragRect, target, targetRect) !== false) {
R
RubaXa 已提交
605 606 607 608
						el.appendChild(dragEl);
						this._animate(dragRect, dragEl);
						target && this._animate(targetRect, target);
					}
R
RubaXa 已提交
609
				}
R
RubaXa 已提交
610 611
				else if (target && !target.animated && target !== dragEl && (target.parentNode[expando] !== void 0)) {
					if (lastEl !== target) {
R
RubaXa 已提交
612
						lastEl = target;
R
RubaXa 已提交
613
						lastCSS = _css(target);
R
RubaXa 已提交
614 615 616
					}


R
RubaXa 已提交
617 618 619 620 621 622 623 624
					var targetRect = target.getBoundingClientRect(),
						width = targetRect.right - targetRect.left,
						height = targetRect.bottom - targetRect.top,
						floating = /left|right|inline/.test(lastCSS.cssFloat + lastCSS.display),
						isWide = (target.offsetWidth > dragEl.offsetWidth),
						isLong = (target.offsetHeight > dragEl.offsetHeight),
						halfway = (floating ? (evt.clientX - targetRect.left) / width : (evt.clientY - targetRect.top) / height) > 0.5,
						nextSibling = target.nextElementSibling,
R
RubaXa 已提交
625
						moveVector = _onMove(rootEl, el, dragEl, dragRect, target, targetRect),
R
RubaXa 已提交
626
						after
R
RubaXa 已提交
627
					;
R
RubaXa 已提交
628

R
RubaXa 已提交
629
					if (moveVector !== false) {
R
RubaXa 已提交
630 631
						_silent = true;
						setTimeout(_unsilent, 30);
R
RubaXa 已提交
632

R
RubaXa 已提交
633
						_cloneHide(isOwner);
R
RubaXa 已提交
634

R
RubaXa 已提交
635 636 637 638
						if (moveVector === 1 || moveVector === -1) {
							after = (moveVector === 1);
						}
						else if (floating) {
R
RubaXa 已提交
639 640 641 642
							after = (target.previousElementSibling === dragEl) && !isWide || halfway && isWide;
						} else {
							after = (nextSibling !== dragEl) && !isLong || halfway && isLong;
						}
R
RubaXa 已提交
643

R
RubaXa 已提交
644 645 646 647 648
						if (after && !nextSibling) {
							el.appendChild(dragEl);
						} else {
							target.parentNode.insertBefore(dragEl, after ? nextSibling : target);
						}
R
RubaXa 已提交
649

R
RubaXa 已提交
650 651 652
						this._animate(dragRect, dragEl);
						this._animate(targetRect, target);
					}
R
RubaXa 已提交
653 654 655 656
				}
			}
		},

657 658 659 660 661 662
		_animate: function (prevRect, target) {
			var ms = this.options.animation;

			if (ms) {
				var currentRect = target.getBoundingClientRect();

R
RubaXa 已提交
663
				_css(target, 'transition', 'none');
664 665 666 667 668 669 670
				_css(target, 'transform', 'translate3d('
					+ (prevRect.left - currentRect.left) + 'px,'
					+ (prevRect.top - currentRect.top) + 'px,0)'
				);

				target.offsetWidth; // repaint

R
RubaXa 已提交
671
				_css(target, 'transition', 'all ' + ms + 'ms');
672 673
				_css(target, 'transform', 'translate3d(0,0,0)');

R
* anim  
RubaXa 已提交
674 675
				clearTimeout(target.animated);
				target.animated = setTimeout(function () {
676
					_css(target, 'transition', '');
677
					_css(target, 'transform', '');
678 679 680 681 682
					target.animated = false;
				}, ms);
			}
		},

683
		_offUpEvents: function () {
684 685
			var ownerDocument = this.el.ownerDocument;

686
			_off(document, 'touchmove', this._onTouchMove);
687 688 689
			_off(ownerDocument, 'mouseup', this._onDrop);
			_off(ownerDocument, 'touchend', this._onDrop);
			_off(ownerDocument, 'touchcancel', this._onDrop);
690
		},
R
RubaXa 已提交
691

R
RubaXa 已提交
692
		_onDrop: function (/**Event*/evt) {
693 694
			var el = this.el,
				options = this.options;
R
RubaXa 已提交
695

R
RubaXa 已提交
696
			clearInterval(this._loopId);
R
RubaXa 已提交
697
			clearInterval(autoScroll.pid);
R
RubaXa 已提交
698
			clearTimeout(this._dragStartTimer);
699

R
RubaXa 已提交
700
			// Unbind events
R
RubaXa 已提交
701
			_off(document, 'mousemove', this._onTouchMove);
702

R
RubaXa 已提交
703
			if (this.nativeDraggable) {
704 705 706
				_off(document, 'drop', this);
				_off(el, 'dragstart', this._onDragStart);
			}
R
RubaXa 已提交
707

708
			this._offUpEvents();
R
RubaXa 已提交
709

R
RubaXa 已提交
710
			if (evt) {
R
RubaXa 已提交
711
				if (moved) {
C
ChiefORZ 已提交
712 713 714
					evt.preventDefault();
					!options.dropBubble && evt.stopPropagation();
				}
R
RubaXa 已提交
715

R
RubaXa 已提交
716
				ghostEl && ghostEl.parentNode.removeChild(ghostEl);
R
RubaXa 已提交
717

R
RubaXa 已提交
718
				if (dragEl) {
R
RubaXa 已提交
719
					if (this.nativeDraggable) {
720 721
						_off(dragEl, 'dragend', this);
					}
R
RubaXa 已提交
722

723
					_disableDraggable(dragEl);
R
RubaXa 已提交
724 725
					_toggleClass(dragEl, this.options.ghostClass, false);

726
					if (rootEl !== parentEl) {
R
RubaXa 已提交
727
						newIndex = _index(dragEl);
728 729 730 731
						if (newIndex != -1) {
							// drag from one list and drop into another
							_dispatchEvent(null, parentEl, 'sort', dragEl, rootEl, oldIndex, newIndex);
							_dispatchEvent(this, rootEl, 'sort', dragEl, rootEl, oldIndex, newIndex);
R
RubaXa 已提交
732

733 734
							// Add event
							_dispatchEvent(null, parentEl, 'add', dragEl, rootEl, oldIndex, newIndex);
735

736 737 738
							// Remove event
							_dispatchEvent(this, rootEl, 'remove', dragEl, rootEl, oldIndex, newIndex);
						}
R
RubaXa 已提交
739
					}
R
RubaXa 已提交
740 741
					else {
						// Remove clone
R
RubaXa 已提交
742
						cloneEl && cloneEl.parentNode.removeChild(cloneEl);
R
RubaXa 已提交
743

R
RubaXa 已提交
744 745 746
						if (dragEl.nextSibling !== nextEl) {
							// Get the index of the dragged element within its parent
							newIndex = _index(dragEl);
747 748 749 750 751
							if (newIndex != -1) {
								// drag & drop within the same list
								_dispatchEvent(this, rootEl, 'update', dragEl, rootEl, oldIndex, newIndex);
								_dispatchEvent(this, rootEl, 'sort', dragEl, rootEl, oldIndex, newIndex);
							}
R
RubaXa 已提交
752
						}
R
RubaXa 已提交
753
					}
754

R
RubaXa 已提交
755 756 757 758 759 760 761
					if (Sortable.active) {
						// Drag end event
						_dispatchEvent(this, rootEl, 'end', dragEl, rootEl, oldIndex, newIndex);

						// Save sorting
						this.save();
					}
R
RubaXa 已提交
762
				}
763
				
764
				// Nulling
R
RubaXa 已提交
765 766
				rootEl =
				dragEl =
767
				parentEl =
R
RubaXa 已提交
768 769
				ghostEl =
				nextEl =
R
RubaXa 已提交
770
				cloneEl =
R
RubaXa 已提交
771

772 773 774
				scrollEl =
				scrollParentEl =

R
RubaXa 已提交
775 776 777
				tapEvt =
				touchEvt =

C
ChiefORZ 已提交
778 779
				moved =

R
RubaXa 已提交
780 781 782
				lastEl =
				lastCSS =

R
RubaXa 已提交
783 784
				activeGroup =
				Sortable.active = null;
R
RubaXa 已提交
785 786 787 788
			}
		},


R
RubaXa 已提交
789 790 791
		handleEvent: function (/**Event*/evt) {
			var type = evt.type;

R
RubaXa 已提交
792
			if (type === 'dragover' || type === 'dragenter') {
R
RubaXa 已提交
793 794 795 796
				if (dragEl) {
					this._onDragOver(evt);
					_globalDragOver(evt);
				}
R
RubaXa 已提交
797
			}
R
RubaXa 已提交
798
			else if (type === 'drop' || type === 'dragend') {
R
RubaXa 已提交
799 800
				this._onDrop(evt);
			}
R
RubaXa 已提交
801 802 803
		},


804 805 806 807 808 809 810 811 812
		/**
		 * Serializes the item into an array of string.
		 * @returns {String[]}
		 */
		toArray: function () {
			var order = [],
				el,
				children = this.el.children,
				i = 0,
R
RubaXa 已提交
813 814
				n = children.length,
				options = this.options;
815 816 817

			for (; i < n; i++) {
				el = children[i];
R
RubaXa 已提交
818
				if (_closest(el, options.draggable, this.el)) {
R
RubaXa 已提交
819
					order.push(el.getAttribute(options.dataIdAttr) || _generateId(el));
R
RubaXa 已提交
820
				}
821 822 823 824 825 826 827 828 829 830 831
			}

			return order;
		},


		/**
		 * Sorts the elements according to the array.
		 * @param  {String[]}  order  order of the items
		 */
		sort: function (order) {
R
RubaXa 已提交
832
			var items = {}, rootEl = this.el;
833 834

			this.toArray().forEach(function (id, i) {
R
RubaXa 已提交
835 836
				var el = rootEl.children[i];

R
RubaXa 已提交
837
				if (_closest(el, this.options.draggable, rootEl)) {
R
RubaXa 已提交
838 839 840
					items[id] = el;
				}
			}, this);
841 842 843

			order.forEach(function (id) {
				if (items[id]) {
R
RubaXa 已提交
844 845
					rootEl.removeChild(items[id]);
					rootEl.appendChild(items[id]);
846 847 848 849 850
				}
			});
		},


R
RubaXa 已提交
851 852 853 854 855 856 857 858 859
		/**
		 * Save the current sorting
		 */
		save: function () {
			var store = this.options.store;
			store && store.set(this);
		},


860 861 862 863 864 865 866 867 868 869 870
		/**
		 * For each element in the set, get the first element that matches the selector by testing the element itself and traversing up through its ancestors in the DOM tree.
		 * @param   {HTMLElement}  el
		 * @param   {String}       [selector]  default: `options.draggable`
		 * @returns {HTMLElement|null}
		 */
		closest: function (el, selector) {
			return _closest(el, selector || this.options.draggable, this.el);
		},


871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887
		/**
		 * Set/get option
		 * @param   {string} name
		 * @param   {*}      [value]
		 * @returns {*}
		 */
		option: function (name, value) {
			var options = this.options;

			if (value === void 0) {
				return options[name];
			} else {
				options[name] = value;
			}
		},


888 889 890 891
		/**
		 * Destroy
		 */
		destroy: function () {
892
			var el = this.el;
R
RubaXa 已提交
893

894
			el[expando] = null;
895

R
RubaXa 已提交
896 897 898
			_off(el, 'mousedown', this._onTapStart);
			_off(el, 'touchstart', this._onTapStart);

R
RubaXa 已提交
899
			if (this.nativeDraggable) {
900 901 902
				_off(el, 'dragover', this);
				_off(el, 'dragenter', this);
			}
R
RubaXa 已提交
903

904
			// Remove draggable attributes
R
RubaXa 已提交
905
			Array.prototype.forEach.call(el.querySelectorAll('[draggable]'), function (el) {
906 907 908
				el.removeAttribute('draggable');
			});

R
RubaXa 已提交
909 910 911 912
			touchDragOverListeners.splice(touchDragOverListeners.indexOf(this._onDragOver), 1);

			this._onDrop();

913
			this.el = el = null;
R
RubaXa 已提交
914 915 916
		}
	};

917

R
RubaXa 已提交
918 919 920 921 922 923 924 925 926
	function _cloneHide(state) {
		if (cloneEl && (cloneEl.state !== state)) {
			_css(cloneEl, 'display', state ? 'none' : '');
			!state && cloneEl.state && rootEl.insertBefore(cloneEl, dragEl);
			cloneEl.state = state;
		}
	}


R
RubaXa 已提交
927
	function _closest(/**HTMLElement*/el, /**String*/selector, /**HTMLElement*/ctx) {
R
RubaXa 已提交
928
		if (el) {
R
RubaXa 已提交
929 930 931
			ctx = ctx || document;
			selector = selector.split('.');

R
RubaXa 已提交
932
			var tag = selector.shift().toUpperCase(),
933
				re = new RegExp('\\s(' + selector.join('|') + ')(?=\\s)', 'g');
R
RubaXa 已提交
934 935

			do {
R
RubaXa 已提交
936
				if (
R
RubaXa 已提交
937
					(tag === '>*' && el.parentNode === ctx) || (
R
raphj 已提交
938
						(tag === '' || el.nodeName.toUpperCase() == tag) &&
R
RubaXa 已提交
939 940
						(!selector.length || ((' ' + el.className + ' ').match(re) || []).length == selector.length)
					)
R
RubaXa 已提交
941 942
				) {
					return el;
R
RubaXa 已提交
943 944
				}
			}
R
RubaXa 已提交
945
			while (el !== ctx && (el = el.parentNode));
R
RubaXa 已提交
946 947
		}

R
RubaXa 已提交
948
		return null;
R
RubaXa 已提交
949 950 951
	}


952
	function _globalDragOver(/**Event*/evt) {
R
RubaXa 已提交
953 954 955 956 957
		evt.dataTransfer.dropEffect = 'move';
		evt.preventDefault();
	}


R
RubaXa 已提交
958
	function _on(el, event, fn) {
R
RubaXa 已提交
959 960 961 962
		el.addEventListener(event, fn, false);
	}


R
RubaXa 已提交
963
	function _off(el, event, fn) {
R
RubaXa 已提交
964 965 966 967
		el.removeEventListener(event, fn, false);
	}


R
RubaXa 已提交
968 969 970
	function _toggleClass(el, name, state) {
		if (el) {
			if (el.classList) {
R
RubaXa 已提交
971 972 973
				el.classList[state ? 'add' : 'remove'](name);
			}
			else {
B
Bogdan Mustiata 已提交
974 975
				var className = (' ' + el.className + ' ').replace(RSPACE, ' ').replace(' ' + name + ' ', ' ');
				el.className = (className + (state ? ' ' + name : '')).replace(RSPACE, ' ');
R
RubaXa 已提交
976 977 978 979 980
			}
		}
	}


R
RubaXa 已提交
981
	function _css(el, prop, val) {
R
RubaXa 已提交
982 983
		var style = el && el.style;

R
RubaXa 已提交
984 985 986
		if (style) {
			if (val === void 0) {
				if (document.defaultView && document.defaultView.getComputedStyle) {
R
RubaXa 已提交
987 988
					val = document.defaultView.getComputedStyle(el, '');
				}
R
RubaXa 已提交
989 990
				else if (el.currentStyle) {
					val = el.currentStyle;
R
RubaXa 已提交
991
				}
R
RubaXa 已提交
992 993 994 995 996 997 998 999 1000

				return prop === void 0 ? val : val[prop];
			}
			else {
				if (!(prop in style)) {
					prop = '-webkit-' + prop;
				}

				style[prop] = val + (typeof val === 'string' ? '' : 'px');
R
RubaXa 已提交
1001 1002 1003 1004 1005
			}
		}
	}


R
RubaXa 已提交
1006 1007
	function _find(ctx, tagName, iterator) {
		if (ctx) {
R
RubaXa 已提交
1008
			var list = ctx.getElementsByTagName(tagName), i = 0, n = list.length;
R
RubaXa 已提交
1009

R
RubaXa 已提交
1010 1011
			if (iterator) {
				for (; i < n; i++) {
R
RubaXa 已提交
1012 1013 1014
					iterator(list[i], i);
				}
			}
R
RubaXa 已提交
1015

R
RubaXa 已提交
1016
			return list;
R
RubaXa 已提交
1017
		}
R
RubaXa 已提交
1018 1019

		return [];
R
RubaXa 已提交
1020 1021 1022
	}


R
RubaXa 已提交
1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038

	function _dispatchEvent(sortable, rootEl, name, targetEl, fromEl, startIndex, newIndex) {
		var evt = document.createEvent('Event'),
			options = (sortable || rootEl[expando]).options,
			onName = 'on' + name.charAt(0).toUpperCase() + name.substr(1);

		evt.initEvent(name, true, true);

		evt.to = rootEl;
		evt.from = fromEl || rootEl;
		evt.item = targetEl || rootEl;
		evt.clone = cloneEl;

		evt.oldIndex = startIndex;
		evt.newIndex = newIndex;

R
RubaXa 已提交
1039 1040
		rootEl.dispatchEvent(evt);

R
RubaXa 已提交
1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052
		if (options[onName]) {
			options[onName].call(sortable, evt);
		}
	}


	function _onMove(fromEl, toEl, dragEl, dragRect, targetEl, targetRect) {
		var evt,
			sortable = fromEl[expando],
			onMoveFn = sortable.options.onMove,
			retVal;

C
ChiefORZ 已提交
1053 1054
		evt = document.createEvent('Event');
		evt.initEvent('move', true, true);
R
RubaXa 已提交
1055

C
ChiefORZ 已提交
1056 1057 1058 1059 1060 1061
		evt.to = toEl;
		evt.from = fromEl;
		evt.dragged = dragEl;
		evt.draggedRect = dragRect;
		evt.related = targetEl || toEl;
		evt.relatedRect = targetRect || toEl.getBoundingClientRect();
R
RubaXa 已提交
1062

C
ChiefORZ 已提交
1063
		fromEl.dispatchEvent(evt);
R
RubaXa 已提交
1064

C
ChiefORZ 已提交
1065
		if (onMoveFn) {
R
RubaXa 已提交
1066 1067 1068
			retVal = onMoveFn.call(sortable, evt);
		}

R
RubaXa 已提交
1069
		return retVal;
R
RubaXa 已提交
1070 1071 1072
	}


R
RubaXa 已提交
1073
	function _disableDraggable(el) {
R
RubaXa 已提交
1074
		el.draggable = false;
R
RubaXa 已提交
1075 1076 1077
	}


R
RubaXa 已提交
1078
	function _unsilent() {
R
RubaXa 已提交
1079 1080 1081 1082
		_silent = false;
	}


R
RubaXa 已提交
1083
	/** @returns {HTMLElement|false} */
R
RubaXa 已提交
1084
	function _ghostInBottom(el, evt) {
R
RubaXa 已提交
1085 1086 1087
		var lastEl = el.lastElementChild,
			rect = lastEl.getBoundingClientRect();

R
RubaXa 已提交
1088
		return (evt.clientY - (rect.top + rect.height) > 5) && lastEl; // min delta
R
RubaXa 已提交
1089 1090 1091
	}


1092 1093 1094 1095 1096 1097 1098
	/**
	 * Generate id
	 * @param   {HTMLElement} el
	 * @returns {String}
	 * @private
	 */
	function _generateId(el) {
R
RubaXa 已提交
1099
		var str = el.tagName + el.className + el.src + el.href + el.textContent,
1100
			i = str.length,
R
RubaXa 已提交
1101
			sum = 0;
1102

1103 1104 1105
		while (i--) {
			sum += str.charCodeAt(i);
		}
1106

1107 1108 1109
		return sum.toString(36);
	}

1110 1111 1112
	/**
	 * Returns the index of an element within its parent
	 * @param el
1113 1114
	 * @returns {number}
	 * @private
1115 1116
	 */
	function _index(/**HTMLElement*/el) {
1117 1118 1119
		if (!el || !el.parentNode) {
			return -1;
		}
1120
		var index = 0;
1121 1122 1123 1124
		while (el && (el = el.previousElementSibling)) {
			if (el.nodeName.toUpperCase() !== 'TEMPLATE') {
				index++;
			}
1125 1126 1127
		}
		return index;
	}
R
RubaXa 已提交
1128

R
RubaXa 已提交
1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149
	function _throttle(callback, ms) {
		var args, _this;

		return function () {
			if (args === void 0) {
				args = arguments;
				_this = this;

				setTimeout(function () {
					if (args.length === 1) {
						callback.call(_this, args[0]);
					} else {
						callback.apply(_this, args);
					}

					args = void 0;
				}, ms);
			}
		};
	}

1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161
	function _extend(dst, src) {
		if (dst && src) {
			for (var key in src) {
				if (src.hasOwnProperty(key)) {
					dst[key] = src[key];
				}
			}
		}

		return dst;
	}

R
RubaXa 已提交
1162

R
RubaXa 已提交
1163 1164 1165 1166 1167 1168
	// Export utils
	Sortable.utils = {
		on: _on,
		off: _off,
		css: _css,
		find: _find,
R
RubaXa 已提交
1169 1170 1171
		is: function (el, selector) {
			return !!_closest(el, selector, el);
		},
1172
		extend: _extend,
R
RubaXa 已提交
1173
		throttle: _throttle,
R
RubaXa 已提交
1174
		closest: _closest,
1175
		toggleClass: _toggleClass,
1176
		index: _index
R
RubaXa 已提交
1177 1178 1179
	};


1180 1181 1182 1183 1184 1185
	/**
	 * Create sortable instance
	 * @param {HTMLElement}  el
	 * @param {Object}      [options]
	 */
	Sortable.create = function (el, options) {
R
RubaXa 已提交
1186
		return new Sortable(el, options);
1187
	};
R
RubaXa 已提交
1188
	
R
RubaXa 已提交
1189 1190

	// Export
R
RubaXa 已提交
1191
	Sortable.version = '1.2.2';
1192
	return Sortable;
R
RubaXa 已提交
1193
});