@ -18,52 +18,51 @@
Sortable = factory ( ) ; // export for Meteor.js
}
else {
/* jshint sub:true */
window [ "Sortable" ] = factory ( ) ;
}
} ) ( function ( ) {
"use strict" ;
var
dra gEl
, ghostEl
, rootEl
, nextEl
var dragEl ,
ghost El ,
cloneEl ,
rootEl ,
nextEl ,
, lastEl
, lastCSS
, lastRect
lastEl ,
lastCSS ,
, activeGroup
activeGroup ,
, tapEvt
, touchEvt
tapEvt ,
touchEvt ,
, expando = 'Sortable' + ( new Date ) . getTime ( )
expando = 'Sortable' + ( new Date ) . getTime ( ) ,
, win = window
, document = win . document
, parseInt = win . parseInt
, supportIEdnd = ! ! document . createElement ( 'div' ) . dragDrop
win = window ,
document = win . document ,
parseInt = win . parseInt ,
supportIEdnd = ! ! document . createElement ( 'div' ) . dragDrop ,
, _silent = false
_silent = false ,
, _create Event = function ( event /**String*/ , item /**HTMLElement*/ ) {
_dispatch Event = function ( rootEl , name , targetEl , fromEl ) {
var evt = document . createEvent ( 'Event' ) ;
evt . initEvent ( event , true , true ) ;
evt . item = item ;
return evt ;
}
, _dispatchEvent = function ( rootEl , name , targetEl ) {
rootEl . dispatchEvent ( _createEvent ( name , targetEl || rootEl ) ) ;
}
evt . initEvent ( name , true , true ) ;
evt . item = targetEl || rootEl ;
evt . from = fromEl || rootEl ;
rootEl . dispatchEvent ( evt ) ;
} ,
, _customEvents = 'onAdd onUpdate onRemove onStart onEnd onFilter' . split ( ' ' )
_customEvents = 'onAdd onUpdate onRemove onStart onEnd onFilter onSort ' . split ( ' ' ) ,
, noop = function ( ) { }
, slice = [ ] . slice
noop = function ( ) { } ,
slice = [ ] . slice ,
, touchDragOverListeners = [ ]
touchDragOverListeners = [ ]
;
@ -78,23 +77,41 @@
this . options = options = ( options || { } ) ;
// Defaults
// Default option s
var defaults = {
group : Math . random ( ) ,
sort : true ,
store : null ,
handle : null ,
draggable : el . children [ 0 ] && el . children [ 0 ] . nodeName || ( /[uo]l/i . test ( el . nodeName ) ? 'li' : '*' ) ,
ghostClass : 'sortable-ghost' ,
ignore : 'a, img' ,
filter : null
filter : null ,
animation : 0 ,
setData : function ( dataTransfer , dragEl ) {
dataTransfer . setData ( 'Text' , dragEl . textContent ) ;
}
} ;
// Set default options
for ( var name in defaults ) {
options [ name ] = options [ name ] || defaults [ name ] ;
! ( name in options ) && ( options [ name ] = defaults [ name ] ) ;
}
if ( ! options . group . name ) {
options . group = { name : options . group } ;
}
[ 'pull' , 'put' ] . forEach ( function ( key ) {
if ( ! ( key in options . group ) ) {
options . group [ key ] = true ;
}
} ) ;
// Define events
_customEvents . forEach ( function ( name ) {
options [ name ] = _bind ( this , options [ name ] || noop ) ;
@ -103,7 +120,7 @@
// Export group name
el [ expando ] = options . group ;
el [ expando ] = options . group . name ;
// Bind all private methods
@ -138,14 +155,13 @@
} ,
_onTapStart : function ( evt /**Event|TouchEvent*/ ) {
var
touch = evt . touches && evt . touches [ 0 ]
, target = ( touch || evt ) . target
, options = this . options
, el = this . el
, filter = options . filter
;
_onTapStart : function ( /**Event|TouchEvent*/ evt ) {
var touch = evt . touches && evt . touches [ 0 ] ,
target = ( touch || evt ) . target ,
options = this . options ,
el = this . el ,
filter = options . filter ;
if ( evt . type === 'mousedown' && evt . button !== 0 ) {
return ; // only left button
@ -200,9 +216,9 @@
if ( touch ) {
// Touch device support
tapEvt = {
target : target
, clientX : touch . clientX
, clientY : touch . clientY
target : target ,
clientX : touch . clientX ,
clientY : touch . clientY
} ;
this . _onDragStart ( tapEvt , true ) ;
@ -222,12 +238,20 @@
if ( document . selection ) {
document . selection . empty ( ) ;
} else {
window . getSelection ( ) . removeAllRanges ( )
window . getSelection ( ) . removeAllRanges ( ) ;
}
} catch ( err ) {
}
} catch ( err ) { }
_dispatchEvent ( dragEl , 'start' ) ;
if ( activeGroup . pull == 'clone' ) {
cloneEl = dragEl . cloneNode ( true ) ;
_css ( cloneEl , 'display' , 'none' ) ;
rootEl . insertBefore ( cloneEl , dragEl ) ;
}
}
} ,
@ -235,16 +259,14 @@
if ( touchEvt ) {
_css ( ghostEl , 'display' , 'none' ) ;
var
target = document . elementFromPoint ( touchEvt . clientX , touchEvt . clientY )
, parent = target
, group = this . options . group
, i = touchDragOverListeners . length
;
var target = document . elementFromPoint ( touchEvt . clientX , touchEvt . clientY ) ,
parent = target ,
groupName = this . options . group . name ,
i = touchDragOverListeners . length ;
if ( parent ) {
do {
if ( parent [ expando ] === group ) {
if ( parent [ expando ] === groupName ) {
while ( i -- ) {
touchDragOverListeners [ i ] ( {
clientX : touchEvt . clientX ,
@ -253,11 +275,13 @@
rootEl : parent
} ) ;
}
break ;
}
target = parent ; // store last element
}
/* jshint boss:true */
while ( parent = parent . parentNode ) ;
}
@ -266,14 +290,12 @@
} ,
_onTouchMove : function ( evt /**TouchEvent*/ ) {
_onTouchMove : function ( /**TouchEvent*/ evt ) {
if ( tapEvt ) {
var
touch = evt . touches [ 0 ]
, dx = touch . clientX - tapEvt . clientX
, dy = touch . clientY - tapEvt . clientY
, translate3d = 'translate3d(' + dx + 'px,' + dy + 'px,0)'
;
var touch = evt . touches [ 0 ] ,
dx = touch . clientX - tapEvt . clientX ,
dy = touch . clientY - tapEvt . clientY ,
translate3d = 'translate3d(' + dx + 'px,' + dy + 'px,0)' ;
touchEvt = touch ;
@ -287,17 +309,16 @@
} ,
_onDragStart : function ( evt /**Event*/ , isTouch /**Boolean*/ ) {
var dataTransfer = evt . dataTransfer ;
_onDragStart : function ( /**Event*/ evt , /**boolean*/ isTouch ) {
var dataTransfer = evt . dataTransfer ,
options = this . options ;
this . _offUpEvents ( ) ;
if ( isTouch ) {
var
rect = dragEl . getBoundingClientRect ( )
, css = _css ( dragEl )
, ghostRect
;
var rect = dragEl . getBoundingClientRect ( ) ,
css = _css ( dragEl ) ,
ghostRect ;
ghostEl = dragEl . cloneNode ( true ) ;
@ -325,7 +346,7 @@
}
else {
dataTransfer . effectAllowed = 'move' ;
dataTransfer . setData ( 'Text' , dragEl . textContent ) ;
options . setData && options . setData . call ( this , dataTransfer , dragEl ) ;
_on ( document , 'drop' , this . _onDrop ) ;
}
@ -334,41 +355,81 @@
} ,
_onDragOver : function ( evt /**Event*/ ) {
if ( ! _silent && ( activeGroup === this . options . group ) && ( evt . rootEl === void 0 || evt . rootEl === this . el ) ) {
var
el = this . el
, target = _closest ( evt . target , this . options . draggable , el )
;
_onDragOver : function ( /**Event*/ evt ) {
var el = this . el ,
target ,
dragRect ,
revert ,
options = this . options ,
group = options . group ,
groupPut = group . put ,
isOwner = ( activeGroup === group ) ,
canSort = options . sort ;
if ( ! _silent &&
( isOwner
? canSort || ( revert = ! rootEl . contains ( dragEl ) )
: activeGroup . pull && activeGroup . name === group . name && groupPut && ( groupPut . indexOf ? groupPut . indexOf ( activeGroup . name ) > - 1 : groupPut )
) &&
( evt . rootEl === void 0 || evt . rootEl === this . el )
) {
target = _closest ( evt . target , options . draggable , el ) ;
dragRect = dragEl . getBoundingClientRect ( ) ;
if ( cloneEl && ( cloneEl . state !== isOwner ) ) {
_css ( cloneEl , 'display' , isOwner ? 'none' : '' ) ;
! isOwner && cloneEl . state && rootEl . insertBefore ( cloneEl , dragEl ) ;
cloneEl . state = isOwner ;
}
if ( revert ) {
if ( cloneEl || nextEl ) {
rootEl . insertBefore ( dragEl , cloneEl || nextEl ) ;
}
else if ( ! canSort ) {
rootEl . appendChild ( dragEl ) ;
}
return ;
}
if ( ( el . children . length === 0 ) || ( el . children [ 0 ] === ghostEl ) ||
( el === evt . target ) && ( target = _ghostInBottom ( el , evt ) )
) {
if ( target ) {
if ( target . animated ) {
return ;
}
targetRect = target . getBoundingClientRect ( ) ;
}
if ( el . children . length === 0 || el . children [ 0 ] === ghostEl || ( el === evt . target ) && _ghostInBottom ( el , evt ) ) {
el . appendChild ( dragEl ) ;
this . _animate ( dragRect , dragEl ) ;
target && this . _animate ( targetRect , target ) ;
}
else if ( target && target !== dragEl && ( target . parentNode [ expando ] !== void 0 ) ) {
else if ( target && ! target . animated && target !== dragEl && ( target . parentNode [ expando ] !== void 0 ) ) {
if ( lastEl !== target ) {
lastEl = target ;
lastCSS = _css ( target ) ;
lastRect = target . getBoundingClientRect ( ) ;
}
var
rect = lastRect
, width = rect . right - rect . left
, height = rect . bottom - rect . top
, floating = /left|right|inline/ . test ( lastCSS . cssFloat + lastCSS . display )
, isWide = ( target . offsetWidth > dragEl . offsetWidth )
, isLong = ( target . offsetHeight > dragEl . offsetHeight )
, halfway = ( floating ? ( evt . clientX - rect . left ) / width : ( evt . clientY - rect . top ) / height ) > . 5
, nextSibling = target . nextElementSibling
, after
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 ,
after
;
_silent = true ;
setTimeout ( _unsilent , 30 ) ;
if ( floating ) {
after = ( target . previousElementSibling === dragEl ) && ! isWide || halfway && isWide
after = ( target . previousElementSibling === dragEl ) && ! isWide || halfway && isWide ;
} else {
after = ( nextSibling !== dragEl ) && ! isLong || halfway && isLong ;
}
@ -378,7 +439,35 @@
} else {
target . parentNode . insertBefore ( dragEl , after ? nextSibling : target ) ;
}
this . _animate ( dragRect , dragEl ) ;
this . _animate ( targetRect , target ) ;
}
}
} ,
_animate : function ( prevRect , target ) {
var ms = this . options . animation ;
if ( ms ) {
var currentRect = target . getBoundingClientRect ( ) ;
_css ( target , 'transition' , 'none' ) ;
_css ( target , 'transform' , 'translate3d('
+ ( prevRect . left - currentRect . left ) + 'px,'
+ ( prevRect . top - currentRect . top ) + 'px,0)'
) ;
target . offsetWidth ; // repaint
_css ( target , 'transition' , 'all ' + ms + 'ms' ) ;
_css ( target , 'transform' , 'translate3d(0,0,0)' ) ;
clearTimeout ( target . animated ) ;
target . animated = setTimeout ( function ( ) {
_css ( target , 'transition' , '' ) ;
target . animated = false ;
} , ms ) ;
}
} ,
@ -389,7 +478,7 @@
_off ( document , 'touchcancel' , this . _onDrop ) ;
} ,
_onDrop : function ( evt /**Event*/ ) {
_onDrop : function ( /**Event*/ evt ) {
clearInterval ( this . _loopId ) ;
// Unbind events
@ -406,27 +495,31 @@
evt . preventDefault ( ) ;
evt . stopPropagation ( ) ;
if ( ghostEl ) {
ghostEl . parentNode . removeChild ( ghostEl ) ;
}
ghostEl && ghostEl . parentNode . removeChild ( ghostEl ) ;
if ( dragEl ) {
_disableDraggable ( dragEl ) ;
_toggleClass ( dragEl , this . options . ghostClass , false ) ;
if ( ! rootEl . contains ( dragEl ) ) {
// Remove event
_dispatchEvent ( rootEl , 'remove' , dragEl ) ;
_dispatchEvent ( dragEl , 'sort' ) ;
_dispatchEvent ( rootEl , 'sort' ) ;
// Add event
_dispatchEvent ( dragEl , 'add' ) ;
_dispatchEvent ( dragEl , 'add' , dragEl , rootEl ) ;
// Remove event
_dispatchEvent ( rootEl , 'remove' , dragEl ) ;
}
else if ( dragEl . nextSibling !== nextEl ) {
// Update event
_dispatchEvent ( dragEl , 'update' ) ;
_dispatchEvent ( dragEl , 'sort' ) ;
cloneEl && cloneEl . parentNode . removeChild ( cloneEl ) ;
}
_dispatchEvent ( dragEl , 'end' ) ;
_dispatchEvent ( root El, 'end' ) ;
}
// Set NULL
@ -434,6 +527,7 @@
dragEl =
ghostEl =
nextEl =
cloneEl =
tapEvt =
touchEvt =
@ -458,8 +552,7 @@
el ,
children = this . el . children ,
i = 0 ,
n = children . length
;
n = children . length ;
for ( ; i < n ; i ++ ) {
el = children [ i ] ;
@ -508,6 +601,23 @@
} ,
/ * *
* 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 ;
}
} ,
/ * *
* Destroy
* /
@ -555,15 +665,13 @@
ctx = ctx || document ;
selector = selector . split ( '.' ) ;
var
tag = selector . shift ( ) . toUpperCase ( )
, re = new RegExp ( '\\s(' + selector . join ( '|' ) + ')\\s' , 'g' )
;
var tag = selector . shift ( ) . toUpperCase ( ) ,
re = new RegExp ( '\\s(' + selector . join ( '|' ) + ')\\s' , 'g' ) ;
do {
if (
( tag === '' || el . nodeName == tag )
&& ( ! selector . length || ( ( ' ' + el . className + ' ' ) . match ( re ) || [ ] ) . length == selector . length )
( tag === '' || el . nodeName == tag ) &&
( ! selector . length || ( ( ' ' + el . className + ' ' ) . match ( re ) || [ ] ) . length == selector . length )
) {
return el ;
}
@ -598,14 +706,16 @@
}
else {
var className = ( ' ' + el . className + ' ' ) . replace ( /\s+/g , ' ' ) . replace ( ' ' + name + ' ' , '' ) ;
el . className = className + ( state ? ' ' + name : '' )
el . className = className + ( state ? ' ' + name : '' ) ;
}
}
}
function _css ( el , prop , val ) {
if ( el && el . style ) {
var style = el && el . style ;
if ( style ) {
if ( val === void 0 ) {
if ( document . defaultView && document . defaultView . getComputedStyle ) {
val = document . defaultView . getComputedStyle ( el , '' ) ;
@ -613,9 +723,15 @@
else if ( el . currentStyle ) {
val = el . currentStyle ;
}
return prop === void 0 ? val : val [ prop ] ;
} else {
el . style [ prop ] = val + ( typeof val === 'string' ? '' : 'px' ) ;
}
else {
if ( ! ( prop in style ) ) {
prop = '-webkit-' + prop ;
}
style [ prop ] = val + ( typeof val === 'string' ? '' : 'px' ) ;
}
}
}
@ -624,19 +740,22 @@
function _find ( ctx , tagName , iterator ) {
if ( ctx ) {
var list = ctx . getElementsByTagName ( tagName ) , i = 0 , n = list . length ;
if ( iterator ) {
for ( ; i < n ; i ++ ) {
iterator ( list [ i ] , i ) ;
}
}
return list ;
}
return [ ] ;
}
function _disableDraggable ( el ) {
return el . draggable = false ;
el . draggable = false ;
}
@ -645,9 +764,10 @@
}
/** @returns {HTMLElement|false} */
function _ghostInBottom ( el , evt ) {
var last = el . lastElementChild . getBoundingClientRect ( ) ;
return evt . clientY - ( las t. top + las t. height ) > 5 ; // min delta
var lastEl = el . lastElementChild , rect = lastEl . getBoundingClientRect ( ) ;
return ( evt . clientY - ( rec t. top + rec t. height ) > 5 ) && lastEl ; // min delta
}
@ -660,8 +780,7 @@
function _generateId ( el ) {
var str = el . tagName + el . className + el . src + el . href + el . textContent ,
i = str . length ,
sum = 0
;
sum = 0 ;
while ( i -- ) {
sum += str . charCodeAt ( i ) ;
@ -680,13 +799,21 @@
bind : _bind ,
closest : _closest ,
toggleClass : _toggleClass ,
createEvent : _createEvent ,
dispatchEvent : _dispatchEvent
} ;
Sortable . version = '0.5.2' ;
Sortable . version = '0.7.0' ;
/ * *
* Create sortable instance
* @ param { HTMLElement } el
* @ param { Object } [ options ]
* /
Sortable . create = function ( el , options ) {
return new Sortable ( el , options ) ;
} ;
// Export
return Sortable ;