
(function($){
    $.extend({
        tablesorter:new function(){
            var parsers=[],widgets=[];this.defaults={
                cssHeader:"header",
                cssAsc:"headerSortUp",
                cssDesc:"headerSortDown",
                sortInitialOrder:"asc",
                sortMultiSortKey:"shiftKey",
                sortForce:null,
                sortAppend:null,
                textExtraction:"simple",
                parsers:{},
                widgets:[],
                widgetZebra:{
                    css:["even","odd"]
                    },
                headers:{},
                widthFixed:false,
                cancelSelection:true,
                sortList:[],
                headerList:[],
                dateFormat:"us",
                decimal:'.',
                debug:false
            };function benchmark(s,d){
                log(s+","+(new Date().getTime()-d.getTime())+"ms");
            }this.benchmark=benchmark;function log(s){
                if(typeof console!="undefined"&&typeof console.debug!="undefined"){
                    console.log(s);
                }else{
                    alert(s);
                }
                }function buildParserCache(table,$headers){
                if(table.config.debug){
                    var parsersDebug="";
                }var rows=table.tBodies[0].rows;if(table.tBodies[0].rows[0]){
                    var list=[],cells=rows[0].cells,l=cells.length;for(var i=0;i<l;i++){
                        var p=false;if($.metadata&&($($headers[i]).metadata()&&$($headers[i]).metadata().sorter)){
                            p=getParserById($($headers[i]).metadata().sorter);
                        }else if((table.config.headers[i]&&table.config.headers[i].sorter)){
                            p=getParserById(table.config.headers[i].sorter);
                        }if(!p){
                            p=detectParserForColumn(table,cells[i]);
                        }if(table.config.debug){
                            parsersDebug+="column:"+i+" parser:"+p.id+"\n";
                        }list.push(p);
                    }
                    }if(table.config.debug){
                    log(parsersDebug);
                }return list;
            };function detectParserForColumn(table,node){
                var l=parsers.length;for(var i=1;i<l;i++){
                    if(parsers[i].is($.trim(getElementText(table.config,node)),table,node)){
                        return parsers[i];
                    }
                    }return parsers[0];
            }function getParserById(name){
                var l=parsers.length;for(var i=0;i<l;i++){
                    if(parsers[i].id.toLowerCase()==name.toLowerCase()){
                        return parsers[i];
                    }
                    }return false;
            }function buildCache(table){
                if(table.config.debug){
                    var cacheTime=new Date();
                }var totalRows=(table.tBodies[0]&&table.tBodies[0].rows.length)||0,totalCells=(table.tBodies[0].rows[0]&&table.tBodies[0].rows[0].cells.length)||0,parsers=table.config.parsers,cache={
                    row:[],
                    normalized:[]
                };for(var i=0;i<totalRows;++i){
                    var c=table.tBodies[0].rows[i],cols=[];cache.row.push($(c));for(var j=0;j<totalCells;++j){
                        cols.push(parsers[j].format(getElementText(table.config,c.cells[j]),table,c.cells[j]));
                    }cols.push(i);cache.normalized.push(cols);cols=null;
                };if(table.config.debug){
                    benchmark("Building cache for "+totalRows+" rows:",cacheTime);
                }return cache;
            };function getElementText(config,node){
                if(!node)return"";var t="";if(config.textExtraction=="simple"){
                    if(node.childNodes[0]&&node.childNodes[0].hasChildNodes()){
                        t=node.childNodes[0].innerHTML;
                    }else{
                        t=node.innerHTML;
                    }
                    }else{
                    if(typeof(config.textExtraction)=="function"){
                        t=config.textExtraction(node);
                    }else{
                        t=$(node).text();
                    }
                    }return t;
            }function appendToTable(table,cache){
                if(table.config.debug){
                    var appendTime=new Date()
                    }var c=cache,r=c.row,n=c.normalized,totalRows=n.length,checkCell=(n[0].length-1),tableBody=$(table.tBodies[0]),rows=[];for(var i=0;i<totalRows;i++){
                    rows.push(r[n[i][checkCell]]);if(!table.config.appender){
                        var o=r[n[i][checkCell]];var l=o.length;for(var j=0;j<l;j++){
                            tableBody[0].appendChild(o[j]);
                        }
                        }
                    }if(table.config.appender){
                    table.config.appender(table,rows);
                }rows=null;if(table.config.debug){
                    benchmark("Rebuilt table:",appendTime);
                }applyWidget(table);setTimeout(function(){
                    $(table).trigger("sortEnd");
                },0);
            };function buildHeaders(table){
                if(table.config.debug){
                    var time=new Date();
                }var meta=($.metadata)?true:false,tableHeadersRows=[];for(var i=0;i<table.tHead.rows.length;i++){
                    tableHeadersRows[i]=0;
                };$tableHeaders=$("thead th",table);$tableHeaders.each(function(index){
                    this.count=0;this.column=index;this.order=formatSortingOrder(table.config.sortInitialOrder);if(checkHeaderMetadata(this)||checkHeaderOptions(table,index))this.sortDisabled=true;if(!this.sortDisabled){
                        $(this).addClass(table.config.cssHeader);
                    }table.config.headerList[index]=this;
                });if(table.config.debug){
                    benchmark("Built headers:",time);log($tableHeaders);
                }return $tableHeaders;
            };function checkCellColSpan(table,rows,row){
                var arr=[],r=table.tHead.rows,c=r[row].cells;for(var i=0;i<c.length;i++){
                    var cell=c[i];if(cell.colSpan>1){
                        arr=arr.concat(checkCellColSpan(table,headerArr,row++));
                    }else{
                        if(table.tHead.length==1||(cell.rowSpan>1||!r[row+1])){
                            arr.push(cell);
                        }
                        }
                    }return arr;
            };function checkHeaderMetadata(cell){
                if(($.metadata)&&($(cell).metadata().sorter===false)){
                    return true;
                };return false;
            }function checkHeaderOptions(table,i){
                if((table.config.headers[i])&&(table.config.headers[i].sorter===false)){
                    return true;
                };return false;
            }function applyWidget(table){
                var c=table.config.widgets;var l=c.length;for(var i=0;i<l;i++){
                    getWidgetById(c[i]).format(table);
                }
                }function getWidgetById(name){
                var l=widgets.length;for(var i=0;i<l;i++){
                    if(widgets[i].id.toLowerCase()==name.toLowerCase()){
                        return widgets[i];
                    }
                    }
                };function formatSortingOrder(v){
                if(typeof(v)!="Number"){
                    i=(v.toLowerCase()=="desc")?1:0;
                }else{
                    i=(v==(0||1))?v:0;
                }return i;
            }function isValueInArray(v,a){
                var l=a.length;for(var i=0;i<l;i++){
                    if(a[i][0]==v){
                        return true;
                    }
                    }return false;
            }function setHeadersCss(table,$headers,list,css){
                $headers.removeClass(css[0]).removeClass(css[1]);var h=[];$headers.each(function(offset){
                    if(!this.sortDisabled){
                        h[this.column]=$(this);
                    }
                    });var l=list.length;for(var i=0;i<l;i++){
                    h[list[i][0]].addClass(css[list[i][1]]);
                }
                }function fixColumnWidth(table,$headers){
                var c=table.config;if(c.widthFixed){
                    var colgroup=$('<colgroup>');$("tr:first td",table.tBodies[0]).each(function(){
                        colgroup.append($('<col>').css('width',$(this).width()));
                    });$(table).prepend(colgroup);
                };
            }function updateHeaderSortCount(table,sortList){
                var c=table.config,l=sortList.length;for(var i=0;i<l;i++){
                    var s=sortList[i],o=c.headerList[s[0]];o.count=s[1];o.count++;
                }
                }function multisort(table,sortList,cache){
                if(table.config.debug){
                    var sortTime=new Date();
                }var dynamicExp="var sortWrapper = function(a,b) {",l=sortList.length;for(var i=0;i<l;i++){
                    var c=sortList[i][0];var order=sortList[i][1];var s=(getCachedSortType(table.config.parsers,c)=="text")?((order==0)?"sortText":"sortTextDesc"):((order==0)?"sortNumeric":"sortNumericDesc");var e="e"+i;dynamicExp+="var "+e+" = "+s+"(a["+c+"],b["+c+"]); ";dynamicExp+="if("+e+") { return "+e+"; } ";dynamicExp+="else { ";
                }var orgOrderCol=cache.normalized[0].length-1;dynamicExp+="return a["+orgOrderCol+"]-b["+orgOrderCol+"];";for(var i=0;i<l;i++){
                    dynamicExp+="}; ";
                }dynamicExp+="return 0; ";dynamicExp+="}; ";eval(dynamicExp);cache.normalized.sort(sortWrapper);if(table.config.debug){
                    benchmark("Sorting on "+sortList.toString()+" and dir "+order+" time:",sortTime);
                }return cache;
            };function sortText(a,b){
                return((a<b)?-1:((a>b)?1:0));
            };function sortTextDesc(a,b){
                return((b<a)?-1:((b>a)?1:0));
            };function sortNumeric(a,b){
                return a-b;
            };function sortNumericDesc(a,b){
                return b-a;
            };function getCachedSortType(parsers,i){
                return parsers[i].type;
            };this.construct=function(settings){
                return this.each(function(){
                    if(!this.tHead||!this.tBodies)return;var $this,$document,$headers,cache,config,shiftDown=0,sortOrder;this.config={};config=$.extend(this.config,$.tablesorter.defaults,settings);$this=$(this);$headers=buildHeaders(this);this.config.parsers=buildParserCache(this,$headers);cache=buildCache(this);var sortCSS=[config.cssDesc,config.cssAsc];fixColumnWidth(this);$headers.click(function(e){
                        $this.trigger("sortStart");var totalRows=($this[0].tBodies[0]&&$this[0].tBodies[0].rows.length)||0;if(!this.sortDisabled&&totalRows>0){
                            var $cell=$(this);var i=this.column;this.order=this.count++%2;if(!e[config.sortMultiSortKey]){
                                config.sortList=[];if(config.sortForce!=null){
                                    var a=config.sortForce;for(var j=0;j<a.length;j++){
                                        if(a[j][0]!=i){
                                            config.sortList.push(a[j]);
                                        }
                                        }
                                    }config.sortList.push([i,this.order]);
                            }else{
                                if(isValueInArray(i,config.sortList)){
                                    for(var j=0;j<config.sortList.length;j++){
                                        var s=config.sortList[j],o=config.headerList[s[0]];if(s[0]==i){
                                            o.count=s[1];o.count++;s[1]=o.count%2;
                                        }
                                        }
                                    }else{
                                    config.sortList.push([i,this.order]);
                                }
                                };setTimeout(function(){
                                setHeadersCss($this[0],$headers,config.sortList,sortCSS);appendToTable($this[0],multisort($this[0],config.sortList,cache));
                            },1);return false;
                        }
                        }).mousedown(function(){
                        if(config.cancelSelection){
                            this.onselectstart=function(){
                                return false
                                };return false;
                        }
                        });$this.bind("update",function(){
                        this.config.parsers=buildParserCache(this,$headers);cache=buildCache(this);
                    }).bind("sorton",function(e,list){
                        $(this).trigger("sortStart");config.sortList=list;var sortList=config.sortList;updateHeaderSortCount(this,sortList);setHeadersCss(this,$headers,sortList,sortCSS);appendToTable(this,multisort(this,sortList,cache));
                    }).bind("appendCache",function(){
                        appendToTable(this,cache);
                    }).bind("applyWidgetId",function(e,id){
                        getWidgetById(id).format(this);
                    }).bind("applyWidgets",function(){
                        applyWidget(this);
                    });if($.metadata&&($(this).metadata()&&$(this).metadata().sortlist)){
                        config.sortList=$(this).metadata().sortlist;
                    }if(config.sortList.length>0){
                        $this.trigger("sorton",[config.sortList]);
                    }applyWidget(this);
                });
            };this.addParser=function(parser){
                var l=parsers.length,a=true;for(var i=0;i<l;i++){
                    if(parsers[i].id.toLowerCase()==parser.id.toLowerCase()){
                        a=false;
                    }
                    }if(a){
                    parsers.push(parser);
                };
            };this.addWidget=function(widget){
                widgets.push(widget);
            };this.formatFloat=function(s){
                var i=parseFloat(s);return(isNaN(i))?0:i;
            };this.formatInt=function(s){
                var i=parseInt(s);return(isNaN(i))?0:i;
            };this.isDigit=function(s,config){
                var DECIMAL='\\'+config.decimal;var exp='/(^[+]?0('+DECIMAL+'0+)?$)|(^([-+]?[1-9][0-9]*)$)|(^([-+]?((0?|[1-9][0-9]*)'+DECIMAL+'(0*[1-9][0-9]*)))$)|(^[-+]?[1-9]+[0-9]*'+DECIMAL+'0+$)/';return RegExp(exp).test($.trim(s));
            };this.clearTableBody=function(table){
                if($.browser.msie){
                    function empty(){
                        while(this.firstChild)this.removeChild(this.firstChild);
                    }empty.apply(table.tBodies[0]);
                }else{
                    table.tBodies[0].innerHTML="";
                }
                };
        }
        });$.fn.extend({
        tablesorter:$.tablesorter.construct
        });var ts=$.tablesorter;ts.addParser({
        id:"text",
        is:function(s){
            return true;
        },
        format:function(s){
            return $.trim(s.toLowerCase());
        },
        type:"text"
    });ts.addParser({
        id:"digit",
        is:function(s,table){
            var c=table.config;return $.tablesorter.isDigit(s,c);
        },
        format:function(s){
            return $.tablesorter.formatFloat(s);
        },
        type:"numeric"
    });ts.addParser({
        id:"currency",
        is:function(s){
            return/^[£$€?.]/.test(s);
        },
        format:function(s){
            return $.tablesorter.formatFloat(s.replace(new RegExp(/[^0-9.]/g),""));
        },
        type:"numeric"
    });ts.addParser({
        id:"ipAddress",
        is:function(s){
            return/^\d{2,3}[\.]\d{2,3}[\.]\d{2,3}[\.]\d{2,3}$/.test(s);
        },
        format:function(s){
            var a=s.split("."),r="",l=a.length;for(var i=0;i<l;i++){
                var item=a[i];if(item.length==2){
                    r+="0"+item;
                }else{
                    r+=item;
                }
                }return $.tablesorter.formatFloat(r);
        },
        type:"numeric"
    });ts.addParser({
        id:"url",
        is:function(s){
            return/^(https?|ftp|file):\/\/$/.test(s);
        },
        format:function(s){
            return jQuery.trim(s.replace(new RegExp(/(https?|ftp|file):\/\//),''));
        },
        type:"text"
    });ts.addParser({
        id:"isoDate",
        is:function(s){
            return/^\d{4}[\/-]\d{1,2}[\/-]\d{1,2}$/.test(s);
        },
        format:function(s){
            return $.tablesorter.formatFloat((s!="")?new Date(s.replace(new RegExp(/-/g),"/")).getTime():"0");
        },
        type:"numeric"
    });ts.addParser({
        id:"percent",
        is:function(s){
            return/\%$/.test($.trim(s));
        },
        format:function(s){
            return $.tablesorter.formatFloat(s.replace(new RegExp(/%/g),""));
        },
        type:"numeric"
    });ts.addParser({
        id:"usLongDate",
        is:function(s){
            return s.match(new RegExp(/^[A-Za-z]{3,10}\.? [0-9]{1,2}, ([0-9]{4}|'?[0-9]{2}) (([0-2]?[0-9]:[0-5][0-9])|([0-1]?[0-9]:[0-5][0-9]\s(AM|PM)))$/));
        },
        format:function(s){
            return $.tablesorter.formatFloat(new Date(s).getTime());
        },
        type:"numeric"
    });ts.addParser({
        id:"shortDate",
        is:function(s){
            return/\d{1,2}[\/\-]\d{1,2}[\/\-]\d{2,4}/.test(s);
        },
        format:function(s,table){
            var c=table.config;s=s.replace(/\-/g,"/");if(c.dateFormat=="us"){
                s=s.replace(/(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{4})/,"$3/$1/$2");
            }else if(c.dateFormat=="uk"){
                s=s.replace(/(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{4})/,"$3/$2/$1");
            }else if(c.dateFormat=="dd/mm/yy"||c.dateFormat=="dd-mm-yy"){
                s=s.replace(/(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{2})/,"$1/$2/$3");
            }return $.tablesorter.formatFloat(new Date(s).getTime());
        },
        type:"numeric"
    });ts.addParser({
        id:"time",
        is:function(s){
            return/^(([0-2]?[0-9]:[0-5][0-9])|([0-1]?[0-9]:[0-5][0-9]\s(am|pm)))$/.test(s);
        },
        format:function(s){
            return $.tablesorter.formatFloat(new Date("2000/01/01 "+s).getTime());
        },
        type:"numeric"
    });ts.addParser({
        id:"metadata",
        is:function(s){
            return false;
        },
        format:function(s,table,cell){
            var c=table.config,p=(!c.parserMetadataName)?'sortValue':c.parserMetadataName;return $(cell).metadata()[p];
        },
        type:"numeric"
    });ts.addWidget({
        id:"zebra",
        format:function(table){
            if(table.config.debug){
                var time=new Date();
            }$("tr:visible",table.tBodies[0]).filter(':even').removeClass(table.config.widgetZebra.css[1]).addClass(table.config.widgetZebra.css[0]).end().filter(':odd').removeClass(table.config.widgetZebra.css[0]).addClass(table.config.widgetZebra.css[1]);if(table.config.debug){
                $.tablesorter.benchmark("Applying Zebra widget",time);
            }
            }
        });
})(jQuery);