2012-02-27 00:49:51 +00:00
|
|
|
/*!
|
2011-10-14 18:19:04 +00:00
|
|
|
* jQuery UI Draggable @VERSION
|
2012-07-04 13:08:08 +00:00
|
|
|
* http://jqueryui.com
|
2008-06-04 02:34:33 +00:00
|
|
|
*
|
2012-07-04 13:08:08 +00:00
|
|
|
* Copyright 2012 jQuery Foundation and other contributors
|
2010-07-09 13:01:04 +00:00
|
|
|
* Dual licensed under the MIT or GPL Version 2 licenses.
|
|
|
|
* http://jquery.org/license
|
2008-11-10 05:18:20 +00:00
|
|
|
*
|
2011-10-14 18:19:04 +00:00
|
|
|
* http://docs.jquery.com/UI/Draggable
|
2008-06-04 02:34:33 +00:00
|
|
|
*
|
|
|
|
* Depends:
|
2009-09-17 10:39:12 +00:00
|
|
|
* jquery.ui.core.js
|
2012-01-07 17:20:49 +00:00
|
|
|
* jquery.ui.interaction.js
|
2009-12-22 19:51:24 +00:00
|
|
|
* jquery.ui.widget.js
|
2008-06-04 02:34:33 +00:00
|
|
|
*/
|
2010-07-13 13:57:58 +00:00
|
|
|
(function( $, undefined ) {
|
2008-06-04 02:34:33 +00:00
|
|
|
|
2012-01-28 03:06:37 +00:00
|
|
|
// create a shallow copy of an object
|
|
|
|
function copy( obj ) {
|
|
|
|
var prop,
|
|
|
|
ret = {};
|
|
|
|
for ( prop in obj ) {
|
|
|
|
ret[ prop ] = obj[ prop ];
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-11-23 17:03:09 +00:00
|
|
|
$.widget( "ui.draggable", $.ui.interaction, {
|
2011-10-14 18:19:04 +00:00
|
|
|
version: "@VERSION",
|
2010-02-05 03:03:50 +00:00
|
|
|
widgetEventPrefix: "drag",
|
2011-09-28 01:33:38 +00:00
|
|
|
|
2010-01-07 03:19:50 +00:00
|
|
|
options: {
|
2012-01-23 02:51:46 +00:00
|
|
|
appendTo: null,
|
2012-01-14 21:15:02 +00:00
|
|
|
handle: null,
|
2012-12-10 22:10:45 +00:00
|
|
|
helper: false,
|
2012-12-10 22:35:15 +00:00
|
|
|
exclude: "input,textarea,button,select,option"
|
2008-11-21 04:01:33 +00:00
|
|
|
},
|
|
|
|
|
2011-10-14 18:19:04 +00:00
|
|
|
// dragEl: element being dragged (original or helper)
|
2011-10-26 01:54:03 +00:00
|
|
|
// position: final CSS position of dragEl
|
2011-10-14 18:19:04 +00:00
|
|
|
// offset: offset of dragEl
|
2012-01-28 03:06:37 +00:00
|
|
|
// originalPosition: CSS position before drag start
|
|
|
|
// originalOffset: offset before drag start
|
2012-01-29 00:34:29 +00:00
|
|
|
// originalPointer: pageX/Y at drag start (offset of pointer)
|
2012-01-28 03:06:37 +00:00
|
|
|
// startPosition: CSS position at drag start (after beforeStart)
|
|
|
|
// startOffset: offset at drag start (after beforeStart)
|
2011-10-26 01:54:03 +00:00
|
|
|
// tempPosition: overridable CSS position of dragEl
|
2011-10-14 18:19:04 +00:00
|
|
|
// overflowOffset: offset of scroll parent
|
2011-10-26 01:54:03 +00:00
|
|
|
// overflow: object containing width and height keys of scroll parent
|
2012-01-23 02:51:46 +00:00
|
|
|
// domPosition: object containing original parent and index when using
|
2012-12-10 22:35:15 +00:00
|
|
|
// appendTo option without a helper
|
2008-11-21 04:01:33 +00:00
|
|
|
|
2011-09-28 01:33:38 +00:00
|
|
|
_create: function() {
|
2011-11-23 17:32:55 +00:00
|
|
|
this._super();
|
2012-01-28 16:12:54 +00:00
|
|
|
|
2011-10-14 18:19:04 +00:00
|
|
|
// Static position elements can't be moved with top/left
|
2011-09-28 01:33:38 +00:00
|
|
|
if ( this.element.css( "position" ) === "static" ) {
|
|
|
|
this.element.css( "position", "relative" );
|
|
|
|
}
|
2012-01-28 16:12:54 +00:00
|
|
|
|
|
|
|
this.element.addClass( "ui-draggable" );
|
2011-10-01 16:38:48 +00:00
|
|
|
},
|
2008-11-23 17:42:24 +00:00
|
|
|
|
2012-01-14 20:49:58 +00:00
|
|
|
/** interaction interface **/
|
2011-10-26 01:54:03 +00:00
|
|
|
|
2012-01-14 21:15:02 +00:00
|
|
|
_isValidTarget: function( element ) {
|
2012-12-10 22:35:15 +00:00
|
|
|
var handle = this.options.handle ? element.is( this.options.handle ) : true,
|
|
|
|
exclude = this.options.exclude ? element.is( this.options.exclude ) : false;
|
|
|
|
|
|
|
|
return ( handle && !exclude );
|
2012-01-14 21:15:02 +00:00
|
|
|
},
|
|
|
|
|
2012-01-15 17:57:39 +00:00
|
|
|
_start: function( event, pointerPosition ) {
|
2012-01-23 02:51:46 +00:00
|
|
|
var offset;
|
|
|
|
|
2011-10-01 16:38:48 +00:00
|
|
|
// The actual dragging element, should always be a jQuery object
|
2012-01-15 20:18:12 +00:00
|
|
|
this.dragEl = this.options.helper ?
|
|
|
|
this._createHelper( pointerPosition ) :
|
|
|
|
this.element;
|
2011-11-06 22:42:35 +00:00
|
|
|
|
2012-01-23 02:51:46 +00:00
|
|
|
// _createHelper() ensures that helpers are in the correct position
|
|
|
|
// in the DOM, but we need to handle appendTo when there is no helper
|
|
|
|
if ( this.options.appendTo && this.dragEl === this.element ) {
|
|
|
|
this.domPosition = {
|
|
|
|
parent: this.element.parent(),
|
|
|
|
index: this.element.index()
|
|
|
|
};
|
|
|
|
offset = this.dragEl.offset();
|
|
|
|
this.dragEl
|
|
|
|
.appendTo( this.options.appendTo )
|
|
|
|
.offset( offset );
|
|
|
|
}
|
|
|
|
|
2011-10-26 01:54:03 +00:00
|
|
|
this.cssPosition = this.dragEl.css( "position" );
|
2011-11-23 14:03:02 +00:00
|
|
|
this.scrollParent = this.element.scrollParent();
|
2008-06-04 02:34:33 +00:00
|
|
|
|
2012-01-21 02:17:11 +00:00
|
|
|
// Cache starting positions
|
2012-01-28 03:06:37 +00:00
|
|
|
this.originalPosition = this.startPosition = this._getPosition();
|
|
|
|
this.originalOffset = this.startOffset = this.dragEl.offset();
|
2012-01-29 00:34:29 +00:00
|
|
|
this.originalPointer = pointerPosition;
|
2008-11-21 04:01:33 +00:00
|
|
|
|
2011-09-28 01:33:38 +00:00
|
|
|
// Cache current position and offset
|
2012-01-28 03:06:37 +00:00
|
|
|
this.position = copy( this.startPosition );
|
|
|
|
this.offset = copy( this.startOffset );
|
2009-01-22 13:10:18 +00:00
|
|
|
|
2011-10-26 01:54:03 +00:00
|
|
|
// Cache the offset of scrollParent, if required for _handleScrolling
|
2012-01-07 17:18:08 +00:00
|
|
|
if ( this.scrollParent[0] !== this.document[0] && this.scrollParent[0].tagName !== "HTML" ) {
|
2011-10-26 01:54:03 +00:00
|
|
|
this.overflowOffset = this.scrollParent.offset();
|
|
|
|
}
|
|
|
|
|
2012-01-07 17:18:08 +00:00
|
|
|
this.overflow = {
|
|
|
|
height: this.scrollParent[0] === this.document[0] ?
|
|
|
|
this.window.height() : this.scrollParent.height(),
|
|
|
|
width: this.scrollParent[0] === this.document[0] ?
|
|
|
|
this.window.width() : this.scrollParent.width()
|
|
|
|
};
|
2011-10-14 18:19:04 +00:00
|
|
|
|
2012-01-15 17:57:39 +00:00
|
|
|
this._preparePosition( pointerPosition );
|
2011-10-26 01:54:03 +00:00
|
|
|
|
2012-01-28 03:06:37 +00:00
|
|
|
// If user cancels beforeStart, don't allow dragging
|
2012-01-28 03:17:27 +00:00
|
|
|
if ( this._trigger( "beforeStart", event,
|
|
|
|
this._originalHash( pointerPosition ) ) === false ) {
|
2012-12-05 03:26:40 +00:00
|
|
|
|
2012-11-08 21:10:43 +00:00
|
|
|
// domPosition needs to be undone even if beforeStart is stopped
|
|
|
|
// Otherwise this.dragEl will remain in the element appendTo is set to
|
|
|
|
this._resetDomPosition();
|
2011-11-23 17:03:09 +00:00
|
|
|
return false;
|
2012-12-05 03:26:40 +00:00
|
|
|
|
2011-10-26 01:54:03 +00:00
|
|
|
}
|
|
|
|
|
2012-01-09 02:48:53 +00:00
|
|
|
this._setCss();
|
2012-01-28 03:06:37 +00:00
|
|
|
this.startPosition = this._getPosition();
|
|
|
|
this.startOffset = this.dragEl.offset();
|
|
|
|
|
2012-01-29 00:34:29 +00:00
|
|
|
this._trigger( "start", event, this._fullHash( pointerPosition ) );
|
2012-01-28 03:06:37 +00:00
|
|
|
this._blockFrames();
|
2011-09-28 01:33:38 +00:00
|
|
|
},
|
2012-12-05 03:26:40 +00:00
|
|
|
|
2012-11-08 21:10:43 +00:00
|
|
|
_resetDomPosition : function() {
|
2012-12-05 03:26:40 +00:00
|
|
|
|
2012-11-08 21:10:43 +00:00
|
|
|
// Nothing to do in this case
|
|
|
|
if ( !this.domPosition ) {
|
|
|
|
return;
|
|
|
|
}
|
2012-12-05 03:26:40 +00:00
|
|
|
|
|
|
|
var parent = this.domPosition.parent,
|
|
|
|
next = parent.children().eq( this.domPosition.index );
|
2012-11-08 21:10:43 +00:00
|
|
|
if ( next.length ) {
|
|
|
|
next.before( this.element );
|
|
|
|
} else {
|
|
|
|
parent.append( this.element );
|
|
|
|
}
|
|
|
|
this.element.offset( this.offset );
|
|
|
|
this.domPosition = null;
|
2012-12-05 03:26:40 +00:00
|
|
|
|
2012-11-08 21:10:43 +00:00
|
|
|
},
|
2008-11-21 04:01:33 +00:00
|
|
|
|
2012-01-15 17:57:39 +00:00
|
|
|
_move: function( event, pointerPosition ) {
|
|
|
|
this._preparePosition( pointerPosition );
|
2011-10-26 01:54:03 +00:00
|
|
|
|
2012-01-08 01:44:51 +00:00
|
|
|
// If user cancels drag, don't move the element
|
2012-01-28 03:17:27 +00:00
|
|
|
if ( this._trigger( "drag", event, this._fullHash( pointerPosition ) ) === false ) {
|
2011-10-26 01:54:03 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-01-09 02:48:53 +00:00
|
|
|
this._setCss();
|
2011-10-26 01:54:03 +00:00
|
|
|
|
|
|
|
// Scroll the scrollParent, if needed
|
2012-01-15 17:57:39 +00:00
|
|
|
this._handleScrolling( pointerPosition );
|
2011-10-26 01:54:03 +00:00
|
|
|
},
|
|
|
|
|
2012-01-15 17:57:39 +00:00
|
|
|
_stop: function( event, pointerPosition ) {
|
|
|
|
this._preparePosition( pointerPosition );
|
2011-10-26 01:54:03 +00:00
|
|
|
|
2012-01-21 03:07:52 +00:00
|
|
|
// If user cancels stop, leave helper there
|
2012-01-28 03:17:27 +00:00
|
|
|
if ( this._trigger( "stop", event, this._fullHash( pointerPosition ) ) !== false ) {
|
2011-10-26 01:54:03 +00:00
|
|
|
if ( this.options.helper ) {
|
|
|
|
this.dragEl.remove();
|
|
|
|
}
|
2012-11-08 21:10:43 +00:00
|
|
|
this._resetDomPosition();
|
2011-10-26 01:54:03 +00:00
|
|
|
}
|
|
|
|
|
2011-11-18 00:48:04 +00:00
|
|
|
this._unblockFrames();
|
2011-10-26 01:54:03 +00:00
|
|
|
},
|
|
|
|
|
2012-01-14 20:49:58 +00:00
|
|
|
/** internal **/
|
|
|
|
|
2012-01-15 20:18:12 +00:00
|
|
|
_createHelper: function( pointerPosition ) {
|
|
|
|
var helper,
|
|
|
|
offset = this.element.offset(),
|
|
|
|
xPos = (pointerPosition.x - offset.left) / this.element.outerWidth(),
|
|
|
|
yPos = (pointerPosition.y - offset.top) / this.element.outerHeight();
|
|
|
|
|
|
|
|
// clone
|
|
|
|
if ( this.options.helper === true ) {
|
|
|
|
helper = this.element.clone()
|
|
|
|
.removeAttr( "id" )
|
|
|
|
.find( "[id]" )
|
|
|
|
.removeAttr( "id" )
|
|
|
|
.end();
|
|
|
|
} else {
|
|
|
|
// TODO: figure out the signature for this; see #4957
|
|
|
|
helper = $( this.options.helper() );
|
|
|
|
}
|
|
|
|
|
2012-01-23 02:51:46 +00:00
|
|
|
// Ensure the helper is in the DOM; obey the appendTo option if it exists
|
|
|
|
if ( this.options.appendTo || !helper.closest( "body" ).length ) {
|
|
|
|
helper.appendTo( this.options.appendTo || this.document[0].body );
|
|
|
|
}
|
|
|
|
|
2012-01-15 20:18:12 +00:00
|
|
|
return helper
|
|
|
|
// Helper must be absolute to function properly
|
|
|
|
.css( "position", "absolute" )
|
|
|
|
.offset({
|
|
|
|
left: pointerPosition.x - helper.outerWidth() * xPos,
|
|
|
|
top: pointerPosition.y - helper.outerHeight() * yPos
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2012-01-14 20:49:58 +00:00
|
|
|
_getPosition: function() {
|
|
|
|
var left, top, position,
|
|
|
|
scrollTop = this.scrollParent.scrollTop(),
|
|
|
|
scrollLeft = this.scrollParent.scrollLeft();
|
|
|
|
|
|
|
|
// If fixed or absolute
|
|
|
|
if ( this.cssPosition !== "relative" ) {
|
|
|
|
position = this.dragEl.position();
|
|
|
|
|
|
|
|
// Take into account scrollbar
|
|
|
|
position.top -= scrollTop;
|
|
|
|
position.left -= scrollLeft;
|
|
|
|
|
|
|
|
return position;
|
|
|
|
}
|
|
|
|
|
|
|
|
// When using relative, css values are checked
|
|
|
|
// Otherwise the position wouldn't account for padding on ancestors
|
|
|
|
left = this.dragEl.css( "left" );
|
|
|
|
top = this.dragEl.css( "top" );
|
|
|
|
|
|
|
|
// Webkit will give back auto if there is no explicit value
|
|
|
|
left = ( left === "auto" ) ? 0: parseInt( left, 10 );
|
|
|
|
top = ( top === "auto" ) ? 0: parseInt( top, 10 );
|
|
|
|
|
|
|
|
return {
|
|
|
|
left: left - scrollLeft,
|
|
|
|
top: top - scrollTop
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
2012-01-15 17:57:39 +00:00
|
|
|
_handleScrolling: function( pointerPosition ) {
|
2012-01-14 20:49:58 +00:00
|
|
|
var scrollTop = this.scrollParent.scrollTop(),
|
|
|
|
scrollLeft = this.scrollParent.scrollLeft(),
|
|
|
|
scrollSensitivity = 20,
|
|
|
|
baseSpeed = 5,
|
|
|
|
speed = function( distance ) {
|
|
|
|
return baseSpeed + Math.round( distance / 2 );
|
|
|
|
},
|
|
|
|
// overflowOffset is only set when scrollParent is not doc/html
|
|
|
|
overflowLeft = this.overflowOffset ?
|
|
|
|
this.overflowOffset.left :
|
|
|
|
scrollLeft,
|
|
|
|
overflowTop = this.overflowOffset ?
|
|
|
|
this.overflowOffset.top :
|
|
|
|
scrollTop,
|
2012-01-15 17:57:39 +00:00
|
|
|
xRight = this.overflow.width + overflowLeft - pointerPosition.x,
|
|
|
|
xLeft = pointerPosition.x- overflowLeft,
|
|
|
|
yBottom = this.overflow.height + overflowTop - pointerPosition.y,
|
|
|
|
yTop = pointerPosition.y - overflowTop;
|
2012-01-14 20:49:58 +00:00
|
|
|
|
|
|
|
// Handle vertical scrolling
|
|
|
|
if ( yBottom < scrollSensitivity ) {
|
|
|
|
this.scrollParent.scrollTop( scrollTop +
|
|
|
|
speed( scrollSensitivity - yBottom ) );
|
|
|
|
} else if ( yTop < scrollSensitivity ) {
|
|
|
|
this.scrollParent.scrollTop( scrollTop -
|
|
|
|
speed( scrollSensitivity - yTop ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle horizontal scrolling
|
|
|
|
if ( xRight < scrollSensitivity ) {
|
|
|
|
this.scrollParent.scrollLeft( scrollLeft +
|
|
|
|
speed( scrollSensitivity - xRight ) );
|
|
|
|
} else if ( xLeft < scrollSensitivity ) {
|
|
|
|
this.scrollParent.scrollLeft( scrollLeft -
|
|
|
|
speed( scrollSensitivity - xLeft ) );
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2011-10-26 01:54:03 +00:00
|
|
|
// Uses event to determine new position of draggable, before any override from callbacks
|
2011-11-27 22:27:55 +00:00
|
|
|
// TODO: handle absolute element inside relative parent like a relative element
|
2012-01-15 17:57:39 +00:00
|
|
|
_preparePosition: function( pointerPosition ) {
|
2012-01-29 00:34:29 +00:00
|
|
|
var leftDiff = pointerPosition.x - this.originalPointer.x,
|
|
|
|
topDiff = pointerPosition.y - this.originalPointer.y,
|
2011-10-14 18:19:04 +00:00
|
|
|
newLeft = leftDiff + this.startPosition.left,
|
|
|
|
newTop = topDiff + this.startPosition.top;
|
2008-11-21 04:01:33 +00:00
|
|
|
|
2011-10-26 01:54:03 +00:00
|
|
|
// Save off new values for .css() in various callbacks using this function
|
2011-09-28 01:33:38 +00:00
|
|
|
this.position = {
|
2011-10-01 16:05:36 +00:00
|
|
|
left: newLeft,
|
|
|
|
top: newTop
|
2011-09-28 01:33:38 +00:00
|
|
|
};
|
2008-11-21 04:01:33 +00:00
|
|
|
|
2011-10-26 01:54:03 +00:00
|
|
|
// Save off values to compare user override against automatic coordinates
|
|
|
|
this.tempPosition = {
|
|
|
|
left: newLeft,
|
|
|
|
top: newTop
|
2011-11-06 21:39:17 +00:00
|
|
|
};
|
2011-11-18 00:48:04 +00:00
|
|
|
|
2011-09-28 01:33:38 +00:00
|
|
|
// Refresh offset cache with new positions
|
2011-11-13 22:14:19 +00:00
|
|
|
this.offset.left = this.startOffset.left + leftDiff;
|
|
|
|
this.offset.top = this.startOffset.top + topDiff;
|
2011-10-26 01:54:03 +00:00
|
|
|
},
|
2008-06-04 02:34:33 +00:00
|
|
|
|
2012-01-09 02:48:53 +00:00
|
|
|
// Places draggable where event, or user via event/callback, indicates
|
|
|
|
_setCss: function() {
|
2012-01-14 20:49:58 +00:00
|
|
|
var newLeft = this.position.left,
|
|
|
|
newTop = this.position.top;
|
2008-06-04 02:34:33 +00:00
|
|
|
|
2011-09-28 01:33:38 +00:00
|
|
|
// User overriding left/top so shortcut math is no longer valid
|
2012-01-14 20:49:58 +00:00
|
|
|
if ( this.tempPosition.left !== this.position.left ||
|
|
|
|
this.tempPosition.top !== this.position.top ) {
|
2012-01-07 17:18:08 +00:00
|
|
|
// Reset offset based on difference of expected and overridden values
|
2012-01-14 20:49:58 +00:00
|
|
|
this.offset.left += newLeft - this.tempPosition.left;
|
|
|
|
this.offset.top += newTop - this.tempPosition.top;
|
2011-09-28 01:33:38 +00:00
|
|
|
}
|
2011-10-14 18:19:04 +00:00
|
|
|
|
|
|
|
// TODO: does this work with nested scrollable parents?
|
|
|
|
if ( this.cssPosition !== "fixed" ) {
|
2012-01-14 20:49:58 +00:00
|
|
|
newLeft += this.scrollParent.scrollLeft();
|
|
|
|
newTop += this.scrollParent.scrollTop();
|
2011-10-09 04:54:47 +00:00
|
|
|
}
|
2008-11-10 05:18:20 +00:00
|
|
|
|
2011-09-28 01:33:38 +00:00
|
|
|
this.dragEl.css({
|
2011-11-18 00:48:04 +00:00
|
|
|
left: newLeft,
|
|
|
|
top: newTop
|
2011-09-28 01:33:38 +00:00
|
|
|
});
|
|
|
|
},
|
2008-11-10 05:18:20 +00:00
|
|
|
|
2012-01-28 03:17:27 +00:00
|
|
|
_originalHash: function( pointerPosition ) {
|
2011-11-06 21:20:56 +00:00
|
|
|
var ret = {
|
2011-11-13 22:14:19 +00:00
|
|
|
position: this.position,
|
2012-01-28 03:06:37 +00:00
|
|
|
offset: copy( this.offset ),
|
|
|
|
pointer: copy( pointerPosition )
|
2008-06-04 02:34:33 +00:00
|
|
|
};
|
2011-11-06 22:42:35 +00:00
|
|
|
|
2011-11-06 21:20:56 +00:00
|
|
|
if ( this.options.helper ) {
|
|
|
|
ret.helper = this.dragEl;
|
|
|
|
}
|
2011-11-06 22:42:35 +00:00
|
|
|
|
2011-11-06 21:20:56 +00:00
|
|
|
return ret;
|
2011-11-06 22:42:35 +00:00
|
|
|
},
|
|
|
|
|
2012-01-29 00:34:29 +00:00
|
|
|
_fullHash: function( pointerPosition ) {
|
2012-01-28 03:17:27 +00:00
|
|
|
return $.extend( this._originalHash( pointerPosition ), {
|
|
|
|
originalPosition: copy( this.originalPosition ),
|
2012-01-29 00:34:29 +00:00
|
|
|
originalOffset: copy( this.originalOffset ),
|
|
|
|
originalPointer: copy( this.originalPointer )
|
2012-01-28 03:17:27 +00:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2011-11-23 14:03:02 +00:00
|
|
|
_blockFrames: function() {
|
|
|
|
|
|
|
|
this.iframeBlocks = this.document.find( "iframe" ).map(function() {
|
|
|
|
var iframe = $( this ),
|
|
|
|
iframeOffset = iframe.offset();
|
|
|
|
|
|
|
|
return $( "<div>" )
|
|
|
|
.css({
|
|
|
|
position: "absolute",
|
|
|
|
width: iframe.outerWidth(),
|
|
|
|
height: iframe.outerHeight(),
|
|
|
|
top: iframeOffset.top,
|
|
|
|
left: iframeOffset.left
|
|
|
|
})
|
2012-12-10 22:14:37 +00:00
|
|
|
.appendTo( iframe.parent() )[0];
|
2011-11-23 14:03:02 +00:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2011-11-06 22:42:35 +00:00
|
|
|
_unblockFrames: function() {
|
2011-11-27 22:27:55 +00:00
|
|
|
if ( this.iframeBlocks ) {
|
2011-11-18 00:48:04 +00:00
|
|
|
this.iframeBlocks.remove();
|
|
|
|
delete this.iframeBlocks;
|
2011-11-06 22:42:35 +00:00
|
|
|
}
|
2012-01-28 16:12:54 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
_destroy: function() {
|
|
|
|
this.element.removeClass( "ui-draggable" );
|
|
|
|
this._super();
|
2011-09-28 01:33:38 +00:00
|
|
|
}
|
2008-06-04 02:34:33 +00:00
|
|
|
});
|
|
|
|
|
2012-01-31 01:57:52 +00:00
|
|
|
$.widget( "ui.draggable", $.ui.draggable, {
|
|
|
|
// $.widget doesn't know how to handle redefinitions with a custom prefix
|
|
|
|
// custom prefixes are going away anyway, so it's not worth fixing right now
|
|
|
|
widgetEventPrefix: "drag",
|
|
|
|
|
|
|
|
options: {
|
|
|
|
containment: null
|
|
|
|
},
|
|
|
|
|
|
|
|
_create: function() {
|
|
|
|
this._super();
|
2012-07-24 15:20:18 +00:00
|
|
|
this._on({
|
2012-01-31 01:57:52 +00:00
|
|
|
dragstart: "_setContainment",
|
|
|
|
drag: "_contain"
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
_setContainment: function( event, ui ) {
|
|
|
|
var offset, left, top,
|
|
|
|
container = this._getContainer();
|
|
|
|
|
|
|
|
if ( !container ) {
|
|
|
|
this.containment = null;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-11-13 21:37:09 +00:00
|
|
|
offset = container.offset();
|
2012-01-31 01:57:52 +00:00
|
|
|
left = offset.left +
|
2012-11-05 21:49:51 +00:00
|
|
|
(parseFloat( $.css( container[0], "borderLeftWidth", true ) ) || 0) +
|
|
|
|
(parseFloat( $.css( container[0], "paddingLeft", true ) ) || 0);
|
2012-01-31 01:57:52 +00:00
|
|
|
top = offset.top +
|
2012-11-05 21:49:51 +00:00
|
|
|
(parseFloat( $.css( container[0], "borderTopWidth", true ) ) || 0) +
|
|
|
|
(parseFloat( $.css( container[0], "paddingTop", true ) ) || 0);
|
2012-01-31 01:57:52 +00:00
|
|
|
|
|
|
|
this.containment = {
|
|
|
|
left: left,
|
|
|
|
top: top,
|
|
|
|
right: left + container.width(),
|
|
|
|
bottom: top + container.height(),
|
|
|
|
leftDiff: ui.originalOffset.left - ui.originalPosition.left,
|
|
|
|
topDiff: ui.originalOffset.top - ui.originalPosition.top,
|
|
|
|
width: this.dragEl.outerWidth(),
|
|
|
|
height: this.dragEl.outerHeight()
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
_contain: function( event, ui ) {
|
|
|
|
var containment = this.containment;
|
|
|
|
|
|
|
|
if ( !containment ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ui.position.left = Math.max( ui.position.left,
|
|
|
|
containment.left - containment.leftDiff );
|
|
|
|
ui.position.left = Math.min( ui.position.left,
|
|
|
|
containment.right - containment.width - containment.leftDiff );
|
|
|
|
|
|
|
|
ui.position.top = Math.max( ui.position.top,
|
|
|
|
containment.top - containment.topDiff );
|
|
|
|
ui.position.top = Math.min( ui.position.top,
|
|
|
|
containment.bottom - containment.height - containment.topDiff );
|
|
|
|
},
|
|
|
|
|
|
|
|
_getContainer: function() {
|
|
|
|
var container,
|
|
|
|
containment = this.options.containment;
|
|
|
|
|
|
|
|
if ( !containment ) {
|
|
|
|
container = null;
|
|
|
|
} else if ( containment === "parent" ) {
|
|
|
|
container = this.element.parent();
|
|
|
|
} else {
|
|
|
|
container = $( containment );
|
|
|
|
if ( !container.length ) {
|
|
|
|
container = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return container;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2011-10-14 18:19:04 +00:00
|
|
|
})( jQuery );
|
2012-12-10 22:50:13 +00:00
|
|
|
|
|
|
|
// DEPRECATED
|
|
|
|
if ( $.uiBackCompat !== false ) {
|
|
|
|
|
|
|
|
// axis option
|
|
|
|
$.widget( "ui.draggable", $.ui.draggable, {
|
|
|
|
options: {
|
|
|
|
axis: false
|
|
|
|
},
|
|
|
|
|
|
|
|
_create : function() {
|
|
|
|
|
2012-12-11 16:55:08 +00:00
|
|
|
var startLeft, startTop;
|
2012-12-10 22:50:13 +00:00
|
|
|
|
|
|
|
this._super();
|
|
|
|
|
|
|
|
// If movement should only move left/right
|
2012-12-11 23:20:48 +00:00
|
|
|
if ( this.options.axis === "x" ) {
|
2012-12-10 22:50:13 +00:00
|
|
|
|
|
|
|
// Cache starting top position to keep it from moving
|
2012-12-11 23:20:48 +00:00
|
|
|
this.element.on( "dragbeforestart", function( e, ui ) {
|
2012-12-11 16:55:08 +00:00
|
|
|
startTop = ui.position.top;
|
2012-12-10 22:50:13 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// On drag, make sure top does not change so axis is locked
|
2012-12-11 23:20:48 +00:00
|
|
|
this.element.on( "drag", function( e, ui ) {
|
2012-12-11 16:55:08 +00:00
|
|
|
ui.position.top = startTop;
|
2012-12-10 22:50:13 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
// If movement should only move up/down
|
2012-12-11 23:20:48 +00:00
|
|
|
else if ( this.options.axis === "y" ) {
|
2012-12-10 22:50:13 +00:00
|
|
|
|
|
|
|
// Cache starting left position to keep it from moving
|
2012-12-11 23:20:48 +00:00
|
|
|
this.element.on( "dragbeforestart", function( e, ui ) {
|
2012-12-11 16:55:08 +00:00
|
|
|
startLeft = ui.position.left;
|
2012-12-10 22:50:13 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// On drag, make sure top does not change so axis is locked
|
2012-12-11 23:20:48 +00:00
|
|
|
this.element.on( "drag", function( e, ui ) {
|
2012-12-11 16:55:08 +00:00
|
|
|
ui.position.left = startLeft;
|
2012-12-10 22:50:13 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2012-12-10 23:03:58 +00:00
|
|
|
// cursor option
|
|
|
|
$.widget( "ui.draggable", $.ui.draggable, {
|
|
|
|
options: {
|
2012-12-11 23:20:48 +00:00
|
|
|
cursor: "auto"
|
2012-12-10 23:03:58 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
_create : function() {
|
|
|
|
|
2012-12-11 16:55:08 +00:00
|
|
|
var startCursor, self, body;
|
2012-12-10 23:03:58 +00:00
|
|
|
|
|
|
|
this._super();
|
|
|
|
|
|
|
|
if ( this.options.cursor ) {
|
|
|
|
|
|
|
|
self = this;
|
|
|
|
body = $( document.body );
|
|
|
|
|
|
|
|
// Cache original cursor to set back
|
2012-12-11 23:20:48 +00:00
|
|
|
this.element.on( "dragbeforestart", function( e, ui ) {
|
|
|
|
startCursor = body.css( "cursor" );
|
2012-12-10 23:03:58 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// Set cursor to what user wants during drag
|
2012-12-11 23:20:48 +00:00
|
|
|
this.element.on( "drag", function( e, ui ) {
|
|
|
|
body.css( "cursor", self.options.cursor );
|
2012-12-10 23:03:58 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// Set back cursor to whatever default was
|
2012-12-11 23:20:48 +00:00
|
|
|
this.element.on( "dragstop", function( e, ui ) {
|
2012-12-10 23:03:58 +00:00
|
|
|
|
|
|
|
// Make sure something was actually reported back before setting body
|
2012-12-11 16:55:08 +00:00
|
|
|
if ( startCursor ) {
|
2012-12-11 23:20:48 +00:00
|
|
|
body.css( "cursor", startCursor );
|
2012-12-10 23:03:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
2012-12-11 16:55:08 +00:00
|
|
|
// cursorAt option
|
|
|
|
$.widget( "ui.draggable", $.ui.draggable, {
|
|
|
|
options: {
|
|
|
|
cursorAt: false
|
|
|
|
},
|
|
|
|
|
|
|
|
_create : function() {
|
|
|
|
|
2012-12-11 16:56:48 +00:00
|
|
|
var self = this,
|
|
|
|
cursorAt;
|
2012-12-11 16:55:08 +00:00
|
|
|
|
|
|
|
this._super();
|
|
|
|
|
2012-12-12 00:16:20 +00:00
|
|
|
// No need to continue
|
2012-12-11 16:55:08 +00:00
|
|
|
if ( !this.options.cursorAt ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
cursorAt = this.options.cursorAt;
|
|
|
|
|
2012-12-11 23:20:48 +00:00
|
|
|
this.element.on( "dragbeforestart", function( e, ui ) {
|
2012-12-11 16:56:48 +00:00
|
|
|
|
|
|
|
var elem = self.dragEl;
|
|
|
|
|
2012-12-11 16:55:08 +00:00
|
|
|
if ( "top" in cursorAt ) {
|
|
|
|
ui.position.top += ui.pointer.y - ui.offset.top - cursorAt.top;
|
|
|
|
}
|
|
|
|
if ( "left" in cursorAt ) {
|
|
|
|
ui.position.left += ui.pointer.x - ui.offset.left - cursorAt.left;
|
|
|
|
}
|
|
|
|
if ( "bottom" in cursorAt ) {
|
|
|
|
ui.position.top += ui.pointer.y - ui.offset.top - elem.outerHeight() + cursorAt.bottom;
|
|
|
|
}
|
|
|
|
if ( "right" in cursorAt ) {
|
|
|
|
ui.position.left += ui.pointer.x - ui.offset.left - elem.outerWidth() + cursorAt.right;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
2012-12-11 23:20:48 +00:00
|
|
|
// grid option
|
|
|
|
$.widget( "ui.draggable", $.ui.draggable, {
|
|
|
|
options: {
|
|
|
|
grid: false
|
|
|
|
},
|
|
|
|
|
|
|
|
_create : function() {
|
|
|
|
|
|
|
|
var x, y, currentX, currentY;
|
|
|
|
|
|
|
|
this._super();
|
|
|
|
|
2012-12-12 00:16:20 +00:00
|
|
|
// No need to continue
|
2012-12-11 23:20:48 +00:00
|
|
|
if ( !this.options.grid ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-12-12 00:16:20 +00:00
|
|
|
// Save off the intended intervals
|
2012-12-11 23:20:48 +00:00
|
|
|
x = this.options.grid[0];
|
|
|
|
y = this.options.grid[1];
|
|
|
|
|
|
|
|
this.element.on( "dragbeforestart", function( e, ui ) {
|
|
|
|
|
2012-12-12 00:16:20 +00:00
|
|
|
// Save off the start position
|
2012-12-11 23:20:48 +00:00
|
|
|
currentX = ui.position.left;
|
|
|
|
currentY = ui.position.top;
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
this.element.on( "drag", function( e, ui ) {
|
|
|
|
|
2012-12-12 00:16:20 +00:00
|
|
|
// If x is actually something, check that user is at least half way to next point
|
2012-12-11 23:20:48 +00:00
|
|
|
if ( x ) {
|
|
|
|
if ( ui.position.left - currentX > x/2 ) {
|
|
|
|
currentX = currentX + x;
|
|
|
|
}
|
|
|
|
else if ( currentX - ui.position.left > x/2 ) {
|
|
|
|
currentX = currentX - x;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-12 00:16:20 +00:00
|
|
|
// If y is actually something, check that user is at least half way to next point
|
2012-12-11 23:20:48 +00:00
|
|
|
if ( y ) {
|
|
|
|
if ( ui.position.top - currentY > y/2 ) {
|
|
|
|
currentY = currentY + y;
|
|
|
|
}
|
|
|
|
else if ( currentY - ui.position.top > y/2 ) {
|
|
|
|
currentY = currentY - y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-12 00:16:20 +00:00
|
|
|
// If there threshold wasn't crossed these variables wouldn't be changed
|
|
|
|
// Otherwise this will now bump the draggable to the next spot on grid
|
2012-12-11 23:20:48 +00:00
|
|
|
ui.position.left = currentX;
|
|
|
|
ui.position.top = currentY;
|
|
|
|
|
2012-12-12 00:16:20 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
// opacity option
|
|
|
|
$.widget( "ui.draggable", $.ui.draggable, {
|
|
|
|
options: {
|
|
|
|
opacity: false
|
|
|
|
},
|
|
|
|
|
|
|
|
_create : function() {
|
|
|
|
|
|
|
|
var self = this,
|
|
|
|
originalOpacity;
|
|
|
|
|
|
|
|
this._super();
|
|
|
|
|
|
|
|
// No need to continue
|
|
|
|
if ( !this.options.opacity ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.element.on( "dragbeforestart", function( e, ui ) {
|
|
|
|
|
|
|
|
// Cache the original opacity of draggable element to reset later
|
|
|
|
originalOpacity = self.dragEl.css( 'opacity' );
|
|
|
|
|
|
|
|
// Set draggable element to new opacity
|
|
|
|
self.dragEl.css( 'opacity', self.options.opacity );
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
this.element.on( "dragstop", function( e, ui ) {
|
|
|
|
|
|
|
|
// Reset opacity
|
|
|
|
self.dragEl.css( 'opacity', originalOpacity );
|
2012-12-11 23:20:48 +00:00
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
2012-12-12 00:21:37 +00:00
|
|
|
// TODO: handle droppables
|
2012-12-12 00:28:47 +00:00
|
|
|
// revert + revertDuration options
|
2012-12-12 00:21:37 +00:00
|
|
|
$.widget( "ui.draggable", $.ui.draggable, {
|
|
|
|
options: {
|
2012-12-12 00:28:47 +00:00
|
|
|
revert: false,
|
|
|
|
revertDuration: 500
|
2012-12-12 00:21:37 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
_create : function() {
|
|
|
|
|
|
|
|
var self = this,
|
|
|
|
originalLeft, originalTop, originalPosition;
|
|
|
|
|
|
|
|
this._super();
|
|
|
|
|
|
|
|
// No need to continue
|
|
|
|
if ( !this.options.revert ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.element.on( "dragbeforestart", function( e, ui ) {
|
|
|
|
|
|
|
|
// Cache the original css of draggable element to reset later
|
|
|
|
originalLeft = self.dragEl.css( 'left' );
|
|
|
|
originalTop = self.dragEl.css( 'top' );
|
|
|
|
originalPosition = self.dragEl.css( 'position' );
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
this.element.on( "dragstop", function( e, ui ) {
|
|
|
|
|
|
|
|
// Reset to before drag
|
2012-12-12 00:28:47 +00:00
|
|
|
self.dragEl.animate({
|
2012-12-12 00:21:37 +00:00
|
|
|
left: originalLeft,
|
|
|
|
top: originalTop,
|
|
|
|
position: originalPosition
|
2012-12-12 00:28:47 +00:00
|
|
|
}, self.options.revertDuration );
|
2012-12-12 00:21:37 +00:00
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
2012-12-10 22:50:13 +00:00
|
|
|
}
|