/*** Copyright (c) 2007 Andrew Tetlaw & Millstream Web Software* http://www.millstream.com.au/view/code/tablekit/* Version: 1.2.1 2007-03-11* * Permission is hereby granted, free of charge, to any person* obtaining a copy of this software and associated documentation* files (the "Software"), to deal in the Software without* restriction, including without limitation the rights to use, copy,* modify, merge, publish, distribute, sublicense, and/or sell copies* of the Software, and to permit persons to whom the Software is* furnished to do so, subject to the following conditions:* * The above copyright notice and this permission notice shall be* included in all copies or substantial portions of the Software.* * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE* SOFTWARE.* * */// Use the TableKit class constructure if you'd prefer to init your tables as JS objectsvar TableKit = Class.create();TableKit.prototype = {	initialize : function(elm, options) {		var table = $(elm);		if(table.tagName !== "TABLE") {			return;		}		TableKit.register(table,Object.extend(TableKit.options,options || {}));		this.id = table.id;		var op = TableKit.option('sortable resizable editable', this.id);		if(op.sortable) {			TableKit.Sortable.init(table);		} 		if(op.resizable) {			TableKit.Resizable.init(table);		}		if(op.editable) {			TableKit.Editable.init(table);		}	},	sort : function(column, order) {		TableKit.Sortable.sort(this.id, column, order);	},	resizeColumn : function(column, w) {		TableKit.Resizable.resize(this.id, column, w);	},	editCell : function(row, column) {		TableKit.Editable.editCell(this.id, row, column);	}};Object.extend(TableKit, {	getBodyRows : function(table) {		table = $(table);		var id = table.id;		if(!TableKit.rows[id]) {			TableKit.rows[id] = (table.tHead && table.tHead.rows.length > 0) ? $A(table.tBodies[0].rows) : $A(table.rows).without(table.rows[0]);		}		return TableKit.rows[id];	},	getHeaderCells : function(table, cell) {		if(!table) { table = $(cell).up('table'); }		var id = table.id;		if(!TableKit.heads[id]) {			TableKit.heads[id] = $A((table.tHead && table.tHead.rows.length > 0) ? table.tHead.rows[table.tHead.rows.length-1].cells : table.rows[0].cells);		}		return TableKit.heads[id];	},	getCellIndex : function(cell) {		return $A(cell.parentNode.cells).indexOf(cell);	},	getRowIndex : function(row) {		return $A(row.parentNode.rows).indexOf(row);	},	getCellText : function(cell, refresh) {		if(!cell) { return ""; }		TableKit.registerCell(cell);		var data = TableKit.cells[cell.id];		if(refresh || data.refresh || !data.textContent) {			data.textContent = cell.textContent ? cell.textContent : cell.innerText;			data.refresh = false;		}		return data.textContent;	},	register : function(table, options) {		if(!table.id) {			TableKit._tblcount += 1;			table.id = "tablekit-table-" + TableKit._tblcount;		}		var id = table.id;		TableKit.tables[id] = TableKit.tables[id] ? Object.extend(TableKit.tables[id], options || {}) : Object.extend({sortable:false,resizable:false,editable:false}, options || {});	},	registerCell : function(cell) {		if(!cell.id) {			TableKit._cellcount += 1;			cell.id = "tablekit-cell-" + TableKit._cellcount;		}		if(!TableKit.cells[cell.id]) {			TableKit.cells[cell.id] = {textContent : '', htmlContent : '', active : false};		}	},	isSortable : function(table) {		return TableKit.tables[table.id] ? TableKit.tables[table.id].sortable : false;	},	isResizable : function(table) {		return TableKit.tables[table.id] ? TableKit.tables[table.id].resizable : false;	},	isEditable : function(table) {		return TableKit.tables[table.id] ? TableKit.tables[table.id].editable : false;	},	setup : function(o) {		Object.extend(TableKit.options, o || {} );	},	option : function(s, id, o1, o2) {		o1 = o1 || TableKit.options;		o2 = o2 || (id ? (TableKit.tables[id] ? TableKit.tables[id] : {}) : {});		var key = id + s;		if(!TableKit._opcache[key]){			TableKit._opcache[key] = $A($w(s)).inject([],function(a,v){				a.push(a[v] = o2[v] || o1[v]);				return a;			});		}		return TableKit._opcache[key];	},	e : function(event) {		return event || window.event;	},	tables : {},	_opcache : {},	cells : {},	rows : {},	heads : {},	options : {		autoLoad : true,		stripe : true,		sortable : true,		resizable : true,		editable : true,		rowEvenClass : 'roweven',		rowOddClass : 'rowodd',		sortableSelector : ['table.sortable'],		columnClass : 'sortcol',		descendingClass : 'sortdesc',		ascendingClass : 'sortasc',		noSortClass : 'nosort',		sortFirstAscendingClass : 'sortfirstasc',		sortFirstDecendingClass : 'sortfirstdesc',		resizableSelector : ['table.resizable'],		minWidth : 10,		showHandle : true,		resizeOnHandleClass : 'resize-handle-active',		editableSelector : ['table.editable'],		formClassName : 'editable-cell-form',		noEditClass : 'noedit',		editAjaxURI : '/',		editAjaxOptions : {}	},	_tblcount : 0,	_cellcount : 0,	load : function() {		if(TableKit.options.autoLoad) {			if(TableKit.options.sortable) {				$A(TableKit.options.sortableSelector).each(function(s){					$$(s).each(function(t) {						TableKit.Sortable.init(t);					});				});			}			if(TableKit.options.resizable) {				$A(TableKit.options.resizableSelector).each(function(s){					$$(s).each(function(t) {						TableKit.Resizable.init(t);					});				});			}			if(TableKit.options.editable) {				$A(TableKit.options.editableSelector).each(function(s){					$$(s).each(function(t) {						TableKit.Editable.init(t);					});				});			}		}	}});TableKit.Rows = {	stripe : function(table) {		var rows = TableKit.getBodyRows(table);		rows.each(function(r,i) {			TableKit.Rows.addStripeClass(table,r,i);		});	},	addStripeClass : function(t,r,i) {		t = t || r.up('table');		var op = TableKit.option('rowEvenClass rowOddClass', t.id);		var css = ((i+1)%2 === 0 ? op[0] : op[1]);		// using prototype's assClassName/RemoveClassName was not efficient for large tables, hence:		var cn = r.className.split(/\s+/);		var newCn = [];		for(var x = 0, l = cn.length; x < l; x += 1) {			if(cn[x] !== op[0] && cn[x] !== op[1]) { newCn.push(cn[x]); }		}		newCn.push(css);		r.className = newCn.join(" ");	}};TableKit.Sortable = {	init : function(elm, options){		var table = $(elm);		if(table.tagName !== "TABLE") {			return;		}		TableKit.register(table,Object.extend(options || {},{sortable:true}));		var sortFirst;		var cells = TableKit.getHeaderCells(table);		var op = TableKit.option('noSortClass columnClass sortFirstAscendingClass sortFirstDecendingClass', table.id);		cells.each(function(c){			c = $(c);			if(!c.hasClassName(op.noSortClass)) {				Event.observe(c, 'mousedown', TableKit.Sortable._sort);				c.addClassName(op.columnClass);				if(c.hasClassName(op.sortFirstAscendingClass) || c.hasClassName(op.sortFirstDecendingClass)) {					sortFirst = c;				}			}		});		if(sortFirst) {			if(sortFirst.hasClassName(op.sortFirstAscendingClass)) {				TableKit.Sortable.sort(table, sortFirst, 1);			} else {				TableKit.Sortable.sort(table, sortFirst, -1);			}		} else { // just add row stripe classes			TableKit.Rows.stripe(table);		}	},	reload : function(table) {		table = $(table);		var cells = TableKit.getHeaderCells(table);		var op = TableKit.option('noSortClass columnClass', table.id);		cells.each(function(c){			c = $(c);			if(!c.hasClassName(op.noSortClass)) {				Event.stopObserving(c, 'mousedown', TableKit.Sortable._sort);				c.removeClassName(op.columnClass);			}		});		TableKit.Sortable.init(table);	},	_sort : function(e) {		if(TableKit.Resizable._onHandle) {return;}		e = TableKit.e(e);		Event.stop(e);		var cell = Event.element(e);		while(!(cell.tagName && cell.tagName.match(/td|th/gi))) {			cell = cell.parentNode;		}		TableKit.Sortable.sort(null, cell);	},	sort : function(table, index, order) {		var cell;		if(typeof index === 'number') {			if(!table || (table.tagName && table.tagName !== "TABLE")) {				return;			}			table = $(table);			index = Math.min(table.rows[0].cells.length, index);			index = Math.max(1, index);			index -= 1;			cell = (table.tHead && table.tHead.rows.length > 0) ? $(table.tHead.rows[table.tHead.rows.length-1].cells[index]) : $(table.rows[0].cells[index]);		} else {			cell = $(index);			table = table ? $(table) : cell.up('table');			index = TableKit.getCellIndex(cell);		}		var op = TableKit.option('noSortClass descendingClass ascendingClass', table.id);				if(cell.hasClassName(op.noSortClass)) {return;}					order = order ? order : (cell.hasClassName(op.descendingClass) ? 1 : -1);		var rows = TableKit.getBodyRows(table);		if(cell.hasClassName(op.ascendingClass) || cell.hasClassName(op.descendingClass)) {			rows.reverse(); // if it was already sorted we just need to reverse it.		} else {			var datatype = TableKit.Sortable.getDataType(cell,index,table);			var tkst = TableKit.Sortable.types;			rows.sort(function(a,b) {				return order * tkst[datatype].compare(TableKit.getCellText(a.cells[index]),TableKit.getCellText(b.cells[index]));			});		}		var tb = table.tBodies[0];		var tkr = TableKit.Rows;		rows.each(function(r,i) {			tb.appendChild(r);			tkr.addStripeClass(table,r,i);		});		var hcells = TableKit.getHeaderCells(null, cell);		$A(hcells).each(function(c,i){			c = $(c);			c.removeClassName(op.ascendingClass);			c.removeClassName(op.descendingClass);			if(index === i) {				if(order === 1) {					c.removeClassName(op.descendingClass);					c.addClassName(op.ascendingClass);				} else {					c.removeClassName(op.ascendingClass);					c.addClassName(op.descendingClass);				}			}		});	},	types : {},	detectors : [],	addSortType : function() {		$A(arguments).each(function(o){			TableKit.Sortable.types[o.name] = o;		});	},	getDataType : function(cell,index,table) {		cell = $(cell);		index = (index || index === 0) ? index : TableKit.getCellIndex(cell);				var colcache = TableKit.Sortable._coltypecache;		var cache = colcache[table.id] ? colcache[table.id] : (colcache[table.id] = {});				if(!cache[index]) {			var t = '';			// first look for a data type id on the heading row cell			if(cell.id && TableKit.Sortable.types[cell.id]) {				t = cell.id;			}			t = cell.classNames().detect(function(n){ // then look for a data type classname on the heading row cell				return (TableKit.Sortable.types[n]) ? true : false;			});			if(!t) {				var rows = TableKit.getBodyRows(table);				cell = rows[0].cells[index]; // grab same index cell from body row to try and match data type				t = TableKit.Sortable.detectors.detect(						function(d){							return TableKit.Sortable.types[d].detect(TableKit.getCellText(cell));						});			}			cache[index] = t;		}		return cache[index];	},	_coltypecache : {}};TableKit.Sortable.detectors = $A($w('date-iso date date-eu date-au time currency datasize number casesensitivetext text')); // setting it here because Safari complained when I did it above...TableKit.Sortable.Type = Class.create();TableKit.Sortable.Type.prototype = {	initialize : function(name, options){		this.name = name;		options = Object.extend({			normal : function(v){				return v;			},			pattern : /.*/		}, options || {});		this.normal = options.normal;		this.pattern = options.pattern;		if(options.compare) {			this.compare = options.compare;		}		if(options.detect) {			this.detect = options.detect;		}	},	compare : function(a,b){		return TableKit.Sortable.Type.compare(this.normal(a), this.normal(b));	},	detect : function(v){		return this.pattern.test(v);	}};TableKit.Sortable.Type.compare = function(a,b) {	return a < b ? -1 : a === b ? 0 : 1;};TableKit.Sortable.addSortType(	new TableKit.Sortable.Type('number', {		pattern : /^[-+]?[\d]*\.?[\d]+(?:[eE][-+]?[\d]+)?/,		normal : function(v) {			// This will grab the first thing that looks like a number from a string, so you can use it to order a column of various srings containing numbers.			v = parseFloat(v.replace(/^.*?([-+]?[\d]*\.?[\d]+(?:[eE][-+]?[\d]+)?).*$/,"$1"));			return isNaN(v) ? 0 : v;		}}),	new TableKit.Sortable.Type('text',{		normal : function(v) {			return v ? v.toLowerCase() : '';		}}),	new TableKit.Sortable.Type('casesensitivetext',{pattern : /^[A-Z]+$/}),	new TableKit.Sortable.Type('datasize',{		pattern : /^[-+]?[\d]*\.?[\d]+(?:[eE][-+]?[\d]+)?\s?[k|m|g|t]b$/i,		normal : function(v) {			var r = v.match(/^([-+]?[\d]*\.?[\d]+([eE][-+]?[\d]+)?)\s?([k|m|g|t]?b)?/i);			var b = r[1] ? Number(r[1]).valueOf() : 0;			var m = r[3] ? r[3].substr(0,1).toLowerCase() : '';			var result = b;			switch(m) {				case  'k':					result = b * 1024;					break;				case  'm':									result = b * 1024 * 1024;					break;				case  'g':					result = b * 1024 * 1024 * 1024;					break;				case  't':					result = b * 1024 * 1024 * 1024 * 1024;					break;			}			return result;		}}),	new TableKit.Sortable.Type('date-au',{		pattern : /^\d{2}\/\d{2}\/\d{4}\s?(?:\d{1,2}\:\d{2}(?:\:\d{2})?\s?[a|p]?m?)?/i,		normal : function(v) {			if(!this.pattern.test(v)) {return 0;}			var r = v.match(/^(\d{2})\/(\d{2})\/(\d{4})\s?(?:(\d{1,2})\:(\d{2})(?:\:(\d{2}))?\s?([a|p]?m?))?/i);			var yr_num = r[3];			var mo_num = parseInt(r[2],10)-1;			var day_num = r[1];			var hr_num = r[4] ? r[4] : 0;			if(r[7] && r[7].toLowerCase().indexOf('p') !== -1) {				hr_num = parseInt(r[4],10) + 12;			}			var min_num = r[5] ? r[5] : 0;			var sec_num = r[6] ? r[6] : 0;			return new Date(yr_num, mo_num, day_num, hr_num, min_num, sec_num, 0).valueOf();		}}),	new TableKit.Sortable.Type('date-us',{		pattern : /^\d{2}\/\d{2}\/\d{4}\s?(?:\d{1,2}\:\d{2}(?:\:\d{2})?\s?[a|p]?m?)?/i,		normal : function(v) {			if(!this.pattern.test(v)) {return 0;}			var r = v.match(/^(\d{2})\/(\d{2})\/(\d{4})\s?(?:(\d{1,2})\:(\d{2})(?:\:(\d{2}))?\s?([a|p]?m?))?/i);			var yr_num = r[3];			var mo_num = parseInt(r[1],10)-1;			var day_num = r[2];			var hr_num = r[4] ? r[4] : 0;			if(r[7] && r[7].toLowerCase().indexOf('p') !== -1) {				hr_num = parseInt(r[4],10) + 12;			}			var min_num = r[5] ? r[5] : 0;			var sec_num = r[6] ? r[6] : 0;			return new Date(yr_num, mo_num, day_num, hr_num, min_num, sec_num, 0).valueOf();		}}),	new TableKit.Sortable.Type('date-eu',{		pattern : /^\d{2}-\d{2}-\d{4}/i,		normal : function(v) {			if(!this.pattern.test(v)) {return 0;}			var r = v.match(/^(\d{2})-(\d{2})-(\d{4})/);			var yr_num = r[3];			var mo_num = parseInt(r[2],10)-1;			var day_num = r[1];			return new Date(yr_num, mo_num, day_num).valueOf();		}}),	new TableKit.Sortable.Type('date-iso',{		pattern : /[\d]{4}-[\d]{2}-[\d]{2}(?:T[\d]{2}\:[\d]{2}(?:\:[\d]{2}(?:\.[\d]+)?)?(Z|([-+][\d]{2}:[\d]{2})?)?)?/, // 2005-03-26T19:51:34Z		normal : function(v) {			if(!this.pattern.test(v)) {return 0;}		    var d = v.match(/([\d]{4})(-([\d]{2})(-([\d]{2})(T([\d]{2}):([\d]{2})(:([\d]{2})(\.([\d]+))?)?(Z|(([-+])([\d]{2}):([\d]{2})))?)?)?)?/);				    var offset = 0;		    var date = new Date(d[1], 0, 1);		    if (d[3]) { date.setMonth(d[3] - 1) ;}		    if (d[5]) { date.setDate(d[5]); }		    if (d[7]) { date.setHours(d[7]); }		    if (d[8]) { date.setMinutes(d[8]); }		    if (d[10]) { date.setSeconds(d[10]); }		    if (d[12]) { date.setMilliseconds(Number("0." + d[12]) * 1000); }		    if (d[14]) {		        offset = (Number(d[16]) * 60) + Number(d[17]);		        offset *= ((d[15] === '-') ? 1 : -1);		    }		    offset -= date.getTimezoneOffset();		    if(offset !== 0) {		    	var time = (Number(date) + (offset * 60 * 1000));		    	date.setTime(Number(time));		    }			return date.valueOf();		}}),	new TableKit.Sortable.Type('date',{		pattern: /^(?:sun|mon|tue|wed|thu|fri|sat)\,\s\d{1,2}\s(?:jan|feb|mar|apr|may|jun|jul|aug|sep|oct|nov|dec)\s\d{4}(?:\s\d{2}\:\d{2}(?:\:\d{2})?(?:\sGMT(?:[+-]\d{4})?)?)?/i, //Mon, 18 Dec 1995 17:28:35 GMT		compare : function(a,b) { // must be standard javascript date format			if(a && b) {				return TableKit.Sortable.Type.compare(new Date(a),new Date(b));			} else {				return TableKit.Sortable.Type.compare(a ? 1 : 0, b ? 1 : 0);			}		}}),	new TableKit.Sortable.Type('time',{		pattern : /^\d{1,2}\:\d{2}(?:\:\d{2})?(?:\s[a|p]m)?$/i,		compare : function(a,b) {			var d = new Date();			var ds = d.getMonth() + "/" + d.getDate() + "/" + d.getFullYear() + " ";			return TableKit.Sortable.Type.compare(new Date(ds + a),new Date(ds + b));		}}),	new TableKit.Sortable.Type('currency',{		pattern : /^[$ŁĄ€¤]/, // dollar,pound,yen,euro,generic currency symbol		normal : function(v) {			return v ? parseFloat(v.replace(/[^-\d\.]/g,'')) : 0;		}}));TableKit.Resizable = {	init : function(elm, options){		var table = $(elm);		if(table.tagName !== "TABLE") {return;}		TableKit.register(table,Object.extend(options || {},{resizable:true}));		 		var cells = TableKit.getHeaderCells(table);		cells.each(function(c){			c = $(c);			Event.observe(c, 'mouseover', TableKit.Resizable.initDetect);			Event.observe(c, 'mouseout', TableKit.Resizable.killDetect);		});	},	resize : function(table, index, w) {		var cell;		if(typeof index === 'number') {			if(!table || (table.tagName && table.tagName !== "TABLE")) {return;}			table = $(table);			index = Math.min(table.rows[0].cells.length, index);			index = Math.max(1, index);			index -= 1;			cell = (table.tHead && table.tHead.rows.length > 0) ? $(table.tHead.rows[table.tHead.rows.length-1].cells[index]) : $(table.rows[0].cells[index]);		} else {			cell = $(index);			table = table ? $(table) : cell.up('table');			index = TableKit.getCellIndex(cell);		}		var pad = parseInt(cell.getStyle('paddingLeft'),10) + parseInt(cell.getStyle('paddingRight'),10);		w = Math.max(w-pad, TableKit.option('minWidth', table.id)[0]);				cell.setStyle({'width' : w + 'px'});	},	initDetect : function(e) {		e = TableKit.e(e);		var cell = Event.element(e);		Event.observe(cell, 'mousemove', TableKit.Resizable.detectHandle);		Event.observe(cell, 'mousedown', TableKit.Resizable.startResize);	},	detectHandle : function(e) {		e = TableKit.e(e);		var cell = Event.element(e);  		if(TableKit.Resizable.pointerPos(cell,Event.pointerX(e),Event.pointerY(e))){  			cell.addClassName(TableKit.option('resizeOnHandleClass', cell.up('table').id)[0]);  			TableKit.Resizable._onHandle = true;  		} else {  			cell.removeClassName(TableKit.option('resizeOnHandleClass', cell.up('table').id)[0]);  			TableKit.Resizable._onHandle = false;  		}	},	killDetect : function(e) {		e = TableKit.e(e);		TableKit.Resizable._onHandle = false;		var cell = Event.element(e);		Event.stopObserving(cell, 'mousemove', TableKit.Resizable.detectHandle);		Event.stopObserving(cell, 'mousedown', TableKit.Resizable.startResize);		cell.removeClassName(TableKit.option('resizeOnHandleClass', cell.up('table').id)[0]);	},	startResize : function(e) {		e = TableKit.e(e);		if(!TableKit.Resizable._onHandle) {return;}		var cell = Event.element(e);		Event.stopObserving(cell, 'mousemove', TableKit.Resizable.detectHandle);		Event.stopObserving(cell, 'mousedown', TableKit.Resizable.startResize);		Event.stopObserving(cell, 'mouseout', TableKit.Resizable.killDetect);		TableKit.Resizable._cell = cell;		var table = cell.up('table');		TableKit.Resizable._tbl = table;		if(TableKit.option('showHandle', table.id)[0]) {			TableKit.Resizable._handle = $(document.createElement('div')).addClassName('resize-handle').setStyle({				'top' : Position.cumulativeOffset(cell)[1] + 'px',				'left' : Event.pointerX(e) + 'px',				'height' : table.getDimensions().height + 'px'			});			document.body.appendChild(TableKit.Resizable._handle);		}		Event.observe(document, 'mousemove', TableKit.Resizable.drag);		Event.observe(document, 'mouseup', TableKit.Resizable.endResize);		Event.stop(e);	},	endResize : function(e) {		e = TableKit.e(e);		var cell = TableKit.Resizable._cell;		TableKit.Resizable.resize(null, cell, (Event.pointerX(e) - Position.cumulativeOffset(cell)[0]));		Event.stopObserving(document, 'mousemove', TableKit.Resizable.drag);		Event.stopObserving(document, 'mouseup', TableKit.Resizable.endResize);		if(TableKit.option('showHandle', TableKit.Resizable._tbl.id)[0]) {			$$('div.resize-handle').each(function(elm){				document.body.removeChild(elm);			});		}		Event.observe(cell, 'mouseout', TableKit.Resizable.killDetect);		TableKit.Resizable._tbl = TableKit.Resizable._handle = TableKit.Resizable._cell = null;		Event.stop(e);	},	drag : function(e) {		e = TableKit.e(e);		if(TableKit.Resizable._handle === null) {			try {				TableKit.Resizable.resize(TableKit.Resizable._tbl, TableKit.Resizable._cell, (Event.pointerX(e) - Position.cumulativeOffset(TableKit.Resizable._cell)[0]));			} catch(e) {}		} else {			TableKit.Resizable._handle.setStyle({'left' : Event.pointerX(e) + 'px'});		}		return false;	},	pointerPos : function(element, x, y) {    	var offset = Position.cumulativeOffset(element);	    return (y >= offset[1] &&	            y <  offset[1] + element.offsetHeight &&	            x >= offset[0] + element.offsetWidth - 5 &&	            x <  offset[0] + element.offsetWidth);  	},	_onHandle : false,	_cell : null,	_tbl : null,	_handle : null};TableKit.Editable = {	init : function(elm, options){		var table = $(elm);		if(table.tagName !== "TABLE") {return;}		TableKit.register(table,Object.extend(options || {},{editable:true}));		Event.observe(table.tBodies[0], 'click', TableKit.Editable._editCell);	},	_editCell : function(e) {		e = TableKit.e(e);		var cell = Event.findElement(e,'td');		TableKit.Editable.editCell(null, cell);	},	editCell : function(table, index, cindex) {		var cell, row;		if(typeof index === 'number') {			if(!table || (table.tagName && table.tagName !== "TABLE")) {return;}			table = $(table);			index = Math.min(table.tBodies[0].rows.length, index);			index = Math.max(1, index);			index -= 1;			cindex = Math.min(table.rows[0].cells.length, cindex);			cindex = Math.max(1, cindex);			cindex -= 1;			row = $(table.tBodies[0].rows[index]);			cell = $(row.cells[cindex]);		} else {			cell = $(index);			table = (table && table.tagName && table.tagName !== "TABLE") ? $(table) : cell.up('table');			row = cell.up('tr');		}		var op = TableKit.option('noEditClass', table.id);		if(cell.hasClassName(op.noEditClass)) {return;}				var head = $(TableKit.getHeaderCells(table, cell)[TableKit.getCellIndex(cell)]);		if(head.hasClassName(op.noEditClass)) {return;}				TableKit.registerCell(cell);		var data = TableKit.cells[cell.id];		if(data.active) {return;}		data.htmlContent = cell.innerHTML;		var ftype = TableKit.Editable.types['text-input'];		if(head.id && TableKit.Editable.types[head.id]) {			ftype = TableKit.Editable.types[head.id];		} else {			var n = head.classNames().detect(function(n){					return (TableKit.Editable.types[n]) ? true : false;			});			ftype = n ? TableKit.Editable.types[n] : ftype;		}		ftype.edit(cell);		data.active = true;	},	types : {},	addCellEditor : function(o) {		if(o && o.name) { TableKit.Editable.types[o.name] = o; }	}};TableKit.Editable.CellEditor = Class.create();TableKit.Editable.CellEditor.prototype = {	initialize : function(name, options){		this.name = name;		this.options = Object.extend({			element : 'input',			attributes : {name : 'value', type : 'text'},			selectOptions : [],			showSubmit : true,			submitText : 'OK',			showCancel : true,			cancelText : 'Cancel',			ajaxURI : null,			ajaxOptions : null		}, options || {});	},	edit : function(cell) {		cell = $(cell);		var op = this.options;		var table = cell.up('table');				var form = $(document.createElement("form"));		form.id = cell.id + '-form';		form.addClassName(TableKit.option('formClassName', table.id)[0]);		form.onsubmit = this._submit.bindAsEventListener(this);				var field = document.createElement(op.element);			$H(op.attributes).each(function(v){				field[v.key] = v.value;			});			switch(op.element) {				case 'input':				case 'textarea':				field.value = TableKit.getCellText(cell);				break;								case 'select':				var txt = TableKit.getCellText(cell);				$A(op.selectOptions).each(function(v){					field.options[field.options.length] = new Option(v[0], v[1]);					if(txt === v[1]) {						field.options[field.options.length-1].selected = 'selected';					}				});				break;			}			form.appendChild(field);			if(op.element === 'textarea') {				form.appendChild(document.createElement("br"));			}			if(op.showSubmit) {				var okButton = document.createElement("input");				okButton.type = "submit";				okButton.value = op.submitText;				okButton.className = 'editor_ok_button';				form.appendChild(okButton);			}			if(op.showCancel) {				var cancelLink = document.createElement("a");				cancelLink.href = "#";				cancelLink.appendChild(document.createTextNode(op.cancelText));				cancelLink.onclick = this._cancel.bindAsEventListener(this);				cancelLink.className = 'editor_cancel';      				form.appendChild(cancelLink);			}			cell.innerHTML = '';			cell.appendChild(form);	},	_submit : function(e) {		var cell = Event.findElement(e,'td');		var form = Event.findElement(e,'form');		Event.stop(e);		this.submit(cell,form);	},	submit : function(cell, form) {		var op = this.options;		form = form ? form : cell.down('form');		var head = $(TableKit.getHeaderCells(null, cell)[TableKit.getCellIndex(cell)]);		var row = cell.up('tr');		var table = cell.up('table');		var s = '&row=' + (TableKit.getRowIndex(row)+1) + '&cell=' + (TableKit.getCellIndex(cell)+1) + '&id=' + row.id + '&field=' + head.id + '&' + Form.serialize(form);		this.ajax = new Ajax.Updater(cell, op.ajaxURI || TableKit.option('editAjaxURI', table.id)[0], Object.extend(op.ajaxOptions || TableKit.option('editAjaxOptions', table.id)[0], {			postBody : s,			onComplete : function() {				var data = TableKit.cells[cell.id];				data.active = false;				data.refresh = true; // mark cell cache for refreshing, in case cell contents has changed and sorting is applied			}		}));	},	_cancel : function(e) {		var cell = Event.findElement(e,'td');		Event.stop(e);		this.cancel(cell);	},	cancel : function(cell) {		this.ajax = null;		var data = TableKit.cells[cell.id];		cell.innerHTML = data.htmlContent;		data.htmlContent = '';		data.active = false;	},	ajax : null};TableKit.Editable.textInput = function(n,attributes) {	TableKit.Editable.addCellEditor(new TableKit.Editable.CellEditor(n, {		element : 'input',		attributes : Object.extend({name : 'value', type : 'text'}, attributes||{})	}));};TableKit.Editable.textInput('text-input');TableKit.Editable.multiLineInput = function(n,attributes) {	TableKit.Editable.addCellEditor(new TableKit.Editable.CellEditor(n, {		element : 'textarea',		attributes : Object.extend({name : 'value', rows : '5', cols : '20'}, attributes||{})	}));	};	TableKit.Editable.multiLineInput('multi-line-input');TableKit.Editable.selectInput = function(n,attributes,selectOptions) {	TableKit.Editable.addCellEditor(new TableKit.Editable.CellEditor(n, {		element : 'select',		attributes : Object.extend({name : 'value'}, attributes||{}),		'selectOptions' : selectOptions	}));	};/*TableKit.Bench = {	bench : [],	start : function(){		TableKit.Bench.bench[0] = new Date().getTime();	},	end : function(s){		TableKit.Bench.bench[1] = new Date().getTime();		alert(s + ' ' + ((TableKit.Bench.bench[1]-TableKit.Bench.bench[0])/1000)+' seconds.') //console.log(s + ' ' + ((TableKit.Bench.bench[1]-TableKit.Bench.bench[0])/1000)+' seconds.')		TableKit.Bench.bench = [];	}} */if(window.FastInit) {	FastInit.addOnLoad(TableKit.load);} else {	Event.observe(window, 'load', TableKit.load);}
