2015-02-10 00:44:47 +00:00
|
|
|
/*! tablesorter Editable Content widget - updated 2/9/2015 (v2.19.1)
|
2013-04-13 01:39:07 +00:00
|
|
|
* Requires tablesorter v2.8+ and jQuery 1.7+
|
|
|
|
* by Rob Garrison
|
|
|
|
*/
|
2013-04-13 02:15:57 +00:00
|
|
|
/*jshint browser:true, jquery:true, unused:false */
|
2013-04-13 01:39:07 +00:00
|
|
|
/*global jQuery: false */
|
2014-09-27 16:04:13 +00:00
|
|
|
;( function( $ ){
|
|
|
|
'use strict';
|
2013-04-13 01:39:07 +00:00
|
|
|
|
2014-09-27 16:04:13 +00:00
|
|
|
var tse = $.tablesorter.editable = {
|
|
|
|
|
|
|
|
editComplete: function( c, wo, $cell, refocus ) {
|
|
|
|
$cell
|
|
|
|
.removeClass( 'tseditable-last-edited-cell' )
|
|
|
|
.trigger( wo.editable_editComplete, [ c ] );
|
|
|
|
// restore focus last cell after updating
|
|
|
|
if ( refocus ) {
|
|
|
|
setTimeout( function() {
|
|
|
|
$cell.focus();
|
|
|
|
}, 50 );
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
selectAll: function( cell ) {
|
|
|
|
setTimeout( function() {
|
|
|
|
// select all text in contenteditable
|
|
|
|
// see http://stackoverflow.com/a/6150060/145346
|
|
|
|
var sel, range = document.createRange();
|
|
|
|
range.selectNodeContents( cell );
|
|
|
|
sel = window.getSelection();
|
|
|
|
sel.removeAllRanges();
|
|
|
|
sel.addRange( range );
|
|
|
|
}, 100 );
|
|
|
|
},
|
|
|
|
|
|
|
|
update: function( c, wo ) {
|
|
|
|
var indx, tmp, $t,
|
2015-02-03 18:58:41 +00:00
|
|
|
colIndex = [],
|
2014-09-27 16:04:13 +00:00
|
|
|
cols = [];
|
2015-02-03 18:58:41 +00:00
|
|
|
if ( !wo.editable_columnsArray && $.type( wo.editable_columns ) === 'string' && wo.editable_columns.indexOf( '-' ) >= 0 ) {
|
2014-09-27 16:04:13 +00:00
|
|
|
// editable_columns can contain a range string ( i.e. '2-4' )
|
|
|
|
tmp = wo.editable_columns.split( /\s*-\s*/ );
|
|
|
|
indx = parseInt( tmp[ 0 ], 10 ) || 0;
|
|
|
|
tmp = parseInt( tmp[ 1 ], 10 ) || ( c.columns - 1 );
|
|
|
|
if ( tmp > c.columns ) {
|
|
|
|
tmp = c.columns - 1;
|
|
|
|
}
|
|
|
|
for ( ; indx <= tmp; indx++ ) {
|
2015-02-03 18:58:41 +00:00
|
|
|
colIndex.push( indx );
|
2014-09-27 16:04:13 +00:00
|
|
|
cols.push( 'td:nth-child(' + ( indx + 1 ) + ')' );
|
|
|
|
}
|
|
|
|
} else if ( $.isArray( wo.editable_columns ) ) {
|
2015-02-03 18:58:41 +00:00
|
|
|
$.each( wo.editable_columnsArray || wo.editable_columns, function( i, col ) {
|
2014-09-27 16:04:13 +00:00
|
|
|
if ( col < c.columns ) {
|
2015-02-03 18:58:41 +00:00
|
|
|
colIndex.push( col );
|
2014-09-27 16:04:13 +00:00
|
|
|
cols.push( 'td:nth-child(' + ( col + 1 ) + ')' );
|
2013-11-25 12:20:12 +00:00
|
|
|
}
|
2014-09-27 16:04:13 +00:00
|
|
|
});
|
|
|
|
}
|
2015-02-03 18:58:41 +00:00
|
|
|
if ( !wo.editable_columnsArray ) {
|
|
|
|
wo.editable_columnsArray = colIndex;
|
|
|
|
wo.editable_columnsArray.sort(function(a,b){ return a - b; });
|
|
|
|
}
|
2014-09-27 16:04:13 +00:00
|
|
|
tmp = $( '<div>' ).wrapInner( wo.editable_wrapContent ).children().length || $.isFunction( wo.editable_wrapContent );
|
|
|
|
// IE does not allow making TR/TH/TD cells directly editable ( issue #404 )
|
|
|
|
// so add a div or span inside ( it's faster than using wrapInner() )
|
|
|
|
c.$tbodies.find( cols.join( ',' ) ).not( '.' + wo.editable_noEdit ).each( function() {
|
|
|
|
// test for children, if they exist, then make the children editable
|
|
|
|
$t = $( this );
|
|
|
|
|
|
|
|
if ( tmp && $t.children().length === 0 ) {
|
|
|
|
$t.wrapInner( wo.editable_wrapContent );
|
|
|
|
}
|
|
|
|
if ( $t.children().length ) {
|
|
|
|
// make all children content editable
|
|
|
|
$t.children().not( '.' + wo.editable_noEdit ).each( function() {
|
|
|
|
var $this = $( this );
|
|
|
|
if ( wo.editable_trimContent ) {
|
|
|
|
$this.text( function( i, txt ) {
|
|
|
|
return $.trim( txt );
|
|
|
|
});
|
2014-07-25 12:50:36 +00:00
|
|
|
}
|
2014-09-27 16:04:13 +00:00
|
|
|
$this.prop( 'contenteditable', true );
|
2013-11-25 12:20:12 +00:00
|
|
|
});
|
2014-09-27 16:04:13 +00:00
|
|
|
} else {
|
|
|
|
if ( wo.editable_trimContent ) {
|
|
|
|
$t.text( function( i, txt ) {
|
|
|
|
return $.trim( txt );
|
|
|
|
});
|
|
|
|
}
|
|
|
|
$t.prop( 'contenteditable', true );
|
2013-11-25 12:20:12 +00:00
|
|
|
}
|
2014-09-27 16:04:13 +00:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
bindEvents: function( c, wo ) {
|
|
|
|
c.$table
|
2015-03-11 23:57:35 +00:00
|
|
|
.off( ( 'updateComplete pagerComplete '.split( ' ' ).join( '.tseditable ' ) ).replace( /\s+/g, ' ' ) )
|
|
|
|
.on( 'updateComplete pagerComplete '.split( ' ' ).join( '.tseditable ' ), function() {
|
2015-02-03 18:58:41 +00:00
|
|
|
tse.update( c, c.widgetOptions );
|
2014-09-27 16:04:13 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
c.$tbodies
|
2015-02-20 22:13:40 +00:00
|
|
|
.off( ( 'mouseleave focus blur focusout keydown '.split( ' ' ).join( '.tseditable ' ) ).replace( /\s+/g, ' ' ) )
|
2014-09-27 16:04:13 +00:00
|
|
|
.on( 'mouseleave.tseditable', function() {
|
|
|
|
if ( c.$table.data( 'contentFocused' ) ) {
|
|
|
|
// change to 'true' instead of element to allow focusout to process
|
|
|
|
c.$table.data( 'contentFocused', true );
|
|
|
|
$( ':focus' ).trigger( 'focusout' );
|
2014-08-27 00:24:43 +00:00
|
|
|
}
|
2014-09-27 16:04:13 +00:00
|
|
|
})
|
|
|
|
.on( 'focus.tseditable', '[contenteditable]', function( e ) {
|
|
|
|
clearTimeout( $( this ).data( 'timer' ) );
|
|
|
|
c.$table.data( 'contentFocused', e.target );
|
|
|
|
var $this = $( this ),
|
|
|
|
selAll = wo.editable_selectAll,
|
|
|
|
column = $this.closest( 'td' ).index(),
|
|
|
|
txt = $.trim( $this.text() );
|
|
|
|
if ( wo.editable_enterToAccept ) {
|
|
|
|
// prevent enter from adding into the content
|
|
|
|
$this.on( 'keydown.tseditable', function( e ){
|
|
|
|
if ( e.which === 13 ) {
|
|
|
|
e.preventDefault();
|
2014-09-02 20:29:28 +00:00
|
|
|
}
|
|
|
|
});
|
2014-08-27 00:24:43 +00:00
|
|
|
}
|
2014-09-27 16:04:13 +00:00
|
|
|
$this.data({ before : txt, original: txt });
|
|
|
|
|
|
|
|
if ( typeof wo.editable_focused === 'function' ) {
|
|
|
|
wo.editable_focused( txt, column, $this );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( selAll ) {
|
|
|
|
if ( typeof selAll === 'function' ) {
|
|
|
|
if ( selAll( txt, column, $this ) ) {
|
|
|
|
tse.selectAll( $this[0] );
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
tse.selectAll( $this[0] );
|
2013-04-13 01:39:07 +00:00
|
|
|
}
|
2014-09-27 16:04:13 +00:00
|
|
|
}
|
|
|
|
})
|
2015-02-20 22:13:40 +00:00
|
|
|
.on( 'blur focusout keydown '.split( ' ' ).join( '.tseditable ' ), '[contenteditable]', function( e ) {
|
2014-09-27 16:04:13 +00:00
|
|
|
if ( !c.$table.data( 'contentFocused' ) ) { return; }
|
|
|
|
var t, validate,
|
|
|
|
valid = false,
|
|
|
|
$this = $( e.target ),
|
|
|
|
txt = $.trim( $this.text() ),
|
|
|
|
column = $this.closest( 'td' ).index();
|
|
|
|
if ( e.which === 27 ) {
|
|
|
|
// user cancelled
|
|
|
|
$this.html( $.trim( $this.data( 'original' ) ) ).trigger( 'blur.tseditable' );
|
|
|
|
c.$table.data( 'contentFocused', false );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// accept on enter ( if set ), alt-enter ( always ) or if autoAccept is set and element is blurred or unfocused
|
|
|
|
t = e.which === 13 && ( wo.editable_enterToAccept || e.altKey ) || wo.editable_autoAccept && e.type !== 'keydown';
|
|
|
|
// change if new or user hits enter ( if option set )
|
|
|
|
if ( t && $this.data( 'before' ) !== txt ) {
|
2014-08-26 21:19:47 +00:00
|
|
|
|
2014-09-27 16:04:13 +00:00
|
|
|
validate = wo.editable_validate;
|
|
|
|
valid = txt;
|
|
|
|
|
|
|
|
if ( typeof( validate ) === 'function' ) {
|
|
|
|
valid = validate( txt, $this.data( 'original' ), column, $this );
|
|
|
|
} else if ( typeof ( validate = $.tablesorter.getColumnData( c.table, validate, column ) ) === 'function' ) {
|
|
|
|
valid = validate( txt, $this.data( 'original' ), column, $this );
|
2014-08-26 21:19:47 +00:00
|
|
|
}
|
|
|
|
|
2014-09-27 16:04:13 +00:00
|
|
|
if ( t && valid !== false ) {
|
|
|
|
c.$table.find( '.tseditable-last-edited-cell' ).removeClass( 'tseditable-last-edited-cell' );
|
|
|
|
$this
|
|
|
|
.addClass( 'tseditable-last-edited-cell' )
|
|
|
|
.html( $.trim( valid ) )
|
|
|
|
.data( 'before', valid )
|
|
|
|
.data( 'original', valid )
|
|
|
|
.trigger( 'change' );
|
|
|
|
c.$table.trigger( 'updateCell', [ $this.closest( 'td' ), false, function() {
|
|
|
|
if ( wo.editable_autoResort ) {
|
|
|
|
setTimeout( function() {
|
|
|
|
c.$table.trigger( 'sorton', [ c.sortList, function() {
|
|
|
|
tse.editComplete( c, wo, c.$table.find( '.tseditable-last-edited-cell' ), true );
|
|
|
|
}, true ] );
|
|
|
|
}, 10 );
|
|
|
|
} else {
|
|
|
|
tse.editComplete( c, wo, c.$table.find( '.tseditable-last-edited-cell' ) );
|
2014-08-26 21:19:47 +00:00
|
|
|
}
|
2014-09-27 16:04:13 +00:00
|
|
|
} ] );
|
2013-04-13 01:39:07 +00:00
|
|
|
return false;
|
|
|
|
}
|
2014-09-27 16:04:13 +00:00
|
|
|
} else if ( !valid && e.type !== 'keydown' ) {
|
|
|
|
clearTimeout( $this.data( 'timer' ) );
|
|
|
|
$this.data( 'timer', setTimeout( function() {
|
|
|
|
if ( $.isFunction( wo.editable_blur ) ) {
|
|
|
|
wo.editable_blur( $.trim( $this.text() ), column, $this );
|
2014-08-26 19:23:13 +00:00
|
|
|
}
|
2014-09-27 16:04:13 +00:00
|
|
|
}, 100 ) );
|
|
|
|
// restore original content on blur
|
|
|
|
$this.html( $.trim( $this.data( 'original' ) ) );
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2014-08-26 19:23:13 +00:00
|
|
|
|
2014-09-27 16:04:13 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
$.tablesorter.addWidget({
|
|
|
|
id: 'editable',
|
|
|
|
options : {
|
|
|
|
editable_columns : [],
|
|
|
|
editable_enterToAccept : true,
|
|
|
|
editable_autoAccept : true,
|
|
|
|
editable_autoResort : false,
|
|
|
|
editable_wrapContent : '<div>', // wrap the cell content... makes this widget work in IE, and with autocomplete
|
|
|
|
editable_trimContent : true, // trim content inside of contenteditable ( remove tabs & carriage returns )
|
|
|
|
editable_validate : null, // function( text, originalText ){ return text; }
|
|
|
|
editable_focused : null, // function( text, columnIndex, $element ) {}
|
|
|
|
editable_blur : null, // function( text, columnIndex, $element ) { }
|
|
|
|
editable_selectAll : false, // true/false or function( text, columnIndex, $element ) { return true; }
|
|
|
|
editable_noEdit : 'no-edit',
|
|
|
|
editable_editComplete : 'editComplete'
|
|
|
|
},
|
|
|
|
init: function( table, thisWidget, c, wo ){
|
|
|
|
if ( !wo.editable_columns.length ) { return; }
|
|
|
|
tse.update( c, wo );
|
|
|
|
tse.bindEvents( c, wo );
|
2013-04-13 01:39:07 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2014-09-27 16:04:13 +00:00
|
|
|
})( jQuery );
|