123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272 |
- /* vim: set expandtab sw=4 ts=4 sts=4: */
- /**
- * @fileoverview A jquery plugin that allows drag&drop sorting in tables.
- * Coded because JQuery UI sortable doesn't support tables. Also it has no animation
- *
- * @name Sortable Table JQuery plugin
- *
- * @requires jQuery
- *
- */
- /* Options:
- $('table').sortableTable({
- ignoreRect: { top, left, width, height } - relative coordinates on each element. If the user clicks
- in this area, it is not seen as a drag&drop request. Useful for toolbars etc.
- events: {
- start: callback function when the user starts dragging
- drop: callback function after an element has been dropped
- }
- })
- */
- /* Commands:
- $('table').sortableTable('init') - equivalent to $('table').sortableTable()
- $('table').sortableTable('refresh') - if the table has been changed, refresh correctly assigns all events again
- $('table').sortableTable('destroy') - removes all events from the table
- */
- /* Setup:
- Can be applied on any table, there is just one convention.
- Each cell (<td>) has to contain one and only one element (preferably div or span)
- which is the actually draggable element.
- */
- (function($) {
- jQuery.fn.sortableTable = function(method) {
-
- var methods = {
- init : function(options) {
- var tb = new sortableTableInstance(this, options);
- tb.init();
- $(this).data('sortableTable',tb);
- },
- refresh : function( ) {
- $(this).data('sortableTable').refresh();
- },
- destroy : function( ) {
- $(this).data('sortableTable').destroy();
- }
- };
- if ( methods[method] ) {
- return methods[method].apply( this, Array.prototype.slice.call( arguments, 1 ));
- } else if ( typeof method === 'object' || ! method ) {
- return methods.init.apply( this, arguments );
- } else {
- $.error( 'Method ' + method + ' does not exist on jQuery.sortableTable' );
- }
-
- function sortableTableInstance(table, options) {
- var down = false;
- var $draggedEl, oldCell, previewMove, id;
-
- if(!options) options = {};
-
- /* Mouse handlers on the child elements */
- var onMouseUp = function(e) {
- dropAt(e.pageX, e.pageY);
- }
-
- var onMouseDown = function(e) {
- $draggedEl = $(this).children();
- if($draggedEl.length == 0) return;
- if(options.ignoreRect && insideRect({x: e.pageX - $draggedEl.offset().left, y: e.pageY - $draggedEl.offset().top}, options.ignoreRect)) return;
-
- down = true;
- oldCell = this;
- //move(e.pageX,e.pageY);
-
- if(options.events && options.events.start)
- options.events.start(this);
- return false;
- }
-
- var globalMouseMove = function(e) {
- if(down) {
- move(e.pageX,e.pageY);
- if(inside($(oldCell), e.pageX, e.pageY)) {
- if(previewMove != null) {
- moveTo(previewMove);
- previewMove = null;
- }
- } else
- $(table).find('td').each(function() {
- if(inside($(this), e.pageX, e.pageY)) {
- if($(previewMove).attr('class') != $(this).children().first().attr('class')) {
- if(previewMove != null) moveTo(previewMove);
- previewMove = $(this).children().first();
- if(previewMove.length > 0)
- moveTo($(previewMove), { pos: {
- top: $(oldCell).offset().top - $(previewMove).parent().offset().top,
- left: $(oldCell).offset().left - $(previewMove).parent().offset().left
- } });
- }
-
- return false;
- }
- });
- }
-
- return false;
- }
-
- var globalMouseOut = function() {
- if(down) {
- down = false;
- if(previewMove) moveTo(previewMove);
- moveTo($draggedEl);
- previewMove = null;
- }
- }
-
- // Initialize sortable table
- this.init = function() {
- id = 1;
- // Add some required css to each child element in the <td>s
- $(table).find('td').children().each(function() {
- // Remove any old occurences of our added draggable-num class
- $(this).attr('class',$(this).attr('class').replace(/\s*draggable\-\d+/g,''));
- $(this).addClass('draggable-' + (id++));
- });
-
- // Mouse events
- $(table).find('td').bind('mouseup',onMouseUp);
- $(table).find('td').bind('mousedown',onMouseDown);
- $(document).mousemove(globalMouseMove);
- $(document).bind('mouseleave', globalMouseOut);
- }
-
- // Call this when the table has been updated
- this.refresh = function() {
- this.destroy();
- this.init();
- }
-
- this.destroy = function() {
- // Add some required css to each child element in the <td>s
- $(table).find('td').children().each(function() {
- // Remove any old occurences of our added draggable-num class
- $(this).attr('class',$(this).attr('class').replace(/\s*draggable\-\d+/g,''));
- });
-
- // Mouse events
- $(table).find('td').unbind('mouseup',onMouseUp)
- $(table).find('td').unbind('mousedown',onMouseDown);
-
- $(document).unbind('mousemove',globalMouseMove);
- $(document).unbind('mouseleave',globalMouseOut);
- }
-
- function switchElement(drag, dropTo) {
- var dragPosDiff = {
- left: $(drag).children().first().offset().left - $(dropTo).offset().left,
- top: $(drag).children().first().offset().top - $(dropTo).offset().top
- };
-
- var dropPosDiff = null;
- if($(dropTo).children().length > 0) {
- dropPosDiff = {
- left: $(dropTo).children().first().offset().left - $(drag).offset().left,
- top: $(dropTo).children().first().offset().top - $(drag).offset().top
- };
- }
-
- /* I love you append(). It moves the DOM Elements so gracefully <3 */
- // Put the element in the way to old place
- $(drag).append($(dropTo).children().first()).children()
- .stop(true,true)
- .bind('mouseup',onMouseUp);
-
- if(dropPosDiff)
- $(drag).append($(dropTo).children().first()).children()
- .css('left',dropPosDiff.left + 'px')
- .css('top',dropPosDiff.top + 'px');
-
- // Put our dragged element into the space we just freed up
- $(dropTo).append($(drag).children().first()).children()
- .bind('mouseup',onMouseUp)
- .css('left',dragPosDiff.left + 'px')
- .css('top',dragPosDiff.top + 'px');
-
- moveTo($(dropTo).children().first(), { duration: 100 });
- moveTo($(drag).children().first(), { duration: 100 });
-
- if(options.events && options.events.drop) {
- // Drop event. The drag child element is moved into the drop element
- // and vice versa. So the parameters are switched.
-
- // Calculate row and column index
- colIdx = $(dropTo).prevAll().length;
- rowIdx = $(dropTo).parent().prevAll().length;
-
- options.events.drop(drag,dropTo, { col: colIdx, row: rowIdx });
- }
- }
-
- function move(x,y) {
- $draggedEl.offset({
- top: Math.min($(document).height(), Math.max(0, y - $draggedEl.height()/2)),
- left: Math.min($(document).width(), Math.max(0, x - $draggedEl.width()/2))
- });
- }
-
- function inside($el, x,y) {
- var off = $el.offset();
- return y >= off.top && x >= off.left && x < off.left + $el.width() && y < off.top + $el.height();
- }
-
- function insideRect(pos, r) {
- return pos.y > r.top && pos.x > r.left && pos.y < r.top + r.height && pos.x < r.left + r.width;
- }
-
- function dropAt(x,y) {
- if(!down) return;
- down = false;
-
- var switched = false;
-
- $(table).find('td').each(function() {
- if($(this).children().first().attr('class') != $(oldCell).children().first().attr('class') && inside($(this), x, y)) {
- switchElement(oldCell, this);
- switched = true;
- return;
- }
- });
- if(!switched) {
- if(previewMove) moveTo(previewMove);
- moveTo($draggedEl);
- }
-
- previewMove = null;
- }
-
- function moveTo(elem, opts) {
- if(!opts) opts = {};
- if(!opts.pos) opts.pos = { left: 0, top: 0 };
- if(!opts.duration) opts.duration = 200;
-
- $(elem).css('position','relative');
- $(elem).animate({ top: opts.pos.top, left: opts.pos.left }, {
- duration: opts.duration,
- complete: function() {
- if(opts.pos.left == 0 && opts.pos.top == 0) {
- $(elem)
- .css('position','')
- .css('left','')
- .css('top','');
- }
- }
- });
- }
- }
- }
-
- })( jQuery );
|