// Copyright 2007 - 2012 Gennadiy Shvets
// The program is distributed under the terms of the GNU General
// Public License 3.0
//
// See http://www.allmyscripts.com/Table_Sort/index.html for usage details.
// Script version 1.9
var TSort_Store;
var TSort_All;
function TSort_StoreDef() {
this.sorting = [];
this.nodes = [];
this.rows = [];
this.row_clones = [];
this.row_nosort = [];
this.sort_state = [];
this.initialized = 0;
this.append_classes = 0;
this.n_columns = 3;
// this.last_sorted = -1;
this.history = [];
this.sort_keys = [];
this.sort_colors = ['#FF0000', '#800080', '#0000FF'];
};
function tsInitOnload() {
// If TSort_All is not initialized - do it now (simulate old behavior)
if (TSort_All == null)
tsRegister();
for (var id in TSort_All) {
tsSetTable(id);
tsInit();
}
if (window.onload_sort_table)
window.onload_sort_table();
}
function tsInit() {
if (TSort_Data.push == null)
return;
var table_id = TSort_Data[0];
var table = document.getElementById(table_id);
// Find thead & tbody data
var cols, i, node, len, tr;
var thead = table.getElementsByTagName('thead')[0];
var tbody = table.getElementsByTagName('tbody')[0];
if (thead == null) {
thead = document.createElement('thead');
table.insertBefore(thead, tbody);
tr = tbody.getElementsByTagName('tr');
len = tr.length;
if (len == 0) {
alert('Cannot find THEAD and TH tags!');
return;
}
var trh = new Array();
for (i = 0; i < len; i++) {
cols = tr[i].getElementsByTagName('th');
if (!cols.length) continue;
trh.push(tr[i]);
}
len = trh.length;
for (i = 0; i < len; i++) {
tbody.removeChild(trh[i]);
thead.appendChild(trh[i]);
}
}
tr = thead.getElementsByTagName('tr');
if (tr.length == 0) return;
if (tr.length > 1) {
var cols0 = tr[0].getElementsByTagName('th');
if (cols0.length == 0)
cols0 = tr[0].getElementsByTagName('td');
var cols1;
var cols1 = tr[1].getElementsByTagName('th');
if (cols1.length == 0)
cols1 = tr[1].getElementsByTagName('td');
cols = new Array();
var j0, j1, n;
len = cols0.length;
for (j0 = 0, j1 = 0; j0 < len; j0++) {
node = cols0[j0];
n = node.colSpan;
if (n > 1) {
while (n > 0) {
cols.push(cols1[j1++]);
n--;
}
} else {
if (node.rowSpan == 1)
j1++;
cols.push(node);
}
}
} else {
cols = tr[0].getElementsByTagName('th');
if (cols.length == 0)
cols = tr[0].getElementsByTagName('td');
}
var cols_len = cols.length;
for (var i = 0; i < cols_len; i++) {
if (i >= TSort_Data.length - 1)
break;
node = cols[i];
var sorting = TSort_Data[i + 1].toLowerCase();
if (sorting == null) sorting = '';
TSort_Store.sorting.push(sorting);
if ((sorting != null) && (sorting != '')) {
// node.tsort_col_id = i;
// node.tsort_table_id = table_id;
// node.onclick = tsDraw;
node.innerHTML = "" + node.innerHTML +
'';
node.style.cursor = "pointer";
}
}
// Parse body rows
var rows = tbody.getElementsByTagName('tr');
if (rows.length == 0) return;
var date = new Date();
var text, a, cn, k;
var attached = TSort_Store.row_nosort;
for (i = 0, k = -1; i < rows.length; i++) {
var row = rows[i];
cn = row.className;
if ((cn != null) && (cn.match(/(^| )_nosort( |$)/))) {
// Save a reference to the TR element
var new_row = row.cloneNode(true);
if (attached[k + 1] == null)
attached[k + 1] = new Array(new_row);
else
attached[k + 1].push(new_row);
continue;
}
var cols = row.getElementsByTagName('td');
len = cols.length;
var row_data = [];
for (j = 0; j < len; j++) {
// Get cell text
text = cols[j].innerHTML;
var sorting = TSort_Store.sorting[j];
if (sorting != 's') {
text = text.replace(/<[^>]+>/g, '');
text = text.replace(/\ /, ' ');
}
text = text.replace(/^\s+/, '');
text = text.replace(/\s+$/, '');
if (sorting == 'h') {
text = text.toLowerCase();
} else if (sorting == 's')
text = text.toLowerCase();
else if (sorting == 'i') {
text = parseInt(text);
if (isNaN(text)) text = 0;
} else if (sorting == 'n') {
text = text.replace(/(\d)\,(?=\d\d\d)/g, "$1");
text = parseInt(text);
if (isNaN(text)) text = 0;
} else if (sorting == 'c') {
text = text.replace(/^(\-?)\$/, "$1");
text = text.replace(/(\d)\,(?=\d\d\d)/g, "$1");
text = parseFloat(text);
if (isNaN(text)) text = 0;
} else if (sorting == 'f') {
text = parseFloat(text);
if (isNaN(text)) text = 0;
} else if (sorting == 'g') {
text = text.replace(/(\d)\,(?=\d\d\d)/g, "$1");
text = parseFloat(text);
if (isNaN(text)) text = 0;
} else if (sorting == 'd') {
if (text.match(/^\d\d\d\d\-\d\d?\-\d\d?(?: \d\d?:\d\d?:\d\d?)?$/)) {
a = text.split(/[\s\-:]/);
text = (a[3] == null) ?
Date.UTC(a[0], a[1] - 1, a[2], 0, 0, 0, 0) :
Date.UTC(a[0], a[1] - 1, a[2], a[3], a[4], a[5], 0);
} else
text = Date.parse(text);
if (isNaN(text)) text = 0;
}
row_data.push(text);
}
// Initialize the rest of the columns, that are not in
for (; j < cols_len; j++) {
// Get cell text
var sorting = TSort_Store.sorting[j];
text = '';
if ((sorting == 'h') || (sorting == 's'))
text = '';
else
text = 0;
row_data.push(text);
}
TSort_Store.rows.push(row_data);
// Save a reference to the TR element
var new_row = row.cloneNode(true);
k++;
new_row.tsort_row_id = k;
TSort_Store.row_clones[k] = new_row;
}
TSort_Store.initialized = 1;
if (TSort_Store.cookie) {
var allc = document.cookie;
i = allc.indexOf(TSort_Store.cookie + '=');
if (i != -1) {
i += TSort_Store.cookie.length + 1;
len = allc.indexOf(";", i);
text = decodeURIComponent(allc.substring(i, (len == -1) ?
allc.length : len));
TSort_Store.initial = (text == '') ? null : text.split(/\s*,\s*/);
}
}
var initial = TSort_Store.initial;
if (initial != null) {
var itype = typeof initial;
if ((itype == 'number') || (itype == 'string'))
tsDraw(initial);
else {
for (i = initial.length - 1; i >= 0; i--)
tsDraw(initial[i]);
}
}
}
function tsDraw(p_id, p_table) {
if (p_table != null)
tsSetTable(p_table);
if ((TSort_Store == null) || (TSort_Store.initialized == 0))
return;
var i = 0;
var sort_keys = TSort_Store.sort_keys;
var id;
var new_order = '';
if (p_id != null) {
if (typeof p_id == 'number')
id = p_id;
else if ((typeof p_id == 'string') && (p_id.match(/^\d+[ADU]$/i))) {
id = p_id.replace(/^(\d+)[ADU]$/i, "$1");
new_order = p_id.replace(/^\d+([ADU])$/i, "$1").toUpperCase();
}
}
if (id == null) {
id = this.tsort_col_id;
if ((p_table == null) && (this.tsort_table_id != null))
tsSetTable(this.tsort_table_id);
}
var table_id = TSort_Data[0];
var order = TSort_Store.sort_state[id];
if (new_order == 'U') {
if (order != null) {
TSort_Store.sort_state[id] = null;
obj = document.getElementById('TS_' + id + '_' + table_id);
if (obj != null) obj.innerHTML = '';
}
} else if (new_order != '') {
TSort_Store.sort_state[id] = (new_order == 'A') ? true : false;
// Add column number to the sort keys array
sort_keys.unshift(id);
i = 1;
} else {
if ((order == null) || (order == true)) {
TSort_Store.sort_state[id] = (order == null) ? true : false;
// Add column number to the sort keys array
sort_keys.unshift(id);
i = 1;
} else {
TSort_Store.sort_state[id] = null;
obj = document.getElementById('TS_' + id + '_' + table_id);
if (obj != null) obj.innerHTML = '';
}
}
var len = sort_keys.length;
// This will either remove the column completely from the sort_keys
// array (i = 0) or remove duplicate column number if present (i = 1).
while (i < len) {
if (sort_keys[i] == id) {
sort_keys.splice(i, 1);
len--;
break;
}
i++;
}
if (len > TSort_Store.n_columns) {
i = sort_keys.pop();
obj = document.getElementById('TS_' + i + '_' + table_id);
if (obj != null) obj.innerHTML = '';
TSort_Store.sort_state[i] = null;
}
// Sort the rows
TSort_Store.row_clones.sort(tsSort);
// Save the currently selected order
var new_tbody = document.createElement('tbody');
var row_clones = TSort_Store.row_clones;
len = row_clones.length;
var classes = TSort_Store.classes;
var alen, j, cn;
var arows = TSort_Store.row_nosort[0];
if (classes == null) {
if (arows != null) {
alen = arows.length;
for (j = 0; j < alen; j++)
new_tbody.appendChild(arows[j].cloneNode(true));
}
for (i = 0; i < len; i++) {
row = row_clones[i];
new_tbody.appendChild(row.cloneNode(true));
arows = TSort_Store.row_nosort[row.tsort_row_id + 1];
if (arows == null) continue;
alen = arows.length;
for (j = 0; j < alen; j++)
new_tbody.appendChild(arows[j].cloneNode(true));
}
} else {
var clone;
var cl = 0;
var cl_len = classes.length;
var append = TSort_Store.append_classes;
if (arows != null) {
alen = arows.length;
for (j = 0; j < alen; j++) {
clone = arows[j].cloneNode(true);
cn = clone.className;
clone.className = ((append) && (cn != null) && (cn.length > 0)) ?
cn + ' ' + classes[cl] : classes[cl];
new_tbody.appendChild(clone);
}
cl++;
if (cl >= cl_len) cl = 0;
}
for (i = 0; i < len; i++) {
row = row_clones[i];
clone = row.cloneNode(true);
cn = clone.className;
clone.className = ((append) && (cn != null) && (cn.length > 0)) ?
cn + ' ' + classes[cl] : classes[cl];
new_tbody.appendChild(clone);
arows = TSort_Store.row_nosort[row.tsort_row_id + 1];
if (arows != null) {
alen = arows.length;
for (j = 0; j < alen; j++) {
clone = arows[j].cloneNode(true);
cn = clone.className;
clone.className = ((append) && (cn != null) && (cn.length > 0)) ?
cn + ' ' + classes[cl] : classes[cl];
new_tbody.appendChild(clone);
}
}
cl++;
if (cl >= cl_len) cl = 0;
}
}
// Replace table body
var table = document.getElementById(table_id);
var tbody = table.getElementsByTagName('tbody')[0];
table.removeChild(tbody);
table.appendChild(new_tbody);
var obj, color, icon, state;
len = sort_keys.length;
var sorting = new Array();
for (i = 0; i < len; i++) {
id = sort_keys[i];
obj = document.getElementById('TS_' + id + '_' + table_id);
if (obj == null) continue;
state = (TSort_Store.sort_state[id]) ? 0 : 1;
icon = TSort_Store.icons[state];
obj.innerHTML = (icon.match(/)) ? icon :
'' + icon + '';
sorting.push(id + ((state) ? 'D' : 'A'));
}
if (TSort_Store.cookie) {
// Store the contents of "sorting" array into a cookie for 30 days
var date = new Date();
date.setTime(date.getTime() + 2592000);
document.cookie = TSort_Store.cookie + "=" +
encodeURIComponent(sorting.join(',')) + "; expires=" +
date.toGMTString() + "; path=/";
}
}
function tsSort(a, b) {
var data_a = TSort_Store.rows[a.tsort_row_id];
var data_b = TSort_Store.rows[b.tsort_row_id];
var sort_keys = TSort_Store.sort_keys;
var len = sort_keys.length;
var id;
var type;
var order;
var result;
for (var i = 0; i < len; i++) {
id = sort_keys[i];
type = TSort_Store.sorting[id];
var v_a = data_a[id];
var v_b = data_b[id];
if (v_a == v_b) continue;
if ((type == 'i') || (type == 'f') || (type == 'd'))
result = v_a - v_b;
else
result = (v_a < v_b) ? -1 : 1;
order = TSort_Store.sort_state[id];
return (order) ? result : 0 - result;
}
return (a.tsort_row_id < b.tsort_row_id) ? -1 : 1;
}
function tsRegister() {
if (typeof TSort_Data == 'undefined') return;
if (TSort_All == null)
TSort_All = new Object();
var ts_obj = new TSort_StoreDef();
ts_obj.sort_data = TSort_Data;
TSort_Data = null;
if (typeof TSort_Classes != 'undefined') {
ts_obj.classes = TSort_Classes;
TSort_Classes = null;
}
if (typeof TSort_Initial != 'undefined') {
ts_obj.initial = TSort_Initial;
TSort_Initial = null;
}
if (typeof TSort_Cookie != 'undefined') {
ts_obj.cookie = TSort_Cookie;
TSort_Cookie = null;
}
if (typeof TSort_Icons != 'undefined') {
ts_obj.icons = TSort_Icons;
TSort_Icons = null;
}
if (ts_obj.icons == null)
ts_obj.icons = new Array("\u2193", "\u2191");
if (typeof TSort_AppendClasses != 'undefined') {
ts_obj.append_classes = TSort_AppendClasses;
TSort_AppendClasses = null;
}
if (typeof TSort_NColumns != 'undefined') {
ts_obj.n_columns = TSort_NColumns;
TSort_NColumns = null;
if (ts_obj.n_columns == null)
ts_obj.n_columns = 3;
}
if (ts_obj.sort_data != null)
TSort_All[ts_obj.sort_data[0]] = ts_obj;
}
function tsSetTable(p_id) {
TSort_Store = TSort_All[p_id];
if (TSort_Store == null) {
alert("Cannot set table '" + p_id + "' - table is not registered");
return;
}
TSort_Data = TSort_Store.sort_data;
}
if (window.addEventListener)
window.addEventListener("load", tsInitOnload, false);
else if (window.attachEvent)
window.attachEvent("onload", tsInitOnload);
else {
if ((window.onload_sort_table == null) && (window.onload != null))
window.onload_sort_table = window.onload;
// Assign new onload function
window.onload = tsInitOnload;
}