visual_db 0.1.0 → 0.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: 698c5b761f2c0623ec6e7b21094fedae9e8bb365
4
- data.tar.gz: e51187658cad7b249f7f19bfcb1bedc079b430ea
3
+ metadata.gz: 8dc6aa78f67c08e060b0707d61109ce1a1a47b7a
4
+ data.tar.gz: 260895a6aeb9f448dc4bbfc605cdcce96493dc6e
5
5
  SHA512:
6
- metadata.gz: e071c52883093ba317422c6e8f78b3c739e929443a39c7aa28f409a29f92133cdd4cc26bb523bd8fc2f7d3e5a8bd4a31a58f4732f053e444b5a360b3d947ba13
7
- data.tar.gz: 87050a80e155f02af6f73da297dce925679343c11b611ca8329942dd22409fea3509162eef87344c3a792a6875e0b4784a31bf4cb75b2d0c4465523568f09eba
6
+ metadata.gz: e542ea7dc9025cf1ae55f118c4c9cb327bfd99513b9d66a9f6ce3de15135321eb01d0124c5a2a41030ce51258b9cd38b270830071dac30a0188edc166c6fde79
7
+ data.tar.gz: 70827442114e16ee10070960c8323cefc22593388788b4bcf18a3dcbc22925dd2f8c563bfa5e1539c909e0444afd397163145a2636327325a9e308358bdb1658
data/README.md ADDED
@@ -0,0 +1,37 @@
1
+ # Visual DB [![Gem Version](https://badge.fury.io/rb/visual_db.svg)](http://badge.fury.io/rb/visual_db)
2
+
3
+ Visual DB is web utility to connect to local or remote MySQL instances and view & modify data. Visual DB is meant to be quick and easy to use, and reduce some potential for errors in user queries. Visual DB is a resource well suited for those just getting started with SQL syntax.
4
+
5
+ ### Screenshots
6
+
7
+ * ![screenshot](https://github.com/at1as/at1as.github.io/blob/master/github_repo_assets/visual_db.jpg)
8
+
9
+ ### Features
10
+
11
+ * Download Tables as CSV
12
+ * View/Create/Delete Databases
13
+ * Connect to local or remote mySQL instances
14
+ * Accessible via browser of your choice
15
+ * SQL Query templates based on user input
16
+ * Tables sortable by column header
17
+
18
+ ### Installation
19
+
20
+ To install the [Ruby Gem](https://rubygems.org/gems/visual_db) (which is generally more stable, but several commits behind this repository):
21
+
22
+ ```bash
23
+ gem install visual_db
24
+ ```
25
+
26
+ To install this repo directory, simply clone assign appropriate permissions, and install necessary dependencies.
27
+
28
+
29
+ ### Dependencies
30
+
31
+ * See Gemfile for dependencies
32
+ * Developed and tested on Ruby 2.2, Mac OS 10.10, and MySQL 5.6.21
33
+
34
+ ### TODO
35
+
36
+ * UI Work
37
+ * More UI guided SQL query generation
@@ -1,3 +1,3 @@
1
1
  module VisualDb
2
- VERSION = "0.1.0"
2
+ VERSION = "0.1.1"
3
3
  end
data/lib/visual_db.rb CHANGED
@@ -109,6 +109,13 @@ end
109
109
  ## General Filters
110
110
  before do
111
111
  connection_redir '/' unless NO_AUTH_PATHS.include? request.path_info
112
+ begin
113
+ $sql_conn.ping() if $sql_conn
114
+ rescue Mysql::Error => e
115
+ error_logger("#{e} (error no. #{$sql_conn.errno rescue "\"nil\""}). Please reconnect")
116
+ $sql_conn = false
117
+ redirect "/?error=true"
118
+ end
112
119
  $db_error = nil unless params[:error]
113
120
  $db_success = nil unless params[:success]
114
121
  $table_cols = nil unless params[:query_action] == "showcols"
@@ -0,0 +1 @@
1
+ function dean_addEvent(t,e,r){if(t.addEventListener)t.addEventListener(e,r,!1);else{r.$$guid||(r.$$guid=dean_addEvent.guid++),t.events||(t.events={});var o=t.events[e];o||(o=t.events[e]={},t["on"+e]&&(o[0]=t["on"+e])),o[r.$$guid]=r,t["on"+e]=handleEvent}}function removeEvent(t,e,r){t.removeEventListener?t.removeEventListener(e,r,!1):t.events&&t.events[e]&&delete t.events[e][r.$$guid]}function handleEvent(t){var e=!0;t=t||fixEvent(((this.ownerDocument||this.document||this).parentWindow||window).event);var r=this.events[t.type];for(var o in r)this.$$handleEvent=r[o],this.$$handleEvent(t)===!1&&(e=!1);return e}function fixEvent(t){return t.preventDefault=fixEvent.preventDefault,t.stopPropagation=fixEvent.stopPropagation,t}var stIsIE=!1;if(sorttable={init:function(){arguments.callee.done||(arguments.callee.done=!0,_timer&&clearInterval(_timer),document.createElement&&document.getElementsByTagName&&(sorttable.DATE_RE=/^(\d\d?)[\/\.-](\d\d?)[\/\.-]((\d\d)?\d\d)$/,forEach(document.getElementsByTagName("table"),function(t){-1!=t.className.search(/\bsortable\b/)&&sorttable.makeSortable(t)})))},makeSortable:function(t){if(0==t.getElementsByTagName("thead").length&&(the=document.createElement("thead"),the.appendChild(t.rows[0]),t.insertBefore(the,t.firstChild)),null==t.tHead&&(t.tHead=t.getElementsByTagName("thead")[0]),1==t.tHead.rows.length){sortbottomrows=[];for(var e=0;e<t.rows.length;e++)-1!=t.rows[e].className.search(/\bsortbottom\b/)&&(sortbottomrows[sortbottomrows.length]=t.rows[e]);if(sortbottomrows){null==t.tFoot&&(tfo=document.createElement("tfoot"),t.appendChild(tfo));for(var e=0;e<sortbottomrows.length;e++)tfo.appendChild(sortbottomrows[e]);delete sortbottomrows}headrow=t.tHead.rows[0].cells;for(var e=0;e<headrow.length;e++)headrow[e].className.match(/\bsorttable_nosort\b/)||(mtch=headrow[e].className.match(/\bsorttable_([a-z0-9]+)\b/),mtch&&(override=mtch[1]),headrow[e].sorttable_sortfunction=mtch&&"function"==typeof sorttable["sort_"+override]?sorttable["sort_"+override]:sorttable.guessType(t,e),headrow[e].sorttable_columnindex=e,headrow[e].sorttable_tbody=t.tBodies[0],dean_addEvent(headrow[e],"click",sorttable.innerSortFunction=function(){if(-1!=this.className.search(/\bsorttable_sorted\b/))return sorttable.reverse(this.sorttable_tbody),this.className=this.className.replace("sorttable_sorted","sorttable_sorted_reverse"),this.removeChild(document.getElementById("sorttable_sortfwdind")),sortrevind=document.createElement("span"),sortrevind.id="sorttable_sortrevind",sortrevind.innerHTML=stIsIE?'&nbsp<font face="webdings">5</font>':"&nbsp;&#x25B4;",void this.appendChild(sortrevind);if(-1!=this.className.search(/\bsorttable_sorted_reverse\b/))return sorttable.reverse(this.sorttable_tbody),this.className=this.className.replace("sorttable_sorted_reverse","sorttable_sorted"),this.removeChild(document.getElementById("sorttable_sortrevind")),sortfwdind=document.createElement("span"),sortfwdind.id="sorttable_sortfwdind",sortfwdind.innerHTML=stIsIE?'&nbsp<font face="webdings">6</font>':"&nbsp;&#x25BE;",void this.appendChild(sortfwdind);theadrow=this.parentNode,forEach(theadrow.childNodes,function(t){1==t.nodeType&&(t.className=t.className.replace("sorttable_sorted_reverse",""),t.className=t.className.replace("sorttable_sorted",""))}),sortfwdind=document.getElementById("sorttable_sortfwdind"),sortfwdind&&sortfwdind.parentNode.removeChild(sortfwdind),sortrevind=document.getElementById("sorttable_sortrevind"),sortrevind&&sortrevind.parentNode.removeChild(sortrevind),this.className+=" sorttable_sorted",sortfwdind=document.createElement("span"),sortfwdind.id="sorttable_sortfwdind",sortfwdind.innerHTML=stIsIE?'&nbsp<font face="webdings">6</font>':"&nbsp;&#x25BE;",this.appendChild(sortfwdind),row_array=[],col=this.sorttable_columnindex,rows=this.sorttable_tbody.rows;for(var t=0;t<rows.length;t++)row_array[row_array.length]=[sorttable.getInnerText(rows[t].cells[col]),rows[t]];row_array.sort(this.sorttable_sortfunction),tb=this.sorttable_tbody;for(var t=0;t<row_array.length;t++)tb.appendChild(row_array[t][1]);delete row_array}))}},guessType:function(t,e){sortfn=sorttable.sort_alpha;for(var r=0;r<t.tBodies[0].rows.length;r++)if(text=sorttable.getInnerText(t.tBodies[0].rows[r].cells[e]),""!=text){if(text.match(/^-?[£$¤]?[\d,.]+%?$/))return sorttable.sort_numeric;if(possdate=text.match(sorttable.DATE_RE),possdate){if(first=parseInt(possdate[1]),second=parseInt(possdate[2]),first>12)return sorttable.sort_ddmm;if(second>12)return sorttable.sort_mmdd;sortfn=sorttable.sort_ddmm}}return sortfn},getInnerText:function(t){if(!t)return"";if(hasInputs="function"==typeof t.getElementsByTagName&&t.getElementsByTagName("input").length,null!=t.getAttribute("sorttable_customkey"))return t.getAttribute("sorttable_customkey");if("undefined"!=typeof t.textContent&&!hasInputs)return t.textContent.replace(/^\s+|\s+$/g,"");if("undefined"!=typeof t.innerText&&!hasInputs)return t.innerText.replace(/^\s+|\s+$/g,"");if("undefined"!=typeof t.text&&!hasInputs)return t.text.replace(/^\s+|\s+$/g,"");switch(t.nodeType){case 3:if("input"==t.nodeName.toLowerCase())return t.value.replace(/^\s+|\s+$/g,"");case 4:return t.nodeValue.replace(/^\s+|\s+$/g,"");case 1:case 11:for(var e="",r=0;r<t.childNodes.length;r++)e+=sorttable.getInnerText(t.childNodes[r]);return e.replace(/^\s+|\s+$/g,"");default:return""}},reverse:function(t){newrows=[];for(var e=0;e<t.rows.length;e++)newrows[newrows.length]=t.rows[e];for(var e=newrows.length-1;e>=0;e--)t.appendChild(newrows[e]);delete newrows},sort_numeric:function(t,e){return aa=parseFloat(t[0].replace(/[^0-9.-]/g,"")),isNaN(aa)&&(aa=0),bb=parseFloat(e[0].replace(/[^0-9.-]/g,"")),isNaN(bb)&&(bb=0),aa-bb},sort_alpha:function(t,e){return t[0].toLowerCase()==e[0].toLowerCase()?0:t[0].toLowerCase()<e[0].toLowerCase()?-1:1},sort_ddmm:function(t,e){return mtch=t[0].match(sorttable.DATE_RE),y=mtch[3],m=mtch[2],d=mtch[1],1==m.length&&(m="0"+m),1==d.length&&(d="0"+d),dt1=y+m+d,mtch=e[0].match(sorttable.DATE_RE),y=mtch[3],m=mtch[2],d=mtch[1],1==m.length&&(m="0"+m),1==d.length&&(d="0"+d),dt2=y+m+d,dt1==dt2?0:dt1<dt2?-1:1},sort_mmdd:function(t,e){return mtch=t[0].match(sorttable.DATE_RE),y=mtch[3],d=mtch[2],m=mtch[1],1==m.length&&(m="0"+m),1==d.length&&(d="0"+d),dt1=y+m+d,mtch=e[0].match(sorttable.DATE_RE),y=mtch[3],d=mtch[2],m=mtch[1],1==m.length&&(m="0"+m),1==d.length&&(d="0"+d),dt2=y+m+d,dt1==dt2?0:dt1<dt2?-1:1},shaker_sort:function(t,e){for(var r=0,o=t.length-1,n=!0;n;){n=!1;for(var s=r;o>s;++s)if(e(t[s],t[s+1])>0){var a=t[s];t[s]=t[s+1],t[s+1]=a,n=!0}if(o--,!n)break;for(var s=o;s>r;--s)if(e(t[s],t[s-1])<0){var a=t[s];t[s]=t[s-1],t[s-1]=a,n=!0}r++}}},document.addEventListener&&document.addEventListener("DOMContentLoaded",sorttable.init,!1),/WebKit/i.test(navigator.userAgent))var _timer=setInterval(function(){/loaded|complete/.test(document.readyState)&&sorttable.init()},10);window.onload=sorttable.init,dean_addEvent.guid=1,fixEvent.preventDefault=function(){this.returnValue=!1},fixEvent.stopPropagation=function(){this.cancelBubble=!0},Array.forEach||(Array.forEach=function(t,e,r){for(var o=0;o<t.length;o++)e.call(r,t[o],o,t)}),Function.prototype.forEach=function(t,e,r){for(var o in t)"undefined"==typeof this.prototype[o]&&e.call(r,t[o],o,t)},String.forEach=function(t,e,r){Array.forEach(t.split(""),function(o,n){e.call(r,o,n,t)})};var forEach=function(t,e,r){if(t){var o=Object;if(t instanceof Function)o=Function;else{if(t.forEach instanceof Function)return void t.forEach(e,r);"string"==typeof t?o=String:"number"==typeof t.length&&(o=Array)}o.forEach(t,e,r)}};
data/views/tables.erb CHANGED
@@ -12,7 +12,7 @@
12
12
  <link href="/css/main.css" rel="stylesheet">
13
13
 
14
14
  <!-- Table Sorting -->
15
- <script src="/js/sorttable.js"></script>
15
+ <script src="/js/sorttable.min.js"></script>
16
16
  </head>
17
17
  <body>
18
18
 
data/visual_db.gemspec CHANGED
@@ -5,7 +5,7 @@ require 'visual_db/version'
5
5
 
6
6
  Gem::Specification.new do |spec|
7
7
  spec.name = "visual_db"
8
- spec.date = "2015-02-13"
8
+ spec.date = "2015-04-04"
9
9
  spec.version = VisualDb::VERSION
10
10
  spec.authors = ["Jason Willems"]
11
11
  spec.email = ["jason@willems.ca"]
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: visual_db
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.0
4
+ version: 0.1.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Jason Willems
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2015-02-13 00:00:00.000000000 Z
11
+ date: 2015-04-04 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: sinatra
@@ -110,6 +110,7 @@ files:
110
110
  - Gemfile
111
111
  - Gemfile.lock
112
112
  - LICENSE.txt
113
+ - README.md
113
114
  - bin/visual_db
114
115
  - lib/visual_db.rb
115
116
  - lib/visual_db/version.rb
@@ -117,7 +118,7 @@ files:
117
118
  - public/css/bootstrap.min.css
118
119
  - public/css/main.css
119
120
  - public/css/starter-template.css
120
- - public/js/sorttable.js
121
+ - public/js/sorttable.min.js
121
122
  - views/about.erb
122
123
  - views/databases.erb
123
124
  - views/index.erb
@@ -1,495 +0,0 @@
1
- /*
2
- SortTable
3
- version 2
4
- 7th April 2007
5
- Stuart Langridge, http://www.kryogenix.org/code/browser/sorttable/
6
-
7
- Instructions:
8
- Download this file
9
- Add <script src="sorttable.js"></script> to your HTML
10
- Add class="sortable" to any table you'd like to make sortable
11
- Click on the headers to sort
12
-
13
- Thanks to many, many people for contributions and suggestions.
14
- Licenced as X11: http://www.kryogenix.org/code/browser/licence.html
15
- This basically means: do what you want with it.
16
- */
17
-
18
-
19
- var stIsIE = /*@cc_on!@*/false;
20
-
21
- sorttable = {
22
- init: function() {
23
- // quit if this function has already been called
24
- if (arguments.callee.done) return;
25
- // flag this function so we don't do the same thing twice
26
- arguments.callee.done = true;
27
- // kill the timer
28
- if (_timer) clearInterval(_timer);
29
-
30
- if (!document.createElement || !document.getElementsByTagName) return;
31
-
32
- sorttable.DATE_RE = /^(\d\d?)[\/\.-](\d\d?)[\/\.-]((\d\d)?\d\d)$/;
33
-
34
- forEach(document.getElementsByTagName('table'), function(table) {
35
- if (table.className.search(/\bsortable\b/) != -1) {
36
- sorttable.makeSortable(table);
37
- }
38
- });
39
-
40
- },
41
-
42
- makeSortable: function(table) {
43
- if (table.getElementsByTagName('thead').length == 0) {
44
- // table doesn't have a tHead. Since it should have, create one and
45
- // put the first table row in it.
46
- the = document.createElement('thead');
47
- the.appendChild(table.rows[0]);
48
- table.insertBefore(the,table.firstChild);
49
- }
50
- // Safari doesn't support table.tHead, sigh
51
- if (table.tHead == null) table.tHead = table.getElementsByTagName('thead')[0];
52
-
53
- if (table.tHead.rows.length != 1) return; // can't cope with two header rows
54
-
55
- // Sorttable v1 put rows with a class of "sortbottom" at the bottom (as
56
- // "total" rows, for example). This is B&R, since what you're supposed
57
- // to do is put them in a tfoot. So, if there are sortbottom rows,
58
- // for backwards compatibility, move them to tfoot (creating it if needed).
59
- sortbottomrows = [];
60
- for (var i=0; i<table.rows.length; i++) {
61
- if (table.rows[i].className.search(/\bsortbottom\b/) != -1) {
62
- sortbottomrows[sortbottomrows.length] = table.rows[i];
63
- }
64
- }
65
- if (sortbottomrows) {
66
- if (table.tFoot == null) {
67
- // table doesn't have a tfoot. Create one.
68
- tfo = document.createElement('tfoot');
69
- table.appendChild(tfo);
70
- }
71
- for (var i=0; i<sortbottomrows.length; i++) {
72
- tfo.appendChild(sortbottomrows[i]);
73
- }
74
- delete sortbottomrows;
75
- }
76
-
77
- // work through each column and calculate its type
78
- headrow = table.tHead.rows[0].cells;
79
- for (var i=0; i<headrow.length; i++) {
80
- // manually override the type with a sorttable_type attribute
81
- if (!headrow[i].className.match(/\bsorttable_nosort\b/)) { // skip this col
82
- mtch = headrow[i].className.match(/\bsorttable_([a-z0-9]+)\b/);
83
- if (mtch) { override = mtch[1]; }
84
- if (mtch && typeof sorttable["sort_"+override] == 'function') {
85
- headrow[i].sorttable_sortfunction = sorttable["sort_"+override];
86
- } else {
87
- headrow[i].sorttable_sortfunction = sorttable.guessType(table,i);
88
- }
89
- // make it clickable to sort
90
- headrow[i].sorttable_columnindex = i;
91
- headrow[i].sorttable_tbody = table.tBodies[0];
92
- dean_addEvent(headrow[i],"click", sorttable.innerSortFunction = function(e) {
93
-
94
- if (this.className.search(/\bsorttable_sorted\b/) != -1) {
95
- // if we're already sorted by this column, just
96
- // reverse the table, which is quicker
97
- sorttable.reverse(this.sorttable_tbody);
98
- this.className = this.className.replace('sorttable_sorted',
99
- 'sorttable_sorted_reverse');
100
- this.removeChild(document.getElementById('sorttable_sortfwdind'));
101
- sortrevind = document.createElement('span');
102
- sortrevind.id = "sorttable_sortrevind";
103
- sortrevind.innerHTML = stIsIE ? '&nbsp<font face="webdings">5</font>' : '&nbsp;&#x25B4;';
104
- this.appendChild(sortrevind);
105
- return;
106
- }
107
- if (this.className.search(/\bsorttable_sorted_reverse\b/) != -1) {
108
- // if we're already sorted by this column in reverse, just
109
- // re-reverse the table, which is quicker
110
- sorttable.reverse(this.sorttable_tbody);
111
- this.className = this.className.replace('sorttable_sorted_reverse',
112
- 'sorttable_sorted');
113
- this.removeChild(document.getElementById('sorttable_sortrevind'));
114
- sortfwdind = document.createElement('span');
115
- sortfwdind.id = "sorttable_sortfwdind";
116
- sortfwdind.innerHTML = stIsIE ? '&nbsp<font face="webdings">6</font>' : '&nbsp;&#x25BE;';
117
- this.appendChild(sortfwdind);
118
- return;
119
- }
120
-
121
- // remove sorttable_sorted classes
122
- theadrow = this.parentNode;
123
- forEach(theadrow.childNodes, function(cell) {
124
- if (cell.nodeType == 1) { // an element
125
- cell.className = cell.className.replace('sorttable_sorted_reverse','');
126
- cell.className = cell.className.replace('sorttable_sorted','');
127
- }
128
- });
129
- sortfwdind = document.getElementById('sorttable_sortfwdind');
130
- if (sortfwdind) { sortfwdind.parentNode.removeChild(sortfwdind); }
131
- sortrevind = document.getElementById('sorttable_sortrevind');
132
- if (sortrevind) { sortrevind.parentNode.removeChild(sortrevind); }
133
-
134
- this.className += ' sorttable_sorted';
135
- sortfwdind = document.createElement('span');
136
- sortfwdind.id = "sorttable_sortfwdind";
137
- sortfwdind.innerHTML = stIsIE ? '&nbsp<font face="webdings">6</font>' : '&nbsp;&#x25BE;';
138
- this.appendChild(sortfwdind);
139
-
140
- // build an array to sort. This is a Schwartzian transform thing,
141
- // i.e., we "decorate" each row with the actual sort key,
142
- // sort based on the sort keys, and then put the rows back in order
143
- // which is a lot faster because you only do getInnerText once per row
144
- row_array = [];
145
- col = this.sorttable_columnindex;
146
- rows = this.sorttable_tbody.rows;
147
- for (var j=0; j<rows.length; j++) {
148
- row_array[row_array.length] = [sorttable.getInnerText(rows[j].cells[col]), rows[j]];
149
- }
150
- /* If you want a stable sort, uncomment the following line */
151
- //sorttable.shaker_sort(row_array, this.sorttable_sortfunction);
152
- /* and comment out this one */
153
- row_array.sort(this.sorttable_sortfunction);
154
-
155
- tb = this.sorttable_tbody;
156
- for (var j=0; j<row_array.length; j++) {
157
- tb.appendChild(row_array[j][1]);
158
- }
159
-
160
- delete row_array;
161
- });
162
- }
163
- }
164
- },
165
-
166
- guessType: function(table, column) {
167
- // guess the type of a column based on its first non-blank row
168
- sortfn = sorttable.sort_alpha;
169
- for (var i=0; i<table.tBodies[0].rows.length; i++) {
170
- text = sorttable.getInnerText(table.tBodies[0].rows[i].cells[column]);
171
- if (text != '') {
172
- if (text.match(/^-?[�$�]?[\d,.]+%?$/)) {
173
- return sorttable.sort_numeric;
174
- }
175
- // check for a date: dd/mm/yyyy or dd/mm/yy
176
- // can have / or . or - as separator
177
- // can be mm/dd as well
178
- possdate = text.match(sorttable.DATE_RE)
179
- if (possdate) {
180
- // looks like a date
181
- first = parseInt(possdate[1]);
182
- second = parseInt(possdate[2]);
183
- if (first > 12) {
184
- // definitely dd/mm
185
- return sorttable.sort_ddmm;
186
- } else if (second > 12) {
187
- return sorttable.sort_mmdd;
188
- } else {
189
- // looks like a date, but we can't tell which, so assume
190
- // that it's dd/mm (English imperialism!) and keep looking
191
- sortfn = sorttable.sort_ddmm;
192
- }
193
- }
194
- }
195
- }
196
- return sortfn;
197
- },
198
-
199
- getInnerText: function(node) {
200
- // gets the text we want to use for sorting for a cell.
201
- // strips leading and trailing whitespace.
202
- // this is *not* a generic getInnerText function; it's special to sorttable.
203
- // for example, you can override the cell text with a customkey attribute.
204
- // it also gets .value for <input> fields.
205
-
206
- if (!node) return "";
207
-
208
- hasInputs = (typeof node.getElementsByTagName == 'function') &&
209
- node.getElementsByTagName('input').length;
210
-
211
- if (node.getAttribute("sorttable_customkey") != null) {
212
- return node.getAttribute("sorttable_customkey");
213
- }
214
- else if (typeof node.textContent != 'undefined' && !hasInputs) {
215
- return node.textContent.replace(/^\s+|\s+$/g, '');
216
- }
217
- else if (typeof node.innerText != 'undefined' && !hasInputs) {
218
- return node.innerText.replace(/^\s+|\s+$/g, '');
219
- }
220
- else if (typeof node.text != 'undefined' && !hasInputs) {
221
- return node.text.replace(/^\s+|\s+$/g, '');
222
- }
223
- else {
224
- switch (node.nodeType) {
225
- case 3:
226
- if (node.nodeName.toLowerCase() == 'input') {
227
- return node.value.replace(/^\s+|\s+$/g, '');
228
- }
229
- case 4:
230
- return node.nodeValue.replace(/^\s+|\s+$/g, '');
231
- break;
232
- case 1:
233
- case 11:
234
- var innerText = '';
235
- for (var i = 0; i < node.childNodes.length; i++) {
236
- innerText += sorttable.getInnerText(node.childNodes[i]);
237
- }
238
- return innerText.replace(/^\s+|\s+$/g, '');
239
- break;
240
- default:
241
- return '';
242
- }
243
- }
244
- },
245
-
246
- reverse: function(tbody) {
247
- // reverse the rows in a tbody
248
- newrows = [];
249
- for (var i=0; i<tbody.rows.length; i++) {
250
- newrows[newrows.length] = tbody.rows[i];
251
- }
252
- for (var i=newrows.length-1; i>=0; i--) {
253
- tbody.appendChild(newrows[i]);
254
- }
255
- delete newrows;
256
- },
257
-
258
- /* sort functions
259
- each sort function takes two parameters, a and b
260
- you are comparing a[0] and b[0] */
261
- sort_numeric: function(a,b) {
262
- aa = parseFloat(a[0].replace(/[^0-9.-]/g,''));
263
- if (isNaN(aa)) aa = 0;
264
- bb = parseFloat(b[0].replace(/[^0-9.-]/g,''));
265
- if (isNaN(bb)) bb = 0;
266
- return aa-bb;
267
- },
268
- sort_alpha: function(a,b) {
269
- if (a[0]==b[0]) return 0;
270
- if (a[0]<b[0]) return -1;
271
- return 1;
272
- },
273
- sort_ddmm: function(a,b) {
274
- mtch = a[0].match(sorttable.DATE_RE);
275
- y = mtch[3]; m = mtch[2]; d = mtch[1];
276
- if (m.length == 1) m = '0'+m;
277
- if (d.length == 1) d = '0'+d;
278
- dt1 = y+m+d;
279
- mtch = b[0].match(sorttable.DATE_RE);
280
- y = mtch[3]; m = mtch[2]; d = mtch[1];
281
- if (m.length == 1) m = '0'+m;
282
- if (d.length == 1) d = '0'+d;
283
- dt2 = y+m+d;
284
- if (dt1==dt2) return 0;
285
- if (dt1<dt2) return -1;
286
- return 1;
287
- },
288
- sort_mmdd: function(a,b) {
289
- mtch = a[0].match(sorttable.DATE_RE);
290
- y = mtch[3]; d = mtch[2]; m = mtch[1];
291
- if (m.length == 1) m = '0'+m;
292
- if (d.length == 1) d = '0'+d;
293
- dt1 = y+m+d;
294
- mtch = b[0].match(sorttable.DATE_RE);
295
- y = mtch[3]; d = mtch[2]; m = mtch[1];
296
- if (m.length == 1) m = '0'+m;
297
- if (d.length == 1) d = '0'+d;
298
- dt2 = y+m+d;
299
- if (dt1==dt2) return 0;
300
- if (dt1<dt2) return -1;
301
- return 1;
302
- },
303
-
304
- shaker_sort: function(list, comp_func) {
305
- // A stable sort function to allow multi-level sorting of data
306
- // see: http://en.wikipedia.org/wiki/Cocktail_sort
307
- // thanks to Joseph Nahmias
308
- var b = 0;
309
- var t = list.length - 1;
310
- var swap = true;
311
-
312
- while(swap) {
313
- swap = false;
314
- for(var i = b; i < t; ++i) {
315
- if ( comp_func(list[i], list[i+1]) > 0 ) {
316
- var q = list[i]; list[i] = list[i+1]; list[i+1] = q;
317
- swap = true;
318
- }
319
- } // for
320
- t--;
321
-
322
- if (!swap) break;
323
-
324
- for(var i = t; i > b; --i) {
325
- if ( comp_func(list[i], list[i-1]) < 0 ) {
326
- var q = list[i]; list[i] = list[i-1]; list[i-1] = q;
327
- swap = true;
328
- }
329
- } // for
330
- b++;
331
-
332
- } // while(swap)
333
- }
334
- }
335
-
336
- /* ******************************************************************
337
- Supporting functions: bundled here to avoid depending on a library
338
- ****************************************************************** */
339
-
340
- // Dean Edwards/Matthias Miller/John Resig
341
-
342
- /* for Mozilla/Opera9 */
343
- if (document.addEventListener) {
344
- document.addEventListener("DOMContentLoaded", sorttable.init, false);
345
- }
346
-
347
- /* for Internet Explorer */
348
- /*@cc_on @*/
349
- /*@if (@_win32)
350
- document.write("<script id=__ie_onload defer src=javascript:void(0)><\/script>");
351
- var script = document.getElementById("__ie_onload");
352
- script.onreadystatechange = function() {
353
- if (this.readyState == "complete") {
354
- sorttable.init(); // call the onload handler
355
- }
356
- };
357
- /*@end @*/
358
-
359
- /* for Safari */
360
- if (/WebKit/i.test(navigator.userAgent)) { // sniff
361
- var _timer = setInterval(function() {
362
- if (/loaded|complete/.test(document.readyState)) {
363
- sorttable.init(); // call the onload handler
364
- }
365
- }, 10);
366
- }
367
-
368
- /* for other browsers */
369
- window.onload = sorttable.init;
370
-
371
- // written by Dean Edwards, 2005
372
- // with input from Tino Zijdel, Matthias Miller, Diego Perini
373
-
374
- // http://dean.edwards.name/weblog/2005/10/add-event/
375
-
376
- function dean_addEvent(element, type, handler) {
377
- if (element.addEventListener) {
378
- element.addEventListener(type, handler, false);
379
- } else {
380
- // assign each event handler a unique ID
381
- if (!handler.$$guid) handler.$$guid = dean_addEvent.guid++;
382
- // create a hash table of event types for the element
383
- if (!element.events) element.events = {};
384
- // create a hash table of event handlers for each element/event pair
385
- var handlers = element.events[type];
386
- if (!handlers) {
387
- handlers = element.events[type] = {};
388
- // store the existing event handler (if there is one)
389
- if (element["on" + type]) {
390
- handlers[0] = element["on" + type];
391
- }
392
- }
393
- // store the event handler in the hash table
394
- handlers[handler.$$guid] = handler;
395
- // assign a global event handler to do all the work
396
- element["on" + type] = handleEvent;
397
- }
398
- };
399
- // a counter used to create unique IDs
400
- dean_addEvent.guid = 1;
401
-
402
- function removeEvent(element, type, handler) {
403
- if (element.removeEventListener) {
404
- element.removeEventListener(type, handler, false);
405
- } else {
406
- // delete the event handler from the hash table
407
- if (element.events && element.events[type]) {
408
- delete element.events[type][handler.$$guid];
409
- }
410
- }
411
- };
412
-
413
- function handleEvent(event) {
414
- var returnValue = true;
415
- // grab the event object (IE uses a global event object)
416
- event = event || fixEvent(((this.ownerDocument || this.document || this).parentWindow || window).event);
417
- // get a reference to the hash table of event handlers
418
- var handlers = this.events[event.type];
419
- // execute each event handler
420
- for (var i in handlers) {
421
- this.$$handleEvent = handlers[i];
422
- if (this.$$handleEvent(event) === false) {
423
- returnValue = false;
424
- }
425
- }
426
- return returnValue;
427
- };
428
-
429
- function fixEvent(event) {
430
- // add W3C standard event methods
431
- event.preventDefault = fixEvent.preventDefault;
432
- event.stopPropagation = fixEvent.stopPropagation;
433
- return event;
434
- };
435
- fixEvent.preventDefault = function() {
436
- this.returnValue = false;
437
- };
438
- fixEvent.stopPropagation = function() {
439
- this.cancelBubble = true;
440
- }
441
-
442
- // Dean's forEach: http://dean.edwards.name/base/forEach.js
443
- /*
444
- forEach, version 1.0
445
- Copyright 2006, Dean Edwards
446
- License: http://www.opensource.org/licenses/mit-license.php
447
- */
448
-
449
- // array-like enumeration
450
- if (!Array.forEach) { // mozilla already supports this
451
- Array.forEach = function(array, block, context) {
452
- for (var i = 0; i < array.length; i++) {
453
- block.call(context, array[i], i, array);
454
- }
455
- };
456
- }
457
-
458
- // generic enumeration
459
- Function.prototype.forEach = function(object, block, context) {
460
- for (var key in object) {
461
- if (typeof this.prototype[key] == "undefined") {
462
- block.call(context, object[key], key, object);
463
- }
464
- }
465
- };
466
-
467
- // character enumeration
468
- String.forEach = function(string, block, context) {
469
- Array.forEach(string.split(""), function(chr, index) {
470
- block.call(context, chr, index, string);
471
- });
472
- };
473
-
474
- // globally resolve forEach enumeration
475
- var forEach = function(object, block, context) {
476
- if (object) {
477
- var resolve = Object; // default
478
- if (object instanceof Function) {
479
- // functions have a "length" property
480
- resolve = Function;
481
- } else if (object.forEach instanceof Function) {
482
- // the object implements a custom forEach method so use that
483
- object.forEach(block, context);
484
- return;
485
- } else if (typeof object == "string") {
486
- // the object is a string
487
- resolve = String;
488
- } else if (typeof object.length == "number") {
489
- // the object is array-like
490
- resolve = Array;
491
- }
492
- resolve.forEach(object, block, context);
493
- }
494
- };
495
-